python使用telnetlib
1 前言
目前,本篇僅記錄前段時間搜索得到的關於python使用Telnet的技術博客,由於受領新任務,未進一步驗證和深入研究與應用。
參考鏈接:
python官網:https://docs.python.org/2/library/telnetlib.html#telnet-example
python調用telnet,登陸遠程機器執行命令:https://www.oschina.net/code/snippet_102130_6381
Python使用Telnetlib實現遠程操作:https://blog.csdn.net/lingfeng5/article/details/73744698
python中telnetlib模塊的使用:https://blog.csdn.net/five3/article/details/8099997
python自動化運維之Telnetlib:https://blog.csdn.net/w1418899532/article/details/82701346
2 原理
3 實踐
3.1 通過執行腳本,可以登陸遠程機器,並執行命令

!/usr/bin/env python def telnetdo(HOST=None, USER=None, PASS=None, COMMAND=None): #define a function import telnetlib, sys if not HOST: try: HOST = sys.argv[1] USER = sys.argv[2] PASS = sys.argv[3] COMMAND = sys.argv[4] except: print "Usage: telnetdo.py host user pass command" return msg = ['Debug messages:\n'] # tn = telnetlib.Telnet() # try: tn.open(HOST) except: print "Cannot open host" return #msg.append(tn.expect(['login:'], 5)) # tn.read_until("login:") tn.write(USER + '\n') if PASS: #msg.append(tn.expect(['Password:'], 5)) tn.read_until("Password:") tn.write(PASS + '\n') #msg.append(tn.expect([USER], 5)) tn.write(COMMAND + '\n') tn.write("exit\n") #msg.append(tn.expect(['#'], 5)) tmp = tn.read_all() tn.close() del tn return tmp if __name__ == '__main__': print telnetdo()
3.2 python自動化運維之Telnetlib
前言:
遠程連接中興設備(系統使用的中興網卡)時使用的事Telnet連接,連接時設有二次驗證,每次輸入用戶名密碼和執行命令是個繁瑣的過程,使用Python自帶的telnetlib庫可以編寫腳本,實現批量登錄服務器並執行命令查詢數據量。
1.導入telnetlib庫可以直接使用。
from telnetlib import Telnet
2.配置服務器、用戶名、密碼,cmd命令等

##############################需要配置信息########################## #關聯設備服務器IP列表 Hosts=['192.168.1.xx','192.168.1.xx','192.168.1.xx','192.168.1.xx','192.168.1.xx','192.168.1.xx'] # 登錄用戶名 username = 'xx' # 登錄密碼 password = 'xx' #EN命令 EN = 'xx' #二次驗證口令 ZXR10 = 'xx' # 命令提示符 finish = 'xx#' #需要執行的命令 commands = ['show lte data processing report','show interface xgei1/1'] #輸出內容寫入文件 g_outfilePath = '/home/tnOutResult' ####################################################################
IP、username、pwd等都以參數形式傳入功能函數。
def do_telnet(Hosts, username, password, finish, commands):
3.功能函數
功能函數主要實現Telnet的遠程登錄、命令執行。

# 連接Telnet服務器 tn = Telnet(host, port=23, timeout=10) #tn.set_debuglevel(2) # 輸入登錄用戶名 tn.read_until('Username:') tn.write(username + '\n') # 輸入登錄密碼 tn.read_until('Password:') tn.write(password + '\n') # 輸入命令en tn.read_until('ZXR10>') tn.write(EN + '\n') # 輸入二次驗證口令 tn.read_until('Password:') tn.write(ZXR10 + '\n') # 登錄完畢后執行命令 tn.read_until(finish) for command in commands: tn.write(command + '\n') time.sleep(2) for i in range(10): tn.write(' ') time.sleep(0.1) Outresult = tn.read_very_eager() #執行完畢后,終止Telnet連接(或輸入exit退出) #tn.read_until(finish) tn.close() # tn.write('exit\n')
read_very_eager()函數是telnetlib模塊里的read方法,這里要設置延時time.sleep(2),才能保證數據讀取完畢。
telnetlib模塊里的read方法介紹:
read_until():當結果中存在想要的信息時返回。
read_some():只要有結果就返回。
read_very_lazy():返回緩沖區中的數據。
telnetlib采用緩沖的處理方式,因此數據並不是一下子就返回的,而是先放在了緩沖區中。許多的讀取處理都是圍繞着這個緩沖區來的。而緩沖區的信息何時到達就不說不清楚 了,也許很快,也許很慢,也許分別到達,也許一下子就收到了。因此,對於數據不一定到齊的這種情況,就采用了read_until()來判斷緩沖區中的數據是否有想要的內容,如果沒有就等待,除非到達了超時時間。
3.2 python中telnetlib模塊的使用
python下能支持telnet的模塊telnetlib是內置模塊,直接import就可以了,其基本的使用方法也是比較簡單的。

#encoding=utf-8 def do_telnet(Host, username, password, finish, commands): import telnetlib '''Telnet遠程登錄:Windows客戶端連接Linux服務器''' # 連接Telnet服務器 tn = telnetlib.Telnet(Host, port=23, timeout=10) tn.set_debuglevel(2) # 輸入登錄用戶名 tn.read_until('login: ') tn.write(username + '\n') # 輸入登錄密碼 tn.read_until('password: ') tn.write(password + '\n') # 登錄完畢后執行命令 tn.read_until(finish) for command in commands: tn.write('%s\n' % command) #執行完畢后,終止Telnet連接(或輸入exit退出) tn.read_until(finish) tn.close() # tn.write('exit\n') if __name__=='__main__': # 配置選項 Host = '10.255.254.205' # Telnet服務器IP username = 'administrator' # 登錄用戶名 password = 'dell1950' # 登錄密碼 finish = ':~$ ' # 命令提示符 commands = ['echo "test"']
其中port和timeout是可選的參數,而timeout的只是在初始化socket連接時起作用,而一旦連接成功后如果出現等待那就不會起作用了,比如使用read_until方式獲取內容時返回的內容與指定的內容沒有吻合,那么就會造成提示等待的情況,這時timeout是不會起作用的,而這個socket連接會一直保持着,永生不死。
那么如何解決這個問題呢,其實還有一種比較原始的方法,就是使用sleep方法來代替read_until方法,這樣就不會出現種情況,因為到點就會自己輸入,最多也就是最后得不到想要的結果,但是這個方式很不穩定,兼容性也不好;另一種方法是使用線程來啟動這個函數,然后對子線程進行超時設置,這樣就可以達到間接控制這個telnet連接的目的了。

import threading pars = replace_db_keyworlds(vars_dict, pars) configs = pars.split(r'@') host = configs[0].encode() user = configs[1] passwd = configs[2] finish = configs[3] commands = configs[4].split(r'\n') th1 = threading.Thread(target=do_telnet, args=(host.encode('utf-8'), user.encode('utf-8'), passwd.encode('utf-8'), finish.encode('utf-8'), commands)) th1.start() th1.join(20) ##20秒超時時間
還有一個需要注意的是,傳遞給Telnet方法的字符串都會被解一次碼,所以如果你傳遞過去需要write的字符串是已經解碼的unicode的話,那么就會報錯的,所以在傳遞發送的字符串之前還是先編成utf-8為妥,其它字符不知道支持不,我只試了utf-8,也沒看源碼。
此外,貌似還有一個pexpect的第三方模塊可以支持telnet等一系列的協議連接,並支持交互式的通信,只是這個模塊夠用就沒學習了,這里先備注一下。