@underpostnet/underpost 2.7.9

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (297) hide show
  1. package/.dockerignore +14 -0
  2. package/.env.development +6 -0
  3. package/.env.production +6 -0
  4. package/.env.test +6 -0
  5. package/.github/workflows/ghpkg.yml +115 -0
  6. package/.github/workflows/publish.yml +84 -0
  7. package/.github/workflows/pwa-microservices-template.page.yml +54 -0
  8. package/.github/workflows/pwa-microservices-template.test.yml +30 -0
  9. package/.nycrc +9 -0
  10. package/.prettierignore +13 -0
  11. package/.prettierrc +9 -0
  12. package/.vscode/extensions.json +72 -0
  13. package/.vscode/settings.json +106 -0
  14. package/AUTHORS.md +10 -0
  15. package/CHANGELOG.md +85 -0
  16. package/Dockerfile +90 -0
  17. package/LICENSE +21 -0
  18. package/README.md +96 -0
  19. package/bin/cron.js +47 -0
  20. package/bin/db.js +180 -0
  21. package/bin/deploy.js +859 -0
  22. package/bin/file.js +109 -0
  23. package/bin/index.js +82 -0
  24. package/bin/ssl.js +55 -0
  25. package/bin/util.js +204 -0
  26. package/bin/vs.js +35 -0
  27. package/conf.js +265 -0
  28. package/docker-compose.yml +67 -0
  29. package/jsconfig.json +7 -0
  30. package/jsdoc.json +32 -0
  31. package/nodemon.json +6 -0
  32. package/package.json +134 -0
  33. package/prometheus.yml +36 -0
  34. package/src/api/core/core.controller.js +69 -0
  35. package/src/api/core/core.model.js +11 -0
  36. package/src/api/core/core.router.js +23 -0
  37. package/src/api/core/core.service.js +31 -0
  38. package/src/api/crypto/crypto.controller.js +51 -0
  39. package/src/api/crypto/crypto.model.js +23 -0
  40. package/src/api/crypto/crypto.router.js +20 -0
  41. package/src/api/crypto/crypto.service.js +64 -0
  42. package/src/api/default/default.controller.js +69 -0
  43. package/src/api/default/default.model.js +20 -0
  44. package/src/api/default/default.router.js +23 -0
  45. package/src/api/default/default.service.js +31 -0
  46. package/src/api/file/file.controller.js +53 -0
  47. package/src/api/file/file.model.js +19 -0
  48. package/src/api/file/file.router.js +21 -0
  49. package/src/api/file/file.service.js +80 -0
  50. package/src/api/instance/instance.controller.js +69 -0
  51. package/src/api/instance/instance.model.js +36 -0
  52. package/src/api/instance/instance.router.js +33 -0
  53. package/src/api/instance/instance.service.js +48 -0
  54. package/src/api/test/test.controller.js +59 -0
  55. package/src/api/test/test.model.js +14 -0
  56. package/src/api/test/test.router.js +21 -0
  57. package/src/api/test/test.service.js +35 -0
  58. package/src/api/user/user.build.js +16 -0
  59. package/src/api/user/user.controller.js +70 -0
  60. package/src/api/user/user.model.js +65 -0
  61. package/src/api/user/user.router.js +345 -0
  62. package/src/api/user/user.service.js +479 -0
  63. package/src/api.js +23 -0
  64. package/src/client/Default.index.js +40 -0
  65. package/src/client/components/core/Account.js +290 -0
  66. package/src/client/components/core/AgGrid.js +160 -0
  67. package/src/client/components/core/Auth.js +19 -0
  68. package/src/client/components/core/Badge.js +32 -0
  69. package/src/client/components/core/Blockchain.js +41 -0
  70. package/src/client/components/core/Blog.js +9 -0
  71. package/src/client/components/core/BtnIcon.js +101 -0
  72. package/src/client/components/core/CalendarCore.js +458 -0
  73. package/src/client/components/core/Chat.js +64 -0
  74. package/src/client/components/core/ColorPalette.js +5267 -0
  75. package/src/client/components/core/CommonJs.js +742 -0
  76. package/src/client/components/core/Content.js +193 -0
  77. package/src/client/components/core/Css.js +846 -0
  78. package/src/client/components/core/CssCore.js +844 -0
  79. package/src/client/components/core/D3Chart.js +44 -0
  80. package/src/client/components/core/Docs.js +331 -0
  81. package/src/client/components/core/DropDown.js +164 -0
  82. package/src/client/components/core/EventsUI.js +46 -0
  83. package/src/client/components/core/FileExplorer.js +699 -0
  84. package/src/client/components/core/FullScreen.js +45 -0
  85. package/src/client/components/core/Input.js +346 -0
  86. package/src/client/components/core/JoyStick.js +77 -0
  87. package/src/client/components/core/Keyboard.js +73 -0
  88. package/src/client/components/core/LoadingAnimation.js +178 -0
  89. package/src/client/components/core/LogIn.js +187 -0
  90. package/src/client/components/core/LogOut.js +58 -0
  91. package/src/client/components/core/Logger.js +26 -0
  92. package/src/client/components/core/Modal.js +1814 -0
  93. package/src/client/components/core/NotificationManager.js +84 -0
  94. package/src/client/components/core/Panel.js +613 -0
  95. package/src/client/components/core/PanelForm.js +469 -0
  96. package/src/client/components/core/Polyhedron.js +162 -0
  97. package/src/client/components/core/Recover.js +204 -0
  98. package/src/client/components/core/Responsive.js +68 -0
  99. package/src/client/components/core/RichText.js +53 -0
  100. package/src/client/components/core/Router.js +76 -0
  101. package/src/client/components/core/Scroll.js +34 -0
  102. package/src/client/components/core/SignUp.js +125 -0
  103. package/src/client/components/core/SocketIo.js +72 -0
  104. package/src/client/components/core/Stream.js +113 -0
  105. package/src/client/components/core/ToggleSwitch.js +87 -0
  106. package/src/client/components/core/ToolTip.js +26 -0
  107. package/src/client/components/core/Translate.js +446 -0
  108. package/src/client/components/core/Validator.js +100 -0
  109. package/src/client/components/core/VanillaJs.js +463 -0
  110. package/src/client/components/core/Wallet.js +106 -0
  111. package/src/client/components/core/WebComponent.js +44 -0
  112. package/src/client/components/core/Webhook.js +25 -0
  113. package/src/client/components/core/Worker.js +280 -0
  114. package/src/client/components/default/CommonDefault.js +29 -0
  115. package/src/client/components/default/CssDefault.js +13 -0
  116. package/src/client/components/default/ElementsDefault.js +38 -0
  117. package/src/client/components/default/LogInDefault.js +41 -0
  118. package/src/client/components/default/LogOutDefault.js +28 -0
  119. package/src/client/components/default/MenuDefault.js +389 -0
  120. package/src/client/components/default/RoutesDefault.js +48 -0
  121. package/src/client/components/default/SettingsDefault.js +16 -0
  122. package/src/client/components/default/SignUpDefault.js +9 -0
  123. package/src/client/components/default/SocketIoDefault.js +54 -0
  124. package/src/client/components/default/TranslateDefault.js +7 -0
  125. package/src/client/public/default/android-chrome-144x144.png +0 -0
  126. package/src/client/public/default/android-chrome-192x192.png +0 -0
  127. package/src/client/public/default/android-chrome-256x256.png +0 -0
  128. package/src/client/public/default/android-chrome-36x36.png +0 -0
  129. package/src/client/public/default/android-chrome-384x384.png +0 -0
  130. package/src/client/public/default/android-chrome-48x48.png +0 -0
  131. package/src/client/public/default/android-chrome-512x512.png +0 -0
  132. package/src/client/public/default/android-chrome-72x72.png +0 -0
  133. package/src/client/public/default/android-chrome-96x96.png +0 -0
  134. package/src/client/public/default/apple-touch-icon-1024x1024.png +0 -0
  135. package/src/client/public/default/apple-touch-icon-114x114.png +0 -0
  136. package/src/client/public/default/apple-touch-icon-120x120.png +0 -0
  137. package/src/client/public/default/apple-touch-icon-144x144.png +0 -0
  138. package/src/client/public/default/apple-touch-icon-152x152.png +0 -0
  139. package/src/client/public/default/apple-touch-icon-167x167.png +0 -0
  140. package/src/client/public/default/apple-touch-icon-180x180.png +0 -0
  141. package/src/client/public/default/apple-touch-icon-57x57.png +0 -0
  142. package/src/client/public/default/apple-touch-icon-60x60.png +0 -0
  143. package/src/client/public/default/apple-touch-icon-72x72.png +0 -0
  144. package/src/client/public/default/apple-touch-icon-76x76.png +0 -0
  145. package/src/client/public/default/apple-touch-icon-precomposed.png +0 -0
  146. package/src/client/public/default/apple-touch-icon.png +0 -0
  147. package/src/client/public/default/apple-touch-startup-image-1125x2436.png +0 -0
  148. package/src/client/public/default/apple-touch-startup-image-1136x640.png +0 -0
  149. package/src/client/public/default/apple-touch-startup-image-1170x2532.png +0 -0
  150. package/src/client/public/default/apple-touch-startup-image-1179x2556.png +0 -0
  151. package/src/client/public/default/apple-touch-startup-image-1242x2208.png +0 -0
  152. package/src/client/public/default/apple-touch-startup-image-1242x2688.png +0 -0
  153. package/src/client/public/default/apple-touch-startup-image-1284x2778.png +0 -0
  154. package/src/client/public/default/apple-touch-startup-image-1290x2796.png +0 -0
  155. package/src/client/public/default/apple-touch-startup-image-1334x750.png +0 -0
  156. package/src/client/public/default/apple-touch-startup-image-1488x2266.png +0 -0
  157. package/src/client/public/default/apple-touch-startup-image-1536x2048.png +0 -0
  158. package/src/client/public/default/apple-touch-startup-image-1620x2160.png +0 -0
  159. package/src/client/public/default/apple-touch-startup-image-1640x2160.png +0 -0
  160. package/src/client/public/default/apple-touch-startup-image-1668x2224.png +0 -0
  161. package/src/client/public/default/apple-touch-startup-image-1668x2388.png +0 -0
  162. package/src/client/public/default/apple-touch-startup-image-1792x828.png +0 -0
  163. package/src/client/public/default/apple-touch-startup-image-2048x1536.png +0 -0
  164. package/src/client/public/default/apple-touch-startup-image-2048x2732.png +0 -0
  165. package/src/client/public/default/apple-touch-startup-image-2160x1620.png +0 -0
  166. package/src/client/public/default/apple-touch-startup-image-2160x1640.png +0 -0
  167. package/src/client/public/default/apple-touch-startup-image-2208x1242.png +0 -0
  168. package/src/client/public/default/apple-touch-startup-image-2224x1668.png +0 -0
  169. package/src/client/public/default/apple-touch-startup-image-2266x1488.png +0 -0
  170. package/src/client/public/default/apple-touch-startup-image-2388x1668.png +0 -0
  171. package/src/client/public/default/apple-touch-startup-image-2436x1125.png +0 -0
  172. package/src/client/public/default/apple-touch-startup-image-2532x1170.png +0 -0
  173. package/src/client/public/default/apple-touch-startup-image-2556x1179.png +0 -0
  174. package/src/client/public/default/apple-touch-startup-image-2688x1242.png +0 -0
  175. package/src/client/public/default/apple-touch-startup-image-2732x2048.png +0 -0
  176. package/src/client/public/default/apple-touch-startup-image-2778x1284.png +0 -0
  177. package/src/client/public/default/apple-touch-startup-image-2796x1290.png +0 -0
  178. package/src/client/public/default/apple-touch-startup-image-640x1136.png +0 -0
  179. package/src/client/public/default/apple-touch-startup-image-750x1334.png +0 -0
  180. package/src/client/public/default/apple-touch-startup-image-828x1792.png +0 -0
  181. package/src/client/public/default/assets/background/white.jpg +0 -0
  182. package/src/client/public/default/assets/background/white0-min.jpg +0 -0
  183. package/src/client/public/default/assets/background/white0.jpg +0 -0
  184. package/src/client/public/default/assets/logo/base-icon.png +0 -0
  185. package/src/client/public/default/assets/mailer/api-user-check.png +0 -0
  186. package/src/client/public/default/assets/mailer/api-user-invalid-token.png +0 -0
  187. package/src/client/public/default/assets/mailer/api-user-recover.png +0 -0
  188. package/src/client/public/default/browserconfig.xml +12 -0
  189. package/src/client/public/default/favicon-16x16.png +0 -0
  190. package/src/client/public/default/favicon-32x32.png +0 -0
  191. package/src/client/public/default/favicon-48x48.png +0 -0
  192. package/src/client/public/default/favicon.ico +0 -0
  193. package/src/client/public/default/manifest.webmanifest +69 -0
  194. package/src/client/public/default/mstile-144x144.png +0 -0
  195. package/src/client/public/default/mstile-150x150.png +0 -0
  196. package/src/client/public/default/mstile-310x150.png +0 -0
  197. package/src/client/public/default/mstile-310x310.png +0 -0
  198. package/src/client/public/default/mstile-70x70.png +0 -0
  199. package/src/client/public/default/plantuml/client-conf.svg +1 -0
  200. package/src/client/public/default/plantuml/client-schema.svg +1 -0
  201. package/src/client/public/default/plantuml/cron-conf.svg +1 -0
  202. package/src/client/public/default/plantuml/cron-schema.svg +1 -0
  203. package/src/client/public/default/plantuml/server-conf.svg +1 -0
  204. package/src/client/public/default/plantuml/server-schema.svg +1 -0
  205. package/src/client/public/default/plantuml/ssr-conf.svg +1 -0
  206. package/src/client/public/default/plantuml/ssr-schema.svg +1 -0
  207. package/src/client/public/default/site.webmanifest +69 -0
  208. package/src/client/public/default/sitemap +148 -0
  209. package/src/client/public/default/yandex-browser-50x50.png +0 -0
  210. package/src/client/public/default/yandex-browser-manifest.json +9 -0
  211. package/src/client/public/doc/favicon.ico +0 -0
  212. package/src/client/public/doc/sitemap +148 -0
  213. package/src/client/public/test/favicon.ico +0 -0
  214. package/src/client/public/test/sitemap +148 -0
  215. package/src/client/services/core/core.service.js +170 -0
  216. package/src/client/services/crypto/crypto.service.js +70 -0
  217. package/src/client/services/default/default.management.js +343 -0
  218. package/src/client/services/default/default.service.js +89 -0
  219. package/src/client/services/file/file.service.js +70 -0
  220. package/src/client/services/instance/instance.management.js +74 -0
  221. package/src/client/services/instance/instance.service.js +89 -0
  222. package/src/client/services/test/test.service.js +70 -0
  223. package/src/client/services/user/user.management.js +50 -0
  224. package/src/client/services/user/user.service.js +89 -0
  225. package/src/client/ssr/Render.js +237 -0
  226. package/src/client/ssr/common/Alert.js +75 -0
  227. package/src/client/ssr/common/SsrCore.js +91 -0
  228. package/src/client/ssr/common/Translate.js +26 -0
  229. package/src/client/ssr/common/Worker.js +28 -0
  230. package/src/client/ssr/components/body/CacheControl.js +114 -0
  231. package/src/client/ssr/components/body/DefaultSplashScreen.js +90 -0
  232. package/src/client/ssr/components/email/DefaultRecoverEmail.js +21 -0
  233. package/src/client/ssr/components/email/DefaultVerifyEmail.js +17 -0
  234. package/src/client/ssr/components/head/Css.js +241 -0
  235. package/src/client/ssr/components/head/DefaultScripts.js +3 -0
  236. package/src/client/ssr/components/head/Production.js +1 -0
  237. package/src/client/ssr/components/head/Pwa.js +146 -0
  238. package/src/client/ssr/components/head/PwaDefault.js +60 -0
  239. package/src/client/ssr/components/head/Seo.js +14 -0
  240. package/src/client/ssr/pages/404.js +12 -0
  241. package/src/client/ssr/pages/500.js +12 -0
  242. package/src/client/ssr/pages/maintenance.js +14 -0
  243. package/src/client/ssr/pages/offline.js +21 -0
  244. package/src/client/sw/default.sw.js +205 -0
  245. package/src/client/sw/template.sw.js +84 -0
  246. package/src/client.build.js +22 -0
  247. package/src/client.dev.js +21 -0
  248. package/src/db/DataBaseProvider.js +45 -0
  249. package/src/db/mariadb/MariaDB.js +33 -0
  250. package/src/db/mongo/MongooseDB.js +124 -0
  251. package/src/dns.js +22 -0
  252. package/src/index.js +43 -0
  253. package/src/mailer/EmailRender.js +69 -0
  254. package/src/mailer/MailerProvider.js +96 -0
  255. package/src/proxy.js +22 -0
  256. package/src/runtime/lampp/Lampp.js +115 -0
  257. package/src/runtime/nginx/Nginx.js +3 -0
  258. package/src/runtime/xampp/Xampp.js +49 -0
  259. package/src/server/auth.js +235 -0
  260. package/src/server/backup.js +120 -0
  261. package/src/server/client-build-live.js +98 -0
  262. package/src/server/client-build.js +754 -0
  263. package/src/server/client-dev-server.js +60 -0
  264. package/src/server/client-formatted.js +58 -0
  265. package/src/server/client-icons.js +151 -0
  266. package/src/server/conf.js +929 -0
  267. package/src/server/crypto.js +91 -0
  268. package/src/server/dns.js +118 -0
  269. package/src/server/downloader.js +42 -0
  270. package/src/server/logger.js +190 -0
  271. package/src/server/network.js +209 -0
  272. package/src/server/peer.js +33 -0
  273. package/src/server/process.js +66 -0
  274. package/src/server/prompt-optimizer.js +28 -0
  275. package/src/server/proxy.js +118 -0
  276. package/src/server/runtime.js +463 -0
  277. package/src/server/ssl.js +120 -0
  278. package/src/server.js +25 -0
  279. package/src/ws/IoInterface.js +45 -0
  280. package/src/ws/IoServer.js +39 -0
  281. package/src/ws/core/channels/core.ws.chat.js +23 -0
  282. package/src/ws/core/channels/core.ws.mailer.js +35 -0
  283. package/src/ws/core/channels/core.ws.stream.js +31 -0
  284. package/src/ws/core/core.ws.connection.js +28 -0
  285. package/src/ws/core/core.ws.emit.js +14 -0
  286. package/src/ws/core/core.ws.server.js +24 -0
  287. package/src/ws/core/management/core.ws.chat.js +8 -0
  288. package/src/ws/core/management/core.ws.mailer.js +16 -0
  289. package/src/ws/core/management/core.ws.stream.js +8 -0
  290. package/src/ws/default/channels/default.ws.main.js +16 -0
  291. package/src/ws/default/default.ws.connection.js +22 -0
  292. package/src/ws/default/default.ws.emit.js +14 -0
  293. package/src/ws/default/default.ws.server.js +20 -0
  294. package/src/ws/default/management/default.ws.main.js +8 -0
  295. package/startup.js +11 -0
  296. package/supervisord-openssh-server.conf +5 -0
  297. package/test/api.test.js +60 -0
