nicecall


Namenicecall JSON
Version 1.0.18 PyPI version JSON
download
home_pagehttp://go.pelicandd.com/n/python-nicecall
SummaryA library which provides a slightly more convinient way to launch processes, compared to Python's subprocess module.
upload_time2023-07-24 21:45:31
maintainer
docs_urlNone
authorArseni Mourzenko
requires_python
licenseMIT
keywords system subprocess process
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            While |python-subprocess|_ is great, it may not be the easiest library to use. This is the reason I created ``nicecall``: it allows to do simple tasks with processes very easily.

.. |python-subprocess| replace:: Python's ``subprocess`` library
.. _python-subprocess: https://docs.python.org/3/library/subprocess.html

Note that ``nicecall`` is not a substitute to ``subprocess``, because much of ``subprocess`` functionality doesn't exist. For instance, one can't use ``stdin`` or pipes with ``nicecall``. The goal is not to replace ``subprocess``, but only to provide an easy way to do the most common tasks.

How to use the library
----------------------

Note: make sure you also check `the tests <http://source.pelicandd.com/codebase/nicecall/tests/>`_ which
give a few examples of how to use the library. Most pieces of code below are in ``tests/smoke/test_docs.py``.

Fluent interface
~~~~~~~~~~~~~~~~

The library uses method chaining, which allows to add logic on the fly before actually launching the process. Methods such as ``on_stdout``, ``ignore``, etc. create a copy of the object, modify this copy, and return it to the caller. This makes it possible to reuse base objects in multiple locations in your code, reducing code duplication.

Exit code
~~~~~~~~~

Let's start by executing a task::

    result = nicecall.Process().execute(["touch", "/tmp/hello"])

The ``result`` contains the exit code, which makes it possible to determine whether the process terminated successfully. Below, the value of ``result`` is expected to be zero. You may also make it fail::

    result = nicecall.Process().execute(["touch", "/tmp/a/b/c/d"])

The ``result`` should now be ``1``, assuming you don't have ``/tmp/a/b/c`` directory.

Sometimes, one may prefer raising an exception if the process fails, instead of checking for the exit code manually. The ``raise_if_error()`` method can be used for that. When called, exit code different from zero will lead to ``subprocess.CalledProcessError`` being raised.

``stdout`` and ``stderr``
~~~~~~~~~~~~~~~~~~~~~~~~~

One can also perform a bunch of actions on ``stdout`` and ``stderr``. Let's display ``stdout`` in terminal::

    nicecall.Process().on_stdout(print).execute(["echo", "a\nb\nc"])

The output should be::

    a
    b
    c

If you just want to read ``stdout`` output, then instead of ``execute``, you can use ``read_stdout``. The following piece of code produces the same output::

    lines = nicecall.Process().read_stdout(["echo", "a\nb\nc"])
    print("\n".join(lines))

Note that when using ``read_stdout``, a non-zero exit code *will* lead to ``subprocess.CalledProcessError`` being raised, independently of the presence of ``raise_if_error()`` in the chain. The reason for this choice is that with ``read_stdout``, there is no way to check for the actual exit code, and plainly ignoring it would lead to difficult to debug situations.

Also note that ``read_stdout`` stores ``stdout`` in a list and is *not* lazy. This means that:

 * The method is appropriate only for commands which output a small quantity of lines.

 * Looping through the lines and stopping in the middle wouldn't affect neither the processing of the exit code, nor the other actions such as logging which may have been specified earlier in the chain through ``on_stdout``.

 * The lines from ``stderr`` are not returned. The ``on_stderr`` method can still be used, exactly in the same way it is with ``execute``.

 * All the actions defined through ``on_stdout`` will be performed *before* the list is returned.

 * Filters added through ``keep`` and ``ignore`` apply to the lines in the result as well.

Logging
~~~~~~~

