使用带有couchdb的ember-resource - 如何保存我的文档?

时间:2012-05-24 21:18:11

标签: javascript couchdb ember.js

我正在使用ember.js和couchdb实现一个应用程序。我选择ember-resource作为数据库访问层,因为它很好地支持嵌套的JSON文档。

由于couchdb在每个文档中使用_rev属性进行乐观锁定,因此在将数据保存到couchdb之后,必须在我的应用程序中更新此属性。

我实现这一点的想法是在保存到数据库后立即重新加载数据,然后用文档的其余部分重新获取新的_rev。

这是我的代码:

// Since we use CouchDB, we have to make sure that we invalidate and re-fetch
// every document right after saving it. CouchDB uses an optimistic locking
// scheme based on the attribute "_rev" in the documents, so we reload it in
// order to have the correct _rev value.
didSave:     function() {
    this._super.apply(this, arguments);
    this.forceReload();
},

// reload resource after save is done, expire to make reload really do something
forceReload: function() {
    this.expire();  // Everything OK up to this location
    Ember.run.next(this, function() {
        this.fetch()  // Sub-Document is reset here, and *not* refetched!
            .fail(function(error) {
                App.displayError(error);
            })
            .done(function() {
                App.log("App.Resource.forceReload fetch done, got revision " + self.get('_rev'));
            });
    });
}

这适用于大多数情况,但如果我有嵌套模型,则在执行fetch之前,子模型将替换为旧版本的数据!

有趣的是,正确的(更新的)数据存储在数据库中,并且在获取之后错误的(旧的)数据在内存模型中,尽管_rev属性是正确的(以及主对象的所有属性)

以下是我的对象定义的一部分:

App.TaskDefinition = App.Resource.define({

url: App.dbPrefix + 'courseware',

schema: {
    id:          String,
    _rev:        String,
    type:        String,
    name:       String,
    comment:    String,
    task:        {
        type:   'App.Task',
        nested: true
    }
}
});

App.Task = App.Resource.define({

schema: {
    id: String,
    title:       String,
    description: String,

    startImmediate: Boolean,
    holdOnComment: Boolean,
    ..... // other attributes and sub-objects
}
});

可能存在问题的任何想法?

非常感谢任何建议!

亲切的问候, 托马斯

1 个答案:

答案 0 :(得分:5)

我找到了一个更好的解决方案,可以从数据库中恢复新版本,而无需在每次保存后重新加载数据。

如果在save中使用更新选项调用ember-resource,则来自db的响应将合并到资源对象中。 couchdb对保存请求的回答类似于

{"ok":true,"id":"mydocument-123","rev":"10-5f3cb46df301143a966251148f88663d"}

所以我只是将对象的_rev属性设置为db中的rev值。

我的应用程序特定资源类作为所有couchdb对象的超类如下:

App.Resource = Ember.Resource.extend({

    save: function(options) {
        options = options || {};
        options.update = true; // sets id and rev after saving in ember-resource; rev is copied to _rev in didSave callback!
        this.set('rev', undefined);
        return this._super(options);
    },

    // we get the new revision in the "rev" attribute of the response from the save-request.
    // since we set options.update to true, we get this value in the attribute "rev", which we simply copy to "_rev"
    didSave:     function() {
        this._super.apply(this, arguments);
        this.set('_rev', this.get('rev'));
    }

    .... // other properties and functions
});

资源定义如下:

App.TaskExecution = App.Resource.define({

    ....

    schema: {

        id:   String,
        _rev: String,
        rev:  String,
        type: String,
        .....  // other attributes
    }

});

到目前为止工作得很好....