sql >> Databasteknik >  >> NoSQL >> MongoDB

Slå upp med en rad objekt

Du behöver i princip $unwind arrayen först. MongoDB kan ännu inte arbeta med den "inre" egenskapen för ett objekt inom en array som en källa för $lookup .

För effektivitetens skull borde vi verkligen använda $concatArrays först för att "ansluta" till arraykällan och gör sedan bara en $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 ett nötskal

  1. Ta samman de två arrayerna på källkod och tagga dem, sedan $unwind först

  2. Gör $lookup på den kombinerade detaljen och $unwind det

  3. Gör $lookup på den andra utländska källan och $unwind det

  4. $group dokumentet tillsammans med en enda array.

  5. $filter genom fältet "taggnamn" eller "typ" la vi till för att "separera" arrayerna.

Du kan följa samma typ av process genom att helt enkelt använda $unwind på varje array, gör sedan "join" och gruppera igen. Men egentligen krävs det mycket fler steg än att bara "kombinera" i första hand.




  1. MongoDB:Kan inte ansluta till ny replikuppsättningsmaster

  2. Rails, Sidekiq - Redis NOAUTH

  3. Räkna i Spring Data MongoDB-förrådet

  4. När du ska använda CouchDB över MongoDB och vice versa