A common thing, at least in my case, is to log ``stdout`` or ``stderr`` to syslog. With ``nicecall``, it's easy::

    # Initialize logging.
    log_handler = logging.handlers.SysLogHandler(address="/dev/log")
    formatter = logging.Formatter("demo: [%(levelname)s] %(message)s")
    log_handler.setFormatter(formatter)
    log_handler.setLevel(logging.DEBUG)

    demo_logger = logging.getLogger("demo")
    demo_logger.setLevel(logging.DEBUG)
    demo_logger.addHandler(log_handler)

    ...

    # Log stdout.
    logger = nicecall.StdoutLogger("test")
    nicecall.Process().on_stdout(logger.log).execute(["echo", "a\nb"])

Note that ``nicecall.StdoutLogger`` can be initialized with either the name of the logger, or the instance of the logger itself.

The library itself logs calls (``INFO`` level) and call failures (``WARNING`` level) through the logger named ``nicecall.process``. For instance, executing ``touch /tmp/a/b/c/d`` will produce two log messages::

    INFO:nicecall.process:Called “touch /tmp/a/b/c/d”.
    WARNING:nicecall.process:“touch /tmp/a/b/c/d” failed with exit code 1.

Filtering
~~~~~~~~~

Sometimes, you don't want to process specific content such as empty lines or whitespace. This is what filters are about::

    nicecall \
        .Process() \
        .ignore(nicecall.filters.whitespace) \
        .on_stdout(print) \
        .execute(["echo", "a\n\nb"])

Here, ``a`` and ``b`` will be displayed in terminal; however, the empty line will be ignored. The reverse is called ``keep``. Both ``keep`` and ``ignore`` accept any function which takes a string as a parameter and returns a boolean. For instance, this will print only ``stdout`` content longer than fifteen characters::

    nicecall \
        .Process() \
        .keep(lambda line: len(line) > 15) \
        .on_stdout(print) \
        .execute(["echo", "Hello World!\nWhat a lovely day!"])

Multiple ``keep`` and ``ignore`` methods can be combined. The output will keep the lines which match *all* predicates from ``keep`` methods and *none* from ``ignore`` ones.

Filters apply to both ``stdout`` and ``stderr``; there is no way to apply them to only one of the streams.

Testing
-------

In order to be able to test your code, the library provides a ``NullProcess`` class, a stub and a mock.

``NullProcess``
~~~~~~~~~~~~~~~

This class creates an object which will *not* launch any process when ``execute`` is called. The purpose of this class is to replace the actual ``Process`` class during testing.

Stub
~~~~

The stub makes it possible to emulate ``Process`` without actually doing the system calls. The difference with ``NullProcess`` is that the stub makes it possible to define the exit codes and ``stdout``/``stderr`` output for specific commands.

The stub allows to define associations between the arguments and the expected response. For instance, imagine a situation where the tested code is expected to perform two calls: one to create a directory, another one to create a file in it. We want to test how the code under testing will perform if the second command fails: are the developers handling this edge case? For this purpose, one can use the stub like this::

    stub = nicecall.tests.ProcessStub()
    stub.add_match(["mkdir", "/tmp/a"], 0)
    stub.add_match(
        ["touch", "/tmp/a/b"],
        1,
        stderr=["touch: cannot touch '/tmp/a/b': No such file or directory"])

The ``stub`` can now be passed to the code under tests instead of ``nicecall.Process()``. The tested code will run, perform a ``mkdir``, and, when executing the ``touch`` command, will get back the exit code ``1`` and a call to the actions, if any, set through ``on_stderr``.

Mock
~~~~

The mock performs in a similar way to a stub, but also records the activity of the code under tests, i.e. the parameters which were passed to different methods of the mock. Usually, the mock is used this way::

    with nicecall.tests.ProcessMockContext() as context:
        # Code under tests goes here.
        # The mock is `context.mock`.
        ...

        # Follows the assertions. In this example, I'm just ensuring that the
        # code under tests added `print` to the `stdout` actions, i.e. ran the
        # `...on_stdout(print)...` command.
        actual = context.on_stdout_actions
        expected = [print]
        self.assertCountEqual(expected, actual)

