2

nano我有一个 nodejs 应用程序,我使用以下脚本连接到我的 couchdb :

const { connectionString } = require('../config');

const nano = require('nano')(connectionString);

// creates database or fails silent if exists
nano.db.create('foo');

module.exports = {
  foo: nano.db.use('foo')
}

该脚本在每次服务器启动时运行,因此它会在每次服务器(重新)启动时尝试创建数据库“foo”,如果数据库已经存在,它只会静默失败。

我非常喜欢这个想法,因为这样我实际上是在应用程序级别维护数据库,并且在我决定添加新数据库时不必手动创建数据库。

将这种方法更进一步,我还尝试从应用程序级别维护我的设计文档。

... 
nano.db.create('foo');

const foo = nano.db.use('foo');

const design = {
  _id: "_design/foo",
  views: {
    by_name: {
      map: function(doc) {
        emit(doc.name, null);
      }
    }
  }
}

foo.insert(design, (err) => {
  if(err)
    console.log('design insert failed');
})

module.exports = {
  foo
}

显然,如果它不存在,这只会插入设计文档。但是如果我更新了我的设计文档并想要更新它怎么办?

我试过了:

foo.get("_design/foo", (err, doc) => {
  if(err)
    return foo.insert(design);

  design._rev = doc._rev
  foo.insert(design);
})

现在的问题是每次服务器重新启动时都会更新设计文档(例如,每次重新启动时都会获得一个新的_rev)。

现在...我的问题:)

1:这是用数据库和设计引导我的 CouchDB 的糟糕方法吗?我应该考虑将一些迁移步骤作为我的部署过程的一部分吗?

2:我的设计文档有很多_revs,基本上是每次部署和服务器重启,这是一个问题吗?即使文件本身没有改变?如果是这样,有没有办法只在文档更改时才更新文档?(我想在我的应用程序中手动将 _rev 设置为某个值,但非常不确定这是否是个好主意)。

4

3 回答 3

2

我尝试了像@Bernhard Gschwantner 建议的 md5 比较。但是我遇到了一些困难,因为我想在我的代码中用纯 javascript 在设计文档中编写 map/reduce 函数。

const design = {
  _id: "_design/foo",
  views: {
    by_name: {
      map: function(doc) {
        emit(doc.name, null);
      }
    }
  }
}

从 CouchDb 获取设计文档时,会返回转换为字符串的 map/reduce 函数:

  ...
  "by_name": {
    "map": "function (doc) {\n        emit(doc.name, null);\n      }"
  },
  ...

显然 md5 比较在这里不起作用。

我最终得到了一个非常简单的解决方案,只需在设计文档上添加一个版本号:

const design = {
  _id: "_design/foo",
  version: 1,
  views: {
    by_name: {
      map: function(doc) {
        emit(doc.name, null);
      }
    }
  }
}

当我更新设计文档时,我只是增加版本号并将其与数据库中的版本号进行比较:

const fooDesign = {...}
foo.get('_design/foo', (err, design) => {
  if(err)
    return foo.insert(fooDesign);

  console.log('comparing foo design version', design.version, fooDesign.version);
  if(design.version !== fooDisign.version) {
    fooDesign._rev = design._rev;
    foo.insert(fooDesign, (err) => {
      if(err)
        return console.log('error updating foo design', err);
      console.log('foo design updated to version', fooDesign.version)
    });
  }
});
于 2016-07-16T12:26:11.377 回答
2
  1. 你的方法似乎很合理。如果检查仅在重新启动时发生,这甚至不会是性能问题。
  2. 太多_rev的 s可能会成为一个问题。_revs 的历史记录_revs_info与文档本身一样保存并存储(有关详细信息,请参阅 CouchDB 文档)。根据您的设置,创建不必要的修订可能是一个错误的决定。

对于一些需要特定视图的服务器端脚本,我们也遇到了类似的挑战。我们的解决方案是计算新旧设计文档的哈希值并进行比较。您可以为此作业使用任何散列函数,例如sha1md5。请记住在散列之前从旧文档中删除 _rev,否则您每次都会得到不同的散列值。

于 2016-07-15T11:18:54.843 回答
1

再次回顾您的问题:在最近的一个项目中,我使用了 Johannes Schmidt 的出色couchdb-push模块。您可以免费获得条件更新,以及从其依赖项继承的许多 其他 好处couchdb-compile

那个图书馆对我来说是一个隐藏的宝石。强烈推荐!

于 2017-01-05T19:56:38.430 回答