Skip to content

Commit a48f644

Browse files
authored
fix: keep a backreference for previously merged identifiers (#9183)
* fix: keep a backreference for previously merged identifiers * update tests
1 parent fa270d7 commit a48f644

File tree

5 files changed

+171
-15
lines changed

5 files changed

+171
-15
lines changed

.vscode/settings.json

+38-2
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,43 @@
11
{
2+
"eslint.format.enable": true,
3+
"eslint.workingDirectories": [{ "mode": "auto" }, "packages/*", "tests/*"],
4+
"eslint.options": { "reportUnusedDisableDirectives": "error" },
5+
"editor.codeActionsOnSave": {
6+
"source.fixAll.eslint": "explicit"
7+
},
28
"files.associations": {
39
"turbo.json": "jsonc"
410
},
5-
"eslint.workingDirectories": [{ "mode": "auto" }, "packages/*", "tests/*"],
6-
"editor.formatOnSave": true
11+
"editor.defaultFormatter": "esbenp.prettier-vscode",
12+
"editor.formatOnSave": true,
13+
"[javascript]": {
14+
"editor.defaultFormatter": "esbenp.prettier-vscode",
15+
"editor.formatOnSave": true
16+
},
17+
"[typescript]": {
18+
"editor.defaultFormatter": "esbenp.prettier-vscode",
19+
"editor.formatOnSave": true
20+
},
21+
"[handlebars]": {
22+
"editor.defaultFormatter": "esbenp.prettier-vscode",
23+
"editor.formatOnSave": true
24+
},
25+
"[json]": {
26+
"editor.defaultFormatter": "esbenp.prettier-vscode",
27+
"editor.formatOnSave": true
28+
},
29+
"[glimmer-js]": {
30+
"editor.defaultFormatter": "esbenp.prettier-vscode",
31+
"editor.formatOnSave": true
32+
},
33+
"[glimmer-ts]": {
34+
"editor.defaultFormatter": "esbenp.prettier-vscode",
35+
"editor.formatOnSave": true
36+
},
37+
"[markdown]": {
38+
"editor.detectIndentation": false,
39+
"editor.insertSpaces": false,
40+
"editor.tabSize": 4
41+
},
42+
"typescript.preferences.importModuleSpecifier": "project-relative"
743
}

packages/store/src/-private/caches/identifier-cache.ts

+32-5
Original file line numberDiff line numberDiff line change
@@ -84,6 +84,7 @@ type StableCache = {
8484
resources: IdentifierMap;
8585
documents: Map<string, StableDocumentIdentifier>;
8686
resourcesByType: TypeMap;
87+
polymorphicLidBackMap: Map<string, string[]>;
8788
};
8889

8990
export type KeyInfoMethod = (resource: unknown, known: StableRecordIdentifier | null) => KeyInfo;
@@ -251,6 +252,7 @@ export class IdentifierCache {
251252
resources: new Map<string, StableRecordIdentifier>(),
252253
resourcesByType: Object.create(null) as TypeMap,
253254
documents: new Map<string, StableDocumentIdentifier>(),
255+
polymorphicLidBackMap: new Map<string, string[]>(),
254256
};
255257
}
256258

@@ -560,16 +562,33 @@ export class IdentifierCache {
560562
const kept = this._merge(identifier, existingIdentifier, data);
561563
const abandoned = kept === identifier ? existingIdentifier : identifier;
562564

565+
// get any backreferences before forgetting this identifier, as it will be removed from the cache
566+
// and we will no longer be able to find them
567+
const abandonedBackReferences = this._cache.polymorphicLidBackMap.get(abandoned.lid);
568+
// delete the backreferences for the abandoned identifier so that forgetRecordIdentifier
569+
// does not try to remove them.
570+
if (abandonedBackReferences) this._cache.polymorphicLidBackMap.delete(abandoned.lid);
571+
563572
// cleanup the identifier we no longer need
564573
this.forgetRecordIdentifier(abandoned);
565574

566-
// ensure a secondary cache entry for this id for the identifier we do keep
567-
// keyOptions.id.set(newId, kept);
575+
// ensure a secondary cache entry for the original lid for the abandoned identifier
576+
this._cache.resources.set(abandoned.lid, kept);
577+
578+
// backReferences let us know which other identifiers are pointing at this identifier
579+
// so we can delete them later if we forget this identifier
580+
const keptBackReferences = this._cache.polymorphicLidBackMap.get(kept.lid) ?? [];
581+
keptBackReferences.push(abandoned.lid);
568582

569-
// ensure a secondary cache entry for this id for the abandoned identifier's type we do keep
570-
// let baseKeyOptions = getTypeIndex(this._cache.resourcesByType, existingIdentifier.type);
571-
// baseKeyOptions.id.set(newId, kept);
583+
// update the backreferences from the abandoned identifier to be for the kept identifier
584+
if (abandonedBackReferences) {
585+
abandonedBackReferences.forEach((lid) => {
586+
keptBackReferences.push(lid);
587+
this._cache.resources.set(lid, kept);
588+
});
589+
}
572590

591+
this._cache.polymorphicLidBackMap.set(kept.lid, keptBackReferences);
573592
return kept;
574593
}
575594

