猫鼬模型无法访问正在进行的连接

时间:2019-02-19 14:44:05

标签: mongodb express mongoose transactions

在当前的Express应用程序中,我想使用mongodb多文档事务的新功能。

首先重要的是指出我如何连接和处理模型

我的app.js(服务器)首先使用db.connect()连接到数据库。

我的db.index文件中需要所有模型。由于将使用相同的猫鼬参考来启动模型,因此我假设将来在不同路径中对模型的要求都指向已连接且相同的连接。如果我对这些假设有误,请纠正我。

我将连接引用保存在状态对象中,并在以后需要进行交易时返回

./ db / index.ts

const fs       = require('fs');
const path     = require('path');
const mongoose = require('mongoose');

const state = {
 connection = null,
}

// require all models
const modelFiles = fs.readdirSync(path.join(__dirname, 'models'));
modelFiles
.filter(fn => fn.endsWith('.js') && fn !== 'index.js')
.forEach(fn => require(path.join(__dirname, 'models', fn)));

const connect = async () => {
  state.connection = await mongoose.connect(.....);
  return;
}

const get = () => state.connection; 

module.exports = {
 connect,
 get,
}

我的模型文件包含我所需的架构

./ db / models / example.model.ts

const mongoose   = require('mongoose');
const Schema     = mongoose.Schema;
const ExampleSchema = new Schema({...);

const ExampleModel = mongoose.model('Example', ExampleSchema);
module.exports  = ExampleModel;

现在我尝试进行基本交易的路线。 F

./ routes / item.route.ts

const ExampleModel = require('../db/models/example.model');

router.post('/changeQty', async (req,res,next) => {

  const connection = db.get().connection;
  const session = await connection.startSession(); // works fine

  // start a transaction
  session.startTransaction(); // also fine

  const {someData} = req.body.data;

  try{

     // jsut looping that data and preparing the promises
     let promiseArr = [];
     someData.forEach(data => {
        // !!! THIS TRHOWS ERROR !!!
        let p = ExampleModel.findOneAndUpdate(
          {_id : data.id},
          {$incr : {qty : data.qty}},
          {new : true, runValidators : true}
        ).session(session).exec();
        promiseArr.push(p);
     })        

     // running the promises parallel
     await Promise.all(promiseArr);

     await session.commitTransaction();

    return res.status(..)....;

  }catch(err){
     await session.abortTransaction();
    // MongoError : Given transaction number 1 does not match any in-progress transactions.
    return res.status(500).json({err : err}); 
  }finally{
    session.endSession();
  }

})

但是我总是收到以下错误,这可能与我的模型的连接引用有关。我认为,他们无权访问启动会话的连接,因此他们不知道该会话。

  

MongoError:给定的交易号1与任何进行中的交易都不匹配   交易。

也许我需要用直接连接参考来初始化db.connect中的模型?

某处存在一个大错误,希望您能引导我走正确的道路。感谢您的帮助,

1 个答案:

答案 0 :(得分:1)

这是因为您正在并行执行操作:

所以您有很多比赛条件。只需使用async/await

并使您的生活更轻松。

let p = await ExampleModel.findOneAndUpdate(
              {_id : data.id},
              {$incr : {qty : data.qty}},
              {new : true, runValidators : true}
            ).session(session).exec();

参考:https://github.com/Automattic/mongoose/issues/7311

如果这不起作用,请尝试一个接一个地执行promise,而不是 promise.all()