I just found out that it is possible to sort the result of Map/Reduce with a list function.
Let's take the simple example that you want to count all documents grouped by a field called type. The following map function emits the values of the type fields of all documents:
function(doc) { emit(doc.type, 1); }
To sum up the documents with the same value in the type field, we just need this well-known reduce function:
function(key, values) { return sum(values) }By default CouchDB yields the result ordered by the keys. But if you want to order the result by occurrences of the type of the document you either have to sort it in your app or you use a list function like this:
function(head, req) { var row var rows=[] while(row = getRow()) { rows.push(row) } rows.sort(function(a,b) { return b.value-a.value }) send(JSON.stringify({"rows" : rows})) }If you save the list function as sort and the Map/Reduce-functions as count together in a design document, you can fetch your sorted result like this:
curl http://.../design-doc/_list/sort/count?group=true
Of course there are other options to sort a view result. I didn't found much documentation on this topic, but this thread at stackoverflow is very informative.
Back to the couch - Cheers!
Good stuff, keep these posts coming!
ReplyDeleteThank you,
Dennis
This list function is exactly what I was looking for. Unfortunately, when I tried to implement it, I got the following error:
ReplyDelete{"error":"compilation_error","reason":"Expression does not eval to a function...}
I implemented it exactly as you listed above and tried to access it via browser at the URL you have listed. Any ideas on what is missing/wrong?
Thanks
Chris
csj790@yahoo.com
Hi Chris,
ReplyDeletei guess its an issue with quoting. Did you implement the list function in your browser using futon? Than you have to quote the two double-quotes.
Here is a copy and paste version:
{
"sort": "function(head, req) {
var row
var rows=[]
while(row = getRow()) {
rows.push(row)
}
rows.sort(function(a,b) {
return b.value-a.value
})
send(JSON.stringify({\"rows\" : rows}))
}
"
}
BTW, I use CouchDB in version 1.0.1.
cheers,
Arbo
I know the rendering is probably not going to work, but I have included a snapshot of my design doc. based on this doc, I am going to
ReplyDelete..._design/example/_list/sort/srcIP?group=True
I am also using CouchDB 1.0.1
{
"_id" : "_design/example",
"_rev" : "27-514ca602f74c626c4e3cb331520e32b7",
"views" : {
"srcIP" : {
"map" : "function(doc) {
emit(doc.in_srcIP, 1);
}",
"reduce" : "function(key, values, rereduce) {
return sum(values);
}"
"dstIP" : {
"map" : "function(doc) {
emit(doc.in_dstIP, 1);
}",
"reduce" : "function(key, values, rereduce) {
return sum(values);
}"
}
},
"lists" : {
"sort" : "function(head, req) {
var row
var rows=[]
while(row = getRow()) {
rows.push(row)
}
rows.sort(function(a,b) {
return b.value-a.value
})
send(JSON.stringify({\"rows\" : rows}))
}"
}
}
Ok, I got it. The newlines were not properly quoted. Try to paste the following:
ReplyDelete{
"sort": "\u000afunction(head, req) { \u000a var row\u000a var rows=[]\u000a while(row = getRow()) { \u000a rows.push(row) \u000a } \u000a rows.sort(function(a,b) { \u000a return b.value-a.value\u000a }) \u000a send(JSON.stringify({\"rows\" : rows}))\u000a}"
}
Editing lists functions in futon is not very relaxing. Try to use a texteditor and curl or similar.
cheers,
Arbo
This comment has been removed by the author.
ReplyDeleteThat worked! Thank you tremendously!
ReplyDeleteFound out the root cause to my problem. When I imported the doc into Couch, it didn't create a \n or line break after each line of the list function so to create separation between each line, I had to put a separator (;) in place which resulted in the following list function:
ReplyDelete"lists":{
"sort":"function(head, req) {
var row;
var rows=[];
while(row = getRow()) {
rows.push(row)
};
rows.sort(function(a,b) {
return b.value-a.value
});
send(JSON.stringify({\"rows\" : rows}))
}"
This comment has been removed by the author.
ReplyDeleteExactly what I was looking for!
ReplyDeleteThanks so much!
ReplyDeleteI really appreciate information shared above. It’s of great help. If someone want to learn Online (Virtual) instructor lead live training in Couch DB, kindly contact us http://www.maxmunus.com/contact
ReplyDeleteMaxMunus Offer World Class Virtual Instructor led training on Couch DB. We have industry expert trainer. We provide Training Material and Software Support. MaxMunus has successfully conducted 100000+ trainings in India, USA, UK, Australlia, Switzerland, Qatar, Saudi Arabia, Bangladesh, Bahrain and UAE etc.
For Demo Contact us.
Nitesh Kumar
MaxMunus
E-mail: nitesh@maxmunus.com
Skype id: nitesh_maxmunus
Ph:(+91) 8553912023
http://www.maxmunus.com/