123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818 |
- =============================
- User authentication in Django
- =============================
- .. module:: django.contrib.auth
- :synopsis: Django's authentication framework.
- Django comes with a user authentication system. It handles user accounts,
- groups, permissions and cookie-based user sessions. This document explains how
- things work.
- Overview
- ========
- The auth system consists of:
- * Users
- * Permissions: Binary (yes/no) flags designating whether a user may perform
- a certain task.
- * Groups: A generic way of applying labels and permissions to more than one
- user.
- Installation
- ============
- Authentication support is bundled as a Django application in
- ``django.contrib.auth``. To install it, do the following:
- 1. Put ``'django.contrib.auth'`` and ``'django.contrib.contenttypes'`` in
- your :setting:`INSTALLED_APPS` setting.
- (The :class:`~django.contrib.auth.models.Permission` model in
- :mod:`django.contrib.auth` depends on :mod:`django.contrib.contenttypes`.)
- 2. Run the command ``manage.py syncdb``.
- Note that the default :file:`settings.py` file created by
- :djadmin:`django-admin.py startproject <startproject>` includes
- ``'django.contrib.auth'`` and ``'django.contrib.contenttypes'`` in
- :setting:`INSTALLED_APPS` for convenience. If your :setting:`INSTALLED_APPS`
- already contains these apps, feel free to run :djadmin:`manage.py syncdb
- <syncdb>` again; you can run that command as many times as you'd like, and each
- time it'll only install what's needed.
- The :djadmin:`syncdb` command creates the necessary database tables, creates
- permission objects for all installed apps that need 'em, and prompts you to
- create a superuser account the first time you run it.
- Once you've taken those steps, that's it.
- Users
- =====
- .. class:: models.User
- API reference
- -------------
- Fields
- ~~~~~~
- .. class:: models.User
- :class:`~django.contrib.auth.models.User` objects have the following
- fields:
- .. attribute:: models.User.username
- Required. 30 characters or fewer. Alphanumeric characters only
- (letters, digits and underscores).
- .. versionchanged:: 1.2
- Usernames may now contain ``@``, ``+``, ``.`` and ``-`` characters.
- .. attribute:: models.User.first_name
- Optional. 30 characters or fewer.
- .. attribute:: models.User.last_name
- Optional. 30 characters or fewer.
- .. attribute:: models.User.email
- Optional. Email address.
- .. attribute:: models.User.password
- Required. A hash of, and metadata about, the password. (Django doesn't
- store the raw password.) Raw passwords can be arbitrarily long and can
- contain any character. See the "Passwords" section below.
- .. attribute:: models.User.is_staff
- Boolean. Designates whether this user can access the admin site.
- .. attribute:: models.User.is_active
- Boolean. Designates whether this user account should be considered
- active. We recommend that you set this flag to ``False`` instead of
- deleting accounts; that way, if your applications have any foreign keys
- to users, the foreign keys won't break.
- This doesn't necessarily control whether or not the user can log in.
- Authentication backends aren't required to check for the ``is_active``
- flag, so if you want to reject a login based on ``is_active`` being
- ``False``, it's up to you to check that in your own login view.
- However, the :class:`~django.contrib.auth.forms.AuthenticationForm`
- used by the :func:`~django.contrib.auth.views.login` view *does*
- perform this check, as do the permission-checking methods such as
- :meth:`~models.User.has_perm` and the authentication in the Django
- admin. All of those functions/methods will return ``False`` for
- inactive users.
- .. attribute:: models.User.is_superuser
- Boolean. Designates that this user has all permissions without
- explicitly assigning them.
- .. attribute:: models.User.last_login
- A datetime of the user's last login. Is set to the current date/time by
- default.
- .. attribute:: models.User.date_joined
- A datetime designating when the account was created. Is set to the
- current date/time by default when the account is created.
- Methods
- ~~~~~~~
- .. class:: models.User
- :class:`~django.contrib.auth.models.User` objects have two many-to-many
- fields: ``groups`` and ``user_permissions``.
- :class:`~django.contrib.auth.models.User` objects can access their related
- objects in the same way as any other :doc:`Django model
- </topics/db/models>`:
- .. code-block:: python
- myuser.groups = [group_list]
- myuser.groups.add(group, group, ...)
- myuser.groups.remove(group, group, ...)
- myuser.groups.clear()
- myuser.user_permissions = [permission_list]
- myuser.user_permissions.add(permission, permission, ...)
- myuser.user_permissions.remove(permission, permission, ...)
- myuser.user_permissions.clear()
- In addition to those automatic API methods,
- :class:`~django.contrib.auth.models.User` objects have the following custom
- methods:
- .. method:: models.User.is_anonymous()
- Always returns ``False``. This is a way of differentiating
- :class:`~django.contrib.auth.models.User` and
- :class:`~django.contrib.auth.models.AnonymousUser` objects.
- Generally, you should prefer using
- :meth:`~django.contrib.auth.models.User.is_authenticated()` to this
- method.
- .. method:: models.User.is_authenticated()
- Always returns ``True``. This is a way to tell if the user has been
- authenticated. This does not imply any permissions, and doesn't check
- if the user is active - it only indicates that the user has provided a
- valid username and password.
- .. method:: models.User.get_full_name()
- Returns the :attr:`~django.contrib.auth.models.User.first_name` plus
- the :attr:`~django.contrib.auth.models.User.last_name`, with a space in
- between.
- .. method:: models.User.set_password(raw_password)
- Sets the user's password to the given raw string, taking care of the
- password hashing. Doesn't save the
- :class:`~django.contrib.auth.models.User` object.
- .. method:: models.User.check_password(raw_password)
- Returns ``True`` if the given raw string is the correct password for
- the user. (This takes care of the password hashing in making the
- comparison.)
- .. method:: models.User.set_unusable_password()
- Marks the user as having no password set. This isn't the same as
- having a blank string for a password.
- :meth:`~django.contrib.auth.models.User.check_password()` for this user
- will never return ``True``. Doesn't save the
- :class:`~django.contrib.auth.models.User` object.
- You may need this if authentication for your application takes place
- against an existing external source such as an LDAP directory.
- .. method:: models.User.has_usable_password()
- Returns ``False`` if
- :meth:`~django.contrib.auth.models.User.set_unusable_password()` has
- been called for this user.
- .. method:: models.User.get_group_permissions(obj=None)
- Returns a set of permission strings that the user has, through his/her
- groups.
- .. versionadded:: 1.2
- If ``obj`` is passed in, only returns the group permissions for
- this specific object.
- .. method:: models.User.get_all_permissions(obj=None)
- Returns a set of permission strings that the user has, both through
- group and user permissions.
- .. versionadded:: 1.2
- If ``obj`` is passed in, only returns the permissions for this
- specific object.
- .. method:: models.User.has_perm(perm, obj=None)
- Returns ``True`` if the user has the specified permission, where perm is
- in the format ``"<app label>.<permission codename>"``. (see
- `permissions`_ section below). If the user is inactive, this method will
- always return ``False``.
- .. versionadded:: 1.2
- If ``obj`` is passed in, this method won't check for a permission for
- the model, but for this specific object.
- .. method:: models.User.has_perms(perm_list, obj=None)
- Returns ``True`` if the user has each of the specified permissions,
- where each perm is in the format
- ``"<app label>.<permission codename>"``. If the user is inactive,
- this method will always return ``False``.
- .. versionadded:: 1.2
- If ``obj`` is passed in, this method won't check for permissions for
- the model, but for the specific object.
- .. method:: models.User.has_module_perms(package_name)
- Returns ``True`` if the user has any permissions in the given package
- (the Django app label). If the user is inactive, this method will
- always return ``False``.
- .. method:: models.User.email_user(subject, message, from_email=None)
- Sends an email to the user. If
- :attr:`~django.contrib.auth.models.User.from_email` is ``None``, Django
- uses the :setting:`DEFAULT_FROM_EMAIL`.
- .. method:: models.User.get_profile()
- Returns a site-specific profile for this user. Raises
- :exc:`django.contrib.auth.models.SiteProfileNotAvailable` if the
- current site doesn't allow profiles, or
- :exc:`django.core.exceptions.ObjectDoesNotExist` if the user does not
- have a profile. For information on how to define a site-specific user
- profile, see the section on `storing additional user information`_ below.
- .. _storing additional user information: #storing-additional-information-about-users
- Manager functions
- ~~~~~~~~~~~~~~~~~
- .. class:: models.UserManager
- The :class:`~django.contrib.auth.models.User` model has a custom manager
- that has the following helper functions:
- .. method:: models.UserManager.create_user(username, email=None, password=None)
- .. versionchanged:: 1.4
- The ``email`` parameter was made optional.
- Creates, saves and returns a :class:`~django.contrib.auth.models.User`.
- The :attr:`~django.contrib.auth.models.User.username` and
- :attr:`~django.contrib.auth.models.User.password` are set as given. The
- domain portion of :attr:`~django.contrib.auth.models.User.email` is
- automatically converted to lowercase, and the returned
- :class:`~django.contrib.auth.models.User` object will have
- :attr:`~models.User.is_active` set to ``True``.
- If no password is provided,
- :meth:`~django.contrib.auth.models.User.set_unusable_password()` will
- be called.
- See `Creating users`_ for example usage.
- .. method:: models.UserManager.make_random_password(length=10, allowed_chars='abcdefghjkmnpqrstuvwxyzABCDEFGHJKLMNPQRSTUVWXYZ23456789')
- Returns a random password with the given length and given string of
- allowed characters. (Note that the default value of ``allowed_chars``
- doesn't contain letters that can cause user confusion, including:
- * ``i``, ``l``, ``I``, and ``1`` (lowercase letter i, lowercase
- letter L, uppercase letter i, and the number one)
- * ``o``, ``O``, and ``0`` (uppercase letter o, lowercase letter o,
- and zero)
- Basic usage
- -----------
- .. _topics-auth-creating-users:
- Creating users
- ~~~~~~~~~~~~~~
- The most basic way to create users is to use the
- :meth:`~django.contrib.auth.models.UserManager.create_user` helper function
- that comes with Django::
- >>> from django.contrib.auth.models import User
- >>> user = User.objects.create_user('john', 'lennon@thebeatles.com', 'johnpassword')
- # At this point, user is a User object that has already been saved
- # to the database. You can continue to change its attributes
- # if you want to change other fields.
- >>> user.is_staff = True
- >>> user.save()
- You can also create users using the Django admin site. Assuming you've enabled
- the admin site and hooked it to the URL ``/admin/``, the "Add user" page is at
- ``/admin/auth/user/add/``. You should also see a link to "Users" in the "Auth"
- section of the main admin index page. The "Add user" admin page is different
- than standard admin pages in that it requires you to choose a username and
- password before allowing you to edit the rest of the user's fields.
- Also note: if you want your own user account to be able to create users using
- the Django admin site, you'll need to give yourself permission to add users
- *and* change users (i.e., the "Add user" and "Change user" permissions). If
- your account has permission to add users but not to change them, you won't be
- able to add users. Why? Because if you have permission to add users, you have
- the power to create superusers, which can then, in turn, change other users. So
- Django requires add *and* change permissions as a slight security measure.
- Changing passwords
- ~~~~~~~~~~~~~~~~~~
- .. versionadded:: 1.2
- The ``manage.py changepassword`` command was added.
- :djadmin:`manage.py changepassword *username* <changepassword>` offers a method
- of changing a User's password from the command line. It prompts you to
- change the password of a given user which you must enter twice. If
- they both match, the new password will be changed immediately. If you
- do not supply a user, the command will attempt to change the password
- whose username matches the current user.
- You can also change a password programmatically, using
- :meth:`~django.contrib.auth.models.User.set_password()`:
- .. code-block:: python
- >>> from django.contrib.auth.models import User
- >>> u = User.objects.get(username__exact='john')
- >>> u.set_password('new password')
- >>> u.save()
- Don't set the :attr:`~django.contrib.auth.models.User.password` attribute
- directly unless you know what you're doing. This is explained in the next
- section.
- Passwords
- ---------
- The :attr:`~django.contrib.auth.models.User.password` attribute of a
- :class:`~django.contrib.auth.models.User` object is a string in this format::
- hashtype$salt$hash
- That's hashtype, salt and hash, separated by the dollar-sign character.
- Hashtype is either ``sha1`` (default), ``md5`` or ``crypt`` -- the algorithm
- used to perform a one-way hash of the password. Salt is a random string used
- to salt the raw password to create the hash. Note that the ``crypt`` method is
- only supported on platforms that have the standard Python ``crypt`` module
- available.
- For example::
- sha1$a1976$a36cc8cbf81742a8fb52e221aaeab48ed7f58ab4
- The :meth:`~django.contrib.auth.models.User.set_password` and
- :meth:`~django.contrib.auth.models.User.check_password` functions handle the
- setting and checking of these values behind the scenes.
- Previous Django versions, such as 0.90, used simple MD5 hashes without password
- salts. For backwards compatibility, those are still supported; they'll be
- converted automatically to the new style the first time
- :meth:`~django.contrib.auth.models.User.check_password()` works correctly for
- a given user.
- Anonymous users
- ---------------
- .. class:: models.AnonymousUser
- :class:`django.contrib.auth.models.AnonymousUser` is a class that
- implements the :class:`django.contrib.auth.models.User` interface, with
- these differences:
- * :attr:`~django.contrib.auth.models.User.id` is always ``None``.
- * :attr:`~django.contrib.auth.models.User.is_staff` and
- :attr:`~django.contrib.auth.models.User.is_superuser` are always
- ``False``.
- * :attr:`~django.contrib.auth.models.User.is_active` is always ``False``.
- * :attr:`~django.contrib.auth.models.User.groups` and
- :attr:`~django.contrib.auth.models.User.user_permissions` are always
- empty.
- * :meth:`~django.contrib.auth.models.User.is_anonymous()` returns ``True``
- instead of ``False``.
- * :meth:`~django.contrib.auth.models.User.is_authenticated()` returns
- ``False`` instead of ``True``.
- * :meth:`~django.contrib.auth.models.User.set_password()`,
- :meth:`~django.contrib.auth.models.User.check_password()`,
- :meth:`~django.contrib.auth.models.User.save()`,
- :meth:`~django.contrib.auth.models.User.delete()`,
- :meth:`~django.contrib.auth.models.User.set_groups()` and
- :meth:`~django.contrib.auth.models.User.set_permissions()` raise
- :exc:`NotImplementedError`.
- In practice, you probably won't need to use
- :class:`~django.contrib.auth.models.AnonymousUser` objects on your own, but
- they're used by Web requests, as explained in the next section.
- .. _topics-auth-creating-superusers:
- Creating superusers
- -------------------
- :djadmin:`manage.py syncdb <syncdb>` prompts you to create a superuser the
- first time you run it after adding ``'django.contrib.auth'`` to your
- :setting:`INSTALLED_APPS`. If you need to create a superuser at a later date,
- you can use a command line utility::
- manage.py createsuperuser --username=joe --email=joe@example.com
- You will be prompted for a password. After you enter one, the user will be
- created immediately. If you leave off the :djadminopt:`--username` or the
- :djadminopt:`--email` options, it will prompt you for those values.
- If you're using an older release of Django, the old way of creating a superuser
- on the command line still works::
- python /path/to/django/contrib/auth/create_superuser.py
- ...where :file:`/path/to` is the path to the Django codebase on your
- filesystem. The ``manage.py`` command is preferred because it figures out the
- correct path and environment for you.
- .. _auth-profiles:
- Storing additional information about users
- ------------------------------------------
- If you'd like to store additional information related to your users, Django
- provides a method to specify a site-specific related model -- termed a "user
- profile" -- for this purpose.
- To make use of this feature, define a model with fields for the
- additional information you'd like to store, or additional methods
- you'd like to have available, and also add a
- :class:`~django.db.models.Field.OneToOneField` named ``user`` from your model
- to the :class:`~django.contrib.auth.models.User` model. This will ensure only
- one instance of your model can be created for each
- :class:`~django.contrib.auth.models.User`. For example::
- from django.contrib.auth.models import User
- class UserProfile(models.Model):
- # This field is required.
- user = models.OneToOneField(User)
- # Other fields here
- accepted_eula = models.BooleanField()
- favorite_animal = models.CharField(max_length=20, default="Dragons.")
- To indicate that this model is the user profile model for a given site, fill in
- the setting :setting:`AUTH_PROFILE_MODULE` with a string consisting of the
- following items, separated by a dot:
- 1. The name of the application (case sensitive) in which the user
- profile model is defined (in other words, the
- name which was passed to :djadmin:`manage.py startapp <startapp>` to create
- the application).
- 2. The name of the model (not case sensitive) class.
- For example, if the profile model was a class named ``UserProfile`` and was
- defined inside an application named ``accounts``, the appropriate setting would
- be::
- AUTH_PROFILE_MODULE = 'accounts.UserProfile'
- When a user profile model has been defined and specified in this manner, each
- :class:`~django.contrib.auth.models.User` object will have a method --
- :class:`~django.contrib.auth.models.User.get_profile()` -- which returns the
- instance of the user profile model associated with that
- :class:`~django.contrib.auth.models.User`.
- The method :class:`~django.contrib.auth.models.User.get_profile()`
- does not create a profile if one does not exist. You need to register a handler
- for the User model's :attr:`django.db.models.signals.post_save` signal and, in
- the handler, if ``created`` is ``True``, create the associated user profile::
- # in models.py
- from django.contrib.auth.models import User
- from django.db.models.signals import post_save
- # definition of UserProfile from above
- # ...
- def create_user_profile(sender, instance, created, **kwargs):
- if created:
- UserProfile.objects.create(user=instance)
- post_save.connect(create_user_profile, sender=User)
- .. seealso:: :doc:`/topics/signals` for more information on Django's signal
- dispatcher.
- Authentication in Web requests
- ==============================
- Until now, this document has dealt with the low-level APIs for manipulating
- authentication-related objects. On a higher level, Django can hook this
- authentication framework into its system of
- :class:`request objects <django.http.HttpRequest>`.
- First, install the
- :class:`~django.contrib.sessions.middleware.SessionMiddleware` and
- :class:`~django.contrib.auth.middleware.AuthenticationMiddleware`
- middlewares by adding them to your :setting:`MIDDLEWARE_CLASSES` setting. See
- the :doc:`session documentation </topics/http/sessions>` for more information.
- Once you have those middlewares installed, you'll be able to access
- :attr:`request.user <django.http.HttpRequest.user>` in views.
- :attr:`request.user <django.http.HttpRequest.user>` will give you a
- :class:`~django.contrib.auth.models.User` object representing the currently
- logged-in user. If a user isn't currently logged in,
- :attr:`request.user <django.http.HttpRequest.user>` will be set to an instance
- of :class:`~django.contrib.auth.models.AnonymousUser` (see the previous
- section). You can tell them apart with
- :meth:`~django.contrib.auth.models.User.is_authenticated()`, like so::
- if request.user.is_authenticated():
- # Do something for authenticated users.
- else:
- # Do something for anonymous users.
- .. _how-to-log-a-user-in:
- How to log a user in
- --------------------
- Django provides two functions in :mod:`django.contrib.auth`:
- :func:`~django.contrib.auth.authenticate()` and
- :func:`~django.contrib.auth.login()`.
- .. function:: authenticate()
- To authenticate a given username and password, use
- :func:`~django.contrib.auth.authenticate()`. It takes two keyword
- arguments, ``username`` and ``password``, and it returns a
- :class:`~django.contrib.auth.models.User` object if the password is valid
- for the given username. If the password is invalid,
- :func:`~django.contrib.auth.authenticate()` returns ``None``. Example::
- from django.contrib.auth import authenticate
- user = authenticate(username='john', password='secret')
- if user is not None:
- if user.is_active:
- print "You provided a correct username and password!"
- else:
- print "Your account has been disabled!"
- else:
- print "Your username and password were incorrect."
- .. function:: login()
- To log a user in, in a view, use :func:`~django.contrib.auth.login()`. It
- takes an :class:`~django.http.HttpRequest` object and a
- :class:`~django.contrib.auth.models.User` object.
- :func:`~django.contrib.auth.login()` saves the user's ID in the session,
- using Django's session framework, so, as mentioned above, you'll need to
- make sure to have the session middleware installed.
- This example shows how you might use both
- :func:`~django.contrib.auth.authenticate()` and
- :func:`~django.contrib.auth.login()`::
- from django.contrib.auth import authenticate, login
- def my_view(request):
- username = request.POST['username']
- password = request.POST['password']
- user = authenticate(username=username, password=password)
- if user is not None:
- if user.is_active:
- login(request, user)
- # Redirect to a success page.
- else:
- # Return a 'disabled account' error message
- else:
- # Return an 'invalid login' error message.
- .. admonition:: Calling ``authenticate()`` first
- When you're manually logging a user in, you *must* call
- :func:`~django.contrib.auth.authenticate()` before you call
- :func:`~django.contrib.auth.login()`.
- :func:`~django.contrib.auth.authenticate()`
- sets an attribute on the :class:`~django.contrib.auth.models.User` noting
- which authentication backend successfully authenticated that user (see the
- `backends documentation`_ for details), and this information is needed
- later during the login process.
- .. _backends documentation: #other-authentication-sources
- Manually managing a user's password
- -----------------------------------
- .. currentmodule:: django.contrib.auth.utils
- .. versionadded:: 1.4
- The :mod:`django.contrib.auth.utils` module provides a set of functions
- to create and validate hashed password. You can use them independently
- from the ``User`` model.
- .. function:: check_password()
- If you'd like to manually authenticate a user by comparing a plain-text
- password to the hashed password in the database, use the convenience
- function :func:`django.contrib.auth.utils.check_password`. It takes two
- arguments: the plain-text password to check, and the full value of a
- user's ``password`` field in the database to check against, and returns
- ``True`` if they match, ``False`` otherwise.
- .. function:: make_password()
- .. versionadded:: 1.4
- Creates a hashed password in the format used by this application. It takes
- two arguments: hashing algorithm to use and the password in plain-text.
- Currently supported algorithms are: ``'sha1'``, ``'md5'`` and ``'crypt'``
- if you have the ``crypt`` library installed. If the second argument is
- ``None``, an unusable password is returned (a one that will be never
- accepted by :func:`django.contrib.auth.utils.check_password`).
- .. function:: is_password_usable()
- .. versionadded:: 1.4
- Checks if the given string is a hashed password that has a chance
- of being verified against :func:`django.contrib.auth.utils.check_password`.
- How to log a user out
- ---------------------
- .. currentmodule:: django.contrib.auth
- .. function:: logout()
- To log out a user who has been logged in via
- :func:`django.contrib.auth.login()`, use
- :func:`django.contrib.auth.logout()` within your view. It takes an
- :class:`~django.http.HttpRequest` object and has no return value.
- Example::
- from django.contrib.auth import logout
- def logout_view(request):
- logout(request)
- # Redirect to a success page.
- Note that :func:`~django.contrib.auth.logout()` doesn't throw any errors if
- the user wasn't logged in.
- When you call :func:`~django.contrib.auth.logout()`, the session data for
- the current request is completely cleaned out. All existing data is
- removed. This is to prevent another person from using the same Web browser
- to log in and have access to the previous user's session data. If you want
- to put anything into the session that will be available to the user
- immediately after logging out, do that *after* calling
- :func:`django.contrib.auth.logout()`.
- .. _topics-auth-signals:
- Login and logout signals
- ------------------------
- .. versionadded:: 1.3
- The auth framework uses two :doc:`signals </topics/signals>` that can be used
- for notification when a user logs in or out.
- .. data:: django.contrib.auth.signals.user_logged_in
- Sent when a user logs in successfully.
- Arguments sent with this signal:
- ``sender``
- As above: the class of the user that just logged in.
- ``request``
- The current :class:`~django.http.HttpRequest` instance.
- ``user``
- The user instance that just logged in.
- .. data:: django.contrib.auth.signals.user_logged_out
- Sent when the logout method is called.
- ``sender``
- As above: the class of the user that just logged out or ``None``
- if the user was not authenticated.
- ``request``
- The current :class:`~django.http.HttpRequest` instance.
- ``user``
- The user instance that just logged out or ``None`` if the
- user was not authenticated.
- Limiting access to logged-in users
- ----------------------------------
- The raw way
- ~~~~~~~~~~~
- The simple, raw way to limit access to pages is to check
- :meth:`request.user.is_authenticated()
- <django.contrib.auth.models.User.is_authenticated()>` and either redirect to a
- login page::
- from django.http import HttpResponseRedirect
- def my_view(request):
- if not request.user.is_authenticated():
- return HttpResponseRedirect('/login/?next=%s' % request.path)
- # ...
- ...or display an error message::
- def my_view(request):
- if not request.user.is_authenticated():
- return render_to_response('myapp/login_error.html')
- # ...
- The login_required decorator
- ~~~~~~~~~~~~~~~~~~~~~~~~~~~~
- .. function:: decorators.login_required([redirect_field_name=REDIRECT_FIELD_NAME, login_url=None])
- As a shortcut, you can use the convenient
- :func:`~django.contrib.auth.decorators.login_required` decorator::
- from django.contrib.auth.decorators import login_required
- @login_required
- def my_view(request):
- ...
- :func:`~django.contrib.auth.decorators.login_required` does the following:
- * If the user isn't logged in, redirect to
- :setting:`settings.LOGIN_URL <LOGIN_URL>`, passing the current absolute
- path in the query string. Example: ``/accounts/login/?next=/polls/3/``.
- * If the user is logged in, execute the view normally. The view code is
- free to assume the user is logged in.
- By default, the path that the user should be redirected to upon
- successful authentication is stored in a query string parameter called
- ``"next"``. If you would prefer to use a different name for this parameter,
- :func:`~django.contrib.auth.decorators.login_required` takes an
- optional ``redirect_field_name`` parameter::
- from django.contrib.auth.decorators import login_required
- @login_required(redirect_field_name='my_redirect_field')
- def my_view(request):
- ...
- Note that if you provide a value to ``redirect_field_name``, you will most
- likely need to customize your login template as well, since the template
- context variable which stores the redirect path will use the value of
- ``redirect_field_name`` as its key rather than ``"next"`` (the default).
- .. versionadded:: 1.3
- :func:`~django.contrib.auth.decorators.login_required` also takes an
- optional ``login_url`` parameter. Example::
- from django.contrib.auth.decorators import login_required
- @login_required(login_url='/accounts/login/')
- def my_view(request):
- ...
- Note that if you don't specify the ``login_url`` parameter, you'll need to map
- the appropriate Django view to :setting:`settings.LOGIN_URL <LOGIN_URL>`. For
- example, using the defaults, add the following line to your URLconf::
- (r'^accounts/login/$', 'django.contrib.auth.views.login'),
- .. function:: views.login(request, [template_name, redirect_field_name, authentication_form])
- **URL name:** ``login``
- See :doc:`the URL documentation </topics/http/urls>` for details on using
- named URL patterns.
- Here's what ``django.contrib.auth.views.login`` does:
- * If called via ``GET``, it displays a login form that POSTs to the
- same URL. More on this in a bit.
- * If called via ``POST``, it tries to log the user in. If login is
- successful, the view redirects to the URL specified in ``next``. If
- ``next`` isn't provided, it redirects to
- :setting:`settings.LOGIN_REDIRECT_URL <LOGIN_REDIRECT_URL>` (which
- defaults to ``/accounts/profile/``). If login isn't successful, it
- redisplays the login form.
- It's your responsibility to provide the login form in a template called
- ``registration/login.html`` by default. This template gets passed four
- template context variables:
- * ``form``: A :class:`~django.forms.Form` object representing the login
- form. See the :doc:`forms documentation </topics/forms/index>` for
- more on ``Form`` objects.
- * ``next``: The URL to redirect to after successful login. This may
- contain a query string, too.
- * ``site``: The current :class:`~django.contrib.sites.models.Site`,
- according to the :setting:`SITE_ID` setting. If you don't have the
- site framework installed, this will be set to an instance of
- :class:`~django.contrib.sites.models.RequestSite`, which derives the
- site name and domain from the current
- :class:`~django.http.HttpRequest`.
- * ``site_name``: An alias for ``site.name``. If you don't have the site
- framework installed, this will be set to the value of
- :attr:`request.META['SERVER_NAME'] <django.http.HttpRequest.META>`.
- For more on sites, see :doc:`/ref/contrib/sites`.
- If you'd prefer not to call the template :file:`registration/login.html`,
- you can pass the ``template_name`` parameter via the extra arguments to
- the view in your URLconf. For example, this URLconf line would use
- :file:`myapp/login.html` instead::
- (r'^accounts/login/$', 'django.contrib.auth.views.login', {'template_name': 'myapp/login.html'}),
- You can also specify the name of the ``GET`` field which contains the URL
- to redirect to after login by passing ``redirect_field_name`` to the view.
- By default, the field is called ``next``.
- Here's a sample :file:`registration/login.html` template you can use as a
- starting point. It assumes you have a :file:`base.html` template that
- defines a ``content`` block:
- .. code-block:: html+django
- {% extends "base.html" %}
- {% load url from future %}
- {% block content %}
- {% if form.errors %}
- <p>Your username and password didn't match. Please try again.</p>
- {% endif %}
- <form method="post" action="{% url 'django.contrib.auth.views.login' %}">
- {% csrf_token %}
- <table>
- <tr>
- <td>{{ form.username.label_tag }}</td>
- <td>{{ form.username }}</td>
- </tr>
- <tr>
- <td>{{ form.password.label_tag }}</td>
- <td>{{ form.password }}</td>
- </tr>
- </table>
- <input type="submit" value="login" />
- <input type="hidden" name="next" value="{{ next }}" />
- </form>
- {% endblock %}
- .. versionadded:: 1.2
- If you are using alternate authentication (see
- :ref:`authentication-backends`) you can pass a custom authentication form
- to the login view via the ``authentication_form`` parameter. This form must
- accept a ``request`` keyword argument in its ``__init__`` method, and
- provide a ``get_user`` method which returns the authenticated user object
- (this method is only ever called after successful form validation).
- .. _forms documentation: ../forms/
- .. _site framework docs: ../sites/
- .. versionadded:: 1.4
- The :func:`~views.login` view and the :ref:`other-built-in-views` now all
- return a :class:`~django.template.response.TemplateResponse` instance,
- which allows you to easily customize the response data before rendering.
- For more details, see the
- :doc:`TemplateResponse documentation </ref/template-response>`.
- .. _other-built-in-views:
- Other built-in views
- --------------------
- .. module:: django.contrib.auth.views
- In addition to the :func:`~views.login` view, the authentication system
- includes a few other useful built-in views located in
- :mod:`django.contrib.auth.views`:
- .. function:: logout(request, [next_page, template_name, redirect_field_name])
- Logs a user out.
- **URL name:** ``logout``
- See :doc:`the URL documentation </topics/http/urls>` for details on using
- named URL patterns.
- **Optional arguments:**
- * ``next_page``: The URL to redirect to after logout.
- * ``template_name``: The full name of a template to display after
- logging the user out. Defaults to
- :file:`registration/logged_out.html` if no argument is supplied.
- * ``redirect_field_name``: The name of a ``GET`` field containing the
- URL to redirect to after log out. Overrides ``next_page`` if the given
- ``GET`` parameter is passed.
- **Template context:**
- * ``title``: The string "Logged out", localized.
- * ``site``: The current :class:`~django.contrib.sites.models.Site`,
- according to the :setting:`SITE_ID` setting. If you don't have the
- site framework installed, this will be set to an instance of
- :class:`~django.contrib.sites.models.RequestSite`, which derives the
- site name and domain from the current
- :class:`~django.http.HttpRequest`.
- * ``site_name``: An alias for ``site.name``. If you don't have the site
- framework installed, this will be set to the value of
- :attr:`request.META['SERVER_NAME'] <django.http.HttpRequest.META>`.
- For more on sites, see :doc:`/ref/contrib/sites`.
- .. function:: logout_then_login(request[, login_url])
- Logs a user out, then redirects to the login page.
- **URL name:** No default URL provided
- **Optional arguments:**
- * ``login_url``: The URL of the login page to redirect to.
- Defaults to :setting:`settings.LOGIN_URL <LOGIN_URL>` if not supplied.
- .. function:: password_change(request[, template_name, post_change_redirect, password_change_form])
- Allows a user to change their password.
- **URL name:** ``password_change``
- **Optional arguments:**
- * ``template_name``: The full name of a template to use for
- displaying the password change form. Defaults to
- :file:`registration/password_change_form.html` if not supplied.
- * ``post_change_redirect``: The URL to redirect to after a successful
- password change.
- .. versionadded:: 1.2
- * ``password_change_form``: A custom "change password" form which must
- accept a ``user`` keyword argument. The form is responsible for
- actually changing the user's password. Defaults to
- :class:`~django.contrib.auth.forms.PasswordChangeForm`.
- **Template context:**
- * ``form``: The password change form (see ``password_change_form`` above).
- .. function:: password_change_done(request[, template_name])
- The page shown after a user has changed their password.
- **URL name:** ``password_change_done``
- **Optional arguments:**
- * ``template_name``: The full name of a template to use.
- Defaults to :file:`registration/password_change_done.html` if not
- supplied.
- .. function:: password_reset(request[, is_admin_site, template_name, email_template_name, password_reset_form, token_generator, post_reset_redirect, from_email])
- Allows a user to reset their password by generating a one-time use link
- that can be used to reset the password, and sending that link to the
- user's registered email address.
- .. versionchanged:: 1.3
- The ``from_email`` argument was added.
- .. versionchanged:: 1.4
- Users flagged with an unusable password (see
- :meth:`~django.contrib.auth.models.User.set_unusable_password()`
- will not be able to request a password reset to prevent misuse
- when using an external authentication source like LDAP.
- **URL name:** ``password_reset``
- **Optional arguments:**
- * ``template_name``: The full name of a template to use for
- displaying the password reset form. Defaults to
- :file:`registration/password_reset_form.html` if not supplied.
- * ``email_template_name``: The full name of a template to use for
- generating the email with the new password. Defaults to
- :file:`registration/password_reset_email.html` if not supplied.
- * ``subject_template_name``: The full name of a template to use for
- the subject of the email with the new password. Defaults
- to :file:`registration/password_reset_subject.txt` if not supplied.
- .. versionadded:: 1.4
- * ``password_reset_form``: Form that will be used to set the password.
- Defaults to :class:`~django.contrib.auth.forms.PasswordResetForm`.
- * ``token_generator``: Instance of the class to check the password. This
- will default to ``default_token_generator``, it's an instance of
- ``django.contrib.auth.tokens.PasswordResetTokenGenerator``.
- * ``post_reset_redirect``: The URL to redirect to after a successful
- password change.
- * ``from_email``: A valid email address. By default Django uses
- the :setting:`DEFAULT_FROM_EMAIL`.
- **Template context:**
- * ``form``: The form (see ``password_reset_form`` above) for resetting
- the user's password.
- **Email template context:**
- * ``email``: An alias for ``user.email``
- * ``user``: The current :class:`~django.contrib.auth.models.User`,
- according to the ``email`` form field. Only active users are able to
- reset their passwords (``User.is_active is True``).
- * ``site_name``: An alias for ``site.name``. If you don't have the site
- framework installed, this will be set to the value of
- :attr:`request.META['SERVER_NAME'] <django.http.HttpRequest.META>`.
- For more on sites, see :doc:`/ref/contrib/sites`.
- * ``domain``: An alias for ``site.domain``. If you don't have the site
- framework installed, this will be set to the value of
- ``request.get_host()``.
- * ``protocol``: http or https
- * ``uid``: The user's id encoded in base 36.
- * ``token``: Token to check that the password is valid.
- Sample ``registration/password_reset_email.html`` (email body template):
- .. code-block:: html+django
- {% load url from future %}
- Someone asked for password reset for email {{ email }}. Follow the link below:
- {{ protocol}}://{{ site_name }}{% url 'auth_password_reset_confirm' uidb36=uid token=token %}
- The same template context is used for subject template. Subject must be
- single line plain text string.
- .. function:: password_reset_done(request[, template_name])
- The page shown after a user has been emailed a link to reset their
- password. This view is called by default if the :func:`password_reset` view
- doesn't have an explicit ``post_reset_redirect`` URL set.
- **URL name:** ``password_reset_done``
- **Optional arguments:**
- * ``template_name``: The full name of a template to use.
- Defaults to :file:`registration/password_reset_done.html` if not
- supplied.
- .. function:: password_reset_confirm(request[, uidb36, token, template_name, token_generator, set_password_form, post_reset_redirect])
- Presents a form for entering a new password.
- **URL name:** ``password_reset_confirm``
- **Optional arguments:**
- * ``uidb36``: The user's id encoded in base 36. Defaults to ``None``.
- * ``token``: Token to check that the password is valid. Defaults to
- ``None``.
- * ``template_name``: The full name of a template to display the confirm
- password view. Default value is :file:`registration/password_reset_confirm.html`.
- * ``token_generator``: Instance of the class to check the password. This
- will default to ``default_token_generator``, it's an instance of
- ``django.contrib.auth.tokens.PasswordResetTokenGenerator``.
- * ``set_password_form``: Form that will be used to set the password.
- Defaults to :class:`~django.contrib.auth.forms.SetPasswordForm`
- * ``post_reset_redirect``: URL to redirect after the password reset
- done. Defaults to ``None``.
- **Template context:**
- * ``form``: The form (see ``set_password_form`` above) for setting the
- new user's password.
- * ``validlink``: Boolean, True if the link (combination of uidb36 and
- token) is valid or unused yet.
- .. function:: password_reset_complete(request[,template_name])
- Presents a view which informs the user that the password has been
- successfully changed.
- **URL name:** ``password_reset_complete``
- **Optional arguments:**
- * ``template_name``: The full name of a template to display the view.
- Defaults to :file:`registration/password_reset_complete.html`.
- Helper functions
- ----------------
- .. currentmodule:: django.contrib.auth.views
- .. function:: redirect_to_login(next[, login_url, redirect_field_name])
- Redirects to the login page, and then back to another URL after a
- successful login.
- **Required arguments:**
- * ``next``: The URL to redirect to after a successful login.
- **Optional arguments:**
- * ``login_url``: The URL of the login page to redirect to.
- Defaults to :setting:`settings.LOGIN_URL <LOGIN_URL>` if not supplied.
- * ``redirect_field_name``: The name of a ``GET`` field containing the
- URL to redirect to after log out. Overrides ``next`` if the given
- ``GET`` parameter is passed.
- Built-in forms
- --------------
- .. module:: django.contrib.auth.forms
- If you don't want to use the built-in views, but want the convenience of not
- having to write forms for this functionality, the authentication system
- provides several built-in forms located in :mod:`django.contrib.auth.forms`:
- .. class:: AdminPasswordChangeForm
- A form used in the admin interface to change a user's password.
- .. class:: AuthenticationForm
- A form for logging a user in.
- .. class:: PasswordChangeForm
- A form for allowing a user to change their password.
- .. class:: PasswordResetForm
- A form for generating and emailing a one-time use link to reset a
- user's password.
- .. class:: SetPasswordForm
- A form that lets a user change his/her password without entering the old
- password.
- .. class:: UserChangeForm
- A form used in the admin interface to change a user's information and
- permissions.
- .. class:: UserCreationForm
- A form for creating a new user.
- Limiting access to logged-in users that pass a test
- ---------------------------------------------------
- .. currentmodule:: django.contrib.auth.decorators
- To limit access based on certain permissions or some other test, you'd do
- essentially the same thing as described in the previous section.
- The simple way is to run your test on :attr:`request.user
- <django.http.HttpRequest.user>` in the view directly. For example, this view
- checks to make sure the user is logged in and has the permission
- ``polls.can_vote``::
- def my_view(request):
- if not request.user.has_perm('polls.can_vote'):
- return HttpResponse("You can't vote in this poll.")
- # ...
- .. function:: user_passes_test(func, [login_url=None])
- As a shortcut, you can use the convenient ``user_passes_test`` decorator::
- from django.contrib.auth.decorators import user_passes_test
- @user_passes_test(lambda u: u.has_perm('polls.can_vote'))
- def my_view(request):
- ...
- We're using this particular test as a relatively simple example. However,
- if you just want to test whether a permission is available to a user, you
- can use the :func:`~django.contrib.auth.decorators.permission_required()`
- decorator, described later in this document.
- :func:`~django.contrib.auth.decorators.user_passes_test` takes a required
- argument: a callable that takes a
- :class:`~django.contrib.auth.models.User` object and returns ``True`` if
- the user is allowed to view the page. Note that
- :func:`~django.contrib.auth.decorators.user_passes_test` does not
- automatically check that the :class:`~django.contrib.auth.models.User` is
- not anonymous.
- :func:`~django.contrib.auth.decorators.user_passes_test()` takes an
- optional ``login_url`` argument, which lets you specify the URL for your
- login page (:setting:`settings.LOGIN_URL <LOGIN_URL>` by default).
- For example::
- from django.contrib.auth.decorators import user_passes_test
- @user_passes_test(lambda u: u.has_perm('polls.can_vote'), login_url='/login/')
- def my_view(request):
- ...
- The permission_required decorator
- ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
- .. function:: permission_required([login_url=None, raise_exception=False])
- It's a relatively common task to check whether a user has a particular
- permission. For that reason, Django provides a shortcut for that case: the
- :func:`~django.contrib.auth.decorators.permission_required()` decorator.
- Using this decorator, the earlier example can be written as::
- from django.contrib.auth.decorators import permission_required
- @permission_required('polls.can_vote')
- def my_view(request):
- ...
- As for the :meth:`User.has_perm` method, permission names take the form
- ``"<app label>.<permission codename>"`` (i.e. ``polls.can_vote`` for a
- permission on a model in the ``polls`` application).
- Note that :func:`~django.contrib.auth.decorators.permission_required()`
- also takes an optional ``login_url`` parameter. Example::
- from django.contrib.auth.decorators import permission_required
- @permission_required('polls.can_vote', login_url='/loginpage/')
- def my_view(request):
- ...
- As in the :func:`~decorators.login_required` decorator, ``login_url``
- defaults to :setting:`settings.LOGIN_URL <LOGIN_URL>`.
- .. versionchanged:: 1.4
- Added ``raise_exception`` parameter. If given, the decorator will raise
- :exc:`~django.core.exceptions.PermissionDenied`, prompting
- :ref:`the 403 (HTTP Forbidden) view<http_forbidden_view>` instead of
- redirecting to the login page.
- .. currentmodule:: django.contrib.auth
- Limiting access to generic views
- --------------------------------
- To limit access to a :doc:`generic view </ref/generic-views>`, write a thin
- wrapper around the view, and point your URLconf to your wrapper instead of the
- generic view itself. For example::
- from django.views.generic.date_based import object_detail
- @login_required
- def limited_object_detail(*args, **kwargs):
- return object_detail(*args, **kwargs)
- .. _permissions:
- Permissions
- ===========
- Django comes with a simple permissions system. It provides a way to assign
- permissions to specific users and groups of users.
- It's used by the Django admin site, but you're welcome to use it in your own
- code.
- The Django admin site uses permissions as follows:
- * Access to view the "add" form and add an object is limited to users with
- the "add" permission for that type of object.
- * Access to view the change list, view the "change" form and change an
- object is limited to users with the "change" permission for that type of
- object.
- * Access to delete an object is limited to users with the "delete"
- permission for that type of object.
- Permissions are set globally per type of object, not per specific object
- instance. For example, it's possible to say "Mary may change news stories," but
- it's not currently possible to say "Mary may change news stories, but only the
- ones she created herself" or "Mary may only change news stories that have a
- certain status, publication date or ID." The latter functionality is something
- Django developers are currently discussing.
- Default permissions
- -------------------
- When ``django.contrib.auth`` is listed in your :setting:`INSTALLED_APPS`
- setting, it will ensure that three default permissions -- add, change and
- delete -- are created for each Django model defined in one of your installed
- applications.
- These permissions will be created when you run :djadmin:`manage.py syncdb
- <syncdb>`; the first time you run ``syncdb`` after adding
- ``django.contrib.auth`` to :setting:`INSTALLED_APPS`, the default permissions
- will be created for all previously-installed models, as well as for any new
- models being installed at that time. Afterward, it will create default
- permissions for new models each time you run :djadmin:`manage.py syncdb
- <syncdb>`.
- Assuming you have an application with an
- :attr:`~django.db.models.Options.app_label` ``foo`` and a model named ``Bar``,
- to test for basic permissions you should use:
- * add: ``user.has_perm('foo.add_bar')``
- * change: ``user.has_perm('foo.change_bar')``
- * delete: ``user.has_perm('foo.delete_bar')``
- .. _custom-permissions:
- Custom permissions
- ------------------
- To create custom permissions for a given model object, use the ``permissions``
- :ref:`model Meta attribute <meta-options>`.
- This example Task model creates three custom permissions, i.e., actions users
- can or cannot do with Task instances, specific to your application::
- class Task(models.Model):
- ...
- class Meta:
- permissions = (
- ("view_task", "Can see available tasks"),
- ("change_task_status", "Can change the status of tasks"),
- ("close_task", "Can remove a task by setting its status as closed"),
- )
- The only thing this does is create those extra permissions when you run
- :djadmin:`manage.py syncdb <syncdb>`. Your code is in charge of checking the
- value of these permissions when an user is trying to access the functionality
- provided by the application (viewing tasks, changing the status of tasks,
- closing tasks.) Continuing the above example, the following checks if a user may
- view tasks::
- user.has_perm('app.view_task')
- API reference
- -------------
- .. currentmodule:: django.contrib.auth.models
- .. class:: models.Permission
- Fields
- ~~~~~~
- :class:`~django.contrib.auth.models.Permission` objects have the following
- fields:
- .. attribute:: Permission.name
- Required. 50 characters or fewer. Example: ``'Can vote'``.
- .. attribute:: Permission.content_type
- Required. A reference to the ``django_content_type`` database table, which
- contains a record for each installed Django model.
- .. attribute:: Permission.codename
- Required. 100 characters or fewer. Example: ``'can_vote'``.
- Methods
- ~~~~~~~
- :class:`~django.contrib.auth.models.Permission` objects have the standard
- data-access methods like any other :doc:`Django model </ref/models/instances>`.
- .. currentmodule:: django.contrib.auth
- Programmatically creating permissions
- -------------------------------------
- While custom permissions can be defined within a model's ``Meta`` class, you
- can also create permissions directly. For example, you can create the
- ``can_publish`` permission for a ``BlogPost`` model in ``myapp``::
- from django.contrib.auth.models import Group, Permission
- from django.contrib.contenttypes.models import ContentType
- content_type = ContentType.objects.get(app_label='myapp', model='BlogPost')
- permission = Permission.objects.create(codename='can_publish',
- name='Can Publish Posts',
- content_type=content_type)
- The permission can then be assigned to a
- :class:`~django.contrib.auth.models.User` via its ``user_permissions``
- attribute or to a :class:`~django.contrib.auth.models.Group` via its
- ``permissions`` attribute.
- Authentication data in templates
- ================================
- The currently logged-in user and his/her permissions are made available in the
- :doc:`template context </ref/templates/api>` when you use
- :class:`~django.template.context.RequestContext`.
- .. admonition:: Technicality
- Technically, these variables are only made available in the template context
- if you use :class:`~django.template.context.RequestContext` *and* your
- :setting:`TEMPLATE_CONTEXT_PROCESSORS` setting contains
- ``"django.contrib.auth.context_processors.auth"``, which is default. For
- more, see the :ref:`RequestContext docs <subclassing-context-requestcontext>`.
- Users
- -----
- When rendering a template :class:`~django.template.context.RequestContext`, the
- currently logged-in user, either a :class:`~django.contrib.auth.models.User`
- instance or an :class:`~django.contrib.auth.models.AnonymousUser` instance, is
- stored in the template variable ``{{ user }}``:
- .. code-block:: html+django
- {% if user.is_authenticated %}
- <p>Welcome, {{ user.username }}. Thanks for logging in.</p>
- {% else %}
- <p>Welcome, new user. Please log in.</p>
- {% endif %}
- This template context variable is not available if a ``RequestContext`` is not
- being used.
- Permissions
- -----------
- The currently logged-in user's permissions are stored in the template variable
- ``{{ perms }}``. This is an instance of
- :class:`django.contrib.auth.context_processors.PermWrapper`, which is a
- template-friendly proxy of permissions.
- .. versionchanged:: 1.3
- Prior to version 1.3, ``PermWrapper`` was located in
- ``django.contrib.auth.context_processors``.
- In the ``{{ perms }}`` object, single-attribute lookup is a proxy to
- :meth:`User.has_module_perms <django.contrib.auth.models.User.has_module_perms>`.
- This example would display ``True`` if the logged-in user had any permissions
- in the ``foo`` app::
- {{ perms.foo }}
- Two-level-attribute lookup is a proxy to
- :meth:`User.has_perm <django.contrib.auth.models.User.has_perm>`. This example
- would display ``True`` if the logged-in user had the permission
- ``foo.can_vote``::
- {{ perms.foo.can_vote }}
- Thus, you can check permissions in template ``{% if %}`` statements:
- .. code-block:: html+django
- {% if perms.foo %}
- <p>You have permission to do something in the foo app.</p>
- {% if perms.foo.can_vote %}
- <p>You can vote!</p>
- {% endif %}
- {% if perms.foo.can_drive %}
- <p>You can drive!</p>
- {% endif %}
- {% else %}
- <p>You don't have permission to do anything in the foo app.</p>
- {% endif %}
- Groups
- ======
- Groups are a generic way of categorizing users so you can apply permissions, or
- some other label, to those users. A user can belong to any number of groups.
- A user in a group automatically has the permissions granted to that group. For
- example, if the group ``Site editors`` has the permission
- ``can_edit_home_page``, any user in that group will have that permission.
- Beyond permissions, groups are a convenient way to categorize users to give
- them some label, or extended functionality. For example, you could create a
- group ``'Special users'``, and you could write code that could, say, give them
- access to a members-only portion of your site, or send them members-only email
- messages.
- API reference
- -------------
- .. class:: models.Group
- Fields
- ~~~~~~
- :class:`~django.contrib.auth.models.Group` objects have the following fields:
- .. attribute:: Group.name
- Required. 80 characters or fewer. Any characters are permitted. Example:
- ``'Awesome Users'``.
- .. attribute:: Group.permissions
- Many-to-many field to :class:`~django.contrib.auth.models.Permissions`::
- group.permissions = [permission_list]
- group.permissions.add(permission, permission, ...)
- group.permissions.remove(permission, permission, ...)
- group.permissions.clear()
- .. _authentication-backends:
- Other authentication sources
- ============================
- The authentication that comes with Django is good enough for most common cases,
- but you may have the need to hook into another authentication source -- that
- is, another source of usernames and passwords or authentication methods.
- For example, your company may already have an LDAP setup that stores a username
- and password for every employee. It'd be a hassle for both the network
- administrator and the users themselves if users had separate accounts in LDAP
- and the Django-based applications.
- So, to handle situations like this, the Django authentication system lets you
- plug in other authentication sources. You can override Django's default
- database-based scheme, or you can use the default system in tandem with other
- systems.
- See the :doc:`authentication backend reference </ref/authbackends>`
- for information on the authentication backends included with Django.
- Specifying authentication backends
- ----------------------------------
- Behind the scenes, Django maintains a list of "authentication backends" that it
- checks for authentication. When somebody calls
- :func:`django.contrib.auth.authenticate()` -- as described in :ref:`How to log
- a user in <how-to-log-a-user-in>` above -- Django tries authenticating across
- all of its authentication backends. If the first authentication method fails,
- Django tries the second one, and so on, until all backends have been attempted.
- The list of authentication backends to use is specified in the
- :setting:`AUTHENTICATION_BACKENDS` setting. This should be a tuple of Python
- path names that point to Python classes that know how to authenticate. These
- classes can be anywhere on your Python path.
- By default, :setting:`AUTHENTICATION_BACKENDS` is set to::
- ('django.contrib.auth.backends.ModelBackend',)
- That's the basic authentication scheme that checks the Django users database.
- The order of :setting:`AUTHENTICATION_BACKENDS` matters, so if the same
- username and password is valid in multiple backends, Django will stop
- processing at the first positive match.
- .. note::
- Once a user has authenticated, Django stores which backend was used to
- authenticate the user in the user's session, and re-uses the same backend
- for subsequent authentication attempts for that user. This effectively means
- that authentication sources are cached, so if you change
- :setting:`AUTHENTICATION_BACKENDS`, you'll need to clear out session data if
- you need to force users to re-authenticate using different methods. A simple
- way to do that is simply to execute ``Session.objects.all().delete()``.
- Writing an authentication backend
- ---------------------------------
- An authentication backend is a class that implements two methods:
- ``get_user(user_id)`` and ``authenticate(**credentials)``.
- The ``get_user`` method takes a ``user_id`` -- which could be a username,
- database ID or whatever -- and returns a ``User`` object.
- The ``authenticate`` method takes credentials as keyword arguments. Most of
- the time, it'll just look like this::
- class MyBackend(object):
- def authenticate(self, username=None, password=None):
- # Check the username/password and return a User.
- But it could also authenticate a token, like so::
- class MyBackend(object):
- def authenticate(self, token=None):
- # Check the token and return a User.
- Either way, ``authenticate`` should check the credentials it gets, and it
- should return a ``User`` object that matches those credentials, if the
- credentials are valid. If they're not valid, it should return ``None``.
- The Django admin system is tightly coupled to the Django ``User`` object
- described at the beginning of this document. For now, the best way to deal with
- this is to create a Django ``User`` object for each user that exists for your
- backend (e.g., in your LDAP directory, your external SQL database, etc.) You
- can either write a script to do this in advance, or your ``authenticate``
- method can do it the first time a user logs in.
- Here's an example backend that authenticates against a username and password
- variable defined in your ``settings.py`` file and creates a Django ``User``
- object the first time a user authenticates::
- from django.conf import settings
- from django.contrib.auth.models import User, check_password
- class SettingsBackend(object):
- """
- Authenticate against the settings ADMIN_LOGIN and ADMIN_PASSWORD.
- Use the login name, and a hash of the password. For example:
- ADMIN_LOGIN = 'admin'
- ADMIN_PASSWORD = 'sha1$4e987$afbcf42e21bd417fb71db8c66b321e9fc33051de'
- """
- supports_inactive_user = False
- def authenticate(self, username=None, password=None):
- login_valid = (settings.ADMIN_LOGIN == username)
- pwd_valid = check_password(password, settings.ADMIN_PASSWORD)
- if login_valid and pwd_valid:
- try:
- user = User.objects.get(username=username)
- except User.DoesNotExist:
- # Create a new user. Note that we can set password
- # to anything, because it won't be checked; the password
- # from settings.py will.
- user = User(username=username, password='get from settings.py')
- user.is_staff = True
- user.is_superuser = True
- user.save()
- return user
- return None
- def get_user(self, user_id):
- try:
- return User.objects.get(pk=user_id)
- except User.DoesNotExist:
- return None
- Handling authorization in custom backends
- -----------------------------------------
- Custom auth backends can provide their own permissions.
- The user model will delegate permission lookup functions
- (:meth:`~django.contrib.auth.models.User.get_group_permissions()`,
- :meth:`~django.contrib.auth.models.User.get_all_permissions()`,
- :meth:`~django.contrib.auth.models.User.has_perm()`, and
- :meth:`~django.contrib.auth.models.User.has_module_perms()`) to any
- authentication backend that implements these functions.
- The permissions given to the user will be the superset of all permissions
- returned by all backends. That is, Django grants a permission to a user that
- any one backend grants.
- The simple backend above could implement permissions for the magic admin
- fairly simply::
- class SettingsBackend(object):
- # ...
- def has_perm(self, user_obj, perm, obj=None):
- if user_obj.username == settings.ADMIN_LOGIN:
- return True
- else:
- return False
- This gives full permissions to the user granted access in the above example.
- Notice that the backend auth functions all take the user object as an argument,
- and they also accept the same arguments given to the associated
- :class:`django.contrib.auth.models.User` functions.
- A full authorization implementation can be found in
- `django/contrib/auth/backends.py`_, which is the default backend and queries
- the ``auth_permission`` table most of the time.
- .. _django/contrib/auth/backends.py: http://code.djangoproject.com/browser/django/trunk/django/contrib/auth/backends.py
- .. _anonymous_auth:
- Authorization for anonymous users
- ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
- .. versionchanged:: 1.2
- An anonymous user is one that is not authenticated i.e. they have provided no
- valid authentication details. However, that does not necessarily mean they are
- not authorized to do anything. At the most basic level, most Web sites
- authorize anonymous users to browse most of the site, and many allow anonymous
- posting of comments etc.
- Django's permission framework does not have a place to store permissions for
- anonymous users. However, it has a foundation that allows custom authentication
- backends to specify authorization for anonymous users. This is especially useful
- for the authors of re-usable apps, who can delegate all questions of authorization
- to the auth backend, rather than needing settings, for example, to control
- anonymous access.
- Authorization for inactive users
- ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
- .. versionadded:: 1.3
- An inactive user is a one that is authenticated but has its attribute
- ``is_active`` set to ``False``. However this does not mean they are not
- authorized to do anything. For example they are allowed to activate their
- account.
- The support for anonymous users in the permission system allows for
- anonymous users to have permissions to do something while inactive
- authenticated users do not.
- To enable this on your own backend, you must set the class attribute
- ``supports_inactive_user`` to ``True``.
- A nonexisting ``supports_inactive_user`` attribute will raise a
- ``PendingDeprecationWarning`` if used in Django 1.3. In Django 1.4, this
- warning will be updated to a ``DeprecationWarning`` which will be displayed
- loudly. Additionally ``supports_inactive_user`` will be set to ``False``.
- Django 1.5 will assume that every backend supports inactive users being
- passed to the authorization methods.
- Handling object permissions
- ---------------------------
- Django's permission framework has a foundation for object permissions, though
- there is no implementation for it in the core. That means that checking for
- object permissions will always return ``False`` or an empty list (depending on
- the check performed).
|