Skip to main content

Asynchronous parallel SSH library

Project description

Non-blocking, asynchronous parallel SSH client library.

Run SSH commands over many - hundreds/hundreds of thousands - number of servers asynchronously and with minimal system load on the client host.

Native code based client with extremely high performance - based on libssh2 C library.

License Latest Version https://travis-ci.org/ParallelSSH/parallel-ssh.svg?branch=master https://ci.appveyor.com/api/projects/status/github/parallelssh/parallel-ssh?svg=true&branch=master https://codecov.io/gh/ParallelSSH/parallel-ssh/branch/master/graph/badge.svg https://img.shields.io/pypi/wheel/parallel-ssh.svg Latest documentation

Installation

pip install parallel-ssh

Usage Example

See documentation on read the docs for more complete examples.

Run uname on two remote hosts in parallel with sudo.

from __future__ import print_function

from pssh.pssh_client import ParallelSSHClient

hosts = ['myhost1', 'myhost2']
client = ParallelSSHClient(hosts)

output = client.run_command('uname')
for host, host_output in output.items():
    for line in host_output.stdout:
        print(line)
Output:
Linux
Linux

Native client

Starting from version 1.2.0, a new client is supported in parallel-ssh which offers much greater performance and reduced overhead than the current default client.

The new client is based on libssh2 via the ssh2-python extension library and supports non-blocking mode natively. Binary wheel packages with libssh2 included are provided for Linux, OSX and Windows platforms and all supported Python versions.

See this post for a performance comparison of the available clients.

To make use of this new client, ParallelSSHClient can be imported from pssh.clients.native instead. Their respective APIs are almost identical.

The new client will become the default and will replace the current pssh.pssh_client in a new major version of the library - 2.0.0.

The paramiko based client will become an optional install via pip extras, available under pssh.clients.miko.

For example:

from pprint import pprint
from pssh.clients.native import ParallelSSHClient

hosts = ['myhost1', 'myhost2']
client = ParallelSSHClient(hosts)

output = client.run_command('uname')
for host, host_output in output.items():
    for line in host_output.stdout:
        print(line)

See documentation for a feature comparison of the two clients.

Native Code Client Features

  • Highest performance and least overhead of any Python SSH libraries

  • Thread safe - makes use of native threads for blocking calls like authentication

  • Natively non-blocking utilising libssh2 via ssh2-python - no monkey patching of the Python standard library

  • Significantly reduced overhead in CPU and memory usage

Exit codes

Once either standard output is iterated on to completion, or client.join(output) is called, exit codes become available in host output. Iteration ends only when remote command has completed, though it may be interrupted and resumed at any point.

for host in output:
    print(output[host].exit_code)
Output:
0
0

The client’s join function can be used to wait for all commands in output object to finish:

client.join(output)

Similarly, output and exit codes are available after client.join is called:

from pprint import pprint

output = client.run_command('exit 0')

# Wait for commands to complete and gather exit codes.
# Output is updated in-place.
client.join(output)
pprint(output.values()[0].exit_code)

# Output remains available in output generators
for host, host_output in output.items():
    for line in host_output.stdout:
        pprint(line)
Output:
0
<..stdout..>

There is also a built in host logger that can be enabled to log output from remote hosts. The helper function pssh.utils.enable_host_logger will enable host logging to stdout.

To log output without having to iterate over output generators, the consume_output flag must be enabled - for example:

from pssh.utils import enable_host_logger

enable_host_logger()
client.join(client.run_command('uname'), consume_output=True)
Output:
[localhost]       Linux

SFTP

SFTP is supported natively.

To copy a local file to remote hosts in parallel:

from pssh.clients import ParallelSSHClient
from pssh.utils import enable_logger, logger
from gevent import joinall

enable_logger(logger)
hosts = ['myhost1', 'myhost2']
client = ParallelSSHClient(hosts)
cmds = client.copy_file('../test', 'test_dir/test')
joinall(cmds, raise_error=True)
Output:
Copied local file ../test to remote destination myhost1:test_dir/test
Copied local file ../test to remote destination myhost2:test_dir/test

