occaecatidicta
Version:
118 lines (104 loc) • 4.17 kB
text/typescript
import { getLogger } from 'omelox-logger'; import { RouteRecord } from '../../util/constants';
import { HandlerCallback } from './handlerService';
import { BeforeHandlerFilter, AfterHandlerFilter } from '../../interfaces/IHandlerFilter';
import { FrontendOrBackendSession } from '../../server/server';
import * as path from 'path';
let logger = getLogger('omelox', path.basename(__filename));
/**
* Filter service.
* Register and fire before and after filters.
*/
export class FilterService {
befores: BeforeHandlerFilter[] = []; // before filters
afters: AfterHandlerFilter[] = []; // after filters
name = 'filter';
removeBefore(filter: BeforeHandlerFilter) {
let index = this.befores.findIndex(v => v === filter);
if (index >= 0) {
this.befores.splice(index);
}
}
removeAfter(filter: AfterHandlerFilter) {
let index = this.afters.findIndex(v => v === filter);
if (index >= 0) {
this.afters.splice(index);
}
}
/**
* Add before filter into the filter chain.
*
* @param filter {Object|Function} filter instance or filter function.
*/
before(filter: BeforeHandlerFilter) {
this.befores.push(filter);
}
/**
* Add after filter into the filter chain.
*
* @param filter {Object|Function} filter instance or filter function.
*/
after(filter: AfterHandlerFilter) {
this.afters.unshift(filter);
}
/**
* TODO: other insert method for filter? such as unshift
*/
/**
* Do the before filter.
* Fail over if any filter pass err parameter to the next function.
*
* @param msg {Object} clienet request msg
* @param session {Object} a session object for current request
* @param cb {Function} cb(err) callback function to invoke next chain node
*/
beforeFilter(routeRecord: RouteRecord , msg: any, session: FrontendOrBackendSession, cb: HandlerCallback) {
let index = 0, self = this;
let next = function (err?: any, resp?: any) {
if (err || index >= self.befores.length) {
cb(err, resp);
return;
}
let handler = self.befores[index++];
if (typeof handler === 'function') {
handler(routeRecord , msg, session, next);
} else if (typeof handler.before === 'function') {
handler.before(routeRecord , msg, session, next);
} else {
logger.error('meet invalid before filter, handler or handler.before should be function.');
next(new Error('invalid before filter.'));
}
}; // end of next
next();
}
/**
* Do after filter chain.
* Give server a chance to do clean up jobs after request responsed.
* After filter can not change the request flow before.
* After filter should call the next callback to let the request pass to next after filter.
*
* @param err {Object} error object
* @param session {Object} session object for current request
* @param {Object} resp response object send to client
* @param cb {Function} cb(err) callback function to invoke next chain node
*/
afterFilter(err: Error, routeRecord: RouteRecord , msg: any, session: FrontendOrBackendSession, resp: any, cb: HandlerCallback) {
let index = 0, self = this;
function next(err: Error) {
// if done
if (index >= self.afters.length) {
cb(err);
return;
}
let handler = self.afters[index++];
if (typeof handler === 'function') {
handler(err, routeRecord , msg, session, resp, next);
} else if (typeof handler.after === 'function') {
handler.after(err, routeRecord , msg, session, resp, next);
} else {
logger.error('meet invalid after filter, handler or handler.after should be function.');
next(new Error('invalid after filter.'));
}
} // end of next
next(err);
}
}