1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175
|
.. _authentication:
Authentication
==============
The :py:class:`Authentication` class provides a means of authenticating users
of the site. It is designed to work out-of-the-box with a simple ``User`` model,
but can be heavily customized.
The :py:class:`Auth` system is comprised of a single class which is responsible
for coordinating incoming requests to your project with known users. It provides
the following:
* views for login and logout
* model to store user data (or you can provide your own)
* mechanism for identifying users across requests (uses session storage)
All of these pieces can be customized, but the default out-of-box implementation
aims to provide a good starting place.
The auth system is also designed to work closely with the :ref:`admin-interface`.
Getting started
---------------
In order to provide a method for users to authenticate with your site, instantiate
an :py:class:`Auth` backend for your project:
.. code-block:: python
from flask import Flask
from flask_peewee.auth import Auth
from flask_peewee.db import Database
app = Flask(__name__)
db = Database(app)
# needed for authentication
auth = Auth(app, db)
.. note::
``user`` is reserverd keyword in Postgres. Pass db_table to Auth to override db table.
Marking areas of the site as login required
-------------------------------------------
If you want to mark specific areas of your site as requiring auth, you can
decorate views using the :py:meth:`Auth.login_required` decorator:
.. code-block:: python
@app.route('/private/')
@auth.login_required
def private_timeline():
user = auth.get_logged_in_user()
# ... display the private timeline for the logged-in user
If the request comes from someone who has not logged-in with the site, they are
redirected to the :py:meth:`Auth.login` view, which allows the user to authenticate.
After successfully logging-in, they will be redirected to the page they requested
initially.
Retrieving the current user
---------------------------
Whenever in a `request context <http://flask.pocoo.org/docs/reqcontext/>`_, the
currently logged-in user is available by calling :py:meth:`Auth.get_logged_in_user`,
which will return ``None`` if the requesting user is not logged in.
The auth system also registers a pre-request hook that stores the currently logged-in
user in the special flask variable ``g``.
Accessing the user in the templates
-----------------------------------
The auth system registers a template context processor which makes the logged-in
user available in any template:
.. code-block:: html
{% if user %}
<p>Hello {{ user.username }}</p>
{% else %}
<p>Please <a href="{{ url_for('auth.login') }}?next={{ request.path }}">log in</a></p>
{% endif %}
Using a custom "User" model
---------------------------
It is easy to use your own model for the ``User``, though depending on the amount
of changes it may be necessary to override methods in both the :py:class:`Auth` and
:py:class:`Admin` classes.
Unless you want to override the default behavior of the :py:class:`Auth` class' mechanism
for actually authenticating users (which you may want to do if relying on a 3rd-party
for auth) -- you will want to be sure your ``User`` model implements two methods:
* ``set_password(password)`` -- takes a raw password and stores an encrypted version on model
* ``check_password(password)`` -- returns whether or not the supplied password matches
the one stored on the model instance
.. note::
The :py:class:`BaseUser` mixin provides default implementations of these two methods.
Here's a simple example of extending the auth system to use a custom user model:
.. code-block:: python
from flask_peewee.auth import BaseUser # <-- implements set_password and check_password
app = Flask(__name__)
db = Database(app)
# create our custom user model. note that we're mixing in BaseUser in order to
# use the default auth methods it implements, "set_password" and "check_password"
class User(db.Model, BaseUser):
username = CharField()
password = CharField()
email = CharField()
# ... our custom fields ...
is_superuser = BooleanField()
# create a modeladmin for it
class UserAdmin(ModelAdmin):
columns = ('username', 'email', 'is_superuser',)
# Make sure the user's password is hashed, after it's been changed in
# the admin interface. If we don't do this, the password will be saved
# in clear text inside the database and login will be impossible.
def save_model(self, instance, form, adding=False):
orig_password = instance.password
user = super(UserAdmin, self).save_model(instance, form, adding)
if orig_password != form.password.data:
user.set_password(form.password.data)
user.save()
return user
# subclass Auth so we can return our custom classes
class CustomAuth(Auth):
def get_user_model(self):
return User
def get_model_admin(self):
return UserAdmin
# instantiate the auth
auth = CustomAuth(app, db)
Here's how you might integrate the custom auth with the admin area of your site:
.. code-block:: python
# subclass Admin to check for whether the user is a superuser
class CustomAdmin(Admin):
def check_user_permission(self, user):
return user.is_superuser
# instantiate the admin
admin = CustomAdmin(app, auth)
admin.register(User, UserAdmin)
admin.setup()
|