Skip to content

Commit 1d85e5c

Browse files
committed
[BUGFIX beta] extract polymorphic belongsTo in RESTSerializer
This change adds the correct extraction of a polymorphic belongsTo specified in the payload in the following form: ``` js { id: 123, // ... message: 1, messageType: 'post' } ``` where the model is specified as: ``` js DS.Model.extend({ message: DS.belongsTo('message', { polymorphic: true }) }) ``` --- This commit introduces a new hook `keyForPolymorphicType` which can be overwritten to customize the key which holds the polymorphic type.
1 parent 921e8c9 commit 1d85e5c

File tree

3 files changed

+178
-1
lines changed

3 files changed

+178
-1
lines changed

packages/ember-data/lib/serializers/json-serializer.js

+32-1
Original file line numberDiff line numberDiff line change
@@ -585,6 +585,29 @@ export default Serializer.extend({
585585
return { id: coerceId(relationshipHash), type: relationshipModelName };
586586
},
587587

588+
/**
589+
Returns a polymorphic relationship formatted as a JSON-API "relationship object".
590+
591+
http://jsonapi.org/format/#document-resource-object-relationships
592+
593+
`relationshipOptions` is a hash which contains more information about the
594+
polymorphic relationship which should be extracted:
595+
- `resourceHash` complete hash of the resource the relationship should be
596+
extracted from
597+
- `relationshipKey` key under which the value for the relationship is
598+
extracted from the resourceHash
599+
- `relationshipMeta` meta information about the relationship
600+
601+
@method extractPolymorphicRelationship
602+
@param {Object} relationshipModelName
603+
@param {Object} relationshipHash
604+
@param {Object} relationshipOptions
605+
@return {Object}
606+
*/
607+
extractPolymorphicRelationship: function(relationshipModelName, relationshipHash, relationshipOptions) {
608+
return this.extractRelationship(relationshipModelName, relationshipHash);
609+
},
610+
588611
/**
589612
Returns the resource's relationships formatted as a JSON-API "relationships object".
590613
@@ -605,7 +628,15 @@ export default Serializer.extend({
605628
let data = null;
606629
let relationshipHash = resourceHash[relationshipKey];
607630
if (relationshipMeta.kind === 'belongsTo') {
608-
data = this.extractRelationship(relationshipMeta.type, relationshipHash);
631+
if (relationshipMeta.options.polymorphic) {
632+
// extracting a polymorphic belongsTo may need more information
633+
// than the type and the hash (which might only be an id) for the
634+
// relationship, hence we pass the key, resource and
635+
// relationshipMeta too
636+
data = this.extractPolymorphicRelationship(relationshipMeta.type, relationshipHash, { key, resourceHash, relationshipMeta });
637+
} else {
638+
data = this.extractRelationship(relationshipMeta.type, relationshipHash);
639+
}
609640
} else if (relationshipMeta.kind === 'hasMany') {
610641
data = Ember.isNone(relationshipHash) ? null : relationshipHash.map((item) => this.extractRelationship(relationshipMeta.type, item));
611642
}

packages/ember-data/lib/serializers/rest-serializer.js

+75
Original file line numberDiff line numberDiff line change
@@ -55,6 +55,37 @@ var camelize = Ember.String.camelize;
5555
*/
5656
var RESTSerializer = JSONSerializer.extend({
5757

58+
/**
59+
`keyForPolymorphicType` can be used to define a custom key when
60+
serializing and deserializing a polymorphic type. By default, the
61+
returned key is `${key}Type`.
62+
63+
Example
64+
65+
```app/serializers/post.js
66+
import DS from 'ember-data';
67+
68+
export default DS.RESTSerializer.extend({
69+
keyForPolymorphicType: function(key, relationship) {
70+
var relationshipKey = this.keyForRelationship(key);
71+
72+
return 'type-' + relationshipKey;
73+
}
74+
});
75+
```
76+
77+
@method keyForPolymorphicType
78+
@param {String} key
79+
@param {String} typeClass
80+
@param {String} method
81+
@return {String} normalized key
82+
*/
83+
keyForPolymorphicType: function(key, typeClass, method) {
84+
var relationshipKey = this.keyForRelationship(key);
85+
86+
return `${relationshipKey}Type`;
87+
},
88+
5889
/**
5990
Normalizes a part of the JSON payload returned by
6091
the server. You should override this method, munge the hash
@@ -689,6 +720,50 @@ var RESTSerializer = JSONSerializer.extend({
689720
} else {
690721
json[key + "Type"] = Ember.String.camelize(belongsTo.modelName);
691722
}
723+
},
724+
725+
/**
726+
You can use this method to customize how a polymorphic relationship should
727+
be extracted.
728+
729+
@method extractPolymorphicRelationship
730+
@param {Object} relationshipType
731+
@param {Object} relationshipHash
732+
@param {Object} relationshipOptions
733+
@return {Object}
734+
*/
735+
extractPolymorphicRelationship: function(relationshipType, relationshipHash, relationshipOptions) {
736+
var { key, resourceHash, relationshipMeta } = relationshipOptions;
737+
738+
// A polymorphic belongsTo relationship can be present in the payload
739+
// either in the form where the `id` and the `type` are given:
740+
//
741+
// {
742+
// message: { id: 1, type: 'post' }
743+
// }
744+
//
745+
// or by the `id` and a `<relationship>Type` attribute:
746+
//
747+
// {
748+
// message: 1,
749+
// messageType: 'post'
750+
// }
751+
//
752+
// The next code checks if the latter case is present and returns the
753+
// corresponding JSON-API representation. The former case is handled within
754+
// the base class JSONSerializer.
755+
var isPolymorphic = relationshipMeta.options.polymorphic;
756+
var typeProperty = this.keyForPolymorphicType(key, relationshipType, 'deserialize');
757+
758+
if (isPolymorphic && resourceHash.hasOwnProperty(typeProperty) && typeof relationshipHash !== 'object') {
759+
let type = this.modelNameFromPayloadKey(resourceHash[typeProperty]);
760+
return {
761+
id: relationshipHash,
762+
type: type
763+
};
764+
}
765+
766+
return this._super(...arguments);
692767
}
693768
});
694769

packages/ember-data/tests/integration/serializers/rest-serializer-test.js

+71
Original file line numberDiff line numberDiff line change
@@ -455,6 +455,41 @@ test('serializeBelongsTo with async polymorphic', function() {
455455
deepEqual(json, expected, 'returned JSON is correct');
456456
});
457457

458+
test('keyForPolymorphicType can be used to overwrite how the type of a polymorphic record is looked up for normalization', function() {
459+
var json = {
460+
doomsdayDevice: {
461+
id: '1',
462+
evilMinion: '2',
463+
typeForEvilMinion: 'evilMinion'
464+
}
465+
};
466+
467+
var expected = {
468+
data: {
469+
type: 'doomsday-device',
470+
id: '1',
471+
attributes: {},
472+
relationships: {
473+
evilMinion: {
474+
data: {
475+
type: 'evil-minion',
476+
id: '2'
477+
}
478+
}
479+
}
480+
},
481+
included: []
482+
};
483+
484+
env.restSerializer.keyForPolymorphicType = function() {
485+
return 'typeForEvilMinion';
486+
};
487+
488+
var normalized = env.restSerializer.normalizeResponse(env.store, DoomsdayDevice, json, null, 'findRecord');
489+
490+
deepEqual(normalized, expected, 'normalized JSON is correct');
491+
});
492+
458493
test('serializeIntoHash uses payloadKeyFromModelName to normalize the payload root key', function() {
459494
run(function() {
460495
league = env.store.createRecord('home-planet', { name: "Umber", id: "123" });
@@ -475,6 +510,42 @@ test('serializeIntoHash uses payloadKeyFromModelName to normalize the payload ro
475510
});
476511
});
477512

513+
test('normalizeResponse with async polymorphic belongsTo, using <relationshipName>Type', function() {
514+
env.registry.register('serializer:application', DS.RESTSerializer.extend());
515+
var store = env.store;
516+
env.adapter.findRecord = (store, type) => {
517+
if (type.modelName === 'doomsday-device') {
518+
return {
519+
doomsdayDevice: {
520+
id: 1,
521+
name: "DeathRay",
522+
evilMinion: 1,
523+
evilMinionType: 'yellowMinion'
524+
}
525+
};
526+
}
527+
528+
equal(type.modelName, 'yellow-minion');
529+
530+
return {
531+
yellowMinion: {
532+
id: 1,
533+
type: 'yellowMinion',
534+
name: 'Alex',
535+
eyes: 3
536+
}
537+
};
538+
};
539+
540+
run(function() {
541+
store.findRecord('doomsday-device', 1).then((deathRay) => {
542+
return deathRay.get('evilMinion');
543+
}).then((evilMinion) => {
544+
equal(evilMinion.get('eyes'), 3);
545+
});
546+
});
547+
});
548+
478549
test('normalizeResponse with async polymorphic belongsTo', function() {
479550
env.registry.register('serializer:application', DS.RESTSerializer.extend({
480551
isNewSerializerAPI: true

0 commit comments

Comments
 (0)