Ember-Data:"mappings"如何工作



我目前正在尝试将ember+emberdata+router+asp.net web api组合在一起。大多数似乎都能工作,但当ember数据试图通过我的模型的适配器findAll时,我收到了一条错误消息。

在我的后端,我有一个这样的模型(C#):

public class Genre {
    [Key]
    public int Id { get; set; }
    [Required]
    [StringLength(50, MinimumLength=3)]
    public string Name { get; set; }
}

在我的应用程序中,我使用ember数据来表示它:

App.Genre = DS.Model.extend({
    id: DS.attr("number"),
    name: DS.attr("string")
}).reopenClass({
    url: 'api/genre'
});

我还在我的应用程序中使用RESTAdapter定义了一个商店,如下所示:

App.store = DS.Store.create({
    revision: 4,
    adapter: DS.RESTAdapter.create({
        bulkCommit: false
    })
});

商店在我的控制器中使用如下:

App.GenreController = Ember.ArrayController.extend({
    content: App.store.findAll(App.Genre),
    selectedGenre: null
});

路由器被定义为

App.router = Em.Router.create({
    enableLogging: true,
    location: 'hash',
    root: Ember.Route.extend({
        //...
        genre: Em.Route.extend({
            route: '/genre',
            index: Ember.Route.extend({
                connectOutlets: function (router, context) {
                    router.get('applicationController').connectOutlet('genre');
                }
            })
        }),
        //...
    })
})

当我运行我的应用程序时,对于每个具有相同结构的对象,我都会得到以下消息:

未捕获错误:断言失败:您的服务器返回了一个哈希密钥0,但没有映射

作为参考,以下是服务返回的json:

[
  {
    "id": 1,
    "name": "Action"
  },
  {
    "id": 2,
    "name": "Drama"
  },
  {
    "id": 3,
    "name": "Comedy"
  },
  {
    "id": 4,
    "name": "Romance"
  }
]

我不能确切地说出问题是什么,由于断言提到我需要映射,我想知道:

  1. 这个映射是什么以及如何使用它
  2. 由于返回的json是一个数组,我应该在应用程序中使用不同类型的控制器吗?或者在ember数据中使用这种类型的json时,我应该知道什么吗?还是应该更改服务器中的JsonFormatter选项

欢迎任何帮助。

如果你觉得这还不足以理解问题,我肯定可以补充更多信息。

EDIT:我在后台更改了一些内容,现在我在服务器中的findAll()等效操作将输出序列化为以下json:

{
  "genres": [
      { "id": 1, "name": "Action" },
      { "id": 2, "name": "Drama" },
      { "id": 3, "name": "Comedy" },
      { "id": 4, "name": "Romance" }
   ]
}

但我仍然无法在客户端中填充我的模型,我的错误消息已更改为:

未捕获错误:断言失败:您的服务器返回了一个哈希关键流派,但你没有映射

不确定我还做错了什么。

引发此异常的方法是sideload,并检查如下映射:

sideload: function (store, type, json, root) {
        var sideloadedType, mappings, loaded = {};
        loaded[root] = true;
        for (var prop in json) {
            if (!json.hasOwnProperty(prop)) { continue; }
            if (prop === root) { continue; }
            sideloadedType = type.typeForAssociation(prop);
            if (!sideloadedType) {
                mappings = get(this, 'mappings');
                Ember.assert("Your server returned a hash with the key " + prop + " but you have no mappings", !!mappings);
//...

这个调用sideloadedType = type.typeForAssociation(prop);返回undefined,然后我得到错误消息。方法typeForAssociation()检查返回空Ember.Map的for 'associationsByName'密钥。

目前还没有解决方案。

顺便说一句

我现在的行动是这样的:

    // GET api/genres
    public object GetGenres() {
        return new { genres = context.Genres.AsQueryable() };
    }
    // GET api/genres
    //[Queryable]
    //public IQueryable<Genre> GetGenres()
    //{
    //    return context.Genres.AsQueryable();
    //}

我不得不删除原来的实现,它被json序列化了。NET,因为我找不到配置选项来生成EmberData所期望的json输出(如{resource_name : [json, json,...]}中所示)。这样做的副作用是我失去了内置的OData支持,但我想保留它。有人知道我如何配置它为集合生成不同的json吗?

映射可以在DS.RESTAdapter中定义

App.Store = DS.Store.extend({
  adapter: DS.RESTAdapter.create({
    bulkCommit: true,
    mappings: {
      genres: App.Genre
    },
    // you can also define plurals, if there is a unregular plural
    // usually, RESTAdapter simply add a 's' for plurals.
    // for example at work we have to define something like this
    plurals: {
      business_process: 'business_processes' 
      //else it tries to fetch business_processs
    }
  }),
  revision: 4
});

希望这能解决你的问题。

更新:

目前,这还没有很好的记录,我不记得是我们自己在阅读代码时发现的,还是Tom Dale指出的。
不管怎样,复数的意义就在这里对于映射,我想我们和你一样受到了同样的错误的驱使,要么我们尝试了,要么汤姆教我们这方面的知识。

RESTAdapter期望返回的JSON的形式为:

{
  "genres": [{
    "id": 1,
    "name": "action"
  },{
    "id": 2,
    "name": "Drama"
  }]
}

测试是一个很好的文档来源,请参阅https://github.com/emberjs/data/blob/master/packages/ember-data/tests/unit/rest_adapter_test.js#L315-329

我使用的是Ember Data rev. 11,而DS.RESTAdapter.create中的plurals配置似乎永远不起作用。我查看了代码,找到了如下解决方案:

App.Adapter = DS.RESTAdapter.extend({
  bulkCommit: false
})
App.Adapter.configure('plurals', {
  series: 'series'
})

最新更新