Обновить объект внутри вложенного массива в Mongodb - PullRequest
1 голос
/ 25 февраля 2020

У меня есть документ, который выглядит следующим образом. Где мне нужно обновить указанный c объект внутри вложенного массива. Мне нужно отредактировать текстовый объект с именем , который имеет локаль из en .

[
    {
        "_id": ObjectId("5e049ebc8e935c407f78c190"),
        "source": "homepage",
        "url": [
            {
                "type": "admindsg",
                "text": [
                    {
                        "locale": "en",
                        "name": "Admin DSG"
                    },
                    {
                        "locale": "nb",
                        "name": "Admin DSG"
                    },
                    {
                        "locale": "li",
                        "name": "Admin DSG"
                    },
                    {
                        "locale": "fi",
                        "name": "Admin DSG"
                    }
                ],
                "value": [
                    {
                        "locale": "en",
                        "link": "https://www.facebook.cloud"
                    },
                    {
                        "locale": "nb",
                        "link": "https://www.facebook.cloud"
                    },
                    {
                        "locale": "li",
                        "link": "https://www.facebook.cloud"
                    },
                    {
                        "locale": "fi",
                        "link": "https://www.facebook.cloud"
                    }
                ],
                "datetime": "2020-02-08 13:36:37"
            },
            {
                "type": "dataauth",
                "text": [
                    {
                        "locale": "en",
                        "name": "Data Authorities"
                    },
                    {
                        "locale": "nb",
                        "name": "Data Authorities"
                    },
                    {
                        "locale": "li",
                        "name": "Data Authorities"
                    },
                    {
                        "locale": "fi",
                        "name": "Data Authorities"
                    }
                ],
                "value": [
                    {
                        "locale": "en",
                        "link": "https://www.facebook.cloud"
                    },
                    {
                        "locale": "nb",
                        "link": "https://www.facebook.cloud"
                    },
                    {
                        "locale": "li",
                        "link": "https://www.facebook.cloud"
                    },
                    {
                        "locale": "fi",
                        "link": "https://www.facebook.cloud"
                    }
                ],
                "datetime": "2020-02-08 13:36:38"
            },
            {
                "type": "blog",
                "text": [
                    {
                        "locale": "en",
                        "name": "facebook blog"
                    },
                    {
                        "locale": "nb",
                        "name": "facebook blog"
                    },
                    {
                        "locale": "li",
                        "name": "facebook blog"
                    },
                    {
                        "locale": "fi",
                        "name": "facebook blog"
                    }
                ],
                "value": [
                    {
                        "locale": "en",
                        "link": "https://www.facebook.no"
                    },
                    {
                        "locale": "nb",
                        "link": "https://www.facebook.no"
                    },
                    {
                        "locale": "li",
                        "link": "https://www.facebook.no"
                    },
                    {
                        "locale": "fi",
                        "link": "https://www.facebook.no"
                    }
                ],
                "datetime": "2020-02-08 13:36:39"
            },
            {
                "type": "guide",
                "text": [
                    {
                        "locale": "en",
                        "name": "Guidelines for you"
                    },
                    {
                        "locale": "nb",
                        "name": "Guidelines for you"
                    },
                    {
                        "locale": "li",
                        "name": "Guidelines for you"
                    },
                    {
                        "locale": "fi",
                        "name": "Guidelines for you"
                    }
                ],
                "value": [
                    {
                        "locale": "en",
                        "link": "https://my.instagram.as/"
                    },
                    {
                        "locale": "nb",
                        "link": "https://my.instagram.as/"
                    },
                    {
                        "locale": "li",
                        "link": "https://my.instagram.as/"
                    },
                    {
                        "locale": "fi",
                        "link": "https://my.instagram.as/"
                    }
                ],
                "datetime": "2020-02-08 13:36:41"
            }
        ]
    }
]

Что должно быть лучшим подход, чтобы сделать это?

Это запрос, который я пробовал:

db.getCollection('general').update({
    "source": "homepage",
    "url.type": "admindsg"
}, {
    "$set": {
        "url.text.$[elem].name": "YOYO"
    }
}, {
    "arrayFilters": [{
        "elem.locale": {
            "$eq": "en"
        }
    }],
    "multi": true
})

Это выдает ошибку, говорящую:

Не найден фильтр массива для идентификатора ' Элемент 'in path' url.text. $ [elem] .name '

enter image description here

Ответы [ 2 ]

2 голосов
/ 25 февраля 2020

url поле является массивом, поэтому вам нужно использовать этот синтаксис:

db.getCollection('general').update({
    "source": "homepage"
}, {
        "$set": {
            "url.$[urlId].text.$[textId].name": "YOYO"
        }
    }, {
        "arrayFilters": [
            { "urlId.type": "admindsg" },
            { "textId.locale": "en" },

        ],
        "multi": true
    })

отфильтрованный позиционный оператор $

0 голосов
/ 25 февраля 2020

Проблема в том, что у вас есть 2 фильтра массива

  • url.type = "admindsg"
  • url.text.local = "en"

Можно использовать конвейер агрегации. Это было бы одно решение:

db.getCollection('general').aggregate([
   { $match: { source: "homepage" } },
   { $unwind: "$url" },
   {
      $set: {
         "url.text": {
            $cond: {
               if: { $eq: ["$url.type", "admindsg"] },
               then: {
                  $reduce: {
                     input: "$url.text",
                     initialValue: [],
                     in: {
                        $concatArrays: [
                           "$$value",
                           [{
                              $cond: {
                                 if: { $eq: ["$$this.locale", "fi"] },
                                 then: {
                                    $mergeObjects: [
                                       { locale: "$$this.locale" },
                                       { name: "Järjestelmänvalvoja DSG" }
                                    ]
                                 },
                                 else: "$$this"
                              }
                           }]
                        ]
                     }
                  }
               },
               else: "$url.text"
            }
         }
      }
   },
   {
      $group: {
         _id: { _id: "$_id", source: "$source" },
         url: { $push: "$$ROOT.url" }
      }
   },
   { $replaceRoot: { newRoot: { $mergeObjects: ["$$ROOT", "$_id"] } } }
]).forEach(function (doc) {
   db.getCollection('general').updateOne(
      { _id: doc._id },
      { $set: { url: doc.url } }
   );
})

Вместо $reduce вы также можете использовать $map:

  {
    $set: {
      "url.text": {
        $cond: {
          if: { $eq: ["$url.type", "admindsg"] },
          then: {
            $map: {
              input: "$url.text",
              in: {
                $cond: {
                  if: { $eq: [ "$$this.locale", "fi" ] },
                  then: {
                    $mergeObjects: [
                      { locale: "$$this.locale" },
                      { name: "Järjestelmänvalvoja DSG" }
                    ]
                  },
                  else: "$$this"
                }
              }
            }
          },
          else: "$url.text"
        }
      }
    }
  },
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...