<?php namespace app\admin\command; use app\admin\model\AuthRule; use ReflectionClass; use ReflectionMethod; use think\Cache; use think\Config; use think\console\Command; use think\console\Input; use think\console\input\Option; use think\console\Output; use think\Exception; use think\Loader; class Menu extends Command { protected $model = null; protected function configure() { $this ->setName('menu') ->addOption('controller', 'c', Option::VALUE_REQUIRED | Option::VALUE_IS_ARRAY, 'controller name,use \'all-controller\' when build all menu', null) ->addOption('delete', 'd', Option::VALUE_OPTIONAL, 'delete the specified menu', '') ->addOption('force', 'f', Option::VALUE_OPTIONAL, 'force delete menu,without tips', null) ->addOption('equal', 'e', Option::VALUE_OPTIONAL, 'the controller must be equal', null) ->setDescription('Build auth menu from controller'); //要执行的controller必须一样,不适用模糊查询 } protected function execute(Input $input, Output $output) { $this->model = new AuthRule(); $adminPath = dirname(__DIR__) . DS; //控制器名 $controller = $input->getOption('controller') ?: ''; if (!$controller) { throw new Exception("please input controller name"); } $force = $input->getOption('force'); //是否为删除模式 $delete = $input->getOption('delete'); //是否控制器完全匹配 $equal = $input->getOption('equal'); if ($delete) { if (in_array('all-controller', $controller)) { throw new Exception("could not delete all menu"); } $ids = []; $list = $this->model->where(function ($query) use ($controller, $equal) { foreach ($controller as $index => $item) { if (stripos($item, '_') !== false) { $item = Loader::parseName($item, 1); } if (stripos($item, '/') !== false) { $controllerArr = explode('/', $item); end($controllerArr); $key = key($controllerArr); $controllerArr[$key] = Loader::parseName($controllerArr[$key]); } else { $controllerArr = [Loader::parseName($item)]; } $item = str_replace('_', '\_', implode('/', $controllerArr)); if ($equal) { $query->whereOr('name', 'eq', $item); } else { $query->whereOr('name', 'like', strtolower($item) . "%"); } } })->select(); foreach ($list as $k => $v) { $output->warning($v->name); $ids[] = $v->id; } if (!$ids) { throw new Exception("There is no menu to delete"); } if (!$force) { $output->info("Are you sure you want to delete all those menu? Type 'yes' to continue: "); $line = fgets(STDIN); if (trim($line) != 'yes') { throw new Exception("Operation is aborted!"); } } AuthRule::destroy($ids); Cache::rm("__menu__"); $output->info("Delete Successed"); return; } if (!in_array('all-controller', $controller)) { foreach ($controller as $index => $item) { if (stripos($item, '_') !== false) { $item = Loader::parseName($item, 1); } if (stripos($item, '/') !== false) { $controllerArr = explode('/', $item); end($controllerArr); $key = key($controllerArr); $controllerArr[$key] = ucfirst($controllerArr[$key]); } else { $controllerArr = [ucfirst($item)]; } $adminPath = dirname(__DIR__) . DS . 'controller' . DS . implode(DS, $controllerArr) . '.php'; if (!is_file($adminPath)) { $output->error("controller not found"); return; } $this->importRule($item); } } else { $authRuleList = AuthRule::select(); //生成权限规则备份文件 file_put_contents(RUNTIME_PATH . 'authrule.json', json_encode(collection($authRuleList)->toArray())); $this->model->where('id', '>', 0)->delete(); $controllerDir = $adminPath . 'controller' . DS; // 扫描新的节点信息并导入 $treelist = $this->import($this->scandir($controllerDir)); } Cache::rm("__menu__"); $output->info("Build Successed!"); } /** * 递归扫描文件夹 * @param string $dir * @return array */ public function scandir($dir) { $result = []; $cdir = scandir($dir); foreach ($cdir as $value) { if (!in_array($value, array(".", ".."))) { if (is_dir($dir . DS . $value)) { $result[$value] = $this->scandir($dir . DS . $value); } else { $result[] = $value; } } } return $result; } /** * 导入规则节点 * @param array $dirarr * @param array $parentdir * @return array */ public function import($dirarr, $parentdir = []) { $menuarr = []; foreach ($dirarr as $k => $v) { if (is_array($v)) { //当前是文件夹 $nowparentdir = array_merge($parentdir, [$k]); $this->import($v, $nowparentdir); } else { //只匹配PHP文件 if (!preg_match('/^(\w+)\.php$/', $v, $matchone)) { continue; } //导入文件 $controller = ($parentdir ? implode('/', $parentdir) . '/' : '') . $matchone[1]; $this->importRule($controller); } } return $menuarr; } protected function importRule($controller) { $controller = str_replace('\\', '/', $controller); if (stripos($controller, '/') !== false) { $controllerArr = explode('/', $controller); end($controllerArr); $key = key($controllerArr); $controllerArr[$key] = ucfirst($controllerArr[$key]); } else { $key = 0; $controllerArr = [ucfirst($controller)]; } $classSuffix = Config::get('controller_suffix') ? ucfirst(Config::get('url_controller_layer')) : ''; $className = "\\app\\admin\\controller\\" . implode("\\", $controllerArr) . $classSuffix; $pathArr = $controllerArr; array_unshift($pathArr, '', 'application', 'admin', 'controller'); $classFile = ROOT_PATH . implode(DS, $pathArr) . $classSuffix . ".php"; $classContent = file_get_contents($classFile); $uniqueName = uniqid("FastAdmin") . $classSuffix; $classContent = str_replace("class " . $controllerArr[$key] . $classSuffix . " ", 'class ' . $uniqueName . ' ', $classContent); $classContent = preg_replace("/namespace\s(.*);/", 'namespace ' . __NAMESPACE__ . ";", $classContent); //临时的类文件 $tempClassFile = __DIR__ . DS . $uniqueName . ".php"; file_put_contents($tempClassFile, $classContent); $className = "\\app\\admin\\command\\" . $uniqueName; //反射机制调用类的注释和方法名 $reflector = new ReflectionClass($className); if (isset($tempClassFile)) { //删除临时文件 @unlink($tempClassFile); } //只匹配公共的方法 $methods = $reflector->getMethods(ReflectionMethod::IS_PUBLIC); $classComment = $reflector->getDocComment(); //判断是否有启用软删除 $softDeleteMethods = ['destroy', 'restore', 'recyclebin']; $withSofeDelete = false; $modelRegexArr = ["/\\\$this\->model\s*=\s*model\(['|\"](\w+)['|\"]\);/", "/\\\$this\->model\s*=\s*new\s+([a-zA-Z\\\]+);/"]; $modelRegex = preg_match($modelRegexArr[0], $classContent) ? $modelRegexArr[0] : $modelRegexArr[1]; preg_match_all($modelRegex, $classContent, $matches); if (isset($matches[1]) && isset($matches[1][0]) && $matches[1][0]) { \think\Request::instance()->module('admin'); $model = model($matches[1][0]); if (in_array('trashed', get_class_methods($model))) { $withSofeDelete = true; } } //忽略的类 if (stripos($classComment, "@internal") !== false) { return; } preg_match_all('#(@.*?)\n#s', $classComment, $annotations); $controllerIcon = 'fa fa-circle-o'; $controllerRemark = ''; //判断注释中是否设置了icon值 if (isset($annotations[1])) { foreach ($annotations[1] as $tag) { if (stripos($tag, '@icon') !== false) { $controllerIcon = substr($tag, stripos($tag, ' ') + 1); } if (stripos($tag, '@remark') !== false) { $controllerRemark = substr($tag, stripos($tag, ' ') + 1); } } } //过滤掉其它字符 $controllerTitle = trim(preg_replace(array('/^\/\*\*(.*)[\n\r\t]/u', '/[\s]+\*\//u', '/\*\s@(.*)/u', '/[\s|\*]+/u'), '', $classComment)); //导入中文语言包 \think\Lang::load(dirname(__DIR__) . DS . 'lang/zh-cn.php'); //先导入菜单的数据 $pid = 0; foreach ($controllerArr as $k => $v) { $key = $k + 1; //驼峰转下划线 $controllerNameArr = array_slice($controllerArr, 0, $key); foreach ($controllerNameArr as &$val) { $val = strtolower(trim(preg_replace("/[A-Z]/", "_\\0", $val), "_")); } unset($val); $name = implode('/', $controllerNameArr); $title = (!isset($controllerArr[$key]) ? $controllerTitle : ''); $icon = (!isset($controllerArr[$key]) ? $controllerIcon : 'fa fa-list'); $remark = (!isset($controllerArr[$key]) ? $controllerRemark : ''); $title = $title ? $title : $v; $rulemodel = $this->model->get(['name' => $name]); if (!$rulemodel) { $this->model ->data(['pid' => $pid, 'name' => $name, 'title' => $title, 'icon' => $icon, 'remark' => $remark, 'ismenu' => 1, 'status' => 'normal']) ->isUpdate(false) ->save(); $pid = $this->model->id; } else { $pid = $rulemodel->id; } } $ruleArr = []; foreach ($methods as $m => $n) { //过滤特殊的类 if (substr($n->name, 0, 2) == '__' || $n->name == '_initialize') { continue; } //未启用软删除时过滤相关方法 if (!$withSofeDelete && in_array($n->name, $softDeleteMethods)) { continue; } //只匹配符合的方法 if (!preg_match('/^(\w+)' . Config::get('action_suffix') . '/', $n->name, $matchtwo)) { unset($methods[$m]); continue; } $comment = $reflector->getMethod($n->name)->getDocComment(); //忽略的方法 if (stripos($comment, "@internal") !== false) { continue; } //过滤掉其它字符 $comment = preg_replace(array('/^\/\*\*(.*)[\n\r\t]/u', '/[\s]+\*\//u', '/\*\s@(.*)/u', '/[\s|\*]+/u'), '', $comment); $title = $comment ? $comment : ucfirst($n->name); //获取主键,作为AuthRule更新依据 $id = $this->getAuthRulePK($name . "/" . strtolower($n->name)); $ruleArr[] = array('id' => $id, 'pid' => $pid, 'name' => $name . "/" . strtolower($n->name), 'icon' => 'fa fa-circle-o', 'title' => $title, 'ismenu' => 0, 'status' => 'normal'); } $this->model->isUpdate(false)->saveAll($ruleArr); } //获取主键 protected function getAuthRulePK($name) { if (!empty($name)) { $id = $this->model ->where('name', $name) ->value('id'); return $id ? $id : null; } } }