<?xml version="1.0" encoding="UTF-8"?>
<?xml-stylesheet type="text/css" href="/stylesheets/rss.css"?>
<rss version="2.0" xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:trackback="http://madskills.com/public/xml/rss/module/trackback/">
  <channel>
    <title>validates_presence_of :purpose : Tag json, everything about json</title>
    <link>http://lenaherrmann.net</link>
    <language>en_US</language>
    <ttl>40</ttl>
    <description>Lena Herrmann</description>
    <item>
      <title>Bulk deletion of documents in CouchDB</title>
      <description>&lt;p&gt;For the couchapp I'm writing, I recently wanted to do a bulk delete operation with CouchDB, but found very little about it on the web. I don't know if my approach is the best way to go, but well, it works - if you know a better way, I'm happy to learn about it.&lt;/p&gt;

&lt;p&gt;In the file &lt;a href="http://github.com/lenalena/couchdb/blob/trunk/share/www/script/jquery.couch.js"&gt;jquery.couch.js&lt;/a&gt; that comes with CouchDB there are a couple of convenience methods to help you dealing with the database, documents and views. There is a method for deletion too (removeDoc), but it only can handle one document at a time. In the style of bulkSave, I added a bulkRemove method:

&lt;div class="CodeRay"&gt;&lt;pre&gt;&lt;notextile&gt;&lt;span class="CodeRay"&gt;bulkRemove: &lt;span class="kw"&gt;function&lt;/span&gt;(docs, options){
  docs.docs = &lt;span class="pd"&gt;$&lt;/span&gt;.each(docs.docs, &lt;span class="kw"&gt;function&lt;/span&gt;(i, doc){doc._deleted = &lt;span class="kw"&gt;true&lt;/span&gt;});
  &lt;span class="pd"&gt;$&lt;/span&gt;.extend(options, {&lt;span class="ke"&gt;successStatus&lt;/span&gt;: &lt;span class="i"&gt;201&lt;/span&gt;});
  ajax({
      &lt;span class="ke"&gt;type&lt;/span&gt;: &lt;span class="s"&gt;&lt;span class="dl"&gt;&amp;quot;&lt;/span&gt;&lt;span class="k"&gt;POST&lt;/span&gt;&lt;span class="dl"&gt;&amp;quot;&lt;/span&gt;&lt;/span&gt;,
      &lt;span class="ke"&gt;contentType&lt;/span&gt;: &lt;span class="s"&gt;&lt;span class="dl"&gt;&amp;quot;&lt;/span&gt;&lt;span class="k"&gt;application/json&lt;/span&gt;&lt;span class="dl"&gt;&amp;quot;&lt;/span&gt;&lt;/span&gt;,
      &lt;span class="ke"&gt;dataType&lt;/span&gt;: &lt;span class="s"&gt;&lt;span class="dl"&gt;&amp;quot;&lt;/span&gt;&lt;span class="k"&gt;json&lt;/span&gt;&lt;span class="dl"&gt;&amp;quot;&lt;/span&gt;&lt;/span&gt;, 
      &lt;span class="ke"&gt;data&lt;/span&gt;: toJSON(docs),
      &lt;span class="ke"&gt;url&lt;/span&gt;: &lt;span class="lv"&gt;this&lt;/span&gt;.uri + &lt;span class="s"&gt;&lt;span class="dl"&gt;&amp;quot;&lt;/span&gt;&lt;span class="k"&gt;_bulk_docs&lt;/span&gt;&lt;span class="dl"&gt;&amp;quot;&lt;/span&gt;&lt;/span&gt; + encodeOptions(options)
    },
    options,
    &lt;span class="s"&gt;&lt;span class="dl"&gt;&amp;quot;&lt;/span&gt;&lt;span class="k"&gt;The documents could not be deleted&lt;/span&gt;&lt;span class="dl"&gt;&amp;quot;&lt;/span&gt;&lt;/span&gt;
  );
},&lt;/span&gt;&lt;/notextile&gt;&lt;/pre&gt;&lt;/div&gt;&lt;/p&gt;

&lt;p&gt;In here I add the attribute "_deleted = true" to every document I want to delete, and then I do a bulk update with the documents. That's it.&lt;/p&gt;

&lt;p&gt;This is one possible use case: In the delete action of my controller I first fetch an array with the post and its comments from the couch (how to do this see below), and then I call the bulkRemove action on that array.

