Skip to main content

Execution helpers for simplified usage of subprocess and ssh.

Project description

exec-helpers

https://travis-ci.com/python-useful-helpers/exec-helpers.svg?branch=master https://dev.azure.com/python-useful-helpers/exec-helpers/_apis/build/status/python-useful-helpers.exec-helpers?branchName=master https://coveralls.io/repos/github/python-useful-helpers/exec-helpers/badge.svg?branch=master Documentation Status https://img.shields.io/pypi/v/exec-helpers.svg https://img.shields.io/pypi/pyversions/exec-helpers.svg https://img.shields.io/pypi/status/exec-helpers.svg https://img.shields.io/github/license/python-useful-helpers/exec-helpers.svg https://img.shields.io/badge/code%20style-black-000000.svg

Execution helpers for simplified usage of subprocess and ssh. Why another subprocess wrapper and why no clear paramiko?

Historically paramiko offers good ssh client, but with specific limitations: you can call command with timeout, but without receiving return code, or call command and wait for return code, but without timeout processing.

In the most cases, we are need just simple SSH client with comfortable API for calls, calls via SSH proxy and checking return code/stderr. This library offers this functionality with deadlock free polling and friendly result objects (with inline decoding of XML Element tree, YAML, JSON, binary or just strings). In addition this library offers the same API for subprocess calls, but with specific limitation: no parallel calls (for protection from race conditions).

Pros:

Python 3.6
Python 3.7
Python 3.8

This package includes:

  • SSHClient - historically the first one helper, which used for SSH connections. Several API calls for sFTP also presents.

  • SSHAuth - class for credentials storage. SSHClient does not store credentials as-is, but uses SSHAuth for it. Objects of this class can be copied between ssh connection objects, also it used for execute_through_host.

  • Subprocess - subprocess.Popen wrapper with timeouts, polling and almost the same API, as SSHClient (except specific flags, like cwd for subprocess and get_tty for ssh).

  • async_api.Subprocess - the same, as Subprocess helper, but works with asyncio. .. note:: for Windows ProactorEventLoop or another non-standard event loop should be used!

  • ExecResult - class for execution results storage. Contains exit code, stdout, stderr and getters for decoding as JSON, YAML, XML (and LXML) element tree, string, bytearray and brief strings (up to 7 lines).

  • ExitCodes - enumerator for standard Linux exit codes. BASH return codes (produced from signal codes) also available.

Installation

Standard: pip install exec-helpers Extras:

  • yaml - install PyYaml for yaml decoding (PyYAML is main decoder, ruamel.YAML also supported as fallback.)

  • xml - install defusedxml for safe XML parsing to xml.etree.ElementTree.Element.

  • lxml - install lxml for advanced XML parsing. Can be unsafe.

  • ALL_FORMATS (all-formats) - install all parsers. When new parsers will be added, it will ne also supported.

Usage

SSHClient

Basic initialization of SSHClient can be done without construction of specific objects:

client = exec_helpers.SSHClient(host, username="username", password="password")

If ssh agent is running - keys will be collected by paramiko automatically, but if keys are in specific location - it should be loaded manually and provided as iterable object of paramiko.RSAKey.

For advanced cases or re-use of credentials, SSHAuth object should be used. It can be collected from connection object via property auth.

Creation from scratch:

auth = exec_helpers.SSHAuth(
    username='username',  # type: typing.Optional[str]
    password='password',  # type: typing.Optional[str]
    key=None,  # type: typing.Optional[paramiko.RSAKey]
    keys=None,  # type: typing.Optional[typing.Iterable[paramiko.RSAKey]],
    key_filename=None,  # type: typing.Union[typing.List[str], None]
    passphrase=None,  # type: typing.Optional[str]
)

Key is a main connection key (always tried first) and keys are alternate keys. Key filename is a filename or list of filenames with keys, which should be loaded. Passphrase is an alternate password for keys, if it differs from main password. If main key now correct for username - alternate keys tried, if correct key found - it became main. If no working key - password is used and None is set as main key.

Context manager is available, connection is closed and lock is released on exit from context.

Subprocess

Context manager is available, subprocess is killed and lock is released on exit from context.

Base methods

