测试开发之路--Flask 之旅 (四):登录与权限控制

简介: 本文介绍了如何使用 Flask-Security 和 Flask-Login 实现权限管理和用户登录功能。首先讲解了 Flask 的消息闪现功能,用于向模板传递信息。接着介绍了如何利用 Flask-Security 的 `roles_required` 和 `roles_accepted` 装饰器保护页面,并展示了如何通过 `current_user` 获取当前用户信息。最后提到了使用 Flask-Login 时应避免升级至 0.4.0 版本,以防出现兼容性问题。通过这些技术,搭建了一个基本的用户权限管理系统。

来源:https://testerhome.com

背景

接着上一篇,我们现在拥有了使用数据库的能力并使用Flask-Security针对Flask-SQLAlchemy的扩展完成了对User和Role表的维护。今天我们再讲讲Flask-Security是怎么做权限管理的并介绍Flask-Security是如何扩展Flask-Login做用户管理的

消息闪现

在讲今天的主题前我们先补充一下之前漏掉的一个东西,就是Flask的消息闪现。 它是Flask的定制方法。可以像模板页面传递一条或多条信息而不需要像模板传递任何参数。我们直接就可以在页面中获取名为这段信息。 举个例子,还记得我们之前讲表单的时候,每个form对象都有一个errors属性么。我们之前是将form这个对象传递给模板页面进行渲染。现在我们直接用flush函数进行传递。如下:

代码语言:javascript

复制

@app.route('/config/save', methods=['POST'])
def save_config():
    form = forms.ConfigForm()
    if form.validate_on_submit():
        update_config(form)
    else:
        flash(form.errors)
        return render_template('detail.html', form=form)
    return redirect(url_for('index'))

上面我们在表单验证不通过的时候向用户发送一个闪现消息。 其中flash函数把form的errors封装了起来。 我们来看看在模板页面中是怎么获取这段error信息的。

代码语言:javascript

复制

{% with messages = get_flashed_messages() %}
            {% if messages %}
                <ul class=flashes>
                    {% for message in messages %}
                        <li><font color="red">{{ message }}</font></li>
                    {% endfor %}
                </ul>
            {% endif %}
        {% endwith %}

上面是页面的代码。 我们可以使用Falsk内置的方法获取所有用flash封装的信息。 我们把这段代码放在base.html里面然后在页面上继承base.html即可。

角色保护

好了,回到主线。当我们拥有了用户和角色以后, 就可以很方便的使用Flask-Security的装饰器来保护我们的页面了。 @roles_required('Admin')可以用来保护一个路由方法。只有当前用户拥有Admin的角色的时候才被准许访问(关于当前用户的管理是Flask-Login的内容,我们暂且知道这个事情就好,之后我会详细解释)。例如下面的使用方法:

代码语言:javascript

复制

@app.route('/config/save', methods=['POST'])
@roles_required('Admin')
def save_config():

注:roles_required这个装饰器一定要放在app.route下面。否则会有问题。 当用户没有Admin权限的时候是无法访问这个路由方法的。它会通过flush函数像页面反馈错误信息。如下:

除了roles_required之外,你还可以使用roles_accepted。 他们之前的区别就是前者必须严格的拥有所指定的权限,后者是只要拥有其中一个权限就可以。如下:

代码语言:javascript

复制

@roles_accepted('editor', 'author')

这段的意思就是要求用户至少拥有这两种role其中的一种就可以访问。 当然除了使用装饰器以外,Flask-Security也是支持以编码的方式控制权限的。例如:

代码语言:javascript

复制

@app.route('/')
@app.route('/index', methods=['GET', 'POST'])
@login_required
def index():
    if current_user.has_role('Admin'):
        names = Env.query.all()
        return render_template('index.html', names=names)
    else:
        names = current_user.envs
        return render_template('index.html', names=names)

这段代码我们使用current_user.has_role方法来判断我们当前用户是属于哪一种角色的(current_user是Flask-Security针对Flask-Login做的扩展,作用是在当前session中维护用户的信息,之后会详细说明)。关于Flask-Security提供的更多方法,请参照文档:https://pythonhosted.org/Flask-Security/api.html