There is similar capability to copy remote files to local ones suffixed with the host’s name with the copy_remote_file function.

Directory recursion is supported in both cases via the recurse parameter - defaults to off.

See SFTP documentation for more examples.

Design And Goals

parallel-ssh’s design goals and motivation are to provide a library for running non-blocking asynchronous SSH commands in parallel with little to no load induced on the system by doing so with the intended usage being completely programmatic and non-interactive.

To meet these goals, API driven solutions are preferred first and foremost. This frees up developers to drive the library via any method desired, be that environment variables, CI driven tasks, command line tools, existing OpenSSH or new configuration files, from within an application et al.

Comparison With Alternatives

There are not many alternatives for SSH libraries in Python. Of the few that do exist, here is how they compare with parallel-ssh.

As always, it is best to use a tool that is suited to the task at hand. parallel-ssh is a library for programmatic and non-interactive use - see Design And Goals. If requirements do not match what it provides then it best not be used. Same applies for the tools described below.

Paramiko

The default SSH client library in parallel-ssh 1.x.x series.

Pure Python code, while having native extensions as dependencies, with poor performance and numerous bugs compared to both OpenSSH binaries and the libssh2 based native clients in parallel-ssh 1.2.x and above. Recent versions have regressed in performance and have blocker issues.

It does not support non-blocking mode, so to make it non-blocking monkey patching must be used which affects all other uses of the Python standard library. However, some functionality like Kerberos (GSS-API) authentication is not provided by other libraries.

asyncssh

Python 3 only asyncio framework using client library. License (EPL) is not compatible with GPL, BSD or other open source licenses and combined works cannot be distributed.

Therefore unsuitable for use in many projects, including parallel-ssh.

Fabric

Port of Capistrano from Ruby to Python. Intended for command line use and is heavily systems administration oriented rather than non-interactive library. Same maintainer as Paramiko.

Uses Paramiko and suffers from the same limitations. More over, uses threads for parallelisation, while not being thread safe, and exhibits very poor performance and extremely high CPU usage even for limited number of hosts - 1 to 10 - with scaling limited to one core.

Library API is non-standard, poorly documented and with numerous issues as API use is not intended.

Ansible

A configuration management and automation tool that makes use of SSH remote commands. Uses, in parts, both Paramiko and OpenSSH binaries.

Similarly to Fabric, uses threads for parallelisation and suffers from the poor scaling that this model offers.

See The State of Python SSH Libraries for what to expect from scaling SSH with threads, as compared to non-blocking I/O with parallel-ssh.

Again similar to Fabric, its intended and documented use is interactive via command line rather than library API based. It may, however, be an option if Ansible is already being used for automation purposes with existing playbooks, the number of hosts is small, and when the use case is interactive via command line.

parallel-ssh is, on the other hand, a suitable option for Ansible as an SSH client that would improve its parallel SSH performance significantly.

ssh2-python

Wrapper to libssh2 C library. Used by parallel-ssh as of 1.2.0 and is by same author.

Does not do parallelisation out of the box but can be made parallel via Python’s threading library relatively easily and as it is a wrapper to a native library that releases Python’s GIL, can scale to multiple cores.

parallel-ssh uses ssh2-python in its native non-blocking mode with event loop and co-operative sockets provided by gevent for an extremely high performance library without the side-effects of monkey patching - see benchmarks.

In addition, parallel-ssh uses native threads to offload CPU blocked tasks like authentication in order to scale to multiple cores while still remaining non-blocking for network I/O.

pssh.ssh2_client.SSHClient is a single host natively non-blocking client for users that do not need parallel capabilities but still want a non-blocking client with native code performance.

Out of all the available Python SSH libraries, libssh2 and ssh2-python have been shown, see benchmarks above, to perform the best with the least resource utilisation and ironically for a native code extension the least amount of dependencies. Only libssh2 C library and its dependencies which are included in binary wheels.

