Pythonを使用してSSHを使用する最も簡単な方法は何ですか?


82

ローカルのPython(3.0)スクリプトからリモートサーバーにSSHで接続し、ログイン/パスワードを入力し、コマンドを実行して、出力をPythonコンソールに出力するにはどうすればよいですか?

大規模な外部ライブラリを使用したり、リモートサーバーに何かをインストールしたりしたくありません。

回答:


42

試したことはありませんが、このpysftpモジュールが役立つ可能性があります。このモジュールはparamikoを使用します。私はすべてがクライアントサイドであると信じています。

興味深いコマンドは、おそらく.execute()リモートマシン上で任意のコマンドを実行するコマンドです。(このモジュールは、FTP文字をより暗示する機能.get().putメソッドも備えています)。

更新:

最初にリンクしたブログ投稿が利用できなくなった後、回答を書き直しました。この回答の古いバージョンを参照するコメントの一部は、奇妙に見えるようになります。


良い発見!エラー応答のカスタマイズを気にしない限り、この追加の抽象化は非常に役立ちます。
Cascabel

sshモジュールがそのトリックを実行しました。シンプルで問題なく動作します。ParamikoAPIを検索する必要はありません。
Christopher Tokar

2
あなたが与えるリンク内のssh.pyファイルへのリンクが壊れています:/
dgorissen 2011年

はい、新しいリンクをお願いします。githubでssh.pyを見つけましたが、同じではありません(そして、それほど良くはありません)
joedborg 2011

1
pysftpパッケージはSFTPのみを提供します。SSHクライアントからはほど遠い。
bortzmeyer 2011年

61

上記のように、Paramikoを使用して自分でコーディングできます。または、質問したすべてのことを実行するためのPythonアプリケーションであるFabricを調べることもできます。

Fabricは、SSHプロトコルを介してアプリケーションのデプロイやシステム管理タスクの実行を合理化するように設計されたPythonライブラリおよびコマンドラインツールです。これは、任意のシェルコマンドを(通常のログインユーザーとして、またはsudoを介して)実行したり、ファイルをアップロードおよびダウンロードしたりするためのツールを提供します。

これはあなたのニーズに合っていると思います。また、大規模なライブラリではなく、サーバーのインストールは必要ありませんが、クライアントへのインストールが必要なparamikoおよびpycryptに依存しています。

アプリはここにありました。これで、ここにあります

* The official, canonical repository is git.fabfile.org
* The official Github mirror is GitHub/bitprophet/fabric

いくつかの良い記事がありますが、過去6か月で変更されているため、注意が必要です。

ファブリックを使用したDjangoのデプロイ

現代のPythonハッカーのツール:Virtualenv、Fabric、Pip

FabricとVirtualenvを使用したシンプルで簡単な導入


後で:ファブリックをインストールするためにparamikoが不要になりました:

$ pip install fabric
Downloading/unpacking fabric
  Downloading Fabric-1.4.2.tar.gz (182Kb): 182Kb downloaded
  Running setup.py egg_info for package fabric
    warning: no previously-included files matching '*' found under directory 'docs/_build'
    warning: no files found matching 'fabfile.py'
Downloading/unpacking ssh>=1.7.14 (from fabric)
  Downloading ssh-1.7.14.tar.gz (794Kb): 794Kb downloaded
  Running setup.py egg_info for package ssh
Downloading/unpacking pycrypto>=2.1,!=2.4 (from ssh>=1.7.14->fabric)
  Downloading pycrypto-2.6.tar.gz (443Kb): 443Kb downloaded
  Running setup.py egg_info for package pycrypto
Installing collected packages: fabric, ssh, pycrypto
  Running setup.py install for fabric
    warning: no previously-included files matching '*' found under directory 'docs/_build'
    warning: no files found matching 'fabfile.py'
    Installing fab script to /home/hbrown/.virtualenvs/fabric-test/bin
  Running setup.py install for ssh
  Running setup.py install for pycrypto
...
Successfully installed fabric ssh pycrypto
Cleaning up...

ただし、これはほとんど表面的なものです。sshはparamikoのフォークであり、両方のライブラリのメンテナは同じであり(Jeff Forcier、Fabricの作成者でもあります)、メンテナはparamikoとsshをparamikoという名前で再結合する予定です。(pbankaによるこの修正。)