Flask-Login

初始化Flask-Login

代码语言:javascript

复制

from flask_login import LoginManager
login_manager = LoginManager()
login_manager.init_app(app)
login_manager.login_view = 'login'

我们使用LoginManager来初始化,并把login_view设置为login方法。 这是为了之后给登录保护使用的。当Flask-Login检测到用户没有登录的时候会把链接重定向到login_view中去。 所以我们设置为login。 一个叫login的路由方法。

登录

终于说到Login了,写帖子的时候我挺纠结的,到底先讲什么后讲什么。因为这三个模块相辅相成。尤其是Flask-Security,它其实就是扩展了其他各种模块。想用它就必须安装其他模块,而且它的封装改变了其他模块的使用方式。 例如Flask-Login有login和logout方法。Flask-Security也有,你需要使用Flask-Security提供的方法进行登录和登出才能做好权限控制,因为原生的Flask-Loing无法保存User和Role相关的信息。你可以这么看待Flask-Security,它的作用就是封装了其他模块。它的底层其实也是调用的其他模块。 举个例子,当你想要做用户登录的时候。我们使用如下的方式:

代码语言:javascript

复制

from flask_security.utils import login_user, logout_user
@app.route('/user/register', methods=['GET', 'POST'])
def register():
    form = forms.RegisterForm()
    if form.validate_on_submit():
        if form.password.data != form.password_again.data:
            errors = '两次输入的密码不同'
            return render_template('register.html', form=form, errors=errors)
        new_user = user_datastore.create_user(email=form.email.data, password=form.password.data)
        normal_role = user_datastore.find_role('User')
        db.session.add(new_user)
        user_datastore.add_role_to_user(new_user, normal_role)
        login_user(new_user)
        return redirect(url_for('index'))
    return render_template('register.html', form=form)

这是一个用户注册的代码。表单验证的部分我先不看了。 我们首先使用Flask-Security的方法在数据库中创建用户信息,给用户添加为普通用户的权限。然后调用login_user方法进行登录。这里需要注意的是,请看我一开始import的是flask_security.utiles中的login_user方法而不是Flask-Login的。就像刚我才说的Flask-Security是封装了其他各种模块的存在。所以现在我们是完全使用Flask-Security的方式来进行登录。 当我们登录了以后,我们的用户信息,也就是User对象会自动的保存在session中。 我们可以通过引入current_user的方式获取当前的用户。如下:

代码语言:javascript

复制

from flask_security import  current_user
@app.route('/user/login', methods=['POST', 'GET'])
def login():
    if not current_user.is_anonymous:
        return redirect(url_for('index'))
    form = forms.LoginForm()
    if form.validate_on_submit():
        user = User.query.filter_by(email=form.email.data).first()
        if user is None:
            form.errors['username_error'] = 'user does not exit'
            return render_template('login.html', form=form)
        if not user.password == form.password.data:
            form.errors['password_error'] = 'password is not right'
            return render_template('login.html', form=form)
        login_user(user, remember=True)
        return redirect(url_for('index'))
    return render_template('login.html', form=form)

上面是登录的代码。我们可以看到我们import的仍然是Flask-Security的current_user而不是Flask-Login的。 我们先判断当前的user是不是匿名用户(未登录的就是匿名用户)。如果不是,说明已经登录了,重定向到首页。这里我们就是引用了current_user。它其实就是我们使用Flask-Security创建的User对象。它包含了所有的User对象的属性和方法。可以看到我们发现用户未登录后,首先判断是不是表单提交以及表单提交是否通过。如果通过了就从数据库中查询出用户的信息。 判断用户是否存在以及填写的密码是否正确。登录后,这个User对象就赋值给current_user了。 我们甚至可以在模板页面中直接使用current_user。如下:

代码语言:javascript

复制

<div class="col-md-12 column">
            <h4>
                你好 {{ current_user.email }}
            </h4>
        </div>

Flask-Login自动会把current_user传递给模板页面。 所以我们就这样直接在页面引用就可以了。 我们把上面的代码写入到base.html,提取每个用户的邮箱。 并显示

