Le mieux est probablement de rechercher en utilisant $where
en plus de la requête normale, tout en gardant les éléments sur le serveur :
db.getCollection('collection').find({
"array": {
"$elemMatch": { "field": "BCD", "enabled": "true" },
},
"$where": function() {
return this.array.map((e,i) => Object.assign(e,{ i }))
.filter( e => e.field === "BCD" && e.enabled === "true" )
.map( e => e.i )[0] <=
this.array.map(e => e.enabled).indexOf("true")
}
})
Et si vous avez MongoDB 3.4 avec prise en charge de $indexOfArray
et $range
, alors cela peut sembler plus long mais c'est en fait plus efficace avec $redact
:
db.getCollection('collection').aggregate([
{ "$match": {
"array": {
"$elemMatch": { "field": "BCD", "enabled": "true" },
}
}},
{ "$redact": {
"$cond": {
"if": {
"$lte": [
{ "$arrayElemAt": [
{ "$map": {
"input": {
"$filter": {
"input": {
"$map": {
"input": {
"$zip": {
"inputs": [
"$array",
{ "$range": [0, { "$size": "$array" }] }
]
}
},
"as": "a",
"in": {
"field": { "$arrayElemAt": [ "$$a.field", 0 ] },
"enabled": { "$arrayElemAt": [ "$$a.enabled", 0 ] },
"index": { "$arrayElemAt": [ "$$a", 1 ] }
}
}
},
"as": "a",
"cond": {
"$and": [
{ "$eq": [ "$$a.field", "BCD" ] },
{ "$eq": [ "$$a.enabled", "true" ] }
]
}
}
},
"as": "a",
"in": "$$a.index"
}},
0
]},
{ "$indexOfArray": [ "$array.enabled", "true" ] }
]
},
"then": "$$KEEP",
"else": "$$PRUNE"
}
}}
])
Il n'y a donc vraiment aucune opération de requête réelle qui applique cela, mais ces deux cas conservent la sélection "sur le serveur" au lieu d'envoyer des données sur le câble au client, puis de filtrer.
Parce que si vous faites cela, cela annule en quelque sorte le but d'utiliser une base de données en premier lieu. Donc, vous voulez vraiment que cela se produise sur le serveur.