web-dev-qa-db-fra.com

Recherche imbriquée MongoDB avec 3 niveaux

J'ai besoin de récupérer l'intégralité de la hiérarchie d'objet unique de la base de données en tant que JSON. En fait, la proposition de toute autre solution pour atteindre ce résultat serait très appréciée. J'ai décidé d'utiliser MongoDB avec son support $ lookup.

J'ai donc trois collections:

fête

{ "_id" : "2", "name" : "party2" }
{ "_id" : "5", "name" : "party5" }
{ "_id" : "4", "name" : "party4" }
{ "_id" : "1", "name" : "party1" }
{ "_id" : "3", "name" : "party3" }    

adresse

{ "_id" : "a3", "street" : "Address3", "party_id" : "2" }
{ "_id" : "a6", "street" : "Address6", "party_id" : "5" }
{ "_id" : "a1", "street" : "Address1", "party_id" : "1" }
{ "_id" : "a5", "street" : "Address5", "party_id" : "5" }
{ "_id" : "a2", "street" : "Address2", "party_id" : "1" }
{ "_id" : "a4", "street" : "Address4", "party_id" : "3" }

addressComment

{ "_id" : "ac2", "address_id" : "a1", "comment" : "Comment2" }
{ "_id" : "ac1", "address_id" : "a1", "comment" : "Comment1" }
{ "_id" : "ac5", "address_id" : "a5", "comment" : "Comment6" }
{ "_id" : "ac4", "address_id" : "a3", "comment" : "Comment4" }
{ "_id" : "ac3", "address_id" : "a2", "comment" : "Comment3" }

Je dois récupérer toutes les parties avec toutes les adresses correspondantes et les commentaires d'adresse dans le cadre du dossier. Mon agrégation:

db.party.aggregate([{
    $lookup: {
        from: "address",
        localField: "_id",
        foreignField: "party_id",
        as: "address"
    }
},
{
    $unwind: "$address"
},
{
    $lookup: {
        from: "addressComment",
        localField: "address._id",
        foreignField: "address_id",
        as: "address.addressComment"
    }
}])

Le résultat est assez bizarre. Certains enregistrements sont ok. Mais Party avec _id 4 est manquant (il n'y a pas d'adresse pour cela). Il y a aussi deux Party _id 1 dans le jeu de résultats (mais avec des adresses différentes):

{
    "_id": "1",
    "name": "party1",
    "address": {
        "_id": "2",
        "street": "Address2",
        "party_id": "1",
        "addressComment": [{
            "_id": "3",
            "address_id": "2",
            "comment": "Comment3"
        }]
    }
}{
    "_id": "1",
    "name": "party1",
    "address": {
        "_id": "1",
        "street": "Address1",
        "party_id": "1",
        "addressComment": [{
            "_id": "1",
            "address_id": "1",
            "comment": "Comment1"
        },
        {
            "_id": "2",
            "address_id": "1",
            "comment": "Comment2"
        }]
    }
}{
    "_id": "3",
    "name": "party3",
    "address": {
        "_id": "4",
        "street": "Address4",
        "party_id": "3",
        "addressComment": []
    }
}{
    "_id": "5",
    "name": "party5",
    "address": {
        "_id": "5",
        "street": "Address5",
        "party_id": "5",
        "addressComment": [{
            "_id": "5",
            "address_id": "5",
            "comment": "Comment5"
        }]
    }
}{
    "_id": "2",
    "name": "party2",
    "address": {
        "_id": "3",
        "street": "Address3",
        "party_id": "2",
        "addressComment": [{
            "_id": "4",
            "address_id": "3",
            "comment": "Comment4"
        }]
    }
}

S'il vous plait, j'ai besoin de votre aide avec ceci. Je suis assez nouveau sur MongoDB mais je pense qu'il peut faire ce dont j'ai besoin.

25
Yuriy

La cause de vos "problèmes" est la deuxième étape d'agrégation - { $unwind: "$address" }. Il supprime l'enregistrement pour la fête avec _id: 4 (car son tableau d'adresses est vide, comme vous le mentionnez) et produit deux enregistrements pour les parties _id: 1 et _id: 5 (car chacun d'eux a deux adresses).

  • Pour empêcher la suppression de parties sans adresses, vous devez définir l'option preserveNullAndEmptyArrays de $unwind étape vers true.

  • Pour éviter la duplication de parties pour ses différentes adresses, vous devez ajouter $group étape d'agrégation vers votre pipeline. Utiliser aussi $project scène avec $filter opérateur pour exclure les enregistrements d'adresse vides en sortie.

db.party.aggregate([{
  $lookup: {
    from: "address",
    localField: "_id",
    foreignField: "party_id",
    as: "address"
  }
}, {
  $unwind: {
    path: "$address",
    preserveNullAndEmptyArrays: true
  }
}, {
  $lookup: {
    from: "addressComment",
    localField: "address._id",
    foreignField: "address_id",
    as: "address.addressComment",
  }
}, {
  $group: {
    _id : "$_id",
    name: { $first: "$name" },
    address: { $Push: "$address" }
  }
}, {
  $project: {
    _id: 1,
    name: 1,
    address: {
      $filter: { input: "$address", as: "a", cond: { $ifNull: ["$$a._id", false] } }
    } 
  }
}]);
43
Shad

Avec le mongodb 3.6 et au-dessus $lookup syntaxe il est assez simple de joindre des champs imbriqués sans utiliser $unwind .

db.party.aggregate([
  { "$lookup": {
    "from": "address",
    "let": { "partyId": "$_id" },
    "pipeline": [
      { "$match": { "$expr": { "$eq": ["$party_id", "$$partyId"] }}},
      { "$lookup": {
        "from": "addressComment",
        "let": { "addressId": "$_id" },
        "pipeline": [
          { "$match": { "$expr": { "$eq": ["$address_id", "$$addressId"] }}}
        ],
        "as": "address"
      }}
    ],
    "as": "address"
  }},
  { "$unwind": "$address" }
])
16
Ashh