MongoDB Aggregation : Double lookup, and merge lookup response to respective object

自闭症网瘾萝莉.ら 提交于 2020-03-23 08:01:07

问题


I'm trying an aggregation but I can't find the right pipeline to do it.

So, this is a part of my document model :

//company.js
{
    "_id" : "5dg8aa8c435b1e2868c841f6",
    "name" : "My Corp",
    "externalId" : "d7f348c9-c69b-69c4-923c-91458c53dc22",
    "professionals_customers" : [ 
        {
            "company" : "6f4d01eb3b948150c2aad9c0"
        }, 
        {
            "company" : "5dg7aa8c366b1e2868c841f6",
            "contact" : "5df8ab5c355b1e2999c841f7"
        }
    ],
}

I try to return the professionnal customers fields hydrated with data, like a classic populate would do.

Company field came from the company collection and contact is provided by the user collection

The desired output must look like :

{
"professionals_customers" : [ 
        {
            "company": {
                "_id": "6f4d01eb3b948150c2aad9c0",
                "name": "Transtar",
                "externalId": "d7f386c9-c79b-49c5-905c-90750c42dc22",
            },
        }, 
        {
            "company": {
                "_id": "5dg7aa8c366b1e2868c841f6",
                "name": "Aperture",
                "externalId": "d7f386c9-c69b-49c4-905c-90750c53dc22", 
            },
            "contact" : {
                "_id": "5df8ab5c355b1e2999c841f7",
                "firstname": "Caroline",
                "lastname": "Glados",
                "externalId": "d7f386c9-c69b-49c4-905c-90750c53dc22", //same externalId as above, the user belongs to the company
            },
        }
    ]
}

At this point I've tried multiple solutions but I can't reach my goal.

let query = [{
        $match : { _id : companyId }
    },{
        $lookup : {
            from: 'companies',
            localField : 'professionals_customers.company',
            foreignField : '_id',
            as : 'professionalsCustomers'
        }
    },{
        $lookup : {
            from: 'users',
            localField : 'professionals_customers.contact',
            foreignField : '_id',
            as : 'contacts'
        }
    }]

At this, point I' ve got two new arrays with all the needed informations, but I don't know how to get the right contact grouped with the right company. Also, maybe it's easier to try to populate the data (with $lookup) keeping the initial struct than trying to regroup professionalCustomers and contacts through the shared externalId.

Additional informations :

-An user that belongs to a company has the same externalId.

-I don't want to use a classical populate, after that, I need to do some other operations


回答1:


Try this query :

db.companies.aggregate([
    { $match: { _id: companyId } },
    { $unwind: "$professionals_customers" },
    {
        $lookup: {
            from: "companies",
            localField: "professionals_customers.company",
            foreignField: "_id",
            as: "professionals_customers.company"
        }
    },
    {
        $lookup: {
            from: "users",
            localField: "professionals_customers.contact",
            foreignField: "_id",
            as: "professionals_customers.contact"
        }
    },
    {
        $addFields: {
            "professionals_customers.company": {
                $arrayElemAt: ["$professionals_customers.company", 0]
            },
            "professionals_customers.contact": {
                $arrayElemAt: ["$professionals_customers.contact", 0]
            }
        }
    },
    {
        $group: { _id: "$_id", professionals_customers: { $push: "$professionals_customers" }, data: { $first: "$$ROOT" } }
    },
    { $addFields: { "data.professionals_customers": "$professionals_customers" } },
    { $replaceRoot: { newRoot: "$data" } }
])

Test : MongoDB-Playground

Note : If needed you need to convert fields/input which is of type string to ObjectId(). Basic thing is you need to check types of two fields being compared or input-to-field-in-DB matches or not.



来源:https://stackoverflow.com/questions/60548338/mongodb-aggregation-double-lookup-and-merge-lookup-response-to-respective-obj

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!