CatchAdmin.php 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456
  1. <?php
  2. declare(strict_types=1);
  3. namespace catcher;
  4. use think\helper\Arr;
  5. class CatchAdmin
  6. {
  7. public static $root = 'catch';
  8. public const VERSION = '2.1.0';
  9. /**
  10. *
  11. * @time 2019年11月30日
  12. * @return string
  13. */
  14. public static function directory(): string
  15. {
  16. return app()->getRootPath() . self::$root . DIRECTORY_SEPARATOR;
  17. }
  18. /**
  19. * 创建目录
  20. *
  21. * @time 2019年12月16日
  22. * @param string $directory
  23. * @return string
  24. */
  25. public static function makeDirectory(string $directory): string
  26. {
  27. if (!is_dir($directory) && !mkdir($directory, 0777, true) && !is_dir($directory)) {
  28. throw new \RuntimeException(sprintf('Directory "%s" was not created', $directory));
  29. }
  30. return $directory;
  31. }
  32. /**
  33. *
  34. * @time 2019年12月04日
  35. * @param $module
  36. * @return string
  37. */
  38. public static function moduleDirectory($module): string
  39. {
  40. return self::makeDirectory(self::directory() . $module . DIRECTORY_SEPARATOR);
  41. }
  42. /**
  43. *
  44. * @time 2019年11月30日
  45. * @return string
  46. */
  47. public static function cacheDirectory(): string
  48. {
  49. return self::makeDirectory(app()->getRuntimePath() . self::$root . DIRECTORY_SEPARATOR);
  50. }
  51. /**
  52. * 备份地址
  53. *
  54. * @time 2019年12月13日
  55. * @return string
  56. */
  57. public static function backupDirectory(): string
  58. {
  59. return self::makeDirectory(self::cacheDirectory() . 'backup' .DIRECTORY_SEPARATOR);
  60. }
  61. /**
  62. *
  63. * @time 2019年12月03日
  64. * @param $module
  65. * @return string
  66. */
  67. public static function moduleMigrationsDirectory($module): string
  68. {
  69. return self::directory() . $module . DIRECTORY_SEPARATOR . 'database' . DIRECTORY_SEPARATOR. 'migrations' . DIRECTORY_SEPARATOR;
  70. }
  71. /**
  72. *
  73. * @time 2019年12月03日
  74. * @param $module
  75. * @return string
  76. */
  77. public static function moduleSeedsDirectory($module): string
  78. {
  79. $seedPath = self::directory() . $module . DIRECTORY_SEPARATOR . 'database' . DIRECTORY_SEPARATOR. 'seeds' . DIRECTORY_SEPARATOR;
  80. self::makeDirectory($seedPath);
  81. return $seedPath;
  82. }
  83. /**
  84. * 获取模块 view path
  85. *
  86. * @time 2019年12月03日
  87. * @param $module
  88. * @return string
  89. */
  90. public static function getModuleViewPath($module): string
  91. {
  92. return self::makeDirectory(self::directory() . $module . DIRECTORY_SEPARATOR . 'view' . DIRECTORY_SEPARATOR);
  93. }
  94. /**
  95. *
  96. * @time 2019年12月03日
  97. * @param $module
  98. * @return string
  99. */
  100. public static function getModuleModelDirectory($module): string
  101. {
  102. return self::makeDirectory(self::directory() . $module . DIRECTORY_SEPARATOR . 'model' . DIRECTORY_SEPARATOR);
  103. }
  104. /**
  105. *
  106. * @time 2019年11月30日
  107. * @return array
  108. */
  109. public static function getModulesDirectory(): array
  110. {
  111. $modules = glob(self::directory() . '*');
  112. foreach ($modules as $key => &$module) {
  113. if (!is_dir($module)) {
  114. unset($modules[$key]);
  115. }
  116. $module .= DIRECTORY_SEPARATOR;
  117. }
  118. return $modules;
  119. }
  120. /**
  121. *
  122. * @time 2019年12月12日
  123. * @param bool $select
  124. * @return array
  125. */
  126. public static function getModulesInfo($select = true): array
  127. {
  128. $modules = [];
  129. if ($select) {
  130. foreach (self::getModulesDirectory() as $module) {
  131. $moduleInfo = self::getModuleInfo($module);
  132. $modules[] = [
  133. 'value' => $moduleInfo['alias'],
  134. 'title' => $moduleInfo['name'],
  135. ];
  136. }
  137. } else {
  138. foreach (self::getModulesDirectory() as $module) {
  139. $moduleInfo = self::getModuleInfo($module);
  140. $modules[$moduleInfo['alias']] = $moduleInfo['name'];
  141. }
  142. }
  143. return $modules;
  144. }
  145. /**
  146. *
  147. * @time 2019年11月30日
  148. * @return array
  149. */
  150. protected static function getModuleServices(): array
  151. {
  152. $services = [];
  153. foreach (self::getModulesDirectory() as $module) {
  154. if (is_dir($module)) {
  155. $moduleInfo = self::getModuleInfo($module);
  156. if (isset($moduleInfo['services']) && !empty($moduleInfo['services'])) {
  157. $services = array_merge($services, $moduleInfo['services']);
  158. }
  159. }
  160. }
  161. return $services;
  162. }
  163. /**
  164. * 获取可用模块
  165. *
  166. * @time 2020年06月23日
  167. * @return array
  168. */
  169. public static function getEnabledService()
  170. {
  171. $services = [];
  172. foreach (self::getModulesDirectory() as $module) {
  173. if (is_dir($module)) {
  174. $moduleInfo = self::getModuleInfo($module);
  175. // 如果没有设置 module.json 默认加载
  176. $moduleServices = $moduleInfo['services'] ?? [];
  177. if (!empty($moduleServices) && $moduleInfo['enable']) {
  178. $services = array_merge($services, $moduleServices);
  179. }
  180. }
  181. }
  182. return $services;
  183. }
  184. /**
  185. * 开启模块
  186. *
  187. * @time 2020年06月23日
  188. * @param $module
  189. * @return bool
  190. */
  191. public static function enableModule($module)
  192. {
  193. $moduleJson = self::moduleDirectory($module) . 'module.json';
  194. if (!file_exists($moduleJson)) {
  195. return true;
  196. }
  197. $info = \json_decode(file_get_contents($moduleJson), true);
  198. $info['enable'] = true;
  199. if (!is_writeable($moduleJson)) {
  200. chmod($moduleJson, 666);
  201. }
  202. file_put_contents($moduleJson, \json_encode($info, JSON_PRETTY_PRINT | JSON_UNESCAPED_UNICODE));
  203. return true;
  204. }
  205. /**
  206. * 关闭模块
  207. *
  208. * @time 2020年06月23日
  209. * @param $module
  210. * @return bool
  211. */
  212. public static function disableModule($module)
  213. {
  214. $moduleJson = self::moduleDirectory($module) . 'module.json';
  215. if (!file_exists($moduleJson)) {
  216. return true;
  217. }
  218. $info = \json_decode(file_get_contents($moduleJson), true);
  219. $info['enable'] = false;
  220. if (!is_writeable($moduleJson)) {
  221. chmod($moduleJson, 666);
  222. }
  223. file_put_contents($moduleJson, \json_encode($info, JSON_PRETTY_PRINT|JSON_UNESCAPED_UNICODE));
  224. return true;
  225. }
  226. /**
  227. *
  228. * @time 2019年11月30日
  229. * @return array
  230. */
  231. protected static function getModuleViews(): array
  232. {
  233. $views = [];
  234. foreach (self::getModulesDirectory() as $module) {
  235. if (is_dir($module . 'view')) {
  236. $moduleInfo = self::getModuleInfo($module);
  237. $moduleName = $moduleInfo['alias'] ?? Arr::last(explode('/', $module));
  238. $views[$moduleName] = $module . 'view' . DIRECTORY_SEPARATOR;
  239. }
  240. }
  241. return $views;
  242. }
  243. /**
  244. * 获取模块信息
  245. *
  246. * @time 2019年11月30日
  247. * @param $module
  248. * @return mixed
  249. */
  250. public static function getModuleInfo($module)
  251. {
  252. if (file_exists($module . DIRECTORY_SEPARATOR . 'module.json')) {
  253. return \json_decode(file_get_contents($module. DIRECTORY_SEPARATOR . 'module.json'), true);
  254. }
  255. return [];
  256. }
  257. /**
  258. * 获取服务
  259. *
  260. * @time 2019年11月30日
  261. * @return array
  262. */
  263. public static function getServices(): array
  264. {
  265. if (file_exists(self::getCacheServicesFile())) {
  266. return self::getCacheServices();
  267. }
  268. return self::getModuleServices();
  269. }
  270. /**
  271. *
  272. * @time 2019年11月30日
  273. * @return mixed
  274. */
  275. public static function getRoutes()
  276. {
  277. if (file_exists(self::getCacheRoutesFile())) {
  278. return [self::getCacheRoutesFile()];
  279. }
  280. return self::getModuleRoutes();
  281. }
  282. /**
  283. *
  284. * @time 2019年11月30日
  285. * @return array|mixed
  286. */
  287. public static function getViews()
  288. {
  289. if (file_exists(self::getCacheViewsFile())) {
  290. return self::getCacheViews();
  291. }
  292. return self::getModuleViews();
  293. }
  294. /**
  295. *
  296. * @time 2019年12月15日
  297. * @return array
  298. */
  299. public static function getModuleRoutes(): array
  300. {
  301. $routeFiles = [];
  302. foreach (self::getModulesDirectory() as $module) {
  303. $moduleInfo = self::getModuleInfo($module);
  304. $moduleAlias = $moduleInfo['alias'] ?? '';
  305. if (!in_array($moduleAlias, ['login']) && file_exists($module . 'route.php')) {
  306. $routeFiles[] = $module . 'route.php';
  307. }
  308. }
  309. return $routeFiles;
  310. }
  311. /**
  312. *
  313. * @time 2019年11月30日
  314. * @return false|int
  315. */
  316. public static function cacheRoutes()
  317. {
  318. $routes = '';
  319. foreach (self::getModuleRoutes() as $route) {
  320. $routes .= trim(str_replace('<?php', '', file_get_contents($route))) . PHP_EOL;
  321. }
  322. return file_put_contents(self::getCacheRoutesFile(), "<?php\r\n " . $routes);
  323. }
  324. /**
  325. *
  326. * @time 2019年11月30日
  327. * @return false|int
  328. */
  329. public static function cacheServices()
  330. {
  331. return file_put_contents(self::getCacheServicesFile(), "<?php\r\n return "
  332. . var_export(self::getEnabledService(), true) . ';');
  333. }
  334. /**
  335. *
  336. * @time 2019年11月30日
  337. * @return false|int
  338. */
  339. public static function cacheViews()
  340. {
  341. return file_put_contents(self::getCacheViewsFile(), "<?php\r\n return "
  342. . var_export(self::getModuleViews(), true) . ';');
  343. }
  344. /**
  345. *
  346. * @time 2019年11月30日
  347. * @return mixed
  348. */
  349. protected static function getCacheViews()
  350. {
  351. return include self::getCacheViewsFile();
  352. }
  353. /**
  354. *
  355. * @time 2019年11月30日
  356. * @return mixed
  357. */
  358. protected static function getCacheServices()
  359. {
  360. return include self::getCacheServicesFile();
  361. }
  362. /**
  363. *
  364. * @time 2019年11月30日
  365. * @return mixed
  366. */
  367. protected static function getCacheViewsFile()
  368. {
  369. return self::cacheDirectory() . 'views.php';
  370. }
  371. /**
  372. *
  373. * @time 2019年11月30日
  374. * @return mixed
  375. */
  376. public static function getCacheServicesFile()
  377. {
  378. return self::cacheDirectory() . 'services.php';
  379. }
  380. /**
  381. *
  382. * @time 2019年11月30日
  383. * @return string
  384. */
  385. protected static function getCacheRoutesFile(): string
  386. {
  387. return self::cacheDirectory() . 'routes.php';
  388. }
  389. }