<?xml version="1.0" encoding="utf-8" ?>
<rss version="2.0" 
   xmlns:creativeCommons="http://backend.userland.com/creativeCommonsRssModule" 
   xmlns:html="http://www.w3.org/1999/html" 
   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" 
   xmlns:slash="http://purl.org/rss/1.0/modules/slash/">
<channel>
   <title>James Westby</title>
   <link>http://jameswestby.net/weblog</link>
   <description></description>
   <language>en</language>
   <copyright>Copyright 2006-2008 James Westby</copyright>
   <ttl>60</ttl>
   <pubDate>Tue, 15 May 2012 02:15 GMT</pubDate>
   <managingEditor>jw+blog@jameswestby.net</managingEditor>
   <generator>PyBlosxom http://pyblosxom.sourceforge.net/ 1.4.3 01/10/2008</generator>
<item>
   <title>Monitoring and Testability</title>
   <guid isPermaLink="false">tech/25-monitoring-and-testability</guid>
   <link>http://jameswestby.net/weblog/tech/25-monitoring-and-testability.html</link>
   <description><![CDATA[
<div class="document">
<p>At UDS last week there was another &quot;Testing in Ubuntu&quot; session. During the
event I gave a brief presentation on monitoring and testability. The thesis
was that there are a lot of parallels between monitoring and testing, so many
that it's worth thinking of monitoring as a type of testing at times. Due to
that great monitoring requires a testable system, as well as thinking about
monitoring right at the start to build a monitorable system as well as a
testable one.</p>
<p>You can watch a video of the talk <a class="reference external" href="http://www.youtube.com/watch?v=zWPp7vmMwOk#t=48m34s">here</a>. (Thanks to the video team for
recording it and getting it online quickly.)</p>
<p>I have two main questions. Firstly, what are the conventional names for
the &quot;passive&quot; and &quot;active&quot; monitoring that I describe? Seecondly, do you
agree with me about monitoring?</p>
</div>

]]></description>
   <category domain="http://jameswestby.net/weblog"></category>
   <pubDate>Tue, 15 May 2012 02:15 GMT</pubDate>
</item>
<item>
   <title>First day of the Linaro Connect</title>
   <guid isPermaLink="false">tech/24-first-day-of-linaro-connect</guid>
   <link>http://jameswestby.net/weblog/tech/24-first-day-of-linaro-connect.html</link>
   <description><![CDATA[
<div class="document">
<p>Today is the first day of the <a class="reference external" href="http://connect.linaro.org">Linaro Connect</a> in Cambridge. Linaro has
gathered to spend a week talking, coding and having fun.</p>
<p>The Infrastructure team is spending most of the week coding, on a few
select topics, chosen to make good use of the time that we have together.</p>
<p>In order to help us focus on our goals for the week I've put together
a hard copy version of <a class="reference external" href="http://status.linaro.org">status.linaro.org</a>.</p>
<img alt="/images/connect-progress-start.jpg" src="/images/connect-progress-start.jpg" />
<p>We'll be updating it during the week as we make progress. I'll report
back on how it looks at the end of the week.</p>
</div>

]]></description>
   <category domain="http://jameswestby.net/weblog"></category>
   <pubDate>Mon, 01 Aug 2011 11:54 GMT</pubDate>
</item>
<item>
   <title>pkgme: handles packaging for you</title>
   <guid isPermaLink="false">tech/23-pkgme-handles-packaging-for-you</guid>
   <link>http://jameswestby.net/weblog/tech/23-pkgme-handles-packaging-for-you.html</link>
   <description><![CDATA[
<div class="document">
<p>If you are an application developer and you want to distribute your new
application for a linux distribution, then you currently have several
hurdles in your path. Beyond picking which one to start with, you either
have a learn a packaging format well enough that you can do the work
yourself, or find someone that can do it for you.</p>
<p>At the early stages though neither of these options is particularly
compelling. You don't want to learn a packaging format, as there is
lots of code to write, and that's what you want to focus on. Finding
someone to do the work for you would be great, but there are far
more applications than skilled packagers, and convincing someone to
help you with something larval is tough: there are going to be a lot
of updates, with plenty of churn, to stay on top of, and it may be
too early for them to tell if the application will be any good.</p>
<p>This is where <tt class="docutils literal">pkgme</tt> comes in. This is a tool that can take care
of the packaging for you, so that you can focus on writing the code,
and skilled packagers can focus on packages that need high-quality
packaging as they will have lots of users.</p>
<p>This isn't a new idea, and there are plenty of tools out there to
generate the packaging for e.g. a Python application. I don't
think it is a particularly good use of developer time to produce
tools like that for every language/project type out there.</p>
<p>Instead, a few of us created <tt class="docutils literal">pkgme</tt>. This is a tool in two parts.
The first part knows about packaging, and how to create the necessary
files to build a working package, but it doesn't know anything about
your application. This knowledge is delegated to a backend, which doesn't
need to understand packaging, and just needs to be able to tell <tt class="docutils literal">pkgme</tt>
certain facts about the application.</p>
<p><tt class="docutils literal">pkgme</tt> is now at a stage where we would like to work with people to
develop backends for whatever application type you would like (Python/
Ruby On Rails/GNOME/KDE/CMake/Autotools/Vala etc.) You don't have to
be an expert on packaging, or indeed on the project type you want to
work on. All it takes is writing a few scripts (in whatever language
makes sense), which can introspect an application and report things
such as the name, version, dependencies, etc.</p>
<p>If this sounds like something that you would like to do then please
<a class="reference external" href="http://pkgme.net/doc/backends/index.html">take a look at the documentation</a>, write the scripts, and then
submit your backend for inclusion in <tt class="docutils literal">pkgme</tt>.</p>
<p>You can also <a class="reference external" href="http://pkgme.net/contact.html">contact the developers</a>, see the <a class="reference external" href="http://pkgme.net/">nascent website at pkgme.net</a>,
or <a class="reference external" href="https://launchpad.net/pkgme">visit the Launchpad page.</a> (We are also very interested in help
with the website and documentation if that is where you skills or interests
lie.)</p>
</div>

]]></description>
   <category domain="http://jameswestby.net/weblog"></category>
   <pubDate>Sun, 06 Feb 2011 19:05 GMT</pubDate>