登录保护

Flask-Security提供了roles_required这种装饰器来进行权限的保护。自然也就提供了login_required的装饰器来进行登录保护。如下:

代码语言:javascript

复制

@app.route('/user/logout')
@login_required
def logout():

当Flask发现用户并没有登录的时候,就会把链接重定向到我们一开始设置的login_view的页面上了。也就是登录页面。

Flask-Login和Flask-Security有个巨坑无比的事情就是,大家尽量不要把Flask-Login的版本升级到0.4.0, 使用0.3.2就可以了。 因为Flask-Security在封装Flask-Login 0.4.0的时候会报一个找不到token_loader的错误。因为Flask-Login在0.4.0的时候已经不使用这个token了。

总结

好了,经过这么一折腾,我们的环境管理平台终于可以给人用了。之后我会再优化完善一些东西,会继续更新一些内容

目录
相关文章
|
5月前
|
存储 安全 API
认证服务---OAuth2.0基本介绍,微博登录测试【上篇】
这篇文章是关于OAuth2.0的介绍和微博登录测试的教程,详细解释了OAuth2.0的基本概念和授权流程,并指导读者如何在新浪微博开放平台进行应用创建、设置回调地址,以及使用Postman工具进行授权测试,为实现第三方微博登录功能做准备。
认证服务---OAuth2.0基本介绍,微博登录测试【上篇】
|
3月前
|
测试技术 数据安全/隐私保护
北邮人论坛登录页面测试用例
北邮人论坛登录页面测试用例
62 1
|
4月前
|
JavaScript 前端开发 数据库
测试开发之路--Flask 之旅 (五):后台管理
本文介绍了如何使用 Flask-Admin 模块为应用添加后台管理功能,包括数据库表管理、自定义视图及服务器文件管理。通过实例展示了如何初始化 Flask-Admin,并实现对用户、角色等表的增删查改操作。此外,还介绍了如何定制视图及管理服务器上的配置文件。这一模块大大提升了应用的管理效率与灵活性。
90 5
测试开发之路--Flask 之旅 (五):后台管理
|
4月前
|
JavaScript 前端开发 数据库
测试开发之路--Flask 之旅 (三):数据库
本文介绍了在 Flask 应用中实现权限管理的过程,包括使用 Flask-SQLAlchemy、Flask-MySQLdb、Flask-Security 和 Flask-Login 等扩展模块进行数据库配置与用户权限设置。首先创建数据库并定义用户、环境和角色模型,接着通过 Flask-Security 初始化用户和角色,并展示了如何便捷地管理权限。后续将深入探讨权限控制的具体应用。
92 4
测试开发之路--Flask 之旅 (三):数据库
|
5月前
|
存储 前端开发 数据库
基于python flask 的图书管理系统,有登录界面,实现简单增删改查,可以做课程设计使用
本文介绍了一个基于Python Flask框架的图书管理系统,该系统具备登录界面,并实现了基本的增删改查功能,适合作为课程设计使用。
121 3
基于python flask 的图书管理系统,有登录界面,实现简单增删改查,可以做课程设计使用
|
8月前
|
前端开发 安全 JavaScript
Python的Flask框架的学习笔记(前后端变量传送,文件上传,网页返回)内含实战:实现一个简单的登录页面
Python的Flask框架的学习笔记(前后端变量传送,文件上传,网页返回)内含实战:实现一个简单的登录页面
197 0
|
6月前
|
JavaScript 测试技术
vue 官方测试工具 unit-jest 实用教程(含实战范例:登录组件的测试)
vue 官方测试工具 unit-jest 实用教程(含实战范例:登录组件的测试)
102 0
|
6月前
|
JSON 数据格式
postman 实用教程(含带 token 访问需登录权限的接口、测试文件上传接口、测试文件下载接口)
postman 实用教程(含带 token 访问需登录权限的接口、测试文件上传接口、测试文件下载接口)
1182 0
|
8月前
|
前端开发 Java 测试技术
selenium+python自动化测试--登录
selenium+python自动化测试--登录
153 2