Flask Restful API权限管理设计与实现

在使用flask设计restful api的时候,有一个很重要的问题就是如何进行权限管理,以及如何进行角色的定义,在网上找了一下没有发现有类似的资料,虽然有些针对网站进行的权限管理设计,但是跟restful api接口的权限管理还是有很多不同的,于是乎自己动手,丰衣足食。为方便后来者,特撰此文!

权限的设计

从本质上思考,我需要为每个API接口设定相应的权限,所以针对API的权限列表跟普通网站的权限设计是不同的,普通网站的权限设计是针对某个功能,比如是否可以comment功能,通常的权限定义如下:

class Permission:
    """
    权限表
    """
    COMMENT = 0x01  # 评论
    MODERATE_COMMENT = 0x02  # 移除评论

但是针对restful api,我们更希望权限是针对我们的api接口,而restful api接口是跟我们路由的endpoint以及http method相关的,所以我们的权限设计应该是类似如下示例中的样子:

# 这里comments是路由的endpoint,接口在判断用户是否有权限的时候
# 可以先获取到endpoint和http method,然后就可以查看其是否有权限
comment_permission = {"comments": {"post": True, "get": True, "delete": False}}

角色的设计

通常,我们在做网站的角色设计时会将角色存储在数据库当中,并会通过或运算(|)赋予角色以特定权限,如下:

class Role(db.Model):
    """
    用户角色
    """
    id = db.Column(db.Integer, primary_key=True)
    # 该用户角色名称
    name = db.Column(db.String(164))
    # 该用户角色是否为默认
    default = db.Column(db.Boolean, default=False, index=True)
    # 该用户角色对应的权限
    permissions = db.Column(db.Integer)
    # 该用户角色和用户的关系
    # 角色为该用户角色的所有用户
    users = db.relationship('User', backref='role', lazy='dynamic')

    @staticmethod
    def insert_roles():
        """
        创建用户角色
        """
        roles = {
            # 定义了两个用户角色(User, Admin)
            'User': (Permission.COMMENT, True),
            'Admin': (Permission.COMMENT |
                      Permission.MODERATE_COMMENT, False)
        }
        for r in roles:
            role = Role.query.filter_by(name=r).first()
            if role is None:
                # 如果用户角色没有创建: 创建用户角色
                role = Role(name=r)
            role.permissions = roles[r][0]
            role.default = roles[r][1]
            db.session.add(role)
            db.session.commit()

这里其实我一直没有搞明白,为什么要将角色存储于数据库当中,在我看来这只会导致更多的I/O操作从而影响系统的性能,因此我在设计角色的时候根本没有考虑存储到数据库中,角色的数据结构在系统运行时,直接存在内存当中,这样在接口调用时,可以直接使用角色相关的数据结构。而且由于我们的权限设计也不太相同,所以我针对restful api设计的Role如下:

USER = 1
ADMIN = 2
VISITOR = 3

Role = {
    USER: {
        "comment": {"post": True, "patch": True, "get": True, "delete": True},
        "share": {"post": True}
    },
    ADMIN: {
        "comment": {"post": True, "patch": True, "get": True, "delete": True},
        "share": {"post": True}
    },
    VISITOR: {
        "comment": {"get": True},
        "share": {"post": True}
    }
}

用户可以被赋予特定的role,如下:

userA = {"name": "John", "role": USER}

那么接口如何判断用户是否有权限访问呢?
首先用户访问接口时都会带有用户信息,restful api一般是通过token来表明身份,系统通过token来获取用户的信息,比如用户名,然后我们可以通过用户名来获取用户的角色role,假设我们访问的接口是comments endpoint的post接口,那么就可以如下判断:

def access_control(user):
    """判断用户是否有访问权限,有就返回True,没有返回False"""
    
    # 首先要获取到API的endpoint和http method,此处代码省略
    ...
    
    role = user.get('role', VISITOR)
    try:
        if not Role[role][endpoint][http_method]:
            return False
        return True
    except KeyError:
        return False

由于基本所有的接口都需要access control,那么我们把上边的代码稍作改变,让它成为一个decorator,同时,user信息也可以直接获取而不需要从参数传递,如下:

from functools import wraps

def get_role():
    # 这里get_resource_by_name用于从数据库中获取该用户的信息,这个需要自己去定义
    # 另外我们可以在登录验证的时候或者token验证的时候讲user name存储于全局变量g中,这样我们可以随时获取该用户名
    user = UserModel.get_resource_by_name(g.user_name)
    return user.get("role", VISITOR)

def access_control(func):
    @wraps(func)
    def wrap_func(*args, **kwargs):
        # 同样要先获取到API的endpoint和http method,此处代码省略
        ...
        
        try:
            if not Role[role][endpoint][http_method]:
                return make_response(
                    jsonify({'error': 'no permission'}), 403)
            return func(*args, **kwargs)
        except KeyError:
            return make_response(
                jsonify({'error': 'no permission'}), 403)
    return wrap_func

以下是一个获取图片resource的使用示例

from flask_restful import Resource

class ImageResource(Resource):
    def __init__(self):
        super(ImageResource, self).__init__()

    @token_auth.login_required
    @access_control
    def get(self, resource_id):
        response = resource_get(resource_id)
        return response

这里另外一个decortor @token_auth.login_required用于token验证,大家可以先不用理会。到这里我们已经可以针对每个接口自动判断该用户是否有权限访问了,而所有权限的变化,都可以通过修改Role中的权限来进行更改,而不需要更改原来的代码,很爽吧,有木有?
不过,笔者在项目中还遇到了另外一个问题,有时候针对一个接口所有的user都应该有权限,但是针对特定的resource,只能resource owner可以操作,举个栗子,比如我们要删除某个评论,但是只允许发布评论的人才有权限删除,也就是comment resource的owner才可以使用delete接口删除,但是我们所有的用户在Role定义的时候delete接口都是True,这个怎么办呢?
这就需要我们在access_control检测完了之后再进一步检测该用户是否是resource owner,所以我们就需要进一步检测,这里添加一个decorator如下:

def get_resource_owner():
    """获取resource的owner"""
    # 自定义,代码省略
    ...

def owner_permission_required(func):
    @wrap(func)
    def wrap_func(*args, **kwargs):
        if g.user_name == get_resource_owner():
            return func(*args, **kwargs)
        return make_response(
            jsonify({'error': 'no permission'}), 403)
    return wrap_func

使用如下:

from flask_restful import Resource

class CommentResource(Resource):
    def __init__(self):
        super(CommentResource, self).__init__()

    @token_auth.login_required
    @access_control
    @owner_permission_required
    @marshal_with(image_fields)
    def delete(self, resource_id):
        response = resource_delete(resource_id)
        return response

注意:decorator的顺序是不能改变的。

至此,Restful API权限管理相关的设计就完成了,如果文章给你带来了启发,记得点赞哦!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 200,302评论 5 470
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 84,232评论 2 377
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 147,337评论 0 332
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,977评论 1 272
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,920评论 5 360
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,194评论 1 277
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,638评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,319评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,455评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,379评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,426评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,106评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,696评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,786评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,996评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,467评论 2 346
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,043评论 2 341

推荐阅读更多精彩内容