</item>
<item>
   <title>Couchapp Walkthrough: Part 4: How data gets to the user</title>
   <guid isPermaLink="false">tech/22-couchapp-walkthrough-part-4-how-data-gets-to-the-user</guid>
   <link>http://jameswestby.net/weblog/tech/22-couchapp-walkthrough-part-4-how-data-gets-to-the-user.html</link>
   <description><![CDATA[
<div class="document">
<p>This was the confusing part when I first ran <tt class="docutils literal">couchapp</tt> to create a new app,
I couldn't really see where the &quot;entry point&quot; of the app was. In the hope that
it might help someone else I'm going to present a quick overview of the default
setup.</p>
<div class="section" id="index-html">
<h1><tt class="docutils literal">index.html</tt></h1>
<p>The <tt class="docutils literal">index.html</tt> page is a static attachement, and the user starts by requesting
it with their browser.</p>
<p>It has some small amount of static HTML, part of which creates a <tt class="docutils literal">div</tt> for the
javascript to put the data in.</p>
<p>Either inline, or in an included file, there is a small bit of javascript that will
initialise the couchapp.</p>
<p>By default this will use the <tt class="docutils literal">div</tt> with the id <tt class="docutils literal">items</tt>, and will attach an
<tt class="docutils literal">evently</tt> widget to it.</p>
</div>
<div class="section" id="evently">
<h1><tt class="docutils literal">evently</tt></h1>
<p>The <tt class="docutils literal">evently</tt> widget that is attached will then either have an <tt class="docutils literal">_init</tt> event,
or a <tt class="docutils literal">_changes</tt> event, either of which will be immediately run by <tt class="docutils literal">evently</tt>.</p>
<p>This event will usually make a couchdb query to get data to transform to HTML and
present to the user (see <cite>part three</cite> for how this works.)</p>
<p>Once that data has been displayed the user any combination of <tt class="docutils literal">evently</tt> widgets or
javascript can be used to make further queries and build an app that works however
you like.</p>
</div>
<div class="section" id="previous-installments">
<h1>Previous installments</h1>
<p>See <a class="reference external" href="http://jameswestby.net/weblog/tech/18-couchapp-walkthrough-part-1.html">part one</a>, <a class="reference external" href="http://jameswestby.net/weblog/tech/19-couchapp-walkthrough-part-2-the-couchapp-tool.html">part two</a>, and <a class="reference external" href="http://jameswestby.net/weblog/tech/20-couchapp-walkthrough-part-3-evently.html">part three.</a></p>
</div>
</div>

]]></description>
   <category domain="http://jameswestby.net/weblog"></category>
   <pubDate>Fri, 24 Dec 2010 01:16 GMT</pubDate>
</item>
<item>
   <title>Introducing soupmatchers</title>
   <guid isPermaLink="false">tech/21-introducing-soupmatchers</guid>
   <link>http://jameswestby.net/weblog/tech/21-introducing-soupmatchers.html</link>
   <description><![CDATA[
<div class="document">
<p>jml <a class="reference external" href="http://code.mumak.net/2010/12/testtools-098-released.html">just announced testtools 0.9.8</a> and in it mentioned the <a class="reference external" href="http://launchpad.net/soupmatchers">soupmatchers</a>
project that I started. Given that I haven't talked about it here before, I wanted to
do a post to introduce it, and explain some of the rationale behind it.</p>
<p>soupmatchers is a library for unit testing HTML, allowing you to assert that certain things
are present or not within an HTML string. Asserting this based on substring matching is going
to be too fragile to be usable, and so soupmatchers works on a parsed representation of the HTML.
It uses the wonderful <a class="reference external" href="http://www.crummy.com/software/BeautifulSoup/">BeautifulSoup</a> library for parsing the HTML, and allows you to assert
the presence or not of tags based on the attributes that you care about.</p>
<pre class="literal-block">
self.assertThat(some_html,
                HTMLContains(Tag('testtools link', 'a',
                attrs={'href': 'https://launchpad.net/testtools'})))
</pre>
<p>You can see more examples in <a class="reference external" href="http://bazaar.launchpad.net/~soupmatchers-dev/soupmatchers/trunk/annotate/head:/README">the README</a>.</p>
<p>Basing this on the testtools matchers frameworks allows you to do this in a semi-declarative way.
I think there is a lot of potential here to improve your unit tests. For instance you can
start to build a suite of matchers tailored to talking about the HTML that your application outputs.
You can have matchers that match areas of the page, and then talk about other elements relative to
them (&quot;This link is placed within the sidebar&quot;). One thing that particularly interests me is to create
a class hierarchy that allows you test particular things across your application. For instance,
you could have an ExternalLink class that asserts that a particular class is set on all of your
external links. Assuming that you use this at the appropriate places in your tests then you will know that the
style that is applied to class will be on all external links. Should you wish to change the way that
external links are represented in the HTML you can change the one class and your tests should tell you
all the places that the code has to be updated.</p>
<p>Please go ahead and try the library and <a class="reference external" href="https://bugs.launchpad.net/soupmatchers/+filebug">let me know</a> how it could be improved.</p>
</div>

]]></description>
   <category domain="http://jameswestby.net/weblog"></category>
   <pubDate>Thu, 23 Dec 2010 02:45 GMT</pubDate>