However, it lacks support for some SSH features present elsewhere like ECDSA keys (PR pending), agent forwarding (PR also pending) and Kerberos authentication - see feature comparison.

Scaling

Some guide lines on scaling parallel-ssh and pool size numbers.

In general, long lived commands with little or no output gathering will scale better. Pool sizes in the multiple thousands have been used successfully with little CPU overhead in the single thread running them in these use cases.

Conversely, many short lived commands with output gathering will not scale as well. In this use case, smaller pool sizes in the hundreds are likely to perform better with regards to CPU overhead in the event loop.

Multiple Python native threads, each of which can get its own event loop, may be used to scale this use case further as number of CPU cores allows. Note that parallel-ssh imports must be done within the target function of the newly started thread for it to receive its own event loop. gevent.get_hub() may be used to confirm that the worker thread event loop differs from the main thread.

Gathering is highlighted here as output generation does not affect scaling. Only when output is gathered either over multiple still running commands, or while more commands are being triggered, is overhead increased.

Technical Details

To understand why this is, consider that in co-operative multi tasking, which is being used in this project via the gevent library, a co-routine (greenlet) needs to yield the event loop to allow others to execute - co-operation. When one co-routine is constantly grabbing the event loop in order to gather output, or when co-routines are constantly trying to start new short-lived commands, it causes contention with other co-routines that also want to use the event loop.

This manifests itself as increased CPU usage in the process running the event loop and reduced performance with regards to scaling improvements from increasing pool size.

On the other end of the spectrum, long lived remote commands that generate no output only need the event loop at the start, when they are establishing connections, and at the end, when they are finished and need to gather exit codes, which results in practically zero CPU overhead at any time other than start or end of command execution.

Output generation is done remotely and has no effect on the event loop until output is gathered - output buffers are iterated on. Only at that point does the event loop need to be held.

User’s group

There is a public ParallelSSH Google group setup for this purpose - both posting and viewing are open to the public.

https://ga-beacon.appspot.com/UA-9132694-7/parallel-ssh/README.rst?pixel

Project details


Release history Release notifications | RSS feed

Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

parallel-ssh-1.6.0.tar.gz (98.5 kB view details)

Uploaded Source

Built Distributions

parallel_ssh-1.6.0-cp36-cp36m-win_amd64.whl (1.3 MB view details)

Uploaded CPython 3.6m Windows x86-64

parallel_ssh-1.6.0-cp36-cp36m-win32.whl (955.2 kB view details)

Uploaded CPython 3.6m Windows x86

parallel_ssh-1.6.0-cp36-cp36m-manylinux1_x86_64.whl (1.6 MB view details)

Uploaded CPython 3.6m

parallel_ssh-1.6.0-cp35-cp35m-win_amd64.whl (1.3 MB view details)

Uploaded CPython 3.5m Windows x86-64

parallel_ssh-1.6.0-cp35-cp35m-win32.whl (954.9 kB view details)

Uploaded CPython 3.5m Windows x86

parallel_ssh-1.6.0-cp35-cp35m-manylinux1_x86_64.whl (1.6 MB view details)

Uploaded CPython 3.5m

parallel_ssh-1.6.0-cp34-cp34m-win_amd64.whl (1.3 MB view details)

Uploaded CPython 3.4m Windows x86-64

parallel_ssh-1.6.0-cp34-cp34m-win32.whl (956.5 kB view details)

Uploaded CPython 3.4m Windows x86

parallel_ssh-1.6.0-cp34-cp34m-manylinux1_x86_64.whl (1.6 MB view details)

Uploaded CPython 3.4m

parallel_ssh-1.6.0-cp27-cp27mu-manylinux1_x86_64.whl (1.6 MB view details)

Uploaded CPython 2.7mu

parallel_ssh-1.6.0-cp27-cp27m-win_amd64.whl (1.3 MB view details)

Uploaded CPython 2.7m Windows x86-64