Main methods are execute, check_call and check_stderr for simple executing, executing and checking return code and executing, checking return code and checking for empty stderr output. This methods are almost the same for SSHClient and Subprocess, except specific flags.

result: ExecResult = helper.execute(
    command,  # type: str
    verbose=False,  # type: bool
    timeout=1 * 60 * 60,  # type: typing.Union[int, float, None]
    # Keyword only:
    log_mask_re=None,  # type: typing.Optional[str]
    stdin=None,  # type: typing.Union[bytes, str, bytearray, None]
    open_stdout=True,  # type: bool
    open_stderr=True,  # type: bool
    **kwargs
)
result: ExecResult = helper.check_call(
    command,  # type: str
    verbose=False,  # type: bool
    timeout=1 * 60 * 60,  # type: type: typing.Union[int, float, None]
    error_info=None,  # type: typing.Optional[str]
    expected=(0,),  # type: typing.Iterable[typing.Union[int, ExitCodes]]
    raise_on_err=True,  # type: bool
    # Keyword only:
    log_mask_re=None,  # type: typing.Optional[str]
    stdin=None,  # type: typing.Union[bytes, str, bytearray, None]
    open_stdout=True,  # type: bool
    open_stderr=True,  # type: bool
    exception_class=CalledProcessError,  # typing.Type[CalledProcessError]
    **kwargs
)
result: ExecResult = helper.check_stderr(
    command,  # type: str
    verbose=False,  # type: bool
    timeout=1 * 60 * 60,  # type: type: typing.Union[int, float, None]
    error_info=None,  # type: typing.Optional[str]
    raise_on_err=True,  # type: bool
    # Keyword only:
    expected=(0,),  # typing.Iterable[typing.Union[int, ExitCodes]]
    log_mask_re=None,  # type: typing.Optional[str]
    stdin=None,  # type: typing.Union[bytes, str, bytearray, None]
    open_stdout=True,  # type: bool
    open_stderr=True,  # type: bool
    exception_class=CalledProcessError,  # typing.Type[CalledProcessError]
)
result: ExecResult = helper(  # Lazy way: instances are callable and uses `execute`.
    command,  # type: str
    verbose=False,  # type: bool
    timeout=1 * 60 * 60,  # type: typing.Union[int, float, None]
    # Keyword only:
    log_mask_re=None,  # type: typing.Optional[str]
    stdin=None,  # type: typing.Union[bytes, str, bytearray, None]
    open_stdout=True,  # type: bool
    open_stderr=True,  # type: bool
    **kwargs
)

If no STDOUT or STDERR required, it is possible to disable this FIFO pipes via **kwargs with flags open_stdout=False and open_stderr=False.

The next command level uses lower level and kwargs are forwarded, so expected exit codes are forwarded from check_stderr. Implementation specific flags are always set via kwargs.

If required to mask part of command from logging, log_mask_re attribute can be set global over instance or provided with command. All regex matched groups will be replaced by ‘<*masked*>’.

result: ExecResult = helper.execute(
    command="AUTH='top_secret_key'; run command",  # type: str
    verbose=False,  # type: bool
    timeout=1 * 60 * 60,  # type: typing.Optional[int]
    log_mask_re=r"AUTH\s*=\s*'(\w+)'"  # type: typing.Optional[str]
)

result.cmd will be equal to AUTH=’<*masked*>’; run command

ExecResult

Execution result object has a set of useful properties:

  • cmd - Command

  • exit_code - Command return code. If possible to decode using enumerators for Linux -> it used.

  • stdin -> str. Text representation of stdin.

  • stdout -> typing.Tuple[bytes]. Raw stdout output.

  • stderr -> typing.Tuple[bytes]. Raw stderr output.

  • stdout_bin -> bytearray. Binary stdout output.

  • stderr_bin -> bytearray. Binary stderr output.

  • stdout_str -> str. Text representation of output.

  • stderr_str -> str. Text representation of output.

  • stdout_brief -> str. Up to 7 lines from stdout (3 first and 3 last if >7 lines).

  • stderr_brief -> str. Up to 7 lines from stderr (3 first and 3 last if >7 lines).

  • stdout_json - STDOUT decoded as JSON.

  • stdout_yaml - STDOUT decoded as YAML. Accessible only if PyYAML or ruamel.YAML library installed. (Extras: yaml)

  • stdout_xml - STDOUT decoded as XML to ElementTree using defusedxml library. Accessible only if defusedxml library installed. (Extras: xml)

  • stdout_lxml - STDOUT decoded as XML to ElementTree using lxml library. Accessible only if lxml library installed. (Extras: lxml) Can be insecure.

  • timestamp -> typing.Optional(datetime.datetime). Timestamp for received exit code.

