fusion-plugin-rpc
Version:
Fetch data on the server and client with an RPC style interface.
253 lines (220 loc) • 28.1 kB
JavaScript
/** 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.js';
import { formatApiPath } from './utils.js';
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,
// $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(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 false && pluginFactory();
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["src/server.js"],"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"],"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"]}