<?xml version="1.0" encoding="UTF-8"?>
<?xml-stylesheet type="text/xsl" media="screen" href="/~d/styles/atom10full.xsl"?><?xml-stylesheet type="text/css" media="screen" href="http://feeds.feedburner.com/~d/styles/itemcontent.css"?><feed xmlns="http://www.w3.org/2005/Atom" xmlns:openSearch="http://a9.com/-/spec/opensearch/1.1/" xmlns:georss="http://www.georss.org/georss" xmlns:gd="http://schemas.google.com/g/2005" xmlns:feedburner="http://rssnamespace.org/feedburner/ext/1.0" gd:etag="W/&quot;D0EERH8zcSp7ImA9WxBUFE4.&quot;"><id>tag:blogger.com,1999:blog-6471110</id><updated>2010-03-01T01:26:45.189-08:00</updated><title>Jason Pollock's Controversial Originality</title><subtitle type="html">Random thoughts as I work through my career</subtitle><link rel="http://schemas.google.com/g/2005#feed" type="application/atom+xml" href="http://blog.jason.pollock.ca/feeds/posts/default" /><link rel="alternate" type="text/html" href="http://blog.jason.pollock.ca/" /><link rel="next" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default?start-index=26&amp;max-results=25&amp;redirect=false&amp;v=2" /><author><name>Jason Pollock</name><uri>http://www.blogger.com/profile/10823039258279387446</uri><email>noreply@blogger.com</email></author><generator version="7.00" uri="http://www.blogger.com">Blogger</generator><openSearch:totalResults>77</openSearch:totalResults><openSearch:startIndex>1</openSearch:startIndex><openSearch:itemsPerPage>25</openSearch:itemsPerPage><atom10:link xmlns:atom10="http://www.w3.org/2005/Atom" rel="self" type="application/atom+xml" href="http://feeds.feedburner.com/JasonPollocksRandomSpewings" /><feedburner:info uri="jasonpollocksrandomspewings" /><atom10:link xmlns:atom10="http://www.w3.org/2005/Atom" rel="hub" href="http://pubsubhubbub.appspot.com/" /><entry gd:etag="W/&quot;D0MBQnc5fSp7ImA9WxBQF0s.&quot;"><id>tag:blogger.com,1999:blog-6471110.post-5009134798813044030</id><published>2010-01-17T13:57:00.000-08:00</published><updated>2010-01-17T13:57:33.925-08:00</updated><app:edited xmlns:app="http://www.w3.org/2007/app">2010-01-17T13:57:33.925-08:00</app:edited><category scheme="http://www.blogger.com/atom/ns#" term="VPN" /><category scheme="http://www.blogger.com/atom/ns#" term="Hulu" /><category scheme="http://www.blogger.com/atom/ns#" term="proxy" /><title>Cheaper Internet TV</title><content type="html">&lt;p&gt;A while ago, I did some math to show how streaming TV through a US proxy on Amazon was cheaper than torrenting the show, and how it was even competitive with Sky TV if you watches less than 12 hours of TV a week.&lt;/p&gt;
&lt;p&gt;That was if you used Amazon as your US proxy.  With Amazon, you are paying for both traffic and time, which skews the price considerably.  If you already have a US server (several of my friends do) then you already have your US proxy with a huge bundle of data.  However, the rest of us need another, easier way.&lt;/p&gt;
&lt;p&gt;Of course, the US being the US, and the Internet being the Internet, there is already several groups offering a range of methods, starting from free and getting more expensive from there.&lt;/p&gt;
&lt;p&gt;First off, some math:&lt;/p&gt;
&lt;p&gt;From last time, we saw the price for Amazon was:&lt;/p&gt;
&lt;pre&gt;
Amazon Cost:
  Instance: US$0.1*(NZ$1/USD$0.67) = NZ$0.15
  Traffic (based on 1 episode of Eureka)
  (304MB/43min)*(60min/hr)*(1GB/1024MB)*(USD$0.27/1GB)*(NZD$1/USD$0.67) = NZ$0.17
  Total   : NZ$0.32/hr
NZ Bandwidth cost:
  (304MB/43min)*(60min/hr)*(1GB/1024MB)*(NZ$1.50/GB) = NZ$0.62/hr
Total cost per hour: NZ$0.94/hr
&lt;/pre&gt;
&lt;p&gt;If we can reduce the Amazon cost, we can substantially reduce our total cost.&lt;/p&gt;
&lt;p&gt;My goal is to get the "Amazon" portion of the charge down to 8c/day.  That would make the price for a Hulu stream into:&lt;/p&gt;
&lt;pre&gt;
NZ$0.62/hr + $0.08/day
&lt;/pre&gt;
&lt;p&gt;So, the break even point against the Sky basic plan would become:&lt;/p&gt;
&lt;pre&gt;($11.74 - 0.08*7)/0.62 = 18 hours/week.&lt;/pre&gt;
&lt;p&gt;Not quite to the 3hrs/day that I have previously boasted about, but close.  Within spitting distance, even.  If you can get bandwidth for less than NZ$1.50/GB, the break even point will change dramatically.  Given that the Southern Cross prices have just dropped by 50%, I hope to see some end-user price changes soon.&lt;/p&gt;
&lt;p&gt;So, I went looking for a VPN provider.  First I tried free, &lt;a href="http://www.hotspotshield.com/"&gt;Hotspot Shield&lt;/a&gt; from AnchroFree.  I quickly gave up on them.  They don't have enough bandwidth, and they seem to be specifically &lt;a href="http://haugland.ca/entries/general/hulu-blocks-hotspot-shield-again"&gt;targetted&lt;/a&gt; by Hulu.&lt;/p&gt;
&lt;p&gt;I thought about using TOR, but quickly discarded that one.  It would be just plain rude.&lt;/p&gt;
&lt;p&gt;I looked at using the free proxy add-on for Firefox.  However, they just aren't reliable enough for the next stage - a set top box.&lt;/p&gt;
&lt;p&gt;So, I went looking at commercial offerings.  I tried StrongVPN.com, WiTopia.net and HideMyNet.com.&lt;/p&gt;
&lt;p&gt;VPN is still pretty new and arcane.  There is a lot of magic, and many things that can go wrong.  I found it very frustrating to get properly configured, and ultimately had to rely on the support teams at a couple of the providers to get configured.  Thankfully, both StrongVPN.com and WiTopia.net offered good support.  HideMyNet (and others) were not so good.  I had other issues too.  Sometimes it felt odd giving these companies my credit card details.  So, I only went with ones that used PayPal.&lt;/p&gt;
&lt;p&gt;Notes:&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;Avoid PPTP.  It's generally provided by the OS, and is flakey and hard to get configured properly.  It may be cheaper, but it's not really worth it.&lt;/li&gt;
&lt;li&gt;Go with an OpenVPN based solution.  This is available on Windows, OS X and Linux.  The benefit is that it is it's own tool, separate from the network stack and therefore a lot easier to diagnose.&lt;/li&gt;
&lt;li&gt;CPU, CPU, CPU.  My 2.4Ghz AMD single core Athlon 3800+ wasn't fast enough to stream more than 700kbps reliably.  My 2.5Ghz Core2Duo is.&lt;/li&gt;
&lt;/ol&gt;
&lt;p&gt;StrongVPN.com offers points of presence all over the world.  However, they are moderately expensive at US$85/yr, and charge money to change PoPs (Point of Presence).  In other words, while they do have PoPs in the UK (for access to BBC iPlayer or Spotify), if you want to switch between UK/US PoPs, they charge money.  It really sounds like they should fix their provisioning and access control systems to me.&lt;/p&gt;
&lt;p&gt;WiTopia.net is the company that I've ultimately gone with.  They offer instant switching between VPN PoPs, UK and US PoPs and a low annual fee of US$60.  Even better, they have US PoPs close to the Southern Cross US interconnect points in LA and San Francisco.&lt;/p&gt;
&lt;p&gt;Since I've gone with WiTopia.net, Hulu came along and banned them from accessing their streams.  However, a quick support message and I was on another PoP, which wasn't filtered (yet).  However, I've also got access to BBC's iPlayer and Pandora, which kicks ass.&lt;/p&gt;
&lt;p&gt;I just wish NetFlix was supported on Linux, then I would sign up for a NetFlix streaming account.&lt;/p&gt;
&lt;p&gt;The final cost analysis works out as:&lt;/p&gt;
&lt;pre&gt;
WiTopia.net Cost:
  Cost: NZ$80/yr
NZ Bandwidth cost:
  (304MB/43min)*(60min/hr)*(1GB/1024MB)*(NZ$1.50/GB) = NZ$0.62/hr

Break-Even point vs Sky: 
  ($11.74 - $80/365)/0.69 = 17 hours.
&lt;/pre&gt;
&lt;p&gt;In other words, if you watch less than 17 hours of Sky specific TV per day (it isn't available on FreeView), you are better off paying for a proxy account and obtaining the content from the US or the UK.&lt;/p&gt;&lt;div class="blogger-post-footer"&gt;&lt;img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/6471110-5009134798813044030?l=blog.jason.pollock.ca' alt='' /&gt;&lt;/div&gt;&lt;img src="http://feeds.feedburner.com/~r/JasonPollocksRandomSpewings/~4/69eWmVQgmmE" height="1" width="1"/&gt;</content><link rel="replies" type="application/atom+xml" href="http://blog.jason.pollock.ca/feeds/5009134798813044030/comments/default" title="Post Comments" /><link rel="replies" type="text/html" href="https://www.blogger.com/comment.g?blogID=6471110&amp;postID=5009134798813044030" title="4 Comments" /><link rel="edit" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/5009134798813044030?v=2" /><link rel="self" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/5009134798813044030?v=2" /><link rel="alternate" type="text/html" href="http://feedproxy.google.com/~r/JasonPollocksRandomSpewings/~3/69eWmVQgmmE/cheaper-internet-tv.html" title="Cheaper Internet TV" /><author><name>Jason Pollock</name><uri>http://www.blogger.com/profile/10823039258279387446</uri><email>noreply@blogger.com</email><gd:extendedProperty name="OpenSocialUserId" value="15378423220352804988" /></author><thr:total xmlns:thr="http://purl.org/syndication/thread/1.0">4</thr:total><feedburner:origLink>http://blog.jason.pollock.ca/2010/01/cheaper-internet-tv.html</feedburner:origLink></entry><entry gd:etag="W/&quot;DEYCRnc-eip7ImA9WxNaFkw.&quot;"><id>tag:blogger.com,1999:blog-6471110.post-8816498277488231352</id><published>2009-11-30T13:22:00.000-08:00</published><updated>2009-11-30T13:22:47.952-08:00</updated><app:edited xmlns:app="http://www.w3.org/2007/app">2009-11-30T13:22:47.952-08:00</app:edited><title>Topic Relevant Comment Spam</title><content type="html">&lt;p&gt;On my previous post &lt;a href="http://blog.jason.pollock.ca/2009/11/streaming-audio-to-your-iphone.html"&gt;"Streaming Audio to your iPhone"&lt;/a&gt;, I received a very odd comment.  It was related to the subject, so it didn't immediately
look like spam, but it wasn't topical.&lt;/p&gt;
&lt;p&gt;Here's the comment (I've removed it from the original post):&lt;/p&gt;
&lt;blockquote&gt;
&lt;p&gt;Web casting, or broadcasting over the internet, is a media file (audio-video mostly) distributed over the internet using streaming media technology. Streaming implies media played as a continuous stream and received real time by the browser (end user). Streaming technology enables a single content source to be distributed to many simultaneous viewers. Streaming video bandwidth is typically calculated in gigabytes of data transferred. It is important to estimate how many viewers you can reach, for example in a live webcast, given your bandwidth constraints or conversely, if you are expecting a certain audience size, what bandwidth resources you need to deploy.&lt;/p&gt;
&lt;p&gt;To estimate how many viewers you can reach during a webcast, consider some parlance:&lt;br/&gt;
One viewer: 1 click of a video player button at one location logged on&lt;br/&gt;
One viewer hour: 1 viewer connected for 1 hour&lt;br/&gt;
100 viewer hours: 100 viewers connected for 1 hour…&lt;br/&gt;
&lt;/p&gt;
&lt;p&gt;Typically webcasts will be offered at different bit rates or quality levels corresponding to different user’s internet connection speeds. Bit rate implies the rate at which bits (basic data units) are transferred. It denotes how much data is transmitted in a given amount of time. (bps / Kbps / Mbps…). Quality improves as more bits are used for each second of the playback. Video of 3000 Kbps will look better than one of say 1000Kbps. This is just like quality of a image is represented in resolution, for video (or audio) it is measured by the bit rate.
&lt;/p&gt;
&lt;/blockquote&gt;
&lt;p&gt;It was posted by "Andy".  I was ready to post a comment thanking Andy for the additional information, but I decided to look to see if it was copied from somewhere else.&lt;/p&gt;
&lt;p&gt;It seems that the exact same comment is placed on just about any blog post that mentions streaming.  On the Google search I &lt;a href="http://www.google.co.nz/search?hl=en&amp;client=firefox-a&amp;channel=s&amp;rls=org.mozilla:en-GB:official&amp;q=%22Web+casting,+or+broadcasting+over+the+internet,+is+a+media+file+%28audio-video+mostly%29+distributed+over+the+internet+using+streaming%22&amp;start=0&amp;sa=N"&gt;used&lt;/a&gt;, I found a lot of comments from the past couple of months.&lt;/p&gt;
&lt;p&gt;Seems "Andy" has been a busy boy, and is actually a &lt;a href="http://www.blogger.com/profile/18319442650203826756"&gt;dirty, rotten spammer&lt;/a&gt;.&lt;/p&gt;
&lt;p&gt;"Andy" (blogger), who also posts as "andylock", or "Andy Lock" (Facebook) is an &lt;a href="http://www.facebook.com/people/Andy-Lock/100000142552423"&gt;automated spam program&lt;/a&gt; for vsworld.com, and the website is a flash-only website.  I didn't stay there long enough to really figure out what they were selling, but it looked like some sort of contracting agency in India.&lt;/p&gt;
&lt;p&gt;Still, I found it interesting that it wasn't immediately obvious that the post was spam.&lt;/p&gt;&lt;div class="blogger-post-footer"&gt;&lt;img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/6471110-8816498277488231352?l=blog.jason.pollock.ca' alt='' /&gt;&lt;/div&gt;&lt;img src="http://feeds.feedburner.com/~r/JasonPollocksRandomSpewings/~4/dxmTlnCgLgU" height="1" width="1"/&gt;</content><link rel="replies" type="application/atom+xml" href="http://blog.jason.pollock.ca/feeds/8816498277488231352/comments/default" title="Post Comments" /><link rel="replies" type="text/html" href="https://www.blogger.com/comment.g?blogID=6471110&amp;postID=8816498277488231352" title="0 Comments" /><link rel="edit" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/8816498277488231352?v=2" /><link rel="self" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/8816498277488231352?v=2" /><link rel="alternate" type="text/html" href="http://feedproxy.google.com/~r/JasonPollocksRandomSpewings/~3/dxmTlnCgLgU/topic-relevant-comment-spam.html" title="Topic Relevant Comment Spam" /><author><name>Jason Pollock</name><uri>http://www.blogger.com/profile/10823039258279387446</uri><email>noreply@blogger.com</email><gd:extendedProperty name="OpenSocialUserId" value="15378423220352804988" /></author><thr:total xmlns:thr="http://purl.org/syndication/thread/1.0">0</thr:total><feedburner:origLink>http://blog.jason.pollock.ca/2009/11/topic-relevant-comment-spam.html</feedburner:origLink></entry><entry gd:etag="W/&quot;Ck4DQngzcSp7ImA9WxNaEE8.&quot;"><id>tag:blogger.com,1999:blog-6471110.post-2659350520969365868</id><published>2009-11-23T16:00:00.000-08:00</published><updated>2009-11-23T16:02:53.689-08:00</updated><app:edited xmlns:app="http://www.w3.org/2007/app">2009-11-23T16:02:53.689-08:00</app:edited><category scheme="http://www.blogger.com/atom/ns#" term="iPhone" /><category scheme="http://www.blogger.com/atom/ns#" term="m3u" /><category scheme="http://www.blogger.com/atom/ns#" term="mp3" /><category scheme="http://www.blogger.com/atom/ns#" term="stream" /><category scheme="http://www.blogger.com/atom/ns#" term="embed" /><category scheme="http://www.blogger.com/atom/ns#" term="perl" /><title>Audio Streaming to the iPhone, Take Two</title><content type="html">&lt;p&gt;I wasn't completely happy with the results of using M3U files.  While it did allow me to specify all of the files, I wasn't able to skip forward and back through the song list.&lt;/p&gt;
&lt;p&gt;It turns out that the M3U support is there primarily to support live streams.  For example, if you have a live stream from a video source (like a TV tuner card), the server records it, splits it into small chunks and converts it to H.264.  The iPhone will then use the M3U file as an index to the individual data files, refreshing the M3U file occasionally.  I'll have to set that up next. :)&lt;/p&gt;
&lt;p&gt;So I went back to Google to see what I could find.  I had come across the &amp;lt;OBJECT&amp;gt; tag in my previous searches, but I had discarded it as too difficult.  I didn't want to work in HTML, and object embedding just seemed dirty.  However, if I wanted to get access to skip forward/back, it looked like I was going to have to use it.  I went through and modified the perl program from yesterday to produce an HTML page for each M3U file.  It wasn't until I got to the end that I realized that I had done it wrong.&lt;/p&gt;
&lt;p&gt;Lots of things don't work with the OBJECT tag, primarily the GOTO command.  It seems that the proper way to embed things into a document is with the &lt;a href="http://www.apple.com/quicktime/tutorials/embed2.html"&gt;EMBED&lt;/a&gt; tag.  That isn't to say that the EMBED tag is prettier.  It isn't, they are both nasty.  It is platform and player specific.  Luckily, this only has to work on the iPhone/iPod Touch, so I'm lucky that way.  I would hate to have to support this for multiple clients.&lt;/p&gt;
&lt;p&gt;The first nasty surprise is that while you can have multiple songs in a list, you are limited to 255 of them.  Even worse, they aren't done through object references in the rest of the document, the entire list is in the single object tag.  That makes it harder to do dynamic, on the fly modification of the list.  No small cgi to do shuffles here!&lt;/p&gt;
&lt;p&gt;Still, 255 songs is enough to cover pretty much all of my artist directories.&lt;/p&gt;
&lt;p&gt;It also seems that Mobile Safari ignores GOTO commands.  In regular Safari, you are able to loop back around to the start of the playlist by putting "QTNEXT255=GOTO0" into the embed tag.  Looks like Apple doesn't want playing loops on the iPhone.&lt;/p&gt;
&lt;p&gt;Next, the iPhone ignores the "autohref", "autoplay" and "autostart" parameters.  It always waits for user interaction.  This is because the object is not really embedded, it takes full control of the screen.  If it did start automatically, it would cause problems on many other sites.  It's a small pain, but we'll survive.&lt;/p&gt;
&lt;p&gt;I still wish it properly supported M3U files.  The seamless transitions are nice.  With the QTNEXT, there is a definite pause between tracks.&lt;/p&gt;
&lt;p&gt;Here is the updated Perl code:&lt;/p&gt;
&lt;pre&gt;
#!/usr/bin/perl

