fusion-plugin-rpc
Version:
Fetch data on the server and client with an RPC style interface.
272 lines (227 loc) • 28.1 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = void 0;
var _koaBodyparser = _interopRequireDefault(require("koa-bodyparser"));
var _formidable = _interopRequireDefault(require("formidable"));
var _fusionCore = require("fusion-core");
var _fusionPluginUniversalEvents = require("fusion-plugin-universal-events");
var _missingHandlerError = _interopRequireDefault(require("./missing-handler-error"));
var _responseError = _interopRequireDefault(require("./response-error"));
var _tokens = require("./tokens.js");
var _utils = require("./utils.js");
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
/** Copyright (c) 2018 Uber Technologies, Inc.
*
* This source code is licensed under the MIT license found in the
* LICENSE file in the root directory of this source tree.
*
*
*/
/* eslint-env node */
const statKey = 'rpc:method';
/* Helper function */
function hasHandler(handlers, method) {
return Object.prototype.hasOwnProperty.call(handlers, method);
}
class RPC {
constructor(emitter, handlers, ctx) {
if (!ctx || !ctx.headers) {
throw new Error('fusion-plugin-rpc requires `ctx`');
}
this.ctx = ctx;
this.emitter = emitter;
this.handlers = handlers;
return this;
}
async request(method, args) {
const startTime = ms();
if (!this.ctx) {
throw new Error('fusion-plugin-rpc requires `ctx`');
}
if (!this.emitter) {
throw new Error('fusion-plugin-rpc requires `emitter`');
}
const scopedEmitter = this.emitter.from(this.ctx);
if (!this.handlers) {
throw new Error('fusion-plugin-rpc requires `handlers`');
}
if (!hasHandler(this.handlers, method)) {
const e = new _missingHandlerError.default(method);
if (scopedEmitter) {
scopedEmitter.emit('rpc:error', {
method,
origin: 'server',
error: e
});
}
throw e;
}
try {
const result = await this.handlers[method](args, this.ctx);
if (scopedEmitter) {
scopedEmitter.emit(statKey, {
method,
status: 'success',
origin: 'server',
timing: ms() - startTime
});
}
return result;
} catch (e) {
if (scopedEmitter) {
scopedEmitter.emit(statKey, {
method,
error: e,
status: 'failure',
origin: 'server',
timing: ms() - startTime
});
}
throw e;
}
}
}
const pluginFactory = () => (0, _fusionCore.createPlugin)({
deps: {
RouteTags: _fusionCore.RouteTagsToken.optional,
emitter: _fusionPluginUniversalEvents.UniversalEventsToken,
handlers: _tokens.RPCHandlersToken,
bodyParserOptions: _tokens.BodyParserOptionsToken.optional,
rpcConfig: _tokens.RPCHandlersConfigToken.optional
},
provides: deps => {
const {
emitter,
handlers
} = deps;
const service = {
from: (0, _fusionCore.memoize)(ctx => new RPC(emitter, handlers, ctx))
};
return service;
},
middleware: deps => {
const {
emitter,
handlers,
bodyParserOptions,
rpcConfig
} = deps;
if (!handlers) throw new Error('Missing handlers registered to RPCHandlersToken');
if (!emitter) throw new Error('Missing emitter registered to UniversalEventsToken');
const parseBody = (0, _koaBodyparser.default)(bodyParserOptions);
const apiPath = (0, _utils.formatApiPath)(rpcConfig && rpcConfig.apiPath ? rpcConfig.apiPath : 'api');
return async (ctx, next) => {
await next();
const routeTags = deps.RouteTags && deps.RouteTags.from(ctx) || {};
const scopedEmitter = emitter.from(ctx);
if (ctx.method === 'POST' && ctx.path.startsWith(apiPath)) {
const startTime = ms(); // eslint-disable-next-line no-useless-escape
const pathMatch = new RegExp(`${apiPath}([^/]+)`, 'i');
const [, method] = ctx.path.match(pathMatch) || [];
if (hasHandler(handlers, method)) {
routeTags.name = method;
let body;
try {
if (ctx.req && ctx.req.headers && ctx.req.headers['content-type'] && ctx.req.headers['content-type'].indexOf('multipart/form-data') !== -1) {
const form = new _formidable.default.IncomingForm();
body = await new Promise((resolve, reject) => {
form.parse(ctx.req, (err, fields, files) => {
if (err) {
reject(err);
}
resolve({ ...fields,
...files
});
});
});
} else {
await parseBody(ctx, () => Promise.resolve());
}
} catch (e) {
ctx.body = {
status: 'failure',
data: {
message: e.message,
code: e.type || 'ERR_BAD_BODY',
meta: e.meta
}
};
if (scopedEmitter) {
scopedEmitter.emit(statKey, {
method,
error: e,
status: 'failure',
origin: 'browser',
timing: ms() - startTime
});
} // don't try to call handler
return;
}
try {
const result = await handlers[method](body || ctx.request.body, ctx);
ctx.body = {
status: 'success',
data: result
};
if (scopedEmitter) {
scopedEmitter.emit(statKey, {
method,
status: 'success',
origin: 'browser',
timing: ms() - startTime
});
}
} catch (e) {
const error = e instanceof _responseError.default ? e : new Error(process.env.NODE_ENV !== "production" ? 'UnknownError - Use ResponseError from fusion-plugin-rpc (or fusion-plugin-rpc-redux-react if you are using React) package for more detailed error messages' : 'Internal Server Error');
ctx.body = {
status: 'failure',
data: {
message: error.message,
// $FlowFixMe
code: error.code,
// $FlowFixMe
meta: error.meta
}
};
if (scopedEmitter) {
scopedEmitter.emit(statKey, {
method,
error: e,
status: 'failure',
origin: 'browser',
timing: ms() - startTime
});
}
}
} else {
const e = new _missingHandlerError.default(method);
ctx.body = {
status: 'failure',
data: {
message: e.message,
code: e.code
}
};
ctx.status = 404;
if (scopedEmitter) {
scopedEmitter.emit('rpc:error', {
origin: 'browser',
method,
error: e
});
}
}
}
};
}
});
/* Helper functions */
function ms() {
const [seconds, ns] = process.hrtime();
return Math.round(seconds * 1000 + ns / 1e6);
}
var _default = false && pluginFactory();
exports.default = _default;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["src/server.js"],"names":["statKey","hasHandler","handlers","method","Object","prototype","hasOwnProperty","call","RPC","constructor","emitter","ctx","headers","Error","request","args","startTime","ms","scopedEmitter","from","e","MissingHandlerError","emit","origin","error","result","status","timing","pluginFactory","deps","RouteTags","RouteTagsToken","optional","UniversalEventsToken","RPCHandlersToken","bodyParserOptions","BodyParserOptionsToken","rpcConfig","RPCHandlersConfigToken","provides","service","middleware","parseBody","apiPath","next","routeTags","path","startsWith","pathMatch","RegExp","match","name","body","req","indexOf","form","formidable","IncomingForm","Promise","resolve","reject","parse","err","fields","files","data","message","code","type","meta","ResponseError","seconds","ns","process","hrtime","Math","round"],"mappings":"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","sourcesContent":["/** Copyright (c) 2018 Uber Technologies, Inc.\n *\n * This source code is licensed under the MIT license found in the\n * LICENSE file in the root directory of this source tree.\n *\n * @flow\n */\n\n/* eslint-env node */\n\nimport bodyparser from 'koa-bodyparser';\nimport formidable from 'formidable';\n\nimport {createPlugin, memoize, RouteTagsToken} from 'fusion-core';\nimport type {Context} from 'fusion-core';\nimport {UniversalEventsToken} from 'fusion-plugin-universal-events';\nimport type {Fetch} from 'fusion-tokens';\n\nimport MissingHandlerError from './missing-handler-error';\nimport ResponseError from './response-error';\nimport {\n  BodyParserOptionsToken,\n  RPCHandlersToken,\n  RPCHandlersConfigToken,\n} from './tokens.js';\nimport type {HandlerType} from './tokens.js';\nimport type {RPCPluginType, IEmitter} from './types.js';\nimport {formatApiPath} from './utils.js';\n\nconst statKey = 'rpc:method';\n\n/* Helper function */\nfunction hasHandler(handlers: HandlerType, method: string): boolean {\n  return Object.prototype.hasOwnProperty.call(handlers, method);\n}\n\nclass RPC {\n  ctx: ?Context;\n  emitter: ?IEmitter;\n  handlers: ?HandlerType;\n  fetch: ?Fetch;\n\n  constructor(emitter: IEmitter, handlers: any, ctx: Context): RPC {\n    if (!ctx || !ctx.headers) {\n      throw new Error('fusion-plugin-rpc requires `ctx`');\n    }\n    this.ctx = ctx;\n    this.emitter = emitter;\n    this.handlers = handlers;\n\n    return this;\n  }\n\n  async request<TArgs, TResult>(method: string, args: TArgs): Promise<TResult> {\n    const startTime = ms();\n\n    if (!this.ctx) {\n      throw new Error('fusion-plugin-rpc requires `ctx`');\n    }\n    if (!this.emitter) {\n      throw new Error('fusion-plugin-rpc requires `emitter`');\n    }\n    const scopedEmitter = this.emitter.from(this.ctx);\n\n    if (!this.handlers) {\n      throw new Error('fusion-plugin-rpc requires `handlers`');\n    }\n    if (!hasHandler(this.handlers, method)) {\n      const e = new MissingHandlerError(method);\n      if (scopedEmitter) {\n        scopedEmitter.emit('rpc:error', {\n          method,\n          origin: 'server',\n          error: e,\n        });\n      }\n      throw e;\n    }\n    try {\n      const result = await this.handlers[method](args, this.ctx);\n      if (scopedEmitter) {\n        scopedEmitter.emit(statKey, {\n          method,\n          status: 'success',\n          origin: 'server',\n          timing: ms() - startTime,\n        });\n      }\n      return result;\n    } catch (e) {\n      if (scopedEmitter) {\n        scopedEmitter.emit(statKey, {\n          method,\n          error: e,\n          status: 'failure',\n          origin: 'server',\n          timing: ms() - startTime,\n        });\n      }\n      throw e;\n    }\n  }\n}\n\nconst pluginFactory: () => RPCPluginType = () =>\n  createPlugin({\n    deps: {\n      RouteTags: RouteTagsToken.optional,\n      emitter: UniversalEventsToken,\n      handlers: RPCHandlersToken,\n      bodyParserOptions: BodyParserOptionsToken.optional,\n      rpcConfig: RPCHandlersConfigToken.optional,\n    },\n\n    provides: (deps) => {\n      const {emitter, handlers} = deps;\n\n      const service = {\n        from: memoize((ctx) => new RPC(emitter, handlers, ctx)),\n      };\n      return service;\n    },\n\n    middleware: (deps) => {\n      const {emitter, handlers, bodyParserOptions, rpcConfig} = deps;\n      if (!handlers)\n        throw new Error('Missing handlers registered to RPCHandlersToken');\n      if (!emitter)\n        throw new Error('Missing emitter registered to UniversalEventsToken');\n      const parseBody = bodyparser(bodyParserOptions);\n\n      const apiPath = formatApiPath(\n        rpcConfig && rpcConfig.apiPath ? rpcConfig.apiPath : 'api'\n      );\n\n      return async (ctx, next) => {\n        await next();\n        const routeTags = (deps.RouteTags && deps.RouteTags.from(ctx)) || {};\n        const scopedEmitter = emitter.from(ctx);\n        if (ctx.method === 'POST' && ctx.path.startsWith(apiPath)) {\n          const startTime = ms();\n          // eslint-disable-next-line no-useless-escape\n          const pathMatch = new RegExp(`${apiPath}([^/]+)`, 'i');\n          const [, method] = ctx.path.match(pathMatch) || [];\n          if (hasHandler(handlers, method)) {\n            routeTags.name = method;\n            let body;\n            try {\n              if (\n                ctx.req &&\n                ctx.req.headers &&\n                ctx.req.headers['content-type'] &&\n                ctx.req.headers['content-type'].indexOf(\n                  'multipart/form-data'\n                ) !== -1\n              ) {\n                const form = new formidable.IncomingForm();\n                body = await new Promise((resolve, reject) => {\n                  form.parse(ctx.req, (err, fields: {[string]: any}, files) => {\n                    if (err) {\n                      reject(err);\n                    }\n\n                    resolve({\n                      ...fields,\n                      ...files,\n                    });\n                  });\n                });\n              } else {\n                await parseBody(ctx, () => Promise.resolve());\n              }\n            } catch (e) {\n              ctx.body = {\n                status: 'failure',\n                data: {\n                  message: e.message,\n                  code: e.type || 'ERR_BAD_BODY',\n                  meta: e.meta,\n                },\n              };\n              if (scopedEmitter) {\n                scopedEmitter.emit(statKey, {\n                  method,\n                  error: e,\n                  status: 'failure',\n                  origin: 'browser',\n                  timing: ms() - startTime,\n                });\n              }\n              // don't try to call handler\n              return;\n            }\n\n            try {\n              const result = await handlers[method](\n                body || ctx.request.body,\n                ctx\n              );\n              ctx.body = {\n                status: 'success',\n                data: result,\n              };\n              if (scopedEmitter) {\n                scopedEmitter.emit(statKey, {\n                  method,\n                  status: 'success',\n                  origin: 'browser',\n                  timing: ms() - startTime,\n                });\n              }\n            } catch (e) {\n              const error =\n                e instanceof ResponseError\n                  ? e\n                  : new Error(\n                      __DEV__\n                        ? 'UnknownError - Use ResponseError from fusion-plugin-rpc (or fusion-plugin-rpc-redux-react if you are using React) package for more detailed error messages'\n                        : 'Internal Server Error'\n                    );\n              ctx.body = {\n                status: 'failure',\n                data: {\n                  message: error.message,\n                  // $FlowFixMe\n                  code: error.code,\n                  // $FlowFixMe\n                  meta: error.meta,\n                },\n              };\n              if (scopedEmitter) {\n                scopedEmitter.emit(statKey, {\n                  method,\n                  error: e,\n                  status: 'failure',\n                  origin: 'browser',\n                  timing: ms() - startTime,\n                });\n              }\n            }\n          } else {\n            const e = new MissingHandlerError(method);\n            ctx.body = {\n              status: 'failure',\n              data: {\n                message: e.message,\n                code: e.code,\n              },\n            };\n            ctx.status = 404;\n            if (scopedEmitter) {\n              scopedEmitter.emit('rpc:error', {\n                origin: 'browser',\n                method,\n                error: e,\n              });\n            }\n          }\n        }\n      };\n    },\n  });\n\n/* Helper functions */\nfunction ms() {\n  const [seconds, ns] = process.hrtime();\n  return Math.round(seconds * 1000 + ns / 1e6);\n}\n\nexport default ((__NODE__ && pluginFactory(): any): RPCPluginType);\n"]}