@bigtyphoon/melo 1.7.6

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 (296) hide show
  1. package/LICENSE +22 -0
  2. package/README.md +55 -0
  3. package/bin/commadtest.ts +10 -0
  4. package/bin/commands/add.ts +43 -0
  5. package/bin/commands/init.ts +292 -0
  6. package/bin/commands/kill.ts +21 -0
  7. package/bin/commands/list.ts +60 -0
  8. package/bin/commands/masterha.ts +40 -0
  9. package/bin/commands/restart.ts +48 -0
  10. package/bin/commands/start.ts +65 -0
  11. package/bin/commands/stop.ts +26 -0
  12. package/bin/melo.ts +30 -0
  13. package/bin/utils/constants.ts +27 -0
  14. package/bin/utils/utils.ts +130 -0
  15. package/dist/bin/commadtest.js +9 -0
  16. package/dist/bin/commands/add.js +40 -0
  17. package/dist/bin/commands/init.js +279 -0
  18. package/dist/bin/commands/kill.js +21 -0
  19. package/dist/bin/commands/list.js +65 -0
  20. package/dist/bin/commands/masterha.js +36 -0
  21. package/dist/bin/commands/restart.js +45 -0
  22. package/dist/bin/commands/start.js +58 -0
  23. package/dist/bin/commands/stop.js +20 -0
  24. package/dist/bin/melo.js +26 -0
  25. package/dist/bin/utils/constants.js +28 -0
  26. package/dist/bin/utils/utils.js +134 -0
  27. package/dist/lib/application.js +888 -0
  28. package/dist/lib/common/manager/appManager.js +112 -0
  29. package/dist/lib/common/manager/taskManager.js +39 -0
  30. package/dist/lib/common/remote/backend/msgRemote.js +63 -0
  31. package/dist/lib/common/remote/frontend/channelRemote.js +78 -0
  32. package/dist/lib/common/remote/frontend/sessionRemote.js +76 -0
  33. package/dist/lib/common/service/backendSessionService.js +337 -0
  34. package/dist/lib/common/service/channelService.js +514 -0
  35. package/dist/lib/common/service/connectionService.js +95 -0
  36. package/dist/lib/common/service/filterService.js +112 -0
  37. package/dist/lib/common/service/handlerService.js +187 -0
  38. package/dist/lib/common/service/sessionService.js +610 -0
  39. package/dist/lib/components/backendSession.js +14 -0
  40. package/dist/lib/components/channel.js +13 -0
  41. package/dist/lib/components/connection.js +12 -0
  42. package/dist/lib/components/connector.js +437 -0
  43. package/dist/lib/components/dictionary.js +93 -0
  44. package/dist/lib/components/master.js +39 -0
  45. package/dist/lib/components/monitor.js +25 -0
  46. package/dist/lib/components/protobuf.js +156 -0
  47. package/dist/lib/components/proxy.js +236 -0
  48. package/dist/lib/components/pushScheduler.js +62 -0
  49. package/dist/lib/components/remote.js +127 -0
  50. package/dist/lib/components/server.js +63 -0
  51. package/dist/lib/components/session.js +20 -0
  52. package/dist/lib/connectors/commands/handshake.js +119 -0
  53. package/dist/lib/connectors/commands/heartbeat.js +67 -0
  54. package/dist/lib/connectors/commands/kick.js +15 -0
  55. package/dist/lib/connectors/common/coder.js +90 -0
  56. package/dist/lib/connectors/common/handler.js +57 -0
  57. package/dist/lib/connectors/hybrid/IHybridSocket.js +3 -0
  58. package/dist/lib/connectors/hybrid/switcher.js +100 -0
  59. package/dist/lib/connectors/hybrid/tcpprocessor.js +40 -0
  60. package/dist/lib/connectors/hybrid/tcpsocket.js +171 -0
  61. package/dist/lib/connectors/hybrid/wsprocessor.js +49 -0
  62. package/dist/lib/connectors/hybridconnector.js +89 -0
  63. package/dist/lib/connectors/hybridsocket.js +139 -0
  64. package/dist/lib/connectors/mqtt/generate.js +113 -0
  65. package/dist/lib/connectors/mqtt/mqttadaptor.js +81 -0
  66. package/dist/lib/connectors/mqtt/protocol.js +48 -0
  67. package/dist/lib/connectors/mqttconnector.js +107 -0
  68. package/dist/lib/connectors/mqttsocket.js +59 -0
  69. package/dist/lib/connectors/sioconnector.js +135 -0
  70. package/dist/lib/connectors/siosocket.js +69 -0
  71. package/dist/lib/connectors/udpconnector.js +76 -0
  72. package/dist/lib/connectors/udpsocket.js +93 -0
  73. package/dist/lib/filters/handler/serial.js +44 -0
  74. package/dist/lib/filters/handler/time.js +32 -0
  75. package/dist/lib/filters/handler/timeout.js +45 -0
  76. package/dist/lib/filters/handler/toobusy.js +36 -0
  77. package/dist/lib/filters/rpc/rpcLog.js +43 -0
  78. package/dist/lib/filters/rpc/toobusy.js +41 -0
  79. package/dist/lib/index.js +81 -0
  80. package/dist/lib/interfaces/IComponent.js +3 -0
  81. package/dist/lib/interfaces/IConnector.js +3 -0
  82. package/dist/lib/interfaces/IHandlerFilter.js +3 -0
  83. package/dist/lib/interfaces/ILifeCycle.js +3 -0
  84. package/dist/lib/interfaces/IPlugin.js +3 -0
  85. package/dist/lib/interfaces/IPushScheduler.js +3 -0
  86. package/dist/lib/interfaces/ISocket.js +3 -0
  87. package/dist/lib/interfaces/IStore.js +3 -0
  88. package/dist/lib/interfaces/define.js +3 -0
  89. package/dist/lib/master/master.js +129 -0
  90. package/dist/lib/master/starter.js +236 -0
  91. package/dist/lib/master/watchdog.js +120 -0
  92. package/dist/lib/melo.js +125 -0
  93. package/dist/lib/modules/console.js +436 -0
  94. package/dist/lib/modules/masterwatcher.js +98 -0
  95. package/dist/lib/modules/monitorwatcher.js +124 -0
  96. package/dist/lib/modules/onlineUser.js +69 -0
  97. package/dist/lib/modules/restartNotifyModule.js +107 -0
  98. package/dist/lib/modules/watchServer.js +737 -0
  99. package/dist/lib/monitor/monitor.js +80 -0
  100. package/dist/lib/pushSchedulers/buffer.js +96 -0
  101. package/dist/lib/pushSchedulers/direct.js +58 -0
  102. package/dist/lib/pushSchedulers/multi.js +80 -0
  103. package/dist/lib/server/server.js +500 -0
  104. package/dist/lib/util/appUtil.js +306 -0
  105. package/dist/lib/util/constants.js +117 -0
  106. package/dist/lib/util/countDownLatch.js +51 -0
  107. package/dist/lib/util/events.js +20 -0
  108. package/dist/lib/util/handlerHelper.js +8 -0
  109. package/dist/lib/util/log.js +14 -0
  110. package/dist/lib/util/moduleUtil.js +101 -0
  111. package/dist/lib/util/pathUtil.js +134 -0
  112. package/dist/lib/util/remoterHelper.js +8 -0
  113. package/dist/lib/util/utils.js +358 -0
  114. package/dist/test/application.js +522 -0
  115. package/dist/test/config/log4js.json +28 -0
  116. package/dist/test/config/master.json +9 -0
  117. package/dist/test/config/servers.json +6 -0
  118. package/dist/test/filters/handler/serial.js +41 -0
  119. package/dist/test/filters/handler/time.js +41 -0
  120. package/dist/test/filters/handler/timeout.js +41 -0
  121. package/dist/test/filters/handler/toobusy.js +57 -0
  122. package/dist/test/filters/rpc/rpcLog.js +22 -0
  123. package/dist/test/filters/rpc/toobusy.js +39 -0
  124. package/dist/test/manager/mockChannelManager.js +77 -0
  125. package/dist/test/manager/taskManager.js +68 -0
  126. package/dist/test/mock-base/app/servers/other-file +0 -0
  127. package/dist/test/mock-plugin/components/mockPlugin.js +10 -0
  128. package/dist/test/mock-plugin/events/mockEvent.js +12 -0
  129. package/dist/test/modules/console.js +242 -0
  130. package/dist/test/pomelo.js +19 -0
  131. package/dist/test/remote/channelRemote.js +159 -0
  132. package/dist/test/service/channel.js +134 -0
  133. package/dist/test/service/channelService.js +216 -0
  134. package/dist/test/service/connectionService.js +114 -0
  135. package/dist/test/service/filterService.js +144 -0
  136. package/dist/test/service/handlerService.js +65 -0
  137. package/dist/test/service/sessionService.js +387 -0
  138. package/dist/test/util/countDownLatch.js +70 -0
  139. package/dist/test/util/pathUtil.js +108 -0
  140. package/dist/test/util/utils.js +140 -0
  141. package/lib/application.ts +1240 -0
  142. package/lib/common/manager/appManager.ts +118 -0
  143. package/lib/common/manager/taskManager.ts +50 -0
  144. package/lib/common/remote/backend/msgRemote.ts +134 -0
  145. package/lib/common/remote/frontend/channelRemote.ts +91 -0
  146. package/lib/common/remote/frontend/sessionRemote.ts +91 -0
  147. package/lib/common/service/backendSessionService.ts +388 -0
  148. package/lib/common/service/channelService.ts +609 -0
  149. package/lib/common/service/connectionService.ts +112 -0
  150. package/lib/common/service/filterService.ts +118 -0
  151. package/lib/common/service/handlerService.ts +224 -0
  152. package/lib/common/service/sessionService.ts +731 -0
  153. package/lib/components/backendSession.ts +14 -0
  154. package/lib/components/channel.ts +11 -0
  155. package/lib/components/connection.ts +13 -0
  156. package/lib/components/connector.ts +533 -0
  157. package/lib/components/dictionary.ts +121 -0
  158. package/lib/components/master.ts +41 -0
  159. package/lib/components/monitor.ts +30 -0
  160. package/lib/components/protobuf.ts +208 -0
  161. package/lib/components/proxy.ts +282 -0
  162. package/lib/components/pushScheduler.ts +70 -0
  163. package/lib/components/remote.ts +166 -0
  164. package/lib/components/server.ts +71 -0
  165. package/lib/components/session.ts +22 -0
  166. package/lib/connectors/commands/handshake.ts +155 -0
  167. package/lib/connectors/commands/heartbeat.ts +83 -0
  168. package/lib/connectors/commands/kick.ts +11 -0
  169. package/lib/connectors/common/coder.ts +93 -0
  170. package/lib/connectors/common/handler.ts +62 -0
  171. package/lib/connectors/hybrid/IHybridSocket.ts +9 -0
  172. package/lib/connectors/hybrid/switcher.ts +142 -0
  173. package/lib/connectors/hybrid/tcpprocessor.ts +43 -0
  174. package/lib/connectors/hybrid/tcpsocket.ts +223 -0
  175. package/lib/connectors/hybrid/wsprocessor.ts +57 -0
  176. package/lib/connectors/hybridconnector.ts +134 -0
  177. package/lib/connectors/hybridsocket.ts +168 -0
  178. package/lib/connectors/mqtt/generate.ts +103 -0
  179. package/lib/connectors/mqtt/mqttadaptor.ts +114 -0
  180. package/lib/connectors/mqtt/protocol.ts +49 -0
  181. package/lib/connectors/mqttconnector.ts +134 -0
  182. package/lib/connectors/mqttsocket.ts +79 -0
  183. package/lib/connectors/sioconnector.ts +161 -0
  184. package/lib/connectors/siosocket.ts +85 -0
  185. package/lib/connectors/udpconnector.ts +113 -0
  186. package/lib/connectors/udpsocket.ts +110 -0
  187. package/lib/filters/handler/serial.ts +46 -0
  188. package/lib/filters/handler/time.ts +35 -0
  189. package/lib/filters/handler/timeout.ts +50 -0
  190. package/lib/filters/handler/toobusy.ts +37 -0
  191. package/lib/filters/rpc/rpcLog.ts +42 -0
  192. package/lib/filters/rpc/toobusy.ts +41 -0
  193. package/lib/index.ts +74 -0
  194. package/lib/interfaces/IComponent.ts +47 -0
  195. package/lib/interfaces/IConnector.ts +20 -0
  196. package/lib/interfaces/IHandlerFilter.ts +15 -0
  197. package/lib/interfaces/ILifeCycle.ts +16 -0
  198. package/lib/interfaces/IPlugin.ts +65 -0
  199. package/lib/interfaces/IPushScheduler.ts +52 -0
  200. package/lib/interfaces/ISocket.ts +26 -0
  201. package/lib/interfaces/IStore.ts +10 -0
  202. package/lib/interfaces/define.ts +15 -0
  203. package/lib/master/master.ts +148 -0
  204. package/lib/master/starter.ts +234 -0
  205. package/lib/master/watchdog.ts +135 -0
  206. package/lib/melo.ts +152 -0
  207. package/lib/modules/console.ts +465 -0
  208. package/lib/modules/masterwatcher.ts +120 -0
  209. package/lib/modules/monitorwatcher.ts +151 -0
  210. package/lib/modules/onlineUser.ts +78 -0
  211. package/lib/modules/restartNotifyModule.ts +128 -0
  212. package/lib/modules/watchServer.ts +766 -0
  213. package/lib/monitor/monitor.ts +99 -0
  214. package/lib/pushSchedulers/buffer.ts +117 -0
  215. package/lib/pushSchedulers/direct.ts +65 -0
  216. package/lib/pushSchedulers/multi.ts +94 -0
  217. package/lib/server/server.ts +554 -0
  218. package/lib/util/appUtil.ts +313 -0
  219. package/lib/util/constants.ts +154 -0
  220. package/lib/util/countDownLatch.ts +72 -0
  221. package/lib/util/events.ts +15 -0
  222. package/lib/util/handlerHelper.ts +5 -0
  223. package/lib/util/log.ts +11 -0
  224. package/lib/util/moduleUtil.ts +110 -0
  225. package/lib/util/pathUtil.ts +132 -0
  226. package/lib/util/remoterHelper.ts +68 -0
  227. package/lib/util/utils.ts +365 -0
  228. package/package.json +93 -0
  229. package/template/game-server/.vscode/launch.json +27 -0
  230. package/template/game-server/app/servers/connector/handler/entryHandler.ts +50 -0
  231. package/template/game-server/app/servers/connector/remote/authRemoter.ts +36 -0
  232. package/template/game-server/app.ts +30 -0
  233. package/template/game-server/app.ts.mqtt +32 -0
  234. package/template/game-server/app.ts.sio +36 -0
  235. package/template/game-server/app.ts.sio.wss +34 -0
  236. package/template/game-server/app.ts.udp +31 -0
  237. package/template/game-server/app.ts.wss +40 -0
  238. package/template/game-server/config/adminServer.json +5 -0
  239. package/template/game-server/config/adminUser.json +22 -0
  240. package/template/game-server/config/clientProtos.json +1 -0
  241. package/template/game-server/config/dictionary.json +1 -0
  242. package/template/game-server/config/log4js.json +150 -0
  243. package/template/game-server/config/master.json +8 -0
  244. package/template/game-server/config/serverProtos.json +1 -0
  245. package/template/game-server/config/servers.json +12 -0
  246. package/template/game-server/copy.js +5 -0
  247. package/template/game-server/package.json +28 -0
  248. package/template/game-server/preload.ts +40 -0
  249. package/template/game-server/tsconfig.json +36 -0
  250. package/template/npm-install.bat +4 -0
  251. package/template/npm-install.sh +5 -0
  252. package/template/readme.md +31 -0
  253. package/template/shared/server.crt +15 -0
  254. package/template/shared/server.key +15 -0
  255. package/template/web-server/app.js +25 -0
  256. package/template/web-server/app.js.https +36 -0
  257. package/template/web-server/bin/component.bat +1 -0
  258. package/template/web-server/bin/component.sh +1 -0
  259. package/template/web-server/package.json +10 -0
  260. package/template/web-server/public/css/base.css +76 -0
  261. package/template/web-server/public/image/logo.png +0 -0
  262. package/template/web-server/public/image/sp.png +0 -0
  263. package/template/web-server/public/index.html +57 -0
  264. package/template/web-server/public/index.html.sio +58 -0
  265. package/template/web-server/public/js/lib/build/build.js +1730 -0
  266. package/template/web-server/public/js/lib/build/build.js.wss +1721 -0
  267. package/template/web-server/public/js/lib/component.json +6 -0
  268. package/template/web-server/public/js/lib/local/boot/component.json +11 -0
  269. package/template/web-server/public/js/lib/local/boot/index.js +11 -0
  270. package/template/web-server/public/js/lib/meloclient.js +456 -0
  271. package/template/web-server/public/js/lib/meloclient.js.wss +456 -0
  272. package/template/web-server/public/js/lib/socket.io.js +3 -0
  273. package/test/application.ts +607 -0
  274. package/test/filters/handler/serial.ts +47 -0
  275. package/test/filters/handler/time.ts +47 -0
  276. package/test/filters/handler/timeout.ts +46 -0
  277. package/test/filters/handler/toobusy.ts +59 -0
  278. package/test/filters/rpc/rpcLog.ts +20 -0
  279. package/test/filters/rpc/toobusy.ts +40 -0
  280. package/test/manager/mockChannelManager.ts +92 -0
  281. package/test/manager/taskManager.ts +78 -0
  282. package/test/mock-base/app/servers/other-file +0 -0
  283. package/test/mock-plugin/components/mockPlugin.ts +8 -0
  284. package/test/mock-plugin/events/mockEvent.ts +12 -0
  285. package/test/modules/console.ts +264 -0
  286. package/test/pomelo.ts +18 -0
  287. package/test/remote/channelRemote.ts +176 -0
  288. package/test/service/channel.ts +161 -0
  289. package/test/service/channelService.ts +243 -0
  290. package/test/service/connectionService.ts +142 -0
  291. package/test/service/filterService.ts +165 -0
  292. package/test/service/handlerService.ts +77 -0
  293. package/test/service/sessionService.ts +464 -0
  294. package/test/util/countDownLatch.ts +81 -0
  295. package/test/util/pathUtil.ts +122 -0
  296. package/test/util/utils.ts +165 -0
