web-dev-qa-db-ja.com

PythonでのSFTP? (プラットフォームに依存しない)

私は、パスワードもハードコードされたハードコードされた場所にファイルを転送する簡単なツールに取り組んでいます。私はpython初心者ですが、ftplibのおかげで簡単でした:

import ftplib

info= ('someuser', 'password')    #hard-coded

def putfile(file, site, dir, user=(), verbose=True):
    """
    upload a file by ftp to a site/directory
    login hard-coded, binary transfer
    """
    if verbose: print 'Uploading', file
    local = open(file, 'rb')    
    remote = ftplib.FTP(site)   
    remote.login(*user)         
    remote.cwd(dir)
    remote.storbinary('STOR ' + file, local, 1024)
    remote.quit()
    local.close()
    if verbose: print 'Upload done.'

if __== '__main__':
    site = 'somewhere.com'            #hard-coded
    dir = './uploads/'                #hard-coded
    import sys, getpass
    putfile(sys.argv[1], site, dir, user=info)

問題は、sFTPをサポートするライブラリが見つからないことです。このようなことを安全に行うための通常の方法は何ですか?

編集:ここでの回答のおかげで、私はそれがParamikoで動作するようになり、これが構文でした。

import paramiko

Host = "THEHOST.com"                    #hard-coded
port = 22
transport = paramiko.Transport((Host, port))

password = "THEPASSWORD"                #hard-coded
username = "THEUSERNAME"                #hard-coded
transport.connect(username = username, password = password)

sftp = paramiko.SFTPClient.from_transport(transport)

import sys
path = './THETARGETDIRECTORY/' + sys.argv[1]    #hard-coded
localpath = sys.argv[1]
sftp.put(localpath, path)

sftp.close()
transport.close()
print 'Upload done.'

再度、感謝します!

166
Mark Wilbur

Paramiko はSFTPをサポートします。私はそれを使いました、そして、私はツイストを使いました。どちらにもそれぞれの場所がありますが、Paramikoから始める方が簡単な場合があります。

102
Brian Clapper

Pysftp https://pypi.python.org/pypi/pysftp を確認する必要があります。これはparamikoに依存しますが、ほとんどの一般的な使用例を数行のコードにラップします。

import pysftp
import sys

path = './THETARGETDIRECTORY/' + sys.argv[1]    #hard-coded
localpath = sys.argv[1]

Host = "THEHOST.com"                    #hard-coded
password = "THEPASSWORD"                #hard-coded
username = "THEUSERNAME"                #hard-coded

with pysftp.Connection(Host, username=username, password=password) as sftp:
    sftp.put(localpath, path)

print 'Upload done.'
62
Dundee MT

簡単でシンプルにしたい場合は、 Fabric もご覧ください。これは、RubyのCapistranoのような自動化された展開ツールですが、Pythonにとってはよりシンプルで、もちろんです。 Paramikoの上に構築されています。

「自動展開」をしたくないかもしれませんが、それでもファブリックは完全にユースケースに適合します。 Fabricのシンプルさを示すために、スクリプトのfabファイルとコマンドは次のようになります(テストされていませんが、99%は確実に機能します)。

fab_putfile.py:

from fabric.api import *

env.hosts = ['THEHOST.com']
env.user = 'THEUSER'
env.password = 'THEPASSWORD'

def put_file(file):
    put(file, './THETARGETDIRECTORY/') # it's copied into the target directory

次に、fabコマンドでファイルを実行します。

fab -f fab_putfile.py put_file:file=./path/to/my/file

これで完了です! :)

15
hopla

以下は、pysftpと秘密鍵を使用したサンプルです。

import pysftp

def upload_file(file_path):

    private_key = "~/.ssh/your-key.pem"  # can use password keyword in Connection instead
    srv = pysftp.Connection(Host="your-Host", username="user-name", private_key=private_key)
    srv.chdir('/var/web/public_files/media/uploads')  # change directory on remote server
    srv.put(file_path)  # To download a file, replace put with get
    srv.close()  # Close connection

pysftpは、paramikoとpycryptoを利用する使いやすいsftpモジュールです。 sftpへのシンプルなインターフェースを提供します。

data = srv.listdir()  # Get the directory and file listing in a list
srv.get(file_path)  # Download a file from remote server
srv.execute('pwd') # Execute a command on the server

その他のコマンドとPySFTPについて here

8
radtek

Twisted は、あなたがしていることを手助けし、ドキュメントをチェックします。たくさんの例があります。また、大きな開発者/ユーザーコミュニティが背後にある成熟した製品です。

3

RSAキーを使用して here を参照してください

スニペット:

import pysftp
import paramiko
from base64 import decodebytes

keydata = b"""L+WsiL5VL51ecJi3LVjmblkAdUTU+xbmXmUArIU5+8N6ua76jO/+T""" 
key = paramiko.RSAKey(data=decodebytes(keydata)) 
cnopts = pysftp.CnOpts()
cnopts.hostkeys.add(Host, 'ssh-rsa', key)


with pysftp.Connection(Host=host, username=username, password=password, cnopts=cnopts) as sftp:   
  with sftp.cd(directory):
    sftp.put(file_to_sent_to_ftp)
1
Abhijeet

パラミコはとても遅いです。サブプロセスとシェルを使用します。以下に例を示します。

remote_file_name = "filename"
remotedir = "/remote/dir"
localpath = "/local/file/dir"
    ftp_cmd_p = """
    #!/bin/sh
    lftp -u username,password sftp://ip:port <<EOF
    cd {remotedir}
    lcd {localpath}
    get {filename}
    EOF
    """
subprocess.call(ftp_cmd_p.format(remotedir=remotedir,
                                 localpath=localpath,
                                 filename=remote_file_name 
                                 ), 
                Shell=True, stdout=sys.stdout, stderr=sys.stderr)
1
杨李思

pexpect module を使用できます

これは良い紹介記事です

child = pexpect.spawn ('/usr/bin/sftp ' + [email protected] )
child.expect ('.* password:')
child.sendline (your_password)
child.expect ('sftp> ')
child.sendline ('dir')
child.expect ('sftp> ')
file_list = child.before
child.sendline ('bye')

私はこれをテストしていませんが、うまくいくはずです

1
MIkee

Pysftpに言及する回答がたくさんありますので、pysftpのコンテキストマネージャラッパーが必要な場合は、使用すると次のように見えるコードがさらに少ないソリューションがあります

path = "sftp://user:p@[email protected]/path/to/file.txt"

# Read a file
with open_sftp(path) as f:
    s = f.read() 
print s

# Write to a file
with open_sftp(path, mode='w') as f:
    f.write("Some content.") 

(完全な)例: http://www.prschmid.com/2016/09/simple-opensftp-context-manager-for.html

このコンテキストマネージャーでは、最初に接続できない場合に自動再試行ロジックが焼き付けられます(実稼働環境で予想以上に頻繁に発生します...)

open_sftpのコンテキストマネージャーの要旨: https://Gist.github.com/prschmid/80a19c22012e42d4d6e791c1e4eb8515

0
prschmid