Skip to content

Commit

Permalink
[BUGFIX] use internalModel promise if already loading (#5562)
Browse files Browse the repository at this point in the history
* use internalModel promise if already loading

* add tests

* prettier to the prettier gods
  • Loading branch information
runspired authored Aug 9, 2018
1 parent 9ebd2fe commit 5dad50b
Show file tree
Hide file tree
Showing 3 changed files with 144 additions and 36 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -288,7 +288,12 @@ export default class BelongsToRelationship extends Relationship {
}

function proxyRecord(internalModel) {
return resolve(internalModel).then(resolvedInternalModel => {
let promise = internalModel;
if (internalModel && internalModel.isLoading()) {
promise = internalModel._promiseProxy;
}

return resolve(promise).then(resolvedInternalModel => {
return resolvedInternalModel ? resolvedInternalModel.getRecord() : null;
});
}
Expand Down
14 changes: 8 additions & 6 deletions addon/-record-data-private/system/store.js
Original file line number Diff line number Diff line change
Expand Up @@ -1386,6 +1386,7 @@ Store = Service.extend({
return RSVP.resolve(null);
}

let internalModel = resource.data ? this._internalModelForResource(resource.data) : null;
let {
relationshipIsStale,
allInverseRecordsAreLoaded,
Expand All @@ -1401,6 +1402,13 @@ Store = Service.extend({
relationshipIsStale ||
(!allInverseRecordsAreLoaded && !relationshipIsEmpty));

// short circuit if we are already loading
if (internalModel && internalModel.isLoading()) {
return internalModel._promiseProxy.then(() => {
return internalModel.getRecord();
});
}

// fetch via link
if (shouldFindViaLink) {
return this._fetchBelongsToLinkFromResource(resource, parentInternalModel, relationshipMeta);
Expand All @@ -1421,23 +1429,17 @@ Store = Service.extend({
return RSVP.resolve(null);
}

let internalModel = this._internalModelForResource(resource.data);

return this._findByInternalModel(internalModel);
}

let resourceIsLocal = !localDataIsEmpty && resource.data.id === null;

if (resourceIsLocal) {
let internalModel = this._internalModelForResource(resource.data);

return RSVP.resolve(internalModel.getRecord());
}

// fetch by data
if (!localDataIsEmpty) {
let internalModel = this._internalModelForResource(resource.data);

return this._fetchRecord(internalModel).then(() => {
return internalModel.getRecord();
});
Expand Down
159 changes: 130 additions & 29 deletions tests/integration/relationships/belongs-to-test.js
Original file line number Diff line number Diff line change
Expand Up @@ -2,38 +2,139 @@ import { get } from '@ember/object';
import { run } from '@ember/runloop';
import RSVP, { resolve } from 'rsvp';
import setupStore from 'dummy/tests/helpers/store';

import { module, test } from 'qunit';
import JSONAPIAdapter from 'ember-data/adapters/json-api';
import JSONAPISerializer from 'ember-data/serializers/json-api';
import { setupTest } from 'ember-qunit';
import Store from 'ember-data/store';
import Model from 'ember-data/model';
import { attr, belongsTo } from '@ember-decorators/data';
import testInDebug, { testRecordData } from 'dummy/tests/helpers/test-in-debug';
import {
setup as setupModelFactoryInjections,
reset as resetModelFactoryInjection,
} from 'dummy/tests/helpers/model-factory-injection';
import { module, test } from 'qunit';

import DS from 'ember-data';
import { ModelData } from 'ember-data/-private';

const { attr, hasMany, belongsTo } = DS;
const { attr: DSattr, hasMany: DShasMany, belongsTo: DSbelongsTo } = DS;
const { hash } = RSVP;

let env, store, User, Message, Post, Comment, Book, Book1, Chapter, Author, NewMessage, Section;

module('integration/relationship/belongs-to BelongsTo Relationships (new-style)', function(hooks) {
let store;
setupTest(hooks);

class Person extends Model {
@belongsTo('pet', { inverse: 'bestHuman', async: true })
bestDog;
@attr
name;
}

class Pet extends Model {
@belongsTo('person', { inverse: 'bestDog', async: false })
bestHuman;
@attr
name;
}

hooks.beforeEach(function() {
let { owner } = this;
owner.register('service:store', Store);
owner.register('model:person', Person);
owner.register('model:pet', Pet);
owner.register(
'serializer:application',
JSONAPISerializer.extend({
normalizeResponse(_, __, payload) {
return payload;
},
})
);
store = owner.lookup('service:store');
});

test("async belongsTo chains the related record's loading promise when present", async function(assert) {
let petFindRecordCalls = 0;
this.owner.register(
'adapter:pet',
JSONAPIAdapter.extend({
findRecord() {
assert.equal(++petFindRecordCalls, 1, 'We call findRecord only once for our pet');
return resolve({
data: {
type: 'pet',
id: '1',
attributes: { name: 'Shen' },
relationships: {
bestHuman: {
data: { type: 'person', id: '1' },
},
},
},
});
},
findBelongsTo() {
return this.store.adapterFor('person').findRecord();
},
})
);
let personFindRecordCalls = 0;
this.owner.register(
'adapter:person',
JSONAPIAdapter.extend({
findRecord() {
assert.equal(++personFindRecordCalls, 1, 'We call findRecord only once for our person');
return resolve({
data: {
type: 'person',
id: '1',
attributes: { name: 'Chris' },
relationships: {
bestDog: {
data: { type: 'pet', id: '1' },
links: {
related: './pet/1',
},
},
},
},
});
},
findBelongsTo() {
return this.store.adapterFor('pet').findRecord();
},
})
);

let person = await store.findRecord('person', '1');
let petRequest = store.findRecord('pet', '1');
let personPetRequest = person.get('bestDog');
let personPet = await personPetRequest;
let pet = await petRequest;

assert.ok(personPet === pet, 'We ended up in the same state');
});
});

module('integration/relationship/belongs_to Belongs-To Relationships', {
beforeEach() {
User = DS.Model.extend({
name: attr('string'),
messages: hasMany('message', { polymorphic: true, async: false }),
favouriteMessage: belongsTo('message', { polymorphic: true, inverse: null, async: false }),
name: DSattr('string'),
messages: DShasMany('message', { polymorphic: true, async: false }),
favouriteMessage: DSbelongsTo('message', { polymorphic: true, inverse: null, async: false }),
});

Message = DS.Model.extend({
user: belongsTo('user', { inverse: 'messages', async: false }),
created_at: attr('date'),
user: DSbelongsTo('user', { inverse: 'messages', async: false }),
created_at: DSattr('date'),
});

Post = Message.extend({
title: attr('string'),
comments: hasMany('comment', { async: false, inverse: null }),
title: DSattr('string'),
comments: DShasMany('comment', { async: false, inverse: null }),
});

Comment = Message.extend({
Expand All @@ -42,27 +143,27 @@ module('integration/relationship/belongs_to Belongs-To Relationships', {
});

Book = DS.Model.extend({
name: attr('string'),
author: belongsTo('author', { async: false }),
chapters: hasMany('chapters', { async: false, inverse: 'book' }),
name: DSattr('string'),
author: DSbelongsTo('author', { async: false }),
chapters: DShasMany('chapters', { async: false, inverse: 'book' }),
});

Book1 = DS.Model.extend({
name: attr('string'),
name: DSattr('string'),
});

Chapter = DS.Model.extend({
title: attr('string'),
book: belongsTo('book', { async: false, inverse: 'chapters' }),
title: DSattr('string'),
book: DSbelongsTo('book', { async: false, inverse: 'chapters' }),
});

Author = DS.Model.extend({
name: attr('string'),
books: hasMany('books', { async: false }),
name: DSattr('string'),
books: DShasMany('books', { async: false }),
});

Section = DS.Model.extend({
name: attr('string'),
name: DSattr('string'),
});

env = setupStore({
Expand Down Expand Up @@ -110,14 +211,14 @@ test('returning a null relationship from payload sets the relationship to null o
env.registry.register(
'model:app',
DS.Model.extend({
name: attr('string'),
team: belongsTo('team', { async: true }),
name: DSattr('string'),
team: DSbelongsTo('team', { async: true }),
})
);
env.registry.register(
'model:team',
DS.Model.extend({
apps: hasMany('app', { async: true }),
apps: DShasMany('app', { async: true }),
})
);
run(() => {
Expand Down Expand Up @@ -854,7 +955,7 @@ test('Destroying a record with an unloaded aync belongsTo association does not f
run(post, 'destroyRecord');
});

testInDebug('A sync belongsTo errors out if the record is unlaoded', function(assert) {
testInDebug('A sync belongsTo errors out if the record is unloaded', function(assert) {
let message;
run(() => {
message = env.store.push({
Expand Down Expand Up @@ -969,7 +1070,7 @@ testInDebug('Passing a model as type to belongsTo should not work', function(ass
User = DS.Model.extend();

DS.Model.extend({
user: belongsTo(User, { async: false }),
user: DSbelongsTo(User, { async: false }),
});
}, /The first argument to DS.belongsTo must be a string/);
});
Expand All @@ -978,7 +1079,7 @@ test('belongsTo hasAnyRelationshipData async loaded', function(assert) {
assert.expect(1);

Book.reopen({
author: belongsTo('author', { async: true }),
author: DSbelongsTo('author', { async: true }),
});

env.adapter.findRecord = function(store, type, id, snapshot) {
Expand Down Expand Up @@ -1030,7 +1131,7 @@ test('belongsTo hasAnyRelationshipData async not loaded', function(assert) {
assert.expect(1);

Book.reopen({
author: belongsTo('author', { async: true }),
author: DSbelongsTo('author', { async: true }),
});

env.adapter.findRecord = function(store, type, id, snapshot) {
Expand Down Expand Up @@ -1079,7 +1180,7 @@ test('belongsTo hasAnyRelationshipData NOT created', function(assert) {
assert.expect(2);

Book.reopen({
author: belongsTo('author', { async: true }),
author: DSbelongsTo('author', { async: true }),
});

run(() => {
Expand Down Expand Up @@ -1825,7 +1926,7 @@ testRecordData(
// Expect assertion failure as Book1 ModelData
// doesn't have relationship attribute
// and inverse is not set to null in
// belongsTo
// DSbelongsTo
assert.expectAssertion(() => {
run(() => {
env.store.push(data);
Expand Down

0 comments on commit 5dad50b

Please sign in to comment.