# 进阶篇三 实战开发之权限控制
[TOC]
## 权限控制的作用
>[info] 在项目中,特别是后台管理的项目中,权限控制非常重要,有些页面要求不限制访问,有些页面要求登陆后才能访问,还有些重要页面,必须是指定的角色才能访问.在MVC项目中,我们可以通过控制每一个控制器的节点的权限达到我们的需求.
## 巧用控制器基类 达到权限控制的目的
首先 咱们先点有一套 验证登录 设置登录 设置退出的方法,
注意这个方法都是放在 base/controller中
### 全局登录函数组
详细使用方法参照
https://www.kancloud.cn/mikkle/thinkphp5_study/349347
* 检查全局登录
~~~
/**
* 检测是否登录
* Power by Mikkle
* QQ:776329498
* @return bool
*/
public function checkLoginGlobal()
{
$check_success = false;
switch ($this->loginType) {
case 1;
case "session";
$this->uuid = Session::get('uuid', 'Global');
$this->member_info = Session::get('member_info', 'Global');
if ($this->uuid && $this->member_info) {
$check_success = true;
}
break;
case 2;
case "cache";
$session_id_check = Cookie::get("session_id");
$this->uuid = Cache::get("uuid_{$session_id_check}");
$this->member_info = Cache::get("member_info_{$session_id_check}");
if ($this->uuid && $this->member_info) {
$check_success = true;
}
//刷新 缓存有效期
Cache::set("uuid_{$session_id_check}", $this->uuid);
Cache::set("member_info_{$session_id_check}", $this->member_info);
break;
case 3:
case "redis":
//这部分不方便共享 不好意思
break;
}
return $check_success;
}
~~~
* 设置全局登录
~~~
/**
* 设置全局登录
* #User: Mikkle
* #Email:776329498@qq.com
* #Date:
*/
public function setLoginGlobal($member_info = [], $login_code = 0)
{
$set_success = false ;
if ($member_info) {
switch ($this->loginType) {
case 1:
case "session":
Session::set('member_info', $member_info, 'Global');
Session::set('uuid', $member_info['uuid'], 'Global');
if ((Session::has("uuid", "Global"))) {
$set_success = true;
}
break;
case 2:
case "cache":
$session_id = $this->create_uuid("SN");
Cookie::set("session_id", $session_id);
Cache::set("member_info_$session_id", $member_info);
Cache::set("uuid_$session_id", $member_info['uuid']);
$session_id_check = Cookie::get("session_id");
if ((Cache::get("uuid_{$session_id_check}"))) {
$set_success = true;
}
break;
case 3:case "redis":
//这部分不方便共享 不好意思
break;
}
}
if (!$set_success) return false;
//保存登录记录
$this->saveLoginInfo($member_info['uuid'],$login_code);
return true;
}
~~~
* 设置全局退出
~~~
/**
* 全局退出
* Power by Mikkle
* QQ:776329498
* @return bool
*/
protected function logoutGlobal(){
switch ($this->loginType) {
case 1:
case "session":
Session::delete('uuid', 'Global');
Session::delete('member_info', 'Global');
break;
case 2:
case "cache":
$session_id_check = Cookie::get("session_id");
Cache::rm("uuid_{$session_id_check}");
Cache::rm("member_info_{$session_id_check}");
Cookie::delete("session_id");
break;
case 3:case "redis":
//这部分不方便共享 不好意思
break;
}
$this->member_info = null;
$this->uuid = null;
return true;
}
~~~
### 在模块控制器基类的构造函数检验权限
比如说现在的wechat 模块
~~~
<?php
/**
* Created by PhpStorm.
* Power by Mikkle
* QQ:776329498
* Date: 2017/4/17
* Time: 9:33
*/
namespace app\wechat\controller;
class Base extends \app\base\controller\Base
{
protected $isLogin = false; //判断是否登陆
protected $uuid; //登陆后的UUID
protected $config_list=[];
/**
* 检查登陆信息
* Power by Mikkle
* QQ:776329498
*/
public function _initialize()
{
if ($this->checkLoginGlobal()) {
$this->isLogin = true;
}
}
}
~~~
这样继承这个基类的方法 都可以通过 $this->uuid 是否存在 来判断是否登录
~~~
public function test(){
//检验用户是否登录
if (!$this->uuid){
return self::showReturnCodeWithOutData(1004);
}
//使用用户信息
dump($this->member_info);
}
~~~
### 重要节点的权限判断
接下来 我们再建立一个Auth权限基类
看源码
~~~
<?php
/**
* Created by PhpStorm.
* Power by Mikkle
* QQ:776329498
* Date: 2017/4/17
* Time: 14:47
*/
namespace app\wechat\controller;
use app\base\model\PersonnelNode;
use app\base\model\PersonnelRoleNodeAccess;
use think\Session;
abstract class Auth extends Base
{
//權限跳过检验的节点
protected $index_array=[
'Index'=>[
'index'=>true,
'login'=>true,
'getmenujson'=>true,
],
];
protected $log_string;
protected $member_info;
public function _initialize()
{
parent::_initialize(); // TODO: Change the autogenerated stub
//检测是否登陆
if(!$this->isLogin) {
$this->redirect('index/login');
}
//检测登陆权限
$auth =$this->checkNodeAuth();
//dump($auth);
if($auth!=true || $auth==false || is_string($auth)){
// dump($this->log_string);
$this->error($this->log_string,"index/index");
}
$this->member_info = Session::get('member_info','Global');
}
/**
* 获取当前访问节点
* Power: Mikkle
* Email:776329498@qq.com
* @return bool
*/
protected function checkNodeAuth()
{
if ($this->checkIsAdmin()) {
return true;
} else {
$request = $this->request;
if ($this->checkIsIndex($request->controller(), $request->action())) {
return true;
}
//检测权限// 当前模块名
$node = new PersonnelNode();
//跳过登录系列的检测以及主页权限
$node_info = $node->getNodeInfo($request->module(), $request->controller(), $request->action());
if (empty($node_info)) {
$this->log_string='此页面访问权限未开放,请联系管理员';
return false;
}
if ($node_info['auth_grade'] > 0) {
return $this->checkUserNodeAuthByNodeGuId($node_info['guid']);
}
return true;
}
}
/**
* 检测节点是否可以默认登录
* Power: Mikkle
* Email:776329498@qq.com
* @param $controller
* @param $action
* @return bool
*/
protected function checkIsIndex($controller,$action)
{
return isset($this->index_array[$controller][$action])? true : false ;
}
/**
* 判断用户是否有节点权限
* Power: Mikkle
* Email:776329498@qq.com
* @param $Guid
* @return bool
*/
protected function checkUserNodeAuthByNodeGuId($Guid)
{
$member = $this->member_info;
$node_list =[];
if (Session::has("role_node_list_{$Guid}")){
$node_list=Session::has("role_node_list_{$Guid}");
}else{
$model = new PersonnelRoleNodeAccess();
$node_list = $model->getRoleMenuList($member['role_id'],1);
}
if (!in_array($Guid, $node_list)) {
$this->log_string="你没有权限,请联系系统管理员";
return false;
}else{
return true;
}
}
/**
* 检测是否是管理员
* Power: Mikkle
* Email:776329498@qq.com
* @return bool
*/
protected function checkIsAdmin()
{
if (Session::has('is_admin')) {
return true;
} else {
return false;
}
}
}
~~~
OK 剩余的就是把授权节点的信息写入权限表[PersonnelRoleNodeAccess]中即可!
### 控制器继承Auth类 即可验证权限
剩余要做的 就是需要验证登录的控制器继承Auth
~~~
<?php
/**
* Created by PhpStorm.
* User: Administrator
* Date: 2017/5/23
* Time: 13:02
*/
namespace app\wechat\controller;
class Project extends Auth
{
}
~~~
感谢大家关注 交流请加QQ群 321449759
![](https://box.kancloud.cn/3499008a08e64306c68873288092a057_286x340.png)
- 序言及更新日志
- 前言一 开发PHP必备的环境(你可以不看)
- LinUX系统ThinkPHP5链接MsSQL数据库的pdo_dblib扩展
- centos7.2挂载硬盘攻略
- Centos系统Redis安装及Redis的PHP扩展安装
- Centos系统增加Swap(系统交换区)的方法
- 前言二 开发PHP软件配置和介绍(你依然可以不看)
- 数据库SQL文件
- 本地Git(版本控制)的搭建
- GIT远程仓库的克隆和推送
- Git常用命令
- PHP面向对象思想实战经验领悟
- PHP面向对象实战----命名空间
- PHP面向对象实战----继承
- 基类实战--底层方法封装
- 基类实战--构造函数实战
- 基类实战--析构函数的使用
- TP5实战开发前篇---控制器(controller)
- 控制器中Request类的使用
- 控制器中基类的使用
- TP5实战开发前篇---模型篇(model)
- TP5实战开发前篇---验证器篇(Validate)
- TP5实战课程入门篇---花拳绣腿
- 模块以及类的文件的建立
- Api开发------单条信息显示
- Api开发---单条信息复杂关联显示
- Api开发---查询信息缓存Cache的应用
- TP5实战技巧---开发思路 引路造桥
- TP5实战技巧---整合基类 化繁为简
- TP5实战课程入门篇---数据操作
- Api开发---数据的添加和修改
- API开发---快速开发API通用接口
- TP5专用微信sdk使用教程
- THINKPHP5微信SDK更新记录及升级指导
- TP5专用SDK 微信参数配置方法
- 微信公众号推送接口对接教程
- 微信推送接口对接示例含扫描登录微信端部分
- TP5专用微信支付SDK使用简介
- TP5专用支付宝支付SDK使用说明
- 使用NW将开发的网站打包成桌面应用
- TP5高阶实战课程 进阶篇概述
- 进阶篇一 实战开发之习惯及要求
- 进阶篇二 实战开发之控制器
- 控制器基类之控制器基类使用方法
- 控制器基类之控制器基类常用方法分享
- 控制器基类之构造函数的使用方法
- 进阶篇三 实战开发之权限控制
- TP5实战源码 --- 全局用户信息验证类Auth
- TP5实战源码 --- 微信Auth实战开发源码
- 进阶篇四 实战开发之模型
- 模型基类之模型基类的用途
- 模型基类之常用数据处理方法
- 模型逻辑层之实战代码(含事务)
- 模型实战开发之模型常用方法
- 模型实战源码 --- 乐观锁的应用
- 模型实战技巧---Model事件功能的使用
- 模型事件实战应用---数据库操作日志
- 进阶篇五 实战开发之缓存(Cache)
- TP5实战源码---应用缓存获取城市信息
- TP5实战源码---应用缓存获取分类详情
- 进阶篇六 TP5类库的封装和使用
- DataEdit快捷操作类库
- ShowCode快捷使用类库
- 阿里大于 短信API接口 TP5专用类库
- DatabaseUpgrade数据库对比及更新类库
- AuthWeb权限类使用说明
- 进阶篇七 服务层的应用
- 服务层源码示例
- 服务层基类源码
- 进阶篇八 应用层Redis数据处理基类
- Redis服务层基类源码
- 进阶篇九 使用Redis类库处理一般的抢购(秒杀)活动示例
- 进阶篇十 某大型项目应用本Redis类源码示例(含事务 乐观锁)
- 进阶篇十一 逻辑层的应用
- 逻辑层基类源码
- 进阶篇 服务层代码示例
- 高阶实战课程 进阶篇持续新增中
- 高阶篇一 TP5命令行之守护任务源码
- TP5实战源码 --- 命令行
- TP5实战源码 --- 通过shell建立PHP守护程序
- 高阶篇二 使用Redis队列发送微信模版消息
- 高阶篇二 之 Worker队列基类源码
- 高阶篇三 TP5实战之Redis缓存应用
- Redis实战源码之Hash专用类库源码
- Redis实战源码之Model类结合
- Redis实战源码之模型Hash基类源码
- Redis实战源码之Hash查询使用技巧
- Redis实战源码之 shell脚本中redis赋值和取值
- 高阶篇四 Swoole的实战应用
- swoole基类代码
- Swoole扩展WebsocketServer专用类
- 基于Swoole的多Room聊天室的程序
- Swoole守护服务shell源码
- 高阶篇五 命令行异步多进程队列类的应用
- tp_worker类源码
- WorkerBase
- WorkerCommand
- WorkerRedis
- Redis类
- CycleWorkBase
- WorkerHookBase异步钩子
- 队列日志SQL
- 高阶篇六 定时执行队列类库以及使用方法
- 定时队列类库源码
- 高阶篇七 异步执行循环队列类库以及使用教程
- CycleWorkBase源码
- 高阶实战课程 进阶篇持续新增中
- Extend便捷类库源码库
- 阿里相关类库
- SendSms--验证码API接口文件
- 权限相关类库目录
- AuthWeb 权限验证类库
- Redis便捷操作类库(20171224更新)
- Redis
- Tools工具类库集
- Curl类库
- DataEdit
- Rand类库
- ShowCode类库
- Upload类库
- 附件集合
- 附件一:微信支付 实战开发源码
- 微信支付类库源代码
- Common_util_pub.php
- DownloadBill_pub.php
- JsApi_pub.php
- NativeCall_pub.php
- NativeLink_pub.php
- OrderQuery_pub.php
- Refund_pub.php
- RefundQuery_pub.php
- SDKRuntimeException.php
- ShortUrl_pub.php
- UnifiedOrder_pub.php
- Wxpay_client_pub.php
- Wxpay_server_pub.php
- WxPayConf_pub.php
- 微信支付回调页面源码
- 附件二 顺丰快递BSP接口实战开发源码
- 顺丰快递BSP接口实战开发源码
- 顺丰BSP基类
- 顺丰BSP基础代码
- 顺丰BSP下单接口
- 顺丰BSP查单接口
- 顺丰BSP确认/取消接口
- 附件三 APP注册登陆接口源码(含融云平台接口)
- 附件四 TP5订单Model(含事务 获取器 修改器等方法)
- 附录五 RSA加密解密
- Rsa文件源码
- 附件六 阿里大于短信接口
- 附件七 AES加解密类
- AES加解密类源码
- 附件八 TP5路由设置源码
- 附件九 TP5 Excel导入导出下载便捷类库
- Excel类库TP5源码
- 附件十 TP5便捷操作Redis类库源码
- TP5源码 Redis操作便捷类库
- 附件十一 TP5源码 上传文件入库类源码
- 上传类Upload源码
- Upload类上传配置文件
- 存储图像文件的数据库SQL文件
- 存储文件的数据库SQL文件
- 附件十二 TP5 图片处理增强类 支持缩略图在线显示
- 附件十三 微信推送消息接口类库源码
- 附件十三 微信推送消息接口类库源码 之 基类
- 附件十四 存储微信昵称的处理方法