SSHClient specific

SSHClient commands support get_pty flag, which enables PTY open on remote side. PTY width and height can be set via keyword arguments, dimensions in pixels are always 0x0.

Possible to call commands in parallel on multiple hosts if it’s not produce huge output:

results: Dict[Tuple[str, int], ExecResult] = SSHClient.execute_together(
    remotes,  # type: typing.Iterable[SSHClient]
    command,  # type: str
    timeout=1 * 60 * 60,  # type: type: typing.Union[int, float, None]
    expected=(0,),  # type: typing.Iterable[typing.Union[int, ExitCodes]]
    raise_on_err=True,  # type: bool
    # Keyword only:
    stdin=None,  # type: typing.Union[bytes, str, bytearray, None]
    open_stdout=True,  # type: bool
    open_stderr=True,  # type: bool
    log_mask_re=None,  # type: typing.Optional[str]
    exception_class=ParallelCallProcessError  # typing.Type[ParallelCallProcessError]
)
results  # type: typing.Dict[typing.Tuple[str, int], exec_result.ExecResult]

Results is a dict with keys = (hostname, port) and and results in values. By default execute_together raises exception if unexpected return code on any remote.

To open new connection using current as proxy is accessible method proxy_to. Basic usage example:

conn: SSHClient = client.proxy_to(host, username="username", password="password")

For execute through SSH host can be used execute_through_host method:

result: ExecResult = client.execute_through_host(
    hostname,  # type: str
    command,  # type: str
    # Keyword only:
    auth=None,  # type: typing.Optional[SSHAuth]
    target_port=22,  # type: int
    timeout=1 * 60 * 60,  # type: type: typing.Union[int, float, None]
    verbose=False,  # type: bool
    stdin=None,  # type: typing.Union[bytes, str, bytearray, None]
    open_stdout=True,  # type: bool
    open_stderr=True,  # type: bool
    log_mask_re=None,  # type: typing.Optional[str]
    get_pty=False,  # type: bool
    width=80,  # type: int
    height=24  # type: int
)

Where hostname is a target hostname, auth is an alternate credentials for target host.

SSH client implements fast sudo support via context manager:

Commands will be run with sudo enforced independently from client settings for normal usage:

with client.sudo(enforce=True):
    ...

Commands will be run without sudo independently from client settings for normal usage:

with client.sudo(enforce=False):
    ...

“Permanent client setting”:

client.sudo_mode = mode  # where mode is True or False

SSH Client supports sFTP for working with remote files:

with client.open(path, mode='r') as f:
    ...

For fast remote paths checks available methods:

  • exists(path) -> bool

>>> conn.exists('/etc/passwd')
True
  • stat(path) -> paramiko.sftp_attr.SFTPAttributes

>>> conn.stat('/etc/passwd')
<SFTPAttributes: [ size=1882 uid=0 gid=0 mode=0o100644 atime=1521618061 mtime=1449733241 ]>
>>> str(conn.stat('/etc/passwd'))
'-rw-r--r--   1 0        0            1882 10 Dec 2015  ?'
  • isfile(path) -> bool

>>> conn.isfile('/etc/passwd')
True
  • isdir(path) -> bool

>>> conn.isdir('/etc/passwd')
False

Additional (non-standard) helpers:

  • mkdir(path: str) - execute mkdir -p path

  • rm_rf(path: str) - execute rm -rf path

  • upload(source: str, target: str) - upload file or from source to target using sFTP.

  • download(destination: str, target: str) - download file from target to destination using sFTP.

Subprocess specific

Keyword arguments:

  • cwd - working directory.

  • env - environment variables dict.

async_api.Subprocess specific

All standard methods are coroutines. Async context manager also available.

Example:

async with helper:
  result: ExecResult = await helper.execute(
      command,  # type: str
      verbose=False,  # type: bool
      timeout=1 * 60 * 60,  # type: typing.Union[int, float, None]
      **kwargs
  )

