$ lookup для каждого элемента в подмассиве - PullRequest
0 голосов
/ 16 февраля 2019

Итак, в моей базе данных у меня есть 3 коллекции, и они выглядят так:

Клиенты:

customers = [
  {_id: 1, username: "jack", ... },
  {_id: 2, username: "jane", ... }
  ...
]

Отзывы:

reviews = [
  { _id: 1, customerID: 1, message: "my message", ...}
  ...
]

Комментарии:

comments = [
  { _id: 1, reviewID: 1, customerID: 2, message: "my response" ...}
  ...
]

Клиенты могут оставлять отзывы, а также комментировать другие отзывы.Итак, я хочу, чтобы mongodb aggregation запросил:

  1. Получите отзывы.

  2. Данные клиента, который сделалобзор.

  3. Комментарии к этому отзыву.

  4. Данные клиентов, которые оставили комментарий к этому отзыву.

т.е.

reviews = [
  {
    _id: 1,
    username: "jack",
    message: "my message"
    comments: [
      { _id: 1, username: "jane", message: "my response", ...},
      ...
    ]
    ...
  }
  ...
]

Ответы [ 2 ]

0 голосов
/ 16 февраля 2019

Вы можете использовать агрегацию ниже с mongodb 3.6 и выше

Reviews.aggregate([
  { "$lookup": {
    "from": Customers.collection.name,
    "let": { "customerID": "$customerID" },
    "pipeline": [
      { "$match": { "$expr": { "$eq": ["$_id", "$$customerID"] } } }
    ],
    "as": "customer"
  }},
  { "$unwind": "$customer" },
  { "$lookup": {
    "from": Comments.collection.name,
    "let": { "reviewID": "$_id" },
    "pipeline": [
      { "$match": { "$expr": { "$eq": ["$reviewID", "$$reviewID"] } } },
      { "$lookup": {
        "from": Customers.collection.name,
        "let": { "customerID": "$customerID" },
        "pipeline": [
          { "$match": { "$expr": { "$eq": ["$_id", "$$customerID"] } } }
        ],
        "as": "customer"
      }},
      { "$unwind": "$customer" },
    ],
    "as": "comments"
  }}
])
0 голосов
/ 16 февраля 2019

Вы можете начать с comments коллекции и $ lookup с customers, чтобы получить customer имя, затем вы можете $ group все комментарии по обзору и $ lookup дважды (с reviews и customer).Каждый раз, когда вы знаете, что это отношение один-к-одному, вы можете использовать $ unwind после $ lookup .Попробуйте:

db.comments.aggregate([
    {
        $lookup: {
            from: "customers",
            localField: "customerID",
            foreignField: "_id",
            as: "customer"
        }
    },
    {
        $unwind: "$customer"
    },
    {
        $project: {
            _id: 1,
            reviewID: 1,
            username: "$customer.username",
            message: 1
        }
    },
    {
        $group: {
            _id: "$reviewID",
            comments: { $push: { _id: "$_id", username: "$username", message: "$message" } }
        }
    },
    {
        $lookup: {
            from: "reviews",
            localField: "_id",
            foreignField: "_id",
            as: "review"
        }
    },
    {
        $unwind: "$review"
    },
    {
        $lookup: {
            from: "customers",
            localField: "review.customerID",
            foreignField: "_id",
            as: "customer"
        }
    },
    {
        $unwind: "$customer"
    },
    {
        $project: {
            _id: 1,
            message: "$review.message",
            username: "$customer.username",
            comments: 1
        }
    }
])

Выходы:

{ "_id" : 1, "comments" : [ { "_id" : 1, "username" : "jane", "message" : "my response" } ], "message" : "my message", "username" : "jack" }

РЕДАКТИРОВАТЬ: Если вы хотите начать с reviews и отфильтровать его для одного фильма, вы можете использовать $поиск с пользовательским конвейером

db.reviews.aggregate([
    {
        $match: {
            movieId: 1,
        }
    },
    {
        $lookup: {
            from: "customers",
            localField: "customerID",
            foreignField: "_id",
            as: "customer"
        }
    },
    {
        $unwind: "$customer"
    },
    {
        $lookup: {
            from: "comments",
            let: { reviewId: "$_id" },
            pipeline: [
                {
                    $match: { $expr: { $eq: [ "$$reviewId", "$reviewID" ] } }
                },
                {
                    $lookup: {
                        from: "customers",
                        localField: "customerID",
                        foreignField: "_id",
                        as: "customer"
                    }
                },
                {
                    $unwind: "$customer"
                },
                {
                    $project: {
                        _id: 1,
                        message: 1,
                        username: "$customer.username"
                    }
                }
            ],
            as: "comments"
        }
    },
    {
        $project: {
            _id: 1,
            message: 1,
            username: "$customer.username",
            comments: 1
        }
    }
])

Выводит тот же вывод

...