これは興味深いリンクのようですので、あなたのリンクが壊れているので更新したいと思います。使用してください:clemesha.org/blog/...を
dlewin

質問者は、「大規模な外部ライブラリ」を使用したくないと指定しませんでしたか?ParamikoとFabricはどちらも、作者が本当に求めているのが単純な1回限りのsshレシピである場合、やり過ぎです。
ゾランパヴロヴィッチ2012

1
@Zoran Pavlovic:すべての答えは、ローカルパッケージ(paramiko、fabric、ssh、libssh2)をインストールするか、サブプロセスを使用してsshを実行することでした。後者はインストールなしのソリューションですが、sshを生成することは素晴らしいアイデアではないと思います。また、彼がsshモジュールをインストールするための答えを選択したため、OPもそうしませんでした。それらのドキュメントは次のように述べています。「ssh.pyは、get、put、executeの3つの一般的なSSH操作を提供します。これはParamikoの高レベルの抽象化です。」したがって、コーディングに重点を置いているlibssh2を好まない限り、準拠する推奨事項はありません。OPの条件を合理的に満たすことができない場合は、適切な解決策を提供することをお勧めします。
hughdbrown 2012

24

余分なモジュールを避けたい場合は、サブプロセスモジュールを使用して実行できます

ssh [host] [command]

出力をキャプチャします。

次のようなものを試してください:

process = subprocess.Popen("ssh example.com ls", shell=True,
    stdout=subprocess.PIPE, stderr=subprocess.STDOUT)
output,stderr = process.communicate()
status = process.poll()
print output

ユーザー名とパスワードを処理するには、サブプロセスを使用してsshプロセスと対話するか、サーバーに公開鍵をインストールしてパスワードプロンプトを回避することができます。


7
しかし、クライアントがWindows上にある場合はどうなるでしょうか?
ネイサン

sshパイプを介してサブプロセスにパスワードを提供するのは難しい場合があります。パイプ(popen())を使用しないのなぜですか?を参照してください。これを回避するにはptypexpectモジュールが必要になる場合があります。
jfs 2014

文字列 'sshsomecomputerでは機能しないようです。python -c "import numpy; print numpy .__ version__" 'コマンド "import"がわからないと表示されます
usethedeathstar 2014

1
@usethedeathstar:リモートコマンド全体を引用符で囲みます:ssh somecomputer'python -c "import this; print this" '
Neil

17

私が書いたlibssh2のためのPythonバインディングを。Libssh2は、SSH2プロトコルを実装するクライアント側ライブラリです。

import socket
import libssh2

sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
sock.connect(('exmaple.com', 22))

session = libssh2.Session()
session.startup(sock)
session.userauth_password('john', '******')

channel = session.channel()
channel.execute('ls -l')

print channel.read(1024)

2
非常に低レベルのようです。たとえば(独自の例)、IPv4またはIPv6(OpenSSHコマンドラインクライアントとは関係ありません)を使用していることを明示的に指定する必要があります。また、ssh-agentで動作させる方法が見つかりませんでした。
bortzmeyer 2011年

2
pylibssh2の良いところは、paramikoのようなsshのネイティブPython実装よりもはるかに高速にファイルを転送することです。
ダミアン

8

ここでは、「最も単純な」の定義が重要です。単純なコードとは、モジュールを使用することを意味します(ただし、「大規模な外部ライブラリ」は誇張です)。

最新の(活発に開発された)モジュールはparamikoだと思います。ダウンロードにはデモスクリプトが付属しており、詳細なオンラインAPIドキュメントが含まれています。pexpectに含まれているPxSSHを試すこともできます。最初のリンクにドキュメントと一緒に短いサンプルがあります。

繰り返しますが、単純さに関しては、優れたエラー検出は常にコードをより複雑に見せることになりますが、サンプルスクリプトから多くのコードを再利用して、それを忘れることができるはずです。


6

hughdbrownのように、私はFabricが好きです。独自の宣言型スクリプト(デプロイなどを作成するため)を実装する一方で、Pythonモジュールとしてインポートして、Fabricスクリプトを記述せずにプログラムで使用することもできることに注意してください。

ファブリックには新しいメンテナーがあり、書き直されている最中です。つまり、(現在)Webで見つかるほとんどのチュートリアルは、現在のバージョンでは機能しません。また、Googleはまだ最初の結果として古いFabricページを表示します。

