RBAC.class.php 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240
  1. <?php
  2. /**
  3. +------------------------------------------------------------------------------
  4. * 基于角色的数据库方式验证类
  5. +------------------------------------------------------------------------------
  6. * @category ORG
  7. * @package ORG
  8. * @subpackage Util
  9. * @author liu21st <liu21st@gmail.com>
  10. * @version $Id$
  11. +------------------------------------------------------------------------------
  12. */
  13. // 配置文件增加设置
  14. // USER_AUTH_ON 是否需要认证
  15. // USER_AUTH_TYPE 认证类型
  16. // USER_AUTH_KEY 认证识别号
  17. // REQUIRE_AUTH_MODULE 需要认证模块
  18. // NOT_AUTH_MODULE 无需认证模块
  19. // USER_AUTH_GATEWAY 认证网关
  20. class RBAC extends Think
  21. {
  22. // 认证方法
  23. static public function authenticate($map,$model='')
  24. {
  25. if(empty($model)) $model = C('USER_AUTH_MODEL');
  26. //使用给定的Map进行认证
  27. return M($model)->where($map)->find();
  28. }
  29. //用于检测用户权限的方法,并保存到Session中
  30. static function saveAccessList($authId=null)
  31. {
  32. if(null===$authId) $authId = $_SESSION[C('USER_AUTH_KEY')];
  33. // 如果使用普通权限模式,保存当前用户的访问权限列表
  34. // 对管理员开发所有权限
  35. if(C('USER_AUTH_TYPE') !=2 && !$_SESSION[C('ADMIN_AUTH_KEY')] )
  36. $_SESSION['_ACCESS_LIST'] = RBAC::getAccessList($authId);
  37. return ;
  38. }
  39. // 取得模块的所属记录访问权限列表 返回有权限的记录ID数组
  40. static function getRecordAccessList($authId=null,$module='') {
  41. if(null===$authId) $authId = $_SESSION[C('USER_AUTH_KEY')];
  42. if(empty($module)) $module = MODULE_NAME;
  43. //获取权限访问列表
  44. $accessList = RBAC::getModuleAccessList($authId,$module);
  45. return $accessList;
  46. }
  47. //检查当前操作是否需要认证
  48. static function checkAccess()
  49. {
  50. //如果项目要求认证,并且当前模块需要认证,则进行权限认证
  51. if( C('USER_AUTH_ON') ){
  52. $_module = array();
  53. $_action = array();
  54. if("" != C('REQUIRE_AUTH_MODULE')) {
  55. //需要认证的模块
  56. $_module['yes'] = explode(',',strtoupper(C('REQUIRE_AUTH_MODULE')));
  57. }else {
  58. //无需认证的模块
  59. $_module['no'] = explode(',',strtoupper(C('NOT_AUTH_MODULE')));
  60. }
  61. //检查当前模块是否需要认证
  62. if((!empty($_module['no']) && !in_array(strtoupper(MODULE_NAME),$_module['no'])) || (!empty($_module['yes']) && in_array(strtoupper(MODULE_NAME),$_module['yes']))) {
  63. if("" != C('REQUIRE_AUTH_ACTION')) {
  64. //需要认证的操作
  65. $_action['yes'] = explode(',',strtoupper(C('REQUIRE_AUTH_ACTION')));
  66. }else {
  67. //无需认证的操作
  68. $_action['no'] = explode(',',strtoupper(C('NOT_AUTH_ACTION')));
  69. }
  70. //检查当前操作是否需要认证
  71. if((!empty($_action['no']) && !in_array(strtoupper(ACTION_NAME),$_action['no'])) || (!empty($_action['yes']) && in_array(strtoupper(ACTION_NAME),$_action['yes']))) {
  72. return true;
  73. }else {
  74. return false;
  75. }
  76. }else {
  77. return false;
  78. }
  79. }
  80. return false;
  81. }
  82. // 登录检查
  83. static public function checkLogin() {
  84. //检查当前操作是否需要认证
  85. if(RBAC::checkAccess()) {
  86. //检查认证识别号
  87. if(!$_SESSION[C('USER_AUTH_KEY')]) {
  88. if(C('GUEST_AUTH_ON')) {
  89. // 开启游客授权访问
  90. if(!isset($_SESSION['_ACCESS_LIST']))
  91. // 保存游客权限
  92. RBAC::saveAccessList(C('GUEST_AUTH_ID'));
  93. }else{
  94. // 禁止游客访问跳转到认证网关
  95. redirect(PHP_FILE.C('USER_AUTH_GATEWAY'));
  96. }
  97. }
  98. }
  99. return true;
  100. }
  101. //权限认证的过滤器方法
  102. static public function AccessDecision($appName=APP_NAME)
  103. {
  104. //检查是否需要认证
  105. if(RBAC::checkAccess()) {
  106. //存在认证识别号,则进行进一步的访问决策
  107. $accessGuid = md5($appName.MODULE_NAME.ACTION_NAME);
  108. if(empty($_SESSION[C('ADMIN_AUTH_KEY')])) {
  109. if(C('USER_AUTH_TYPE')==2) {
  110. //加强验证和即时验证模式 更加安全 后台权限修改可以即时生效
  111. //通过数据库进行访问检查
  112. $accessList = RBAC::getAccessList($_SESSION[C('USER_AUTH_KEY')]);
  113. }else {
  114. // 如果是管理员或者当前操作已经认证过,无需再次认证
  115. if( $_SESSION[$accessGuid]) {
  116. return true;
  117. }
  118. //登录验证模式,比较登录后保存的权限访问列表
  119. $accessList = $_SESSION['_ACCESS_LIST'];
  120. }
  121. //判断是否为组件化模式,如果是,验证其全模块名
  122. $module = defined('P_MODULE_NAME')? P_MODULE_NAME : MODULE_NAME;
  123. if(!isset($accessList[strtoupper($appName)][strtoupper($module)][strtoupper(ACTION_NAME)])) {
  124. $_SESSION[$accessGuid] = false;
  125. return false;
  126. }
  127. else {
  128. $_SESSION[$accessGuid] = true;
  129. }
  130. }else{
  131. //管理员无需认证
  132. return true;
  133. }
  134. }
  135. return true;
  136. }
  137. /**
  138. +----------------------------------------------------------
  139. * 取得当前认证号的所有权限列表
  140. +----------------------------------------------------------
  141. * @param integer $authId 用户ID
  142. +----------------------------------------------------------
  143. * @access public
  144. +----------------------------------------------------------
  145. */
  146. static public function getAccessList($authId)
  147. {
  148. // Db方式权限数据
  149. $db = Db::getInstance();
  150. $table = array('role'=>C('RBAC_ROLE_TABLE'),'user'=>C('RBAC_USER_TABLE'),'access'=>C('RBAC_ACCESS_TABLE'),'node'=>C('RBAC_NODE_TABLE'));
  151. $sql = "select node.id,node.name from ".
  152. $table['role']." as role,".
  153. $table['user']." as user,".
  154. $table['access']." as access ,".
  155. $table['node']." as node ".
  156. "where user.user_id='{$authId}' and user.role_id=role.id and ( access.role_id=role.id or (access.role_id=role.pid and role.pid!=0 ) ) and role.status=1 and access.node_id=node.id and node.level=1 and node.status=1";
  157. $apps = $db->query($sql);
  158. $access = array();
  159. foreach($apps as $key=>$app) {
  160. $appId = $app['id'];
  161. $appName = $app['name'];
  162. // 读取项目的模块权限
  163. $access[strtoupper($appName)] = array();
  164. $sql = "select node.id,node.name from ".
  165. $table['role']." as role,".
  166. $table['user']." as user,".
  167. $table['access']." as access ,".
  168. $table['node']." as node ".
  169. "where user.user_id='{$authId}' and user.role_id=role.id and ( access.role_id=role.id or (access.role_id=role.pid and role.pid!=0 ) ) and role.status=1 and access.node_id=node.id and node.level=2 and node.pid={$appId} and node.status=1";
  170. $modules = $db->query($sql);
  171. // 判断是否存在公共模块的权限
  172. $publicAction = array();
  173. foreach($modules as $key=>$module) {
  174. $moduleId = $module['id'];
  175. $moduleName = $module['name'];
  176. if('PUBLIC'== strtoupper($moduleName)) {
  177. $sql = "select node.id,node.name from ".
  178. $table['role']." as role,".
  179. $table['user']." as user,".
  180. $table['access']." as access ,".
  181. $table['node']." as node ".
  182. "where user.user_id='{$authId}' and user.role_id=role.id and ( access.role_id=role.id or (access.role_id=role.pid and role.pid!=0 ) ) and role.status=1 and access.node_id=node.id and node.level=3 and node.pid={$moduleId} and node.status=1";
  183. $rs = $db->query($sql);
  184. foreach ($rs as $a){
  185. $publicAction[$a['name']] = $a['id'];
  186. }
  187. unset($modules[$key]);
  188. break;
  189. }
  190. }
  191. // 依次读取模块的操作权限
  192. foreach($modules as $key=>$module) {
  193. $moduleId = $module['id'];
  194. $moduleName = $module['name'];
  195. $sql = "select node.id,node.name from ".
  196. $table['role']." as role,".
  197. $table['user']." as user,".
  198. $table['access']." as access ,".
  199. $table['node']." as node ".
  200. "where user.user_id='{$authId}' and user.role_id=role.id and ( access.role_id=role.id or (access.role_id=role.pid and role.pid!=0 ) ) and role.status=1 and access.node_id=node.id and node.level=3 and node.pid={$moduleId} and node.status=1";
  201. $rs = $db->query($sql);
  202. $action = array();
  203. foreach ($rs as $a){
  204. $action[$a['name']] = $a['id'];
  205. }
  206. // 和公共模块的操作权限合并
  207. $action += $publicAction;
  208. $access[strtoupper($appName)][strtoupper($moduleName)] = array_change_key_case($action,CASE_UPPER);
  209. }
  210. }
  211. return $access;
  212. }
  213. // 读取模块所属的记录访问权限
  214. static public function getModuleAccessList($authId,$module) {
  215. // Db方式
  216. $db = Db::getInstance();
  217. $table = array('role'=>C('RBAC_ROLE_TABLE'),'user'=>C('RBAC_USER_TABLE'),'access'=>C('RBAC_ACCESS_TABLE'));
  218. $sql = "select access.node_id from ".
  219. $table['role']." as role,".
  220. $table['user']." as user,".
  221. $table['access']." as access ".
  222. "where user.user_id='{$authId}' and user.role_id=role.id and ( access.role_id=role.id or (access.role_id=role.pid and role.pid!=0 ) ) and role.status=1 and access.module='{$module}' and access.status=1";
  223. $rs = $db->query($sql);
  224. $access = array();
  225. foreach ($rs as $node){
  226. $access[] = $node['node_id'];
  227. }
  228. return $access;
  229. }
  230. }//end class
  231. ?>