UNPKG

fusion-plugin-rpc

Version:

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

226 lines (222 loc) 27.9 kB
/** 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 */ import bodyparser from 'koa-bodyparser'; import formidable from 'formidable'; import { createPlugin, memoize, RouteTagsToken } from 'fusion-core'; import { UniversalEventsToken } from 'fusion-plugin-universal-events'; import MissingHandlerError from './missing-handler-error'; import ResponseError from './response-error'; import { BodyParserOptionsToken, RPCHandlersToken, RPCHandlersConfigToken } from './tokens'; import { formatApiPath } from './utils'; 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(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 = () => createPlugin({ deps: { RouteTags: RouteTagsToken.optional, emitter: UniversalEventsToken, handlers: RPCHandlersToken, bodyParserOptions: BodyParserOptionsToken.optional, rpcConfig: RPCHandlersConfigToken.optional }, provides: deps => { const { emitter, handlers } = deps; const service = { from: 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 = bodyparser(bodyParserOptions); const apiPath = 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.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 ? 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(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); } export default true && pluginFactory(); //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["bodyparser","formidable","createPlugin","memoize","RouteTagsToken","UniversalEventsToken","MissingHandlerError","ResponseError","BodyParserOptionsToken","RPCHandlersToken","RPCHandlersConfigToken","formatApiPath","statKey","hasHandler","handlers","method","Object","prototype","hasOwnProperty","call","RPC","constructor","emitter","ctx","headers","Error","request","args","startTime","ms","scopedEmitter","from","e","emit","origin","error","result","status","timing","pluginFactory","deps","RouteTags","optional","bodyParserOptions","rpcConfig","provides","service","middleware","parseBody","apiPath","next","routeTags","path","startsWith","pathMatch","RegExp","match","name","body","req","indexOf","form","IncomingForm","Promise","resolve","reject","parse","err","fields","files","data","message","code","type","meta","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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"}