前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >php基于RBAC(角色的访问控制)的设计

php基于RBAC(角色的访问控制)的设计

作者头像
友儿
发布2022-09-11 16:16:36
5550
发布2022-09-11 16:16:36
举报
文章被收录于专栏:友儿

权限系统模块对于互联网产品是一个非常重要的功能,可以控制不同的角色合理的访问不同的资源从而达到安全访问的作用

权限控制有哪些模型

1 ACL 2 RBAC 基于角色的访问控制

我们可以看出,ACL是用户和权限直接关系的,而RBAC则是通过角色间接关联用户和权限的。所以我们注意到角色是RBAC系统的一个重要属性。

什么是RBAC模型

RBAC(Role-Based Access Control,基于角色的访问控制),就是用户通过角色与权限进行关联。简单地说,一个用户拥有若干角色,每一个角色拥有若干权限。这样,就构造成“用户-角色-权限”的授权模型。在这种模型中,用户与角色之间,角色与权限之间,一般者是多对多的关系。

为什么要选择RBAC模型

原因如下:

  • 方便用户分组
  • 方便权限分配和回收
  • 扩展方便,可以满足大部分业务需求

这些也就是我们在说权限管理前,应该先知道权限管理要有功能。

图中有重要的RBAC模型5大属性,分别是: 1 用户属性(张三、李四、王五) 2 角色属性(销售经理、销售、前台) 3 用户与角色的关系(张三 是 销售经理 、李四 王五 是 销售) 4 权限(添加客户、编辑客户、删除客户,查看客户) 5 权限与角色的关系(销售 拥有 查看客户的 权 限、销售经理可以 查看/添加/删除/编辑客户的)

**一个RBAC权限模块,必然要实现三个功能

1 用户管理** 用户列表 添加用户 编辑用户 设置用户角色

2 角色管理 角色列表 添加角色 编辑角色 设置角色权限

3 权限管理 权限列表 新增权限 编辑权限

数据表设计

用户表

代码语言:javascript
复制
CREATE TABLE `user` (
  `id` int(11) unsigned NOT NULL AUTO_INCREMENT, 
  `name` varchar(20) NOT NULL DEFAULT '' COMMENT '姓名',
  `email` varchar(30) NOT NULL DEFAULT '' COMMENT '邮箱',
  `is_admin` tinyint(1) NOT NULL DEFAULT '0' COMMENT '是否是超级管理员 1表示是 0 表示不是',
  `status` tinyint(1) NOT NULL DEFAULT '1' COMMENT '状态 1:有效 0:无效',
  `updated_time` timestamp NOT NULL DEFAULT '0000-00-00 00:00:00' COMMENT '最后一次更新时间',
  `created_time` timestamp NOT NULL DEFAULT '0000-00-00 00:00:00' COMMENT '插入时间',
  PRIMARY KEY (`id`),
  KEY `idx_email` (`email`)
) ENGINE=InnoDB  DEFAULT CHARSET=utf8 COMMENT='用户表';

角色表