Testing

The main test mechanism for the package exec-helpers is using tox. Available environments can be collected via tox -l

CI systems

For code checking several CI systems is used in parallel:

  1. Travis CI: is used for checking: PEP8, pylint, bandit, installation possibility and unit tests. Also it’s publishes coverage on coveralls.

  2. Azure Pipelines: is used for windows compatibility checking.

  3. coveralls: is used for coverage display.

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

exec-helpers-6.0.0.tar.gz (70.6 kB view details)

Uploaded Source

Built Distributions

exec_helpers-6.0.0-py3-none-any.whl (61.3 kB view details)

Uploaded Python 3

exec_helpers-6.0.0-cp38-cp38m-win_amd64.whl (937.8 kB view details)

Uploaded CPython 3.8m Windows x86-64

exec_helpers-6.0.0-cp38-cp38m-win32.whl (801.7 kB view details)

Uploaded CPython 3.8m Windows x86

exec_helpers-6.0.0-cp38-cp38-manylinux1_x86_64.whl (4.2 MB view details)

Uploaded CPython 3.8

exec_helpers-6.0.0-cp37-cp37m-win_amd64.whl (887.1 kB view details)

Uploaded CPython 3.7m Windows x86-64

exec_helpers-6.0.0-cp37-cp37m-win32.whl (762.4 kB view details)

Uploaded CPython 3.7m Windows x86

exec_helpers-6.0.0-cp37-cp37m-manylinux1_x86_64.whl (3.8 MB view details)

Uploaded CPython 3.7m

exec_helpers-6.0.0-cp36-cp36m-win_amd64.whl (887.1 kB view details)

Uploaded CPython 3.6m Windows x86-64

exec_helpers-6.0.0-cp36-cp36m-win32.whl (762.4 kB view details)

Uploaded CPython 3.6m Windows x86

exec_helpers-6.0.0-cp36-cp36m-manylinux1_x86_64.whl (3.8 MB view details)

Uploaded CPython 3.6m

File details

Details for the file exec-helpers-6.0.0.tar.gz.

File metadata

  • Download URL: exec-helpers-6.0.0.tar.gz
  • Upload date:
  • Size: 70.6 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.1.1 pkginfo/1.5.0.1 requests/2.22.0 setuptools/42.0.2 requests-toolbelt/0.9.1 tqdm/4.40.2 CPython/3.7.1

File hashes

Hashes for exec-helpers-6.0.0.tar.gz
Algorithm Hash digest
SHA256 beef554291a64836c40e9ea8a8c2dbf0bc2f6bfcde9625ad9df9d9bbe6adb9ee
MD5 cd3ed2321988e3f93e75ba2849b96649
BLAKE2b-256 1354771903a2bab1c00ac2d191baedd22c0d054729f7ce811eb84a6c57e10cdd

See more details on using hashes here.

File details

Details for the file exec_helpers-6.0.0-py3-none-any.whl.

File metadata

  • Download URL: exec_helpers-6.0.0-py3-none-any.whl
  • Upload date:
  • Size: 61.3 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.1.1 pkginfo/1.5.0.1 requests/2.22.0 setuptools/42.0.2 requests-toolbelt/0.9.1 tqdm/4.40.2 CPython/3.7.1

File hashes

Hashes for exec_helpers-6.0.0-py3-none-any.whl
Algorithm Hash digest
SHA256 40f07ec3792fb8c1ccbd8c622e144132989203fe145af839f858ea6ca72b3491
MD5 5fe6948e760c8fa5e441a64874e78f06
BLAKE2b-256 3957595e30b0c40b5755fff55f159f8aa3a067a174fbfc77d92a4a19b5698f1b

See more details on using hashes here.

File details

Details for the file exec_helpers-6.0.0-cp38-cp38m-win_amd64.whl.

File metadata

  • Download URL: exec_helpers-6.0.0-cp38-cp38m-win_amd64.whl
  • Upload date:
  • Size: 937.8 kB
  • Tags: CPython 3.8m, Windows x86-64
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.1.1 pkginfo/1.5.0.1 requests/2.22.0 setuptools/42.0.2 requests-toolbelt/0.9.1 tqdm/4.40.2 CPython/3.8.0

