underpost 2.7.1 → 2.7.3

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