Metadata-Version: 2.0
Name: mountepy
Version: 0.1.10
Summary: Utilities for creating (micro)service tests. Based on Mountebank.
Home-page: https://github.com/butla/mountepy
Author: Michał Bultrowicz
Author-email: michal.bultrowicz@gmail.com
License: Unlicense
Keywords: test http mountebank microservice
Platform: UNKNOWN
Classifier: Programming Language :: Python :: 3.4
Classifier: Operating System :: POSIX
Classifier: Development Status :: 3 - Alpha
Classifier: License :: Freely Distributable
Classifier: Intended Audience :: Developers
Classifier: Natural Language :: English
Requires-Dist: port-for
Requires-Dist: requests

mountepy
========

.. image:: https://travis-ci.org/butla/mountepy.svg?branch=master
    :target: https://travis-ci.org/butla/mountepy
.. image:: https://coveralls.io/repos/butla/mountepy/badge.svg?branch=master&service=github
    :target: https://coveralls.io/github/butla/mountepy?branch=master
.. image:: https://requires.io/github/butla/mountepy/requirements.svg?branch=master
    :target: https://requires.io/github/butla/mountepy/requirements/?branch=master

Utilities for creating (micro)service tests. Based on `Mountebank <http://www.mbtest.org/>`_.

- Test-framework-agnostic (use unittest, nose, py.test or whatever...
  but I like py.test).
- Enables fast and reliable end-to-end testing of microservices. They
  won't be aware that they are in some testing mode.
- Tested on Python 3.4, Ubuntu 14 x64.
- Planned features in the road map below (*this is still under
  development*).
  If you have suggestions just post them as Github issues.

Mountepy actually works by managing HTTP service processes, including
that of Mountebank. Thanks to that you no longer need that "start X
before running the tests" for your application. No. Your tests start
"X", it's put up or down only when it needs to and as many times as you
need.

I recommend Pytest for elastic composition of service process test
fixtures. Your process may start once per test suite, once per test,
etc.

Installation
------------

.. code-block:: bash

    $ pip install mountepy

A standalone distribution of Mountebank (including NodeJS) will be
downloaded on first run.

If you don't want Mountepy to download Mountebank:

1. Install NodeJS and NPM. On Ubuntu it's

.. code-block:: bash

    $ sudo apt-get install -y nodejs-legacy npm

2. Install Mountebank yourself

.. code-block:: bash

    $ npm install -g mountebank --production

Examples
--------

Mountebank acts as a mock for external HTTP services.
Here's how you spawn a Mountebank process, configure it with a stub
of some HTTP service, assert that it's actually responding.
Mountebank process is killed after the ``with`` block.

.. code-block:: python

    # requests is installed alongside Mountepy
    import mountepy, requests

    with mountepy.Mountebank() as mb:
        imposter = mb.add_imposter_simple(path='/something', response='mock response')
        stub_url = 'http://localhost:{}/something'.format(imposter.port)
        assert requests.get(stub_url).text == 'mock response'

It's a good idea to test your service as a whole process.
Let's say that you have an one-file WSGI (e.g. Flask or Bottle) app
that responds to a ``GET`` on its root path (``'\'``) with a string
it sees in ``RET_STR`` environment variable.
Also, the app needs to know on what port to run, so we also pass it
as an environment variable. ``{port}`` is a special value for Mountepy.
It will be filled with the application's port, whether it's passed
during object costruction or automatically selected from free ports.

.. code-block:: python

    # port_for is installed alongside Mountepy
    import mountepy, requests, port_for, os, sys

    service_port = port_for.select_random()
    service = mountepy.HttpService(
        [sys.executable, 'sample_app.py'],
        port=service_port,
        env={
            'PORT': '{port}',
            'RET_STR': 'Just some text.'
        })
    with service:
        assert requests.get(service.base_url).text == 'Just some text.'

Starting a more complex service running on `Gunicorn <http://gunicorn.org/>`_
can look like this:

.. code-block:: python

    import os, sys

    gunicorn_path = os.path.join(os.path.dirname(sys.executable), 'gunicorn')
    service_command = [
        gunicorn_path,
        'your_package.app:get_app()',
        '--bind', ':{port}',
        '--enable-stdio-inheritance',
        '--pythonpath', ','.join(sys.path)]

    service = HttpService(service_command)

    # You can use start/stop methods instead of using the "with" statement.
    # It's the same for Mountebank objects.
    service.start()

    # now you test stuff...
    service.stop()

Testing
-------

Install and run tox

.. code-block:: bash

    $ pip install tox
    $ tox

Motivation (on 2015-12-30)
--------------------------

- Why `Mountebank <https://github.com/bbyars/mountebank>`_? It can be
  deployed as standalone application, is actively developed and
  supports TCP mocks which can be used to simulate broken HTTP
  messages.
- Why not `Pretenders <https://github.com/pretenders/pretenders>`_?
  Doesn't support TCP and the development doesn't seem to be really
  active.
- Why not `WireMock <https://github.com/tomakehurst/wiremock>`_?
  Doesn't support TCP and I don't want to be forced to install Java to
  run tests and it doesn't seem to have more features than Mountebank.
- Why create a new project? There already is a `Python Mountebank
  wrapper <https://github.com/aholyoke/mountebank-python>`_, but it
  doesn't offer much.

Road map
--------

#. Add Pylint.
#. Switch license to Creative Commons Zero.
#. Add to PyPI.
#. Fill all imposter match fields.
#. Fix all TODOs.
#. Fully document methods.
#. Translate this README to rST.
#. Add example of calling services through client generated with
   `Bravado <https://github.com/Yelp/bravado>`_.
#. Remove MANIFEST.in like it's done in PyScaffold.
#. Check if cycling port\_for.is\_available() is as good as
   wait\_for\_port. If not, add that to port\_for.

Notes
-----

- `Bottle <https://github.com/bottlepy/bottle>`_ is used to test HTTP
  services' handler.


