本文是基於RESTful描述的,須要你對這個有初步的瞭解。 RESTful是什麼? Representational State Transfer,簡稱REST,是Roy Fielding博士在2000年他的博士論文中提出來的一種軟件架構風格。 REST比較重要的點是資源和狀態轉換, 所謂"資源",就是網絡上的一個實體,或者說是網絡上的一個具體信息。它能夠是一段文本、一張圖片、一首歌曲、一種服務,總之就是一個具體的實在。 而 「狀態轉換」,則是把對應的HTTP協議裏面,四個表示操做方式的動詞分別對應四種基本操做:javascript
清楚了資源的概念,而後再來對資源進行一下分類,我把資源分爲下面三類:php
「私人資源」:是屬於某一個用戶全部的資源,只有用戶本人才能操做,其餘用戶不能操做。例如用戶的我的信息、訂單、收貨地址等等。 「角色資源」:與私人資源不一樣,角色資源範疇更大,一個角色能夠對應多我的,也就是一羣人。若是給某角色分配了權限,那麼只有身爲該角色的用戶才能擁有這些權限。例如系統資源只可以管理員操做,通常用戶不能操做。 「公共資源」:全部人不管角色都可以訪問並操做的資源。css
而對資源的操做,無非就是分爲四種:html
角色和用戶的概念,自不用多說,你們都懂,可是權限的概念須要提一提。
「權限」,就是資源與操做的一套組合,例如"增長用戶"是一種權限,"刪除用戶"是一種權限,因此對於一種資源所對應的權限有且只有四種。java
須要注意兩種特別狀況:git
角色、用戶、權限的模型應該怎麼樣設計,才能知足它們之間的關係?程序員
對上圖的一些關鍵字段進行說明:github
在sails下稱爲策略(Policy),在java SSH下稱爲過濾器(Filter),不管名稱如何,他們工做原理是大同小異的,主要是在一條HTTP請求訪問一個Controller下的action以前進行檢測。因此在這一層,咱們能夠自定義一些策略/過濾器來實現權限控制。
爲行文方便,下面姑且容許我使用策略這一詞。web
** 策略 (Policy) **thinkphp
下面排版順序對應Policy的運行順序
若是經過全部policy的檢測,則把請求轉發到目標action。
在Sails下,有一個很方便的套件sails-permissions,集成了一套權限管理的方案,本文也是基於該套件的源碼所引出來的權限管理解決方案。
對程序員最大的挑戰,並非可否掌握了哪些編程語言,哪些軟件框架,而是對業務和需求的理解,而後在此基礎上,把要點抽象出來,寫成計算機能理解的語言。
最後,但願這篇文章,可以幫助你對權限管理這一課題增長多一點點理解。
<?php // +---------------------------------------------------------------------- // | ThinkPHP [ WE CAN DO IT JUST THINK IT ] // +---------------------------------------------------------------------- // | Copyright (c) 2006-2014 http://thinkphp.cn All rights reserved. // +---------------------------------------------------------------------- // | Licensed ( http://www.apache.org/licenses/LICENSE-2.0 ) // +---------------------------------------------------------------------- // | Author: liu21st <liu21st@gmail.com> // +---------------------------------------------------------------------- namespace Think\Controller; use Think\Controller; /** * ThinkPHP REST控制器類 */ class RestController extends Controller { // 當前請求類型 protected $_method = ''; // 當前請求的資源類型 protected $_type = ''; // REST容許的請求類型列表 protected $allowMethod = array('get','post','put','delete'); // REST默認請求類型 protected $defaultMethod = 'get'; // REST容許請求的資源類型列表 protected $allowType = array('html','xml','json','rss'); // 默認的資源類型 protected $defaultType = 'html'; // REST容許輸出的資源類型列表 protected $allowOutputType= array( 'xml' => 'application/xml', 'json' => 'application/json', 'html' => 'text/html', ); /** * 架構函數 * @access public */ public function __construct() { // 資源類型檢測 if(''==__EXT__) { // 自動檢測資源類型 $this->_type = $this->getAcceptType(); }elseif(!in_array(__EXT__,$this->allowType)) { // 資源類型非法 則用默認資源類型訪問 $this->_type = $this->defaultType; }else{ // 檢測實際資源類型 $this->_type = $this->getAcceptType() == __EXT__ ? __EXT__ : $this->defaultType; } // 請求方式檢測 $method = strtolower(REQUEST_METHOD); if(!in_array($method,$this->allowMethod)) { // 請求方式非法 則用默認請求方法 $method = $this->defaultMethod; } $this->_method = $method; parent::__construct(); } /** * 魔術方法 有不存在的操做的時候執行 * @access public * @param string $method 方法名 * @param array $args 參數 * @return mixed */ public function __call($method,$args) { if( 0 === strcasecmp($method,ACTION_NAME.C('ACTION_SUFFIX'))) { if(method_exists($this,$method.'_'.$this->_method.'_'.$this->_type)) { // RESTFul方法支持 $fun = $method.'_'.$this->_method.'_'.$this->_type; $this->$fun(); }elseif($this->_method == $this->defaultMethod && method_exists($this,$method.'_'.$this->_type) ){ $fun = $method.'_'.$this->_type; $this->$fun(); }elseif($this->_type == $this->defaultType && method_exists($this,$method.'_'.$this->_method) ){ $fun = $method.'_'.$this->_method; $this->$fun(); }elseif(method_exists($this,'_empty')) { // 若是定義了_empty操做 則調用 $this->_empty($method,$args); }elseif(file_exists_case($this->view->parseTemplate())){ // 檢查是否存在默認模版 若是有直接輸出模版 $this->display(); }else{ E(L('_ERROR_ACTION_').':'.ACTION_NAME); } } } /** * 獲取當前請求的Accept頭信息 * @return string */ protected function getAcceptType(){ $type = array( 'html' => 'text/html,application/xhtml+xml,*/*', 'xml' => 'application/xml,text/xml,application/x-xml', 'json' => 'application/json,text/x-json,application/jsonrequest,text/json', 'js' => 'text/javascript,application/javascript,application/x-javascript', 'css' => 'text/css', 'rss' => 'application/rss+xml', 'yaml' => 'application/x-yaml,text/yaml', 'atom' => 'application/atom+xml', 'pdf' => 'application/pdf', 'text' => 'text/plain', 'png' => 'image/png', 'jpg' => 'image/jpg,image/jpeg,image/pjpeg', 'gif' => 'image/gif', 'csv' => 'text/csv' ); foreach($type as $key=>$val){ $array = explode(',',$val); foreach($array as $k=>$v){ if(stristr($_SERVER['HTTP_ACCEPT'], $v)) { return $key; } } } return false; } // 發送Http狀態信息 protected function sendHttpStatus($code) { static $_status = array( // Informational 1xx 100 => 'Continue', 101 => 'Switching Protocols', // Success 2xx 200 => 'OK', 201 => 'Created', 202 => 'Accepted', 203 => 'Non-Authoritative Information', 204 => 'No Content', 205 => 'Reset Content', 206 => 'Partial Content', // Redirection 3xx 300 => 'Multiple Choices', 301 => 'Moved Permanently', 302 => 'Moved Temporarily ', // 1.1 303 => 'See Other', 304 => 'Not Modified', 305 => 'Use Proxy', // 306 is deprecated but reserved 307 => 'Temporary Redirect', // Client Error 4xx 400 => 'Bad Request', 401 => 'Unauthorized', 402 => 'Payment Required', 403 => 'Forbidden', 404 => 'Not Found', 405 => 'Method Not Allowed', 406 => 'Not Acceptable', 407 => 'Proxy Authentication Required', 408 => 'Request Timeout', 409 => 'Conflict', 410 => 'Gone', 411 => 'Length Required', 412 => 'Precondition Failed', 413 => 'Request Entity Too Large', 414 => 'Request-URI Too Long', 415 => 'Unsupported Media Type', 416 => 'Requested Range Not Satisfiable', 417 => 'Expectation Failed', // Server Error 5xx 500 => 'Internal Server Error', 501 => 'Not Implemented', 502 => 'Bad Gateway', 503 => 'Service Unavailable', 504 => 'Gateway Timeout', 505 => 'HTTP Version Not Supported', 509 => 'Bandwidth Limit Exceeded' ); if(isset($_status[$code])) { header('HTTP/1.1 '.$code.' '.$_status[$code]); // 確保FastCGI模式下正常 header('Status:'.$code.' '.$_status[$code]); } } /** * 編碼數據 * @access protected * @param mixed $data 要返回的數據 * @param String $type 返回類型 JSON XML * @return string */ protected function encodeData($data,$type='') { if(empty($data)) return ''; if('json' == $type) { // 返回JSON數據格式到客戶端 包含狀態信息 $data = json_encode($data); }elseif('xml' == $type){ // 返回xml格式數據 $data = xml_encode($data); }elseif('php'==$type){ $data = serialize($data); }// 默認直接輸出 $this->setContentType($type); //header('Content-Length: ' . strlen($data)); return $data; } /** * 設置頁面輸出的CONTENT_TYPE和編碼 * @access public * @param string $type content_type 類型對應的擴展名 * @param string $charset 頁面輸出編碼 * @return void */ public function setContentType($type, $charset=''){ if(headers_sent()) return; if(empty($charset)) $charset = C('DEFAULT_CHARSET'); $type = strtolower($type); if(isset($this->allowOutputType[$type])) //過濾content_type header('Content-Type: '.$this->allowOutputType[$type].'; charset='.$charset); } /** * 輸出返回數據 * @access protected * @param mixed $data 要返回的數據 * @param String $type 返回類型 JSON XML * @param integer $code HTTP狀態 * @return void */ protected function response($data,$type='',$code=200) { $this->sendHttpStatus($code); exit($this->encodeData($data,strtolower($type))); } }