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