parallel_ssh-1.6.0-cp27-cp27m-win32.whl (956.9 kB view details)

Uploaded CPython 2.7m Windows x86

parallel_ssh-1.6.0-cp27-cp27m-manylinux1_x86_64.whl (1.6 MB view details)

Uploaded CPython 2.7m

parallel_ssh-1.6.0-cp27-cp27m-macosx_10_13_x86_64.whl (1.3 MB view details)

Uploaded CPython 2.7m macOS 10.13+ x86-64

parallel_ssh-1.6.0-cp27-cp27m-macosx_10_12_x86_64.whl (1.3 MB view details)

Uploaded CPython 2.7m macOS 10.12+ x86-64

parallel_ssh-1.6.0-cp27-cp27m-macosx_10_11_x86_64.whl (1.3 MB view details)

Uploaded CPython 2.7m macOS 10.11+ x86-64

parallel_ssh-1.6.0-cp27-cp27m-macosx_10_10_intel.whl (1.3 MB view details)

Uploaded CPython 2.7m macOS 10.10+ intel

File details

Details for the file parallel-ssh-1.6.0.tar.gz.

File metadata

File hashes

Hashes for parallel-ssh-1.6.0.tar.gz
Algorithm Hash digest
SHA256 219398acb59112db0f56b7dfd7adddcf49523c1492f8a422c866304555b35c65
MD5 27f237eb65425580db3803b53663bfd8
BLAKE2b-256 daf782afd051b53cd90cd4594591e3fa6431b6bcedcf479bf242fa32822de4d5

See more details on using hashes here.

File details

Details for the file parallel_ssh-1.6.0-cp36-cp36m-win_amd64.whl.

File metadata

File hashes

Hashes for parallel_ssh-1.6.0-cp36-cp36m-win_amd64.whl
Algorithm Hash digest
SHA256 8713f3780464e75c9d78c62acdc7c4256c64022145ef2c5b6595b3b90594d50c
MD5 7581f61f9e34d6c9cc0dfbac9bc0d56c
BLAKE2b-256 8c5d4e7c7996a862579d0d9758e5f965d87dc53f2bebd7f24fd297fe978a4747

See more details on using hashes here.

File details

Details for the file parallel_ssh-1.6.0-cp36-cp36m-win32.whl.

File metadata

File hashes

Hashes for parallel_ssh-1.6.0-cp36-cp36m-win32.whl
Algorithm Hash digest
SHA256 6001f51029a9b0ca266b32c2494b0b0848c7c17aec99767c5ae1e6bd8d7d2b90
MD5 05a2f2915cc2cc4f9c781034398f4adf
BLAKE2b-256 629764b0f92ee8867a762938efecae8d30295b4e19bd7f2dccc572fd5569b03a

See more details on using hashes here.

File details

Details for the file parallel_ssh-1.6.0-cp36-cp36m-manylinux1_x86_64.whl.

File metadata

File hashes

Hashes for parallel_ssh-1.6.0-cp36-cp36m-manylinux1_x86_64.whl
Algorithm Hash digest
SHA256 e27492bf3a25659b90a00e93683dc4e35d3950d8d6e53e9d3a9311317b31faa3
MD5 6a16d027ddb7423777ba0194f742a678
BLAKE2b-256 cfe3be9c3776e1a250d8b73f618ce1ba6cc48d6b2241041671d8f634e894a955

See more details on using hashes here.

File details

Details for the file parallel_ssh-1.6.0-cp35-cp35m-win_amd64.whl.

File metadata

File hashes

Hashes for parallel_ssh-1.6.0-cp35-cp35m-win_amd64.whl
Algorithm Hash digest
SHA256 7d565973bb1b7cd415dfbca8022b28de090c00cca1a51d156ea6cb5b6614b3dd
MD5 2a0a43bf19fac86b2f0811176bd94bd5
BLAKE2b-256 93d284795745a3d4bd0e6466536996cd3259db7ba432c70fc6f29757c101ae8e

