基本的なページネーションを行う集約クエリを実行したい:
company_id
に属するすべての注文を検索しますorder_number
で注文を並べ替えます100
および残りを渡す2
およびそれらを渡すクエリの内訳は次のとおりです。
db.Order.collection.aggregate([
これにより、一致するすべてのドキュメントが検索されます。
{ '$match' : { "company_id" : ObjectId("54c0...") } },
これはドキュメントをソートします:
{ '$sort' : { 'order_number' : -1 } },
これはドキュメントをカウントし、変更されていないドキュメントを渡しますが、ここから物事が変になるので、間違いを犯していると確信しています。
{
'$group' : {
'_id' : null,
'count' : { '$sum' : 1 },
'entries' : { '$Push' : "$$ROOT" }
}
},
これはいくつかのドキュメントをスキップするようです:
{ "$skip" : 100 },
これはドキュメントを制限することになっていますが、そうではありません:
{ "$limit" : 2 },
これはカウントを返しますが、ドキュメントを配列で返すのではなく、各フィールドで配列を返します。
{ '$project' : {
'count' : 1,
'entries' : {'_id' : "$entries._id", 'order_number' : "$entries.order_number"}
}
}
])
これが結果です:
[
{ "_id" : null,
"count" : 300,
"entries" : [
{
"_id" : [ObjectId('5a5c...'), ObjectId('5a5c...')],
"order_number" : ["4346", "4345"]
},
{
"_id" : [ObjectId('5a5c...'), ObjectId('5a5c...')],
"order_number" : ["4346", "4345"]
},
...
]
}
]
どこで間違っていますか?
合計andを計算してサブセットを返すには、同じデータセットにグループ化とスキップ/制限を適用する必要があります。そのために facets を利用できます
たとえば、3ページ目、1ページあたり10ドキュメントを表示するには:
db.Order.aggregate([
{ '$match' : { "company_id" : ObjectId("54c0...") } },
{ '$sort' : { 'order_number' : -1 } },
{ '$facet' : {
metadata: [ { $count: "total" }, { $addFields: { page: NumberInt(3) } } ],
data: [ { $skip: 20 }, { $limit: 10 } ] // add projection here wish you re-shape the docs
} }
] )
2つのフィールドを持つ単一のドキュメントを返します。
{
"metadata" : [
{
"total" : 300,
"page" : 3
}
],
"data" : [
{
... original document ...
},
{
... another document ...
},
{
... etc up to 10 docs ...
}
]
}