最新のドキュメントについては、http//docs.fabfile.orgを確認してください。


ファブリックはparamikoのフォーク使用していますpypi.python.org/pypi/sshすべてのSSHもののために。
ダミアン

6

paramikoは少し低レベルであり、Fabricはライブラリとしての使用には特に適していないことがわかったので、paramikoを使用して少し優れたインターフェイスを実装するspurという独自のライブラリをまとめました。

import spur

shell = spur.SshShell(hostname="localhost", username="bob", password="password1")
result = shell.run(["echo", "-n", "hello"])
print result.output # prints hello

プログラムの実行中にプログラムの出力を出力することもできます。これは、プログラムが終了する前に長時間実行されるコマンドの出力を確認する場合に便利です。

result = shell.run(["echo", "-n", "hello"], stdout=sys.stdout)

非標準コマンドの実行をサポートしていません。たとえば、一部のルーター(MikroTik)では、コマンドの前に「/」が付いているため、このライブラリはエラーをスローします。ただし、標準のLinuxホストの場合は、かなり良いようです。
Babken Vardanyan 2014

私は、ホスト名にIPアドレスを渡すと、それはIPがknown_hostsファイルに見つかりませんでしたというエラーがスローされます...
rexbelia

@rexbeliaこれはSSHの通常の動作です。正しいサーバーと通信していることを確認するために、SSHはホストからのキーがすでにわかっている場合にのみ受け入れます。解決策は、ドキュメントで説明されているように、関連するキーをknown_hostsに追加するか、missing_host_key引数を適切な値に設定することです。
Michael Williamson

4

ここに到達した人のために、pythonsshサンプルをグーグルで検索します。元の質問と回答は、今ではほとんどデコードされています。paramikoには少し機能が備わっているようです(わかりました。ここでは純粋に推測します。Pythonは初めてです)。paramikoを使用してsshクライアントを直接作成できます。

import base64
import paramiko

client = paramiko.SSHClient()

client.set_missing_host_key_policy(paramiko.AutoAddPolicy())
client.connect('192.168.1.1', username='user', password='password')
stdin, stdout, stderr = client.exec_command('cat /proc/meminfo')
for line in stdout:
    print('... ' + line.strip('\n'))
client.close()

このコードはhttps://github.com/paramiko/paramikoのデモから採用され ました。


1

これは私のために働いた

import subprocess
import sys
HOST="IP"
COMMAND="ifconfig"

def passwordless_ssh(HOST):
        ssh = subprocess.Popen(["ssh", "%s" % HOST, COMMAND],
                       shell=False,
                       stdout=subprocess.PIPE,
                       stderr=subprocess.PIPE)
        result = ssh.stdout.readlines()
        if result == []:
                error = ssh.stderr.readlines()
                print >>sys.stderr, "ERROR: %s" % error
                return "error"
        else:
                return result

1

please refer to paramiko.org, its very useful while doing ssh using python.

import paramiko

import time

ssh = paramiko.SSHClient() #SSHClient() is the paramiko object</n>

#Below lines adds the server key automatically to know_hosts file.use anyone one of the below

ssh.load_system_host_keys() 

ssh.set_missing_host_key_policy(paramiko.AutoAddPolicy())

try:

#Here we are actually connecting to the server.

ssh.connect('10.106.104.24', port=22, username='admin', password='')

time.sleep(5)

#I have mentioned time because some servers or endpoint prints there own information after 
#loggin in e.g. the version, model and uptime information, so its better to give some time 
#before executing the command.

#Here we execute the command, stdin for input, stdout for output, stderr for error

stdin, stdout, stderr = ssh.exec_command('xstatus Time')

#Here we are reading the lines from output.

output = stdout.readlines() 

print(output)


#Below all are the Exception handled by paramiko while ssh. Refer to paramiko.org for more information about exception.


except (BadHostKeyException, AuthenticationException,  
    SSHException, socket.error) as e:           

print(e)

0

リモートマシンを管理し、ファイル操作を実行するために開発したspurparamikoのラッパーであるspurplusをご覧ください

Spurplusは、すぐに使用できるcheck_output()機能を提供します。

import spurplus
with spurplus.connect_with_retries(
        hostname='some-machine.example.com', username='devop') as shell:
     out = shell.check_output(['/path/to/the/command', '--some_argument']) 
     print(out)
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.