sql >> Database teknologi >  >> NoSQL >> MongoDB

Opslag med en række objekter

Du skal dybest set $unwind arrayet først. MongoDB kan endnu ikke arbejde med den "indre" egenskab for et objekt i et array som en kilde til $lookup .

For effektivitetens skyld bør vi virkelig bruge $concatArrays først for at "join" array-kilden, og derefter kun én $lookup operation:

Project.aggregate([
  { "$match": { "project_id": projectId} },
  { "$project": {
    "project_id": 1,
    "updated_at": 1,
    "created_at": 1,
    "owner": 1,
    "name": 1,
    "combined": {
      "$concatArrays": [
        { "$map": {
          "input": {
            "$filter": {
              "input": "$uploaded_files",
              "as": "uf",
              "cond": { "$eq": ["$$uf.upload_id", uploadId ] }
            }
          },
          "as": "uf",
          "in": {
            "$arrayToObject": {
              "$concatArrays": [
                { "$objectToArray": "$$uf" },
                [{ "k": "type", "v": "uploaded_files" }]
              ]
            }
          }
        }},
        { "$map": {
          "input": {
            "$filter": {
              "input": "$file_history",
              "as": "fh",
              "cond": { "$eq": ["$$fh.upload_id", uploadId ] }
            }
          },
          "as": "fh",
          "in": {
            "$arrayToObject": {
              "$concatArrays": [
                { "$objectToArray": "$$fh" },
                [{ "k": "type", "v": "file_history" }]
              ]
            }
          }
        }}
      ]
    }
  }},
  { "$unwind": "$combined" },
  { "$lookup": {
    "from": "files",
    "localField": "combined.file",
    "foreignField": "_id",
    "as": "combined.file"
  }},
  { "$unwind": "$combined.file" },
  { "$lookup": {
    "from": "users",
    "localField": "owner",
    "foreignField": "_id",
    "as": "owner"
  }},
  { "$unwind": "$owner" },
  { "$group": {
    "_id": "$_id",
    "project_id": { "$first": "$project_id" },
    "updated_at": { "$first": "$updated_at" },
    "created_at": { "$first": "$created_at" },
    "owner": { "$first": "$owner" },
    "name": { "$first": "$name" },
    "combined": { "$push": "$combined" }
  }},
  { "$project": {
    "project_id": 1,
    "updated_at": 1,
    "created_at": 1,
    "owner": 1,
    "name": 1,
    "uploaded_files": {
      "$filter": {
        "input": "$combined",
        "as": "cf",
        "cond": { "$eq": [ "$$cf.type", "uploaded_files" ] }
      }    
    },
    "file_history": {
      "$filter": {
        "input": "$combined",
        "as": "cf",
        "cond": { "$eq": [ "$$cf.type", "file_history" ] }
      }    
    }
  }}
])

I en nøddeskal

  1. Bring de to arrays sammen i kilden og tag dem, derefter $unwind først

  2. Udfør $lookup på den kombinerede detalje og $unwind det

  3. Udfør $lookup på den anden udenlandske kilde og $unwind det

  4. $group dokumentet sammen med et enkelt array.

  5. $filter ved feltet "tagnavne" eller "type" tilføjede vi for at "adskille" arrays.

Du kan følge den samme slags proces ved blot at bruge $unwind på hvert array, lav derefter "join" og gruppering igen. Men det kræver virkelig mange flere trin end blot at "kombinere" i første omgang.




  1. UpdateMany i MongoDB kører to gange med $inc

  2. Mongoose nægter at sende gyldig streng til ObjectId

  3. bruger $og med $match i mongodb

  4. Hvordan registrerer jeg AbstractMongoEventListener programmatisk?