代码语言:javascript
复制
CREATE TABLE `role` (
  `id` int(11) unsigned NOT NULL AUTO_INCREMENT,
  `name` varchar(50) NOT NULL DEFAULT '' COMMENT '角色名称',
  `status` tinyint(1) NOT NULL DEFAULT '1' COMMENT '状态 1:有效 0:无效',
  `updated_time` timestamp NOT NULL DEFAULT '0000-00-00 00:00:00' COMMENT '最后一次更新时间',
  `created_time` timestamp NOT NULL DEFAULT '0000-00-00 00:00:00' COMMENT '插入时间',
  PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='角色表';

用户角色表

代码语言:javascript
复制
CREATE TABLE `user_role` (
  `id` int(11) unsigned NOT NULL AUTO_INCREMENT,
  `uid` int(11) NOT NULL DEFAULT '0' COMMENT '用户id',
  `role_id` int(11) NOT NULL DEFAULT '0' COMMENT '角色ID',
  `created_time` timestamp NOT NULL DEFAULT '0000-00-00 00:00:00' COMMENT '插入时间',
  PRIMARY KEY (`id`),
  KEY `idx_uid` (`uid`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='用户角色表';

权限详情表

代码语言:javascript
复制
CREATE TABLE `access` (
  `id` int(11) unsigned NOT NULL AUTO_INCREMENT,
  `title` varchar(50) NOT NULL DEFAULT '' COMMENT '权限名称',
  `urls` varchar(1000) NOT NULL DEFAULT '' COMMENT 'json 数组',
  `status` tinyint(1) NOT NULL DEFAULT '1' COMMENT '状态 1:有效 0:无效',
  `updated_time` timestamp NOT NULL DEFAULT '0000-00-00 00:00:00' COMMENT '最后一次更新时间',
  `created_time` timestamp NOT NULL DEFAULT '0000-00-00 00:00:00' COMMENT '插入时间',
  PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='权限详情表';

角色权限表

代码语言:javascript
复制
CREATE TABLE `role_access` (
  `id` int(11) unsigned NOT NULL AUTO_INCREMENT,
  `role_id` int(11) NOT NULL DEFAULT '0' COMMENT '角色id',
  `access_id` int(11) NOT NULL DEFAULT '0' COMMENT '权限id',
  `created_time` timestamp NOT NULL DEFAULT '0000-00-00 00:00:00' COMMENT '插入时间',
  PRIMARY KEY (`id`),
  KEY `idx_role_id` (`role_id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='角色权限表';

用户操作记录表

代码语言:javascript
复制
CREATE TABLE `app_access_log` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `uid` bigint(20) NOT NULL DEFAULT '0' COMMENT '品牌UID',
  `target_url` varchar(255) NOT NULL DEFAULT '' COMMENT '访问的url',
  `query_params` longtext NOT NULL COMMENT 'get和post参数',
  `ua` varchar(255) NOT NULL DEFAULT '' COMMENT '访问ua',
  `ip` varchar(32) NOT NULL DEFAULT '' COMMENT '访问ip',
  `note` varchar(1000) NOT NULL DEFAULT '' COMMENT 'json格式备注字段',
  `created_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP,
  PRIMARY KEY (`id`),
  KEY `idx_uid` (`uid`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='用户操作记录表';

代码实现

本系统所有页面都是需要登录之后才能访问的, 在框架中加入统一验证方法

代码语言:javascript
复制
public function beforeAction($action) {
    $login_status = $this->checkLoginStatus();
    if ( !$login_status && !in_array( $action->uniqueId,$this->allowAllAction )  ) {
        if(Yii::$app->request->isAjax){
            $this->renderJSON([],"未登录,请返回用户中心",-302);
        }else{
            $this->redirect( UrlService::buildUrl("/user/login") );//返回到登录页面
        }
        return false;
    }
    //保存所有的访问到数据库当中
    $get_params = $this->get( null );
    $post_params = $this->post( null );
    $model_log = new AppAccessLog();
    $model_log->uid = $this->current_user?$this->current_user['id']:0;
    $model_log->target_url = isset( $_SERVER['REQUEST_URI'] )?$_SERVER['REQUEST_URI']:'';
    $model_log->query_params = json_encode( array_merge( $post_params,$get_params ) );
    $model_log->ua = isset( $_SERVER['HTTP_USER_AGENT'] )?$_SERVER['HTTP_USER_AGENT']:'';
    $model_log->ip = isset( $_SERVER['REMOTE_ADDR'] )?$_SERVER['REMOTE_ADDR']:'';
    $model_log->created_time = date("Y-m-d H:i:s");
    $model_log->save( 0 );
    /**
     * 判断权限的逻辑是
     * 取出当前登录用户的所属角色,
     * 在通过角色 取出 所属 权限关系
     * 在权限表中取出所有的权限链接
     * 判断当前访问的链接 是否在 所拥有的权限列表中
     */
    //判断当前访问的链接 是否在 所拥有的权限列表中
    if( !$this->checkPrivilege( $action->getUniqueId() ) ){
        $this->redirect( UrlService::buildUrl( "/error/forbidden" ) );
        return false;
    }
    return true;
}

检查是否有访问指定链接的权限

代码语言:javascript
复制
public function checkPrivilege( $url ){
    //如果是超级管理员 也不需要权限判断
    if( $this->current_user && $this->current_user['is_admin'] ){
        return true;
    }

    //有一些页面是不需要进行权限判断的
    if( in_array( $url,$this->ignore_url ) ){
        return true;
    }

    return in_array( $url, $this->getRolePrivilege( ) );
}

获取某用户的所有权限,取出指定用户的所属角色, 在通过角色取出所属权限关系,在权限表中取出所有的权限链接

代码语言:javascript
复制
public function getRolePrivilege($uid = 0){
    if( !$uid && $this->current_user ){
        $uid = $this->current_user->id;
    }

    if( !$this->privilege_urls ){
        $role_ids = UserRole::find()->where([ 'uid' => $uid ])->select('role_id')->asArray()->column();
        if( $role_ids ){
            //在通过角色 取出 所属 权限关系
            $access_ids = RoleAccess::find()->where([ 'role_id' =>  $role_ids ])->select('access_id')->asArray()->column();
            //在权限表中取出所有的权限链接
            $list = Access::find()->where([ 'id' => $access_ids ])->all();
            if( $list ){
                foreach( $list as $_item  ){
                    $tmp_urls = @json_decode(  $_item['urls'],true );
                    $this->privilege_urls = array_merge( $this->privilege_urls,$tmp_urls );
                }
            }
        }
    }
    return $this->privilege_urls ;
}

更详细的代码,我已经给大家整理好了。有需要的可以查看以下链接。

代码语言:javascript
复制
链接: https://pan.baidu.com/s/1Jr852vqhf8eRkVxilCnsOQ   
提取码: ct1e 
本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 权限控制有哪些模型
  • 什么是RBAC模型
  • 为什么要选择RBAC模型
  • 数据表设计
相关产品与服务
数据库
云数据库为企业提供了完善的关系型数据库、非关系型数据库、分析型数据库和数据库生态工具。您可以通过产品选择和组合搭建,轻松实现高可靠、高可用性、高性能等数据库需求。云数据库服务也可大幅减少您的运维工作量,更专注于业务发展,让企业一站式享受数据上云及分布式架构的技术红利!
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档