UNPKG

@data-client/rest

Version:

Quickly define typed REST resources and endpoints

169 lines (166 loc) 20.7 kB
import { schema } from '@data-client/endpoint'; import RestEndpoint from './RestEndpoint.js'; import { shortenPath } from './RestHelpers.js'; const { Invalidate, Collection: BaseCollection } = schema; /** Creates collection of Endpoints for common operations on a given data/schema. * * @see https://dataclient.io/rest/api/resource */ export default function resource({ path, schema, Endpoint = RestEndpoint, Collection = BaseCollection, nonFilterArgumentKeys, optimistic, paginationField, ...extraOptions }) { if (process.env.NODE_ENV !== 'production') { // if they lowercase and it looks like they meant to use upper-case version if ('endpoint' in extraOptions && Endpoint === RestEndpoint && typeof extraOptions['endpoint'] === 'function' && extraOptions['endpoint'] && Object.prototype.isPrototypeOf.call(RestEndpoint.prototype, extraOptions['endpoint'].prototype)) { console.warn(`You passed 'endpoint' option; did you mean to use Endpoint? https://dataclient.io/rest/api/resource#endpoint This parameter must be capitalized. This warning will not show in production.`); } // if they lowercase and it looks like they meant to use upper-case version if ('collection' in extraOptions && Collection === BaseCollection && typeof extraOptions['collection'] === 'function' && extraOptions['collection'] && Object.prototype.isPrototypeOf.call(BaseCollection.prototype, extraOptions['collection'].prototype)) { console.warn(`You passed 'collection' option; did you mean to use Collection? https://dataclient.io/rest/api/resource#collection This parameter must be capitalized. This warning will not show in production.`); } } const shortenedPath = shortenPath(path); const getName = name => `${schema == null ? void 0 : schema.name}.${name}`; // this accounts for derivative endpoints function extendMember(extended, key, options) { extended[key] = extended[key].extend(options); } const extraMutateOptions = { ...extraOptions }; const extraPartialOptions = { ...extraOptions }; const get = new Endpoint({ ...extraOptions, path, schema, name: getName('get') }); if (optimistic) { extraMutateOptions.getOptimisticResponse = optimisticUpdate; // TODO: Check that schema is a queryable, otherwise this doesn't make sense extraPartialOptions.getOptimisticResponse = optimisticPartial(schema); } const getList = new Endpoint({ ...extraMutateOptions, paginationField: paginationField, movePath: path, path: shortenedPath, schema: new Collection([schema], nonFilterArgumentKeys === undefined ? undefined : { nonFilterArgumentKeys }), name: getName('getList') }); const ret = { get, getList, // TODO(deprecated): remove this once we remove creates create: getList.push.extend({ name: getName('create') }), update: new Endpoint({ ...extraMutateOptions, path, schema, method: 'PUT', name: getName('update') }), // TODO(breaking): Move to getList.move partialUpdate: new Endpoint({ ...extraPartialOptions, path, schema, method: 'PATCH', name: getName('partialUpdate') }), delete: new Endpoint({ ...extraMutateOptions, path, schema: // Entity || Union schema.process || schema._hoistable ? new Invalidate(schema) : schema, method: 'DELETE', name: getName('delete'), process(res, params) { return res && Object.keys(res).length ? res : params; }, getOptimisticResponse: optimistic ? optimisticDelete : undefined }), extend(...args) { if (typeof args[0] === 'string') { const [key, options] = args; if (key in this) { const extended = { ...this }; extendMember(extended, key, options); return extended; } else { return { ...this, [key]: this.get.extend(options) }; } } else if (typeof args[0] === 'function') { const extended = args[0](this); return { ...this, ...extended }; } const overrides = args[0]; const extended = { ...this }; for (const key in overrides) { // eslint-disable-next-line @typescript-eslint/ban-ts-comment // @ts-ignore extendMember(extended, key, overrides[key]); } return extended; } }; return ret; } function optimisticUpdate(snap, params, body) { return { ...params, ...ensurePojo(body) }; } function optimisticPartial(schema) { return function (snap, params, body) { const data = snap.get(schema, params); if (!data) throw snap.abort; return { ...params, ...data, // even tho we don't always have two arguments, the extra one will simply be undefined which spreads fine ...ensurePojo(body) }; }; } function optimisticDelete(snap, params) { return params; } function ensurePojo(body) { return body instanceof FormData ? Object.fromEntries(body.entries()) : body; } //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJzY2hlbWEiLCJSZXN0RW5kcG9pbnQiLCJzaG9ydGVuUGF0aCIsIkludmFsaWRhdGUiLCJDb2xsZWN0aW9uIiwiQmFzZUNvbGxlY3Rpb24iLCJyZXNvdXJjZSIsInBhdGgiLCJFbmRwb2ludCIsIm5vbkZpbHRlckFyZ3VtZW50S2V5cyIsIm9wdGltaXN0aWMiLCJwYWdpbmF0aW9uRmllbGQiLCJleHRyYU9wdGlvbnMiLCJwcm9jZXNzIiwiZW52IiwiTk9ERV9FTlYiLCJPYmplY3QiLCJwcm90b3R5cGUiLCJpc1Byb3RvdHlwZU9mIiwiY2FsbCIsImNvbnNvbGUiLCJ3YXJuIiwic2hvcnRlbmVkUGF0aCIsImdldE5hbWUiLCJuYW1lIiwiZXh0ZW5kTWVtYmVyIiwiZXh0ZW5kZWQiLCJrZXkiLCJvcHRpb25zIiwiZXh0ZW5kIiwiZXh0cmFNdXRhdGVPcHRpb25zIiwiZXh0cmFQYXJ0aWFsT3B0aW9ucyIsImdldCIsImdldE9wdGltaXN0aWNSZXNwb25zZSIsIm9wdGltaXN0aWNVcGRhdGUiLCJvcHRpbWlzdGljUGFydGlhbCIsImdldExpc3QiLCJtb3ZlUGF0aCIsInVuZGVmaW5lZCIsInJldCIsImNyZWF0ZSIsInB1c2giLCJ1cGRhdGUiLCJtZXRob2QiLCJwYXJ0aWFsVXBkYXRlIiwiZGVsZXRlIiwiX2hvaXN0YWJsZSIsInJlcyIsInBhcmFtcyIsImtleXMiLCJsZW5ndGgiLCJvcHRpbWlzdGljRGVsZXRlIiwiYXJncyIsIm92ZXJyaWRlcyIsInNuYXAiLCJib2R5IiwiZW5zdXJlUG9qbyIsImRhdGEiLCJhYm9ydCIsIkZvcm1EYXRhIiwiZnJvbUVudHJpZXMiLCJlbnRyaWVzIl0sInNvdXJjZXMiOlsiLi4vc3JjL3Jlc291cmNlLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IHNjaGVtYSB9IGZyb20gJ0BkYXRhLWNsaWVudC9lbmRwb2ludCc7XG5pbXBvcnQgdHlwZSB7IFNuYXBzaG90SW50ZXJmYWNlLCBRdWVyeWFibGUgfSBmcm9tICdAZGF0YS1jbGllbnQvZW5kcG9pbnQnO1xuXG5pbXBvcnQge1xuICBSZXNvdXJjZUdlbmVyaWNzLFxuICBSZXNvdXJjZU9wdGlvbnMsXG4gIFJlc291cmNlLFxuICBSZXNvdXJjZUludGVyZmFjZSxcbn0gZnJvbSAnLi9yZXNvdXJjZVR5cGVzLmpzJztcbmltcG9ydCBSZXN0RW5kcG9pbnQsIHtcbiAgR2V0RW5kcG9pbnQsXG4gIFBhcnRpYWxSZXN0R2VuZXJpY3MsXG4gIFJlc3RFbmRwb2ludE9wdGlvbnMsXG4gIFJlc3RJbnN0YW5jZUJhc2UsXG59IGZyb20gJy4vUmVzdEVuZHBvaW50LmpzJztcbmltcG9ydCB7IHNob3J0ZW5QYXRoIH0gZnJvbSAnLi9SZXN0SGVscGVycy5qcyc7XG5cbmNvbnN0IHsgSW52YWxpZGF0ZSwgQ29sbGVjdGlvbjogQmFzZUNvbGxlY3Rpb24gfSA9IHNjaGVtYTtcblxuLyoqIENyZWF0ZXMgY29sbGVjdGlvbiBvZiBFbmRwb2ludHMgZm9yIGNvbW1vbiBvcGVyYXRpb25zIG9uIGEgZ2l2ZW4gZGF0YS9zY2hlbWEuXG4gKlxuICogQHNlZSBodHRwczovL2RhdGFjbGllbnQuaW8vcmVzdC9hcGkvcmVzb3VyY2VcbiAqL1xuZXhwb3J0IGRlZmF1bHQgZnVuY3Rpb24gcmVzb3VyY2U8TyBleHRlbmRzIFJlc291cmNlR2VuZXJpY3M+KHtcbiAgcGF0aCxcbiAgc2NoZW1hLFxuICBFbmRwb2ludCA9IFJlc3RFbmRwb2ludCxcbiAgQ29sbGVjdGlvbiA9IEJhc2VDb2xsZWN0aW9uLFxuICBub25GaWx0ZXJBcmd1bWVudEtleXMsXG4gIG9wdGltaXN0aWMsXG4gIHBhZ2luYXRpb25GaWVsZCxcbiAgLi4uZXh0cmFPcHRpb25zXG59OiBSZWFkb25seTxPPiAmIFJlc291cmNlT3B0aW9ucyk6IFJlc291cmNlPE8+IHtcbiAgaWYgKHByb2Nlc3MuZW52Lk5PREVfRU5WICE9PSAncHJvZHVjdGlvbicpIHtcbiAgICAvLyBpZiB0aGV5IGxvd2VyY2FzZSBhbmQgaXQgbG9va3MgbGlrZSB0aGV5IG1lYW50IHRvIHVzZSB1cHBlci1jYXNlIHZlcnNpb25cbiAgICBpZiAoXG4gICAgICAnZW5kcG9pbnQnIGluIGV4dHJhT3B0aW9ucyAmJlxuICAgICAgRW5kcG9pbnQgPT09IFJlc3RFbmRwb2ludCAmJlxuICAgICAgdHlwZW9mIGV4dHJhT3B0aW9uc1snZW5kcG9pbnQnXSA9PT0gJ2Z1bmN0aW9uJyAmJlxuICAgICAgZXh0cmFPcHRpb25zWydlbmRwb2ludCddICYmXG4gICAgICBPYmplY3QucHJvdG90eXBlLmlzUHJvdG90eXBlT2YuY2FsbChcbiAgICAgICAgUmVzdEVuZHBvaW50LnByb3RvdHlwZSxcbiAgICAgICAgKGV4dHJhT3B0aW9uc1snZW5kcG9pbnQnXSBhcyBhbnkpLnByb3RvdHlwZSxcbiAgICAgIClcbiAgICApIHtcbiAgICAgIGNvbnNvbGUud2FybihcbiAgICAgICAgYFlvdSBwYXNzZWQgJ2VuZHBvaW50JyBvcHRpb247IGRpZCB5b3UgbWVhbiB0byB1c2UgRW5kcG9pbnQ/XG5odHRwczovL2RhdGFjbGllbnQuaW8vcmVzdC9hcGkvcmVzb3VyY2UjZW5kcG9pbnRcblRoaXMgcGFyYW1ldGVyIG11c3QgYmUgY2FwaXRhbGl6ZWQuXG5cblRoaXMgd2FybmluZyB3aWxsIG5vdCBzaG93IGluIHByb2R1Y3Rpb24uYCxcbiAgICAgICk7XG4gICAgfVxuICAgIC8vIGlmIHRoZXkgbG93ZXJjYXNlIGFuZCBpdCBsb29rcyBsaWtlIHRoZXkgbWVhbnQgdG8gdXNlIHVwcGVyLWNhc2UgdmVyc2lvblxuICAgIGlmIChcbiAgICAgICdjb2xsZWN0aW9uJyBpbiBleHRyYU9wdGlvbnMgJiZcbiAgICAgIENvbGxlY3Rpb24gPT09IEJhc2VDb2xsZWN0aW9uICYmXG4gICAgICB0eXBlb2YgZXh0cmFPcHRpb25zWydjb2xsZWN0aW9uJ10gPT09ICdmdW5jdGlvbicgJiZcbiAgICAgIGV4dHJhT3B0aW9uc1snY29sbGVjdGlvbiddICYmXG4gICAgICBPYmplY3QucHJvdG90eXBlLmlzUHJvdG90eXBlT2YuY2FsbChcbiAgICAgICAgQmFzZUNvbGxlY3Rpb24ucHJvdG90eXBlLFxuICAgICAgICAoZXh0cmFPcHRpb25zWydjb2xsZWN0aW9uJ10gYXMgYW55KS5wcm90b3R5cGUsXG4gICAgICApXG4gICAgKSB7XG4gICAgICBjb25zb2xlLndhcm4oXG4gICAgICAgIGBZb3UgcGFzc2VkICdjb2xsZWN0aW9uJyBvcHRpb247IGRpZCB5b3UgbWVhbiB0byB1c2UgQ29sbGVjdGlvbj9cbmh0dHBzOi8vZGF0YWNsaWVudC5pby9yZXN0L2FwaS9yZXNvdXJjZSNjb2xsZWN0aW9uXG5UaGlzIHBhcmFtZXRlciBtdXN0IGJlIGNhcGl0YWxpemVkLlxuXG5UaGlzIHdhcm5pbmcgd2lsbCBub3Qgc2hvdyBpbiBwcm9kdWN0aW9uLmAsXG4gICAgICApO1xuICAgIH1cbiAgfVxuICBjb25zdCBzaG9ydGVuZWRQYXRoID0gc2hvcnRlblBhdGgocGF0aCk7XG4gIGNvbnN0IGdldE5hbWUgPSAobmFtZTogc3RyaW5nKSA9PiBgJHsoc2NoZW1hIGFzIGFueSk/Lm5hbWV9LiR7bmFtZX1gO1xuICAvLyB0aGlzIGFjY291bnRzIGZvciBkZXJpdmF0aXZlIGVuZHBvaW50c1xuICBmdW5jdGlvbiBleHRlbmRNZW1iZXIoZXh0ZW5kZWQ6IGFueSwga2V5OiBzdHJpbmcsIG9wdGlvbnM6IGFueSkge1xuICAgIGV4dGVuZGVkW2tleV0gPSBleHRlbmRlZFtrZXldLmV4dGVuZChvcHRpb25zKTtcbiAgfVxuXG4gIGNvbnN0IGV4dHJhTXV0YXRlT3B0aW9ucyA9IHsgLi4uZXh0cmFPcHRpb25zIH07XG4gIGNvbnN0IGV4dHJhUGFydGlhbE9wdGlvbnMgPSB7IC4uLmV4dHJhT3B0aW9ucyB9O1xuICBjb25zdCBnZXQ6IEdldEVuZHBvaW50PHsgcGF0aDogT1sncGF0aCddOyBzY2hlbWE6IE9bJ3NjaGVtYSddIH0+ID1cbiAgICBuZXcgRW5kcG9pbnQoe1xuICAgICAgLi4uZXh0cmFPcHRpb25zLFxuICAgICAgcGF0aCxcbiAgICAgIHNjaGVtYSxcbiAgICAgIG5hbWU6IGdldE5hbWUoJ2dldCcpLFxuICAgIH0pIGFzIGFueTtcbiAgaWYgKG9wdGltaXN0aWMpIHtcbiAgICAoZXh0cmFNdXRhdGVPcHRpb25zIGFzIGFueSkuZ2V0T3B0aW1pc3RpY1Jlc3BvbnNlID0gb3B0aW1pc3RpY1VwZGF0ZTtcbiAgICAvLyBUT0RPOiBDaGVjayB0aGF0IHNjaGVtYSBpcyBhIHF1ZXJ5YWJsZSwgb3RoZXJ3aXNlIHRoaXMgZG9lc24ndCBtYWtlIHNlbnNlXG4gICAgKGV4dHJhUGFydGlhbE9wdGlvbnMgYXMgYW55KS5nZXRPcHRpbWlzdGljUmVzcG9uc2UgPSBvcHRpbWlzdGljUGFydGlhbChcbiAgICAgIHNjaGVtYSBhcyBhbnksXG4gICAgKTtcbiAgfVxuICBjb25zdCBnZXRMaXN0ID0gbmV3IEVuZHBvaW50KHtcbiAgICAuLi5leHRyYU11dGF0ZU9wdGlvbnMsXG4gICAgcGFnaW5hdGlvbkZpZWxkOiBwYWdpbmF0aW9uRmllbGQgYXMgc3RyaW5nLFxuICAgIG1vdmVQYXRoOiBwYXRoLFxuICAgIHBhdGg6IHNob3J0ZW5lZFBhdGgsXG4gICAgc2NoZW1hOiBuZXcgQ29sbGVjdGlvbihcbiAgICAgIFtzY2hlbWEgYXMgYW55XSxcbiAgICAgIG5vbkZpbHRlckFyZ3VtZW50S2V5cyA9PT0gdW5kZWZpbmVkID8gdW5kZWZpbmVkIDogKFxuICAgICAgICB7IG5vbkZpbHRlckFyZ3VtZW50S2V5cyB9XG4gICAgICApLFxuICAgICksXG4gICAgbmFtZTogZ2V0TmFtZSgnZ2V0TGlzdCcpLFxuICB9KTtcbiAgY29uc3QgcmV0ID0ge1xuICAgIGdldCxcbiAgICBnZXRMaXN0LFxuICAgIC8vIFRPRE8oZGVwcmVjYXRlZCk6IHJlbW92ZSB0aGlzIG9uY2Ugd2UgcmVtb3ZlIGNyZWF0ZXNcbiAgICBjcmVhdGU6IGdldExpc3QucHVzaC5leHRlbmQoeyBuYW1lOiBnZXROYW1lKCdjcmVhdGUnKSB9KSxcbiAgICB1cGRhdGU6IG5ldyBFbmRwb2ludCh7XG4gICAgICAuLi5leHRyYU11dGF0ZU9wdGlvbnMsXG4gICAgICBwYXRoLFxuICAgICAgc2NoZW1hLFxuICAgICAgbWV0aG9kOiAnUFVUJyxcbiAgICAgIG5hbWU6IGdldE5hbWUoJ3VwZGF0ZScpLFxuICAgIH0pLFxuICAgIC8vIFRPRE8oYnJlYWtpbmcpOiBNb3ZlIHRvIGdldExpc3QubW92ZVxuICAgIHBhcnRpYWxVcGRhdGU6IG5ldyBFbmRwb2ludCh7XG4gICAgICAuLi5leHRyYVBhcnRpYWxPcHRpb25zLFxuICAgICAgcGF0aCxcbiAgICAgIHNjaGVtYSxcbiAgICAgIG1ldGhvZDogJ1BBVENIJyxcbiAgICAgIG5hbWU6IGdldE5hbWUoJ3BhcnRpYWxVcGRhdGUnKSxcbiAgICB9KSxcbiAgICBkZWxldGU6IG5ldyBFbmRwb2ludCh7XG4gICAgICAuLi5leHRyYU11dGF0ZU9wdGlvbnMsXG4gICAgICBwYXRoLFxuICAgICAgc2NoZW1hOlxuICAgICAgICAvLyBFbnRpdHkgfHwgVW5pb25cbiAgICAgICAgKHNjaGVtYSBhcyBhbnkpLnByb2Nlc3MgfHwgKHNjaGVtYSBhcyBhbnkpLl9ob2lzdGFibGUgP1xuICAgICAgICAgIG5ldyBJbnZhbGlkYXRlKHNjaGVtYSBhcyBhbnkpXG4gICAgICAgIDogc2NoZW1hLFxuICAgICAgbWV0aG9kOiAnREVMRVRFJyxcbiAgICAgIG5hbWU6IGdldE5hbWUoJ2RlbGV0ZScpLFxuICAgICAgcHJvY2VzcyhyZXM6IGFueSwgcGFyYW1zOiBhbnkpIHtcbiAgICAgICAgcmV0dXJuIHJlcyAmJiBPYmplY3Qua2V5cyhyZXMpLmxlbmd0aCA/IHJlcyA6IHBhcmFtcztcbiAgICAgIH0sXG4gICAgICBnZXRPcHRpbWlzdGljUmVzcG9uc2U6IG9wdGltaXN0aWMgPyAob3B0aW1pc3RpY0RlbGV0ZSBhcyBhbnkpIDogdW5kZWZpbmVkLFxuICAgIH0pLFxuICAgIGV4dGVuZChcbiAgICAgIC4uLmFyZ3M6XG4gICAgICAgIHwgW3N0cmluZywgUmVzdEVuZHBvaW50T3B0aW9ucyAmIFBhcnRpYWxSZXN0R2VuZXJpY3NdXG4gICAgICAgIHwgW1Jlc3RFbmRwb2ludE9wdGlvbnMgJiBQYXJ0aWFsUmVzdEdlbmVyaWNzXVxuICAgICAgICB8IFtcbiAgICAgICAgICAgIChcbiAgICAgICAgICAgICAgYmFzZVJlc291cmNlOiBSZXNvdXJjZUludGVyZmFjZSxcbiAgICAgICAgICAgICkgPT4gUmVjb3JkPHN0cmluZywgUmVzdEluc3RhbmNlQmFzZT4sXG4gICAgICAgICAgXVxuICAgICkge1xuICAgICAgaWYgKHR5cGVvZiBhcmdzWzBdID09PSAnc3RyaW5nJykge1xuICAgICAgICBjb25zdCBba2V5LCBvcHRpb25zXSA9IGFyZ3M7XG4gICAgICAgIGlmIChrZXkgaW4gdGhpcykge1xuICAgICAgICAgIGNvbnN0IGV4dGVuZGVkID0geyAuLi50aGlzIH07XG4gICAgICAgICAgZXh0ZW5kTWVtYmVyKGV4dGVuZGVkLCBrZXksIG9wdGlvbnMpO1xuICAgICAgICAgIHJldHVybiBleHRlbmRlZDtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICByZXR1cm4ge1xuICAgICAgICAgICAgLi4udGhpcyxcbiAgICAgICAgICAgIFtrZXldOiB0aGlzLmdldC5leHRlbmQob3B0aW9ucyksXG4gICAgICAgICAgfTtcbiAgICAgICAgfVxuICAgICAgfSBlbHNlIGlmICh0eXBlb2YgYXJnc1swXSA9PT0gJ2Z1bmN0aW9uJykge1xuICAgICAgICBjb25zdCBleHRlbmRlZCA9IGFyZ3NbMF0odGhpcyk7XG4gICAgICAgIHJldHVybiB7XG4gICAgICAgICAgLi4udGhpcyxcbiAgICAgICAgICAuLi5leHRlbmRlZCxcbiAgICAgICAgfTtcbiAgICAgIH1cbiAgICAgIGNvbnN0IG92ZXJyaWRlcyA9IGFyZ3NbMF07XG4gICAgICBjb25zdCBleHRlbmRlZCA9IHsgLi4udGhpcyB9O1xuICAgICAgZm9yIChjb25zdCBrZXkgaW4gb3ZlcnJpZGVzKSB7XG4gICAgICAgIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBAdHlwZXNjcmlwdC1lc2xpbnQvYmFuLXRzLWNvbW1lbnRcbiAgICAgICAgLy8gQHRzLWlnbm9yZVxuICAgICAgICBleHRlbmRNZW1iZXIoZXh0ZW5kZWQsIGtleSwgb3ZlcnJpZGVzW2tleV0pO1xuICAgICAgfVxuICAgICAgcmV0dXJuIGV4dGVuZGVkO1xuICAgIH0sXG4gIH0gYXMgYW55O1xuICByZXR1cm4gcmV0O1xufVxuXG5mdW5jdGlvbiBvcHRpbWlzdGljVXBkYXRlKHNuYXA6IFNuYXBzaG90SW50ZXJmYWNlLCBwYXJhbXM6IGFueSwgYm9keTogYW55KSB7XG4gIHJldHVybiB7XG4gICAgLi4ucGFyYW1zLFxuICAgIC4uLmVuc3VyZVBvam8oYm9keSksXG4gIH07XG59XG5mdW5jdGlvbiBvcHRpbWlzdGljUGFydGlhbChzY2hlbWE6IFF1ZXJ5YWJsZSkge1xuICByZXR1cm4gZnVuY3Rpb24gKHNuYXA6IFNuYXBzaG90SW50ZXJmYWNlLCBwYXJhbXM6IGFueSwgYm9keTogYW55KSB7XG4gICAgY29uc3QgZGF0YSA9IHNuYXAuZ2V0KHNjaGVtYSwgcGFyYW1zKTtcbiAgICBpZiAoIWRhdGEpIHRocm93IHNuYXAuYWJvcnQ7XG4gICAgcmV0dXJuIHtcbiAgICAgIC4uLnBhcmFtcyxcbiAgICAgIC4uLmRhdGEsXG4gICAgICAvLyBldmVuIHRobyB3ZSBkb24ndCBhbHdheXMgaGF2ZSB0d28gYXJndW1lbnRzLCB0aGUgZXh0cmEgb25lIHdpbGwgc2ltcGx5IGJlIHVuZGVmaW5lZCB3aGljaCBzcHJlYWRzIGZpbmVcbiAgICAgIC4uLmVuc3VyZVBvam8oYm9keSksXG4gICAgfTtcbiAgfTtcbn1cbmZ1bmN0aW9uIG9wdGltaXN0aWNEZWxldGUoc25hcDogU25hcHNob3RJbnRlcmZhY2UsIHBhcmFtczogYW55KSB7XG4gIHJldHVybiBwYXJhbXM7XG59XG5mdW5jdGlvbiBlbnN1cmVQb2pvKGJvZHk6IGFueSkge1xuICByZXR1cm4gYm9keSBpbnN0YW5jZW9mIEZvcm1EYXRhID9cbiAgICAgIE9iamVjdC5mcm9tRW50cmllcygoYm9keSBhcyBhbnkpLmVudHJpZXMoKSlcbiAgICA6IGJvZHk7XG59XG4iXSwibWFwcGluZ3MiOiJBQUFBLFNBQVNBLE1BQU0sUUFBUSx1QkFBdUI7QUFTOUMsT0FBT0MsWUFBWSxNQUtaLG1CQUFtQjtBQUMxQixTQUFTQyxXQUFXLFFBQVEsa0JBQWtCO0FBRTlDLE1BQU07RUFBRUMsVUFBVTtFQUFFQyxVQUFVLEVBQUVDO0FBQWUsQ0FBQyxHQUFHTCxNQUFNOztBQUV6RDtBQUNBO0FBQ0E7QUFDQTtBQUNBLGVBQWUsU0FBU00sUUFBUUEsQ0FBNkI7RUFDM0RDLElBQUk7RUFDSlAsTUFBTTtFQUNOUSxRQUFRLEdBQUdQLFlBQVk7RUFDdkJHLFVBQVUsR0FBR0MsY0FBYztFQUMzQkkscUJBQXFCO0VBQ3JCQyxVQUFVO0VBQ1ZDLGVBQWU7RUFDZixHQUFHQztBQUMwQixDQUFDLEVBQWU7RUFDN0MsSUFBSUMsT0FBTyxDQUFDQyxHQUFHLENBQUNDLFFBQVEsS0FBSyxZQUFZLEVBQUU7SUFDekM7SUFDQSxJQUNFLFVBQVUsSUFBSUgsWUFBWSxJQUMxQkosUUFBUSxLQUFLUCxZQUFZLElBQ3pCLE9BQU9XLFlBQVksQ0FBQyxVQUFVLENBQUMsS0FBSyxVQUFVLElBQzlDQSxZQUFZLENBQUMsVUFBVSxDQUFDLElBQ3hCSSxNQUFNLENBQUNDLFNBQVMsQ0FBQ0MsYUFBYSxDQUFDQyxJQUFJLENBQ2pDbEIsWUFBWSxDQUFDZ0IsU0FBUyxFQUNyQkwsWUFBWSxDQUFDLFVBQVUsQ0FBQyxDQUFTSyxTQUNwQyxDQUFDLEVBQ0Q7TUFDQUcsT0FBTyxDQUFDQyxJQUFJLENBQ1Y7QUFDUjtBQUNBO0FBQ0E7QUFDQSwwQ0FDTSxDQUFDO0lBQ0g7SUFDQTtJQUNBLElBQ0UsWUFBWSxJQUFJVCxZQUFZLElBQzVCUixVQUFVLEtBQUtDLGNBQWMsSUFDN0IsT0FBT08sWUFBWSxDQUFDLFlBQVksQ0FBQyxLQUFLLFVBQVUsSUFDaERBLFlBQVksQ0FBQyxZQUFZLENBQUMsSUFDMUJJLE1BQU0sQ0FB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