javascript - 如果存在则更新多个,否则为每个不存在的 LeadId 创建一个新文档

标签 javascript node.js mongodb mongoose

考虑架构:

const mongoose = require("mongoose");
const Schema = mongoose.Schema;
const EightWeekGamePlanSchema = new Schema({
  Week: {
    type: Number,
    required: true
  },
  LeadId: {
    type: String,
    required: true
  },
  PackageId: {
    type: String,
    required: true
  },
  BusinessName: {
    type: String,
    required: true
  },
  PhoneNumberMasque: {
    type: String,
    required: true
  },
  City: {
    type: String,
    required: true
  },
  Rooms: {
    type: Number,
    required: true
  },
  LeadStartDate: {
    type: Date
  },
  LeadEndDate: {
    type: Date
  },

  TargetedToBeClaimedByClientType: {
    type: Number,
    required: true
  },
  TotalClaimsLeftToBeClaimedByClientType: {
    // incresed by 1 every time it's claimed
    type: Number,
    required: true
  },
  TotalClaimsToBeClaimedByClientType: {
    // Stays fixed
    type: Number,
    required: true
  },
  Status: {
    type: Number,
    required: true
  },

  InsertDate: {
    type: Date,
    default: Date.now
  }
});

module.exports = EightWeekGamePlan = mongoose.model(
  "eightweekgameplan",
  EightWeekGamePlanSchema
);

我正在尝试编写一个相当复杂的查询:如果 LeadID 出现在给定数组 winnerLeads< 中,则按 howManyClaims 中的给定数字更新多个文档 并且它们的 TargetedToBeClaimedByClientType 属性等于给定的 groupTarget :

