控制台应用,yzncms本身基于tp5.1框架,里面的队列用不了,bug,坑
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

Index.php 12KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | Yzncms [ 御宅男工作室 ]
  4. // +----------------------------------------------------------------------
  5. // | Copyright (c) 2018 http://yzncms.com All rights reserved.
  6. // +----------------------------------------------------------------------
  7. // | Licensed ( http://www.apache.org/licenses/LICENSE-2.0 )
  8. // +----------------------------------------------------------------------
  9. // | Author: 御宅男 <530765310@qq.com>
  10. // +----------------------------------------------------------------------
  11. // +----------------------------------------------------------------------
  12. // | 接口管理
  13. // +----------------------------------------------------------------------
  14. namespace addons\apidoc\controller;
  15. use think\addons\Controller;
  16. class Index extends Controller
  17. {
  18. protected $config = [
  19. 'title' => 'APi接口文档',
  20. 'copyright' => 'Powered By YznCMS',
  21. 'controllers' => [
  22. ],
  23. 'versions' => [
  24. ],
  25. 'groups' => [],
  26. 'with_cache' => false,
  27. 'responses' => '{
  28. "code":"状态码",
  29. "message":"操作描述",
  30. "data":"业务数据",
  31. "timestamp":"响应时间戳"
  32. }',
  33. 'global_auth_key' => "Authorization",
  34. 'auth' => [
  35. 'with_auth' => false,
  36. 'auth_password' => "123456",
  37. 'headers_key' => "apidocToken",
  38. ],
  39. 'definitions' => "hg\apidoc\Definitions",
  40. 'filter_method' => [
  41. '_empty',
  42. ],
  43. ];
  44. public function index()
  45. {
  46. return $this->fetch();
  47. }
  48. /**
  49. * 获取配置
  50. * @return array
  51. */
  52. public function getConfig()
  53. {
  54. $config = config('apidoc') ? config('apidoc') : config('apidoc.');
  55. $this->config = array_merge($this->config, $config);
  56. if (!empty($this->config['auth'])) {
  57. $this->config['auth'] = [
  58. 'with_auth' => $this->config['auth']['with_auth'],
  59. // 验证类型,password=密码验证,只在进入时做密码验证
  60. 'headers_key' => $this->config['auth']['headers_key'],
  61. ];
  62. }
  63. return json($this->config);
  64. }
  65. /**
  66. * 验证身份
  67. */
  68. public function verifyAuth()
  69. {
  70. $config = config('apidoc') ? config('apidoc') : config('apidoc.');
  71. $this->config = array_merge($this->config, $config);
  72. $request = Request::instance();
  73. $params = $request->param();
  74. if ($this->config['auth']['with_auth'] === true) {
  75. // 密码验证
  76. if (md5($this->config['auth']['auth_password']) === $params['password']) {
  77. $token = md5($params['password'] . strtotime(date('Y-m-d', time())));
  78. return json(array("token" => $token));
  79. } else {
  80. throw new \think\Exception("密码不正确,请重新输入");
  81. }
  82. }
  83. return json($params);
  84. }
  85. public function verifyToken()
  86. {
  87. if (!empty($this->config['auth'])) {
  88. if ($this->config['auth']['with_auth'] === true) {
  89. $token = $this->request->header($this->config['auth']['headers_key']);
  90. if ($token === md5(md5($this->config['auth']['auth_password']) . strtotime(date('Y-m-d', time())))) {
  91. return true;
  92. } else {
  93. throw new \think\exception\HttpException(401, "身份令牌已过期,请重新登录");
  94. }
  95. }
  96. }
  97. return true;
  98. }
  99. /**
  100. * 获取接口列表
  101. * @return array
  102. */
  103. public function getList()
  104. {
  105. $config = config('apidoc') ? config('apidoc') : config('apidoc.');
  106. $this->config = array_merge($this->config, $config);
  107. // 验证token身份
  108. if ($this->config['auth']['with_auth'] === true) {
  109. $tokenRes = $this->verifyToken();
  110. }
  111. $params = $this->request->param();
  112. $version = "";
  113. if (!empty($params) && !empty($params['version'])) {
  114. $version = $params['version'];
  115. }
  116. $cacheFiles = [];
  117. $cacheName = "";
  118. if ($this->config['with_cache']) {
  119. // 获取缓存数据
  120. $cachePath = "../runtime/apidoc/" . $version;
  121. if (file_exists($cachePath) && $params['reload'] == 'false') {
  122. $cacheFilePath = "";
  123. $filePaths = glob($cachePath . '/*.json');
  124. if (count($filePaths) > 0) {
  125. $cacheFilePath = $filePaths[count($filePaths) - 1];
  126. }
  127. if (!empty($params) && !empty($params['cacheFileName'])) {
  128. // 前端传入的缓存文件名
  129. $cacheFileName = $params['cacheFileName'];
  130. $cacheFilePath = $cachePath . "/" . $cacheFileName . '.json';
  131. }
  132. if ($cacheFilePath && file_exists($cacheFilePath)) {
  133. $fileContent = file_get_contents($cacheFilePath);
  134. if (!empty($fileContent)) {
  135. $fileJson = json_decode($fileContent);
  136. $list = $fileJson;
  137. $cacheName = str_replace(".json", "", basename($cacheFilePath));
  138. } else {
  139. $list = $this->getApiList($version);
  140. }
  141. } else {
  142. // 不存在缓存文件,生成数据并存缓存
  143. $list = $this->getApiList($version);
  144. // 生成缓存数据
  145. $cacheName = $this->createJsonFile($list, $version);
  146. }
  147. } else {
  148. // 不存在缓存文件,生成数据并存缓存
  149. $list = $this->getApiList($version);
  150. // 生成缓存数据
  151. $cacheName = $this->createJsonFile($list, $version);
  152. }
  153. $filePaths = glob($cachePath . '/*.json');
  154. if (count($filePaths) > 0) {
  155. foreach ($filePaths as $item) {
  156. $cacheFiles[] = str_replace(".json", "", basename($item));
  157. }
  158. }
  159. } else {
  160. $list = $this->getApiList($version);
  161. }
  162. if (isset($this->config['groups']) && count($this->config['groups']) > 0) {
  163. array_unshift($this->config['groups'], ['title' => '全部', 'name' => 0]);
  164. }
  165. $data = array(
  166. "title" => $this->config['title'],
  167. "version" => $version,
  168. "copyright" => $this->config['copyright'],
  169. "responses" => $this->config['responses'],
  170. "list" => $list,
  171. "cacheFiles" => $cacheFiles,
  172. "cacheName" => $cacheName,
  173. "groups" => $this->config['groups'],
  174. );
  175. $res = [
  176. 'code' => 0,
  177. 'data' => $data,
  178. ];
  179. return json($res);
  180. }
  181. /**
  182. * 获取api接口文档
  183. */
  184. public function getApiList($version)
  185. {
  186. $config = config('apidoc') ? config('apidoc') : config('apidoc.');
  187. $this->config = array_merge($this->config, $config);
  188. $list = [];
  189. $controllers = $this->config['controllers'];
  190. $versionPath = "";
  191. if (!empty($version)) {
  192. foreach ($this->config['versions'] as $item) {
  193. if ($item['title'] == $version && !empty($item['folder'])) {
  194. $versionPath = $item['folder'] . "\\";
  195. }
  196. }
  197. }
  198. foreach ($controllers as $k => $class) {
  199. $class = "app\\" . $versionPath . $class;
  200. if (class_exists($class)) {
  201. $reflection = new \ReflectionClass($class);
  202. $doc_str = $reflection->getDocComment();
  203. $doc = new \addons\apidoc\library\Parser($this->config);
  204. // 解析控制器类的注释
  205. $class_doc = $doc->parseClass($doc_str);
  206. // 获取当前控制器Class的所有方法
  207. $method = $reflection->getMethods(\ReflectionMethod::IS_PUBLIC);
  208. $filter_method = array_merge(['__construct'], $this->config['filter_method']);
  209. $actions = [];
  210. foreach ($method as $j => $action) {
  211. // 过滤不解析的方法
  212. if (!in_array($action->name, $filter_method)) {
  213. // 获取当前方法的注释
  214. $actionDoc = new \addons\apidoc\library\Parser($this->config);
  215. $actionDocStr = $action->getDocComment();
  216. if ($actionDocStr) {
  217. // 解析当前方法的注释
  218. $action_doc = $actionDoc->parseAction($actionDocStr);
  219. // $action_doc['name'] = $class."::".$action->name;
  220. $action_doc['id'] = $k . "-" . $j;
  221. // // 解析方法
  222. $actions[] = $action_doc;
  223. }
  224. }
  225. }
  226. $class_doc['children'] = $actions;
  227. $class_doc['id'] = $k . "";
  228. if (empty($class_doc['title']) && empty($class_doc['controller'])) {
  229. $class_doc['title'] = $controllers[$k];
  230. }
  231. $list[] = $class_doc;
  232. }
  233. }
  234. return $list;
  235. }
  236. /**
  237. * 获取文件夹内的所有文件
  238. * @param string $class
  239. * @param string $action
  240. *
  241. * @return array|bool
  242. */
  243. protected function listDirFiles($app, $isapp = true)
  244. {
  245. $arr = [];
  246. $base = base_path();
  247. if ($isapp) {
  248. $dir = $base . $app;
  249. } else {
  250. $dir = $app;
  251. }
  252. if (is_dir($dir)) {
  253. //如果是目录,则进行下一步操作
  254. $d = opendir($dir); //打开目录
  255. if ($d) {
  256. //目录打开正常
  257. while (($file = readdir($d)) !== false) {
  258. //循环读出目录下的文件,直到读不到为止
  259. if ($file != '.' && $file != '..') {
  260. //排除一个点和两个点
  261. if (is_dir($dir . '/' . $file)) { //如果当前是目录
  262. $arr = array_merge($arr, self::listDirFiles($dir . '/' . $file, false)); //进一步获取该目录里的文件
  263. } else {
  264. if (pathinfo($dir . '/' . $file)['extension'] == 'php') {
  265. $arr[] = str_replace([$base, '/', '.php'], ['', '\\', ''], $dir . '/' . $file); //进一步获取该目录里的文件
  266. }
  267. }
  268. }
  269. }
  270. }
  271. closedir($d); //关闭句柄
  272. }
  273. asort($arr);
  274. return $arr;
  275. }
  276. /**
  277. * 创建接口参数缓存文件
  278. * @param $json
  279. * @param $version
  280. * @return bool|false|string
  281. */
  282. protected function createJsonFile($json, $version)
  283. {
  284. if (empty($json)) {
  285. return false;
  286. }
  287. $fileName = date("Y-m-d H_i_s");
  288. $fileJson = $json;
  289. $fileContent = json_encode($fileJson);
  290. $dir = "../runtime/apidoc/" . $version;
  291. $path = $dir . "/" . $fileName . ".json";
  292. //判断文件夹是否存在
  293. if (!file_exists($dir)) {
  294. mkdir($dir, 0777, true);
  295. }
  296. $myfile = fopen($path, "w") or die("Unable to open file!");
  297. fwrite($myfile, $fileContent);
  298. fclose($myfile);
  299. return $fileName;
  300. }
  301. }