Metadata-Version: 2.1
Name: monitor-requests
Version: 2.1.0
Summary: Check remote calls via request
Home-page: https://github.com/danpozmanter/monitor_requests
Author: Dan Pozmanter
Author-email: UNKNOWN
License: BSD
Keywords: requests testing monitoring
Platform: UNKNOWN
Classifier: Development Status :: 5 - Production/Stable
Classifier: Intended Audience :: Developers
Classifier: License :: OSI Approved :: BSD License
Classifier: Natural Language :: English
Classifier: Operating System :: OS Independent
Classifier: Topic :: Software Development :: Libraries :: Python Modules
Classifier: Topic :: Software Development :: Testing

monitor requests
================

|Build Status| |PyPI|

A tool to check for remote calls via the `requests`_ module.

Find out which external apis your code is calling.

Optionally:
    * Find where calls are taking place.
    * Capture the responses and response status codes.
    * Filter by domain
    * Run as a server for parallel tests

2.x Notes:
    * Now mocks the `send()` method on requests' HTTPAdapter class.
    * Now `monitor-requests` instead of `MonitorRequests`

**Installation**

.. code:: bash

    pip install monitor-requests

**Usage**

.. code:: python

    import unittest
    import monitor_requests
    from example import example_method

    class ExampleTestCase(unittest.TestCase):

        @classmethod
        def setUpClass(cls):
            cls.monitor = monitor_requests.Monitor()

        @classmethod
        def tearDownClass(cls):
            print('External calls for example:')
            # Optionally pass tracebacks=False to skip outputing tracebacks
            # Report will call .stop() by default.
            cls.monitor.report(tracebacks=False)

        def test_example_method(self):
            result = example_method()
            self.assertEqual(result.status_code, 200)

    if __name__ == 'main':
        unittest.main()


To filter what domains are captured:

.. code:: python

    # Only capture facebook and google requests:
    monitor = monitor_requests.Monitor(
        domains=['facebook\.com', 'google\.com']
    )

To set this up inside a django test runner:
(This will only work at the suite level if running tests in serial. Depending on your setup you may need to run with --parallel=1). Alternatively there are instructions further down on how to use `Server Moder` to push data asynchronously to an included `tornado`_ data server.

.. code:: python

    def run_suite(self, suite, **kwargs):
        monitor = monitor_requests.Monitor()
        test_result = super(ReelioTestRunner, self).run_suite(suite, **kwargs)
        monitor.report()
        return test_result

To set up inside a py.test session fixture:

.. code:: python

    @pytest.fixture(scope='session')
    def session_fixture():
        monitor = monitor_requests.Monitor()
        yield
        monitor.report()

To write to a file:

.. code:: python

        @classmethod
        def tearDownClass(cls):
            with open('output.txt', 'w') as f:
                cls.monitor.report(output=f)

For finer tuned control over output:

* Use `debug=True` to show urls, responses, and tracebacks.
* Use `urls=True` to show urls.
* Use `tracebacks=True` or `respones=True` to show tracebacks or responses (urls will be shown as well, as both tracebacks and responses are organized by url).

***Server Mode***

If you want to activate monitor_requests for an entire test suite running parallel, you can run the included `tornado`_ server to persist request data:

.. code:: bash

    monitor_requests_server --port=9003

.. code:: python

    def run_suite(self, suite, **kwargs):
        # Make sure to turn off mocking at the suit or session level.
        monitor = monitor_requests.Monitor(server_port=9003, mocking=False)
        test_result = super(ReelioTestRunner, self).run_suite(suite, **kwargs)
        monitor.report()
        return test_result

You will need to do additional calls in your TestCase classes:

.. code:: python

    class ExampleTestCase(unittest.TestCase):

        @classmethod
        def setUpClass(cls):
            # Same port, and same domain filtering if applicable.
            cls.monitor = monitor_requests.Monitor(server_port=9003)

        @classmethod
        def tearDownClass(cls):
            # Make sure to stop the mocking in the tear down.
            cls.monitor.stop()

Note that here there is no tearDownClass and no call to either stop() or report().
That only happens at the session level.

**Example Output**

With `debug=True`:


.. code:: text

    __________URLS__________

    __________URL________
    URL:      http://facebook.com?param=test
    Methods:  GET
    Requests: 1
    ______Tracebacks_____
    File "example.py", line 22, in <module>
        run()
      File "example.py", line 18, in run
        get_function_fb()
      File "example.py", line 12, in get_function_fb
        return requests.get('http://facebook.com?param=test')
    _______Responses______
    <StatusCode>200</StatusCode>
    <Content><!DOCTYPE html>
    <html lang="en" id="facebook" class="no_js">Etc/Etc</html></Content>

    __________URL________
    URL:      http://google.com
    Methods:  GET
    Requests: 1
    ______Tracebacks_____
    File "example.py", line 22, in <module>
        run()
      File "example.py", line 17, in run
        get_function()
      File "example.py", line 7, in get_function
        return requests.get('http://google.com')
    _______Responses______
    <StatusCode>200</StatusCode>
    <Content><!doctype html><html itemscope="" itemtype="http://schema.org/WebPage" lang="en">Etc/Etc</html></Content>

    ___________Analysis__________

    Total Requests: 2
    Time (Seconds): 1.16714
    URL Count:      2
    Domain Count:   2
    Domains:        facebook.com, google.com

With `debug=False`:


.. code:: text

    ___________Analysis__________

    Total Requests: 2
    Time (Seconds): 1.08454
    URL Count:      2
    Domain Count:   2
    Domains:        facebook.com, google.com


.. _requests: https://github.com/requests/requests
.. _tornado: https://github.com/tornadoweb/tornado
.. |Build Status| image:: https://travis-ci.org/danpozmanter/monitor_requests.svg?branch=master
   :target: https://travis-ci.org/danpozmanter/monitor_requests
.. |PyPI| image:: https://img.shields.io/pypi/v/monitor_requests.svg
   :target: https://pypi.python.org/pypi/monitor_requests/


