332 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			332 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
<?php
 | 
						||
 | 
						||
namespace app\admin\manystore_command;
 | 
						||
 | 
						||
use app\manystore\model\ManystoreAuthRule;
 | 
						||
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 ManystoreAuthRule();
 | 
						||
 | 
						||
        $manystorePath = dirname(__DIR__) . DS;
 | 
						||
        $manystorePathArray = explode(DS,$manystorePath);
 | 
						||
        $manystorePathArray[count($manystorePathArray) - 2] = 'manystore';
 | 
						||
        $manystorePath = implode(DS,$manystorePathArray);
 | 
						||
        //控制器名
 | 
						||
        $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(defined('STDIN') ? STDIN : fopen('php://stdin', 'r'));
 | 
						||
                if (trim($line) != 'yes') {
 | 
						||
                    throw new Exception("Operation is aborted!");
 | 
						||
                }
 | 
						||
            }
 | 
						||
            ManystoreAuthRule::destroy($ids);
 | 
						||
 | 
						||
            Cache::rm("__manystore_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)];
 | 
						||
                }
 | 
						||
                $manystorePath = $manystorePath . 'controller' . DS  . implode(DS, $controllerArr) . '.php';
 | 
						||
                if (!is_file($manystorePath)) {
 | 
						||
                    $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 = $manystorePath . 'controller' . DS;
 | 
						||
            // 扫描新的节点信息并导入
 | 
						||
            $treelist = $this->import($this->scandir($controllerDir));
 | 
						||
        }
 | 
						||
        Cache::rm("__manystore_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\\manystore\\" . implode("\\", $controllerArr) . $classSuffix;
 | 
						||
 | 
						||
        $pathArr = $controllerArr;
 | 
						||
        array_unshift($pathArr, '', 'application', 'manystore', '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\\manystore_command\\" . $uniqueName;
 | 
						||
 | 
						||
        //删除临时文件
 | 
						||
        register_shutdown_function(function () use ($tempClassFile) {
 | 
						||
            if ($tempClassFile) {
 | 
						||
                //删除临时文件
 | 
						||
                @unlink($tempClassFile);
 | 
						||
            }
 | 
						||
        });
 | 
						||
 | 
						||
        //反射机制调用类的注释和方法名
 | 
						||
        $reflector = new ReflectionClass($className);
 | 
						||
 | 
						||
        //只匹配公共的方法
 | 
						||
        $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;
 | 
						||
        }
 | 
						||
    }
 | 
						||
}
 |