Skip to content

Commit 6b416aa

Browse files
gitKrystanMehulKChaudhari
authored andcommitted
fix: Don't notify changes for attributes not registered with the schema (emberjs#9698)
1 parent 340078b commit 6b416aa

File tree

9 files changed

+87
-31
lines changed

9 files changed

+87
-31
lines changed

packages/json-api/src/-private/cache.ts

+14-8
Original file line numberDiff line numberDiff line change
@@ -656,10 +656,12 @@ export default class JSONAPICache implements Cache {
656656
this._capabilities.notifyChange(identifier, 'state', null);
657657
}
658658

659+
const fields = this._capabilities.schema.fields(identifier);
660+
659661
// if no cache entry existed, no record exists / property has been accessed
660662
// and thus we do not need to notify changes to any properties.
661663
if (calculateChanges && existed && data.attributes) {
662-
changedKeys = calculateChangedKeys(cached, data.attributes);
664+
changedKeys = calculateChangedKeys(cached, data.attributes, fields);
663665
}
664666

665667
cached.remoteAttrs = Object.assign(
@@ -682,7 +684,7 @@ export default class JSONAPICache implements Cache {
682684
}
683685

684686
if (data.relationships) {
685-
setupRelationships(this.__graph, this._capabilities, identifier, data);
687+
setupRelationships(this.__graph, fields, identifier, data);
686688
}
687689

688690
if (changedKeys?.size) {
@@ -1007,6 +1009,7 @@ export default class JSONAPICache implements Cache {
10071009
}
10081010
}
10091011

1012+
const fields = this._capabilities.schema.fields(identifier);
10101013
cached.isNew = false;
10111014
let newCanonicalAttributes: ExistingResourceObject['attributes'];
10121015
if (data) {
@@ -1027,7 +1030,6 @@ export default class JSONAPICache implements Cache {
10271030
if (!DEPRECATE_RELATIONSHIP_REMOTE_UPDATE_CLEARING_LOCAL_STATE) {
10281031
// assert against bad API behavior where a belongsTo relationship
10291032
// is saved but the return payload indicates a different final state.
1030-
const fields = this._capabilities.schema.fields(identifier);
10311033
fields.forEach((field, name) => {
10321034
if (field.kind === 'belongsTo') {
10331035
const relationshipData = data.relationships![name]?.data;
@@ -1053,11 +1055,11 @@ export default class JSONAPICache implements Cache {
10531055
cached.inflightRelationships = null;
10541056
}
10551057
}
1056-
setupRelationships(this.__graph, this._capabilities, identifier, data);
1058+
setupRelationships(this.__graph, fields, identifier, data);
10571059
}
10581060
newCanonicalAttributes = data.attributes;
10591061
}
1060-
const changedKeys = newCanonicalAttributes && calculateChangedKeys(cached, newCanonicalAttributes);
1062+
const changedKeys = newCanonicalAttributes && calculateChangedKeys(cached, newCanonicalAttributes, fields);
10611063

10621064
cached.remoteAttrs = Object.assign(
10631065
cached.remoteAttrs || (Object.create(null) as Record<string, unknown>),
@@ -1872,7 +1874,8 @@ function notifyAttributes(
18721874
*/
18731875
function calculateChangedKeys(
18741876
cached: CachedResource,
1875-
updates: Exclude<ExistingResourceObject['attributes'], undefined>
1877+
updates: Exclude<ExistingResourceObject['attributes'], undefined>,
1878+
fields: ReturnType<Store['schema']['fields']>
18761879
): Set<string> {
18771880
const changedKeys = new Set<string>();
18781881
const keys = Object.keys(updates);
@@ -1887,6 +1890,10 @@ function calculateChangedKeys(
18871890

18881891
for (let i = 0; i < length; i++) {
18891892
const key = keys[i];
1893+
if (!fields.has(key)) {
1894+
continue;
1895+
}
1896+
18901897
const value = updates[key];
18911898

18921899
// A value in localAttrs means the user has a local change to
@@ -1962,15 +1969,14 @@ function _isLoading(
19621969

19631970
function setupRelationships(
19641971
graph: Graph,
1965-
capabilities: CacheCapabilitiesManager,
1972+
fields: ReturnType<Store['schema']['fields']>,
19661973
identifier: StableRecordIdentifier,
19671974
data: ExistingResourceObject
19681975
) {
19691976
// TODO @runspired iterating by definitions instead of by payload keys
19701977
// allows relationship payloads to be ignored silently if no relationship
19711978
// definition exists. Ensure there's a test for this and then consider
19721979
// moving this to an assertion. This check should possibly live in the graph.
1973-
const fields = capabilities.schema.fields(identifier);
19741980
for (const [name, field] of fields) {
19751981
if (!isRelationship(field)) continue;
19761982

packages/model/src/-private/model.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -1738,7 +1738,7 @@ class Model extends EmberObject implements MinimalLegacyRecord {
17381738
this.eachComputedProperty((name, meta) => {
17391739
if (isAttributeSchema(meta)) {
17401740
assert(
1741-
"You may not set `id` as an attribute on your model. Please remove any lines that look like: `id: attr('<type>')` from " +
1741+
"You may not set 'id' as an attribute on your model. Please remove any lines that look like: `id: attr('<type>')` from " +
17421742
this.toString(),
17431743
name !== 'id'
17441744
);

tests/main/eslint.config.mjs

+1
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,7 @@ const AllowedImports = [
2323
'@ember/test-helpers',
2424
'@ember/test-waiters',
2525
'@glimmer/component',
26+
'@glimmer/tracking',
2627
'qunit',
2728
];
2829

tests/main/tests/acceptance/relationships/tracking-record-state-test.js

-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,6 @@ import { action } from '@ember/object';
33
import { inject } from '@ember/service';
44
import { click, findAll, render } from '@ember/test-helpers';
55
import Component from '@glimmer/component';
6-
// eslint-disable-next-line no-restricted-imports
76
import { tracked } from '@glimmer/tracking';
87

98
import { module, test } from 'qunit';

tests/main/tests/acceptance/tracking-create-record-test.js

-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,6 @@ import { setComponentTemplate } from '@ember/component';
22
import { inject as service } from '@ember/service';
33
import { render, settled } from '@ember/test-helpers';
44
import Component from '@glimmer/component';
5-
// eslint-disable-next-line no-restricted-imports
65
import { tracked } from '@glimmer/tracking';
76

87
import { module, test } from 'qunit';

tests/main/tests/helpers/create-tracking-context.gjs

-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,5 @@
11
import { render, settled } from '@ember/test-helpers';
22
import Component from '@glimmer/component';
3-
// eslint-disable-next-line no-restricted-imports
43
import { tracked } from '@glimmer/tracking';
54

65
// eslint-disable-next-line @typescript-eslint/require-await
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,70 @@
1+
import { settled } from '@ember/test-helpers';
2+
import { tracked } from '@glimmer/tracking';
3+
4+
import { module, test } from 'qunit';
5+
6+
import { setupRenderingTest } from 'ember-qunit';
7+
8+
import Model, { attr } from '@ember-data/model';
9+
import type Store from '@ember-data/store';
10+
import { Type } from '@warp-drive/core-types/symbols';
11+
12+
import { reactiveContext } from '../../helpers/reactive-context';
13+
14+
class User extends Model {
15+
@attr declare name: string;
16+
17+
// Not an attr, so shouldn't notify if a similarly named attribute changes.
18+
@tracked brotherName: string | undefined;
19+
20+
declare [Type]: 'user';
21+
}
22+
23+
module('@ember-data/json-api | Cache (2)', function (hooks) {
24+
setupRenderingTest(hooks);
25+
26+
hooks.beforeEach(function () {
27+
this.owner.register('model:user', User);
28+
});
29+
30+
test('Does not notify for attributes not in schema', async function (assert) {
31+
const store = this.owner.lookup('service:store') as Store;
32+
const user: User = store.push({
33+
data: {
34+
id: '1',
35+
type: 'user',
36+
attributes: {
37+
name: 'Aino',
38+
},
39+
},
40+
});
41+
42+
const rc = await reactiveContext.call(this, user, {
43+
type: 'user',
44+
identity: null,
45+
fields: [
46+
{ name: 'name', kind: 'field' },
47+
{ name: 'brotherName', kind: 'field' },
48+
],
49+
});
50+
51+
const { counters } = rc;
52+
53+
assert.deepEqual(counters, { name: 1, brotherName: 1 }, 'Test setup: counters initialized');
54+
55+
store.push({
56+
data: {
57+
id: '1',
58+
type: 'user',
59+
attributes: {
60+
name: 'Aino',
61+
brotherName: 'Ellis',
62+
},
63+
},
64+
});
65+
66+
await settled();
67+
68+
assert.deepEqual(counters, { name: 1, brotherName: 1 }, 'brotherName did not notify');
69+
});
70+
});

tests/main/tests/unit/model-test.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -210,7 +210,7 @@ module('unit/model - Model', function (hooks) {
210210
assert.expectAssertion(() => {
211211
const ModelClass = store.modelFor('test-model');
212212
get(ModelClass, 'attributes');
213-
}, /You may not set `id` as an attribute on your model/);
213+
}, /You may not set 'id' as an attribute on your model/);
214214

215215
assert.expectAssertion(() => {
216216
store.push({

tests/main/tests/unit/store/push-test.js

-18
Original file line numberDiff line numberDiff line change
@@ -527,24 +527,6 @@ module('unit/store/push - Store#push', function (hooks) {
527527
assert.notOk(store._instanceCache.peek(pushResult, { bucket: 'record' }), 'record is not materialized');
528528
});
529529

530-
test('_push does not require a modelName to resolve to a modelClass', function (assert) {
531-
const store = this.owner.lookup('service:store');
532-
const originalCall = store.modelFor;
533-
store.modelFor = function () {
534-
assert.notOk('modelFor was triggered as a result of a call to store._push');
535-
};
536-
537-
store._push({
538-
data: {
539-
id: '1',
540-
type: 'person',
541-
},
542-
});
543-
544-
store.modelFor = originalCall;
545-
assert.ok('We made it');
546-
});
547-
548530
test('_push returns an array of identifiers if an array is pushed', function (assert) {
549531
const store = this.owner.lookup('service:store');
550532
const pushResult = store._push({

0 commit comments

Comments
 (0)