UNPKG

fusion-plugin-rpc

Version:

Fetch data on the server and client with an RPC style interface.

234 lines (229 loc) 28 kB
"use strict"; 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"); var _utils = require("./utils"); 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, // @ts-expect-error code: error.code, // @ts-expect-error 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 = true && pluginFactory(); exports.default = _default; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"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","createPlugin","deps","RouteTags","RouteTagsToken","optional","UniversalEventsToken","RPCHandlersToken","bodyParserOptions","BodyParserOptionsToken","rpcConfig","RPCHandlersConfigToken","provides","service","memoize","middleware","parseBody","bodyparser","apiPath","formatApiPath","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"],"sources":["src/server.ts"],"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 */\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';\nimport type {HandlerType} from './tokens';\nimport type {RPCPluginType, IEmitter} from './types';\nimport {formatApiPath} from './utils';\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 | undefined | null;\n  emitter: IEmitter | undefined | null;\n  handlers: HandlerType | undefined | null;\n  fetch: Fetch | undefined | null;\n\n  constructor(emitter: IEmitter, handlers: any, ctx: Context) {\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(\n                    ctx.req,\n                    (\n                      err,\n                      fields: {\n                        [x: string]: any;\n                      },\n                      files\n                    ) => {\n                      if (err) {\n                        reject(err);\n                      }\n\n                      resolve({\n                        ...fields,\n                        ...files,\n                      });\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                  // @ts-expect-error\n                  code: error.code,\n                  // @ts-expect-error\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() as any as RPCPluginType);\n"],"mappings":"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"}