django-request-profiler


Namedjango-request-profiler JSON
Version 1.1 PyPI version JSON
download
home_pagehttps://github.com/yunojuno/django-request-profiler
SummaryA simple Django project profiler for timing HTTP requests.
upload_time2023-11-14 11:43:29
maintainer
docs_urlNone
authorYunoJuno
requires_python>=3.9,<4.0
licenseMIT
keywords
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage
            Django Request Profiler
=======================

**This package now requires Python 3.9 and Django 3.2 and above.**

A very simple request profiler for Django.

Introduction
------------

    Premature optimization is the root of all evil.

There are a lot of very good, and complete, python and django profilers
available. They can give you detailed stack traces and function call timings,
output all the SQL statements that have been run, the templates that have been
rendered, and the state of any / all variables along the way. These tools are
great for optimisation of your application, once you have decided that the
time is right.

``django-request-profiler`` is not intended to help you optimise, but to help
you decide whether you need to optimise in the first place. It is complimentary.

Requirements
------------

1. Small enough to run in production
2. Able to configure profiling at runtime
3. Configurable to target specific URLs or users
4. Record basic request metadata:

- Duration (request-response)
- Request path, remote addr, user-agent
- Response status code, content length
- View function
- Django user and session keys (if appropriate)
- Database query count (if DEBUG=True)

It doesn't need to record all the inner timing information - the goal is to have
a system that can be used to monitor site response times, and to identify
problem areas ahead of time.

Technical details
-----------------

The profiler itself runs as Django middleware, and it simply starts a timer when
it first sees the request, and stops the timer when it is finished with the
response. It should be installed as the first middleware in
``MIDDLEWARE_CLASSES`` in order to record the maximum duration.

It hooks into the  ``process_request`` method to start the timer, the
``process_view`` method to record the view function name, and the
``process_response`` method to stop the timer, record all the request
information and store the instance.

The profiler is controlled by adding ``RuleSet`` instances which are used to
filter which requests are profiled. There can be many, overlapping,
RuleSets, but if any match, the request is profiled. The RuleSet model
defines two core matching methods:

1. uri_regex - in order to profile a subset of the site, you can supply a regex
which is used match the incoming request path. If the url matches, the request
can be profiled.

2. user_filter_type - there are three choices here - profile all users, profile
only authenticated users, and profile authenticated users belonging to a given
Group - e.g. create a groups called "profiling" and add anyone you want to
profile.

These filter properties are an AND (must pass the uri and user filter), but the
rules as a group are an OR - so if a request passes all the filters in any rule,
then it's profiled.

These filters are pretty blunt, and there are plenty of use cases where you may
want more sophisticated control over the profiling. There are two ways to do
this. The first is a setting, ``REQUEST_PROFILER_GLOBAL_EXCLUDE_FUNC``, which is
a function that takes a request as the single argument, and must return True or
False. If it returns False, the profile is cancelled, irrespective of any rules.
The primary use case for this is to exclude common requests that you are not
interested in, e.g. from search engine bots, or from Admin users etc. The
default for this function is to prevent admin user requests from being profiled.

The second control is via the ``cancel()`` method on the ``ProfilingRecord``,
which is accessible via the ``request_profile_complete`` signal. By hooking
in to this signal you can add additional processing, and optionally cancel
the profiler. A typical use case for this is to log requests that have
exceeded a set request duration threshold. In a high volume environment you
may want to, for instance, only profile a random subset of all requests.

.. code:: python

    from django.dispatch import receiver
    from request_profiler.signals import request_profile_complete

    @receiver(request_profiler_complete)
    def on_request_profile_complete(sender, **kwargs):
        profiler = kwargs.get('instance')
        if profiler.elapsed > 2:
            # log long-running requests
            # NB please don't use 'print' for real - use logging
            print u"Long-running request warning: %s" % profiler
        else:
            # calling cancel means that it won't be saved to the db
            profiler.cancel()

An additional scenario where you may want to use the signal is to store
the profiler records async - say if you are recording every request for
a short period, and you don't want to add unnecessary inline database
write operations. In this case you can use the ``stop()`` method, which
will prevent the middleware from saving it directly (it will only save
records where ``profiler.is_running`` is true, and both ``cancel`` and
``stop`` set it to false).

