<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0"
	xmlns:content="http://purl.org/rss/1.0/modules/content/"
	xmlns:wfw="http://wellformedweb.org/CommentAPI/"
	xmlns:dc="http://purl.org/dc/elements/1.1/"
	xmlns:atom="http://www.w3.org/2005/Atom"
	xmlns:sy="http://purl.org/rss/1.0/modules/syndication/"
	xmlns:slash="http://purl.org/rss/1.0/modules/slash/"
	>

<channel>
	<title>Curia &#187; Seth</title>
	<atom:link href="http://blog.curiasolutions.com/author/admin/feed/" rel="self" type="application/rss+xml" />
	<link>http://blog.curiasolutions.com</link>
	<description>Technology thoughts and ideas</description>
	<lastBuildDate>Wed, 16 Dec 2009 01:38:40 +0000</lastBuildDate>
	<generator>http://wordpress.org/?v=2.8.6</generator>
	<language>en</language>
	<sy:updatePeriod>hourly</sy:updatePeriod>
	<sy:updateFrequency>1</sy:updateFrequency>
			<item>
		<title>Python as a PHP replacement?</title>
		<link>http://blog.curiasolutions.com/2009/11/python-as-a-php-replacement/</link>
		<comments>http://blog.curiasolutions.com/2009/11/python-as-a-php-replacement/#comments</comments>
		<pubDate>Mon, 30 Nov 2009 18:11:50 +0000</pubDate>
		<dc:creator>Seth</dc:creator>
				<category><![CDATA[Python]]></category>
		<category><![CDATA[Web Development]]></category>
		<category><![CDATA[php]]></category>
		<category><![CDATA[programming]]></category>

		<guid isPermaLink="false">http://blog.curiasolutions.com/?p=637</guid>
		<description><![CDATA[I recently sat down to coffee with a new acquaintance of mine who spends much of his time implementing F/OSS projects at non-profit organizations, and who had just stepped into a lead web developer position using PHP. After sharing pleasantries we began trading stories and talking about each of our &#8220;tools of the trade.&#8221; When [...]]]></description>
			<content:encoded><![CDATA[<p>I recently sat down to coffee with a new acquaintance of mine who spends much of his time implementing F/OSS projects at non-profit organizations, and who had just stepped into a lead web developer position using PHP. After sharing pleasantries we began trading stories and talking about each of our &#8220;tools of the trade.&#8221; When I mentioned that I used to do most of my web development in PHP, but have spent the past year or so trying to move as completely as possible to Python, his response was: &#8220;Huh, I have never really thought of Python as a PHP replacement.&#8221;</p>
<p>Now, this guy hasn&#8217;t exactly been living under a rock for the past 10 years—his resume was quite impressive and included projects in a number of different programming languages; But as you can imagine, I was rather surprised by his response, and it made me wonder: Has the Python community really been that bad at promoting the strengths of Python for web development? Or, does the nirvana experienced by switching from a language like PHP to Python just make us so at peace with the world that we forget the hordes of developers still stuck with C-style syntax? Either way, it got me thinking about a few of the reasons why <em>I</em> decided to switch from PHP to Python; and why I not only see Python as an excellent PHP replacement, but am surprised it is such a &#8220;best-kept secret&#8221; for web development.</p>
<p>There are already plenty of pages out there discussing <a href="http://wiki.python.org/moin/PythonVsPhp" target="_blank">Python vs. PHP as a language</a>, so I probably won&#8217;t get too technical here. I also want to try avoid turning this into a &#8220;Python is better than PHP because&#8230;&#8221; rant (for more on that, please see the end of this post), so I will simply share with you a few of the main reasons why <em>I</em> decided to replace PHP with Python as <em>my</em> primary language for web development:</p>
<p><span id="more-637"></span></p>
<h4>» Python is more flexible than PHP</h4>
<p>Python was engineered from the beginning to be a general purpose programming language. PHP on the other hand was written to overcome many of the &#8220;limitations&#8221; of the <a href="http://en.wikipedia.org/wiki/Cgi-bin" target="_blank">cgi-bin</a>. This should say a lot about the &#8220;soul&#8221; of each language, but the conclusion I have come to is:</p>
<ul>
<li>PHP is meant to do web things really well</li>
<li>Python is meant to do most things really well</li>
</ul>
<p>Sure, these days you <em>could</em> write a desktop GUI in PHP, but it would most likely <em>not</em> be the right tool for the job (and you&#8217;d probably get laughed at). On a somewhat related note: many PHP developers I know find themselves switching between PHP for their website tasks, and a variety of other languages for other tasks (Shell Scripting for command-line scripts, Perl for text processing, etc.). As a Python programmer, I can typically stick to <em>one programming language</em> no matter what the task at hand requires (especially when armed with <a href="http://ipython.scipy.org/" target="_blank">iPython</a>).</p>
<p>Now, the obvious argument for PHP at this point is: &#8220;But wait, since PHP was written specifically for the web, shouldn&#8217;t that actually give it <em>more</em> credibility as the right tool for the web job?&#8221; Well, 5 or 10 years ago I might have agreed with you, but web development has changed a lot in the past few years (not to mention the introduction of Python&#8217;s <a href="http://en.wikipedia.org/wiki/Web_Server_Gateway_Interface" target="_blank">WSGI spec</a>).</p>
<p>The internet as a platform has grown and mutated drastically and now requires much more than &#8220;a server-side extension to HTML.&#8221; We are now in an age of full-blown internet <em>applications</em>, which require a true &#8220;application-level&#8221; language. In my opinion, PHP has only recently begun to move in that direction with the v5.3 release (as well as updates expected in PHP 6). That alone should give you pause when pondering PHP&#8217;s ability to &#8220;keep up&#8221; with the growth of the web.</p>
<p>In short, the internet continues to transform into an arena that is much less of a website platform, and much more of an application platform. To me, this makes a general purpose language such as Python the obvious choice for &#8220;future-proof&#8221; internet development.</p>
<h4>» Python is typically faster than PHP, <span style="text-decoration: line-through;">even</span> especially on the web</h4>
<p>You might find PHP beating out Python in a simple &#8220;hello world&#8221;-type test, but &#8220;real-world&#8221; applications will typically run much faster in Python. In a Web 2.0 world where everyone has broadband, speed is now an important factor in keeping your visitors happy.</p>
<p>In addition to &#8220;raw, under-the-hood&#8221; speed, Python&#8217;s language philosophy means it will typically <a href="http://groups.google.com/group/comp.lang.python/msg/83907b65bdee30f6" target="_blank">scale better than PHP</a> (although this can be argued a million different ways, since scalability usually involves much more than the language).</p>
<p>I&#8217;ve already done <a href="http://blog.curiasolutions.com/2009/09/the-great-web-development-shootout/" target="_blank">benchmarks</a> comparing a few popular web frameworks, but if you&#8217;re still curious I&#8217;d suggest you <a href="http://www.google.com/search?q=is+python+or+php+faster%3F" target="_blank">dig around with Google</a> for a while.</p>
<h4>» Python is easier to read and maintain than PHP</h4>
<p>Any veteran programmer will tell you that code <em>readability</em> plays a huge factor in code <em>maintainability</em>. With that in mind, Python was designed with some pretty strict syntax rules in regards to code-blocks and whitespace. While this may initially frustrate programmers who are used to a looser &#8220;bracket-encapsulated&#8221; language, it actually forces you to write more readable (and therefore, more maintainable) code.</p>
<p>Other than readability, Python&#8217;s clean syntax also means:</p>
<ul>
<li>You can write more code with less errors (because they&#8217;re so easy to spot). This often comes as a pleasant surprise to programmers migrating from C-style languages (take <a href="http://www.python.org/about/success/esr/" target="_blank">Eric Raymond</a> for example).</li>
<li>You end up typing less — No more semicolons or brackets, and fewer parentheses.</li>
<li>Your eyes don&#8217;t have to move around as much to figure out what&#8217;s going on.</li>
<li>Your first guess about what&#8217;s happening in a code-block is usually correct.</li>
</ul>
<p>When you have time to kill, I&#8217;d recommend a stroll through <a href="http://rosettacode.org/wiki/Main_Page" target="_blank">Rosetta Code</a> to compare how different languages &#8220;look&#8221; when tackling the same problem. Unless you just love brackets, I think you&#8217;ll find the non C-style syntaxes much easier to read.</p>
<h4>» Python is more consistent than PHP</h4>
<p>Continuing somewhat in the vein of readability is the topic of consistency. Since PHP&#8217;s lack of consistency has already been studied exhaustively by the Perl folks, I will spare you my rant and suggest you <a href="http://tnx.nl/php.html" target="_blank">read theirs</a>.</p>
<p>As a side note on this topic: I often hear PHP programmers talk about how awesome PHP&#8217;s documentation is in that you can open a web browser and just type &#8220;php.net/some_function&#8221; to get help. Well, documentation recall usually isn&#8217;t that big of a deal for me (because I have a somewhat photographic memory), but I will say that after switching to Python I found myself turning to its documentation <em>significantly</em> less than I had with PHP, simply because I didn&#8217;t have to keep reminding myself whether that infrequently used function was named some_function or somefunction. With Python, the function/method I need is usually named <em>exactly</em> what I expect it to be named, and returns <em>exactly</em> what I expect it to return.</p>
<p>Oh, and don&#8217;t even get me started on PHP&#8217;s recent <a href="http://blog.fedecarg.com/2008/10/28/php-namespaces-controversy/" target="_blank">namespace</a> <a href="http://michaelkimsal.com/blog/disappointed-with-php-namespace-seperator-decision/" target="_blank">decision</a>.</p>
<p>&nbsp;</p>
<h4 style="text-align: center;">» However, PHP still excels in a few areas «</h4>
<p>Having said all of that, there are a few areas where I believe PHP may still shine brighter than Python:</p>
<h4>» PHP hosting support <span style="text-decoration: line-through;">is</span> may be more prevalent than Python</h4>
<p>It&#8217;s practically unheard of to find yourself on a web host that doesn&#8217;t support PHP out of the box. However, this is becoming less and less of an issue since most Python web applications can now be deployed using tools such as FastCGI or mod_proxy (as opposed to the popular <a href="http://code.google.com/p/modwsgi/" target="_blank">mod_wsgi</a> setup, which often requires root access to configure). These days any &#8220;reputable&#8221; web host is probably going to support some sort of strategy for deploying Python web applications. Of course, this only really applies to shared hosting environments without root access. If you&#8217;ve got a VPS or dedicated server you&#8217;re fine.</p>
<h4>» PHP is arguably &#8220;more seasoned&#8221; on the web than Python</h4>
<p>The communities behind projects like Drupal and WordPress really stand out to me as examples of the maturity of PHP on the web. Although similar Python web projects have been slowly emerging (and <a href="http://plone.org/" target="_blank">Plone</a> has been around for years, if that&#8217;s your cup of tea), they don&#8217;t really compete yet in my opinion.</p>
<p>Of course, I am only talking about specific web projects here. Language wise, Python has been around longer than PHP. Web 2.0 framework wise, Python&#8217;s offerings have been around about as long as PHP&#8217;s (pretty much every language had its own answer to Ruby on Rails in 2005).</p>
<h4>» PHP <em>seems</em> easier than Python at first (and might be for simple deployments)</h4>
<p>There&#8217;s something magical about throwing together a few quick index.php/about.php/contact.php files and uploading them to the httpdocs directory of your webserver and it &#8220;just working.&#8221; A lot of simple websites have no need for acronyms like MVC, ORM, and DRY, and in those cases PHP is an excellent &#8220;server-side extension&#8221; to HTML. However, if and when you begin to scale, you&#8217;ll either find yourself switching to a full-blown framework, or throwing one together to help facilitate your growth. At that point, you&#8217;ll probably begin discovering the &#8220;dark corners&#8221; of PHP, and may find yourself wishing you had &#8220;done it right&#8221; the first time.</p>
<p>&nbsp;</p>
<h4 style="text-align: center;">» So, which is better? «</h4>
<p>Remember, the point of this post is to solidify the fact that Python can be an excellent PHP <em>replacement</em>. When it comes to programming languages, &#8220;better&#8221; is often more a case of personal preference than anything else. However, to <em>me</em> the points discussed above make Python the better choice when it comes to what <em>I</em> like in a web development language.</p>
<p>So, is Python a better web development language than PHP?<br />
Well, does Nike make a better shoe than Adidas?</p>
<p>The most honest answer is probably: &#8220;It depends.&#8221; Python definitely has some significant advantages over PHP in certain circumstances, but the same could probably be said of PHP in other circumstances (although limited to the web I imagine).</p>
<p>&nbsp;</p>
<p>I hope this post has encouraged you PHP programmers out there to take another look at Python. It may take some getting used to (Python&#8217;s way of doing things is inherently quite different than PHP&#8217;s), but you will probably find yourself thanking me later.</p>
]]></content:encoded>
			<wfw:commentRss>http://blog.curiasolutions.com/2009/11/python-as-a-php-replacement/feed/</wfw:commentRss>
		<slash:comments>5</slash:comments>
		</item>
		<item>
		<title>Dead-simple pagination wizardry with TurboGears 2</title>
		<link>http://blog.curiasolutions.com/2009/11/dead-simple-pagination-wizardry-with-turbogears-2/</link>
		<comments>http://blog.curiasolutions.com/2009/11/dead-simple-pagination-wizardry-with-turbogears-2/#comments</comments>
		<pubDate>Wed, 04 Nov 2009 01:00:21 +0000</pubDate>
		<dc:creator>Seth</dc:creator>
				<category><![CDATA[Web Development]]></category>
		<category><![CDATA[turbogears]]></category>

		<guid isPermaLink="false">http://blog.curiasolutions.com/?p=556</guid>
		<description><![CDATA[I recently had to implement pagination in a TG 2 project that I&#8217;m almost finished with. Although I had previously glanced at the paginate webhelper module, I first turned to the updated TurboGears documentation to see what was &#8220;officially&#8221; written up about this subject.
A lot of work has been going into the TG docs in [...]]]></description>
			<content:encoded><![CDATA[<p>I recently had to implement pagination in a TG 2 project that I&#8217;m almost finished with. Although I had previously glanced at the <a href="http://turbogears.org/2.1/docs/modules/thirdparty/webhelpers_paginate.html" target="_blank"><em>paginate</em></a> webhelper module, I first turned to the <a href="http://turbogears.org/2.1/docs/index.html" target="_blank">updated TurboGears documentation</a> to see what was &#8220;officially&#8221; written up about this subject.</p>
<p>A lot of work has been going into the TG docs in preparation for the upcoming v2.1 launch, and one little gem that was recently added was the <a href="http://turbogears.org/2.1/docs/main/Pagination/index.html" target="_blank">Pagination Quickstart</a> (thanks Lukas). While this sent me down the right path towards pagination perfection, there were a few things that came up that I felt like deserved further explanation:</p>
<ol>
<li>The <a href="http://turbogears.org/2.1/docs/modules/tgdecorators.html#tg.decorators.paginate" target="_blank">@paginate() decorator</a> is an extremely simple wrapper for the <em>paginate</em> webhelper module.</li>
<li>When paginating SQLAlchemy queries, <em>paginate</em> runs a duplicate <em>count()</em> query to figure out its current position in the data collection. While the magic of this is nice, the possible performance issues with duplicating complex SQL queries on a high-traffic website could be undesirable.</li>
<li>It&#8217;s common to have controller methods dynamically switch to a &#8220;feed template&#8221; when doing AJAX pagination. Don&#8217;t worry, <em>paginate</em> + <em>override_template</em> makes this a piece of cake!</li>
</ol>
<p>&nbsp;</p>
<blockquote><p>The @paginate() decorator makes pagination a breeze</p></blockquote>
<p>The @paginate decorator is a wrapper for the paginate module that can be added to your controller methods. Simply specify your data collection&#8217;s name as the first argument of the decorator, and pagination is &#8220;automagically&#8221; setup for you.</p>
<p>An example controller method with pagination would look like this:</p>

<div class="wp_syntax"><div class="code"><pre class="python" style="font-family:monospace;"><span style="color: #808080; font-style: italic;"># dead-simple pagination in TurboGears 2</span>
&nbsp;
@expose<span style="color: black;">&#40;</span><span style="color: #483d8b;">'myproject.templates.posts'</span><span style="color: black;">&#41;</span>
@paginate<span style="color: black;">&#40;</span><span style="color: #483d8b;">&quot;posts&quot;</span><span style="color: black;">&#41;</span>
<span style="color: #ff7700;font-weight:bold;">def</span> posts<span style="color: black;">&#40;</span><span style="color: #008000;">self</span><span style="color: black;">&#41;</span>:
    posts = DBSession.<span style="color: black;">query</span><span style="color: black;">&#40;</span>Posts<span style="color: black;">&#41;</span>
    <span style="color: #ff7700;font-weight:bold;">return</span> <span style="color: #008000;">dict</span><span style="color: black;">&#40;</span>posts=posts<span style="color: black;">&#41;</span></pre></div></div>

<p>Really? That was easy!</p>
<p>The documentation on this decorator seems quite clear to me, so instead of repeating its content here I will simply give you a <a href="http://turbogears.org/2.1/docs/modules/tgdecorators.html#tg.decorators.paginate" target="_blank">link for further reading</a>.</p>
<p><span id="more-556"></span>&nbsp;</p>
<blockquote><p>SQLAlchemy pagination performance wizardry</p></blockquote>
<p>The main issue here is that the paginate module has to figure out its current position in the data collection in order to be able to spit out the right &#8220;chunk&#8221; of data (as well as to provide you with fancy stuff like &#8220;page 3 of 6&#8243;). So, by default when you give it an SQLAlchemy query object, it runs a count() query to figure this out. This results in two practically identical SQL queries (one to get the total count, and the other to return your actual collection of data). While this may not be an issue on simple queries, duplicating complex queries on a high-traffic website could be disastrous.</p>
<p>Remember, magic can be unpredictable, so before we dive into this topic there are a few things you should take into consideration:</p>
<ul>
<li>This technique is not compatible with the @paginate() decorator due to the fact that the current page arg isn&#8217;t passed into the controller method when using the decorator, so be ready to get your hands dirty with the full-blown <em>paginate</em> module.</li>
<li>The SQLAlchemy all() query method results in immediate query execution. This may have undesirable side-effects.</li>
<li>I&#8217;m sure there is a cleaner/sexier way to do this, but for the sake of time and in an effort to be verbose this is what I have come up with so far. Please leave a helpful comment below if your magic is cleaner than mine.</li>
</ul>
<p>Ok, so now that we are familiar with the problem, what is a good solution? Well, a rough hack to work around this issue is to use SQLAlchemy&#8217;s limit() and offset() query methods combined with the all() method to get SQLAlchemy to return a list instead of a query object. This puts <em>paginate</em> to work on the list instead of on the database, avoiding the extra count() query. However, as tricky as this may be, there are still problems:</p>
<ol>
<li>In order for paginate to calculate a correct current position based on the &#8220;page&#8221; it thinks it&#8217;s on, you&#8217;ve got to pad the <strong>beginning</strong> of the list you send to the paginate.Page() method with enough items to accurately reflect the &#8220;current&#8221; size of the data collection.</li>
<li>In order for paginate to know there are more records waiting to be loaded, you&#8217;ve got to pad the <strong>end</strong> of the list you send to the paginate.Page() method with at least one more item than each page&#8217;s specified item limit.</li>
</ol>
<p>This may seem confusing at first, but an example controller method should help clear things up for you:</p>

<div class="wp_syntax"><div class="code"><pre class="python" style="font-family:monospace;"><span style="color: #808080; font-style: italic;"># import the paginate webhelper</span>
<span style="color: #ff7700;font-weight:bold;">from</span> webhelpers <span style="color: #ff7700;font-weight:bold;">import</span> paginate
<span style="color: #808080; font-style: italic;"># specify the number of items you want per page</span>
items_per_page = <span style="color: #ff4500;">20</span>
&nbsp;
@expose<span style="color: black;">&#40;</span><span style="color: #483d8b;">'myproject.templates.posts'</span><span style="color: black;">&#41;</span>
<span style="color: #ff7700;font-weight:bold;">def</span> posts<span style="color: black;">&#40;</span><span style="color: #008000;">self</span>, page=<span style="color: #ff4500;">1</span><span style="color: black;">&#41;</span>:
    <span style="color: #808080; font-style: italic;"># specify the limit, plus 1 so the Page() method can tell if extra data is available</span>
    limit = items_per_page + <span style="color: #ff4500;">1</span>
    <span style="color: #808080; font-style: italic;"># specify the offset to start with</span>
    offset = <span style="color: black;">&#40;</span><span style="color: #008000;">int</span><span style="color: black;">&#40;</span>page<span style="color: black;">&#41;</span> - <span style="color: #ff4500;">1</span><span style="color: black;">&#41;</span> <span style="color: #66cc66;">*</span> items_per_page
&nbsp;
    <span style="color: #808080; font-style: italic;"># if the offset is greater than 0, we have to pad the beginning of the list</span>
    <span style="color: #ff7700;font-weight:bold;">if</span> offset: posts = <span style="color: black;">&#91;</span>x <span style="color: #ff7700;font-weight:bold;">for</span> x <span style="color: #ff7700;font-weight:bold;">in</span> <span style="color: #008000;">xrange</span><span style="color: black;">&#40;</span>offset<span style="color: black;">&#41;</span><span style="color: black;">&#93;</span>
    <span style="color: #808080; font-style: italic;"># otherwise we're starting at the beginning (with an empty list)</span>
    <span style="color: #ff7700;font-weight:bold;">else</span>: posts = <span style="color: #008000;">list</span><span style="color: black;">&#40;</span><span style="color: black;">&#41;</span>
&nbsp;
    <span style="color: #808080; font-style: italic;"># load the DB data into the list</span>
    posts.<span style="color: black;">extend</span><span style="color: black;">&#40;</span>DBSession.<span style="color: black;">query</span><span style="color: black;">&#40;</span>Posts<span style="color: black;">&#41;</span>.<span style="color: black;">limit</span><span style="color: black;">&#40;</span>limit<span style="color: black;">&#41;</span>.<span style="color: black;">offset</span><span style="color: black;">&#40;</span>offset<span style="color: black;">&#41;</span>.<span style="color: #008000;">all</span><span style="color: black;">&#40;</span><span style="color: black;">&#41;</span><span style="color: black;">&#41;</span>
&nbsp;
    <span style="color: #808080; font-style: italic;"># setup the pagination object</span>
    pagination = paginate.<span style="color: black;">Page</span><span style="color: black;">&#40;</span>posts, page, items_per_page=items_per_page<span style="color: black;">&#41;</span>
    <span style="color: #808080; font-style: italic;"># either use tmpl_context or return the object in the return dict</span>
    tmpl_context.<span style="color: black;">pagination</span> = pagination
&nbsp;
    <span style="color: #ff7700;font-weight:bold;">return</span> <span style="color: #008000;">dict</span><span style="color: black;">&#40;</span>posts=pagination.<span style="color: black;">items</span><span style="color: black;">&#41;</span></pre></div></div>

<p>Voilà! Now that you&#8217;re dealing with a list, pagination should work as expected without the extra &#8220;count()&#8221; query.</p>
<p>&nbsp;</p>
<blockquote><p>Use override_template() for dynamic template rendering</p></blockquote>
<p>AJAX pagination is a common practice and often provides a richer user experience than &#8220;static&#8221; pagination. Fortunately, AJAX pagination is almost as easy to setup as normal pagination, and only requires <a href="http://turbogears.org/2.1/docs/modules/thirdparty/webhelpers_paginate.html#can-i-use-ajax-ajah" target="_blank">a few extra tweaks</a> to an already simple configuration.</p>
<p>The only &#8220;tricky&#8221; thing here is if you&#8217;re accessing the same controller method for partial data as well as the full page. In such a case, you&#8217;ll need to have the controller method render two separate templates depending on if you&#8217;re asking for the &#8220;full&#8221; page, or just the listing of data. Don&#8217;t worry though, <em>paginate</em> automatically provides you with a &#8220;partial&#8221; arg that you can check against, and override_template() allows you to override whatever template you have defined in the @expose() decorator.</p>
<p>Modifying the previous example&#8217;s controller method for this type of functionality looks like this:</p>

<div class="wp_syntax"><div class="code"><pre class="python" style="font-family:monospace;"><span style="color: #808080; font-style: italic;"># add the override_template import</span>
<span style="color: #ff7700;font-weight:bold;">from</span> tg <span style="color: #ff7700;font-weight:bold;">import</span> override_template
&nbsp;
<span style="color: #808080; font-style: italic;"># add the &quot;partial&quot; arg to the controller method</span>
<span style="color: #ff7700;font-weight:bold;">def</span> posts<span style="color: black;">&#40;</span><span style="color: #008000;">self</span>, page=<span style="color: #ff4500;">1</span>, partial=<span style="color: #ff4500;">0</span><span style="color: black;">&#41;</span>:
    <span style="color: #808080; font-style: italic;"># ...snip...</span>
&nbsp;
    <span style="color: #ff7700;font-weight:bold;">if</span> partial:
        override_template<span style="color: black;">&#40;</span><span style="color: #008000;">self</span>.<span style="color: black;">posts</span>, <span style="color: #483d8b;">'mako:myproject.templates.post_feed'</span><span style="color: black;">&#41;</span>
&nbsp;
    <span style="color: #ff7700;font-weight:bold;">return</span> <span style="color: #008000;">dict</span><span style="color: black;">&#40;</span>posts=pagination.<span style="color: black;">items</span><span style="color: black;">&#41;</span></pre></div></div>

<p>More information on override_template() can be found in <a href="http://turbogears.org/2.1/docs/modules/tgdecorators.html#tg.decorators.override_template" target="_blank">the docs</a>.</p>
<p>&nbsp;</p>
<p>I hope this tutorial has been helpful. I will be submitting a TG docs pull request soon, so please leave comments if you have any suggestions or feedback.</p>
]]></content:encoded>
			<wfw:commentRss>http://blog.curiasolutions.com/2009/11/dead-simple-pagination-wizardry-with-turbogears-2/feed/</wfw:commentRss>
		<slash:comments>4</slash:comments>
		</item>
		<item>
		<title>Reorganizing data with list &amp; dict comprehensions</title>
		<link>http://blog.curiasolutions.com/2009/10/reorganizing-data-with-list-dict-comprehensions/</link>
		<comments>http://blog.curiasolutions.com/2009/10/reorganizing-data-with-list-dict-comprehensions/#comments</comments>
		<pubDate>Sat, 31 Oct 2009 03:58:59 +0000</pubDate>
		<dc:creator>Seth</dc:creator>
				<category><![CDATA[Python]]></category>
		<category><![CDATA[php]]></category>
		<category><![CDATA[programming]]></category>

		<guid isPermaLink="false">http://blog.curiasolutions.com/?p=538</guid>
		<description><![CDATA[While writing scripts, I frequently run into the issue of needing to re-arrange sets of data into a more &#8220;process friendly&#8221; format. A common issue I encounter is needing to turn a list (array) into a dictionary (associative array) or vice versa. More often than not, I find myself needing to be able to access [...]]]></description>
			<content:encoded><![CDATA[<p>While writing scripts, I frequently run into the issue of needing to re-arrange sets of data into a more &#8220;process friendly&#8221; format. A common issue I encounter is needing to turn a list (array) into a dictionary (associative array) or vice versa. More often than not, I find myself needing to be able to access list elements by a key, but since they aren&#8217;t setup in a dictionary I have to pull out a looping technique to reorganize the data for this to be possible.</p>
<p>Take the following set of data for example:</p>

<div class="wp_syntax"><div class="code"><pre class="python" style="font-family:monospace;"><span style="color: black;">&#91;</span><span style="color: black;">&#91;</span><span style="color: #ff4500;">1</span>, <span style="color: #483d8b;">'John Smith'</span>, <span style="color: #483d8b;">'admin'</span><span style="color: black;">&#93;</span>,
 <span style="color: black;">&#91;</span><span style="color: #ff4500;">2</span>, <span style="color: #483d8b;">'Jane Doe'</span>, <span style="color: #483d8b;">'superuser'</span><span style="color: black;">&#93;</span>,
 <span style="color: black;">&#91;</span><span style="color: #ff4500;">3</span>, <span style="color: #483d8b;">'Sam Jones'</span>, <span style="color: #483d8b;">'user'</span><span style="color: black;">&#93;</span><span style="color: black;">&#93;</span></pre></div></div>

<p>What we have here is a few rows of user data. In this example, the data is in a Python list (which in PHP would be an array).</p>
<p>In PHP, this would look something like (using print_r):</p>

<div class="wp_syntax"><div class="code"><pre class="php" style="font-family:monospace;"><span style="color: #990000;">Array</span> <span style="color: #009900;">&#40;</span>
    <span style="color: #009900;">&#91;</span><span style="color: #cc66cc;">0</span><span style="color: #009900;">&#93;</span> <span style="color: #339933;">=&gt;</span> <span style="color: #990000;">Array</span> <span style="color: #009900;">&#40;</span>
        <span style="color: #009900;">&#91;</span><span style="color: #cc66cc;">0</span><span style="color: #009900;">&#93;</span> <span style="color: #339933;">=&gt;</span> <span style="color: #cc66cc;">1</span>
        <span style="color: #009900;">&#91;</span><span style="color: #cc66cc;">1</span><span style="color: #009900;">&#93;</span> <span style="color: #339933;">=&gt;</span> John Smith
        <span style="color: #009900;">&#91;</span><span style="color: #cc66cc;">2</span><span style="color: #009900;">&#93;</span> <span style="color: #339933;">=&gt;</span> admin
    <span style="color: #009900;">&#41;</span>
    <span style="color: #009900;">&#91;</span><span style="color: #cc66cc;">1</span><span style="color: #009900;">&#93;</span> <span style="color: #339933;">=&gt;</span> <span style="color: #990000;">Array</span> <span style="color: #009900;">&#40;</span>
        <span style="color: #009900;">&#91;</span><span style="color: #cc66cc;">0</span><span style="color: #009900;">&#93;</span> <span style="color: #339933;">=&gt;</span> <span style="color: #cc66cc;">2</span>
        <span style="color: #009900;">&#91;</span><span style="color: #cc66cc;">1</span><span style="color: #009900;">&#93;</span> <span style="color: #339933;">=&gt;</span> Jane Doe
<span style="color: #666666; font-style: italic;"># (...etc...)</span></pre></div></div>

<p>So, what if I found myself writing some code that needed to be able to access each record by its first value, which in this case would be the user_id? <span id="more-538"></span></p>
<p>Well, in PHP the easiest way to make this happen would be a good-old-fashioned <em>foreach</em> loop:</p>

<div class="wp_syntax"><div class="code"><pre class="php" style="font-family:monospace;"><span style="color: #b1b100;">foreach</span> <span style="color: #009900;">&#40;</span><span style="color: #000088;">$row</span> <span style="color: #b1b100;">as</span> <span style="color: #000088;">$val</span><span style="color: #009900;">&#41;</span> <span style="color: #009900;">&#123;</span>
    <span style="color: #000088;">$out</span><span style="color: #009900;">&#91;</span><span style="color: #000088;">$val</span><span style="color: #009900;">&#91;</span><span style="color: #cc66cc;">0</span><span style="color: #009900;">&#93;</span><span style="color: #009900;">&#93;</span> <span style="color: #339933;">=</span> <span style="color: #000088;">$val</span><span style="color: #339933;">;</span>
<span style="color: #009900;">&#125;</span></pre></div></div>

<p>Not very &#8220;sexy&#8221; for something I find myself having to more often than I would like, but it works nonetheless.</p>
<p>Now, here&#8217;s where the beauty of Python kicks in. Python has a built-in feature called &#8220;comprehension&#8221; which allows you to modify and/or convert sets of data quickly and easily.</p>
<p>Remember what our original data set looks like? Here&#8217;s a refresher:</p>

<div class="wp_syntax"><div class="code"><pre class="python" style="font-family:monospace;"><span style="color: #66cc66;">&gt;&gt;&gt;</span> <span style="color: #ff7700;font-weight:bold;">print</span> row
<span style="color: black;">&#91;</span><span style="color: black;">&#91;</span><span style="color: #ff4500;">1</span>, <span style="color: #483d8b;">'John Smith'</span>, <span style="color: #483d8b;">'admin'</span><span style="color: black;">&#93;</span>,
 <span style="color: black;">&#91;</span><span style="color: #ff4500;">2</span>, <span style="color: #483d8b;">'Jane Doe'</span>, <span style="color: #483d8b;">'superuser'</span><span style="color: black;">&#93;</span>,
 <span style="color: black;">&#91;</span><span style="color: #ff4500;">3</span>, <span style="color: #483d8b;">'Sam Jones'</span>, <span style="color: #483d8b;">'user'</span><span style="color: black;">&#93;</span><span style="color: black;">&#93;</span></pre></div></div>

<p>Now, in Python all we need to do is use the dictionary comprehension:</p>

<div class="wp_syntax"><div class="code"><pre class="python" style="font-family:monospace;"><span style="color: #66cc66;">&gt;&gt;&gt;</span> out = <span style="color: black;">&#123;</span>val<span style="color: black;">&#91;</span><span style="color: #ff4500;">0</span><span style="color: black;">&#93;</span>: val <span style="color: #ff7700;font-weight:bold;">for</span> val <span style="color: #ff7700;font-weight:bold;">in</span> row<span style="color: black;">&#125;</span> <span style="color: #808080; font-style: italic;"># Python 3.X</span>
<span style="color: #808080; font-style: italic;"># (or, for Python 2.X)</span>
<span style="color: #66cc66;">&gt;&gt;&gt;</span> out = <span style="color: #008000;">dict</span><span style="color: black;">&#40;</span><span style="color: black;">&#40;</span>val<span style="color: black;">&#91;</span><span style="color: #ff4500;">0</span><span style="color: black;">&#93;</span>, val<span style="color: black;">&#41;</span> <span style="color: #ff7700;font-weight:bold;">for</span> val <span style="color: #ff7700;font-weight:bold;">in</span> row<span style="color: black;">&#41;</span> <span style="color: #808080; font-style: italic;"># Python 2.4+</span>
&nbsp;
<span style="color: #66cc66;">&gt;&gt;&gt;</span> <span style="color: #ff7700;font-weight:bold;">print</span> out
<span style="color: black;">&#123;</span><span style="color: #ff4500;">1</span>: <span style="color: black;">&#91;</span><span style="color: #ff4500;">1</span>, <span style="color: #483d8b;">'John Smith'</span>, <span style="color: #483d8b;">'admin'</span><span style="color: black;">&#93;</span>,
 <span style="color: #ff4500;">2</span>: <span style="color: black;">&#91;</span><span style="color: #ff4500;">2</span>, <span style="color: #483d8b;">'Jane Doe'</span>, <span style="color: #483d8b;">'superuser'</span><span style="color: black;">&#93;</span>,
 <span style="color: #ff4500;">3</span>: <span style="color: black;">&#91;</span><span style="color: #ff4500;">3</span>, <span style="color: #483d8b;">'Sam Jones'</span>, <span style="color: #483d8b;">'user'</span><span style="color: black;">&#93;</span><span style="color: black;">&#125;</span></pre></div></div>

<p>See how easy that was?</p>
<p>And what if I had a dictionary in the above format, and needed to turn it into the type of list I had previously? Well, I could simply use the list comprehension to re-generate this set of data as a list:</p>

<div class="wp_syntax"><div class="code"><pre class="python" style="font-family:monospace;"><span style="color: #66cc66;">&gt;&gt;&gt;</span> <span style="color: #ff7700;font-weight:bold;">print</span> <span style="color: black;">&#91;</span>val <span style="color: #ff7700;font-weight:bold;">for</span> val <span style="color: #ff7700;font-weight:bold;">in</span> out.<span style="color: black;">values</span><span style="color: black;">&#40;</span><span style="color: black;">&#41;</span><span style="color: black;">&#93;</span>
<span style="color: black;">&#91;</span><span style="color: black;">&#91;</span><span style="color: #ff4500;">1</span>, <span style="color: #483d8b;">'John Smith'</span>, <span style="color: #483d8b;">'admin'</span><span style="color: black;">&#93;</span>,
 <span style="color: black;">&#91;</span><span style="color: #ff4500;">2</span>, <span style="color: #483d8b;">'Jane Doe'</span>, <span style="color: #483d8b;">'superuser'</span><span style="color: black;">&#93;</span>,
 <span style="color: black;">&#91;</span><span style="color: #ff4500;">3</span>, <span style="color: #483d8b;">'Sam Jones'</span>, <span style="color: #483d8b;">'user'</span><span style="color: black;">&#93;</span><span style="color: black;">&#93;</span></pre></div></div>

<p>This is an extremely simple comprehension example. For a deeper look, check out Wikipedia&#8217;s <a href="http://en.wikipedia.org/wiki/Python_syntax_and_semantics#Functional_programming" target="_blank">Python syntax page</a> or the <a href="http://docs.python.org/3.1/tutorial/datastructures.html" target="_blank">official Python documentation</a>. </p>
]]></content:encoded>
			<wfw:commentRss>http://blog.curiasolutions.com/2009/10/reorganizing-data-with-list-dict-comprehensions/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>The great web technology shootout &#8211; Round 3: Better, Faster, and Shinier</title>
		<link>http://blog.curiasolutions.com/2009/10/the-great-web-technology-shootout-round-3-better-faster-and-shinier/</link>
		<comments>http://blog.curiasolutions.com/2009/10/the-great-web-technology-shootout-round-3-better-faster-and-shinier/#comments</comments>
		<pubDate>Mon, 05 Oct 2009 18:42:52 +0000</pubDate>
		<dc:creator>Seth</dc:creator>
				<category><![CDATA[F/OSS]]></category>
		<category><![CDATA[Web Development]]></category>
		<category><![CDATA[django]]></category>
		<category><![CDATA[performance]]></category>
		<category><![CDATA[pylons]]></category>
		<category><![CDATA[rails]]></category>
		<category><![CDATA[turbogears]]></category>

		<guid isPermaLink="false">http://blog.curiasolutions.com/?p=409</guid>
		<description><![CDATA[[Note: This post is the continuation of a series. Please read Round 1 and Round 2 first if you are just now joining us.]
[Update 10/6: Slight updates to the Pylons test after some input from Ben. Added a couple new charts for you visual folks.]
As I briefly mentioned in Round 1, this whole thing came [...]]]></description>
			<content:encoded><![CDATA[<p>[Note: This post is the continuation of a series. Please read <a href="http://blog.curiasolutions.com/?p=172">Round 1</a> and <a href="http://blog.curiasolutions.com/?p=278">Round 2</a> first if you are just now joining us.]</p>
<p><strong>[Update 10/6: Slight updates to the Pylons test after some input from Ben. Added a couple new charts for you visual folks.]</strong></p>
<p>As I briefly mentioned in <a href="http://blog.curiasolutions.com/?p=172">Round 1</a>, this whole thing came about as an experiment to satisfy my own curiosity. Unfortunately, I wasn&#8217;t expecting these posts to draw the amount of attention they have been getting, and several people informed me of a few &#8220;issues&#8221; with the first round. Since my initial approach to this topic was somewhat casual, I didn&#8217;t really take the time to perform each test in a &#8220;proper scientific fashion.&#8221; Although this was clearly stated in the introduction to round one, it unfortunately resulted in performance estimations that were somewhat less than accurate.</p>
<p>After input from various people much smarter than myself, I quickly went to work tweaking my test environment and building &#8220;proper&#8221; test apps. In the midst of this, a conversation about PHP accelerators prompted me to put PHP under the spotlight, which brought about <a href="http://blog.curiasolutions.com/?p=278">Round 2</a> as an interim round. This gave me a chance to demonstrate the necessity of PHP acceleration, and only continued to solidify my opinion of PHP as an inferior web development language (remember, I just said <em>my opinion</em>).</p>
<p>Which brings us to Round 3. A lot of work has gone into &#8220;doing it right&#8221; this time, so I am fairly confident that these results are a much more accurate representation of each test subject&#8217;s performance estimations. Remember, benchmark test code typically has no real-world value, so &#8220;performance estimations&#8221; are about all I can promise here. <strong>Your mileage <em>will</em> vary.</strong> As a wise person once said:</p>
<blockquote><p>&#8220;All this benchmarking is doing is proving what we already know: More code takes longer to execute.&#8221; &#8211; Ben Bangert (dev lead of Pylons)</p></blockquote>
<p><span id="more-409"></span></p>
<p>&nbsp;</p>
<p>What you should know about Round 3:</p>
<ul>
<li>The hardware/software platform is the same as in <a href="http://blog.curiasolutions.com/?p=172">Round 1</a>.</li>
<li>Python v2.5.4 was used for all the Python tests. Ruby v1.8.6 was used for the Rails test.</li>
<li>Apache v2.2.3 was used, with <a href="http://code.google.com/p/modwsgi/" target="_blank">mod_wsgi</a> v2.5 for the Python tests and <a href="http://www.modrails.com/" target="_blank">Phusion Passenger</a> v2.2.5 for the Rails test.</li>
<li>Note that while the mod_wsgi tests in Round 1 were using &#8220;daemon mode&#8221;, the mod_wsgi tests here are using &#8220;embedded mode&#8221; which is often faster (although <a href="http://blog.dscpl.com.au/2009/03/load-spikes-and-excessive-memory-usage.html" target="_blank">not necessarily the recommended configuration)</a>.
<li>The Python component versions used here were: SQLAlchemy v0.5.6, Genshi v0.5.1, Mako v0.2.5, and Jinja 2.2.1.</li>
<li>ApacheBench was run with -n 10000 -c 10.</li>
<li>Each test was run several times to make sure that there were no anomalies, with the &#8220;optimum average&#8221; chosen as the numbers represented here.</li>
<li>In an attempt to try to make these tests more scientific, I am now including the source code for each test.</li>
</ul>
<p>&nbsp;</p>
<blockquote><h4>Plain WSGI – <a href="http://blog.curiasolutions.com/wp-content/uploads/2009/10/wsgi_test_src.zip">wsgi_test_src.zip</a></h4>
</blockquote>
<p>Let&#8217;s start with a basic &#8220;hello world&#8221; WSGI test to set the baseline:</p>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">The <span style="color: #ff0000;">'hello world'</span> test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">WSGIServer</span><span style="color: #66cc66;">/</span><span style="color: #cc66cc;">0.1</span>
Document Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">12</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">1532.76</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
Apache<span style="color: #66cc66;">/</span>2<span style="color: #66cc66;">.</span>2<span style="color: #66cc66;">.</span>3 <span style="color: #66cc66;">&#40;</span>mod_wsgi<span style="color: #66cc66;">&#41;</span>
Document Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">12</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">5549.73</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>As you can see, mod_wsgi allows Apache to put some serious muscle into Python&#8217;s WSGI. It&#8217;s no wonder that mod_wsgi is quickly becoming the de-facto standard for Apache Python deployment.</p>
<p>&nbsp;</p>
<blockquote><h4>TurboGears v2.0.3 – <a href="http://blog.curiasolutions.com/wp-content/uploads/2009/10/tg2_test_src.zip">tg2_test_src.zip</a></h4>
</blockquote>
<p>As I mentioned in Round 1, <a href="http://www.turbogears.org/" target="_blank">TurboGears</a> has quickly become my web framework of choice. Building a &#8220;best-of-breed&#8221; component stack on top of Pylons <a href="http://percious.com/blog/archives/31" target="_blank">may not be easy</a>, but <a href="http://percious.com/blog/archives/32" target="_blank">when it works</a> I believe it pays off immensely (e.g. Ubuntu).</p>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">The <span style="color: #ff0000;">'hello world'</span> test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">12</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">935.13</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
Genshi template test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>genshi_hello<span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">923</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">601.07</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
Mako template test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>mako_hello<span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">932</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">723.18</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
Jinja2 template test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>jinja_hello<span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">937</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">764.35</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
Database query tests<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>raw_sql<span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">1270</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">569.71</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
Document Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>genshi_sql<span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">2409</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">388.96</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
Document Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>mako_sql<span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">2418</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">515.07</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
Document Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>jinja_sql<span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">2472</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">535.06</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>&nbsp;</p>
<blockquote><h4>TurboGears v2.1a1 – <a href="http://blog.curiasolutions.com/wp-content/uploads/2009/10/tg21_test_src.zip">tg21_test_src.zip</a></h4>
</blockquote>
<p>The TurboGears community is alive and well, and just before finishing this post the release of v2.1 Alpha 1 was announced. Significant improvements were made to TG&#8217;s <a href="http://percious.com/blog/archives/33" target="_blank">object dispatch</a>, so I was excited to see what the new numbers would look like.</p>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">The <span style="color: #ff0000;">'hello world'</span> test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">11</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">1118.92</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
Genshi template test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>genshi_hello<span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">923</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">713.97</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
Mako template test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>mako_hello<span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">932</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">812.64</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
Jinja2 template test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>jinja_hello<span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">937</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">1000.98</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
Database query tests<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>raw_sql<span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">1270</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">673.04</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
Document Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>genshi_sql<span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">2409</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">433.45</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
Document Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>mako_sql<span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">2418</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">601.89</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
Document Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>jinja_sql<span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">2472</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">630.66</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>As you can see, the future of TurboGears looks very bright, and I only expect these numbers to get better as the final 2.1 release approaches (expected in Q1 2010).</p>
<p>&nbsp;</p>
<blockquote><h4>Pylons v0.9.7 – <a href="http://blog.curiasolutions.com/wp-content/uploads/2009/10/pylons_test_src.zip">pylons_test_src.zip</a></h4>
</blockquote>
<p><a href="http://www.pylonshq.com/" target="_blank">Pylons</a> is the foundation which TurboGears is built upon, and for those who don&#8217;t need all &#8220;the extras&#8221; it is an excellent choice.</p>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">The <span style="color: #ff0000;">'hello world'</span> test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>hello<span style="color: #66cc66;">/</span>index
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">12</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">2593.47</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
Mako template test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>hello<span style="color: #66cc66;">/</span>hello
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">932</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">1737.03</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
Database query tests<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>hello<span style="color: #66cc66;">/</span>raw_sql
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">1270</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">964.07</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
Document Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>hello<span style="color: #66cc66;">/</span>hellodb
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">2418</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">831.26</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>&nbsp;</p>
<blockquote><h4>Django v1.1 – <a href="http://blog.curiasolutions.com/wp-content/uploads/2009/10/django_test_src.zip">django_test_src.zip</a></h4>
</blockquote>
<p>While <a href="http://www.djangoproject.com/" target="_blank">Django</a> is probably Python&#8217;s most popular web framework, I have never really clicked with it. Its way of doing things is significantly different than TG/Pylons, but if you don&#8217;t mind its &#8220;<a href="http://docs.djangoproject.com/en/dev/faq/general/#django-appears-to-be-a-mvc-framework-but-you-call-the-controller-the-view-and-the-view-the-template-how-come-you-don-t-use-the-standard-names" target="_blank">MTV</a>&#8221; architectural pattern and can survive without SQLAlchemy (although there are hacks), it is a very capable framework.</p>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">The <span style="color: #ff0000;">'hello world'</span> test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>hello
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">12</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">3376.48</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
Django template test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>hellos
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">936</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">1781.43</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
Database query test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>hellodb
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">2476</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">972.11</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>&nbsp;</p>
<blockquote><h4>Bottle 0.5.8 – <a href="http://blog.curiasolutions.com/wp-content/uploads/2009/10/bottle_test_src.zip">bottle_test_src.zip</a></h4>
</blockquote>
<p><a href="http://bottle.paws.de/" target="_blank">Bottle</a> caught my attention a few months ago while surveying the WSGI landscape for an alternative to PHP when building websites that only require a few pieces of dynamic functionality. As a &#8220;micro-framework&#8221;, it does an excellent job at bridging the gap between &#8220;pure WSGI&#8221; and its &#8220;feature-rich&#8221; big brothers.</p>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">The <span style="color: #ff0000;">'hello world'</span> test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">12</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">5545.96</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
Mako template test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>hello
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">923</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">3588.04</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
Database query test <span style="color: #66cc66;">&#40;</span>SQLAlchemy<span style="color: #66cc66;">&#41;</span><span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>hellodb
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">2413</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">1479.54</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>As you can see, there was practically no difference in speed between Bottle and pure WSGI in a basic &#8220;hello world&#8221; test. Even with the addition of Mako and SQLAlchemy, Bottle performed significantly faster than a bare Pylons or Django setup.</p>
<p>(Note: Mako template inheritance was not used in the Bottle Mako test.)</p>
<p>&nbsp;</p>
<blockquote><h4>Ruby on Rails v2.3.3 – <a href="http://blog.curiasolutions.com/wp-content/uploads/2009/10/rails_test_src.zip">rails_test_src.zip</a></h4>
</blockquote>
<p>What web technology shootout would be complete without <a href="http://www.rubyonrails.org/" target="_blank">Rails</a>? Although I am not a Rails fan (mostly because I can&#8217;t stand Ruby), I believe that in many ways we owe much of the popularity and success of the web framework movement to Rails.</p>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">The <span style="color: #ff0000;">'hello world'</span> test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>hello<span style="color: #66cc66;">/</span>hello
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">12</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">1048.18</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
Rails template test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>hello<span style="color: #66cc66;">/</span>hellos
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">937</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">949.54</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
Database query test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>hello<span style="color: #66cc66;">/</span>hellodb
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">2482</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">734.18</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>(Note: I think my version of Ruby was using a newer version of the SQLite3 library than my version of Python. This may or may not have given Rails an advantage on the database query test.)</p>
<p>&nbsp;</p>
<p>For those of you who like charts, here&#8217;s a comparison of the &#8220;return hello world&#8221; test results for each framework:</p>
<div id="attachment_531" class="wp-caption aligncenter" style="width: 566px"><a href="http://blog.curiasolutions.com/wp-content/uploads/2009/10/hello_world_test.png"><img src="http://blog.curiasolutions.com/wp-content/uploads/2009/10/hello_world_test.png" alt="The return hello world test chart" title="Hello World Test" width="556" height="352" class="size-full wp-image-531" /></a><p class="wp-caption-text">The return hello world test chart</p></div>
<p>And here&#8217;s a comparison of the template test:</p>
<div id="attachment_532" class="wp-caption aligncenter" style="width: 566px"><a href="http://blog.curiasolutions.com/wp-content/uploads/2009/10/template_test.png"><img src="http://blog.curiasolutions.com/wp-content/uploads/2009/10/template_test.png" alt="The template test chart" title="Template Test" width="556" height="263" class="size-full wp-image-532" /></a><p class="wp-caption-text">The template test chart</p></div>
<p>Finally, here&#8217;s the template test with the database query added:</p>
<div id="attachment_530" class="wp-caption aligncenter" style="width: 566px"><a href="http://blog.curiasolutions.com/wp-content/uploads/2009/10/database_test.png"><img src="http://blog.curiasolutions.com/wp-content/uploads/2009/10/database_test.png" alt="Template test with database query" title="Database Query" width="556" height="264" class="size-full wp-image-530" /></a><p class="wp-caption-text">Template test with database query</p></div>
<p>&nbsp;</p>
<h4>Closing thoughts</h4>
<ul>
<li>Gone are the days of Rails as &#8220;the one framework to rule them all.&#8221; C&#8217;mon guys, it&#8217;s not 2005 anymore.</li>
<li>Python has gotten a lot of flak from other language camps for its abundance of web frameworks. While some Pythonists have been embarrassed by this, I only see it as proof of Python&#8217;s success as a language. My retort would be: &#8220;If it was as easy to write a web framework in your language as it is in Python, you&#8217;d have the same problem!&#8221;</li>
</ul>
<p>&nbsp;</p>
<h5>Coming soon: &#8220;Round 4: Micro-frameworks &mdash; Quick &#038; Dirty, and leaving PHP in the dust!&#8221;</h5>
<p>(Note: Please don&#8217;t leave benchmarking requests as comments. If you&#8217;d like me to take a look at your favorite framework, direct message me on twitter. Thanks!)</p>
]]></content:encoded>
			<wfw:commentRss>http://blog.curiasolutions.com/2009/10/the-great-web-technology-shootout-round-3-better-faster-and-shinier/feed/</wfw:commentRss>
		<slash:comments>14</slash:comments>
		</item>
		<item>
		<title>The great web technology shootout &#8211; Round 2: PHP deserves a helping hand</title>
		<link>http://blog.curiasolutions.com/2009/09/the-great-web-technology-shootout-round-2-php-deserves-a-helping-hand/</link>
		<comments>http://blog.curiasolutions.com/2009/09/the-great-web-technology-shootout-round-2-php-deserves-a-helping-hand/#comments</comments>
		<pubDate>Tue, 22 Sep 2009 06:53:41 +0000</pubDate>
		<dc:creator>Seth</dc:creator>
				<category><![CDATA[F/OSS]]></category>
		<category><![CDATA[Web Development]]></category>
		<category><![CDATA[performance]]></category>
		<category><![CDATA[php]]></category>

		<guid isPermaLink="false">http://blog.curiasolutions.com/?p=278</guid>
		<description><![CDATA[[Note: This post is the continuation of a series. Please read Round 1 first if you are just now joining us.]
In Round 1, PHP was looking like quite the tortoise of the group. However, if you&#8217;re familiar with some of the core differences between Python &#038; PHP, you&#8217;ll know that Python has been &#8220;cheating&#8221; slightly.
Let [...]]]></description>
			<content:encoded><![CDATA[<p>[Note: This post is the continuation of a series. Please read <a href="http://blog.curiasolutions.com/?p=277">Round 1</a> first if you are just now joining us.]</p>
<p>In <a href="http://blog.curiasolutions.com/?p=277">Round 1</a>, PHP was looking like quite the tortoise of the group. However, if you&#8217;re familiar with some of the core differences between Python &#038; PHP, you&#8217;ll know that Python has been &#8220;cheating&#8221; slightly.</p>
<p>Let me explain: By default, Python compiles each script into bytecode on its first execution, allowing this bottleneck to be skipped on subsequent runs. PHP, however does not perform this type of optimization by default (in the 5.x line at least), so the PHP interpreter must re-compile each file every time it is run. As you can imagine, this can give PHP (without an accelerator) a huge disadvantage when compared to languages such as Python.</p>
<p><strong>With this in mind, I have decided to take Round 2 to focus solely PHP.</strong> This will hopefully provide a clear picture of the benefits of PHP bytecode caching (at least when it comes to page-views &mdash; the memory benefits are a whole other story), and give you an idea of PHP&#8217;s performance with the help of an accelerator.</strong></p>
<p>There are <a href="http://en.wikipedia.org/wiki/List_of_PHP_accelerators" target="_blank">many PHP accelerators available</a>, but I have chosen <a href="http://php.net/apc" target="_blank">APC</a> for use here (mostly due to its inclusion in the upcoming PHP 6 core).<span id="more-278"></span></p>
<h5>What you should know about Round 2:</h5>
<ol>
<li>The hardware/software platform is the same as in <a href="http://blog.curiasolutions.com/?p=277">Round 1</a>.</li>
<li>PHP has been upgraded to v5.2.9.</li>
<li>The APC tests are using APC v3.0.19.</li>
<li>ApacheBench was run with -n # -c 10, with -n usually being 5000 or 10000, depending on the test subject&#8217;s speed.</li>
<li>Each test was run several times to make sure that there were no anomalies, with the &#8220;optimum average&#8221; chosen as the numbers represented here.</li>
<li>In an attempt to try to make these tests more scientific, I am now including the source code for each test (minus the CMS tests, which were performed on a standard install).</li>
</ol>
<p>My apc.ini settings:</p>

<div class="wp_syntax"><div class="code"><pre class="ini" style="font-family:monospace;"><span style="color: #000099;">extension</span> <span style="color: #000066; font-weight:bold;">=</span><span style="color: #660066;"> apc.so</span>
apc.enabled <span style="color: #000066; font-weight:bold;">=</span><span style="color: #660066;"> 1</span>
apc.shm_size <span style="color: #000066; font-weight:bold;">=</span><span style="color: #660066;"> 96</span>
apc.include_once_override <span style="color: #000066; font-weight:bold;">=</span><span style="color: #660066;"> 1</span>
apc.mmap_file_mask <span style="color: #000066; font-weight:bold;">=</span><span style="color: #660066;"> /tmp/apc.XXXXXX</span></pre></div></div>

<p>&nbsp;</p>
<blockquote><h4>Raw PHP &#8211; <a href='http://blog.curiasolutions.com/wp-content/uploads/2009/09/php_test_src.zip'>php_test_src.zip</a></h4>
</blockquote>
<p>To investigate the speed of &#8220;raw PHP&#8221;, I created a couple of minimal PHP scripts that printed out a simple &#8220;hello world&#8221;, and read a few lines of text from a database. Take a look at the test source code for reference.</p>
<h5>APC Disabled</h5>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">The <span style="color: #ff0000;">'hello world'</span> test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>hello<span style="color: #66cc66;">.</span>php
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">12</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">4199.43</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
The simple database query test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>hello_db<span style="color: #66cc66;">.</span>php
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">2524</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">2797.54</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<h5>APC Enabled</h5>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">The <span style="color: #ff0000;">'hello world'</span> test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>hello<span style="color: #66cc66;">.</span>php
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">12</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">6618.42</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
Simple database query test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>hello_db<span style="color: #66cc66;">.</span>php
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">2524</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">3713.99</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>Right out of the gate, we can see that APC gives PHP a significant boost.</p>
<p>Of course, a script that simply does a &#8220;echo &#8216;hello world&#8217;&#8221; isn&#8217;t really practical, so let&#8217;s take a look at a couple of popular PHP frameworks to see what a full stack looks like.</p>
<p>&nbsp;</p>
<blockquote><h4>CakePHP v1.2.5 &#8211; <a href='http://blog.curiasolutions.com/wp-content/uploads/2009/09/cakephp_test_src.zip'>cakephp_test_src.zip</a></h4>
</blockquote>
<p>CakePHP was the slowest test subject by a large margin in <a href="http://blog.curiasolutions.com/?p=277">Round 1</a>, but if you take a look at the caching section of its core configuration file you can see that the developers have written it with acceleration in mind. It includes special configuration directives for not just APC, but also <a href="http://xcache.lighttpd.net/" target="_blank">Xcache</a> and <a href="http://www.danga.com/memcached/" target="_blank">Memcache</a>.</p>
<p>Still though, without acceleration CakePHP is a sad story indeed:</p>
<h5>APC Disabled &#8211; File Caching Engine (default)</h5>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">The return <span style="color: #ff0000;">'hello world'</span> test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>hellos<span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">13</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">45.25</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
The template hello world test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>hellos<span style="color: #66cc66;">/</span>hello
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">931</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">39.90</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
The simple database query test <span style="color: #66cc66;">&#40;</span>using the ORM<span style="color: #66cc66;">&#41;</span><span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>hellos<span style="color: #66cc66;">/</span>hello_db
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">2469</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">38.80</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>However, with the help of APC things finally begin to look reasonable:</p>
<h5>APC Enabled &#8211; APC Caching Engine (in core.php)</h5>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">The return <span style="color: #ff0000;">'hello world'</span> test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>hellos<span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">13</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">140.98</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
The template hello world test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>hellos<span style="color: #66cc66;">/</span>hello
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">931</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">120.04</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
The simple database query test <span style="color: #66cc66;">&#40;</span>using the ORM<span style="color: #66cc66;">&#41;</span><span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>hellos<span style="color: #66cc66;">/</span>hello_db
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">2469</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">117.56</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>As you can see, APC increases CakePHP&#8217;s speed about threefold in these examples. These results may or may not be acceptable to you (remember that we are running these tests with a concurrency of 10), but I felt much better about CakePHP after pairing it with APC.</p>
<p>&nbsp;</p>
<blockquote><h4>Kohana v3.0 &#8211; <a href='http://blog.curiasolutions.com/wp-content/uploads/2009/09/kohana3_test_src.zip'>kohana3_test_src.zip</a></h4>
</blockquote>
<p>Kohana (a divergent of CodeIgnitor) is a much more lightweight PHP framework than CakePHP, but still gives you a pretty full featureset. Since v3.0 was released shortly after <a href="http://blog.curiasolutions.com/?p=277">Round 1</a>, I decided to use the new version for the tests here in Round 2.</p>
<p>It should be noted that although file caching is enabled in a default CakePHP install, Kohana&#8217;s caching is disabled by default. I have included a couple of tests with caching disabled here to exhibit the performance of a default Kohana install.</p>
<h5>APC Disabled &#8211; Caching Disabled (default)</h5>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">The return <span style="color: #ff0000;">'hello world'</span> test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>hello<span style="color: #66cc66;">/</span>index
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">12</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">86.73</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<h5>APC Disabled &#8211; Caching Enabled (bootstrap.php)</h5>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">The return <span style="color: #ff0000;">'hello world'</span> test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>hello<span style="color: #66cc66;">/</span>index
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">12</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">180.50</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
The template hello world test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>hellos<span style="color: #66cc66;">/</span>index
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">930</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">162.27</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
The simple database query test <span style="color: #66cc66;">&#40;</span>using the Database module<span style="color: #66cc66;">&#41;</span><span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>hellodb<span style="color: #66cc66;">/</span>index
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">2525</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">113.92</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
The simple database query test <span style="color: #66cc66;">&#40;</span>using the ORM module<span style="color: #66cc66;">&#41;</span><span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>helloorm<span style="color: #66cc66;">/</span>index
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">2525</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">99.73</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>As you can see, even without APC Kohana is quite a bit faster than CakePHP.</p>
<h5>APC Enabled &#8211; Caching Disabled</h5>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">The return <span style="color: #ff0000;">'hello world'</span> test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>hello<span style="color: #66cc66;">/</span>index
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">12</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">125.93</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<h5>APC Enabled &#8211; Caching Enabled (bootstrap.php)</h5>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">The return <span style="color: #ff0000;">'hello world'</span> test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>hello<span style="color: #66cc66;">/</span>index
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">12</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">280.67</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
&nbsp;
The template hello world test<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>hellos<span style="color: #66cc66;">/</span>index
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">930</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">253.50</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
The simple database query test <span style="color: #66cc66;">&#40;</span>using the Database module<span style="color: #66cc66;">&#41;</span><span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>hellodb<span style="color: #66cc66;">/</span>index
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">2525</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">173.28</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
The simple database query test <span style="color: #66cc66;">&#40;</span>using the ORM module<span style="color: #66cc66;">&#41;</span><span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>helloorm<span style="color: #66cc66;">/</span>index
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">2525</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">153.90</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>Although not a threefold gain as with CakePHP, Kohana with APC+caching definitely makes a good attempt at shortening the gap seen in <a href="http://blog.curiasolutions.com/?p=277">Round 1</a> between the PHP frameworks and the offerings from other languages. Of course, if raw speed is what you&#8217;re looking for, there are <a href="http://www.doophp.com/benchmark" target="_blank">faster PHP frameworks available</a> (note that the benchmarks on that page were run on a completely different setup and should not be compared with any of the numbers here).</p>
<p>&nbsp;</p>
<p><strong>Next, let&#8217;s take a look at how APC enhances the performance of a few of PHP&#8217;s heavyweights.</strong></p>
<p>&nbsp;</p>
<blockquote><h4>WordPress v2.8.4 (default install)</h4>
</blockquote>
<p>Although I&#8217;ve heard that WordPress has a built-in caching setting, a quick glance at wp-settings.php did not reveal anything. So, there are simply two WordPress tests&mdash;with APC, and without:</p>
<h5>APC Disabled</h5>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">Document Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">5519</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">24.80</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<h5>APC Enabled</h5>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">Document Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">5519</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">74.00</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>&nbsp;</p>
<blockquote><h4>Joomla v1.5.14 (default install, without sample data)</h4>
</blockquote>
<p>The results of these tests were quite disappointing to me, and made me wonder if I was doing something wrong. Sadly, it looks as though Joomla is quite the slow one of the bunch.</p>
<h5>APC Disabled</h5>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">Joomla Caching Disabled <span style="color: #66cc66;">&#40;</span>default<span style="color: #66cc66;">&#41;</span><span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">4006</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">27.90</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
Joomla Caching Enabled <span style="color: #66cc66;">&#40;</span>File<span style="color: #66cc66;">&#41;</span><span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">4006</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">31.46</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<h5>APC Enabled</h5>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">Joomla Caching Disabled<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">4006</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">41.27</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
Joomla Caching Enabled <span style="color: #66cc66;">&#40;</span>APC<span style="color: #66cc66;">&#41;</span><span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">4006</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">45.01</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>&nbsp;</p>
<blockquote><h4>Drupal v6.14 (default install, clean URLs disabled)</h4>
</blockquote>
<p>Drupal has several caching settings, so I decided to see what each one would look like and how much of a difference &#8220;aggressive&#8221; caching made.</p>
<h5>APC Disabled</h5>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">Drupal Caching Disabled <span style="color: #66cc66;">&#40;</span>default<span style="color: #66cc66;">&#41;</span><span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">5548</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">36.72</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
<span style="color: #ff0000;">&quot;Normal&quot;</span> Drupal Caching<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">5548</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">289.69</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
<span style="color: #ff0000;">&quot;Aggressive&quot;</span> Drupal Caching<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">5548</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">335.68</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<h5>APC Enabled</h5>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">Drupal Caching Disabled<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">5548</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">83.93</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
<span style="color: #ff0000;">&quot;Normal&quot;</span> Drupal Caching<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">5548</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">498.37</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span>
&nbsp;
&nbsp;
<span style="color: #ff0000;">&quot;Aggressive&quot;</span> Drupal Caching<span style="color: #66cc66;">:</span>
&nbsp;
<span style="color: #006600;">Document</span> Path<span style="color: #66cc66;">:</span>          <span style="color: #66cc66;">/</span>
Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">5548</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">564.48</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>As you can see, Drupal&#8217;s caching (especially when combined with APC) changes things drastically. These results seem to indicate that Drupal&#8217;s caching does a great job of allowing the CMS to get out of the way.</p>
<p>&nbsp;</p>
<h4>Closing thoughts</h4>
<ul>
<li>The benefits of a PHP accelerator are really too good to pass up. Even without written-in caching, an accelerator can give your code a significant boost.</li>
<li>While these tests have only really explored the impact on website page-load times, a PHP accelerator typically also cuts down on the amount of memory used by your PHP scripts.</li>
<li>PHP 6 has promised to fix a lot of things, including the lack of built-in bytecode caching. It will be interesting to revisit these tests in a few years (?) to see how much optimization the new version brings to the table.</li>
</ul>
<p>&nbsp;</p>
<h5>Can&#8217;t get enough of this madness? Just wait until you read <a href="http://blog.curiasolutions.com/?p=409">Round 3</a>!</h5>
<p>(Note: Please don&#8217;t leave benchmarking requests as comments. If you&#8217;d like me to take a look at your favorite framework, direct message me on twitter. Thanks!)</p>
]]></content:encoded>
			<wfw:commentRss>http://blog.curiasolutions.com/2009/09/the-great-web-technology-shootout-round-2-php-deserves-a-helping-hand/feed/</wfw:commentRss>
		<slash:comments>1</slash:comments>
		</item>
		<item>
		<title>The great web technology shootout &#8211; Round 1: A quick glance at the landscape</title>
		<link>http://blog.curiasolutions.com/2009/09/the-great-web-development-shootout/</link>
		<comments>http://blog.curiasolutions.com/2009/09/the-great-web-development-shootout/#comments</comments>
		<pubDate>Sun, 13 Sep 2009 05:29:28 +0000</pubDate>
		<dc:creator>Seth</dc:creator>
				<category><![CDATA[F/OSS]]></category>
		<category><![CDATA[Web Development]]></category>
		<category><![CDATA[django]]></category>
		<category><![CDATA[performance]]></category>
		<category><![CDATA[php]]></category>
		<category><![CDATA[Python]]></category>
		<category><![CDATA[rails]]></category>
		<category><![CDATA[turbogears]]></category>

		<guid isPermaLink="false">http://blog.curiasolutions.com/?p=172</guid>
		<description><![CDATA[[Update: Some of the results here ended up being less than accurate. Please see Round 3 for an explanation (and updated tests).]
Recently I went on a benchmarking spree and decided to throw ApacheBench at a bunch of the different web development technology platforms I interact with on a day-to-day basis. The results were interesting enough [...]]]></description>
			<content:encoded><![CDATA[<p><strong>[Update: Some of the results here ended up being less than accurate. Please see <a href="http://blog.curiasolutions.com/?p=409">Round 3</a> for an explanation (and updated tests).]</strong></p>
<p>Recently I went on a benchmarking spree and decided to throw <a href="http://en.wikipedia.org/wiki/ApacheBench" target="_blank">ApacheBench</a> at a bunch of the different web development technology platforms I interact with on a day-to-day basis. The results were interesting enough to me that I decided I&#8217;d take a post to share them here.</p>
<p><strong>Disclaimer: The following test results should be taken with a <em>massive</em> grain of salt.</strong> If you know anything about benchmarking, you will know that the slightest adjustments have the potential to change things drastically. While I have tried to perform each test as fairly and accurately as possible, it would be foolish to consider these results as scientific in any way. <strong>It should also be noted that my goal here was not to see how fast each technology performs at its most optimized configuration, but rather what a minimal out-of-the-box experience looks like.</strong></p>
<p>Test platform info:</p>
<ul>
<li>The hardware was an Intel Core2Quad Q9300, 2.5Ghz, 6MB Cache, 1333FSB, 2GB DDR RAM.</li>
<li>The OS was CentOS v5.3 32-bit with a standard Apache Webserver setup.</li>
<li> ApacheBench was used with only the -n and -c flags (1000 requests for the PHP frameworks, 5000 requests for everything else).	</li>
<li> Each ApacheBench test was run 5-10 times, with the &#8220;optimum average&#8221; chosen as the numbers represented here.</li>
<li>The PHP tests were done using the standard Apache PHP module.</li>
<li>The mod_wsgi tests were done in daemon mode set to 2 processes/15 threads.</li>
<li>The SQL tests were done with mysqli ($mysql->query()) on PHP, and SQLAlchemy (conn.execute()) on Python fetching and printing 5 rows of data from a sample database.</li>
</ul>
<p>&nbsp;</p>
<h4>Apache v2.2.3</h4>
<p>We will start with the raw Apache benchmark.</p>
<p>For this test, Apache loaded a simple HTML file with random text:</p>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">6537</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">8356.23</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>As expected, Apache is lightning fast.</p>
<p>Ok, so now that we&#8217;ve set the high water mark, let&#8217;s take a look at some popular web technology platforms&#8230;<span id="more-172"></span></p>
<p>&nbsp;</p>
<h4>PHP v5.2.8</h4>
<p>For this test, I created a PHP file that simply printed the $_SERVER array and some random text:</p>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">6135</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">2812.22</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>Next, I decided to add some mysqli() code to connect to a database and print out a few rows of data:</p>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">283</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">2135.46</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>Not at all bad, but who really wants to build a webapp using plain old PHP anymore?</p>
<p>Next I decided to test the two PHP frameworks that I am most familiar with.</p>
<p>&nbsp;</p>
<h4>CakePHP v1.2.5</h4>
<p>For this test, I did a basic CakePHP install, and added a route that printed &#8220;hello world&#8221; from a template:</p>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">944</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">42.00</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>As you can see, frameworks can be a blessing and a curse. On a side note, CakePHP was the only test that gave me failed requests.</p>
<h4>Kohana v2.3.4</h4>
<p>For this test, I did a basic install and loaded Kohana&#8217;s default welcome/index setup (which seemed lightweight enough to me to qualify as the test focal point):</p>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">1987</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">100.11</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>Kohana fared much better than CakePHP, but it appears as though PHP&#8217;s speed breaks down quickly once you start building a framework around it.</p>
<p>&nbsp;</p>
<h4>Python v2.5.4</h4>
<p>Unlike PHP, Python wasn&#8217;t specifically created for the web. However, several years ago the <a href="http://en.wikipedia.org/wiki/Web_Server_Gateway_Interface" target="_blank">WSGI spec</a> was introduced which makes building web applications with Python a breeze. There are many different ways to depoly a WSGI app, but my tests were limited to Python&#8217;s built-in WSGIServer, and Apache&#8217;s <a href="http://code.google.com/p/modwsgi/" target="_blank">mod_wsgi</a>.</p>
<h4>WSGIServer</h4>
<p>For this test, I used WSGI.org&#8217;s <a href="http://hg.moinmo.in/moin/1.8/raw-file/tip/wiki/server/test.wsgi" target="_blank">test.wsgi</a>:</p>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">4123</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">1658.35</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>These results were quite impressive to me, considering that this is pure Python at work.</p>
<h4>mod_wsgi</h4>
<p>mod_wsgi is Apache&#8217;s module to process WSGI scripts. This test used the exact same script as the previous WSGIServer example:</p>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">9552</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">3994.30</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>Don&#8217;t look now, but those results are better than the vanilla PHP test!</p>
<p>Next, I added SQLAlchemy imports to the test file and did a conn.execute(), in an attempt to try to mimic the PHP mysqli() test:</p>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">10212</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">3379.87</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>I expected to lose quite a bit of speed with the introduction of an ORM, but it looks like SQLAlchemy is quite fast. These results were still over 1,000reqs/s more than the PHP example.</p>
<h4>TornadoServer</h4>
<p>There&#8217;s been a lot of buzz lately about the new <a href="http://www.tornadoweb.org/" target="_blank">TornadoServer</a>, so I thought I&#8217;d throw it in the mix. This test uses their basic &#8220;hello world&#8221; example:</p>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">12</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">3330.00</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>Nothing shocking here at first glance, but the Tornado is built to handle extreme loads. So, I increased the concurrency to 1,000 and threw 10,000 requests at it:</p>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">12</span> bytes
Concurrency Level<span style="color: #66cc66;">:</span>      <span style="color: #cc66cc;">1000</span>
<span style="color: #0000ff;">COMPLETE</span> requests<span style="color: #66cc66;">:</span>      <span style="color: #cc66cc;">10000</span>
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">3288.15</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>Almost exactly the same results! On a side note, none of the other tests subjects were able to handle a concurrency of 1,000.</p>
<p>&nbsp;</p>
<h4>TurboGears v2.0.3 (mod_wsgi)</h4>
<p><a href="http://www.turbogears.com/" target="_blank">TurboGears</a> has quickly become my framework of choice, so naturally it was the first Python framework I tested.</p>
<p>For this test I setup a default quickstart project, and then added a hello() controller method which simply returned a &#8220;hello world&#8221; string:</p>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">11</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">470.18</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>Next, I added a simple Genshi template with 1 level of inheritance:</p>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">1351</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">143.26</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>While those results are not terrible, they should encourage you to check out the other templating options.</p>
<p>Next, I performed the same test with Mako:</p>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">1534</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">394.41</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>As you can see, Mako is significantly faster than Genshi.</p>
<p>TurboGears also supports Jinja2, so I thought I&#8217;d also see how it held up:</p>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">1020</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">401.04</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>I was not expecting Jinja2 to be faster than Mako, so I am not entirely sure about these results. However, the difference is only slight, and might have something to do with the fact that the dotted template names finder is disabled for Jinja templates.</p>
<h4>Pylons v0.9.7 (mod_wsgi)</h4>
<p>Mark Ramm (the current development lead of TurboGears) has been quoted as saying &#8220;TurboGears is to Pylons as Ubuntu is to Debian.&#8221; With that in mind, I naturally wanted to see how much speed I was losing with TurboGears&#8217; &#8220;additions&#8221;.</p>
<p>For this test I tried to mimic the first TurboGears test (basic layout, returning a string):</p>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">11</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">1961.00</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>While I was expecting a gain from TurboGears&#8217; omissions, I wasn&#8217;t quite expecting over four times the increase. I can only assume that the bare Pylons setup is pretty minimal, while TurboGears bootstraps a lot of stuff on for you by default.</p>
<p>Duplicating the Mako test in Pylons was also quite fast:</p>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">9743</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">1183.39</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<h4>Bottle v0.5.7 (mod_wsgi)</h4>
<p><a href="http://bottle.paws.de/" target="_blank">Bottle</a> caught my attention recently as a minimalistic Python web framework, so I thought I&#8217;d put it through the grind as well.</p>
<p>For this test, I used their simple &#8220;hello world&#8221; example:</p>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">12</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">3912.43</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>As you can see, Bottle is very close to bare WSGI as far as speed goes. On a side note, adding a sample template using Bottle&#8217;s default templating package didn&#8217;t seem to change these numbers at all.</p>
<p>Next I tested Bottle with its templating system switched to Mako:</p>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">41</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">3134.61</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>Then I added a SQLAlchemy and performed a query:</p>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">9374</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">2503.27</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>What was exciting to me about this test was that I found myself looking at a minimal Python framework equipped with a full templating system and ORM which all performed faster than the PHP+mysqli() test.</p>
<h4>Django v1.1 (mod_wsgi)</h4>
<p>I don&#8217;t use <a href="http://www.djangoproject.com/" target="_blank">Django</a>, but I thought it would be worth including in this report. This is a basic Django setup printing out a simple hello world template:</p>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">1471</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">564.41</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>As expected, a minimal Django setup is quite fast. (Note: This test included a SQLite database query.)</p>
<p>&nbsp;</p>
<h4>Ruby On Rails v2.2.3 (mod_passenger)</h4>
<p>I also don&#8217;t use <a href="http://rubyonrails.org/" target="_blank">Rails</a>, but I figured this report would be incomplete without Rails on the list. So, I took <a href="http://gems.mediatemple.net/testapp_rails.tgz">MediaTemple&#8217;s Rails test app</a> and loaded up a test page:</p>

<div class="wp_syntax"><div class="code"><pre class="progress" style="font-family:monospace;">Document Length<span style="color: #66cc66;">:</span>        <span style="color: #cc66cc;">1291</span> bytes
Requests per second<span style="color: #66cc66;">:</span>    <span style="color: #cc66cc;">433.57</span> <span style="color: #66cc66;">&#91;</span>#<span style="color: #66cc66;">/</span>sec<span style="color: #66cc66;">&#93;</span> <span style="color: #66cc66;">&#40;</span>mean<span style="color: #66cc66;">&#41;</span></pre></div></div>

<p>Note that the test app used here is doing a little bit more than just returning &#8220;Hello World&#8221;.</p>
<p>&nbsp;</p>
<h4>My Conclusions</h4>
<p>As I said in the disclaimer, to arrive at any sort of decisive conclusions based on these numbers alone would be foolish. However, here is what all of this left me thinking:</p>
<ul>
<li>All of these technologies performed adequately for use with most projects.</li>
<li>CakePHP was the only test that produced less than 100 requests per second. However, there are many articles available on optimizing CakePHP.</li>
<li>These results made me wish I had moved away from PHP much earlier in my web development carrier. There are just better options these days when it comes to choosing a web development language (imho). If you <em>must</em> use PHP for a large project, I strongly suggest that you look into <a href="http://en.wikipedia.org/wiki/PHP_accelerator" target="_blank">an accelerator</a>.</li>
<li>The Bottle+Mako+SQLAlchemy results struck me as an extremely minimal drop-in replacement for simple PHP sites.</li>
<li>mod_wsgi is a huge winner in deploying Python webapps. Way to go Graham!</li>
<li>Do your own bencharking, because <em>your mileage <strong>will</strong> vary</em>!</li>
</ul>
<p>&nbsp;</p>
<h5>Like what you&#8217;ve seen here? Then why not check out <a href="http://blog.curiasolutions.com/?p=278">Round 2</a>?</h5>
<p>(Note: Please don&#8217;t leave benchmarking requests as comments. If you&#8217;d like me to take a look at your favorite framework, direct message me on twitter. Thanks!)</p>
]]></content:encoded>
			<wfw:commentRss>http://blog.curiasolutions.com/2009/09/the-great-web-development-shootout/feed/</wfw:commentRss>
		<slash:comments>5</slash:comments>
		</item>
		<item>
		<title>How to use multiple databases in TurboGears 2.0</title>
		<link>http://blog.curiasolutions.com/2009/07/multiple-databases-in-turbogears-2/</link>
		<comments>http://blog.curiasolutions.com/2009/07/multiple-databases-in-turbogears-2/#comments</comments>
		<pubDate>Fri, 31 Jul 2009 01:46:08 +0000</pubDate>
		<dc:creator>Seth</dc:creator>
				<category><![CDATA[Python]]></category>
		<category><![CDATA[Web Development]]></category>
		<category><![CDATA[turbogears]]></category>

		<guid isPermaLink="false">http://blog.curiasolutions.com/?p=87</guid>
		<description><![CDATA[I recently had to setup a special marketing web-portal for a client of mine that would collect some basic information and throw it into a database to be retrieved later. Since I&#8217;ve already got most of the client&#8217;s web-stuff on TG2 (and in an effort to keep things DRY), I decided I&#8217;d just add a [...]]]></description>
			<content:encoded><![CDATA[<p>I recently had to setup a special marketing web-portal for a client of mine that would collect some basic information and throw it into a database to be retrieved later. Since I&#8217;ve already got most of the client&#8217;s web-stuff on TG2 (and in an effort to keep things DRY), I decided I&#8217;d just add a controller for the new pages to TG and use some mod_proxy kung-fu to make it look like it all lived autonomously. Easy enough, right?</p>
<p>Well, about halfway through this process I decided I wanted to have the collected information dump into its own SQLite DB, keeping it safely away from the rest of my client&#8217;s data. I had heard that setting up multiple databases in TG2 was supposed to be easy, and with some help from Google I soon ran across <a href="http://groups.google.com/group/turbogears/browse_thread/thread/70c14ac308563af5" target="_blank">this thread on the ML</a>. In it, Chris supplies some <a href="http://groups.google.com/group/turbogears/msg/dec19abfaa503bbc" target="_blank">very helpful example code</a> which Mike subsequently posted on his blog as <a href="http://www.blog.pythonlibrary.org/?p=210" target="_blank">a nice tutorial</a>. However, neither of these resources were exhaustive enough to achieve what I was looking for without a bit of &#8220;stumbling around&#8221;, so in an effort to be overly verbose (and perhaps unnecessarily repetitive) I&#8217;ve decided to post what I hope will be a more comprehensive run-down of how to accomplish this task.</p>
<p>Disclaimer: I am in no way a TurboGears or SQLAlchemy expert. There&#8217;s probably an easier/better way to do this, but since there&#8217;s no &#8220;official&#8221; TurboGears tutorial on this topic yet I&#8217;m afraid this is the best method I&#8217;ve found so far. If anyone reading this knows a better way to implement this kind of thing, please leave a comment and I will update this post as the suggestions come in.</p>
<blockquote><p>Step 1: Define your database urls in the [app:main] section of your .ini file(s)</p></blockquote>
<p>This is where the magic begins. Instead of one simple <em>sqlalchemy.url =</em> assignment, you&#8217;ve got to create assignments for each of the databases you want to use:<span id="more-87"></span></p>

<div class="wp_syntax"><div class="code"><pre class="python" style="font-family:monospace;"><span style="color: #808080; font-style: italic;"># in myapp/development.ini (or production.ini, or whatever.ini you are using)</span>
&nbsp;
<span style="color: #808080; font-style: italic;">#sqlalchemy.url = sqlite:///%(here)s/devdata.db</span>
sqlalchemy.<span style="color: black;">first</span>.<span style="color: black;">url</span> = sqlite:///<span style="color: #66cc66;">%</span><span style="color: black;">&#40;</span>here<span style="color: black;">&#41;</span>s/database_1.<span style="color: black;">db</span>
sqlalchemy.<span style="color: black;">second</span>.<span style="color: black;">url</span> = sqlite:///<span style="color: #66cc66;">%</span><span style="color: black;">&#40;</span>here<span style="color: black;">&#41;</span>s/database_2.<span style="color: black;">db</span></pre></div></div>

<blockquote><p>Step 2: Change the way your app loads the databases</p></blockquote>
<p>Now we need to instruct the app to load the multiple databases correctly. This requires telling base_config (in app_cfg.py) to load our own custom AppConfig with the proper multi-db assignments and a call to the model&#8217;s init_model method (more on that in Step 3):</p>

<div class="wp_syntax"><div class="code"><pre class="python" style="font-family:monospace;"><span style="color: #808080; font-style: italic;"># in myapp/config/app_cfg.py</span>
&nbsp;
<span style="color: #808080; font-style: italic;"># make sure these imports are added to the top</span>
<span style="color: #ff7700;font-weight:bold;">from</span> tg.<span style="color: black;">configuration</span> <span style="color: #ff7700;font-weight:bold;">import</span> AppConfig, config
<span style="color: #ff7700;font-weight:bold;">from</span> pylons <span style="color: #ff7700;font-weight:bold;">import</span> config <span style="color: #ff7700;font-weight:bold;">as</span> pylons_config
<span style="color: #ff7700;font-weight:bold;">from</span> myapp.<span style="color: black;">model</span> <span style="color: #ff7700;font-weight:bold;">import</span> init_model
&nbsp;
<span style="color: #808080; font-style: italic;"># add this before base_config =</span>
<span style="color: #ff7700;font-weight:bold;">class</span> MultiDBAppConfig<span style="color: black;">&#40;</span>AppConfig<span style="color: black;">&#41;</span>:
    <span style="color: #ff7700;font-weight:bold;">def</span> setup_sqlalchemy<span style="color: black;">&#40;</span><span style="color: #008000;">self</span><span style="color: black;">&#41;</span>:
        <span style="color: #483d8b;">&quot;&quot;&quot;Setup SQLAlchemy database engine(s)&quot;&quot;&quot;</span>
        <span style="color: #ff7700;font-weight:bold;">from</span> sqlalchemy <span style="color: #ff7700;font-weight:bold;">import</span> engine_from_config
        engine1 = engine_from_config<span style="color: black;">&#40;</span>pylons_config, <span style="color: #483d8b;">'sqlalchemy.first.'</span><span style="color: black;">&#41;</span>
        engine2 = engine_from_config<span style="color: black;">&#40;</span>pylons_config, <span style="color: #483d8b;">'sqlalchemy.second.'</span><span style="color: black;">&#41;</span>
        <span style="color: #808080; font-style: italic;"># engine1 should be assigned to sa_engine as well as your first engine's name</span>
        config<span style="color: black;">&#91;</span><span style="color: #483d8b;">'pylons.app_globals'</span><span style="color: black;">&#93;</span>.<span style="color: black;">sa_engine</span> = engine1
        config<span style="color: black;">&#91;</span><span style="color: #483d8b;">'pylons.app_globals'</span><span style="color: black;">&#93;</span>.<span style="color: black;">sa_engine_first</span> = engine1
        config<span style="color: black;">&#91;</span><span style="color: #483d8b;">'pylons.app_globals'</span><span style="color: black;">&#93;</span>.<span style="color: black;">sa_engine_second</span> = engine2
        <span style="color: #808080; font-style: italic;"># Pass the engine to init_model, to be able to introspect tables</span>
        init_model<span style="color: black;">&#40;</span>engine1, engine2<span style="color: black;">&#41;</span>
&nbsp;
<span style="color: #808080; font-style: italic;">#base_config = AppConfig()</span>
base_config = MultiDBAppConfig<span style="color: black;">&#40;</span><span style="color: black;">&#41;</span></pre></div></div>

<blockquote><p>Step 3: Update your model&#8217;s __init__ to handle multiple sessions and metadata</p></blockquote>
<p>Switching the model&#8217;s init from a single-db config to a multi-db simply means we have to duplicate our DBSession and metata assignments, and then update the init_model method to assign/configure each engine correctly:</p>

<div class="wp_syntax"><div class="code"><pre class="python" style="font-family:monospace;"><span style="color: #808080; font-style: italic;"># in myapp/model/__init__.py</span>
&nbsp;
<span style="color: #808080; font-style: italic;"># after the first maker/DBSession assignment, add a 2nd one</span>
maker2 = sessionmaker<span style="color: black;">&#40;</span>autoflush=<span style="color: #008000;">True</span>, autocommit=<span style="color: #008000;">False</span>,
                     extension=ZopeTransactionExtension<span style="color: black;">&#40;</span><span style="color: black;">&#41;</span><span style="color: black;">&#41;</span>
DBSession2 = scoped_session<span style="color: black;">&#40;</span>maker2<span style="color: black;">&#41;</span>
&nbsp;
<span style="color: #808080; font-style: italic;"># after the first DeclarativeBase assignment, add a 2nd one</span>
DeclarativeBase2 = declarative_base<span style="color: black;">&#40;</span><span style="color: black;">&#41;</span>
&nbsp;
<span style="color: #808080; font-style: italic;"># uncomment the metadata2 line and assign it to DeclarativeBase2.metadata</span>
metadata2 = DeclarativeBase2.<span style="color: black;">metadata</span>
&nbsp;
<span style="color: #808080; font-style: italic;"># finally, modify the init_model method to allow both engines to be passed (see step 2) and</span>
<span style="color: #808080; font-style: italic;"># assign the sessions and metadata to each engine</span>
<span style="color: #ff7700;font-weight:bold;">def</span> init_model<span style="color: black;">&#40;</span>engine1, engine2<span style="color: black;">&#41;</span>:
    <span style="color: #483d8b;">&quot;&quot;&quot;Call me before using any of the tables or classes in the model.&quot;&quot;&quot;</span>
&nbsp;
<span style="color: #808080; font-style: italic;">#    DBSession.configure(bind=engine)</span>
    DBSession.<span style="color: black;">configure</span><span style="color: black;">&#40;</span>bind=engine1<span style="color: black;">&#41;</span>
    DBSession2.<span style="color: black;">configure</span><span style="color: black;">&#40;</span>bind=engine2<span style="color: black;">&#41;</span>
&nbsp;
    metadata.<span style="color: black;">bind</span> = engine1
    metadata2.<span style="color: black;">bind</span> = engine2</pre></div></div>

<blockquote><p>Step 4: Have your model classes inherit from either DeclarativeBase or DeclarativeBase2, and use DBSession or DBSession2 as your database session handler</p></blockquote>
<p>Now that the configuration has all been taken care of, you can instruct your models to use either the first or second DeclarativeBase and DBSession based on what DB engine you want it to access:</p>

<div class="wp_syntax"><div class="code"><pre class="python" style="font-family:monospace;"><span style="color: #808080; font-style: italic;"># myapp/model/spam.py (uses engine1)</span>
<span style="color: #ff7700;font-weight:bold;">from</span> sqlalchemy <span style="color: #ff7700;font-weight:bold;">import</span> Table, ForeignKey, Column
<span style="color: #ff7700;font-weight:bold;">from</span> sqlalchemy.<span style="color: #dc143c;">types</span> <span style="color: #ff7700;font-weight:bold;">import</span> Integer, Unicode, Boolean
<span style="color: #ff7700;font-weight:bold;">from</span> myapp.<span style="color: black;">model</span> <span style="color: #ff7700;font-weight:bold;">import</span> DeclarativeBase
&nbsp;
<span style="color: #ff7700;font-weight:bold;">class</span> Spam<span style="color: black;">&#40;</span>DeclarativeBase<span style="color: black;">&#41;</span>:
    __tablename__ = <span style="color: #483d8b;">'spam'</span>
&nbsp;
        <span style="color: #008000;">self</span>.<span style="color: #008000;">id</span> = <span style="color: #008000;">id</span>
        <span style="color: #008000;">self</span>.<span style="color: black;">variety</span> = variety
&nbsp;
    <span style="color: #008000;">id</span> = Column<span style="color: black;">&#40;</span>Integer, autoincrement=<span style="color: #008000;">True</span>, primary_key=<span style="color: #008000;">True</span><span style="color: black;">&#41;</span>
    variety = Column<span style="color: black;">&#40;</span>Unicode<span style="color: black;">&#40;</span><span style="color: #ff4500;">50</span><span style="color: black;">&#41;</span>, nullable=<span style="color: #008000;">False</span><span style="color: black;">&#41;</span>
&nbsp;
<span style="color: #808080; font-style: italic;">#---------------------------------------------------------------------#</span>
&nbsp;
<span style="color: #808080; font-style: italic;"># myapp/model/eggs.py (uses engine2)</span>
<span style="color: #ff7700;font-weight:bold;">from</span> sqlalchemy <span style="color: #ff7700;font-weight:bold;">import</span> Table, ForeignKey, Column
<span style="color: #ff7700;font-weight:bold;">from</span> sqlalchemy.<span style="color: #dc143c;">types</span> <span style="color: #ff7700;font-weight:bold;">import</span> Integer, Unicode, Boolean
<span style="color: #ff7700;font-weight:bold;">from</span> myapp.<span style="color: black;">model</span> <span style="color: #ff7700;font-weight:bold;">import</span> DeclarativeBase2
&nbsp;
<span style="color: #ff7700;font-weight:bold;">class</span> Eggs<span style="color: black;">&#40;</span>DeclarativeBase2<span style="color: black;">&#41;</span>:
    __tablename__ = <span style="color: #483d8b;">'eggs'</span>
&nbsp;
    <span style="color: #ff7700;font-weight:bold;">def</span> <span style="color: #0000cd;">__init__</span><span style="color: black;">&#40;</span><span style="color: #008000;">self</span>, <span style="color: #008000;">id</span>, pkg_qty<span style="color: black;">&#41;</span>:
        <span style="color: #008000;">self</span>.<span style="color: #008000;">id</span> = <span style="color: #008000;">id</span>
        <span style="color: #008000;">self</span>.<span style="color: black;">pkg_qty</span> = pkg_qty
&nbsp;
    <span style="color: #008000;">id</span> = Column<span style="color: black;">&#40;</span>Integer, autoincrement=<span style="color: #008000;">True</span>, primary_key=<span style="color: #008000;">True</span><span style="color: black;">&#41;</span>
    pkg_qty = Column<span style="color: black;">&#40;</span>Integer, default=<span style="color: #ff4500;">12</span><span style="color: black;">&#41;</span></pre></div></div>

<blockquote><p>Optional: Create and populate each database in websetup.py</p></blockquote>
<p>If you want your setup_app method to populate each database with data, simply use the appropriate metadata/DBSession objects as you would in a single-db setup:</p>

<div class="wp_syntax"><div class="code"><pre class="python" style="font-family:monospace;"><span style="color: #808080; font-style: italic;"># in myapp/websetup.py</span>
<span style="color: #ff7700;font-weight:bold;">def</span> setup_app<span style="color: black;">&#40;</span>command, conf, <span style="color: #008000;">vars</span><span style="color: black;">&#41;</span>:
    <span style="color: #483d8b;">&quot;&quot;&quot;Place any commands to setup myapp here&quot;&quot;&quot;</span>
    load_environment<span style="color: black;">&#40;</span>conf.<span style="color: black;">global_conf</span>, conf.<span style="color: black;">local_conf</span><span style="color: black;">&#41;</span>
    <span style="color: #808080; font-style: italic;"># Load the models</span>
    <span style="color: #ff7700;font-weight:bold;">from</span> myapp <span style="color: #ff7700;font-weight:bold;">import</span> model
    <span style="color: #ff7700;font-weight:bold;">print</span> <span style="color: #483d8b;">&quot;Creating tables for engine1&quot;</span>
    model.<span style="color: black;">metadata</span>.<span style="color: black;">create_all</span><span style="color: black;">&#40;</span><span style="color: black;">&#41;</span>
    <span style="color: #ff7700;font-weight:bold;">print</span> <span style="color: #483d8b;">&quot;Creating tables for engine2&quot;</span>
    model.<span style="color: black;">metadata2</span>.<span style="color: black;">create_all</span><span style="color: black;">&#40;</span><span style="color: black;">&#41;</span>
&nbsp;
    <span style="color: #808080; font-style: italic;"># populate spam table</span>
    spam = <span style="color: black;">&#91;</span>
        model.<span style="color: black;">Spam</span><span style="color: black;">&#40;</span><span style="color: #ff4500;">1</span>, u<span style="color: #483d8b;">'Classic'</span><span style="color: black;">&#41;</span>,
        model.<span style="color: black;">Spam</span><span style="color: black;">&#40;</span><span style="color: #ff4500;">2</span>, u<span style="color: #483d8b;">'Golden Honey Grail'</span><span style="color: black;">&#41;</span>,
    <span style="color: black;">&#93;</span>
    <span style="color: #808080; font-style: italic;"># DBSession is bound to the spam table</span>
    model.<span style="color: black;">DBSession</span>.<span style="color: black;">add_all</span><span style="color: black;">&#40;</span>spam<span style="color: black;">&#41;</span>
&nbsp;
    <span style="color: #808080; font-style: italic;"># populate eggs table</span>
    eggs = <span style="color: black;">&#91;</span>
        model.<span style="color: black;">Eggs</span><span style="color: black;">&#40;</span><span style="color: #ff4500;">1</span>, <span style="color: #ff4500;">12</span><span style="color: black;">&#41;</span>,
        model.<span style="color: black;">Eggs</span><span style="color: black;">&#40;</span><span style="color: #ff4500;">2</span>, <span style="color: #ff4500;">6</span><span style="color: black;">&#41;</span>,
    <span style="color: black;">&#93;</span>
    <span style="color: #808080; font-style: italic;"># DBSession2 is bound to the eggs table</span>
    model.<span style="color: black;">DBSession2</span>.<span style="color: black;">add_all</span><span style="color: black;">&#40;</span>eggs<span style="color: black;">&#41;</span>
&nbsp;
    model.<span style="color: black;">DBSession</span>.<span style="color: black;">flush</span><span style="color: black;">&#40;</span><span style="color: black;">&#41;</span>
    model.<span style="color: black;">DBSession2</span>.<span style="color: black;">flush</span><span style="color: black;">&#40;</span><span style="color: black;">&#41;</span>
    transaction.<span style="color: black;">commit</span><span style="color: black;">&#40;</span><span style="color: black;">&#41;</span>
    <span style="color: #ff7700;font-weight:bold;">print</span> <span style="color: #483d8b;">&quot;Successfully setup&quot;</span></pre></div></div>

<p>Hope this helps! If you have any questions or suggestions please leave a comment below.</p>
]]></content:encoded>
			<wfw:commentRss>http://blog.curiasolutions.com/2009/07/multiple-databases-in-turbogears-2/feed/</wfw:commentRss>
		<slash:comments>1</slash:comments>
		</item>
		<item>
		<title>Why I&#8217;ve fallen in love with Python</title>
		<link>http://blog.curiasolutions.com/2009/07/why-ive-fallen-in-love-with-python/</link>
		<comments>http://blog.curiasolutions.com/2009/07/why-ive-fallen-in-love-with-python/#comments</comments>
		<pubDate>Mon, 27 Jul 2009 07:05:20 +0000</pubDate>
		<dc:creator>Seth</dc:creator>
				<category><![CDATA[F/OSS]]></category>
		<category><![CDATA[Python]]></category>

		<guid isPermaLink="false">http://blog.curiasolutions.com/?p=38</guid>
		<description><![CDATA[In a previous post, I briefly mentioned falling in love with Python. Now that I&#8217;m finally using Python for a large percentage of my development and am more than &#8220;casually acquainted&#8221; with its toolset, I thought it would be fun to highlight a few reasons why Python has become my new language of choice.
In an [...]]]></description>
			<content:encoded><![CDATA[<p>In a <a href="http://blog.curiasolutions.com/?p=3" title="Introducing TurboGears 2">previous post</a>, I briefly mentioned falling in love with Python. Now that I&#8217;m finally using Python for a large percentage of my development and am more than &#8220;casually acquainted&#8221; with its toolset, I thought it would be fun to highlight a few reasons why Python has become my new language of choice.</p>
<p>In an effort to help you understand where I&#8217;m coming from, let me (briefly) rehash some of my programming history: I spent much of the 90&#8217;s doing dynamic web development using Perl (weren&#8217;t <em>those</em> the days!). I eventually migrated to PHP which usually made things much easier on the web; and subsequently replaced most of my console scripting with BASH [shell scripting]. However, I&#8217;m kind of a hack and love languages so I have occasionally been known to write something in C; and although I&#8217;m not a complete stranger to Java and Ruby, I never really felt like I &#8220;clicked&#8221; with either of those languages.</p>
<p>Ok, now that I&#8217;ve hopefully convinced you that I&#8217;m not just a fly-by-night programmer, let me show you some Python code. Brace yourself, as this article is bound to get lengthy&#8230; <span id="more-38"></span></p>
<blockquote><p>Reason #1: &#8220;Whitespace done right&#8221; is actually a good thing</p></blockquote>
<p>The first thing that people either absolutely love or adamantly hate about Python is the fact that code blocks are heavily tied to proper usage of whitespace. At first glance, this causes many curious onlookers from other languages to shy away from Python and continue in their bracket-encapsulated bondage. I&#8217;ll admit, at first I wasn&#8217;t too wild about these new restrictions either (I&#8217;m a programmer—I should be able to do my code how I want!); but the first time I had to go back to doing something in PHP for a client, I quickly realized I didn&#8217;t ever want to put anything inside brackets again:</p>

<div class="wp_syntax"><div class="code"><pre class="python" style="font-family:monospace;"><span style="color: #808080; font-style: italic;"># A sample function that's free from brackets, parenthesis, and semicolons!</span>
<span style="color: #ff7700;font-weight:bold;">def</span> check_number<span style="color: black;">&#40;</span>x<span style="color: black;">&#41;</span>:
    <span style="color: #ff7700;font-weight:bold;">if</span> <span style="color: #008000;">type</span><span style="color: black;">&#40;</span>x<span style="color: black;">&#41;</span> <span style="color: #66cc66;">!</span>= <span style="color: #008000;">type</span><span style="color: black;">&#40;</span><span style="color: #ff4500;">0</span><span style="color: black;">&#41;</span>:
        <span style="color: #ff7700;font-weight:bold;">return</span> <span style="color: #483d8b;">&quot;That isn't a number!&quot;</span>
    <span style="color: #ff7700;font-weight:bold;">elif</span> x <span style="color: #66cc66;">%</span> <span style="color: #ff4500;">2</span>:
        <span style="color: #ff7700;font-weight:bold;">return</span> <span style="color: #483d8b;">&quot;This number is odd&quot;</span>
    <span style="color: #ff7700;font-weight:bold;">else</span>:
        <span style="color: #ff7700;font-weight:bold;">return</span> <span style="color: #483d8b;">&quot;This number is even&quot;</span></pre></div></div>

<p>Not only does this make Python source code incredibly readable, but you also end up typing less!</p>
<blockquote><p>Reason #2: I almost always know what type of object I&#8217;m working with</p></blockquote>
<p>One thing that always annoyed me about Perl was that you set an array or hash using the &#8216;@&#8217; or &#8216;%&#8217; sigils, but typically accessed them using the &#8216;$&#8217; sigil. PHP simplified this a bit, but &#8220;$array = array();&#8221; could be a simple list or an associative array (known as a dictionary in Python). With Python on the other hand, the enclosure defines the type (for most built-in types at least), and since everything is an object there&#8217;s no need for silly sigils:</p>

<div class="wp_syntax"><div class="code"><pre class="python" style="font-family:monospace;"><span style="color: #808080; font-style: italic;"># the type of enclosure makes a distinction between a string , a list, and a dictionary</span>
<span style="color: #dc143c;">string</span> = <span style="color: #483d8b;">'This is a string'</span>
<span style="color: #ff7700;font-weight:bold;">print</span> <span style="color: #dc143c;">string</span> <span style="color: #808080; font-style: italic;"># prints 'This is a string'</span>
&nbsp;
<span style="color: #008000;">list</span> = <span style="color: black;">&#91;</span><span style="color: #483d8b;">'This'</span>, <span style="color: #483d8b;">'is'</span>, <span style="color: #483d8b;">'a'</span>, <span style="color: #483d8b;">'list'</span><span style="color: black;">&#93;</span>
<span style="color: #ff7700;font-weight:bold;">print</span> <span style="color: #008000;">list</span><span style="color: black;">&#91;</span><span style="color: #ff4500;">0</span><span style="color: black;">&#93;</span> <span style="color: #808080; font-style: italic;"># prints 'This'</span>
&nbsp;
<span style="color: #008000;">dict</span> = <span style="color: black;">&#123;</span><span style="color: #483d8b;">'first_key'</span>: <span style="color: #483d8b;">'This'</span>, <span style="color: #483d8b;">'second_key'</span>: <span style="color: #483d8b;">'is'</span>, <span style="color: #483d8b;">'third_key'</span>: <span style="color: #483d8b;">'a'</span>, <span style="color: #483d8b;">'fourth_key'</span>: <span style="color: #483d8b;">'dictionary'</span><span style="color: black;">&#125;</span>
<span style="color: #ff7700;font-weight:bold;">print</span> <span style="color: #008000;">dict</span><span style="color: black;">&#91;</span><span style="color: #483d8b;">'fourth_key'</span><span style="color: black;">&#93;</span> <span style="color: #808080; font-style: italic;"># prints 'dictionary'</span></pre></div></div>

<p>On a somewhat related note, you can use these objects on-the-fly in your scripts without even having to assign them to a name. This isn&#8217;t usually practical of course (and may not be recommended), but it is possible:</p>

<div class="wp_syntax"><div class="code"><pre class="python" style="font-family:monospace;"><span style="color: #ff7700;font-weight:bold;">print</span> <span style="color: black;">&#91;</span><span style="color: #483d8b;">'This'</span>, <span style="color: #483d8b;">'is'</span>, <span style="color: #483d8b;">'a'</span>, <span style="color: #483d8b;">'list'</span><span style="color: black;">&#93;</span><span style="color: black;">&#91;</span><span style="color: #ff4500;">1</span><span style="color: black;">&#93;</span> <span style="color: #808080; font-style: italic;"># prints 'is'</span>
<span style="color: #ff7700;font-weight:bold;">print</span> <span style="color: black;">&#123;</span><span style="color: #483d8b;">'first_key'</span>: <span style="color: #483d8b;">'This'</span>, <span style="color: #483d8b;">'second_key'</span>: <span style="color: #483d8b;">'is'</span>, <span style="color: #483d8b;">'third_key'</span>: <span style="color: #483d8b;">'a'</span>, <span style="color: #483d8b;">'fourth_key'</span>: <span style="color: #483d8b;">'dictionary'</span><span style="color: black;">&#125;</span><span style="color: black;">&#91;</span><span style="color: #483d8b;">'first_key'</span><span style="color: black;">&#93;</span> <span style="color: #808080; font-style: italic;"># prints 'This'</span></pre></div></div>

<blockquote><p>Reason #3: I can throw practically any type of object almost anywhere I want</p></blockquote>
<p>Ok, so that might be a slight exaggeration; but the truth is that since Python treats everything as an object, you&#8217;re free to pack things away in the strangest places. A great way of demonstrating the power and flexibility of Python is in an example showing a few ways different types of objects can be stored and accessed:</p>

<div class="wp_syntax"><div class="code"><pre class="python" style="font-family:monospace;"><span style="color: #808080; font-style: italic;"># create a function</span>
<span style="color: #ff7700;font-weight:bold;">def</span> hello_function<span style="color: black;">&#40;</span><span style="color: black;">&#41;</span>:
    <span style="color: #ff7700;font-weight:bold;">return</span> <span style="color: #483d8b;">'Hello world!'</span>
&nbsp;
<span style="color: #808080; font-style: italic;"># create a file handle</span>
hello_file = <span style="color: #008000;">open</span><span style="color: black;">&#40;</span><span style="color: #483d8b;">'~/hello.txt'</span>, <span style="color: #483d8b;">'r'</span><span style="color: black;">&#41;</span>
&nbsp;
<span style="color: #808080; font-style: italic;"># create a list with the function, the file handle, and a string</span>
<span style="color: #008000;">list</span> = <span style="color: black;">&#91;</span>hello_function, hello_file, <span style="color: #483d8b;">'Hello world!'</span><span style="color: black;">&#93;</span>
<span style="color: #ff7700;font-weight:bold;">print</span> <span style="color: #008000;">list</span><span style="color: black;">&#91;</span><span style="color: #ff4500;">0</span><span style="color: black;">&#93;</span><span style="color: black;">&#40;</span><span style="color: black;">&#41;</span> <span style="color: #808080; font-style: italic;"># prints 'Hello world!'</span>
<span style="color: #ff7700;font-weight:bold;">print</span> <span style="color: #008000;">list</span><span style="color: black;">&#91;</span><span style="color: #ff4500;">1</span><span style="color: black;">&#93;</span>.<span style="color: black;">read</span><span style="color: black;">&#40;</span><span style="color: black;">&#41;</span> <span style="color: #808080; font-style: italic;"># prints 'Hello world!\n'</span>
<span style="color: #ff7700;font-weight:bold;">print</span> <span style="color: #008000;">list</span><span style="color: black;">&#91;</span><span style="color: #ff4500;">2</span><span style="color: black;">&#93;</span> <span style="color: #808080; font-style: italic;"># prints 'Hello world!'</span>
&nbsp;
<span style="color: #808080; font-style: italic;"># or how about a dictionary?</span>
<span style="color: #008000;">dict</span> = <span style="color: black;">&#123;</span><span style="color: #483d8b;">'function'</span>: hello_function, <span style="color: #483d8b;">'file_object'</span>: hello_file, <span style="color: #483d8b;">'string'</span>: <span style="color: #483d8b;">'Hello world!'</span><span style="color: black;">&#125;</span>
<span style="color: #ff7700;font-weight:bold;">print</span> <span style="color: #008000;">dict</span><span style="color: black;">&#91;</span><span style="color: #483d8b;">'function'</span><span style="color: black;">&#93;</span><span style="color: black;">&#40;</span><span style="color: black;">&#41;</span> <span style="color: #808080; font-style: italic;"># prints 'Hello world!'</span>
<span style="color: #ff7700;font-weight:bold;">print</span> <span style="color: #008000;">dict</span><span style="color: black;">&#91;</span><span style="color: #483d8b;">'file_object'</span><span style="color: black;">&#93;</span>.<span style="color: black;">read</span><span style="color: black;">&#40;</span><span style="color: black;">&#41;</span> <span style="color: #808080; font-style: italic;"># prints 'Hello world!\n'</span>
<span style="color: #ff7700;font-weight:bold;">print</span> <span style="color: #008000;">dict</span><span style="color: black;">&#91;</span><span style="color: #483d8b;">'string'</span><span style="color: black;">&#93;</span> <span style="color: #808080; font-style: italic;"># prints 'Hello world!'</span>
&nbsp;
<span style="color: #808080; font-style: italic;"># print types</span>
<span style="color: #ff7700;font-weight:bold;">print</span> <span style="color: #008000;">type</span><span style="color: black;">&#40;</span><span style="color: #008000;">list</span><span style="color: black;">&#91;</span><span style="color: #ff4500;">0</span><span style="color: black;">&#93;</span><span style="color: black;">&#41;</span> <span style="color: #808080; font-style: italic;"># prints &lt;type 'function'&gt;</span>
<span style="color: #ff7700;font-weight:bold;">print</span> <span style="color: #008000;">type</span><span style="color: black;">&#40;</span><span style="color: #008000;">dict</span><span style="color: black;">&#91;</span><span style="color: #483d8b;">'file_object'</span><span style="color: black;">&#93;</span><span style="color: black;">&#41;</span> <span style="color: #808080; font-style: italic;"># prints &lt;type 'file'&gt;</span>
<span style="color: #ff7700;font-weight:bold;">print</span> <span style="color: #008000;">type</span><span style="color: black;">&#40;</span><span style="color: #008000;">list</span><span style="color: black;">&#91;</span><span style="color: #ff4500;">2</span><span style="color: black;">&#93;</span><span style="color: black;">&#41;</span> <span style="color: #808080; font-style: italic;"># prints &lt;type 'str'&gt; (str for string)</span></pre></div></div>

<p>The <em>shelve</em> and <em>pickle</em> modules are also work a look if you like what you&#8217;ve just seen here.</p>
<blockquote><p>Reason #4: The &#8220;in&#8221; operator simplifies sequences and makes membership testing a breeze</p></blockquote>
<p>To explain the power of &#8220;in&#8221;, it&#8217;s probably easier to just show you a few examples:</p>

<div class="wp_syntax"><div class="code"><pre class="python" style="font-family:monospace;"><span style="color: #483d8b;">'p'</span> <span style="color: #ff7700;font-weight:bold;">in</span> <span style="color: #483d8b;">&quot;spam&quot;</span> <span style="color: #808080; font-style: italic;"># == True</span>
<span style="color: #483d8b;">'item1'</span> <span style="color: #ff7700;font-weight:bold;">in</span> <span style="color: black;">&#91;</span><span style="color: #483d8b;">'item1'</span>, <span style="color: #483d8b;">'item2'</span><span style="color: black;">&#93;</span> <span style="color: #808080; font-style: italic;"># == True</span>
<span style="color: #483d8b;">'key'</span> <span style="color: #ff7700;font-weight:bold;">in</span> <span style="color: black;">&#123;</span><span style="color: #483d8b;">&quot;key&quot;</span>: <span style="color: #483d8b;">'value'</span><span style="color: black;">&#125;</span> <span style="color: #808080; font-style: italic;"># == True</span>
&nbsp;
<span style="color: #808080; font-style: italic;"># prints anchor tags for a navigation</span>
<span style="color: #ff7700;font-weight:bold;">for</span> title, link <span style="color: #ff7700;font-weight:bold;">in</span> <span style="color: black;">&#123;</span><span style="color: #483d8b;">'Home'</span>: <span style="color: #483d8b;">'index.html'</span>, <span style="color: #483d8b;">'About'</span>: <span style="color: #483d8b;">'about.html'</span>, <span style="color: #483d8b;">'Contact'</span>: <span style="color: #483d8b;">'contact.html'</span><span style="color: black;">&#125;</span>.<span style="color: black;">iteritems</span><span style="color: black;">&#40;</span><span style="color: black;">&#41;</span>:
    <span style="color: #ff7700;font-weight:bold;">print</span> <span style="color: #483d8b;">'&lt;a href=&quot;%s&quot;&gt;%s&lt;/a&gt;'</span> <span style="color: #66cc66;">%</span> <span style="color: black;">&#40;</span>link, title<span style="color: black;">&#41;</span></pre></div></div>

<blockquote><p>Reason #5: Slicing, stepping, and striding are a walk in the park</p></blockquote>
<p>Probably my most-hated function in PHP is the substr() function. It should just be easier to access a parts of a string. Thankfully, in Python it&#8217;s a piece of cake:</p>

<div class="wp_syntax"><div class="code"><pre class="python" style="font-family:monospace;"><span style="color: #808080; font-style: italic;">#string slices and stepping/striding</span>
<span style="color: #dc143c;">string</span> = <span style="color: #483d8b;">'This is a string'</span>
&nbsp;
<span style="color: #ff7700;font-weight:bold;">print</span> <span style="color: #dc143c;">string</span><span style="color: black;">&#91;</span><span style="color: #ff4500;">3</span><span style="color: black;">&#93;</span> <span style="color: #808080; font-style: italic;"># prints 's'</span>
<span style="color: #ff7700;font-weight:bold;">print</span> <span style="color: #dc143c;">string</span><span style="color: black;">&#91;</span><span style="color: #ff4500;">5</span>:<span style="color: #ff4500;">9</span><span style="color: black;">&#93;</span> <span style="color: #808080; font-style: italic;"># prints 'is a'</span>
<span style="color: #ff7700;font-weight:bold;">print</span> <span style="color: #dc143c;">string</span><span style="color: black;">&#91;</span>::<span style="color: #ff4500;">2</span><span style="color: black;">&#93;</span> <span style="color: #808080; font-style: italic;"># prints 'Ti sasrn' (a stride of 2 means print every other item)</span>
&nbsp;
<span style="color: #808080; font-style: italic;"># works on lists and dictionaries as well</span>
<span style="color: #008000;">list</span> = <span style="color: black;">&#91;</span><span style="color: #483d8b;">'This'</span>, <span style="color: #483d8b;">'is'</span>, <span style="color: #483d8b;">'a'</span>, <span style="color: #483d8b;">'list'</span><span style="color: black;">&#93;</span>
&nbsp;
<span style="color: #ff7700;font-weight:bold;">print</span> <span style="color: #008000;">list</span><span style="color: black;">&#91;</span>::<span style="color: #ff4500;">2</span><span style="color: black;">&#93;</span> <span style="color: #808080; font-style: italic;"># prints ['This', 'a']</span></pre></div></div>

<blockquote><p>Reason #6-XX: Python makes programming fun again!</p></blockquote>
<p>I could go on and on, but this post would be endless. As you can see, Python is a flexible, powerful, and fun programming language. For more Python concepts (I didn&#8217;t have time here to go into list comprehensions, generators, decorators, etc), I&#8217;d suggest you take a gander at Wikipedia&#8217;s page on <a href="http://en.wikipedia.org/wiki/Python_syntax_and_semantics" target="_blank">Python syntax and semantics</a>.</p>
<p>Finally, I&#8217;ll leave you with a few relevant links that are worth taking the time to check out:</p>
<ul>
<li>Bruce Eckel&#8217;s slides: <a href="http://www.mindviewinc.com/downloads/pub/eckel/LovePython.zip" target="_blank">Why I Love Python</a></li>
<li><a href="http://docs.python.org/tutorial/" target="_blank">Python&#8217;s official tutorial</a> (by Guido&mdash;Python&#8217;s creator)</li>
<li><a href="http://diveintopython.org/" target="_blank">Dive Into Python</a> &#8211; Free online book</li>
<li><a href="http://xkcd.com/353/" target="_blank">xkcd&#8217;s famous Python comic</a></li>
</ul>
]]></content:encoded>
			<wfw:commentRss>http://blog.curiasolutions.com/2009/07/why-ive-fallen-in-love-with-python/feed/</wfw:commentRss>
		<slash:comments>2</slash:comments>
		</item>
		<item>
		<title>Making TurboGears and Authorize.net play nice together</title>
		<link>http://blog.curiasolutions.com/2009/07/making-turbogears-and-authorize-net-play-nice-together/</link>
		<comments>http://blog.curiasolutions.com/2009/07/making-turbogears-and-authorize-net-play-nice-together/#comments</comments>
		<pubDate>Sat, 18 Jul 2009 19:18:49 +0000</pubDate>
		<dc:creator>Seth</dc:creator>
				<category><![CDATA[Python]]></category>
		<category><![CDATA[Web Development]]></category>
		<category><![CDATA[turbogears]]></category>

		<guid isPermaLink="false">http://blog.curiasolutions.com/?p=40</guid>
		<description><![CDATA[I recently had to wire up an Authorize.net form for a TurboGears project I&#8217;m building, and since I&#8217;m a bit new to custom form validation in TG I had quite a bit of trouble figuring out the &#8220;right&#8221; way to do it.
The goal was to get the form to go through two layers of validation [...]]]></description>
			<content:encoded><![CDATA[<p>I recently had to wire up an Authorize.net form for a TurboGears project I&#8217;m building, and since I&#8217;m a bit new to custom form validation in TG I had quite a bit of trouble figuring out the &#8220;right&#8221; way to do it.</p>
<p>The goal was to get the form to go through two layers of validation before passing:</p>
<ol>
<li>Use the validation packages provided by TG (<a href="http://toscawidgets.org/" target="_blank">tw.forms</a> and <a href="http://www.formencode.org/" target="_blank">formencode</a>)</li>
<li>If the first layer of validation passes, try to run the authorize.net charge. If this returns a response code of 1 (approved) then all validation has passed. Otherwise, invalidate the form and flash the authorize.net error.</li>
</ol>
<p>tw.forms and formencode are awesome packages, but at first glance there didn&#8217;t seem to be &#8220;one obvious way&#8221; to do things. I found the documentation and tutorials to be scant at best, and eventually went to the <a href="http://groups.google.com/group/turbogears" target="_blank">TurboGears mailing list</a> for help. In spite of the fact that I&#8217;d probably give TurboGears a &#8220;C&#8221; <em>at best</em> for its documentation (not to mention the availability of &#8220;verbose&#8221; tutorials), the folks on the mailing list and IRC channel (when it&#8217;s actually active) seem to be quite helpful.</p>
<p>Eventually (and with additional help from Google&#8217;s <a href="http://www.google.com/codesearch" target="_blank">source code search</a>) I was able to hack together something that worked as planned, and was overjoyed to observe chained validators in action. Below is some example code provided for your hacking pleasure. To download the full example file, see the links at the bottom of the post.<span id="more-40"></span></p>

<div class="wp_syntax"><div class="code"><pre class="python" style="font-family:monospace;"><span style="color: #808080; font-style: italic;"># FancyValidator to process the Credit Card using the authorize package</span>
<span style="color: #ff7700;font-weight:bold;">class</span> ProcessCard<span style="color: black;">&#40;</span>validators.<span style="color: black;">FancyValidator</span><span style="color: black;">&#41;</span>:
    <span style="color: #ff7700;font-weight:bold;">def</span> _to_python<span style="color: black;">&#40;</span><span style="color: #008000;">self</span>, value, state<span style="color: black;">&#41;</span>:
        <span style="color: #ff7700;font-weight:bold;">from</span> authorize <span style="color: #ff7700;font-weight:bold;">import</span> aim <span style="color: #ff7700;font-weight:bold;">as</span> aim_api
&nbsp;
        <span style="color: #808080; font-style: italic;"># Setup the aim Api object.</span>
        aim = aim_api.<span style="color: black;">Api</span><span style="color: black;">&#40;</span>AUTHNET_LOGIN, AUTHNET_KEY, is_test=<span style="color: #008000;">False</span><span style="color: black;">&#41;</span>
&nbsp;
        <span style="color: #808080; font-style: italic;"># Create a transaction against a credit card</span>
        result_dict = aim.<span style="color: black;">transaction</span><span style="color: black;">&#40;</span>
            amount=u<span style="color: #483d8b;">&quot;16.00&quot;</span>,
            card_num=<span style="color: #008000;">unicode</span><span style="color: black;">&#40;</span>value<span style="color: black;">&#91;</span><span style="color: #483d8b;">'card_number'</span><span style="color: black;">&#93;</span><span style="color: black;">&#41;</span>,
            exp_date=<span style="color: #008000;">unicode</span><span style="color: black;">&#40;</span>value<span style="color: black;">&#91;</span><span style="color: #483d8b;">'card_expiry'</span><span style="color: black;">&#93;</span><span style="color: black;">&#41;</span>,
            <span style="color: #808080; font-style: italic;"># ...and others...</span>
            <span style="color: black;">&#41;</span>
&nbsp;
        <span style="color: #ff7700;font-weight:bold;">if</span> result_dict<span style="color: black;">&#91;</span><span style="color: #483d8b;">'code'</span><span style="color: black;">&#93;</span> == <span style="color: #483d8b;">'1'</span>:
            <span style="color: #808080; font-style: italic;"># success</span>
            <span style="color: #ff7700;font-weight:bold;">return</span> value
        <span style="color: #ff7700;font-weight:bold;">else</span>:
            <span style="color: #808080; font-style: italic;"># failure</span>
            <span style="color: #ff7700;font-weight:bold;">raise</span> validators.<span style="color: black;">Invalid</span><span style="color: black;">&#40;</span>result_dict<span style="color: black;">&#91;</span><span style="color: #483d8b;">'reason_text'</span><span style="color: black;">&#93;</span>, value, state<span style="color: black;">&#41;</span>
&nbsp;
<span style="color: #ff7700;font-weight:bold;">class</span> AuthnetForm<span style="color: black;">&#40;</span>TableForm<span style="color: black;">&#41;</span>:
    submit_text=<span style="color: #483d8b;">'Process Card'</span>
&nbsp;
    validator = validators.<span style="color: black;">Schema</span><span style="color: black;">&#40;</span>
        chained_validators = <span style="color: black;">&#91;</span>
            validators.<span style="color: black;">CreditCardValidator</span><span style="color: black;">&#40;</span><span style="color: #483d8b;">'card_type'</span>,<span style="color: #483d8b;">'card_number'</span><span style="color: black;">&#41;</span>,
            validators.<span style="color: black;">CreditCardSecurityCode</span><span style="color: black;">&#40;</span><span style="color: #483d8b;">'card_type'</span>,<span style="color: #483d8b;">'card_cvv'</span><span style="color: black;">&#41;</span>,
            <span style="color: #808080; font-style: italic;"># you could also add an expiry validator, but authnet will handle this for you</span>
            ProcessCard<span style="color: black;">&#40;</span><span style="color: black;">&#41;</span>
        <span style="color: black;">&#93;</span>
    <span style="color: black;">&#41;</span>
&nbsp;
    <span style="color: #ff7700;font-weight:bold;">class</span> fields<span style="color: black;">&#40;</span>WidgetsList<span style="color: black;">&#41;</span>:
        name = TextField<span style="color: black;">&#40;</span>validator=validators.<span style="color: black;">String</span><span style="color: black;">&#40;</span>not_empty=<span style="color: #008000;">True</span><span style="color: black;">&#41;</span><span style="color: black;">&#41;</span>
        <span style="color: #808080; font-style: italic;"># ...and others like address, city, state, zip...</span>
        spacer = Spacer<span style="color: black;">&#40;</span>suppress_label=<span style="color: #008000;">True</span><span style="color: black;">&#41;</span>
        card_type = SingleSelectField<span style="color: black;">&#40;</span>options=<span style="color: black;">&#91;</span><span style="color: black;">&#40;</span><span style="color: #483d8b;">'visa'</span>, <span style="color: #483d8b;">'Visa'</span><span style="color: black;">&#41;</span>, <span style="color: black;">&#40;</span><span style="color: #483d8b;">'mastercard'</span>, <span style="color: #483d8b;">'Master Card'</span><span style="color: black;">&#41;</span>, <span style="color: black;">&#40;</span><span style="color: #483d8b;">'discover'</span>, <span style="color: #483d8b;">'Discover'</span><span style="color: black;">&#41;</span>, <span style="color: black;">&#40;</span><span style="color: #483d8b;">'amex'</span>, <span style="color: #483d8b;">'American Express'</span><span style="color: black;">&#41;</span><span style="color: black;">&#93;</span>, validator=validators.<span style="color: black;">NotEmpty</span><span style="color: black;">&#41;</span>
        card_expiry = CalendarDatePicker<span style="color: black;">&#40;</span>date_format=<span style="color: #483d8b;">&quot;%m/%Y&quot;</span>, validator=validators.<span style="color: black;">NotEmpty</span><span style="color: black;">&#41;</span>
        card_number = TextField<span style="color: black;">&#40;</span>label_text=<span style="color: #483d8b;">'Card #'</span>, validator=validators.<span style="color: black;">NotEmpty</span><span style="color: black;">&#41;</span>
        card_cvv = TextField<span style="color: black;">&#40;</span>label_text=<span style="color: #483d8b;">'CVV Code'</span>, validator=validators.<span style="color: black;">NotEmpty</span><span style="color: black;">&#41;</span>
&nbsp;
authnet_form = AuthnetForm<span style="color: black;">&#40;</span><span style="color: #483d8b;">'authnet_form'</span>, action=url<span style="color: black;">&#40;</span><span style="color: #483d8b;">'/authnet/process/'</span><span style="color: black;">&#41;</span><span style="color: black;">&#41;</span></pre></div></div>

<p>Links of interest:</p>
<ul>
<li><a href="http://blog.curiasolutions.com/wp-content/uploads/2009/07/authnet.py">Download the authnet.py file</a></li>
<li><a href="http://groups.google.com/group/turbogears/t/c721e2d15bb2c134" target="_blank">This post&#8217;s original Google group thread</a></li>
<li><a href="http://www.adroll.com/labs" target="_blank">The Authorize package</a></li>
</ul>
]]></content:encoded>
			<wfw:commentRss>http://blog.curiasolutions.com/2009/07/making-turbogears-and-authorize-net-play-nice-together/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>Version Control Systems: My opinion so far</title>
		<link>http://blog.curiasolutions.com/2009/06/versioning-control-systems-my-opinion-so-far/</link>
		<comments>http://blog.curiasolutions.com/2009/06/versioning-control-systems-my-opinion-so-far/#comments</comments>
		<pubDate>Sat, 20 Jun 2009 18:29:05 +0000</pubDate>
		<dc:creator>Seth</dc:creator>
				<category><![CDATA[F/OSS]]></category>
		<category><![CDATA[DVCS]]></category>

		<guid isPermaLink="false">http://blog.curiasolutions.com/?p=8</guid>
		<description><![CDATA[There&#8217;s been a lot of noise in the development community over the last year or so about DVCS. Being highly opinionated myself, I can&#8217;t help but take a minute to share my own thoughts on the matter.
Let me start by first of all saying: Yes, it&#8217;s true. Distributed Version Control Systems really are everything that [...]]]></description>
			<content:encoded><![CDATA[<p>There&#8217;s been a lot of noise in the development community over the last year or so about DVCS. Being highly opinionated myself, I can&#8217;t help but take a minute to share my own thoughts on the matter.</p>
<p>Let me start by first of all saying: Yes, it&#8217;s true. <span style="text-decoration: underline;">D</span>istributed <span style="text-decoration: underline;">V</span>ersion <span style="text-decoration: underline;">C</span>ontrol <span style="text-decoration: underline;">S</span>ystems really are everything that Centralized Version Control Systems (read: SVN/CVS) fall short on. However, having been a long-time Subversion user myself, as well as the fact that I hardly ever work on a team with more than a couple of other developers, it took a lot of noise to make me even care about the differences between DVCS and CVCS. <em>So what is the difference</em> you might ask? Almost everything; <strong>and that&#8217;s a good thing!</strong> (ok, that&#8217;s a slight exaggeration, but the benefits are so profound it almost feels that way)</p>
<p>Don&#8217;t worry, I&#8217;ll spare you the repetition of what has already been said elsewhere. My story is this: I spent about a month looking at the various DVCS&#8217; and eventually narrowed it down to Git vs Mercurial (why anyone would use Bazaar over Mercurial is beyond me). Since I&#8217;m a Linux user, and since Github impressed me, I finally decided to go with Git on the next project that came my way. Boy oh boy do I wish I had run across <a href="http://importantshock.wordpress.com/2008/08/07/git-vs-mercurial/" target="_blank">this article</a> beforehand&#8230;<span id="more-8"></span></p>
<blockquote><p>Git: MacGyver</p></blockquote>
<p>As the referenced article implies, Git does anything and everything you&#8217;d ever need it to do. Now, as a Linux guy I typically like the options I&#8217;m given with the &#8220;everything plus the kitchen sink&#8221; approach. However, I quickly found myself using 2 or 3 commands to do something I swear <em>everyone</em> must do and really should be only one command. Then there&#8217;s all the command options you have to throw in to make the command do exactly what you want it to do. This quickly became tiresome and once that project was over I decided to give Mercurial a try, hoping it would be much kinder to me.</p>
<blockquote><p>Mercurial: Bond</p></blockquote>
<p>The first thing I realized when using Mercurial on a &#8220;real&#8221; project was that it seemed to think the same way I did (an interesting trend I&#8217;ve noticed with Python programs). Whereas with Git I found myself typing &#8220;git help &lt;command&gt;&#8221; over and over again, I hardly ever needed to look up any special options with the Mercurial commands. Another thing I soon realized was that a lot of the negative press that I&#8217;d read on Mercurial was actually written up about the 1.0 version (or earlier). I was pleased to find that much of these &#8220;issues&#8221; no longer existed in my 1.2.x version of the tool.</p>
<p>I now run almost all of my new projects on Mercurial. I say &#8220;almost all&#8221; because there are a few cases where the project is small and a tool I&#8217;m using in the project already has a Git repository so it makes sense to keep everything streamlined. However, with new tools like <a href="http://hg-git.github.com/" target="_blank">hg-git</a>, even that may become a thing of the past.</p>
<p>In conclusion, I strongly encourage you to check out Mercurial if you&#8217;re jumping in to the DVCS game. Unless you&#8217;re a huge project like the Linux kernel, you probably won&#8217;t ever need the &#8220;advanced functionality&#8221; that Git puts at your fingertips (which is probably in most cases also available in Mercurial, just not right in your face). If you&#8217;re still using SVN (or even, *gasp*, CVS), then why not get with the times? Mercurial has some amazing <a href="http://www.selenic.com/mercurial/wiki/ConvertExtension" target="_blank">conversion options</a> that will retain your history and make the switch a piece of cake. Sure, there are a few new concepts to learn (pull/push and the distributed nature), but it&#8217;s only a minor learning curve that will pay off immensely.</p>
<p>&nbsp;</p>
<p>Links of interest:</p>
<ul>
<li><a href="http://www.selenic.com/mercurial" target="_blank">Mercurial</a></li>
<li><a href="http://www.bitbucket.org/" target="_blank">Bitbucket</a> &#8211; Mercurial&#8217;s version of Github</li>
<li><a href="http://en.wikipedia.org/wiki/Distributed_Version_Control_System" target="_blank">WikiPedia&#8217;s article on DVCS</a></li>
<li><a href="http://betterexplained.com/articles/intro-to-distributed-version-control-illustrated/" target="_blank">Intro to Distributed Versioning Control</a> &#8211; An excellent read</li>
<li><a href="http://mail.python.org/pipermail/python-dev/2009-March/087931.html" target="_blank">&#8220;While git has many fans, it also provokes strong antipathies.&#8221;</a> &#8211; Guido van Rossum</li>
<li><a href="http://www.youtube.com/watch?v=4XpnKHJAok8" target="_blank">&#8220;Subversion is the most pointless project ever started.&#8221;</a> &#8211; Linus Torvalds</li>
</ul>
]]></content:encoded>
			<wfw:commentRss>http://blog.curiasolutions.com/2009/06/versioning-control-systems-my-opinion-so-far/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
	</channel>
</rss>
