Ember.js REST Adapter without JSON root

2020-02-01 08:07发布

The Ember.js REST Adapter expects the JSON to be returned as:

{
    "person": {
        "first_name": "Barack",
        "last_name": "Obama",
        "is_person_of_the_year": true
    }
}

But my API returns the data without a root element:

{
    "first_name": "Barack",
    "last_name": "Obama",
    "is_person_of_the_year": true
}

Is it possible to customize the REST Adapter so that it accepts my JSON data? Right now it's showing "Assertion failed: Your server returned a hash with the key 0 but you have no mapping for it"

UPDATE: Based on Sherwin Yu's answer below, this is what I came up with, seems to work so far: https://gist.github.com/richardkall/5910875

5条回答
祖国的老花朵
2楼-- · 2020-02-01 08:43

You could also normalize it into something ember would expect.

App.PersonSerializer = DS.RESTSerializer.extend({
  normalizePayload: function(type, payload) {
    var typeKey = type.typeKey;
    return {
      typeKey: payload
    }
  }
});
查看更多
家丑人穷心不美
3楼-- · 2020-02-01 08:50

Yes, you can write your own custom REST adapter. Take a look at the source code in the JSONSerializer, RESTSerializer (which extends the JSONSerializer), and the REST adapter.

Basically, the you need to override the extract* methods from the JSONSerializer.

Currently, it looks something like this:

extract: function(loader, json, type, record) {
  var root = this.rootForType(type);

  this.sideload(loader, type, json, root);
  this.extractMeta(loader, type, json);

  if (json[root]) {
    if (record) { loader.updateId(record, json[root]); }
    this.extractRecordRepresentation(loader, type, json[root]);
  }
},

Notice how it checks json[root] -- you'd have to write your custom method based on your expected API response.

Another approach would be to "preprocess" the json from the API to use a root element. You could do this by finding out what methods call extract* (which passes it the json) and before it does so, modify the json to contain the root element.

Hope this helps, please let me know if it's unclear.

查看更多
Ridiculous、
4楼-- · 2020-02-01 08:56

The easiest way is to not use the RESTSerializer but the much simpler JSONSerializer, which does not expect a root element.

Good resources on understanding which serializer to use for a given API can be found in these two blog posts:

http://thejsguy.com/2015/12/05/which-ember-data-serializer-should-i-use.html
http://emberigniter.com/fit-any-backend-into-ember-custom-adapters-serializers/

查看更多
ゆ 、 Hurt°
5楼-- · 2020-02-01 08:57

I solved this by extending DS.RESTSerializer. extractArray method needs to be overloaded when server response is array type.

App.PersonSerializer = DS.RESTSerializer.extend({
    extractSingle: function (store, type, payload, id) {
        var wrappedObj = {};
        wrappedObj[type.typeKey] = payload;
        return this._super(store, type, wrappedObj, id);
    }});
查看更多
仙女界的扛把子
6楼-- · 2020-02-01 09:01

I tried to add root to json. This works for me:

App.Adapter = DS.RESTAdapter.extend({
    findAll: function(store, type, since) {
        var root, adapter;
            var plural; // Insert this line
        root = this.rootForType(type);
        adapter = this;
            plural = this.pluralize(root); // Insert this line
        return this.ajax(
                this.buildURL(root), 
                "GET",
                {
                    data: this.sinceQuery(since)
                }).then(function(json) {
                    eval ( "json = {" + plural + " : json }" ); // Insert this line
                    adapter.didFindAll(store, type, json);
                }).then(null, DS.rejectionHandler);
    }
});
查看更多
登录 后发表回答