Skip to content

Commit

Permalink
Cache identifiers
Browse files Browse the repository at this point in the history
Co-authored-by: igorT <[email protected]>
Co-authored-by: pete-the-pete <>
  • Loading branch information
igorT and igorT committed May 13, 2020
1 parent 4f12aa0 commit a612032
Show file tree
Hide file tree
Showing 3 changed files with 67 additions and 13 deletions.
24 changes: 24 additions & 0 deletions packages/-ember-data/tests/integration/identifiers/cache-test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -54,5 +54,29 @@ module('Integration | Identifiers - cache', function(hooks) {
'getOrCreateRecordIdentifier() return identifier'
);
});

test('identifiers are cached by lid and can be looked up by lid', async function(assert) {
const houseHash = {
type: 'house',
id: '1',
attributes: {
name: 'Moomin',
},
};
const cache = identifierCacheFor(store);
const identifier = cache.getOrCreateRecordIdentifier(houseHash);

assert.strictEqual(
identifier,
cache.getOrCreateRecordIdentifier({ lid: identifier.lid }),
'getOrCreateRecordIdentifier() return identifier'
);

assert.strictEqual(
identifier,
cache.getOrCreateRecordIdentifier({ type: 'not house', lid: identifier.lid }),
'getOrCreateRecordIdentifier() return identifier'
);
});
});
});
47 changes: 36 additions & 11 deletions packages/store/addon/-private/identifiers/cache.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,8 @@ import isNonEmptyString from '../utils/is-non-empty-string';
import isStableIdentifier, { markStableIdentifier, unmarkStableIdentifier } from './is-stable-identifier';
import uuidv4 from './utils/uuid-v4';

type Identifier = import('../ts-interfaces/identifier').Identifier;

type CoreStore = import('../system/core-store').default;
type StableRecordIdentifier = import('../ts-interfaces/identifier').StableRecordIdentifier;
type GenerationMethod = import('../ts-interfaces/identifier').GenerationMethod;
Expand Down Expand Up @@ -137,7 +139,10 @@ export class IdentifierCache {
/**
* @internal
*/
private _getRecordIdentifier(resource: ResourceIdentifierObject, shouldGenerate: true): StableRecordIdentifier;
private _getRecordIdentifier(
resource: ResourceIdentifierObject | Identifier,
shouldGenerate: true
): StableRecordIdentifier;
private _getRecordIdentifier(
resource: ResourceIdentifierObject,
shouldGenerate: false
Expand All @@ -157,18 +162,30 @@ export class IdentifierCache {
return resource;
}

let lid = coerceId(resource.lid);
let identifier: StableRecordIdentifier | undefined = lid !== null ? this._cache.lids[lid] : undefined;

if (identifier !== undefined) {
return identifier;
}

let type = normalizeModelName(resource.type);
let id = coerceId(resource.id);

if (shouldGenerate === false) {
if (!type || !id) {
return;
}
}

// `type` must always be present
if (DEBUG) {
if (!isNonEmptyString(resource.type)) {
throw new Error('resource.type needs to be a string');
}
}

let type = normalizeModelName(resource.type);
let keyOptions = getTypeIndex(this._cache.types, type);
let identifier: StableRecordIdentifier | undefined;
let lid = coerceId(resource.lid);
let id = coerceId(resource.id);

// go straight for the stable RecordIdentifier key'd to `lid`
if (lid !== null) {
Expand Down Expand Up @@ -264,7 +281,9 @@ export class IdentifierCache {
`id` + `type` or `lid` will return the same `lid` value)
- this referential stability of the object itself is guaranteed
*/
getOrCreateRecordIdentifier(resource: ResourceIdentifierObject | ExistingResourceObject): StableRecordIdentifier {
getOrCreateRecordIdentifier(
resource: ResourceIdentifierObject | ExistingResourceObject | Identifier
): StableRecordIdentifier {
return this._getRecordIdentifier(resource, true);
}

Expand Down Expand Up @@ -325,7 +344,7 @@ export class IdentifierCache {
let newId = coerceId(data.id);

const keyOptions = getTypeIndex(this._cache.types, identifier.type);
let existingIdentifier = detectMerge(keyOptions, identifier, data, newId, this._cache.lids);
let existingIdentifier = detectMerge(this._cache.types, identifier, data, newId, this._cache.lids);

if (existingIdentifier) {
identifier = this._mergeRecordIdentifiers(keyOptions, identifier, existingIdentifier, data, newId as string);
Expand Down Expand Up @@ -519,23 +538,29 @@ function performRecordIdentifierUpdate(
}

function detectMerge(
keyOptions: KeyOptions,
typesCache: ConfidentDict<KeyOptions>,
identifier: StableRecordIdentifier,
data: ResourceIdentifierObject | ExistingResourceObject,
newId: string | null,
lids: IdentifierMap
): StableRecordIdentifier | false {
const { id, type, lid } = identifier;
if (id !== null && id !== newId && newId !== null) {
const existingIdentifier = keyOptions.id[newId];
let keyOptions = getTypeIndex(typesCache, identifier.type);
let existingIdentifier = keyOptions.id[newId];

return existingIdentifier !== undefined ? existingIdentifier : false;
} else {
let newType = normalizeModelName(data.type);

// If the ids and type are the same but lid is not the same, we should trigger a merge of the identifiers
if (id !== null && id === newId && newType === type && data.lid && data.lid !== lid) {
const existingIdentifier = lids[data.lid];

let existingIdentifier = lids[data.lid];
return existingIdentifier !== undefined ? existingIdentifier : false;
// If the lids are the same, and ids are the same, but types are different we should trigger a merge of the identifiers
} else if (id !== null && id === newId && newType !== type && data.lid && data.lid === lid) {
let keyOptions = getTypeIndex(typesCache, newType);
let existingIdentifier = keyOptions.id[id];
return existingIdentifier !== undefined ? existingIdentifier : false;
}
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -81,8 +81,13 @@ export default class InternalModelFactory {
constructor(public store: CoreStore) {
this.identifierCache = identifierCacheFor(store);
this.identifierCache.__configureMerge((identifier, matchedIdentifier, resourceData) => {
const intendedIdentifier = identifier.id === resourceData.id ? identifier : matchedIdentifier;
const altIdentifier = identifier.id === resourceData.id ? matchedIdentifier : identifier;
let intendedIdentifier = identifier;
if (identifier.id !== matchedIdentifier.id) {
intendedIdentifier = identifier.id === resourceData.id ? identifier : matchedIdentifier;
} else if (identifier.type !== matchedIdentifier.type) {
intendedIdentifier = identifier.type === resourceData.type ? identifier : matchedIdentifier;
}
let altIdentifier = identifier === intendedIdentifier ? matchedIdentifier : identifier;

// check for duplicate InternalModel's
const map = this.modelMapFor(identifier.type);
Expand Down

0 comments on commit a612032

Please sign in to comment.