为什么需要restful形式的权限管理
最近在写flask应用时使用了 restful 形式的flask.views.MethodView
,但是在对其进行权限管理时遇到了一些问题
flask文档上介绍说用
decorators = []
添加装饰器,但实际使用上,比如
- get 和 post 采用不同的权限
get 不使用 login_required
post 需要 login_required
这样就不能使用 decorators 对视图进行装饰
-
post ,delete, put 都需要 login_required,但是get不需要 而 delete 又需要更高级别的权限,我们可以这样
class AAA(MethodView): def get(self,uid): ... @login_required def post(self): ... @login_required def put(self,uid): ... @login_required @more_required def delete(self,uid): ...
是不是看起来还不错, 但是,如果再加上类似EditBlogPostPermission 这样的权限管理呢? 是不是还需要这样
@login_required def put(self,uid): permission = EditBlogPostPermission(uid) if permission.can(): # Save the edits ... return render_template('edit_post.html') ...
先不论样式丑不丑,最重要的代码的 可维护性 极差,所以我增加了如下代码
怎么实现restful形式的权限管理
同样采用装饰器实现,调用 BasePermission 时会自动调用 call 函数
class BasePermission(object):
def __call__(self, func):
@wraps(func)
def decorator(*args, **kwargs):
meth = getattr(self, request.method.lower(), None)
if meth is None and request.method == 'HEAD':
meth = getattr(self, 'get', None)
assert meth is not None, 'Unimplemented method %r' % request.method
check = meth(*args, **kwargs)
if check:
return check
else:
pass
return func(*args, **kwargs)
return decorator
举个例子,get方式不需要用户登陆,而其它方式需要,并且put方式需要创建主题的作者才能更改
class TopicPermission(BasePermission):
@login_required
def post(self):
pass
def get(self, uid):
pass
@login_required
def put(self, uid):
permission = EditTopicPermission(uid)
if not permission.can():
flash('你没有权限')
return redirect(url_for('topic.topic', uid=uid))
@login_required
def delete(self,uid):
pass
topic_permission = TopicPermission()
假设四种方式都需要同一种权限,都需要用户登陆,总不能每个函数前都加上装饰器吧 所以稍加修改
decorators = ()
def __call__(self, func):
if self.decorators:
for dec in self.decorators:
return dec(func)
OK,这样就可以添加
decorators = [login_required]
来实现四种请求方式采用同一种权限
最后,你就可以在 AAA 这个类里添加
decorators = [topic_permission]
实现restful形式的权限管理
ok,就这样,可能还不完善或有一些问题,如有问题请联系我