fusion-plugin-rpc
Version:
Fetch data on the server and client with an RPC style interface.
118 lines (116 loc) • 14.7 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';
import { formatApiPath } from './utils';
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,"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"],"sources":["src/browser.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 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';\nimport type {RPCPluginType, IEmitter, RPCConfigType} from './types';\nimport {formatApiPath} from './utils';\n\ntype InitializationOpts = {\n  fetch: Fetch;\n  emitter: IEmitter;\n  queryParams: Array<[string, string]>;\n  rpcConfig: RPCConfigType | undefined | null;\n};\n\nconst statKey = 'rpc:method-client';\n\nclass RPC {\n  ctx: Context | undefined | null;\n  emitter: IEmitter | undefined | null;\n  handlers: HandlerType | undefined | null;\n  queryParams: Array<[string, string]>;\n  fetch: Fetch | undefined | null;\n  config: RPCConfigType | undefined | null;\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?: {\n      [x: string]: string;\n    } | null,\n    options?: RequestInit | null\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() as any as RPCPluginType);\n"],"mappings":"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"}