</item>
<item>
   <title>Couchapp Walkthrough: Part 3: Evently</title>
   <guid isPermaLink="false">tech/20-couchapp-walkthrough-part-3-evently</guid>
   <link>http://jameswestby.net/weblog/tech/20-couchapp-walkthrough-part-3-evently.html</link>
   <description><![CDATA[
<div class="document">
<p>[ Apologies to those that saw this half-finished when I published
rather than saving a draft ]</p>
<p>This is the part that it took me a long time to understand: how the different
parts of the default couchapp collaborate to present data to the user.</p>
<p>In this post I'm just going to deal with client-side couchapps using the
default technologies. As explained in the previous post you can use any
combination of HTML and javascript in a couchapp, and you can also do some
of the work server-side in couchdb. However, I'm going to explain what
the couchapp tool gives you when you create a new project, as that is
where you are likely to be starting, and once you understand that you
can choose where to deviate from that model.</p>
<div class="section" id="jquery-events">
<h1>jQuery events</h1>
<p>Our first detour is in to a little bit of background, the excellent
javascript libarary that is heavily used in couchapps.</p>
<p>jQuery allows for events on elements in the DOM. There are standard
events, such as &quot;click&quot; and &quot;submit&quot;, but you are free to define your
own.</p>
<p>These events are given a name, and you can then &quot;trigger&quot; them, and
bind handlers to act when they are triggered.</p>
<p>By building up events from low level ones such as &quot;click&quot;, to more-complex
and app-specific ones such as &quot;item purchased&quot;, you can break down
your code in to smaller chunks, and have different parts of the page
react to the same event, such as having the &quot;buy&quot; link disappear from
the item that the user just bought, as well as having the total of
the shopping cart update.</p>
<p>Events can also have data, or arguments, that travels with them. For instance
the &quot;item purchased&quot; event could have the item that was purchased as the
data, so that handlers can make use of it when they run.</p>
</div>
<div class="section" id="evently">
<h1>evently</h1>
<p>Now that we know something about jQuery events, we can look at something
built on top of them, the &quot;evently&quot; library. This is a layer on top of
jQuery that allows you build up your app from pieces that have a specific
function, and communicate through events.</p>
<p>An evently &quot;widget&quot; can be bound to an element (or several elements if
you want). The widget is a bunch of event handlers which can do anything
you like, but have some conveniences built in for fetching data and
updating the page based on the result.</p>
<p>When an event is triggered the handler you defined is run. If it is
a simple javascript function then that function is run, and can do
anything you like.</p>
<pre class="literal-block">
{click: function() {
        alert(&quot;You clicked!&quot;);
    }
}
</pre>
<p>Often though you want to update the element based on the action. evently
has built in support for the &quot;mustache&quot; templating language, and if you
specify a template in that syntax it will replace the current HTML
of the element that it is attached to with the result of rendering
that template.</p>
<pre class="literal-block">
{click:
    {
        mustache: &quot;&lt;div&gt;You clicked!&lt;/div&gt;&quot;
    }
}
</pre>
<p>Which will put &quot;You clicked!&quot; in to the page instead of in an alert. What
if you don't want to replace the current content, and just want to append
another line? For that use the &quot;render&quot; option.</p>
<pre class="literal-block">
{click:
    {
        mustache: &quot;&lt;div&gt;You clicked!&lt;/div&gt;&quot;,
        render: &quot;append&quot;
    }
}
</pre>
<p>Which would put another &quot;You clicked!&quot; on the page every time you click.
As well as &quot;append&quot; there is also &quot;prepend&quot;, or really any jQuery method
that you want to call.</p>
<p>Simply rendering a static template isn't going to be very useful though,
usually you want something dynamic. For that use the &quot;data&quot; option,
which can just be an object if you want, but that's still not going to
be very dynamic either, so it can be a function that returns an object.</p>
<pre class="literal-block">
{click:
    {
        data: function(e) {
            return {name, &quot;Bob&quot;};
        },
        mustache: &quot;&lt;div&gt;Hi {{name}}!&lt;/div&gt;&quot;
    }
}
</pre>
<p>The data function gets passed the event object from jQuery (so you
can e.g. get the target of the event), and any data for the event
too (so it could see what item you just bought).</p>
<p>That's all well and good, but it doesn't help us get data from couchdb
in to the page. For that we need the opportunity to make a request
to couchdb. We could just fall back to using one function to handle
the event, but then we lose the integration with mustache. Therefore
there is an &quot;async&quot; key that allows us to make an AJAX request
and then use mustache on the result.</p>
<pre class="literal-block">
{click:
    {
        async: function(callback) {
            /* some code that does an async request, and then calls callback with the result */
        },
        data: function(resp) {
            /* Some code that processes the data from the async function to ready it for the template */
        },
        mustache: &quot;A tempate that will be rendered with the result of the data function&quot;
    }
}
</pre>
<p>Now, writing an async method to query a couchdb view is so common in couchapps that
eventy has special support for it. The <tt class="docutils literal">query</tt> key can either be a json
structure that specifies a view and the arguments to it, or a function that
returns such a structure based on things such as the query string in the URL.</p>
<p>There are two further functions that you will find helpful from time to time. The
first is <tt class="docutils literal">before</tt> that allows you to run some code before the rest of the process
starts, and may do something such as trigger another event. The other is its partner
<tt class="docutils literal">after</tt>, which can do much the same things as <tt class="docutils literal">before</tt>, but can also do things
such as modify the HTML that is output.</p>
<p>Lastly there is another thing that can be done with the HTML that is output,
specified with they <tt class="docutils literal">selectors</tt> key. This allows you to perform an action
on particular parts of the html. The keys of this structure are jQuery
selectors that specify which elements the function will be applied to. For
instance you can do something with all the divs in the output, or all the
spans with a certain class, or the form with a particular id.</p>
<p>What you can do to those elements is basically unlimited, as you can run
arbitrary javascript. However, there is built in support for specifying
an evently widget, which will automatically be bound to each element
that matches the selector. This nesting is one of the most powerful and
useful features of evently, and one you should generally be using often.
I will probably talk more about what nested widgets are useful for later.</p>
</div>
<div class="section" id="special-evently-events">
<h1>Special evently events</h1>
<p>evently has two special events. The first of these is <tt class="docutils literal">_init</tt>. This
event is triggered when the widget is created. This means you can
dynamically pre-populate the element, or at least keep the inital
state of the element with the rest of your code, rather than putting
some in the HTML file and the rest in evently code.</p>
<p>The other special event is tied to couchdb, and is the <tt class="docutils literal">_changes</tt>
event, and is triggered whenever the database that the couchapp is
in changes. This means that you can have elements on the page that
dynamically update whenever the database changes, whether that is
through user action, another user doing something, external scripts,
or couchdb replication. This makes it very easy to write &quot;live&quot;
pages that show updates without refreshes, and is very useful for
some applications.</p>
<p>Currently <tt class="docutils literal">_changes</tt> doesn't receive the modified documents, so
it is normally just used to make another request to get the updated
information, whether that be through <tt class="docutils literal">async</tt> or <tt class="docutils literal">view</tt>. If
you wish to get the modified documents in order to update the page
directly and reduce requests then you can write some custom code
to do this.</p>
</div>
<div class="section" id="conclusion">
<h1>Conclusion</h1>
<p>As you have seen, evently is just a thin layer on top of jQuery concepts
such as events and asynchronous events, with some conveniences for
templating and interacting with couchdb.</p>
<p>This combination is well suited to the needs of at least simple
and moderately complex couchapps, while still being very powerful,
and allowing you to fall back to custom javascript at any point.</p>
<p>You can find more about evently at <a class="reference external" href="http://couchapp.org/page/evently">the couchapp site.</a></p>
<p>See <a class="reference external" href="http://jameswestby.net/weblog/tech/18-couchapp-walkthrough-part-1.html">part one</a> and <a class="reference external" href="http://jameswestby.net/weblog/tech/19-couchapp-walkthrough-part-2-the-couchapp-tool.html">part two.</a></p>
</div>
</div>

]]></description>
   <category domain="http://jameswestby.net/weblog"></category>
   <pubDate>Sat, 09 Oct 2010 00:56 GMT</pubDate>
