2005-08-17 04:00:02 +00:00
|
|
|
|
===================
|
|
|
|
|
How to use sessions
|
|
|
|
|
===================
|
|
|
|
|
|
2009-06-18 13:32:12 +00:00
|
|
|
|
.. module:: django.contrib.sessions
|
|
|
|
|
:synopsis: Provides session management for Django projects.
|
|
|
|
|
|
2011-06-26 17:00:24 +00:00
|
|
|
|
Django provides full support for anonymous sessions. The session framework
|
|
|
|
|
lets you store and retrieve arbitrary data on a per-site-visitor basis. It
|
|
|
|
|
stores data on the server side and abstracts the sending and receiving of
|
|
|
|
|
cookies. Cookies contain a session ID -- not the data itself (unless you're
|
|
|
|
|
using the :ref:`cookie based backend<cookie-session-backend>`).
|
2005-08-17 04:00:02 +00:00
|
|
|
|
|
|
|
|
|
Enabling sessions
|
|
|
|
|
=================
|
|
|
|
|
|
2010-08-19 19:27:44 +00:00
|
|
|
|
Sessions are implemented via a piece of :doc:`middleware </ref/middleware>`.
|
2005-08-17 04:00:02 +00:00
|
|
|
|
|
2007-09-15 21:29:14 +00:00
|
|
|
|
To enable session functionality, do the following:
|
2005-08-17 04:00:02 +00:00
|
|
|
|
|
2015-11-07 16:12:37 +01:00
|
|
|
|
* Edit the :setting:`MIDDLEWARE` setting and make sure it contains
|
|
|
|
|
``'django.contrib.sessions.middleware.SessionMiddleware'``. The default
|
|
|
|
|
``settings.py`` created by ``django-admin startproject`` has
|
|
|
|
|
``SessionMiddleware`` activated.
|
2007-09-15 22:36:53 +00:00
|
|
|
|
|
2005-08-17 04:00:02 +00:00
|
|
|
|
If you don't want to use sessions, you might as well remove the
|
2015-11-07 16:12:37 +01:00
|
|
|
|
``SessionMiddleware`` line from :setting:`MIDDLEWARE` and
|
2011-05-20 00:51:25 +00:00
|
|
|
|
``'django.contrib.sessions'`` from your :setting:`INSTALLED_APPS`.
|
|
|
|
|
It'll save you a small bit of overhead.
|
2005-08-17 04:00:02 +00:00
|
|
|
|
|
2013-01-12 18:44:53 -05:00
|
|
|
|
.. _configuring-sessions:
|
|
|
|
|
|
2007-09-15 21:29:14 +00:00
|
|
|
|
Configuring the session engine
|
|
|
|
|
==============================
|
|
|
|
|
|
|
|
|
|
By default, Django stores sessions in your database (using the model
|
|
|
|
|
``django.contrib.sessions.models.Session``). Though this is convenient, in
|
|
|
|
|
some setups it's faster to store session data elsewhere, so Django can be
|
|
|
|
|
configured to store session data on your filesystem or in your cache.
|
|
|
|
|
|
2009-12-22 15:18:51 +00:00
|
|
|
|
Using database-backed sessions
|
|
|
|
|
------------------------------
|
|
|
|
|
|
|
|
|
|
If you want to use a database-backed session, you need to add
|
2011-05-29 17:41:04 +00:00
|
|
|
|
``'django.contrib.sessions'`` to your :setting:`INSTALLED_APPS` setting.
|
2009-12-22 15:18:51 +00:00
|
|
|
|
|
2013-07-25 16:19:36 +01:00
|
|
|
|
Once you have configured your installation, run ``manage.py migrate``
|
2009-12-22 15:18:51 +00:00
|
|
|
|
to install the single database table that stores session data.
|
|
|
|
|
|
2012-10-30 21:59:23 +01:00
|
|
|
|
.. _cached-sessions-backend:
|
|
|
|
|
|
2009-01-10 22:18:14 +00:00
|
|
|
|
Using cached sessions
|
|
|
|
|
---------------------
|
|
|
|
|
|
|
|
|
|
For better performance, you may want to use a cache-based session backend.
|
|
|
|
|
|
|
|
|
|
To store session data using Django's cache system, you'll first need to make
|
2010-08-19 19:27:44 +00:00
|
|
|
|
sure you've configured your cache; see the :doc:`cache documentation
|
|
|
|
|
</topics/cache>` for details.
|
2009-01-10 22:18:14 +00:00
|
|
|
|
|
|
|
|
|
.. warning::
|
|
|
|
|
|
2022-02-02 20:45:37 +01:00
|
|
|
|
You should only use cache-based sessions if you're using the Memcached or
|
|
|
|
|
Redis cache backend. The local-memory cache backend doesn't retain data
|
|
|
|
|
long enough to be a good choice, and it'll be faster to use file or
|
|
|
|
|
database sessions directly instead of sending everything through the file
|
|
|
|
|
or database cache backends. Additionally, the local-memory cache backend is
|
2014-07-16 12:32:57 -04:00
|
|
|
|
NOT multi-process safe, therefore probably not a good choice for production
|
|
|
|
|
environments.
|
2009-01-10 22:18:14 +00:00
|
|
|
|
|
2012-10-30 21:59:23 +01:00
|
|
|
|
If you have multiple caches defined in :setting:`CACHES`, Django will use the
|
|
|
|
|
default cache. To use another cache, set :setting:`SESSION_CACHE_ALIAS` to the
|
|
|
|
|
name of that cache.
|
|
|
|
|
|
2022-10-07 09:39:35 +02:00
|
|
|
|
Once your cache is configured, you have to choose between a database-backed
|
|
|
|
|
cache or a non-persistent cache.
|
|
|
|
|
|
|
|
|
|
The cached database backend (``cached_db``) uses a write-through cache --
|
|
|
|
|
session writes are applied to both the cache and the database. Session reads
|
|
|
|
|
use the cache, or the database if the data has been evicted from the cache. To
|
|
|
|
|
use this backend, set :setting:`SESSION_ENGINE` to
|
|
|
|
|
``"django.contrib.sessions.backends.cached_db"``, and follow the configuration
|
|
|
|
|
instructions for the `using database-backed sessions`_.
|
|
|
|
|
|
|
|
|
|
The cache backend (``cache``) stores session data only in your cache. This is
|
|
|
|
|
faster because it avoids database persistence, but you will have to consider
|
|
|
|
|
what happens when cache data is evicted. Eviction can occur if the cache fills
|
|
|
|
|
up or the cache server is restarted, and it will mean session data is lost,
|
|
|
|
|
including logging out users. To use this backend, set :setting:`SESSION_ENGINE`
|
|
|
|
|
to ``"django.contrib.sessions.backends.cache"``.
|
|
|
|
|
|
|
|
|
|
The cache backend can be made persistent by using a persistent cache, such as
|
|
|
|
|
Redis with appropriate configuration. But unless your cache is definitely
|
|
|
|
|
configured for sufficient persistence, opt for the cached database backend.
|
|
|
|
|
This avoids edge cases caused by unreliable data storage in production.
|
2009-12-22 15:18:51 +00:00
|
|
|
|
|
2007-09-15 21:29:14 +00:00
|
|
|
|
Using file-based sessions
|
|
|
|
|
-------------------------
|
|
|
|
|
|
2011-05-20 00:51:25 +00:00
|
|
|
|
To use file-based sessions, set the :setting:`SESSION_ENGINE` setting to
|
2007-09-15 21:29:14 +00:00
|
|
|
|
``"django.contrib.sessions.backends.file"``.
|
|
|
|
|
|
2011-05-20 00:51:25 +00:00
|
|
|
|
You might also want to set the :setting:`SESSION_FILE_PATH` setting (which
|
|
|
|
|
defaults to output from ``tempfile.gettempdir()``, most likely ``/tmp``) to
|
2021-07-23 07:48:16 +01:00
|
|
|
|
control where Django stores session files. Be sure to check that your web
|
2011-05-20 00:51:25 +00:00
|
|
|
|
server has permissions to read and write to this location.
|
2007-09-15 21:29:14 +00:00
|
|
|
|
|
2011-06-26 17:00:24 +00:00
|
|
|
|
.. _cookie-session-backend:
|
|
|
|
|
|
|
|
|
|
Using cookie-based sessions
|
|
|
|
|
---------------------------
|
|
|
|
|
|
|
|
|
|
To use cookies-based sessions, set the :setting:`SESSION_ENGINE` setting to
|
2011-11-17 21:16:42 +00:00
|
|
|
|
``"django.contrib.sessions.backends.signed_cookies"``. The session data will be
|
2011-06-26 17:00:24 +00:00
|
|
|
|
stored using Django's tools for :doc:`cryptographic signing </topics/signing>`
|
|
|
|
|
and the :setting:`SECRET_KEY` setting.
|
|
|
|
|
|
|
|
|
|
.. note::
|
|
|
|
|
|
2011-11-21 22:03:03 +00:00
|
|
|
|
It's recommended to leave the :setting:`SESSION_COOKIE_HTTPONLY` setting
|
2014-08-02 18:52:07 +02:00
|
|
|
|
on ``True`` to prevent access to the stored data from JavaScript.
|
2011-06-26 17:00:24 +00:00
|
|
|
|
|
|
|
|
|
.. warning::
|
|
|
|
|
|
2011-10-17 15:43:24 +00:00
|
|
|
|
**The session data is signed but not encrypted**
|
2011-06-26 17:00:24 +00:00
|
|
|
|
|
2011-10-17 15:43:24 +00:00
|
|
|
|
When using the cookies backend the session data can be read by the client.
|
2011-06-26 17:00:24 +00:00
|
|
|
|
|
2011-10-17 15:43:24 +00:00
|
|
|
|
A MAC (Message Authentication Code) is used to protect the data against
|
|
|
|
|
changes by the client, so that the session data will be invalidated when being
|
|
|
|
|
tampered with. The same invalidation happens if the client storing the
|
|
|
|
|
cookie (e.g. your user's browser) can't store all of the session cookie and
|
|
|
|
|
drops data. Even though Django compresses the data, it's still entirely
|
2020-04-15 13:11:13 +02:00
|
|
|
|
possible to exceed the :rfc:`common limit of 4096 bytes <2965#section-5.3>`
|
|
|
|
|
per cookie.
|
2011-10-17 15:43:24 +00:00
|
|
|
|
|
|
|
|
|
**No freshness guarantee**
|
|
|
|
|
|
|
|
|
|
Note also that while the MAC can guarantee the authenticity of the data
|
|
|
|
|
(that it was generated by your site, and not someone else), and the
|
|
|
|
|
integrity of the data (that it is all there and correct), it cannot
|
|
|
|
|
guarantee freshness i.e. that you are being sent back the last thing you
|
|
|
|
|
sent to the client. This means that for some uses of session data, the
|
2013-10-02 10:15:18 -04:00
|
|
|
|
cookie backend might open you up to `replay attacks`_. Unlike other session
|
|
|
|
|
backends which keep a server-side record of each session and invalidate it
|
|
|
|
|
when a user logs out, cookie-based sessions are not invalidated when a user
|
2013-11-30 08:37:15 -05:00
|
|
|
|
logs out. Thus if an attacker steals a user's cookie, they can use that
|
2013-10-02 10:15:18 -04:00
|
|
|
|
cookie to login as that user even if the user logs out. Cookies will only
|
|
|
|
|
be detected as 'stale' if they are older than your
|
2011-10-17 15:43:24 +00:00
|
|
|
|
:setting:`SESSION_COOKIE_AGE`.
|
|
|
|
|
|
|
|
|
|
**Performance**
|
|
|
|
|
|
2021-03-26 20:50:27 +00:00
|
|
|
|
Finally, the size of a cookie can have an impact on the speed of your site.
|
2011-06-26 17:00:24 +00:00
|
|
|
|
|
2015-08-08 12:02:32 +02:00
|
|
|
|
.. _`replay attacks`: https://en.wikipedia.org/wiki/Replay_attack
|
2009-12-22 15:18:51 +00:00
|
|
|
|
|
2005-08-17 04:00:02 +00:00
|
|
|
|
Using sessions in views
|
|
|
|
|
=======================
|
|
|
|
|
|
2011-05-20 00:51:25 +00:00
|
|
|
|
When ``SessionMiddleware`` is activated, each :class:`~django.http.HttpRequest`
|
|
|
|
|
object -- the first argument to any Django view function -- will have a
|
|
|
|
|
``session`` attribute, which is a dictionary-like object.
|
2005-08-17 04:00:02 +00:00
|
|
|
|
|
2011-05-20 00:51:25 +00:00
|
|
|
|
You can read it and write to ``request.session`` at any point in your view.
|
|
|
|
|
You can edit it multiple times.
|
2005-08-17 04:00:02 +00:00
|
|
|
|
|
2011-05-20 00:51:25 +00:00
|
|
|
|
.. class:: backends.base.SessionBase
|
|
|
|
|
|
|
|
|
|
This is the base class for all session objects. It has the following
|
|
|
|
|
standard dictionary methods:
|
|
|
|
|
|
|
|
|
|
.. method:: __getitem__(key)
|
2008-06-07 20:28:06 +00:00
|
|
|
|
|
2005-08-17 04:00:02 +00:00
|
|
|
|
Example: ``fav_color = request.session['fav_color']``
|
|
|
|
|
|
2011-05-20 00:51:25 +00:00
|
|
|
|
.. method:: __setitem__(key, value)
|
2008-06-07 20:28:06 +00:00
|
|
|
|
|
2005-08-17 04:00:02 +00:00
|
|
|
|
Example: ``request.session['fav_color'] = 'blue'``
|
|
|
|
|
|
2011-05-20 00:51:25 +00:00
|
|
|
|
.. method:: __delitem__(key)
|
2008-06-07 20:28:06 +00:00
|
|
|
|
|
2005-11-20 17:16:13 +00:00
|
|
|
|
Example: ``del request.session['fav_color']``. This raises ``KeyError``
|
|
|
|
|
if the given ``key`` isn't already in the session.
|
2005-08-17 04:00:02 +00:00
|
|
|
|
|
2011-05-20 00:51:25 +00:00
|
|
|
|
.. method:: __contains__(key)
|
2008-06-07 20:28:06 +00:00
|
|
|
|
|
2006-05-02 01:31:56 +00:00
|
|
|
|
Example: ``'fav_color' in request.session``
|
|
|
|
|
|
2011-05-20 00:51:25 +00:00
|
|
|
|
.. method:: get(key, default=None)
|
2008-06-07 20:28:06 +00:00
|
|
|
|
|
2005-08-17 04:00:02 +00:00
|
|
|
|
Example: ``fav_color = request.session.get('fav_color', 'red')``
|
|
|
|
|
|
2016-04-20 10:36:59 +02:00
|
|
|
|
.. method:: pop(key, default=__not_given)
|
2011-08-20 19:22:34 +00:00
|
|
|
|
|
2015-04-13 08:48:16 -05:00
|
|
|
|
Example: ``fav_color = request.session.pop('fav_color', 'blue')``
|
2011-08-20 19:22:34 +00:00
|
|
|
|
|
2014-01-22 22:26:10 +01:00
|
|
|
|
.. method:: keys()
|
2006-02-10 21:33:07 +00:00
|
|
|
|
|
2014-01-22 22:26:10 +01:00
|
|
|
|
.. method:: items()
|
2006-02-10 21:33:07 +00:00
|
|
|
|
|
2014-01-22 22:26:10 +01:00
|
|
|
|
.. method:: setdefault()
|
2007-12-02 15:27:29 +00:00
|
|
|
|
|
2014-01-22 22:26:10 +01:00
|
|
|
|
.. method:: clear()
|
2008-09-02 03:40:42 +00:00
|
|
|
|
|
2011-05-20 00:51:25 +00:00
|
|
|
|
It also has these methods:
|
2005-08-17 04:00:02 +00:00
|
|
|
|
|
2014-01-22 22:26:10 +01:00
|
|
|
|
.. method:: flush()
|
2008-08-14 03:57:46 +00:00
|
|
|
|
|
2015-05-17 18:35:14 -04:00
|
|
|
|
Deletes the current session data from the session and deletes the session
|
2013-08-18 22:36:36 -07:00
|
|
|
|
cookie. This is used if you want to ensure that the previous session data
|
|
|
|
|
can't be accessed again from the user's browser (for example, the
|
2009-02-16 05:10:31 +00:00
|
|
|
|
:func:`django.contrib.auth.logout()` function calls it).
|
2008-08-14 03:57:46 +00:00
|
|
|
|
|
2014-01-22 22:26:10 +01:00
|
|
|
|
.. method:: set_test_cookie()
|
2008-06-07 20:28:06 +00:00
|
|
|
|
|
2005-08-17 04:00:02 +00:00
|
|
|
|
Sets a test cookie to determine whether the user's browser supports
|
|
|
|
|
cookies. Due to the way cookies work, you won't be able to test this
|
2008-06-16 04:01:33 +00:00
|
|
|
|
until the user's next page request. See `Setting test cookies`_ below for
|
2005-08-17 04:00:02 +00:00
|
|
|
|
more information.
|
|
|
|
|
|
2014-01-22 22:26:10 +01:00
|
|
|
|
.. method:: test_cookie_worked()
|
2008-06-07 20:28:06 +00:00
|
|
|
|
|
2005-08-17 04:00:02 +00:00
|
|
|
|
Returns either ``True`` or ``False``, depending on whether the user's
|
|
|
|
|
browser accepted the test cookie. Due to the way cookies work, you'll
|
|
|
|
|
have to call ``set_test_cookie()`` on a previous, separate page request.
|
2008-06-16 04:01:33 +00:00
|
|
|
|
See `Setting test cookies`_ below for more information.
|
2005-08-17 04:00:02 +00:00
|
|
|
|
|
2014-01-22 22:26:10 +01:00
|
|
|
|
.. method:: delete_test_cookie()
|
2008-06-07 20:28:06 +00:00
|
|
|
|
|
2005-09-23 01:28:44 +00:00
|
|
|
|
Deletes the test cookie. Use this to clean up after yourself.
|
|
|
|
|
|
2019-05-19 22:15:45 +02:00
|
|
|
|
.. method:: get_session_cookie_age()
|
|
|
|
|
|
2022-03-26 20:54:48 +01:00
|
|
|
|
Returns the value of the setting :setting:`SESSION_COOKIE_AGE`. This can
|
|
|
|
|
be overridden in a custom session backend.
|
2019-05-19 22:15:45 +02:00
|
|
|
|
|
2011-05-20 00:51:25 +00:00
|
|
|
|
.. method:: set_expiry(value)
|
2008-06-07 20:28:06 +00:00
|
|
|
|
|
|
|
|
|
Sets the expiration time for the session. You can pass a number of
|
|
|
|
|
different values:
|
|
|
|
|
|
2011-10-14 00:12:01 +00:00
|
|
|
|
* If ``value`` is an integer, the session will expire after that
|
|
|
|
|
many seconds of inactivity. For example, calling
|
|
|
|
|
``request.session.set_expiry(300)`` would make the session expire
|
|
|
|
|
in 5 minutes.
|
2008-06-07 20:28:06 +00:00
|
|
|
|
|
2021-11-29 17:22:23 +00:00
|
|
|
|
* If ``value`` is a ``datetime`` or ``timedelta`` object, the session
|
|
|
|
|
will expire at that specific date/time.
|
2008-06-12 03:36:48 +00:00
|
|
|
|
|
2011-10-14 00:12:01 +00:00
|
|
|
|
* If ``value`` is ``0``, the user's session cookie will expire
|
2021-07-23 07:48:16 +01:00
|
|
|
|
when the user's web browser is closed.
|
2008-06-07 20:28:06 +00:00
|
|
|
|
|
2011-10-14 00:12:01 +00:00
|
|
|
|
* If ``value`` is ``None``, the session reverts to using the global
|
|
|
|
|
session expiry policy.
|
2008-06-07 20:28:06 +00:00
|
|
|
|
|
2010-05-06 01:35:02 +00:00
|
|
|
|
Reading a session is not considered activity for expiration
|
|
|
|
|
purposes. Session expiration is computed from the last time the
|
|
|
|
|
session was *modified*.
|
|
|
|
|
|
2014-01-22 22:26:10 +01:00
|
|
|
|
.. method:: get_expiry_age()
|
2008-06-07 20:28:06 +00:00
|
|
|
|
|
|
|
|
|
Returns the number of seconds until this session expires. For sessions
|
|
|
|
|
with no custom expiration (or those set to expire at browser close), this
|
2011-05-20 00:51:25 +00:00
|
|
|
|
will equal :setting:`SESSION_COOKIE_AGE`.
|
2008-06-07 20:28:06 +00:00
|
|
|
|
|
2012-10-27 21:32:50 +02:00
|
|
|
|
This function accepts two optional keyword arguments:
|
|
|
|
|
|
|
|
|
|
- ``modification``: last modification of the session, as a
|
|
|
|
|
:class:`~datetime.datetime` object. Defaults to the current time.
|
|
|
|
|
- ``expiry``: expiry information for the session, as a
|
2014-10-16 20:00:17 -04:00
|
|
|
|
:class:`~datetime.datetime` object, an :class:`int` (in seconds), or
|
2012-10-27 21:32:50 +02:00
|
|
|
|
``None``. Defaults to the value stored in the session by
|
|
|
|
|
:meth:`set_expiry`, if there is one, or ``None``.
|
|
|
|
|
|
2022-03-26 20:54:48 +01:00
|
|
|
|
.. note::
|
|
|
|
|
|
|
|
|
|
This method is used by session backends to determine the session expiry
|
|
|
|
|
age in seconds when saving the session. It is not really intended for
|
|
|
|
|
usage outside of that context.
|
|
|
|
|
|
|
|
|
|
In particular, while it is **possible** to determine the remaining
|
|
|
|
|
lifetime of a session **just when** you have the correct
|
|
|
|
|
``modification`` value **and** the ``expiry`` is set as a ``datetime``
|
|
|
|
|
object, where you do have the ``modification`` value, it is more
|
|
|
|
|
straight-forward to calculate the expiry by-hand::
|
|
|
|
|
|
|
|
|
|
expires_at = modification + timedelta(seconds=settings.SESSION_COOKIE_AGE)
|
|
|
|
|
|
2014-01-22 22:26:10 +01:00
|
|
|
|
.. method:: get_expiry_date()
|
2008-06-07 20:28:06 +00:00
|
|
|
|
|
|
|
|
|
Returns the date this session will expire. For sessions with no custom
|
|
|
|
|
expiration (or those set to expire at browser close), this will equal the
|
2011-05-20 00:51:25 +00:00
|
|
|
|
date :setting:`SESSION_COOKIE_AGE` seconds from now.
|
2008-06-07 20:28:06 +00:00
|
|
|
|
|
2022-03-26 20:54:48 +01:00
|
|
|
|
This function accepts the same keyword arguments as
|
|
|
|
|
:meth:`get_expiry_age`, and similar notes on usage apply.
|
2012-10-27 21:32:50 +02:00
|
|
|
|
|
2014-01-22 22:26:10 +01:00
|
|
|
|
.. method:: get_expire_at_browser_close()
|
2008-06-07 20:28:06 +00:00
|
|
|
|
|
|
|
|
|
Returns either ``True`` or ``False``, depending on whether the user's
|
2021-07-23 07:48:16 +01:00
|
|
|
|
session cookie will expire when the user's web browser is closed.
|
2008-06-07 20:28:06 +00:00
|
|
|
|
|
2014-01-22 22:26:10 +01:00
|
|
|
|
.. method:: clear_expired()
|
2012-10-27 23:12:08 +02:00
|
|
|
|
|
|
|
|
|
Removes expired sessions from the session store. This class method is
|
|
|
|
|
called by :djadmin:`clearsessions`.
|
|
|
|
|
|
2014-01-22 22:26:10 +01:00
|
|
|
|
.. method:: cycle_key()
|
2013-09-25 12:59:33 -04:00
|
|
|
|
|
|
|
|
|
Creates a new session key while retaining the current session data.
|
|
|
|
|
:func:`django.contrib.auth.login()` calls this method to mitigate against
|
|
|
|
|
session fixation.
|
|
|
|
|
|
2013-08-21 20:12:19 -04:00
|
|
|
|
.. _session_serialization:
|
|
|
|
|
|
|
|
|
|
Session serialization
|
|
|
|
|
---------------------
|
|
|
|
|
|
2015-08-27 09:55:53 -04:00
|
|
|
|
By default, Django serializes session data using JSON. You can use the
|
|
|
|
|
:setting:`SESSION_SERIALIZER` setting to customize the session serialization
|
|
|
|
|
format. Even with the caveats described in :ref:`custom-serializers`, we highly
|
2013-08-21 20:12:19 -04:00
|
|
|
|
recommend sticking with JSON serialization *especially if you are using the
|
|
|
|
|
cookie backend*.
|
|
|
|
|
|
2015-08-27 09:55:53 -04:00
|
|
|
|
For example, here's an attack scenario if you use :mod:`pickle` to serialize
|
|
|
|
|
session data. If you're using the :ref:`signed cookie session backend
|
2021-12-13 21:47:03 -06:00
|
|
|
|
<cookie-session-backend>` and :setting:`SECRET_KEY` (or any key of
|
|
|
|
|
:setting:`SECRET_KEY_FALLBACKS`) is known by an attacker (there isn't an
|
|
|
|
|
inherent vulnerability in Django that would cause it to leak), the attacker
|
|
|
|
|
could insert a string into their session which, when unpickled, executes
|
|
|
|
|
arbitrary code on the server. The technique for doing so is simple and easily
|
|
|
|
|
available on the internet. Although the cookie session storage signs the
|
2015-08-27 09:55:53 -04:00
|
|
|
|
cookie-stored data to prevent tampering, a :setting:`SECRET_KEY` leak
|
|
|
|
|
immediately escalates to a remote code execution vulnerability.
|
|
|
|
|
|
2016-01-24 22:26:11 +01:00
|
|
|
|
Bundled serializers
|
2016-01-03 12:56:22 +02:00
|
|
|
|
~~~~~~~~~~~~~~~~~~~
|
2013-08-21 20:12:19 -04:00
|
|
|
|
|
|
|
|
|
.. class:: serializers.JSONSerializer
|
|
|
|
|
|
|
|
|
|
A wrapper around the JSON serializer from :mod:`django.core.signing`. Can
|
2013-09-03 07:48:03 -04:00
|
|
|
|
only serialize basic data types.
|
|
|
|
|
|
|
|
|
|
In addition, as JSON supports only string keys, note that using non-string
|
|
|
|
|
keys in ``request.session`` won't work as expected:
|
2023-02-09 16:48:46 +01:00
|
|
|
|
|
2013-09-03 07:48:03 -04:00
|
|
|
|
.. code-block:: pycon
|
|
|
|
|
|
|
|
|
|
>>> # initial assignment
|
|
|
|
|
>>> request.session[0] = "bar"
|
|
|
|
|
>>> # subsequent requests following serialization & deserialization
|
|
|
|
|
>>> # of session data
|
|
|
|
|
>>> request.session[0] # KeyError
|
|
|
|
|
>>> request.session["0"]
|
|
|
|
|
'bar'
|
|
|
|
|
|
2016-07-03 12:59:47 +06:00
|
|
|
|
Similarly, data that can't be encoded in JSON, such as non-UTF8 bytes like
|
|
|
|
|
``'\xd9'`` (which raises :exc:`UnicodeDecodeError`), can't be stored.
|
|
|
|
|
|
2013-09-03 07:48:03 -04:00
|
|
|
|
See the :ref:`custom-serializers` section for more details on limitations
|
|
|
|
|
of JSON serialization.
|
2013-08-21 20:12:19 -04:00
|
|
|
|
|
|
|
|
|
.. _custom-serializers:
|
|
|
|
|
|
2016-01-24 22:26:11 +01:00
|
|
|
|
Write your own serializer
|
2016-01-03 12:56:22 +02:00
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~
|
2013-08-21 20:12:19 -04:00
|
|
|
|
|
2020-01-30 09:28:32 +00:00
|
|
|
|
Note that the :class:`~django.contrib.sessions.serializers.JSONSerializer`
|
|
|
|
|
cannot handle arbitrary Python data types. As is often the case, there is a
|
|
|
|
|
trade-off between convenience and security. If you wish to store more advanced
|
|
|
|
|
data types including ``datetime`` and ``Decimal`` in JSON backed sessions, you
|
|
|
|
|
will need to write a custom serializer (or convert such values to a JSON
|
|
|
|
|
serializable object before storing them in ``request.session``). While
|
|
|
|
|
serializing these values is often straightforward
|
2016-08-19 18:02:13 +08:00
|
|
|
|
(:class:`~django.core.serializers.json.DjangoJSONEncoder` may be helpful),
|
2013-08-21 20:12:19 -04:00
|
|
|
|
writing a decoder that can reliably get back the same thing that you put in is
|
|
|
|
|
more fragile. For example, you run the risk of returning a ``datetime`` that
|
|
|
|
|
was actually a string that just happened to be in the same format chosen for
|
|
|
|
|
``datetime``\s).
|
|
|
|
|
|
|
|
|
|
Your serializer class must implement two methods,
|
|
|
|
|
``dumps(self, obj)`` and ``loads(self, data)``, to serialize and deserialize
|
|
|
|
|
the dictionary of session data, respectively.
|
|
|
|
|
|
2005-08-17 04:05:42 +00:00
|
|
|
|
Session object guidelines
|
|
|
|
|
-------------------------
|
|
|
|
|
|
2011-10-14 00:12:01 +00:00
|
|
|
|
* Use normal Python strings as dictionary keys on ``request.session``. This
|
|
|
|
|
is more of a convention than a hard-and-fast rule.
|
2005-08-17 04:05:42 +00:00
|
|
|
|
|
2011-10-14 00:12:01 +00:00
|
|
|
|
* Session dictionary keys that begin with an underscore are reserved for
|
|
|
|
|
internal use by Django.
|
2005-08-17 04:05:42 +00:00
|
|
|
|
|
2011-10-14 00:12:01 +00:00
|
|
|
|
* Don't override ``request.session`` with a new object, and don't access or
|
|
|
|
|
set its attributes. Use it like a Python dictionary.
|
2005-08-17 04:05:42 +00:00
|
|
|
|
|
2005-08-17 04:00:02 +00:00
|
|
|
|
Examples
|
|
|
|
|
--------
|
|
|
|
|
|
|
|
|
|
This simplistic view sets a ``has_commented`` variable to ``True`` after a user
|
|
|
|
|
posts a comment. It doesn't let a user post a comment more than once::
|
|
|
|
|
|
|
|
|
|
def post_comment(request, new_comment):
|
|
|
|
|
if request.session.get("has_commented", False):
|
|
|
|
|
return HttpResponse("You've already commented.")
|
|
|
|
|
c = comments.Comment(comment=new_comment)
|
|
|
|
|
c.save()
|
|
|
|
|
request.session["has_commented"] = True
|
|
|
|
|
return HttpResponse("Thanks for your comment!")
|
|
|
|
|
|
2005-11-13 04:43:07 +00:00
|
|
|
|
This simplistic view logs in a "member" of the site::
|
2005-08-17 04:00:02 +00:00
|
|
|
|
|
|
|
|
|
def login(request):
|
2007-05-02 22:10:12 +00:00
|
|
|
|
m = Member.objects.get(username=request.POST["username"])
|
2022-01-04 05:47:24 +00:00
|
|
|
|
if m.check_password(request.POST["password"]):
|
2005-11-13 04:43:07 +00:00
|
|
|
|
request.session["member_id"] = m.id
|
2005-08-17 04:00:02 +00:00
|
|
|
|
return HttpResponse("You're logged in.")
|
|
|
|
|
else:
|
|
|
|
|
return HttpResponse("Your username and password didn't match.")
|
|
|
|
|
|
2005-11-13 04:43:07 +00:00
|
|
|
|
...And this one logs a member out, according to ``login()`` above::
|
2005-08-17 04:00:02 +00:00
|
|
|
|
|
|
|
|
|
def logout(request):
|
|
|
|
|
try:
|
2005-11-13 04:43:07 +00:00
|
|
|
|
del request.session["member_id"]
|
2005-08-17 04:00:02 +00:00
|
|
|
|
except KeyError:
|
|
|
|
|
pass
|
|
|
|
|
return HttpResponse("You're logged out.")
|
|
|
|
|
|
2011-05-20 00:51:25 +00:00
|
|
|
|
The standard :meth:`django.contrib.auth.logout` function actually does a bit
|
|
|
|
|
more than this to prevent inadvertent data leakage. It calls the
|
|
|
|
|
:meth:`~backends.base.SessionBase.flush` method of ``request.session``.
|
|
|
|
|
We are using this example as a demonstration of how to work with session
|
|
|
|
|
objects, not as a full ``logout()`` implementation.
|
2008-08-14 03:58:00 +00:00
|
|
|
|
|
2005-08-17 04:00:02 +00:00
|
|
|
|
Setting test cookies
|
|
|
|
|
====================
|
|
|
|
|
|
2019-06-17 16:54:55 +02:00
|
|
|
|
As a convenience, Django provides a way to test whether the user's browser
|
|
|
|
|
accepts cookies. Call the :meth:`~backends.base.SessionBase.set_test_cookie`
|
|
|
|
|
method of ``request.session`` in a view, and call
|
2011-05-20 00:51:25 +00:00
|
|
|
|
:meth:`~backends.base.SessionBase.test_cookie_worked` in a subsequent view --
|
2005-08-17 04:00:02 +00:00
|
|
|
|
not in the same view call.
|
|
|
|
|
|
|
|
|
|
This awkward split between ``set_test_cookie()`` and ``test_cookie_worked()``
|
|
|
|
|
is necessary due to the way cookies work. When you set a cookie, you can't
|
|
|
|
|
actually tell whether a browser accepted it until the browser's next request.
|
|
|
|
|
|
2011-05-20 00:51:25 +00:00
|
|
|
|
It's good practice to use
|
|
|
|
|
:meth:`~backends.base.SessionBase.delete_test_cookie()` to clean up after
|
|
|
|
|
yourself. Do this after you've verified that the test cookie worked.
|
2005-09-23 01:28:44 +00:00
|
|
|
|
|
2005-08-17 04:00:02 +00:00
|
|
|
|
Here's a typical usage example::
|
|
|
|
|
|
2015-12-22 10:21:24 -05:00
|
|
|
|
from django.http import HttpResponse
|
|
|
|
|
from django.shortcuts import render
|
|
|
|
|
|
2023-02-28 20:53:28 +01:00
|
|
|
|
|
2005-08-17 04:00:02 +00:00
|
|
|
|
def login(request):
|
2006-11-27 00:08:46 +00:00
|
|
|
|
if request.method == "POST":
|
2005-08-17 04:00:02 +00:00
|
|
|
|
if request.session.test_cookie_worked():
|
2005-09-23 01:28:44 +00:00
|
|
|
|
request.session.delete_test_cookie()
|
2005-08-17 04:00:02 +00:00
|
|
|
|
return HttpResponse("You're logged in.")
|
|
|
|
|
else:
|
|
|
|
|
return HttpResponse("Please enable cookies and try again.")
|
|
|
|
|
request.session.set_test_cookie()
|
2015-12-22 10:21:24 -05:00
|
|
|
|
return render(request, "foo/login_form.html")
|
2005-08-17 04:00:02 +00:00
|
|
|
|
|
|
|
|
|
Using sessions out of views
|
|
|
|
|
===========================
|
|
|
|
|
|
2013-08-29 14:15:58 -04:00
|
|
|
|
.. note::
|
|
|
|
|
|
|
|
|
|
The examples in this section import the ``SessionStore`` object directly
|
|
|
|
|
from the ``django.contrib.sessions.backends.db`` backend. In your own code,
|
|
|
|
|
you should consider importing ``SessionStore`` from the session engine
|
|
|
|
|
designated by :setting:`SESSION_ENGINE`, as below:
|
|
|
|
|
|
2023-02-09 16:48:46 +01:00
|
|
|
|
.. code-block:: pycon
|
|
|
|
|
|
2013-08-29 14:15:58 -04:00
|
|
|
|
>>> from importlib import import_module
|
|
|
|
|
>>> from django.conf import settings
|
|
|
|
|
>>> SessionStore = import_module(settings.SESSION_ENGINE).SessionStore
|
|
|
|
|
|
2007-09-15 22:36:53 +00:00
|
|
|
|
An API is available to manipulate session data outside of a view:
|
2023-02-09 16:48:46 +01:00
|
|
|
|
|
2007-09-15 22:36:53 +00:00
|
|
|
|
.. code-block:: pycon
|
2007-09-15 21:29:14 +00:00
|
|
|
|
|
2007-09-19 23:32:53 +00:00
|
|
|
|
>>> from django.contrib.sessions.backends.db import SessionStore
|
2012-04-15 16:34:13 +00:00
|
|
|
|
>>> s = SessionStore()
|
2013-08-21 20:12:19 -04:00
|
|
|
|
>>> # stored as seconds since epoch since datetimes are not serializable in JSON.
|
|
|
|
|
>>> s["last_login"] = 1376587691
|
2016-07-21 13:39:13 -04:00
|
|
|
|
>>> s.create()
|
2012-04-15 16:34:13 +00:00
|
|
|
|
>>> s.session_key
|
|
|
|
|
'2b1189a188b44ad18c35e113ac6ceead'
|
|
|
|
|
>>> s = SessionStore(session_key="2b1189a188b44ad18c35e113ac6ceead")
|
2007-09-15 21:29:14 +00:00
|
|
|
|
>>> s["last_login"]
|
2013-08-21 20:12:19 -04:00
|
|
|
|
1376587691
|
2007-09-15 21:29:14 +00:00
|
|
|
|
|
2016-07-21 13:39:13 -04:00
|
|
|
|
``SessionStore.create()`` is designed to create a new session (i.e. one not
|
|
|
|
|
loaded from the session store and with ``session_key=None``). ``save()`` is
|
|
|
|
|
designed to save an existing session (i.e. one loaded from the session store).
|
|
|
|
|
Calling ``save()`` on a new session may also work but has a small chance of
|
|
|
|
|
generating a ``session_key`` that collides with an existing one. ``create()``
|
|
|
|
|
calls ``save()`` and loops until an unused ``session_key`` is generated.
|
|
|
|
|
|
2007-09-20 02:02:21 +00:00
|
|
|
|
If you're using the ``django.contrib.sessions.backends.db`` backend, each
|
2019-06-17 16:54:55 +02:00
|
|
|
|
session is a normal Django model. The ``Session`` model is defined in
|
2022-06-30 15:08:49 +03:00
|
|
|
|
:source:`django/contrib/sessions/models.py`. Because it's a normal model, you can
|
2007-09-15 22:36:53 +00:00
|
|
|
|
access sessions using the normal Django database API:
|
2023-02-09 16:48:46 +01:00
|
|
|
|
|
2007-09-15 22:36:53 +00:00
|
|
|
|
.. code-block:: pycon
|
2005-08-17 04:00:02 +00:00
|
|
|
|
|
2006-05-02 01:31:56 +00:00
|
|
|
|
>>> from django.contrib.sessions.models import Session
|
2007-03-25 23:22:04 +00:00
|
|
|
|
>>> s = Session.objects.get(pk="2b1189a188b44ad18c35e113ac6ceead")
|
2005-08-17 04:00:02 +00:00
|
|
|
|
>>> s.expire_date
|
2007-09-15 22:36:53 +00:00
|
|
|
|
datetime.datetime(2005, 8, 20, 13, 35, 12)
|
2005-08-17 04:00:02 +00:00
|
|
|
|
|
2014-05-16 18:18:34 +02:00
|
|
|
|
Note that you'll need to call
|
|
|
|
|
:meth:`~base_session.AbstractBaseSession.get_decoded()` to get the session
|
|
|
|
|
dictionary. This is necessary because the dictionary is stored in an encoded
|
|
|
|
|
format:
|
2023-02-09 16:48:46 +01:00
|
|
|
|
|
2014-05-16 18:18:34 +02:00
|
|
|
|
.. code-block:: pycon
|
2005-08-17 04:00:02 +00:00
|
|
|
|
|
|
|
|
|
>>> s.session_data
|
|
|
|
|
'KGRwMQpTJ19hdXRoX3VzZXJfaWQnCnAyCkkxCnMuMTExY2ZjODI2Yj...'
|
|
|
|
|
>>> s.get_decoded()
|
|
|
|
|
{'user_id': 42}
|
|
|
|
|
|
2005-11-20 17:16:13 +00:00
|
|
|
|
When sessions are saved
|
|
|
|
|
=======================
|
|
|
|
|
|
|
|
|
|
By default, Django only saves to the session database when the session has been
|
|
|
|
|
modified -- that is if any of its dictionary values have been assigned or
|
|
|
|
|
deleted::
|
|
|
|
|
|
|
|
|
|
# Session is modified.
|
|
|
|
|
request.session["foo"] = "bar"
|
|
|
|
|
|
|
|
|
|
# Session is modified.
|
|
|
|
|
del request.session["foo"]
|
|
|
|
|
|
|
|
|
|
# Session is modified.
|
|
|
|
|
request.session["foo"] = {}
|
|
|
|
|
|
|
|
|
|
# Gotcha: Session is NOT modified, because this alters
|
|
|
|
|
# request.session['foo'] instead of request.session.
|
|
|
|
|
request.session["foo"]["bar"] = "baz"
|
|
|
|
|
|
2007-08-12 10:24:21 +00:00
|
|
|
|
In the last case of the above example, we can tell the session object
|
|
|
|
|
explicitly that it has been modified by setting the ``modified`` attribute on
|
|
|
|
|
the session object::
|
|
|
|
|
|
|
|
|
|
request.session.modified = True
|
|
|
|
|
|
2011-05-20 00:51:25 +00:00
|
|
|
|
To change this default behavior, set the :setting:`SESSION_SAVE_EVERY_REQUEST`
|
|
|
|
|
setting to ``True``. When set to ``True``, Django will save the session to the
|
|
|
|
|
database on every single request.
|
2005-11-02 14:26:55 +00:00
|
|
|
|
|
2005-11-20 17:16:13 +00:00
|
|
|
|
Note that the session cookie is only sent when a session has been created or
|
2011-05-29 17:41:04 +00:00
|
|
|
|
modified. If :setting:`SESSION_SAVE_EVERY_REQUEST` is ``True``, the session
|
|
|
|
|
cookie will be sent on every request.
|
2005-11-20 17:16:13 +00:00
|
|
|
|
|
|
|
|
|
Similarly, the ``expires`` part of a session cookie is updated each time the
|
|
|
|
|
session cookie is sent.
|
|
|
|
|
|
2013-08-19 08:29:32 -04:00
|
|
|
|
The session is not saved if the response's status code is 500.
|
2012-07-05 18:09:48 +03:00
|
|
|
|
|
2013-04-04 18:12:12 +02:00
|
|
|
|
.. _browser-length-vs-persistent-sessions:
|
|
|
|
|
|
2006-06-01 22:25:06 +00:00
|
|
|
|
Browser-length sessions vs. persistent sessions
|
|
|
|
|
===============================================
|
|
|
|
|
|
|
|
|
|
You can control whether the session framework uses browser-length sessions vs.
|
2011-05-20 00:51:25 +00:00
|
|
|
|
persistent sessions with the :setting:`SESSION_EXPIRE_AT_BROWSER_CLOSE`
|
|
|
|
|
setting.
|
2006-06-01 22:25:06 +00:00
|
|
|
|
|
2011-05-29 17:41:04 +00:00
|
|
|
|
By default, :setting:`SESSION_EXPIRE_AT_BROWSER_CLOSE` is set to ``False``,
|
|
|
|
|
which means session cookies will be stored in users' browsers for as long as
|
2011-05-20 00:51:25 +00:00
|
|
|
|
:setting:`SESSION_COOKIE_AGE`. Use this if you don't want people to have to
|
|
|
|
|
log in every time they open a browser.
|
2006-06-01 22:25:06 +00:00
|
|
|
|
|
2011-05-29 17:41:04 +00:00
|
|
|
|
If :setting:`SESSION_EXPIRE_AT_BROWSER_CLOSE` is set to ``True``, Django will
|
|
|
|
|
use browser-length cookies -- cookies that expire as soon as the user closes
|
2013-11-30 08:37:15 -05:00
|
|
|
|
their browser. Use this if you want people to have to log in every time they
|
|
|
|
|
open a browser.
|
2006-06-01 22:25:06 +00:00
|
|
|
|
|
2008-06-07 20:28:06 +00:00
|
|
|
|
This setting is a global default and can be overwritten at a per-session level
|
2011-05-20 00:51:25 +00:00
|
|
|
|
by explicitly calling the :meth:`~backends.base.SessionBase.set_expiry` method
|
|
|
|
|
of ``request.session`` as described above in `using sessions in views`_.
|
2008-06-07 20:28:06 +00:00
|
|
|
|
|
2013-03-02 10:11:23 -05:00
|
|
|
|
.. note::
|
|
|
|
|
|
|
|
|
|
Some browsers (Chrome, for example) provide settings that allow users to
|
2022-03-11 08:11:42 +00:00
|
|
|
|
continue browsing sessions after closing and reopening the browser. In
|
2013-03-02 10:11:23 -05:00
|
|
|
|
some cases, this can interfere with the
|
|
|
|
|
:setting:`SESSION_EXPIRE_AT_BROWSER_CLOSE` setting and prevent sessions
|
|
|
|
|
from expiring on browser close. Please be aware of this while testing
|
|
|
|
|
Django applications which have the
|
|
|
|
|
:setting:`SESSION_EXPIRE_AT_BROWSER_CLOSE` setting enabled.
|
|
|
|
|
|
2020-03-25 11:04:52 +00:00
|
|
|
|
.. _clearing-the-session-store:
|
|
|
|
|
|
2012-10-27 23:12:08 +02:00
|
|
|
|
Clearing the session store
|
2006-12-30 07:30:00 +00:00
|
|
|
|
==========================
|
|
|
|
|
|
2012-10-27 23:12:08 +02:00
|
|
|
|
As users create new sessions on your website, session data can accumulate in
|
|
|
|
|
your session store. If you're using the database backend, the
|
|
|
|
|
``django_session`` database table will grow. If you're using the file backend,
|
|
|
|
|
your temporary directory will contain an increasing number of files.
|
2006-12-30 07:30:00 +00:00
|
|
|
|
|
2012-10-27 23:12:08 +02:00
|
|
|
|
To understand this problem, consider what happens with the database backend.
|
2006-12-30 07:30:00 +00:00
|
|
|
|
When a user logs in, Django adds a row to the ``django_session`` database
|
|
|
|
|
table. Django updates this row each time the session data changes. If the user
|
|
|
|
|
logs out manually, Django deletes the row. But if the user does *not* log out,
|
2012-10-27 23:12:08 +02:00
|
|
|
|
the row never gets deleted. A similar process happens with the file backend.
|
|
|
|
|
|
|
|
|
|
Django does *not* provide automatic purging of expired sessions. Therefore,
|
|
|
|
|
it's your job to purge expired sessions on a regular basis. Django provides a
|
|
|
|
|
clean-up management command for this purpose: :djadmin:`clearsessions`. It's
|
|
|
|
|
recommended to call this command on a regular basis, for example as a daily
|
|
|
|
|
cron job.
|
2006-12-30 07:30:00 +00:00
|
|
|
|
|
2012-10-27 23:12:08 +02:00
|
|
|
|
Note that the cache backend isn't vulnerable to this problem, because caches
|
|
|
|
|
automatically delete stale data. Neither is the cookie backend, because the
|
|
|
|
|
session data is stored by the users' browsers.
|
2006-12-30 07:30:00 +00:00
|
|
|
|
|
2005-11-20 17:16:13 +00:00
|
|
|
|
Settings
|
|
|
|
|
========
|
|
|
|
|
|
2013-01-12 18:44:53 -05:00
|
|
|
|
A few :ref:`Django settings <settings-sessions>` give you control over session
|
2011-05-20 00:51:25 +00:00
|
|
|
|
behavior:
|
2005-11-02 14:26:55 +00:00
|
|
|
|
|
2013-01-12 18:44:53 -05:00
|
|
|
|
* :setting:`SESSION_CACHE_ALIAS`
|
|
|
|
|
* :setting:`SESSION_COOKIE_AGE`
|
|
|
|
|
* :setting:`SESSION_COOKIE_DOMAIN`
|
|
|
|
|
* :setting:`SESSION_COOKIE_HTTPONLY`
|
|
|
|
|
* :setting:`SESSION_COOKIE_NAME`
|
|
|
|
|
* :setting:`SESSION_COOKIE_PATH`
|
2018-04-13 20:58:31 -04:00
|
|
|
|
* :setting:`SESSION_COOKIE_SAMESITE`
|
2013-01-12 18:44:53 -05:00
|
|
|
|
* :setting:`SESSION_COOKIE_SECURE`
|
|
|
|
|
* :setting:`SESSION_ENGINE`
|
|
|
|
|
* :setting:`SESSION_EXPIRE_AT_BROWSER_CLOSE`
|
|
|
|
|
* :setting:`SESSION_FILE_PATH`
|
|
|
|
|
* :setting:`SESSION_SAVE_EVERY_REQUEST`
|
2016-01-29 15:04:49 -05:00
|
|
|
|
* :setting:`SESSION_SERIALIZER`
|
2005-11-02 14:26:55 +00:00
|
|
|
|
|
2013-09-25 12:59:33 -04:00
|
|
|
|
.. _topics-session-security:
|
|
|
|
|
|
|
|
|
|
Session security
|
|
|
|
|
================
|
|
|
|
|
|
|
|
|
|
Subdomains within a site are able to set cookies on the client for the whole
|
2014-01-03 12:02:58 -05:00
|
|
|
|
domain. This makes session fixation possible if cookies are permitted from
|
|
|
|
|
subdomains not controlled by trusted users.
|
2013-09-25 12:59:33 -04:00
|
|
|
|
|
|
|
|
|
For example, an attacker could log into ``good.example.com`` and get a valid
|
2013-11-24 21:05:59 -06:00
|
|
|
|
session for their account. If the attacker has control over ``bad.example.com``,
|
|
|
|
|
they can use it to send their session key to you since a subdomain is permitted
|
2013-11-18 19:10:58 -05:00
|
|
|
|
to set cookies on ``*.example.com``. When you visit ``good.example.com``,
|
2013-09-25 12:59:33 -04:00
|
|
|
|
you'll be logged in as the attacker and might inadvertently enter your
|
2019-07-23 07:10:58 -06:00
|
|
|
|
sensitive personal data (e.g. credit card info) into the attacker's account.
|
2013-09-25 12:59:33 -04:00
|
|
|
|
|
|
|
|
|
Another possible attack would be if ``good.example.com`` sets its
|
2017-11-01 10:57:59 -04:00
|
|
|
|
:setting:`SESSION_COOKIE_DOMAIN` to ``"example.com"`` which would cause
|
2013-09-25 12:59:33 -04:00
|
|
|
|
session cookies from that site to be sent to ``bad.example.com``.
|
|
|
|
|
|
2005-08-17 04:00:02 +00:00
|
|
|
|
Technical details
|
|
|
|
|
=================
|
|
|
|
|
|
2013-08-21 20:12:19 -04:00
|
|
|
|
* The session dictionary accepts any :mod:`json` serializable value when using
|
2020-01-30 09:28:32 +00:00
|
|
|
|
:class:`~django.contrib.sessions.serializers.JSONSerializer`.
|
2005-08-17 04:00:02 +00:00
|
|
|
|
|
2011-10-14 00:12:01 +00:00
|
|
|
|
* Session data is stored in a database table named ``django_session`` .
|
2005-08-17 04:00:02 +00:00
|
|
|
|
|
2011-10-14 00:12:01 +00:00
|
|
|
|
* Django only sends a cookie if it needs to. If you don't set any session
|
|
|
|
|
data, it won't send a session cookie.
|
2005-08-17 04:00:02 +00:00
|
|
|
|
|
2014-05-16 18:18:34 +02:00
|
|
|
|
The ``SessionStore`` object
|
|
|
|
|
---------------------------
|
|
|
|
|
|
|
|
|
|
When working with sessions internally, Django uses a session store object from
|
|
|
|
|
the corresponding session engine. By convention, the session store object class
|
|
|
|
|
is named ``SessionStore`` and is located in the module designated by
|
|
|
|
|
:setting:`SESSION_ENGINE`.
|
|
|
|
|
|
|
|
|
|
All ``SessionStore`` classes available in Django inherit from
|
|
|
|
|
:class:`~backends.base.SessionBase` and implement data manipulation methods,
|
|
|
|
|
namely:
|
|
|
|
|
|
|
|
|
|
* ``exists()``
|
|
|
|
|
* ``create()``
|
|
|
|
|
* ``save()``
|
|
|
|
|
* ``delete()``
|
|
|
|
|
* ``load()``
|
|
|
|
|
* :meth:`~backends.base.SessionBase.clear_expired`
|
|
|
|
|
|
|
|
|
|
In order to build a custom session engine or to customize an existing one, you
|
|
|
|
|
may create a new class inheriting from :class:`~backends.base.SessionBase` or
|
|
|
|
|
any other existing ``SessionStore`` class.
|
|
|
|
|
|
2019-06-17 16:54:55 +02:00
|
|
|
|
You can extend the session engines, but doing so with database-backed session
|
|
|
|
|
engines generally requires some extra effort (see the next section for
|
|
|
|
|
details).
|
2014-05-16 18:18:34 +02:00
|
|
|
|
|
|
|
|
|
.. _extending-database-backed-session-engines:
|
|
|
|
|
|
|
|
|
|
Extending database-backed session engines
|
|
|
|
|
=========================================
|
|
|
|
|
|
|
|
|
|
Creating a custom database-backed session engine built upon those included in
|
|
|
|
|
Django (namely ``db`` and ``cached_db``) may be done by inheriting
|
|
|
|
|
:class:`~base_session.AbstractBaseSession` and either ``SessionStore`` class.
|
|
|
|
|
|
|
|
|
|
``AbstractBaseSession`` and ``BaseSessionManager`` are importable from
|
|
|
|
|
``django.contrib.sessions.base_session`` so that they can be imported without
|
|
|
|
|
including ``django.contrib.sessions`` in :setting:`INSTALLED_APPS`.
|
|
|
|
|
|
|
|
|
|
.. class:: base_session.AbstractBaseSession
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
The abstract base session model.
|
|
|
|
|
|
|
|
|
|
.. attribute:: session_key
|
|
|
|
|
|
|
|
|
|
Primary key. The field itself may contain up to 40 characters. The
|
|
|
|
|
current implementation generates a 32-character string (a random
|
|
|
|
|
sequence of digits and lowercase ASCII letters).
|
|
|
|
|
|
|
|
|
|
.. attribute:: session_data
|
|
|
|
|
|
|
|
|
|
A string containing an encoded and serialized session dictionary.
|
|
|
|
|
|
|
|
|
|
.. attribute:: expire_date
|
|
|
|
|
|
|
|
|
|
A datetime designating when the session expires.
|
|
|
|
|
|
|
|
|
|
Expired sessions are not available to a user, however, they may still
|
|
|
|
|
be stored in the database until the :djadmin:`clearsessions` management
|
|
|
|
|
command is run.
|
|
|
|
|
|
|
|
|
|
.. classmethod:: get_session_store_class()
|
|
|
|
|
|
|
|
|
|
Returns a session store class to be used with this session model.
|
|
|
|
|
|
|
|
|
|
.. method:: get_decoded()
|
|
|
|
|
|
|
|
|
|
Returns decoded session data.
|
|
|
|
|
|
|
|
|
|
Decoding is performed by the session store class.
|
|
|
|
|
|
|
|
|
|
You can also customize the model manager by subclassing
|
|
|
|
|
:class:`~django.contrib.sessions.base_session.BaseSessionManager`:
|
|
|
|
|
|
|
|
|
|
.. class:: base_session.BaseSessionManager
|
|
|
|
|
|
|
|
|
|
.. method:: encode(session_dict)
|
|
|
|
|
|
|
|
|
|
Returns the given session dictionary serialized and encoded as a string.
|
|
|
|
|
|
|
|
|
|
Encoding is performed by the session store class tied to a model class.
|
|
|
|
|
|
|
|
|
|
.. method:: save(session_key, session_dict, expire_date)
|
|
|
|
|
|
|
|
|
|
Saves session data for a provided session key, or deletes the session
|
|
|
|
|
in case the data is empty.
|
|
|
|
|
|
|
|
|
|
Customization of ``SessionStore`` classes is achieved by overriding methods
|
|
|
|
|
and properties described below:
|
|
|
|
|
|
|
|
|
|
.. class:: backends.db.SessionStore
|
|
|
|
|
|
|
|
|
|
Implements database-backed session store.
|
|
|
|
|
|
|
|
|
|
.. classmethod:: get_model_class()
|
|
|
|
|
|
|
|
|
|
Override this method to return a custom session model if you need one.
|
|
|
|
|
|
|
|
|
|
.. method:: create_model_instance(data)
|
|
|
|
|
|
|
|
|
|
Returns a new instance of the session model object, which represents
|
|
|
|
|
the current session state.
|
|
|
|
|
|
|
|
|
|
Overriding this method provides the ability to modify session model
|
|
|
|
|
data before it's saved to database.
|
|
|
|
|
|
|
|
|
|
.. class:: backends.cached_db.SessionStore
|
|
|
|
|
|
|
|
|
|
Implements cached database-backed session store.
|
|
|
|
|
|
|
|
|
|
.. attribute:: cache_key_prefix
|
|
|
|
|
|
|
|
|
|
A prefix added to a session key to build a cache key string.
|
|
|
|
|
|
|
|
|
|
Example
|
|
|
|
|
-------
|
|
|
|
|
|
|
|
|
|
The example below shows a custom database-backed session engine that includes
|
|
|
|
|
an additional database column to store an account ID (thus providing an option
|
|
|
|
|
to query the database for all active sessions for an account)::
|
|
|
|
|
|
|
|
|
|
from django.contrib.sessions.backends.db import SessionStore as DBStore
|
|
|
|
|
from django.contrib.sessions.base_session import AbstractBaseSession
|
|
|
|
|
from django.db import models
|
|
|
|
|
|
2023-02-28 20:53:28 +01:00
|
|
|
|
|
2014-05-16 18:18:34 +02:00
|
|
|
|
class CustomSession(AbstractBaseSession):
|
|
|
|
|
account_id = models.IntegerField(null=True, db_index=True)
|
|
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
|
def get_session_store_class(cls):
|
|
|
|
|
return SessionStore
|
|
|
|
|
|
2023-02-28 20:53:28 +01:00
|
|
|
|
|
2014-05-16 18:18:34 +02:00
|
|
|
|
class SessionStore(DBStore):
|
|
|
|
|
@classmethod
|
|
|
|
|
def get_model_class(cls):
|
|
|
|
|
return CustomSession
|
|
|
|
|
|
|
|
|
|
def create_model_instance(self, data):
|
2017-01-22 12:27:14 +05:30
|
|
|
|
obj = super().create_model_instance(data)
|
2014-05-16 18:18:34 +02:00
|
|
|
|
try:
|
|
|
|
|
account_id = int(data.get("_auth_user_id"))
|
|
|
|
|
except (ValueError, TypeError):
|
|
|
|
|
account_id = None
|
|
|
|
|
obj.account_id = account_id
|
|
|
|
|
return obj
|
|
|
|
|
|
|
|
|
|
If you are migrating from the Django's built-in ``cached_db`` session store to
|
|
|
|
|
a custom one based on ``cached_db``, you should override the cache key prefix
|
|
|
|
|
in order to prevent a namespace clash::
|
|
|
|
|
|
|
|
|
|
class SessionStore(CachedDBStore):
|
|
|
|
|
cache_key_prefix = "mysessions.custom_cached_db_backend"
|
|
|
|
|
|
|
|
|
|
# ...
|
|
|
|
|
|
2005-11-02 14:26:55 +00:00
|
|
|
|
Session IDs in URLs
|
|
|
|
|
===================
|
|
|
|
|
|
|
|
|
|
The Django sessions framework is entirely, and solely, cookie-based. It does
|
|
|
|
|
not fall back to putting session IDs in URLs as a last resort, as PHP does.
|
|
|
|
|
This is an intentional design decision. Not only does that behavior make URLs
|
|
|
|
|
ugly, it makes your site vulnerable to session-ID theft via the "Referer"
|
|
|
|
|
header.
|