UNPKG

@sex-pomelo/sex-pomelo

Version:

[![NPM version][npm-image-pomelo]][npm-url-pomelo] [![NPM version][npm-image-down]][npm-url-pomelo]

482 lines (427 loc) 12.6 kB
/** * Implementation of server component. * Init and start server instance. */ "use strict"; const reFilename = __filename.substring(__filename.indexOf("node_modules")) const logger = require('@sex-pomelo/sex-pomelo-logger').getLogger('pomelo', reFilename); const fs = require('fs'); const path = require('path'); const pathUtil = require('../util/pathUtil'); const Loader = require('pomelo-loader'); const utils = require('../util/utils'); const schedule = require('@sex-pomelo/sex-pomelo-scheduler'); const events = require('../util/events'); const Constants = require('../util/constants'); const FilterService = require('../common/service/filterService'); const HandlerService = require('../common/service/handlerService'); const Application = require('../application'); const ST_INITED = 0; // server inited const ST_STARTED = 1; // server started const ST_STOPED = 2; // server stoped /** * Server factory function. * * @param {Application} app current application context * @return {Server} server instance */ module.exports.create = function(app, opts) { return new Server(app, opts); }; class Server{ /** * * @param {Application} app application instance * @param {object} opts config */ constructor(app, opts){ this.opts = opts || {}; this.app = app; this.globalFilterService = null; this.filterService = null; this.handlerService = null; this.crons = []; this.jobs = {}; this.state = ST_INITED; app.event.on(events.ADD_CRONS, this.addCrons.bind(this)); app.event.on(events.REMOVE_CRONS, this.removeCrons.bind(this)); } /** * Server lifecycle callback */ start () { if(this.state > ST_INITED) { return; } this.globalFilterService = initFilter(true, this.app); this.filterService = initFilter(false, this.app); this.handlerService = initHandler(this.app, this.opts); this.cronHandlers = loadCronHandlers(this.app); loadCrons(this, this.app); this.state = ST_STARTED; } afterStart() { scheduleCrons(this, this.crons); }; /** * Stop server */ stop() { this.state = ST_STOPED; }; /** * Global handler. * * @param {Object} msg request message * @param {Object} session session object * @param {Callback} callback function */ globalHandle(msg, session, cb) { if(this.state !== ST_STARTED) { utils.invokeCallback(cb, new Error('server not started')); return; } let routeRecord = parseRoute(msg.route); if(!routeRecord) { utils.invokeCallback(cb, new Error('meet unknown route message %j', msg.route)); return; } if (routeRecord.method === 'constructor') { logger.warn('attack session:', session, msg); this.app.sessionService.kickBySessionId(session.id, 'attack'); return; } let self = this; let dispatch = function(err, resp, opts) { if(err) { handleError(true, self, err, msg, session, resp, opts, function(err, resp, opts) { response(true, self, err, msg, session, resp, opts, cb); }); return; } if(self.app.getServerType() !== routeRecord.serverType) { doForward(self.app, msg, session, routeRecord, function(err, resp, opts) { response(true, self, err, msg, session, resp, opts, cb); }); } else { doHandle(self, msg, session, routeRecord, function(err, resp, opts) { response(true, self, err, msg, session, resp, opts, cb); }); } }; beforeFilter(true, self, msg, session, dispatch); } /** * Handle request */ handle(msg, session, cb) { if(this.state !== ST_STARTED) { cb(new Error('server not started')); return; } let routeRecord = parseRoute(msg.route); doHandle(this, msg, session, routeRecord, cb); } /** * Add crons at runtime. * * @param {Array} crons would be added in application */ addCrons(crons) { this.cronHandlers = loadCronHandlers(this.app); for(let i=0, l=crons.length; i<l; i++) { let cron = crons[i]; checkAndAdd(cron, this.crons, this); } scheduleCrons(this, crons); } /** * Remove crons at runtime. * * @param {Array} crons would be removed in application */ removeCrons(crons) { for(let i=0, l=crons.length; i<l; i++) { let cron = crons[i]; let id = parseInt(cron.id); if(!!this.jobs[id]) { schedule.cancelJob(this.jobs[id]); } else { logger.warn('cron is not in application: %j', cron); } } } } let initFilter = function(isGlobal, app) { let service = new FilterService(); let befores, afters; if(isGlobal) { befores = app.get(Constants.KEYWORDS.GLOBAL_BEFORE_FILTER); afters = app.get(Constants.KEYWORDS.GLOBAL_AFTER_FILTER); } else { befores = app.get(Constants.KEYWORDS.BEFORE_FILTER); afters = app.get(Constants.KEYWORDS.AFTER_FILTER); } let i, l; if(befores) { for(i=0, l=befores.length; i<l; i++) { service.before(befores[i]); } } if(afters) { for(i=0, l=afters.length; i<l; i++) { service.after(afters[i]); } } return service; }; let initHandler = function(app, opts) { return new HandlerService(app, opts); }; /** * Load cron handlers from current application * @access private */ let loadCronHandlers = function(app) { let p = pathUtil.getCronPath(app.getBase(), app.getServerType()); if(p) { return Loader.load(p, app); } }; /** * Load crons from configure file * @access private */ let loadCrons = function(server, app) { let env = app.get(Constants.RESERVED.ENV); let p = path.join(app.getBase(), Constants.FILEPATH.CRON); if(!fs.existsSync(p)) { p = path.join(app.getBase(), Constants.FILEPATH.CONFIG_DIR, env, path.basename(Constants.FILEPATH.CRON)); if (!fs.existsSync(p)) { return; } } app.loadConfigBaseApp(Constants.RESERVED.CRONS, Constants.FILEPATH.CRON); let crons = app.get(Constants.RESERVED.CRONS); for(let serverType in crons) { if(app.serverType === serverType) { let list = crons[serverType]; for(let i = 0; i<list.length; i++) { if(!list[i].serverId) { checkAndAdd(list[i], server.crons, server); } else { if(app.serverId === list[i].serverId) { checkAndAdd(list[i], server.crons, server); } } } } } }; /** * Fire before filter chain if any * @access private */ let beforeFilter = function(isGlobal, server, msg, session, cb) { let fm; if(isGlobal) { fm = server.globalFilterService; } else { fm = server.filterService; } if(fm) { fm.beforeFilter(msg, session, cb); } else { utils.invokeCallback(cb); } }; /** * Fire after filter chain if have * @access private */ let afterFilter = function(isGlobal, server, err, msg, session, resp, opts, cb) { let fm; if(isGlobal) { fm = server.globalFilterService; } else { fm = server.filterService; } if(fm) { if(isGlobal) { fm.afterFilter(err, msg, session, resp, function() { // do nothing }); } else { fm.afterFilter(err, msg, session, resp, function(err) { cb(err, resp, opts); }); } } }; /** * pass err to the global error handler if specified * @access private */ let handleError = function(isGlobal, server, err, msg, session, resp, opts, cb) { let handler; if(isGlobal) { handler = server.app.get(Constants.RESERVED.GLOBAL_ERROR_HANDLER); } else { handler = server.app.get(Constants.RESERVED.ERROR_HANDLER); } if(!handler) { logger.debug('no default error handler to resolve unknown exception. ' + err.stack); utils.invokeCallback(cb, err, resp, opts); } else { if(handler.length === 5) { handler(err, msg, resp, session, cb); } else { handler(err, msg, resp, session, opts, cb); } } }; /** * Send response to client and fire after filter chain if any. * @access private */ let response = function(isGlobal, server, err, msg, session, resp, opts, cb) { if(isGlobal) { cb(err, resp, opts); // after filter should not interfere response afterFilter(isGlobal, server, err, msg, session, resp, opts, cb); } else { afterFilter(isGlobal, server, err, msg, session, resp, opts, cb); } }; /** * Parse route string. * @access private * @param {String} route route string, such as: serverName.handlerName.methodName * @return {Object} parse result object or null for illeagle route string */ let parseRoute = function(route) { if(!route) { return null; } let ts = route.split('.'); if(ts.length !== 3) { return null; } return { route: route, serverType: ts[0], handler: ts[1], method: ts[2] }; }; let doForward = function(app, msg, session, routeRecord, cb) { let finished = false; //should route to other servers try { app.sysrpc[routeRecord.serverType].msgRemote.forwardMessage( // app.sysrpc[routeRecord.serverType].msgRemote.forwardMessage2( session, msg, // msg.oldRoute || msg.route, // msg.body, // msg.aesPassword, // msg.compressGzip, session.export(), function(err, resp, opts) { if(err) { logger.error('fail to process remote message:' + err.stack); } finished = true; utils.invokeCallback(cb, err, resp, opts); } ); } catch(err) { if(!finished) { logger.error('fail to forward message:' + routeRecord.serverType + ' ' + err.stack); utils.invokeCallback(cb, err); } } }; let doHandle = function(server, msg, session, routeRecord, cb) { let originMsg = msg; msg = msg.body || {}; msg.__route__ = originMsg.route; let self = server; let handle = function(err, resp, opts) { if(err) { // error from before filter handleError(false, self, err, msg, session, resp, opts, function(err, resp, opts) { response(false, self, err, msg, session, resp, opts, cb); }); return; } self.handlerService.handle(routeRecord, msg, session, function(err, resp, opts) { if(err) { //error from handler handleError(false, self, err, msg, session, resp, opts, function(err, resp, opts) { response(false, self, err, msg, session, resp, opts, cb); }); return; } response(false, self, err, msg, session, resp, opts, cb); }); }; //end of handle beforeFilter(false, server, msg, session, handle); }; /** * Schedule crons * @access private */ let scheduleCrons = function(server, crons) { let handlers = server.cronHandlers; for(let i = 0; i<crons.length; i++) { let cronInfo = crons[i]; let time = cronInfo.time; let action = cronInfo.action; let jobId = cronInfo.id; if(!time || !action || !jobId) { logger.error('cron miss necessary parameters: %j', cronInfo); continue; } if(action.indexOf('.') < 0) { logger.error('cron action is error format: %j', cronInfo); continue; } let cron = action.split('.')[0]; let job = action.split('.')[1]; let handler = handlers[cron]; if(!handler) { logger.error('could not find cron: %j', cronInfo); continue; } if(typeof handler[job] !== 'function') { logger.error('could not find cron job: %j, %s', cronInfo, job); continue; } let id = schedule.scheduleJob(time, handler[job].bind(handler)); server.jobs[jobId] = id; } }; /** * If cron is not in crons then put it in the array. * @access private */ let checkAndAdd = function(cron, crons, server) { if(!containCron(cron.id, crons)) { server.crons.push(cron); } else { logger.warn('cron is duplicated: %j', cron); } }; /** * Check if cron is in crons. * @access private */ let containCron = function(id, crons) { for(let i=0, l=crons.length; i<l; i++) { if(id === crons[i].id) { return true; } } return false; };