@@ -0,0 +1,929 @@
1
+ import fs from 'fs-extra';
2
+ import dotenv from 'dotenv';
3
+ import { cap, capFirst, getCapVariableName, newInstance, range, timer } from '../client/components/core/CommonJs.js';
4
+ import * as dir from 'path';
5
+ import cliProgress from 'cli-progress';
6
+ import cliSpinners from 'cli-spinners';
7
+ import logUpdate from 'log-update';
8
+ import colors from 'colors';
9
+ import { loggerFactory } from './logger.js';
10
+ import { shellExec, shellCd, getRootDirectory } from './process.js';
11
+ import { DefaultConf } from '../../conf.js';
12
+ import ncp from 'copy-paste';
13
+ import read from 'read';
14
+ import splitFile from 'split-file';
15
+ import axios from 'axios';
16
+ import https from 'https';
17
+
18
+ // axios.defaults.baseURL = BASE_URL;
19
+
20
+ const httpsAgent = new https.Agent({
21
+ rejectUnauthorized: false,
22
+ });
23
+
24
+ axios.defaults.httpsAgent = httpsAgent;
25
+
26
+ colors.enable();
27
+ dotenv.config();
28
+
29
+ const logger = loggerFactory(import.meta);
30
+
31
+ // monitoring: https://app.pm2.io/
32
+
33
+ const Config = {
34
+ default: DefaultConf,
35
+ build: async function (options = { folder: '' }) {
36
+ if (!fs.existsSync(`./tmp`)) fs.mkdirSync(`./tmp`, { recursive: true });
37
+ fs.writeFileSync(`./tmp/await-deploy`, '', 'utf8');
38
+ if (fs.existsSync(`./engine-private/conf/${process.argv[2]}`)) return loadConf(process.argv[2]);
39
+ if (fs.existsSync(`./engine-private/replica/${process.argv[2]}`)) return loadConf(process.argv[2]);
40
+
41
+ if (process.argv[2] === 'deploy') return;
42
+
43
+ if (process.argv[2] === 'proxy') {
44
+ this.default.server = {};
45
+ for (const deployId of process.argv[3].split(',')) {
46
+ let confPath = `./engine-private/conf/${deployId}/conf.server.json`;
47
+ const privateConfDevPath = fs.existsSync(`./engine-private/replica/${deployId}/conf.server.json`)
48
+ ? `./engine-private/replica/${deployId}/conf.server.json`
49
+ : `./engine-private/conf/${deployId}/conf.server.dev.${process.argv[4]}.json`;
50
+ const confDevPath = fs.existsSync(privateConfDevPath)
51
+ ? privateConfDevPath
52
+ : `./engine-private/conf/${deployId}/conf.server.dev.json`;
53
+
54
+ if (process.env.NODE_ENV === 'development' && fs.existsSync(confDevPath)) confPath = confDevPath;
55
+ const serverConf = JSON.parse(fs.readFileSync(confPath, 'utf8'));
56
+
57
+ for (const host of Object.keys(loadReplicas(serverConf))) {
58
+ if (serverConf[host]['/'])
59
+ this.default.server[host] = {
60
+ ...this.default.server[host],
61
+ ...serverConf[host],
62
+ };
63
+ else
64
+ this.default.server[host] = {
65
+ ...serverConf[host],
66
+ ...this.default.server[host],
67
+ };
68
+ }
69
+ }
70
+ }
71
+ if (!options || !options.folder)
72
+ options = {
73
+ ...options,
74
+ folder: `./conf`,
75
+ };
76
+ if (!fs.existsSync(options.folder)) fs.mkdirSync(options.folder, { recursive: true });
77
+ for (const confType of Object.keys(this.default)) {
78
+ fs.writeFileSync(
79
+ `${options.folder}/conf.${confType}.json`,
80
+ JSON.stringify(this.default[confType], null, 4),
81
+ 'utf8',
82
+ );
83
+ }
84
+ },
85
+ };
86
+
87
+ const loadConf = (deployId) => {
88
+ const folder = fs.existsSync(`./engine-private/replica/${deployId}`)
89
+ ? `./engine-private/replica/${deployId}`
90
+ : `./engine-private/conf/${deployId}`;
91
+ if (!fs.existsSync(`./conf`)) fs.mkdirSync(`./conf`);
92
+ if (!fs.existsSync(`./tmp`)) fs.mkdirSync(`./tmp`, { recursive: true });
93
+ const isValidDeployId = fs.existsSync(`${folder}`);
94
+ if (!isValidDeployId) {
95
+ logger.info(`Save new deploy conf: '${deployId}'`);
96
+ shellExec(`node bin/deploy save ${deployId}`);
97
+ return loadConf(deployId);
98
+ }
99
+ for (const typeConf of Object.keys(Config.default)) {
100
+ let srcConf = isValidDeployId
101
+ ? fs.readFileSync(`${folder}/conf.${typeConf}.json`, 'utf8')
102
+ : JSON.stringify(Config.default[typeConf]);
103
+ if (process.env.NODE_ENV === 'development' && typeConf === 'server') {
104
+ const devConfPath = `${folder}/conf.${typeConf}.dev${process.argv[3] ? `.${process.argv[3]}` : ''}.json`;
105
+ if (fs.existsSync(devConfPath)) srcConf = fs.readFileSync(devConfPath, 'utf8');
106
+ }
107
+ if (typeConf === 'server') srcConf = JSON.stringify(loadReplicas(JSON.parse(srcConf)), null, 4);
108
+ fs.writeFileSync(`./conf/conf.${typeConf}.json`, srcConf, 'utf8');
109
+ }
110
+ if (!isValidDeployId) return {};
111
+ fs.writeFileSync(`./.env.production`, fs.readFileSync(`${folder}/.env.production`, 'utf8'), 'utf8');
112
+ fs.writeFileSync(`./.env.development`, fs.readFileSync(`${folder}/.env.development`, 'utf8'), 'utf8');
113
+ fs.writeFileSync(`./.env.test`, fs.readFileSync(`${folder}/.env.test`, 'utf8'), 'utf8');
114
+ if (process.env.NODE_ENV) {
115
+ fs.writeFileSync(`./.env`, fs.readFileSync(`${folder}/.env.${process.env.NODE_ENV}`, 'utf8'), 'utf8');
116
+ const env = dotenv.parse(fs.readFileSync(`${folder}/.env.${process.env.NODE_ENV}`, 'utf8'));
117
+ process.env = {
118
+ ...process.env,
119
+ ...env,
120
+ };
121
+ }
122
+ fs.writeFileSync(`./package.json`, fs.readFileSync(`${folder}/package.json`, 'utf8'), 'utf8');
123
+ return { folder, deployId };
124
+ };
125
+
126
+ const loadReplicas = (confServer) => {
127
+ for (const host of Object.keys(confServer)) {
128
+ for (const path of Object.keys(confServer[host])) {
129
+ const { replicas, singleReplica } = confServer[host][path];
130
+ if (replicas && (process.argv[2] === 'proxy' || !singleReplica))
131
+ for (const replicaPath of replicas) {
132
+ confServer[host][replicaPath] = newInstance(confServer[host][path]);
133
+ delete confServer[host][replicaPath].replicas;
134
+ }
135
+ }
136
+ }
137
+ return confServer;
138
+ };
139
+
140
+ const cloneConf = async (
141
+ { toOptions, fromOptions },
142
+ fromDefaultOptions = { deployId: 'dd-default', clientId: 'default' },
143
+ ) => {
144
+ if (!fromOptions.deployId) fromOptions.deployId = fromDefaultOptions.deployId;
145
+ if (!fromOptions.clientId) fromOptions.clientId = fromDefaultOptions.clientId;
146
+
147
+ const confFromFolder = `./engine-private/conf/${fromOptions.deployId}`;
148
+ const confToFolder = `./engine-private/conf/${toOptions.deployId}`;
149
+
150
+ const toClientVariableName = getCapVariableName(toOptions.clientId);
151
+ const fromClientVariableName = getCapVariableName(fromOptions.clientId);
152
+
153
+ const formattedSrc = (dataConf) =>
154
+ JSON.stringify(dataConf, null, 4)
155
+ .replaceAll(fromClientVariableName, toClientVariableName)
156
+ .replaceAll(fromOptions.clientId, toOptions.clientId);
157
+
158
+ const isMergeConf = fs.existsSync(confToFolder);
159
+ if (!isMergeConf) fs.mkdirSync(confToFolder, { recursive: true });
160
+
161
+ fs.writeFileSync(
162
+ `${confToFolder}/.env.production`,
163
+ fs.readFileSync(`${confFromFolder}/.env.production`, 'utf8'),
164
+ 'utf8',
165
+ );
166
+ fs.writeFileSync(
167
+ `${confToFolder}/.env.development`,
168
+ fs.readFileSync(`${confFromFolder}/.env.development`, 'utf8'),
169
+ 'utf8',
170
+ );
171
+ fs.writeFileSync(`${confToFolder}/.env.test`, fs.readFileSync(`${confFromFolder}/.env.test`, 'utf8'), 'utf8');
172
+
173
+ for (const confTypeId of ['server', 'client', 'cron', 'ssr']) {
174
+ const confFromData = JSON.parse(fs.readFileSync(`${confFromFolder}/conf.${confTypeId}.json`, 'utf8'));
175
+ fs.writeFileSync(`${confToFolder}/conf.${confTypeId}.json`, formattedSrc(confFromData), 'utf8');
176
+ }
177
+
178
+ const packageData = JSON.parse(fs.readFileSync(`${confFromFolder}/package.json`, 'utf8'));
179
+ packageData.scripts.start = packageData.scripts.start.replaceAll(fromOptions.deployId, toOptions.deployId);
180
+ fs.writeFileSync(`${confToFolder}/package.json`, JSON.stringify(packageData, null, 4), 'utf8');
181
+ };
182
+
183
+ const addClientConf = async (
184
+ { toOptions, fromOptions },
185
+ fromDefaultOptions = { deployId: 'dd-default', clientId: 'default' },
186
+ ) => {
187
+ if (!fromOptions.deployId) fromOptions.deployId = fromDefaultOptions.deployId;
188
+ if (!fromOptions.clientId) fromOptions.clientId = fromDefaultOptions.clientId;
189
+
190
+ const confFromFolder = `./engine-private/conf/${fromOptions.deployId}`;
191
+ const confToFolder = `./engine-private/conf/${toOptions.deployId}`;
192
+
193
+ const toClientConf = JSON.parse(fs.readFileSync(`${confToFolder}/conf.client.json`, 'utf8'));
194
+ const fromClientConf = JSON.parse(fs.readFileSync(`${confFromFolder}/conf.client.json`, 'utf8'));
195
+
196
+ const toClientVariableName = getCapVariableName(toOptions.clientId);
197
+ const fromClientVariableName = getCapVariableName(fromOptions.clientId);
198
+
199
+ const { host, path } = toOptions;
200
+
201
+ toClientConf[fromOptions.clientId] = fromClientConf[fromOptions.clientId];
202
+
203
+ fs.writeFileSync(`${confToFolder}/conf.client.json`, JSON.stringify(toClientConf, null, 4), 'utf8');
204
+
205
+ const toServerConf = JSON.parse(fs.readFileSync(`${confToFolder}/conf.server.json`, 'utf8'));
206
+ const fromServerConf = JSON.parse(fs.readFileSync(`${confToFolder}/conf.server.json`, 'utf8'));
207
+
208
+ toServerConf[host][path].client = fromOptions.clientId;
209
+ toServerConf[host][path].runtime = 'nodejs';
210
+ toServerConf[host][path].apis = fromClientConf[fromOptions.clientId].services;
211
+
212
+ fs.writeFileSync(`${confToFolder}/conf.server.json`, JSON.stringify(toServerConf, null, 4), 'utf8');
213
+
214
+ const fromSsrConf = JSON.parse(fs.readFileSync(`${confFromFolder}/conf.ssr.json`, 'utf8'));
215
+ const toSsrConf = JSON.parse(fs.readFileSync(`${confToFolder}/conf.ssr.json`, 'utf8'));
216
+
217
+ toSsrConf[fromClientVariableName] = fromSsrConf[fromClientVariableName];
218
+
219
+ fs.writeFileSync(`${confToFolder}/conf.ssr.json`, JSON.stringify(toSsrConf, null, 4), 'utf8');
220
+ };
221
+
222
+ const buildClientSrc = async (
223
+ { toOptions, fromOptions },
224
+ fromDefaultOptions = { deployId: 'dd-default', clientId: 'default' },
225
+ ) => {
226
+ if (!fromOptions.deployId) fromOptions.deployId = fromDefaultOptions.deployId;
227
+ if (!fromOptions.clientId) fromOptions.clientId = fromDefaultOptions.clientId;
228
+
229
+ const confFromFolder = `./src/client/components/${fromOptions.clientId}`;
230
+ const confToFolder = `./src/client/components/${toOptions.clientId}`;
231
+
232
+ const toClientVariableName = getCapVariableName(toOptions.clientId);
233
+ const fromClientVariableName = getCapVariableName(fromOptions.clientId);
234
+
235
+ const formattedSrc = (src) =>
236
+ src.replaceAll(fromClientVariableName, toClientVariableName).replaceAll(fromOptions.clientId, toOptions.clientId);
237
+
238
+ const isMergeConf = fs.existsSync(confToFolder);
239
+ if (!isMergeConf) fs.mkdirSync(confToFolder, { recursive: true });
240
+
241
+ const files = await fs.readdir(confFromFolder, { recursive: true });
242
+ for (const relativePath of files) {
243
+ const fromFilePath = dir.resolve(`${confFromFolder}/${relativePath}`);
244
+ const toFilePath = dir.resolve(`${confToFolder}/${relativePath}`);
245
+
246
+ fs.writeFileSync(formattedSrc(toFilePath), formattedSrc(fs.readFileSync(fromFilePath, 'utf8')), 'utf8');
247
+ }
248
+
249
+ fs.writeFileSync(
250
+ `./src/client/ssr/components/head/${toClientVariableName}Scripts.js`,
251
+ formattedSrc(fs.readFileSync(`./src/client/ssr/components/head/${fromClientVariableName}Scripts.js`, 'utf8')),
252
+ 'utf8',
253
+ );
254
+
255
+ fs.writeFileSync(
256
+ `./src/client/${toClientVariableName}.index.js`,
257
+ formattedSrc(fs.readFileSync(`./src/client/${fromClientVariableName}.index.js`, 'utf8')),
258
+ 'utf8',
259
+ );
260
+
261
+ fs.copySync(`./src/client/public/${fromOptions.clientId}`, `./src/client/public/${toOptions.clientId}`);
262
+ };
263
+
264
+ const buildApiSrc = async (
265
+ { toOptions, fromOptions },
266
+ fromDefaultOptions = { apiId: 'default', deployId: 'dd-default', clientId: 'default' },
267
+ ) => {
268
+ if (!fromOptions.apiId) fromOptions.apiId = fromDefaultOptions.apiId;
269
+ if (!fromOptions.deployId) fromOptions.deployId = fromDefaultOptions.deployId;
270
+ if (!fromOptions.clientId) fromOptions.clientId = fromDefaultOptions.clientId;
271
+
272
+ const toClientVariableName = getCapVariableName(toOptions.apiId);
273
+ const fromClientVariableName = getCapVariableName(fromOptions.apiId);
274
+
275
+ const formattedSrc = (src) =>
276
+ src.replaceAll(fromClientVariableName, toClientVariableName).replaceAll(fromOptions.apiId, toOptions.apiId);
277
+
278
+ const apiToFolder = `./src/api/${toOptions.apiId}`;
279
+ const apiFromFolder = `./src/api/${fromOptions.apiId}`;
280
+
281
+ const isMergeConf = fs.existsSync(apiToFolder);
282
+ if (!isMergeConf) fs.mkdirSync(apiToFolder, { recursive: true });
283
+
284
+ for (const srcApiType of ['model', 'controller', 'service', 'router']) {
285
+ fs.writeFileSync(
286
+ `${apiToFolder}/${toOptions.apiId}.${srcApiType}.js`,
287
+ formattedSrc(fs.readFileSync(`${apiFromFolder}/${fromOptions.apiId}.${srcApiType}.js`, 'utf8')),
288
+ 'utf8',
289
+ );
290
+ }
291
+
292
+ fs.mkdirSync(`./src/client/services/${toOptions.apiId}`, { recursive: true });
293
+ if (fs.existsSync(`./src/client/services/${fromOptions.apiId}/${fromOptions.apiId}.service.js`))
294
+ fs.writeFileSync(
295
+ `./src/client/services/${toOptions.apiId}/${toOptions.apiId}.service.js`,
296
+ formattedSrc(
297
+ fs.readFileSync(`./src/client/services/${fromOptions.apiId}/${fromOptions.apiId}.service.js`, 'utf8'),
298
+ ),
299
+ 'utf8',
300
+ );
301
+ return;
302
+ if (fs.existsSync(`./src/client/services/${fromOptions.apiId}/${fromOptions.apiId}.management.js`))
303
+ fs.writeFileSync(
304
+ `./src/client/services/${toOptions.apiId}/${toOptions.apiId}.management.js`,
305
+ formattedSrc(
306
+ fs.readFileSync(`./src/client/services/${fromOptions.apiId}/${fromOptions.apiId}.management.js`, 'utf8'),
307
+ ),
308
+ 'utf8',
309
+ );
310
+ };
311
+
312
+ const addApiConf = async (
313
+ { toOptions, fromOptions },
314
+ fromDefaultOptions = { apiId: 'default', deployId: 'dd-default', clientId: 'default' },
315
+ ) => {
316
+ if (!fromOptions.apiId) fromOptions.apiId = fromDefaultOptions.apiId;
317
+ if (!fromOptions.deployId) fromOptions.deployId = fromDefaultOptions.deployId;
318
+ if (!fromOptions.clientId) fromOptions.clientId = fromDefaultOptions.clientId;
319
+
320
+ const toClientVariableName = getCapVariableName(toOptions.apiId);
321
+ const fromClientVariableName = getCapVariableName(fromOptions.apiId);
322
+
323
+ const confFromFolder = `./engine-private/conf/${fromOptions.deployId}`;
324
+ const confToFolder = `./engine-private/conf/${toOptions.deployId}`;
325
+
326
+ const confServer = JSON.parse(fs.readFileSync(`${confToFolder}/conf.server.json`, 'utf8'));
327
+ for (const host of Object.keys(confServer))
328
+ for (const path of Object.keys(confServer[host]))
329
+ if (confServer[host][path].apis) confServer[host][path].apis.push(toOptions.apiId);
330
+ fs.writeFileSync(`${confToFolder}/conf.server.json`, JSON.stringify(confServer, null, 4), 'utf8');
331
+
332
+ const confClient = JSON.parse(fs.readFileSync(`${confToFolder}/conf.client.json`, 'utf8'));
333
+ confClient[toOptions.clientId].services.push(toOptions.apiId);
334
+ fs.writeFileSync(`${confToFolder}/conf.client.json`, JSON.stringify(confClient, null, 4), 'utf8');
335
+ };
336
+
337
+ const addWsConf = async (
338
+ { toOptions, fromOptions },
339
+ fromDefaultOptions = { wsId: 'default', deployId: 'dd-default', host: 'default.net', paths: '/' },
340
+ ) => {
341
+ if (!fromOptions.wsId) fromOptions.wsId = fromDefaultOptions.wsId;
342
+ if (!fromOptions.deployId) fromOptions.deployId = fromDefaultOptions.deployId;
343
+ if (!fromOptions.host) fromOptions.host = fromDefaultOptions.host;
344
+ if (!fromOptions.paths) fromOptions.paths = fromDefaultOptions.paths;
345
+
346
+ const toClientVariableName = getCapVariableName(toOptions.apiId);
347
+ const fromClientVariableName = getCapVariableName(fromOptions.apiId);
348
+
349
+ const confFromFolder = `./engine-private/conf/${fromOptions.deployId}`;
350
+ const confToFolder = `./engine-private/conf/${toOptions.deployId}`;
351
+
352
+ const paths = toOptions.paths.split(',');
353
+
354
+ const confServer = JSON.parse(fs.readFileSync(`${confToFolder}/conf.server.json`, 'utf8'));
355
+ for (const host of Object.keys(confServer))
356
+ for (const path of Object.keys(confServer[host]))
357
+ if (host === toOptions.host && paths.includes(path) && confServer[host][path])
358
+ confServer[host][path].ws = toOptions.wsId;
359
+ fs.writeFileSync(`${confToFolder}/conf.server.json`, JSON.stringify(confServer, null, 4), 'utf8');
360
+ };
361
+
362
+ const buildWsSrc = async (
363
+ { toOptions, fromOptions },
364
+ fromDefaultOptions = { wsId: 'default', deployId: 'dd-default', host: 'default.net', paths: '/' },
365
+ ) => {
366
+ if (!fromOptions.wsId) fromOptions.wsId = fromDefaultOptions.wsId;
367
+ if (!fromOptions.deployId) fromOptions.deployId = fromDefaultOptions.deployId;
368
+ if (!fromOptions.host) fromOptions.host = fromDefaultOptions.host;
369
+ if (!fromOptions.paths) fromOptions.paths = fromDefaultOptions.paths;
370
+
371
+ const toClientVariableName = getCapVariableName(toOptions.wsId);
372
+ const fromClientVariableName = getCapVariableName(fromOptions.wsId);
373
+
374
+ const confFromFolder = `./src/ws/${fromOptions.wsId}`;
375
+ const confToFolder = `./src/ws/${toOptions.wsId}`;
376
+
377
+ const paths = toOptions.paths.split(',');
378
+
379
+ const formattedSrc = (src) =>
380
+ src.replaceAll(fromClientVariableName, toClientVariableName).replaceAll(fromOptions.wsId, toOptions.wsId);
381
+
382
+ const files = await fs.readdir(confFromFolder, { recursive: true });
383
+ for (const relativePath of files) {
384
+ const fromFilePath = dir.resolve(`${confFromFolder}/${relativePath}`);
385
+ const toFilePath = dir.resolve(`${confToFolder}/${relativePath}`);
386
+
387
+ if (fs.lstatSync(fromFilePath).isDirectory() && !fs.existsSync(formattedSrc(toFilePath)))
388
+ fs.mkdirSync(formattedSrc(toFilePath), { recursive: true });
389
+
390
+ if (fs.lstatSync(fromFilePath).isFile() && !fs.existsSync(formattedSrc(toFilePath))) {
391
+ fs.writeFileSync(formattedSrc(toFilePath), formattedSrc(fs.readFileSync(fromFilePath, 'utf8')), 'utf8');
392
+ }
393
+ }
394
+ };
395
+
396
+ const cloneSrcComponents = async ({ toOptions, fromOptions }) => {
397
+ const toClientVariableName = getCapVariableName(toOptions.componentsFolder);
398
+ const fromClientVariableName = getCapVariableName(fromOptions.componentsFolder);
399
+
400
+ const formattedSrc = (src) =>
401
+ src
402
+ .replaceAll(fromClientVariableName, toClientVariableName)
403
+ .replaceAll(fromOptions.componentsFolder, toOptions.componentsFolder);
404
+
405
+ const confFromFolder = `./src/client/components/${fromOptions.componentsFolder}`;
406
+ const confToFolder = `./src/client/components/${toOptions.componentsFolder}`;
407
+
408
+ fs.mkdirSync(confToFolder, { recursive: true });
409
+
410
+ const files = await fs.readdir(confFromFolder);
411
+ for (const relativePath of files) {
412
+ const fromFilePath = dir.resolve(`${confFromFolder}/${relativePath}`);
413
+ const toFilePath = dir.resolve(`${confToFolder}/${relativePath}`);
414
+
415
+ fs.writeFileSync(formattedSrc(toFilePath), formattedSrc(fs.readFileSync(fromFilePath, 'utf8')), 'utf8');
416
+ }
417
+ };
418
+
419
+ const buildProxyRouter = () => {
420
+ const confServer = JSON.parse(fs.readFileSync(`./conf/conf.server.json`, 'utf8'));
421
+ let currentPort = parseInt(process.env.PORT) + 1;
422
+ const proxyRouter = {};
423
+ const singleReplicaHosts = [];
424
+ for (const host of Object.keys(confServer)) {
425
+ for (const path of Object.keys(confServer[host])) {
426
+ if (confServer[host][path].singleReplica && !singleReplicaHosts.includes(host)) {
427
+ singleReplicaHosts.push(host);
428
+ currentPort++;
429
+ continue;
430
+ }
431
+ confServer[host][path].port = newInstance(currentPort);
432
+ for (const port of confServer[host][path].proxy) {
433
+ if (!(port in proxyRouter)) proxyRouter[port] = {};
434
+ proxyRouter[port][`${host}${path}`] = {
435
+ // target: `http://${host}:${confServer[host][path].port}${path}`,
436
+ target: `http://localhost:${confServer[host][path].port - singleReplicaHosts.length}`,
437
+ // target: `http://127.0.0.1:${confServer[host][path].port}`,
438
+ proxy: confServer[host][path].proxy,
439
+ redirect: confServer[host][path].redirect,
440
+ host,
441
+ path,
442
+ };
443
+ }
444
+ currentPort++;
445
+ if (confServer[host][path].peer) {
446
+ const peerPath = path === '/' ? `/peer` : `${path}/peer`;
447
+ confServer[host][peerPath] = newInstance(confServer[host][path]);
448
+ confServer[host][peerPath].port = newInstance(currentPort);
449
+ for (const port of confServer[host][path].proxy) {
450
+ if (!(port in proxyRouter)) proxyRouter[port] = {};
451
+ proxyRouter[port][`${host}${peerPath}`] = {
452
+ // target: `http://${host}:${confServer[host][peerPath].port}${peerPath}`,
453
+ target: `http://localhost:${confServer[host][peerPath].port - singleReplicaHosts.length}`,
454
+ // target: `http://127.0.0.1:${confServer[host][peerPath].port}`,
455
+ proxy: confServer[host][peerPath].proxy,
456
+ host,
457
+ path: peerPath,
458
+ };
459
+ }
460
+ currentPort++;
461
+ }
462
+ }
463
+ }
464
+ return proxyRouter;
465
+ };
466
+
467
+ const cliBar = async (time = 5000) => {
468
+ // create new progress bar
469
+ const b = new cliProgress.SingleBar({
470
+ format: 'Delay | {bar} | {percentage}% || {value}/{total} Chunks || Speed: {speed}',
471
+ barCompleteChar: '\u2588',
472
+ barIncompleteChar: '\u2591',
473
+ hideCursor: true,
474
+ });
475
+
476
+ const maxValueDisplay = 200;
477
+ const minValueDisplay = 0;
478
+ const steps = 10;
479
+ const incrementValue = 200 / steps;
480
+ const delayTime = time / steps;
481
+ // initialize the bar - defining payload token "speed" with the default value "N/A"
482
+ b.start(maxValueDisplay, minValueDisplay, {
483
+ speed: 'N/A',
484
+ });
485
+
486
+ // update values
487
+ // b1.increment();
488
+ // b1.update(20);
489
+
490
+ for (const step of range(1, steps)) {
491
+ b.increment(incrementValue);
492
+ await timer(delayTime);
493
+ }
494
+
495
+ // stop the bar
496
+ b.stop();
497
+ };
498
+
499
+ const cliSpinner = async (time = 5000, message0, message1, color, type = 'dots') => {
500
+ const { frames, interval } = cliSpinners[type];
501
+ const steps = parseInt(time / interval);
502
+ let index = 0;
503
+ for (const step of range(1, steps)) {
504
+ const msg = `${message0 ? message0 : ''}${frames[index]}${message1 ? message1 : ''}`;
505
+ logUpdate(color ? msg[color] : msg);
506
+ await timer(interval);
507
+ index++;
508
+ if (index === frames.length) index = 0;
509
+ }
510
+ };
511
+
512
+ const buildReplicaId = ({ deployId, replica }) => `${deployId}-${replica.slice(1)}`;
513
+
514
+ const getDataDeploy = (
515
+ options = { buildSingleReplica: false, deployGroupId: '', deployId: '', disableSyncEnvPort: false },
516
+ ) => {
517
+ let dataDeploy = JSON.parse(fs.readFileSync(`./engine-private/deploy/${options.deployGroupId}.json`, 'utf8'));
518
+
519
+ if (options.deployId) dataDeploy = dataDeploy.filter((d) => d === options.deployId);
520
+
521
+ dataDeploy = dataDeploy.map((deployId) => {
522
+ return {
523
+ deployId,
524
+ };
525
+ });
526
+
527
+ if (options && options.buildSingleReplica && fs.existsSync(`./engine-private/replica`))
528
+ fs.removeSync(`./engine-private/replica`);
529
+
530
+ let buildDataDeploy = [];
531
+ for (const deployObj of dataDeploy) {
532
+ const serverConf = loadReplicas(
533
+ JSON.parse(fs.readFileSync(`./engine-private/conf/${deployObj.deployId}/conf.server.json`, 'utf8')),
534
+ );
535
+ let replicaDataDeploy = [];
536
+ for (const host of Object.keys(serverConf))
537
+ for (const path of Object.keys(serverConf[host])) {
538
+ if (serverConf[host][path].replicas && serverConf[host][path].singleReplica) {
539
+ if (options && options.buildSingleReplica) shellExec(Cmd.replica(deployObj.deployId, host, path));
540
+ replicaDataDeploy = replicaDataDeploy.concat(
541
+ serverConf[host][path].replicas.map((r) => {
542
+ return {
543
+ deployId: buildReplicaId({ deployId: deployObj.deployId, replica: r }),
544
+ replicaHost: host,
545
+ };
546
+ }),
547
+ );
548
+ }
549
+ }
550
+ buildDataDeploy.push(deployObj);
551
+ if (replicaDataDeploy.length > 0) buildDataDeploy = buildDataDeploy.concat(replicaDataDeploy);
552
+ }
553
+
554
+ const enableSyncEnvPort = !options.disableSyncEnvPort && options.buildSingleReplica;
555
+ if (enableSyncEnvPort) shellExec(Cmd.syncPorts(options.deployGroupId));
556
+
557
+ logger.info('buildDataDeploy', { buildDataDeploy, enableSyncEnvPort });
558
+
559
+ return buildDataDeploy;
560
+ };
561
+
562
+ const validateTemplatePath = (absolutePath = '') => {
563
+ const host = 'default.net';
564
+ const path = '/';
565
+ const client = 'default';
566
+ const ssr = 'Default';
567
+ const confServer = DefaultConf.server[host][path];
568
+ const confClient = DefaultConf.client[client];
569
+ const confSsr = DefaultConf.ssr[ssr];
570
+ const clients = Object.keys(confClient).concat(['core', 'test', 'default']);
571
+
572
+ if (absolutePath.match('src/api') && !confServer.apis.find((p) => absolutePath.match(`src/api/${p}/`))) {
573
+ return false;
574
+ }
575
+ if (
576
+ absolutePath.match('src/client/services/') &&
577
+ !clients.find((p) => absolutePath.match(`src/client/services/${p}/`))
578
+ ) {
579
+ return false;
580
+ }
581
+ if (absolutePath.match('src/client/public/') && !clients.find((p) => absolutePath.match(`src/client/public/${p}/`))) {
582
+ return false;
583
+ }
584
+ if (
585
+ absolutePath.match('src/client/components/') &&
586
+ !clients.find((p) => absolutePath.match(`src/client/components/${p}/`))
587
+ ) {
588
+ return false;
589
+ }
590
+ if (absolutePath.match('src/client/sw/') && !clients.find((p) => absolutePath.match(`src/client/sw/${p}.sw.js`))) {
591
+ return false;
592
+ }
593
+ if (
594
+ absolutePath.match('src/client/ssr/components/body') &&
595
+ !confSsr.body.find((p) => absolutePath.match(`src/client/ssr/components/body/${p}.js`))
596
+ ) {
597
+ return false;
598
+ }
599
+ if (
600
+ absolutePath.match('src/client/ssr/components/head') &&
601
+ !confSsr.head.find((p) => absolutePath.match(`src/client/ssr/components/head/${p}.js`))
602
+ ) {
603
+ return false;
604
+ }
605
+ if (absolutePath.match('hardhat/')) return false;
606
+ if (
607
+ absolutePath.match('/client') &&
608
+ absolutePath.match('.index.js') &&
609
+ !absolutePath.match('/offline') &&
610
+ !clients.find((p) => absolutePath.match(`src/client/${capFirst(p)}.index.js`))
611
+ ) {
612
+ return false;
613
+ }
614
+ if (absolutePath.match('src/ws/') && !clients.find((p) => absolutePath.match(`src/ws/${p}/`))) {
615
+ return false;
616
+ }
617
+ return true;
618
+ };
619
+
620
+ const deployTest = async (dataDeploy) => {
621
+ const failed = [];
622
+ for (const deploy of dataDeploy) {
623
+ const deployServerConfPath = fs.existsSync(`./engine-private/replica/${deploy.deployId}/conf.server.json`)
624
+ ? `./engine-private/replica/${deploy.deployId}/conf.server.json`
625
+ : `./engine-private/conf/${deploy.deployId}/conf.server.json`;
626
+ const serverConf = loadReplicas(JSON.parse(fs.readFileSync(deployServerConfPath, 'utf8')));
627
+ let fail = false;
628
+ for (const host of Object.keys(serverConf))
629
+ for (const path of Object.keys(serverConf[host])) {
630
+ const { singleReplica } = serverConf[host][path];
631
+ if (singleReplica) continue;
632
+ const urlTest = `https://${host}${path}`;
633
+ try {
634
+ const result = await axios.get(urlTest);
635
+ const test = result.data.split('<title>');
636
+ if (test[1])
637
+ logger.info('Success deploy', {
638
+ ...deploy,
639
+ result: test[1].split('</title>')[0],
640
+ urlTest,
641
+ });
642
+ else {
643
+ logger.error('Error deploy', {
644
+ ...deploy,
645
+ result: result.data,
646
+ urlTest,
647
+ });
648
+ fail = true;
649
+ }
650
+ } catch (error) {
651
+ logger.error('Error deploy', {
652
+ ...deploy,
653
+ message: error.message,
654
+ urlTest,
655
+ });
656
+ fail = true;
657
+ }
658
+ }
659
+ if (fail) failed.push(deploy);
660
+ }
661
+ return { failed };
662
+ };
663
+
664
+ const getDeployGroupId = () => {
665
+ const deployGroupIndexArg = process.argv.findIndex((a) => a.match(`deploy-group:`));
666
+ if (deployGroupIndexArg > -1) return process.argv[deployGroupIndexArg].split(':')[1].trim();
667
+ return 'dd';
668
+ };
669
+
670
+ const getDeployId = () => {
671
+ const deployIndexArg = process.argv.findIndex((a) => a.match(`deploy-id:`));
672
+ if (deployIndexArg > -1) return process.argv[deployIndexArg].split(':')[1].trim();
673
+ for (const deployId of process.argv) {
674
+ if (fs.existsSync(`./engine-private/conf/${deployId}`)) return deployId;
675
+ else if (fs.existsSync(`./engine-private/replica/${deployId}`)) return deployId;
676
+ }
677
+ return 'default';
678
+ };
679
+
680
+ const getCronBackUpFolder = (host = '', path = '') => {
681
+ return `${host}${path.replace(/\\/g, '/').replace(`/`, '-')}`;
682
+ };
683
+
684
+ const execDeploy = async (options = { deployId: 'default' }) => {
685
+ const { deployId } = options;
686
+ shellExec(Cmd.delete(deployId));
687
+ shellExec(Cmd.conf(deployId));
688
+ shellExec(Cmd.run(deployId));
689
+ return await new Promise(async (resolve) => {
690
+ const maxTime = 1000 * 60 * 5;
691
+ const minTime = 7 * 1000;
692
+ const intervalTime = 1000;
693
+ let currentTime = 0;
694
+ const attempt = () => {
695
+ if (currentTime >= minTime && !fs.existsSync(`./tmp/await-deploy`)) {
696
+ clearInterval(processMonitor);
697
+ return resolve(true);
698
+ }
699
+ cliSpinner(
700
+ intervalTime,
701
+ `[deploy.js] `,
702
+ ` Load instance | elapsed time ${currentTime / 1000}s / ${maxTime / 1000}s`,
703
+ 'yellow',
704
+ 'material',
705
+ );
706
+ currentTime += intervalTime;
707
+ if (currentTime >= maxTime) return resolve(false);
708
+ };
709
+ const processMonitor = setInterval(attempt, intervalTime);
710
+ });
711
+ };
712
+
713
+ const deployRun = async (dataDeploy, reset) => {
714
+ if (!fs.existsSync(`./tmp`)) fs.mkdirSync(`./tmp`, { recursive: true });
715
+ if (reset) fs.writeFileSync(`./tmp/runtime-router.json`, '{}', 'utf8');
716
+ await fixDependencies();
717
+ for (const deploy of dataDeploy) await execDeploy(deploy);
718
+ const { failed } = await deployTest(dataDeploy);
719
+ if (failed.length > 0) {
720
+ for (const deploy of failed) logger.error(deploy.deployId, Cmd.run(deploy.deployId));
721
+ await read({ prompt: 'Press enter to retry failed processes\n' });
722
+ await deployRun(failed);
723
+ } else logger.info(`Deploy process successfully`);
724
+ };
725
+
726
+ const restoreMacroDb = async (deployGroupId = '') => {
727
+ const dataDeploy = await getDataDeploy({ deployGroupId, buildSingleReplica: false });
728
+ for (const deployGroup of dataDeploy) {
729
+ if (!deployGroup.replicaHost) {
730
+ const deployServerConfPath = `./engine-private/conf/${deployGroup.deployId}/conf.server.json`;
731
+ const serverConf = JSON.parse(fs.readFileSync(deployServerConfPath, 'utf8'));
732
+
733
+ for (const host of Object.keys(serverConf)) {
734
+ for (const path of Object.keys(serverConf[host])) {
735
+ const { db, singleReplica } = serverConf[host][path];
736
+ if (db && !singleReplica) {
737
+ const cmd = `node bin/db ${host}${path} import ${deployGroup.deployId} cron`;
738
+ shellExec(cmd);
739
+ }
740
+ }
741
+ }
742
+ }
743
+ }
744
+ };
745
+
746
+ const mergeBackUp = async (baseBackJsonPath, outputFilePath) => {
747
+ const names = JSON.parse(fs.readFileSync(baseBackJsonPath, 'utf8')).map((p) =>
748
+ p.replaceAll(`\\`, '/').replaceAll('C:/', '/').replaceAll('c:/', '/'),
749
+ );
750
+ await new Promise((resolve) => {
751
+ splitFile
752
+ .mergeFiles(names, outputFilePath)
753
+ .then(() => {
754
+ resolve();
755
+ })
756
+ .catch((err) => {
757
+ console.log('Error: ', err);
758
+ resolve();
759
+ });
760
+ });
761
+ };
762
+
763
+ const getRestoreCronCmd = async (options = { host: '', path: '', conf: {}, deployId: '' }) => {
764
+ const { host, path, conf, deployId } = options;
765
+ const { runtime, db, git, directory } = conf[host][path];
766
+ const { provider, name, user, password = '', backupPath = '' } = db;
767
+
768
+ if (['xampp', 'lampp'].includes(runtime)) {
769
+ logger.info('Create database', `node bin/db ${host}${path} create ${deployId}`);
770
+ shellExec(`node bin/db ${host}${path} create ${deployId}`);
771
+ }
772
+
773
+ if (git) {
774
+ if (directory && !fs.existsSync(directory)) fs.mkdirSync(directory, { recursive: true });
775
+
776
+ shellExec(`git clone ${git}`);
777
+
778
+ // fs.mkdirSync(`./public/${host}${path}`, { recursive: true });
779
+
780
+ if (fs.existsSync(`./${git.split('/').pop()}`))
781
+ fs.moveSync(`./${git.split('/').pop()}`, directory ? directory : `./public/${host}${path}`, {
782
+ overwrite: true,
783
+ });
784
+ }
785
+
786
+ let cmd, currentBackupTimestamp, baseBackUpPath;
787
+
788
+ if (process.argv.includes('cron')) {
789
+ baseBackUpPath = `${process.cwd()}/engine-private/cron-backups/${getCronBackUpFolder(host, path)}`;
790
+
791
+ const files = await fs.readdir(baseBackUpPath, { withFileTypes: true });
792
+
793
+ currentBackupTimestamp = files
794
+ .map((fileObj) => parseInt(fileObj.name))
795
+ .sort((a, b) => a - b)
796
+ .reverse()[0];
797
+ }
798
+
799
+ switch (provider) {
800
+ case 'mariadb':
801
+ {
802
+ if (process.argv.includes('cron')) {
803
+ cmd = `mysql -u ${user} -p${password} ${name} < ${baseBackUpPath}/${currentBackupTimestamp}/${name}.sql`;
804
+ if (fs.existsSync(`${baseBackUpPath}/${currentBackupTimestamp}/${name}-parths.json`))
805
+ await mergeBackUp(
806
+ `${baseBackUpPath}/${currentBackupTimestamp}/${name}-parths.json`,
807
+ `${baseBackUpPath}/${currentBackupTimestamp}/${name}.sql`,
808
+ );
809
+ } else {
810
+ cmd = `mysql -u ${user} -p${password} ${name} < ${
811
+ backupPath ? backupPath : `./engine-private/sql-backups/${name}.sql`
812
+ }`;
813
+ if (
814
+ fs.existsSync(
815
+ `${
816
+ backupPath ? backupPath.split('/').slice(0, -1).join('/') : `./engine-private/sql-backups`
817
+ }/${name}-parths.json`,
818
+ )
819
+ )
820
+ await mergeBackUp(
821
+ `${
822
+ backupPath ? backupPath.split('/').slice(0, -1).join('/') : `./engine-private/sql-backups`
823
+ }/${name}-parths.json`,
824
+ `${
825
+ backupPath ? backupPath.split('/').slice(0, -1).join('/') : `./engine-private/sql-backups`
826
+ }/${name}.sql`,
827
+ );
828
+ }
829
+ }
830
+ break;
831
+
832
+ case 'mongoose':
833
+ {
834
+ if (process.argv.includes('cron')) {
835
+ cmd = `mongorestore -d ${name} ${baseBackUpPath}/${currentBackupTimestamp}/${name}`;
836
+ } else cmd = `mongorestore -d ${name} ${backupPath ? backupPath : `./engine-private/mongodb-backup/${name}`}`;
837
+ }
838
+ break;
839
+ }
840
+
841
+ // logger.info('Restore', cmd);
842
+
843
+ return cmd;
844
+ };
845
+
846
+ const Cmd = {
847
+ delete: (deployId) => `pm2 delete ${deployId}`,
848
+ run: (deployId) => `node bin/deploy run ${deployId}`,
849
+ build: (deployId) => `node bin/deploy build-full-client ${deployId}${process.argv.includes('l') ? ' l' : ''}`,
850
+ conf: (deployId, env) => `node bin/deploy conf ${deployId} ${env ? env : 'production'}`,
851
+ replica: (deployId, host, path) => `node bin/deploy build-single-replica ${deployId} ${host} ${path}`,
852
+ syncPorts: (deployGroupId) => `node bin/deploy sync-env-port ${deployGroupId}`,
853
+ cron: (deployId, job, expression) => {
854
+ shellExec(Cmd.delete(`${deployId}-${job}`));
855
+ return `env-cmd -f .env.production pm2 start bin/cron.js --no-autorestart --instances 1 --cron "${expression}" --name ${deployId}-${job} -- ${job} ${deployId}`;
856
+ },
857
+ };
858
+
859
+ const fixDependencies = async () => {
860
+ // sed -i "$line_number s,.*,$new_text," "$file"
861
+ // sed -i "$line_number c \\$new_text" "$file"
862
+ const dep = fs.readFileSync(`./node_modules/peer/dist/module.mjs`, 'utf8');
863
+ const errorLine = `import {WebSocketServer as $hSjDC$WebSocketServer} from "ws";`;
864
+
865
+ fs.writeFileSync(
866
+ `./node_modules/peer/dist/module.mjs`,
867
+ dep.replaceAll(
868
+ errorLine,
869
+ `import WebSocketServer from "ws";
870
+ let $hSjDC$WebSocketServer = WebSocketServer.Server;`,
871
+ ),
872
+ 'utf8',
873
+ );
874
+ };
875
+
876
+ const maintenancePath = `${getRootDirectory()}/public/${process.env.DEFAULT_DEPLOY_HOST}${
877
+ process.env.DEFAULT_DEPLOY_PATH
878
+ }/maintenance.html`;
879
+
880
+ const maintenanceMiddleware = (req, res, port, proxyRouter) => {
881
+ if (process.argv.includes('maintenance') && fs.existsSync(maintenancePath)) {
882
+ if (req.method.toUpperCase() === 'GET') return res.status(503).sendFile(maintenancePath);
883
+ return res.status(503).json({
884
+ status: 'error',
885
+ message: 'Server is under maintenance',
886
+ });
887
+ }
888
+ };
889
+
890
+ const setUpProxyMaintenanceServer = ({ deployGroupId }) => {
891
+ shellExec(`pm2 kill`);
892
+ const proxyDeployId = fs.readFileSync(`./engine-private/deploy/${deployGroupId}.proxy`, 'utf8').trim();
893
+ shellExec(`node bin/deploy conf ${proxyDeployId} production`);
894
+ shellExec(`node bin/deploy run ${proxyDeployId} maintenance`);
895
+ };
896
+
897
+ export {
898
+ Cmd,
899
+ Config,
900
+ loadConf,
901
+ loadReplicas,
902
+ cloneConf,
903
+ getCapVariableName,
904
+ buildClientSrc,
905
+ buildApiSrc,
906
+ addApiConf,
907
+ addClientConf,
908
+ addWsConf,
909
+ buildWsSrc,
910
+ cloneSrcComponents,
911
+ buildProxyRouter,
912
+ cliBar,
913
+ cliSpinner,
914
+ getDataDeploy,
915
+ validateTemplatePath,
916
+ buildReplicaId,
917
+ restoreMacroDb,
918
+ getDeployGroupId,
919
+ execDeploy,
920
+ deployRun,
921
+ getCronBackUpFolder,
922
+ getRestoreCronCmd,
923
+ mergeBackUp,
924
+ fixDependencies,
925
+ getDeployId,
926
+ maintenancePath,
927
+ maintenanceMiddleware,
928
+ setUpProxyMaintenanceServer,
929
+ };