File hashes

Hashes for exec_helpers-6.0.0-cp38-cp38m-win_amd64.whl
Algorithm Hash digest
SHA256 ddc3b4f628abc3fff5a407c277cbb14d254a332a54f04fca43c3dce29c840b1b
MD5 2e287a535b6c02a732212cb12462012e
BLAKE2b-256 19fd38463539b9117a0c5ac5e294521628c02d0bfb5f9f0b61c0e7766506d38b

See more details on using hashes here.

File details

Details for the file exec_helpers-6.0.0-cp38-cp38m-win32.whl.

File metadata

  • Download URL: exec_helpers-6.0.0-cp38-cp38m-win32.whl
  • Upload date:
  • Size: 801.7 kB
  • Tags: CPython 3.8m, Windows x86
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.1.1 pkginfo/1.5.0.1 requests/2.22.0 setuptools/42.0.2 requests-toolbelt/0.9.1 tqdm/4.40.2 CPython/3.8.0

File hashes

Hashes for exec_helpers-6.0.0-cp38-cp38m-win32.whl
Algorithm Hash digest
SHA256 ef4be5361e699c11f090d91dc4b2b24f12a89f81920c4f2f51dfa7d80dcc4cc1
MD5 c0dcb8481a6491c372a39dc579efac1d
BLAKE2b-256 3c9c4fa4261789e7ed753f36d64bfdc60bd3f74e30a54077be215b622102392d

See more details on using hashes here.

File details

Details for the file exec_helpers-6.0.0-cp38-cp38-manylinux1_x86_64.whl.

File metadata

  • Download URL: exec_helpers-6.0.0-cp38-cp38-manylinux1_x86_64.whl
  • Upload date:
  • Size: 4.2 MB
  • Tags: CPython 3.8
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.1.1 pkginfo/1.5.0.1 requests/2.22.0 setuptools/42.0.2 requests-toolbelt/0.9.1 tqdm/4.40.2 CPython/3.7.1

File hashes

Hashes for exec_helpers-6.0.0-cp38-cp38-manylinux1_x86_64.whl
Algorithm Hash digest
SHA256 59a50905f5d80108d290ca90be8786b0999a585c9ab17e69c7c2b4a039d9c521
MD5 b7141ed363017c2ef57465cd28173219
BLAKE2b-256 2d38662c5ff69e0014ca60a5f6e1f7d9df293b71bfe56b5789ae41da3b3364ea

See more details on using hashes here.

File details

Details for the file exec_helpers-6.0.0-cp37-cp37m-win_amd64.whl.

File metadata

  • Download URL: exec_helpers-6.0.0-cp37-cp37m-win_amd64.whl
  • Upload date:
  • Size: 887.1 kB
  • Tags: CPython 3.7m, Windows x86-64
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.1.1 pkginfo/1.5.0.1 requests/2.22.0 setuptools/42.0.2 requests-toolbelt/0.9.1 tqdm/4.40.2 CPython/3.7.5

File hashes

Hashes for exec_helpers-6.0.0-cp37-cp37m-win_amd64.whl
Algorithm Hash digest
SHA256 63e75b20f559d0141cb8f8d0eccd9d05c6e2f4d153d48e2636d49d5589043db1
MD5 80dae8dd2594714ffcd8046684e34dc3
BLAKE2b-256 fb585a0287da5e3fa2d6685a6deb33f19e32dd0fec8e8508e2b4dd6514f95860

See more details on using hashes here.

File details

Details for the file exec_helpers-6.0.0-cp37-cp37m-win32.whl.

File metadata

  • Download URL: exec_helpers-6.0.0-cp37-cp37m-win32.whl
  • Upload date:
  • Size: 762.4 kB
  • Tags: CPython 3.7m, Windows x86
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.1.1 pkginfo/1.5.0.1 requests/2.22.0 setuptools/42.0.2 requests-toolbelt/0.9.1 tqdm/4.40.2 CPython/3.7.5

File hashes

