带有事务的 Mongock 迁移失败

Mongock migration failure with transaction

我正在 运行 启动一个 spring boot 2.4.3 应用程序,我有一个支持事务的 mongodb 4 集群(我 运行 它在本地使用 运行-rs)。我也在使用 spring 数据 mongodb。我正在尝试为数据库迁移集成 mongock(最新版本 - 4.3.8),但我遇到了一个我无法解决的问题。正如我们所知,最新版本的 mongock 默认使用事务。我有以下变更集,它只是为我的一个实体创建了一个 mongodb 集合。

@ChangeSet(order = "001", id = "initSessionCollection", author = "Hristo")
public void init(MongockTemplate mongockTemplate) {
  mongockTemplate.createCollection(Session.class);
}

当我 运行 应用程序迁移失败并出现以下错误时:

2021-04-13 16:36:18.092  WARN 70141 --- [           main] c.g.c.m.d.m.s.v.SpringDataMongoV3Driver  : Error in Mongock's transaction

com.github.cloudyrock.mongock.exception.MongockException: Error in method[InitDb.init] : Command failed with error 251 (NoSuchTransaction): 'Given transaction number 1 does not match any in-progress transactions.' on server localhost:27017. The full response is {"errorLabels": ["TransientTransactionError"], "operationTime": {"$timestamp": {"t": 1618320977, "i": 3}}, "ok": 0.0, "errmsg": "Given transaction number 1 does not match any in-progress transactions.", "code": 251, "codeName": "NoSuchTransaction", "$clusterTime": {"clusterTime": {"$timestamp": {"t": 1618320977, "i": 3}}, "signature": {"hash": {"$binary": {"base64": "AAAAAAAAAAAAAAAAAAAAAAAAAAA=", "subType": "00"}}, "keyId": 0}}}; nested exception is com.mongodb.MongoCommandException: Command failed with error 251 (NoSuchTransaction): 'Given transaction number 1 does not match any in-progress transactions.' on server localhost:27017. The full response is {"errorLabels": ["TransientTransactionError"], "operationTime": {"$timestamp": {"t": 1618320977, "i": 3}}, "ok": 0.0, "errmsg": "Given transaction number 1 does not match any in-progress transactions.", "code": 251, "codeName": "NoSuchTransaction", "$clusterTime": {"clusterTime": {"$timestamp": {"t": 1618320977, "i": 3}}, "signature": {"hash": {"$binary": {"base64": "AAAAAAAAAAAAAAAAAAAAAAAAAAA=", "subType": "00"}}, "keyId": 0}}}

如果我按预期禁用了 mongock 的事务,则迁移应用成功。我不想禁用数据库迁移的事务,但我不知道问题出在哪里。

编辑:我发现了问题所在 - 因为 Spring 引导包含多文档事务,并且此类事务不支持创建集合命令。但是我应该如何处理集合的创建?

如您所述,MongoDB 事务模型有一些局限性。请看官方MongoDB documentation.

考虑到这一点,您不能将该操作包装在事务中,因此如果您正在使用事务,则不能在标准 ChangeLog 中使用它。

不过,好消息是,一如既往:D,Mongock 团队考虑到了这一点,并将提供一种预交易操作机制。

我们预计将在 6 月左右准备好发布候选版本。但是,如果这是紧急需求,请给我们发送电子邮件至 dev@cloudyrock.io,我们将设法找到解决方案。