</item>
<item>
   <title>Couchapp Walkthrough: Part 2: The couchapp tool</title>
   <guid isPermaLink="false">tech/19-couchapp-walkthrough-part-2-the-couchapp-tool</guid>
   <link>http://jameswestby.net/weblog/tech/19-couchapp-walkthrough-part-2-the-couchapp-tool.html</link>
   <description><![CDATA[
<div class="document">
<p>Today I would like to talk about the couchapp tool. This is something that you can use when working on couchapps, and provides a way to quickly iterate your design.</p>
<p>However, rather confusingly, the couchapp tool isn't actually required for couchapps. If you get a design document with HTML attachments in to your database then you have a couchapp.</p>
<p>Why would you want to use such a tool then? Firstly because it will generate a skeleton couchapp for you, so that you don't have to remember how to organise it, and if it is your first couchapp it's good to start from something working.</p>
<p>More importantly though, the couchapp tool is useful as you develop as it allows you to edit the parts of your app in their native format. This means that if you are writing a HTML snippet you can just put some HTML in a file, rather than having to write it as part of a deeply nested JSON structure and deal with the errors that you would make if you did that. Also it means that you can use things like syntax highlighting in your preferred text editor without any special magic.</p>
<div class="section" id="how-it-works">
<h1>How it works</h1>
<p>At it's core couchapp is a rather simple tool. It walks a filesystem tree and assembles the things that it finds there in to a JSON document.</p>
<p>For instance, if it finds a directory at the root of the tree called <tt class="docutils literal">_attachments</tt> it puts the content of each file there in to a list which it puts in to the dict with an &quot;_attachments&quot; key, Which is one of the ways that couchdb accepts document attachments. Therefore if you have an <tt class="docutils literal">_attachments/index.html</tt> file in your tree it will be attached to your design document when the JSON structure is sent to couchdb.</p>
<p>This continues across the tree, so the contents of the &quot;views&quot; directory will become the &quot;views&quot; key of the documents, which is how you do map/reduce queries on the database.</p>
<p>couchapp has various conventions for dealing with files. For instance if it finds a &quot;.js&quot; file it treats it as a javascript snippet which will be encoded in to a string in the resulting document. &quot;.html&quot; files outside of the &quot;_attachments&quot; directory will also be encoded as strings. If it finds a &quot;.json&quot; file then it treats it as literal JSON that will be embebedded.</p>
<p>This way it builds up the JSON structure that a couchapp expects, and will send it to the couchdb of your choice when it is done.</p>
<p>In addition to this functionality the tool can also generate you a skeleton app, and also add new pieces to your app, such as new views.</p>
</div>
<div class="section" id="getting-it">
<h1>Getting It</h1>
<p>couchapp is a python tool, so you can install it using pip or similar. However, Ubuntu users can <a class="reference external" href="https://launchpad.net/~couchapp/+archive/couchapp">install it from a PPA</a> (yay for daily builds with recipes!).</p>
</div>
<div class="section" id="using-it">
<h1>Using It</h1>
<p>To use it run</p>
<pre class="literal-block">
couchapp generate myapp
</pre>
<p>which will create you a new skeleton in myapp.</p>
<pre class="literal-block">
cd myapp
ls
</pre>
<p>You will see for instance the <tt class="docutils literal">_attachments</tt> and <tt class="docutils literal">views</tt> directories, and an <tt class="docutils literal">_attachments/index.html.</tt></p>
<p>To get your app in to couchdb you can run</p>
<pre class="literal-block">
couchapp push http://localhost:5984/mydb
</pre>
<p>and it will tell you the URL to visit to see your new app.</p>
<p>If you want to use desktopcouch you can run</p>
<pre class="literal-block">
couchapp push desktopcouch://
</pre>
<p>though I think it has a bug that it prints the wrong URLs when pushing to desktopcouch.</p>
<p>Once you have looked at the HTML generated by your app you should look at the design document that couchapp created. Go to</p>
<pre class="literal-block">
http://localhost:5984/_utils
</pre>
<p>or</p>
<pre class="literal-block">
xdg-open ~/.local/share/desktop-couch/couchdb.html
</pre>
<p>if you are using desktopcouch.</p>
<p>Click to the mydb database and you will see a document called <tt class="docutils literal">_design/myapp</tt>. Click on this and you will see the content of the design document; you are looking at a couchapp in its raw form.</p>
<p>If you compare what is in that design document with what is in the myapp directory that the tool created you should start to see how it generates it from the filesystem.</p>
<p>Now try making a change on the filesystem, for instance edit <tt class="docutils literal">_attachments/index.html</tt> and put your name somewhere in the body. Then push again, running</p>
<pre class="literal-block">
couchapp push http://localhost:5984/mydb
</pre>
<p>and refresh the page in your browser and you should see the change. (Just click on <tt class="docutils literal">index.html</tt> in the design document to get back to viewing your app from there).</p>
<p>I will go in to more detail about the content of the couchapp that was generated for you in another post.</p>
<p>See the <a class="reference external" href="/weblog/tech/18-couchapp-walkthrough-part-1.html">previous installment.</a></p>
</div>
</div>

]]></description>
   <category domain="http://jameswestby.net/weblog"></category>
   <pubDate>Wed, 29 Sep 2010 23:48 GMT</pubDate>