&lt;div class="CodeRay"&gt;&lt;pre&gt;&lt;notextile&gt;&lt;span class="CodeRay"&gt;couchapp.design.view(&lt;span class="s"&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="k"&gt;post_with_comments&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;/span&gt;, {
  &lt;span class="ke"&gt;startkey&lt;/span&gt;: [params[&lt;span class="s"&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="k"&gt;id&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;/span&gt;]],
  &lt;span class="ke"&gt;endkey&lt;/span&gt;: [params[&lt;span class="s"&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="k"&gt;id&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;/span&gt;], {}],
  &lt;span class="ke"&gt;success&lt;/span&gt;: &lt;span class="kw"&gt;function&lt;/span&gt;(json) {
    &lt;span class="kw"&gt;if&lt;/span&gt; (json.rows.length &amp;gt; &lt;span class="i"&gt;0&lt;/span&gt;) { 
      &lt;span class="kw"&gt;var&lt;/span&gt; post_and_comments = json.rows.map(&lt;span class="kw"&gt;function&lt;/span&gt;(row) {&lt;span class="kw"&gt;return&lt;/span&gt; row.value}); 
      couchapp.db.bulkRemove({&lt;span class="ke"&gt;docs&lt;/span&gt;: post_and_comments}, {
        &lt;span class="ke"&gt;success&lt;/span&gt;: &lt;span class="kw"&gt;function&lt;/span&gt;() {
          flash = {&lt;span class="ke"&gt;message&lt;/span&gt;: &lt;span class="s"&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="k"&gt;Post deleted.&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;/span&gt;, &lt;span class="ke"&gt;type&lt;/span&gt;: &lt;span class="s"&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="k"&gt;notice&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;/span&gt;}
          redirect(&lt;span class="s"&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="k"&gt;#/outlines&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;/span&gt;, flash);
        },
        &lt;span class="ke"&gt;error&lt;/span&gt;: &lt;span class="kw"&gt;function&lt;/span&gt;(response_code, msg) {
          flash = {&lt;span class="ke"&gt;message&lt;/span&gt;: &lt;span class="s"&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="k"&gt;Error deleting post: &lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;/span&gt; + msg, &lt;span class="ke"&gt;type&lt;/span&gt;: &lt;span class="s"&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="k"&gt;error&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;/span&gt;};
        }
      });
    }      
  }
});&lt;/span&gt;&lt;/notextile&gt;&lt;/pre&gt;&lt;/div&gt;&lt;/p&gt;

&lt;p&gt;Note that the "flash" object I'm using is my own Sammy.js implementation of a Rails-like flash to show a message after a redirect. Maybe I'll share that in another blog post.&lt;/p&gt;

&lt;p&gt;As a bonus, this is the view I'm using to retrieve a post and its comments in one request. I'm following the pattern Christopher Lenz is recommending in his article on &lt;a href="http://www.cmlenz.net/archives/2007/10/couchdb-joins"&gt;CouchDB Joins&lt;/a&gt;.

&lt;div class="CodeRay"&gt;&lt;pre&gt;&lt;notextile&gt;&lt;span class="CodeRay"&gt;&lt;span class="kw"&gt;function&lt;/span&gt;(doc) {
  &lt;span class="kw"&gt;if&lt;/span&gt; (doc.type == &lt;span class="s"&gt;&lt;span class="dl"&gt;&amp;quot;&lt;/span&gt;&lt;span class="k"&gt;Post&lt;/span&gt;&lt;span class="dl"&gt;&amp;quot;&lt;/span&gt;&lt;/span&gt;) {
    emit([doc._id, &lt;span class="i"&gt;0&lt;/span&gt;], doc);
  } &lt;span class="kw"&gt;else&lt;/span&gt; &lt;span class="kw"&gt;if&lt;/span&gt; (doc.type == &lt;span class="s"&gt;&lt;span class="dl"&gt;&amp;quot;&lt;/span&gt;&lt;span class="k"&gt;Comment&lt;/span&gt;&lt;span class="dl"&gt;&amp;quot;&lt;/span&gt;&lt;/span&gt;) {
    emit([doc.post_id, &lt;span class="i"&gt;1&lt;/span&gt;, Date.parse(doc.created_at)], doc);
  }
}&lt;/span&gt;&lt;/notextile&gt;&lt;/pre&gt;&lt;/div&gt;&lt;/p&gt;

&lt;p&gt;When you call this view with

&lt;div class="CodeRay"&gt;&lt;pre&gt;&lt;notextile&gt;&lt;span class="CodeRay"&gt;startkey: [params[&lt;span class="s"&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="k"&gt;id&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;/span&gt;]],
&lt;span class="ke"&gt;endkey&lt;/span&gt;: [params[&lt;span class="s"&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="k"&gt;id&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;/span&gt;], {}]&lt;/span&gt;&lt;/notextile&gt;&lt;/pre&gt;&lt;/div&gt;&lt;/p&gt;

&lt;p&gt;you get an array with the post as the first element, and then all the comments that have the post's ID as post_id.&lt;/p&gt;

&lt;p&gt;This can of course also be used for things other than deleting: to show all the comments for a post, I get the array's first element and extract the title and content. Then I remove it from the array with

&lt;div class="CodeRay"&gt;&lt;pre&gt;&lt;notextile&gt;&lt;span class="CodeRay"&gt;json.rows.splice(&lt;span class="i"&gt;0&lt;/span&gt;,&lt;span class="i"&gt;1&lt;/span&gt;);        &lt;/span&gt;&lt;/notextile&gt;&lt;/pre&gt;&lt;/div&gt;&lt;/p&gt;

&lt;p&gt;You now have an array with only the post's comments and you can display them somehow.&lt;/p&gt;

</description>
      <pubDate>Tue, 22 Dec 2009 17:37:00 +0000</pubDate>
      <guid isPermaLink="false">urn:uuid:150befc7-1e89-47d5-ae0c-47d803336e73</guid>
      <comments>http://lenaherrmann.net/2009/12/22/bulk-deletion-of-documents-in-couchdb#comments</comments>
      <category>couchdb</category>
      <category>javascript</category>
      <category>json</category>
      <trackback:ping>http://lenaherrmann.net/trackbacks?article_id=8</trackback:ping>
      <link>http://lenaherrmann.net/2009/12/22/bulk-deletion-of-documents-in-couchdb</link>
    </item>
  </channel>
</rss>

