Api.php 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202
  1. <?php
  2. namespace app\admin\command;
  3. use app\admin\command\Api\library\Builder;
  4. use think\Config;
  5. use think\console\Command;
  6. use think\console\Input;
  7. use think\console\input\Option;
  8. use think\console\Output;
  9. use think\Exception;
  10. class Api extends Command
  11. {
  12. protected function configure()
  13. {
  14. $site = Config::get('site');
  15. $this
  16. ->setName('api')
  17. ->addOption('url', 'u', Option::VALUE_OPTIONAL, 'default api url', '')
  18. ->addOption('module', 'm', Option::VALUE_OPTIONAL, 'module name(admin/index/api)', 'api')
  19. ->addOption('output', 'o', Option::VALUE_OPTIONAL, 'output index file name', 'api.html')
  20. ->addOption('template', 'e', Option::VALUE_OPTIONAL, '', 'index.html')
  21. ->addOption('force', 'f', Option::VALUE_OPTIONAL, 'force override general file', false)
  22. ->addOption('title', 't', Option::VALUE_OPTIONAL, 'document title', $site['name'])
  23. ->addOption('class', 'c', Option::VALUE_OPTIONAL | Option::VALUE_IS_ARRAY, 'extend class', null)
  24. ->addOption('language', 'l', Option::VALUE_OPTIONAL, 'language', 'zh-cn')
  25. ->addOption('addon', 'a', Option::VALUE_OPTIONAL, 'addon name', null)
  26. ->addOption('controller', 'r', Option::VALUE_REQUIRED | Option::VALUE_IS_ARRAY, 'controller name', null)
  27. ->setDescription('Build Api document from controller');
  28. }
  29. protected function execute(Input $input, Output $output)
  30. {
  31. $apiDir = __DIR__ . DS . 'Api' . DS;
  32. $force = $input->getOption('force');
  33. $url = $input->getOption('url');
  34. $language = $input->getOption('language');
  35. $language = $language ? $language : 'zh-cn';
  36. $langFile = $apiDir . 'lang' . DS . $language . '.php';
  37. if (!is_file($langFile)) {
  38. throw new Exception('language file not found');
  39. }
  40. $lang = include_once $langFile;
  41. // 目标目录
  42. $output_dir = ROOT_PATH . 'public' . DS;
  43. $output_file = $output_dir . $input->getOption('output');
  44. if (is_file($output_file) && !$force) {
  45. throw new Exception("api index file already exists!\nIf you need to rebuild again, use the parameter --force=true ");
  46. }
  47. // 模板文件
  48. $template_dir = $apiDir . 'template' . DS;
  49. $template_file = $template_dir . $input->getOption('template');
  50. if (!is_file($template_file)) {
  51. throw new Exception('template file not found');
  52. }
  53. // 额外的类
  54. $classes = $input->getOption('class');
  55. // 标题
  56. $title = $input->getOption('title');
  57. // 模块
  58. $module = $input->getOption('module');
  59. // 插件
  60. $addon = $input->getOption('addon');
  61. $moduleDir = $addonDir = '';
  62. if ($addon) {
  63. $addonInfo = get_addon_info($addon);
  64. if (!$addonInfo) {
  65. throw new Exception('addon not found');
  66. }
  67. $moduleDir = ADDON_PATH . $addon . DS;
  68. } else {
  69. $moduleDir = APP_PATH . $module . DS;
  70. }
  71. if (!is_dir($moduleDir)) {
  72. throw new Exception('module not found');
  73. }
  74. if (version_compare(PHP_VERSION, '7.0.0', '<')) {
  75. if (extension_loaded('Zend OPcache')) {
  76. $configuration = opcache_get_configuration();
  77. $directives = $configuration['directives'];
  78. $configName = request()->isCli() ? 'opcache.enable_cli' : 'opcache.enable';
  79. if (!$directives[$configName]) {
  80. throw new Exception("Please make sure {$configName} is turned on, Get help:https://forum.fastadmin.net/d/1321");
  81. }
  82. } else {
  83. throw new Exception("Please make sure opcache already enabled, Get help:https://forum.fastadmin.net/d/1321");
  84. }
  85. }
  86. //控制器名
  87. $controller = $input->getOption('controller') ?: [];
  88. if (!$controller) {
  89. $controllerDir = $moduleDir . Config::get('url_controller_layer') . DS;
  90. $files = new \RecursiveIteratorIterator(
  91. new \RecursiveDirectoryIterator($controllerDir),
  92. \RecursiveIteratorIterator::LEAVES_ONLY
  93. );
  94. foreach ($files as $name => $file) {
  95. if (!$file->isDir() && $file->getExtension() == 'php') {
  96. $filePath = $file->getRealPath();
  97. $classes[] = $this->get_class_from_file($filePath);
  98. }
  99. }
  100. } else {
  101. foreach ($controller as $index => $item) {
  102. $filePath = $moduleDir . Config::get('url_controller_layer') . DS . $item . '.php';
  103. $classes[] = $this->get_class_from_file($filePath);
  104. }
  105. }
  106. $classes = array_unique(array_filter($classes));
  107. $config = [
  108. 'sitename' => config('site.name'),
  109. 'title' => $title,
  110. 'author' => config('site.name'),
  111. 'description' => '',
  112. 'apiurl' => $url,
  113. 'language' => $language,
  114. ];
  115. try {
  116. $builder = new Builder($classes);
  117. $content = $builder->render($template_file, ['config' => $config, 'lang' => $lang]);
  118. } catch (\Exception $e) {
  119. print_r($e);
  120. }
  121. if (!file_put_contents($output_file, $content)) {
  122. throw new Exception('Cannot save the content to ' . $output_file);
  123. }
  124. $output->info("Build Successed!");
  125. }
  126. /**
  127. * get full qualified class name
  128. *
  129. * @param string $path_to_file
  130. * @return string
  131. * @author JBYRNE http://jarretbyrne.com/2015/06/197/
  132. */
  133. protected function get_class_from_file($path_to_file)
  134. {
  135. //Grab the contents of the file
  136. $contents = file_get_contents($path_to_file);
  137. //Start with a blank namespace and class
  138. $namespace = $class = "";
  139. //Set helper values to know that we have found the namespace/class token and need to collect the string values after them
  140. $getting_namespace = $getting_class = false;
  141. //Go through each token and evaluate it as necessary
  142. foreach (token_get_all($contents) as $token) {
  143. //If this token is the namespace declaring, then flag that the next tokens will be the namespace name
  144. if (is_array($token) && $token[0] == T_NAMESPACE) {
  145. $getting_namespace = true;
  146. }
  147. //If this token is the class declaring, then flag that the next tokens will be the class name
  148. if (is_array($token) && $token[0] == T_CLASS) {
  149. $getting_class = true;
  150. }
  151. //While we're grabbing the namespace name...
  152. if ($getting_namespace === true) {
  153. //If the token is a string or the namespace separator...
  154. if (is_array($token) && in_array($token[0], [T_STRING, T_NS_SEPARATOR])) {
  155. //Append the token's value to the name of the namespace
  156. $namespace .= $token[1];
  157. } elseif ($token === ';') {
  158. //If the token is the semicolon, then we're done with the namespace declaration
  159. $getting_namespace = false;
  160. }
  161. }
  162. //While we're grabbing the class name...
  163. if ($getting_class === true) {
  164. //If the token is a string, it's the name of the class
  165. if (is_array($token) && $token[0] == T_STRING) {
  166. //Store the token's value as the class name
  167. $class = $token[1];
  168. //Got what we need, stope here
  169. break;
  170. }
  171. }
  172. }
  173. //Build the fully-qualified class name and return it
  174. return $namespace ? $namespace . '\\' . $class : $class;
  175. }
  176. }