web-dev-qa-db-ja.com

ページネーションにMongoDB集約を使用する方法は?

基本的なページネーションを行う集約クエリを実行したい:

  1. 特定のcompany_idに属するすべての注文を検索します
  2. order_numberで注文を並べ替えます
  3. 文書の総数を数える
  4. 例えばにスキップします文書番号100および残りを渡す
  5. ドキュメントの数を制限します2およびそれらを渡す
  6. カウントとドキュメントから選択したいくつかのフィールドを返すことで終了します

クエリの内訳は次のとおりです。

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"]
      },
      ...
    ]
  }
]

どこで間違っていますか?

8
JohnSmith1976

合計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 ...
        }
    ]
}
22
Alex Blex