@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,500 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.manualReloadCrons = exports.Server = exports.create = void 0;
4
+ /**
5
+ * Implementation of server component.
6
+ * Init and start server instance.
7
+ */
8
+ const melo_logger_1 = require("@bigtyphoon/melo-logger");
9
+ const path = require("path");
10
+ const pathUtil = require("../util/pathUtil");
11
+ const Loader = require("@bigtyphoon/melo-loader");
12
+ const melo_loader_1 = require("@bigtyphoon/melo-loader");
13
+ const utils = require("../util/utils");
14
+ const schedule = require("@bigtyphoon/melo-scheduler");
15
+ const events_1 = require("../util/events");
16
+ const Constants = require("../util/constants");
17
+ const filterService_1 = require("../common/service/filterService");
18
+ const handlerService_1 = require("../common/service/handlerService");
19
+ const events_2 = require("events");
20
+ let logger = (0, melo_logger_1.getLogger)('melo', path.basename(__filename));
21
+ let ST_INITED = 0; // server inited
22
+ let ST_STARTED = 1; // server started
23
+ let ST_STOPED = 2; // server stoped
24
+ /**
25
+ * Server factory function.
26
+ *
27
+ * @param {Object} app current application context
28
+ * @return {Object} erver instance
29
+ */
30
+ function create(app, opts) {
31
+ return new Server(app, opts);
32
+ }
33
+ exports.create = create;
34
+ class Server extends events_2.EventEmitter {
35
+ constructor(app, opts) {
36
+ super();
37
+ this.globalFilterService = null;
38
+ this.filterService = null;
39
+ this.handlerService = null;
40
+ this.cronHandlers = null;
41
+ this.crons = [];
42
+ this.jobs = {};
43
+ this.state = ST_INITED;
44
+ this.opts = opts || {};
45
+ this.app = app;
46
+ app.event.on(events_1.default.ADD_CRONS, this.addCrons.bind(this));
47
+ app.event.on(events_1.default.REMOVE_CRONS, this.removeCrons.bind(this));
48
+ }
49
+ /**
50
+ * Server lifecycle callback
51
+ */
52
+ start() {
53
+ if (this.state > ST_INITED) {
54
+ return;
55
+ }
56
+ this.globalFilterService = initFilter(true, this.app);
57
+ this.filterService = initFilter(false, this.app);
58
+ this.handlerService = initHandler(this.app, this.opts);
59
+ this.loadCrons();
60
+ this.state = ST_STARTED;
61
+ }
62
+ loadCrons(manualReload = false, clear = false) {
63
+ if (manualReload) {
64
+ logger.info('loadCrons remove crons', this.crons);
65
+ this.removeCrons(this.crons);
66
+ if (clear) {
67
+ this.crons = [];
68
+ }
69
+ }
70
+ this.cronHandlers = loadCronHandlers(this.app, manualReload);
71
+ loadCrons(this, this.app, manualReload);
72
+ if (manualReload) {
73
+ scheduleCrons(this, this.crons);
74
+ }
75
+ }
76
+ afterStart() {
77
+ scheduleCrons(this, this.crons);
78
+ }
79
+ /**
80
+ * Stop server
81
+ */
82
+ stop() {
83
+ this.state = ST_STOPED;
84
+ }
85
+ /**
86
+ * Global handler.
87
+ *
88
+ * @param {Object} msg request message
89
+ * @param {Object} session session object
90
+ * @param {Callback} callback function
91
+ */
92
+ globalHandle(msg, session, cb) {
93
+ if (this.state !== ST_STARTED) {
94
+ utils.invokeCallback(cb, new Error('server not started'));
95
+ return;
96
+ }
97
+ let routeRecord = parseRoute(msg.route);
98
+ if (!routeRecord) {
99
+ utils.invokeCallback(cb, new Error(`meet unknown route message ${msg.route}`));
100
+ return;
101
+ }
102
+ if (routeRecord.method === 'constructor') {
103
+ logger.warn('attack session:', session, msg);
104
+ this.app.sessionService.kickBySessionId(session.id, 'attack');
105
+ return;
106
+ }
107
+ let self = this;
108
+ let dispatch = function (err, resp) {
109
+ if (err) {
110
+ handleError(true, self, err, msg, session, resp, function (err, resp) {
111
+ response(true, self, err, routeRecord, msg, session, resp, cb);
112
+ });
113
+ return;
114
+ }
115
+ if (self.app.getServerType() !== routeRecord.serverType) {
116
+ doForward(self.app, msg, session, routeRecord, function (err, resp) {
117
+ response(true, self, err, routeRecord, msg, session, resp, cb);
118
+ });
119
+ }
120
+ else {
121
+ doHandle(self, msg, session, routeRecord, function (err, resp) {
122
+ response(true, self, err, routeRecord, msg, session, resp, cb);
123
+ });
124
+ }
125
+ };
126
+ beforeFilter(true, self, routeRecord, msg, session, dispatch);
127
+ }
128
+ /**
129
+ * Handle request
130
+ */
131
+ handle(msg, session, cb) {
132
+ if (this.state !== ST_STARTED) {
133
+ cb(new Error('server not started'));
134
+ return;
135
+ }
136
+ let routeRecord = parseRoute(msg.route);
137
+ doHandle(this, msg, session, routeRecord, cb);
138
+ }
139
+ /**
140
+ * Add crons at runtime.
141
+ *
142
+ * @param {Array} crons would be added in application
143
+ */
144
+ addCrons(crons) {
145
+ this.cronHandlers = loadCronHandlers(this.app);
146
+ for (let i = 0, l = crons.length; i < l; i++) {
147
+ let cron = crons[i];
148
+ checkAndAdd(cron, this.crons, this);
149
+ }
150
+ scheduleCrons(this, crons);
151
+ }
152
+ /**
153
+ * Remove crons at runtime.
154
+ *
155
+ * @param {Array} crons would be removed in application
156
+ */
157
+ removeCrons(crons) {
158
+ for (let i = 0, l = crons.length; i < l; i++) {
159
+ let cron = crons[i];
160
+ let id = cron.id;
161
+ if (!!this.jobs[id]) {
162
+ schedule.cancelJob(this.jobs[id]);
163
+ delete this.jobs[id];
164
+ }
165
+ else {
166
+ logger.warn('cron is not in application: %j', cron);
167
+ }
168
+ }
169
+ }
170
+ }
171
+ exports.Server = Server;
172
+ // 重置 crons 缓存, 手动添加的crons只会取消任务重新加载任务。
173
+ // clear 控制重新加载之前是否先清除原有的.
174
+ // 有的 cron 是通过运行时动态添加进来的. 所以不能直接清除, 所以只能添加选项自己来控制
175
+ function manualReloadCrons(app, clear = false) {
176
+ if (!app.components.__server__) {
177
+ return;
178
+ }
179
+ logger.info('manualReloadCrons start');
180
+ app.components.__server__.server.loadCrons(true, clear);
181
+ logger.info('manualReloadCrons finish');
182
+ }
183
+ exports.manualReloadCrons = manualReloadCrons;
184
+ let initFilter = function (isGlobal, app) {
185
+ let service = new filterService_1.FilterService();
186
+ let befores, afters;
187
+ if (isGlobal) {
188
+ befores = app.get(Constants.KEYWORDS.GLOBAL_BEFORE_FILTER);
189
+ afters = app.get(Constants.KEYWORDS.GLOBAL_AFTER_FILTER);
190
+ }
191
+ else {
192
+ befores = app.get(Constants.KEYWORDS.BEFORE_FILTER);
193
+ afters = app.get(Constants.KEYWORDS.AFTER_FILTER);
194
+ }
195
+ let i, l;
196
+ if (befores) {
197
+ for (i = 0, l = befores.length; i < l; i++) {
198
+ service.before(befores[i]);
199
+ }
200
+ }
201
+ if (afters) {
202
+ for (i = 0, l = afters.length; i < l; i++) {
203
+ service.after(afters[i]);
204
+ }
205
+ }
206
+ return service;
207
+ };
208
+ let initHandler = function (app, opts) {
209
+ return new handlerService_1.HandlerService(app, opts);
210
+ };
211
+ /**
212
+ * Load cron handlers from current application
213
+ */
214
+ let loadCronHandlers = function (app, manualReload = false) {
215
+ let all = {};
216
+ let p = pathUtil.getCronPath(app.getBase(), app.getServerType());
217
+ if (p) {
218
+ let crons = Loader.load(p, app, manualReload, true, melo_loader_1.LoaderPathType.PINUS_CRONNER);
219
+ for (let name in crons) {
220
+ all[name] = crons[name];
221
+ }
222
+ }
223
+ for (let plugin of app.usedPlugins) {
224
+ if (plugin.cronPath) {
225
+ if (!_checkCanRequire(plugin.cronPath)) {
226
+ logger.error(`插件[${plugin.name}的cronPath[${plugin.cronPath}不存在。]]`);
227
+ continue;
228
+ }
229
+ let crons = Loader.load(plugin.cronPath, app, manualReload, true, melo_loader_1.LoaderPathType.PINUS_CRONNER);
230
+ for (let name in crons) {
231
+ all[name] = crons[name];
232
+ }
233
+ }
234
+ }
235
+ return all;
236
+ };
237
+ const clearRequireCache = function (path) {
238
+ const moduleObj = require.cache[path];
239
+ if (!moduleObj) {
240
+ return;
241
+ }
242
+ if (moduleObj.parent) {
243
+ moduleObj.parent.children.splice(moduleObj.parent.children.indexOf(moduleObj), 1);
244
+ }
245
+ delete require.cache[path];
246
+ };
247
+ function _checkCanRequire(path, manualReload = false) {
248
+ try {
249
+ path = require.resolve(path);
250
+ if (manualReload) {
251
+ clearRequireCache(path);
252
+ }
253
+ }
254
+ catch (err) {
255
+ return null;
256
+ }
257
+ return path;
258
+ }
259
+ /**
260
+ * Load crons from configure file
261
+ */
262
+ let loadCrons = function (server, app, manualReload = false) {
263
+ let env = app.get(Constants.RESERVED.ENV);
264
+ let p = path.join(app.getBase(), Constants.FILEPATH.CRON);
265
+ if (!_checkCanRequire(p, manualReload)) {
266
+ p = path.join(app.getBase(), Constants.FILEPATH.CONFIG_DIR, env, path.basename(Constants.FILEPATH.CRON));
267
+ if (!_checkCanRequire(p, manualReload)) {
268
+ return;
269
+ }
270
+ }
271
+ app.loadConfigBaseApp(Constants.RESERVED.CRONS, Constants.FILEPATH.CRON);
272
+ let crons = app.get(Constants.RESERVED.CRONS);
273
+ for (let serverType in crons) {
274
+ if (app.serverType === serverType) {
275
+ let list = crons[serverType];
276
+ for (let i = 0; i < list.length; i++) {
277
+ if (!list[i].serverId) {
278
+ checkAndAdd(list[i], server.crons, server, manualReload);
279
+ }
280
+ else {
281
+ if (app.serverId === list[i].serverId) {
282
+ checkAndAdd(list[i], server.crons, server, manualReload);
283
+ }
284
+ }
285
+ }
286
+ }
287
+ }
288
+ };
289
+ /**
290
+ * Fire before filter chain if any
291
+ */
292
+ let beforeFilter = function (isGlobal, server, routeRecord, msg, session, cb) {
293
+ let fm;
294
+ if (isGlobal) {
295
+ fm = server.globalFilterService;
296
+ }
297
+ else {
298
+ fm = server.filterService;
299
+ }
300
+ if (fm) {
301
+ fm.beforeFilter(routeRecord, msg, session, cb);
302
+ }
303
+ else {
304
+ utils.invokeCallback(cb);
305
+ }
306
+ };
307
+ /**
308
+ * Fire after filter chain if have
309
+ */
310
+ let afterFilter = function (isGlobal, server, err, routeRecord, msg, session, resp, cb) {
311
+ let fm;
312
+ if (isGlobal) {
313
+ fm = server.globalFilterService;
314
+ }
315
+ else {
316
+ fm = server.filterService;
317
+ }
318
+ if (fm) {
319
+ if (isGlobal) {
320
+ fm.afterFilter(err, routeRecord, msg, session, resp, function () {
321
+ // do nothing
322
+ });
323
+ }
324
+ else {
325
+ fm.afterFilter(err, routeRecord, msg, session, resp, function (err) {
326
+ cb(err, resp);
327
+ });
328
+ }
329
+ }
330
+ };
331
+ /**
332
+ * pass err to the global error handler if specified
333
+ */
334
+ let handleError = function (isGlobal, server, err, msg, session, resp, cb) {
335
+ let handler;
336
+ if (isGlobal) {
337
+ handler = server.app.get(Constants.RESERVED.GLOBAL_ERROR_HANDLER);
338
+ }
339
+ else {
340
+ handler = server.app.get(Constants.RESERVED.ERROR_HANDLER);
341
+ }
342
+ if (!handler) {
343
+ logger.error(`${server.app.serverId} no default error handler msg[${JSON.stringify(msg)}] to resolve unknown exception: sessionId:${JSON.stringify(session.export())} , error stack: ${err.stack}`);
344
+ utils.invokeCallback(cb, err, resp);
345
+ }
346
+ else {
347
+ if (handler.length === 5) {
348
+ handler(err, msg, resp, session, cb);
349
+ }
350
+ else {
351
+ handler(err, msg, resp, session, cb);
352
+ }
353
+ }
354
+ };
355
+ /**
356
+ * Send response to client and fire after filter chain if any.
357
+ */
358
+ let response = function (isGlobal, server, err, routeRecord, msg, session, resp, cb) {
359
+ if (isGlobal) {
360
+ cb(err, resp);
361
+ // after filter should not interfere response
362
+ afterFilter(isGlobal, server, err, routeRecord, msg, session, resp, cb);
363
+ }
364
+ else {
365
+ afterFilter(isGlobal, server, err, routeRecord, msg, session, resp, cb);
366
+ }
367
+ };
368
+ /**
369
+ * Parse route string.
370
+ *
371
+ * @param {String} route route string, such as: serverName.handlerName.methodName
372
+ * @return {Object} parse result object or null for illeagle route string
373
+ */
374
+ let parseRoute = function (route) {
375
+ if (!route) {
376
+ return null;
377
+ }
378
+ let ts = route.split('.');
379
+ if (ts.length !== 3) {
380
+ return null;
381
+ }
382
+ return {
383
+ route: route,
384
+ serverType: ts[0],
385
+ handler: ts[1],
386
+ method: ts[2]
387
+ };
388
+ };
389
+ let doForward = function (app, msg, session, routeRecord, cb) {
390
+ let finished = false;
391
+ // should route to other servers
392
+ try {
393
+ app.sysrpc[routeRecord.serverType].msgRemote.forwardMessage(
394
+ // app.sysrpc[routeRecord.serverType].msgRemote.forwardMessage2(
395
+ session, msg,
396
+ // msg.oldRoute || msg.route,
397
+ // msg.body,
398
+ // msg.aesPassword,
399
+ // msg.compressGzip,
400
+ session.export()).then(function (resp) {
401
+ finished = true;
402
+ utils.invokeCallback(cb, null, resp);
403
+ }).catch(function (err) {
404
+ logger.error(app.serverId + ' fail to process remote message:' + err.stack);
405
+ utils.invokeCallback(cb, err);
406
+ });
407
+ }
408
+ catch (err) {
409
+ if (!finished) {
410
+ logger.error(app.serverId + ' fail to forward message:' + err.stack);
411
+ utils.invokeCallback(cb, err);
412
+ }
413
+ }
414
+ };
415
+ let doHandle = function (server, msg, session, routeRecord, cb) {
416
+ msg = msg.body || {};
417
+ let self = server;
418
+ let handle = function (err, resp) {
419
+ if (err) {
420
+ // error from before filter
421
+ handleError(false, self, err, msg, session, resp, function (err, resp) {
422
+ response(false, self, err, routeRecord, msg, session, resp, cb);
423
+ });
424
+ return;
425
+ }
426
+ self.handlerService.handle(routeRecord, msg, session, function (err, resp) {
427
+ if (err) {
428
+ // error from handler
429
+ handleError(false, self, err, msg, session, resp, function (err, resp) {
430
+ response(false, self, err, routeRecord, msg, session, resp, cb);
431
+ });
432
+ return;
433
+ }
434
+ response(false, self, err, routeRecord, msg, session, resp, cb);
435
+ });
436
+ }; // end of handle
437
+ beforeFilter(false, server, routeRecord, msg, session, handle);
438
+ };
439
+ /**
440
+ * Schedule crons
441
+ */
442
+ let scheduleCrons = function (server, crons) {
443
+ let handlers = server.cronHandlers;
444
+ for (let i = 0; i < crons.length; i++) {
445
+ let cronInfo = crons[i];
446
+ let time = cronInfo.time;
447
+ let action = cronInfo.action;
448
+ let jobId = cronInfo.id;
449
+ if (!time || !action || !jobId) {
450
+ logger.error(server.app.serverId + ' cron miss necessary parameters: %j', cronInfo);
451
+ continue;
452
+ }
453
+ if (action.indexOf('.') < 0) {
454
+ logger.error(server.app.serverId + ' cron action is error format: %j', cronInfo);
455
+ continue;
456
+ }
457
+ let cron = action.split('.')[0];
458
+ let job = action.split('.')[1];
459
+ let handler = handlers[cron];
460
+ if (!handler) {
461
+ logger.error('could not find cron: %j', cronInfo);
462
+ continue;
463
+ }
464
+ if (typeof handler[job] !== 'function') {
465
+ logger.error('could not find cron job: %j, %s', cronInfo, job);
466
+ continue;
467
+ }
468
+ let id = schedule.scheduleJob(time, handler[job].bind(handler));
469
+ server.jobs[jobId] = id;
470
+ }
471
+ };
472
+ /**
473
+ * If cron is not in crons then put it in the array.
474
+ */
475
+ let checkAndAdd = function (cron, crons, server, replace = false) {
476
+ const orgCron = containCron(cron.id, crons);
477
+ if (!orgCron) {
478
+ server.crons.push(cron);
479
+ }
480
+ else {
481
+ logger.warn('cron is duplicated: %j', cron);
482
+ if (replace) {
483
+ logger.warn('replace time and action org:%j, new:%j', orgCron, cron);
484
+ orgCron.time = cron.time;
485
+ orgCron.action = cron.action;
486
+ }
487
+ }
488
+ };
489
+ /**
490
+ * Check if cron is in crons.
491
+ */
492
+ let containCron = function (id, crons) {
493
+ for (let i = 0, l = crons.length; i < l; i++) {
494
+ if (id === crons[i].id) {
495
+ return crons[i];
496
+ }
497
+ }
498
+ return null;
499
+ };
500
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2VydmVyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vbGliL3NlcnZlci9zZXJ2ZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUE7OztHQUdHO0FBQ0gseURBQW9EO0FBQ3BELDZCQUE2QjtBQUM3Qiw2Q0FBNkM7QUFDN0Msa0RBQWtEO0FBQ2xELHlEQUF5RDtBQUN6RCx1Q0FBdUM7QUFDdkMsdURBQXVEO0FBQ3ZELDJDQUFtRDtBQUNuRCwrQ0FBK0M7QUFFL0MsbUVBQWdFO0FBQ2hFLHFFQUEwRztBQUUxRyxtQ0FBc0M7QUFHdEMsSUFBSSxNQUFNLEdBQUcsSUFBQSx1QkFBUyxFQUFDLE1BQU0sRUFBRSxJQUFJLENBQUMsUUFBUSxDQUFDLFVBQVUsQ0FBQyxDQUFDLENBQUM7QUFHMUQsSUFBSSxTQUFTLEdBQUcsQ0FBQyxDQUFDLENBQUksZ0JBQWdCO0FBQ3RDLElBQUksVUFBVSxHQUFHLENBQUMsQ0FBQyxDQUFHLGlCQUFpQjtBQUN2QyxJQUFJLFNBQVMsR0FBRyxDQUFDLENBQUMsQ0FBSSxnQkFBZ0I7QUFjdEM7Ozs7O0dBS0c7QUFDSCxTQUFnQixNQUFNLENBQUMsR0FBZ0IsRUFBRSxJQUFtQjtJQUN4RCxPQUFPLElBQUksTUFBTSxDQUFDLEdBQUcsRUFBRSxJQUFJLENBQUMsQ0FBQztBQUNqQyxDQUFDO0FBRkQsd0JBRUM7QUFFRCxNQUFhLE1BQU8sU0FBUSxxQkFBWTtJQVlwQyxZQUFZLEdBQWdCLEVBQUUsSUFBb0I7UUFDOUMsS0FBSyxFQUFFLENBQUM7UUFUWix3QkFBbUIsR0FBa0IsSUFBSSxDQUFDO1FBQzFDLGtCQUFhLEdBQWtCLElBQUksQ0FBQztRQUNwQyxtQkFBYyxHQUFtQixJQUFJLENBQUM7UUFDdEMsaUJBQVksR0FBNEQsSUFBSSxDQUFDO1FBQzdFLFVBQUssR0FBVyxFQUFFLENBQUM7UUFDbkIsU0FBSSxHQUFpQyxFQUFFLENBQUM7UUFDeEMsVUFBSyxHQUFHLFNBQVMsQ0FBQztRQUlkLElB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