Hashes for exec_helpers-6.0.0-cp37-cp37m-win32.whl
Algorithm Hash digest
SHA256 0eae1e85d3441b2292302f6dec1f6f570a1cc5addf4ab3f1b4777a6d558932d7
MD5 c9d829b4180aab0fb8216e13e1e52085
BLAKE2b-256 3e087020b65234e5c828f8fad6a880fd1222bd8fffb9562f98fb55fc4721e108

See more details on using hashes here.

File details

Details for the file exec_helpers-6.0.0-cp37-cp37m-manylinux1_x86_64.whl.

File metadata

  • Download URL: exec_helpers-6.0.0-cp37-cp37m-manylinux1_x86_64.whl
  • Upload date:
  • Size: 3.8 MB
  • Tags: CPython 3.7m
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.1.1 pkginfo/1.5.0.1 requests/2.22.0 setuptools/42.0.2 requests-toolbelt/0.9.1 tqdm/4.40.2 CPython/3.7.1

File hashes

Hashes for exec_helpers-6.0.0-cp37-cp37m-manylinux1_x86_64.whl
Algorithm Hash digest
SHA256 a0befacfeb9b30a8bf05249c482a144ea198b7bd84e9885618698b5eb01f7079
MD5 1c29ae55a85c6a223956a529f77a970f
BLAKE2b-256 306f422adb9c3c53d547e4b383ac2d847ae3000f55fc842c26aa95afc4b40c34

See more details on using hashes here.

File details

Details for the file exec_helpers-6.0.0-cp36-cp36m-win_amd64.whl.

File metadata

  • Download URL: exec_helpers-6.0.0-cp36-cp36m-win_amd64.whl
  • Upload date:
  • Size: 887.1 kB
  • Tags: CPython 3.6m, Windows x86-64
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.1.1 pkginfo/1.5.0.1 requests/2.22.0 setuptools/42.0.2 requests-toolbelt/0.9.1 tqdm/4.40.2 CPython/3.6.8

File hashes

Hashes for exec_helpers-6.0.0-cp36-cp36m-win_amd64.whl
Algorithm Hash digest
SHA256 5fd2a1eef59a626ad7b4bf08aecdae303ab6254bc286d682e855f16185c7b63a
MD5 66c7f05758ccb56e5e71b7e92ecc06c9
BLAKE2b-256 565149bb27097c93a1d29585fe1432fbc6d9b649f9818461a1dc8f9ea9f3ae25

See more details on using hashes here.

File details

Details for the file exec_helpers-6.0.0-cp36-cp36m-win32.whl.

File metadata

  • Download URL: exec_helpers-6.0.0-cp36-cp36m-win32.whl
  • Upload date:
  • Size: 762.4 kB
  • Tags: CPython 3.6m, Windows x86
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.1.1 pkginfo/1.5.0.1 requests/2.22.0 setuptools/42.0.2 requests-toolbelt/0.9.1 tqdm/4.40.2 CPython/3.6.8

File hashes

Hashes for exec_helpers-6.0.0-cp36-cp36m-win32.whl
Algorithm Hash digest
SHA256 43011e6ab35ac38a5de65de8df67c90312bbf4a576065371d92d20f9a5f8803a
MD5 795a1249e658fd1ad009de81cdb22e2b
BLAKE2b-256 3db0d43d64dd4d1bc2febf2d303487d5973cc3c590ff65c40aafe72fcd6efe97

See more details on using hashes here.

File details

Details for the file exec_helpers-6.0.0-cp36-cp36m-manylinux1_x86_64.whl.

File metadata

  • Download URL: exec_helpers-6.0.0-cp36-cp36m-manylinux1_x86_64.whl
  • Upload date:
  • Size: 3.8 MB
  • Tags: CPython 3.6m
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.1.1 pkginfo/1.5.0.1 requests/2.22.0 setuptools/42.0.2 requests-toolbelt/0.9.1 tqdm/4.40.2 CPython/3.7.1

File hashes

Hashes for exec_helpers-6.0.0-cp36-cp36m-manylinux1_x86_64.whl
Algorithm Hash digest
SHA256 4880d5f5cf4af0db6a6262995f99b24f822a7c5bbd7d6d7ac93a7de983ae6a84
MD5 775c4a04acc9644562662cdb3269bf90
BLAKE2b-256 64976d2c538cf0b19337f58bbb95b74bc2ac3b3f3ee66a6bfdd7b209a9e362f4

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