Reputation: 7063
Passing an object to an existing field in $set
or $addFields
merges objects rather than replaces them, e.g.
https://mongoplayground.net/p/aXe-rExjCXr
// Collection
[
{
"_id": "123",
"options": {
"size": "Large",
"color": "Red"
}
}
]
// Aggregate
db.collection.aggregate([
{
$set: {
options: {
size: "Small"
}
}
}
]);
// Expect
[
{
"_id": "123",
"options": {
"size": "Small"
}
}
]
// Actual
[
{
"_id": "123",
"options": {
"size": "Small",
"color": "Red" // <-- Not expected?
}
}
]
(It get's even weirder with arrays)
Is it possible to have it behave like non-object values and simply replace the field?
For context, I want to use this in an aggregate update pipeline.
Upvotes: 1
Views: 2266
Reputation: 8705
This is the expected behaviour, and as far as i know there is not plan to change, as far as i remembered there was a jira with this, but they closed it, meaning that it will not change i think.
$set/$addFields
replace always except
$project
replace always except
Solutions
$unset
the old field first for example, and then $set
$literal
to avoid this, but when we use $literal
we have to be sure that we dont use expressions because they will not be evaluated.
(expressions like path references, variables, operators etc)Upvotes: 3