@orbit/record-cache
Version:
Orbit base classes used to access and maintain a set of records.
184 lines • 29.4 kB
JavaScript
import { deepGet } from '@orbit/utils';
import { cloneRecordIdentity, equalRecordIdentities, uniqueRecordIdentities } from '@orbit/records';
export function recordAdded(schema, record) {
const ops = [];
if (record.relationships) {
const recordIdentity = cloneRecordIdentity(record);
schema.eachRelationship(record.type, (relationship, relationshipDef) => {
const relationshipData = deepGet(record, [
'relationships',
relationship,
'data'
]);
if (relationshipData) {
const relatedRecords = recordArrayFromData(relationshipData);
Array.prototype.push.apply(ops, addRelatedRecordsOps(schema, recordIdentity, relationshipDef, relatedRecords));
}
});
}
return ops;
}
export function relatedRecordAdded(schema, record, relationship, relatedRecord) {
const ops = [];
if (relatedRecord) {
const { type } = record;
const relationshipDef = schema.getRelationship(type, relationship);
const inverseRelationship = relationshipDef.inverse;
if (inverseRelationship) {
ops.push(addRelationshipOp(schema, relatedRecord, inverseRelationship, record));
}
}
return ops;
}
export function relatedRecordRemoved(schema, record, relationship, relatedRecord, currentRelatedRecord) {
const ops = [];
if (currentRelatedRecord) {
const { type } = record;
const relationshipDef = schema.getRelationship(type, relationship);
const inverseRelationship = relationshipDef.inverse;
if (inverseRelationship) {
ops.push(removeRelationshipOp(schema, relatedRecord, inverseRelationship, record));
}
}
return ops;
}
export function relatedRecordReplaced(schema, record, relationship, relatedRecord, currentRelatedRecord) {
const ops = [];
if (!equalRecordIdentities(relatedRecord, currentRelatedRecord)) {
const { type } = record;
const relationshipDef = schema.getRelationship(type, relationship);
const inverseRelationship = relationshipDef.inverse;
if (inverseRelationship) {
if (currentRelatedRecord) {
ops.push(removeRelationshipOp(schema, currentRelatedRecord, inverseRelationship, record));
}
if (relatedRecord) {
ops.push(addRelationshipOp(schema, relatedRecord, inverseRelationship, record));
}
}
}
return ops;
}
export function relatedRecordsReplaced(schema, record, relationship, relatedRecords, currentRelatedRecords) {
const ops = [];
const { type } = record;
const relationshipDef = schema.getRelationship(type, relationship);
let addedRecords;
if (currentRelatedRecords && currentRelatedRecords.length > 0) {
let removedRecords = uniqueRecordIdentities(currentRelatedRecords, relatedRecords);
Array.prototype.push.apply(ops, removeRelatedRecordsOps(schema, record, relationshipDef, removedRecords));
addedRecords = uniqueRecordIdentities(relatedRecords, currentRelatedRecords);
}
else {
addedRecords = relatedRecords;
}
Array.prototype.push.apply(ops, addRelatedRecordsOps(schema, record, relationshipDef, addedRecords));
return ops;
}
export function recordRemoved(schema, record) {
const ops = [];
if (record && record.relationships) {
const recordIdentity = cloneRecordIdentity(record);
schema.eachRelationship(record.type, (relationship, relationshipDef) => {
const relationshipData = deepGet(record, [
'relationships',
relationship,
'data'
]);
if (relationshipData) {
const relatedRecords = recordArrayFromData(relationshipData);
Array.prototype.push.apply(ops, removeRelatedRecordsOps(schema, recordIdentity, relationshipDef, relatedRecords));
}
});
}
return ops;
}
export function recordUpdated(schema, record, currentRecord) {
const ops = [];
if (record.relationships) {
const recordIdentity = cloneRecordIdentity(record);
schema.eachRelationship(record.type, (relationship, relationshipDef) => {
var _a;
const relationshipData = deepGet(record, [
'relationships',
relationship,
'data'
]);
const currentRelationshipData = currentRecord &&
deepGet(currentRecord, ['relationships', relationship, 'data']);
if (relationshipData !== undefined) {
// TODO - remove deprecated `type` check
if (((_a = relationshipDef.kind) !== null && _a !== void 0 ? _a : relationshipDef.type) === 'hasMany') {
Array.prototype.push.apply(ops, relatedRecordsReplaced(schema, recordIdentity, relationship, relationshipData || [], currentRelationshipData || []));
}
else {
Array.prototype.push.apply(ops, relatedRecordReplaced(schema, recordIdentity, relationship, relationshipData || null, currentRelationshipData || null));
}
}
});
}
return ops;
}
function addRelatedRecordsOps(schema, record, relationshipDef, relatedRecords) {
if (relatedRecords.length > 0 && relationshipDef.inverse) {
const inverse = relationshipDef.inverse;
return relatedRecords.map((relatedRecord) => addRelationshipOp(schema, relatedRecord, inverse, record));
}
return [];
}
export function removeRelatedRecordsOps(schema, record, relationshipDef, relatedRecords) {
if (relatedRecords.length > 0) {
if (relationshipDef.dependent === 'remove') {
return removeDependentRecords(relatedRecords);
}
else if (relationshipDef.inverse) {
const inverse = relationshipDef.inverse;
return relatedRecords.map((relatedRecord) => removeRelationshipOp(schema, relatedRecord, inverse, record));
}
}
return [];
}
export function addRelationshipOp(schema, record, relationship, relatedRecord) {
var _a;
const { type } = record;
const relationshipDef = schema.getRelationship(type, relationship);
// TODO - remove deprecated `type` check
const isHasMany = ((_a = relationshipDef.kind) !== null && _a !== void 0 ? _a : relationshipDef.type) === 'hasMany';
return {
op: isHasMany ? 'addToRelatedRecords' : 'replaceRelatedRecord',
record,
relationship,
relatedRecord
};
}
export function removeRelationshipOp(schema, record, relationship, relatedRecord) {
var _a;
const { type } = record;
const relationshipDef = schema.getRelationship(type, relationship);
// TODO - remove deprecated `type` check
const isHasMany = ((_a = relationshipDef.kind) !== null && _a !== void 0 ? _a : relationshipDef.type) === 'hasMany';
return {
op: isHasMany ? 'removeFromRelatedRecords' : 'replaceRelatedRecord',
record,
relationship,
relatedRecord: isHasMany ? relatedRecord : null
};
}
export function recordArrayFromData(data) {
if (Array.isArray(data)) {
return data;
}
else if (data) {
return [data];
}
else {
return [];
}
}
function removeDependentRecords(relatedRecords) {
return relatedRecords.map((record) => ({
op: 'removeRecord',
record
}));
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"schema-consistency-utils.js","sourceRoot":"","sources":["../../../../src/operation-processors/utils/schema-consistency-utils.ts"],"names":[],"mappings":"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","sourcesContent":["import { deepGet } from '@orbit/utils';\nimport {\n  cloneRecordIdentity,\n  equalRecordIdentities,\n  uniqueRecordIdentities,\n  InitializedRecord,\n  RecordIdentity,\n  RecordOperation,\n  RelationshipDefinition,\n  RecordSchema\n} from '@orbit/records';\n\nexport function recordAdded(\n  schema: RecordSchema,\n  record: InitializedRecord\n): RecordOperation[] {\n  const ops: RecordOperation[] = [];\n\n  if (record.relationships) {\n    const recordIdentity = cloneRecordIdentity(record);\n\n    schema.eachRelationship(record.type, (relationship, relationshipDef) => {\n      const relationshipData = deepGet(record, [\n        'relationships',\n        relationship,\n        'data'\n      ]) as RecordIdentity | RecordIdentity[] | null | undefined;\n\n      if (relationshipData) {\n        const relatedRecords = recordArrayFromData(\n          relationshipData\n        ) as RecordIdentity[];\n\n        Array.prototype.push.apply(\n          ops,\n          addRelatedRecordsOps(\n            schema,\n            recordIdentity,\n            relationshipDef,\n            relatedRecords\n          )\n        );\n      }\n    });\n  }\n\n  return ops;\n}\n\nexport function relatedRecordAdded(\n  schema: RecordSchema,\n  record: RecordIdentity,\n  relationship: string,\n  relatedRecord: RecordIdentity\n): RecordOperation[] {\n  const ops: RecordOperation[] = [];\n\n  if (relatedRecord) {\n    const { type } = record;\n    const relationshipDef = schema.getRelationship(type, relationship);\n    const inverseRelationship = relationshipDef.inverse;\n\n    if (inverseRelationship) {\n      ops.push(\n        addRelationshipOp(schema, relatedRecord, inverseRelationship, record)\n      );\n    }\n  }\n\n  return ops;\n}\n\nexport function relatedRecordRemoved(\n  schema: RecordSchema,\n  record: RecordIdentity,\n  relationship: string,\n  relatedRecord: RecordIdentity,\n  currentRelatedRecord?: RecordIdentity | null\n): RecordOperation[] {\n  const ops: RecordOperation[] = [];\n\n  if (currentRelatedRecord) {\n    const { type } = record;\n    const relationshipDef = schema.getRelationship(type, relationship);\n    const inverseRelationship = relationshipDef.inverse;\n\n    if (inverseRelationship) {\n      ops.push(\n        removeRelationshipOp(schema, relatedRecord, inverseRelationship, record)\n      );\n    }\n  }\n\n  return ops;\n}\n\nexport function relatedRecordReplaced(\n  schema: RecordSchema,\n  record: RecordIdentity,\n  relationship: string,\n  relatedRecord: RecordIdentity | null,\n  currentRelatedRecord?: RecordIdentity | null\n): RecordOperation[] {\n  const ops: RecordOperation[] = [];\n\n  if (\n    !equalRecordIdentities(\n      relatedRecord as RecordIdentity,\n      currentRelatedRecord as RecordIdentity\n    )\n  ) {\n    const { type } = record;\n    const relationshipDef = schema.getRelationship(type, relationship);\n    const inverseRelationship = relationshipDef.inverse;\n\n    if (inverseRelationship) {\n      if (currentRelatedRecord) {\n        ops.push(\n          removeRelationshipOp(\n            schema,\n            currentRelatedRecord,\n            inverseRelationship,\n            record\n          )\n        );\n      }\n\n      if (relatedRecord) {\n        ops.push(\n          addRelationshipOp(schema, relatedRecord, inverseRelationship, record)\n        );\n      }\n    }\n  }\n\n  return ops;\n}\n\nexport function relatedRecordsReplaced(\n  schema: RecordSchema,\n  record: RecordIdentity,\n  relationship: string,\n  relatedRecords: RecordIdentity[],\n  currentRelatedRecords?: RecordIdentity[]\n): RecordOperation[] {\n  const ops: RecordOperation[] = [];\n  const { type } = record;\n  const relationshipDef = schema.getRelationship(type, relationship);\n\n  let addedRecords: RecordIdentity[];\n\n  if (currentRelatedRecords && currentRelatedRecords.length > 0) {\n    let removedRecords = uniqueRecordIdentities(\n      currentRelatedRecords,\n      relatedRecords\n    );\n    Array.prototype.push.apply(\n      ops,\n      removeRelatedRecordsOps(schema, record, relationshipDef, removedRecords)\n    );\n\n    addedRecords = uniqueRecordIdentities(\n      relatedRecords,\n      currentRelatedRecords\n    );\n  } else {\n    addedRecords = relatedRecords;\n  }\n\n  Array.prototype.push.apply(\n    ops,\n    addRelatedRecordsOps(schema, record, relationshipDef, addedRecords)\n  );\n\n  return ops;\n}\n\nexport function recordRemoved(\n  schema: RecordSchema,\n  record?: InitializedRecord\n): RecordOperation[] {\n  const ops: RecordOperation[] = [];\n\n  if (record && record.relationships) {\n    const recordIdentity = cloneRecordIdentity(record);\n\n    schema.eachRelationship(record.type, (relationship, relationshipDef) => {\n      const relationshipData = deepGet(record, [\n        'relationships',\n        relationship,\n        'data'\n      ]) as RecordIdentity | RecordIdentity[] | null | undefined;\n\n      if (relationshipData) {\n        const relatedRecords = recordArrayFromData(\n          relationshipData\n        ) as RecordIdentity[];\n\n        Array.prototype.push.apply(\n          ops,\n          removeRelatedRecordsOps(\n            schema,\n            recordIdentity,\n            relationshipDef,\n            relatedRecords\n          )\n        );\n      }\n    });\n  }\n\n  return ops;\n}\n\nexport function recordUpdated(\n  schema: RecordSchema,\n  record: InitializedRecord,\n  currentRecord?: InitializedRecord\n): RecordOperation[] {\n  const ops: RecordOperation[] = [];\n\n  if (record.relationships) {\n    const recordIdentity = cloneRecordIdentity(record);\n\n    schema.eachRelationship(record.type, (relationship, relationshipDef) => {\n      const relationshipData = deepGet(record, [\n        'relationships',\n        relationship,\n        'data'\n      ]) as RecordIdentity | RecordIdentity[] | null | undefined;\n\n      const currentRelationshipData =\n        currentRecord &&\n        (deepGet(currentRecord, ['relationships', relationship, 'data']) as\n          | RecordIdentity\n          | RecordIdentity[]\n          | null\n          | undefined);\n\n      if (relationshipData !== undefined) {\n        // TODO - remove deprecated `type` check\n        if (\n          (relationshipDef.kind ?? (relationshipDef as any).type) === 'hasMany'\n        ) {\n          Array.prototype.push.apply(\n            ops,\n            relatedRecordsReplaced(\n              schema,\n              recordIdentity,\n              relationship,\n              (relationshipData as RecordIdentity[]) || [],\n              (currentRelationshipData as RecordIdentity[]) || []\n            )\n          );\n        } else {\n          Array.prototype.push.apply(\n            ops,\n            relatedRecordReplaced(\n              schema,\n              recordIdentity,\n              relationship,\n              (relationshipData as RecordIdentity) || null,\n              (currentRelationshipData as RecordIdentity) || null\n            )\n          );\n        }\n      }\n    });\n  }\n\n  return ops;\n}\n\nfunction addRelatedRecordsOps(\n  schema: RecordSchema,\n  record: RecordIdentity,\n  relationshipDef: RelationshipDefinition,\n  relatedRecords: RecordIdentity[]\n): RecordOperation[] {\n  if (relatedRecords.length > 0 && relationshipDef.inverse) {\n    const inverse = relationshipDef.inverse;\n    return relatedRecords.map((relatedRecord) =>\n      addRelationshipOp(schema, relatedRecord, inverse, record)\n    );\n  }\n  return [];\n}\n\nexport function removeRelatedRecordsOps(\n  schema: RecordSchema,\n  record: RecordIdentity,\n  relationshipDef: RelationshipDefinition,\n  relatedRecords: RecordIdentity[]\n): RecordOperation[] {\n  if (relatedRecords.length > 0) {\n    if (relationshipDef.dependent === 'remove') {\n      return removeDependentRecords(relatedRecords);\n    } else if (relationshipDef.inverse) {\n      const inverse = relationshipDef.inverse;\n      return relatedRecords.map((relatedRecord) =>\n        removeRelationshipOp(schema, relatedRecord, inverse, record)\n      );\n    }\n  }\n  return [];\n}\n\nexport function addRelationshipOp(\n  schema: RecordSchema,\n  record: RecordIdentity,\n  relationship: string,\n  relatedRecord: RecordIdentity\n): RecordOperation {\n  const { type } = record;\n  const relationshipDef = schema.getRelationship(type, relationship);\n  // TODO - remove deprecated `type` check\n  const isHasMany =\n    (relationshipDef.kind ?? (relationshipDef as any).type) === 'hasMany';\n\n  return {\n    op: isHasMany ? 'addToRelatedRecords' : 'replaceRelatedRecord',\n    record,\n    relationship,\n    relatedRecord\n  } as RecordOperation;\n}\n\nexport function removeRelationshipOp(\n  schema: RecordSchema,\n  record: RecordIdentity,\n  relationship: string,\n  relatedRecord: RecordIdentity\n): RecordOperation {\n  const { type } = record;\n  const relationshipDef = schema.getRelationship(type, relationship);\n  // TODO - remove deprecated `type` check\n  const isHasMany =\n    (relationshipDef.kind ?? (relationshipDef as any).type) === 'hasMany';\n\n  return {\n    op: isHasMany ? 'removeFromRelatedRecords' : 'replaceRelatedRecord',\n    record,\n    relationship,\n    relatedRecord: isHasMany ? relatedRecord : null\n  } as RecordOperation;\n}\n\nexport function recordArrayFromData(\n  data: RecordIdentity | RecordIdentity[] | null\n): RecordIdentity[] {\n  if (Array.isArray(data)) {\n    return data;\n  } else if (data) {\n    return [data];\n  } else {\n    return [];\n  }\n}\n\nfunction removeDependentRecords(\n  relatedRecords: RecordIdentity[]\n): RecordOperation[] {\n  return relatedRecords.map(\n    (record) =>\n      ({\n        op: 'removeRecord',\n        record\n      } as RecordOperation)\n  );\n}\n"]}