错误地编写了MongoDB聚合管道的$match阶段。

huangapple go评论115阅读模式
英文:

incorrectly drafted MongoDB aggregation pipeline $match stage

问题

我正在尝试使用go.mongodb.org/mongo-driver/mongo库在golang语言中进行查询(下面我附上了纯mongodb查询的工作代码),以下是golang查询代码。我无法正确使用matchStage,我尝试了许多变体,我确定我只是非常粗心或者根本不理解。

如何同时使用$match$expr$and$lte来创建一个正确的matchStage

  1. func (r *Mongo) ChatHistory(ctx context.Context, chatID string, f *Filter) ([]*Message, error) {
  2. matchStage := bson.D{
  3. primitive.E{
  4. Key: "$match",
  5. Value: bson.D{
  6. primitive.E{Key: "$expr", Value: bson.D{
  7. primitive.E{Key: "$and", Value: bson.A{
  8. bson.D{
  9. primitive.E{Key: "$lte", Value: bson.D{
  10. primitive.E{
  11. Key: "$create_date",
  12. Value: f.Date, // int64
  13. },
  14. }},
  15. },
  16. }},
  17. }},
  18. },
  19. },
  20. }
  21. sortStage := bson.D{
  22. {
  23. Key: "$sort", Value: bson.D{
  24. primitive.E{Key: "create_date", Value: -1},
  25. },
  26. },
  27. }
  28. limitStage := bson.D{primitive.E{Key: "$limit", Value: f.Count}}
  29. cursor, err := r.colMessage.Aggregate(ctx, mongo.Pipeline{matchStage, sortStage, limitStage})
  30. if err != nil {
  31. l.Error().Err(err).Msg("failed find")
  32. return nil, err
  33. }
  34. var res []*Message
  35. if err = cursor.All(ctx, &res); err != nil {
  36. l.Error().Err(err).Msg("failed find all documents")
  37. return nil, err
  38. }
  39. if err = cursor.Close(ctx); err != nil {
  40. l.Error().Err(err).Msg("failed close cursor")
  41. return nil, err
  42. }
  43. return res, nil
  44. }

错误:(InvalidPipelineOperator) Unrecognized expression '$create_date'

MongoDB playground链接

英文:

I'm trying to make a query in golang language (below I've attached working code of pure mongodb query) using go.mongodb.org/mongo-driver/mongo library, below is golang query code. I can't get matchStage to work correctly, I've tried many variants and I'm sure I'm just very inattentive or just don't understand

How can I use $match, $expr, $and and $lte at once to make a correct matchStage?

  1. func (r *Mongo) ChatHistory(ctx context.Context, chatID string, f *Filter) ([]*Message, error) {
  2. matchStage := bson.D{
  3. primitive.E{
  4. Key: "$match",
  5. Value: bson.D{
  6. primitive.E{Key: "$expr", Value: bson.D{
  7. primitive.E{Key: "$and", Value: bson.A{
  8. bson.D{
  9. primitive.E{Key: "$lte", Value: bson.D{
  10. primitive.E{
  11. Key: "$create_date",
  12. Value: f.Date, // int64
  13. },
  14. }},
  15. },
  16. }},
  17. }},
  18. },
  19. },
  20. }
  21. sortStage := bson.D{
  22. {
  23. Key: "$sort", Value: bson.D{
  24. primitive.E{Key: "create_date", Value: -1},
  25. },
  26. },
  27. }
  28. limitStage := bson.D{primitive.E{Key: "$limit", Value: f.Count}}
  29. cursor, err := r.colMessage.Aggregate(ctx, mongo.Pipeline{matchStage, sortStage, limitStage})
  30. if err != nil {
  31. l.Error().Err(err).Msg("failed find")
  32. return nil, err
  33. }
  34. var res []*Message
  35. if err = cursor.All(ctx, &res); err != nil {
  36. l.Error().Err(err).Msg("failed find all documents")
  37. return nil, err
  38. }
  39. if err = cursor.Close(ctx); err != nil {
  40. l.Error().Err(err).Msg("failed close cursor")
  41. return nil, err
  42. }
  43. return res, nil
  44. }

Error: (InvalidPipelineOperator) Unrecognized expression '$create_date'

MongoDB playground link

答案1

得分: 2

$lte的值必须是一个数组而不是一个文档:

  1. matchStage := bson.D{
  2. primitive.E{
  3. Key: "$match",
  4. Value: bson.D{
  5. primitive.E{Key: "$expr", Value: bson.D{
  6. primitive.E{Key: "$and", Value: bson.A{
  7. bson.D{
  8. primitive.E{Key: "$lte", Value: bson.A{
  9. "$create_date",
  10. f.Date, // int64
  11. }},
  12. },
  13. }},
  14. }},
  15. },
  16. },
  17. }

