InstallProjectCommand.php 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373
  1. <?php
  2. namespace catcher\command\install;
  3. use catcher\CatchAdmin;
  4. use catcher\library\InstallLocalModule;
  5. use think\console\Command;
  6. use think\console\Input;
  7. use think\console\input\Option;
  8. use think\console\Output;
  9. use think\facade\Console;
  10. class InstallProjectCommand extends Command
  11. {
  12. protected $databaseLink = [];
  13. protected $defaultModule = ['permissions', 'system'];
  14. protected function configure()
  15. {
  16. $this->setName('catch:install')
  17. ->addOption('reinstall', '-r',Option::VALUE_NONE, 'reinstall back')
  18. ->setDescription('install project');
  19. }
  20. /**
  21. *
  22. * @time 2019年11月29日
  23. * @param Input $input
  24. * @param Output $output
  25. * @return int|void|null
  26. */
  27. protected function execute(Input $input, Output $output)
  28. {
  29. if ($input->getOption('reinstall')) {
  30. $this->reInstall();
  31. $this->project();
  32. } else {
  33. $this->detectionEnvironment();
  34. $this->firstStep();
  35. $this->secondStep();
  36. $this->thirdStep();
  37. $this->finished();
  38. $this->project();
  39. }
  40. }
  41. /**
  42. * 环境检测
  43. *
  44. * @time 2019年11月29日
  45. * @return void
  46. */
  47. protected function detectionEnvironment(): void
  48. {
  49. $this->output->info('environment begin to check...');
  50. if (version_compare(PHP_VERSION, '7.1.0', '<')) {
  51. $this->output->error('php version should >= 7.1.0');
  52. exit();
  53. }
  54. $this->output->info('php version ' . PHP_VERSION);
  55. if (!extension_loaded('mbstring')) {
  56. $this->output->error('mbstring extension not install');exit();
  57. }
  58. $this->output->info('mbstring extension is installed');
  59. if (!extension_loaded('json')) {
  60. $this->output->error('json extension not install');
  61. exit();
  62. }
  63. $this->output->info('json extension is installed');
  64. if (!extension_loaded('openssl')) {
  65. $this->output->error('openssl extension not install');
  66. exit();
  67. }
  68. $this->output->info('openssl extension is installed');
  69. if (!extension_loaded('pdo')) {
  70. $this->output->error('pdo extension not install');
  71. exit();
  72. }
  73. $this->output->info('pdo extension is installed');
  74. if (!extension_loaded('xml')) {
  75. $this->output->error('xml extension not install');
  76. exit();
  77. }
  78. $this->output->info('xml extension is installed');
  79. $this->output->info('🎉 environment checking finished');
  80. }
  81. /**
  82. * 安装第一步
  83. *
  84. * @time 2019年11月29日
  85. * @return mixed
  86. */
  87. protected function firstStep()
  88. {
  89. if (file_exists($this->app->getRootPath() . '.env')) {
  90. return false;
  91. }
  92. // 设置 app domain
  93. $appDomain = strtolower($this->output->ask($this->input, '👉 first, you should set app domain: '));
  94. if (strpos('http://', $appDomain) === false || strpos('https://', $appDomain) === false) {
  95. $appDomain = 'http://' . $appDomain;
  96. }
  97. $answer = strtolower($this->output->ask($this->input, '🤔️ Did You Need to Set Database information? (Y/N): '));
  98. if ($answer === 'y' || $answer === 'yes') {
  99. $charset = $this->output->ask($this->input, '👉 please input database charset, default (utf8mb4):') ? : 'utf8mb4';
  100. $database = '';
  101. while (!$database) {
  102. $database = $this->output->ask($this->input, '👉 please input database name: ');
  103. if ($database) {
  104. break;
  105. }
  106. }
  107. $host = $this->output->ask($this->input, '👉 please input database host, default (127.0.0.1):') ? : '127.0.0.1';
  108. $port = $this->output->ask($this->input, '👉 please input database host port, default (3306):') ? : '3306';
  109. $prefix = $this->output->ask($this->input, '👉 please input table prefix, default (null):') ? : '';
  110. $username = $this->output->ask($this->input, '👉 please input database username default (root): ') ? : 'root';
  111. $password = '';
  112. $tryTimes = 0;
  113. while (!$password) {
  114. $password = $this->output->ask($this->input, '👉 please input database password: ');
  115. if ($password) {
  116. break;
  117. }
  118. // 尝试三次以上未填写,视为密码空
  119. $tryTimes++;
  120. if (!$password && $tryTimes > 2) {
  121. break;
  122. }
  123. }
  124. $this->databaseLink = [$host, $database, $username, $password, $port, $charset, $prefix];
  125. $this->generateEnvFile($host, $database, $username, $password, $port, $charset, $prefix, $appDomain);
  126. }
  127. }
  128. /**
  129. * 安装第二部
  130. *
  131. * @time 2019年11月29日
  132. * @return void
  133. */
  134. protected function secondStep(): void
  135. {
  136. if (file_exists($this->getEnvFilePath())) {
  137. $connections = \config('database.connections');
  138. // 因为 env file 导致安装失败
  139. if (!$this->databaseLink) {
  140. unlink($this->getEnvFilePath());
  141. $this->execute($this->input, $this->output);
  142. } else {
  143. [
  144. $connections['mysql']['hostname'],
  145. $connections['mysql']['database'],
  146. $connections['mysql']['username'],
  147. $connections['mysql']['password'],
  148. $connections['mysql']['hostport'],
  149. $connections['mysql']['charset'],
  150. $connections['mysql']['prefix'],
  151. ] = $this->databaseLink ?: [
  152. env('mysql.hostname')
  153. ];
  154. \config([
  155. 'connections' => $connections,
  156. ], 'database');
  157. $this->migrateAndSeeds();
  158. }
  159. }
  160. }
  161. /**
  162. * 生成表结构
  163. *
  164. * @time 2020年01月20日
  165. * @return void
  166. */
  167. protected function migrateAndSeeds(): void
  168. {
  169. foreach ($this->defaultModule as $m) {
  170. $module = new InstallLocalModule($m);
  171. $module->installModuleTables();
  172. $module->installModuleSeeds();
  173. $this->output->info('🎉 module [' . $m . '] installed successfully');
  174. }
  175. }
  176. /**
  177. * 回滚数据
  178. *
  179. * @time 2020年09月07日
  180. * @return void
  181. */
  182. protected function migrateRollback()
  183. {
  184. foreach ($this->defaultModule as $m) {
  185. $module = new InstallLocalModule($m);
  186. $module->rollbackModuleTable();
  187. $this->output->info('🎉' . $m . ' tables rollback successfully');
  188. }
  189. }
  190. /**
  191. * 安装第四步
  192. *
  193. * @time 2019年11月29日
  194. * @return void
  195. */
  196. protected function thirdStep(): void
  197. {
  198. // Console::call('catch:cache');
  199. }
  200. /**
  201. * finally
  202. *
  203. * @time 2019年11月30日
  204. * @return void
  205. */
  206. protected function finished(): void
  207. {
  208. // todo something
  209. // create jwt
  210. Console::call('jwt:create');
  211. // create service
  212. Console::call('catch-service:discover');
  213. }
  214. /**
  215. * generate env file
  216. *
  217. * @time 2019年11月29日
  218. * @param $host
  219. * @param $database
  220. * @param $username
  221. * @param $password
  222. * @param $port
  223. * @param $charset
  224. * @param $prefix
  225. * @param $appDomain
  226. * @return void
  227. */
  228. protected function generateEnvFile($host, $database, $username, $password, $port, $charset, $prefix, $appDomain): void
  229. {
  230. try {
  231. $env = \parse_ini_file(root_path() . '.example.env', true);
  232. $env['APP']['DOMAIN'] = $appDomain;
  233. $env['DATABASE']['HOSTNAME'] = $host;
  234. $env['DATABASE']['DATABASE'] = $database;
  235. $env['DATABASE']['USERNAME'] = $username;
  236. $env['DATABASE']['PASSWORD'] = $password;
  237. $env['DATABASE']['HOSTPORT'] = $port;
  238. $env['DATABASE']['CHARSET'] = $charset;
  239. if ($prefix) {
  240. $env['DATABASE']['PREFIX'] = $prefix;
  241. }
  242. $dotEnv = '';
  243. foreach ($env as $key => $e) {
  244. if (is_string($e)) {
  245. $dotEnv .= sprintf('%s = %s', $key, $e === '1' ? 'true' : ($e === '' ? 'false' : $e)) . PHP_EOL;
  246. $dotEnv .= PHP_EOL;
  247. } else {
  248. $dotEnv .= sprintf('[%s]', $key) . PHP_EOL;
  249. foreach ($e as $k => $v) {
  250. $dotEnv .= sprintf('%s = %s', $k, $v === '1' ? 'true' : ($v === '' ? 'false' : $v)) . PHP_EOL;
  251. }
  252. $dotEnv .= PHP_EOL;
  253. }
  254. }
  255. if ($this->getEnvFile()) {
  256. $this->output->info('env file has been generated');
  257. }
  258. if ((new \mysqli($host, $username, $password, null, $port))->query(sprintf('CREATE DATABASE IF NOT EXISTS %s DEFAULT CHARSET %s COLLATE %s_general_ci;',
  259. $database, $charset, $charset))) {
  260. $this->output->info(sprintf('🎉 create database %s successfully', $database));
  261. } else {
  262. $this->output->warning(sprintf('create database %s failed,you need create database first by yourself', $database));
  263. }
  264. } catch (\Exception $e) {
  265. $this->output->error($e->getMessage());
  266. exit(0);
  267. }
  268. file_put_contents(root_path() . '.env', $dotEnv);
  269. }
  270. /**
  271. *
  272. * @time 2019年11月29日
  273. * @return string
  274. */
  275. protected function getEnvFile(): string
  276. {
  277. return file_exists(root_path() . '.env') ? root_path() . '.env' : '';
  278. }
  279. protected function project()
  280. {
  281. $year = date('Y');
  282. $this->output->info('🎉 project is installed, welcome!');
  283. $this->output->info(sprintf('
  284. /-------------------- welcome to use -------------------------\
  285. | __ __ ___ __ _ |
  286. | _________ _/ /______/ /_ / | ____/ /___ ___ (_)___ |
  287. | / ___/ __ `/ __/ ___/ __ \ / /| |/ __ / __ `__ \/ / __ \ |
  288. | / /__/ /_/ / /_/ /__/ / / / / ___ / /_/ / / / / / / / / / / |
  289. | \___/\__,_/\__/\___/_/ /_/ /_/ |_\__,_/_/ /_/ /_/_/_/ /_/ |
  290. | |
  291. \ __ __ __ __ _ __ _ __ enjoy it ! _ __ __ __ __ __ __ ___ _ @ 2017 ~ %s
  292. 版本: %s
  293. 初始账号: catch@admin.com
  294. 初始密码: catchadmin
  295. ', $year, CatchAdmin::VERSION));
  296. exit(0);
  297. }
  298. protected function reInstall(): void
  299. {
  300. $ask = strtolower($this->output->ask($this->input,'reset project? (Y/N)'));
  301. if ($ask === 'y' || $ask === 'yes' ) {
  302. $this->migrateRollback();
  303. $this->migrateAndSeeds();
  304. $this->finished();
  305. }
  306. }
  307. /**
  308. * 获取 env path
  309. *
  310. * @time 2020年04月06日
  311. * @return string
  312. */
  313. protected function getEnvFilePath()
  314. {
  315. return root_path() . '.env';
  316. }
  317. }