The mock makes it possible to check the following elements:

 * ``executed_args``: the ``args`` which were used when calling ``execute()`` method.

 * ``ignore_predicates``: the list of predicates added by the tested code using the ``ignore`` method.

 * ``keep_predicates``: same as previous, but for ``keep``.

 * ``on_stdout_actions``: the list of actions added by the tested code using the ``on_stdout`` method.

 * ``on_stderr_actions``: same as previous, but for ``on_stderr``.

Classes
-------

``process.py``
~~~~~~~~~~~~~~

The class is the entry point of the library. It makes it possible to specify different options before actually starting the process.

 * ``execute``: actually executes the process and blocks until the process finishes.

   *Parameters:*

   ``args`` is an array which indicates the process to start, and its parameters. Example: ``["touch", "/tmp/hello"]``.

   *Returns:*

   Returns the exit code.

 * ``keep``: specifies a filter to apply to determine if the line of ``stdout`` or ``stderr`` should be processed by the actions specified through ``on_stdout`` and ``on_stderr``.

   The method can be called multiple times and mixed with ``ignore`` to aggregate multiple filters.

   *Parameters:*

   ``predicate`` is a function which takes a string as a parameter and returns a boolean value: ``true`` if the line should be processed, or ``false`` otherwise.

   *Returns:*

   Returns a new instance of the ``Process`` class with the new filter.

 * ``ignore``: see ``keep``. Here, the predicate is reverted.

 * ``on_stdout``: adds an action to perform when a line from ``stdout`` is received.

   The method can be called multiple times if multiple actions should be performed for every line of ``stdout``.

   *Parameters:*

   ``action``: a function which takes a string as a parameter and doesn't return anything.

   *Returns:*

   Returns a new instance of the ``Process`` class with the new action.

 * ``on_stderr``: see ``on_stdout``. Here, it deals with ``stderr`` instead.

``filters.py``
~~~~~~~~~~~~~~

The file contains a bunch of filters which can be used in ``Process.keep`` and ``Process.ignore``.

``logger.py``
~~~~~~~~~~~~~

This class is used to log output from ``stdout`` or ``stderr``.

Compatibility
-------------

The library was written for Python 3 under Linux. I haven't tested it neither with Python 2, nor under Windows.

Reliability
-----------

While I used Test Driven Development when creating this library and naturally have a 100% branch coverage, I don't know neither Python, nor ``subprocess`` well enough to be sure that the library can be used reliably in production. Use at own risk.

Contributing
------------

If you want to contribute, contact me at `arseni.mourzenko@pelicandd.com <mailto:arseni.mourzenko@pelicandd.com>`_. You'll be able to contribute to the project using the `official SVN repository <http://source.pelicandd.com/codebase/nicecall/>`_. If you find it more convinient to clone the source to GitHub, you can do that too.

The source code of the library and the corresponding documentation are covered by the `MIT License <https://opensource.org/licenses/MIT>`_.

            

