<?xml version="1.0" encoding="UTF-8"?>
<?xml-stylesheet type="text/xsl" media="screen" href="/~d/styles/rss2full.xsl"?><?xml-stylesheet type="text/css" media="screen" href="http://feeds.feedburner.com/~d/styles/itemcontent.css"?><rss xmlns:atom="http://www.w3.org/2005/Atom" version="2.0"><channel><title>Review Board Developer Blogs</title><link>http://www.reviewboard.org/news/</link><description>Latest Review Board developer blog posts</description><language>en-us</language><lastBuildDate>Mon, 28 Mar 2011 13:14:17 -0700</lastBuildDate><atom10:link xmlns:atom10="http://www.w3.org/2005/Atom" rel="self" type="application/rss+xml" href="http://feeds.feedburner.com/ReviewBoard" /><feedburner:info xmlns:feedburner="http://rssnamespace.org/feedburner/ext/1.0" uri="reviewboard" /><atom10:link xmlns:atom10="http://www.w3.org/2005/Atom" rel="hub" href="http://pubsubhubbub.appspot.com/" /><item><title>Lady Gaga
</title><link>http://mikeconley.ca/blog/2011/03/28/lady-gaga/</link><description>&lt;p&gt;When I&amp;#8217;m not hacking on &lt;a href="http://www.mozillamessaging.com/en-US/thunderbird/"&gt;Thunderbird&lt;/a&gt;, &lt;a href="http://www.markusproject.org"&gt;MarkUs&lt;/a&gt;, or &lt;a href="http://www.reviewboard.org"&gt;Review Board&lt;/a&gt;, I usually spend my time playing in a band called &lt;a href="http://www.myspace.com/thejohnsonreport"&gt;The Johnson Report&lt;/a&gt;.&lt;/p&gt;
&lt;p&gt;I normally don&amp;#8217;t play the &amp;#8220;band card&amp;#8221;, but I&amp;#8217;m particularly proud of this latest achievement, and wanted to share.&lt;/p&gt;
&lt;p&gt;&lt;a href="https://www.youtube.com/watch?v=Hikfh1TMdEg"&gt;Here&amp;#8217;s us playing some Lady Gaga.&lt;/a&gt;&lt;/p&gt;
&lt;p&gt;&lt;/p&gt;&lt;div class="feedflare"&gt;
&lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=dYgsfjs3AYo:P3nvvwysyiA:yIl2AUoC8zA"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?d=yIl2AUoC8zA" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=dYgsfjs3AYo:P3nvvwysyiA:V_sGLiPBpWU"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=dYgsfjs3AYo:P3nvvwysyiA:V_sGLiPBpWU" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=dYgsfjs3AYo:P3nvvwysyiA:F7zBnMyn0Lo"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=dYgsfjs3AYo:P3nvvwysyiA:F7zBnMyn0Lo" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=dYgsfjs3AYo:P3nvvwysyiA:gIN9vFwOqvQ"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=dYgsfjs3AYo:P3nvvwysyiA:gIN9vFwOqvQ" border="0"&gt;&lt;/img&gt;&lt;/a&gt;
&lt;/div&gt;</description><author>mike.d.conley@gmail.com (mikeconley)</author><pubDate>Mon, 28 Mar 2011 13:14:17 -0700</pubDate><guid>http://mikeconley.ca/blog/2011/03/28/lady-gaga/</guid></item><item><title>That’s all, folks! or Becoming Randall Stevens
</title><link>http://mikeconley.ca/blog/2010/12/23/thats-all-folks-or-becoming-randall-stevens/</link><description>&lt;p&gt;Once again, I&amp;#8217;ve let a month&amp;#8217;s worth of dust gather on my blog.  But I have a good reason for being so busy!&lt;/p&gt;
&lt;p&gt;Several good reasons, actually.&lt;/p&gt;
&lt;p&gt;And here they are:&lt;/p&gt;
&lt;h3&gt;UCOSP has wrapped for the semester&lt;/h3&gt;
&lt;p&gt;This semester, I was a teaching assistant for the &lt;a href="http://www.ucosp.ca"&gt;UCOSP (Undergraduate Capstone Open-source Projects) course&lt;/a&gt;.  I helped out with two projects:  &lt;a href="http://www.markusproject.org"&gt;MarkUs&lt;/a&gt; and &lt;a href="http://www.reviewboard.org"&gt;Review Board&lt;/a&gt;.&lt;/p&gt;
&lt;p&gt;This semester, we saw some outstanding work for both projects.  Lots of great students, lots of good code, &lt;a href="http://www.reviewboard.org/news/2010/12/11/ucosp-2010-project-screencasts/"&gt;lots of leaps forward&lt;/a&gt;.&lt;/p&gt;
&lt;p&gt;I&amp;#8217;m looking forward to helping out next semester with UCOSP.&lt;/p&gt;
&lt;p&gt;I won&amp;#8217;t be doing it as a paid teaching assistant though.  Why?  Well&amp;#8230;&lt;/p&gt;
&lt;h3&gt;I&amp;#8217;ve finished school&lt;/h3&gt;
&lt;p&gt;&lt;a href="http://mikeconley.ca/blog/2010/12/23/the-wisdom-of-peers-a-motive-for-exploring-peer-code-review-in-the-classroom/"&gt;My research paper&lt;/a&gt; was signed off by my two readers, and I just wrote my last final exam a few nights ago.  Unofficial grades have been posted, and I&amp;#8217;ve passed what I needed to pass.&lt;/p&gt;
&lt;p&gt;So that&amp;#8217;s that &amp;#8211; I&amp;#8217;m a Master of Computer Sciences, I guess.  Awesome!&lt;/p&gt;
&lt;h3&gt;I got a job!&lt;/h3&gt;
&lt;p&gt;I&amp;#8217;ve been hired by &lt;a href="http://www.mozillamessaging.com"&gt;Mozilla Messaging&lt;/a&gt; to work on &lt;a href="http://www.mozillamessaging.com/en-GB/thunderbird/"&gt;the Thunderbird project&lt;/a&gt;!  I&amp;#8217;m 100% psyched about this opportunity, and look forward to peeling into the code.  An added bonus:  since Thunderbird is an open-source project, I&amp;#8217;m absolutely free to discuss the code and the various things I&amp;#8217;m doing with it.  No NDAs for me!  So stay tuned &amp;#8211; I&amp;#8217;ll have lots to say about Thunderbird and the Mozilla Framework code.  Just give me some time to wade through it.&lt;/p&gt;
&lt;h3&gt;Zihuatanejo&lt;/h3&gt;
&lt;p&gt;It&amp;#8217;s been a pretty long road.  I&amp;#8217;ve been in school, in one form or another, for over two decades.  It&amp;#8217;s strange that it&amp;#8217;s over.  I&amp;#8217;m extremely excited about my next adventures, but I think I&amp;#8217;m going to miss school.&lt;/p&gt;
&lt;p&gt;Still, I can&amp;#8217;t help but be a bit dramatic&amp;#8230;&lt;/p&gt;
&lt;div class="shashin_image" style="width: 298px; margin-left: auto; margin-right: auto;"&gt;&lt;a class="highslide" href="http://lh3.ggpht.com/_Kib24bTtAAU/TRA5pkQoAjI/AAAAAAAABcE/xspZS96MbeQ/shawshank_redemption_4.jpg?imgmax=640" id="shashin_thumb_link_3"&gt;&lt;img alt="" height="161" id="shashin_thumb_image_3" src="http://lh3.ggpht.com/_Kib24bTtAAU/TRA5pkQoAjI/AAAAAAAABcE/xspZS96MbeQ/shawshank_redemption_4.jpg?imgmax=288" title="" width="288" /&gt;&lt;/a&gt;&lt;/div&gt;
&lt;blockquote&gt;&lt;p&gt;In 1966, Andy Dufresne escaped from Shawshank prison. All they found of him was a muddy set of prison clothes, a bar of soap, and an old rock hammer, damn near worn down to the nub. I remember thinking it would take a man six hundred years to tunnel through the wall with it. Old Andy did it in less than twenty. Oh, Andy loved geology. I imagine it appealed to his meticulous nature. An ice age here, million years of mountain building there. Geology is the study of pressure and time. That&amp;#8217;s all it takes really, pressure, and time. &amp;#8230;Andy crawled to freedom through five hundred yards of shit smelling foulness I can&amp;#8217;t even imagine, or maybe I just don&amp;#8217;t want to. Five hundred yards&amp;#8230; that&amp;#8217;s the length of five football fields, just shy of half a mile&amp;#8230;&lt;/p&gt;
&lt;p&gt;Andy Dufresne &amp;#8211; who crawled through a river of shit and came out clean on the other side.&lt;/p&gt;&lt;/blockquote&gt;
&lt;p&gt;P.S.:  &lt;a href="http://mikeconley.ca/blog/2009/04/21/thats-all-folks-now-for-celebration-rituals/"&gt;Here are some celebration rituals, if so inclined.&lt;/a&gt;&lt;/p&gt;&lt;div class="feedflare"&gt;
&lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=zgDmlx8m2Pg:9v0Hch9D5o0:yIl2AUoC8zA"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?d=yIl2AUoC8zA" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=zgDmlx8m2Pg:9v0Hch9D5o0:V_sGLiPBpWU"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=zgDmlx8m2Pg:9v0Hch9D5o0:V_sGLiPBpWU" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=zgDmlx8m2Pg:9v0Hch9D5o0:F7zBnMyn0Lo"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=zgDmlx8m2Pg:9v0Hch9D5o0:F7zBnMyn0Lo" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=zgDmlx8m2Pg:9v0Hch9D5o0:gIN9vFwOqvQ"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=zgDmlx8m2Pg:9v0Hch9D5o0:gIN9vFwOqvQ" border="0"&gt;&lt;/img&gt;&lt;/a&gt;
&lt;/div&gt;</description><author>mike.d.conley@gmail.com (mikeconley)</author><pubDate>Thu, 23 Dec 2010 19:28:01 -0800</pubDate><guid>http://mikeconley.ca/blog/2010/12/23/thats-all-folks-or-becoming-randall-stevens/</guid></item><item><title>Stallin’…
</title><link>http://mikeconley.ca/blog/2010/11/03/stallin/</link><description>&lt;p&gt;I know, I know.  I left you all hanging at the edge of your seat with my last blog post, and I still haven&amp;#8217;t posted my idea for &lt;a href="http://mikeconley.ca/blog/2010/10/23/recognizing-good-code-review/"&gt;recognizing good code review&lt;/a&gt;.&lt;/p&gt;
&lt;p&gt;I&amp;#8217;m bogged down with school work, and I&amp;#8217;m aiming to have the first draft of my research paper done next week.  So that&amp;#8217;s taking 100% of my resources.&lt;/p&gt;
&lt;p&gt;Just be patient.  I&amp;#8217;ll post my idea soon.&lt;/p&gt;
&lt;div class="wp_likes" id="wp_likes_post-1882"&gt;&lt;a class="like" href="javascript:wp_likes.like(1882);" title=""&gt;&lt;img alt="" border="0" src="http://mikeconley.ca/blog/wp-content/plugins/wp-likes/images/like.png" /&gt;Like&lt;/a&gt;&lt;span class="text"&gt;&lt;/span&gt;
&lt;div class="unlike"&gt;&lt;a href="javascript:wp_likes.unlike(1882);"&gt;Unlike&lt;/a&gt;&lt;/div&gt;
&lt;/div&gt;&lt;div class="feedflare"&gt;
&lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=KPm6g-DKB68:qEWVoCYe9Gw:yIl2AUoC8zA"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?d=yIl2AUoC8zA" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=KPm6g-DKB68:qEWVoCYe9Gw:V_sGLiPBpWU"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=KPm6g-DKB68:qEWVoCYe9Gw:V_sGLiPBpWU" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=KPm6g-DKB68:qEWVoCYe9Gw:F7zBnMyn0Lo"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=KPm6g-DKB68:qEWVoCYe9Gw:F7zBnMyn0Lo" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=KPm6g-DKB68:qEWVoCYe9Gw:gIN9vFwOqvQ"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=KPm6g-DKB68:qEWVoCYe9Gw:gIN9vFwOqvQ" border="0"&gt;&lt;/img&gt;&lt;/a&gt;
&lt;/div&gt;</description><author>mike.d.conley@gmail.com (mikeconley)</author><pubDate>Wed, 03 Nov 2010 19:33:49 -0700</pubDate><guid>http://mikeconley.ca/blog/2010/11/03/stallin/</guid></item><item><title>Recognizing Good Code Review
</title><link>http://mikeconley.ca/blog/2010/10/23/recognizing-good-code-review/</link><description>&lt;p&gt;While the benefits of code review are &lt;a href="http://www.mfagan.com/ibmfagan.pdf"&gt;proven&lt;/a&gt;, &lt;a href="http://smartbear.com/docs/BestPracticesForPeerCodeReview.pdf"&gt;documented&lt;/a&gt;, &lt;a href="http://smartbear.com/codecollab-code-review-book.php"&gt;numerous&lt;/a&gt; and &lt;a href="http://www.reviewboard.org"&gt;awesome&lt;/a&gt;, it doesn&amp;#8217;t change the fact that most people, in general, don&amp;#8217;t like doing it.&lt;/p&gt;
&lt;p&gt;I guess code review just isn&amp;#8217;t really all that fun.&lt;/p&gt;
&lt;p&gt;So a few months ago, &lt;a href="http://mikeconley.ca/blog/2010/03/09/turning-peer-code-review-into-a-game/"&gt;I broadcast the idea of turning code review into a game.&lt;/a&gt; It was my way of trying to mix things up &amp;#8211; &amp;#8220;let&amp;#8217;s add points, and have reviewers/developers competing to be the best participant in the code review process&amp;#8221;.&lt;/p&gt;
&lt;p&gt;Well, if there&amp;#8217;s one thing that &lt;a href="http://www.third-bit.com"&gt;my supervisor Greg&lt;/a&gt; has taught me, it&amp;#8217;s how I shouldn&amp;#8217;t rush headlong into something before all of the facts are in.  So before I decide to do something like game-ifize code review, I should take a look at some prior work in the area&amp;#8230;&lt;/p&gt;
&lt;p&gt;Enter this guy:  &lt;a href="http://cargocollective.com/codingconduct"&gt;Sebastian Deterding&lt;/a&gt;.&lt;/p&gt;
&lt;p&gt;In particular, check out the following slide-show.  Flip through it if you have the time.  If you don&amp;#8217;t have the time, scroll down, where I get to the salient point with respect to game-ificating code review.&lt;/p&gt;
&lt;p&gt;&lt;a href="http://www.slideshare.net/dings/pawned-gamification-and-its-discontents"&gt;Here&amp;#8217;s the slide-show.&lt;/a&gt; Be sure to read the narrative at the bottom.&lt;/p&gt;
&lt;h3&gt;The Salient Point&lt;/h3&gt;
&lt;p&gt;Sebastian seems to be saying that adding points to apps and trying to incite competition does not make something a game.  If it did, &lt;a href="http://progresswars.com/"&gt;then this should be countless hours of fun&lt;/a&gt;.&lt;/p&gt;
&lt;p&gt;Without play, there is no game.  Points do not equal a game.  It&amp;#8217;s not nearly that simple.&lt;/p&gt;
&lt;h3&gt;Free Pizza and Pop&lt;/h3&gt;
&lt;p&gt;I&amp;#8217;m going to divert for a second here.&lt;/p&gt;
&lt;p&gt;Last week, a company set themselves up a couple of booths in the lobby of the Bahen Center where I work.  They were there to recruit university students to work for their company &amp;#8211; either as interns, or full-timers.&lt;/p&gt;
&lt;p&gt;They were also handing out free pizza and pop.&lt;/p&gt;
&lt;p&gt;Needless to say, I wanted a few slices &amp;#8211; but I figured it would be polite if I engaged them in conversation before waltzing off with some of the free food and drink they&amp;#8217;d brought.&lt;/p&gt;
&lt;p&gt;So I sparked up a conversation with one of the recruiters, and he told me about the company.  I&amp;#8217;m going to call this recruiter Vlad.&lt;/p&gt;
&lt;p&gt;I ended up gently steering the conversation towards code review, and I asked my inevitable question:&lt;/p&gt;
&lt;blockquote&gt;&lt;p&gt;&amp;#8220;So, do you guys do code review?&amp;#8221;&lt;/p&gt;&lt;/blockquote&gt;
&lt;p&gt;I felt like a dentist asking a patient if he&amp;#8217;s been flossing.  Vlad waffled a bit, but the general impression was:&lt;/p&gt;
&lt;blockquote&gt;&lt;p&gt;&amp;#8220;Not as much as we should.  We don&amp;#8217;t have a prescribed workflow. It&amp;#8217;d be hard to persuade all of the teams to do it.&amp;#8221;&lt;/p&gt;&lt;/blockquote&gt;
&lt;p&gt;And then we started talking about code review in general.  It turns out that Vlad had worked in a few companies where they&amp;#8217;d done code review, and he always felt a little short changed.  He said something along the lines of:&lt;/p&gt;
&lt;blockquote&gt;&lt;p&gt;&amp;#8220;I never felt compelled to do reviews.  They just sort of happened&amp;#8230;and I did it, and it felt like&amp;#8230;unrecognized effort.  I mean, what&amp;#8217;s the incentive?  Do you know what I mean?  There&amp;#8217;s incentive for the software, but I&amp;#8217;m talking incentive for &lt;em&gt;me&lt;/em&gt;.  And some people did really lousy reviews&amp;#8230;but my reviews were treated the same as theirs.  I didn&amp;#8217;t get recognized, and didn&amp;#8217;t get rewarded if I did a good review.  So it was hard for me to do them.  I want to be recognized for my good reviews, for my good contributions.&amp;#8221;&lt;/p&gt;&lt;/blockquote&gt;
&lt;p&gt;I wish I&amp;#8217;d had a tape-recorder running so I could have gotten Vlad&amp;#8217;s exact words.  But that&amp;#8217;s what I remember him saying.&lt;/p&gt;
&lt;h3&gt;Feedback and Recognition&lt;/h3&gt;
&lt;p&gt;Maybe instead of trying to game-ulize code review, I can instead hear what Vlad is saying and work off of that.&lt;/p&gt;
&lt;p&gt;With the code review that Vlad participated in, &lt;strong&gt;all of the feedback went to the code author, and none went to the reviewers&lt;/strong&gt;.  And the reviewers are the ones who are doing all of the heavy lifting!  As a reviewer, Vlad also wants feedback, and recognition for code review done well.&lt;/p&gt;
&lt;p&gt;There&amp;#8217;s a company in Toronto that specializes in feedback like this.  They&amp;#8217;re one of the major players in the Toronto start-up scene, and have built a pretty sweet suite of tools to facilitate quick and easy feedback/recognition.&lt;/p&gt;
&lt;p&gt;The company is called &lt;a href="http://www.rypple.com"&gt;Rypple&lt;/a&gt;.  And maybe that&amp;#8217;s the name of the application, too.  (checks website) Yeah, it&amp;#8217;s both.&lt;/p&gt;
&lt;p&gt;So Rypple has this feature called Kudos that let&amp;#8217;s people publicly acknowledge the good work of their team.&lt;/p&gt;
&lt;p&gt;Normally, I don&amp;#8217;t pimp companies.  And it upsets me when people comment on my blog, and their sub-text is to try to sell their product or service.  However, I think this video is relevant, so I&amp;#8217;m posting their demo video so you can see how Kudos work:&lt;/p&gt;
&lt;p&gt;&lt;a href="http://vimeo.com/11811215"&gt;Click here if you can&amp;#8217;t see the video.&lt;/a&gt;&lt;/p&gt;
&lt;p&gt;&lt;/p&gt;
&lt;h3&gt;The Idea&lt;/h3&gt;
&lt;p&gt;So Rypple&amp;#8217;s idea is to have a feed that the team subscribes to, and publicly display things like Kudos.  The badges for the Kudos are also limited in how many you can give per week, so they&amp;#8217;re a valuable commodity that can&amp;#8217;t just be handed out all over the place.  Cool idea.&lt;/p&gt;
&lt;p&gt;So there&amp;#8217;s one approach &amp;#8211; use a service like Rypple to give your reviewers better feedback and recognition.&lt;/p&gt;
&lt;p&gt;Or maybe we could build an extension for &lt;a href="http://www.reviewboard.org"&gt;Review Board&lt;/a&gt; that does something similar, and more oriented around code review.&lt;/p&gt;
&lt;p&gt;It&amp;#8217;s not oriented like a game, like I had originally envisioned.  But somehow, I think this idea has more meaning and traction than just &amp;#8220;adding points&amp;#8221;.&lt;/p&gt;
&lt;p&gt;More on this idea in a few days.  But please, comment if you have any thoughts or ideas to add.&lt;/p&gt;&lt;div class="feedflare"&gt;
&lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=354S44VLXwo:bylkv8pOFag:yIl2AUoC8zA"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?d=yIl2AUoC8zA" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=354S44VLXwo:bylkv8pOFag:V_sGLiPBpWU"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=354S44VLXwo:bylkv8pOFag:V_sGLiPBpWU" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=354S44VLXwo:bylkv8pOFag:F7zBnMyn0Lo"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=354S44VLXwo:bylkv8pOFag:F7zBnMyn0Lo" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=354S44VLXwo:bylkv8pOFag:gIN9vFwOqvQ"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=354S44VLXwo:bylkv8pOFag:gIN9vFwOqvQ" border="0"&gt;&lt;/img&gt;&lt;/a&gt;
&lt;/div&gt;</description><author>mike.d.conley@gmail.com (mikeconley)</author><pubDate>Sat, 23 Oct 2010 19:49:34 -0700</pubDate><guid>http://mikeconley.ca/blog/2010/10/23/recognizing-good-code-review/</guid></item><item><title>Review Board Issue Tracking:  A Sneak Peek
</title><link>http://mikeconley.ca/blog/2010/10/21/review-board-issue-tracking-a-sneak-peek/</link><description>&lt;p&gt;So I wrote my (hopefully) last mid-term ever last night, and in celebration, I thought I&amp;#8217;d put together a little video showing off the issue tracking feature I&amp;#8217;m hoping to put into &lt;a href="http://www.reviewboard.org"&gt;Review Board&lt;/a&gt;.&lt;/p&gt;
&lt;p&gt;It&amp;#8217;s still in it&amp;#8217;s very early stages.  The code hasn&amp;#8217;t been reviewed.  I&amp;#8217;m still really really open to suggestions and feedback on this.  So please, comment here, or on &lt;a href="http://groups.google.com/group/reviewboard-dev"&gt;the reviewboard-dev list&lt;/a&gt;.&lt;/p&gt;
&lt;p&gt;So here it is &amp;#8211; enjoy!&lt;/p&gt;
&lt;p&gt;&lt;a href="http://www.youtube.com/watch?v=ruVlGGdyvYM"&gt;(Click here if you can&amp;#8217;t see the video)&lt;/a&gt;&lt;br /&gt;
&lt;/p&gt;&lt;div class="feedflare"&gt;
&lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=HirnriQTXEs:hZEulZTTC0I:yIl2AUoC8zA"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?d=yIl2AUoC8zA" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=HirnriQTXEs:hZEulZTTC0I:V_sGLiPBpWU"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=HirnriQTXEs:hZEulZTTC0I:V_sGLiPBpWU" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=HirnriQTXEs:hZEulZTTC0I:F7zBnMyn0Lo"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=HirnriQTXEs:hZEulZTTC0I:F7zBnMyn0Lo" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=HirnriQTXEs:hZEulZTTC0I:gIN9vFwOqvQ"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=HirnriQTXEs:hZEulZTTC0I:gIN9vFwOqvQ" border="0"&gt;&lt;/img&gt;&lt;/a&gt;
&lt;/div&gt;</description><author>mike.d.conley@gmail.com (mikeconley)</author><pubDate>Thu, 21 Oct 2010 18:27:41 -0700</pubDate><guid>http://mikeconley.ca/blog/2010/10/21/review-board-issue-tracking-a-sneak-peek/</guid></item><item><title>Look down. Now back up again. The Defects are now Issues.
</title><link>http://mikeconley.ca/blog/2010/07/10/look-down-now-back-up-again-the-defects-are-now-issues/</link><description>&lt;p&gt;From my software development experience, there are a few different words for the generic notion of a &amp;#8220;problem&amp;#8221;.&lt;/p&gt;
&lt;p&gt;A &lt;strong&gt;bug&lt;/strong&gt; or a &lt;strong&gt;defect&lt;/strong&gt;, for example, is defined as the following from &lt;a href="http://en.wikipedia.org/wiki/Software_bug"&gt;Wikipedia&lt;/a&gt;:&lt;/p&gt;
&lt;blockquote&gt;&lt;p&gt;A &lt;strong&gt;software bug&lt;/strong&gt; is the common term used to describe an error flaw,  mistake, failure,  or fault in a computer program or system that produces an incorrect or unexpected result, or  causes it to behave in unintended ways. Most bugs arise from mistakes  and errors made by people in either a program&amp;#8217;s source  code or its design, and a few are caused by compilers producing incorrect code.&lt;/p&gt;&lt;/blockquote&gt;
&lt;p&gt;An &lt;strong&gt;issue&lt;/strong&gt; goes a level higher &amp;#8211; a bug is an issue, but an issue might not be a bug.  &lt;a href="http://en.wikipedia.org/wiki/Issue_%28computers%29#Issue"&gt;Wikipedia says&lt;/a&gt;:&lt;/p&gt;
&lt;blockquote&gt;&lt;p&gt;In computing, the term &lt;strong&gt;issue&lt;/strong&gt; is a unit of work to accomplish an  improvement in a system. An issue could be a bug, a requested feature,  task, missing documentation, and so forth. The word &amp;#8220;issue&amp;#8221;  is popularly misused in lieu of &amp;#8220;problem.&amp;#8221;  This usage is probably related.&lt;/p&gt;&lt;/blockquote&gt;
&lt;p&gt;Where am I going with all of this?&lt;/p&gt;
&lt;p&gt;Well, remember when I said &lt;a href="http://mikeconley.ca/blog/2010/07/10/filing-defects-in-review-board/"&gt;I was going to add defect reporting/tracking capabilities&lt;/a&gt; to &lt;a href="http://www.reviewboard.org"&gt;Review Board&lt;/a&gt;?  I asked for some feedback on &lt;a href="http://mikeconley.ca/blog/2010/07/10/filing-defects-in-review-board/"&gt;my UI mockups&lt;/a&gt; on the developer mailing list, and &lt;a href="http://groups.google.com/group/reviewboard-dev/browse_thread/thread/48705cfb39296991"&gt;an interesting conversation on terminology erupted&lt;/a&gt;.&lt;/p&gt;
&lt;p&gt;Anyhow, the long and the short of it is &amp;#8211; we&amp;#8217;re going to be calling &amp;#8220;problems still existing within a review request revision&amp;#8221; &lt;strong&gt;issues&lt;/strong&gt;.  And this is distinct from the sort of thing that might show up in an issue tracker.&lt;/p&gt;
&lt;p&gt;Maybe down the line, we&amp;#8217;ll have a way for administrators to set their own word for it.  From the thread, it sounds like everybody and their brother has their own favourite terminology.  &amp;#8221;Issue&amp;#8221; will have to do for now.&lt;/p&gt;
&lt;p&gt;Thanks again to everyone on the list who contributed to the conversation.&lt;/p&gt;
&lt;div class="wp_likes" id="wp_likes_post-1582"&gt;&lt;a class="like" href="javascript:wp_likes.like(1582);" title=""&gt;&lt;img alt="" border="0" src="http://mikeconley.ca/blog/wp-content/plugins/wp-likes/images/like.png" /&gt;Like&lt;/a&gt;&lt;span class="text"&gt;&lt;/span&gt;
&lt;div class="unlike"&gt;&lt;a href="javascript:wp_likes.unlike(1582);"&gt;Unlike&lt;/a&gt;&lt;/div&gt;
&lt;/div&gt;&lt;div class="feedflare"&gt;
&lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=N2myp6h7wX0:iTxebx8DJes:yIl2AUoC8zA"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?d=yIl2AUoC8zA" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=N2myp6h7wX0:iTxebx8DJes:V_sGLiPBpWU"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=N2myp6h7wX0:iTxebx8DJes:V_sGLiPBpWU" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=N2myp6h7wX0:iTxebx8DJes:F7zBnMyn0Lo"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=N2myp6h7wX0:iTxebx8DJes:F7zBnMyn0Lo" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=N2myp6h7wX0:iTxebx8DJes:gIN9vFwOqvQ"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=N2myp6h7wX0:iTxebx8DJes:gIN9vFwOqvQ" border="0"&gt;&lt;/img&gt;&lt;/a&gt;
&lt;/div&gt;</description><author>mike.d.conley@gmail.com (mikeconley)</author><pubDate>Sun, 11 Jul 2010 02:44:46 -0700</pubDate><guid>http://mikeconley.ca/blog/2010/07/10/look-down-now-back-up-again-the-defects-are-now-issues/</guid></item><item><title>Filing Defects in Review Board
</title><link>http://mikeconley.ca/blog/2010/07/10/filing-defects-in-review-board/</link><description>&lt;p&gt;In my last post, I talked about &lt;a href="http://mikeconley.ca/blog/2010/07/05/review-board-statistics-extensions-karma-stopwatch-and-fixit/"&gt;an extension for Review Board&lt;/a&gt; that would allow users to register “defects”, “TODOs” or “problems” with code that&amp;#8217;s up for review.&lt;/p&gt;
&lt;p&gt;After chatting with the lead RB devs for a bit, we&amp;#8217;ve decided to scrap the extension.&lt;/p&gt;
&lt;p&gt;[audible gasp, booing, hissing]&lt;/p&gt;
&lt;p&gt;Instead, we&amp;#8217;re just going to put it in the core of Review Board.&lt;/p&gt;
&lt;p&gt;[thundering applause]&lt;/p&gt;
&lt;h3&gt;Defects&lt;/h3&gt;
&lt;p&gt;Why is this useful?  I&amp;#8217;ve got a few reasons for you:&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;It&amp;#8217;ll be easier for reviewees to keep track of things left to fix, and similarly, it&amp;#8217;ll be harder for reviewees to accidentally skip over fixing a defect that a reviewer has found&lt;/li&gt;
&lt;li&gt;My statistics extension will be able to calculate useful things like &lt;strong&gt;defect detection rate&lt;/strong&gt;, and &lt;strong&gt;defect density&lt;/strong&gt;&lt;/li&gt;
&lt;li&gt;Maybe it&amp;#8217;s just me, but checking things off as &amp;#8220;fixed&amp;#8221; or &amp;#8220;completed&amp;#8221; is &lt;em&gt;really&lt;/em&gt; satisfying&lt;/li&gt;
&lt;li&gt;Who knows, down the line, I might code up an extension that lets you &lt;a href="http://mikeconley.ca/blog/2010/03/09/turning-peer-code-review-into-a-game/"&gt;turn finding/closing defects into a game&lt;/a&gt;&lt;/li&gt;
&lt;/ol&gt;
&lt;p&gt;However, since we&amp;#8217;re adding this to the core of &lt;a href="http://www.reviewboard.org"&gt;Review Board&lt;/a&gt;, we have to keep it simple.  One of Review Board&amp;#8217;s biggest strengths is in its total lack of clutter.  No bells.  No whistles.  Just the things you need to get the job done.  Let the extensions bring the bells and whistles.&lt;/p&gt;
&lt;p&gt;So that means creating a bare-bones defect-tracking mechanism and UI, and leaving it open for extension.  Because who knows, maybe &lt;a href="http://groups.google.com/group/reviewboard/browse_thread/thread/e224b4afce4a8cfc/d3501fd3018fae93"&gt;there&lt;/a&gt; &lt;a href="http://groups.google.com/group/reviewboard/browse_thread/thread/2b4dc38ad73b82b2/9d410350f1c769ae"&gt;are&lt;/a&gt; &lt;a href="http://groups.google.com/group/reviewboard/browse_thread/thread/eb6d98fad2c1b3c5/bbf2592fcc70b53c"&gt;some&lt;/a&gt; &lt;a href="http://groups.google.com/group/reviewboard/browse_thread/thread/8dfbea6d40a9f160/a4319afa5b1404b1"&gt;people&lt;/a&gt; out there who want to customize what &lt;em&gt;kind &lt;/em&gt;of defects they&amp;#8217;re filing.&lt;/p&gt;
&lt;p&gt;I&amp;#8217;ve come up with a design that I think is pretty simple and clean.  And it doesn&amp;#8217;t rock the boat &amp;#8211; if you&amp;#8217;re not interested in filing defects, your Review Board experience stays the same.&lt;/p&gt;
&lt;h4&gt;Filing a Defect&lt;/h4&gt;
&lt;p&gt;I propose adding a simple checkbox to the comment dialog to indicate that this comment files a defect, like so:&lt;/p&gt;
&lt;div class="wp-caption alignnone" id="attachment_1560" style="width: 310px;"&gt;&lt;a href="http://mikeconley.ca/blog/wp-content/uploads/2010/07/comment_checkbox_shot.png"&gt;&lt;img alt="Comment Defect Checkbox Screenshot" class="size-medium wp-image-1560 " height="119" src="http://mikeconley.ca/blog/wp-content/uploads/2010/07/comment_checkbox_shot-300x119.png" title="Comment Defect Checkbox Screenshot" width="300" /&gt;&lt;/a&gt;&lt;p class="wp-caption-text"&gt;No bells.  No whistles.  Just a simple little checkbox.&lt;/p&gt;&lt;/div&gt;
&lt;p&gt;While I&amp;#8217;m in there, I&amp;#8217;ll try to toss in some hooks so that extension developers can add more fields &amp;#8211; for example, the classification or the priority of the defect.  By default, however, it&amp;#8217;s just a bare-bones little checkbox.&lt;/p&gt;
&lt;p&gt;So far, so good.  You&amp;#8217;ve filed a defect.  Maybe this is how it&amp;#8217;ll look like in the in-line comment viewer:&lt;/p&gt;
&lt;div class="wp-caption alignnone" id="attachment_1569" style="width: 310px;"&gt;&lt;a href="http://mikeconley.ca/blog/wp-content/uploads/2010/07/comments_interface_shot_report_inline.png"&gt;&lt;img alt="The inline comment viewer is showing that a defect report has been filed." class="size-medium wp-image-1569 " height="128" src="http://mikeconley.ca/blog/wp-content/uploads/2010/07/comments_interface_shot_report_inline-300x128.png" title="comments_interface_shot_report_inline" width="300" /&gt;&lt;/a&gt;&lt;p class="wp-caption-text"&gt;A defect has been reported!&lt;/p&gt;&lt;/div&gt;
&lt;h4&gt;Two Choices&lt;/h4&gt;
&lt;p&gt;A reviewer can file defects reports, and the &lt;strong&gt;reviewee&lt;/strong&gt; is able to act on them.&lt;/p&gt;
&lt;p&gt;Lets say I&amp;#8217;m the reviewee.  I&amp;#8217;ve just gotten a review, and I&amp;#8217;ve got my editor / IDE with my patch waiting in the background.  I see a few defect reports have been filed.  For the ones I completely agree with, I fix them in my editor, and then go back to Review Board and mark them as &lt;strong&gt;Fixed&lt;/strong&gt;.&lt;/p&gt;
&lt;div class="wp-caption alignnone" id="attachment_1567" style="width: 310px;"&gt;&lt;a href="http://mikeconley.ca/blog/wp-content/uploads/2010/07/comments_interface_shot_fixed_inline.png"&gt;&lt;img alt="The defect report has been marked as being fixed." class="size-medium wp-image-1567" height="128" src="http://mikeconley.ca/blog/wp-content/uploads/2010/07/comments_interface_shot_fixed_inline-300x128.png" title="comments_interface_shot_fixed_inline" width="300" /&gt;&lt;/a&gt;&lt;p class="wp-caption-text"&gt;All fixed!&lt;/p&gt;&lt;/div&gt;
&lt;p&gt;It&amp;#8217;s also possible that I might not agree with one or more of the defect reports.  In this case, I&amp;#8217;ll reply to the comment to argue my case.  I might also mark the defect report as &lt;strong&gt;Pass&lt;/strong&gt;, which means, &amp;#8220;I&amp;#8217;ve seen it, but I think I&amp;#8217;ll pass on that&amp;#8221;.&lt;/p&gt;
&lt;div class="wp-caption alignnone" id="attachment_1568" style="width: 310px;"&gt;&lt;a href="http://mikeconley.ca/blog/wp-content/uploads/2010/07/comments_interface_shot_passed_inline.png"&gt;&lt;img alt="The defect report has been marked as &amp;quot;pass&amp;quot;." class="size-medium wp-image-1568" height="128" src="http://mikeconley.ca/blog/wp-content/uploads/2010/07/comments_interface_shot_passed_inline-300x128.png" title="comments_interface_shot_passed_inline" width="300" /&gt;&lt;/a&gt;&lt;p class="wp-caption-text"&gt;I think I'll pass on that, thanks.&lt;/p&gt;&lt;/div&gt;
&lt;p&gt;These comments and defect reports are also visible in the review request details page:&lt;/p&gt;
&lt;div class="wp-caption alignnone" id="attachment_1564" style="width: 310px;"&gt;&lt;a href="http://mikeconley.ca/blog/wp-content/uploads/2010/07/comments_interface_shot_report.png"&gt;&lt;img alt="A defect report has been filed, and we're in the review request detail page." class="size-medium wp-image-1564" height="86" src="http://mikeconley.ca/blog/wp-content/uploads/2010/07/comments_interface_shot_report-300x86.png" title="comments_interface_shot_report" width="300" /&gt;&lt;/a&gt;&lt;p class="wp-caption-text"&gt;A defect has been filed.&lt;/p&gt;&lt;/div&gt;
&lt;div class="wp-caption alignnone" id="attachment_1565" style="width: 310px;"&gt;&lt;a href="http://mikeconley.ca/blog/wp-content/uploads/2010/07/comments_interface_shot_fixed.png"&gt;&lt;img alt="The defect is marked as fixed, and we're in the review request detail page." class="size-medium wp-image-1565" height="86" src="http://mikeconley.ca/blog/wp-content/uploads/2010/07/comments_interface_shot_fixed-300x86.png" title="comments_interface_shot_fixed" width="300" /&gt;&lt;/a&gt;&lt;p class="wp-caption-text"&gt;All fixed up.&lt;/p&gt;&lt;/div&gt;
&lt;div class="wp-caption alignnone" id="attachment_1566" style="width: 310px;"&gt;&lt;a href="http://mikeconley.ca/blog/wp-content/uploads/2010/07/comments_interface_shot_passed.png"&gt;&lt;img alt="We're passing on the defect report, and we're in the review request detail page." class="size-medium wp-image-1566" height="86" src="http://mikeconley.ca/blog/wp-content/uploads/2010/07/comments_interface_shot_passed-300x86.png" title="comments_interface_shot_passed" width="300" /&gt;&lt;/a&gt;&lt;p class="wp-caption-text"&gt;It's all good - just pass this defect report.&lt;/p&gt;&lt;/div&gt;
&lt;h3&gt;Thoughts?&lt;/h3&gt;
&lt;p&gt;What do you think?  Am I on the right track?  Am I missing a case?  Does &amp;#8220;pass&amp;#8221; make sense?  Will this be useful?  I&amp;#8217;d love to hear your thoughts.&lt;/p&gt;
&lt;div class="wp_likes" id="wp_likes_post-1558"&gt;&lt;a class="like" href="javascript:wp_likes.like(1558);" title=""&gt;&lt;img alt="" border="0" src="http://mikeconley.ca/blog/wp-content/plugins/wp-likes/images/like.png" /&gt;Like&lt;/a&gt;&lt;span class="text"&gt;&lt;/span&gt;
&lt;div class="unlike"&gt;&lt;a href="javascript:wp_likes.unlike(1558);"&gt;Unlike&lt;/a&gt;&lt;/div&gt;
&lt;/div&gt;&lt;div class="feedflare"&gt;
&lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=h34c_6vmkeU:8wXPC3Akmuo:yIl2AUoC8zA"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?d=yIl2AUoC8zA" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=h34c_6vmkeU:8wXPC3Akmuo:V_sGLiPBpWU"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=h34c_6vmkeU:8wXPC3Akmuo:V_sGLiPBpWU" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=h34c_6vmkeU:8wXPC3Akmuo:F7zBnMyn0Lo"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=h34c_6vmkeU:8wXPC3Akmuo:F7zBnMyn0Lo" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=h34c_6vmkeU:8wXPC3Akmuo:gIN9vFwOqvQ"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=h34c_6vmkeU:8wXPC3Akmuo:gIN9vFwOqvQ" border="0"&gt;&lt;/img&gt;&lt;/a&gt;
&lt;/div&gt;</description><author>mike.d.conley@gmail.com (mikeconley)</author><pubDate>Sat, 10 Jul 2010 20:08:50 -0700</pubDate><guid>http://mikeconley.ca/blog/2010/07/10/filing-defects-in-review-board/</guid></item><item><title>Review Board Statistics Extensions: Karma, Stopwatch, and FixIt
</title><link>http://mikeconley.ca/blog/2010/07/05/review-board-statistics-extensions-karma-stopwatch-and-fixit/</link><description>&lt;p&gt;I just spent the long weekend in Ottawa and Qu﻿﻿é﻿﻿﻿bec City with my parents and my girlfriend Em.&lt;/p&gt;
&lt;p&gt;During the long drive back to Toronto from Qu﻿﻿é﻿﻿﻿bec City, I had plenty of time to think about &lt;a href="http://mikeconley.ca/blog/category/technology/computer-science/gsoc-computer-science/"&gt;my GSoC project&lt;/a&gt;, and where I want to go with it once &lt;a href="http://code.google.com/soc/"&gt;GSoC&lt;/a&gt; is done.&lt;/p&gt;
&lt;p&gt;Here&amp;#8217;s what I came up with.&lt;/p&gt;
&lt;h3&gt;Detach Reviewing Time from Statistics&lt;/h3&gt;
&lt;p&gt;I think it&amp;#8217;s a safe assumption that my reviewing-time extension isn&amp;#8217;t going to be the only one to generate useful statistical data.&lt;/p&gt;
&lt;p&gt;So why not give extension developers an easy mechanism to display statistical data for their extension?&lt;/p&gt;
&lt;p&gt;First, I&amp;#8217;m going to extract the reviewing-time recording portion of the extension. Then, RB-Stats (or whatever I end up calling it), will introduce it&amp;#8217;s &lt;em&gt;own &lt;/em&gt;set of hooks for other extensions to register with.  This way, if users want some stats, there will be one place to go to get them.  And if an extension developer wants to make some statistics available, a lot of the hard work will already be done for them.&lt;/p&gt;
&lt;p&gt;And if an extension has the capability of combining its data with another extensions data to create a new statistic, we&amp;#8217;ll let RB-Stats manage all of that business.&lt;/p&gt;
&lt;h3&gt;Stopwatch&lt;/h3&gt;
&lt;p&gt;The reviewing-time feature of RB-Stats will become an extension on its own, and register its data with RB-Stats.  Once RB-Stats and Stopwatch are done, we should be feature equivalent with &lt;a href="http://mikeconley.ca/blog/2010/06/24/review-board-statistics-extension-demo-time/"&gt;my demo&lt;/a&gt;.&lt;/p&gt;
&lt;h3&gt;Review Karma&lt;/h3&gt;
&lt;p&gt;I kind of breezed past this in &lt;a href="http://mikeconley.ca/blog/2010/06/24/review-board-statistics-extension-demo-time/"&gt;my demo&lt;/a&gt;, but I&amp;#8217;m interested in displaying &amp;#8220;review karma&amp;#8221;.  Review karma is the reviews/review-requests ratio.&lt;/p&gt;
&lt;p&gt;But I&amp;#8217;m not sure karma is the right word.  It suggests that a low ratio (many review requests, few reviews) is a bad thing.  I&amp;#8217;m not so sure that&amp;#8217;s true.&lt;/p&gt;
&lt;p&gt;Still, I wonder what the impact will be to display review karma?  Not just in the RB-Stats statistics view, but next to user names?  Will there be an impact on review activity when we display this &amp;#8220;reputation&amp;#8221; value?&lt;/p&gt;
&lt;h3&gt;FixIt&lt;/h3&gt;
&lt;p&gt;This is a big one.&lt;/p&gt;
&lt;p&gt;Most code review tools allow reviewers to register &amp;#8220;defects&amp;#8221;, &amp;#8220;todos&amp;#8221; or &amp;#8220;problems&amp;#8221; with the code up for review.  This makes it easier for reviewees to keep track of things to fix, and things that have already been taken care of.  It&amp;#8217;s also useful in that it helps generate interesting statistics like defect density and defect detection rate (assuming Stopwatch is installed and enabled).&lt;/p&gt;
&lt;p&gt;I&amp;#8217;m going to tackle this extension as soon as RB-Stats, Stopwatch and Karma are done.  At this point, I&amp;#8217;m quite confident that the current extension framework can more or less handle this.&lt;/p&gt;
&lt;p&gt;Got any more ideas for me?  Or maybe an extension wish-list?  Let  me know.&lt;/p&gt;
&lt;div class="wp_likes" id="wp_likes_post-1541"&gt;&lt;a class="like" href="javascript:wp_likes.like(1541);" title=""&gt;&lt;img alt="" border="0" src="http://mikeconley.ca/blog/wp-content/plugins/wp-likes/images/like.png" /&gt;Like&lt;/a&gt;&lt;span class="text"&gt;&lt;/span&gt;
&lt;div class="unlike"&gt;&lt;a href="javascript:wp_likes.unlike(1541);"&gt;Unlike&lt;/a&gt;&lt;/div&gt;
&lt;/div&gt;&lt;div class="feedflare"&gt;
&lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=P3zHcH9q0Kc:CpdApxnrf6w:yIl2AUoC8zA"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?d=yIl2AUoC8zA" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=P3zHcH9q0Kc:CpdApxnrf6w:V_sGLiPBpWU"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=P3zHcH9q0Kc:CpdApxnrf6w:V_sGLiPBpWU" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=P3zHcH9q0Kc:CpdApxnrf6w:F7zBnMyn0Lo"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=P3zHcH9q0Kc:CpdApxnrf6w:F7zBnMyn0Lo" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=P3zHcH9q0Kc:CpdApxnrf6w:gIN9vFwOqvQ"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=P3zHcH9q0Kc:CpdApxnrf6w:gIN9vFwOqvQ" border="0"&gt;&lt;/img&gt;&lt;/a&gt;
&lt;/div&gt;</description><author>mike.d.conley@gmail.com (mikeconley)</author><pubDate>Mon, 05 Jul 2010 19:22:13 -0700</pubDate><guid>http://mikeconley.ca/blog/2010/07/05/review-board-statistics-extensions-karma-stopwatch-and-fixit/</guid></item><item><title>Review Board Statistics Extension – Demo Time
</title><link>http://mikeconley.ca/blog/2010/06/24/review-board-statistics-extension-demo-time/</link><description>&lt;p&gt;If I&amp;#8217;ve learned anything from &lt;a href="http://www.third-bit.com/"&gt;my supervisor&lt;/a&gt;, it&amp;#8217;s to demo.  Demo often. Step out of the lab and introduce what you&amp;#8217;ve been working on to the world. Hit the pavement and &lt;em&gt;show&lt;/em&gt;, rather than tell.&lt;/p&gt;
&lt;p&gt;So here&amp;#8217;s a video of me demoing my statistics extension for &lt;a href="http://www.reviewboard.org"&gt;Review Board&lt;/a&gt;.  It&amp;#8217;s still in the early phases, but a lot of the groundwork has been taken care of.&lt;/p&gt;
&lt;p&gt;And sorry for the video quality.  Desktop capture on Ubuntu turned out to be surprisingly difficult for my laptop, and that&amp;#8217;s the best I could do.&lt;/p&gt;
&lt;p&gt;So, without further ado, here&amp;#8217;s my demo (&lt;a href="http://www.youtube.com/watch?v=ZY4IqP26PnY"&gt;click here if you can&amp;#8217;t see it&lt;/a&gt;):&lt;/p&gt;
&lt;p&gt;&lt;/p&gt;
&lt;p&gt;Not bad!  And I haven&amp;#8217;t even reached the midterm of &lt;a href="http://code.google.com/soc/"&gt;GSoC&lt;/a&gt; yet.  Still plenty of time to enhance, document, test, and polish.&lt;/p&gt;
&lt;p&gt;If you have any questions or comments, I&amp;#8217;d love to hear them.&lt;/p&gt;
&lt;div class="wp_likes" id="wp_likes_post-1532"&gt;&lt;a class="like" href="javascript:wp_likes.like(1532);" title=""&gt;&lt;img alt="" border="0" src="http://mikeconley.ca/blog/wp-content/plugins/wp-likes/images/like.png" /&gt;Like&lt;/a&gt;&lt;span class="text"&gt;&lt;/span&gt;
&lt;div class="unlike"&gt;&lt;a href="javascript:wp_likes.unlike(1532);"&gt;Unlike&lt;/a&gt;&lt;/div&gt;
&lt;/div&gt;&lt;div class="feedflare"&gt;
&lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=n-ayE1Y7-_c:XCykhkNzjCU:yIl2AUoC8zA"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?d=yIl2AUoC8zA" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=n-ayE1Y7-_c:XCykhkNzjCU:V_sGLiPBpWU"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=n-ayE1Y7-_c:XCykhkNzjCU:V_sGLiPBpWU" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=n-ayE1Y7-_c:XCykhkNzjCU:F7zBnMyn0Lo"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=n-ayE1Y7-_c:XCykhkNzjCU:F7zBnMyn0Lo" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=n-ayE1Y7-_c:XCykhkNzjCU:gIN9vFwOqvQ"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=n-ayE1Y7-_c:XCykhkNzjCU:gIN9vFwOqvQ" border="0"&gt;&lt;/img&gt;&lt;/a&gt;
&lt;/div&gt;</description><author>mike.d.conley@gmail.com (mikeconley)</author><pubDate>Thu, 24 Jun 2010 20:22:54 -0700</pubDate><guid>http://mikeconley.ca/blog/2010/06/24/review-board-statistics-extension-demo-time/</guid></item><item><title>GSoC Update:  My Review Board Statistics Extension
</title><link>http://mikeconley.ca/blog/2010/06/21/gsoc-update-my-review-board-statistics-extension/</link><description>&lt;h3&gt;The Primary Goal&lt;/h3&gt;
&lt;p&gt;From the very beginning, my GSoC project has been mainly focused towards one primary goal:  I want to build an extension for &lt;a href="http://www.reviewboard.org"&gt;Review Board&lt;/a&gt; that will allow me to collect information about how long reviewers actually spend reviewing code.&lt;/p&gt;
&lt;p&gt;That&amp;#8217;s easier said than done.  When I started, the Review Board extension framework wasn&amp;#8217;t really in a state to allow such an extension to exist.&lt;/p&gt;
&lt;p&gt;So I&amp;#8217;ve been tooling around in the Review Board code for the past 2 months, preparing the framework, and getting it ready to handle my extension.&lt;/p&gt;
&lt;p&gt;And last night, it started to work.  I can now give rough estimates on how long a reviewer has spent reviewing code.&lt;/p&gt;
&lt;h3&gt;How It Works&lt;/h3&gt;
&lt;p&gt;My extension adds a new table to the database which stores &amp;#8220;reviewing sessions&amp;#8221;.  Each reviewing session is associated to a particular review request and user, and also has a field to store the number of seconds that a user has spent in review.&lt;/p&gt;
&lt;p&gt;I&amp;#8217;ve created a TemplateHook that allows me to inject Javascript into key areas of Review Board (in particular, the diff viewer, and the screenshot viewer).  The Javascript does the following:  every 10 seconds, we check to see if the mouse has moved on the body of the HTML document.  If it has, we send an &amp;#8220;activity&amp;#8221; notification to the server.&lt;/p&gt;
&lt;p&gt;The server receives this activity notification through &lt;a href="http://www.reviewboard.org/docs/manual/dev/webapi/#webapiguide"&gt;the Web API&lt;/a&gt;, and checks to see if the time lapsed since the last session update was greater than 10 seconds.  If it is, we increment the working session by 10 seconds and return a 200 HTTP code.  If it isn&amp;#8217;t, we don&amp;#8217;t change anything and return a 304 HTTP code.&lt;/p&gt;
&lt;p&gt;Next, my extension waits for a user to publish a review.  When it notices that a review is being published, it finds the working session for that user and review request, and then attaches it to the published review.  If the user then starts looking at the diff or screenshots again, a new working session is created.&lt;/p&gt;
&lt;p&gt;The result?  A pretty decent estimate of how long a user has spent reviewing the code.  No time gets recorded if the user gets up and has a sandwich.  No time gets recorded if the user is on another tab reading &lt;a href="http://www.reddit.com"&gt;Reddit&lt;/a&gt;.&lt;/p&gt;
&lt;p&gt;&lt;a href="http://mikeconley.ca/blog/wp-content/uploads/2010/06/Reviewing-Time.png"&gt;&lt;img alt="An image showing how reviewing time is displayed to the user" class="alignnone size-medium wp-image-1518" height="124" src="http://mikeconley.ca/blog/wp-content/uploads/2010/06/Reviewing-Time-300x124.png" title="Reviewing Time" width="300" /&gt;&lt;/a&gt;&lt;/p&gt;
&lt;p&gt;Not bad.  For a first draft, anyhow.&lt;/p&gt;
&lt;p&gt;I think I&amp;#8217;m going to try to chart the data somehow, so that users can track their inspection rates.  I&amp;#8217;ll let you know how that goes.&lt;/p&gt;
&lt;div class="wp_likes" id="wp_likes_post-1516"&gt;&lt;a class="like" href="javascript:wp_likes.like(1516);" title=""&gt;&lt;img alt="" border="0" src="http://mikeconley.ca/blog/wp-content/plugins/wp-likes/images/like.png" /&gt;Like&lt;/a&gt;&lt;span class="text"&gt;&lt;/span&gt;
&lt;div class="unlike"&gt;&lt;a href="javascript:wp_likes.unlike(1516);"&gt;Unlike&lt;/a&gt;&lt;/div&gt;
&lt;/div&gt;&lt;div class="feedflare"&gt;
&lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=7ykVzkLeS9E:4TDyfUnqjw8:yIl2AUoC8zA"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?d=yIl2AUoC8zA" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=7ykVzkLeS9E:4TDyfUnqjw8:V_sGLiPBpWU"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=7ykVzkLeS9E:4TDyfUnqjw8:V_sGLiPBpWU" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=7ykVzkLeS9E:4TDyfUnqjw8:F7zBnMyn0Lo"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=7ykVzkLeS9E:4TDyfUnqjw8:F7zBnMyn0Lo" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=7ykVzkLeS9E:4TDyfUnqjw8:gIN9vFwOqvQ"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=7ykVzkLeS9E:4TDyfUnqjw8:gIN9vFwOqvQ" border="0"&gt;&lt;/img&gt;&lt;/a&gt;
&lt;/div&gt;</description><author>mike.d.conley@gmail.com (mikeconley)</author><pubDate>Mon, 21 Jun 2010 18:19:22 -0700</pubDate><guid>http://mikeconley.ca/blog/2010/06/21/gsoc-update-my-review-board-statistics-extension/</guid></item><item><title>Review Board Tests:  SCMTool Segmentation Fault
</title><link>http://mikeconley.ca/blog/2010/06/02/review-board-tests-scmtool-segmentation-fault/</link><description>&lt;p&gt;I can&amp;#8217;t honestly say I&amp;#8217;ve been doing much test-driven development on Review Board.  Django is a really cool web-framework, but I miss all of the nice testing tools from the Rails ecosystem.&lt;/p&gt;
&lt;p&gt;Anyhow, today, I decided to run the tests, and BAM &amp;#8211; Segmentation Fault:&lt;/p&gt;
&lt;blockquote&gt;&lt;p&gt;Testing Perforce binary diff parsing &amp;#8230; SKIP: perforce/p4python is not installed&lt;br /&gt;
Testing PerforceTool.get_changeset &amp;#8230; SKIP: perforce/p4python is not installed&lt;br /&gt;
Testing Perforce empty and normal diff parsing &amp;#8230; SKIP: perforce/p4python is not installed&lt;br /&gt;
Testing Perforce empty diff parsing &amp;#8230; SKIP: perforce/p4python is not installed&lt;br /&gt;
Testing PerforceTool.get_file &amp;#8230; SKIP: perforce/p4python is not installed&lt;br /&gt;
Testing parsing SVN diff with binary file &amp;#8230; ok&lt;br /&gt;
Testing SVNTool.get_file &amp;#8230; &lt;strong&gt;Segmentation Fault&lt;/strong&gt;&lt;/p&gt;&lt;/blockquote&gt;
&lt;p&gt;Yikes.  Getting a segfault in Python is unusual, and a little jarring.  However, it did let me narrow down the problem a bit:  it must have to do with the pysvn bindings that Review Board uses to talk to Subversion, because those are written in C++ (which can certainly segfault).&lt;/p&gt;
&lt;p&gt;The machine I was using had Ubuntu Hardy on it, and the Hardy packages have pysvn 1.5.2-1.  Taking a look at &lt;a href="http://pysvn.tigris.org/"&gt;the pysvn homepage&lt;/a&gt;, the most recent version is actually 1.7.2.&lt;/p&gt;
&lt;p&gt;So, I uninstalled pysvn, &lt;a href="http://pysvn.tigris.org/project_downloads.html"&gt;downloaded the source&lt;/a&gt; for 1.7.2, compiled it, and installed it manually.&lt;/p&gt;
&lt;p&gt;Segfault gone.  Tests pass.  Awesome sauce.&lt;/p&gt;
&lt;div class="wp_likes" id="wp_likes_post-1393"&gt;&lt;a class="like" href="javascript:wp_likes.like(1393);" title=""&gt;&lt;img alt="" border="0" src="http://mikeconley.ca/blog/wp-content/plugins/wp-likes/images/like.png" /&gt;Like&lt;/a&gt;&lt;span class="text"&gt;&lt;/span&gt;
&lt;div class="unlike"&gt;&lt;a href="javascript:wp_likes.unlike(1393);"&gt;Unlike&lt;/a&gt;&lt;/div&gt;
&lt;/div&gt;&lt;div class="feedflare"&gt;
&lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=-iWlzFaN1bM:k1KXAdtW2Wk:yIl2AUoC8zA"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?d=yIl2AUoC8zA" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=-iWlzFaN1bM:k1KXAdtW2Wk:V_sGLiPBpWU"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=-iWlzFaN1bM:k1KXAdtW2Wk:V_sGLiPBpWU" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=-iWlzFaN1bM:k1KXAdtW2Wk:F7zBnMyn0Lo"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=-iWlzFaN1bM:k1KXAdtW2Wk:F7zBnMyn0Lo" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=-iWlzFaN1bM:k1KXAdtW2Wk:gIN9vFwOqvQ"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=-iWlzFaN1bM:k1KXAdtW2Wk:gIN9vFwOqvQ" border="0"&gt;&lt;/img&gt;&lt;/a&gt;
&lt;/div&gt;</description><author>mike.d.conley@gmail.com (mikeconley)</author><pubDate>Wed, 02 Jun 2010 20:24:28 -0700</pubDate><guid>http://mikeconley.ca/blog/2010/06/02/review-board-tests-scmtool-segmentation-fault/</guid></item><item><title>Python Eggs:  Sunny Side Up, and Other Goodies (or How I Learned to Stop Worrying and Start Coding)
</title><link>http://mikeconley.ca/blog/2010/05/12/python-eggs-sunny-side-up-and-other-goodies/</link><description>&lt;h3&gt;Cooking with Eggs&lt;/h3&gt;
&lt;p&gt;Every now and then, the computer gods smile and give me a freebie.&lt;/p&gt;
&lt;p&gt;I&amp;#8217;ve been worrying my mind out over a few problems / obstacles for &lt;a href="http://mikeconley.ca/blog/2010/04/27/my-gsoc-project-review-board-extensions/"&gt;my Review Board extensions GSoC project&lt;/a&gt;.  In particular, I&amp;#8217;ve been worrying about dealing with extension dependencies, conflicts, and installation.&lt;/p&gt;
&lt;p&gt;I racked my brain.  I came up with scenarios.  I drew lots of big scary diagrams on a wipe board.&lt;/p&gt;
&lt;p&gt;And then light dawned.&lt;/p&gt;
&lt;h3&gt;Batteries Come Included&lt;/h3&gt;
&lt;p&gt;Enter &lt;a href="http://peak.telecommunity.com/DevCenter/setuptools"&gt;Setuptools&lt;/a&gt; and &lt;a href="http://peak.telecommunity.com/DevCenter/PythonEggs"&gt;Python Eggs&lt;/a&gt;.&lt;/p&gt;
&lt;p&gt;All of those things I was worried about having to build and account for?  When using Python Eggs, It&amp;#8217;s all built in&lt;em&gt;.&lt;/em&gt; Dependencies?  Taken care of. Conflicts?  Don&amp;#8217;t worry about it.  Installation?  That&amp;#8217;s what Setuptools and Python Eggs were built for!&lt;/p&gt;
&lt;p&gt;&lt;a href="http://peak.telecommunity.com/DevCenter/setuptools#extensible-applications-and-frameworks"&gt;In fact, it even looks like Setuptools was designed with extensible applications in mind.&lt;/a&gt;&lt;/p&gt;
&lt;h3&gt;Wait, really?  How?&lt;/h3&gt;
&lt;p&gt;Here&amp;#8217;s the setup.py file for &lt;a href="http://github.com/chipx86/rb-extension-pack/tree/master/rbreports/"&gt;the rb-reports extension in the rb-extensions-pack on Github:&lt;/a&gt;&lt;/p&gt;
&lt;pre&gt;from setuptools import setup, find_packages

PACKAGE="RB-Reports"
VERSION="0.1"

setup(
    name=PACKAGE,
    version=VERSION,
    description="""Reports extension for Review Board""",
    author="Christian Hammond",
    packages=["rbreports"],
    entry_points={
        'reviewboard.extensions':
        '%s = rbreports.extension:ReportsExtension' % PACKAGE,
    },
    package_data={
        'rbreports': [
            'htdocs/css/*.css',
            'htdocs/js/*.js',
            'templates/rbreports/*.html',
            'templates/rbreports/*.txt',
        ],
    }
)
&lt;/pre&gt;
&lt;p&gt;Pay particular attention to the &amp;#8220;entry_points&amp;#8221; parameter.  What this is doing, is registering rbreports.extension:ReportsExtension to the entry point &amp;#8220;reviewboard.extensions&amp;#8221;.&lt;/p&gt;
&lt;p&gt;&amp;#8220;Hold up!&amp;#8221;, I hear you asking. &amp;#8220;What&amp;#8217;s an entry point?&amp;#8221;&lt;/p&gt;
&lt;h3&gt;Entry Points&lt;/h3&gt;
&lt;p&gt;An entry point is a unique identifier associated with an application that can accept extensions.&lt;/p&gt;
&lt;p&gt;The unique identifier for Review Board extensions is &amp;#8220;reviewboard.extensions&amp;#8221;.&lt;/p&gt;
&lt;p&gt;This is the first handshake, more or less, between Review Board and any extensions:  in order for Review Board to &amp;#8220;see&amp;#8221; the extension, the extension must register an entry point at &amp;#8220;reviewboard.extensions&amp;#8221;.&lt;/p&gt;
&lt;p&gt;&lt;a href="http://reinout.vanrees.org/weblog/2010/01/06/zest-releaser-entry-points.html"&gt;This blog post shows how extensions can be found and loaded up.&lt;/a&gt;&lt;/p&gt;
&lt;h3&gt;Other Goodies&lt;/h3&gt;
&lt;h4&gt;INSTALLED_APPS and Django&lt;/h4&gt;
&lt;p&gt;I remember also being worried about how to create tables in Django for extension models.  I thought &amp;#8220;holy smokes, I&amp;#8217;m going to have to either shoehorn some raw SQL into the extension manager, or maybe even trust the extension developers to write the CREATE TABLE queries themselves!&amp;#8221;.&lt;/p&gt;
&lt;p&gt;Luckily, there&amp;#8217;s a better alternative.&lt;/p&gt;
&lt;p&gt;Django knows about its applications through a dictionary called &lt;a href="http://docs.djangoproject.com/en/dev/ref/settings/#installed-apps"&gt;INSTALLED_APPS.&lt;/a&gt; When you add a new model to a Django project, you simply add the model app to the INSTALLED_APPS dictionary, and run &amp;#8220;manage.py syncdb&amp;#8221;.  Django does the magic, bingo-bango, and boom &amp;#8211; tables created.&lt;/p&gt;
&lt;p&gt;So if a new extension has some tables it needs created, I simply insert the app name of the extension into INSTALLED_APPS when the extension is installed, and &lt;a href="http://groups.google.com/group/django-users/browse_thread/thread/34b501d2d1f88496?pli=1"&gt;call syncdb programmatically&lt;/a&gt;.  Tables created:  no sweat.&lt;/p&gt;
&lt;h4&gt;django-evolution&lt;/h4&gt;
&lt;p&gt;Creating tables is easy.  But what if an extension gets updated, and the table needs to be modified?  Sounds like we&amp;#8217;ve got a mess on our hands.&lt;/p&gt;
&lt;p&gt;And don&amp;#8217;t expect Django to save you.  When you modify a model in Django, they expect you to into that DB and alter that table by hand:&lt;/p&gt;
&lt;blockquote&gt;&lt;p&gt;[syncdb] creates the tables if they don’t yet exist. Note that syncdb does &lt;em&gt;not&lt;/em&gt; sync changes in models or deletions of models; if you make a change to a model or delete a model, and you want to update the database, syncdb will not handle that.&lt;br /&gt;
From &lt;a href="http://www.djangobook.com/"&gt;The Django Book&lt;/a&gt; &amp;#8211; &lt;a href="http://www.djangobook.com/en/2.0/chapter05/"&gt;Chapter 5: Models&lt;/a&gt;&lt;/p&gt;&lt;/blockquote&gt;
&lt;p&gt;Thankfully, there&amp;#8217;s a mechanism that&amp;#8217;s already built into Review Board that makes this trouble go away:  &lt;a href="http://code.google.com/p/django-evolution/"&gt;django-evolution&lt;/a&gt;.  Django-evolution, when used properly, will automatically detect changes in application models, and alter the database tables accordingly.  This is how Review Board does upgrades.&lt;/p&gt;
&lt;p&gt;And to top that off, RB co-founder &lt;a href="http://www.chipx86.com/blog/"&gt;Christian Hammond&lt;/a&gt; just became the django-evolution maintainer.&lt;/p&gt;
&lt;p&gt;Wow.  Everything is falling neatly into place.&lt;/p&gt;
&lt;div class="wp_likes" id="wp_likes_post-1350"&gt;&lt;a class="like" href="javascript:wp_likes.like(1350);" title=""&gt;&lt;img alt="" border="0" src="http://mikeconley.ca/blog/wp-content/plugins/wp-likes/images/like.png" /&gt;Like&lt;/a&gt;&lt;span class="text"&gt;&lt;/span&gt;
&lt;div class="unlike"&gt;&lt;a href="javascript:wp_likes.unlike(1350);"&gt;Unlike&lt;/a&gt;&lt;/div&gt;
&lt;/div&gt;&lt;div class="feedflare"&gt;
&lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=ED3RnkNnQBg:D47PBdZkX10:yIl2AUoC8zA"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?d=yIl2AUoC8zA" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=ED3RnkNnQBg:D47PBdZkX10:V_sGLiPBpWU"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=ED3RnkNnQBg:D47PBdZkX10:V_sGLiPBpWU" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=ED3RnkNnQBg:D47PBdZkX10:F7zBnMyn0Lo"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=ED3RnkNnQBg:D47PBdZkX10:F7zBnMyn0Lo" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=ED3RnkNnQBg:D47PBdZkX10:gIN9vFwOqvQ"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=ED3RnkNnQBg:D47PBdZkX10:gIN9vFwOqvQ" border="0"&gt;&lt;/img&gt;&lt;/a&gt;
&lt;/div&gt;</description><author>mike.d.conley@gmail.com (mikeconley)</author><pubDate>Wed, 12 May 2010 20:20:28 -0700</pubDate><guid>http://mikeconley.ca/blog/2010/05/12/python-eggs-sunny-side-up-and-other-goodies/</guid></item><item><title>Python Metaclasses in Review Board
</title><link>http://mikeconley.ca/blog/2010/05/04/python-metaclasses-in-review-board/</link><description>&lt;p&gt;So, after &lt;a href="http://mikeconley.ca/blog/2010/04/30/code-spelunking-review-board-extensions-2/"&gt;diving into the Review Board extension code&lt;/a&gt;, I hit a little snag.&lt;/p&gt;
&lt;p&gt;It turns out I never learned about Python metaclasses.  And the extension code in Djblets / Review Board uses them.  In &lt;a href="http://www.mikeconley.ca/images/rb_ext/Map.html"&gt;the map I made of the Review Board extension code&lt;/a&gt;, I represented my confusion with an image of some kind of quantum-divide-by-zero implosion.&lt;/p&gt;
&lt;p&gt;I&amp;#8217;ll get to the &lt;em&gt;why&lt;/em&gt; for metaclasses in a second.  First, I&amp;#8217;ll demonstrate &lt;em&gt;how&lt;/em&gt; it&amp;#8217;s currently being used in the code.&lt;/p&gt;
&lt;h3&gt;The Way Things Are&lt;/h3&gt;
&lt;p&gt;This snippit is from the Djblets library, in the extensions folder, in base.py:&lt;/p&gt;
&lt;pre&gt;...

&lt;a href="#extension_hook" name="extension_hook"&gt;class ExtensionHook(object):&lt;/a&gt;
    def __init__(self, extension):
        self.extension = extension
        self.extension.hooks.add(self)
        self.__class__.add_hook(self)

    def shutdown(self):
        self.__class__.remove_hook(self)

&lt;a href="#extension_hook_point" name="extension_hook_point"&gt;class ExtensionHookPoint(type):&lt;/a&gt;
    def __init__(cls, name, bases, attrs):
        super(ExtensionHookPoint, cls).__init__(name, bases, attrs)

        if not hasattr(cls, "hooks"):
            cls.hooks = []

    def add_hook(cls, hook):
        cls.hooks.append(hook)

    def remove_hook(cls, hook):
        cls.hooks.remove(hook)

...&lt;/pre&gt;
&lt;p&gt;And here are those classes being used in the Review Board extension directory, where it defines its hooks:&lt;/p&gt;
&lt;pre&gt;...

&lt;a href="#dashboard_hook" name="dashboard_hook"&gt;class DashboardHook(ExtensionHook):&lt;/a&gt;
    __metaclass__ = ExtensionHookPoint

    def get_entries(self):
        raise NotImplemented

&lt;a href="#navigation_bar_hook" name="navigation_bar_hook"&gt;class NavigationBarHook(ExtensionHook):&lt;/a&gt;
    """
    A hook for adding entries to the main navigation bar.
    """
    __metaclass__ = ExtensionHookPoint

    def get_entry(self):
        """
        Returns the entry to add to the navigation bar.

        This should be a dict with the following keys:

            * `label`: The label to display
            * `url`:   The URL to point to.
        """
        raise NotImplemented
...&lt;/pre&gt;
&lt;p&gt;The idea here is that, if someone wanted to write an extension, they could add a hook to the navigation bar by subclassing the hooks, like so:&lt;/p&gt;
&lt;p&gt;(from &lt;a href="http://github.com/chipx86/rb-extension-pack/blob/master/rbreports/rbreports/extension.py"&gt;the rbreports prototype extension&lt;/a&gt;)&lt;/p&gt;
&lt;pre&gt;...

class ReportsDashboardHook(DashboardHook):
    def get_entries(self):
        return [{
            'label': 'Reports',
            'url': settings.SITE_ROOT + 'reports/',
        }]

class ReportsExtension(Extension):
    is_configurable = True

    def __init__(self):
        Extension.__init__(self)

        self.url_hook = URLHook(self, patterns('',
            (r'^reports/', include('rbreports.urls'))))

        self.dashboard_hook = ReportsDashboardHook(self)
...&lt;/pre&gt;
&lt;p&gt;So you can see the __metaclass__ thing up there in the &lt;a href="#dashboard_hook"&gt;DashboardHook&lt;/a&gt;.  &lt;a href="#dashboard_hook"&gt;DashboardHook&lt;/a&gt; subclasses &lt;a href="#extension_hook"&gt;ExtensionHook&lt;/a&gt;, and has &lt;a href="#extension_hook_point"&gt;ExtensionHookPoint&lt;/a&gt; as a metaclass.&lt;/p&gt;
&lt;p&gt;&lt;em&gt;Why?&lt;/em&gt; And what is a metaclass anyways?  Why is this useful at all?&lt;/p&gt;
&lt;p&gt;Well, luckily, I think I&amp;#8217;ve figured it out.&lt;/p&gt;
&lt;h3&gt;Behold &amp;#8211; Metaclasses&lt;/h3&gt;
&lt;blockquote&gt;&lt;p&gt;Metaclasses are deeper magic than 99% of users should ever worry about.         If you wonder whether you need them, you don&amp;#8217;t (the people who actually need them         know with certainty that they need them, and don&amp;#8217;t need an explanation about why).&lt;br /&gt;
&amp;#8211; Python Guru Tim Peters&lt;/p&gt;&lt;/blockquote&gt;
&lt;p&gt;There are &lt;a href="http://onlamp.com/pub/a/python/2003/04/17/metaclasses.html"&gt;plenty&lt;/a&gt; &lt;a href="http://www.voidspace.org.uk/python/articles/metaclasses.shtml"&gt;of resources&lt;/a&gt; &lt;a href="http://stackoverflow.com/questions/100003/what-is-a-metaclass-in-python"&gt;available&lt;/a&gt; &lt;a href="http://www.vrplumber.com/programming/metaclasses.pdf"&gt;regarding&lt;/a&gt; &lt;a href="http://stackoverflow.com/questions/392160/what-are-your-concrete-use-cases-for-metaclasses-in-python"&gt;Python&amp;#8217;s&lt;/a&gt; metaclasses.  And I ended up reading a ton of them trying to figure out just what the hell metaclasses actually do.&lt;/p&gt;
&lt;p&gt;As it turns out, &lt;a href="http://en.wikipedia.org/wiki/Metaclasses"&gt;the best explanation I got was from Wikipedia&lt;/a&gt;:&lt;/p&gt;
&lt;blockquote&gt;&lt;p&gt;In object-oriented programming, a metaclass is a class whose instances are classes. Just as an ordinary class defines the behavior of certain objects, a metaclass defines the behavior of certain classes and their instances.&lt;/p&gt;&lt;/blockquote&gt;
&lt;p&gt;&lt;a href="http://en.wikipedia.org/wiki/Metaclasses#Python_example"&gt;In particular, check out their example on Cars&lt;/a&gt;.  That, for me,  more or less set the record straight on how metaclasses are used.&lt;/p&gt;
&lt;p&gt;But why does Review Board use them when defining those hooks, like &lt;a href="#dashboard_hook"&gt;DashboardHook&lt;/a&gt;?&lt;/p&gt;
&lt;h3&gt;Why?&lt;/h3&gt;
&lt;p&gt;Forget the metaclasses for just a second.&lt;/p&gt;
&lt;p&gt;Something is missing from that code that I posted up.&lt;/p&gt;
&lt;p&gt;I&amp;#8217;ll pose it as a question:  How exactly is Review Board supposed to &lt;em&gt;know&lt;/em&gt; about ReportsDashboardHook?  Like&amp;#8230;if we&amp;#8217;re rendering the Dashboard, and want to fire off calls to all of our DashboardHooks&amp;#8230;how do we do it?  How can we find all of the active extension subclasses for DashboardHook?&lt;/p&gt;
&lt;p&gt;One way you could do it, is to dive into the extensions directories, and use regular expressions to find defined classes.  That sounds like a lot of work, brittle, and prone to error.  What else?&lt;/p&gt;
&lt;p&gt;Next, you could go back to those extension files, &amp;#8220;eval&amp;#8221; them (&lt;em&gt;shudder&lt;/em&gt;), and extract the globals(), looking for &lt;a href="#extension_hook"&gt;ExtensionHook&lt;/a&gt; subclasses.  That also sounds like lots of work, brittle, and prone to error.&lt;/p&gt;
&lt;p&gt;You could go for &lt;a href="http://docs.python.org/library/stdtypes.html#class.__subclasses__"&gt;class.__subclasses__&lt;/a&gt;&amp;#8230;but this gives you every subclass, and we just want the hooks that are for active extensions.  We could filter out all of the ones that aren&amp;#8217;t activated, but we&amp;#8217;d have to do that for every hook, and that blows.&lt;/p&gt;
&lt;p&gt;Next, we could have developers add their hooks to a registry after defining them.  So, we could have:&lt;/p&gt;
&lt;pre&gt;class ReportsDashboardHook(ExtensionHook):
  # ...
  # code goes here
  # ...

dashboard_hook_list.append(ReportsDashboardHook)&lt;/pre&gt;
&lt;p&gt;And then when an extension is shut down, we just remove it from the global hook list.  That doesn&amp;#8217;t look so bad, does it?  The only problem, is now we have to trust that extension developers will know to add those hooks to the global_hook_list?  It&amp;#8217;s another step, another thing to forget, and another point of failure.&lt;/p&gt;
&lt;p&gt;And it seems silly &amp;#8211; I mean, why go through all this effort?  Review Board has already seen these hooks&amp;#8230;why should it be so hard to just access the list activated hooks easily?&lt;/p&gt;
&lt;p&gt;&lt;strong&gt;And there it is.&lt;/strong&gt; &lt;em&gt;That&amp;#8217;s&lt;/em&gt; why I think we&amp;#8217;re using metaclasses.&lt;/p&gt;
&lt;p&gt;If you go back to &lt;a href="#extension_hook"&gt;ExtensionHook&lt;/a&gt; and &lt;a href="#extension_hook_point"&gt;ExtensionHookPoint&lt;/a&gt;, you&amp;#8217;ll notice that ExtensionHookPoint has a list as an instance variable called hooks.  That&amp;#8217;s the global hook list for that class of hook (&lt;a href="#dashboard_hook"&gt;DashboardHook&lt;/a&gt;).  And then &lt;a href="#extension_hook"&gt;ExtensionHook&lt;/a&gt;, in its constructor, automatically adds itself to the Extension&amp;#8217;s internal list of hooks, as well as the global hook list, with the add_hook method.  Shutting down the hook removes that hook from the global hook list.&lt;/p&gt;
&lt;p&gt;So now, all we need to do to add our implemented hook to the list of hooks, is to simply subclass &lt;a href="#dashboard_hook"&gt;DashboardHook&lt;/a&gt;, or &lt;a href="#navigation_bar_hook"&gt;NavigationBarHook&lt;/a&gt;, or &lt;a href="http://www.mikeconley.ca/images/rb_ext/Map.html"&gt;any of those Review Board hooks&lt;/a&gt;.  No need to add the hook to the global list manually &amp;#8211; Djblets / Review Board will take care of it.  So now, we can get all of the DashboardHook subclasses for activated hooks, easily and consistently, with no extra work for the extension developer.  Same with the NavigationBarHook subclasses.&lt;/p&gt;
&lt;p&gt;Smooth as glass.  Nice.&lt;/p&gt;
&lt;p&gt;Anyhow, that&amp;#8217;s my understanding of it.  If I&amp;#8217;m totally off, I&amp;#8217;ll let you know when I find out.&lt;/p&gt;
&lt;div class="wp_likes" id="wp_likes_post-1270"&gt;&lt;a class="like" href="javascript:wp_likes.like(1270);" title=""&gt;&lt;img alt="" border="0" src="http://mikeconley.ca/blog/wp-content/plugins/wp-likes/images/like.png" /&gt;Like&lt;/a&gt;&lt;span class="text"&gt;&lt;/span&gt;
&lt;div class="unlike"&gt;&lt;a href="javascript:wp_likes.unlike(1270);"&gt;Unlike&lt;/a&gt;&lt;/div&gt;
&lt;/div&gt;&lt;div class="feedflare"&gt;
&lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=eDViMtH8V1k:GdZwk7DRG24:yIl2AUoC8zA"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?d=yIl2AUoC8zA" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=eDViMtH8V1k:GdZwk7DRG24:V_sGLiPBpWU"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=eDViMtH8V1k:GdZwk7DRG24:V_sGLiPBpWU" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=eDViMtH8V1k:GdZwk7DRG24:F7zBnMyn0Lo"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=eDViMtH8V1k:GdZwk7DRG24:F7zBnMyn0Lo" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=eDViMtH8V1k:GdZwk7DRG24:gIN9vFwOqvQ"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=eDViMtH8V1k:GdZwk7DRG24:gIN9vFwOqvQ" border="0"&gt;&lt;/img&gt;&lt;/a&gt;
&lt;/div&gt;</description><author>mike.d.conley@gmail.com (mikeconley)</author><pubDate>Wed, 05 May 2010 01:31:43 -0700</pubDate><guid>http://mikeconley.ca/blog/2010/05/04/python-metaclasses-in-review-board/</guid></item><item><title>Looking Back on Review Board
</title><link>http://www.chipx86.com/blog/2010/05/04/looking-back-on-review-board/</link><description>&lt;p&gt;Just over 3.5 years ago, David Trowbridge and I spent some time discussing the annoyances of the typical patch submission and code review processes in the open source projects we participated in and at companies, and decided to play with some ideas for improving this. At the time, we knew very little about what we intended to do. We had a name for it pretty early on, but that was about all we had. We didn&amp;#8217;t even know whether we&amp;#8217;d get past an early prototyping stage. But here it is, over 3 years later, and we have the leading open source code review tool with an active support and development community, hundreds of companies using it, and exciting new innovations for aiding in the code review process.&lt;/p&gt;
&lt;p&gt;I was thinking a few days ago about how far we&amp;#8217;ve come and some of the decisions we made along the way. I went digging through our commit history in order to relive some of the past of our little project. Since so few people were even aware of Review Board&amp;#8217;s existence at the time, I thought I&amp;#8217;d share some of our history with you. Particularly the interesting and funny bits.&lt;/p&gt;
&lt;p&gt;&lt;b&gt;&amp;#8220;Add the reviewboard.&amp;#8221;&lt;/b&gt;&lt;/p&gt;
&lt;p&gt;Commit #1. The very first thing we put in our Subversion tree on September 27, 2006. I don&amp;#8217;t even remember what was in this change now. We transitioned to Git last year and this commit is now just plain empty. Maybe it was jut the directory structure? Who can say.&lt;/p&gt;
&lt;p&gt;Early on, we didn&amp;#8217;t refer to &amp;#8220;Review Board&amp;#8221; as a proper name. It was generally &amp;#8220;the reviewboard&amp;#8221; or something similar. The codebase was young. We didn&amp;#8217;t actually do code review on the project at this point (and it shows!). The first few months are littered with odd or nonsensical commit messages, small breakages, and bad decisions.&lt;/p&gt;
&lt;p&gt;A few of my favorite commit messages are:&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;&amp;#8220;I suck. Make submitting of reviews.&amp;#8221;&lt;/li&gt;
&lt;li&gt;&amp;#8220;Don&amp;#8217;t stuff the list of files in the bug list. It&amp;#8217;s impolite.&amp;#8221;&lt;/li&gt;
&lt;li&gt;&amp;#8220;Avoid failing out with Christian&amp;#8217;s wacko form&amp;#8221;&lt;/li&gt;
&lt;li&gt;&amp;#8220;Gum.&amp;#8221;&lt;/li&gt;
&lt;li&gt;&amp;#8220;Holy apple pancakes. It worked!&amp;#8221;&lt;/li&gt;
&lt;li&gt;&amp;#8220;I suck&amp;#8230; The array was empty&amp;#8230; The tests never had a chance to fail. :(&amp;#8221;&lt;/li&gt;
&lt;li&gt;&amp;#8220;&amp;#8216;This is a summary&amp;#8217; sucks. Now we use fortune for the summary, description, and testing done. &amp;#8216;You&amp;#8217;re ugly and your mother dresses you funny.&amp;#8217;&amp;#8221;&lt;/li&gt;
&lt;li&gt;&amp;#8220;Unbreak things before ChipX86 notices&amp;#8221;&lt;/li&gt;
&lt;li&gt;&amp;#8220;I&amp;#8217;m just&amp;#8230; garhgh&amp;#8221;&lt;/li&gt;
&lt;/ul&gt;
&lt;p&gt;Nowadays, our commit messages look nothing like that, but that&amp;#8217;s the fun of a new project. You get to go commit-crazy while you try to figure out what you&amp;#8217;re building.&lt;/p&gt;
&lt;p&gt;&lt;b&gt;Dashboard, quips and fortunes&lt;/b&gt;&lt;/p&gt;
&lt;p&gt;The UI of old looked quite different than the UI of today.&lt;/p&gt;
&lt;p&gt;We had a dashboard from the very beginning (before the review request pages, even) but it wasn&amp;#8217;t anything like the dashboard we had today. It was a simple page with a table containing all outgoing review requests and a table containing incoming review requests. But it also had one more thing: quips.&lt;/p&gt;
&lt;p&gt;The beginning of quips functionality was being built. Quips are just little random quotes that are inserted in the UI. I think the plan was to put quips on certain pages, making Review Board a little more fun. We were using them in the dashboard for empty lists, with variations all saying something about the dashboard being empty. Quips are a neat feature that just never survived the early days of development.&lt;/p&gt;
&lt;p&gt;Fortunes are similar. On Linux/Unix systems, there&amp;#8217;s a little program called &amp;#8220;fortune&amp;#8221; that just displays a random quote. Since we at first had to test review request functionality without actually having a repository backend of any sort, and we didn&amp;#8217;t want to input all the information each time, we just used fortune to generate the summary, description and testing done text. This made for some really funny review requests early on, but this is of course something that had no reason to survive initial development.&lt;/p&gt;
&lt;p&gt;Sometimes we would create a bunch of review requests just to see what kind of quotes we&amp;#8217;d get. :)&lt;/p&gt;
&lt;p&gt;&lt;b&gt;Multiple repositories? Almost didn&amp;#8217;t happen.&lt;/b&gt;&lt;/p&gt;
&lt;p&gt;One of the really critical parts of Review Board today is the ability to talk to a variety of different types of repositories in one instance. But, it turns out, this almost didn&amp;#8217;t happen.&lt;/p&gt;
&lt;p&gt;The initial goals were not that ambitious. Review Board talked to one repository per instance. Everything was basically hard-coded with one repository in mind. That type of repository, as well as its information, was customizable. You just couldn&amp;#8217;t have more than one. At the time, this wasn&amp;#8217;t a problem, but it didn&amp;#8217;t take long until we had a need to talk to two repositories.&lt;/p&gt;
&lt;p&gt;We discussed this and at first decided that if we needed to talk to two repositories, we could just set up two instances. It would have been a lot of work to update it for multiple repositories, after all. And really, this was a small project. Who would really need more than a couple repositories? This started to nag at me, though, and so I spent a couple nights rewriting all of the code as an experiment. It ended up working pretty nicely, and we were able to ditch the multi-instance model.&lt;/p&gt;
&lt;p&gt;&lt;b&gt;The importance of rewards&lt;/b&gt;&lt;/p&gt;
&lt;p&gt;It&amp;#8217;s always nice to have a little reward for milestones. Developers sometimes compete over cool bug numbers, revisions, etc. Initially, we were going to use quips to add some fun to the site, but we ended up settling on our current trophy system.&lt;/p&gt;
&lt;p&gt;One of our first Review Board instances started to approach review request #1000, which was a huge milestone for us. I decided to commemorate the event by staying up and quickly hacking in a hidden feature for showing a trophy for review request #1000. The way we implemented it, you&amp;#8217;d see the first ever trophy at 1,000, and from there you&amp;#8217;d see it at every milestone number (1,000, 2,000, 3,000, 10,000, etc.). I didn&amp;#8217;t want to stop there, though, so I added support for a second type of trophy, one that has confused people with its appearance to this day. Mission complete.&lt;/p&gt;
&lt;p&gt;Of course, when we updated the server and someone finally hit 1000, it triggered a bug in the new trophy code and broke his review request. Oh well, I tried.&lt;/p&gt;
&lt;p&gt;&lt;b&gt;Diff viewers are hard&lt;/b&gt;&lt;/p&gt;
&lt;p&gt;If I could pick one point during the whole history of Review Board where I was ready to completely give up, it would be during the creation of our diff viewer. All three diff viewers.&lt;/p&gt;
&lt;p&gt;See, the first diff viewer was a complete and total hack. We generated a side-by-side diff using the diff tools and just parsed the output, basically generating a table of that. It was ugly, though, and limiting. It also caused problems where text on a row would either be truncated or would break the parser. I spent a long time working on this before I totally gave up and went on to try a new approach.&lt;/p&gt;
&lt;p&gt;My second approach was closer to what we have today, but also limiting and very, very buggy. We were using Python&amp;#8217;s built-in diff generation module, which implements a basic diff algorithm. It gave us insert and delete information, but not replace information. We had to hack that in ourselves, and it was really a hack. Try taking a bunch of inserts and deletes and find out which of those are really changed lines. No, really, try it. It&amp;#8217;s harder than you think, and it&amp;#8217;ll often be wrong.&lt;/p&gt;
&lt;p&gt;Still, we stuck with this for a long time. It was slow, buggy, and didn&amp;#8217;t generate the sort of output people expected from diff tools. Most people see diffs from GNU Diff, which implements the Meyers Diff Algorithm (with a few additions and tweaks). These Meyers diffs are much nicer to view than what Python gave us. Another problem we hit was that we didn&amp;#8217;t have real line number information, so we had to output fake line numbers. They weren&amp;#8217;t really line numbers so much as row numbers in the table. Ugh. Even getting this far was really hard and frustrating, and the result still wasn&amp;#8217;t good.&lt;/p&gt;
&lt;p&gt;Attempt #3. I decided to build our own diff parser and generator from scratch. What a project. I knew nothing about diff generation and hardly knew where to start. I spent probably a good month or so just trying to work on this new diff code, and was so close to giving up so many times. It ended up being completely worth it, though, as we ended up with a very nice, extensible diff parser.&lt;/p&gt;
&lt;p&gt;Without that third attempt, we&amp;#8217;d be in the stone age. Review Board would not be as nice to use. We wouldn&amp;#8217;t have inter-line diffs (where we highlight what changed in a replace line), syntax highlighting, move detection (coming in 1.5), or function/class headers (where we show which function/class the part of the diff is in &amp;#8212; also coming in 1.5).&lt;/p&gt;
&lt;p&gt;&lt;b&gt;What else&amp;#8230;&lt;/b&gt;&lt;/p&gt;
&lt;p&gt;Well, there&amp;#8217;s a lot more I could talk about. Our initial attempts at JavaScript code for the UI, our trials and challenges with database migration, or our early problems storing diffs with different encodings in databases. This is getting long, though, so I&amp;#8217;ll cover these in another post on lessons learned.&lt;/p&gt;&lt;div class="feedflare"&gt;
&lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=-bsNUPXMJII:zyYZy9angkQ:yIl2AUoC8zA"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?d=yIl2AUoC8zA" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=-bsNUPXMJII:zyYZy9angkQ:V_sGLiPBpWU"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=-bsNUPXMJII:zyYZy9angkQ:V_sGLiPBpWU" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=-bsNUPXMJII:zyYZy9angkQ:F7zBnMyn0Lo"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=-bsNUPXMJII:zyYZy9angkQ:F7zBnMyn0Lo" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=-bsNUPXMJII:zyYZy9angkQ:gIN9vFwOqvQ"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=-bsNUPXMJII:zyYZy9angkQ:gIN9vFwOqvQ" border="0"&gt;&lt;/img&gt;&lt;/a&gt;
&lt;/div&gt;</description><author>chipx86@chipx86.com (chipx86)</author><pubDate>Tue, 04 May 2010 09:53:00 -0700</pubDate><guid>http://www.chipx86.com/blog/2010/05/04/looking-back-on-review-board/</guid></item><item><title>The Review Board Extension Life-Cycle
</title><link>http://mikeconley.ca/blog/2010/05/02/reviewboard-extension-lifecycle/</link><description>&lt;p&gt;According to &lt;a href="http://socghop.appspot.com/document/show/gsoc_program/google/gsoc2010/timeline"&gt;the timeline&lt;/a&gt;, I&amp;#8217;m still in the community-bonding period for GSoC.  Coding for &lt;a href="http://mikeconley.ca/blog/2010/04/27/my-gsoc-project-review-board-extensions/"&gt;my project&lt;/a&gt; is supposed to start sometime towards the end of May.&lt;/p&gt;
&lt;p&gt;So I&amp;#8217;m using the time to do the following:&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;Close as many small, easy tickets as I can for the  upcoming Review Board release.  I&amp;#8217;ve already posted some patches for review.  More forthcoming.&lt;/li&gt;
&lt;li&gt;Get to know the tools I&amp;#8217;ll be using.  Review Board is hosted on Github, and I&amp;#8217;m relatively new to the whole DVCS thing.  I&amp;#8217;ve been figuring out &lt;a href="http://progit.org"&gt;how to use Git&lt;/a&gt;, how to post patches, merging, branching, etc.&lt;/li&gt;
&lt;li&gt;Get to know the area I&amp;#8217;ll be working in.  I&amp;#8217;ve been figuring out how Django apps organize themselves.  I&amp;#8217;ve also drawn up &lt;a href="http://mikeconley.ca/blog/2010/04/30/code-spelunking-review-board-extensions-2/"&gt;a map of the current state of the extension framework&lt;/a&gt; to help me visualize it.&lt;/li&gt;
&lt;li&gt;Get to know the other developers working on Review Board.  I&amp;#8217;ve been hanging out in the #reviewboard-soc FreeNode IRC channel.  Very nice, and helpful people to work with.&lt;/li&gt;
&lt;li&gt;Develop a plan of attack for my project&lt;/li&gt;
&lt;/ul&gt;
&lt;p&gt;And this last point is the one I want to talk about.&lt;/p&gt;
&lt;h3&gt;The Review Board Extension Life-Cycle&lt;/h3&gt;
&lt;p&gt;An extension isn&amp;#8217;t just some isolated piece of code that gets crammed into an application.  When you&amp;#8217;ve got multiple extensions already installed and running, installing and activating a new extension is like introducing a new animal into an ecosystem.  You have to make sure that your new animal plays nice with the others, and that, in the morning, there won&amp;#8217;t be a pile of rotting corpses where your application used to be.&lt;/p&gt;
&lt;p&gt;I may have gotten carried away with my metaphor.&lt;/p&gt;
&lt;p&gt;So let&amp;#8217;s look at what I&amp;#8217;m envisioning as the life-cycle for a Review Board extension.  I&amp;#8217;ll start right from the top.&lt;/p&gt;
&lt;h4&gt;Getting the Extension&lt;/h4&gt;
&lt;p&gt;Ideally, this will work as nicely as WordPress&amp;#8217;s implementation:  a Review Board administrator is given a catalog of extensions to choose from within the Administrator interface, and one click later, the desired extensions are downloaded and ready to be installed.&lt;/p&gt;
&lt;p&gt;For all you system administrators out there, that last idea might make your toes curl.  A Review Board administrator is not necessarily a system administrator, and the system administrator knows what he/she likes on their machine.  An application that can go and download other applications can be dangerous.  We have to ensure that the application that we&amp;#8217;re downloading is the one we&amp;#8217;re &lt;em&gt;trying &lt;/em&gt;to download.  The last thing we need is some man-in-the-middle to do something cute and bork the code review machine.  And we want to ensure that the extension functions as advertised.  No hidden features.  No self-destruct mechanisms.  No back doors.&lt;/p&gt;
&lt;p&gt;Do I have a plan for this part?  Well&amp;#8230;.no, not really.  I don&amp;#8217;t imagine I&amp;#8217;ll get that far &amp;#8211; I consider it a little out of my scope.  So, for my project, I think it&amp;#8217;ll satisfy if the system admin (or Review Board admin) can manually download the extension, decompress it, and place it where Review Board can work with it.  That other stuff can come later (and I&amp;#8217;ll try to design so that it can come later &lt;em&gt;easily&lt;/em&gt;).&lt;/p&gt;
&lt;h4&gt;Installing the Extension&lt;/h4&gt;
&lt;p&gt;Ok, so at this point, we&amp;#8217;ve got our extension downloaded in a place where Review Board can see it.&lt;/p&gt;
&lt;p&gt;So now what?&lt;/p&gt;
&lt;p&gt;Now we need to &lt;em&gt;install&lt;/em&gt; the extension.  To me, that means letting the extension put its roots into the RB install by creating database tables, preparing initial data, and generally doing everything to make conditions suitable for the extension to function.&lt;/p&gt;
&lt;p&gt;When an extension install begins, I imagine it will consider the following questions (in no particular order):&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;Do I (the extension) depend on other extensions to function?  If so, are those extensions present?  If not, let the user know so that they can go get them.&lt;/li&gt;
&lt;li&gt;Are there some extensions already installed that will conflict with me, or make me behave badly?  If so, let the user know so that they can either remove that conflict, or find an alternative extension.&lt;/li&gt;
&lt;li&gt;Is the user entirely aware of what I can do?  Make sure that my capabilities, limitations, and behavioural quirks are known to the user.&lt;/li&gt;
&lt;/ol&gt;
&lt;p&gt;Once those 3 questions are answered, and everything is looking good for the install, the extension will create the database tables it needs (if any) in order to function.  If anything goes wrong during this process, the database changes will be rolled back, and the user will get a full read out about what went wrong.&lt;/p&gt;
&lt;p&gt;If nothing goes wrong, the extension will be installed.  The user might then be asked to set some initial operating parameters for the extension.&lt;/p&gt;
&lt;p&gt;Ok great &amp;#8211; the extension is installed.  Now what?&lt;/p&gt;
&lt;h4&gt;Activating the Extension&lt;/h4&gt;
&lt;p&gt;For something that sounds so dramatic, the explanation about what happens is pretty short:  Review Board simply becomes aware that the extension is activated, and passes data through the necessary hooks in order for the extension to function properly.  Upon activation, the extension should do a quick double-check to ensure that all prerequisites for the extension have been met.  This is because we can&amp;#8217;t trust users to activate an extension immediately after downloading them.&lt;/p&gt;
&lt;h4&gt;The Extension Runs&lt;/h4&gt;
&lt;p&gt;While it&amp;#8217;s activated, the extension will probably react to various events that happen on Review Board.  Tables will be updated.  View methods will be run.  Templates will be rendered.&lt;/p&gt;
&lt;p&gt;If anything, ever, goes horribly wrong with an extension, the following will happen:&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;A log entry will be written, dumping the error, and information about what the user was trying to do&lt;/li&gt;
&lt;li&gt;An error message will be displayed to the user, trying to tell them what exactly happened&lt;/li&gt;
&lt;li&gt;The extension (and its dependents) will be deactivated.  They will no longer react to events on Review Board.  Their tables will still be there, the settings will still, but the extension will be, in essence, asleep.  Dormant.  Non-reactive.&lt;/li&gt;
&lt;/ol&gt;
&lt;p&gt;The administrator could try to reactivate the extension at this point.  They might try to contact the extension developer for support.&lt;/p&gt;
&lt;h4&gt;The Extension is Un-installed&lt;/h4&gt;
&lt;p&gt;If the user wants to rid themselves of an extension, they must first deactivate it.  This will put it (and its dependents) in the dormant state.  It just switches them off, nothing else.&lt;/p&gt;
&lt;p&gt;Deactivated extensions can then be un-installed.  If the user chooses to un-install the extension, the extension database tables and settings will be wiped out.&lt;/p&gt;
&lt;p&gt;The extension itself won&amp;#8217;t be deleted though &amp;#8211; at least, not within the scope of my project.  The extension files will need to be removed from Review Board manually.&lt;/p&gt;
&lt;p&gt;At this point, any dependents that this uninstalled extension had will no longer be able to be activated.&lt;/p&gt;
&lt;p&gt;Anyhow, that&amp;#8217;s how I envision the life-cycle.  It&amp;#8217;s my first go at it, so I&amp;#8217;d love to hear some feedback if you have any.&lt;/p&gt;&lt;div class="feedflare"&gt;
&lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=83ziWRET4Ow:xsD6sgFMKkk:yIl2AUoC8zA"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?d=yIl2AUoC8zA" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=83ziWRET4Ow:xsD6sgFMKkk:V_sGLiPBpWU"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=83ziWRET4Ow:xsD6sgFMKkk:V_sGLiPBpWU" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=83ziWRET4Ow:xsD6sgFMKkk:F7zBnMyn0Lo"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=83ziWRET4Ow:xsD6sgFMKkk:F7zBnMyn0Lo" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=83ziWRET4Ow:xsD6sgFMKkk:gIN9vFwOqvQ"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=83ziWRET4Ow:xsD6sgFMKkk:gIN9vFwOqvQ" border="0"&gt;&lt;/img&gt;&lt;/a&gt;
&lt;/div&gt;</description><author>mike.d.conley@gmail.com (mikeconley)</author><pubDate>Mon, 03 May 2010 02:55:07 -0700</pubDate><guid>http://mikeconley.ca/blog/2010/05/02/reviewboard-extension-lifecycle/</guid></item><item><title>Code Spelunking: Review Board Extensions
</title><link>http://mikeconley.ca/blog/2010/04/30/code-spelunking-review-board-extensions-2/</link><description>&lt;p&gt;So this summer, &lt;a href="http://mikeconley.ca/blog/2010/04/27/my-gsoc-project-review-board-extensions/"&gt;I&amp;#8217;m working on Review Board for the Google Summer of Code.&lt;/a&gt;&lt;/p&gt;
&lt;p&gt;Until my GSoC acceptance, my romps into the code had been relatively shallow.  But with my proposal being given the green light, I&amp;#8217;ve started doing more extensive explorations.&lt;/p&gt;
&lt;p&gt;Review Board is built using &lt;a href="http://www.djangoproject.com/"&gt;the Django web framework&lt;/a&gt;.  I haven&amp;#8217;t worked with Django before, but I have quite a bit of experience with Rails, so that should be an asset.  Using a web framework means having (relatively) predictable source code layout, and Review Board is no exception.&lt;/p&gt;
&lt;h4&gt;Djblets&lt;/h4&gt;
&lt;p&gt;At one point or another, the Review Board developers realized that a lot of their code wasn&amp;#8217;t Review Board specific, and could be abstracted out into an external library.&lt;/p&gt;
&lt;p&gt;That library is called &lt;a href="http://github.com/djblets/djblets"&gt;Djblets&lt;/a&gt;.&lt;/p&gt;
&lt;p&gt;Among other things, Djblets adds a DataGrid component for easy record sorting and pagination.  There are improvements to Django&amp;#8217;s Authentication system.  Functions for easily displaying a user&amp;#8217;s &lt;a href="http://en.gravatar.com/"&gt;Gravatar&lt;/a&gt;.&lt;/p&gt;
&lt;p&gt;And, low and behold, there is a branch of Djblets that provides classes and functions for giving a Django application an extension framework.  The classes are abstract enough so that, in your Django application, you can specify different types and behaviours for your Hooks.&lt;/p&gt;
&lt;h4&gt;Djblets -&amp;gt; Review Board&lt;/h4&gt;
&lt;p&gt;The Review Board extension branch takes these Djblets extension classes, and extends them into DashboardHooks, NavigationBarHooks, ReviewRequestDetailHooks&amp;#8230;lots of different hooks.&lt;/p&gt;
&lt;p&gt;So, Djblets creates the foundation abstractions.  Review Board makes these abstractions a little more specific.  And then an extension writer needs to instantiate and use these classes to design their extensions.  It sounds complicated, I know.&lt;/p&gt;
&lt;h4&gt;So Let&amp;#8217;s Map It Out&lt;/h4&gt;
&lt;p&gt;When I start learning a new code base, I do a lot of drawing.  To me, getting to now a code base is like getting to know a city, and that means walking around it, and mapping it out.&lt;/p&gt;
&lt;p&gt;So I&amp;#8217;ve taken the liberty of mapping out the extension classes that I&amp;#8217;ve found, and how they relate to one another.  Note that at the bottom of my map, a simple extension (RB Reports) is using some of those classes to hook itself into Review Board.  You can find this, and other extensions,&lt;a href="http://github.com/chipx86/rb-extension-pack"&gt;here&lt;/a&gt;.&lt;/p&gt;
&lt;div class="wp-caption alignnone" id="attachment_1222" style="width: 310px;"&gt;&lt;a href="http://www.mikeconley.ca/images/rb_ext/Map.html"&gt;&lt;img alt="My map of the extension framework" class="size-medium wp-image-1222" height="238" src="http://mikeconley.ca/blog/wp-content/uploads/2010/04/Map-300x238.png" title="Extensions Map" width="300" /&gt;&lt;/a&gt;&lt;p class="wp-caption-text"&gt;Click here to check out my map of the current state of the extension framework&lt;/p&gt;&lt;/div&gt;
&lt;p&gt;Now, before someone in the department starts complaining about my misuse of UML:  I&amp;#8217;m not a UML guy.  I just wanted an easy piece of diagramming software, and the one that I found (&lt;a href="http://www.gnome.org/projects/dia/"&gt;Dia&lt;/a&gt;), did UML.  I just wanted something to draw boxes and lines. So please don&amp;#8217;t freak out if you think I&amp;#8217;m using the wrong symbols.&lt;/p&gt;
&lt;p&gt;One symbol you might be wondering about is the blue quantum-flux-capacitor-implosion.&lt;/p&gt;
&lt;p&gt;I&amp;#8217;ll save that for a future post.&lt;/p&gt;&lt;div class="feedflare"&gt;
&lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=aAuTZB0_nlo:dxBa5lcRSOM:yIl2AUoC8zA"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?d=yIl2AUoC8zA" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=aAuTZB0_nlo:dxBa5lcRSOM:V_sGLiPBpWU"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=aAuTZB0_nlo:dxBa5lcRSOM:V_sGLiPBpWU" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=aAuTZB0_nlo:dxBa5lcRSOM:F7zBnMyn0Lo"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=aAuTZB0_nlo:dxBa5lcRSOM:F7zBnMyn0Lo" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=aAuTZB0_nlo:dxBa5lcRSOM:gIN9vFwOqvQ"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=aAuTZB0_nlo:dxBa5lcRSOM:gIN9vFwOqvQ" border="0"&gt;&lt;/img&gt;&lt;/a&gt;
&lt;/div&gt;</description><author>mike.d.conley@gmail.com (mikeconley)</author><pubDate>Sat, 01 May 2010 00:25:04 -0700</pubDate><guid>http://mikeconley.ca/blog/2010/04/30/code-spelunking-review-board-extensions-2/</guid></item><item><title>My GSoC Project:  Review Board Extensions
</title><link>http://mikeconley.ca/blog/2010/04/27/my-gsoc-project-review-board-extensions/</link><description>&lt;p&gt;If you didn&amp;#8217;t already know, &lt;a href="http://www.reviewboard.org"&gt;Review Board&lt;/a&gt; is an open-source web-based code review tool.  &lt;a href="http://www.markusproject.org"&gt;The MarkUs Team&lt;/a&gt; has been using Review Board for &lt;a href="http://mikeconley.ca/blog/2010/02/02/pre-commit-code-review-in-markus-development/"&gt;pre-commit code review&lt;/a&gt; for about a year now.  This has given the team a number of advantages:&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;For a team that usually has a 4 month turnover, this allows us to quickly get new team members up to speed with how to contribute to MarkUs.  We review every change that they propose, and give them tips/guidance on how to make it fit in well with the application.  They learn, and the applications code stays healthy.&lt;/li&gt;
&lt;li&gt;We catch defects before they enter the code base.  Simple as that.&lt;/li&gt;
&lt;li&gt;We get a good sense of what other people are working on, and what is going on in the code.  Review Board has become a central conversation and learning hub for the developers on the MarkUs team.&lt;/li&gt;
&lt;/ol&gt;
&lt;p&gt;So, the long and the short of it:  &lt;em&gt;I like Review Board.  Review Board helps us write better code.  I want to make Review Board better.&lt;/em&gt;&lt;/p&gt;
&lt;p&gt;So what am I proposing?&lt;/p&gt;
&lt;h3&gt;How to Avoid A Bloated Software Monster&lt;/h3&gt;
&lt;p&gt;You can never make some people happy.&lt;/p&gt;
&lt;p&gt;No matter how decent your software is, someone will eventually come up to you and say:&lt;/p&gt;
&lt;blockquote&gt;&lt;p&gt;Wow!  Your software would be perfect if only it had feature XYZ!  Sadly, because you don&amp;#8217;t have feature XYZ, I can&amp;#8217;t use it.  Please implement, k thx!&lt;/p&gt;&lt;/blockquote&gt;
&lt;p&gt;And so you either have to politely say &amp;#8220;no&amp;#8221;, and lose that user, or say &amp;#8220;yes&amp;#8221;, and add feature XYZ to the application.  And for users out there who don&amp;#8217;t need, or don&amp;#8217;t care about feature XYZ, that new feature just becomes a distraction and adds no value.  Make this happen a bunch of times, and you&amp;#8217;ve got yourself a bloated mutha for a piece of software.&lt;/p&gt;
&lt;p&gt;And we don&amp;#8217;t want a bloated piece of software.  But we &lt;em&gt;do &lt;/em&gt;want to make our users happy, and provide feature XYZ for them if they want it.&lt;/p&gt;
&lt;p&gt;So what&amp;#8217;s the solution?  &lt;strong&gt;We provide an extension framework&lt;/strong&gt; (which is also sometimes called a plug-in architecture).&lt;/p&gt;
&lt;p&gt;An extension framework allows developers to easily expand a piece of software to do new things.  So, if a user wants feature XYZ, we (or someone else) just creates and make available an extension that implements the feature.  The user installs the extension, activates it, and bam &amp;#8211; our user is happy as a clam with their new feature.&lt;/p&gt;
&lt;p&gt;And if we make it super-easy to develop them, third-party developers can write new, wonderful, interesting extensions to do things that&amp;#8230;well, we wouldn&amp;#8217;t have considered in the first place. It&amp;#8217;s a new place for innovation.  What&amp;#8217;s that old cliché?&lt;/p&gt;
&lt;blockquote&gt;&lt;p&gt;If you build it [the plug-in framework], they will come [the third-party developers who write awesome things]&lt;/p&gt;&lt;/blockquote&gt;
&lt;p&gt;And the developers do come.  Just look at &lt;a href="https://addons.mozilla.org/en-US/firefox/"&gt;Firefox add-ons&lt;/a&gt; or &lt;a href="http://wordpress.org/extend/plugins/"&gt;WordPress plugins&lt;/a&gt;.  Entire &lt;em&gt;ecosystems&lt;/em&gt; of extensions, doing things that the original developers would probably have never dreamed of doing on their own.  Hell, &lt;em&gt;I&amp;#8217;ve &lt;/em&gt;even written &lt;a href="../category/technology/firefox-extensions/alertcheck-firefox-extensions-technology/"&gt;a Firefox add-on.&lt;/a&gt; And users love customizing their Firefox / WordPress with those extensions.  It &lt;em&gt;adds value&lt;/em&gt;.&lt;/p&gt;
&lt;p&gt;So we get wins all over the place:&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;Our user gets their feature&lt;/li&gt;
&lt;li&gt;The software gets more attractive because it&amp;#8217;s flexible and customizable&lt;/li&gt;
&lt;li&gt;The original software developers get to focus on the core piece of software, and let the third-party developers focus on the fringe features&lt;/li&gt;
&lt;/ul&gt;
&lt;p&gt;And this is where I think I can help Review Board.&lt;/p&gt;
&lt;p&gt;(Before I go on, if you&amp;#8217;re interested, &lt;a href="http://www.fogcreek.com/fogbugz/blog/post/the-fogbugz-plugin-architecture.aspx"&gt;here&amp;#8217;s another article on the how and the why of plug-in architectures&lt;/a&gt;)&lt;/p&gt;
&lt;h3&gt;Review Board Extensions&lt;/h3&gt;
&lt;p&gt;So if you look at the &lt;a href="http://code.google.com/p/reviewboard/w/list"&gt;Review Board Wiki&lt;/a&gt;, or glance at the &lt;a href="http://www.reviewboard.org/mailing-lists/"&gt;mailing lists&lt;/a&gt; you see numerous requests from users for new features, for example:&lt;/p&gt;
&lt;blockquote&gt;&lt;p&gt;It would be nice if the review board had a &amp;#8220;next comment&amp;#8221; button that is always available to click, or had a collapse/expand button. This would make it easier to see other people&amp;#8217;s comments in cases like this.&lt;/p&gt;
&lt;p&gt;&amp;#8230;&lt;/p&gt;
&lt;p&gt;It will be nice to have post-commit support. Instead of every post-commit review being a separate URL, if we could setup default rules for post-commit reviews to update an existing review providing the diff-between-diff features, it would be very useful.&lt;/p&gt;&lt;/blockquote&gt;
&lt;p&gt;The Review Board developers could smell the threat of bloated feature-creep from a mile away.  So, &lt;a href="http://github.com/chipx86/reviewboard/tree/extensions"&gt;in a separate branch&lt;/a&gt;, they began working on integrating an extension framework into Review Board.&lt;/p&gt;
&lt;p&gt;The extension branch, however, has been gathering dust, while the developers focus on more critical patches and releases.&lt;/p&gt;
&lt;p&gt;My GSoC proposal is to finish off a draft of the extension framework, document it, and build a very simple extension for it.  My simple extension will allow me to record basic statistics about Review Board reviewers &amp;#8211; for example, how long they spend on a particular review, their inspection rate, etc.&lt;/p&gt;
&lt;p&gt;Having been a project lead &lt;a href="http://www.markusproject.org"&gt;MarkUs&lt;/a&gt; for so long, it&amp;#8217;s going to be a good experience to be back on &amp;#8220;the bottom&amp;#8221; &amp;#8211; to be the new developer who doesn&amp;#8217;t entirely have a sense of the application code yet.  It&amp;#8217;s going to be good to go code spelunking again.  I&amp;#8217;ve done some preliminary explorations, and it&amp;#8217;s reminding me of my first experiences with MarkUs.  Like &lt;a href="http://www.youtube.com/watch?v=7N7i7_ts_eg"&gt;a submarine using its sonar&lt;/a&gt;, I&amp;#8217;m slowly getting a sense of the code terrain.&lt;/p&gt;
&lt;p&gt;I&amp;#8217;ll let you know what my first few sweeps find.&lt;/p&gt;&lt;div class="feedflare"&gt;
&lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=cYuIs-CKyTw:xzh8mt2C3dk:yIl2AUoC8zA"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?d=yIl2AUoC8zA" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=cYuIs-CKyTw:xzh8mt2C3dk:V_sGLiPBpWU"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=cYuIs-CKyTw:xzh8mt2C3dk:V_sGLiPBpWU" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=cYuIs-CKyTw:xzh8mt2C3dk:F7zBnMyn0Lo"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=cYuIs-CKyTw:xzh8mt2C3dk:F7zBnMyn0Lo" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=cYuIs-CKyTw:xzh8mt2C3dk:gIN9vFwOqvQ"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=cYuIs-CKyTw:xzh8mt2C3dk:gIN9vFwOqvQ" border="0"&gt;&lt;/img&gt;&lt;/a&gt;
&lt;/div&gt;</description><author>mike.d.conley@gmail.com (mikeconley)</author><pubDate>Tue, 27 Apr 2010 19:54:24 -0700</pubDate><guid>http://mikeconley.ca/blog/2010/04/27/my-gsoc-project-review-board-extensions/</guid></item><item><title>Ping!
</title><link>http://mikeconley.ca/blog/2010/04/27/ping/</link><description>&lt;p&gt;I&amp;#8217;ve &lt;a href="http://mikeconley.ca/blog/2010/01/19/markus-squad-hows-refactor-my-code-belated-happy-holidays-im-not-dead/"&gt;done it&lt;/a&gt; &lt;a href="http://mikeconley.ca/blog/2009/08/20/still-alive/"&gt;again&lt;/a&gt;:  I&amp;#8217;ve let dust gather on my blog.&lt;/p&gt;
&lt;p&gt;Quick update:&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;I&amp;#8217;ve finished my courses for this semester, and have gone into &lt;a href="http://mikeconley.ca/blog/2010/03/29/does-peer-grading-make-students-better-programmers/"&gt;full-blown research mode&lt;/a&gt;.&lt;/li&gt;
&lt;li&gt;My research proposal is going through ethics review, in order to make sure that I&amp;#8217;m not going to blow things up (or hurt anybody if I do)&lt;/li&gt;
&lt;li&gt;While my paperwork is reviewed, I&amp;#8217;m refining my procedure and apparatus.  Better and better.&lt;/li&gt;
&lt;li&gt;I&amp;#8217;ve been accepted into &lt;a href="http://socghop.appspot.com"&gt;Google Summer of Code&lt;/a&gt; this year &amp;#8211; I&amp;#8217;ll be working on &lt;a href="http://www.reviewboard.org"&gt;Review Board&lt;/a&gt;.  Details about my project will be the subject of an upcoming post, which I will toss up shortly.&lt;/li&gt;
&lt;li&gt;I may or may not be co-directing a radio play.  I&amp;#8217;ll let you know.&lt;/li&gt;
&lt;li&gt;&lt;a href="http://www.markusproject.org"&gt;The MarkUs team&lt;/a&gt; is about to release version 0.7, and a fresh batch of Summer students will soon be here at UofT to work on it!&lt;/li&gt;
&lt;li&gt;I have &lt;em&gt;not &lt;/em&gt;forgotten about the &lt;a href="http://www.uc.utoronto.ca/content/view/340/2083/"&gt;UCDP&lt;/a&gt; &lt;a href="http://mikeconley.ca/blog/category/personal/poland/"&gt;trip to Poland&lt;/a&gt;.  I still have to tell you what we saw and did at Auschwitz.  Cripes &amp;#8211; it&amp;#8217;s almost a year since I returned, and I&amp;#8217;m only half-way through the whole story.  And there&amp;#8217;s a ton more to tell.  Coming soon.&lt;/li&gt;
&lt;/ul&gt;
&lt;p&gt;Stay tuned.&lt;/p&gt;&lt;div class="feedflare"&gt;
&lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=Y21VMNxYIeU:lPHzVBT0r5Y:yIl2AUoC8zA"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?d=yIl2AUoC8zA" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=Y21VMNxYIeU:lPHzVBT0r5Y:V_sGLiPBpWU"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=Y21VMNxYIeU:lPHzVBT0r5Y:V_sGLiPBpWU" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=Y21VMNxYIeU:lPHzVBT0r5Y:F7zBnMyn0Lo"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=Y21VMNxYIeU:lPHzVBT0r5Y:F7zBnMyn0Lo" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=Y21VMNxYIeU:lPHzVBT0r5Y:gIN9vFwOqvQ"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=Y21VMNxYIeU:lPHzVBT0r5Y:gIN9vFwOqvQ" border="0"&gt;&lt;/img&gt;&lt;/a&gt;
&lt;/div&gt;</description><author>mike.d.conley@gmail.com (mikeconley)</author><pubDate>Tue, 27 Apr 2010 18:03:05 -0700</pubDate><guid>http://mikeconley.ca/blog/2010/04/27/ping/</guid></item><item><title>Review Board 1.0 Released!
</title><link>http://www.chipx86.com/blog/2009/06/20/review-board-10-released/</link><description>&lt;p&gt;&lt;a href="http://www.review-board.org/"&gt;&lt;img alt="Review Board 1.0" class="alignleft size-full wp-image-319" height="123" src="http://www.chipx86.com/blog/wp-content/uploads/2009/06/rb10.png" title="Review Board 1.0" width="130" /&gt;&lt;/a&gt;&lt;/p&gt;
&lt;p&gt;Tonight, we hit a milestone in the &lt;a href="http://www.review-board.org/"&gt;Review Board&lt;/a&gt; project that we&amp;#8217;ve been working toward for over two years. We finally pushed out our 1.0 release. A lot of blood, sweat and tears went into this release (okay, so not literally, but it was A LOT OF WORK!). The last few months in particular have been challenging, as we&amp;#8217;ve had to solve some tricky bugs and scalability problems, but the end result is pretty great.&lt;/p&gt;
&lt;p&gt;Just a short while ago, we &lt;a href="http://www.review-board.org/news/2009/06/20/review-board-10-released/"&gt;announced&lt;/a&gt; the release and put up an &lt;a href="http://www.review-board.org/docs/releasenotes/dev/reviewboard/1.0/"&gt;overview&lt;/a&gt; of the entire release and product. We&amp;#8217;ve already had some nice congratulatory e-mails and tweets, which is really nice :)&lt;/p&gt;
&lt;p&gt;Some stats for this release:&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;2 years, 9 months, 25 days have passed since our first commit.&lt;/li&gt;
&lt;li&gt;120 contributors have contributed to Review Board so far (in terms of code contributions).&lt;/li&gt;
&lt;li&gt;2,019 commits were made.&lt;/li&gt;
&lt;li&gt;899 review requests have been posted to our project&amp;#8217;s actual &lt;a href="http://reviews.review-board.org/"&gt;Review Board server&lt;/a&gt;. 1,650 users are registered on there.&lt;/li&gt;
&lt;li&gt;Our &lt;a href="http://demo.review-board.org/"&gt;demo server&lt;/a&gt;, in comparison, has 2,082 review requests filed and 10,154 users.&lt;/li&gt;
&lt;li&gt;938 bugs were filed. 812 were fixed.&lt;/li&gt;
&lt;li&gt;232 feature requests were filed. 101 were implemented. Most remaining ones are scheduled for releases.&lt;/li&gt;
&lt;li&gt;An estimated 200+ companies are now using Review Board. 26 have let us &lt;a href="http://www.review-board.org/users/"&gt;list them publicly&lt;/a&gt;.&lt;/li&gt;
&lt;li&gt;The largest known Review Board install has over 83,000 filed review requests and over 2,000 users, doing upwards of 10GB of traffic per day.&lt;/li&gt;
&lt;li&gt;5 presentations on Review Board are known to have been given, 3 by us, 2 by others.&lt;/li&gt;
&lt;li&gt;552 users have joined our main mailing list, and 3,674 e-mails have been sent.&lt;/li&gt;
&lt;/ul&gt;
&lt;p&gt;Now that Review Board 1.0 is out, we can get started on some awesome new features we&amp;#8217;ve had planned. I have a little notebook full of ideas for our 1.1 and 1.5 releases (which may become 1.5 and 2.0, respectively, as this list grows). Some of the new features are actually ready to be committed within the next couple of days, so those of you using nightlies will start to see them soon.&lt;/p&gt;
&lt;p&gt;We were accepted into this year&amp;#8217;s &lt;a href="http://review-board.org/summer-of-code/"&gt;Summer of Code&lt;/a&gt;, and have three students working on exciting projects for us, so hopefully we&amp;#8217;ll start to see these trickle into the upcoming nightlies as well. Among these projects include diff viewer improvements (moved region detection, better whitespace-only change detection), IDE integration with Eclipse, and improved notification hooks and e-mail support.&lt;/p&gt;
&lt;p&gt;We&amp;#8217;re also working on providing support for third-party extensions, which will allow developers to extend Review Board in new, exciting ways without having to modify Review Board itself. This is especially handy for companies who wish to integrate better with their sandboxes, bug trackers or unit testing services. This will likely land in 1.5 (2.0?) at the earliest, as it&amp;#8217;s a large change, but the code for this mostly works today. It&amp;#8217;s just a matter of getting the codebase ready and figuring out what APIs we want to stabilize and expose.&lt;/p&gt;
&lt;p&gt;As I mentioned in the &lt;a href="http://www.review-board.org/news/2009/06/20/review-board-10-released/"&gt;release announcement&lt;/a&gt;, we&amp;#8217;re planning a release party, tentatively on July 11th, 2009, in the Bay Area (somewhere around Palo Alto, CA). If any Review Board users want to join us, please &lt;a href="http://tinyurl.com/rb1-release-party"&gt;RSVP!&lt;/a&gt;&lt;/p&gt;&lt;div class="feedflare"&gt;
&lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=vySeBU86Ujg:uJHUZuLTCYc:yIl2AUoC8zA"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?d=yIl2AUoC8zA" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=vySeBU86Ujg:uJHUZuLTCYc:V_sGLiPBpWU"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=vySeBU86Ujg:uJHUZuLTCYc:V_sGLiPBpWU" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=vySeBU86Ujg:uJHUZuLTCYc:F7zBnMyn0Lo"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=vySeBU86Ujg:uJHUZuLTCYc:F7zBnMyn0Lo" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=vySeBU86Ujg:uJHUZuLTCYc:gIN9vFwOqvQ"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=vySeBU86Ujg:uJHUZuLTCYc:gIN9vFwOqvQ" border="0"&gt;&lt;/img&gt;&lt;/a&gt;
&lt;/div&gt;</description><author>chipx86@chipx86.com (chipx86)</author><pubDate>Sun, 21 Jun 2009 07:47:36 -0700</pubDate><guid>http://www.chipx86.com/blog/2009/06/20/review-board-10-released/</guid></item><item><title>Review Board: Summer of Code, Roadmap and Future Plans
</title><link>http://www.chipx86.com/blog/2009/04/02/review-board-summer-of-code-roadmap-and-future-plans/</link><description>&lt;p&gt;&lt;b&gt;Summer of Code&lt;/b&gt;&lt;/p&gt;
&lt;p&gt;This year, we (the &lt;a href="http://www.review-board.org/"&gt;Review Board&lt;/a&gt; project) was given the opportunity to participate in Google&amp;#8217;s Summer of Code. We&amp;#8217;ve received some great student proposals so far, and I think we&amp;#8217;ll see exciting work done on Review Board this summer.&lt;/p&gt;
&lt;p&gt;The deadline for Summer of Code is coming up fast (April 3rd, 19:00 UTC). If you&amp;#8217;re interested in working on Review Board and haven&amp;#8217;t yet applied, it&amp;#8217;s not too late, but you&amp;#8217;ll want to hurry. Skim through our &lt;a href="http://www.review-board.org/wiki/Summer_of_Code_Ideas"&gt;ideas&lt;/a&gt; page and, if you find something interesting or have a great idea not listed here, then &lt;a href="http://socghop.appspot.com/"&gt;apply&lt;/a&gt; and tell us your plans. I can say we&amp;#8217;ve received several proposals so far for the installer and admin UI, so unless you feel strongly about either of those, you&amp;#8217;ll increase your chances with other proposals.&lt;/p&gt;
&lt;p&gt;We&amp;#8217;re also offering &lt;a href="http://www.review-board.org/summer-of-code/hosting/"&gt;free Review Board hosting&lt;/a&gt; for open source projects participating in Summer of Code. If you&amp;#8217;re a mentoring organization and would like to give Review Board a try for reviewing and managing student code, go ahead and contact us and we&amp;#8217;ll get you set up.&lt;/p&gt;
&lt;p&gt;&lt;b&gt;Roadmap&lt;/b&gt;&lt;/p&gt;
&lt;p&gt;We&amp;#8217;re finally nearing 1.0. We recently put out our 1.0 beta 2 release and are now in a feature freeze. We&amp;#8217;re working to get some bug, performance and usability fixes in for beta 3, which I&amp;#8217;m shooting for in a few weeks. Then we&amp;#8217;ll branch for 1.0, put out a Release Candidate or two, and then finally release 1.0!&lt;/p&gt;
&lt;p&gt;There&amp;#8217;s a lot of really cool features planned after 1.0, namely &lt;b&gt;extensions&lt;/b&gt; and &lt;b&gt;policy customization&lt;/b&gt;.&lt;/p&gt;
&lt;p&gt;&lt;b&gt;Extensions&lt;/b&gt;&lt;/p&gt;
&lt;p&gt;Our bug tracker is filled with feature requests for all kinds of things, ranging from bug tracker integration, instant messaging, a method for offering bribes for code reviews, and so on. We clearly can&amp;#8217;t put all the requested features in the codebase, so we&amp;#8217;ve decided instead to add support for third-party extensions. Coming soon, developers will be able to write extensions to Review Board in the form of Python modules to extend or alter the functionality of Review Board. The extension framework will allow them to do the following:&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;Access the database using the existing Review Board database models.&lt;/li&gt;
&lt;li&gt;Add new database models for storing data.&lt;/li&gt;
&lt;li&gt;Listen for signals (new review request published, review request submitted, etc.) and act on them.&lt;/li&gt;
&lt;li&gt;Add custom URLs.&lt;/li&gt;
&lt;li&gt;Replace existing URLs, for advanced capabilities such as replacing the diff viewer.&lt;/li&gt;
&lt;li&gt;Add new API handlers.&lt;/li&gt;
&lt;li&gt;Add &amp;#8220;action&amp;#8221; links to existing review requests and reviews.&lt;/li&gt;
&lt;li&gt;Add columns and sidebar entries to the dashboard.&lt;/li&gt;
&lt;li&gt;Add pages to the administration UI.&lt;/li&gt;
&lt;li&gt;Communicate with other extensions.&lt;/li&gt;
&lt;li&gt;Provide a settings page, which stores data in Review Board-provided models (we even auto-generate the settings page for the extension by default).&lt;/li&gt;
&lt;li&gt;And more!&lt;/li&gt;
&lt;/ul&gt;
&lt;p&gt;A lot of this already exists in a private development branch, and it will be one of our primary focuses as soon as 1.0 goes out.&lt;/p&gt;
&lt;p&gt;In time, we&amp;#8217;ll add a new section to the Review Board website where developers can list their extensions for download and for sale. Administrators will be able to browse and search for extensions directly from the administration UI and install them without having to even open a terminal (in most cases).&lt;/p&gt;
&lt;p&gt;We&amp;#8217;re hoping this will solve a lot of in-house integration issues. For example, many companies have custom sandbox architectures, bug trackers, and statistics software which they&amp;#8217;ll now be able to tie in with Review Board.&lt;/p&gt;
&lt;p&gt;&lt;b&gt;Policy Customization&lt;/b&gt;&lt;/p&gt;
&lt;p&gt;We&amp;#8217;ve found that a lot of companies have very specific ways they want to handle policy and access restrictions. For example, many companies want to limit who can see certain parts of a repository (and therefore certain diffs), or want to allow anybody to create review groups, or want to disallow people from joining review groups. Some also want to dictate what constitutes approval for submitting a change.&lt;/p&gt;
&lt;p&gt;We&amp;#8217;re looking into the various requests and attempting to come up with a policy model that is flexible enough to handle these needs. One of the ideas is to provide some basic level of access control on a per-repository, per-path, and per-group basis. We&amp;#8217;d then piggy-back on the extension framework to allow for more specific policy control. The advantage is that developers could write their own policy rules that interface with some part of their company&amp;#8217;s infrastructure.&lt;/p&gt;
&lt;p&gt;If people have any input on this, we&amp;#8217;d love to hear it.&lt;/p&gt;&lt;div class="feedflare"&gt;
&lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=QRspFP8F_60:KQd3NmatfKY:yIl2AUoC8zA"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?d=yIl2AUoC8zA" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=QRspFP8F_60:KQd3NmatfKY:V_sGLiPBpWU"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=QRspFP8F_60:KQd3NmatfKY:V_sGLiPBpWU" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=QRspFP8F_60:KQd3NmatfKY:F7zBnMyn0Lo"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=QRspFP8F_60:KQd3NmatfKY:F7zBnMyn0Lo" border="0"&gt;&lt;/img&gt;&lt;/a&gt; &lt;a href="http://feeds.feedburner.com/~ff/ReviewBoard?a=QRspFP8F_60:KQd3NmatfKY:gIN9vFwOqvQ"&gt;&lt;img src="http://feeds.feedburner.com/~ff/ReviewBoard?i=QRspFP8F_60:KQd3NmatfKY:gIN9vFwOqvQ" border="0"&gt;&lt;/img&gt;&lt;/a&gt;
&lt;/div&gt;</description><author>chipx86@chipx86.com (chipx86)</author><pubDate>Thu, 02 Apr 2009 21:53:39 -0700</pubDate><guid>http://www.chipx86.com/blog/2009/04/02/review-board-summer-of-code-roadmap-and-future-plans/</guid></item></channel></rss>

