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.
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] } }
}
}
}]);
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" }
])