7

我对烧瓶很陌生,我正在尝试使用装饰器的强大功能:p 我读了很多东西,在这里找到了很多关于 python 装饰器的主题,但没有什么真正有用的。

@app.route('groups/<id_group>')
@group_required(id_group)
@login_required
def groups_groupIndex(id_group):
    #do some stuff
    return render_template('index_group.html')

这是我得到的错误:

@group_required(id_group), NameError: name 'id_group' is not defined

好的,id_group 尚未定义,但我不明白为什么我可以在函数 groups_groupIndex 中使用来自 URL 的 id_group 参数,但不能在装饰器中使用!

我尝试移动/切换装饰器,但每次都会发生相同的错误。

这是我的装饰器,但它似乎工作正常

def group_required(group_id):
    def decorated(func):
        @wraps(func)
        def inner (*args, **kwargs):
            #Core_usergroup : table to match users and groups
            groups = Core_usergroup.query.filter_by(user_id = g.user.id).all()
            for group in groups:
                #if the current user is in the group : return func
                if int(group.group_id) == int(group_id) :
                    return func(*args, **kwargs)
            flash(gettext('You have no right on this group'))
            return render_template('access_denied.html')     
        return inner
    return decorated

也许我看不到我应该看到的装饰器......我可以这样使用我的装饰器还是需要我重写一些不同的东西?

4

1 回答 1

14

您定义group_id为函数参数;这使它成为该函数中的本地名称。

这不会使该名称可用于其他范围;装饰器所在的全局命名空间看不到该名称。

但是,包装函数可以。调用时将从@apps.route()包装器传递该参数:

def group_required(func):
    @wraps(func)
    def wrapper(group_id, *args, **kwargs):
        #Core_usergroup : table to match users and groups
        groups = Core_usergroup.query.filter_by(user_id = g.user.id).all()
        for group in groups:
            #if the current user is in the group : return func
            if int(group.group_id) == int(group_id) :
                return func(*args, **kwargs)
        flash(gettext('You have no right on this group'))
        return render_template('access_denied.html')     
    return wrapper

请注意,此装饰器不会将group_id参数传递给装饰函数;使用return func(group_id, *args, **kwargs)而不是您仍然需要在视图函数中访问该值。

于 2013-08-19T09:32:58.720 回答