一:簡介
paramiko是用python語言寫的一個模塊,遵循SSH2協議,支持以加密和認證的方式,進行遠程服務器的連接。
由於使用的是python這樣的能夠跨平台運行的語言,所以所有python支持的平台,如Linux, Solaris, BSD, MacOS X, Windows等,paramiko都可以支持,因此,如果需要使用SSH從一個平台連接到另外一個平台,進行一系列的操作時,paramiko是最佳工具之一。
舉個常見的例子,現有這樣的需求:需要使用windows客戶端,遠程連接到Linux服務器,查看上面的日志狀態,大家通常使用的方法會是:
1:用telnet
2:用PUTTY
3:用WinSCP
4:用XManager等…
那現在如果需求又增加一條,要從服務器上下載文件,該怎么辦?那常用的辦法可能會是:
1:Linux上安裝FTP並配置
2:Linux上安裝Sambe並配置…
大家會發現,常見的解決方法都會需要對遠程服務器必要的配置,如果遠程服務器只有一兩台還好說,如果有N台,還需要逐台進行配置,或者需要使用代碼進行以上操作時,上面的辦法就不太方便了。
使用paramiko可以很好的解決以上問題,比起前面的方法,它僅需要在本地上安裝相應的軟件(python以及PyCrypto),對遠程服務器沒有配置要求,對於連接多台服務器,進行復雜的連接操作特別有幫助。
二:安裝
安裝paramiko有兩個先決條件,python和另外一個名為PyCrypto的模塊。
通常安裝標准的python模塊,只需要在模塊的根目錄下運行:
python setup.py build
python setup.py install
備注:安裝前先檢查是否安裝gcc(yum -y install gcc)
2.1 PyCrypto安裝
wget http://ftp.dlitz.net/pub/dlitz/crypto/pycrypto/pycrypto-2.6.tar.gz
tar -zxvf pycrypto-2.6.tar.gz
cd pycrypto-2.6/
python setup.py build && python setup.py install
測試:
python>> import Crypto
(編譯時報錯:error: command 'gcc' failed with exit status 1;這是因為缺少python-dev的軟件包,所yum -y install python-devel)
2.2 paramiko安裝
wget http://www.lag.net/paramiko/download/paramiko-1.7.7.1.tar.gz
tar xvzf paramiko-1.7.7.1.tar.gz
cd paramiko-1.7.7.1/
python setup.py build && python setup.py install
Crypto error: 'module' object has no attribute 'HAVE_DECL_MPZ_POWM_SEC'
測試:
python>> import paramiko
(Crypto error: 'module' object has no attribute 'HAVE_DECL_MPZ_POWM_SEC'
找到 /usr/lib/python2.7/site-packages/Crypto/Util/number.py
把if _fastmath is not None and not _fastmath.HAVE_DECL_MPZ_POWM_SEC:
注釋了
#if _fastmath is not None and not _fastmath.HAVE_DECL_MPZ_POWM_SEC:
)
三: 使用paramiko
#設置ssh連接的遠程主機地址和端口
t=paramiko.Transport((ip,port))
#設置登錄名和密碼
t.connect(username=username,password=password)
#連接成功后打開一個channel
chan=t.open_session()
#設置會話超時時間
chan.settimeout(session_timeout)
#打開遠程的terminal
chan.get_pty()
#激活terminal
chan.invoke_shell()
然后就可以通過chan.send('command')和chan.recv(recv_buffer)來遠程執行命令以及本地獲取反饋。
paramiko有兩個模塊SSHClient()和SFTPClient()
利用SSHClient()

1 #coding:utf-8 2 import paramiko 3 4 #創建SSH對象 5 ssh = paramiko.SSHClient() 6 # 允許連接不在know_hosts文件中的主機 7 ssh.set_missing_host_key_policy(paramiko.AutoAddPolicy()) 8 # 連接服務器 9 ssh.connect(hostname='192.168.2.103', port=22, username='root', password='123456') 10 11 # 執行命令 12 stdin, stdout, stderr = ssh.exec_command('ls') 13 # 獲取命令結果 14 result = stdout.read() 15 print (str(result,encoding='utf-8')) 16 # 關閉連接 17 ssh.close()
SSHClient()里面有一個transport變量,這個是用於獲取連接的,因此我們也可以單獨的獲取到transport變量,然后執行連接操作

1 #coding:utf-8 2 import paramiko 3 4 transport = paramiko.Transport(('192.168.2.103', 22)) 5 transport.connect(username='root', password='123456') 6 7 ssh = paramiko.SSHClient() 8 ssh._transport = transport 9 10 stdin, stdout, stderr = ssh.exec_command('df') 11 print (str(stdout.read(),encoding='utf-8')) 12 13 transport.close()
用transport實現上傳下載以及命令的執行:

1 #coding:utf-8 2 import paramiko 3 import uuid 4 5 class SSHConnection(object): 6 7 def __init__(self, host='192.168.2.103', port=22, username='root',pwd='123456'): 8 self.host = host 9 self.port = port 10 self.username = username 11 self.pwd = pwd 12 self.__k = None 13 14 def connect(self): 15 transport = paramiko.Transport((self.host,self.port)) 16 transport.connect(username=self.username,password=self.pwd) 17 self.__transport = transport 18 19 def close(self): 20 self.__transport.close() 21 22 def upload(self,local_path,target_path): 23 # 連接,上傳 24 # file_name = self.create_file() 25 sftp = paramiko.SFTPClient.from_transport(self.__transport) 26 # 將location.py 上傳至服務器 /tmp/test.py 27 sftp.put(local_path, target_path) 28 29 def download(self,remote_path,local_path): 30 sftp = paramiko.SFTPClient.from_transport(self.__transport) 31 sftp.get(remote_path,local_path) 32 33 def cmd(self, command): 34 ssh = paramiko.SSHClient() 35 ssh._transport = self.__transport 36 # 執行命令 37 stdin, stdout, stderr = ssh.exec_command(command) 38 # 獲取命令結果 39 result = stdout.read() 40 print (str(result,encoding='utf-8')) 41 return result 42 43 ssh = SSHConnection() 44 ssh.connect() 45 ssh.cmd("ls") 46 ssh.upload('s1.py','/tmp/ks77.py') 47 ssh.download('/tmp/test.py','kkkk',) 48 ssh.cmd("df") 49 ssh.close()
四,與linux連接
下面是兩種使用paramiko連接到linux服務器的代碼
方式一:
1 ssh = paramiko.SSHClient() 2 ssh.set_missing_host_key_policy(paramiko.AutoAddPolicy()) 3 ssh.connect("某IP地址",22,"用戶名", "口令")
方式二:
1 t = paramiko.Transport((“主機”,”端口”)) 2 t.connect(username = “用戶名”, password = “口令”)
SFTPClient()也是使用transport來實現的,因此如果有需求需要執行命令和上傳文件糅合在一起的話,那么就需要使用transport的方式來實現。
如果連接遠程主機需要提供密鑰,上面第二行代碼可改成:
t.connect(username = “用戶名”, password = “口令”, hostkey=”密鑰”)
3.1 windows對linux運行任意命令,並將結果輸出
如果linux服務器開放了22端口,在windows端,我們可以使用paramiko遠程連接到該服務器,並執行任意命令,然后通過 print或其它方式得到該結果,
代碼如下

1 #coding:Utf8 2 3 import paramiko 4 5 ssh = paramiko.SSHClient() 6 ssh.set_missing_host_key_policy(paramiko.AutoAddPolicy()) 7 ssh.connect("某IP地址",22,"用戶名", "口令") 8 stdin, stdout, stderr = ssh.exec_command("你的命令") 9 print stdout.readlines() 10 ssh.close()
其中的”你的命令”可以任意linux支持的命令
3.2 從widnows端下載linux服務器上的文件

1 coding:utf8 2 3 import paramiko 4 5 t = paramiko.Transport((“主機”,”端口”)) 6 t.connect(username = “用戶名”, password = “口令”) 7 sftp = paramiko.SFTPClient.from_transport(t) 8 remotepath=’/var/log/system.log’ 9 localpath=’/tmp/system.log’ 10 sftp.get(remotepath, localpath) 11 t.close()
3.3 從widnows端上傳文件到linux服務器

1 import paramiko 2 3 t = paramiko.Transport((“主機”,”端口”)) 4 t.connect(username = “用戶名”, password = “口令”) 5 sftp = paramiko.SFTPClient.from_transport(t) 6 remotepath=’/var/log/system.log’ 7 localpath=’/tmp/system.log’ 8 sftp.put(localpath,remotepath) 9 t.close()
3.4 在Linux上安裝paramiko模塊
安裝scrapy這個應用中遇到的問題
c/_cffi_backend.c:2:20: fatal error: Python.h: No such file or directory
sudo apt-get install python-dev
c/_cffi_backend.c:13:17: fatal error: ffi.h: No such file or directory
1 sudo apt-get install libffi-dev
* make sure the development packages of libxml2 and libxslt are installed *
1 sudo apt-get install libxslt1-dev
1.下載安裝wget http://peak.telecommunity.com/dist/ez_setup.py
2.python ez_setup.py
3.easy_install paramiko