2013-03-17 18:21:05 +01:00
|
|
|
====================
|
|
|
|
Deployment checklist
|
|
|
|
====================
|
|
|
|
|
2021-07-28 11:56:56 +01:00
|
|
|
The internet is a hostile environment. Before deploying your Django project,
|
2013-03-17 18:21:05 +01:00
|
|
|
you should take some time to review your settings, with security, performance,
|
|
|
|
and operations in mind.
|
|
|
|
|
|
|
|
Django includes many :doc:`security features </topics/security>`. Some are
|
|
|
|
built-in and always enabled. Others are optional because they aren't always
|
|
|
|
appropriate, or because they're inconvenient for development. For example,
|
|
|
|
forcing HTTPS may not be suitable for all websites, and it's impractical for
|
|
|
|
local development.
|
|
|
|
|
|
|
|
Performance optimizations are another category of trade-offs with convenience.
|
|
|
|
For instance, caching is useful in production, less so for local development.
|
|
|
|
Error reporting needs are also widely different.
|
|
|
|
|
|
|
|
The following checklist includes settings that:
|
|
|
|
|
|
|
|
- must be set properly for Django to provide the expected level of security;
|
|
|
|
- are expected to be different in each environment;
|
|
|
|
- enable optional security features;
|
|
|
|
- enable performance optimizations;
|
|
|
|
- provide error reporting.
|
|
|
|
|
|
|
|
Many of these settings are sensitive and should be treated as confidential. If
|
|
|
|
you're releasing the source code for your project, a common practice is to
|
|
|
|
publish suitable settings for development, and to use a private settings
|
|
|
|
module for production.
|
|
|
|
|
2014-09-12 14:50:36 -04:00
|
|
|
Run ``manage.py check --deploy``
|
|
|
|
================================
|
|
|
|
|
2016-01-11 20:59:34 -05:00
|
|
|
Some of the checks described below can be automated using the :option:`check
|
|
|
|
--deploy` option. Be sure to run it against your production settings file as
|
|
|
|
described in the option's documentation.
|
2014-09-12 14:50:36 -04:00
|
|
|
|
2024-08-07 14:47:05 +01:00
|
|
|
Switch away from ``manage.py runserver``
|
|
|
|
========================================
|
|
|
|
|
|
|
|
The :djadmin:`runserver` command is not designed for a production setting. Be
|
|
|
|
sure to switch to a production-ready WSGI or ASGI server. For a few common
|
|
|
|
options, see :doc:`WSGI servers </howto/deployment/wsgi/index>` or
|
|
|
|
:doc:`ASGI servers </howto/deployment/asgi/index>`.
|
|
|
|
|
2013-03-17 18:21:05 +01:00
|
|
|
Critical settings
|
|
|
|
=================
|
|
|
|
|
|
|
|
:setting:`SECRET_KEY`
|
|
|
|
---------------------
|
|
|
|
|
|
|
|
**The secret key must be a large random value and it must be kept secret.**
|
|
|
|
|
|
|
|
Make sure that the key used in production isn't used anywhere else and avoid
|
|
|
|
committing it to source control. This reduces the number of vectors from which
|
|
|
|
an attacker may acquire the key.
|
|
|
|
|
|
|
|
Instead of hardcoding the secret key in your settings module, consider loading
|
|
|
|
it from an environment variable::
|
|
|
|
|
|
|
|
import os
|
2023-02-28 20:53:28 +01:00
|
|
|
|
2013-03-17 18:21:05 +01:00
|
|
|
SECRET_KEY = os.environ["SECRET_KEY"]
|
|
|
|
|
|
|
|
or from a file::
|
|
|
|
|
|
|
|
with open("/etc/secret_key.txt") as f:
|
|
|
|
SECRET_KEY = f.read().strip()
|
|
|
|
|
2021-12-13 21:47:03 -06:00
|
|
|
If rotating secret keys, you may use :setting:`SECRET_KEY_FALLBACKS`::
|
|
|
|
|
|
|
|
import os
|
2023-02-28 20:53:28 +01:00
|
|
|
|
2021-12-13 21:47:03 -06:00
|
|
|
SECRET_KEY = os.environ["CURRENT_SECRET_KEY"]
|
|
|
|
SECRET_KEY_FALLBACKS = [
|
|
|
|
os.environ["OLD_SECRET_KEY"],
|
|
|
|
]
|
|
|
|
|
|
|
|
Ensure that old secret keys are removed from ``SECRET_KEY_FALLBACKS`` in a
|
|
|
|
timely manner.
|
|
|
|
|
2013-03-17 18:21:05 +01:00
|
|
|
:setting:`DEBUG`
|
|
|
|
----------------
|
|
|
|
|
|
|
|
**You must never enable debug in production.**
|
|
|
|
|
|
|
|
You're certainly developing your project with :setting:`DEBUG = True <DEBUG>`,
|
|
|
|
since this enables handy features like full tracebacks in your browser.
|
|
|
|
|
|
|
|
For a production environment, though, this is a really bad idea, because it
|
|
|
|
leaks lots of information about your project: excerpts of your source code,
|
|
|
|
local variables, settings, libraries used, etc.
|
|
|
|
|
|
|
|
Environment-specific settings
|
|
|
|
=============================
|
|
|
|
|
|
|
|
:setting:`ALLOWED_HOSTS`
|
|
|
|
------------------------
|
|
|
|
|
|
|
|
When :setting:`DEBUG = False <DEBUG>`, Django doesn't work at all without a
|
|
|
|
suitable value for :setting:`ALLOWED_HOSTS`.
|
|
|
|
|
|
|
|
This setting is required to protect your site against some CSRF attacks. If
|
|
|
|
you use a wildcard, you must perform your own validation of the ``Host`` HTTP
|
|
|
|
header, or otherwise ensure that you aren't vulnerable to this category of
|
|
|
|
attacks.
|
|
|
|
|
2021-07-23 07:48:16 +01:00
|
|
|
You should also configure the web server that sits in front of Django to
|
2015-06-11 10:00:33 -04:00
|
|
|
validate the host. It should respond with a static error page or ignore
|
|
|
|
requests for incorrect hosts instead of forwarding the request to Django. This
|
|
|
|
way you'll avoid spurious errors in your Django logs (or emails if you have
|
2021-08-15 20:11:25 +01:00
|
|
|
error reporting configured that way). For example, on nginx you might set up a
|
2015-06-11 10:00:33 -04:00
|
|
|
default server to return "444 No Response" on an unrecognized host:
|
|
|
|
|
|
|
|
.. code-block:: nginx
|
|
|
|
|
|
|
|
server {
|
|
|
|
listen 80 default_server;
|
|
|
|
return 444;
|
|
|
|
}
|
|
|
|
|
2013-03-17 18:21:05 +01:00
|
|
|
:setting:`CACHES`
|
|
|
|
-----------------
|
|
|
|
|
|
|
|
If you're using a cache, connection parameters may be different in development
|
2016-12-28 19:42:46 -05:00
|
|
|
and in production. Django defaults to per-process :ref:`local-memory caching
|
|
|
|
<local-memory-caching>` which may not be desirable.
|
2013-03-17 18:21:05 +01:00
|
|
|
|
|
|
|
Cache servers often have weak authentication. Make sure they only accept
|
|
|
|
connections from your application servers.
|
|
|
|
|
|
|
|
:setting:`DATABASES`
|
|
|
|
--------------------
|
|
|
|
|
|
|
|
Database connection parameters are probably different in development and in
|
|
|
|
production.
|
|
|
|
|
2013-03-17 19:29:22 +01:00
|
|
|
Database passwords are very sensitive. You should protect them exactly like
|
|
|
|
:setting:`SECRET_KEY`.
|
|
|
|
|
2013-03-17 18:21:05 +01:00
|
|
|
For maximum security, make sure database servers only accept connections from
|
|
|
|
your application servers.
|
|
|
|
|
|
|
|
If you haven't set up backups for your database, do it right now!
|
|
|
|
|
|
|
|
:setting:`EMAIL_BACKEND` and related settings
|
|
|
|
---------------------------------------------
|
|
|
|
|
|
|
|
If your site sends emails, these values need to be set correctly.
|
|
|
|
|
2015-07-28 09:44:08 -04:00
|
|
|
By default, Django sends email from webmaster@localhost and root@localhost.
|
|
|
|
However, some mail providers reject email from these addresses. To use
|
|
|
|
different sender addresses, modify the :setting:`DEFAULT_FROM_EMAIL` and
|
|
|
|
:setting:`SERVER_EMAIL` settings.
|
2015-06-11 10:00:33 -04:00
|
|
|
|
2013-03-17 18:21:05 +01:00
|
|
|
:setting:`STATIC_ROOT` and :setting:`STATIC_URL`
|
|
|
|
------------------------------------------------
|
|
|
|
|
|
|
|
Static files are automatically served by the development server. In
|
|
|
|
production, you must define a :setting:`STATIC_ROOT` directory where
|
|
|
|
:djadmin:`collectstatic` will copy them.
|
|
|
|
|
2013-03-07 14:15:39 -05:00
|
|
|
See :doc:`/howto/static-files/index` for more information.
|
2013-03-17 18:21:05 +01:00
|
|
|
|
|
|
|
:setting:`MEDIA_ROOT` and :setting:`MEDIA_URL`
|
|
|
|
----------------------------------------------
|
|
|
|
|
|
|
|
Media files are uploaded by your users. They're untrusted! Make sure your web
|
2016-09-30 00:59:00 +01:00
|
|
|
server never attempts to interpret them. For instance, if a user uploads a
|
|
|
|
``.php`` file, the web server shouldn't execute it.
|
2013-03-17 18:21:05 +01:00
|
|
|
|
|
|
|
Now is a good time to check your backup strategy for these files.
|
|
|
|
|
|
|
|
HTTPS
|
|
|
|
=====
|
|
|
|
|
|
|
|
Any website which allows users to log in should enforce site-wide HTTPS to
|
|
|
|
avoid transmitting access tokens in clear. In Django, access tokens include
|
|
|
|
the login/password, the session cookie, and password reset tokens. (You can't
|
|
|
|
do much to protect password reset tokens if you're sending them by email.)
|
|
|
|
|
|
|
|
Protecting sensitive areas such as the user account or the admin isn't
|
2013-03-17 19:29:22 +01:00
|
|
|
sufficient, because the same session cookie is used for HTTP and HTTPS. Your
|
|
|
|
web server must redirect all HTTP traffic to HTTPS, and only transmit HTTPS
|
|
|
|
requests to Django.
|
2013-03-17 18:21:05 +01:00
|
|
|
|
|
|
|
Once you've set up HTTPS, enable the following settings.
|
|
|
|
|
|
|
|
:setting:`CSRF_COOKIE_SECURE`
|
|
|
|
-----------------------------
|
|
|
|
|
|
|
|
Set this to ``True`` to avoid transmitting the CSRF cookie over HTTP
|
|
|
|
accidentally.
|
|
|
|
|
|
|
|
:setting:`SESSION_COOKIE_SECURE`
|
|
|
|
--------------------------------
|
|
|
|
|
|
|
|
Set this to ``True`` to avoid transmitting the session cookie over HTTP
|
|
|
|
accidentally.
|
|
|
|
|
|
|
|
Performance optimizations
|
|
|
|
=========================
|
|
|
|
|
|
|
|
Setting :setting:`DEBUG = False <DEBUG>` disables several features that are
|
|
|
|
only useful in development. In addition, you can tune the following settings.
|
|
|
|
|
2020-03-25 11:04:52 +00:00
|
|
|
Sessions
|
|
|
|
--------
|
|
|
|
|
|
|
|
Consider using :ref:`cached sessions <cached-sessions-backend>` to improve
|
|
|
|
performance.
|
|
|
|
|
|
|
|
If using database-backed sessions, regularly :ref:`clear old sessions
|
|
|
|
<clearing-the-session-store>` to avoid storing unnecessary data.
|
|
|
|
|
2013-05-09 15:42:14 +02:00
|
|
|
:setting:`CONN_MAX_AGE`
|
|
|
|
-----------------------
|
|
|
|
|
2013-05-27 11:14:16 +02:00
|
|
|
Enabling :ref:`persistent database connections
|
|
|
|
<persistent-database-connections>` can result in a nice speed-up when
|
|
|
|
connecting to the database accounts for a significant part of the request
|
|
|
|
processing time.
|
2013-05-09 15:42:14 +02:00
|
|
|
|
|
|
|
This helps a lot on virtualized hosts with limited network performance.
|
|
|
|
|
2014-12-17 22:10:57 +01:00
|
|
|
:setting:`TEMPLATES`
|
|
|
|
--------------------
|
2013-03-17 18:21:05 +01:00
|
|
|
|
|
|
|
Enabling the cached template loader often improves performance drastically, as
|
2022-03-22 11:26:05 +01:00
|
|
|
it avoids compiling each template every time it needs to be rendered. When
|
|
|
|
:setting:`DEBUG = False <DEBUG>`, the cached template loader is enabled
|
|
|
|
automatically. See :class:`django.template.loaders.cached.Loader` for more
|
|
|
|
information.
|
2013-03-17 18:21:05 +01:00
|
|
|
|
|
|
|
Error reporting
|
|
|
|
===============
|
|
|
|
|
|
|
|
By the time you push your code to production, it's hopefully robust, but you
|
|
|
|
can't rule out unexpected errors. Thankfully, Django can capture errors and
|
|
|
|
notify you accordingly.
|
|
|
|
|
|
|
|
:setting:`LOGGING`
|
|
|
|
------------------
|
|
|
|
|
|
|
|
Review your logging configuration before putting your website in production,
|
|
|
|
and check that it works as expected as soon as you have received some traffic.
|
|
|
|
|
|
|
|
See :doc:`/topics/logging` for details on logging.
|
|
|
|
|
|
|
|
:setting:`ADMINS` and :setting:`MANAGERS`
|
|
|
|
-----------------------------------------
|
|
|
|
|
|
|
|
:setting:`ADMINS` will be notified of 500 errors by email.
|
|
|
|
|
|
|
|
:setting:`MANAGERS` will be notified of 404 errors.
|
|
|
|
:setting:`IGNORABLE_404_URLS` can help filter out spurious reports.
|
|
|
|
|
|
|
|
See :doc:`/howto/error-reporting` for details on error reporting by email.
|
|
|
|
|
2015-07-25 06:37:51 -04:00
|
|
|
.. admonition:: Error reporting by email doesn't scale very well
|
2013-03-17 18:21:05 +01:00
|
|
|
|
|
|
|
Consider using an error monitoring system such as Sentry_ before your
|
|
|
|
inbox is flooded by reports. Sentry can also aggregate logs.
|
|
|
|
|
2017-05-20 17:51:21 +02:00
|
|
|
.. _Sentry: https://docs.sentry.io/
|
2013-03-17 18:21:05 +01:00
|
|
|
|
2013-08-01 18:07:59 -04:00
|
|
|
Customize the default error views
|
|
|
|
---------------------------------
|
|
|
|
|
|
|
|
Django includes default views and templates for several HTTP error codes. You
|
|
|
|
may want to override the default templates by creating the following templates
|
|
|
|
in your root template directory: ``404.html``, ``500.html``, ``403.html``, and
|
2018-10-11 12:46:46 +01:00
|
|
|
``400.html``. The :ref:`default error views <error-views>` that use these
|
2021-07-23 07:48:16 +01:00
|
|
|
templates should suffice for 99% of web applications, but you can
|
2018-10-11 12:46:46 +01:00
|
|
|
:ref:`customize them <customizing-error-views>` as well.
|