另外请注意,你可以从复合字面量中省略primitive.E类型:

  1. matchStage := bson.D{
  2. {
  3. Key: "$match",
  4. Value: bson.D{
  5. {Key: "$expr", Value: bson.D{
  6. {Key: "$and", Value: bson.A{
  7. bson.D{
  8. {Key: "$lte", Value: bson.A{
  9. "$create_date",
  10. f.Date, // int64
  11. }},
  12. },
  13. }},
  14. }},
  15. },
  16. },
  17. }

但请注意,你在Mongo Playground上的表达式是不正确的。引用自文档

$match接受一个指定查询条件的文档。查询语法与读操作查询语法相同...

在使用$expr时,你必须使用$eq,例如:

  1. matchStage := bson.D{
  2. {
  3. Key: "$match",
  4. Value: bson.D{
  5. {Key: "$expr", Value: bson.D{
  6. {Key: "$and", Value: bson.A{
  7. bson.D{
  8. {Key: "$eq", Value: bson.A{
  9. "$chat_id",
  10. chatID,
  11. }},
  12. },
  13. bson.D{
  14. {Key: "$lte", Value: bson.A{
  15. "$create_date",
  16. f.Date, // int64
  17. }},
  18. },
  19. }},
  20. }},
  21. },
  22. },
  23. }

在这里尝试:https://mongoplayground.net/p/SBEJD-Fyhjl

你应该在$match中使用一个普通的查询文档。

看看这个等效的、更简单的解决方案:

  1. matchStage := bson.D{
  2. {
  3. Key: "$match",
  4. Value: bson.D{
  5. {Key: "chat_id", Value: chatID},
  6. {Key: "create_date", Value: bson.D{
  7. {
  8. Key: "$lte",
  9. Value: f.Date, // int64
  10. }},
  11. },
  12. },
  13. },
  14. }

如果你使用bson.M而不是bson.D,甚至更简单:

  1. matchStage := bson.M{
  2. "$match": bson.M{
  3. "chat_id": chatID,
  4. "create_date": bson.M{"$lte": f.Date},
  5. },
  6. }

当然,在这种情况下,你不能使用mongo.Pipeline作为管道,但[]any[]bson.M也可以。

英文:

Value of $lte must be an array not a document:

  1. matchStage := bson.D{
  2. primitive.E{
  3. Key: "$match",
  4. Value: bson.D{
  5. primitive.E{Key: "$expr", Value: bson.D{
  6. primitive.E{Key: "$and", Value: bson.A{
  7. bson.D{
  8. primitive.E{Key: "$lte", Value: bson.A{
  9. "$create_date",
  10. f.Date, // int64
  11. }},
  12. },
  13. }},
  14. }},
  15. },
  16. },
  17. }

Also note that you can leave out the primitive.E type from the composite literal:

  1. matchStage := bson.D{
  2. {
  3. Key: "$match",
  4. Value: bson.D{
  5. {Key: "$expr", Value: bson.D{
  6. {Key: "$and", Value: bson.A{
  7. bson.D{
  8. {Key: "$lte", Value: bson.A{
  9. "$create_date",
  10. f.Date, // int64
  11. }},
  12. },
  13. }},
  14. }},
  15. },
  16. },
  17. }

But note that your expression on the mongo playground is incorrect. Quoting from the doc:

> $match takes a document that specifies the query conditions. The query syntax is identical to the read operation query syntax
...

When using $expr, you have to use $eq, for example:

  1. matchStage := bson.D{
  2. {
  3. Key: "$match",
  4. Value: bson.D{
  5. {Key: "$expr", Value: bson.D{
  6. {Key: "$and", Value: bson.A{
  7. bson.D{
  8. {Key: "$eq", Value: bson.A{
  9. "$chat_id",
  10. chatID,
  11. }},
  12. },
  13. bson.D{
  14. {Key: "$lte", Value: bson.A{
  15. "$create_date",
  16. f.Date, // int64
  17. }},
  18. },
  19. }},
  20. }},
  21. },
  22. },
  23. }

Try it here: https://mongoplayground.net/p/SBEJD-Fyhjl

You should use a normal query document in $match.

See this equivalent, much simpler solution:

  1. matchStage := bson.D{
  2. {
  3. Key: "$match",
  4. Value: bson.D{
  5. {Key: "chat_id", Value: chatID},
  6. {Key: "create_date", Value: bson.D{
  7. {
  8. Key: "$lte",
  9. Value: f.Date, // int64
  10. }},
  11. },
  12. },
  13. },
  14. }

And even much-much simpler if you use bson.M instead of bson.D:

  1. matchStage := bson.M{
  2. "$match": bson.M{
  3. "chat_id": chatID,
  4. "create_date": bson.M{"$lte": f.Date},
  5. },
  6. }

Of course in this last case you can't use mongo.Pipeline for the pipeline, but []any or []bson.M would also do.

huangapple
  • 本文由 发表于 2023年2月3日 04:37:21
  • 转载请务必保留本文链接:https://go.coder-hub.com/75328672.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定