router.post("/add-claims-to-group", auth, async (req, res) => {
    const { howManyClaims, groupTarget, winnerLeads, week } = req.body;
    EightWeekGamePlan.updateMany(
        {
          LeadId: {
            $in: winnerLeads
          },
          TargetedToBeClaimedByClientType: groupTarget
        },
        { $inc: { TotalClaimsToBeClaimedByClientType: howManyClaims } },
        { multi: true },
        (err, writeResult) => {
          if (err) {
            console.log(err);
          } else {
            console.log(writeResult);
          }
        }
      );
}

否则,执行某种 Create 操作:获取属于 LeadID 的现有 EightWeekGamePlan 文档之一,复制它并设置其 TotalClaimsToBeClaimedByClientType 属性与 howManyClaims

是否可以在一个查询中结合更新和创建操作?

最佳答案

根据您的查询,由于您正在执行 .updateMany(),因此您不必执行 { multi: true }。无论如何,通常您可以使用 {upset: true} 进行更新插入,但理想情况下,它会根据过滤条件创建一个新文档,并仅在数据库中找不到匹配项时才从输入查询中更新字段。但由于这里我们在过滤条件中有一个列表($in),它可能无法正常工作,请尝试以下操作:

let winnerLeads = [1, 2, 3, 31, 5]
let groupTarget = 1
let howManyClaims = 2
let bulkArr = []
for (i of winnerLeads) {
    bulkArr.push({
        updateOne: {
            "filter": {
                LeadId: i,
                TargetedToBeClaimedByClientType: groupTarget
            },
            // If you wanted it to be incremented rather than replace the field, then try `$inc` instead of `$set`.
            "update": { $set: { TotalClaimsToBeClaimedByClientType: howManyClaims } },
            "upsert": true
        }
    })
}
db.EightWeekGamePlan.bulkWrite(bulkArr);

收集数据:

/* 1 */
{
    "_id" : ObjectId("5e06eb8f400289966e00fac2"),
    "LeadId" : 1,
    "TotalClaimsToBeClaimedByClientType" : 1.0,
    "TargetedToBeClaimedByClientType" : 1
}

/* 2 */
{
    "_id" : ObjectId("5e06eb98400289966e00fb88"),
    "LeadId" : 2,
    "TotalClaimsToBeClaimedByClientType" : 1.0,
    "TargetedToBeClaimedByClientType" : 1
}

/* 3 */
{
    "_id" : ObjectId("5e06eba0400289966e00fc47"),
    "LeadId" : 3,
    "TotalClaimsToBeClaimedByClientType" : 0,
    "TargetedToBeClaimedByClientType" : 11
}

/* 4 */
{
    "_id" : ObjectId("5e06ebac400289966e00fd4b"),
    "LeadId" : 4,
    "TotalClaimsToBeClaimedByClientType" : 1,
    "TargetedToBeClaimedByClientType" : 11
}

/* 5 */
{
    "_id" : ObjectId("5e06ecef400289966e01273a"),
    "LeadId" : 5,
    "TotalClaimsToBeClaimedByClientType" : 1.0,
    "TargetedToBeClaimedByClientType" : 1
}

结果:

/* 1 */
{
    "_id" : ObjectId("5e06eb8f400289966e00fac2"),
    "LeadId" : 1,
    "TotalClaimsToBeClaimedByClientType" : 2.0,
    "TargetedToBeClaimedByClientType" : 1
}

/* 2 */
{
    "_id" : ObjectId("5e06eb98400289966e00fb88"),
    "LeadId" : 2,
    "TotalClaimsToBeClaimedByClientType" : 2.0,
    "TargetedToBeClaimedByClientType" : 1
}

/* 3 */
{
    "_id" : ObjectId("5e06eba0400289966e00fc47"),
    "LeadId" : 3,
    "TotalClaimsToBeClaimedByClientType" : 0,
    "TargetedToBeClaimedByClientType" : 11
}

/* 4 */
{
    "_id" : ObjectId("5e06ebac400289966e00fd4b"),
    "LeadId" : 4,
    "TotalClaimsToBeClaimedByClientType" : 1,
    "TargetedToBeClaimedByClientType" : 11
}

/* 5 */
{
    "_id" : ObjectId("5e06ecef400289966e01273a"),
    "LeadId" : 5,
    "TotalClaimsToBeClaimedByClientType" : 2,
    "TargetedToBeClaimedByClientType" : 1
}

/* 6 */
{
    "_id" : ObjectId("5e071eb1400289966e0597a0"),
    "TargetedToBeClaimedByClientType" : 1.0,
    "LeadId" : 3.0,
    "TotalClaimsToBeClaimedByClientType" : 2.0
}

/* 7 */
{
    "_id" : ObjectId("5e071e62400289966e059168"),
    "TargetedToBeClaimedByClientType" : 1.0,
    "LeadId" : 31.0,
    "TotalClaimsToBeClaimedByClientType" : 2.0
}

基本上bulkWrite除了写入结果外不会返回任何文档,您可以在数据库中验证更新操作结果,另外从上面的结果6得到插入为LeadId : 3 + TargetedToBeClaimedByClientType": 1.0(因此 LeadId:3 是重复的)组合不存在于数据库中并且 7 被插入,因为 LeadId : 31 是数据库中不存在,剩余 125TotalClaimsToBeClaimedByClientType 已更新。

引用: bulkWrite

关于javascript - 如果存在则更新多个,否则为每个不存在的 LeadId 创建一个新文档,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59506916/

相关文章:

javascript - Flow.js map 是不可写的吗?

php - 从 javascript 向 php 传递参数

javascript - PayPal Express Checkout.js - Checkout.js 中未定义的 indexOf

MongoDb - 如何使用 $set 更新嵌套对象的多个元素?

regex - 这两个 MongoDB $regex 语法有什么区别?

javascript - Google map JavaScript API 警告 : NoApiKeys

node.js - 使用 Node js 在 mongoDB 中创建一个新的对象 ID

javascript - 如何通过异步等待使用 map 数据

node.js - 从 ExpressJS 应用程序迁移到 ElectronJS 应用程序

arrays - 如何使用节点在 Mongoose 中获取深度嵌套数组中的对象