@@ -596,6 +615,14 @@ export class IdentifierCache {
596615
this._cache.resources.delete(identifier.lid);
597616
typeSet.lid.delete(identifier.lid);
598617

618+
const backReferences = this._cache.polymorphicLidBackMap.get(identifier.lid);
619+
if (backReferences) {
620+
backReferences.forEach((lid) => {
621+
this._cache.resources.delete(lid);
622+
});
623+
this._cache.polymorphicLidBackMap.delete(identifier.lid);
624+
}
625+
599626
if (DEBUG) {
600627
identifier[DEBUG_STALE_CACHE_OWNER] = identifier[CACHE_OWNER];
601628
}

tests/main/tests/integration/identifiers/configuration-test.ts

+41-3
Original file line numberDiff line numberDiff line change
@@ -382,6 +382,10 @@ module('Integration | Identifiers - configuration', function (hooks) {
382382
if (bucket !== 'record') {
383383
throw new Error('Test cannot generate an lid for a non-record');
384384
}
385+
if (typeof resource === 'object' && resource !== null && 'lid' in resource && typeof resource.lid === 'string') {
386+
generateLidCalls++;
387+
return resource.lid;
388+
}
385389
if (typeof resource !== 'object' || resource === null || !('type' in resource)) {
386390
throw new Error('Test cannot generate an lid for a non-object');
387391
}
@@ -393,12 +397,12 @@ module('Integration | Identifiers - configuration', function (hooks) {
393397
return `${resource.type}:${resource.id}`;
394398
});
395399
let forgetMethodCalls = 0;
396-
// eslint-disable-next-line prefer-const
400+
397401
let expectedIdentifier;
398402

399403
let testMethod = (identifier) => {
400404
forgetMethodCalls++;
401-
assert.strictEqual(expectedIdentifier, identifier, `We forgot the expected identifier ${expectedIdentifier}`);
405+
assert.strictEqual(identifier, expectedIdentifier, `We forgot the expected identifier ${expectedIdentifier}`);
402406
};
403407

404408
setIdentifierForgetMethod((identifier) => {
@@ -434,7 +438,11 @@ module('Integration | Identifiers - configuration', function (hooks) {
434438
const finalUserByIdIdentifier = recordIdentifierFor(userById);
435439

436440
assert.strictEqual(generateLidCalls, 2, 'We generated no new lids when we looked up the originals');
437-
assert.strictEqual(store.identifierCache._cache.resources.size, 1, 'We now have only 1 identifier in the cache');
441+
assert.strictEqual(
442+
store.identifierCache._cache.resources.size,
443+
2,
444+
'We keep a back reference identifier in the cache'
445+
);
438446
assert.strictEqual(forgetMethodCalls, 1, 'We abandoned an identifier');
439447

440448
assert.notStrictEqual(
@@ -453,6 +461,36 @@ module('Integration | Identifiers - configuration', function (hooks) {
453461
'We are using the identifier by id for the result of findRecord with username'
454462
);
455463

464+
const recordA = store.peekRecord({ lid: finalUserByUsernameIdentifier.lid });
465+
const recordB = store.peekRecord({ lid: finalUserByIdIdentifier.lid });
466+
const recordC = store.peekRecord({ lid: originalUserByUsernameIdentifier.lid });
467+
const recordD = store.peekRecord('user', '@runspired');
468+
const recordE = store.peekRecord('user', '1');
469+
470+
assert.strictEqual(recordA, recordB, 'We have a single record for both identifiers');
471+
assert.strictEqual(recordA, recordC, 'We have a single record for both identifiers');
472+
assert.strictEqual(recordA, recordD, 'We have a single record for both identifiers');
473+
assert.strictEqual(recordA, recordE, 'We have a single record for both identifiers');
474+
475+
const regeneratedIdentifier = store.identifierCache.getOrCreateRecordIdentifier({
476+
lid: finalUserByUsernameIdentifier.lid,
477+
});
478+
const regeneratedIdentifier2 = store.identifierCache.getOrCreateRecordIdentifier({
479+
id: '@runspired',
480+
type: 'user',
481+
});
482+
assert.strictEqual(regeneratedIdentifier, finalUserByUsernameIdentifier, 'We regenerate the same identifier');
483+
assert.strictEqual(regeneratedIdentifier2, finalUserByUsernameIdentifier, 'We regenerate the same identifier');
484+
485+
expectedIdentifier = finalUserByIdIdentifier;
486+
store.unloadRecord(recordA);
487+
await settled();
488+
assert.strictEqual(
489+
store.identifierCache._cache.resources.size,
490+
0,
491+
'We have no identifiers or backreferences in the cache'
492+
);
493+
456494
// end test before store teardown
457495
testMethod = () => {};
458496
});

tests/main/tests/integration/identifiers/scenarios-test.ts

+7-5
Original file line numberDiff line numberDiff line change
@@ -468,12 +468,14 @@ module('Integration | Identifiers - scenarios', function (hooks) {
468468

469469
// ensure we truly are in a good state internally
470470
const lidCache = store.identifierCache._cache.resources;
471-
const lids = [...lidCache.values()];
472-
assert.strictEqual(
473-
lidCache.size,
474-
1,
475-
`We only have the lid '${identifierByUsername.lid}' in ['${lids.join("', '")}']`
471+
assert.strictEqual(lidCache.size, 2, `We should have both lids in the cache still since one is a backreference`);
472+
assert.deepEqual(
473+
[...lidCache.keys()],
474+
['remote:user:1:9001', 'remote:user:@runspired:9000'],
475+
'We have the expected keys'
476476
);
477+
const lids = [...lidCache.values()];
478+
assert.arrayStrictEquals(lids, [identifierById, identifierById], 'We have the expected values');
477479

478480
// ensure we still use secondary caching for @runspired post-merging of the identifiers
479481
const recordByUsernameAgain = (await store.findRecord('user', '@runspired')) as User;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,53 @@
1+
import { settled } from '@ember/test-helpers';
2+
3+
import { module, test } from 'qunit';
4+
5+
import type Store from 'ember-data/store';
6+
import { setupTest } from 'ember-qunit';
7+
8+
import Model, { attr } from '@ember-data/model';
9+
import { recordIdentifierFor } from '@ember-data/store';
10+
11+
class Person extends Model {
12+
@attr declare name: string;
13+
}
14+
class Employee extends Person {}
15+
16+
module('integration/records/polymorphic-find-record - Polymorphic findRecord', function (hooks) {
17+
setupTest(hooks);
18+
19+
test('when findRecord with abstract type returns concrete type', async function (assert) {
20+
this.owner.register('model:person', Person);
21+
this.owner.register('model:employee', Employee);
22+
23+
const store = this.owner.lookup('service:store') as Store;
24+
const adapter = store.adapterFor('application');
25+
26+
adapter.findRecord = () => {
27+
return Promise.resolve({
28+
data: {
29+
id: '1',
30+
type: 'employee',
31+
attributes: {
32+
name: 'Rey Skybarker',
33+
},
34+
},
35+
});
36+
};
37+
38+
const person = (await store.findRecord('person', '1')) as Employee;
39+
assert.ok(person instanceof Employee, 'record is an instance of Employee');
40+
assert.strictEqual(person.name, 'Rey Skybarker', 'name is correct');
41+
assert.strictEqual(recordIdentifierFor(person).type, 'employee', 'identifier has the concrete type');
42+
43+
const employee = store.peekRecord('employee', '1');
44+
const person2 = store.peekRecord('person', '1');
45+
assert.strictEqual(employee, person, 'peekRecord returns the same instance for concrete type');
46+
assert.strictEqual(person2, person, 'peekRecord returns the same instance for abstract type');
47+
assert.strictEqual(store.identifierCache._cache.resources.size, 2, 'identifier cache contains backreferences');
48+
49+
person.unloadRecord();
50+
await settled();
51+
assert.strictEqual(store.identifierCache._cache.resources.size, 0, 'identifier cache is empty');
52+
});
53+
});

0 commit comments

Comments
 (0)