用另一个装饰器包裹@roles_accepted 装饰器!在下面的代码中,用于提取角色的变量名被传递到外部装饰器中,尽管它可能是硬编码的。
def roles_accepted_from_route(variable_name):
def decorator(func):
@wraps(func)
def wrapper(*args, **kwargs):
_variable = kwargs.get(variable_name, None)
if _variable:
print(_variable, _variable[:3])
@roles_accepted('admin', _variable[:3])
def inner_wrapper(*args, **kwargs):
print(f'roles_accepted returned True, variable: {_variable[:3]}')
return func(*args, **kwargs)
return inner_wrapper(*args, **kwargs)
return wrapper
return decorator
并按如下方式使用:
@app.route('/<string:_id>')
@roles_accepted_from_route(variable_name='_id')
def some_route(_id):
return f'Route Variable: {_id}'
下面的单文件示例,几乎没有错误检查。在第一个 Flask 请求中创建一个具有email: 'fred@example.net', password: 'password' 角色的用户'LON'。
from functools import wraps
from flask import Flask
from flask_security import roles_accepted
from flask_security.utils import hash_password
from flask_sqlalchemy import SQLAlchemy
from flask_security import Security, SQLAlchemyUserDatastore, UserMixin, RoleMixin, current_user
app = Flask(__name__)
app.config['DEBUG'] = True
app.config['SECRET_KEY'] = 'super-secret'
app.config['SQLALCHEMY_DATABASE_URI'] = 'sqlite:///test.db3'
app.config['SECURITY_PASSWORD_SALT'] = '5ce39dc7add2284076de45b923d74dd00a052117cdf0ab900548565681e56fce'
# Create database connection object
db = SQLAlchemy(app)
# Define models
roles_users = db.Table('roles_users',
db.Column('user_id', db.Integer(), db.ForeignKey('user.id')),
db.Column('role_id', db.Integer(), db.ForeignKey('role.id')))
class Role(db.Model, RoleMixin):
id = db.Column(db.Integer(), primary_key=True)
name = db.Column(db.String(80), unique=True)
description = db.Column(db.String(255))
class User(db.Model, UserMixin):
id = db.Column(db.Integer, primary_key=True)
email = db.Column(db.String(255), unique=True)
password = db.Column(db.String(255))
active = db.Column(db.Boolean())
confirmed_at = db.Column(db.DateTime())
roles = db.relationship('Role', secondary=roles_users,
backref=db.backref('users', lazy='dynamic'))
# Setup Flask-Security
user_datastore = SQLAlchemyUserDatastore(db, User, Role)
security = Security(app, user_datastore)
# Create a user to test with
@app.before_first_request
def create_user():
db.drop_all()
db.create_all()
_lon_role = user_datastore.create_role(name='LON')
_user = user_datastore.create_user(email='fred@example.net', password=hash_password('password'))
user_datastore.add_role_to_user(_user, _lon_role)
db.session.commit()
def roles_accepted_from_route(variable_name):
def decorator(func):
@wraps(func)
def wrapper(*args, **kwargs):
_variable = kwargs.get(variable_name, None)
if _variable:
print(_variable, _variable[:3])
@roles_accepted('admin', _variable[:3])
def inner_wrapper(*args, **kwargs):
print(f'roles_accepted returned True, variable: {_variable[:3]}')
return func(*args, **kwargs)
return inner_wrapper(*args, **kwargs)
return wrapper
return decorator
@app.route('/')
def home():
if current_user.is_authenticated:
return f'You are logged in as : {current_user.email} and have roles: {",".join([r.name for r in current_user.roles])}'
else:
return f'<a href="/login">Login</a>'
@app.route('/<string:_id>')
@roles_accepted_from_route(variable_name='_id')
def some_route(_id):
return f'Route Variable: {_id}'
if __name__ == '__main__':
app.run()