Ember-Data: как работают "отображения"

В настоящее время я пытаюсь что-то соединить с веб-интерфейсом ember + emberdata + router + asp.net. Кажется, что большинство из них работает, но я застрял в сообщении об ошибке, которое я получаю, когда ember-data пытается findAll через адаптер для моих моделей.

В моем бэкэнде у меня есть такая модель (C#):

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

Который в моем приложении я представляю так, используя ember-data:

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');
                }
            })
        }),

        //...
    })
})

Когда я запускаю свое приложение, я получаю следующее сообщение для каждого объекта, имеющего такую ​​же структуру:

Uncaught Error: утверждение не выполнено: ваш сервер вернул хэш с ключом 0, но у вас нет сопоставлений

Для справки вот json, который возвращает сервис:

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

Я не могу точно сказать, в чем проблема, и так как утверждение говорит о том, что мне нужно отображение, я хотел бы знать:

  1. Что это за отображение и как его использовать.
  2. Поскольку возвращаемый json является массивом, должен ли я использовать в своем приложении контроллер другого типа, или есть что-то, о чем я должен знать при работе с этим типом json в ember-data? или я должен изменить параметры JsonFormatter на сервере?

Любая помощь приветствуется.

Я определенно могу добавить больше информации, если вы чувствуете, что этого недостаточно для понимания проблемы.

РЕДАКТИРОВАТЬ: я изменил несколько вещей в моем бэкэнде и теперь мой findAll() Эквивалентное действие на сервере сериализует вывод как следующий json:

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

Но я все еще не могу заставить его заполнить мои модели в клиенте, и мое сообщение об ошибке изменилось на это:

Uncaught Ошибка: утверждение не удалось: ваш сервер возвратил хэш с ключевыми жанрами, но у вас нет сопоставлений

Не уверен, что еще я могу делать не так.

Метод, который выбрасывает это исключение: 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() чеки для 'associationsByName' ключ, который возвращает пустой Ember.Map,

До сих пор нет решения для этого на данный момент.

Кстати...

Мое действие теперь так:

    // 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, так как я не смог найти параметры конфигурации для создания вывода json, как ожидает Ember-Data (как в {resource_name : [json, json,...]}). Побочным эффектом этого является то, что я потерял встроенную поддержку OData, но я бы хотел оставить ее. Кто-нибудь знает, как я могу настроить его для производства различных JSON для коллекции?

3 ответа

Решение

Сопоставление может быть определено в 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
});

Надеюсь, что это решит вашу проблему.

Обновить:

В настоящее время это плохо документировано, я не помню, нашли ли мы это сами, читая код, или, возможно, Том Дейл указал на это.
Во всяком случае, здесь есть смысл для множественного числа. Для отображений, я думаю, что мы были вызваны той же ошибкой, что и вы, и либо мы пытались, либо Том учил нас этому.

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 и кажется, что plurals конфиг в DS.RESTAdapter.create никогда не работает. Я посмотрел на коды и нашел решение следующим образом:

App.Adapter = DS.RESTAdapter.extend({
  bulkCommit: false
})

App.Adapter.configure('plurals', {
  series: 'series'
})
Другие вопросы по тегам