如何使用Flask为Jinja2模板提供类变量?

时间:2015-03-25 13:44:47

标签: python flask jinja2

我有一个使用flask-login的模板,我有一个基于bitset的权限模型。在我的模板中,我想基于权限公开页面中的内容。我的用户模型中有一个函数can(permission),用于控制用户可以执行的操作。

class User(UserMixin, db.Model):
    __tablename__ = 'users'
    id = db.Column(db.Integer, primary_key=True)
    email = db.Column(db.String(64), unique=True)
    fullname = db.Column(db.String(75))
    password_hash = db.Column(db.String(128))
    role_id = db.Column(db.Integer, db.ForeignKey('roles.id'))

    @property
    def password(self):
        raise AttributeError('password is not a readable attribute')

    @password.setter
    def password(self, password):
        self.password_hash = generate_password_hash(password)

    def verify_password(self, password):
        return check_password_hash(self.password_hash, password)

    def can(self, permissions):
        return self.role is not None and (self.role.permissions & permissions) == permissions

    def __repr__(self):
        return '<User: {}>'.format(self.email)

class Permission:
    FOLLOW = 0x01
    COMMENT = 0x02
    WRITE_ARTICLES = 0x04
    MODERATE_COMMENTS = 0x08
    ADMINISTER = 0x80

当我尝试在模板中调用can()函数并向其传递权限时,我收到一条错误,即未定义权限,可能是因为Permission类不在模板范围内。以下是我的base.html中所有其他模板扩展的一部分:

{% if current_user.can(Permission.ADMINISTER) %}
<h3>Administration</h3>
<ul>
    <li><a href="{{ url_for('main.add_user') }}">Add User</a></li>
    <li><a href="{{ url_for('main.change_password') }}">Change Password</a></li>
    <li><a href="{{ url_for('main.lock_user') }}">Lock User</a></li>
</ul>
{% endif %}

我正在松散地关注Miguel Grinberg关于Flask Web Development的书,他正在base.html (line 32)中做同样的事情。

如何让Permission类可用于我的模板?

1 个答案:

答案 0 :(得分:7)

最明显,但最不方便的是,在渲染需要它的模板时,只需传递Permission类。

return render_template('page.html', Permission=Permission)

通过告诉Flask 自动将其添加到模板上下文,更进一步。在设置应用程序时,在某处装饰上下文处理器。

@app.context_processor
def include_permission_class():
    return {'Permission': Permission}

由于性能和清晰度原因,上下文仅适用于呈现的即时模板。如果您需要{% import %}{% include %},则可以将with context添加到需要它的每个块:{% include 'other_template.html' with context%}。或者,您可以将该类添加到Jinja env的全局命名空间。

app.add_template_global(Permission, 'Permission')