mirror of
https://github.com/wassname/flask-security.git
synced 2026-06-27 16:10:11 +08:00
Add the ability to specify additional fields on the user model that can be used for logging in.
This commit is contained in:
@@ -84,7 +84,8 @@ _default_config = {
|
||||
'EMAIL_SUBJECT_PASSWORDLESS': 'Login instructions',
|
||||
'EMAIL_SUBJECT_PASSWORD_NOTICE': 'Your password has been reset',
|
||||
'EMAIL_SUBJECT_PASSWORD_CHANGE_NOTICE': 'Your password has been changed',
|
||||
'EMAIL_SUBJECT_PASSWORD_RESET': 'Password reset instructions'
|
||||
'EMAIL_SUBJECT_PASSWORD_RESET': 'Password reset instructions',
|
||||
'USER_IDENTITY_ATTRIBUTES': ['email']
|
||||
}
|
||||
|
||||
#: Default Flask-Security messages
|
||||
|
||||
+28
-18
@@ -9,6 +9,8 @@
|
||||
:license: MIT, see LICENSE for more details.
|
||||
"""
|
||||
|
||||
from .utils import get_identity_attributes
|
||||
|
||||
|
||||
class Datastore(object):
|
||||
def __init__(self, db):
|
||||
@@ -179,14 +181,14 @@ class SQLAlchemyUserDatastore(SQLAlchemyDatastore, UserDatastore):
|
||||
SQLAlchemyDatastore.__init__(self, db)
|
||||
UserDatastore.__init__(self, user_model, role_model)
|
||||
|
||||
def get_user(self, id_or_email):
|
||||
returned = None
|
||||
if self._is_numeric(id_or_email):
|
||||
returned = self.user_model.query.get(id_or_email)
|
||||
if not returned:
|
||||
returned = self.user_model.query.filter(
|
||||
self.user_model.email.ilike(id_or_email)).first()
|
||||
return returned
|
||||
def get_user(self, identifier):
|
||||
if self._is_numeric(identifier):
|
||||
return self.user_model.query.get(identifier)
|
||||
for attr in get_identity_attributes():
|
||||
query = getattr(self.user_model, attr).ilike(identifier)
|
||||
rv = self.user_model.query.filter(query).first()
|
||||
if rv is not None:
|
||||
return rv
|
||||
|
||||
def _is_numeric(self, value):
|
||||
try:
|
||||
@@ -210,12 +212,18 @@ class MongoEngineUserDatastore(MongoEngineDatastore, UserDatastore):
|
||||
MongoEngineDatastore.__init__(self, db)
|
||||
UserDatastore.__init__(self, user_model, role_model)
|
||||
|
||||
def get_user(self, id_or_email):
|
||||
def get_user(self, identifier):
|
||||
from mongoengine import ValidationError
|
||||
try:
|
||||
return self.user_model.objects(id=id_or_email).first()
|
||||
return self.user_model.objects(id=identifier).first()
|
||||
except ValidationError:
|
||||
return self.user_model.objects(email__iexact=id_or_email).first()
|
||||
pass
|
||||
for attr in get_identity_attributes():
|
||||
query_key = '%s__iexact' % attr
|
||||
query = {query_key: identifier}
|
||||
rv = self.user_model.objects(**query).first()
|
||||
if rv is not None:
|
||||
return rv
|
||||
|
||||
def find_user(self, **kwargs):
|
||||
try:
|
||||
@@ -254,16 +262,18 @@ class PeeweeUserDatastore(PeeweeDatastore, UserDatastore):
|
||||
UserDatastore.__init__(self, user_model, role_model)
|
||||
self.UserRole = role_link
|
||||
|
||||
def get_user(self, id_or_email):
|
||||
def get_user(self, identifier):
|
||||
try:
|
||||
return self.user_model.get(self.user_model.id == id_or_email)
|
||||
return self.user_model.get(self.user_model.id == identifier)
|
||||
except ValueError:
|
||||
pass
|
||||
try:
|
||||
return self.user_model.get(self.user_model.email ** id_or_email)
|
||||
except self.user_model.DoesNotExist:
|
||||
pass
|
||||
return None
|
||||
|
||||
for attr in get_identity_attributes():
|
||||
column = getattr(self.user_model, attr)
|
||||
try:
|
||||
return self.user_model.get(column ** identifier)
|
||||
except self.user_model.DoesNotExist:
|
||||
pass
|
||||
|
||||
def find_user(self, **kwargs):
|
||||
try:
|
||||
|
||||
@@ -220,6 +220,7 @@ class LoginForm(Form, NextFormMixin):
|
||||
self.password.errors.append(get_message('PASSWORD_NOT_PROVIDED')[0])
|
||||
return False
|
||||
|
||||
|
||||
self.user = _datastore.get_user(self.email.data)
|
||||
|
||||
if self.user is None:
|
||||
|
||||
@@ -279,6 +279,16 @@ def get_token_status(token, serializer, max_age=None):
|
||||
return expired, invalid, user
|
||||
|
||||
|
||||
def get_identity_attributes(app=None):
|
||||
app = app or current_app
|
||||
attrs = app.config['SECURITY_USER_IDENTITY_ATTRIBUTES']
|
||||
try:
|
||||
attrs = [f.strip() for f in attrs.split(',')]
|
||||
except AttributeError:
|
||||
pass
|
||||
return attrs
|
||||
|
||||
|
||||
@contextmanager
|
||||
def capture_passwordless_login_requests():
|
||||
login_requests = []
|
||||
|
||||
@@ -826,3 +826,14 @@ class ConfirmableExtendFormsTest(SecurityTest):
|
||||
def test_send_confirmation(self):
|
||||
r = self._get('/confirm', follow_redirects=True)
|
||||
self.assertIn("My Send Confirmation Email Address Field", r.data)
|
||||
|
||||
|
||||
class AdditionalUserIdentityAttributes(SecurityTest):
|
||||
|
||||
AUTH_CONFIG = {
|
||||
'SECURITY_USER_IDENTITY_ATTRIBUTES': ('email', 'username')
|
||||
}
|
||||
|
||||
def test_authenticate(self):
|
||||
r = self.authenticate(email='matt')
|
||||
self.assertIn('Hello matt@lp.com', r.data)
|
||||
|
||||
@@ -128,17 +128,17 @@ def create_roles():
|
||||
|
||||
|
||||
def create_users(count=None):
|
||||
users = [('matt@lp.com', 'password', ['admin'], True),
|
||||
('joe@lp.com', 'password', ['editor'], True),
|
||||
('dave@lp.com', 'password', ['admin', 'editor'], True),
|
||||
('jill@lp.com', 'password', ['author'], True),
|
||||
('tiya@lp.com', 'password', [], False)]
|
||||
users = [('matt@lp.com', 'matt', 'password', ['admin'], True),
|
||||
('joe@lp.com', 'joe', 'password', ['editor'], True),
|
||||
('dave@lp.com', 'dave', 'password', ['admin', 'editor'], True),
|
||||
('jill@lp.com', 'jill', 'password', ['author'], True),
|
||||
('tiya@lp.com', 'tiya', 'password', [], False)]
|
||||
count = count or len(users)
|
||||
|
||||
for u in users[:count]:
|
||||
pw = encrypt_password(u[1])
|
||||
ds.create_user(email=u[0], password=pw,
|
||||
roles=u[2], active=u[3])
|
||||
pw = encrypt_password(u[2])
|
||||
ds.create_user(email=u[0], username=u[1], password=pw,
|
||||
roles=u[3], active=u[4])
|
||||
ds.commit()
|
||||
|
||||
|
||||
|
||||
@@ -30,6 +30,7 @@ def create_app(config, **kwargs):
|
||||
|
||||
class User(db.Document, UserMixin):
|
||||
email = db.StringField(unique=True, max_length=255)
|
||||
username = db.StringField(max_length=255)
|
||||
password = db.StringField(required=True, max_length=255)
|
||||
last_login_at = db.DateTimeField()
|
||||
current_login_at = db.DateTimeField()
|
||||
|
||||
@@ -29,6 +29,7 @@ def create_app(config, **kwargs):
|
||||
|
||||
class User(db.Model, UserMixin):
|
||||
email = TextField()
|
||||
username = TextField()
|
||||
password = TextField()
|
||||
last_login_at = DateTimeField(null=True)
|
||||
current_login_at = DateTimeField(null=True)
|
||||
|
||||
@@ -32,6 +32,7 @@ def create_app(config, **kwargs):
|
||||
class User(db.Model, UserMixin):
|
||||
id = db.Column(db.Integer, primary_key=True)
|
||||
email = db.Column(db.String(255), unique=True)
|
||||
username = db.Column(db.String(255))
|
||||
password = db.Column(db.String(255))
|
||||
last_login_at = db.Column(db.DateTime())
|
||||
current_login_at = db.Column(db.DateTime())
|
||||
|
||||
Reference in New Issue
Block a user