use File::Find;
use File::Basename;

use vars qw/*name *dir *prune/;

my $URL_BASE= "http://10.10.10.5/";

my @M3Us;

sub createPlaybackHTMLHeader {
    my ($filename, $target, $song) = @_;

    my $title = basename(dirname($filename));

    # open the file, 
    open(HTML, "&amp;gt;$target");

    # Write the HTML/Head elements,

    print HTML &amp;lt;&amp;lt;END;
&amp;lt;html&amp;gt;
  &amp;lt;head&amp;gt;
    &amp;lt;title&amp;gt;$title&amp;lt;/title&amp;gt;
    &amp;lt;meta name="viewport" content="width=device-width; initial-scale=1.25"/&amp;gt;
  &amp;lt;/head&amp;gt;
  &amp;lt;body&amp;gt;
    &amp;lt;p&amp;gt;Play all music in the "$title" directory, click the button below&amp;lt;/p&amp;gt;
    &amp;lt;embed src="$song"&amp;gt;
      autoplay="true"
      controller="true"
END
;
    # close the file.
    close(HTML);
}

sub createPlaybackHTMLFooter {
    my ($filename, $target) = @_;

    # open the file, 
    open(HTML, "&amp;gt;&amp;gt;$target");

    # Write the HTML/Head elements,

    print HTML &amp;lt;&amp;lt;END;
      qtnext255="GOTO0"
    &amp;lt;/embed&amp;gt;
  &amp;lt;/body&amp;gt;
&amp;lt;/html&amp;gt;

END
;
    # close the file.
    close(HTML);
}

sub createPlaybackAddSong {
    my ($filename,$target, $song, $count) = @_;

    open(HTML, "&amp;gt;&amp;gt;$target");
    print HTML &amp;lt;&amp;lt;END;
      qtnext$count="&amp;lt;$song&amp;gt; T&amp;lt;myself&amp;gt;"
END
;
    # close the file.
    close(HTML);
}

sub convertM3UToHTML {
    my ($m3ufile, $target) = @_;

    print "Converting $m3ufile to $target...";

    if ( ! -f $m3ufile ) {
 print "No m3ufile, returning\n";
 return;
    }

    # Open that m3u file, and convert it to a playback html page.
    open(M3U, $m3ufile) or die "Can't open m3u file";
    my @songs=&amp;lt;M3U&amp;gt;;
    close(M3U);

    if (@songs &amp;gt; 255) {
 print "Too many songs! Oh well\n";
 @songs = @songs[0..254];
    } elsif (@songs == 0) {
 print "No songs, returning\n";
 return;
    } else {
 print "No problems!\n";
    }

    my $first_song = shift @songs;

    chomp $first_song;

    createPlaybackHTMLHeader($m3ufile, $target, $first_song);
    my $count = 1;

    foreach my $song (@songs) {
 chomp $song;
 createPlaybackAddSong($m3ufile, $target, $song, $count);
 $count += 1;
    }

    createPlaybackHTMLFooter($m3ufile, $target);
}

sub entering {
    print "entering Directory boundary ", $File::Find::name, "\n";

    push @M3Us, $File::Find::dir;

    if ($File::Find::name =~ /.AppleDouble/) {
 return;
    }

    return sort(@_);
}

sub leaving {
    print "leaving Directory boundary ", $File::Find::name, "\n";

    my $directory = pop @M3Us;

    # open the file, 
    my $source = $directory . "/PLAY.m3u";
    my $target = $directory . "/PLAY_ALL.html";

    convertM3UToHTML($source, $target);
}

sub wanted {
   print "Checking ", $File::Find::name,"\n";

   if ( $File::Find::name =~ /\.mp3/ ) {
       print "MP3 Found ", $File::Find::name, "\n";
       my $url = $File::Find::name;
       $url =~ s/ /%20/g;
       $url =~ s/\/export\///g;
       $url = $URL_BASE . $url;
       foreach my $dir (@M3Us) {
    my $m3u_file = $dir . "/PLAY.m3u";
    open (M3U, "&amp;gt;&amp;gt;$m3u_file") or die "Boom!";
    print M3U "$url\n";
    close(M3U);
       }
   }
}

sub cleaner {
    if ($File::Find::name =~ /PLAY.m3u/ ) {
 print "Removing ", $File::Find::name, "\n";
 unlink($File::Find::name);
    }

    if ($File::Find::name =~ /PLAY_ALL.html/ ) {
 print "Removing ", $File::Find::name, "\n";
 unlink($File::Find::name);
    }
}

find ({ wanted =&amp;gt; \&amp;cleaner},"/export/mp3");

find ({ wanted =&amp;gt; \&amp;wanted , preprocess =&amp;gt; \&amp;entering, postprocess =&amp;gt; \&amp;leaving},"/export/mp3");

open(M3U, "/export/mp3/PLAY.m3u") or die "Unable to open play";

my @main_m3u = &amp;lt;M3U&amp;gt;;
close(M3U);

my @music = grep(!/AudioBooks/, @main_m3u);
my @random = sort { int(rand(3))-1 } @music;

open(M3U, "&amp;gt;/export/mp3/MUSIC.m3u") or die "Unable to open MUSIC.m3u";

print M3U @music;
close(M3U);

convertM3UToHTML("/export/mp3/MUSIC.m3u", "/export/mp3/MUSIC_PLAY.html");

open(M3U, "&amp;gt;/export/mp3/RANDOM.m3u") or die "Unable to open RANDOM.m3u";
print M3U @random;
close(M3U);

convertM3UToHTML("/export/mp3/RANDOM.m3u", "/export/mp3/RANDOM_PLAY.html");
&lt;/pre&gt;&lt;div class="blogger-post-footer"&gt;&lt;img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/6471110-2659350520969365868?l=blog.jason.pollock.ca' alt='' /&gt;&lt;/div&gt;&lt;img src="http://feeds.feedburner.com/~r/JasonPollocksRandomSpewings/~4/2j0uSk4yEw0" height="1" width="1"/&gt;</content><link rel="replies" type="application/atom+xml" href="http://blog.jason.pollock.ca/feeds/2659350520969365868/comments/default" title="Post Comments" /><link rel="replies" type="text/html" href="https://www.blogger.com/comment.g?blogID=6471110&amp;postID=2659350520969365868" title="0 Comments" /><link rel="edit" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/2659350520969365868?v=2" /><link rel="self" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/2659350520969365868?v=2" /><link rel="alternate" type="text/html" href="http://feedproxy.google.com/~r/JasonPollocksRandomSpewings/~3/2j0uSk4yEw0/audio-streaming-to-iphone-take-two.html" title="Audio Streaming to the iPhone, Take Two" /><author><name>Jason Pollock</name><uri>http://www.blogger.com/profile/10823039258279387446</uri><email>noreply@blogger.com</email><gd:extendedProperty name="OpenSocialUserId" value="15378423220352804988" /></author><thr:total xmlns:thr="http://purl.org/syndication/thread/1.0">0</thr:total><feedburner:origLink>http://blog.jason.pollock.ca/2009/11/audio-streaming-to-iphone-take-two.html</feedburner:origLink></entry><entry gd:etag="W/&quot;A0cGQ309eyp7ImA9WxNbGUk.&quot;"><id>tag:blogger.com,1999:blog-6471110.post-7601062345712763900</id><published>2009-11-22T20:00:00.000-08:00</published><updated>2009-11-22T20:03:42.363-08:00</updated><app:edited xmlns:app="http://www.w3.org/2007/app">2009-11-22T20:03:42.363-08:00</app:edited><title>Streaming Audio to your iPhone</title><content type="html">&lt;p&gt;I'm trying to hide the CDs and DVDs that are steadily taking over my house.  I've managed to get rid of all of the DVDs, they've all been ripped to the server and are now happily up in the attic.  The last problem are the CDs.&lt;/p&gt;
&lt;p&gt;Yes, I have them all ripped to the server.  The problem is that I don't have any way of getting at them everywhere in the house.  Specifically, my wife likes to listen to murder mysteries as she cooks (me, I like cooking to the news).  Since I don't have a network enabled stereo, I have to put up with a stack of CDs sitting on the counter.&lt;/p&gt;
&lt;p&gt;I have given up waiting for a network enabled stereo system.  The squeezebox just doesn't do it for me.  I want something small, with wifi and wired, with a radio, clock and a streaming mp3 player.  It should also have .m3u support and a "shuffle".&lt;/p&gt;
&lt;p&gt;I decided to see what I could do with what we already have...&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;First realization.  We both have iPhones.
&lt;li&gt;Second realization.  We have a WiFi network in the house.
&lt;li&gt;Third realization.  The iPhone will happily stream music off of a web page.
&lt;/ul&gt;
&lt;p&gt;So the project was formed.  First, I tried the SqueezeBox server (previously SlimServer) a server which serves up streams to network attached stereo component equipment.  I've got a 1st generation SliMP3, it was a great device.  The server just doesn't play well with the iPhone.  So, I decided to go the bare bones route.  I decided to setup Apache on my file server so that it will serve up music to the iPhones in the house.  Then the phones can be used to stream the music/books wherever anyone is in the house.  I would finally be allowed to hide all of the CDs!  Perfect!&lt;/p&gt;
&lt;p&gt;Setting up Apache was pretty easy.  I followed the instructions that are easily &lt;a href="http://www.google.co.nz/#hl=en&amp;source=hp&amp;q=setting+up+apache+on+ubuntu&amp;btnG=Google+Search&amp;meta=&amp;aq=1&amp;oq=setting+up+apache+on+&amp;fp=1&amp;cad=b"&gt;Googled&lt;/a&gt;.  I didn't create any HTML files, but I did configure the auto_index module (/etc/apache2/mods-available/autoindex.conf).  My changed settings were:&lt;/p&gt;

&lt;pre&gt;
IndexOptions SuppressDescription SuppressSize SuppressLastModified
IndexOptions SuppressHTMLPreamble
HeaderName /include/iPhone_Header.html
&lt;/pre&gt;
&lt;p&gt;NOTE: HeaderName is relative to the DocumentRoot, not the filesystem.&lt;/p&gt;
&lt;p&gt;This allowed me to keep the listing to just the filenames, and replaced the standard HTML header with one of my own:&lt;/p&gt;
&lt;pre&gt;
&amp;lt;html&amp;gt;
&amp;lt;head&amp;gt;
&amp;lt;meta name="viewport" content="width=device-width; initial-scale=1.25"/&amp;gt;
&amp;lt;/head&amp;gt;
&amp;lt;body&amp;gt;
&lt;/pre&gt;
&lt;p&gt;This header provides a hint to the iPhone of where to set the viewport.  It seems to work for my listings, making the file list usable.&lt;/p&gt;
&lt;p&gt;Once we have that, we've got a server with a directory listing that we can scroll through and play individual MP3s.  We don't have any playlists though, which is pretty unpleasant.&lt;/p&gt;
&lt;p&gt;Bring in the Perl!&lt;/p&gt;
&lt;p&gt;I've written a small piece of Perl code to iterate over my MP3 tree and create M3U files in each directory containing URLs for all of the MP3s that are children of that directory.  Because of how I store my MP3s, that gives me album, artist and full play lists.  I then randomize the playlist to give me a shuffled list.&lt;/p&gt;
&lt;p&gt;The new iPhone release is able to play M3U files, with one problem.  You can't skip to the next track, which is pretty poor.  But now I've got a method of delivering music to any room in the house.  I just have to get one of those 3rd party iPod speaker systems, and I can get rid of all of the CDs in the kitchen!&lt;/p&gt;

&lt;pre&gt;
#!/usr/bin/perl

use File::Find;

use vars qw/*name *dir *prune/;

my $URL_BASE= "http://10.10.10.5/";

my @M3Us;

sub entering {
    print "entering Directory boundary ", $File::Find::name, "\n";

    push @M3Us, $File::Find::dir;
    return sort(@_);
}

sub leaving {
    print "leaving Directory boundary ", $File::Find::name, "\n";

    pop @M3Us;
}

sub wanted {
   print "Checking ", $File::Find::name,"\n";

   if ( $File::Find::name =~ /\.mp3/ ) {
       print "MP3 Found ", $File::Find::name, "\n";
       my $url = $File::Find::name;
       $url =~ s/ /%20/g;
       $url =~ s/\/export\///g;
       $url = $URL_BASE . $url;
       foreach my $dir (@M3Us) {
    my $m3u_file = $dir . "/PLAY.m3u";
    open (M3U, "&amp;gt;&amp;gt;$m3u_file") or die "Boom!";
    print M3U "$url\n";
    close(M3U);
       }
   }
}

sub cleaner {
    if ($File::Find::name =~ /PLAY.m3u/ ) {
 print "Removing ", $File::Find::name, "\n";
 unlink($File::Find::name);
    }
}

find ({ wanted =&amp;gt; \&amp;cleaner},"/export/mp3");

find ({ wanted =&amp;gt; \&amp;wanted , preprocess =&amp;gt; \&amp;entering, postprocess =&amp;gt; \&amp;leaving},"/export/mp3");

open(M3U, "/export/mp3/PLAY.m3u") or die "Unable to open play";

my @main_m3u = &amp;lt;M3U&amp;gt;;
close(M3U);

my @music = grep(!/AudioBooks/, @main_m3u);
my @random = sort { int(rand(3))-1 } @music;

open(M3U, "&amp;gt;/export/mp3/MUSIC.m3u") or die "Unable to open MUSIC.m3u";
print M3U "@music";
close(M3U);

open(M3U, "&amp;gt;/export/mp3/RANDOM.m3u") or die "Unable to open RANDOM.m3u";
print M3U "@random";
close(M3U);
&lt;/pre&gt;&lt;div class="blogger-post-footer"&gt;&lt;img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/6471110-7601062345712763900?l=blog.jason.pollock.ca' alt='' /&gt;&lt;/div&gt;&lt;img src="http://feeds.feedburner.com/~r/JasonPollocksRandomSpewings/~4/f97E0vtqLEo" height="1" width="1"/&gt;</content><link rel="replies" type="application/atom+xml" href="http://blog.jason.pollock.ca/feeds/7601062345712763900/comments/default" title="Post Comments" /><link rel="replies" type="text/html" href="https://www.blogger.com/comment.g?blogID=6471110&amp;postID=7601062345712763900" title="3 Comments" /><link rel="edit" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/7601062345712763900?v=2" /><link rel="self" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/7601062345712763900?v=2" /><link rel="alternate" type="text/html" href="http://feedproxy.google.com/~r/JasonPollocksRandomSpewings/~3/f97E0vtqLEo/streaming-audio-to-your-iphone.html" title="Streaming Audio to your iPhone" /><author><name>Jason Pollock</name><uri>http://www.blogger.com/profile/10823039258279387446</uri><email>noreply@blogger.com</email><gd:extendedProperty name="OpenSocialUserId" value="15378423220352804988" /></author><thr:total xmlns:thr="http://purl.org/syndication/thread/1.0">3</thr:total><feedburner:origLink>http://blog.jason.pollock.ca/2009/11/streaming-audio-to-your-iphone.html</feedburner:origLink></entry><entry gd:etag="W/&quot;C0MBQn89fSp7ImA9WxNbE08.&quot;"><id>tag:blogger.com,1999:blog-6471110.post-8932177860891720737</id><published>2009-11-15T13:29:00.000-08:00</published><updated>2009-11-15T13:44:13.165-08:00</updated><app:edited xmlns:app="http://www.w3.org/2007/app">2009-11-15T13:44:13.165-08:00</app:edited><category scheme="http://www.blogger.com/atom/ns#" term="Bug" /><category scheme="http://www.blogger.com/atom/ns#" term="Oracle" /><category scheme="http://www.blogger.com/atom/ns#" term="SQL" /><title>Oracle's BETWEEN keyword</title><content type="html">&lt;p&gt;I came across a new Oracle keyword today, BETWEEN.&lt;/p&gt;
&lt;p&gt;At first, I thought it was pretty cool.  I would be able to simplify the majority of the range checks that I perform.  Before I really started using it though, I decided to look at what it actually did.  Ouch!&lt;/p&gt;
&lt;p&gt;Google, my ever present documentation source, &lt;a href="http://www.adp-gmbh.ch/ora/sql/between.html"&gt;told&lt;/a&gt; me that between doesn't work the way I thought it would.  It's inclusive of both ends of the range.  Who would want that?  You never want a range that is inclusive of both ends!  Otherwise, elements in contiguous ranges have indeterminate ownership!&lt;/p&gt;
&lt;p&gt;Let's try an example:&lt;/p&gt;
&lt;p&gt;We want all rows with a date on the row that has a date of today.&lt;/p&gt;
&lt;pre&gt;
  select id from test_table where my_date between 
    trunc(sysdate) and trunc(sysdate) + 1;
&lt;/pre&gt;
&lt;p&gt;Great you would think, almost too easy!&lt;/p&gt;
&lt;p&gt;You would be correct too, it was too easy.  It doesn't work.  Since it is inclusive of the end of range, you get all values that have a my_date of 12:00:00 tomorrow.  What you really want is:&lt;/p&gt;
&lt;pre&gt;
  select id from test_table where 
      (my_date &gt;= trunc(sysdate)) and 
      (my_date &lt; trunc(sysdate) + 1);
&lt;/pre&gt;
&lt;p&gt;When using new keywords and abstractions, you should always know what they are doing.&lt;/p&gt;&lt;div class="blogger-post-footer"&gt;&lt;img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/6471110-8932177860891720737?l=blog.jason.pollock.ca' alt='' /&gt;&lt;/div&gt;&lt;img src="http://feeds.feedburner.com/~r/JasonPollocksRandomSpewings/~4/y69bO5ymi70" height="1" width="1"/&gt;</content><link rel="replies" type="application/atom+xml" href="http://blog.jason.pollock.ca/feeds/8932177860891720737/comments/default" title="Post Comments" /><link rel="replies" type="text/html" href="https://www.blogger.com/comment.g?blogID=6471110&amp;postID=8932177860891720737" title="0 Comments" /><link rel="edit" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/8932177860891720737?v=2" /><link rel="self" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/8932177860891720737?v=2" /><link rel="alternate" type="text/html" href="http://feedproxy.google.com/~r/JasonPollocksRandomSpewings/~3/y69bO5ymi70/oracles-between-keyword.html" title="Oracle's BETWEEN keyword" /><author><name>Jason Pollock</name><uri>http://www.blogger.com/profile/10823039258279387446</uri><email>noreply@blogger.com</email><gd:extendedProperty name="OpenSocialUserId" value="15378423220352804988" /></author><thr:total xmlns:thr="http://purl.org/syndication/thread/1.0">0</thr:total><feedburner:origLink>http://blog.jason.pollock.ca/2009/11/oracles-between-keyword.html</feedburner:origLink></entry><entry gd:etag="W/&quot;DEUHRHk4fSp7ImA9WxNUEUQ.&quot;"><id>tag:blogger.com,1999:blog-6471110.post-8204598689124352005</id><published>2009-11-02T13:04:00.001-08:00</published><updated>2009-11-02T13:10:35.735-08:00</updated><app:edited xmlns:app="http://www.w3.org/2007/app">2009-11-02T13:10:35.735-08:00</app:edited><title>std::auto_ptr and GDB.</title><content type="html">&lt;p&gt;I needed to gain access to the contents of an auto_ptr inside of GDB.  However, GDB doesn't like the overloaded -&amp;gt; operator, so the simple foo-&amp;gt;fnImInterestedIn() doesn't work.  Here's the simple pattern:&lt;/p&gt;

&lt;pre&gt;
class bar_t {
  public:
     int fnImInterestedIn();
};

std::auto_ptr&amp;lt;bar_t&amp;gt; foo;
(gdb) p ((struct bar_t *)foo._M_ptr)-&amp;gt;fnImInterestedIn()
&lt;/pre&gt;&lt;div class="blogger-post-footer"&gt;&lt;img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/6471110-8204598689124352005?l=blog.jason.pollock.ca' alt='' /&gt;&lt;/div&gt;&lt;img src="http://feeds.feedburner.com/~r/JasonPollocksRandomSpewings/~4/BNcpOxGaRw0" height="1" width="1"/&gt;</content><link rel="replies" type="application/atom+xml" href="http://blog.jason.pollock.ca/feeds/8204598689124352005/comments/default" title="Post Comments" /><link rel="replies" type="text/html" href="https://www.blogger.com/comment.g?blogID=6471110&amp;postID=8204598689124352005" title="0 Comments" /><link rel="edit" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/8204598689124352005?v=2" /><link rel="self" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/8204598689124352005?v=2" /><link rel="alternate" type="text/html" href="http://feedproxy.google.com/~r/JasonPollocksRandomSpewings/~3/BNcpOxGaRw0/stdautoptr-and-gdb.html" title="std::auto_ptr and GDB." /><author><name>Jason Pollock</name><uri>http://www.blogger.com/profile/10823039258279387446</uri><email>noreply@blogger.com</email><gd:extendedProperty name="OpenSocialUserId" value="15378423220352804988" /></author><thr:total xmlns:thr="http://purl.org/syndication/thread/1.0">0</thr:total><feedburner:origLink>http://blog.jason.pollock.ca/2009/11/stdautoptr-and-gdb.html</feedburner:origLink></entry><entry gd:etag="W/&quot;Ck4FR3c9eSp7ImA9WxNUEUQ.&quot;"><id>tag:blogger.com,1999:blog-6471110.post-6099608728523879579</id><published>2009-11-02T11:27:00.001-08:00</published><updated>2009-11-02T11:41:56.961-08:00</updated><app:edited xmlns:app="http://www.w3.org/2007/app">2009-11-02T11:41:56.961-08:00</app:edited><category scheme="http://www.blogger.com/atom/ns#" term="Hulu" /><category scheme="http://www.blogger.com/atom/ns#" term="Amazon" /><category scheme="http://www.blogger.com/atom/ns#" term="EC2" /><category scheme="http://www.blogger.com/atom/ns#" term="proxy" /><title>Hulu Proxy Apocalypse</title><content type="html">&lt;p&gt;It seems that the great Hulu apocalypse has hit more than just Witopia.net, Amazon's EC2 instances also seem to be blocked.&lt;/p&gt;
&lt;p&gt;Of course, there are a tonne of other cloud providers out there, several of them even cheaper than Amazon.  Personally, I'm still with Witopia.  Like any good company, they had a new address range up immediately.&lt;/p&gt;
&lt;p&gt;Hulu is stuck in the same game as Apple.  They're playing whack-a-mole with the hackers.  The only problem, every time they want to block an access method, it costs them money.  However, it is free for us to invent a way around a block (it's a hobby), and there are a lot more of us than there are developers at Hulu.&lt;/p&gt;
&lt;p&gt;This is yet another example of "Don't piss off the nerds".&lt;/p&gt;
&lt;p&gt;Personally, here are the lessons that I would take from this.&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;There is a market for International access.  People are willing to _pay_ to get access.  I am currently paying US$12/month to access Hulu, Pandora, and TV.com.  I would pay that for unfettered access to Hulu.
&lt;li&gt;Hulu isn't going to win with black lists.  They're going to have to implement a white list, which is a lot more expensive to maintain.
&lt;li&gt;Not a single person who was using Hulu simply stopped watching TV when Witopia and Amazon were turned off, they just went back to bittorrent.
&lt;li&gt;Don't piss off the nerds.  They can out-spend you.
&lt;/ol&gt;
&lt;p&gt;There are plenty of cloud providers out there.  The same solution will probably work on other sites.  Install Squid and give it a try!&lt;/p&gt;&lt;div class="blogger-post-footer"&gt;&lt;img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/6471110-6099608728523879579?l=blog.jason.pollock.ca' alt='' /&gt;&lt;/div&gt;&lt;img src="http://feeds.feedburner.com/~r/JasonPollocksRandomSpewings/~4/z5Pwj-0Lbv4" height="1" width="1"/&gt;</content><link rel="replies" type="application/atom+xml" href="http://blog.jason.pollock.ca/feeds/6099608728523879579/comments/default" title="Post Comments" /><link rel="replies" type="text/html" href="https://www.blogger.com/comment.g?blogID=6471110&amp;postID=6099608728523879579" title="1 Comments" /><link rel="edit" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/6099608728523879579?v=2" /><link rel="self" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/6099608728523879579?v=2" /><link rel="alternate" type="text/html" href="http://feedproxy.google.com/~r/JasonPollocksRandomSpewings/~3/z5Pwj-0Lbv4/hulu-proxy-apocalypse.html" title="Hulu Proxy Apocalypse" /><author><name>Jason Pollock</name><uri>http://www.blogger.com/profile/10823039258279387446</uri><email>noreply@blogger.com</email><gd:extendedProperty name="OpenSocialUserId" value="15378423220352804988" /></author><thr:total xmlns:thr="http://purl.org/syndication/thread/1.0">1</thr:total><feedburner:origLink>http://blog.jason.pollock.ca/2009/11/hulu-proxy-apocalypse.html</feedburner:origLink></entry><entry gd:etag="W/&quot;D08GRX4_fip7ImA9WxNRFkw.&quot;"><id>tag:blogger.com,1999:blog-6471110.post-1342761048752291194</id><published>2009-09-05T16:58:00.000-07:00</published><updated>2009-09-10T13:50:24.046-07:00</updated><app:edited xmlns:app="http://www.w3.org/2007/app">2009-09-10T13:50:24.046-07:00</app:edited><category scheme="http://www.blogger.com/atom/ns#" term="Hulu" /><category scheme="http://www.blogger.com/atom/ns#" term="Amazon" /><category scheme="http://www.blogger.com/atom/ns#" term="S92A" /><category scheme="http://www.blogger.com/atom/ns#" term="EC2" /><title>Using Amazon EC2 to access Hulu</title><content type="html">&lt;p&gt;As more and more content moves onto the Internet, it is frequently provided on a region-by-region basis.  A lot of people want to have access to Hulu, Pandora, ABC, NBC, Netflix and the US Amazon and iTunes online stores.&lt;/p&gt;
&lt;p&gt;I was one of them.  S92A also provided an impetus to my research.  If downloading US content was going to result in disconnection, I needed another way to get my North American TV fix.  It would be even better if it was legal.&lt;/p&gt;
&lt;p&gt;A little-known section of the NZ copyright law makes it legal to break DRM if the only purpose is to provide a region lock.  To me, that indicates that if I can get around the geographic IP block on these web sites, I am no longer breaking NZ copyright law by watching the shows.&lt;/p&gt;
&lt;p&gt;Perfect.&lt;/p&gt;
&lt;p&gt;There are a couple of methods to do this.  The &lt;a href="http://alestic.com/2009/05/openvpn-ec2"&gt;ultimate method&lt;/a&gt; is using an OpenVPN server on EC2.  I didn't start there.  I started by using Squid.&lt;/p&gt;
&lt;p&gt;First, you will need to learn how to construct an Amazon EC2 instance.  This requires setting up an account, downloading the tools and starting an instance.  Nothing too difficult, and all described by &lt;a href="http://www.robertsosinski.com/2008/01/26/starting-amazon-ec2-with-mac-os-x/"&gt;Robert Sosinski&lt;/a&gt;.&lt;/p&gt;
&lt;p&gt;Since this will be a network proxy, we don't need a fast CPU, or a lot of memory.  The smallest EC2 machine image is perfectly usable.  I used a Fedora instance, since that was what I was familiar with at the time.&lt;/p&gt;
&lt;p&gt;From the steps in Robert's instructions, I would leave out allowing access to port 80 (ec2-authorize default -p 80).  We don't need it for this.&lt;/p&gt;
&lt;p&gt;Now that you have a working image, we need to get Squid working.  A funny aside, you can tell how mature the Internet is getting by the search results for open source project names.  It used to be that if you Google'ed "Squid" you got the HTTP proxy.  Now you get cephalopods.&lt;/p&gt;
&lt;p&gt;I wanted a set of instructions that could be easily scripted so that I didn't have to leave the instance running, or store anything on S3, Amazon's storage system.&lt;/p&gt;
&lt;pre&gt;
export EC2HOST=ec2-xx-xxx-xx-xx.compute-1.amazonaws.com
&lt;/pre&gt;
&lt;p&gt;First we install and start squid.&lt;/p&gt;
&lt;pre&gt;
ssh -i ec2-keypair root@$EC2HOST  "yum -y install squid"
ssh -i ec2-keypair root@$EC2HOST  "/etc/init.d/squid start"
&lt;/pre&gt;
&lt;p&gt;Finally, we setup a local tunnel from our local machine to the Squid proxy.&lt;/p&gt;
&lt;pre&gt;
ssh -i ec2-keypair -N -L3128:localhost:3128 root@$EC2HOST 
&lt;/pre&gt;
&lt;p&gt;To make use of the proxy, all we need to do is point Firefox (or your preferred browser) at localhost:3128.  Voila we now have access to the US.&lt;/p&gt;
&lt;p&gt;However, if we try to use Hulu, only some of the videos work.  We don't get the "not available in your region" error message anymore, instead we get a "unable to play the video at this time".  Something else is going on.&lt;/p&gt;
&lt;p&gt;Hulu is using multiple layers of security.  They are not only checking the source of the HTTP stream, the actual RTMPE stream is protected as well.  Time to add more stealth.&lt;/p&gt;
&lt;p&gt;First, we tell squid to not tell anyone downstream who they are proxying for:&lt;/p&gt;
&lt;pre&gt;
ssh -i ec2-keypair root@$EC2HOST  "echo "forwarded_for off &gt;&gt; /etc/squid/squid.conf"
ssh -i ec2-keypair root@$EC2HOST  "/etc/init.d/squid restart"
&lt;/pre&gt;
&lt;p&gt;However, that doesn't fix all of the problems.  Reading up on the protocol that the flash player uses (&lt;a href="http://en.wikipedia.org/wiki/Real_Time_Messaging_Protocol"&gt;RTMP&lt;/a&gt;), we see that while it will tunnel over HTTP, it will first try to make a direct connection.  It is that direct connection which is causing us problems, so we will turn it off.&lt;/p&gt;
&lt;pre&gt;
sudo ipfw add 2000 deny tcp from any to any 1935 out
&lt;/pre&gt;
&lt;p&gt;Now, when we try to use Hulu, we see that all of the videos are working, the RTMP stream is properly using the HTTP proxy, and Hulu is no longer restricting our access.&lt;/p&gt;
&lt;p&gt;However, this isn't perfect.  Amazon EC2 seems to rate limit the instances.  Even though you are paying per byte of transfer, EC2 doesn't let you have more than 1mbps per connection.  That means that while we can watch Hulu, we can't get reliable access to the HD content.&lt;/p&gt;
&lt;H4&gt;So, how expensive is it?&lt;/H4&gt;
&lt;/p&gt;Here's some math....
&lt;ul&gt;
&lt;li&gt;NZ Sky subscription, basic plan (no movies, no sports). $11.74/week
&lt;li&gt;Hulu Video
&lt;pre&gt;
Amazon Cost:
  Instance: US$0.1*(NZ$1/USD$0.67) = NZ$0.15
  Traffic (based on 1 episode of Eureka)
  (304MB/43min)*(60min/hr)*(1GB/1024MB)*(USD$0.27/1GB)*(NZD$1/USD$0.67) = NZ$0.17
  Total   : NZ$0.32/hr
NZ Bandwidth cost:
  (304MB/43min)*(60min/hr)*(1GB/1024MB)*(NZ$1.50/GB) = NZ$0.62/hr
Total cost per hour: NZ$0.94/hr
&lt;/pre&gt;
&lt;li&gt;Bittorrent cost per hour
&lt;pre&gt;
  SeedRatio = 1.0
  (350MB/40min)*(60min/hr)*(2-(1-SeedRatio))*(1GB/1024MB)*(NZ$1.50/GB) = NZ$1.53/hr
&lt;/pre&gt;
&lt;/ul&gt;
&lt;p&gt;Break even point of Squid Proxy:
&lt;ul&gt;
&lt;li&gt;vs Bittorrent - instant
&lt;li&gt;vs Sky        - 11.74/0.94 = 12 hours.
&lt;/ul&gt;
&lt;/p&gt;
&lt;p&gt;Therefore, to have any value, you need to be watching content on Sky TV which isn't available on FreeView for more than 12 hours a week in order to justify paying for Sky TV.&lt;/p&gt;
&lt;p&gt;So, in this case, going legal is cheaper.&lt;/p&gt;&lt;div class="blogger-post-footer"&gt;&lt;img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/6471110-1342761048752291194?l=blog.jason.pollock.ca' alt='' /&gt;&lt;/div&gt;&lt;img src="http://feeds.feedburner.com/~r/JasonPollocksRandomSpewings/~4/6MSfY5M7aWY" height="1" width="1"/&gt;</content><link rel="replies" type="application/atom+xml" href="http://blog.jason.pollock.ca/feeds/1342761048752291194/comments/default" title="Post Comments" /><link rel="replies" type="text/html" href="https://www.blogger.com/comment.g?blogID=6471110&amp;postID=1342761048752291194" title="1 Comments" /><link rel="edit" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/1342761048752291194?v=2" /><link rel="self" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/1342761048752291194?v=2" /><link rel="alternate" type="text/html" href="http://feedproxy.google.com/~r/JasonPollocksRandomSpewings/~3/6MSfY5M7aWY/using-amazon-ec2-to-access-hulu.html" title="Using Amazon EC2 to access Hulu" /><author><name>Jason Pollock</name><uri>http://www.blogger.com/profile/10823039258279387446</uri><email>noreply@blogger.com</email><gd:extendedProperty name="OpenSocialUserId" value="15378423220352804988" /></author><thr:total xmlns:thr="http://purl.org/syndication/thread/1.0">1</thr:total><feedburner:origLink>http://blog.jason.pollock.ca/2009/09/using-amazon-ec2-to-access-hulu.html</feedburner:origLink></entry><entry gd:etag="W/&quot;D04FSXk_cCp7ImA9WxNRFkw.&quot;"><id>tag:blogger.com,1999:blog-6471110.post-4580192827239574156</id><published>2009-03-22T22:48:00.000-07:00</published><updated>2009-09-10T13:51:58.748-07:00</updated><app:edited xmlns:app="http://www.w3.org/2007/app">2009-09-10T13:51:58.748-07:00</app:edited><category scheme="http://www.blogger.com/atom/ns#" term="PVR" /><category scheme="http://www.blogger.com/atom/ns#" term="Boxee" /><category scheme="http://www.blogger.com/atom/ns#" term="TV" /><title>The Future of Cable-TV</title><content type="html">&lt;p&gt;There's a bit of a discussion going on between &lt;a href="http://blogmaverick.com/"&gt;Mark Cuban&lt;/a&gt; and &lt;a href="blog.boxee.tv"&gt;Avner Ronen&lt;/a&gt;.  It started with someone writing an &lt;a href="http://www.contentinople.com/author.asp?section_id=450&amp;doc_id=173912"&gt;article&lt;/a&gt; about &lt;a href="www.boxee.org"&gt;boxee&lt;/a&gt;, the company that Avner heads.
&lt;/p&gt;
&lt;p&gt;Mark seemed to take exception to the article, and posted, &lt;a href="http://blogmaverick.com/2009/03/20/why-do-internet-people-think-content-people-are-stupid/"&gt;"Why Do Internet People Think Content People Are Stupid?"&lt;/a&gt;  Avner then followed up on his &lt;a href="http://blog.boxee.tv/2009/03/21/a-lively-debate-with-mark-cuban/"&gt;blog&lt;/a&gt;.&lt;/p&gt;

&lt;p&gt;I've been thinking about Cable and Broadcast TV for a while.  I find myself agreeing with Avner, that the Internet will allow customers to 
have a la carte access to shows and channels.
&lt;/p&gt;

&lt;p&gt;First off, why would this happen?  We only need to look at PVRs to see where it is going.  PVRs are changing the relationship between broadcast networks, content
creators and viewers.  Currently, a PVR changes how a viewer makes use of the TV network.  They can watch any show they are interested in, regardless of 
when it was broadcast, or even if they know it is being broadcast.  People with PVRs timeshift TV shows, frequently watching a show up to a week after broadcast.  
This change in viewing habits is showing up in the TV show's ratings, with some shows seeing a drop of up over 30% viewership as their viewers record it for 
later &lt;a href="http://danowen.blogspot.com/2009/03/tv-ratings-can-pvrs-save-dollhouse-and.html"&gt;viewing.&lt;/a&gt;&lt;/p&gt;
&lt;p&gt;What happens when the majority of your customers have PVRs?  They no longer watch "whatever is on".  They will watch the shows that they want to watch, when 
they want to watch them.  The broadcast schedule becomes less important.  In fact, it doesn't matter when the show is broadcast anymore.  Your viewers will
automatically follow the show through scheduling changes.&lt;/p&gt;
&lt;p&gt;Some consultants say that people will still want to watch the show during the broadcast slot so that they can talk about it the next day at work.
That is true, there is some pressure to watch a show as it comes out, but it doesn't need to be watched immediately.  It can be watched just as easily +1,+2,+6
hours later.  The person could even elect to watch the show the next morning over coffee or on the way to work on the train.&lt;/p&gt;
&lt;p&gt;So, if the timeslot doesn't matter, what does that do to a broadcast network?  Frankly, it turns them into two things.  First and foremost, they are aggregators.
They decide what it is that you are going to watch.  Second, they are a very efficient one to many data network.  Because of their sunk costs in the form of 
spectrum and transmitters they are able to deliver content to homes across the world cheaply.&lt;/p&gt;
&lt;p&gt;Now, remember, in the world of the PVR, it doesn't matter when a show is broadcast - it'll be found and recorded.  This lowers the value represented by prime-time slots.
Everyone is watching TV at that time, but it isn't needed to show them pre-recorded content.  It also increases the value of the 2-6AM slots.  The time when a
broadcaster would usually send out a test-signal or infomercials are now perfectly positioned to show syndicated content.&lt;/p&gt;
&lt;p&gt;Remember, it's all about feeding the PVR.  You want to fill their viewing hours with _your_ content, not the other guy's.&lt;/p&gt;
&lt;p&gt;So, we're not in a world where insanely popular shows are broadcast to people's PVRs at 2AM on a Monday morning.  Just in case you think this is crazy, this is
exactly how bittorrent works with background downloaders like &lt;a href="http://www.ted.nu/"&gt;TED&lt;/a&gt; (torrent episode downloader).  TV shows show up on the pirate networks in the middle of the night,
and are available for watching on your local PVR a couple of hours later.&lt;/p&gt;
&lt;p&gt;So, now that they're feeding your PVR, what happens next?  Aggregation.  The networks provide aggregation and editing services.  They are tastemakers.  
The only problem?  They demonstrate this taste through their line-up and schedule.  As I've already shown, the schedule is unimportant - people use PVRs.  That 
leaves only the line-up.  Those "up-next" teasers for the next show?  Of little value - the viewer can't watch the next show!  So, the line-up also becomes 
unimportant.  There is no way to package shows to make viewers more "sticky".  They will choose shows from networks a la carte.  Again, we can already see this 
behaviour - both on PVRs, and bittorrent.  There are also better "tastemakers" than networks.  They're called bloggers.  There are a lot more of them, they
are the reviewers of the Internet age.  Even better, they are perceived as more trustworthy, because they don't review things for a living.  True "tastemaking".&lt;/p&gt;
&lt;p&gt;In the world of the PVR, syndication is dead.  In the world of 500 channels, how many versions of CBS do you need?  How many times a day do you need to watch
the same episode of "The Simpsons"?  If your viewers have a PVR, the answer is you only need 1 instance.  All those cable networks that are syndicating your shows
to fill in their schedules?  If you put those shows on your own network, with your own ads, you can have that revenue too.&lt;/p&gt;
&lt;p&gt;So, the value of syndication has just collapsed.  There's only room for one copy of each episode per week (perhaps even per year).  
All that revenue broadcasters get from cable companies for retransmitting their channels?  Gone - the cable company only needs one, and they'll take the 
local one - they can frequently get that for free.&lt;/p&gt;
&lt;p&gt;Let's recap.  We've killed the schedule, aggregation and syndication.  We have just turned a local TV broadcaster into a broadband pipe that specialises in delivery 
of video and audio content supported by local advertising.&lt;/p&gt;
&lt;p&gt;Does that sound familiar?  It should.  It's an ISP running a rewriting HTTP proxy, that inserts advertisements into pages that its customers view.&lt;/p&gt;
&lt;p&gt;At that point, the model shifts.  Content producers sell their own advertisements and purchase time on the broadcast networks.  This is how I see it proceeding:&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;Content producers sell their own advertisements.&lt;/li&gt;
&lt;li&gt;They put their back catalog up on the Internet at VoD.&lt;/li&gt;
&lt;li&gt;They make the content available on a P2P network.&lt;/li&gt;
&lt;li&gt;In markets where they have significant penetration, they purchase time from the local broadcasters, and feed that information to their customers.&lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;Now that would be an interesting synergy.  You watch a show on Hulu through Boxee.  When you are finished, up pops a dialog box "Would you like to schedule this show 
for recording in your area?".  If you select yes, it instructs your local PVR to record that show the next time it comes around.&lt;/p&gt;
&lt;p&gt;All of a sudden, there is a continuous flow between Internet VoD, P2P and OTA (or cable) broadcast.  Each with their own strengths.&lt;/p&gt;
&lt;p&gt;That'll be cool.&lt;/p&gt;&lt;div class="blogger-post-footer"&gt;&lt;img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/6471110-4580192827239574156?l=blog.jason.pollock.ca' alt='' /&gt;&lt;/div&gt;&lt;img src="http://feeds.feedburner.com/~r/JasonPollocksRandomSpewings/~4/Ny-uKzo8z0c" height="1" width="1"/&gt;</content><link rel="replies" type="application/atom+xml" href="http://blog.jason.pollock.ca/feeds/4580192827239574156/comments/default" title="Post Comments" /><link rel="replies" type="text/html" href="https://www.blogger.com/comment.g?blogID=6471110&amp;postID=4580192827239574156" title="0 Comments" /><link rel="edit" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/4580192827239574156?v=2" /><link rel="self" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/4580192827239574156?v=2" /><link rel="alternate" type="text/html" href="http://feedproxy.google.com/~r/JasonPollocksRandomSpewings/~3/Ny-uKzo8z0c/future-of-cable-tv.html" title="The Future of Cable-TV" /><author><name>Jason Pollock</name><uri>http://www.blogger.com/profile/10823039258279387446</uri><email>noreply@blogger.com</email><gd:extendedProperty name="OpenSocialUserId" value="15378423220352804988" /></author><thr:total xmlns:thr="http://purl.org/syndication/thread/1.0">0</thr:total><feedburner:origLink>http://blog.jason.pollock.ca/2009/03/future-of-cable-tv.html</feedburner:origLink></entry><entry gd:etag="W/&quot;CkcFQHwycSp7ImA9WxVWFE4.&quot;"><id>tag:blogger.com,1999:blog-6471110.post-273695405503685107</id><published>2009-02-23T14:44:00.001-08:00</published><updated>2009-02-23T15:00:11.299-08:00</updated><app:edited xmlns:app="http://www.w3.org/2007/app">2009-02-23T15:00:11.299-08:00</app:edited><category scheme="http://www.blogger.com/atom/ns#" term="copyright" /><category scheme="http://www.blogger.com/atom/ns#" term="S92A" /><title>New Zealand S92A</title><content type="html">&lt;p&gt;Just to make sure that my stance isn't lost now that the blackout's been lifted.
&lt;/p&gt;
&lt;p&gt;&lt;a href="www.creativefreedom.org.nz"&gt;&lt;img src="http://creativefreedom.org.nz/library/black-out/blackout-day7.png"&gt;&lt;/a&gt;
&lt;/p&gt;
&lt;p&gt;I would like to see several changes to the TCF, including the right to see the evidence of your accusers.&lt;/p&gt;&lt;div class="blogger-post-footer"&gt;&lt;img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/6471110-273695405503685107?l=blog.jason.pollock.ca' alt='' /&gt;&lt;/div&gt;&lt;img src="http://feeds.feedburner.com/~r/JasonPollocksRandomSpewings/~4/GSAqopApIa4" height="1" width="1"/&gt;</content><link rel="replies" type="application/atom+xml" href="http://blog.jason.pollock.ca/feeds/273695405503685107/comments/default" title="Post Comments" /><link rel="replies" type="text/html" href="https://www.blogger.com/comment.g?blogID=6471110&amp;postID=273695405503685107" title="0 Comments" /><link rel="edit" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/273695405503685107?v=2" /><link rel="self" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/273695405503685107?v=2" /><link rel="alternate" type="text/html" href="http://feedproxy.google.com/~r/JasonPollocksRandomSpewings/~3/GSAqopApIa4/new-zealand-s92a.html" title="New Zealand S92A" /><author><name>Jason Pollock</name><uri>http://www.blogger.com/profile/10823039258279387446</uri><email>noreply@blogger.com</email><gd:extendedProperty name="OpenSocialUserId" value="15378423220352804988" /></author><thr:total xmlns:thr="http://purl.org/syndication/thread/1.0">0</thr:total><feedburner:origLink>http://blog.jason.pollock.ca/2009/02/new-zealand-s92a.html</feedburner:origLink></entry><entry gd:etag="W/&quot;CUUFRng7eCp7ImA9WxVXE08.&quot;"><id>tag:blogger.com,1999:blog-6471110.post-5910624936783128135</id><published>2009-02-10T19:29:00.000-08:00</published><updated>2009-02-10T19:33:37.600-08:00</updated><app:edited xmlns:app="http://www.w3.org/2007/app">2009-02-10T19:33:37.600-08:00</app:edited><category scheme="http://www.blogger.com/atom/ns#" term="prepaid" /><title>Holy Cow</title><content type="html">&lt;p&gt;Wow, the price plans are changing.  Even the prepaid plans are getting into the whole "flat rate" thing.  I love the idea of a plan where if you don't use the phone, you aren't charged for the plan.  The addition of unlimited on-net calling?  Perfect.&lt;/p&gt;
&lt;p&gt;
&lt;a href="http://www.engadgetmobile.com/2009/02/10/verizon-prepaid-pricing-changes-coming-february-11th/"&gt;Verizon prepaid pricing changes coming February 11th&lt;/a&gt;&lt;/p&gt;&lt;p&gt;
(from Engadget Mobile)
&lt;/p&gt;&lt;div class="blogger-post-footer"&gt;&lt;img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/6471110-5910624936783128135?l=blog.jason.pollock.ca' alt='' /&gt;&lt;/div&gt;&lt;img src="http://feeds.feedburner.com/~r/JasonPollocksRandomSpewings/~4/NFBz9wtrui8" height="1" width="1"/&gt;</content><link rel="replies" type="application/atom+xml" href="http://blog.jason.pollock.ca/feeds/5910624936783128135/comments/default" title="Post Comments" /><link rel="replies" type="text/html" href="https://www.blogger.com/comment.g?blogID=6471110&amp;postID=5910624936783128135" title="0 Comments" /><link rel="edit" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/5910624936783128135?v=2" /><link rel="self" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/5910624936783128135?v=2" /><link rel="alternate" type="text/html" href="http://feedproxy.google.com/~r/JasonPollocksRandomSpewings/~3/NFBz9wtrui8/holy-cow.html" title="Holy Cow" /><author><name>Jason Pollock</name><uri>http://www.blogger.com/profile/10823039258279387446</uri><email>noreply@blogger.com</email><gd:extendedProperty name="OpenSocialUserId" value="15378423220352804988" /></author><thr:total xmlns:thr="http://purl.org/syndication/thread/1.0">0</thr:total><feedburner:origLink>http://blog.jason.pollock.ca/2009/02/holy-cow.html</feedburner:origLink></entry><entry gd:etag="W/&quot;AkIDSHg6eCp7ImA9WxVXEkQ.&quot;"><id>tag:blogger.com,1999:blog-6471110.post-4477770223467628536</id><published>2009-02-10T12:27:00.000-08:00</published><updated>2009-02-10T12:42:59.610-08:00</updated><app:edited xmlns:app="http://www.w3.org/2007/app">2009-02-10T12:42:59.610-08:00</app:edited><category scheme="http://www.blogger.com/atom/ns#" term="process" /><category scheme="http://www.blogger.com/atom/ns#" term="development" /><category scheme="http://www.blogger.com/atom/ns#" term="risk managmenet" /><title>Risk Management, are you getting your money's worth?</title><content type="html">&lt;p&gt;For the first time in the better part of a decade, I'm tech lead on a new project for my current employer.  For the intervening years I've been working for other employers, other projects....  I wasn't slacking, honest!&lt;/p&gt;
&lt;p&gt;Anyways, it gives me a perfect opportunity to compare and contrast the organization of 10 years ago with today.&lt;/p&gt;
&lt;p&gt;Holy Cow!  It takes them way to long to start up a project.  They spend too long deciding what to do, and not enough time actually doing it.  Of course, this is all in the name of "risk mitigation".&lt;/p&gt;
&lt;p&gt;I'm finding it both extremely frustrating and hilarious.  It makes me want to throw things.  The company is going to spend well over 30k to make sure that the 150k they spend on the project is a success.  Even before writing the SRS.  They are going to spend 30k writing a Product Concept Document, Project Definition Workshops, Project Initiation Gate Meeting, and a Product Scoping Document with estimates.  I'm even willing to bet that I am underestimating how much they've spent on this.&lt;/p&gt;
&lt;p&gt;The funny thing?  All that work will be thrown out as soon as the SRS is written.&lt;/p&gt;
&lt;p&gt;The next shock was the testing cost.  For every day of development, there is a day of testing, more risk mitigation.  Then there's another day added for "overhead".  So, that small 100 day project?  It's actually 300 days.&lt;/p&gt;
&lt;p&gt;So, I looked at it from a math point of view.  First the project budget:&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;Pre-SRS work - 45k&lt;/li&gt;
&lt;li&gt;SRS - 45k&lt;/li&gt;
&lt;li&gt;Development effort - 100k&lt;/li&gt;
&lt;li&gt;Testing effort - 100k&lt;/li&gt;
&lt;li&gt;Management overhead - 100k&lt;/li&gt;
&lt;/ol&gt;
Total Project cost: 390k&lt;br/&gt;
Risk of complete failure: 10%&lt;/br&gt;
&lt;p&gt;Even worse, the calendar time and effort are unrelated.  The calendar time for steps 1+2, 3, 4 are all the same.&lt;/p&gt;
&lt;p&gt;Now, let's have a look at a riskier way of doing it:&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;Pre-SRS work - 15k&lt;/li&gt;
&lt;li&gt;SRS - 7k&lt;/li&gt;
&lt;li&gt;Development effort - 75k&lt;/li&gt;
&lt;li&gt;Testing effort - 50k&lt;/li&gt;
&lt;li&gt;Management overhead - 50k&lt;/li&gt;
&lt;/ol&gt;
Total Project cost: 197k&lt;br/&gt;
Risk of complete failure: 50%&lt;br/&gt;
&lt;p&gt;We've gotten rid of all of the risk mitigation.  Not only that, we've shrunk the time in steps 1+2 by 75%!  That's a huge time to market win.&lt;/p&gt;
&lt;p&gt;Let's see if the risk reward makes sense.&lt;/p&gt;
The cost of a failure is 50% (odds of failure) * the cost of the project:&lt;br/&gt;
&lt;pre&gt;
  Risky Way: 197k * 0.5 = 95k
  Safe Way: 390k * 0.1 = 39k
&lt;/pre&gt;
&lt;p&gt;Therefore, the amortized cost of a project using the:&lt;/p&gt;
&lt;pre&gt;
  Risky Way: 197k + 95k = 292k
  Safe Way:  390k + 39k = 429k
&lt;/pre&gt;

&lt;p&gt;What failure rate would be needed to justify the extra cost?  70%? 80%? 90%?  To justify the extra money spent (assuming 0% the safe way), the failure rate would have to be:&lt;/p&gt;
&lt;pre&gt;
390k - 197k
----------- = 97%
   197k
&lt;/pre&gt;

&lt;p&gt;Failure is the cost of total failure, as in the project has to be thrown away and started over.&lt;/p&gt;
&lt;p&gt;Add in the time to market benefits (on the order of 30% for these assumptions), and it starts to look pretty convincing.&lt;/p&gt;
&lt;p&gt;Everyone wonders why so many businesses are CMM level 0/1.  Have you considered that they might actually be right?&lt;/p&gt;&lt;div class="blogger-post-footer"&gt;&lt;img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/6471110-4477770223467628536?l=blog.jason.pollock.ca' alt='' /&gt;&lt;/div&gt;&lt;img src="http://feeds.feedburner.com/~r/JasonPollocksRandomSpewings/~4/CsrFTeECa_Q" height="1" width="1"/&gt;</content><link rel="replies" type="application/atom+xml" href="http://blog.jason.pollock.ca/feeds/4477770223467628536/comments/default" title="Post Comments" /><link rel="replies" type="text/html" href="https://www.blogger.com/comment.g?blogID=6471110&amp;postID=4477770223467628536" title="1 Comments" /><link rel="edit" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/4477770223467628536?v=2" /><link rel="self" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/4477770223467628536?v=2" /><link rel="alternate" type="text/html" href="http://feedproxy.google.com/~r/JasonPollocksRandomSpewings/~3/CsrFTeECa_Q/risk-management-are-you-getting-your.html" title="Risk Management, are you getting your money's worth?" /><author><name>Jason Pollock</name><uri>http://www.blogger.com/profile/10823039258279387446</uri><email>noreply@blogger.com</email><gd:extendedProperty name="OpenSocialUserId" value="15378423220352804988" /></author><thr:total xmlns:thr="http://purl.org/syndication/thread/1.0">1</thr:total><feedburner:origLink>http://blog.jason.pollock.ca/2009/02/risk-management-are-you-getting-your.html</feedburner:origLink></entry><entry gd:etag="W/&quot;DUECQXs5fyp7ImA9WxVREEg.&quot;"><id>tag:blogger.com,1999:blog-6471110.post-5130372647796135937</id><published>2009-01-15T14:10:00.000-08:00</published><updated>2009-01-15T14:14:20.527-08:00</updated><app:edited xmlns:app="http://www.w3.org/2007/app">2009-01-15T14:14:20.527-08:00</app:edited><title>Prepaid is Dead, redux.</title><content type="html">&lt;p&gt;I just saw this in the news today:&lt;/p&gt;
&lt;p&gt;&lt;a href="http://www.reuters.com/article/technologyNews/idUSTRE50E0BU20090115"&gt;Boost sees $50 unlimited plan battling Leap, Metro&lt;/a&gt;
&lt;/p&gt;
&lt;p&gt;Unlimited calling and texting for US$50/month.  It's a race to the bottom with all-you-can-eat plans.&lt;/p&gt;
&lt;p&gt;If you're charging per minute, per sms, per byte, the question is "why?".  Save yourself a lot of money and quit billing for the core service!  That Intec Billing Engine you're thinking of buying?  Get rid of it, or use it for something other than charging for calls.&lt;/p&gt;&lt;div class="blogger-post-footer"&gt;&lt;img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/6471110-5130372647796135937?l=blog.jason.pollock.ca' alt='' /&gt;&lt;/div&gt;&lt;img src="http://feeds.feedburner.com/~r/JasonPollocksRandomSpewings/~4/J_W_yrICFrQ" height="1" width="1"/&gt;</content><link rel="replies" type="application/atom+xml" href="http://blog.jason.pollock.ca/feeds/5130372647796135937/comments/default" title="Post Comments" /><link rel="replies" type="text/html" href="https://www.blogger.com/comment.g?blogID=6471110&amp;postID=5130372647796135937" title="2 Comments" /><link rel="edit" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/5130372647796135937?v=2" /><link rel="self" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/5130372647796135937?v=2" /><link rel="alternate" type="text/html" href="http://feedproxy.google.com/~r/JasonPollocksRandomSpewings/~3/J_W_yrICFrQ/prepaid-is-dead-redux.html" title="Prepaid is Dead, redux." /><author><name>Jason Pollock</name><uri>http://www.blogger.com/profile/10823039258279387446</uri><email>noreply@blogger.com</email><gd:extendedProperty name="OpenSocialUserId" value="15378423220352804988" /></author><thr:total xmlns:thr="http://purl.org/syndication/thread/1.0">2</thr:total><feedburner:origLink>http://blog.jason.pollock.ca/2009/01/prepaid-is-dead-redux.html</feedburner:origLink></entry><entry gd:etag="W/&quot;CUEFSHk6eSp7ImA9WxRUFkg.&quot;"><id>tag:blogger.com,1999:blog-6471110.post-9028754709571241657</id><published>2008-11-25T14:50:00.000-08:00</published><updated>2008-11-25T14:53:39.711-08:00</updated><app:edited xmlns:app="http://www.w3.org/2007/app">2008-11-25T14:53:39.711-08:00</app:edited><title>Adding value?</title><content type="html">&lt;p&gt;Ask yourself, "Can I be replaced with a shell script?"  Specifically:&lt;/p&gt;
&lt;pre&gt;
 % yes 'No'
&lt;/pre&gt;
&lt;p&gt;Then ask yourself, "What value am I adding?"&lt;/p&gt;&lt;div class="blogger-post-footer"&gt;&lt;img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/6471110-9028754709571241657?l=blog.jason.pollock.ca' alt='' /&gt;&lt;/div&gt;&lt;img src="http://feeds.feedburner.com/~r/JasonPollocksRandomSpewings/~4/FB5c7DohFZw" height="1" width="1"/&gt;</content><link rel="replies" type="application/atom+xml" href="http://blog.jason.pollock.ca/feeds/9028754709571241657/comments/default" title="Post Comments" /><link rel="replies" type="text/html" href="https://www.blogger.com/comment.g?blogID=6471110&amp;postID=9028754709571241657" title="0 Comments" /><link rel="edit" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/9028754709571241657?v=2" /><link rel="self" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/9028754709571241657?v=2" /><link rel="alternate" type="text/html" href="http://feedproxy.google.com/~r/JasonPollocksRandomSpewings/~3/FB5c7DohFZw/adding-value.html" title="Adding value?" /><author><name>Jason Pollock</name><uri>http://www.blogger.com/profile/10823039258279387446</uri><email>noreply@blogger.com</email><gd:extendedProperty name="OpenSocialUserId" value="15378423220352804988" /></author><thr:total xmlns:thr="http://purl.org/syndication/thread/1.0">0</thr:total><feedburner:origLink>http://blog.jason.pollock.ca/2008/11/adding-value.html</feedburner:origLink></entry><entry gd:etag="W/&quot;D0cNQ389fCp7ImA9WxRTGUQ.&quot;"><id>tag:blogger.com,1999:blog-6471110.post-1103247895727134144</id><published>2008-09-09T14:05:00.000-07:00</published><updated>2008-09-09T14:18:12.164-07:00</updated><app:edited xmlns:app="http://www.w3.org/2007/app">2008-09-09T14:18:12.164-07:00</app:edited><category scheme="http://www.blogger.com/atom/ns#" term="Google" /><category scheme="http://www.blogger.com/atom/ns#" term="Performance" /><category scheme="http://www.blogger.com/atom/ns#" term="Google Chrome" /><category scheme="http://www.blogger.com/atom/ns#" term="Firefox" /><title>Google Chrome, a Contrarian View.</title><content type="html">&lt;p&gt;I read the comic about Google Chrome.  I love the architecture!  We downloaded it here in the office, and sure enough - it forks a new process for every page load.  This is the browser architecture that I've been waiting years for, one where a script on one page won't kill performance in another window.
&lt;/p&gt;&lt;p&gt;
So, I went home and tried it on my single core Athlon at home.  Ouch!  The performance is terrible.  The browser keeps locking up, it stops responding to mouse clicks, keypresses, anything and everything.  It acts like alpha quality software, not the "beta" it is supposed to be.
&lt;/p&gt;&lt;p&gt;
I've got a 2 year old system (Athlon64 3800+), the only difference between it and new ones is that it is single core.  It's still fast enough to play new games on, but it can't handle Google Chrome.
&lt;/p&gt;&lt;p&gt;
I can only assume that Google Chrome has a tendency to starve some of their child processes on single core machines.
&lt;/p&gt;&lt;p&gt;
In the meantime, I'll stick with Firefox 3.  It leaves Google Chrome in the dust.
&lt;/p&gt;&lt;div class="blogger-post-footer"&gt;&lt;img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/6471110-1103247895727134144?l=blog.jason.pollock.ca' alt='' /&gt;&lt;/div&gt;&lt;img src="http://feeds.feedburner.com/~r/JasonPollocksRandomSpewings/~4/fAza7hl4CIE" height="1" width="1"/&gt;</content><link rel="replies" type="application/atom+xml" href="http://blog.jason.pollock.ca/feeds/1103247895727134144/comments/default" title="Post Comments" /><link rel="replies" type="text/html" href="https://www.blogger.com/comment.g?blogID=6471110&amp;postID=1103247895727134144" title="3 Comments" /><link rel="edit" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/1103247895727134144?v=2" /><link rel="self" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/1103247895727134144?v=2" /><link rel="alternate" type="text/html" href="http://feedproxy.google.com/~r/JasonPollocksRandomSpewings/~3/fAza7hl4CIE/google-chrome-contrarian-view.html" title="Google Chrome, a Contrarian View." /><author><name>Jason Pollock</name><uri>http://www.blogger.com/profile/10823039258279387446</uri><email>noreply@blogger.com</email><gd:extendedProperty name="OpenSocialUserId" value="15378423220352804988" /></author><thr:total xmlns:thr="http://purl.org/syndication/thread/1.0">3</thr:total><feedburner:origLink>http://blog.jason.pollock.ca/2008/09/google-chrome-contrarian-view.html</feedburner:origLink></entry><entry gd:etag="W/&quot;A0UFQXg4fCp7ImA9WxdbEEg.&quot;"><id>tag:blogger.com,1999:blog-6471110.post-5848966330602697175</id><published>2008-08-06T14:37:00.000-07:00</published><updated>2008-08-06T14:46:50.634-07:00</updated><app:edited xmlns:app="http://www.w3.org/2007/app">2008-08-06T14:46:50.634-07:00</app:edited><category scheme="http://www.blogger.com/atom/ns#" term="rant" /><category scheme="http://www.blogger.com/atom/ns#" term="python" /><category scheme="http://www.blogger.com/atom/ns#" term="duck typing" /><title>Duck Typing Sucks.</title><content type="html">&lt;p&gt;I'm working with some code that's written in Python, and makes extensive use of duck typing.  In duck typing the premise is that if it quacks like a duck, it is a duck.
&lt;/p&gt;
&lt;p&gt;Of course, if you've got code that doesn't tell you &lt;b&gt;what&lt;/b&gt; it expects to receive, you can't hope to figure out what the type should be.
&lt;/p&gt;
&lt;p&gt;For example, let's say you've got 3 code blocks:
&lt;/p&gt;
&lt;pre&gt;
class A:
    def funkyDuck():
        # random cool stuff

class B:
    def funkyDuck():
        # entirely different cool stuff

class C:
    def funkyDuck():
        # a third entirely cool thing

&lt;/pre&gt;

Now, imagine you're looking at code that goes:

&lt;pre&gt;
some_random_object.funkyDuck()
&lt;/pre&gt;
Which one gets called?  By the way, they're all in different files, so the only way to find them is with "grep".

Added bonus, the undocumented hash!

&lt;pre&gt;
    def addApplications(self, appDict):
        """ Add some application modules to the defaults.
            All binaries and libraries in any of the source dirs will be soft linked
            to a directory with the name of the product.
            @type  appDict: dict of string -&gt; (list, string)
            @param appDict: The source dirs and OS user name for each product
        """
&lt;/pre&gt;

&lt;p&gt;Guess I'm supposed to guess what appDict actually looks like.
&lt;/p&gt;&lt;p&gt;Hrm.  Perhaps my problem isn't with duck typing itself, but it's use here.
&lt;/p&gt;&lt;p&gt;Hint: When your test framework is just as complicated as the program under test, you're doing something very, very wrong.&lt;/p&gt;&lt;div class="blogger-post-footer"&gt;&lt;img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/6471110-5848966330602697175?l=blog.jason.pollock.ca' alt='' /&gt;&lt;/div&gt;&lt;img src="http://feeds.feedburner.com/~r/JasonPollocksRandomSpewings/~4/tC1rC2892QU" height="1" width="1"/&gt;</content><link rel="replies" type="application/atom+xml" href="http://blog.jason.pollock.ca/feeds/5848966330602697175/comments/default" title="Post Comments" /><link rel="replies" type="text/html" href="https://www.blogger.com/comment.g?blogID=6471110&amp;postID=5848966330602697175" title="4 Comments" /><link rel="edit" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/5848966330602697175?v=2" /><link rel="self" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/5848966330602697175?v=2" /><link rel="alternate" type="text/html" href="http://feedproxy.google.com/~r/JasonPollocksRandomSpewings/~3/tC1rC2892QU/duck-typing-sucks.html" title="Duck Typing Sucks." /><author><name>Jason Pollock</name><uri>http://www.blogger.com/profile/10823039258279387446</uri><email>noreply@blogger.com</email><gd:extendedProperty name="OpenSocialUserId" value="15378423220352804988" /></author><thr:total xmlns:thr="http://purl.org/syndication/thread/1.0">4</thr:total><feedburner:origLink>http://blog.jason.pollock.ca/2008/08/duck-typing-sucks.html</feedburner:origLink></entry><entry gd:etag="W/&quot;DkEBRXs-eyp7ImA9WxdVF08.&quot;"><id>tag:blogger.com,1999:blog-6471110.post-979367451698669792</id><published>2008-07-22T03:57:00.000-07:00</published><updated>2008-07-22T04:04:14.553-07:00</updated><app:edited xmlns:app="http://www.w3.org/2007/app">2008-07-22T04:04:14.553-07:00</app:edited><category scheme="http://www.blogger.com/atom/ns#" term="poison" /><category scheme="http://www.blogger.com/atom/ns#" term="multi-threading" /><category scheme="http://www.blogger.com/atom/ns#" term="DNS" /><category scheme="http://www.blogger.com/atom/ns#" term="attack" /><category scheme="http://www.blogger.com/atom/ns#" term="Bug" /><category scheme="http://www.blogger.com/atom/ns#" term="math" /><title>One in a Million</title><content type="html">&lt;p&gt; I remember discussing &lt;a href="http://arstechnica.com/articles/paedia/cpu/hyperthreading.ars"&gt;multi-threaded programming&lt;/a&gt; with a senior engineer the better part of a decade ago.  I was reacting rather strongly to his cavalier attitude towards multi-threaded code, deadlocks and race conditions.&lt;/p&gt;
&lt;blockquote&gt;
"That is such a small hole, it's one in a million that it will happen."
&lt;/blockquote&gt;
&lt;p&gt;At the time, we were a small company, and I was a young engineer.  One in a million, that doesn't sound that big.  Of course, one in a million happens more frequently than you would expect.  It didn't take very long for our systems to be processing a million transactions a day, and then millions per hour.  Every time the holes had to get smaller and smaller.&lt;/p&gt;
&lt;p&gt;They still have a fault in there where the entire system crashes when the system goes completely idle for a couple of minutes.  It doesn't happen all the time, it's a "one in about 20 million", but it only has to happen once.  So, it crashed about once a week.  That one took forever to track down.  Still haven't been able to fix it entirely, but they know that it's there, and the hole is smaller, probably "1 in a couple billion" now, below the rate that it's restarted for maintenance.  Fixed, for now.&lt;/p&gt;
&lt;p&gt;It's interesting see that the DNS engineers are learning the &lt;a href="http://addxorrol.blogspot.com/2008/07/on-dans-request-for-no-speculation.html"&gt;same lesson&lt;/a&gt;.  Compute power has now shifted in favour of the attackers.  As we saw with &lt;a href="http://en.wikipedia.org/wiki/Captcha"&gt;Captcha&lt;/a&gt;, it doesn't matter that you only get through rarely - you only have to get through once in a while.  Even just guessing a DNS TXID (2^16), you've got a 1 in a 65536 chance of guessing the right answer.&lt;/p&gt;
&lt;p&gt;In pure mathematical terms, the likelihood of getting the TXID &lt;b&gt;wrong&lt;/b&gt; 400,000 times in a row is 0.2%.  In other words, pretty certain.&lt;/p&gt;
&lt;pre&gt;(1-(1/2^16)))^400000)&lt;/pre&gt;
&lt;p&gt;Let's look at the 50/50 point, where the likelihood of seeing X number of failures is 0.5.&lt;/p&gt;
&lt;p&gt;That turns out to be somewhere around 45000 attempts, simple for loop range.  Since I am spoofing packets, I don't care about a response, even better, filling up the pipe will expand the race situation, giving me a larger chance to get through.&lt;/p&gt;
&lt;p&gt;Still, it sounds like the attacker has to also be requesting lookups to poison the server.  Let's say they take 2ms each, and they're being nice by only doing them one at a time.  Even using the large 400k count, it will take at most 15 minutes of concerted effort to poison the cache.  The 50/50 point?  One minute, 20 seconds.&lt;/p&gt;
&lt;p&gt;Look out for those one in a million situations, they happen more often than you'd like.&lt;/p&gt;&lt;div class="blogger-post-footer"&gt;&lt;img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/6471110-979367451698669792?l=blog.jason.pollock.ca' alt='' /&gt;&lt;/div&gt;&lt;img src="http://feeds.feedburner.com/~r/JasonPollocksRandomSpewings/~4/PnFJMwKnXSY" height="1" width="1"/&gt;</content><link rel="replies" type="application/atom+xml" href="http://blog.jason.pollock.ca/feeds/979367451698669792/comments/default" title="Post Comments" /><link rel="replies" type="text/html" href="https://www.blogger.com/comment.g?blogID=6471110&amp;postID=979367451698669792" title="0 Comments" /><link rel="edit" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/979367451698669792?v=2" /><link rel="self" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/979367451698669792?v=2" /><link rel="alternate" type="text/html" href="http://feedproxy.google.com/~r/JasonPollocksRandomSpewings/~3/PnFJMwKnXSY/one-in-million.html" title="One in a Million" /><author><name>Jason Pollock</name><uri>http://www.blogger.com/profile/10823039258279387446</uri><email>noreply@blogger.com</email><gd:extendedProperty name="OpenSocialUserId" value="15378423220352804988" /></author><thr:total xmlns:thr="http://purl.org/syndication/thread/1.0">0</thr:total><feedburner:origLink>http://blog.jason.pollock.ca/2008/07/one-in-million.html</feedburner:origLink></entry><entry gd:etag="W/&quot;DU8DQXg5eip7ImA9WxdVFkU.&quot;"><id>tag:blogger.com,1999:blog-6471110.post-4890307209547378734</id><published>2008-07-21T17:48:00.000-07:00</published><updated>2008-07-21T17:51:10.622-07:00</updated><app:edited xmlns:app="http://www.w3.org/2007/app">2008-07-21T17:51:10.622-07:00</app:edited><category scheme="http://www.blogger.com/atom/ns#" term="hungarian notation" /><category scheme="http://www.blogger.com/atom/ns#" term="document names" /><title>Naming Documents</title><content type="html">&lt;p&gt;I thought we killed off &lt;a href="http://en.wikipedia.org/wiki/Hungarian_notation"&gt;Hungarian Notation&lt;/a&gt; because it doesn't work?  Did I miss a memo somewhere?  It seems that Hungarian Notation is making a comeback, this time in document names.&lt;/p&gt;&lt;p&gt;There's nothing like having your meta data encoded in the file name, with three letter acronyms for everything.  It is impossible to decode, and becomes really fun when the team names change once a year!&lt;/p&gt;
&lt;i&gt;
"Say where's the SRS?"

"Is it in SRS.doc?"

"Nope"

"How about esgEng_SIT_DR4_SRS.doc?"
"Nope"

"Oh, wait, Engineering was renamed last year for 6 months....  How about esg_PEN_SIT_DR3_SRS.doc?"

"Ah, that's got it."

&lt;/i&gt;&lt;p&gt;&lt;/p&gt;&lt;p&gt;Encoding meta data in the filename is stupid.  How about putting it in the file in the meta data portion where it belongs, and then using a search tool to search it?  Or, maybe a directory structure that represents your tag tree.&lt;/p&gt;
&lt;pre&gt;
esgEng/SIT/DR4/SRS.doc
&lt;/pre&gt;
&lt;p&gt;Then, when you need to change the meta data, you rename the tree:&lt;/p&gt;
&lt;pre&gt;esg/PEN/SIT/DR3/SRS.doc&lt;/pre&gt;&lt;p&gt;But Jason, I need to know who produced the document without opening it!&lt;/p&gt;&lt;p&gt;Use the checksum, and search/store that.  Anything else can be mistakenly altered or lost.  Use the file's checksum.  It is a much more reliable descriptor of the file than the filename!  I have learned from experience to, never, ever trust a filename, they lie.&lt;/p&gt;&lt;p&gt;Or better yet, you buy a document repository off the shelf for 100k, and shove
the problem at them.  Personally, I just use the Google Search Appliance.  Now that they've got it looking at the correct bits of data, it's very useful.  Much better than any of the searches built into the various corporate portals.&lt;/p&gt;&lt;div class="blogger-post-footer"&gt;&lt;img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/6471110-4890307209547378734?l=blog.jason.pollock.ca' alt='' /&gt;&lt;/div&gt;&lt;img src="http://feeds.feedburner.com/~r/JasonPollocksRandomSpewings/~4/7soc2KNiPH0" height="1" width="1"/&gt;</content><link rel="replies" type="application/atom+xml" href="http://blog.jason.pollock.ca/feeds/4890307209547378734/comments/default" title="Post Comments" /><link rel="replies" type="text/html" href="https://www.blogger.com/comment.g?blogID=6471110&amp;postID=4890307209547378734" title="0 Comments" /><link rel="edit" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/4890307209547378734?v=2" /><link rel="self" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/4890307209547378734?v=2" /><link rel="alternate" type="text/html" href="http://feedproxy.google.com/~r/JasonPollocksRandomSpewings/~3/7soc2KNiPH0/naming-documents.html" title="Naming Documents" /><author><name>Jason Pollock</name><uri>http://www.blogger.com/profile/10823039258279387446</uri><email>noreply@blogger.com</email><gd:extendedProperty name="OpenSocialUserId" value="15378423220352804988" /></author><thr:total xmlns:thr="http://purl.org/syndication/thread/1.0">0</thr:total><feedburner:origLink>http://blog.jason.pollock.ca/2008/07/naming-documents.html</feedburner:origLink></entry><entry gd:etag="W/&quot;D04FRHYzeip7ImA9WxdWGEw.&quot;"><id>tag:blogger.com,1999:blog-6471110.post-1629391695724060533</id><published>2008-07-11T15:35:00.000-07:00</published><updated>2008-07-11T15:38:35.882-07:00</updated><app:edited xmlns:app="http://www.w3.org/2007/app">2008-07-11T15:38:35.882-07:00</app:edited><category scheme="http://www.blogger.com/atom/ns#" term="OSX" /><category scheme="http://www.blogger.com/atom/ns#" term="vmware" /><category scheme="http://www.blogger.com/atom/ns#" term="Apple" /><category scheme="http://www.blogger.com/atom/ns#" term="starcraft" /><category scheme="http://www.blogger.com/atom/ns#" term="diablo ii" /><title>Getting Starcraft working on OSX</title><content type="html">&lt;p&gt;I've started playing Starcraft again after ages away, but I found that my brand spanking new MacBook Pro wouldn't work!&lt;/p&gt;
&lt;p&gt;The new versions of OSX have an updated video driver which does not support 256 colour modes.  This means that &lt;a href="www.blizzard.com/starcraft"&gt;Starcraft&lt;/a&gt;, &lt;a href="http://www.blizzard.com/diablo2/"&gt;Diablo II&lt;/a&gt;, or basically any older game will &lt;a href="http://discussions.apple.com/thread.jspa?messageID=7513113"&gt;no longer work&lt;/a&gt;.  Not good when the applications were still sold by Apple as OSX compatible games (they've since been removed).&lt;/p&gt;
&lt;p&gt;I have now figured out a simple (albeit more expensive - US$79.99 + Windows license) way to solve the problem.  I decided to run the game under &lt;a href="http://www.vmware.com/products/fusion/"&gt;VMWare Fusion&lt;/a&gt;.&lt;/p&gt;
&lt;p&gt;However there were two problems with this.  First, when running in full screen mode, the display doesn't stretch to fill the screen.  So on my laptop, I end up with a postage stamp display right in the middle of the screen.  Second, the mouse is not restricted to that little postage stamp, it moves freely all the way around the display.  When attempting to scroll the display in the game, you move the mouse to the edge, since the mouse will leave the bounds, the game doesn't scroll very well.&lt;/p&gt;
&lt;p&gt;The first was fixed with a quick google. &lt;/p&gt;
&lt;p&gt;In the file "Preferences/VMWare Fusion/preferences" add&lt;/p&gt;
&lt;pre&gt;
    pref.autoFitFullScreen = "fitHostToGuest"
&lt;/pre&gt;
&lt;p&gt;This will stretch the display.  If you have a widescreen display, you will still have black bars on the left and the right, but we're making progress.&lt;/p&gt;
&lt;p&gt;Now to mouse capture.  There doesn't appear to be an option to prevent the mouse from being taken back by the host OS, so I took a more drastic approach.  I uninstalled VMWare Tools.  This requires a reboot of the VM, but when you are done, you will no longer be able to move the mouse outside of the VM!&lt;/p&gt;
&lt;p&gt;To get back to the host OS, press CTRL-CMD.  To re-install VMWare Toole, select "Install VMWare Tools" under "Virtual Machine"&lt;/p&gt;
&lt;p&gt;You should now be able to play Diablo II and Starcraft in all their 256 colour beauty.&lt;/p&gt;
&lt;p&gt;Now, if I can only control my nerves enough to work my trackball....&lt;/p&gt;&lt;div class="blogger-post-footer"&gt;&lt;img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/6471110-1629391695724060533?l=blog.jason.pollock.ca' alt='' /&gt;&lt;/div&gt;&lt;img src="http://feeds.feedburner.com/~r/JasonPollocksRandomSpewings/~4/8hOlci_oD2o" height="1" width="1"/&gt;</content><link rel="replies" type="application/atom+xml" href="http://blog.jason.pollock.ca/feeds/1629391695724060533/comments/default" title="Post Comments" /><link rel="replies" type="text/html" href="https://www.blogger.com/comment.g?blogID=6471110&amp;postID=1629391695724060533" title="0 Comments" /><link rel="edit" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/1629391695724060533?v=2" /><link rel="self" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/1629391695724060533?v=2" /><link rel="alternate" type="text/html" href="http://feedproxy.google.com/~r/JasonPollocksRandomSpewings/~3/8hOlci_oD2o/getting-starcraft-working-on-osx.html" title="Getting Starcraft working on OSX" /><author><name>Jason Pollock</name><uri>http://www.blogger.com/profile/10823039258279387446</uri><email>noreply@blogger.com</email><gd:extendedProperty name="OpenSocialUserId" value="15378423220352804988" /></author><thr:total xmlns:thr="http://purl.org/syndication/thread/1.0">0</thr:total><feedburner:origLink>http://blog.jason.pollock.ca/2008/07/getting-starcraft-working-on-osx.html</feedburner:origLink></entry><entry gd:etag="W/&quot;DE4DRnwzcCp7ImA9WxdWF08.&quot;"><id>tag:blogger.com,1999:blog-6471110.post-2477493483591398243</id><published>2008-07-10T14:39:00.000-07:00</published><updated>2008-07-10T14:56:17.288-07:00</updated><app:edited xmlns:app="http://www.w3.org/2007/app">2008-07-10T14:56:17.288-07:00</app:edited><category scheme="http://www.blogger.com/atom/ns#" term="process" /><category scheme="http://www.blogger.com/atom/ns#" term="airlines" /><category scheme="http://www.blogger.com/atom/ns#" term="efficiency" /><title>Keeping Efficiency Gains</title><content type="html">&lt;p&gt;&lt;a href="http://www.aircanada.com/"&gt;Air Canada&lt;/a&gt; has implemented &lt;a href="http://www.aircanada.com/en/travelinfo/traveller/checkin/checkin.html"&gt;self check-in&lt;/a&gt; at &lt;a href="http://www.gtaa.com/en/home/"&gt;Pearson Airport&lt;/a&gt; in Toronto.  For all flights, you no longer go to an attendant to obtain you boarding pass, you enter all the required data into a computer, and then check your bags.&lt;/p&gt;
&lt;p&gt;This was supposedly done for efficiency reasons.  However, most of the efficiencies in the system have been lost.  Amazingly, when you get to the front of the second line (to check in your bags), the check in clerk does &lt;u&gt;exactly&lt;/u&gt; the same amount of work as before.  The first thing they do is verify all of the details that you provided the computer!&lt;/p&gt;&lt;p&gt;Let's go over that again.  After having provided my information to a computer, I have to provide it a second time to a human who verifies
the first data.  The time taken for verification was exactly the same as it would have taken for them to key it in in the first time.  Instantly, you have a net loss of efficiency.  Even worse, they only have a single set of scales for each pair of desks.  That means that there is substantial dead time while you wait for the person at the desk next to you to finish weighing their bags.  More lost efficiency.&lt;/p&gt;&lt;p&gt;I can see how it happened.  Someone checked in to the wrong flight, or their bags went to the wrong place.  Perhaps they got to US customs (you go through US customs in Canada when flying to the US), and didn't have the correct forms or all their data provided, and were sent back, maybe they even missed a flight.&lt;/p&gt;
&lt;p&gt;So, the check-in clerks, who are also looking to protect their jobs, add in the task of checking customer provided data.&lt;/p&gt;
&lt;p&gt;However, they don't check the data of people who don't have checked baggage, they go straight through to customs.  This shows the stupidity of the additional check.  Is Air Canada saying that people with checked luggage are more likely to enter incorrect data?  I doubt it.&lt;/p&gt;&lt;p&gt;If you make efficiency gains in your organisation, make sure you protect them.  Guard them jealously. If you don't, you will see them frittered away.&lt;/p&gt;&lt;div class="blogger-post-footer"&gt;&lt;img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/6471110-2477493483591398243?l=blog.jason.pollock.ca' alt='' /&gt;&lt;/div&gt;&lt;img src="http://feeds.feedburner.com/~r/JasonPollocksRandomSpewings/~4/4aOhxBAjms8" height="1" width="1"/&gt;</content><link rel="replies" type="application/atom+xml" href="http://blog.jason.pollock.ca/feeds/2477493483591398243/comments/default" title="Post Comments" /><link rel="replies" type="text/html" href="https://www.blogger.com/comment.g?blogID=6471110&amp;postID=2477493483591398243" title="0 Comments" /><link rel="edit" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/2477493483591398243?v=2" /><link rel="self" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/2477493483591398243?v=2" /><link rel="alternate" type="text/html" href="http://feedproxy.google.com/~r/JasonPollocksRandomSpewings/~3/4aOhxBAjms8/keeping-efficiency-gains.html" title="Keeping Efficiency Gains" /><author><name>Jason Pollock</name><uri>http://www.blogger.com/profile/10823039258279387446</uri><email>noreply@blogger.com</email><gd:extendedProperty name="OpenSocialUserId" value="15378423220352804988" /></author><thr:total xmlns:thr="http://purl.org/syndication/thread/1.0">0</thr:total><feedburner:origLink>http://blog.jason.pollock.ca/2008/07/keeping-efficiency-gains.html</feedburner:origLink></entry><entry gd:etag="W/&quot;C04BRXcyfCp7ImA9WxdQEkQ.&quot;"><id>tag:blogger.com,1999:blog-6471110.post-4339295364981782816</id><published>2008-06-12T10:29:00.000-07:00</published><updated>2008-06-12T10:32:34.994-07:00</updated><app:edited xmlns:app="http://www.w3.org/2007/app">2008-06-12T10:32:34.994-07:00</app:edited><title>The Truth about Vacations.</title><content type="html">This invariably happens to me.  I go on vacation, and then the server goes down.  It doesn't go down during the year.  It's had an uptime of over a year!  However, it's down and stayed down.

So, if you've been trying to email me, I apologize, it won't get through.  The same goes if you're wondering where blog.jason.pollock.ca went to, or why I'm not on jabber anymore.

I'm trying to get it going again, but email will probably be down until I get back home.  The eBook Repositories will definitely not be updating, so the Baen Books will slowly rot.&lt;div class="blogger-post-footer"&gt;&lt;img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/6471110-4339295364981782816?l=blog.jason.pollock.ca' alt='' /&gt;&lt;/div&gt;&lt;img src="http://feeds.feedburner.com/~r/JasonPollocksRandomSpewings/~4/DP9kdU-o2Fw" height="1" width="1"/&gt;</content><link rel="replies" type="application/atom+xml" href="http://blog.jason.pollock.ca/feeds/4339295364981782816/comments/default" title="Post Comments" /><link rel="replies" type="text/html" href="https://www.blogger.com/comment.g?blogID=6471110&amp;postID=4339295364981782816" title="0 Comments" /><link rel="edit" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/4339295364981782816?v=2" /><link rel="self" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/4339295364981782816?v=2" /><link rel="alternate" type="text/html" href="http://feedproxy.google.com/~r/JasonPollocksRandomSpewings/~3/DP9kdU-o2Fw/truth-about-vacations.html" title="The Truth about Vacations." /><author><name>Jason Pollock</name><uri>http://www.blogger.com/profile/10823039258279387446</uri><email>noreply@blogger.com</email><gd:extendedProperty name="OpenSocialUserId" value="15378423220352804988" /></author><thr:total xmlns:thr="http://purl.org/syndication/thread/1.0">0</thr:total><feedburner:origLink>http://blog.jason.pollock.ca/2008/06/truth-about-vacations.html</feedburner:origLink></entry><entry gd:etag="W/&quot;DkIEQHw5eyp7ImA9WxdRGEU.&quot;"><id>tag:blogger.com,1999:blog-6471110.post-4845321801994767465</id><published>2008-06-07T17:09:00.000-07:00</published><updated>2008-06-07T17:21:41.223-07:00</updated><app:edited xmlns:app="http://www.w3.org/2007/app">2008-06-07T17:21:41.223-07:00</app:edited><category scheme="http://www.blogger.com/atom/ns#" term="iPhone" /><category scheme="http://www.blogger.com/atom/ns#" term="AppTapp Installer" /><category scheme="http://www.blogger.com/atom/ns#" term="Installer.app" /><title>Installer.app and chmod/chown</title><content type="html">&lt;p&gt;&lt;i&gt;
How is it that I would go about chmodding the directories created by CopyPath?
&lt;/i&gt;&lt;/p&gt;

&lt;p&gt;
It seems that if the .zip contains the permissions already, they are maintained when used by CopyPath.
&lt;/p&gt;
&lt;p&gt;
In other words, if you have:
&lt;/p&gt;

&lt;pre&gt;
foo/bar
&lt;/pre&gt;
&lt;p&gt;
in your .zip, and do a CopyPath("foo", "~/foo") in the XML document, the permissions are set up correctly.  At least I haven't had any problems if my installer looks like that.
&lt;/p&gt;
&lt;p&gt;
However, if you create implicitly create directories by using CopyPath on the individual files, they are created without permissions, and a chmod/chown is needed.  For example, CopyPath("foo/bar", "nuts/baz/bar") will create "nuts/baz", with no permissions.
&lt;/p&gt;
&lt;p&gt;
I have to do this with the Gutenberg SciFi and Baen Books libraries, since I don't host the .zips themselves.
&lt;/p&gt;
&lt;p&gt;
To manage the chmod/chown problem, the packages ensure that the BSD subsystem is installed (or Cydia), and then runs them as part of the install steps.
&lt;/p&gt;
&lt;p&gt;
Here is an example:
&lt;/p&gt;
&lt;pre&gt;
      &amp;lt;key&amp;gt;preflight&amp;lt;/key&amp;gt;
      &amp;lt;array&amp;gt;
        &amp;lt;array&amp;gt;
          &amp;lt;string&amp;gt;IfNot&amp;lt;/string&amp;gt;
          &amp;lt;array&amp;gt;
              &amp;lt;array&amp;gt;
                &amp;lt;string&amp;gt;InstalledPackage&amp;lt;/string&amp;gt;
                &amp;lt;string&amp;gt;com.natetrue.iphone.iphone_binkit&amp;lt;/string&amp;gt;
              &amp;lt;/array&amp;gt;
          &amp;lt;/array&amp;gt;
          &amp;lt;array&amp;gt;
              &amp;lt;array&amp;gt;
                &amp;lt;string&amp;gt;AbortOperation&amp;lt;/string&amp;gt;
                &amp;lt;string&amp;gt;You must install the "BSD Subsystem" package from Installer first!  It's in the System Category.&amp;lt;/string&amp;gt;
              &amp;lt;/array&amp;gt;
          &amp;lt;/array&amp;gt;
        &amp;lt;/array&amp;gt;
      &amp;lt;/array&amp;gt;
&lt;/pre&gt;
&lt;p&gt;
Which checks for the BSD subsystem as part of the pre-install.  If the BSD subsystem is not installed, the user gets a pop-up error message and the package will not install.
&lt;/p&gt;
&lt;p&gt;
And then the install
&lt;/p&gt;
&lt;pre&gt;
      &amp;lt;key&amp;gt;install&amp;lt;/key&amp;gt;
      &amp;lt;array&amp;gt;
        &amp;lt;array&amp;gt;
          &amp;lt;string&amp;gt;CopyPath&amp;lt;/string&amp;gt;
          &amp;lt;string&amp;gt;1013.txt&amp;lt;/string&amp;gt;
          &amp;lt;string&amp;gt;~/Media/EBooks/The_First_Men_in_the_Moon/1013.txt&amp;lt;/string&amp;gt;
        &amp;lt;/array&amp;gt;
        &amp;lt;array&amp;gt;
          &amp;lt;string&amp;gt;If&amp;lt;/string&amp;gt;
          &amp;lt;array&amp;gt;
              &amp;lt;array&amp;gt;
                &amp;lt;string&amp;gt;FirmwareVersionIs&amp;lt;/string&amp;gt;
                &amp;lt;array&amp;gt;
                  &amp;lt;string&amp;gt;1.1.3&amp;lt;/string&amp;gt;
                  &amp;lt;string&amp;gt;1.1.4&amp;lt;/string&amp;gt;
                &amp;lt;/array&amp;gt;
              &amp;lt;/array&amp;gt;
          &amp;lt;/array&amp;gt;
          &amp;lt;array&amp;gt;
              &amp;lt;array&amp;gt;
                &amp;lt;string&amp;gt;SetStatus&amp;lt;/string&amp;gt;
                &amp;lt;string&amp;gt;Changing Permissions&amp;lt;/string&amp;gt;
              &amp;lt;/array&amp;gt;                              
              &amp;lt;array&amp;gt;
                &amp;lt;string&amp;gt;Exec&amp;lt;/string&amp;gt;
                &amp;lt;string&amp;gt;/bin/chmod -R 755 /var/mobile/Media/EBooks&amp;lt;/string&amp;gt;
              &amp;lt;/array&amp;gt;
              &amp;lt;array&amp;gt;
                &amp;lt;string&amp;gt;Exec&amp;lt;/string&amp;gt;
                &amp;lt;string&amp;gt;/usr/bin/chown -R mobile /var/mobile/Media/EBooks/.&amp;lt;/string&amp;gt;
              &amp;lt;/array&amp;gt;
              &amp;lt;array&amp;gt;                 
                &amp;lt;string&amp;gt;SetStatus&amp;lt;/string&amp;gt;
                &amp;lt;string&amp;gt;Running GutenMark&amp;lt;/string&amp;gt;
              &amp;lt;/array&amp;gt;
              &amp;lt;array&amp;gt;
                &amp;lt;string&amp;gt;Exec&amp;lt;/string&amp;gt;
                &amp;lt;string&amp;gt;/usr/bin/GutenMark --profile=en --no-toc --config=/etc/GutenMark.cfg /var/mobile/Media/EBooks/The_First_Men_in_the_Moon/1013.txt /var/mobile/Media/EBooks/The_First_Men_in_the_Moon/1013-h.htm&amp;lt;/string&amp;gt;
              &amp;lt;/array&amp;gt;
              &amp;lt;array&amp;gt;
                &amp;lt;string&amp;gt;SetStatus&amp;lt;/string&amp;gt;
                &amp;lt;string&amp;gt;Running GutenSplit&amp;lt;/string&amp;gt;
              &amp;lt;/array&amp;gt;
              &amp;lt;array&amp;gt;
                &amp;lt;string&amp;gt;Exec&amp;lt;/string&amp;gt;
                &amp;lt;string&amp;gt;/usr/bin/GutenSplit --no-toc /var/mobile/Media/EBooks/The_First_Men_in_the_Moon/1013-h.htm /var/mobile/Media/EBooks/The_First_Men_in_the_Moon/Chapter_&amp;lt;/string&amp;gt;
              &amp;lt;/array&amp;gt;
          &amp;lt;/array&amp;gt;
        &amp;lt;/array&amp;gt;
&lt;/pre&gt;
&lt;p&gt;
That's not the full install segment, but it should give you an idea.  I have to separate 1.1.3/4 from earlier firmwares because they use different users and store files differently.
&lt;/p&gt;
&lt;p&gt;
To help, I've created a little perl library that allows me to write packages without having to mess with the nasty XML.
It's part of the &lt;a href="http://code.google.com/p/iphoneebookrepo/"&gt;iphoneebookrepo&lt;/a&gt; project over at google code.  Specifically &lt;a href="http://code.google.com/p/iphoneebookrepo/source/browse/trunk/PlistMaker.pm"&gt;PlistMaker.pm&lt;/a&gt;.
&lt;/p&gt;
&lt;p&gt;
In terms of the perl library, it ends up looking like this:
&lt;/p&gt;
&lt;pre&gt;
my $gutenberg_html_install =
 p_if(IsFirmwareVersion("1.1.3", "1.1.4"),
      SetStatus("Changing Permissions").
      Exec("/bin/chmod -R 755 /var/mobile/Media/EBooks").
      Exec("/usr/bin/chown -R mobile /var/mobile/Media/EBooks/.").
      SetStatus("Running GutenSplit").
      Exec("/usr/bin/GutenSplit -1 -2 -3 -4 --no-toc --no-skip /var/mobile/Media/EBooks/$BookDir/${BookNumber}-h.htm /var/mobile/Media/EBooks/$BookDir/Chapter_")
 ).
 p_if(IsFirmwareVersion("1.0.0", "1.0.1", "1.0.2", "1.1.1", "1.1.2" ),
      SetStatus("Changing Permissions").
      Exec("/bin/chmod -R 755 /var/root/Media/EBooks") .
      SetStatus("Running GutenSplit").
      Exec("/usr/bin/GutenSplit -1 -2 -3 -4 --no-toc --no-skip /var/root/Media/EBooks/$BookDir/${BookNumber}-h.htm /var/root/Media/EBooks/$BookDir/Chapter_")
 );

my %BookDescriptor = (
 bundleIdentifier =&amp;gt; "ca.pollock.gutenberg.$BookNumber",
 name =&amp;gt; $BookTitle,
 version =&amp;gt; "1.0",
 location =&amp;gt; $BookURL,
 category =&amp;gt; "Gutenberg SciFi (" .$BookLang . ")",
 size =&amp;gt; $BookSize,
 hash =&amp;gt; $BookMD5,
 url =&amp;gt; "http://jason-pollock.blogspot.com/",
 maintainer =&amp;gt; "Jason Pollock",
 contact =&amp;gt; 'jason@pollock.ca',
 description =&amp;gt; "by $BookAuthor",
 install =&amp;gt; $Book_install,
 preflight =&amp;gt; $Book_preflight,
 postflight =&amp;gt; "" ,
 uninstall =&amp;gt; $Book_uninstall,
 update =&amp;gt; ""
 );

my $BookPLIST = CreatePackage(%BookDescriptor);
&lt;/pre&gt;
&lt;p&gt;
One "Gotcha" to be aware of...  The Exec command is actually performing it's own parameter splitting prior to calling "Exec".  It does not do any sort of "~" expansion, nor does it understand spaces.  So, if you are doing a chmod on a directory with spaces in the name, it won't work, regardless of escapes or quotes.
&lt;/p&gt;&lt;div class="blogger-post-footer"&gt;&lt;img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/6471110-4845321801994767465?l=blog.jason.pollock.ca' alt='' /&gt;&lt;/div&gt;&lt;img src="http://feeds.feedburner.com/~r/JasonPollocksRandomSpewings/~4/-_D6ZNds30w" height="1" width="1"/&gt;</content><link rel="replies" type="application/atom+xml" href="http://blog.jason.pollock.ca/feeds/4845321801994767465/comments/default" title="Post Comments" /><link rel="replies" type="text/html" href="https://www.blogger.com/comment.g?blogID=6471110&amp;postID=4845321801994767465" title="0 Comments" /><link rel="edit" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/4845321801994767465?v=2" /><link rel="self" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/4845321801994767465?v=2" /><link rel="alternate" type="text/html" href="http://feedproxy.google.com/~r/JasonPollocksRandomSpewings/~3/-_D6ZNds30w/installerapp-and-chmodchown.html" title="Installer.app and chmod/chown" /><author><name>Jason Pollock</name><uri>http://www.blogger.com/profile/10823039258279387446</uri><email>noreply@blogger.com</email><gd:extendedProperty name="OpenSocialUserId" value="15378423220352804988" /></author><thr:total xmlns:thr="http://purl.org/syndication/thread/1.0">0</thr:total><feedburner:origLink>http://blog.jason.pollock.ca/2008/06/installerapp-and-chmodchown.html</feedburner:origLink></entry><entry gd:etag="W/&quot;CE8DR349fip7ImA9WxdTFU0.&quot;"><id>tag:blogger.com,1999:blog-6471110.post-2582955974392127569</id><published>2008-05-11T03:46:00.000-07:00</published><updated>2008-05-11T03:47:56.066-07:00</updated><app:edited xmlns:app="http://www.w3.org/2007/app">2008-05-11T03:47:56.066-07:00</app:edited><title>The iPhone Gutenberg Science Fiction eBook Repository!</title><content type="html">&lt;p&gt;Yes, I managed to finish it.  I've managed to convince &lt;a href="http://www.gutenberg.org"&gt;Project Gutenberg&lt;/a&gt; books to install onto the iPhone.&lt;/p&gt;
&lt;p&gt;To make use of the repository:&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;Obtain an iPhone.&lt;/li&gt;
&lt;li&gt;Jailbreak the iPhone, I recommend ziphone for this task.&lt;/li&gt;&lt;li&gt;Install Books, the iPhone &lt;a href="http://code.google.com/p/iphoneebooks/"&gt;ebook reader&lt;/a&gt; using the Installer application (1.4+ only!)&lt;/li&gt;
&lt;li&gt;Add the repository to Installer:&lt;/li&gt;
&lt;ol&gt;
&lt;li&gt;Start Installer&lt;/li&gt;
&lt;li&gt;Click "Sources"&lt;/li&gt;
&lt;li&gt;Click "Edit"&lt;/li&gt;
&lt;li&gt;Click "Add"&lt;/li&gt;
&lt;li&gt;Enter "http://library.pollock.ca/gutenberg_scifi"&lt;/li&gt;
&lt;li&gt;Click "OK"&lt;/li&gt;
&lt;li&gt;Click "Done"&lt;/li&gt;
&lt;li&gt;Click "Refresh" &lt;/li&gt;
&lt;/ol&gt;
&lt;li&gt;Select "Install" at the bottom of the screen. You will see "Gutenberg SciFi" is now available as an application category! &lt;/li&gt;
&lt;li&gt;Before you can install a book, you will need to install &lt;a href="http://www.sandroid.org/GutenMark/"&gt;GutenMark&lt;/a&gt;, it is under Utilities.  I am using it to format the files on your phone to save me bandwidth.&lt;/li&gt;&lt;/ol&gt;
&lt;p&gt;There are currently 163 books in the library, in English and French.  The list of books I used is from Gutenberg's &lt;a href="http://www.gutenberg.org/wiki/Gutenberg:The_CD_and_DVD_Project"&gt;SciFi CD&lt;/a&gt; collection.  Since I can convert any Gutenberg book at this point, if there is a favourite you are looking for, please let me know.&lt;/p&gt;
&lt;p&gt;The installation software has a preference for the HTML version of the book if it exists, however it will use the TXT version and pass it through GutenMark
if it doesn't.  Both versions will be split into chapters using the very handy GutenSplit.&lt;/p&gt;&lt;div class="blogger-post-footer"&gt;&lt;img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/6471110-2582955974392127569?l=blog.jason.pollock.ca' alt='' /&gt;&lt;/div&gt;&lt;img src="http://feeds.feedburner.com/~r/JasonPollocksRandomSpewings/~4/8VUQFs5tV_k" height="1" width="1"/&gt;</content><link rel="replies" type="application/atom+xml" href="http://blog.jason.pollock.ca/feeds/2582955974392127569/comments/default" title="Post Comments" /><link rel="replies" type="text/html" href="https://www.blogger.com/comment.g?blogID=6471110&amp;postID=2582955974392127569" title="10 Comments" /><link rel="edit" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/2582955974392127569?v=2" /><link rel="self" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/2582955974392127569?v=2" /><link rel="alternate" type="text/html" href="http://feedproxy.google.com/~r/JasonPollocksRandomSpewings/~3/8VUQFs5tV_k/iphone-gutenberg-science-fiction-ebook.html" title="The iPhone Gutenberg Science Fiction eBook Repository!" /><author><name>Jason Pollock</name><uri>http://www.blogger.com/profile/10823039258279387446</uri><email>noreply@blogger.com</email><gd:extendedProperty name="OpenSocialUserId" value="15378423220352804988" /></author><thr:total xmlns:thr="http://purl.org/syndication/thread/1.0">10</thr:total><feedburner:origLink>http://blog.jason.pollock.ca/2008/05/iphone-gutenberg-science-fiction-ebook.html</feedburner:origLink></entry><entry gd:etag="W/&quot;AkMHQXs4fSp7ImA9WxZaF0o.&quot;"><id>tag:blogger.com,1999:blog-6471110.post-8040562404264683704</id><published>2008-05-02T18:30:00.000-07:00</published><updated>2008-05-02T18:33:50.535-07:00</updated><app:edited xmlns:app="http://www.w3.org/2007/app">2008-05-02T18:33:50.535-07:00</app:edited><category scheme="http://www.blogger.com/atom/ns#" term="telephony" /><category scheme="http://www.blogger.com/atom/ns#" term="911" /><category scheme="http://www.blogger.com/atom/ns#" term="VoIP" /><title>VoIP and 911</title><content type="html">&lt;p&gt;I see that another infant has died because a VoIP call wasn't properly routed to a 911 &lt;a href="http://www.cbc.ca/technology/story/2008/05/02/crtc-voip.html"&gt;operator&lt;/a&gt;.  Situations like this caused some very extreme results for US VoIP carriers - Vonage wasn't allowed to accept new customers for a couple of months until they sorted it out!&lt;/p&gt;

&lt;p&gt;Every time I see one of these stories, I try to come up with a solution. Here's what I would do for this problem.&lt;/p&gt;

&lt;p&gt;There are a couple of problems in the current solution that need to be solved.  First, the call was routed based on the customer's billing data.  That isn't accurate in the VoIP world.  It's not even accurate in the fixed line world!  Many people have phones that are billed to PO boxes or businesses that are not the physical location of the line.  Second, E911 has the ability to locate an individual caller.  Without that, if the call is disconnected, or the caller cannot speak, the 911 operator has no way of determining where the person needing help is.&lt;/p&gt;

&lt;p&gt;First, the easy one.  Routing based on billing data.  This is pretty easy, DON'T DO IT!&lt;/p&gt;

&lt;p&gt;We need to find some reasonably accurate geolocation data in a VoIP call.  Thankfully, it's right there.  The source IP address.  Home VoIP will be at the end of either a cable or DSL modem.  Either is a physical port.  With a static IP address, it's well defined where they are.  Even with a dynamic address, the address range is still geographically limited.  Definitely accurate enough to select a 911 call center.&lt;/p&gt;

&lt;p&gt;We can make use of commercial IP address &lt;a href="http://www.ip2location.com/"&gt;geolocation databases&lt;/a&gt; to route to the nearest 911 operator.  There are existing commercial databases that provide just this information.  However, the data may be inaccurate.  So, you add the data to the call.  You now have a call with location data of the call and the billing data.  If they disagree, you route based on the IP address and inform the call center to verify the region information.&lt;/p&gt;

&lt;p&gt;Perhaps it's too simple a solution.  VoIP telephony people (and most Internet people) view the Internet as a cloud, they don't trust IP addresses to identify locations (Telephony people have the opposite problem, they put too much trust in the phone number.)&lt;/p&gt;

&lt;p&gt;However, society has started using IP addresses as not only identification of location, but of a specific user too.  For an example, have a look at the &lt;a href="http://recordingindustryvspeople.blogspot.com/"&gt;RIAA lawsuits&lt;/a&gt; in the US.  The IP address is sufficient to sue an individual for copyright infringement.  It is used to identify not only the location the connection is made from, but to make a reasonable assertion to the identity of the individual!&lt;/p&gt;

&lt;p&gt;So, current commercial IP address geolocation products have the information we need to properly route 911 calls to the local call center.&lt;/p&gt;

&lt;p&gt;How do we get E911 levels of location accuracy?  To do that, we need the help of the ISPs.&lt;/p&gt;
&lt;p&gt;Remember, everyone is connecting through DSL or cable, or some other fixed-line connection.  We will deal with mobile later.  That means that the ISP has a mapping between the IP address and the physical port (at least for DSL, cable too probably).  However, gaining access to this data will require government regulation and probably some money changing hands (carriers bill for 911 services).&lt;/p&gt;

&lt;p&gt;If this database is also made available to the 911 operator, then we have a complete system, a database that maps an IP address to a call center, and then on from the IP address to the physical location.  There is probably even a standard that we can use, since E911 services for mobile phones have the same problems.&lt;/p&gt;

&lt;p&gt;So, what about mobile?  Well, in the mobile world, we still have the IP address to location mapping, and the carrier still has the IP address to physical location mapping.  However, it is a little more indirect.  There is a mapping from IP address to phone number, and then the carrier can use their existing E911 services to locate the actual device.&lt;/p&gt;

&lt;p&gt;Is any of this even expensive code?  No, it isn't.  That's the interesting bit.  It's all information that is added on to the existing 911 call.  Even without a correct geolocation of the IP address, the call center operator can still correct the problem by asking "Where are you?".  This system will be there to provide additional information to the 911 operator, not replace them.  Since the 911 operator themselves are a key component of this solution, additional training will undoubtedly be needed, including a change in some of the scripts they read from.  Adding the city to the address for confirmation would solve many problems.&lt;/p&gt;

&lt;p&gt;Where does this break down?&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;VPN - If the phone is in a branch office, that routes IP traffic through a VPN to a different POP, it won't work.  However, they've already got this exact same problem with corporate PBXs anyways, this isn't new.&lt;/li&gt;
&lt;li&gt;Calls from outside of the jurisdiction.  If someone takes the device overseas to use it for cheap (free) long distance.&lt;/li&gt;
&lt;li&gt;Carriers making frequent changes to their routing rules.  If a subnet moves from one region to another, 911 calls may have the wrong location until the database is updated.&lt;/li&gt;
&lt;/ol&gt;&lt;div class="blogger-post-footer"&gt;&lt;img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/6471110-8040562404264683704?l=blog.jason.pollock.ca' alt='' /&gt;&lt;/div&gt;&lt;img src="http://feeds.feedburner.com/~r/JasonPollocksRandomSpewings/~4/H8cwuSCH-l4" height="1" width="1"/&gt;</content><link rel="replies" type="application/atom+xml" href="http://blog.jason.pollock.ca/feeds/8040562404264683704/comments/default" title="Post Comments" /><link rel="replies" type="text/html" href="https://www.blogger.com/comment.g?blogID=6471110&amp;postID=8040562404264683704" title="0 Comments" /><link rel="edit" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/8040562404264683704?v=2" /><link rel="self" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/8040562404264683704?v=2" /><link rel="alternate" type="text/html" href="http://feedproxy.google.com/~r/JasonPollocksRandomSpewings/~3/H8cwuSCH-l4/voip-and-911.html" title="VoIP and 911" /><author><name>Jason Pollock</name><uri>http://www.blogger.com/profile/10823039258279387446</uri><email>noreply@blogger.com</email><gd:extendedProperty name="OpenSocialUserId" value="15378423220352804988" /></author><thr:total xmlns:thr="http://purl.org/syndication/thread/1.0">0</thr:total><feedburner:origLink>http://blog.jason.pollock.ca/2008/05/voip-and-911.html</feedburner:origLink></entry><entry gd:etag="W/&quot;DkADQngyfSp7ImA9WxZaFEs.&quot;"><id>tag:blogger.com,1999:blog-6471110.post-5278979941582607577</id><published>2008-04-29T03:14:00.000-07:00</published><updated>2008-04-29T03:26:13.695-07:00</updated><app:edited xmlns:app="http://www.w3.org/2007/app">2008-04-29T03:26:13.695-07:00</app:edited><category scheme="http://www.blogger.com/atom/ns#" term="Baen Free Library" /><category scheme="http://www.blogger.com/atom/ns#" term="iPhone" /><category scheme="http://www.blogger.com/atom/ns#" term="repository" /><category scheme="http://www.blogger.com/atom/ns#" term="eBook" /><title>Baen Free Library iPhone eBook Repository Updated!</title><content type="html">&lt;p&gt;Since STE has released v 1.4 of the iPhone EBook reader, I have finally had to update the repository to support 1.4.  The major change?  It now places books in ~/Media/EBooks instead of /var/root/Media/EBooks.  This is because Books.app only looks in ~/ now!&lt;/p&gt;
&lt;p&gt;I took the opportunity and added some additional features:&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;Cover Art!  The books now display their cover art in the main title list.
&lt;li&gt;Chapter Sorting! The Books.app team removed a hack they had specifically put in for Baen, it was causing performance problems.  I have renamed the html files to avoid the problem.
&lt;/ol&gt;
&lt;p&gt;Well, I thought it was more features than that when I started.&lt;/p&gt;
&lt;p&gt;If you already have the book installed, it should upgrade through Installer.app.  I have tested a couple of books, so I'm reasonably confident it will work.&lt;/p&gt;
&lt;p&gt;Also, since webscriptions has a habit of updating the .zip files every week or so, I now scan their site every hour to ensure that the source plist is up to date.&lt;/p&gt;
&lt;p&gt;Again, if you run into any problems, file a bug report over at the google code &lt;a href="http://code.google.com/p/iphoneebookrepo/issues/list"&gt;project&lt;/a&gt;&lt;/p&gt;&lt;div class="blogger-post-footer"&gt;&lt;img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/6471110-5278979941582607577?l=blog.jason.pollock.ca' alt='' /&gt;&lt;/div&gt;&lt;img src="http://feeds.feedburner.com/~r/JasonPollocksRandomSpewings/~4/dKc4ieaUcrI" height="1" width="1"/&gt;</content><link rel="replies" type="application/atom+xml" href="http://blog.jason.pollock.ca/feeds/5278979941582607577/comments/default" title="Post Comments" /><link rel="replies" type="text/html" href="https://www.blogger.com/comment.g?blogID=6471110&amp;postID=5278979941582607577" title="0 Comments" /><link rel="edit" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/5278979941582607577?v=2" /><link rel="self" type="application/atom+xml" href="http://www.blogger.com/feeds/6471110/posts/default/5278979941582607577?v=2" /><link rel="alternate" type="text/html" href="http://feedproxy.google.com/~r/JasonPollocksRandomSpewings/~3/dKc4ieaUcrI/baen-free-library-iphone-ebook.html" title="Baen Free Library iPhone eBook Repository Updated!" /><author><name>Jason Pollock</name><uri>http://www.blogger.com/profile/10823039258279387446</uri><email>noreply@blogger.com</email><gd:extendedProperty name="OpenSocialUserId" value="15378423220352804988" /></author><thr:total xmlns:thr="http://purl.org/syndication/thread/1.0">0</thr:total><feedburner:origLink>http://blog.jason.pollock.ca/2008/04/baen-free-library-iphone-ebook.html</feedburner:origLink></entry></feed>