See more details on using hashes here.

File details

Details for the file parallel_ssh-1.6.0-cp35-cp35m-win32.whl.

File metadata

File hashes

Hashes for parallel_ssh-1.6.0-cp35-cp35m-win32.whl
Algorithm Hash digest
SHA256 ab2c35937841888f4540a532507e216472d9397ffbb2a1c4f44e54de4b1e945f
MD5 de4599899bd41960ab68cf58455d50b6
BLAKE2b-256 1a22e30d3e8de9ff926013a62b8c774cbe71ef9138072533c16e1cf52d7132ea

See more details on using hashes here.

File details

Details for the file parallel_ssh-1.6.0-cp35-cp35m-manylinux1_x86_64.whl.

File metadata

File hashes

Hashes for parallel_ssh-1.6.0-cp35-cp35m-manylinux1_x86_64.whl
Algorithm Hash digest
SHA256 72ae426b6eed3f6f0a24b59a75acac5017c51e3543edddbfda598002b3ea3622
MD5 d2933a7cba3d4f5f8b76c64825e3162e
BLAKE2b-256 21c829e064e4d2fe3d602835df938dd000332dbccca3ee40c89900785e201320

See more details on using hashes here.

File details

Details for the file parallel_ssh-1.6.0-cp34-cp34m-win_amd64.whl.

File metadata

File hashes

Hashes for parallel_ssh-1.6.0-cp34-cp34m-win_amd64.whl
Algorithm Hash digest
SHA256 0603e03f6e8c12d92c99ff4f3c5df0aa6f2a46719adb5eb33e421aa5c5849938
MD5 fc2980ad2f0eb7adac18f09979f8794d
BLAKE2b-256 7aaf18dc98c00a3946eca69ee6a8766a4312131b228735b377af98704ba21bfa

See more details on using hashes here.

File details

Details for the file parallel_ssh-1.6.0-cp34-cp34m-win32.whl.

File metadata

File hashes

Hashes for parallel_ssh-1.6.0-cp34-cp34m-win32.whl
Algorithm Hash digest
SHA256 502e1086c49ece299a6da3cec9db830caa611502dfe472dfe77659328b580521
MD5 5e4e1ad80beea973386ff499d4fdf012
BLAKE2b-256 d3388c19fe21515ba173af9f9b032eb30b191d429f02707efea6ea74a45f9e65

See more details on using hashes here.

File details

Details for the file parallel_ssh-1.6.0-cp34-cp34m-manylinux1_x86_64.whl.

File metadata

File hashes

Hashes for parallel_ssh-1.6.0-cp34-cp34m-manylinux1_x86_64.whl
Algorithm Hash digest
SHA256 c1c0e255066bdbed01273dbe6eb873c221e3c64c8dd096a7a306db89428a3c4b
MD5 609f17d0783bef7f737bbcc354a7c619
BLAKE2b-256 6ce76079adef396f1ca931035b39a7b2d7774d0c6c9be6d64d07e0b277256001

See more details on using hashes here.

File details

Details for the file parallel_ssh-1.6.0-cp27-cp27mu-manylinux1_x86_64.whl.

File metadata

File hashes

Hashes for parallel_ssh-1.6.0-cp27-cp27mu-manylinux1_x86_64.whl
Algorithm Hash digest
SHA256 b3f2d037e65256fcb5e1a8df687e57f9ebeb1e55546db544f17938928f278373
MD5 2a2f673d066f53e17761701984c8e3c5
BLAKE2b-256 9e6b6017e699aa17d0127af7ffd1f45e0ce3e33d58ec8d798aec1ae8ea11a5c6

See more details on using hashes here.

File details

Details for the file parallel_ssh-1.6.0-cp27-cp27m-win_amd64.whl.

File metadata

File hashes

