How to query a nested document in mongo?

How do I query a mongo document containing subset of nested array

  • Here is a doc I have: var docIHave = { _id: "someId", things: [ { name: "thing1", stuff: [1,2,3,4,5,6,7,8,9] }, { name: "thing2", stuff: [4,5,6,7,8,9,10,11,12,13,14] }, { name: "thing3", stuff: [1,4,6,8,11,21,23,30] } ] } This is the doc I want: var docIWant = { _id: "someId", things: [ { name: "thing1", stuff: [5,6,7,8,9] }, { name: "thing2", stuff: [5,6,7,8,9,10,11] }, { name: "thing3", stuff: [6,8,11] } ] } stuff´s of docIWant should only contain items greater than min=4 and smaller than max=12. Background: I have a meteor app and I subscribe to a collection giving me docIHave. Based on parameters min and max I need the docIWant "on the fly". The original document should not be modified. I need a query or procedure that returns me docIWant with the subset of stuff. A practical code example would be greatly appreciated.

  • Answer:

    Use http://docs.mongodb.org/manual/core/aggregation-introduction/ like following : First use http://docs.mongodb.org/manual/reference/operator/aggregation/unwind/ this will unwind stuff and then use http://docs.mongodb.org/manual/reference/operator/aggregation/match/ to find elements greater than 4. After that http://docs.mongodb.org/manual/reference/operator/aggregation/group/ data based on things.name and add required fields in $project. The query will be as following: db.collection.aggregate([ { $unwind: "$things" }, { $unwind: "$things.stuff" }, { $match: { "things.stuff": { $gt: 4, $lt:12 } } }, { $group: { "_id": "$things.name", "stuff": { $push: "$things.stuff" } } }, { $project: { "thingName": "$_id", "stuff": 1 } }])

Hans at Stack Overflow Visit the source

Was this solution helpful to you?

Other answers

Use the http://docs.mongodb.org/manual/core/aggregation-introduction/ for this. In the aggregation pipeline, consider the http://docs.mongodb.org/manual/reference/operator/aggregation/match/#pipe._S_match operator as your first pipeline stage. This is quite necessary to optimize your aggregation as you would need to filter documents that match the given criteria first before passing them on further down the pipeline. Next use the http://docs.mongodb.org/manual/reference/operator/aggregation/unwind/#pipe._S_unwind operator. This deconstructs the things array field from the input documents to output a document for each element. Each output document is the input document with the value of the array field replaced by the element. Another http://docs.mongodb.org/manual/reference/operator/aggregation/unwind/#pipe._S_unwind operation would be needed on the things.stuff array as well. The next pipeline stage would then filter dopcuments where the deconstructed things.stuff match the given min and max criteria. Use a http://docs.mongodb.org/manual/reference/operator/aggregation/match/#pipe._S_match operator for this. A http://docs.mongodb.org/manual/reference/operator/aggregation/group/#pipe._S_group operator is then required to group the input documents by a specified identifier expression and applies the accumulator expression http://docs.mongodb.org/manual/reference/operator/aggregation/push/#grp._S_push to each group. This creates an array expression to each group. Typically your aggregation should end up like this (although I haven't actually tested it but this should get you going in the right direction): db.collection.aggregate([ { "$match": { "things.stuff": { "$gt": 4, "$lte": 11 } } }, { "$unwind": "$things" }, { "$unwind": "$things.stuff" }, { "$match": { "things.stuff": { "$gt": 4, "$lte": 11 } } }, { "$group": { "_id": { "_id": "$_id", "things": "$things" }, "stuff": { "$push": "$things.stuff" } } }, { "$group": { "_id": "$_id._id", "things": { "$push": { "name": "$_id.things.name", "stuff": "$stuff" } } } } ])

chridam

If you need to transform the document on the client for display purposes, you could do something like this: Template.myTemplate.helpers({ transformedDoc: function() { // get the bounds - maybe these are stored in session vars var min = Session.get('min'); var max = Session.get('max'); // fetch the doc somehow that needs to be transformed var doc = SomeCollection.findOne(); // transform the thing.stuff arrays _.each(doc.things, function(thing) { thing.stuff = _.reject(thing.stuff, function(n) { return (n < min) || (n > max); }); }); // return the transformed doc return doc; } }); Then in your template: {{#each transformedDoc.things}}...{{/each}}

David Weldon

Related Q & A:

Just Added Q & A:

Find solution

For every problem there is a solution! Proved by Solucija.

  • Got an issue and looking for advice?

  • Ask Solucija to search every corner of the Web for help.

  • Get workable solutions and helpful tips in a moment.

Just ask Solucija about an issue you face and immediately get a list of ready solutions, answers and tips from other Internet users. We always provide the most suitable and complete answer to your question at the top, along with a few good alternatives below.