</item>
<item>
   <title>Couchapp Walkthrough - Part 1</title>
   <guid isPermaLink="false">tech/18-couchapp-walkthrough-part-1</guid>
   <link>http://jameswestby.net/weblog/tech/18-couchapp-walkthrough-part-1.html</link>
   <description><![CDATA[
<div class="document">
<p>Couchapps are a particular way of using couchdb that allow you to
serve web applications directly from the database. These applications generate
HTML and javascript to present data from couchdb to the user,
and then update the database and the UI based on their actions.</p>
<p>Of course there are plenty of frameworks out there that do this sort of thing,
and more and more of them are adding couchdb support. What
makes couchapps particularly interesting are two things. Firstly, the
ease with which they can be developed and deployed. As they are served
directly from couchdb they require little infrastructure, and the
couchapp tool allows for a rapid iteration. In addition, the conveniences
that are provided mean that simple things can be done very quickly with
little code.</p>
<p>The other thing that makes couchapps attractive is that they live inside
the database. This means that the code lives alongside the data, and will
travel with it as it is replicated. This means that you can easily have
an app that you have fast, local access to on your desktop, while
at the same time replicating to a server so that you can access the same
data from your phone while you are out. Again, this doesn't require
couchapps, and they won't be suitable for all needs, but they are certainly
an interesting idea.</p>
<p>You can read more about couchapps at <a class="reference external" href="http://couchapp.org">http://couchapp.org</a>.</p>
<p>Intrigued by couchapps I set out to play with them over a weekend. Unfortunately
the documentation is rather lacking currently, so I wouldn't recommend experimenting
yourself if you are not happy digging around for answers, and sometimes
not finding them outside the code. In order to
go a little way to rectifying this, I intend to write a few posts about
the things I wish I had known when I started out. I found everything to be
a little strange at first, and it wasn't even clear where the entry point
of a couchapp was for instance. Hopefully these posts will be found using
google by others who are struggling in a similar way.</p>
<div class="section" id="architecture">
<h1>Architecture</h1>
<p>Firstly something about the pieces that make up a couchapp (or at least those
that the tool and documentation recommend,) and the way that they all fit together.</p>
<p>At the core is the couchdb database itself. It is a collection of &quot;documents&quot;,
each of which can have attachments. Some of these documents are known as
&quot;design documents,&quot; and they start with a prefix of &quot;_design.&quot; Design
documents can have &quot;view&quot; functions, and various other special fields
that can be used to query or manipulate other documents.</p>
<p>A couchapp is a design document with an attachment, usually called index.html.
These attachments are served directly by couchdb and can be accessed at a
known URL. You can put anything you like in that html file, and you could
just have a static page if you wanted. Usually however though it is is
an HTML page that uses javascript in order to display the results
of queries on the database. The user will then access the attachment on
the design document, and will interact with the resulting page.</p>
<p>In theory you can do anything you like in that page, but it is usual
to make use of standard tools in order to query the database and
provide information and opportunity for interaction to the user.</p>
<p>The first standard tool is jQuery, with a couple of plugins for
working with couchdb and couchapps specifically. These allow for
querying views in the database and acting on the results, retrieving
and updating documents, and plenty more.</p>
<p>In addition the couchapp tool sets you up with another jQuery
plugin called &quot;evently&quot;, which is a way to structure interactions
with jQuery, and change the page based on various events. I will
go in to more detail about how evently works in a later post.</p>
<p>In addition to all the client-side tools for interacting with the
database, it is also possible to make use of couchdb features such
as shows, lists, update handlers validation functions in order to move
some of the processing server-side. This is useful for various reasons,
including being more accessible, allowing search engines to index the
content, and not having to trust the client not to take malicious
actions.</p>
<p>The two approaches can be combined, and you can prototype with the
client-side tools, and then move some of the work to the server-side
facilities later.</p>
<p>Stay tuned for more on how a simple couchapp generates content based
on what is in the db.</p>
</div>
</div>

]]></description>
   <category domain="http://jameswestby.net/weblog"></category>
   <pubDate>Tue, 28 Sep 2010 23:48 GMT</pubDate>
</item>
<item>
   <title>Directly logging in a user in Django tests</title>
   <guid isPermaLink="false">tech/17-directly-logging-in-a-user-in-django-tests</guid>
   <link>http://jameswestby.net/weblog/tech/17-directly-logging-in-a-user-in-django-tests.html</link>
   <description><![CDATA[
<div class="document">
<p>The <a class="reference external" href="http://docs.djangoproject.com/en/dev/topics/testing/#django.test.client.Client.login">examples for Django testing</a> point you towards hardcoding a
username and password for a user to impersonate in tests, and
the API of the test client encourages this too.</p>
<p>However, Django has a nice pluggable authentication system that
means you can easily use something such as OpenID instead of
passwords.</p>
<p>Putting the passwords in your tests ties you to having the password
support enabled, and while you could do this for just the tests, it's
completely out of the scope of most tests (I'm not talking about any
tests for the actual login process here.)</p>
<p>When I saw this while reviewing code recently I worked with <a class="reference external" href="https://launchpad.net/~zkrynicki">Zygmunt</a>
to write a Client subclass that didn't have this restriction. With this
subclass you can just choose a User object, and have that client login
as that user, without them having to have a password at all. Doing
this decoples your tests from the implementation of the authentication
system, and makes them target the code you want to test more precisely.</p>
<p>Here's the code:</p>
<pre class="literal-block">
from django.conf import settings
from django.contrib.auth import login
from django.http import HttpRequest
from django.test.client import Client


class TestClient(Client):

    def login_user(self, user):
        &quot;&quot;&quot;
        Login as specified user, does not depend on auth backend (hopefully)

        This is based on Client.login() with a small hack that does not
        require the call to authenticate()
        &quot;&quot;&quot;
        if not 'django.contrib.sessions' in settings.INSTALLED_APPS:
            raise AssertionError(&quot;Unable to login without django.contrib.sessions in INSTALLED_APPS&quot;)
        user.backend = &quot;%s.%s&quot; % (&quot;django.contrib.auth.backends&quot;,
                                  &quot;ModelBackend&quot;)
        engine = import_module(settings.SESSION_ENGINE)

        # Create a fake request to store login details.
        request = HttpRequest()
        if self.session:
            request.session = self.session
        else:
            request.session = engine.SessionStore()
        login(request, user)

        # Set the cookie to represent the session.
        session_cookie = settings.SESSION_COOKIE_NAME
        self.cookies[session_cookie] = request.session.session_key
        cookie_data = {
            'max-age': None,
            'path': '/',
            'domain': settings.SESSION_COOKIE_DOMAIN,
            'secure': settings.SESSION_COOKIE_SECURE or None,
            'expires': None,
        }
        self.cookies[session_cookie].update(cookie_data)

        # Save the session values.
        request.session.save()
</pre>
<p>Then you can use it in your tests like this:</p>
<pre class="literal-block">
from django.contrib.auth.models import User


client = TestClient()
user = User(username=&quot;eve&quot;)
user.save()
client.login_user(user)
</pre>
<p>Then any requests you make with that client will be authenticated as the user that
was created.</p>
<p><a class="reference external" href="http://code.djangoproject.com/ticket/14350">Ticket submitted</a> with Django to have this available for everyone in future.</p>
</div>

]]></description>
   <category domain="http://jameswestby.net/weblog"></category>
   <pubDate>Tue, 28 Sep 2010 00:50 GMT</pubDate>
