NodeService.php 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | ThinkAdmin
  4. // +----------------------------------------------------------------------
  5. // | 版权所有 2014~2019 广州楚才信息科技有限公司 [ http://www.cuci.cc ]
  6. // +----------------------------------------------------------------------
  7. // | 官方网站: http://demo.thinkadmin.top
  8. // +----------------------------------------------------------------------
  9. // | 开源协议 ( https://mit-license.org )
  10. // +----------------------------------------------------------------------
  11. // | gitee 代码仓库:https://gitee.com/zoujingli/ThinkAdmin
  12. // | github 代码仓库:https://github.com/zoujingli/ThinkAdmin
  13. // +----------------------------------------------------------------------
  14. namespace app\admin\service;
  15. use library\tools\Data;
  16. use library\tools\Node;
  17. use think\Db;
  18. use think\facade\App;
  19. use think\facade\Cache;
  20. use think\facade\Request;
  21. /**
  22. * 功能节点管理服务
  23. * Class NodeService
  24. * @package app\admin\service
  25. */
  26. class NodeService
  27. {
  28. /**
  29. * 获取标准访问节点
  30. * @param string $node
  31. * @return string
  32. */
  33. public static function full($node = null)
  34. {
  35. if (empty($node)) return self::current();
  36. if (count(explode('/', $node)) === 1) {
  37. $node = Request::module() . '/' . Request::controller() . '/' . $node;
  38. }
  39. return self::parseString(trim($node, " /"));
  40. }
  41. /**
  42. * 判断是否已经登录
  43. * @return boolean
  44. */
  45. public static function islogin()
  46. {
  47. return session('admin_user.id') ? true : false;
  48. }
  49. /**
  50. * 获取当前访问节点
  51. * @return string
  52. */
  53. public static function current()
  54. {
  55. return self::parseString(Request::module() . '/' . Request::controller() . '/' . Request::action());
  56. }
  57. /**
  58. * 检查密码是否合法
  59. * @param string $password
  60. * @return array
  61. */
  62. public static function checkpwd($password)
  63. {
  64. $password = trim($password);
  65. if (!strlen($password) >= 6) {
  66. return ['code' => 0, 'msg' => '密码必须大于6字符!'];
  67. }
  68. if (!preg_match("/^(?![\d]+$)(?![a-zA-Z]+$)(?![^\da-zA-Z]+$).{6,32}$/", $password)) {
  69. return ['code' => 0, 'msg' => '密码必需包含大小写字母、数字、符号任意两者组合!'];
  70. } else {
  71. return ['code' => 1, 'msg' => '密码复杂度通过验证!'];
  72. }
  73. }
  74. /**
  75. * 获取可选菜单节点
  76. * @return array
  77. * @throws \ReflectionException
  78. */
  79. public static function getMenuNodeList()
  80. {
  81. static $nodes = [];
  82. if (count($nodes) > 0) return $nodes;
  83. foreach (self::getMethodList() as $node => $method) if ($method['menu']) {
  84. $nodes[] = ['node' => $node, 'title' => $method['title']];
  85. }
  86. return $nodes;
  87. }
  88. /**
  89. * 获取系统菜单树数据
  90. * @return array
  91. * @throws \ReflectionException
  92. * @throws \think\db\exception\DataNotFoundException
  93. * @throws \think\db\exception\ModelNotFoundException
  94. * @throws \think\exception\DbException
  95. */
  96. public static function getMenuNodeTree()
  97. {
  98. $list = Db::name('SystemMenu')->where(['status' => '1'])->order('sort desc,id asc')->select();
  99. return self::buildMenuData(Data::arr2tree($list), self::getMethodList());
  100. }
  101. /**
  102. * 后台主菜单权限过滤
  103. * @param array $menus 当前菜单列表
  104. * @param array $nodes 系统权限节点
  105. * @return array
  106. * @throws \ReflectionException
  107. */
  108. private static function buildMenuData($menus, $nodes)
  109. {
  110. foreach ($menus as $key => &$menu) {
  111. if (!empty($menu['sub'])) $menu['sub'] = self::buildMenuData($menu['sub'], $nodes);
  112. if (!empty($menu['sub'])) $menu['url'] = '#';
  113. elseif (preg_match('/^https?\:/i', $menu['url'])) continue;
  114. elseif ($menu['url'] === '#') unset($menus[$key]);
  115. else {
  116. $node = join('/', array_slice(explode('/', preg_replace('/[\W]/', '/', $menu['url'])), 0, 3));
  117. $menu['url'] = url($menu['url']) . (empty($menu['params']) ? '' : "?{$menu['params']}");
  118. if (!self::checkAuth($node)) unset($menus[$key]);
  119. }
  120. }
  121. return $menus;
  122. }
  123. /**
  124. * 获取授权节点列表
  125. * @return array
  126. * @throws \ReflectionException
  127. */
  128. public static function getAuthList()
  129. {
  130. static $nodes = [];
  131. if (count($nodes) > 0) return $nodes;
  132. $nodes = Cache::tag('system')->get('NodeAuthList', []);
  133. if (count($nodes) > 0) return $nodes;
  134. foreach (self::getMethodList() as $key => $node) {
  135. if ($node['auth']) $nodes[$key] = $node['title'];
  136. }
  137. Cache::tag('system')->set('NodeAuthList', $nodes);
  138. return $nodes;
  139. }
  140. /**
  141. * 强制验证访问权限
  142. * --- 需要加载控制器解析注释
  143. * @param null|string $node
  144. * @return boolean
  145. * @throws \ReflectionException
  146. */
  147. public static function forceAuth($node = null)
  148. {
  149. if (session('admin_user.username') === 'admin') return true;
  150. $real = is_null($node) ? self::current() : self::full($node);
  151. list($module, $controller, $action) = explode('/', $real);
  152. if (class_exists($class = App::parseClass($module, 'controller', $controller))) {
  153. $reflection = new \ReflectionClass($class);
  154. if ($reflection->hasMethod($action)) {
  155. $comment = preg_replace("/\s/", '', $reflection->getMethod($action)->getDocComment());
  156. if (stripos($comment, '@authtrue') === false) return true;
  157. return in_array($real, (array)session('admin_user.nodes'));
  158. }
  159. }
  160. return true;
  161. }
  162. /**
  163. * 检查指定节点授权
  164. * --- 需要读取缓存或扫描所有节点
  165. * @param null|string $node
  166. * @return boolean
  167. * @throws \ReflectionException
  168. */
  169. public static function checkAuth($node = null)
  170. {
  171. if (session('admin_user.username') === 'admin') return true;
  172. $real = is_null($node) ? self::current() : self::full($node);
  173. if (isset(self::getAuthList()[$real])) {
  174. return in_array($real, (array)session('admin_user.nodes'));
  175. } else {
  176. return true;
  177. }
  178. }
  179. /**
  180. * 获取授权节点列表
  181. * @param array $checkeds
  182. * @return array
  183. * @throws \ReflectionException
  184. */
  185. public static function getAuthTree($checkeds = [])
  186. {
  187. static $nodes = [];
  188. if (count($nodes) > 0) return $nodes;
  189. foreach (self::getAuthList() as $node => $title) {
  190. $pnode = substr($node, 0, strripos($node, '/'));
  191. $nodes[$node] = ['node' => $node, 'title' => $title, 'pnode' => $pnode, 'checked' => in_array($node, $checkeds)];
  192. }
  193. foreach (self::getClassList() as $node => $title) foreach (array_keys($nodes) as $key) {
  194. if (stripos($key, "{$node}/") !== false) {
  195. $pnode = substr($node, 0, strripos($node, '/'));
  196. $nodes[$node] = ['node' => $node, 'title' => $title, 'pnode' => $pnode, 'checked' => in_array($node, $checkeds)];
  197. $nodes[$pnode] = ['node' => $pnode, 'title' => ucfirst($pnode), 'checked' => in_array($pnode, $checkeds)];
  198. }
  199. }
  200. return $nodes = Data::arr2tree($nodes, 'node', 'pnode', '_sub_');
  201. }
  202. /**
  203. * 初始化用户权限
  204. * @param boolean $force 是否重置系统权限
  205. * @throws \think\db\exception\DataNotFoundException
  206. * @throws \think\db\exception\ModelNotFoundException
  207. * @throws \think\exception\DbException
  208. */
  209. public static function applyUserAuth($force = false)
  210. {
  211. if ($force) {
  212. Cache::tag('system')->rm('NodeAuthList');
  213. Cache::tag('system')->rm('NodeClassData');
  214. Cache::tag('system')->rm('NodeMethodData');
  215. }
  216. if (($uid = session('admin_user.id'))) {
  217. session('admin_user', Db::name('SystemUser')->where(['id' => $uid])->find());
  218. }
  219. if (($aids = session('admin_user.authorize'))) {
  220. $where = [['status', 'eq', '1'], ['id', 'in', explode(',', $aids)]];
  221. $subsql = Db::name('SystemAuth')->field('id')->where($where)->buildSql();
  222. session('admin_user.nodes', array_unique(Db::name('SystemAuthNode')->whereRaw("auth in {$subsql}")->column('node')));
  223. } else {
  224. session('admin_user.nodes', []);
  225. }
  226. }
  227. /**
  228. * 获取控制器节点列表
  229. * @return array
  230. * @throws \ReflectionException
  231. */
  232. public static function getClassList()
  233. {
  234. static $nodes = [];
  235. if (count($nodes) > 0) return $nodes;
  236. $nodes = Cache::tag('system')->get('NodeClassData', []);
  237. if (count($nodes) > 0) return $nodes;
  238. self::eachController(function (\ReflectionClass $reflection, $prenode) use (&$nodes) {
  239. list($node, $comment) = [trim($prenode, ' / '), $reflection->getDocComment()];
  240. $nodes[$node] = preg_replace('/^\/\*\*\*(.*?)\*.*?$/', '$1', preg_replace("/\s/", '', $comment));
  241. if (stripos($nodes[$node], '@') !== false) $nodes[$node] = '';
  242. });
  243. Cache::tag('system')->set('NodeClassData', $nodes);
  244. return $nodes;
  245. }
  246. /**
  247. * 获取方法节点列表
  248. * @return array
  249. * @throws \ReflectionException
  250. */
  251. public static function getMethodList()
  252. {
  253. static $nodes = [];
  254. if (count($nodes) > 0) return $nodes;
  255. $nodes = Cache::tag('system')->get('NodeMethodData', []);
  256. if (count($nodes) > 0) return $nodes;
  257. self::eachController(function (\ReflectionClass $reflection, $prenode) use (&$nodes) {
  258. foreach ($reflection->getMethods(\ReflectionMethod::IS_PUBLIC) as $method) {
  259. $action = strtolower($method->getName());
  260. list($node, $comment) = ["{$prenode}{$action}", preg_replace("/\s/", '', $method->getDocComment())];
  261. $nodes[$node] = [
  262. 'auth' => stripos($comment, '@authtrue') !== false,
  263. 'menu' => stripos($comment, '@menutrue') !== false,
  264. 'title' => preg_replace('/^\/\*\*\*(.*?)\*.*?$/', '$1', $comment),
  265. ];
  266. if (stripos($nodes[$node]['title'], '@') !== false) $nodes[$node]['title'] = '';
  267. }
  268. });
  269. Cache::tag('system')->set('NodeMethodData', $nodes);
  270. return $nodes;
  271. }
  272. /**
  273. * 控制器扫描回调
  274. * @param callable $callable
  275. * @throws \ReflectionException
  276. */
  277. public static function eachController($callable)
  278. {
  279. foreach (self::scanPath(env('app_path') . "*/controller/") as $file) {
  280. if (!preg_match("|/(\w+)/controller/(.+)\.php$|", $file, $matches)) continue;
  281. list($module, $controller) = [$matches[1], strtr($matches[2], '/', '.')];
  282. if (class_exists($class = substr(strtr(env('app_namespace') . $matches[0], '/', '\\'), 0, -4))) {
  283. call_user_func($callable, new \ReflectionClass($class), Node::parseString("{$module}/{$controller}/"));
  284. }
  285. }
  286. }
  287. /**
  288. * 驼峰转下划线规则
  289. * @param string $node 节点名称
  290. * @return string
  291. */
  292. public static function parseString($node)
  293. {
  294. if (count($nodes = explode('/', $node)) > 1) {
  295. $dots = [];
  296. foreach (explode('.', $nodes[1]) as $dot) {
  297. $dots[] = trim(preg_replace("/[A-Z]/", "_\\0", $dot), "_");
  298. }
  299. $nodes[1] = join('.', $dots);
  300. }
  301. return strtolower(join('/', $nodes));
  302. }
  303. /**
  304. * 获取所有PHP文件列表
  305. * @param string $dirname 扫描目录
  306. * @param array $data 额外数据
  307. * @param string $ext 有文件后缀
  308. * @return array
  309. */
  310. private static function scanPath($dirname, $data = [], $ext = 'php')
  311. {
  312. foreach (glob("{$dirname}*") as $file) {
  313. if (is_dir($file)) {
  314. $data = array_merge($data, self::scanPath("{$file}/"));
  315. } elseif (is_file($file) && pathinfo($file, 4) === $ext) {
  316. $data[] = str_replace('\\', '/', $file);
  317. }
  318. }
  319. return $data;
  320. }
  321. }