使用余烬数据的非CRUD操作

时间:2012-03-14 16:17:16

标签: javascript ember.js ember-data

说我有以下的ember-data模型:

App.Person = DS.Model.extend({
    firstName: DS.attr('string'),
    lastName:  DS.attr('string'),
    starred:   DS.attr('boolean')
});

这与使用以下非常标准的CRUD API的Rails应用程序进行通信:

GET    /people    - get a list of people
POST   /people    - create a new person
GET    /people/id - get a specific person
PUT    /people/id - update a specific person
DELETE /people/id - delete a specific person

这一切都使用标准存储/适配器映射到Ember-Data。

但是,我们可以说,为了让一个人“明星”或“明星”,API不允许我们通过标准更新操作来执行此操作。此操作有一个特定的API端点:

POST   /people/id/star   - mark a person as "starred"
POST   /people/id/unstar - mark a person as "unstarred"

如何在Ember Data中使用此API?

看起来我需要以某种方式扩展DS.Store和DS.RESTAdapter,但我不确定让他们了解这些不同操作的最佳方法。对于应用程序的通用适配器必须知道主演人员,这也感觉有点不对。

请注意,我无法控制API,因此我无法让POST /people/id知道“主演”,以便它符合标准更新。

5 个答案:

答案 0 :(得分:10)

有一段时间,当时可能无法实现,但您可以直接调用适配器的ajax方法:

YourApp.Store.prototype.adapter.ajax(url, action, {data: hashOfParams})

例如:

YourApp.Store.prototype.adapter.ajax('/products', 'GET', {data: {ids: [1,2,3]}})

对于你的问题:

YourApp.Store.prototype.adapter.ajax('/people' + id + '/star','POST')

编辑 - 使用buildURL很有用,特别是如果您在适配器上设置了命名空间:

url = YourApp.Store.prototype.adapter.buildURL('people',id) + '/star'

编辑2 - 您还可以使用container.lookup('adapter:application')获取适配器,如果您无法全局访问应用程序(例如ES6模块/ ember-cli),这将非常有用)

编辑3 - 以上是指Ember / Ember-CLI的过时版本。现在我在mixin中定义了这个函数 -

  customAjax: (method, type, id, action, hash = null) ->
    #note that you can now also get the adapter from the store -
    #adapter = @store.adapterFor('application')
    adapter = @container.lookup('adapter:application')
    url = adapter.buildURL(type, id) + '/' + action
    hash['data'] = $.extend({}, hash) if hash #because rails
    adapter.ajax(url, method, hash).then (result) =>
      return result

并称之为 -

@customAjax('PUT', 'modelClass', modelId, 'nonCRUDActionName', optionalHashOfAdditionalData).then (response) =>
  #do something with response

答案 1 :(得分:2)

我经历了同样的问题,并使用此处解释的修改版本解决了它: Non-crud rails actions in Ember.js

基本上,您需要使用Jquery AJAX调用,对我来说重要的部分是更改contentType以便在请求标头中发送auth数据,而不是在表单数据中发送(默认情况下,如果您在通话中包含数据)。另外,请确保在AJAX部分中提供适当的上下文,否则'这个'不会在“成功”中取得成功。回调:

App.ItemController = Ember.ObjectController.extend
    actions:
        starItem: ->
            controller = this
            item = @get 'model'
            id = item.id
            token = @auth.get 'authToken'
            $.ajax
                type: 'PUT'
                context: controller
                data: '{ "auth_token": "' + token + '"}'
                url: '/api/v1/items/' + id + '/star'
                contentType: 'application/json; charset=UTF-8'
                dataType: 'json'
                success: ->
                    if @get 'item.starred_by_user'
                        @set 'model.starred_by_user', false
                        item.decrementProperty('total_stars') 
                    else 
                        @set 'model.starred_by_user', true
                        item.incrementProperty('total_stars')

答案 2 :(得分:0)

也许是一个好的开始:https://github.com/beautifulnode/ember.request

还没有尝试过,但在你的情况下似乎很有希望......

答案 3 :(得分:0)

我认为,在CQRS中,star和unstar是命令。因此,您应该声明模型PersonStar并使用它。

答案 4 :(得分:-1)

在讨论组中,耶胡达提到这还不可能。