@orbit/record-cache
Version:
Orbit base classes used to access and maintain a set of records.
243 lines • 33.7 kB
JavaScript
import { equalRecordIdentities, equalRecordIdentitySets, RecordNotFoundException, recordsInclude } from '@orbit/records';
import { deepGet, deepSet, eq } from '@orbit/utils';
export const AsyncInverseTransformOperators = {
async addRecord(cache, operation,
// eslint-disable-next-line @typescript-eslint/no-unused-vars
options) {
const op = operation;
const { type, id } = op.record;
const current = await cache.getRecordAsync(op.record);
if (current) {
if (eq(current, op.record)) {
return;
}
else {
return {
op: 'updateRecord',
record: current
};
}
}
else {
return {
op: 'removeRecord',
record: { type, id }
};
}
},
async updateRecord(cache, operation, options) {
const op = operation;
const currentRecord = await cache.getRecordAsync(op.record);
const replacement = op.record;
const { type, id } = replacement;
if (currentRecord) {
let result = { type, id };
let changed = false;
['attributes', 'keys'].forEach((grouping) => {
if (replacement[grouping]) {
Object.keys(replacement[grouping]).forEach((field) => {
let value = replacement[grouping][field];
let currentValue = deepGet(currentRecord, [grouping, field]);
if (!eq(value, currentValue)) {
changed = true;
deepSet(result, [grouping, field], currentValue === undefined ? null : currentValue);
}
});
}
});
if (replacement.relationships) {
Object.keys(replacement.relationships).forEach((field) => {
let data = deepGet(replacement, ['relationships', field, 'data']);
if (data !== undefined) {
let currentData = deepGet(currentRecord, [
'relationships',
field,
'data'
]);
let relationshipChanged;
if (Array.isArray(data)) {
if (currentData) {
relationshipChanged = !equalRecordIdentitySets(currentData, data);
}
else {
relationshipChanged = true;
currentData = [];
}
}
else {
if (currentData) {
relationshipChanged = !equalRecordIdentities(currentData, data);
}
else {
relationshipChanged = true;
currentData = null;
}
}
if (relationshipChanged) {
changed = true;
deepSet(result, ['relationships', field, 'data'], currentData);
}
}
});
}
if (changed) {
return {
op: 'updateRecord',
record: result
};
}
}
else {
if (options === null || options === void 0 ? void 0 : options.raiseNotFoundExceptions) {
throw new RecordNotFoundException(type, id);
}
else {
return {
op: 'removeRecord',
record: { type, id }
};
}
}
},
async removeRecord(cache, operation, options) {
const op = operation;
const { record } = op;
const currentRecord = await cache.getRecordAsync(record);
if (currentRecord) {
return {
op: 'addRecord',
record: currentRecord
};
}
else {
if (options === null || options === void 0 ? void 0 : options.raiseNotFoundExceptions) {
throw new RecordNotFoundException(record.type, record.id);
}
}
},
async replaceKey(cache, operation, options) {
const op = operation;
const { record, key } = op;
const currentRecord = await cache.getRecordAsync(record);
if (currentRecord === undefined) {
if (options === null || options === void 0 ? void 0 : options.raiseNotFoundExceptions) {
throw new RecordNotFoundException(record.type, record.id);
}
}
const currentValue = currentRecord && deepGet(currentRecord, ['keys', key]);
if (!eq(currentValue, op.value)) {
const { type, id } = op.record;
return {
op: 'replaceKey',
record: { type, id },
key,
value: currentValue
};
}
},
async replaceAttribute(cache, operation, options) {
const op = operation;
const { record, attribute } = op;
const currentRecord = await cache.getRecordAsync(record);
if (currentRecord === undefined) {
if (options === null || options === void 0 ? void 0 : options.raiseNotFoundExceptions) {
throw new RecordNotFoundException(record.type, record.id);
}
}
const currentValue = currentRecord && deepGet(currentRecord, ['attributes', attribute]);
if (!eq(currentValue, op.value)) {
const { type, id } = record;
return {
op: 'replaceAttribute',
record: { type, id },
attribute,
value: currentValue
};
}
},
async addToRelatedRecords(cache, operation, options) {
const op = operation;
const { record, relationship, relatedRecord } = op;
const currentRelatedRecords = await cache.getRelatedRecordsAsync(record, relationship);
if (currentRelatedRecords === undefined) {
if (options === null || options === void 0 ? void 0 : options.raiseNotFoundExceptions) {
if ((await cache.getRecordAsync(record)) === undefined) {
throw new RecordNotFoundException(record.type, record.id);
}
}
}
if (currentRelatedRecords === undefined ||
!recordsInclude(currentRelatedRecords, relatedRecord)) {
return {
op: 'removeFromRelatedRecords',
record,
relationship,
relatedRecord
};
}
},
async removeFromRelatedRecords(cache, operation, options) {
const op = operation;
const { record, relationship, relatedRecord } = op;
const currentRelatedRecords = await cache.getRelatedRecordsAsync(record, relationship);
if (currentRelatedRecords === undefined) {
if (options === null || options === void 0 ? void 0 : options.raiseNotFoundExceptions) {
if ((await cache.getRecordAsync(record)) === undefined) {
throw new RecordNotFoundException(record.type, record.id);
}
}
}
if (currentRelatedRecords !== undefined &&
recordsInclude(currentRelatedRecords, relatedRecord)) {
return {
op: 'addToRelatedRecords',
record,
relationship,
relatedRecord
};
}
},
async replaceRelatedRecords(cache, operation, options) {
const op = operation;
const { record, relationship, relatedRecords } = op;
const currentRelatedRecords = await cache.getRelatedRecordsAsync(record, relationship);
if (currentRelatedRecords === undefined) {
if (options === null || options === void 0 ? void 0 : options.raiseNotFoundExceptions) {
if ((await cache.getRecordAsync(record)) === undefined) {
throw new RecordNotFoundException(record.type, record.id);
}
}
}
if (currentRelatedRecords === undefined ||
!equalRecordIdentitySets(currentRelatedRecords, relatedRecords)) {
return {
op: 'replaceRelatedRecords',
record,
relationship,
relatedRecords: currentRelatedRecords || []
};
}
},
async replaceRelatedRecord(cache, operation, options) {
const op = operation;
const { record, relationship, relatedRecord } = op;
const currentRelatedRecord = await cache.getRelatedRecordAsync(record, relationship);
if (currentRelatedRecord === undefined) {
if (options === null || options === void 0 ? void 0 : options.raiseNotFoundExceptions) {
if ((await cache.getRecordAsync(record)) === undefined) {
throw new RecordNotFoundException(record.type, record.id);
}
}
}
if (currentRelatedRecord === undefined ||
!equalRecordIdentities(currentRelatedRecord, relatedRecord)) {
return {
op: 'replaceRelatedRecord',
record,
relationship,
relatedRecord: currentRelatedRecord || null
};
}
}
};
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"async-inverse-transform-operators.js","sourceRoot":"","sources":["../../../src/operators/async-inverse-transform-operators.ts"],"names":[],"mappings":"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","sourcesContent":["import { RequestOptions } from '@orbit/data';\nimport {\n  AddRecordOperation,\n  AddToRelatedRecordsOperation,\n  equalRecordIdentities,\n  equalRecordIdentitySets,\n  InitializedRecord,\n  RecordIdentity,\n  RecordNotFoundException,\n  RecordOperation,\n  recordsInclude,\n  RemoveFromRelatedRecordsOperation,\n  RemoveRecordOperation,\n  ReplaceAttributeOperation,\n  ReplaceKeyOperation,\n  ReplaceRelatedRecordOperation,\n  ReplaceRelatedRecordsOperation,\n  UpdateRecordOperation\n} from '@orbit/records';\nimport { deepGet, deepSet, Dict, eq } from '@orbit/utils';\nimport { AsyncRecordAccessor } from '../record-accessor';\n\nexport interface AsyncInverseTransformOperator {\n  (\n    cache: AsyncRecordAccessor,\n    operation: RecordOperation,\n    options?: RequestOptions\n  ): Promise<RecordOperation | undefined>;\n}\n\nexport const AsyncInverseTransformOperators: Dict<AsyncInverseTransformOperator> = {\n  async addRecord(\n    cache: AsyncRecordAccessor,\n    operation: RecordOperation,\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    options?: RequestOptions\n  ): Promise<RecordOperation | undefined> {\n    const op = operation as AddRecordOperation;\n    const { type, id } = op.record;\n    const current = await cache.getRecordAsync(op.record);\n\n    if (current) {\n      if (eq(current, op.record)) {\n        return;\n      } else {\n        return {\n          op: 'updateRecord',\n          record: current\n        };\n      }\n    } else {\n      return {\n        op: 'removeRecord',\n        record: { type, id }\n      };\n    }\n  },\n\n  async updateRecord(\n    cache: AsyncRecordAccessor,\n    operation: RecordOperation,\n    options?: RequestOptions\n  ): Promise<RecordOperation | undefined> {\n    const op = operation as UpdateRecordOperation;\n    const currentRecord = await cache.getRecordAsync(op.record);\n    const replacement: InitializedRecord = op.record;\n    const { type, id } = replacement;\n\n    if (currentRecord) {\n      let result = { type, id };\n      let changed = false;\n\n      ['attributes', 'keys'].forEach((grouping) => {\n        if ((replacement as any)[grouping]) {\n          Object.keys((replacement as any)[grouping]).forEach((field) => {\n            let value = (replacement as any)[grouping][field];\n            let currentValue = deepGet(currentRecord, [grouping, field]);\n            if (!eq(value, currentValue)) {\n              changed = true;\n              deepSet(\n                result,\n                [grouping, field],\n                currentValue === undefined ? null : currentValue\n              );\n            }\n          });\n        }\n      });\n\n      if (replacement.relationships) {\n        Object.keys(replacement.relationships).forEach((field) => {\n          let data = deepGet(replacement, ['relationships', field, 'data']);\n          if (data !== undefined) {\n            let currentData = deepGet(currentRecord, [\n              'relationships',\n              field,\n              'data'\n            ]);\n            let relationshipChanged;\n\n            if (Array.isArray(data)) {\n              if (currentData) {\n                relationshipChanged = !equalRecordIdentitySets(\n                  currentData,\n                  data\n                );\n              } else {\n                relationshipChanged = true;\n                currentData = [];\n              }\n            } else {\n              if (currentData) {\n                relationshipChanged = !equalRecordIdentities(currentData, data);\n              } else {\n                relationshipChanged = true;\n                currentData = null;\n              }\n            }\n\n            if (relationshipChanged) {\n              changed = true;\n              deepSet(result, ['relationships', field, 'data'], currentData);\n            }\n          }\n        });\n      }\n\n      if (changed) {\n        return {\n          op: 'updateRecord',\n          record: result\n        };\n      }\n    } else {\n      if (options?.raiseNotFoundExceptions) {\n        throw new RecordNotFoundException(type, id);\n      } else {\n        return {\n          op: 'removeRecord',\n          record: { type, id }\n        };\n      }\n    }\n  },\n\n  async removeRecord(\n    cache: AsyncRecordAccessor,\n    operation: RecordOperation,\n    options?: RequestOptions\n  ): Promise<RecordOperation | undefined> {\n    const op = operation as RemoveRecordOperation;\n    const { record } = op;\n    const currentRecord = await cache.getRecordAsync(record);\n\n    if (currentRecord) {\n      return {\n        op: 'addRecord',\n        record: currentRecord\n      };\n    } else {\n      if (options?.raiseNotFoundExceptions) {\n        throw new RecordNotFoundException(record.type, record.id);\n      }\n    }\n  },\n\n  async replaceKey(\n    cache: AsyncRecordAccessor,\n    operation: RecordOperation,\n    options?: RequestOptions\n  ): Promise<RecordOperation | undefined> {\n    const op = operation as ReplaceKeyOperation;\n    const { record, key } = op;\n    const currentRecord = await cache.getRecordAsync(record);\n\n    if (currentRecord === undefined) {\n      if (options?.raiseNotFoundExceptions) {\n        throw new RecordNotFoundException(record.type, record.id);\n      }\n    }\n\n    const currentValue = currentRecord && deepGet(currentRecord, ['keys', key]);\n\n    if (!eq(currentValue, op.value)) {\n      const { type, id } = op.record;\n\n      return {\n        op: 'replaceKey',\n        record: { type, id },\n        key,\n        value: currentValue\n      };\n    }\n  },\n\n  async replaceAttribute(\n    cache: AsyncRecordAccessor,\n    operation: RecordOperation,\n    options?: RequestOptions\n  ): Promise<RecordOperation | undefined> {\n    const op = operation as ReplaceAttributeOperation;\n    const { record, attribute } = op;\n    const currentRecord = await cache.getRecordAsync(record);\n\n    if (currentRecord === undefined) {\n      if (options?.raiseNotFoundExceptions) {\n        throw new RecordNotFoundException(record.type, record.id);\n      }\n    }\n\n    const currentValue =\n      currentRecord && deepGet(currentRecord, ['attributes', attribute]);\n\n    if (!eq(currentValue, op.value)) {\n      const { type, id } = record;\n\n      return {\n        op: 'replaceAttribute',\n        record: { type, id },\n        attribute,\n        value: currentValue\n      };\n    }\n  },\n\n  async addToRelatedRecords(\n    cache: AsyncRecordAccessor,\n    operation: RecordOperation,\n    options?: RequestOptions\n  ): Promise<RecordOperation | undefined> {\n    const op = operation as AddToRelatedRecordsOperation;\n    const { record, relationship, relatedRecord } = op;\n    const currentRelatedRecords = await cache.getRelatedRecordsAsync(\n      record,\n      relationship\n    );\n\n    if (currentRelatedRecords === undefined) {\n      if (options?.raiseNotFoundExceptions) {\n        if ((await cache.getRecordAsync(record)) === undefined) {\n          throw new RecordNotFoundException(record.type, record.id);\n        }\n      }\n    }\n\n    if (\n      currentRelatedRecords === undefined ||\n      !recordsInclude(currentRelatedRecords, relatedRecord)\n    ) {\n      return {\n        op: 'removeFromRelatedRecords',\n        record,\n        relationship,\n        relatedRecord\n      };\n    }\n  },\n\n  async removeFromRelatedRecords(\n    cache: AsyncRecordAccessor,\n    operation: RecordOperation,\n    options?: RequestOptions\n  ): Promise<RecordOperation | undefined> {\n    const op = operation as RemoveFromRelatedRecordsOperation;\n    const { record, relationship, relatedRecord } = op;\n    const currentRelatedRecords = await cache.getRelatedRecordsAsync(\n      record,\n      relationship\n    );\n\n    if (currentRelatedRecords === undefined) {\n      if (options?.raiseNotFoundExceptions) {\n        if ((await cache.getRecordAsync(record)) === undefined) {\n          throw new RecordNotFoundException(record.type, record.id);\n        }\n      }\n    }\n\n    if (\n      currentRelatedRecords !== undefined &&\n      recordsInclude(currentRelatedRecords, relatedRecord)\n    ) {\n      return {\n        op: 'addToRelatedRecords',\n        record,\n        relationship,\n        relatedRecord\n      };\n    }\n  },\n\n  async replaceRelatedRecords(\n    cache: AsyncRecordAccessor,\n    operation: RecordOperation,\n    options?: RequestOptions\n  ): Promise<RecordOperation | undefined> {\n    const op = operation as ReplaceRelatedRecordsOperation;\n    const { record, relationship, relatedRecords } = op;\n    const currentRelatedRecords = await cache.getRelatedRecordsAsync(\n      record,\n      relationship\n    );\n\n    if (currentRelatedRecords === undefined) {\n      if (options?.raiseNotFoundExceptions) {\n        if ((await cache.getRecordAsync(record)) === undefined) {\n          throw new RecordNotFoundException(record.type, record.id);\n        }\n      }\n    }\n\n    if (\n      currentRelatedRecords === undefined ||\n      !equalRecordIdentitySets(currentRelatedRecords, relatedRecords)\n    ) {\n      return {\n        op: 'replaceRelatedRecords',\n        record,\n        relationship,\n        relatedRecords: currentRelatedRecords || []\n      };\n    }\n  },\n\n  async replaceRelatedRecord(\n    cache: AsyncRecordAccessor,\n    operation: RecordOperation,\n    options?: RequestOptions\n  ): Promise<RecordOperation | undefined> {\n    const op = operation as ReplaceRelatedRecordOperation;\n    const { record, relationship, relatedRecord } = op;\n    const currentRelatedRecord = await cache.getRelatedRecordAsync(\n      record,\n      relationship\n    );\n\n    if (currentRelatedRecord === undefined) {\n      if (options?.raiseNotFoundExceptions) {\n        if ((await cache.getRecordAsync(record)) === undefined) {\n          throw new RecordNotFoundException(record.type, record.id);\n        }\n      }\n    }\n\n    if (\n      currentRelatedRecord === undefined ||\n      !equalRecordIdentities(\n        currentRelatedRecord as RecordIdentity,\n        relatedRecord as RecordIdentity\n      )\n    ) {\n      return {\n        op: 'replaceRelatedRecord',\n        record,\n        relationship,\n        relatedRecord: currentRelatedRecord || null\n      };\n    }\n  }\n};\n"]}