Start.php 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410
  1. <?php
  2. namespace Proginn;
  3. use Exception;
  4. use Medoo\Medoo;
  5. use Proginn\Config;
  6. use Redis;
  7. class Start
  8. {
  9. const IP_LOCK_KEY = 'docker:ip:';
  10. /**
  11. * 数据库链接
  12. *
  13. * @var Medoo $connection
  14. *
  15. */
  16. protected static $connection;
  17. /**
  18. * 数据库链接
  19. *
  20. * @var Redis $redis
  21. *
  22. */
  23. protected static $redis;
  24. protected $startIP = 180936414; // 10.200.222.222
  25. protected $domainSuffix = '.test.proginn.com';
  26. protected $containersBasePath = '/workspace/containers/';
  27. public function __construct($argv)
  28. {
  29. // 防止git仓库有人改了文件权限,容器里面shell脚本无法执行
  30. system("sudo chmod +x " . ROOT_DIR . '/shell/*');
  31. $params = $this->parseArgs($argv);
  32. $action = $params['action'];
  33. $proginnBranch = $params['proginn-branch'] ?? 'master';
  34. $rooterBranch = $params['rooter-branch'] ?? 'master';
  35. $proginnFrontendBranch = $params['proginn-frontend-branch'] ?? 'master';
  36. $name = $params['name'];
  37. switch ($action) {
  38. case 'start':
  39. $this->start($name, $proginnBranch, $rooterBranch, $proginnFrontendBranch);
  40. break;
  41. case 'remove':
  42. $this->remove($name);
  43. break;
  44. case 'dev':
  45. $this->dev("dev");
  46. break;
  47. case 'nginx':
  48. $this->nginx($name, $proginnBranch, $rooterBranch, $proginnFrontendBranch);//同步nginx配置,不用每次都重新拉取
  49. break;
  50. default:
  51. echo "Usage: php index.php --action=start|remove --name=mydev --proginn-branch=dev --rooter-branch=dev --proginn-frontend-branch dev\n";
  52. break;
  53. }
  54. }
  55. public function dev($name)
  56. {
  57. $this->containersBasePath = '/workspace/projects/';
  58. if (preg_match('/[^a-zA-Z0-9_]+/', $name)) {
  59. echo "创建失败,容器名:{$name}只能字母、数字及下划线组成。\n";
  60. return;
  61. }
  62. $db = $this->getConnection();
  63. $row = $db->get('sites', '*', [
  64. 'name' => $name,
  65. ]);
  66. if (!empty($row)) {
  67. echo "创建失败,容器:{$name}已经存在,请确认。\n";
  68. return;
  69. }
  70. $ip = $this->getIp();
  71. $ipv4 = long2ip($ip);
  72. $domain = $name;
  73. $fullDomain = $name . $this->domainSuffix;
  74. $this->getConnection()->query('start transaction');
  75. try {
  76. $id = $this->getConnection()->insert('sites', [
  77. 'proginn_branch' => "dev",
  78. 'domain' => $domain,
  79. 'ip' => $ip,
  80. 'ipv4' => $ipv4,
  81. 'created_at' => time(),
  82. 'name' => $name,
  83. 'rooter_branch' => "dev",
  84. 'proginn_frontend_branch' => "dev",
  85. 'full_domain' => $fullDomain,
  86. ]);
  87. if (!$id) {
  88. throw new Exception('创建数据库记录失败');
  89. }
  90. // 容器存储目录
  91. $directory = $this->containersBasePath;
  92. // 创建日志相关目录
  93. if(!file_exists($directory . '/log/pm2')) mkdir($directory . '/log/pm2', 0777, true); //
  94. if(!file_exists($directory . '/log/proginn_cache')) mkdir($directory . '/log/proginn_cache', 0777, true); //
  95. // 配置存放目录
  96. if(!file_exists($directory . '/config')) mkdir($directory . '/config', 0777, true); //
  97. // yml 文件
  98. $template = file_get_contents(ROOT_DIR . '/dockerfile/template.yml');
  99. $template = str_replace('containers/<containerName>', "project", $template);
  100. $template = str_replace('<ip>', $ipv4, $template);
  101. $template = str_replace('<proginn-branch>', "dev", $template);
  102. $template = str_replace('<rooter-branch>', "dev", $template);
  103. $template = str_replace('<proginn-frontend-branch>', "dev", $template);
  104. $template = str_replace('<domain>', $domain, $template);
  105. $template = str_replace('<fullDomain>', $fullDomain, $template);
  106. $template = str_replace('<containerName>', "dev", $template);
  107. file_put_contents($directory . 'config/docker.yml', $template);
  108. // nginx 配置
  109. $proxy = file_get_contents(ROOT_DIR . '/config/nginx/template/proxy.nginx.conf');
  110. $proxy = str_replace('{{domain}}', $domain, $proxy);
  111. $proxy = str_replace('{{ip}}', $ipv4, $proxy);
  112. file_put_contents("/workspace/commonContainers/nginx/conf.d/{$name}.conf", $proxy);
  113. $proginn = file_get_contents(ROOT_DIR . '/config/nginx/template/proginn.nginx.conf');
  114. $proginn = str_replace('{{domain}}', $domain, $proginn);
  115. $proginn = str_replace('{{ip}}', $ipv4, $proginn);
  116. file_put_contents("{$directory}/config/nginx.conf", $proginn);
  117. $log = file_get_contents(ROOT_DIR . '/config/logrotate/log.conf');
  118. $log = str_replace('{{containerName}}', $name, $log);
  119. file_put_contents($directory . 'config/logrotate.conf', $log);
  120. system("sudo cp -f {$directory}config/logrotate.conf /etc/logrotate.d/{$name}");
  121. // 写入项目变量 Found orphan containers
  122. file_put_contents($directory . 'config/.env', "COMPOSE_PROJECT_NAME={$name}");
  123. // 启动容器
  124. system("docker-compose -p {$name} -f {$directory}config/docker.yml up -d");
  125. // 重载Nginx
  126. system("docker exec nginx nginx -s reload");
  127. $this->getConnection()->query('commit');
  128. } catch (\Throwable $e) {
  129. echo "exception:" . $e->__toString() . "\n";
  130. $this->getConnection()->query('rollback');
  131. if (file_exists($directory . 'config/docker.yml')) {
  132. system("docker-compose -p {$name} -f {$directory}config/docker.yml rm -s -f");
  133. }
  134. $redis = $this->getRedis();
  135. $lockKey = static::IP_LOCK_KEY . $ip;
  136. $redis->del($lockKey);
  137. // 容器存储目录
  138. $directory = $this->containersBasePath . $name;
  139. system("sudo rm -rf {$directory}");
  140. system("rm -f /workspace/commonContainers/nginx/conf.d/{$name}.conf");
  141. system("docker exec nginx nginx -s reload");
  142. system("sudo rm -f /etc/logrotate.d/{$name}");
  143. }
  144. }
  145. public function nginx($name, $proginnBranch, $rooterBranch, $proginnFrontendBranch)
  146. {
  147. if (preg_match('/[^a-zA-Z0-9_]+/', $name)) {
  148. echo "创建失败,容器名:{$name}只能字母、数字及下划线组成。\n";
  149. return;
  150. }
  151. if (preg_match('/[^a-zA-Z0-9_]+/', $proginnBranch)) {
  152. echo "创建失败,proginn分支名:{$name}只能字母、数字及下划线组成。\n";
  153. return;
  154. }
  155. if (preg_match('/[^a-zA-Z0-9_]+/', $rooterBranch)) {
  156. echo "创建失败,rooter分支名:{$name}只能字母、数字及下划线组成。\n";
  157. return;
  158. }
  159. if (preg_match('/[^a-zA-Z0-9_]+/', $proginnFrontendBranch)) {
  160. echo "创建失败,proginn-frontend分支名:{$name}只能字母、数字及下划线组成。\n";
  161. return;
  162. }
  163. $db = $this->getConnection();
  164. $row = $db->get('sites', '*', [
  165. 'name' => $name,
  166. ]);
  167. if (empty($row)) {
  168. echo "获取失败,容器:{$name}不存在,请确认。\n";
  169. return;
  170. }
  171. $ip = $this->getIp();
  172. $ipv4 = long2ip($ip);
  173. $domain = $name;
  174. $this->getConnection()->query('start transaction');
  175. try {
  176. // 容器存储目录
  177. $directory = $this->containersBasePath . $name;
  178. // nginx 配置
  179. $proginn = file_get_contents(ROOT_DIR . '/config/nginx/template/proginn.nginx.conf');
  180. $proginn = str_replace('{{domain}}', $domain, $proginn);
  181. $proginn = str_replace('{{ip}}', $ipv4, $proginn);
  182. file_put_contents("{$directory}/config/nginx.conf", $proginn);
  183. // 重载Nginx
  184. system("docker exec nginx nginx -s reload");
  185. $this->getConnection()->query('commit');
  186. } catch (\Throwable $e) {
  187. echo "exception:" . $e->__toString() . "\n";
  188. $this->getConnection()->query('rollback');
  189. }
  190. }
  191. public function start($name, $proginnBranch, $rooterBranch, $proginnFrontendBranch)
  192. {
  193. if (preg_match('/[^a-zA-Z0-9_]+/', $name)) {
  194. echo "创建失败,容器名:{$name}只能字母、数字及下划线组成。\n";
  195. return;
  196. }
  197. if (preg_match('/[^a-zA-Z0-9_]+/', $proginnBranch)) {
  198. echo "创建失败,proginn分支名:{$name}只能字母、数字及下划线组成。\n";
  199. return;
  200. }
  201. if (preg_match('/[^a-zA-Z0-9_]+/', $rooterBranch)) {
  202. echo "创建失败,rooter分支名:{$name}只能字母、数字及下划线组成。\n";
  203. return;
  204. }
  205. if (preg_match('/[^a-zA-Z0-9_]+/', $proginnFrontendBranch)) {
  206. echo "创建失败,proginn-frontend分支名:{$name}只能字母、数字及下划线组成。\n";
  207. return;
  208. }
  209. $db = $this->getConnection();
  210. $row = $db->get('sites', '*', [
  211. 'name' => $name,
  212. ]);
  213. if (!empty($row)) {
  214. echo "创建失败,容器:{$name}已经存在,请确认。\n";
  215. return;
  216. }
  217. $ip = $this->getIp();
  218. $ipv4 = long2ip($ip);
  219. $domain = $name;
  220. $fullDomain = $name . $this->domainSuffix;
  221. $this->getConnection()->query('start transaction');
  222. try {
  223. $id = $this->getConnection()->insert('sites', [
  224. 'proginn_branch' => $proginnBranch,
  225. 'domain' => $domain,
  226. 'ip' => $ip,
  227. 'ipv4' => $ipv4,
  228. 'created_at' => time(),
  229. 'name' => $name,
  230. 'rooter_branch' => $rooterBranch,
  231. 'proginn_frontend_branch' => $proginnFrontendBranch,
  232. 'full_domain' => $fullDomain,
  233. ]);
  234. if (!$id) {
  235. throw new Exception('创建数据库记录失败');
  236. }
  237. // 容器存储目录
  238. $directory = $this->containersBasePath . $name;
  239. // 创建容器存储目录
  240. mkdir($directory, 0777, true);
  241. // 拷贝项目
  242. system("cp -Rf /workspace/projects/proginn {$directory}");
  243. system("cp -Rf /workspace/projects/proginn-frontend {$directory}");
  244. system("cp -Rf /workspace/projects/boss {$directory}");
  245. system("cp -Rf /workspace/projects/festival {$directory}");
  246. system("cp -Rf /workspace/projects/docker-test {$directory}");
  247. system("cp -Rf /workspace/projects/proginn-user {$directory}");
  248. system("cp -Rf /workspace/projects/proginn-bituni {$directory}");
  249. // 创建日志相关目录
  250. mkdir($directory . '/log/pm2', 0777, true); //
  251. mkdir($directory . '/log/proginn_cache', 0777, true); //
  252. // 配置存放目录
  253. mkdir($directory . '/config', 0777, true); //
  254. // yml 文件
  255. $template = file_get_contents(ROOT_DIR . '/dockerfile/template.yml');
  256. $template = str_replace('<containerName>', $name, $template);
  257. $template = str_replace('<ip>', $ipv4, $template);
  258. $template = str_replace('<proginn-branch>', $proginnBranch, $template);
  259. $template = str_replace('<rooter-branch>', $rooterBranch, $template);
  260. $template = str_replace('<proginn-frontend-branch>', $proginnFrontendBranch, $template);
  261. $template = str_replace('<domain>', $domain, $template);
  262. $template = str_replace('<fullDomain>', $fullDomain, $template);
  263. file_put_contents($directory . '/config/docker.yml', $template);
  264. // nginx 配置
  265. $proxy = file_get_contents(ROOT_DIR . '/config/nginx/template/proxy.nginx.conf');
  266. $proxy = str_replace('{{domain}}', $domain, $proxy);
  267. $proxy = str_replace('{{ip}}', $ipv4, $proxy);
  268. file_put_contents("/workspace/commonContainers/nginx/conf.d/{$name}.conf", $proxy);
  269. $proginn = file_get_contents(ROOT_DIR . '/config/nginx/template/proginn.nginx.conf');
  270. $proginn = str_replace('{{domain}}', $domain, $proginn);
  271. $proginn = str_replace('{{ip}}', $ipv4, $proginn);
  272. file_put_contents("{$directory}/config/nginx.conf", $proginn);
  273. $log = file_get_contents(ROOT_DIR . '/config/logrotate/log.conf');
  274. $log = str_replace('{{containerName}}', $name, $log);
  275. file_put_contents($directory . '/config/logrotate.conf', $log);
  276. system("sudo cp -f {$directory}/config/logrotate.conf /etc/logrotate.d/{$name}");
  277. // 写入项目变量 Found orphan containers
  278. file_put_contents($directory . '/config/.env', "COMPOSE_PROJECT_NAME={$name}");
  279. // 启动容器
  280. system("docker-compose -p {$name} -f {$directory}/config/docker.yml up -d");
  281. // 重载Nginx
  282. system("docker exec nginx nginx -s reload");
  283. $this->getConnection()->query('commit');
  284. } catch (\Throwable $e) {
  285. echo "exception:" . $e->__toString() . "\n";
  286. $this->getConnection()->query('rollback');
  287. if (file_exists($directory . '/config/docker.yml')) {
  288. system("docker-compose -p {$name} -f {$directory}/config/docker.yml rm -s -f");
  289. }
  290. $redis = $this->getRedis();
  291. $lockKey = static::IP_LOCK_KEY . $ip;
  292. $redis->del($lockKey);
  293. // 容器存储目录
  294. $directory = $this->containersBasePath . $name;
  295. system("sudo rm -rf {$directory}");
  296. system("rm -f /workspace/commonContainers/nginx/conf.d/{$name}.conf");
  297. system("docker exec nginx nginx -s reload");
  298. system("sudo rm -f /etc/logrotate.d/{$name}");
  299. }
  300. }
  301. protected function remove($name)
  302. {
  303. $db = $this->getConnection();
  304. $row = $db->get('sites', '*', [
  305. 'name' => $name,
  306. ]);
  307. if (empty($row)) {
  308. echo "删除失败,容器:{$name}不存在,请确认。\n";
  309. return;
  310. }
  311. $this->getConnection()->query('start transaction');
  312. try {
  313. // 容器存储目录
  314. $directory = $this->containersBasePath . $name;
  315. $db->delete('sites', [
  316. 'name' => $name,
  317. ]);
  318. if (file_exists($directory . '/config/docker.yml')) {
  319. system("docker-compose -p {$name} -f {$directory}/config/docker.yml rm -s -f");
  320. }
  321. // 容器存储目录
  322. $directory = $this->containersBasePath . $name;
  323. system("sudo rm -rf {$directory}");
  324. system("rm -f /workspace/commonContainers/nginx/conf.d/{$name}.conf");
  325. system("docker exec nginx nginx -s reload");
  326. $redis = $this->getRedis();
  327. $lockKey = static::IP_LOCK_KEY . $row['ip'];
  328. $redis->del($lockKey);
  329. $this->getConnection()->query('commit');
  330. } catch (\Throwable $e) {
  331. echo "exception:" . $e->__toString() . "\n";
  332. $this->getConnection()->query('rollback');
  333. }
  334. }
  335. protected function getIp()
  336. {
  337. $ip = $this->startIP;
  338. $redis = $this->getRedis();
  339. while (true) {
  340. $ip++;
  341. $lockKey = static::IP_LOCK_KEY . $ip;
  342. $lock = $redis->setnx($lockKey, $ip);
  343. if ($lock) {
  344. break;
  345. }
  346. }
  347. return $ip;
  348. }
  349. protected function parseArgs($argv)
  350. {
  351. unset($argv[0]);
  352. $params = [];
  353. foreach ($argv as $i => $arg) {
  354. $arg = explode('=', $arg);
  355. $key = trim($arg[0], "-");
  356. $val = $arg[1] ?? true;
  357. $params[$key] = $val;
  358. }
  359. return $params;
  360. }
  361. protected function getRedis()
  362. {
  363. if (static::$redis === null) {
  364. static::$redis = new Redis();
  365. static::$redis->connect(Config::REDIS_HOST, Config::REDIS_PORT, 3);
  366. static::$redis->auth(Config::REDIS_PASS);
  367. }
  368. return static::$redis;
  369. }
  370. /**
  371. * @return Medoo
  372. */
  373. protected function getConnection()
  374. {
  375. if (static::$connection === null) {
  376. static::$connection = new Medoo([
  377. 'database_type' => 'mysql',
  378. 'database_name' => Config::DB_NAME,
  379. 'server' => Config::DB_HOST,
  380. 'username' => Config::DB_USER,
  381. 'password' => Config::DB_PASS,
  382. 'charset' => Config::DB_CHAR,
  383. 'port' => Config::DB_PORT,
  384. 'prefix' => '',
  385. 'logging' => true,
  386. ]);
  387. }
  388. return static::$connection;
  389. }
  390. }