@orbit/record-cache
Version:
Orbit base classes used to access and maintain a set of records.
165 lines • 25.1 kB
JavaScript
import { cloneRecordIdentity, equalRecordIdentities, mergeRecords, RecordNotFoundException, recordsInclude } from '@orbit/records';
import { clone, deepGet, deepSet } from '@orbit/utils';
export const AsyncTransformOperators = {
async addRecord(cache, operation,
// eslint-disable-next-line @typescript-eslint/no-unused-vars
options) {
const op = operation;
const { record } = op;
await cache.setRecordAsync(record);
if (cache.keyMap) {
cache.keyMap.pushRecord(record);
}
return record;
},
async updateRecord(cache, operation, options) {
const op = operation;
const { record } = 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 mergedRecord = mergeRecords(currentRecord || null, record);
await cache.setRecordAsync(mergedRecord);
if (cache.keyMap) {
cache.keyMap.pushRecord(mergedRecord);
}
return mergedRecord;
},
async removeRecord(cache, operation, options) {
const op = operation;
const record = await cache.removeRecordAsync(op.record);
if (record === undefined) {
if (options === null || options === void 0 ? void 0 : options.raiseNotFoundExceptions) {
throw new RecordNotFoundException(op.record.type, op.record.id);
}
}
return record;
},
async replaceKey(cache, operation, options) {
const op = operation;
const currentRecord = await cache.getRecordAsync(op.record);
let record;
if (currentRecord) {
record = clone(currentRecord);
}
else {
record = cloneRecordIdentity(op.record);
if (options === null || options === void 0 ? void 0 : options.raiseNotFoundExceptions) {
throw new RecordNotFoundException(record.type, record.id);
}
}
deepSet(record, ['keys', op.key], op.value);
await cache.setRecordAsync(record);
if (cache.keyMap) {
cache.keyMap.pushRecord(record);
}
return record;
},
async replaceAttribute(cache, operation, options) {
const op = operation;
const currentRecord = await cache.getRecordAsync(op.record);
let record;
if (currentRecord) {
record = clone(currentRecord);
}
else {
record = cloneRecordIdentity(op.record);
if (options === null || options === void 0 ? void 0 : options.raiseNotFoundExceptions) {
throw new RecordNotFoundException(record.type, record.id);
}
}
deepSet(record, ['attributes', op.attribute], op.value);
await cache.setRecordAsync(record);
return record;
},
async addToRelatedRecords(cache, operation, options) {
const op = operation;
const { relationship, relatedRecord } = op;
const currentRecord = await cache.getRecordAsync(op.record);
let record;
if (currentRecord) {
record = clone(currentRecord);
}
else {
record = cloneRecordIdentity(op.record);
if (options === null || options === void 0 ? void 0 : options.raiseNotFoundExceptions) {
throw new RecordNotFoundException(record.type, record.id);
}
}
const relatedRecords = deepGet(record, ['relationships', relationship, 'data']) || [];
if (!recordsInclude(relatedRecords, relatedRecord)) {
relatedRecords.push(relatedRecord);
deepSet(record, ['relationships', relationship, 'data'], relatedRecords);
await cache.setRecordAsync(record);
}
return record;
},
async removeFromRelatedRecords(cache, operation, options) {
const op = operation;
const currentRecord = await cache.getRecordAsync(op.record);
const { relationship, relatedRecord } = op;
let record;
if (currentRecord) {
record = clone(currentRecord);
let relatedRecords = deepGet(record, [
'relationships',
relationship,
'data'
]);
if (relatedRecords) {
relatedRecords = relatedRecords.filter((r) => !equalRecordIdentities(r, relatedRecord));
if (deepSet(record, ['relationships', relationship, 'data'], relatedRecords)) {
await cache.setRecordAsync(record);
}
}
return record;
}
else {
if (options === null || options === void 0 ? void 0 : options.raiseNotFoundExceptions) {
throw new RecordNotFoundException(op.record.type, op.record.id);
}
}
},
async replaceRelatedRecords(cache, operation, options) {
const op = operation;
const currentRecord = await cache.getRecordAsync(op.record);
const { relationship, relatedRecords } = op;
let record;
if (currentRecord) {
record = clone(currentRecord);
}
else {
record = cloneRecordIdentity(op.record);
if (options === null || options === void 0 ? void 0 : options.raiseNotFoundExceptions) {
throw new RecordNotFoundException(record.type, record.id);
}
}
if (deepSet(record, ['relationships', relationship, 'data'], relatedRecords)) {
await cache.setRecordAsync(record);
}
return record;
},
async replaceRelatedRecord(cache, operation, options) {
const op = operation;
const currentRecord = await cache.getRecordAsync(op.record);
const { relationship, relatedRecord } = op;
let record;
if (currentRecord) {
record = clone(currentRecord);
}
else {
record = cloneRecordIdentity(op.record);
if (options === null || options === void 0 ? void 0 : options.raiseNotFoundExceptions) {
throw new RecordNotFoundException(record.type, record.id);
}
}
if (deepSet(record, ['relationships', relationship, 'data'], relatedRecord)) {
await cache.setRecordAsync(record);
}
return record;
}
};
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"async-transform-operators.js","sourceRoot":"","sources":["../../../src/operators/async-transform-operators.ts"],"names":[],"mappings":"AACA,OAAO,EAGL,mBAAmB,EACnB,qBAAqB,EAErB,YAAY,EAEZ,uBAAuB,EAGvB,cAAc,EAQf,MAAM,gBAAgB,CAAC;AACxB,OAAO,EAAE,KAAK,EAAE,OAAO,EAAE,OAAO,EAAQ,MAAM,cAAc,CAAC;AAW7D,MAAM,CAAC,MAAM,uBAAuB,GAAiC;IACnE,KAAK,CAAC,SAAS,CACb,KAA0B,EAC1B,SAA0B;IAC1B,6DAA6D;IAC7D,OAAwB;QAExB,MAAM,EAAE,GAAG,SAA+B,CAAC;QAC3C,MAAM,EAAE,MAAM,EAAE,GAAG,EAAE,CAAC;QACtB,MAAM,KAAK,CAAC,cAAc,CAAC,MAAM,CAAC,CAAC;QAEnC,IAAI,KAAK,CAAC,MAAM,EAAE;YAChB,KAAK,CAAC,MAAM,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC;SACjC;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;IAED,KAAK,CAAC,YAAY,CAChB,KAA0B,EAC1B,SAA0B,EAC1B,OAAwB;QAExB,MAAM,EAAE,GAAG,SAAkC,CAAC;QAC9C,MAAM,EAAE,MAAM,EAAE,GAAG,EAAE,CAAC;QACtB,MAAM,aAAa,GAAG,MAAM,KAAK,CAAC,cAAc,CAAC,MAAM,CAAC,CAAC;QAEzD,IAAI,aAAa,KAAK,SAAS,EAAE;YAC/B,IAAI,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,uBAAuB,EAAE;gBACpC,MAAM,IAAI,uBAAuB,CAAC,MAAM,CAAC,IAAI,EAAE,MAAM,CAAC,EAAE,CAAC,CAAC;aAC3D;SACF;QAED,MAAM,YAAY,GAAG,YAAY,CAAC,aAAa,IAAI,IAAI,EAAE,MAAM,CAAC,CAAC;QAEjE,MAAM,KAAK,CAAC,cAAc,CAAC,YAAY,CAAC,CAAC;QAEzC,IAAI,KAAK,CAAC,MAAM,EAAE;YAChB,KAAK,CAAC,MAAM,CAAC,UAAU,CAAC,YAAY,CAAC,CAAC;SACvC;QAED,OAAO,YAAY,CAAC;IACtB,CAAC;IAED,KAAK,CAAC,YAAY,CAChB,KAA0B,EAC1B,SAA0B,EAC1B,OAAwB;QAExB,MAAM,EAAE,GAAG,SAAkC,CAAC;QAC9C,MAAM,MAAM,GAAG,MAAM,KAAK,CAAC,iBAAiB,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC;QAExD,IAAI,MAAM,KAAK,SAAS,EAAE;YACxB,IAAI,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,uBAAuB,EAAE;gBACpC,MAAM,IAAI,uBAAuB,CAAC,EAAE,CAAC,MAAM,CAAC,IAAI,EAAE,EAAE,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;aACjE;SACF;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;IAED,KAAK,CAAC,UAAU,CACd,KAA0B,EAC1B,SAA0B,EAC1B,OAAwB;QAExB,MAAM,EAAE,GAAG,SAAgC,CAAC;QAC5C,MAAM,aAAa,GAAG,MAAM,KAAK,CAAC,cAAc,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC;QAC5D,IAAI,MAAyB,CAAC;QAE9B,IAAI,aAAa,EAAE;YACjB,MAAM,GAAG,KAAK,CAAC,aAAa,CAAC,CAAC;SAC/B;aAAM;YACL,MAAM,GAAG,mBAAmB,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC;YAExC,IAAI,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,uBAAuB,EAAE;gBACpC,MAAM,IAAI,uBAAuB,CAAC,MAAM,CAAC,IAAI,EAAE,MAAM,CAAC,EAAE,CAAC,CAAC;aAC3D;SACF;QAED,OAAO,CAAC,MAAM,EAAE,CAAC,MAAM,EAAE,EAAE,CAAC,GAAG,CAAC,EAAE,EAAE,CAAC,KAAK,CAAC,CAAC;QAC5C,MAAM,KAAK,CAAC,cAAc,CAAC,MAAM,CAAC,CAAC;QAEnC,IAAI,KAAK,CAAC,MAAM,EAAE;YAChB,KAAK,CAAC,MAAM,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC;SACjC;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;IAED,KAAK,CAAC,gBAAgB,CACpB,KAA0B,EAC1B,SAA0B,EAC1B,OAAwB;QAExB,MAAM,EAAE,GAAG,SAAsC,CAAC;QAClD,MAAM,aAAa,GAAG,MAAM,KAAK,CAAC,cAAc,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC;QAC5D,IAAI,MAAyB,CAAC;QAE9B,IAAI,aAAa,EAAE;YACjB,MAAM,GAAG,KAAK,CAAC,aAAa,CAAC,CAAC;SAC/B;aAAM;YACL,MAAM,GAAG,mBAAmB,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC;YAExC,IAAI,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,uBAAuB,EAAE;gBACpC,MAAM,IAAI,uBAAuB,CAAC,MAAM,CAAC,IAAI,EAAE,MAAM,CAAC,EAAE,CAAC,CAAC;aAC3D;SACF;QAED,OAAO,CAAC,MAAM,EAAE,CAAC,YAAY,EAAE,EAAE,CAAC,SAAS,CAAC,EAAE,EAAE,CAAC,KAAK,CAAC,CAAC;QACxD,MAAM,KAAK,CAAC,cAAc,CAAC,MAAM,CAAC,CAAC;QAEnC,OAAO,MAAM,CAAC;IAChB,CAAC;IAED,KAAK,CAAC,mBAAmB,CACvB,KAA0B,EAC1B,SAA0B,EAC1B,OAAwB;QAExB,MAAM,EAAE,GAAG,SAAyC,CAAC;QACrD,MAAM,EAAE,YAAY,EAAE,aAAa,EAAE,GAAG,EAAE,CAAC;QAC3C,MAAM,aAAa,GAAG,MAAM,KAAK,CAAC,cAAc,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC;QAC5D,IAAI,MAAyB,CAAC;QAE9B,IAAI,aAAa,EAAE;YACjB,MAAM,GAAG,KAAK,CAAC,aAAa,CAAC,CAAC;SAC/B;aAAM;YACL,MAAM,GAAG,mBAAmB,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC;YAExC,IAAI,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,uBAAuB,EAAE;gBACpC,MAAM,IAAI,uBAAuB,CAAC,MAAM,CAAC,IAAI,EAAE,MAAM,CAAC,EAAE,CAAC,CAAC;aAC3D;SACF;QAED,MAAM,cAAc,GAClB,OAAO,CAAC,MAAM,EAAE,CAAC,eAAe,EAAE,YAAY,EAAE,MAAM,CAAC,CAAC,IAAI,EAAE,CAAC;QAEjE,IAAI,CAAC,cAAc,CAAC,cAAc,EAAE,aAAa,CAAC,EAAE;YAClD,cAAc,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;YAEnC,OAAO,CAAC,MAAM,EAAE,CAAC,eAAe,EAAE,YAAY,EAAE,MAAM,CAAC,EAAE,cAAc,CAAC,CAAC;YACzE,MAAM,KAAK,CAAC,cAAc,CAAC,MAAM,CAAC,CAAC;SACpC;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;IAED,KAAK,CAAC,wBAAwB,CAC5B,KAA0B,EAC1B,SAA0B,EAC1B,OAAwB;QAExB,MAAM,EAAE,GAAG,SAA8C,CAAC;QAC1D,MAAM,aAAa,GAAG,MAAM,KAAK,CAAC,cAAc,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC;QAC5D,MAAM,EAAE,YAAY,EAAE,aAAa,EAAE,GAAG,EAAE,CAAC;QAC3C,IAAI,MAAyB,CAAC;QAE9B,IAAI,aAAa,EAAE;YACjB,MAAM,GAAG,KAAK,CAAC,aAAa,CAAC,CAAC;YAC9B,IAAI,cAAc,GAAqB,OAAO,CAAC,MAAM,EAAE;gBACrD,eAAe;gBACf,YAAY;gBACZ,MAAM;aACP,CAAC,CAAC;YACH,IAAI,cAAc,EAAE;gBAClB,cAAc,GAAG,cAAc,CAAC,MAAM,CACpC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,qBAAqB,CAAC,CAAC,EAAE,aAAa,CAAC,CAChD,CAAC;gBAEF,IACE,OAAO,CACL,MAAM,EACN,CAAC,eAAe,EAAE,YAAY,EAAE,MAAM,CAAC,EACvC,cAAc,CACf,EACD;oBACA,MAAM,KAAK,CAAC,cAAc,CAAC,MAAM,CAAC,CAAC;iBACpC;aACF;YACD,OAAO,MAAM,CAAC;SACf;aAAM;YACL,IAAI,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,uBAAuB,EAAE;gBACpC,MAAM,IAAI,uBAAuB,CAAC,EAAE,CAAC,MAAM,CAAC,IAAI,EAAE,EAAE,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;aACjE;SACF;IACH,CAAC;IAED,KAAK,CAAC,qBAAqB,CACzB,KAA0B,EAC1B,SAA0B,EAC1B,OAAwB;QAExB,MAAM,EAAE,GAAG,SAA2C,CAAC;QACvD,MAAM,aAAa,GAAG,MAAM,KAAK,CAAC,cAAc,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC;QAC5D,MAAM,EAAE,YAAY,EAAE,cAAc,EAAE,GAAG,EAAE,CAAC;QAC5C,IAAI,MAAyB,CAAC;QAE9B,IAAI,aAAa,EAAE;YACjB,MAAM,GAAG,KAAK,CAAC,aAAa,CAAC,CAAC;SAC/B;aAAM;YACL,MAAM,GAAG,mBAAmB,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC;YAExC,IAAI,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,uBAAuB,EAAE;gBACpC,MAAM,IAAI,uBAAuB,CAAC,MAAM,CAAC,IAAI,EAAE,MAAM,CAAC,EAAE,CAAC,CAAC;aAC3D;SACF;QAED,IACE,OAAO,CAAC,MAAM,EAAE,CAAC,eAAe,EAAE,YAAY,EAAE,MAAM,CAAC,EAAE,cAAc,CAAC,EACxE;YACA,MAAM,KAAK,CAAC,cAAc,CAAC,MAAM,CAAC,CAAC;SACpC;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;IAED,KAAK,CAAC,oBAAoB,CACxB,KAA0B,EAC1B,SAA0B,EAC1B,OAAwB;QAExB,MAAM,EAAE,GAAG,SAA0C,CAAC;QACtD,MAAM,aAAa,GAAG,MAAM,KAAK,CAAC,cAAc,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC;QAC5D,MAAM,EAAE,YAAY,EAAE,aAAa,EAAE,GAAG,EAAE,CAAC;QAC3C,IAAI,MAAyB,CAAC;QAE9B,IAAI,aAAa,EAAE;YACjB,MAAM,GAAG,KAAK,CAAC,aAAa,CAAC,CAAC;SAC/B;aAAM;YACL,MAAM,GAAG,mBAAmB,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC;YAExC,IAAI,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,uBAAuB,EAAE;gBACpC,MAAM,IAAI,uBAAuB,CAAC,MAAM,CAAC,IAAI,EAAE,MAAM,CAAC,EAAE,CAAC,CAAC;aAC3D;SACF;QAED,IACE,OAAO,CAAC,MAAM,EAAE,CAAC,eAAe,EAAE,YAAY,EAAE,MAAM,CAAC,EAAE,aAAa,CAAC,EACvE;YACA,MAAM,KAAK,CAAC,cAAc,CAAC,MAAM,CAAC,CAAC;SACpC;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;CACF,CAAC","sourcesContent":["import { RequestOptions } from '@orbit/data';\nimport {\n  AddRecordOperation,\n  AddToRelatedRecordsOperation,\n  cloneRecordIdentity,\n  equalRecordIdentities,\n  InitializedRecord,\n  mergeRecords,\n  RecordIdentity,\n  RecordNotFoundException,\n  RecordOperation,\n  RecordOperationResult,\n  recordsInclude,\n  RemoveFromRelatedRecordsOperation,\n  RemoveRecordOperation,\n  ReplaceAttributeOperation,\n  ReplaceKeyOperation,\n  ReplaceRelatedRecordOperation,\n  ReplaceRelatedRecordsOperation,\n  UpdateRecordOperation\n} from '@orbit/records';\nimport { clone, deepGet, deepSet, Dict } from '@orbit/utils';\nimport { AsyncRecordAccessor } from '../record-accessor';\n\nexport interface AsyncTransformOperator {\n  (\n    cache: AsyncRecordAccessor,\n    operation: RecordOperation,\n    options?: RequestOptions\n  ): Promise<RecordOperationResult>;\n}\n\nexport const AsyncTransformOperators: Dict<AsyncTransformOperator> = {\n  async addRecord(\n    cache: AsyncRecordAccessor,\n    operation: RecordOperation,\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    options?: RequestOptions\n  ): Promise<RecordOperationResult> {\n    const op = operation as AddRecordOperation;\n    const { record } = op;\n    await cache.setRecordAsync(record);\n\n    if (cache.keyMap) {\n      cache.keyMap.pushRecord(record);\n    }\n\n    return record;\n  },\n\n  async updateRecord(\n    cache: AsyncRecordAccessor,\n    operation: RecordOperation,\n    options?: RequestOptions\n  ): Promise<RecordOperationResult> {\n    const op = operation as UpdateRecordOperation;\n    const { record } = 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 mergedRecord = mergeRecords(currentRecord || null, record);\n\n    await cache.setRecordAsync(mergedRecord);\n\n    if (cache.keyMap) {\n      cache.keyMap.pushRecord(mergedRecord);\n    }\n\n    return mergedRecord;\n  },\n\n  async removeRecord(\n    cache: AsyncRecordAccessor,\n    operation: RecordOperation,\n    options?: RequestOptions\n  ): Promise<RecordOperationResult> {\n    const op = operation as RemoveRecordOperation;\n    const record = await cache.removeRecordAsync(op.record);\n\n    if (record === undefined) {\n      if (options?.raiseNotFoundExceptions) {\n        throw new RecordNotFoundException(op.record.type, op.record.id);\n      }\n    }\n\n    return record;\n  },\n\n  async replaceKey(\n    cache: AsyncRecordAccessor,\n    operation: RecordOperation,\n    options?: RequestOptions\n  ): Promise<RecordOperationResult> {\n    const op = operation as ReplaceKeyOperation;\n    const currentRecord = await cache.getRecordAsync(op.record);\n    let record: InitializedRecord;\n\n    if (currentRecord) {\n      record = clone(currentRecord);\n    } else {\n      record = cloneRecordIdentity(op.record);\n\n      if (options?.raiseNotFoundExceptions) {\n        throw new RecordNotFoundException(record.type, record.id);\n      }\n    }\n\n    deepSet(record, ['keys', op.key], op.value);\n    await cache.setRecordAsync(record);\n\n    if (cache.keyMap) {\n      cache.keyMap.pushRecord(record);\n    }\n\n    return record;\n  },\n\n  async replaceAttribute(\n    cache: AsyncRecordAccessor,\n    operation: RecordOperation,\n    options?: RequestOptions\n  ): Promise<RecordOperationResult> {\n    const op = operation as ReplaceAttributeOperation;\n    const currentRecord = await cache.getRecordAsync(op.record);\n    let record: InitializedRecord;\n\n    if (currentRecord) {\n      record = clone(currentRecord);\n    } else {\n      record = cloneRecordIdentity(op.record);\n\n      if (options?.raiseNotFoundExceptions) {\n        throw new RecordNotFoundException(record.type, record.id);\n      }\n    }\n\n    deepSet(record, ['attributes', op.attribute], op.value);\n    await cache.setRecordAsync(record);\n\n    return record;\n  },\n\n  async addToRelatedRecords(\n    cache: AsyncRecordAccessor,\n    operation: RecordOperation,\n    options?: RequestOptions\n  ): Promise<RecordOperationResult> {\n    const op = operation as AddToRelatedRecordsOperation;\n    const { relationship, relatedRecord } = op;\n    const currentRecord = await cache.getRecordAsync(op.record);\n    let record: InitializedRecord;\n\n    if (currentRecord) {\n      record = clone(currentRecord);\n    } else {\n      record = cloneRecordIdentity(op.record);\n\n      if (options?.raiseNotFoundExceptions) {\n        throw new RecordNotFoundException(record.type, record.id);\n      }\n    }\n\n    const relatedRecords: RecordIdentity[] =\n      deepGet(record, ['relationships', relationship, 'data']) || [];\n\n    if (!recordsInclude(relatedRecords, relatedRecord)) {\n      relatedRecords.push(relatedRecord);\n\n      deepSet(record, ['relationships', relationship, 'data'], relatedRecords);\n      await cache.setRecordAsync(record);\n    }\n\n    return record;\n  },\n\n  async removeFromRelatedRecords(\n    cache: AsyncRecordAccessor,\n    operation: RecordOperation,\n    options?: RequestOptions\n  ): Promise<RecordOperationResult> {\n    const op = operation as RemoveFromRelatedRecordsOperation;\n    const currentRecord = await cache.getRecordAsync(op.record);\n    const { relationship, relatedRecord } = op;\n    let record: InitializedRecord;\n\n    if (currentRecord) {\n      record = clone(currentRecord);\n      let relatedRecords: RecordIdentity[] = deepGet(record, [\n        'relationships',\n        relationship,\n        'data'\n      ]);\n      if (relatedRecords) {\n        relatedRecords = relatedRecords.filter(\n          (r) => !equalRecordIdentities(r, relatedRecord)\n        );\n\n        if (\n          deepSet(\n            record,\n            ['relationships', relationship, 'data'],\n            relatedRecords\n          )\n        ) {\n          await cache.setRecordAsync(record);\n        }\n      }\n      return record;\n    } else {\n      if (options?.raiseNotFoundExceptions) {\n        throw new RecordNotFoundException(op.record.type, op.record.id);\n      }\n    }\n  },\n\n  async replaceRelatedRecords(\n    cache: AsyncRecordAccessor,\n    operation: RecordOperation,\n    options?: RequestOptions\n  ): Promise<RecordOperationResult> {\n    const op = operation as ReplaceRelatedRecordsOperation;\n    const currentRecord = await cache.getRecordAsync(op.record);\n    const { relationship, relatedRecords } = op;\n    let record: InitializedRecord;\n\n    if (currentRecord) {\n      record = clone(currentRecord);\n    } else {\n      record = cloneRecordIdentity(op.record);\n\n      if (options?.raiseNotFoundExceptions) {\n        throw new RecordNotFoundException(record.type, record.id);\n      }\n    }\n\n    if (\n      deepSet(record, ['relationships', relationship, 'data'], relatedRecords)\n    ) {\n      await cache.setRecordAsync(record);\n    }\n\n    return record;\n  },\n\n  async replaceRelatedRecord(\n    cache: AsyncRecordAccessor,\n    operation: RecordOperation,\n    options?: RequestOptions\n  ): Promise<RecordOperationResult> {\n    const op = operation as ReplaceRelatedRecordOperation;\n    const currentRecord = await cache.getRecordAsync(op.record);\n    const { relationship, relatedRecord } = op;\n    let record: InitializedRecord;\n\n    if (currentRecord) {\n      record = clone(currentRecord);\n    } else {\n      record = cloneRecordIdentity(op.record);\n\n      if (options?.raiseNotFoundExceptions) {\n        throw new RecordNotFoundException(record.type, record.id);\n      }\n    }\n\n    if (\n      deepSet(record, ['relationships', relationship, 'data'], relatedRecord)\n    ) {\n      await cache.setRecordAsync(record);\n    }\n\n    return record;\n  }\n};\n"]}