Raw data

            {
    "_id": null,
    "home_page": "http://go.pelicandd.com/n/python-nicecall",
    "name": "nicecall",
    "maintainer": "",
    "docs_url": null,
    "requires_python": "",
    "maintainer_email": "",
    "keywords": "system subprocess process",
    "author": "Arseni Mourzenko",
    "author_email": "arseni.mourzenko@pelicandd.com",
    "download_url": "",
    "platform": null,
    "description": "While |python-subprocess|_ is great, it may not be the easiest library to use. This is the reason I created ``nicecall``: it allows to do simple tasks with processes very easily.\n\n.. |python-subprocess| replace:: Python's ``subprocess`` library\n.. _python-subprocess: https://docs.python.org/3/library/subprocess.html\n\nNote that ``nicecall`` is not a substitute to ``subprocess``, because much of ``subprocess`` functionality doesn't exist. For instance, one can't use ``stdin`` or pipes with ``nicecall``. The goal is not to replace ``subprocess``, but only to provide an easy way to do the most common tasks.\n\nHow to use the library\n----------------------\n\nNote: make sure you also check `the tests <http://source.pelicandd.com/codebase/nicecall/tests/>`_ which\ngive a few examples of how to use the library. Most pieces of code below are in ``tests/smoke/test_docs.py``.\n\nFluent interface\n~~~~~~~~~~~~~~~~\n\nThe library uses method chaining, which allows to add logic on the fly before actually launching the process. Methods such as ``on_stdout``, ``ignore``, etc. create a copy of the object, modify this copy, and return it to the caller. This makes it possible to reuse base objects in multiple locations in your code, reducing code duplication.\n\nExit code\n~~~~~~~~~\n\nLet's start by executing a task::\n\n    result = nicecall.Process().execute([\"touch\", \"/tmp/hello\"])\n\nThe ``result`` contains the exit code, which makes it possible to determine whether the process terminated successfully. Below, the value of ``result`` is expected to be zero. You may also make it fail::\n\n    result = nicecall.Process().execute([\"touch\", \"/tmp/a/b/c/d\"])\n\nThe ``result`` should now be ``1``, assuming you don't have ``/tmp/a/b/c`` directory.\n\nSometimes, one may prefer raising an exception if the process fails, instead of checking for the exit code manually. The ``raise_if_error()`` method can be used for that. When called, exit code different from zero will lead to ``subprocess.CalledProcessError`` being raised.\n\n``stdout`` and ``stderr``\n~~~~~~~~~~~~~~~~~~~~~~~~~\n\nOne can also perform a bunch of actions on ``stdout`` and ``stderr``. Let's display ``stdout`` in terminal::\n\n    nicecall.Process().on_stdout(print).execute([\"echo\", \"a\\nb\\nc\"])\n\nThe output should be::\n\n    a\n    b\n    c\n\nIf you just want to read ``stdout`` output, then instead of ``execute``, you can use ``read_stdout``. The following piece of code produces the same output::\n\n    lines = nicecall.Process().read_stdout([\"echo\", \"a\\nb\\nc\"])\n    print(\"\\n\".join(lines))\n\nNote that when using ``read_stdout``, a non-zero exit code *will* lead to ``subprocess.CalledProcessError`` being raised, independently of the presence of ``raise_if_error()`` in the chain. The reason for this choice is that with ``read_stdout``, there is no way to check for the actual exit code, and plainly ignoring it would lead to difficult to debug situations.\n\nAlso note that ``read_stdout`` stores ``stdout`` in a list and is *not* lazy. This means that:\n\n * The method is appropriate only for commands which output a small quantity of lines.\n\n * Looping through the lines and stopping in the middle wouldn't affect neither the processing of the exit code, nor the other actions such as logging which may have been specified earlier in the chain through ``on_stdout``.\n\n * The lines from ``stderr`` are not returned. The ``on_stderr`` method can still be used, exactly in the same way it is with ``execute``.\n\n * All the actions defined through ``on_stdout`` will be performed *before* the list is returned.\n\n * Filters added through ``keep`` and ``ignore`` apply to the lines in the result as well.\n\nLogging\n~~~~~~~\n\nA common thing, at least in my case, is to log ``stdout`` or ``stderr`` to syslog. With ``nicecall``, it's easy::\n\n    # Initialize logging.\n    log_handler = logging.handlers.SysLogHandler(address=\"/dev/log\")\n    formatter = logging.Formatter(\"demo: [%(levelname)s] %(message)s\")\n    log_handler.setFormatter(formatter)\n    log_handler.setLevel(logging.DEBUG)\n\n    demo_logger = logging.getLogger(\"demo\")\n    demo_logger.setLevel(logging.DEBUG)\n    demo_logger.addHandler(log_handler)\n\n    ...\n\n    # Log stdout.\n    logger = nicecall.StdoutLogger(\"test\")\n    nicecall.Process().on_stdout(logger.log).execute([\"echo\", \"a\\nb\"])\n\nNote that ``nicecall.StdoutLogger`` can be initialized with either the name of the logger, or the instance of the logger itself.\n\nThe library itself logs calls (``INFO`` level) and call failures (``WARNING`` level) through the logger named ``nicecall.process``. For instance, executing ``touch /tmp/a/b/c/d`` will produce two log messages::\n\n    INFO:nicecall.process:Called \u201ctouch /tmp/a/b/c/d\u201d.\n    WARNING:nicecall.process:\u201ctouch /tmp/a/b/c/d\u201d failed with exit code 1.\n\nFiltering\n~~~~~~~~~\n\nSometimes, you don't want to process specific content such as empty lines or whitespace. This is what filters are about::\n\n    nicecall \\\n        .Process() \\\n        .ignore(nicecall.filters.whitespace) \\\n        .on_stdout(print) \\\n        .execute([\"echo\", \"a\\n\\nb\"])\n\nHere, ``a`` and ``b`` will be displayed in terminal; however, the empty line will be ignored. The reverse is called ``keep``. Both ``keep`` and ``ignore`` accept any function which takes a string as a parameter and returns a boolean. For instance, this will print only ``stdout`` content longer than fifteen characters::\n\n    nicecall \\\n        .Process() \\\n        .keep(lambda line: len(line) > 15) \\\n        .on_stdout(print) \\\n        .execute([\"echo\", \"Hello World!\\nWhat a lovely day!\"])\n\nMultiple ``keep`` and ``ignore`` methods can be combined. The output will keep the lines which match *all* predicates from ``keep`` methods and *none* from ``ignore`` ones.\n\nFilters apply to both ``stdout`` and ``stderr``; there is no way to apply them to only one of the streams.\n\nTesting\n-------\n\nIn order to be able to test your code, the library provides a ``NullProcess`` class, a stub and a mock.\n\n``NullProcess``\n~~~~~~~~~~~~~~~\n\nThis class creates an object which will *not* launch any process when ``execute`` is called. The purpose of this class is to replace the actual ``Process`` class during testing.\n\nStub\n~~~~\n\nThe stub makes it possible to emulate ``Process`` without actually doing the system calls. The difference with ``NullProcess`` is that the stub makes it possible to define the exit codes and ``stdout``/``stderr`` output for specific commands.\n\nThe stub allows to define associations between the arguments and the expected response. For instance, imagine a situation where the tested code is expected to perform two calls: one to create a directory, another one to create a file in it. We want to test how the code under testing will perform if the second command fails: are the developers handling this edge case? For this purpose, one can use the stub like this::\n\n    stub = nicecall.tests.ProcessStub()\n    stub.add_match([\"mkdir\", \"/tmp/a\"], 0)\n    stub.add_match(\n        [\"touch\", \"/tmp/a/b\"],\n        1,\n        stderr=[\"touch: cannot touch '/tmp/a/b': No such file or directory\"])\n\nThe ``stub`` can now be passed to the code under tests instead of ``nicecall.Process()``. The tested code will run, perform a ``mkdir``, and, when executing the ``touch`` command, will get back the exit code ``1`` and a call to the actions, if any, set through ``on_stderr``.\n\nMock\n~~~~\n\nThe mock performs in a similar way to a stub, but also records the activity of the code under tests, i.e. the parameters which were passed to different methods of the mock. Usually, the mock is used this way::\n\n    with nicecall.tests.ProcessMockContext() as context:\n        # Code under tests goes here.\n        # The mock is `context.mock`.\n        ...\n\n        # Follows the assertions. In this example, I'm just ensuring that the\n        # code under tests added `print` to the `stdout` actions, i.e. ran the\n        # `...on_stdout(print)...` command.\n        actual = context.on_stdout_actions\n        expected = [print]\n        self.assertCountEqual(expected, actual)\n\nThe mock makes it possible to check the following elements:\n\n * ``executed_args``: the ``args`` which were used when calling ``execute()`` method.\n\n * ``ignore_predicates``: the list of predicates added by the tested code using the ``ignore`` method.\n\n * ``keep_predicates``: same as previous, but for ``keep``.\n\n * ``on_stdout_actions``: the list of actions added by the tested code using the ``on_stdout`` method.\n\n * ``on_stderr_actions``: same as previous, but for ``on_stderr``.\n\nClasses\n-------\n\n``process.py``\n~~~~~~~~~~~~~~\n\nThe class is the entry point of the library. It makes it possible to specify different options before actually starting the process.\n\n * ``execute``: actually executes the process and blocks until the process finishes.\n\n   *Parameters:*\n\n   ``args`` is an array which indicates the process to start, and its parameters. Example: ``[\"touch\", \"/tmp/hello\"]``.\n\n   *Returns:*\n\n   Returns the exit code.\n\n * ``keep``: specifies a filter to apply to determine if the line of ``stdout`` or ``stderr`` should be processed by the actions specified through ``on_stdout`` and ``on_stderr``.\n\n   The method can be called multiple times and mixed with ``ignore`` to aggregate multiple filters.\n\n   *Parameters:*\n\n   ``predicate`` is a function which takes a string as a parameter and returns a boolean value: ``true`` if the line should be processed, or ``false`` otherwise.\n\n   *Returns:*\n\n   Returns a new instance of the ``Process`` class with the new filter.\n\n * ``ignore``: see ``keep``. Here, the predicate is reverted.\n\n * ``on_stdout``: adds an action to perform when a line from ``stdout`` is received.\n\n   The method can be called multiple times if multiple actions should be performed for every line of ``stdout``.\n\n   *Parameters:*\n\n   ``action``: a function which takes a string as a parameter and doesn't return anything.\n\n   *Returns:*\n\n   Returns a new instance of the ``Process`` class with the new action.\n\n * ``on_stderr``: see ``on_stdout``. Here, it deals with ``stderr`` instead.\n\n``filters.py``\n~~~~~~~~~~~~~~\n\nThe file contains a bunch of filters which can be used in ``Process.keep`` and ``Process.ignore``.\n\n``logger.py``\n~~~~~~~~~~~~~\n\nThis class is used to log output from ``stdout`` or ``stderr``.\n\nCompatibility\n-------------\n\nThe library was written for Python 3 under Linux. I haven't tested it neither with Python 2, nor under Windows.\n\nReliability\n-----------\n\nWhile I used Test Driven Development when creating this library and naturally have a 100% branch coverage, I don't know neither Python, nor ``subprocess`` well enough to be sure that the library can be used reliably in production. Use at own risk.\n\nContributing\n------------\n\nIf you want to contribute, contact me at `arseni.mourzenko@pelicandd.com <mailto:arseni.mourzenko@pelicandd.com>`_. You'll be able to contribute to the project using the `official SVN repository <http://source.pelicandd.com/codebase/nicecall/>`_. If you find it more convinient to clone the source to GitHub, you can do that too.\n\nThe source code of the library and the corresponding documentation are covered by the `MIT License <https://opensource.org/licenses/MIT>`_.\n",
    "bugtrack_url": null,
    "license": "MIT",
    "summary": "A library which provides a slightly more convinient way to launch processes, compared to Python's subprocess module.",
    "version": "1.0.18",
    "project_urls": {
        "Homepage": "http://go.pelicandd.com/n/python-nicecall"
    },
    "split_keywords": [
        "system",
        "subprocess",
        "process"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "8bbf39f0bbdd029e72f5fcc383f5d4bd75a0b0609802913b13a8f4e499a1ebe5",
                "md5": "e9acad4ab243a81876d9d0f8295799df",
                "sha256": "10600e71c13cc3f92715ce6306d34e255bf5f1e7880c4566283a2a79925a2978"
            },
            "downloads": -1,
            "filename": "nicecall-1.0.18-py2.py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "e9acad4ab243a81876d9d0f8295799df",
            "packagetype": "bdist_wheel",
            "python_version": "py2.py3",
            "requires_python": null,
            "size": 9439,
            "upload_time": "2023-07-24T21:45:31",
            "upload_time_iso_8601": "2023-07-24T21:45:31.783016Z",
            "url": "https://files.pythonhosted.org/packages/8b/bf/39f0bbdd029e72f5fcc383f5d4bd75a0b0609802913b13a8f4e499a1ebe5/nicecall-1.0.18-py2.py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2023-07-24 21:45:31",
    "github": false,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "lcname": "nicecall"
}
        
Elapsed time: 0.33688s