fusion-plugin-rpc
Version:
Fetch data on the server and client with an RPC style interface.
125 lines (114 loc) • 14.8 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 browser */
import { createPlugin, memoize } from 'fusion-core';
import { UniversalEventsToken } from 'fusion-plugin-universal-events';
import { I18nToken } from 'fusion-plugin-i18n';
import { FetchToken } from 'fusion-tokens';
import { RPCHandlersConfigToken, RPCQueryParamsToken } from './tokens.js';
import { formatApiPath } from './utils.js';
const statKey = 'rpc:method-client';
class RPC {
constructor({
fetch,
emitter,
rpcConfig,
queryParams
}) {
this.fetch = fetch;
this.config = rpcConfig || {};
this.emitter = emitter;
this.queryParams = queryParams;
this.apiPath = formatApiPath(rpcConfig && rpcConfig.apiPath ? rpcConfig.apiPath : 'api');
}
request(rpcId, args, headers, options) {
if (!this.fetch) {
throw new Error('fusion-plugin-rpc requires `fetch`');
}
if (!this.emitter) {
throw new Error('Missing emitter registered to UniversalEventsToken');
}
const fetch = this.fetch;
const emitter = this.emitter;
const apiPath = this.apiPath;
const startTime = Date.now();
const queryParams = this.queryParams.length > 0 ? `?${this.queryParams.map(([k, v]) => `${encodeURIComponent(k)}=${encodeURIComponent(v)}`).join('&')}` : '';
return fetch(`${apiPath}${rpcId}${queryParams}`, args instanceof FormData ? { ...options,
method: 'POST',
headers: { // Content-Type will be set automatically
...(headers || {})
},
body: args
} : { ...options,
method: 'POST',
// $FlowFixMe
headers: {
'Content-Type': 'application/json',
...(headers || {})
},
body: JSON.stringify(args || {})
}).then(r => r.json()).then(args => {
const {
status,
data
} = args;
if (status === 'success') {
emitter.emit(statKey, {
method: rpcId,
status: 'success',
timing: Date.now() - startTime
});
return data;
} else {
emitter.emit(statKey, {
method: rpcId,
error: data,
status: 'failure',
timing: Date.now() - startTime
});
return Promise.reject(data ? data : {});
}
});
}
}
const pluginFactory = () => createPlugin({
deps: {
fetch: FetchToken,
emitter: UniversalEventsToken,
i18n: I18nToken.optional,
rpcConfig: RPCHandlersConfigToken.optional,
queryParams: RPCQueryParamsToken.optional
},
provides: deps => {
const {
fetch = window.fetch,
emitter,
rpcConfig,
i18n,
queryParams
} = deps;
return {
from: memoize(ctx => {
const queryParamsValue = queryParams && queryParams.from(ctx) || [];
const locale = i18n && i18n.from(ctx).locale || '';
const localeCode = typeof locale === 'string' ? locale : locale.code;
if (localeCode) {
queryParamsValue.push(['localeCode', localeCode]);
}
return new RPC({
fetch,
emitter,
rpcConfig,
queryParams: queryParamsValue
});
})
};
}
});
export default true && pluginFactory();
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["src/browser.js"],"names":["createPlugin","memoize","UniversalEventsToken","I18nToken","FetchToken","RPCHandlersConfigToken","RPCQueryParamsToken","formatApiPath","statKey","RPC","constructor","fetch","emitter","rpcConfig","queryParams","config","apiPath","request","rpcId","args","headers","options","Error","startTime","Date","now","length","map","k","v","encodeURIComponent","join","FormData","method","body","JSON","stringify","then","r","json","status","data","emit","timing","error","Promise","reject","pluginFactory","deps","i18n","optional","provides","window","from","ctx","queryParamsValue","locale","localeCode","code","push"],"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 browser */\n\nimport {createPlugin, memoize, type Context} from 'fusion-core';\nimport {UniversalEventsToken} from 'fusion-plugin-universal-events';\nimport {I18nToken} from 'fusion-plugin-i18n';\nimport {FetchToken} from 'fusion-tokens';\nimport type {Fetch} from 'fusion-tokens';\n\nimport {\n  type HandlerType,\n  RPCHandlersConfigToken,\n  RPCQueryParamsToken,\n} from './tokens.js';\nimport type {RPCPluginType, IEmitter, RPCConfigType} from './types.js';\nimport {formatApiPath} from './utils.js';\n\ntype InitializationOpts = {\n  fetch: Fetch,\n  emitter: IEmitter,\n  queryParams: Array<[string, string]>,\n  rpcConfig: ?RPCConfigType,\n};\n\nconst statKey = 'rpc:method-client';\n\nclass RPC {\n  ctx: ?Context;\n  emitter: ?IEmitter;\n  handlers: ?HandlerType;\n  queryParams: Array<[string, string]>;\n  fetch: ?Fetch;\n  config: ?RPCConfigType;\n  apiPath: string;\n  constructor({fetch, emitter, rpcConfig, queryParams}: InitializationOpts) {\n    this.fetch = fetch;\n    this.config = rpcConfig || {};\n    this.emitter = emitter;\n    this.queryParams = queryParams;\n\n    this.apiPath = formatApiPath(\n      rpcConfig && rpcConfig.apiPath ? rpcConfig.apiPath : 'api'\n    );\n  }\n\n  request<TArgs, TResult>(\n    rpcId: string,\n    args: TArgs,\n    headers: ?{[string]: string},\n    options: ?RequestOptions\n  ): Promise<TResult> {\n    if (!this.fetch) {\n      throw new Error('fusion-plugin-rpc requires `fetch`');\n    }\n    if (!this.emitter) {\n      throw new Error('Missing emitter registered to UniversalEventsToken');\n    }\n    const fetch = this.fetch;\n    const emitter = this.emitter;\n    const apiPath = this.apiPath;\n\n    const startTime = Date.now();\n    const queryParams =\n      this.queryParams.length > 0\n        ? `?${this.queryParams\n            .map(\n              ([k, v]) => `${encodeURIComponent(k)}=${encodeURIComponent(v)}`\n            )\n            .join('&')}`\n        : '';\n\n    return fetch(\n      `${apiPath}${rpcId}${queryParams}`,\n      args instanceof FormData\n        ? {\n            ...options,\n            method: 'POST',\n            headers: {\n              // Content-Type will be set automatically\n              ...(headers || {}),\n            },\n            body: args,\n          }\n        : {\n            ...options,\n            method: 'POST',\n            // $FlowFixMe\n            headers: {\n              'Content-Type': 'application/json',\n              ...(headers || {}),\n            },\n            body: JSON.stringify(args || {}),\n          }\n    )\n      .then((r) => r.json())\n      .then((args) => {\n        const {status, data} = args;\n        if (status === 'success') {\n          emitter.emit(statKey, {\n            method: rpcId,\n            status: 'success',\n            timing: Date.now() - startTime,\n          });\n          return data;\n        } else {\n          emitter.emit(statKey, {\n            method: rpcId,\n            error: data,\n            status: 'failure',\n            timing: Date.now() - startTime,\n          });\n          return Promise.reject(data ? data : {});\n        }\n      });\n  }\n}\n\nconst pluginFactory: () => RPCPluginType = () =>\n  createPlugin({\n    deps: {\n      fetch: FetchToken,\n      emitter: UniversalEventsToken,\n      i18n: I18nToken.optional,\n      rpcConfig: RPCHandlersConfigToken.optional,\n      queryParams: RPCQueryParamsToken.optional,\n    },\n    provides: (deps) => {\n      const {\n        fetch = window.fetch,\n        emitter,\n        rpcConfig,\n        i18n,\n        queryParams,\n      } = deps;\n\n      return {\n        from: memoize((ctx) => {\n          const queryParamsValue = (queryParams && queryParams.from(ctx)) || [];\n          const locale = (i18n && i18n.from(ctx).locale) || '';\n          const localeCode = typeof locale === 'string' ? locale : locale.code;\n          if (localeCode) {\n            queryParamsValue.push(['localeCode', localeCode]);\n          }\n          return new RPC({\n            fetch,\n            emitter,\n            rpcConfig,\n            queryParams: queryParamsValue,\n          });\n        }),\n      };\n    },\n  });\n\nexport default ((__BROWSER__ && pluginFactory(): any): RPCPluginType);\n"]}