Hashes for parallel_ssh-1.6.0-cp27-cp27m-win_amd64.whl
Algorithm Hash digest
SHA256 2414d96ee0c9c5e8b0c2b911e2bb517c3a00c3f901af812638a449cfddf59785
MD5 9efd4d5b40321a3b871f63280ced596c
BLAKE2b-256 62164bd76a8c1fbdb7047afc255f4aa113a16ee51646e1f82d0de02512dcac35

See more details on using hashes here.

File details

Details for the file parallel_ssh-1.6.0-cp27-cp27m-win32.whl.

File metadata

File hashes

Hashes for parallel_ssh-1.6.0-cp27-cp27m-win32.whl
Algorithm Hash digest
SHA256 4143b863b564a23ed4194ea7f6dbeabd80d0f1f4b0ba1008b5782759a373e744
MD5 a94d68f5f186cd9f66ef83a4449f2b90
BLAKE2b-256 823ef73ded0238f429d16efa78bdbb1eb2a87bc25d8a94e2da80f123deb1ee23

See more details on using hashes here.

File details

Details for the file parallel_ssh-1.6.0-cp27-cp27m-manylinux1_x86_64.whl.

File metadata

File hashes

Hashes for parallel_ssh-1.6.0-cp27-cp27m-manylinux1_x86_64.whl
Algorithm Hash digest
SHA256 c2676a58394bac3dd46bd5f7780861c9488947e81b450a040622825746398906
MD5 e20c08c040762fe3dd7836c9ec9088e5
BLAKE2b-256 06e9c4f561fa55977a5acff00fd48fa0edb7a199d58e3454b1528250e708e5ca

See more details on using hashes here.

File details

Details for the file parallel_ssh-1.6.0-cp27-cp27m-macosx_10_13_x86_64.whl.

File metadata

File hashes

Hashes for parallel_ssh-1.6.0-cp27-cp27m-macosx_10_13_x86_64.whl
Algorithm Hash digest
SHA256 093456b90afddef25d571fc4c731b18ab539886281c8806eb5f84073435d498a
MD5 8da1e9c385de259187b97e689f3f7c3f
BLAKE2b-256 37e631d10f090505acebfac2a820ee5f3614fcfa56b9673917f51cec4b53b220

See more details on using hashes here.

File details

Details for the file parallel_ssh-1.6.0-cp27-cp27m-macosx_10_12_x86_64.whl.

File metadata

File hashes

Hashes for parallel_ssh-1.6.0-cp27-cp27m-macosx_10_12_x86_64.whl
Algorithm Hash digest
SHA256 ff4ffb6002a7d98ae416f282dc42acc880140ff7ebdaf214434114dce394e45b
MD5 da06e38efd1b80cd2fe2e0ef2469aa69
BLAKE2b-256 669fa38721ca6239285f66a0647baea42045114974a74f4869f11f92ecf74fd6

See more details on using hashes here.

File details

Details for the file parallel_ssh-1.6.0-cp27-cp27m-macosx_10_11_x86_64.whl.

File metadata

File hashes

Hashes for parallel_ssh-1.6.0-cp27-cp27m-macosx_10_11_x86_64.whl
Algorithm Hash digest
SHA256 986532d8de945973e45b392c7225d3356f59321c8f3c4919b8d9a90f7bf6ddc4
MD5 f8618bb6efeffbf1484734c9f52dfeb5
BLAKE2b-256 b1704a05e3395b2da6850105b68f20fc27faea5c364dd6302bc6a823c410d74a

See more details on using hashes here.

File details

Details for the file parallel_ssh-1.6.0-cp27-cp27m-macosx_10_10_intel.whl.

File metadata

File hashes

Hashes for parallel_ssh-1.6.0-cp27-cp27m-macosx_10_10_intel.whl
Algorithm Hash digest
SHA256 d65ec1c49e4af1585d6a198fc6850c60bbb4e2f01b8be2c45c1975df59550dca
MD5 8552456509b80c5df122dbd974bb6844
BLAKE2b-256 7696a8fded7157e9828115d5cd1e8f7e41c2636c2aa3846f4a904109742ab343

See more details on using hashes here.

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page