您好,登錄后才能下訂單哦!
小編這次要給大家分享的是paramiko執(zhí)行命令超時怎么辦,文章內(nèi)容豐富,感興趣的小伙伴可以來了解一下,希望大家閱讀完這篇文章之后能夠有所收獲。
問題:paramiko遠(yuǎn)程執(zhí)行命令,需要等到命令返回信息,如果命令執(zhí)行時間比較長,返回信息就需要等很久
方案:
1、使用nohup + 待執(zhí)行命令 + & ,使用后臺執(zhí)行的方式,應(yīng)該可以快速返回
2、設(shè)置paramiko的執(zhí)行命令等待超時時間
stdin, stdout, stderr = self.client.exec_command(cmd,timeout=10,get_pty=True)
其實上面的兩種方案都不可行:方案1,需要優(yōu)化,下面這種直接調(diào)用的方式會導(dǎo)致test.sh啟動不起來
stdin, stdout, stderr = self.client.exec_command(‘sh ~/test.sh &',get_pty=True)
方案2,對于需要等待很久的命令,如果timeout小于程序的執(zhí)行時間,還是會失敗,命令執(zhí)行失敗
最終的解決方案來自參考1
把執(zhí)行很久的sh文件A放入另一個sh文件B中,然后paramiko執(zhí)行文件B。摘錄原文:
paramiko遠(yuǎn)程執(zhí)行后臺腳本“阻塞”問題
我寫的遠(yuǎn)程命令通道上線之后,發(fā)現(xiàn)在遠(yuǎn)程腳本中后臺再執(zhí)行另一個腳本,通道會一直等待后臺腳本執(zhí)行完成才會返回,有時甚至?xí)┧馈?/p>
1、復(fù)現(xiàn)過程如下:
①、編寫測試腳本
腳本1:test.sh
#!/bin/bash sleep 30 echo test end exit 0
腳本2:run.sh
#!/bin/bash bash /tmp/test.sh & echo run ok! exit 0
腳本3:test.py
import paramiko client = paramiko.SSHClient() client = paramiko.SSHClient() client.set_missing_host_key_policy(paramiko.AutoAddPolicy()) client.connect(hostname='192.168.1.10', port=22, username='root', password='123456', timeout=300,allow_agent=False,look_for_keys=False) stdin,stdout,stderr=client.exec_command("bash /tmp/run.sh") result_info = "" for line in stdout.readlines(): result_info += line print result_info
將test.sh和run.sh傳到遠(yuǎn)程服務(wù)器上,比如放到192.168.1.10:/tmp/下。
②、發(fā)起遠(yuǎn)程執(zhí)行
在本地執(zhí)行 python test.py,會發(fā)現(xiàn)整個腳本不會立即打印run ok,而是等30s之后才打印包括test.sh的所有輸出信息。
2、解決辦法
將遠(yuǎn)程腳本的標(biāo)準(zhǔn)輸出stdout重定向到錯誤輸出stderr即可,test.py 修改如下:
import paramiko client = paramiko.SSHClient() client = paramiko.SSHClient() client.set_missing_host_key_policy(paramiko.AutoAddPolicy()) client.connect(hostname='192.168.1.10', port=22, username='root', password='123456', timeout=300,allow_agent=False,look_for_keys=False) stdin,stdout,stderr=client.exec_command("bash /tmp/run.sh 1>&2") result_info = "" for line in stderr.readlines(): result_info += line print result_info
現(xiàn)在執(zhí)行,就能立即得到結(jié)果了。其實原因很簡單,因為bash /tmp/test.sh & 雖然是后臺執(zhí)行,但是依然會產(chǎn)生標(biāo)準(zhǔn)輸出,一旦產(chǎn)生標(biāo)準(zhǔn)輸出,paramiko就會認(rèn)為命令還未執(zhí)行完成,且stdout的buffer大于stderr,因此產(chǎn)生等待問題。
這里只要將腳本執(zhí)行的標(biāo)準(zhǔn)輸出重定向到錯誤輸出(1>&2),然后paramiko就可以使用stderr快速讀取遠(yuǎn)程打屏信息了。
基于上面paramiko的原理:有stdout輸出,就認(rèn)為命令沒有執(zhí)行完成。得出下面的解決方案,對于需要執(zhí)行很久test.sh,我們首先把標(biāo)準(zhǔn)輸出都發(fā)給標(biāo)準(zhǔn)錯誤輸出(1>&2),然后后臺啟動(&)
stdin, stdout, stderr = self.client.exec_command(‘bash ~/test.sh 1>&2 &',get_pty=True)
項目實踐,還有下面的方案:去掉參數(shù)get_pty,這樣就不會回傳標(biāo)準(zhǔn)輸出信息和標(biāo)準(zhǔn)錯誤信息
self.client.exec_command(‘bash ~/test.sh')
看完這篇關(guān)于paramiko執(zhí)行命令超時怎么辦的文章,如果覺得文章內(nèi)容寫得不錯的話,可以把它分享出去給更多人看到。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。