Project

General

Profile

Actions

MasonRecipe » History » Revision 2

« Previous | Revision 2/20 (diff) | Next »
Anonymous, 2006-08-06 07:57


= Using Mason with lighttpd (via FastCGI) =

Mason can be used with Lighttpd and is (arguably :-) easier to setup that Apache + mod_perl.

Here's my current setup. Requirements:

  • Mason handles files ending with '/' or '.html' or '.css'.
  • Other files statically served by lighttpd
  • Load perl modules at fastcgi process start

First we need a mason handler, that lighttpd will use via FastCGI. I've called it mason_lighttpd_handler.fcgi. It does not need to be in any particular location.

{{{
#!/usr/bin/perl

use strict;
use HTML::Mason::CGIHandler;
use CGI::Fast;

{
package HTML::Mason::Commands;

  1. anything you want available to components
  2. use DBI;
  3. use Other::Module;
    }
  1. lazily-instantiated variables
    my $cgi;
    my $h;

$ENV{MASON_COMP_ROOT} = '/your/comp/root/goes/here';
$ENV{MASON_DATA_ROOT} = '/your/data/root/goes/here'';

while ($cgi = new CGI::Fast) {

  1. this seems to be necessary as lighttpd does not provide the
  2. PATH_INFO and the QUERY_STRING environment variables
  3. Note that the below seems to work but I have no idea if it's the
  4. 'right' way to parse the REQUEST_URI out into PATH_INFO and QUERY_STRING
    my $uri = $ENV{REQUEST_URI};
    if ($uri =~ /\?/) {
    $uri =~ /^(.*?)\?(.*)/;
    $ENV{PATH_INFO} = $1;
    $ENV{QUERY_STRING} = $2;
    } else {
    $ENV{PATH_INFO} = $uri;
    $ENV{QUERY_STRING} = "";
    }
  1. this is lazily instantiated because %ENV is not set at startup time
    if (! $h) {
    $h = HTML::Mason::CGIHandler->new(comp_root => $ENV{MASON_COMP_ROOT},
    data_dir => $ENV{MASON_DATA_ROOT},
    error_mode => 'fatal',
    allow_globals => [qw($dbh $user $test)],
    error_format => 'html');
    }
  1. hand off to mason
    eval { $h->handle_cgi_object($cgi) };
    if (my $raw_error = $@) { # print out a pretty system error page and log $raw_error
    print $raw_error;
    }
    }

exit 0;

}}}

Next we need to tell Lighttpd to process requests for this site and process them via this script and FastCGI. Here is the relevant fragment. I'm using a regexp for the hostname expression
but you may need something less complicated - see the Lighttpd docs for more information:

{{{
$HTTP["host"] =~ "hostnameexpression" {
server.document-root = "/path/to/your/document/root"

fastcgi.map-extensions   = ( ".css" => ".html", "/" => ".html" )    # map .css and '/' to .html so they will be handled by FastCGI
index-file.names = ( "index.html" )
url.access-deny = ( ".mhtml" )
fastcgi.server = ( ".html" =>
((
"socket" => "/tmp/fastcgi.socket",
"bin-path" => "/path/to/the/mason_lighttpd_handler.fcgi",
"check-local" => "disable"
))
)
}
}}}

That's it!

When you start lighttpd it will start several copies of your mason handler script, passing requests to them as appropriate.

One nice side-effect of this is that if you change a perl module (not a mason component) that would require you to restart/reload an apache mod_perl server, in this case you can just kill the perl fastcgi processes - lighttpd will notice and restart them, of course with your new module code in place.

Add debugging (via 'warn') to your handler script if you are getting unexpected results, you can see the output from it in the lighttpd eror log.

Let me know (Justin Hawkins <> if there are problems with this, particular with the REQUEST_URI parsing. This is very much in the category of "seems to work" at this stage.

Updated by Anonymous over 18 years ago · 20 revisions