Skip to content

Commit 4f7cf7b

Browse files
committed
[BUGFIX beta] serializePolymorphicType uses keyForPolymorphicType
Currently `keyForAttribute` is used to get the key under which the type of a polymorphic record is serialized. This is not correct, as the new `keyForPolymorphicType` hook should be used. This commit uses the new hook and falls back to the previous vesion, if the key generated via the old method (keyForAttribute) is different to the new version and the `keyForPolymorphicType` has not been overwritten. A deprecation warning is logged if this is the case.
1 parent 1d85e5c commit 4f7cf7b

File tree

2 files changed

+62
-3
lines changed

2 files changed

+62
-3
lines changed

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

+22-3
Original file line numberDiff line numberDiff line change
@@ -703,7 +703,7 @@ var RESTSerializer = JSONSerializer.extend({
703703

704704
/**
705705
You can use this method to customize how polymorphic objects are serialized.
706-
By default the JSON Serializer creates the key by appending `Type` to
706+
By default the REST Serializer creates the key by appending `Type` to
707707
the attribute and value from the model's camelcased model name.
708708
709709
@method serializePolymorphicType
@@ -714,11 +714,30 @@ var RESTSerializer = JSONSerializer.extend({
714714
serializePolymorphicType: function(snapshot, json, relationship) {
715715
var key = relationship.key;
716716
var belongsTo = snapshot.belongsTo(key);
717+
var typeKey = this.keyForPolymorphicType(key, relationship.type, 'serialize');
718+
719+
// old way of getting the key for the polymorphic type
717720
key = this.keyForAttribute ? this.keyForAttribute(key, "serialize") : key;
721+
key = `${key}Type`;
722+
723+
// The old way of serializing the type of a polymorphic record used
724+
// `keyForAttribute`, which is not correct. The next code checks if the old
725+
// way is used and if it differs from the new way of using
726+
// `keyForPolymorphicType`. If this is the case, a deprecation warning is
727+
// logged and the old way is restored (so nothing breaks).
728+
if (key !== typeKey && this.keyForPolymorphicType === RESTSerializer.prototype.keyForPolymorphicType) {
729+
Ember.deprecate("The key to serialize the type of a polymorphic record is created via keyForAttribute which has been deprecated. Use the keyForPolymorphicType hook instead.", false, {
730+
id: 'ds.rest-serializer.deprecated-key-for-polymorphic-type',
731+
until: '3.0.0'
732+
});
733+
734+
typeKey = key;
735+
}
736+
718737
if (Ember.isNone(belongsTo)) {
719-
json[key + "Type"] = null;
738+
json[typeKey] = null;
720739
} else {
721-
json[key + "Type"] = Ember.String.camelize(belongsTo.modelName);
740+
json[typeKey] = camelize(belongsTo.modelName);
722741
}
723742
},
724743

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

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

458+
test('serializeBelongsTo logs deprecation when old behavior for getting polymorphic type key is used', function() {
459+
var evilMinion, doomsdayDevice;
460+
var json = {};
461+
var expected = { evilMinion: '1', myCustomKeyType: 'evilMinion' };
462+
463+
env.restSerializer.keyForAttribute = function() {
464+
return 'myCustomKey';
465+
};
466+
467+
run(function() {
468+
evilMinion = env.store.createRecord('evil-minion', { id: 1, name: 'Tomster' });
469+
doomsdayDevice = env.store.createRecord('doomsday-device', { id: 2, name: 'Yehuda', evilMinion: evilMinion });
470+
});
471+
472+
expectDeprecation(function() {
473+
env.restSerializer.serializeBelongsTo(doomsdayDevice._createSnapshot(), json, { key: 'evilMinion', options: { polymorphic: true, async: true } });
474+
}, "The key to serialize the type of a polymorphic record is created via keyForAttribute which has been deprecated. Use the keyForPolymorphicType hook instead.");
475+
476+
deepEqual(json, expected, 'returned JSON is correct');
477+
});
478+
479+
test('keyForPolymorphicType can be used to overwrite how the type of a polymorphic record is serialized', function() {
480+
var evilMinion, doomsdayDevice;
481+
var json = {};
482+
var expected = { evilMinion: '1', typeForEvilMinion: 'evilMinion' };
483+
484+
env.restSerializer.keyForPolymorphicType = function() {
485+
return 'typeForEvilMinion';
486+
};
487+
488+
run(function() {
489+
evilMinion = env.store.createRecord('evil-minion', { id: 1, name: 'Tomster' });
490+
doomsdayDevice = env.store.createRecord('doomsday-device', { id: 2, name: 'Yehuda', evilMinion: evilMinion });
491+
});
492+
493+
env.restSerializer.serializeBelongsTo(doomsdayDevice._createSnapshot(), json, { key: 'evilMinion', options: { polymorphic: true, async: true } });
494+
495+
deepEqual(json, expected, 'returned JSON is correct');
496+
});
497+
458498
test('keyForPolymorphicType can be used to overwrite how the type of a polymorphic record is looked up for normalization', function() {
459499
var json = {
460500
doomsdayDevice: {

0 commit comments

Comments
 (0)