</item>
<item>
   <title>Improving the usability of launchpadlib-using code</title>
   <guid isPermaLink="false">tech/16-Improving-the-usability-of-launchpadlib-using-code</guid>
   <link>http://jameswestby.net/weblog/tech/16-Improving-the-usability-of-launchpadlib-using-code.html</link>
   <description><![CDATA[
<div class="document">
<p>Normally when you write some code using launchpadlib you end up with Launchpad
showing your users something like this:</p>
<img alt="/images/lplib-before.png" src="/images/lplib-before.png" />
<p>This isn't great, how is the user supposed to know which option to click? What
do you do if they don't choose the option you want?</p>
<p>Instead it's possible to limit the choices that the user has to make to only
those that your application can use, plus the option to deny all access, by
changing the way you create your Launchpad object.</p>
<pre class="literal-block">
from launchpadlib.launchpad import Launchpad

lp = Launchpad.get_token_and_login(&quot;testing&quot;, allow_access_levels=[&quot;WRITE_PUBLIC&quot;])
</pre>
<p>This will present your users with something like this:</p>
<img alt="/images/lplib-after.png" src="/images/lplib-after.png" />
<p>which is easier to understand. There could be further improvements, but they would
happen on the Launchpad side.</p>
<p>This approach works for both Launchpad.get_token_and_login and Launchpad.login_with.</p>
<p>The values that you can pass here aren't documented, and should probably be constants
in launchpadlib, rather than hardcoded in every application, but for now you can
use:</p>
<ul class="simple">
<li>READ_PUBLIC</li>
<li>READ_PRIVATE</li>
<li>WRITE_PUBLIC</li>
<li>WRITE_PRIVATE</li>
</ul>
</div>

]]></description>
   <category domain="http://jameswestby.net/weblog"></category>
   <pubDate>Wed, 07 Jul 2010 20:17 GMT</pubDate>
</item>
<item>
   <title>Re: Getting the hobbyist back</title>
   <guid isPermaLink="false">tech/15-Re-Getting-the-hobbyist-back</guid>
   <link>http://jameswestby.net/weblog/tech/15-Re-Getting-the-hobbyist-back.html</link>
   <description><![CDATA[
<div class="document">
<p>Dear <a class="reference external" href="http://blogs.gnome.org/bolsh/2010/04/29/getting-the-hobbyist-back/">Mr Neary</a>, thanks for your thought provoking post, I think it is a
problem we need to be aware of as Free Software matures.</p>
<p>Firstly though I would like to say that the apparent ageism present in your
argument isn't helpful to your point. Your comments appear to diminish the
contributions of a whole generation of people. In addition, we shouldn't just
be concerned with attracting young people to contribute, the same changes will
have likely reduced the chances that people of all ages will get involved.</p>
<p>Aside from that though there is much to discuss. You talk about the changes in
Free Software since you got involved, and it mirrors my observations. While these
changes may have forced fewer people to learn all the details of how the system
works, they have certainly allowed more people to use the software, bringing many
different skills to the party with them.</p>
<p>I would contend that often the experience for those looking to do the compilation
that you rate as important has parallels to the experience of just using the software
you present from a few years ago. If we can change that experience as much as we
have the installation and first use experience then we will empower more people to
take part in those activities.</p>
<p>It is instructive then to look at how the changes came about to see if there are
any pointers for us. I think there are two causes of the change that are of interest
to this discussion.</p>
<p>Firstly, one change has been an increased focus on user experience. Designing
and building software that serves the users needs has made it much more palatable
for people, and reduced the investment that people have to make before using it.
In the same way I think we should focus on developer experience, making it more
pleasant to perform some of the tasks needed to be a hobbyist. Yes, this means
hiding some of the complexity to start with, but that doesn't mean that it can't
be delved in to later. Progressive exposure will help people to learn by not
requiring them to master the art before being able to do anything.</p>
<p>Secondly, there has been a push to make informed decisions on behalf of the user
when providing them with the initial experience. You no longer get a base system
after installation, upon which you are expected to select from the thousands of
packages to build your perfect environment. Neither are you led to download multiple
CDs that contain the entire contents of a distribution, much of which is installed
by default. Instead you are given an environment that is already equipped to do
common tasks, where each task is covered by an application that has been selected
by experts on your behalf.</p>
<p>We should do something similar with developer tools, making opinionated decisions
for the new developer, and allowing them to change things as they learn, similar
to the way in which you are still free to choose from the thousands of packages
in the distribution repositories. Doing this makes documentation easier to write,
allows for knowledge sharing, and reduces the chances of paralysis of choice.</p>
<p>There are obviously difficulties with this given that often the choice of tool
that one person makes on a project dicatates or heavily influences the choice
other people have to make. If you choose autotools for your projects then I can't
build it with CMake. Our development tools are important to us as they shape
the environment in which we work, so there are strong opinions, but perhaps
consistency could become more of a priority. There are also things we can do
with libraries, format specifications and wrappers to allow choice while still
providing a good experience for the fledgling developer.</p>
<p>Obviously as we are talking about free software the code will always be available,
but that isn't enough in my mind. It needs to be easier to go from code to
something you can install and remove, allowing you to dig deeper once you have
achieved that.</p>
<p>I believe that our effort around things like <a class="reference external" href="https://dev.launchpad.net/BuildBranchToArchive">https://dev.launchpad.net/BuildBranchToArchive</a>
will go some way to helping with this.</p>
</div>

]]></description>
   <category domain="http://jameswestby.net/weblog"></category>
   <pubDate>Fri, 30 Apr 2010 13:59 GMT</pubDate>
