摘要:参考链接官方关于事务的介绍中文社区关于的介绍如果不想进行第二步,可以直接为你创建一个新的复制集我只是个前端啊,为什么要这么折磨我
主管前几天发现mongoDB已经升级到4.0了,迫不及待得让我实现他期待已久的事务回滚,发现还是有很多坑啊!
下面是我将已有的本地mongoDB升级到支持事务回滚的历程,分享出来,有错误的地方欢迎指正!
以mac为例哈
升级mongodb至4.0.0
$ brew upgrade mongodb
升级或安装mongodb.js v3.1.0 以上
$ npm i mongodb --save-dev
坑
升级完上述后,db.js里连接mongo时,
会警告,让你在connect的option里加一个字段
useNewUrlParser:true,
这时如果有用户验证,还需要在connect的option里再加一个字段
authSource:用户所在的db,一般为admin
否则会报验证失败,找不到用户的错误,比如
const mongoClient = await MongoClient.connect(mongoClientUrl, { auth: { user: config.dbUserName, password: config.dbUserPassword, }, authSource:"admin", useNewUrlParser:true, });2.将已有的数据库改造成复制集 目前事务回滚只能在复制集上操作,多带带的mongodb server是不能操作事务的
关掉所有的mongod
在平时启动mongod的命令后添加--replSet rs0,比如
$ mongod -dbpath ./db --port 27017 --replSet rs0
再开个shell,创一个不同端口的mongo实例,比如
$ mongod -dbpath ./db_repl --port 27018 --replSet rs0
连接27017的mongo实例,并设置
$ mongo $ rs.initiate() $ rs.add("localhost:27018");
完成
3.写回滚代码在db.js里新增了一个方法
export const getSession = async function() { return await state.mongoClient.startSession(); };
每次在需要回滚的mongo代码前调用这个方法拿到session,开始回滚的标记
const session = await db.getSession(); session.startTransaction({ readConcern: {level: "snapshot"}, writeConcern: {w: "majority"}, });
在每次调用mongodb.js的方法操作数据库时,都要带上session,比如
db .collection(this.collecitonName) .insertOne(doc,{session});
在你处理错误并觉得需要回滚了,执行
await session.abortTransaction();
在你觉得没问题,一起正常结束时,执行
await session.commitTransaction();
我的代码里可能有一些封装的代码没有放上去,导致不一定能理解。我只是举个栗子去实现,具体的代码实现可以看参考链接1
4.总结useNewUrlParser这个属性会在url里识别验证用户所需的db,未升级前是不需要指定的,升级到一定要指定,不管是在url后面,还是用authSource
事务回滚只能在复制集上操作,我猜测实现的原理可能是这样:先记录主节点的session,然后回滚的话,通过这个session查找副节点的数据快照,然后将这快照再应用到主节点上,实现回滚。当然,实际情况应该相当复杂,不然mongoDB也不会用3年时间来实现这个操作。
5.参考链接mongoDB官方关于事务的介绍
mongoDB中文社区关于replSet的介绍
如果不想进行第二步,run-rs可以直接为你创建一个新的mongoDB复制集
我只是个前端啊,为什么要这么折磨我QAQ文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。
转载请注明本文地址:https://www.ucloud.cn/yun/19295.html
摘要:最近自己在写一个小的项目,写的时候才发现自己会的东西太少了,总是遇到各种各样的坑。那么这个时候你就需要在里面添加一个手动回滚的机制了。这样就了,当然也可以通过去批量实现这种效果,只是暂时我还没有研究明白,所以就先记录这个最简单的了。 最近自己在写一个小的项目,写的时候才发现自己会的东西太少了,总是遇到各种各样的坑。 今天主要记录一下自己在写数据库存储的时候想到要是出现错误,是不是要回滚...
摘要:如上图所示,的实际上是已中间件的形式放在应用层,不用依赖数据库对协议的支持,完全剥离了分布式事务方案对数据库在协议支持上的要求。 微信公众号「后端进阶」,专注后端技术分享:Java、Golang、WEB框架、分布式中间件、服务治理等等。 在微服务架构体系下,我们可以按照业务模块分层设计,单独部署,减轻了服务部署压力,也解耦了业务的耦合,避免了应用逐渐变成一个庞然怪物,从而可以轻松扩展,...
阅读 3076·2023-04-26 00:53
阅读 3521·2021-11-19 09:58
阅读 1693·2021-09-29 09:35
阅读 3277·2021-09-28 09:46
阅读 3850·2021-09-22 15:38
阅读 2691·2019-08-30 15:55
阅读 3005·2019-08-23 14:10
阅读 3821·2019-08-22 18:17