========== Middleware ========== Middleware is a framework of hooks into Django's request/response processing. It's a light, low-level "plugin" system for globally altering Django's input or output. Each middleware component is responsible for doing some specific function. For example, Django includes a middleware component, :class:`~django.contrib.auth.middleware.AuthenticationMiddleware`, that associates users with requests using sessions. This document explains how middleware works, how you activate middleware, and how to write your own middleware. Django ships with some built-in middleware you can use right out of the box. They're documented in the :doc:`built-in middleware reference </ref/middleware>`. Writing your own middleware =========================== A middleware factory is a callable that takes a ``get_response`` callable and returns a middleware. A middleware is a callable that takes a request and returns a response, just like a view. A middleware can be written as a function that looks like this:: def simple_middleware(get_response): # One-time configuration and initialization. def middleware(request): # Code to be executed for each request before # the view (and later middleware) are called. response = get_response(request) # Code to be executed for each request/response after # the view is called. return response return middleware Or it can be written as a class whose instances are callable, like this:: class SimpleMiddleware: def __init__(self, get_response): self.get_response = get_response # One-time configuration and initialization. def __call__(self, request): # Code to be executed for each request before # the view (and later middleware) are called. response = self.get_response(request) # Code to be executed for each request/response after # the view is called. return response The ``get_response`` callable provided by Django might be the actual view (if this is the last listed middleware) or it might be the next middleware in the chain. The current middleware doesn't need to know or care what exactly it is, just that it represents whatever comes next. The above is a slight simplification -- the ``get_response`` callable for the last middleware in the chain won't be the actual view but rather a wrapper method from the handler which takes care of applying :ref:`view middleware <view-middleware>`, calling the view with appropriate URL arguments, and applying :ref:`template-response <template-response-middleware>` and :ref:`exception <exception-middleware>` middleware. Middleware can either support only synchronous Python (the default), only asynchronous Python, or both. See :ref:`async-middleware` for details of how to advertise what you support, and know what kind of request you are getting. Middleware can live anywhere on your Python path. ``__init__(get_response)`` -------------------------- Middleware factories must accept a ``get_response`` argument. You can also initialize some global state for the middleware. Keep in mind a couple of caveats: * Django initializes your middleware with only the ``get_response`` argument, so you can't define ``__init__()`` as requiring any other arguments. * Unlike the ``__call__()`` method which is called once per request, ``__init__()`` is called only *once*, when the web server starts. Marking middleware as unused ---------------------------- It's sometimes useful to determine at startup time whether a piece of middleware should be used. In these cases, your middleware's ``__init__()`` method may raise :exc:`~django.core.exceptions.MiddlewareNotUsed`. Django will then remove that middleware from the middleware process and log a debug message to the :ref:`django-request-logger` logger when :setting:`DEBUG` is ``True``. Activating middleware ===================== To activate a middleware component, add it to the :setting:`MIDDLEWARE` list in your Django settings. In :setting:`MIDDLEWARE`, each middleware component is represented by a string: the full Python path to the middleware factory's class or function name. For example, here's the default value created by :djadmin:`django-admin startproject <startproject>`:: MIDDLEWARE = [ "django.middleware.security.SecurityMiddleware", "django.contrib.sessions.middleware.SessionMiddleware", "django.middleware.common.CommonMiddleware", "django.middleware.csrf.CsrfViewMiddleware", "django.contrib.auth.middleware.AuthenticationMiddleware", "django.contrib.messages.middleware.MessageMiddleware", "django.middleware.clickjacking.XFrameOptionsMiddleware", ] A Django installation doesn't require any middleware — :setting:`MIDDLEWARE` can be empty, if you'd like — but it's strongly suggested that you at least use :class:`~django.middleware.common.CommonMiddleware`. The order in :setting:`MIDDLEWARE` matters because a middleware can depend on other middleware. For instance, :class:`~django.contrib.auth.middleware.AuthenticationMiddleware` stores the authenticated user in the session; therefore, it must run after :class:`~django.contrib.sessions.middleware.SessionMiddleware`. See :ref:`middleware-ordering` for some common hints about ordering of Django middleware classes. Middleware order and layering ============================= During the request phase, before calling the view, Django applies middleware in the order it's defined in :setting:`MIDDLEWARE`, top-down. You can think of it like an onion: each middleware class is a "layer" that wraps the view, which is in the core of the onion. If the request passes through all the layers of the onion (each one calls ``get_response`` to pass the request in to the next layer), all the way to the view at the core, the response will then pass through every layer (in reverse order) on the way back out. If one of the layers decides to short-circuit and return a response without ever calling its ``get_response``, none of the layers of the onion inside that layer (including the view) will see the request or the response. The response will only return through the same layers that the request passed in through. Other middleware hooks ====================== Besides the basic request/response middleware pattern described earlier, you can add three other special methods to class-based middleware: .. _view-middleware: ``process_view()`` ------------------ .. method:: process_view(request, view_func, view_args, view_kwargs) ``request`` is an :class:`~django.http.HttpRequest` object. ``view_func`` is the Python function that Django is about to use. (It's the actual function object, not the name of the function as a string.) ``view_args`` is a list of positional arguments that will be passed to the view, and ``view_kwargs`` is a dictionary of keyword arguments that will be passed to the view. Neither ``view_args`` nor ``view_kwargs`` include the first view argument (``request``). ``process_view()`` is called just before Django calls the view. It should return either ``None`` or an :class:`~django.http.HttpResponse` object. If it returns ``None``, Django will continue processing this request, executing any other ``process_view()`` middleware and, then, the appropriate view. If it returns an :class:`~django.http.HttpResponse` object, Django won't bother calling the appropriate view; it'll apply response middleware to that :class:`~django.http.HttpResponse` and return the result. .. note:: Accessing :attr:`request.POST <django.http.HttpRequest.POST>` inside middleware before the view runs or in ``process_view()`` will prevent any view running after the middleware from being able to :ref:`modify the upload handlers for the request <modifying_upload_handlers_on_the_fly>`, and should normally be avoided. The :class:`~django.middleware.csrf.CsrfViewMiddleware` class can be considered an exception, as it provides the :func:`~django.views.decorators.csrf.csrf_exempt` and :func:`~django.views.decorators.csrf.csrf_protect` decorators which allow views to explicitly control at what point the CSRF validation should occur. .. _exception-middleware: ``process_exception()`` ----------------------- .. method:: process_exception(request, exception) ``request`` is an :class:`~django.http.HttpRequest` object. ``exception`` is an ``Exception`` object raised by the view function. Django calls ``process_exception()`` when a view raises an exception. ``process_exception()`` should return either ``None`` or an :class:`~django.http.HttpResponse` object. If it returns an :class:`~django.http.HttpResponse` object, the template response and response middleware will be applied and the resulting response returned to the browser. Otherwise, :ref:`default exception handling <error-views>` kicks in. Again, middleware are run in reverse order during the response phase, which includes ``process_exception``. If an exception middleware returns a response, the ``process_exception`` methods of the middleware classes above that middleware won't be called at all. .. _template-response-middleware: ``process_template_response()`` ------------------------------- .. method:: process_template_response(request, response) ``request`` is an :class:`~django.http.HttpRequest` object. ``response`` is the :class:`~django.template.response.TemplateResponse` object (or equivalent) returned by a Django view or by a middleware. ``process_template_response()`` is called just after the view has finished executing, if the response instance has a ``render()`` method, indicating that it is a :class:`~django.template.response.TemplateResponse` or equivalent. It must return a response object that implements a ``render`` method. It could alter the given ``response`` by changing ``response.template_name`` and ``response.context_data``, or it could create and return a brand-new :class:`~django.template.response.TemplateResponse` or equivalent. You don't need to explicitly render responses -- responses will be automatically rendered once all template response middleware has been called. Middleware are run in reverse order during the response phase, which includes ``process_template_response()``. Dealing with streaming responses ================================ Unlike :class:`~django.http.HttpResponse`, :class:`~django.http.StreamingHttpResponse` does not have a ``content`` attribute. As a result, middleware can no longer assume that all responses will have a ``content`` attribute. If they need access to the content, they must test for streaming responses and adjust their behavior accordingly:: if response.streaming: response.streaming_content = wrap_streaming_content(response.streaming_content) else: response.content = alter_content(response.content) .. note:: ``streaming_content`` should be assumed to be too large to hold in memory. Response middleware may wrap it in a new generator, but must not consume it. Wrapping is typically implemented as follows:: def wrap_streaming_content(content): for chunk in content: yield alter_content(chunk) :class:`~django.http.StreamingHttpResponse` allows both synchronous and asynchronous iterators. The wrapping function must match. Check :attr:`StreamingHttpResponse.is_async <django.http.StreamingHttpResponse.is_async>` if your middleware needs to support both types of iterator. .. versionchanged:: 4.2 Support for streaming responses with asynchronous iterators was added. Exception handling ================== Django automatically converts exceptions raised by the view or by middleware into an appropriate HTTP response with an error status code. :ref:`Certain exceptions <error-views>` are converted to 4xx status codes, while an unknown exception is converted to a 500 status code. This conversion takes place before and after each middleware (you can think of it as the thin film in between each layer of the onion), so that every middleware can always rely on getting some kind of HTTP response back from calling its ``get_response`` callable. Middleware don't need to worry about wrapping their call to ``get_response`` in a ``try/except`` and handling an exception that might have been raised by a later middleware or the view. Even if the very next middleware in the chain raises an :class:`~django.http.Http404` exception, for example, your middleware won't see that exception; instead it will get an :class:`~django.http.HttpResponse` object with a :attr:`~django.http.HttpResponse.status_code` of 404. You can set :setting:`DEBUG_PROPAGATE_EXCEPTIONS` to ``True`` to skip this conversion and propagate exceptions upward. .. _async-middleware: Asynchronous support ==================== Middleware can support any combination of synchronous and asynchronous requests. Django will adapt requests to fit the middleware's requirements if it cannot support both, but at a performance penalty. By default, Django assumes that your middleware is capable of handling only synchronous requests. To change these assumptions, set the following attributes on your middleware factory function or class: * ``sync_capable`` is a boolean indicating if the middleware can handle synchronous requests. Defaults to ``True``. * ``async_capable`` is a boolean indicating if the middleware can handle asynchronous requests. Defaults to ``False``. If your middleware has both ``sync_capable = True`` and ``async_capable = True``, then Django will pass it the request without converting it. In this case, you can work out if your middleware will receive async requests by checking if the ``get_response`` object you are passed is a coroutine function, using ``asgiref.sync.iscoroutinefunction``. The ``django.utils.decorators`` module contains :func:`~django.utils.decorators.sync_only_middleware`, :func:`~django.utils.decorators.async_only_middleware`, and :func:`~django.utils.decorators.sync_and_async_middleware` decorators that allow you to apply these flags to middleware factory functions. The returned callable must match the sync or async nature of the ``get_response`` method. If you have an asynchronous ``get_response``, you must return a coroutine function (``async def``). ``process_view``, ``process_template_response`` and ``process_exception`` methods, if they are provided, should also be adapted to match the sync/async mode. However, Django will individually adapt them as required if you do not, at an additional performance penalty. Here's an example of how to create a middleware function that supports both:: from asgiref.sync import iscoroutinefunction from django.utils.decorators import sync_and_async_middleware @sync_and_async_middleware def simple_middleware(get_response): # One-time configuration and initialization goes here. if iscoroutinefunction(get_response): async def middleware(request): # Do something here! response = await get_response(request) return response else: def middleware(request): # Do something here! response = get_response(request) return response return middleware .. note:: If you declare a hybrid middleware that supports both synchronous and asynchronous calls, the kind of call you get may not match the underlying view. Django will optimize the middleware call stack to have as few sync/async transitions as possible. Thus, even if you are wrapping an async view, you may be called in sync mode if there is other, synchronous middleware between you and the view. When using an asynchronous class-based middleware, you must ensure that instances are correctly marked as coroutine functions:: from asgiref.sync import iscoroutinefunction, markcoroutinefunction class AsyncMiddleware: async_capable = True sync_capable = False def __init__(self, get_response): self.get_response = get_response if iscoroutinefunction(self.get_response): markcoroutinefunction(self) async def __call__(self, request): response = await self.get_response(request) # Some logic ... return response .. _upgrading-middleware: Upgrading pre-Django 1.10-style middleware ========================================== .. class:: django.utils.deprecation.MiddlewareMixin :module: Django provides ``django.utils.deprecation.MiddlewareMixin`` to ease creating middleware classes that are compatible with both :setting:`MIDDLEWARE` and the old ``MIDDLEWARE_CLASSES``, and support synchronous and asynchronous requests. All middleware classes included with Django are compatible with both settings. The mixin provides an ``__init__()`` method that requires a ``get_response`` argument and stores it in ``self.get_response``. The ``__call__()`` method: #. Calls ``self.process_request(request)`` (if defined). #. Calls ``self.get_response(request)`` to get the response from later middleware and the view. #. Calls ``self.process_response(request, response)`` (if defined). #. Returns the response. If used with ``MIDDLEWARE_CLASSES``, the ``__call__()`` method will never be used; Django calls ``process_request()`` and ``process_response()`` directly. In most cases, inheriting from this mixin will be sufficient to make an old-style middleware compatible with the new system with sufficient backwards-compatibility. The new short-circuiting semantics will be harmless or even beneficial to the existing middleware. In a few cases, a middleware class may need some changes to adjust to the new semantics. These are the behavioral differences between using :setting:`MIDDLEWARE` and ``MIDDLEWARE_CLASSES``: #. Under ``MIDDLEWARE_CLASSES``, every middleware will always have its ``process_response`` method called, even if an earlier middleware short-circuited by returning a response from its ``process_request`` method. Under :setting:`MIDDLEWARE`, middleware behaves more like an onion: the layers that a response goes through on the way out are the same layers that saw the request on the way in. If a middleware short-circuits, only that middleware and the ones before it in :setting:`MIDDLEWARE` will see the response. #. Under ``MIDDLEWARE_CLASSES``, ``process_exception`` is applied to exceptions raised from a middleware ``process_request`` method. Under :setting:`MIDDLEWARE`, ``process_exception`` applies only to exceptions raised from the view (or from the ``render`` method of a :class:`~django.template.response.TemplateResponse`). Exceptions raised from a middleware are converted to the appropriate HTTP response and then passed to the next middleware. #. Under ``MIDDLEWARE_CLASSES``, if a ``process_response`` method raises an exception, the ``process_response`` methods of all earlier middleware are skipped and a ``500 Internal Server Error`` HTTP response is always returned (even if the exception raised was e.g. an :class:`~django.http.Http404`). Under :setting:`MIDDLEWARE`, an exception raised from a middleware will immediately be converted to the appropriate HTTP response, and then the next middleware in line will see that response. Middleware are never skipped due to a middleware raising an exception.