UNPKG

pretur.sync

Version:

Front-end back-end communication infrastructure

271 lines 32.7 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); var tslib_1 = require("tslib"); function buildResponder(options) { var errorToBundle = options.errorToBundle, pool = options.pool, operator = options.operator, validator = options.validator, transact = options.transact; return function responder(requests, context) { return tslib_1.__awaiter(this, void 0, void 0, function () { var responses, _i, requests_1, request, requestId, _a, tr, _b, data, count, error_1, error_2, _c, _d, error_3, error, transaction, _e, generatedIds, errors, error_4, batch, e, _f, _g, _h, action, id, type, failed, failReasons, batchTr, _j, _k, reqItem, action, type, itemId, _l, generatedIds, errors, error_5, error_6, _m, _o, resItem, _p, _q, error_7; return tslib_1.__generator(this, function (_r) { switch (_r.label) { case 0: if (!Array.isArray(requests) || requests.length === 0) { return [2 /*return*/, []]; } responses = []; _i = 0, requests_1 = requests; _r.label = 1; case 1: if (!(_i < requests_1.length)) return [3 /*break*/, 46]; request = requests_1[_i]; requestId = request.requestId; _a = request.type; switch (_a) { case 'select': return [3 /*break*/, 2]; case 'operate': return [3 /*break*/, 10]; case 'mutate': return [3 /*break*/, 14]; case 'batchMutate': return [3 /*break*/, 25]; case 'validate': return [3 /*break*/, 41]; } return [3 /*break*/, 45]; case 2: _r.trys.push([2, 8, , 9]); if (!transact || !pool) { throw new Error('select requests require transact and pool to operate.'); } return [4 /*yield*/, transact()]; case 3: tr = _r.sent(); _r.label = 4; case 4: _r.trys.push([4, 6, , 7]); return [4 /*yield*/, pool.resolve(tr, request.scope, request.model, request.query, context)]; case 5: _b = _r.sent(), data = _b.data, count = _b.count; tr.commit(); responses.push({ errors: [], count: count, data: data, requestId: requestId, type: 'select' }); return [3 /*break*/, 7]; case 6: error_1 = _r.sent(); responses.push({ errors: [errorToBundle(error_1)], count: 0, data: [], requestId: requestId, type: 'select', }); tr.rollback(); return [3 /*break*/, 7]; case 7: return [3 /*break*/, 9]; case 8: error_2 = _r.sent(); responses.push({ count: 0, data: [], errors: [errorToBundle(error_2)], requestId: requestId, type: 'select', }); return [3 /*break*/, 9]; case 9: return [3 /*break*/, 45]; case 10: _r.trys.push([10, 12, , 13]); if (!operator) { throw new Error('operate requests require an operator.'); } _d = (_c = responses).push; return [4 /*yield*/, operator(request, context)]; case 11: _d.apply(_c, [_r.sent()]); return [3 /*break*/, 13]; case 12: error_3 = _r.sent(); responses.push({ errors: [errorToBundle(error_3)], name: request.name, requestId: requestId, type: 'operate', }); return [3 /*break*/, 13]; case 13: return [3 /*break*/, 45]; case 14: if (!transact || !pool) { error = new Error('mutate requests require both transact and pool.'); responses.push({ action: request.action, errors: [errorToBundle(error)], requestId: requestId, type: request.type, }); return [3 /*break*/, 45]; } return [4 /*yield*/, transact()]; case 15: transaction = _r.sent(); _r.label = 16; case 16: _r.trys.push([16, 22, , 24]); return [4 /*yield*/, pool.sync(transaction, request, context)]; case 17: _e = _r.sent(), generatedIds = _e.generatedIds, errors = _e.errors; if (!(errors.length > 0)) return [3 /*break*/, 19]; return [4 /*yield*/, transaction.rollback()]; case 18: _r.sent(); responses.push({ action: request.action, errors: errors, requestId: requestId, type: request.type, }); return [3 /*break*/, 21]; case 19: return [4 /*yield*/, transaction.commit()]; case 20: _r.sent(); responses.push({ action: request.action, errors: [], generatedIds: generatedIds, requestId: requestId, type: request.type, }); _r.label = 21; case 21: return [3 /*break*/, 24]; case 22: error_4 = _r.sent(); return [4 /*yield*/, transaction.rollback()]; case 23: _r.sent(); responses.push({ action: request.action, errors: [errorToBundle(error_4)], requestId: requestId, type: request.type, }); return [3 /*break*/, 24]; case 24: return [3 /*break*/, 45]; case 25: batch = { errors: [], queue: [], requestId: requestId, type: request.type, }; if (!transact || !pool) { e = [errorToBundle(new Error('mutate requests require both transact and pool.'))]; for (_f = 0, _g = request.queue; _f < _g.length; _f++) { _h = _g[_f], action = _h.action, id = _h.requestId, type = _h.type; batch.queue.push({ action: action, errors: e, requestId: id, type: type }); } responses.push(tslib_1.__assign({}, batch, { errors: e })); return [3 /*break*/, 45]; } failed = false; failReasons = []; return [4 /*yield*/, transact()]; case 26: batchTr = _r.sent(); _r.label = 27; case 27: _r.trys.push([27, 36, , 38]); _j = 0, _k = request.queue; _r.label = 28; case 28: if (!(_j < _k.length)) return [3 /*break*/, 35]; reqItem = _k[_j]; action = reqItem.action, type = reqItem.type, itemId = reqItem.requestId; if (!!failed) return [3 /*break*/, 33]; _r.label = 29; case 29: _r.trys.push([29, 31, , 32]); return [4 /*yield*/, pool.sync(batchTr, reqItem, context)]; case 30: _l = _r.sent(), generatedIds = _l.generatedIds, errors = _l.errors; if (errors.length > 0) { failed = true; failReasons = errors; batchTr.rollback(); batch.queue.push({ action: action, type: type, requestId: itemId, errors: errors }); } else { batch.queue.push({ action: action, errors: errors, generatedIds: generatedIds, requestId: itemId, type: type, }); } return [3 /*break*/, 32]; case 31: error_5 = _r.sent(); failed = true; failReasons = [errorToBundle(error_5)]; batchTr.rollback(); batch.queue.push({ action: action, errors: failReasons, requestId: itemId, type: type }); return [3 /*break*/, 32]; case 32: return [3 /*break*/, 34]; case 33: batch.queue.push({ action: reqItem.action, errors: failReasons, requestId: reqItem.requestId, type: reqItem.type, }); _r.label = 34; case 34: _j++; return [3 /*break*/, 28]; case 35: return [3 /*break*/, 38]; case 36: error_6 = _r.sent(); return [4 /*yield*/, batchTr.rollback()]; case 37: _r.sent(); for (_m = 0, _o = batch.queue; _m < _o.length; _m++) { resItem = _o[_m]; resItem.errors = resItem.errors ? resItem.errors.concat([errorToBundle(error_6)]) : [errorToBundle(error_6)]; } failed = true; return [3 /*break*/, 38]; case 38: if (!!failed) return [3 /*break*/, 40]; return [4 /*yield*/, batchTr.commit()]; case 39: _r.sent(); _r.label = 40; case 40: responses.push(batch); return [3 /*break*/, 45]; case 41: _r.trys.push([41, 43, , 44]); if (!validator) { throw new Error('validate requests require a validator.'); } _q = (_p = responses).push; return [4 /*yield*/, validator(request, context)]; case 42: _q.apply(_p, [_r.sent()]); return [3 /*break*/, 44]; case 43: error_7 = _r.sent(); responses.push({ errors: [errorToBundle(error_7)], name: request.name, requestId: requestId, type: 'validate', }); return [3 /*break*/, 44]; case 44: return [3 /*break*/, 45]; case 45: _i++; return [3 /*break*/, 1]; case 46: return [2 /*return*/, responses]; } }); }); }; } exports.buildResponder = buildResponder; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"responder.js","sourceRoot":"","sources":["../src/responder.ts"],"names":[],"mappings":"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","sourcesContent":["import { EmptySpec } from 'pretur.spec';\r\nimport { Bundle } from 'pretur.i18n';\r\nimport { Query } from './query';\r\nimport { Request, OperateRequest, ValidateRequest, MutateRequest } from './request';\r\nimport {\r\n  Response, OperateResponse, ValidateResponse, MutateResponse, BatchMutateResponse,\r\n} from './response';\r\n\r\n// Manually import some types to decouple the librarires\r\nexport interface TransactionLike {\r\n  commit(): Promise<void>;\r\n  rollback(): Promise<void>;\r\n}\r\n\r\nexport interface ResolveResult {\r\n  data: any[];\r\n  count: number;\r\n}\r\n\r\nexport interface SyncResult {\r\n  errors: Bundle[];\r\n  generatedIds?: any;\r\n}\r\n\r\nexport interface PoolLike<C> {\r\n  resolve(\r\n    transaction: any,\r\n    scope: string,\r\n    model: string,\r\n    query: Query<EmptySpec>,\r\n    context?: C,\r\n  ): Promise<ResolveResult>;\r\n  sync(\r\n    transaction: any,\r\n    item: MutateRequest,\r\n    context?: C,\r\n  ): Promise<SyncResult>;\r\n}\r\n\r\nexport interface ResponderOptions<C> {\r\n  errorToBundle: (error: Error) => Bundle;\r\n  pool?: PoolLike<C>;\r\n  operator?: (request: OperateRequest, context?: C) => PromiseLike<OperateResponse>;\r\n  validator?: (request: ValidateRequest, context?: C) => PromiseLike<ValidateResponse>;\r\n  transact?: () => PromiseLike<TransactionLike>;\r\n}\r\n\r\nexport function buildResponder<C>(options: ResponderOptions<C>) {\r\n  const { errorToBundle, pool, operator, validator, transact } = options;\r\n\r\n  return async function responder(requests: Request[], context?: any): Promise<Response[]> {\r\n    if (!Array.isArray(requests) || requests.length === 0) {\r\n      return [];\r\n    }\r\n\r\n    const responses: Response[] = [];\r\n\r\n    for (const request of requests) {\r\n      const requestId = request.requestId;\r\n\r\n      switch (request.type) {\r\n        case 'select':\r\n          try {\r\n            if (!transact || !pool) {\r\n              throw new Error('select requests require transact and pool to operate.');\r\n            }\r\n            const tr = await transact();\r\n            try {\r\n              const { data, count } = await pool.resolve(\r\n                tr,\r\n                request.scope,\r\n                request.model,\r\n                request.query,\r\n                context,\r\n              );\r\n              tr.commit();\r\n              responses.push({ errors: [], count, data, requestId, type: 'select' });\r\n            } catch (error) {\r\n\r\n              responses.push({\r\n                errors: [errorToBundle(error)],\r\n                count: 0,\r\n                data: [],\r\n                requestId,\r\n                type: 'select',\r\n              });\r\n              tr.rollback();\r\n            }\r\n          } catch (error) {\r\n            responses.push({\r\n              count: 0,\r\n              data: [],\r\n              errors: [errorToBundle(error)],\r\n              requestId,\r\n              type: 'select',\r\n            });\r\n          }\r\n          break;\r\n        case 'operate':\r\n          try {\r\n            if (!operator) {\r\n              throw new Error('operate requests require an operator.');\r\n            }\r\n            responses.push(await operator(request, context));\r\n          } catch (error) {\r\n            responses.push({\r\n              errors: [errorToBundle(error)],\r\n              name: request.name,\r\n              requestId,\r\n              type: 'operate',\r\n            });\r\n          }\r\n          break;\r\n        case 'mutate':\r\n          if (!transact || !pool) {\r\n            const error = new Error('mutate requests require both transact and pool.');\r\n            responses.push(<MutateResponse>{\r\n              action: request.action,\r\n              errors: [errorToBundle(error)],\r\n              requestId,\r\n              type: request.type,\r\n            });\r\n            break;\r\n          }\r\n\r\n          const transaction = await transact();\r\n          try {\r\n            const { generatedIds, errors } = await pool.sync(transaction, request, context);\r\n            if (errors.length > 0) {\r\n              await transaction.rollback();\r\n              responses.push(<MutateResponse>{\r\n                action: request.action,\r\n                errors,\r\n                requestId,\r\n                type: request.type,\r\n              });\r\n            } else {\r\n              await transaction.commit();\r\n              responses.push(<MutateResponse>{\r\n                action: request.action,\r\n                errors: [],\r\n                generatedIds,\r\n                requestId,\r\n                type: request.type,\r\n              });\r\n            }\r\n          } catch (error) {\r\n            await transaction.rollback();\r\n            responses.push(<MutateResponse>{\r\n              action: request.action,\r\n              errors: [errorToBundle(error)],\r\n              requestId,\r\n              type: request.type,\r\n            });\r\n          }\r\n          break;\r\n        case 'batchMutate':\r\n          const batch: BatchMutateResponse = {\r\n            errors: [],\r\n            queue: [],\r\n            requestId,\r\n            type: request.type,\r\n          };\r\n\r\n          if (!transact || !pool) {\r\n            const e = [errorToBundle(new Error('mutate requests require both transact and pool.'))];\r\n\r\n            for (const { action, requestId: id, type } of request.queue) {\r\n              batch.queue.push(<MutateResponse>{ action, errors: e, requestId: id, type });\r\n            }\r\n\r\n            responses.push({ ...batch, errors: e });\r\n            break;\r\n          }\r\n\r\n          let failed = false;\r\n          let failReasons: Bundle[] = [];\r\n          const batchTr = await transact();\r\n          try {\r\n            for (const reqItem of request.queue) {\r\n              const { action, type, requestId: itemId } = reqItem;\r\n              if (!failed) {\r\n                try {\r\n                  const { generatedIds, errors } = await pool.sync(batchTr, reqItem, context);\r\n                  if (errors.length > 0) {\r\n                    failed = true;\r\n                    failReasons = errors;\r\n                    batchTr.rollback();\r\n                    batch.queue.push({ action, type, requestId: itemId, errors });\r\n                  } else {\r\n                    batch.queue.push({\r\n                      action,\r\n                      errors,\r\n                      generatedIds,\r\n                      requestId: itemId,\r\n                      type,\r\n                    });\r\n                  }\r\n                } catch (error) {\r\n                  failed = true;\r\n                  failReasons = [errorToBundle(error)];\r\n                  batchTr.rollback();\r\n                  batch.queue.push({ action, errors: failReasons, requestId: itemId, type });\r\n                }\r\n              } else {\r\n                batch.queue.push({\r\n                  action: reqItem.action,\r\n                  errors: failReasons,\r\n                  requestId: reqItem.requestId,\r\n                  type: reqItem.type,\r\n                });\r\n              }\r\n            }\r\n\r\n          } catch (error) {\r\n            await batchTr.rollback();\r\n            for (const resItem of batch.queue) {\r\n              resItem.errors = resItem.errors\r\n                ? [...resItem.errors, errorToBundle(error)]\r\n                : [errorToBundle(error)];\r\n            }\r\n            failed = true;\r\n          }\r\n\r\n          if (!failed) {\r\n            await batchTr.commit();\r\n          }\r\n\r\n          responses.push(batch);\r\n          break;\r\n        case 'validate':\r\n          try {\r\n            if (!validator) {\r\n              throw new Error('validate requests require a validator.');\r\n            }\r\n            responses.push(await validator(request, context));\r\n          } catch (error) {\r\n            responses.push({\r\n              errors: [errorToBundle(error)],\r\n              name: request.name,\r\n              requestId,\r\n              type: 'validate',\r\n            });\r\n          }\r\n          break;\r\n      }\r\n    }\r\n\r\n    return responses;\r\n  };\r\n}\r\n"]}