.. code:: python

    from django.dispatch import receiver
    from request_profiler.signals import request_profile_complete

    @receiver(request_profiler_complete)
    def on_request_profile_complete(sender, **kwargs):
        profiler = kwargs.get('instance')
        # stop the profiler to prevent it from being saved automatically
        profiler.stop()
        assert not profiler.is_running
        # add a job to a queue to perform the save itself
        queue.enqueue(profiler.save)


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

For use as the app in Django project, use pip:

.. code:: shell

    $ pip install django-request-profiler
    # For hacking on the project, pull from Git:
    $ git pull git@github.com:yunojuno/django-request-profiler.git

Tests
-----

The app installer contains a test suite that can be run using the Django
test runner:

.. code:: shell

    $ pip install -r requirements.txt
    $ python manage.py test test_app request_profiler

If you want to test coverage you'll need to add some dependencies:

.. code:: shell

    $ pip install coverage django-coverage
    $ python manage.py test_coverage test_app request_profiler

The tests also run using `tox <https://testrun.org/tox/latest/>`_:

.. code:: shell

    $ pip install tox
    $ tox

**Note: To test with a custom user model, you should override the default User model
by providing a value for the AUTH_USER_MODEL (in testapp/settings) setting that references a custom model**

The tests run on `Travis <https://travis-ci.org/yunojuno/django-request-profiler>`_ on commits to master.

Usage
-----

Once installed, add the app and middleware to your project's settings file.
In order to add the database tables, you should run the ``migrate`` command:

.. code:: bash

    $ python manage.py migrate request_profiler

NB the middleware must be the **first** item in ``MIDDLEWARE_CLASSES``.

.. code:: python

    INSTALLED_APPS = (
        'django.contrib.admin',
        'django.contrib.auth',
        'django.contrib.contenttypes',
        'django.contrib.sessions',
        'django.contrib.messages',
        'django.contrib.staticfiles',
        'request_profiler',
    )

    MIDDLEWARE_CLASSES = [
        # this package's middleware
        'request_profiler.middleware.ProfilingMiddleware',
        # default django middleware
        'django.middleware.common.CommonMiddleware',
        'django.contrib.sessions.middleware.SessionMiddleware',
        'django.contrib.auth.middleware.AuthenticationMiddleware',
        'django.middleware.csrf.CsrfViewMiddleware',
        'django.contrib.messages.middleware.MessageMiddleware',
    ]

Configuration
-------------

To configure the app, open the admin site, and add a new request profiler
'Rule set'. The default options will result in all non-admin requests being
profiled.

Licence
-------

MIT (see LICENCE)

            