</item>
<item>
   <title>Caution: python-multiprocessing, threads and glib don't mix</title>
   <guid isPermaLink="false">tech/14-caution-python-multiprocessing-and-glib-dont-mix</guid>
   <link>http://jameswestby.net/weblog/tech/14-caution-python-multiprocessing-and-glib-dont-mix.html</link>
   <description><![CDATA[
<div class="document">
<p>If you don't want to read this article, then just steer clear of
python-multiprocessing, threads and glib in the same application. Let me
explain why.</p>
<p>There's a rather <a class="reference external" href="https://bugs.edge.launchpad.net/ubuntu/+source/gwibber/+bug/554005">famous bug</a> in <a class="reference external" href="https://launchpad.net/gwibber">Gwibber</a> in Ubuntu Lucid, where
a gwibber-service process will start taking 100% of the CPU time
of one of your cores if it can. While looking in to why this bug
happened I learnt a lot about how multiprocessing and GLib work,
and wanted to record some of this so that others may avoid the
bear traps.</p>
<p>Python's <a class="reference external" href="http://docs.python.org/library/multiprocessing.html">multiprocessing module</a> is a nice module to allow you to
easily run some code in a subprocess, to get around the restriction of
the GIL for example. It makes it really easy to run a particular function
in a subprocess, which is a step up from what you had to do before it
existed. However, when using it you should be aware how the way it works
can interact with the rest of your app, because there are some possible
nasties lurking there.</p>
<p><cite>GLib</cite> is a set of building blocks for apps, most notably used by GTK+.
It provides an object system, a mainloop and lots more besides. What we are
most interested here is the mainloop, signals, and thread integration that
it provides.</p>
<p>Let's start the explanation by looking at how multiprocessing does its thing.
When you start a subprocess using multiprocessing.Process, or something that
uses it, it causes a fork(2), which starts a new process with a copy of the
programs current memory, with some exceptions. This is really nice for
multiprocessing, as you can just run any code from that program in the
subprocess and pass the result back without too much difficulty.</p>
<p>The problems occur because there isn't an exec(3) to accompany the fork(2).
This is what makes multiprocessing so easy to use, but doesn't insert a clean
process boundary between the processes. Most notably for this example, it
means the child inherits the file descriptors of the parent (critically even
those marked FD_CLOEXEC).</p>
<p>The other piece to this puzzle is how the GLib mainloop communicates
between threads. It requires some mechanism where one thread can alert
another that something of interest happened. To do this when you tell
GLib that you will be using threads in your app by calling g_thread_init
(gobject.threads_init() in Python) then it will create a pipe for use by
glib to alert other threads.  It also creates a watcher thread that
polls one end of this pipe so that it can act when a thread wishes to
pass something on to the mainloop.</p>
<p>The final part of the puzzle is what your app does in a subprocess with
mutliprocessing. If you purely do something such as number crunching
then you won't have any issues. If however you use some glib functions
that will cause the child to communicate with the mainloop then you
will see problems.</p>
<p>As the child inherits the file descriptors of the parent it will use the
same pipe for communication. Therefore if a function in the child writes
to this pipe then it can put the parent in to a confused state. What
happens in gwibber is that it uses some gnome-keyring functions and that
puts the parent in to a state where the watcher thread created by
g_thread_init busy-polls on the pipe, taking up as much CPU time as it can
get from one core.</p>
<p>In summary, you will see issues if you use python-multiprocessing from
a thread and use some glib functions in the children.</p>
<p>There are some ways to fix this, but no silver bullet:</p>
<blockquote>
<ul class="simple">
<li>Don't use threads, just use multiprocessing. However, you can't
communicate with glib signals between subprocesses, and there's
no equivalent built in to multiprocessing.</li>
<li>Don't use glib functions from the children.</li>
<li>Don't use multiprocessing to run the children, use exec(3) a script
that does what you want, but this isn't as flexible or as
convenient.</li>
</ul>
</blockquote>
<p>It may be possible to use the support for different GMainContexts for
different threads to work around this, but:</p>
<blockquote>
<ul class="simple">
<li>You can't access this from Python, and</li>
<li>I'm not sure that every library you use will correctly implement it,
and so you may still get issues.</li>
</ul>
</blockquote>
<p>Note that none of the parties here are doing anything particularly
wrong, it's a bad interaction caused by some decisions that are known to
cause issues with concurrency. I also think there are issues when using
DBus from multiprocessing children, but I haven't thoroughly
investigated that. I'm not entirely sure why the multiprocessing child
seems to have to be run from a non-main thread in the parent to trigger
this, any insight would be welcome. You can find a small script to
reproduce the problem <a class="reference external" href="http://jameswestby.net/scratch/multiprocessing_bug.py">here</a>.</p>
<p>Or, to put it another way, global state bad for concurrency.</p>
</div>

]]></description>
   <category domain="http://jameswestby.net/weblog"></category>
   <pubDate>Thu, 08 Apr 2010 12:01 GMT</pubDate>
</item>
<item>
   <title>Some of my best friends are Unicorns</title>
   <guid isPermaLink="false">tech/13-some-of-my-best-friends-are-unicorns</guid>
   <link>http://jameswestby.net/weblog/tech/13-some-of-my-best-friends-are-unicorns.html</link>
   <description><![CDATA[
<div class="document">
<p>As my contribution to Ada Lovelace Day 2010 I would like to mention
<a class="reference external" href="http://emmajane.net">Emma Jane Hogbin</a>.</p>
<p>Emma is an Ubuntu Member, published author, documentation evangelist,
conference organiser, Drupal theming expert, tireless conference presenter,
and many more things as well.</p>
<p>Her enthusiasm is infectious, and her passion for solving problems for people
is admirable. She is a constant source of inspiration to me, and that continues
even as she <a class="reference external" href="http://www.emmajane.net/happy-birthday-agnes-ones-you">branches out in to new things</a>.</p>
<p>(Hat tip for the title to the ever excellent Sharrow)</p>
</div>

]]></description>
   <category domain="http://jameswestby.net/weblog"></category>
   <pubDate>Thu, 25 Mar 2010 01:38 GMT</pubDate>