@@ -0,0 +1,465 @@
1
+ /*!
2
+ * Melo -- consoleModule serverStop stop/kill
3
+ * Copyright(c) 2012 fantasyni <fantasyni@163.com>
4
+ * MIT Licensed
5
+ */
6
+ import { getLogger } from '@bigtyphoon/melo-logger';
7
+ import * as countDownLatch from '../util/countDownLatch';
8
+ import * as utils from '../util/utils';
9
+ import * as Constants from '../util/constants';
10
+ import * as starter from '../master/starter';
11
+ import { exec } from 'child_process';
12
+ import { Application } from '../application';
13
+ import { IModule, MonitorCallback, MasterAgent, MasterCallback } from '@bigtyphoon/melo-admin';
14
+ import { MonitorAgent } from '@bigtyphoon/melo-admin';
15
+ import { ServerInfo } from '../util/constants';
16
+ import * as path from 'path';
17
+ import * as os from 'os';
18
+
19
+ let logger = getLogger('melo', path.basename(__filename));
20
+
21
+
22
+ export interface ConsoleModuleOptions {
23
+ app?: Application;
24
+
25
+ }
26
+
27
+ export class ConsoleModule implements IModule {
28
+ app: Application;
29
+
30
+ static moduleId = '__console__';
31
+
32
+ constructor(opts: ConsoleModuleOptions) {
33
+ opts = opts || {};
34
+ this.app = opts.app;
35
+ }
36
+
37
+ monitorHandler(agent: MonitorAgent, msg: any, cb: MonitorCallback) {
38
+ let serverId = agent.id;
39
+ switch (msg.signal) {
40
+ case 'stop':
41
+ if (agent.type === Constants.RESERVED.MASTER) {
42
+ return;
43
+ }
44
+ this.app.stop(true);
45
+ break;
46
+ case 'list':
47
+ let serverType = agent.type;
48
+ let pid = process.pid;
49
+ let heapUsed = (process.memoryUsage().heapUsed / (1024 * 1024)).toFixed(2);
50
+ let rss = (process.memoryUsage().rss / (1024 * 1024)).toFixed(2);
51
+ let heapTotal = (process.memoryUsage().heapTotal / (1024 * 1024)).toFixed(2);
52
+ let uptime = (process.uptime() / 60).toFixed(2);
53
+ utils.invokeCallback(cb, {
54
+ serverId: serverId,
55
+ body: {
56
+ serverId: serverId,
57
+ serverType: serverType,
58
+ pid: pid,
59
+ rss: rss,
60
+ heapTotal: heapTotal,
61
+ heapUsed: heapUsed,
62
+ uptime: uptime
63
+ }
64
+ });
65
+ break;
66
+ case 'kill':
67
+ utils.invokeCallback(cb, serverId);
68
+ if (agent.type !== 'master') {
69
+ setTimeout(function () {
70
+ process.exit(-1);
71
+ }, Constants.TIME.TIME_WAIT_MONITOR_KILL);
72
+ }
73
+ break;
74
+ case 'addCron':
75
+ this.app.addCrons([msg.cron]);
76
+ break;
77
+ case 'removeCron':
78
+ this.app.removeCrons([msg.cron]);
79
+ break;
80
+ case 'blacklist':
81
+ if (this.app.isFrontend()) {
82
+ let connector = this.app.components.__connector__;
83
+ connector.blacklist = connector.blacklist.concat(msg.blacklist);
84
+ }
85
+ break;
86
+ case 'restart':
87
+ if (agent.type === Constants.RESERVED.MASTER) {
88
+ return;
89
+ }
90
+ let self = this;
91
+ let server = this.app.get(Constants.RESERVED.CURRENT_SERVER);
92
+ utils.invokeCallback(cb, server);
93
+ process.nextTick(function () {
94
+ self.app.stop(true);
95
+ });
96
+ break;
97
+ default:
98
+ logger.error('receive error signal: %j', msg);
99
+ break;
100
+ }
101
+ }
102
+
103
+ clientHandler(agent: MasterAgent, msg: any, cb: MasterCallback) {
104
+ let app = this.app;
105
+ switch (msg.signal) {
106
+ case 'kill':
107
+ kill(app, agent, msg, cb);
108
+ break;
109
+ case 'stop':
110
+ stop(app, agent, msg, cb);
111
+ break;
112
+ case 'list':
113
+ list(app, agent, msg, cb);
114
+ break;
115
+ case 'add':
116
+ add(app, agent, msg, cb);
117
+ break;
118
+ case 'addCron':
119
+ addCron(app, agent, msg, cb);
120
+ break;
121
+ case 'removeCron':
122
+ removeCron(app, agent, msg, cb);
123
+ break;
124
+ case 'blacklist':
125
+ blacklist(app, agent, msg, cb);
126
+ break;
127
+ case 'restart':
128
+ restart(app, agent, msg, cb);
129
+ break;
130
+ default:
131
+ utils.invokeCallback(cb, new Error('The command cannot be recognized, please check.'), null);
132
+ break;
133
+ }
134
+ }
135
+ }
136
+
137
+ let kill = function (app: Application, agent: MasterAgent, msg: any, cb: MasterCallback) {
138
+ let sid, record;
139
+ let serverIds: string[] = [];
140
+ let count = Object.keys(agent.idMap).length;
141
+ let latch = countDownLatch.createCountDownLatch(count, { timeout: Constants.TIME.TIME_WAIT_MASTER_KILL }, function (isTimeout) {
142
+ if (!isTimeout) {
143
+ utils.invokeCallback(cb, null, { code: 'ok' });
144
+ } else {
145
+ utils.invokeCallback(cb, null, { code: 'remained', serverIds: serverIds });
146
+ }
147
+ setTimeout(function () {
148
+ process.exit(-1);
149
+ }, Constants.TIME.TIME_WAIT_MONITOR_KILL);
150
+ });
151
+
152
+ let agentRequestCallback = function (msg: string) {
153
+ for (let i = 0; i < serverIds.length; ++i) {
154
+ if (serverIds[i] === msg) {
155
+ serverIds.splice(i, 1);
156
+ latch.done();
157
+ break;
158
+ }
159
+ }
160
+ };
161
+
162
+ for (sid in agent.idMap) {
163
+ record = agent.idMap[sid];
164
+ serverIds.push(record.id);
165
+ agent.request(record.id, ConsoleModule.moduleId, { signal: msg.signal }, agentRequestCallback);
166
+ }
167
+ };
168
+
169
+ let stop = function (app: Application, agent: MasterAgent, msg: any, cb: MasterCallback) {
170
+ let serverIds = msg.ids;
171
+ if (!!serverIds.length) {
172
+ let servers = app.getServers();
173
+ app.set(Constants.RESERVED.STOP_SERVERS, serverIds);
174
+ for (let i = 0; i < serverIds.length; i++) {
175
+ let serverId = serverIds[i];
176
+ if (!servers[serverId]) {
177
+ utils.invokeCallback(cb, new Error('Cannot find the server to stop.'), null);
178
+ } else {
179
+ agent.notifyById(serverId, ConsoleModule.moduleId, { signal: msg.signal });
180
+ }
181
+ }
182
+ utils.invokeCallback(cb, null, { status: 'part' });
183
+ } else {
184
+ let servers = app.getServers();
185
+ let serverIds: any = [];
186
+ for (let key in servers) {
187
+ serverIds.push(key);
188
+ }
189
+ app.set(Constants.RESERVED.STOP_SERVERS, serverIds);
190
+ agent.notifyAll(ConsoleModule.moduleId, { signal: msg.signal });
191
+ setTimeout(function () {
192
+ app.stop(true);
193
+ utils.invokeCallback(cb, null, { status: 'all' });
194
+ }, Constants.TIME.TIME_WAIT_STOP);
195
+ }
196
+ };
197
+
198
+ let restart = function (app: Application, agent: MasterAgent, msg: any, cb: MasterCallback) {
199
+ let successFlag: boolean;
200
+ let successIds: string[] = [];
201
+ let serverIds = msg.ids;
202
+ let type = msg.type;
203
+ let servers;
204
+ if (!serverIds.length && !!type) {
205
+ servers = app.getServersByType(type);
206
+ if (!servers) {
207
+ utils.invokeCallback(cb, new Error('restart servers with unknown server type: ' + type));
208
+ return;
209
+ }
210
+ for (let i = 0; i < servers.length; i++) {
211
+ serverIds.push(servers[i].id);
212
+ }
213
+ } else if (!serverIds.length) {
214
+ servers = app.getServers();
215
+ for (let key in servers) {
216
+ serverIds.push(key);
217
+ }
218
+ }
219
+ let count = serverIds.length;
220
+ let latch = countDownLatch.createCountDownLatch(count, { timeout: Constants.TIME.TIME_WAIT_COUNTDOWN }, function () {
221
+ if (!successFlag) {
222
+ utils.invokeCallback(cb, new Error('all servers start failed.'));
223
+ return;
224
+ }
225
+ utils.invokeCallback(cb, null, utils.arrayDiff(serverIds, successIds));
226
+ });
227
+
228
+ let request = function (id: string) {
229
+ return (function () {
230
+ agent.request(id, ConsoleModule.moduleId, { signal: msg.signal }, function (msg) {
231
+ if (!msg || !Object.keys(msg).length) {
232
+ latch.done();
233
+ return;
234
+ }
235
+ setTimeout(function () {
236
+ runServer(app, msg, function (err, status) {
237
+ if (!!err) {
238
+ logger.error('restart ' + id + ' failed.', err.message, 'status:', status);
239
+ } else {
240
+ successIds.push(id);
241
+ successFlag = true;
242
+ }
243
+ latch.done();
244
+ });
245
+ }, Constants.TIME.TIME_WAIT_RESTART);
246
+ });
247
+ })();
248
+ };
249
+
250
+ for (let j = 0; j < serverIds.length; j++) {
251
+ request(serverIds[j]);
252
+ }
253
+ };
254
+
255
+ let list = function (app: Application, agent: MasterAgent, msg: any, cb: MasterCallback) {
256
+ let sid, record;
257
+ let serverInfo: any = {};
258
+ let count = Object.keys(agent.idMap).length;
259
+ let latch = countDownLatch.createCountDownLatch(count, { timeout: Constants.TIME.TIME_WAIT_COUNTDOWN }, function () {
260
+ utils.invokeCallback(cb, null, { msg: serverInfo });
261
+ });
262
+
263
+ let callback = function (msg: { serverId: string, body: any }) {
264
+ serverInfo[msg.serverId] = msg.body;
265
+ latch.done();
266
+ };
267
+ for (sid in agent.idMap) {
268
+ record = agent.idMap[sid];
269
+ agent.request(record.id, ConsoleModule.moduleId, { signal: msg.signal }, callback);
270
+ }
271
+ };
272
+
273
+ let add = function (app: Application, agent: MasterAgent, msg: any, cb: MasterCallback) {
274
+ if (checkCluster(msg)) {
275
+ startCluster(app, msg, cb);
276
+ } else {
277
+ startServer(app, msg, cb);
278
+ }
279
+ reset(ServerInfo);
280
+ };
281
+
282
+ let addCron = function (app: Application, agent: MasterAgent, msg: any, cb: MasterCallback) {
283
+ let cron = parseArgs(msg, CronInfo, cb);
284
+ sendCronInfo(cron, agent, msg, CronInfo, cb);
285
+ };
286
+
287
+ let removeCron = function (app: Application, agent: MasterAgent, msg: any, cb: MasterCallback) {
288
+ let cron = parseArgs(msg, RemoveCron, cb);
289
+ sendCronInfo(cron, agent, msg, RemoveCron, cb);
290
+ };
291
+
292
+ let blacklist = function (app: Application, agent: MasterAgent, msg: any, cb: MasterCallback) {
293
+ let ips = msg.args;
294
+ for (let i = 0; i < ips.length; i++) {
295
+ if (!(new RegExp(/(\d+)\.(\d+)\.(\d+)\.(\d+)/g).test(ips[i]))) {
296
+ utils.invokeCallback(cb, new Error('blacklist ip: ' + ips[i] + ' is error format.'), null);
297
+ return;
298
+ }
299
+ }
300
+ agent.notifyAll(ConsoleModule.moduleId, { signal: msg.signal, blacklist: msg.args });
301
+ process.nextTick(function () {
302
+ cb(null, { status: 'ok' });
303
+ });
304
+ };
305
+
306
+ let parseArgs = function (msg: any, info: any, cb: (err?: string | Error, data?: any) => void) {
307
+ let rs: { [key: string]: string } = {};
308
+ let args = msg.args;
309
+ for (let i = 0; i < args.length; i++) {
310
+ if (args[i].indexOf('=') < 0) {
311
+ cb(new Error('Error server parameters format.'), null);
312
+ return;
313
+ }
314
+ let pairs = args[i].split('=');
315
+ let key = pairs[0];
316
+ if (!!info[key]) {
317
+ info[key] = 1;
318
+ }
319
+ rs[pairs[0]] = pairs[1];
320
+ }
321
+ return rs;
322
+ };
323
+
324
+ let sendCronInfo = function (cron: any, agent: MasterAgent, msg: any, info: any, cb: Function) {
325
+ if (isReady(info) && (cron.serverId || cron.serverType)) {
326
+ if (!!cron.serverId) {
327
+ agent.notifyById(cron.serverId, ConsoleModule.moduleId, { signal: msg.signal, cron: cron });
328
+ } else {
329
+ agent.notifyByType(cron.serverType, ConsoleModule.moduleId, { signal: msg.signal, cron: cron });
330
+ }
331
+ process.nextTick(function () {
332
+ cb(null, { status: 'ok' });
333
+ });
334
+ } else {
335
+ cb(new Error('Miss necessary server parameters.'), null);
336
+ }
337
+ reset(info);
338
+ };
339
+
340
+ let startServer = function (app: Application, msg: any, cb: (err?: Error | string, result?: any) => void) {
341
+ let server = parseArgs(msg, ServerInfo, cb);
342
+ if (isReady(ServerInfo)) {
343
+ runServer(app, server as any, cb);
344
+ } else {
345
+ cb(new Error('Miss necessary server parameters.'), null);
346
+ }
347
+ };
348
+
349
+ let runServer = function (app: Application, server: ServerInfo, cb: (err?: Error, result?: any) => void) {
350
+ utils.checkPort(app, server, function (status) {
351
+ if (status === 'busy') {
352
+ utils.invokeCallback(cb, new Error('Port occupied already, check your server to add.'));
353
+ } else {
354
+ starter.run(app, server, function (err) {
355
+ if (err) {
356
+ err = String(err)
357
+ const checkErrCorrect = 'https://nodejs.org/en/docs/inspector'
358
+ const idx = err.indexOf(checkErrCorrect)
359
+ if (idx === -1 || idx + checkErrCorrect.length + 10 < err.length) {
360
+ utils.invokeCallback(cb, new Error(err), null);
361
+ return;
362
+ }
363
+ }
364
+ });
365
+ process.nextTick(function () {
366
+ utils.invokeCallback(cb, null, { status: 'ok' });
367
+ });
368
+ }
369
+ });
370
+ };
371
+
372
+ let startCluster = function (app: Application, msg: any, cb: MasterCallback) {
373
+ let serverMap = {};
374
+ let fails: ServerInfo[] = [];
375
+ let successFlag: boolean;
376
+ let serverInfo = parseArgs(msg, ClusterInfo, cb) as any;
377
+ utils.loadCluster(app, serverInfo, serverMap);
378
+ let count = Object.keys(serverMap).length;
379
+ let latch = countDownLatch.createCountDownLatch(count, () => {
380
+ if (!successFlag) {
381
+ utils.invokeCallback(cb, new Error('all servers start failed.'));
382
+ return;
383
+ }
384
+ utils.invokeCallback(cb, null, fails);
385
+ });
386
+
387
+ let start = function (server: ServerInfo) {
388
+ return (function () {
389
+ utils.checkPort(app, server, function (status) {
390
+ if (status === 'busy') {
391
+ fails.push(server);
392
+ latch.done();
393
+ } else {
394
+ starter.run(app, server, function (err) {
395
+ if (err) {
396
+ fails.push(server);
397
+ if(latch.count) {
398
+ latch.done();
399
+ }
400
+ }
401
+ });
402
+ process.nextTick(function () {
403
+ successFlag = true;
404
+ if(latch.count) {
405
+ latch.done();
406
+ }
407
+ });
408
+ }
409
+ });
410
+ })();
411
+ };
412
+ for (let key in serverMap) {
413
+ let server = (serverMap as any)[key];
414
+ start(server);
415
+ }
416
+ };
417
+
418
+ let checkCluster = function (msg: any) {
419
+ let flag = false;
420
+ let args = msg.args;
421
+ for (let i = 0; i < args.length; i++) {
422
+ if (utils.startsWith(args[i], Constants.RESERVED.CLUSTER_COUNT)) {
423
+ flag = true;
424
+ }
425
+ }
426
+ return flag;
427
+ };
428
+
429
+ let isReady = function (info: any) {
430
+ for (let key in info) {
431
+ if (info[key]) {
432
+ return false;
433
+ }
434
+ }
435
+ return true;
436
+ };
437
+
438
+ let reset = function (info: any) {
439
+ for (let key in info) {
440
+ info[key] = 0;
441
+ }
442
+ };
443
+
444
+ let ServerInfo = {
445
+ host: 0,
446
+ port: 0,
447
+ id: 0,
448
+ serverType: 0
449
+ };
450
+
451
+ let CronInfo = {
452
+ id: 0,
453
+ action: 0,
454
+ time: 0
455
+ };
456
+
457
+ let RemoveCron = {
458
+ id: 0
459
+ };
460
+
461
+ let ClusterInfo = {
462
+ host: 0,
463
+ port: 0,
464
+ clusterCount: 0
465
+ };
@@ -0,0 +1,120 @@
1
+ import { getLogger } from '@bigtyphoon/melo-logger';
2
+ import * as utils from '../util/utils';
3
+ import * as Constants from '../util/constants';
4
+ import { Watchdog} from '../master/watchdog';
5
+ import { Application } from '../application';
6
+ import { IModule, ConsoleService, MasterAgent, MasterCallback } from '@bigtyphoon/melo-admin';
7
+ import * as path from 'path';
8
+ let logger = getLogger('melo', path.basename(__filename));
9
+
10
+
11
+
12
+ export class MasterWatcherModule implements IModule {
13
+ app: Application;
14
+ service: ConsoleService;
15
+ id: string;
16
+ watchdog: Watchdog;
17
+
18
+ static moduleId = Constants.KEYWORDS.MASTER_WATCHER;
19
+
20
+
21
+ constructor(opts: {app: Application}, consoleService: ConsoleService) {
22
+ this.app = opts.app;
23
+ this.service = consoleService;
24
+ this.id = this.app.getServerId();
25
+
26
+ this.watchdog = new Watchdog(this.app, this.service);
27
+ this.service.on('register', this.onServerAdd.bind(this));
28
+ this.service.on('disconnect', this.onServerLeave.bind(this));
29
+ this.service.on('reconnect', this.onServerReconnect.bind(this));
30
+ }
31
+
32
+ // ----------------- bind methods -------------------------
33
+
34
+ onServerAdd(record: any) {
35
+ logger.debug('masterwatcher receive add server event, with server: %j', record);
36
+ if (!record || record.type === 'client' || !record.serverType) {
37
+ return;
38
+ }
39
+ this.watchdog.addServer(record);
40
+ }
41
+
42
+ onServerReconnect(record: any) {
43
+ logger.debug('masterwatcher receive reconnect server event, with server: %j', record);
44
+ if (!record || record.type === 'client' || !record.serverType) {
45
+ logger.warn('onServerReconnect receive wrong message: %j', record);
46
+ return;
47
+ }
48
+ this.watchdog.reconnectServer(record);
49
+ }
50
+
51
+ onServerLeave(id: string, type: string) {
52
+ logger.debug('masterwatcher receive remove server event, with server: %s, type: %s', id, type);
53
+ if (!id) {
54
+ logger.warn('onServerLeave receive server id is empty.');
55
+ return;
56
+ }
57
+ if (type !== 'client') {
58
+ this.watchdog.removeServer(id);
59
+ }
60
+ }
61
+
62
+ // ----------------- module methods -------------------------
63
+
64
+ start(cb: () => void) {
65
+ utils.invokeCallback(cb);
66
+ }
67
+
68
+ masterHandler(agent: MasterAgent, msg: any, cb: MasterCallback) {
69
+ if (!msg) {
70
+ logger.warn('masterwatcher receive empty message.');
71
+ return;
72
+ }
73
+ let func = (masterMethods as any)[msg.action];
74
+ if (!func) {
75
+ logger.info('masterwatcher unknown action: %j', msg.action);
76
+ return;
77
+ }
78
+ func(this, agent, msg, cb);
79
+ }
80
+ }
81
+
82
+ // ----------------- monitor request methods -------------------------
83
+
84
+ let subscribe = function (module: MasterWatcherModule, agent: MasterAgent, msg: any, cb: MasterCallback) {
85
+ if (!msg) {
86
+ utils.invokeCallback(cb, new Error('masterwatcher subscribe empty message.'));
87
+ return;
88
+ }
89
+
90
+ module.watchdog.subscribe(msg.id);
91
+ utils.invokeCallback(cb, null, module.watchdog.query());
92
+ };
93
+
94
+ let unsubscribe = function (module: MasterWatcherModule, agent: MasterAgent, msg: any, cb: MasterCallback) {
95
+ if (!msg) {
96
+ utils.invokeCallback(cb, new Error('masterwatcher unsubscribe empty message.'));
97
+ return;
98
+ }
99
+ module.watchdog.unsubscribe(msg.id);
100
+ utils.invokeCallback(cb);
101
+ };
102
+
103
+ let query = function (module: MasterWatcherModule, agent: MasterAgent, msg: any, cb: MasterCallback) {
104
+ utils.invokeCallback(cb, null, module.watchdog.query());
105
+ };
106
+
107
+ let record = function (module: MasterWatcherModule, agent: MasterAgent, msg: any, cb: MasterCallback) {
108
+ if (!msg) {
109
+ utils.invokeCallback(cb, new Error('masterwatcher record empty message.'));
110
+ return;
111
+ }
112
+ module.watchdog.record(msg.id);
113
+ };
114
+
115
+ let masterMethods = {
116
+ 'subscribe': subscribe,
117
+ 'unsubscribe': unsubscribe,
118
+ 'query': query,
119
+ 'record': record
120
+ };