diff --git a/packages/ember-data/lib/serializers/embedded-records-mixin.js b/packages/ember-data/lib/serializers/embedded-records-mixin.js index 1fe96c6c068..2373e42aa02 100644 --- a/packages/ember-data/lib/serializers/embedded-records-mixin.js +++ b/packages/ember-data/lib/serializers/embedded-records-mixin.js @@ -1,3 +1,5 @@ +var get = Ember.get; +var set = Ember.set; var forEach = Ember.EnumerableUtils.forEach; var camelize = Ember.String.camelize; @@ -91,6 +93,16 @@ var camelize = Ember.String.camelize; */ var EmbeddedRecordsMixin = Ember.Mixin.create({ + init: function() { + this._super.apply(this, arguments); + + if (this.get('isNewSerializerAPI')) { + this._extractEmbeddedRecords = this._newExtractEmbeddedRecords; + this._extractEmbeddedBelongsTo = this._newExtractEmbeddedBelongsTo; + this._extractEmbeddedHasMany = this._newExtractEmbeddedHasMany; + } + }, + /** Normalize the record and recursively normalize/extract all the embedded records while pushing them into the store as they are encountered @@ -120,7 +132,12 @@ var EmbeddedRecordsMixin = Ember.Mixin.create({ **/ normalize: function(typeClass, hash, prop) { var normalizedHash = this._super(typeClass, hash, prop); - return extractEmbeddedRecords(this, this.store, typeClass, normalizedHash); + return this._extractEmbeddedRecords(this, this.store, typeClass, normalizedHash); + }, + + newNormalize: function(typeClass, hash, prop) { + var normalizedHash = this._super(typeClass, hash, prop); + return this._extractEmbeddedRecords(this, this.store, typeClass, normalizedHash); }, keyForRelationship: function(key, typeClass, method) { @@ -385,63 +402,93 @@ var EmbeddedRecordsMixin = Ember.Mixin.create({ attrsOption: function(attr) { var attrs = this.get('attrs'); return attrs && (attrs[camelize(attr)] || attrs[attr]); - } -}); + }, -// chooses a relationship kind to branch which function is used to update payload -// does not change payload if attr is not embedded -function extractEmbeddedRecords(serializer, store, typeClass, partial) { - - typeClass.eachRelationship(function(key, relationship) { - if (serializer.hasDeserializeRecordsOption(key)) { - var embeddedTypeClass = store.modelFor(relationship.type); - if (relationship.kind === "hasMany") { - if (relationship.options.polymorphic) { - extractEmbeddedHasManyPolymorphic(store, key, partial); - } else { - extractEmbeddedHasMany(store, key, embeddedTypeClass, partial); + _extractEmbeddedRecords: function(serializer, store, typeClass, partial) { + + typeClass.eachRelationship(function(key, relationship) { + if (serializer.hasDeserializeRecordsOption(key)) { + var embeddedTypeClass = store.modelFor(relationship.type); + if (relationship.kind === "hasMany") { + if (relationship.options.polymorphic) { + this._extractEmbeddedHasManyPolymorphic(store, key, partial); + } else { + this._extractEmbeddedHasMany(store, key, embeddedTypeClass, partial); + } } - } - if (relationship.kind === "belongsTo") { - if (relationship.options.polymorphic) { - extractEmbeddedBelongsToPolymorphic(store, key, partial); - } else { - extractEmbeddedBelongsTo(store, key, embeddedTypeClass, partial); + if (relationship.kind === "belongsTo") { + if (relationship.options.polymorphic) { + this._extractEmbeddedBelongsToPolymorphic(store, key, partial); + } else { + this._extractEmbeddedBelongsTo(store, key, embeddedTypeClass, partial); + } } } + }, this); + + return partial; + }, + + _extractEmbeddedHasMany: function(store, key, embeddedTypeClass, hash) { + if (!hash[key]) { + return hash; } - }); - return partial; -} + var ids = []; -// handles embedding for `hasMany` relationship -function extractEmbeddedHasMany(store, key, embeddedTypeClass, hash) { - if (!hash[key]) { + var embeddedSerializer = store.serializerFor(embeddedTypeClass.modelName); + forEach(hash[key], function(data) { + var embeddedRecord = embeddedSerializer.normalize(embeddedTypeClass, data, null); + store.push(embeddedTypeClass.modelName, embeddedRecord); + ids.push(embeddedRecord.id); + }); + + hash[key] = ids; return hash; - } + }, - var ids = []; + _extractEmbeddedHasManyPolymorphic: function(store, key, hash) { + if (!hash[key]) { + return hash; + } - var embeddedSerializer = store.serializerFor(embeddedTypeClass.modelName); - forEach(hash[key], function(data) { - var embeddedRecord = embeddedSerializer.normalize(embeddedTypeClass, data, null); - store.push(embeddedTypeClass.modelName, embeddedRecord); - ids.push(embeddedRecord.id); - }); + var ids = []; - hash[key] = ids; - return hash; -} + forEach(hash[key], function(data) { + var modelName = data.type; + var embeddedSerializer = store.serializerFor(modelName); + var embeddedTypeClass = store.modelFor(modelName); + // var primaryKey = embeddedSerializer.get('primaryKey'); -function extractEmbeddedHasManyPolymorphic(store, key, hash) { - if (!hash[key]) { + var embeddedRecord = embeddedSerializer.normalize(embeddedTypeClass, data, null); + store.push(embeddedTypeClass.modelName, embeddedRecord); + ids.push({ id: embeddedRecord.id, type: modelName }); + }); + + hash[key] = ids; return hash; - } + }, + + _extractEmbeddedBelongsTo: function(store, key, embeddedTypeClass, hash) { + if (!hash[key]) { + return hash; + } + + var embeddedSerializer = store.serializerFor(embeddedTypeClass.modelName); + var embeddedRecord = embeddedSerializer.normalize(embeddedTypeClass, hash[key], null); + store.push(embeddedTypeClass.modelName, embeddedRecord); + + hash[key] = embeddedRecord.id; + //TODO Need to add a reference to the parent later so relationship works between both `belongsTo` records + return hash; + }, - var ids = []; + _extractEmbeddedBelongsToPolymorphic: function(store, key, hash) { + if (!hash[key]) { + return hash; + } - forEach(hash[key], function(data) { + var data = hash[key]; var modelName = data.type; var embeddedSerializer = store.serializerFor(modelName); var embeddedTypeClass = store.modelFor(modelName); @@ -449,43 +496,76 @@ function extractEmbeddedHasManyPolymorphic(store, key, hash) { var embeddedRecord = embeddedSerializer.normalize(embeddedTypeClass, data, null); store.push(embeddedTypeClass.modelName, embeddedRecord); - ids.push({ id: embeddedRecord.id, type: modelName }); - }); - hash[key] = ids; - return hash; -} - -function extractEmbeddedBelongsTo(store, key, embeddedTypeClass, hash) { - if (!hash[key]) { + hash[key] = embeddedRecord.id; + hash[key + 'Type'] = modelName; return hash; - } + }, + + _newExtractEmbeddedRecords: function(serializer, store, typeClass, partial) { + typeClass.eachRelationship((key, relationship) => { + if (serializer.hasDeserializeRecordsOption(key)) { + let embeddedTypeClass = store.modelFor(relationship.type); + if (relationship.kind === "hasMany") { + this._extractEmbeddedHasMany(store, key, embeddedTypeClass, partial, relationship); + } + if (relationship.kind === "belongsTo") { + this._extractEmbeddedBelongsTo(store, key, embeddedTypeClass, partial, relationship); + } + } + }, this); + return partial; + }, + + _newExtractEmbeddedHasMany: function(store, key, typeClass, hash, relationshipMeta) { + let relationshipHash = get(hash, `data.relationships.${key}.data`); + if (!relationshipHash) { + return; + } - var embeddedSerializer = store.serializerFor(embeddedTypeClass.modelName); - var embeddedRecord = embeddedSerializer.normalize(embeddedTypeClass, hash[key], null); - store.push(embeddedTypeClass.modelName, embeddedRecord); + let hasMany = relationshipHash.map(item => { + let embeddedTypeClass = typeClass; + if (relationshipMeta.options.polymorphic) { + let modelName = item.type; + embeddedTypeClass = store.modelFor(modelName); + } - hash[key] = embeddedRecord.id; - //TODO Need to add a reference to the parent later so relationship works between both `belongsTo` records - return hash; -} + let embeddedSerializer = store.serializerFor(embeddedTypeClass.modelName); + let { data, included } = embeddedSerializer.normalize(embeddedTypeClass, item, null); + hash.included = hash.included || []; + hash.included.push(data); + hash.included.push(...included); -function extractEmbeddedBelongsToPolymorphic(store, key, hash) { - if (!hash[key]) { - return hash; - } + return { id: data.id, type: embeddedTypeClass.modelName }; + }); - var data = hash[key]; - var modelName = data.type; - var embeddedSerializer = store.serializerFor(modelName); - var embeddedTypeClass = store.modelFor(modelName); + let relationship = { data: hasMany }; + set(hash, `data.relationships.${key}`, relationship); + }, - var embeddedRecord = embeddedSerializer.normalize(embeddedTypeClass, data, null); - store.push(embeddedTypeClass.modelName, embeddedRecord); + _newExtractEmbeddedBelongsTo: function(store, key, typeClass, hash, relationshipMeta) { + let relationshipHash = get(hash, `data.relationships.${key}.data`); + if (!relationshipHash) { + return; + } + + let embeddedTypeClass = typeClass; + if (relationshipMeta.options.polymorphic) { + let modelName = relationshipHash.type; + embeddedTypeClass = store.modelFor(modelName); + } + + let embeddedSerializer = store.serializerFor(embeddedTypeClass.modelName); + let { data, included } = embeddedSerializer.normalize(embeddedTypeClass, relationshipHash, null); + hash.included = hash.included || []; + hash.included.push(data); + hash.included.push(...included); - hash[key] = embeddedRecord.id; - hash[key + 'Type'] = modelName; - return hash; -} + let belongsTo = { id: data.id, type: embeddedTypeClass.modelName }; + + let relationship = { data: belongsTo }; + set(hash, `data.relationships.${key}`, relationship); + } +}); export default EmbeddedRecordsMixin; diff --git a/packages/ember-data/tests/integration/serializers/embedded-records-mixin-new-test.js b/packages/ember-data/tests/integration/serializers/embedded-records-mixin-new-test.js new file mode 100644 index 00000000000..61a95db034e --- /dev/null +++ b/packages/ember-data/tests/integration/serializers/embedded-records-mixin-new-test.js @@ -0,0 +1,1196 @@ +var HomePlanet, SuperVillain, EvilMinion, SecretLab, SecretWeapon, BatCave, Comment, env; +var run = Ember.run; +var LightSaber; +var TestSerializer; + +module("integration/embedded_records_mixin - EmbeddedRecordsMixin (new API)", { + setup: function() { + SuperVillain = DS.Model.extend({ + firstName: DS.attr('string'), + lastName: DS.attr('string'), + homePlanet: DS.belongsTo("homePlanet", { inverse: 'villains' }), + secretLab: DS.belongsTo("secretLab"), + secretWeapons: DS.hasMany("secretWeapon"), + evilMinions: DS.hasMany("evilMinion") + }); + HomePlanet = DS.Model.extend({ + name: DS.attr('string'), + villains: DS.hasMany('superVillain', { inverse: 'homePlanet' }) + }); + SecretLab = DS.Model.extend({ + minionCapacity: DS.attr('number'), + vicinity: DS.attr('string'), + superVillain: DS.belongsTo('superVillain') + }); + BatCave = SecretLab.extend({ + infiltrated: DS.attr('boolean') + }); + SecretWeapon = DS.Model.extend({ + name: DS.attr('string'), + superVillain: DS.belongsTo('superVillain') + }); + LightSaber = SecretWeapon.extend({ + color: DS.attr('string') + }); + EvilMinion = DS.Model.extend({ + superVillain: DS.belongsTo('superVillain'), + name: DS.attr('string') + }); + Comment = DS.Model.extend({ + body: DS.attr('string'), + root: DS.attr('boolean'), + children: DS.hasMany('comment', { inverse: null }) + }); + TestSerializer = DS.RESTSerializer.extend({ + isNewSerializerAPI: true + }); + env = setupStore({ + superVillain: SuperVillain, + homePlanet: HomePlanet, + secretLab: SecretLab, + batCave: BatCave, + secretWeapon: SecretWeapon, + lightSaber: LightSaber, + evilMinion: EvilMinion, + comment: Comment + }); + env.store.modelFor('superVillain'); + env.store.modelFor('homePlanet'); + env.store.modelFor('secretLab'); + env.store.modelFor('batCave'); + env.store.modelFor('secretWeapon'); + env.store.modelFor('lightSaber'); + env.store.modelFor('evilMinion'); + env.store.modelFor('comment'); + + env.registry.register('serializer:application', TestSerializer.extend(DS.EmbeddedRecordsMixin)); + }, + + teardown: function() { + run(env.store, 'destroy'); + } +}); + +test("extractSingle with embedded objects", function() { + env.registry.register('adapter:super-villain', DS.RESTAdapter); + env.registry.register('serializer:super-villain', TestSerializer.extend()); + env.registry.register('serializer:home-planet', TestSerializer.extend(DS.EmbeddedRecordsMixin, { + attrs: { + villains: { embedded: 'always' } + } + })); + + var serializer = env.container.lookup("serializer:home-planet"); + var json_hash = { + homePlanet: { + id: "1", + name: "Umber", + villains: [{ + id: "2", + firstName: "Tom", + lastName: "Dale" + }] + } + }; + var json; + + run(function() { + json = serializer.normalizeResponse(env.store, HomePlanet, json_hash, '1', 'find'); + }); + + deepEqual(json, { + "data": { + "id": "1", + "type": "home-planet", + "attributes": { + "name": "Umber" + }, + "relationships": { + "villains": { + "data": [ + { "id": "2", "type": "super-villain" } + ] + } + } + }, + "included": [ + { + "id": "2", + "type": "super-villain", + "attributes": { + "firstName": "Tom", + "lastName": "Dale" + }, + "relationships": {} + } + ] + }); +}); + +test("extractSingle with embedded objects inside embedded objects", function() { + env.registry.register('adapter:super-villain', DS.RESTAdapter); + env.registry.register('serializer:home-planet', TestSerializer.extend(DS.EmbeddedRecordsMixin, { + attrs: { + villains: { embedded: 'always' } + } + })); + env.registry.register('serializer:super-villain', TestSerializer.extend(DS.EmbeddedRecordsMixin, { + attrs: { + evilMinions: { embedded: 'always' } + } + })); + env.registry.register('serializer:evil-minion', TestSerializer); + + var serializer = env.container.lookup("serializer:home-planet"); + var json_hash = { + homePlanet: { + id: "1", + name: "Umber", + villains: [{ + id: "2", + firstName: "Tom", + lastName: "Dale", + evilMinions: [{ + id: "3", + name: "Alex" + }] + }] + } + }; + var json; + + run(function() { + json = serializer.normalizeResponse(env.store, HomePlanet, json_hash, '1', 'find'); + }); + + deepEqual(json, { + "data": { + "id": "1", + "type": "home-planet", + "attributes": { + "name": "Umber" + }, + "relationships": { + "villains": { + "data": [ + { "id": "2", "type": "super-villain" } + ] + } + } + }, + "included": [{ + "id": "2", + "type": "super-villain", + "attributes": { + "firstName": "Tom", + "lastName": "Dale" + }, + "relationships": { + "evilMinions": { + "data": [ + { "id": "3", "type": "evil-minion" } + ] + } + } + }, { + "id": "3", + "type": "evil-minion", + "attributes": { + "name": "Alex" + }, + "relationships": {} + }] + }); +}); + +test("extractSingle with embedded objects of same type", function() { + env.registry.register('adapter:comment', DS.RESTAdapter); + env.registry.register('serializer:comment', TestSerializer.extend(DS.EmbeddedRecordsMixin, { + attrs: { + children: { embedded: 'always' } + } + })); + + var serializer = env.container.lookup("serializer:comment"); + var json_hash = { + comment: { + id: "1", + body: "Hello", + root: true, + children: [{ + id: "2", + body: "World", + root: false + }, + { + id: "3", + body: "Foo", + root: false + }] + } + }; + var json; + run(function() { + json = serializer.normalizeResponse(env.store, Comment, json_hash, '1', 'find'); + }); + + deepEqual(json, { + "data": { + "id": "1", + "type": "comment", + "attributes": { + "body": "Hello", + "root": true + }, + "relationships": { + "children": { + "data": [ + { "id": "2", "type": "comment" }, + { "id": "3", "type": "comment" } + ] + } + } + }, + "included": [{ + "id": "2", + "type": "comment", + "attributes": { + "body": "World", + "root": false + }, + "relationships": {} + }, { + "id": "3", + "type": "comment", + "attributes": { + "body": "Foo", + "root": false + }, + "relationships": {} + }] + }, "Primary record was correct"); +}); + +test("extractSingle with embedded objects inside embedded objects of same type", function() { + env.registry.register('adapter:comment', DS.RESTAdapter); + env.registry.register('serializer:comment', TestSerializer.extend(DS.EmbeddedRecordsMixin, { + attrs: { + children: { embedded: 'always' } + } + })); + + var serializer = env.container.lookup("serializer:comment"); + var json_hash = { + comment: { + id: "1", + body: "Hello", + root: true, + children: [{ + id: "2", + body: "World", + root: false, + children: [{ + id: "4", + body: "Another", + root: false + }] + }, + { + id: "3", + body: "Foo", + root: false + }] + } + }; + var json; + run(function() { + json = serializer.normalizeResponse(env.store, Comment, json_hash, '1', 'find'); + }); + + deepEqual(json, { + "data": { + "id": "1", + "type": "comment", + "attributes": { + "body": "Hello", + "root": true + }, + "relationships": { + "children": { + "data": [ + { "id": "2", "type": "comment" }, + { "id": "3", "type": "comment" } + ] + } + } + }, + "included": [{ + "id": "2", + "type": "comment", + "attributes": { + "body": "World", + "root": false + }, + "relationships": { + "children": { + "data": [ + { "id": "4", "type": "comment" } + ] + } + } + }, { + "id": "4", + "type": "comment", + "attributes": { + "body": "Another", + "root": false + }, + "relationships": {} + }, { + "id": "3", + "type": "comment", + "attributes": { + "body": "Foo", + "root": false + }, + "relationships": {} + }] + }, "Primary record was correct"); +}); + +test("extractSingle with embedded objects of same type, but from separate attributes", function() { + HomePlanet.reopen({ + reformedVillains: DS.hasMany('superVillain', { inverse: null }) + }); + + env.registry.register('adapter:home-planet', DS.RESTAdapter); + env.registry.register('serializer:home-planet', TestSerializer.extend(DS.EmbeddedRecordsMixin, { + attrs: { + villains: { embedded: 'always' }, + reformedVillains: { embedded: 'always' } + } + })); + env.registry.register('serializer:super-villain', TestSerializer); + + var serializer = env.container.lookup("serializer:home-planet"); + var json_hash = { + homePlanet: { + id: "1", + name: "Earth", + villains: [{ + id: "1", + firstName: "Tom" + }, { + id: "3", + firstName: "Yehuda" + }], + reformedVillains: [{ + id: "2", + firstName: "Alex" + },{ + id: "4", + firstName: "Erik" + }] + } + }; + var json; + run(function() { + json = serializer.normalizeResponse(env.store, HomePlanet, json_hash, '1', 'find'); + }); + + deepEqual(json, { + "data": { + "id": "1", + "type": "home-planet", + "attributes": { + "name": "Earth" + }, + "relationships": { + "villains": { + "data": [ + { "id": "1", "type": "super-villain" }, + { "id": "3", "type": "super-villain" } + ] + }, + "reformedVillains": { + "data": [ + { "id": "2", "type": "super-villain" }, + { "id": "4", "type": "super-villain" } + ] + } + } + }, + "included": [{ + "id": "1", + "type": "super-villain", + "attributes": { + "firstName": "Tom" + }, + "relationships": {} + }, { + "id": "3", + "type": "super-villain", + "attributes": { + "firstName": "Yehuda" + }, + "relationships": {} + }, { + "id": "2", + "type": "super-villain", + "attributes": { + "firstName": "Alex" + }, + "relationships": {} + }, { + "id": "4", + "type": "super-villain", + "attributes": { + "firstName": "Erik" + }, + "relationships": {} + }] + }, "Primary hash was correct"); +}); + +test("extractArray with embedded objects", function() { + env.registry.register('adapter:super-villain', DS.RESTAdapter); + env.registry.register('serializer:home-planet', TestSerializer.extend(DS.EmbeddedRecordsMixin, { + attrs: { + villains: { embedded: 'always' } + } + })); + env.registry.register('serializer:super-villain', TestSerializer); + + var serializer = env.container.lookup("serializer:home-planet"); + + var json_hash = { + homePlanets: [{ + id: "1", + name: "Umber", + villains: [{ + id: "1", + firstName: "Tom", + lastName: "Dale" + }] + }] + }; + var array; + + run(function() { + array = serializer.normalizeResponse(env.store, HomePlanet, json_hash, null, 'findAll'); + }); + + deepEqual(array, { + "data": [{ + "id": "1", + "type": "home-planet", + "attributes": { + "name": "Umber" + }, + "relationships": { + "villains": { + "data": [ + { "id": "1", "type": "super-villain" } + ] + } + } + }], + "included": [{ + "id": "1", + "type": "super-villain", + "attributes": { + "firstName": "Tom", + "lastName": "Dale" + }, + "relationships": {} + }] + }); +}); + +test("extractArray with embedded objects with custom primary key", function() { + expect(1); + env.registry.register('adapter:super-villain', DS.RESTAdapter); + env.registry.register('serializer:super-villain', TestSerializer.extend({ + primaryKey: 'villain_id' + })); + env.registry.register('serializer:home-planet', TestSerializer.extend(DS.EmbeddedRecordsMixin, { + attrs: { + villains: { embedded: 'always' } + } + })); + + var serializer = env.container.lookup("serializer:home-planet"); + + var json_hash = { + homePlanets: [{ + id: "1", + name: "Umber", + villains: [{ + villain_id: "2", + firstName: "Alex", + lastName: "Baizeau" + }] + }] + }; + var array; + + run(function() { + array = serializer.normalizeResponse(env.store, HomePlanet, json_hash, null, 'findAll'); + }); + + deepEqual(array, { + "data": [{ + "id": "1", + "type": "home-planet", + "attributes": { + "name": "Umber" + }, + "relationships": { + "villains": { + "data": [ + { "id": "2", "type": "super-villain" } + ] + } + } + }], + "included": [{ + "id": "2", + "type": "super-villain", + "attributes": { + "firstName": "Alex", + "lastName": "Baizeau" + }, + "relationships": {} + }] + }); +}); + +test("extractArray with embedded objects with identical relationship and attribute key ", function() { + expect(1); + env.registry.register('adapter:super-villain', DS.RESTAdapter); + env.registry.register('serializer:home-planet', TestSerializer.extend(DS.EmbeddedRecordsMixin, { + attrs: { + villains: { embedded: 'always' } + }, + //Makes the keyForRelationship and keyForAttribute collide. + keyForRelationship: function(key, type) { + return this.keyForAttribute(key, type); + } + })); + env.registry.register('serializer:super-villain', TestSerializer); + + var serializer = env.container.lookup("serializer:home-planet"); + + var json_hash = { + homePlanets: [{ + id: "1", + name: "Umber", + villains: [{ + id: "1", + firstName: "Alex", + lastName: "Baizeau" + }] + }] + }; + var array; + + run(function() { + array = serializer.normalizeResponse(env.store, HomePlanet, json_hash, null, 'findAll'); + }); + + deepEqual(array, { + "data": [{ + "id": "1", + "type": "home-planet", + "attributes": { + "name": "Umber" + }, + "relationships": { + "villains": { + "data": [ + { "id": "1", "type": "super-villain" } + ] + } + } + }], + "included": [{ + "id": "1", + "type": "super-villain", + "attributes": { + "firstName": "Alex", + "lastName": "Baizeau" + }, + "relationships": {} + }] + }); +}); +test("extractArray with embedded objects of same type as primary type", function() { + env.registry.register('adapter:comment', DS.RESTAdapter); + env.registry.register('serializer:comment', TestSerializer.extend(DS.EmbeddedRecordsMixin, { + attrs: { + children: { embedded: 'always' } + } + })); + + var serializer = env.container.lookup("serializer:comment"); + + var json_hash = { + comments: [{ + id: "1", + body: "Hello", + root: true, + children: [{ + id: "2", + body: "World", + root: false + }, + { + id: "3", + body: "Foo", + root: false + }] + }] + }; + var array; + + run(function() { + array = serializer.normalizeResponse(env.store, Comment, json_hash, null, 'findAll'); + }); + + deepEqual(array, { + "data": [{ + "id": "1", + "type": "comment", + "attributes": { + "body": "Hello", + "root": true + }, + "relationships": { + "children": { + "data": [ + { "id": "2", "type": "comment" }, + { "id": "3", "type": "comment" } + ] + } + } + }], + "included": [{ + "id": "2", + "type": "comment", + "attributes": { + "body": "World", + "root": false + }, + "relationships": {} + }, { + "id": "3", + "type": "comment", + "attributes": { + "body": "Foo", + "root": false + }, + "relationships": {} + }] + }, "Primary array is correct"); +}); + +test("extractArray with embedded objects of same type, but from separate attributes", function() { + HomePlanet.reopen({ + reformedVillains: DS.hasMany('superVillain') + }); + + env.registry.register('adapter:home-planet', DS.RESTAdapter); + env.registry.register('serializer:home-planet', TestSerializer.extend(DS.EmbeddedRecordsMixin, { + attrs: { + villains: { embedded: 'always' }, + reformedVillains: { embedded: 'always' } + } + })); + env.registry.register('serializer:super-villain', TestSerializer); + + var serializer = env.container.lookup("serializer:home-planet"); + var json_hash = { + homePlanets: [{ + id: "1", + name: "Earth", + villains: [{ + id: "1", + firstName: "Tom" + },{ + id: "3", + firstName: "Yehuda" + }], + reformedVillains: [{ + id: "2", + firstName: "Alex" + },{ + id: "4", + firstName: "Erik" + }] + },{ + id: "2", + name: "Mars", + villains: [{ + id: "1", + firstName: "Tom" + },{ + id: "3", + firstName: "Yehuda" + }], + reformedVillains: [{ + id: "5", + firstName: "Peter" + },{ + id: "6", + firstName: "Trek" + }] + }] + }; + + var json; + run(function() { + json = serializer.normalizeResponse(env.store, HomePlanet, json_hash, null, 'findAll'); + }); + + deepEqual(json, { + "data": [{ + "id": "1", + "type": "home-planet", + "attributes": { + "name": "Earth" + }, + "relationships": { + "reformedVillains": { + "data": [ + { "id": "2", "type": "super-villain" }, + { "id": "4", "type": "super-villain" } + ] + }, + "villains": { + "data": [ + { "id": "1", "type": "super-villain" }, + { "id": "3", "type": "super-villain" } + ] + } + } + }, { + "id": "2", + "type": "home-planet", + "attributes": { + "name": "Mars" + }, + "relationships": { + "reformedVillains": { + "data": [ + { "id": "5", "type": "super-villain" }, + { "id": "6", "type": "super-villain" } + ] + }, + "villains": { + "data": [ + { "id": "1", "type": "super-villain" }, + { "id": "3", "type": "super-villain" } + ] + } + } + }], + "included": [{ + "id": "1", + "type": "super-villain", + "attributes": { + "firstName": "Tom" + }, + "relationships": {} + }, { + "id": "3", + "type": "super-villain", + "attributes": { + "firstName": "Yehuda" + }, + "relationships": {} + }, { + "id": "2", + "type": "super-villain", + "attributes": { + "firstName": "Alex" + }, + "relationships": {} + }, { + "id": "4", + "type": "super-villain", + "attributes": { + "firstName": "Erik" + }, + "relationships": {} + }, { + "id": "1", + "type": "super-villain", + "attributes": { + "firstName": "Tom" + }, + "relationships": {} + }, { + "id": "3", + "type": "super-villain", + "attributes": { + "firstName": "Yehuda" + }, + "relationships": {} + }, { + "id": "5", + "type": "super-villain", + "attributes": { + "firstName": "Peter" + }, + "relationships": {} + }, { + "id": "6", + "type": "super-villain", + "attributes": { + "firstName": "Trek" + }, + "relationships": {} + }] + }, "Primary array was correct"); +}); + +test("extractSingle with embedded object (belongsTo relationship)", function() { + env.registry.register('adapter:super-villain', DS.RESTAdapter); + env.registry.register('serializer:super-villain', TestSerializer.extend(DS.EmbeddedRecordsMixin, { + attrs: { + secretLab: { embedded: 'always' } + } + })); + //env.registry.register('serializer:secret-lab', TestSerializer); + + var serializer = env.container.lookup("serializer:super-villain"); + + var json_hash = { + superVillain: { + id: "1", + firstName: "Tom", + lastName: "Dale", + homePlanet: "123", + evilMinions: ["1", "2", "3"], + secretLab: { + minionCapacity: 5000, + vicinity: "California, USA", + id: "101" + }, + secretWeapons: [] + } + }; + var json; + + run(function() { + json = serializer.normalizeResponse(env.store, SuperVillain, json_hash, '1', 'find'); + }); + + deepEqual(json, { + "data": { + "id": "1", + "type": "super-villain", + "attributes": { + "firstName": "Tom", + "lastName": "Dale" + }, + "relationships": { + "evilMinions": { + "data": [ + { "id": "1", "type": "evil-minion" }, + { "id": "2", "type": "evil-minion" }, + { "id": "3", "type": "evil-minion" } + ] + }, + "homePlanet": { + "data": { "id": "123", "type": "home-planet" } + }, + "secretLab": { + "data": { "id": "101", "type": "secret-lab" } + }, + "secretWeapons": { + "data": [] + } + } + }, + "included": [{ + "id": "101", + "type": "secret-lab", + "attributes": { + "minionCapacity": 5000, + "vicinity": "California, USA" + }, + "relationships": {} + }] + }); +}); + +test("extractSingle with multiply-nested belongsTo", function() { + env.registry.register('adapter:evil-minion', DS.RESTAdapter); + env.registry.register('serializer:evil-minion', TestSerializer.extend(DS.EmbeddedRecordsMixin, { + attrs: { + superVillain: { embedded: 'always' } + } + })); + env.registry.register('serializer:super-villain', TestSerializer.extend(DS.EmbeddedRecordsMixin, { + attrs: { + homePlanet: { embedded: 'always' } + } + })); + + var serializer = env.container.lookup("serializer:evil-minion"); + var json_hash = { + evilMinion: { + id: "1", + name: "Alex", + superVillain: { + id: "1", + firstName: "Tom", + lastName: "Dale", + evilMinions: ["1"], + homePlanet: { + id: "1", + name: "Umber", + villains: ["1"] + } + } + } + }; + var json; + + run(function() { + json = serializer.normalizeResponse(env.store, EvilMinion, json_hash, '1', 'find'); + }); + + deepEqual(json, { + "data": { + "id": "1", + "type": "evil-minion", + "attributes": { + "name": "Alex" + }, + "relationships": { + "superVillain": { + "data": { "id": "1", "type": "super-villain" } + } + } + }, + "included": [{ + "id": "1", + "type": "super-villain", + "attributes": { + "firstName": "Tom", + "lastName": "Dale" + }, + "relationships": { + "evilMinions": { + "data": [ + { "id": "1", "type": "evil-minion" } + ] + }, + "homePlanet": { + "data": { "id": "1", "type": "home-planet" } + } + } + }, { + "id": "1", + "type": "home-planet", + "attributes": { + "name": "Umber" + }, + "relationships": { + "villains": { + "data": [ + { "id": "1", "type": "super-villain" } + ] + } + } + }] + }, "Primary hash was correct"); +}); + +test("extractSingle with polymorphic hasMany", function() { + SuperVillain.reopen({ + secretWeapons: DS.hasMany("secretWeapon", { polymorphic: true }) + }); + + env.registry.register('adapter:super-villain', DS.RESTAdapter); + env.registry.register('serializer:super-villain', TestSerializer.extend(DS.EmbeddedRecordsMixin, { + attrs: { + secretWeapons: { embedded: 'always' } + } + })); + var serializer = env.container.lookup("serializer:super-villain"); + + var json_hash = { + superVillain: { + id: "1", + firstName: "Tom", + lastName: "Dale", + secretWeapons: [ + { + id: "1", + type: "LightSaber", + name: "Tom's LightSaber", + color: "Red" + }, + { + id: "1", + type: "SecretWeapon", + name: "The Death Star" + } + ] + } + }; + var json; + + run(function() { + json = serializer.normalizeResponse(env.store, SuperVillain, json_hash, '1', 'findAll'); + }); + + deepEqual(json, { + "data": { + "id": "1", + "type": "super-villain", + "attributes": { + "firstName": "Tom", + "lastName": "Dale" + }, + "relationships": { + "secretWeapons": { + "data": [ + { "id": "1", "type": "light-saber" }, + { "id": "1", "type": "secret-weapon" } + ] + } + } + }, + "included": [{ + "id": "1", + "type": "light-saber", + "attributes": { + "color": "Red", + "name": "Tom's LightSaber" + }, + "relationships": {} + }, { + "id": "1", + "type": "secret-weapon", + "attributes": { + "name": "The Death Star" + }, + "relationships": {} + }] + }, "Primary hash was correct"); +}); + +test("extractSingle with polymorphic belongsTo", function() { + SuperVillain.reopen({ + secretLab: DS.belongsTo("secretLab", { polymorphic: true }) + }); + + env.registry.register('adapter:super-villain', DS.RESTAdapter); + env.registry.register('serializer:super-villain', TestSerializer.extend(DS.EmbeddedRecordsMixin, { + attrs: { + secretLab: { embedded: 'always' } + } + })); + var serializer = env.container.lookup("serializer:super-villain"); + + var json_hash = { + superVillain: { + id: "1", + firstName: "Tom", + lastName: "Dale", + secretLab: { + id: "1", + type: "bat-cave", + infiltrated: true + } + } + }; + + var json; + + run(function() { + json = serializer.normalizeResponse(env.store, SuperVillain, json_hash, '1', 'find'); + }); + + deepEqual(json, { + "data": { + "id": "1", + "type": "super-villain", + "attributes": { + "firstName": "Tom", + "lastName": "Dale" + }, + "relationships": { + "secretLab": { + "data": { "id": "1", "type": "bat-cave" } + } + } + }, + "included": [{ + "id": "1", + "type": "bat-cave", + "attributes": { + "infiltrated": true + }, + "relationships": {} + }] + }, "Primary has was correct"); +}); + +test("normalize with custom belongsTo primary key", function() { + env.registry.register('adapter:evil-minion', DS.RESTAdapter); + env.registry.register('serializer:evil-minion', TestSerializer.extend(DS.EmbeddedRecordsMixin, { + attrs: { + superVillain: { embedded: 'always' } + } + })); + env.registry.register('serializer:super-villain', TestSerializer.extend({ + primaryKey: 'custom' + })); + + var serializer = env.container.lookup("serializer:evil-minion"); + var json_hash = { + evil_minion: { + id: "1", + name: "Alex", + superVillain: { + custom: "1", + firstName: "Tom", + lastName: "Dale" + } + } + }; + var json; + + run(function() { + json = serializer.normalizeResponse(env.store, EvilMinion, json_hash, '1', 'find'); + }); + + deepEqual(json, { + "data": { + "id": "1", + "type": "evil-minion", + "attributes": { + "name": "Alex" + }, + "relationships": { + "superVillain": { + "data": { "id": "1", "type": "super-villain" } + } + } + }, + "included": [{ + "id": "1", + "type": "super-villain", + "attributes": { + "firstName": "Tom", + "lastName": "Dale" + }, + "relationships": {} + }] + }, "Primary hash was correct"); +});