# 進階篇三 實戰開發之權限控制
[TOC]
## 權限控制的作用
>[info] 在項目中,特別是后臺管理的項目中,權限控制非常重要,有些頁面要求不限制訪問,有些頁面要求登陸后才能訪問,還有些重要頁面,必須是指定的角色才能訪問.在MVC項目中,我們可以通過控制每一個控制器的節點的權限達到我們的需求.
## 巧用控制器基類 達到權限控制的目的
首先 咱們先點有一套 驗證登錄 設置登錄 設置退出的方法,
注意這個方法都是放在 base/controller中
### 全局登錄函數組
詳細使用方法參照
http://www.hmoore.net/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

- 序言及更新日志
- 前言一 開發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 圖片處理增強類 支持縮略圖在線顯示
- 附件十三 微信推送消息接口類庫源碼
- 附件十三 微信推送消息接口類庫源碼 之 基類
- 附件十四 存儲微信昵稱的處理方法