使用Flask-Login的几个基本步骤
定义安全的用户模型
- 不要直接在数据库中保存用户的密码,应该存储他们的哈希值。werkzeug.security 模块中的
generate_password_hash, check_password_hash
函数来进行哈希值产生和验证。 - 密码应该是只写的, 只有写入和验证两个选项。
from flask_login import UserMixin
from werkzeug.security import generate_password_hash, check_password_hash
from myapp import db, login_manager
class User(UserMixin, db.Model):
__tablename__ = "users"
id = db.Column(db.Integer, primary_key=True)
username = db.Column(db.String(64), nullable=False, unique=True, index=True)
email = db.Column(db.String(64), nullable=False, unique=True, index=True)
password_hash = db.Column(db.String(128), nullable=False)
role_id = db.Column(db.Integer, db.ForeignKey('roles.id'))
def __init__(self, username, email, password, role_id):
self.username = username
self.email = email
self.password = password
self.role_id = role_id
@property
def password(self):
raise AttributeError("Error, password is a write only attribute")
@password.setter
def password(self, password):
print "password", password
self.password_hash = generate_password_hash(password)
print "password hash:" , self.password_hash
def verify_password(self, password):
return check_password_hash(self.password_hash, password)
使用FLask-Login插件包装app,定义一个 login_manager
在app定义的地方加入如下代码:
from flask_login import LoginManager
login_manager = LoginManager()
login_manager.session_protection = "strong"
## 在这里 auth 为一个 bluepint, login是其中的一个 function
login_manager.login_view = "auth.login"
login_manager.init_app(app)
定义 login_manager的user_loader函数
@login_manager.user_loader
def load_user(id):
return User.query.get(int(id))
使用 flask_login.login_required 保护一些只能被登入用户看到的视图
from flask_login import login_required
@app.route("/secret")
@login_required
def secret():
return "Only logined user can view this"
使用 flask_login.login_user 函数使用户登入
from flask_login import login_user
@auth.route("/login", methods=["GET", "POST"])
def login():
form = LoginForm()
# 验证表单,的每一项是否符合要求
if form.validate_on_submit():
user = User.query.filter_by(email=form.email.data).first()
if user is not None and user.verify_password(form.password.data):
login_user(user, form.remember_me.data)
return redirect(request.args.get('next') or url_for('hello'))
flash('Invalid username or password.')
#如果来的是get请求,那么直接返回模板
return render_template('auth/login.html', form=form)
其他相关主题
- Flask-Login可以搭配使用Flask-WTF进行登陆表单的生成和验证 一种简单的表单使用方式如下
继承一个 flask_wtf.Form 类
# coding: utf-8
from flask_wtf import Form
from wtforms import StringField, PasswordField, BooleanField, SubmitField
from wtforms.validators import Required, Email, Length
# 每一个form都可以用flask_wtf.Form里面的一些属性和方法
# 但同时需要引入 wtforms 包来完善功能
class LoginForm(Form):
email = StringField("Email", validators=[Required(), Length(1,64), Email()])
password = PasswordField('Password', validators=[Required()])
remember_me = BooleanField("Keep me login")
submit = SubmitField("Login")
在模板中渲染表单
## 在模板的开头导入 wtf
\{\% import "bootstrap/wtf.html" as wtf \%\}
# 在模板中需要写表单的地方生成表单
\{\{ wtf.quick_form(form) \}\}
在view函数中使用 form对象
- 可参见本页上面的 auth.login视图