Raw data

            {
    "_id": null,
    "home_page": "https://github.com/yunojuno/django-request-profiler",
    "name": "django-request-profiler",
    "maintainer": "",
    "docs_url": null,
    "requires_python": ">=3.9,<4.0",
    "maintainer_email": "",
    "keywords": "",
    "author": "YunoJuno",
    "author_email": "code@yunojuno.com",
    "download_url": "https://files.pythonhosted.org/packages/0c/04/e74bd18194cb3e2f49485c67d8de6152d5007eb94925458e02a1499d5ec5/django_request_profiler-1.1.tar.gz",
    "platform": null,
    "description": "Django Request Profiler\n=======================\n\n**This package now requires Python 3.9 and Django 3.2 and above.**\n\nA very simple request profiler for Django.\n\nIntroduction\n------------\n\n    Premature optimization is the root of all evil.\n\nThere are a lot of very good, and complete, python and django profilers\navailable. They can give you detailed stack traces and function call timings,\noutput all the SQL statements that have been run, the templates that have been\nrendered, and the state of any / all variables along the way. These tools are\ngreat for optimisation of your application, once you have decided that the\ntime is right.\n\n``django-request-profiler`` is not intended to help you optimise, but to help\nyou decide whether you need to optimise in the first place. It is complimentary.\n\nRequirements\n------------\n\n1. Small enough to run in production\n2. Able to configure profiling at runtime\n3. Configurable to target specific URLs or users\n4. Record basic request metadata:\n\n- Duration (request-response)\n- Request path, remote addr, user-agent\n- Response status code, content length\n- View function\n- Django user and session keys (if appropriate)\n- Database query count (if DEBUG=True)\n\nIt doesn't need to record all the inner timing information - the goal is to have\na system that can be used to monitor site response times, and to identify\nproblem areas ahead of time.\n\nTechnical details\n-----------------\n\nThe profiler itself runs as Django middleware, and it simply starts a timer when\nit first sees the request, and stops the timer when it is finished with the\nresponse. It should be installed as the first middleware in\n``MIDDLEWARE_CLASSES`` in order to record the maximum duration.\n\nIt hooks into the  ``process_request`` method to start the timer, the\n``process_view`` method to record the view function name, and the\n``process_response`` method to stop the timer, record all the request\ninformation and store the instance.\n\nThe profiler is controlled by adding ``RuleSet`` instances which are used to\nfilter which requests are profiled. There can be many, overlapping,\nRuleSets, but if any match, the request is profiled. The RuleSet model\ndefines two core matching methods:\n\n1. uri_regex - in order to profile a subset of the site, you can supply a regex\nwhich is used match the incoming request path. If the url matches, the request\ncan be profiled.\n\n2. user_filter_type - there are three choices here - profile all users, profile\nonly authenticated users, and profile authenticated users belonging to a given\nGroup - e.g. create a groups called \"profiling\" and add anyone you want to\nprofile.\n\nThese filter properties are an AND (must pass the uri and user filter), but the\nrules as a group are an OR - so if a request passes all the filters in any rule,\nthen it's profiled.\n\nThese filters are pretty blunt, and there are plenty of use cases where you may\nwant more sophisticated control over the profiling. There are two ways to do\nthis. The first is a setting, ``REQUEST_PROFILER_GLOBAL_EXCLUDE_FUNC``, which is\na function that takes a request as the single argument, and must return True or\nFalse. If it returns False, the profile is cancelled, irrespective of any rules.\nThe primary use case for this is to exclude common requests that you are not\ninterested in, e.g. from search engine bots, or from Admin users etc. The\ndefault for this function is to prevent admin user requests from being profiled.\n\nThe second control is via the ``cancel()`` method on the ``ProfilingRecord``,\nwhich is accessible via the ``request_profile_complete`` signal. By hooking\nin to this signal you can add additional processing, and optionally cancel\nthe profiler. A typical use case for this is to log requests that have\nexceeded a set request duration threshold. In a high volume environment you\nmay want to, for instance, only profile a random subset of all requests.\n\n.. code:: python\n\n    from django.dispatch import receiver\n    from request_profiler.signals import request_profile_complete\n\n    @receiver(request_profiler_complete)\n    def on_request_profile_complete(sender, **kwargs):\n        profiler = kwargs.get('instance')\n        if profiler.elapsed > 2:\n            # log long-running requests\n            # NB please don't use 'print' for real - use logging\n            print u\"Long-running request warning: %s\" % profiler\n        else:\n            # calling cancel means that it won't be saved to the db\n            profiler.cancel()\n\nAn additional scenario where you may want to use the signal is to store\nthe profiler records async - say if you are recording every request for\na short period, and you don't want to add unnecessary inline database\nwrite operations. In this case you can use the ``stop()`` method, which\nwill prevent the middleware from saving it directly (it will only save\nrecords where ``profiler.is_running`` is true, and both ``cancel`` and\n``stop`` set it to false).\n\n.. code:: python\n\n    from django.dispatch import receiver\n    from request_profiler.signals import request_profile_complete\n\n    @receiver(request_profiler_complete)\n    def on_request_profile_complete(sender, **kwargs):\n        profiler = kwargs.get('instance')\n        # stop the profiler to prevent it from being saved automatically\n        profiler.stop()\n        assert not profiler.is_running\n        # add a job to a queue to perform the save itself\n        queue.enqueue(profiler.save)\n\n\nInstallation\n------------\n\nFor use as the app in Django project, use pip:\n\n.. code:: shell\n\n    $ pip install django-request-profiler\n    # For hacking on the project, pull from Git:\n    $ git pull git@github.com:yunojuno/django-request-profiler.git\n\nTests\n-----\n\nThe app installer contains a test suite that can be run using the Django\ntest runner:\n\n.. code:: shell\n\n    $ pip install -r requirements.txt\n    $ python manage.py test test_app request_profiler\n\nIf you want to test coverage you'll need to add some dependencies:\n\n.. code:: shell\n\n    $ pip install coverage django-coverage\n    $ python manage.py test_coverage test_app request_profiler\n\nThe tests also run using `tox <https://testrun.org/tox/latest/>`_:\n\n.. code:: shell\n\n    $ pip install tox\n    $ tox\n\n**Note: To test with a custom user model, you should override the default User model\nby providing a value for the AUTH_USER_MODEL (in testapp/settings) setting that references a custom model**\n\nThe tests run on `Travis <https://travis-ci.org/yunojuno/django-request-profiler>`_ on commits to master.\n\nUsage\n-----\n\nOnce installed, add the app and middleware to your project's settings file.\nIn order to add the database tables, you should run the ``migrate`` command:\n\n.. code:: bash\n\n    $ python manage.py migrate request_profiler\n\nNB the middleware must be the **first** item in ``MIDDLEWARE_CLASSES``.\n\n.. code:: python\n\n    INSTALLED_APPS = (\n        'django.contrib.admin',\n        'django.contrib.auth',\n        'django.contrib.contenttypes',\n        'django.contrib.sessions',\n        'django.contrib.messages',\n        'django.contrib.staticfiles',\n        'request_profiler',\n    )\n\n    MIDDLEWARE_CLASSES = [\n        # this package's middleware\n        'request_profiler.middleware.ProfilingMiddleware',\n        # default django middleware\n        'django.middleware.common.CommonMiddleware',\n        'django.contrib.sessions.middleware.SessionMiddleware',\n        'django.contrib.auth.middleware.AuthenticationMiddleware',\n        'django.middleware.csrf.CsrfViewMiddleware',\n        'django.contrib.messages.middleware.MessageMiddleware',\n    ]\n\nConfiguration\n-------------\n\nTo configure the app, open the admin site, and add a new request profiler\n'Rule set'. The default options will result in all non-admin requests being\nprofiled.\n\nLicence\n-------\n\nMIT (see LICENCE)\n",
    "bugtrack_url": null,
    "license": "MIT",
    "summary": "A simple Django project profiler for timing HTTP requests.",
    "version": "1.1",
    "project_urls": {
        "Homepage": "https://github.com/yunojuno/django-request-profiler",
        "Repository": "https://github.com/yunojuno/django-request-profiler"
    },
    "split_keywords": [],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "cbe87127ff2c6e7fbdb2920c63d94fed9451371298e1bdea2e1963ae773dbd47",
                "md5": "fcb91b851a86cbdd8b5473c9b339823f",
                "sha256": "00052d49bf3f6028281e787c65155c5527e7e6533f910994cc7fc2737e07da5a"
            },
            "downloads": -1,
            "filename": "django_request_profiler-1.1-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "fcb91b851a86cbdd8b5473c9b339823f",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": ">=3.9,<4.0",
            "size": 17300,
            "upload_time": "2023-11-14T11:43:27",
            "upload_time_iso_8601": "2023-11-14T11:43:27.536185Z",
            "url": "https://files.pythonhosted.org/packages/cb/e8/7127ff2c6e7fbdb2920c63d94fed9451371298e1bdea2e1963ae773dbd47/django_request_profiler-1.1-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "0c04e74bd18194cb3e2f49485c67d8de6152d5007eb94925458e02a1499d5ec5",
                "md5": "1f07c02394cf6899c1f250ca2dcb9248",
                "sha256": "d42c056ee2b3630f5f8f6a412982f6893b5d56221408ae3888d5653bbed69910"
            },
            "downloads": -1,
            "filename": "django_request_profiler-1.1.tar.gz",
            "has_sig": false,
            "md5_digest": "1f07c02394cf6899c1f250ca2dcb9248",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": ">=3.9,<4.0",
            "size": 15010,
            "upload_time": "2023-11-14T11:43:29",
            "upload_time_iso_8601": "2023-11-14T11:43:29.513336Z",
            "url": "https://files.pythonhosted.org/packages/0c/04/e74bd18194cb3e2f49485c67d8de6152d5007eb94925458e02a1499d5ec5/django_request_profiler-1.1.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2023-11-14 11:43:29",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "yunojuno",
    "github_project": "django-request-profiler",
    "travis_ci": false,
    "coveralls": true,
    "github_actions": true,
    "tox": true,
    "lcname": "django-request-profiler"
}
        
Elapsed time: 0.14206s