<?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>Casey A. McLaughlin &#187; Web Development</title>
	<atom:link href="http://www.caseymclaughlin.com/sections/web-development/feed/" rel="self" type="application/rss+xml" />
	<link>http://www.caseymclaughlin.com</link>
	<description>Online Portfolio</description>
	<lastBuildDate>Mon, 08 Mar 2010 21:38:28 +0000</lastBuildDate>
	<language>en</language>
	<sy:updatePeriod>hourly</sy:updatePeriod>
	<sy:updateFrequency>1</sy:updateFrequency>
	<generator>http://wordpress.org/?v=3.0.1</generator>
		<item>
		<title>Gone is Gone: The Case Against &#8220;Delete&#8221;</title>
		<link>http://www.caseymclaughlin.com/2010/03/gone-is-gone-the-case-against-delete/</link>
		<comments>http://www.caseymclaughlin.com/2010/03/gone-is-gone-the-case-against-delete/#comments</comments>
		<pubDate>Mon, 08 Mar 2010 21:35:36 +0000</pubDate>
		<dc:creator>McLaughlin Casey</dc:creator>
				<category><![CDATA[Web Development]]></category>

		<guid isPermaLink="false">http://www.caseymclaughlin.com/?p=195</guid>
		<description><![CDATA[		<span class="Z3988" title="ctx_ver=Z39.88-2004&amp;rft_val_fmt=info%3Aofi%2Ffmt%3Akev%3Amtx%3Adc&amp;rfr_id=info%3Asid%2Focoins.info%3Agenerator&amp;rft.title=Gone is Gone: The Case Against &#8220;Delete&#8221;&amp;rft.aulast=McLaughlin&amp;rft.aufirst=Casey&amp;rft.subject=Web Development&amp;rft.source=Casey A. McLaughlin&amp;rft.date=2010-03-08&amp;rft.type=&amp;rft.format=text&amp;rft.identifier=http://www.caseymclaughlin.com/2010/03/gone-is-gone-the-case-against-delete/&amp;rft.language=English"></span>
I love, love, love getting rid of crap.  Maybe that&#8217;s why I like to read Unclutterer, or think that the 100 Things Challenge is such a great idea.  In real-life, I&#8217;m a minimalist, but when it comes to programming web apps, I&#8217;ve become a digital pack-rat. I don&#8217;t care if my database grows into a giant data-monster, [...]]]></description>
			<content:encoded><![CDATA[		<span class="Z3988" title="ctx_ver=Z39.88-2004&amp;rft_val_fmt=info%3Aofi%2Ffmt%3Akev%3Amtx%3Adc&amp;rfr_id=info%3Asid%2Focoins.info%3Agenerator&amp;rft.title=Gone is Gone: The Case Against &#8220;Delete&#8221;&amp;rft.aulast=McLaughlin&amp;rft.aufirst=Casey&amp;rft.subject=Web Development&amp;rft.source=Casey A. McLaughlin&amp;rft.date=2010-03-08&amp;rft.type=&amp;rft.format=text&amp;rft.identifier=http://www.caseymclaughlin.com/2010/03/gone-is-gone-the-case-against-delete/&amp;rft.language=English"></span>
<p><a href="http://www.caseymclaughlin.com/wp-content/uploads/2010/03/a_bomb2.jpg"><img class="size-thumbnail wp-image-201 alignright" title="The Effects of Deleting Data" src="http://www.caseymclaughlin.com/wp-content/uploads/2010/03/a_bomb2-150x150.jpg" alt="The Effects of Deleting Data" width="150" height="150" /></a>I love, love, love getting rid of crap.  Maybe that&#8217;s why I like to read <a title="Unclutterer Blog - People who think up my ally" href="http://unclutterer.com/">Unclutterer</a>, or think that the <a title="100 Things Challenge - Great Concept!" href="http://www.guynameddave.com/100-thing-challenge.html">100 Things Challenge</a> is such a great idea.  In real-life, I&#8217;m a minimalist, but when it comes to programming web apps, I&#8217;ve become a digital pack-rat.</p>
<p>I don&#8217;t care if my database grows into a giant data-monster, ready to ravage my server&#8217;s hard disk and slow queries down to a crawl; I&#8217;ll deal with it.  The key here is to not allow any important history of data to get lost.</p>
<p><img title="More..." src="http://www.caseymclaughlin.com/wp-includes/js/tinymce/plugins/wordpress/img/trans.gif" alt="" /><span id="more-195"></span></p>
<h2>Why Deleting is Bad</h2>
<p>Here&#8217;s an example, to get us started:</p>
<p>Let&#8217;s say you have a simple online store, and you&#8217;re selling <a title="The Definition of Importance" href="http://stupid.com">important consumer items</a> to important people.  You have a database that keeps track of your customers, your products, and your transactions.  It may look something like this:</p>
<p><a href="http://www.caseymclaughlin.com/wp-content/uploads/2010/03/simple-db.png"><img title="Database Example" src="http://www.caseymclaughlin.com/wp-content/uploads/2010/03/simple-db.png" alt="Database Example" width="625" height="136" /></a>So, you&#8217;ve been in business for a while, and you have a bunch of customers in the system, quite a few products, and a lengthy transaction history.  Recently, you&#8217;ve decided to stop selling your <a title="An Important Asset for every Household" href="http://www.stupid.com/fun/D-MOLD.html">Jell-O Brain Mold</a> and you&#8217;re ready to remove it from your online store.  Common sense says that you would go to your database, find the record referring to your product, and delete it.</p>
<p>But wait!  If you do this, then the entire history of transactions linked to those products become orphaned (the<em>products_id</em> field will point to a non-existent record).  So, if you want to go back in at a future date to see the history of how many of those items that you sold, you will be in trouble.  You haven&#8217;t just deleted the item from your website per se, you&#8217;ve actually deleted the record of the item ever existing in the first place.</p>
<p>And, you don&#8217;t want to do that.</p>
<h2>Flag It!</h2>
<p>The useful alternative would be to add a flag to the data, so that instead of deleting the record, you could mark it as inactive.  This way, it wouldn&#8217;t appear on your website, but it would still exist in the database for future reference.</p>
<p>Your tables will now look something like:</p>
<p><a href="http://www.caseymclaughlin.com/wp-content/uploads/2010/03/simple-db-2.png"><img title="DB with Flags" src="http://www.caseymclaughlin.com/wp-content/uploads/2010/03/simple-db-2.png" alt="DB with Flags" width="625" height="151" /></a>Now you can tell your application to only show the products with `active` = &#8220;TRUE&#8221; on your website, but refer to products with `active` = &#8220;FALSE&#8221; when viewing older transaction histories.</p>
<p>This works pretty well, and if you program your User Interface correctly, your website users should never actually be deleting records; instead they should be marking them inactive so they &#8220;go away&#8221; but never truly die.</p>
<h2>Audit History</h2>
<p>Okay, now it gets a bit more complex.  In some cases, you don&#8217;t just want to mark a record &#8220;active&#8221; or &#8220;inactive&#8221; (that is, give it the ability to have two states), you also want to keep a complete revision history of that record, including what change was made, who made it, and when.  This way you can see what a record looked like at a particular moment in time.</p>
<p>This is not a new concept.  Think about your bank account.  It&#8217;s pretty much a bunch of numbers stored in a database somewhere.  Do you think that your checking account balance is simply a single field in a single record stored in a database?  Heck no!  Every single debit and credit is stored as a separate transaction, showing a complete history of changes to the account.</p>
<p>There are many ways to accomplish this, really, and for me to put up a tutorial, would be to <a title="Stack Overflow Thread on the Subject" href="http://stackoverflow.com/questions/323065/how-to-version-control-a-record-in-a-database">reinvent </a><a title="Arnold's (complex) solution" href="http://www.adaniels.nl/articles/versioning-mysql-data/">the</a> <a title="Another Tutorial for MS SQL" href="http://www.codeproject.com/KB/architecture/LisRecordVersioning.aspx">wheel</a>.  The key is to think about which data you want to have complete histories on, and which data is not expendable.  Design your application around these constraints.</p>
<h2>Sometimes you Just Have to Kill It</h2>
<p>I wrote an application that contains patient records.  When an administrator clicks &#8220;Delete&#8221;, the patient doesn&#8217;t really go away; all of the information remains in the database.  Sometimes, however, we <strong>must </strong>delete the record permanently. Usually this is because a patient opts out of the program and requests to be removed from the study.  I use a special word for this kind of delete: <strong>Purge</strong>.</p>
<p>But, before the record is purged, the user has to click a checkbox, and go through two <strong>&#8220;Are You Sure?&#8221;</strong> screens informing him or her that not only will the patient record be deleted, but also the complete history of that record, and all data associated with it.  I try to have the application show the data in question on the screen the user has to click yes to.</p>
<p>&#8230;well, that&#8217;s enough for now.  Hope you&#8217;re thinking about data consistency, so you too can become a digital pack rat.</p>
]]></content:encoded>
			<wfw:commentRss>http://www.caseymclaughlin.com/2010/03/gone-is-gone-the-case-against-delete/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>Biting the Distributed Development Bullet</title>
		<link>http://www.caseymclaughlin.com/2009/12/make-codeigniter-svn-friendly/</link>
		<comments>http://www.caseymclaughlin.com/2009/12/make-codeigniter-svn-friendly/#comments</comments>
		<pubDate>Mon, 21 Dec 2009 20:15:46 +0000</pubDate>
		<dc:creator>McLaughlin Casey</dc:creator>
				<category><![CDATA[Web Development]]></category>
		<category><![CDATA[management]]></category>
		<category><![CDATA[php]]></category>
		<category><![CDATA[project management]]></category>
		<category><![CDATA[projects]]></category>
		<category><![CDATA[subversion]]></category>
		<category><![CDATA[version control]]></category>

		<guid isPermaLink="false">http://www.caseymclaughlin.com/?p=129</guid>
		<description><![CDATA[You're a dummy for not using version control.  If I can do it, anybody can.]]></description>
			<content:encoded><![CDATA[		<span class="Z3988" title="ctx_ver=Z39.88-2004&amp;rft_val_fmt=info%3Aofi%2Ffmt%3Akev%3Amtx%3Adc&amp;rfr_id=info%3Asid%2Focoins.info%3Agenerator&amp;rft.title=Biting the Distributed Development Bullet&amp;rft.aulast=McLaughlin&amp;rft.aufirst=Casey&amp;rft.subject=Web Development&amp;rft.source=Casey A. McLaughlin&amp;rft.date=2009-12-21&amp;rft.type=&amp;rft.format=text&amp;rft.identifier=http://www.caseymclaughlin.com/2009/12/make-codeigniter-svn-friendly/&amp;rft.language=English"></span>
<p><a href="http://www.caseymclaughlin.com/wp-content/uploads/2009/12/programmer.jpg"><img class="alignright size-thumbnail wp-image-131" title="programmer" src="http://www.caseymclaughlin.com/wp-content/uploads/2009/12/programmer-150x150.jpg" alt="Will Code for Food" width="150" height="150" /></a>Like most development n00bs, after I started using version control, I never looked back.  And <a title="Stack Overflow Thread" href="http://spedr.com/k1z1">neither should you</a>.</p>
<p>It required me to change the whole way that my office operated, though.  See, the way that we were doing things before seemed so simple; we just edited files and shouted at each other if we collided in our editing.  As it turns out, this was a bad thing, and there have been better alternatives around way longer than I&#8217;ve even been in this business.</p>
<p>But like most crotchety, over-egotistical developer-types it took a while for me to come around.  Here&#8217;s a brief history of my flirtations and eventual commitment to version control.</p>
<h3><span id="more-129"></span>Our Blissful, Youthful Ignorance</h3>
<p>This wasn&#8217;t really that long ago&#8230;</p>
<div id="attachment_139" class="wp-caption aligncenter" style="width: 310px"><a href="http://www.caseymclaughlin.com/wp-content/uploads/2009/12/Workflow-SSH.png"><img class="size-medium wp-image-139" title="The Old Workflow" src="http://www.caseymclaughlin.com/wp-content/uploads/2009/12/Workflow-SSH-300x237.png" alt="The old, SSH-based Workflow" width="300" height="237" /></a><p class="wp-caption-text">1..2..3... Easy Peasy Japanesey.. and stupid.</p></div>
<p>We would all work directly on the files directly on our development server.  We would sit down, SSH into the server using our favorite editor and code away, never thinking of change management, reverting back to old versions, or who would be responsible for what.  Conflict resolution seemed easy&#8230; if I was editing a file named<em> welcome.php</em>, I would simply shout across the room, &#8220;Hey Mike&#8211;don&#8217;t work on <em>welcome.php</em> right now, or I&#8217;ll have to beat you!&#8221;.</p>
<p>Our revision control system consisted of me creating and storing weekly backups of the entire database and file system.  If we made a change that killed the system, we&#8217;d simply either lose that week&#8217;s work or spend hours debugging it.  It was all gloriously simple.</p>
<h3>Version Control Puberty &#8212; The Angsty Years</h3>
<p>One day, I decided to implement Subversion, mainly because other developers were sneering at me and telling me I was a dummy.  And while I didn&#8217;t want to be a dummy, the whole thing seemed overly complicated.  Why would I want to increase complication when everything was working so simple?  I also had some fundamental issues with this new version control idea:</p>
<ul>
<li>First off, everybody has to have to get their own copy of the entire file set.  Bad!  Doom! Insecure! I have to trust my developers with the codebase? ::Gasp::</li>
<li>Secondly, what do you mean that developers can&#8217;t see their changes on the server immediately upon commit without some hacked hook system?  This means they&#8217;ll have to run <a title="Wikipedia LAMP" href="http://en.wikipedia.org/wiki/LAMP_%28software_bundle%29">LAMP</a> on their local boxes, too!  Geez.. now we have to have local dev environment setup procedures, too..</li>
<li>Finally, SVN requires management of its own.  You know.. backing up, securing access, and generally managing.  More overhead and more complication!</li>
</ul>
<p>Grudgingly, in order to remove the &#8220;dummy&#8221; label from my operations, I implemented Subversion.  It was difficult at first to understand how it would fit easily into our procedures, but with some <a title="Subversion Book - Basic Work Cycle (a very helpful chapter)" href="http://svnbook.red-bean.com/en/1.5/svn.tour.cycle.html">very helpful free online material</a>, we started using it.</p>
<p>And sure, I <em>did</em> have to write procedures for setting up local dev environments AND install a subversion clients AND manage a <a title="Hands Down, VisualSVN is the easist-to-use SVN Server for Windows" href="http://www.visualsvn.com/server/">repository server</a>, but you know what?</p>
<p>All of that stuff was way worth it.</p>
<h3>All Grown-up Now</h3>
<p>Let me repeat that: whatever frustrations version control causes, they are all worth it.</p>
<p>After being SVN disciples for a while now, our development procedures have matured.  Here&#8217;s how we do things:</p>
<div id="attachment_137" class="wp-caption aligncenter" style="width: 310px"><a href="http://www.caseymclaughlin.com/wp-content/uploads/2009/12/Workflow-SVN.png"><img class="size-medium wp-image-137" title="SVN Workflow" src="http://www.caseymclaughlin.com/wp-content/uploads/2009/12/Workflow-SVN-300x242.png" alt="The way we do things now" width="300" height="242" /></a><p class="wp-caption-text">The way we do things now.. Much better!</p></div>
<p>I&#8217;ve since tried other VCS systems like <a title="Mercurial Homepage" href="http://mercurial.selenic.com/">Mercurial</a>, <a title="Git Homepage" href="http://git-scm.com/">Git</a>, and <a title="Bazaar Homepage" href="http://bazaar.canonical.com/en/">Bazaar</a>, but have not found any compelling reason to switch.   Either way, using version control has saved our sorry butts on more than one occasion and has made life easier in about 1,001 ways.</p>
<p>As a bonus, it has also forced us to use better procedures and generate better development documentation, which is amazing, because we&#8217;re pretty lazy.  Now if only we could find a tool to better force us to behave when it comes to unit testing&#8230;</p>
<p>Well, that&#8217;s my story.  If I&#8217;ve inspired you to use VCS, hooray!  <a title="Awesome SVN Primer I found on the front page of Google search" href="http://www.snipe.net/2009/03/getting-started-with-subversion/">Here&#8217;s a helpful place to get started</a>.  If not, you&#8217;re a dummy.</p>
]]></content:encoded>
			<wfw:commentRss>http://www.caseymclaughlin.com/2009/12/make-codeigniter-svn-friendly/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>Fixed GeSHI for CodeIgniter Link</title>
		<link>http://www.caseymclaughlin.com/2009/09/fixed-geshi-for-codeigniter-link/</link>
		<comments>http://www.caseymclaughlin.com/2009/09/fixed-geshi-for-codeigniter-link/#comments</comments>
		<pubDate>Wed, 23 Sep 2009 16:11:54 +0000</pubDate>
		<dc:creator>McLaughlin Casey</dc:creator>
				<category><![CDATA[General]]></category>
		<category><![CDATA[Web Development]]></category>
		<category><![CDATA[announcement]]></category>
		<category><![CDATA[ci]]></category>
		<category><![CDATA[geshi]]></category>
		<category><![CDATA[meta]]></category>
		<category><![CDATA[projects]]></category>
		<category><![CDATA[this site]]></category>

		<guid isPermaLink="false">http://www.caseymclaughlin.com/?p=119</guid>
		<description><![CDATA[		<span class="Z3988" title="ctx_ver=Z39.88-2004&amp;rft_val_fmt=info%3Aofi%2Ffmt%3Akev%3Amtx%3Adc&amp;rfr_id=info%3Asid%2Focoins.info%3Agenerator&amp;rft.title=Fixed GeSHI for CodeIgniter Link&amp;rft.aulast=McLaughlin&amp;rft.aufirst=Casey&amp;rft.subject=General&amp;rft.subject=Web Development&amp;rft.source=Casey A. McLaughlin&amp;rft.date=2009-09-23&amp;rft.type=&amp;rft.format=text&amp;rft.identifier=http://www.caseymclaughlin.com/2009/09/fixed-geshi-for-codeigniter-link/&amp;rft.language=English"></span>
Thanks all who notified me that the GesHi for CodeIgniter download link was serving up a big, ugly 404 page.  I plan on releasing a new version soon, but in the meantime, I&#8217;ve fixed the link. Any questions, comments, or suggestions are always welcome.  Also, if anybody has tried this plugin on CI 1.7.2, let [...]]]></description>
			<content:encoded><![CDATA[		<span class="Z3988" title="ctx_ver=Z39.88-2004&amp;rft_val_fmt=info%3Aofi%2Ffmt%3Akev%3Amtx%3Adc&amp;rfr_id=info%3Asid%2Focoins.info%3Agenerator&amp;rft.title=Fixed GeSHI for CodeIgniter Link&amp;rft.aulast=McLaughlin&amp;rft.aufirst=Casey&amp;rft.subject=General&amp;rft.subject=Web Development&amp;rft.source=Casey A. McLaughlin&amp;rft.date=2009-09-23&amp;rft.type=&amp;rft.format=text&amp;rft.identifier=http://www.caseymclaughlin.com/2009/09/fixed-geshi-for-codeigniter-link/&amp;rft.language=English"></span>
<p><a href="http://www.caseymclaughlin.com/wp-content/uploads/2009/09/funny-car-repairs-engine.jpg"><img class="alignright size-thumbnail wp-image-121" title="funny-car-repairs-engine" src="http://www.caseymclaughlin.com/wp-content/uploads/2009/09/funny-car-repairs-engine-150x150.jpg" alt="funny-car-repairs-engine" width="150" height="150" /></a>Thanks all who notified me that the GesHi for CodeIgniter download link was serving up a big, ugly <strong>404</strong> page.  I plan on releasing a new version soon, but in the meantime, I&#8217;ve fixed the link.</p>
<p>Any questions, comments, or suggestions are always welcome.  Also, if anybody has tried this plugin on CI 1.7.2, let me know how it works.</p>
]]></content:encoded>
			<wfw:commentRss>http://www.caseymclaughlin.com/2009/09/fixed-geshi-for-codeigniter-link/feed/</wfw:commentRss>
		<slash:comments>1</slash:comments>
		</item>
		<item>
		<title>CodeIgniter Preflight Checker</title>
		<link>http://www.caseymclaughlin.com/2009/06/codeigniter-preflight-checker/</link>
		<comments>http://www.caseymclaughlin.com/2009/06/codeigniter-preflight-checker/#comments</comments>
		<pubDate>Fri, 26 Jun 2009 18:16:07 +0000</pubDate>
		<dc:creator>McLaughlin Casey</dc:creator>
				<category><![CDATA[Web Development]]></category>
		<category><![CDATA[codeigniter]]></category>
		<category><![CDATA[php]]></category>
		<category><![CDATA[portability]]></category>
		<category><![CDATA[portable]]></category>

		<guid isPermaLink="false">http://www.caseymclaughlin.com/?p=84</guid>
		<description><![CDATA[This post builds on last week's discussion of how to make an application portable by demonstrating how to implement a "pre-flight" check before your application runs.]]></description>
			<content:encoded><![CDATA[		<span class="Z3988" title="ctx_ver=Z39.88-2004&amp;rft_val_fmt=info%3Aofi%2Ffmt%3Akev%3Amtx%3Adc&amp;rfr_id=info%3Asid%2Focoins.info%3Agenerator&amp;rft.title=CodeIgniter Preflight Checker&amp;rft.aulast=McLaughlin&amp;rft.aufirst=Casey&amp;rft.subject=Web Development&amp;rft.source=Casey A. McLaughlin&amp;rft.date=2009-06-26&amp;rft.type=&amp;rft.format=text&amp;rft.identifier=http://www.caseymclaughlin.com/2009/06/codeigniter-preflight-checker/&amp;rft.language=English"></span>
<p><a href="http://www.flickr.com/photos/mike_miley/2996159793/"><img class="alignright size-thumbnail wp-image-86" title="preflight_check" src="http://www.caseymclaughlin.com/wp-content/uploads/2009/06/preflight_check-150x150.jpg" alt="preflight_check" width="150" height="150" /></a>In <a title="Last Post: Making PHP Applications Portable" href="http://www.caseymclaughlin.com/2009/06/making-php-applications-portable/">my last post</a>, I discussed do&#8217;s and don&#8217;ts for making a PHP application portable.  This week, I will show you how to create a preflight checker in the <a title="CodeIgniter Homepage" href="http://www.codeigniter.com">CodeIgniter PHP framework</a> to ensure that an environment will support your application.</p>
<p><span id="more-84"></span></p>
<p><em>Note: This article assumes that you already have a working CodeIgniter application setup.  If you are interested in getting started with CodeIgniter, there are some <a title="CodeIgniter Video Tutorials" href="http://codeigniter.com/tutorials/">really</a> <a title="CodeIgniter Official User Guide" href="http://codeigniter.com/user_guide/">great</a> <a title="Net Tuts &quot;Getting started with CodeIgniter&quot; Article" href="http://net.tutsplus.com/videos/screencasts/easy-development-with-codeigniter/">tutorials </a>on the web.  If you&#8217;re not using CodeIgniter, skip to the &#8220;What to Check&#8221; section below for a general idea of how to apply this to PHP apps in general.</em></p>
<h2>Setting up the Hook</h2>
<p>The best way to setup a preflight checker is to create a hook.  Make sure that you <a title="CodeIgniter Hooks Documentation" href="http://codeigniter.com/user_guide/general/hooks.html">enable hooks</a> in your main configuration file before getting started.  We&#8217;ll be setting up a pre-system hook, so add the following code to your <em>application/config/hooks.php</em> file:</p>

<div class="wp_syntax"><div class="code"><pre class="php" style="font-family:monospace;"><span style="color: #666666; font-style: italic;">/*Preflight Check Hook*/</span>
<span style="color: #000088;">$hook</span><span style="color: #009900;">&#91;</span><span style="color: #0000ff;">'pre_system'</span><span style="color: #009900;">&#93;</span> <span style="color: #339933;">=</span> <span style="color: #990000;">array</span><span style="color: #009900;">&#40;</span><span style="color: #0000ff;">'class'</span> <span style="color: #339933;">=&gt;</span> <span style="color: #0000ff;">''</span><span style="color: #339933;">,</span>
<span style="color: #0000ff;">'function'</span> <span style="color: #339933;">=&gt;</span> <span style="color: #0000ff;">'preflight_check'</span><span style="color: #339933;">,</span>
<span style="color: #0000ff;">'filename'</span> <span style="color: #339933;">=&gt;</span> <span style="color: #0000ff;">'preflight_check_hook.php'</span><span style="color: #339933;">,</span>
<span style="color: #0000ff;">'filepath'</span> <span style="color: #339933;">=&gt;</span> <span style="color: #0000ff;">'hooks'</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span></pre></div></div>

<p>This will tell CodeIgniter to run this hook every time the system starts.  As you can infer from the above code, we&#8217;ll now have to create a file in the <em>application/hooks/</em> folder (if it doesn&#8217;t exist, create it) named <strong>preflight_check_hook.php</strong> and add a function to that file named <strong>preflight_check()</strong>:</p>

<div class="wp_syntax"><div class="code"><pre class="php" style="font-family:monospace;"><span style="color: #009933; font-style: italic;">/**
 * Preflight Check checks all of the prerequistes before the first
 * pageview for each session.
 *
 * If a problem is found, the program halts execution.
 */</span>
<span style="color: #000000; font-weight: bold;">function</span> preflight_check<span style="color: #009900;">&#40;</span><span style="color: #009900;">&#41;</span>
<span style="color: #009900;">&#123;</span>
  <span style="color: #666666; font-style: italic;">//all your checks go here</span>
<span style="color: #009900;">&#125;</span></pre></div></div>

<p>Now we&#8217;re ready to add checks so we can test the server environment.</p>
<h2>What to Check</h2>
<p>Technically, you can check whatever you want here, but I tend to check the things that are likely to be different from one server environment to the next.  Some of the more important ones are:</p>
<ul>
<li>Whether the version of PHP or CodeIgniter is new enough to support your application.  You can check this by using the <em>phpversion()</em> function</li>
<li>Necessary PHP Extensions installed?  You can check this by using the <em>function_exists()</em> function.</li>
<li>PHP.INI variables set correctly?  You can check this by using the <em>ini_get()<strong> </strong></em>function.</li>
<li>Whether certain folders are really writable on the server or not</li>
<li>Whether files exist or not.</li>
</ul>
<p>Of course, you can check for most anything you want to in the preflight check, and even try to tweak the environment a little using <em>ini_set()</em>.  What you <span style="text-decoration: underline;">can&#8217;t</span> do is access most of the functionality in CodeIgniter, since this hook will run before most of your libraries are loaded.  There are a few useful CodeIgniter variables and functions that you will have access to, though:</p>
<ul>
<li><em>show_error()</em> &#8212; For printing out errors (non CodeIgniter users should use the <em>die()</em> function)</li>
<li><em>is_really_writable()</em> &#8212; For testing if a file is writable or not (non CodeIgniter users should use the <em>is_readable()</em> function)</li>
<li><em>$application_folder </em>and <em>$system_folder</em></li>
</ul>
<p>Rather than bore you with lengthy explanations of each type of check, it will be easier to simply show you an example of how I typically do it.  Below is an example preflight check hook that I ripped almost verbatim from one of my applications:</p>

<div class="wp_syntax"><div class="code"><pre class="php" style="font-family:monospace;"><span style="color: #000000; font-weight: bold;">function</span> preflight_check<span style="color: #009900;">&#40;</span><span style="color: #009900;">&#41;</span>
<span style="color: #009900;">&#123;</span>
	<span style="color: #000000; font-weight: bold;">global</span> <span style="color: #000088;">$application_folder</span><span style="color: #339933;">;</span>
	<span style="color: #000000; font-weight: bold;">global</span> <span style="color: #000088;">$system_folder</span><span style="color: #339933;">;</span>
&nbsp;
	<span style="color: #666666; font-style: italic;">//Ensure proper version of PHP</span>
	<span style="color: #b1b100;">if</span> <span style="color: #009900;">&#40;</span><span style="color: #990000;">phpversion</span><span style="color: #009900;">&#40;</span><span style="color: #009900;">&#41;</span> <span style="color: #339933;">&lt;</span> <span style="color: #0000ff;">'5.1'</span><span style="color: #009900;">&#41;</span>
	  show_error<span style="color: #009900;">&#40;</span><span style="color: #0000ff;">&quot;This application requires at least PHP 5.1 to run properly!&quot;</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span>
&nbsp;
	<span style="color: #666666; font-style: italic;">//Ensure SQLITE extension is installed</span>
	<span style="color: #b1b100;">if</span> <span style="color: #009900;">&#40;</span> <span style="color: #339933;">!</span> <span style="color: #990000;">function_exists</span><span style="color: #009900;">&#40;</span><span style="color: #0000ff;">'mysql_connect'</span><span style="color: #009900;">&#41;</span><span style="color: #009900;">&#41;</span>
	  show_error<span style="color: #009900;">&#40;</span><span style="color: #0000ff;">&quot;The sqlite PHP extension is required to run this application!  Refer to &lt;a href=&quot;</span>http<span style="color: #339933;">:</span><span style="color: #666666; font-style: italic;">//us.php.net/manual/en/book.sqlite.php&quot;&gt;the PHP documentation for this extension&lt;/a&gt; for more information.&quot;);</span>
&nbsp;
	<span style="color: #666666; font-style: italic;">//Ensure GD2 is is installed</span>
	<span style="color: #b1b100;">if</span> <span style="color: #009900;">&#40;</span> <span style="color: #339933;">!</span> <span style="color: #990000;">function_exists</span><span style="color: #009900;">&#40;</span><span style="color: #0000ff;">&quot;gd_info&quot;</span><span style="color: #009900;">&#41;</span><span style="color: #009900;">&#41;</span>
		show_error<span style="color: #009900;">&#40;</span><span style="color: #0000ff;">&quot;The PHP GD2 image library is required to run this software.  Check with your server administrator or hosting provider, or &lt;a title=&quot;</span>Installation Instructions <span style="color: #b1b100;">for</span> GD2<span style="color: #339933;">&gt;</span>refer to the PHP documentation <span style="color: #b1b100;">for</span> installation instructions<span style="color: #339933;">&lt;/</span>a<span style="color: #339933;">&gt;.&amp;</span>quot<span style="color: #339933;">;</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span> 	<span style="color: #666666; font-style: italic;">//Check if the system cache directory is writable 	if ( ! is_really_writable($system_folder . &quot; href=&quot;http://us2.php.net/manual/en/intro.image.php&quot;&gt;&lt;/a&gt;&lt;a title=&quot;JSON extension installation instructions&quot; href=&quot;http://us3.php.net/manual/en/book.json.php&quot;&gt;manually install it&lt;/a&gt;.)&quot;);</span>
&nbsp;
	<span style="color: #666666; font-style: italic;">//Check memory limit and attempt to set it to what this application needs</span>
	<span style="color: #b1b100;">if</span> <span style="color: #009900;">&#40;</span>get_bytes<span style="color: #009900;">&#40;</span><span style="color: #990000;">ini_get</span><span style="color: #009900;">&#40;</span><span style="color: #0000ff;">'php_value memory_limit'</span><span style="color: #009900;">&#41;</span><span style="color: #009900;">&#41;</span> <span style="color: #339933;">&lt;</span> <span style="color: #cc66cc;">33554432</span><span style="color: #009900;">&#41;</span>
	<span style="color: #009900;">&#123;</span>
		<span style="color: #b1b100;">if</span><span style="color: #009900;">&#40;</span> <span style="color: #339933;">!</span> <span style="color: #990000;">ini_set</span><span style="color: #009900;">&#40;</span><span style="color: #0000ff;">'php_value_memory_limit'</span><span style="color: #339933;">,</span> <span style="color: #0000ff;">'32m'</span><span style="color: #009900;">&#41;</span>
			show_error<span style="color: #009900;">&#40;</span><span style="color: #0000ff;">&quot;The memory limit in your PHP.INI file must be set to at least 32 megabytes (32m).&quot;</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span>
	<span style="color: #009900;">&#125;</span>
<span style="color: #009900;">&#125;</span></pre></div></div>

<p>You can copy this code into your <em>application/hooks/preflight_check_hook.php</em> and try it out.  Basically, it will halt execution (via the <em>show_error()</em> function) and show an error on the screen that alerts the user to something not being setup correctly if any one of these checks fails.</p>
<h2>What&#8217;s the Advantage over just letting the program fail on its own?</h2>
<p>Technically, nothing; your preflight checker will check things that would normally cause your application to fail at some point in the execution anyway.  This is simply a way for you to alert your admins/developers/users that they need to fix their environment right off the bat.  Plus it&#8217;s more graceful and clean than PHP error messages.</p>
<p>Practically, however, this may save a lot of headaches, especially if you distribute the application or expect people to download a copy onto their own desktops for development.  Oh, and if your users/developers/admins have PHP error reporting turned off on their system, then a carefully coded preflight checker will save them from getting a mysterious blank screen when the environment doesn&#8217;t match the requirements.</p>
<h2>Keeping this thing from running for every page view</h2>
<p>You probably won&#8217;t want this script to run every time a  page is loaded on your site, especially if the script includes file operations, lest your server take a performance hit.  Typically, you will want to run this preflight checker only once per session.  By adding a few lines of code to the top and bottom of this function, you can make it run only once per session (unless it fails).</p>

<div class="wp_syntax"><div class="code"><pre class="php" style="font-family:monospace;"><span style="color: #000000; font-weight: bold;">function</span> preflight_check<span style="color: #009900;">&#40;</span><span style="color: #009900;">&#41;</span>
<span style="color: #009900;">&#123;</span>
	<span style="color: #666666; font-style: italic;">//See if this has already been run during the current session</span>
	<span style="color: #990000;">session_start</span><span style="color: #009900;">&#40;</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span>
	<span style="color: #b1b100;">if</span> <span style="color: #009900;">&#40;</span><span style="color: #990000;">isset</span><span style="color: #009900;">&#40;</span><span style="color: #000088;">$_SESSION</span><span style="color: #009900;">&#91;</span><span style="color: #0000ff;">'preflight_check_pass'</span><span style="color: #009900;">&#93;</span><span style="color: #009900;">&#41;</span> <span style="color: #339933;">&amp;</span>amp<span style="color: #339933;">;&amp;</span>amp<span style="color: #339933;">;</span> <span style="color: #000088;">$_SESSION</span><span style="color: #009900;">&#91;</span><span style="color: #0000ff;">'preflight_check_pass'</span><span style="color: #009900;">&#93;</span> <span style="color: #339933;">==</span> <span style="color: #0000ff;">'pass'</span><span style="color: #009900;">&#41;</span>
		<span style="color: #b1b100;">return</span> <span style="color: #009900; font-weight: bold;">TRUE</span><span style="color: #339933;">;</span>
&nbsp;
	<span style="color: #000000; font-weight: bold;">global</span> <span style="color: #000088;">$application_folder</span><span style="color: #339933;">;</span>
	<span style="color: #000000; font-weight: bold;">global</span> <span style="color: #000088;">$system_folder</span><span style="color: #339933;">;</span>
&nbsp;
	<span style="color: #666666; font-style: italic;">//Ensure proper version of PHP</span>
	<span style="color: #b1b100;">if</span> <span style="color: #009900;">&#40;</span><span style="color: #990000;">phpversion</span><span style="color: #009900;">&#40;</span><span style="color: #009900;">&#41;</span> <span style="color: #339933;">&lt;</span> <span style="color: #0000ff;">'5.1'</span><span style="color: #009900;">&#41;</span>
	  show_error<span style="color: #009900;">&#40;</span><span style="color: #0000ff;">&quot;This application requires at least PHP 5.1 to run properly!&quot;</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span>
&nbsp;
	<span style="color: #666666; font-style: italic;">// All of the</span>
	<span style="color: #666666; font-style: italic;">// rest of your checks</span>
	<span style="color: #666666; font-style: italic;">// go in here...</span>
&nbsp;
	<span style="color: #666666; font-style: italic;">// ...</span>
&nbsp;
	<span style="color: #666666; font-style: italic;">//Woo hoo.. the check passed!</span>
	<span style="color: #000088;">$_SESSION</span><span style="color: #009900;">&#91;</span><span style="color: #0000ff;">'preflight_check_pass'</span><span style="color: #009900;">&#93;</span> <span style="color: #339933;">=</span> <span style="color: #0000ff;">'pass'</span><span style="color: #339933;">;</span>
<span style="color: #009900;">&#125;</span></pre></div></div>

<p>You&#8217;ll notice that this example uses PHP&#8217;s built-in session system rather than CodeIgniter&#8217;s session library.  This is because CodeIgniter&#8217;s will not have been loaded yet when this hook runs, and because we&#8217;re not storing any particularly sensitive information in this session, so we don&#8217;t have to take advantage of CI&#8217;s advanced session security features.</p>
<p>Well, that&#8217;s about it for now.  If you have any comments or see any issues with the examples in today&#8217;s post, feel free to share by leaving a comment.</p>
]]></content:encoded>
			<wfw:commentRss>http://www.caseymclaughlin.com/2009/06/codeigniter-preflight-checker/feed/</wfw:commentRss>
		<slash:comments>2</slash:comments>
		</item>
		<item>
		<title>Making PHP Applications Portable</title>
		<link>http://www.caseymclaughlin.com/2009/06/making-php-applications-portable/</link>
		<comments>http://www.caseymclaughlin.com/2009/06/making-php-applications-portable/#comments</comments>
		<pubDate>Tue, 23 Jun 2009 19:27:27 +0000</pubDate>
		<dc:creator>McLaughlin Casey</dc:creator>
				<category><![CDATA[Web Development]]></category>
		<category><![CDATA[codeigniter]]></category>
		<category><![CDATA[php]]></category>
		<category><![CDATA[portability]]></category>
		<category><![CDATA[portable]]></category>

		<guid isPermaLink="false">http://www.caseymclaughlin.com/?p=56</guid>
		<description><![CDATA[This post is the first in a two-part series explaining the virtues of portability when designing PHP applications, and provides a checklist of do's and don'ts (mostly don'ts) for ensuring that your application will run in any PHP server environment.]]></description>
			<content:encoded><![CDATA[		<span class="Z3988" title="ctx_ver=Z39.88-2004&amp;rft_val_fmt=info%3Aofi%2Ffmt%3Akev%3Amtx%3Adc&amp;rfr_id=info%3Asid%2Focoins.info%3Agenerator&amp;rft.title=Making PHP Applications Portable&amp;rft.aulast=McLaughlin&amp;rft.aufirst=Casey&amp;rft.subject=Web Development&amp;rft.source=Casey A. McLaughlin&amp;rft.date=2009-06-23&amp;rft.type=&amp;rft.format=text&amp;rft.identifier=http://www.caseymclaughlin.com/2009/06/making-php-applications-portable/&amp;rft.language=English"></span>
<p><img class="alignright size-full wp-image-62" title="portable_computer_guy" src="http://www.caseymclaughlin.com/wp-content/uploads/2009/06/portable_computer_guy.jpg" alt="portable_computer_guy" width="225" height="158" />&#8220;Portable&#8221; means that your application is able to run on any PHP webserver with minimal fuss.</p>
<p>Ensuring portability is an absolute must for programs that are packaged and distributed for public consumption, like WordPress or MediaWiki.  But, you should aim for portability in all of your applications.  The ideal is a one or two-step installation process and only two or three system requirements:  1. <em>Webserver (IIS/Apache/etc)</em>, 2. <em>PHP v5.X+</em>, 3. <em>MySQL 4+</em>.</p>
<p><span id="more-56"></span></p>
<h2>Why Make it Portable?</h2>
<p>For one, it keeps things simple.  The concept of portability meshes well with the <a title="Keep It Simple, Stupid." href="http://en.wikipedia.org/wiki/KISS_principle">KISS</a> concept that I value so highly.</p>
<p>Secondly, it makes deploying your application to a new server simple if the need ever arises.</p>
<p>Thirdly, it enables you to easily distribute your package to the world if that is your intention.  I, for one, simply hate installing web applications on my server with long lists of prerequisites and a lengthy install process, especially if one of those steps is to install <em>another</em> set of libraries (e.g. from PEAR) before installing the application itself.</p>
<p>Finally, portability makes distributed development easier.  Many applications are developed on Windows but run on Linux.  Portable applications eliminate the headaches involved in trying to get the to two environments (development &amp; production) to match.</p>
<h2>How to Make Your Application Portable</h2>
<p>A truly portable application will work out-of-the-box, and require minimal configuration.  It will also automatically check the environment for specific configuration or prerequisites that <em>do </em>happen to be required.  Below are five things to try and avoid, and one thing to try and do to ensure your application is portable:</p>
<h3>1. Don&#8217;t rely on exotic PHP extensions</h3>
<p>I&#8217;m talking about extensions that are not bundled with PHP or found pre-installed on most hosting providers.  If you do require a specific PHP extension that is not bundled with PHP, you should check for the existence of the extension at the beginning of the program&#8217;s execution.</p>
<p>For example, many of my applications require JSON libraries, which some versions of PHP5 do not include, so I check for the existence of the <em>json_encode</em> function when the application first runs.</p>
<h3>2. Don&#8217;t rely on separate installation of PEAR Libraries</h3>
<p>Don&#8217;t get me wrong; PEAR is great.  But, you shouldn&#8217;t predicate the successful execution of your application on your users installing external libraries.  Most PEAR libraries are dependent on other PEAR libraries, and require the user to figure out and install multiple packages.  Your application should not require this pre-setup, and it should run with all of the required libraries and classes included.</p>
<h3>3. Don&#8217;t rely on command-line configuration</h3>
<p>If you want your application to be truly portable, you have to cater to the crowd who has only FTP access to their servers.  Yes, these folks are few-and-far-between, but coding with them in mind ensures that your application setup procedure is as simple as possible.  This means that your installation instructions should be completely void of anything that says <em>&#8220;login to the command line and run xyz command&#8221;</em>.</p>
<p>Definitely don&#8217;t rely on the user to have administrative privileges on the server.  This means to try to avoide the necessity for external software or packages (although this is unavoidable sometimes).</p>
<h3>4. Don&#8217;t rely on the program to be run on a specific operating system</h3>
<p>This one is  a stretch for some people, but truly portable applications can run on Windows, Linux, or MAC servers without fuss.  I make all my applications portable enough to run on any OS  simply because we use SVN for development, which requires my developers to download fully working copies of the project to their desktops.  Since my developers use Windows &amp; Mac, and my servers run Linux, my applications have to be cross-platform compatible.</p>
<h3>5. Don&#8217;t rely on PHP.INI customizations</h3>
<p>A lot of PHP applications I have tried won&#8217;t work with magic quotes (or vice versa), or won&#8217;t work without the user increasing the amount of memory or upload file size  for PHP.   Before making these nice-to-have&#8217;s required, ask yourself if you can find a different way to solve the problem.</p>
<p>For situations where it is unavoidable to ensure that a specific directive is set, your application should automatically attempt to override the default PHP.INI customization via <em>ini_set() </em>and then inform the user in <span style="text-decoration: underline;">plain English</span> that the ini value needs to be changed.</p>
<h3>6. <span style="text-decoration: underline;">Do</span> include a mechanism in the application to test for required prerequisites at the very start of execution (or as part of the installation process)</h3>
<p>Rather than have your application spit out a generic PHP error message when it comes across an environment setting that it doesn&#8217;t like, you should include a function that performs a &#8220;pre-flight checklist&#8221; at the beginning of runtime for each session, or at least the very first time your application is run.</p>
<p>Next week, I will show you how to do exactly this by creating a hook in <a title="CodeIgniter Homepage" href="http://codeigniter.com">CodeIgniter</a> that thoroughly tests the environment, and informs the user how to remedy issues.</p>
<h6>For Further Reading, I recommend:</h6>
<ul>
<li><a title="&quot;Writing Portable PHP Code&quot; article" href="http://www.phpwact.org/php/writing_portable_php_code">Writing Portable PHP Code</a> from <a title="Web Application Toolkit" href="http://www.phpwact.org">PHPwact.org</a></li>
<li><a title="&quot;Making Your PHP Code Portable&quot; Article" href="http://www.mtdev.com/2002/09/make-your-php-code-portable">Making Your PHP Code Portable</a> from <a title="mtDev" href="http://mtdev.com">mtdev.com</a></li>
</ul>
]]></content:encoded>
			<wfw:commentRss>http://www.caseymclaughlin.com/2009/06/making-php-applications-portable/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
	</channel>
</rss>
