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
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:
- How can I change a value in an array?Best solution by Stack Overflow
- How do I display this XML document?Best solution by Stack Overflow
- How do I audit a policy document? What areas should I look at and how should it be written?Best solution by Yahoo! Answers
- How do I attach a word document to an email?Best solution by Yahoo! Answers
- How do i change a document to a picture?Best solution by Answerbag.com
Just Added Q & A:
- How many active mobile subscribers are there in China?Best solution by Quora
- How to find the right vacation?Best solution by bookit.com
- How To Make Your Own Primer?Best solution by thekrazycouponlady.com
- How do you get the domain & range?Best solution by ChaCha
- How do you open pop up blockers?Best solution by Yahoo! Answers
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.