</item>
<item>
   <title>Commit access is no more</title>
   <guid isPermaLink="false">tech/12-commit-access-is-no-more</guid>
   <link>http://jameswestby.net/weblog/tech/12-commit-access-is-no-more.html</link>
   <description><![CDATA[
<div class="document">
<p>Many projects that I work on, or follow the development of, and granted there may
be a large selection bias here, are showing some of the same tendencies. Combined
these indicate to me that we need to change the way we look at becoming a trusted
member of the project.</p>
<p>The obvious change here is the move to distributed version control. I'm obviously
a fan of this change, and for many reasons. One of those is the democratisation of
the tools. There is no longer a special set of people that gets to use the best
tools, with everyone else having to make do. Now you get to use the same tools
whether you were the founder of the project, or someone working on your first
change. That's extremely beneficial as it means that we don't partition our efforts
to improve the tools we use. It also means that new contributors have an easier
time getting started, as they get to use better tools. These two influences combine
as well: a long time contributor can describe how they achieve something, and the
new contributor can directly apply it, as they use the same tools.</p>
<p>This change does mean that getting &quot;commit access&quot; isn't about getting the ability
to commit anymore; everyone can commit anytime to their own branch. Some projects,
e.g. Bazaar, don't even hand out &quot;commit access&quot; in the literal sense, the project
blessed code is handled by a robot, you just get the ability to have the robot merge
a branch.</p>
<p>While it is true that getting &quot;commit access&quot; was never really about the tools,
it was and is about being trusted to shepherd the shared code, a lot of projects
still don't treat it that way. Once a developer gets &quot;commit access&quot; they just
start committing every half-cooked patch they have to trunk. The full use of
distributed version control, with many branches, just emphasises the shared
code aspect. Anyone is free to create a branch with their half-baked idea and
see if anyone else likes it. The &quot;blessed&quot; branch is just that, one that the
project as a whole decides they will collaborate on.</p>
<p>This leads to my second change, code review. This is something that I also deeply
believe in; it is vital to quality, and a point at which open source software
becomes a massive advantage, so something we should exploit to the full. I see
it used increasingly in many projects, and many moving up jml's <a class="reference external" href="http://mumak.net/stuff/your-code-sucks.html">code review
&quot;ladder&quot;</a> towards pre-merge review of every change. There seems to be increasing
acceptance that code review is valuable, or at least that it is something a good
project does.</p>
<p>Depending on the project the relationship of code review and &quot;commit access&quot; can
vary, but at the least, someone with &quot;commit access&quot; can make their code review
count. Some projects will not allow even those with &quot;commit access&quot; to act
unilaterally, requiring multiple reviews, and some may even relegate the concept,
working off votes from whoever is interested in the change.</p>
<p>At the very least, most projects will have code review when a new contributor
wishes to make a change. This typically means that when you are granted &quot;commit
access&quot; you are able or expected to review other people's code, even though
you may never have done so before. Some projects also require every contribution
to be reviewed, meaning that &quot;commit access&quot; doesn't grant you the ability to
do as you wish, it instead just puts the onus on you to review the code of others
as well as write your own.</p>
<p>As code review becomes more prevalent we need to re-examine what we see as
&quot;commit access,&quot; and how people show that they are ready for it. It may be
that the concept becomes &quot;trusted reviewer&quot; or similar, but at the least
code review will be a large part of it. Therefore I feel that we shouldn't
just be looking at a person's code contributions, but also their code review
contributions. Code review is a skill, some people are very good at it, some
people are very very bad at it. You can improve with practice and teaching,
and you can set bad examples for others if you are not careful. We will
have to make sure that review runs through the blood of a project, everyone
reviews the code of everyone else, and the reviews are reviewed.</p>
<p>The final change that I see as related is that of empowering non-code
contributors. More and more projects are valuing these contributors, and
one important part of doing that is trusting them with responsibility. It
may be that sometimes trusting them means giving them &quot;commit access&quot;,
if they are working on improving the inline help for instance. Yes, it may
be that distributed version control and code review mean that they do
not have to do this, but those arguments could be made for code contributors
too.</p>
<p>This leads me to another, and perhaps the most important, aspect of the
&quot;commit access&quot; idea: trust. The fundamental, though sometimes unspoken,
measure we use to gauge if someone should get &quot;commit access&quot; is whether
we believe them to be trustworthy. Do we trust them to introduce code without
review? Do we trust them to review other people's changes? Do we trust them
to change only those areas they are experts in, or to speak to someone
else if they are not? This is the rule we should be applying when making
this decision, and we should be sure to be aware that this is what we
are doing. There will often be other considerations as well, but this
decision will always factor.</p>
<p>These ideas are not new, and the influences described here did not create
them. However the confluence of them, and the changes that will likely
happen in our projects over the next few years, mean that we must be sure
to confront them. We must discard the &quot;commit access&quot; idea as many projects
have seen it, and come up with new responsibilities that better reflect
the tasks people are doing, the new ways projects operate, and that
reward the interactions that make our projects stronger.</p>
</div>

]]></description>
   <category domain="http://jameswestby.net/weblog"></category>
   <pubDate>Fri, 27 Nov 2009 05:54 GMT</pubDate>
</item>
<item>
   <title>Merging file descriptor output</title>
   <guid isPermaLink="false">tech/11-merging-file-descriptor-output</guid>
   <link>http://jameswestby.net/weblog/tech/11-merging-file-descriptor-output.html</link>
   <description><![CDATA[
<div class="document">
<p>I have a problem that I believe will be easy for someone with a bit of
UNIX coding knowledge to solve, so I appeal to those that can to help.</p>
<p>I'm trying to write a DBus service that will spawn a command, and provide
the output to the user. The service runs on the system bus as root, and
so it is a form of privilege escalation. However, the command may be long
running, and produce a lot of output as it works, so I want to allow the
calling process to get this output before the command completes.</p>
<p>My current approach uses <cite>gobject.spawn_async</cite> and so gets file descriptors
back, one for stdout and one for stderr. I currently have a thread that
uses <cite>select</cite> to wait for output, and then uses DBus signals to allow the
client to access it. This works great, except that stdout and stderr can
become interleaved in the middle of lines.</p>
<p>I believe that I can't just wait for full lines before signalling, as
a command might do something like print &quot;Username: &quot; and then wait for
input. I could normally do full lines, and then if the child blocks on
stdin send whatever it has written so far, but that doesn't seem ideal.
(I haven't implemented anything about proving input on stdin so far,
but I don't want a solution that makes it difficult to do so).</p>
<p>It seems to me that this is something that will be implemented somewhere,
for instance my shell can run commands and then interleave the output
in a desirable manner, but I haven't found how yet. Any suggestions
are welcome, but this is from python, so system calls that I can't make
directly from python would be a pain, though I'm not that bothered about
portability.</p>
</div>

]]></description>
   <category domain="http://jameswestby.net/weblog"></category>
   <pubDate>Mon, 08 Jun 2009 23:38 GMT</pubDate>
</item>
</channel>
</rss>
