Hello World
一個簡單的hello world例子:
複製代碼 代碼如下:
#!/usr/bin/env perl
use strict;
use warnings;
use CGI::Fast;
while(my $q = new CGI::Fast)
{
print $q->header("text/plain");
print "Hello World";
}
和CGI的區別僅在於多了一個迴圈來接受請求,CGI::Fast對象和CGI介面是一樣的,而且該指令碼也可以當做CGI指令碼使用。
搭建nginx + FastCGI 環境
Perl使用CGI::Fast包來提供FastCGI服務,該包提供兩種方式來啟動FastCGI進程,一個是直接使用該包提供的服務將當前進程變為FastCGI進程,另外一個是使用第三方工具spawn-fcgi來啟動。
nginx配置方式例子:
複製代碼 代碼如下:
location / {
fastcgi_pass 127.0.0.1:8184;
fastcgi_param SCRIPT_FILENAME /scripts$fastcgi_script_name;
include fastcgi_params;
}
配置好nginx後,使用spawn-fcgi來啟動前面的Hello World:
複製代碼 代碼如下:
$ spawn-fcgi -n -a 127.0.0.1 -p 8184 -f ./main.pl
調試支援
在前面的命令列裡使用了參數-n,讓spawn-fcgi不要fork出多個進程,並阻塞,允許使用者Ctrl+C來關閉,產品伺服器可以去掉這個參數來充分利用伺服器的多核來提供更高的並發數。我之前寫了一個bash指令碼,允許在檔案改動的情況下重啟服務,方便調試perl程式,代碼如下:
複製代碼 代碼如下:
#!/bin/bash
#PID檔案和需要啟動的指令碼
PID_FILE=service.pid
MAIN=main.pl
#關閉之前啟動的進程
term() {
test -e $PID_FILE || return
pid=`cat $PID_FILE`
kill -s -0 $pid || return
echo "Terminating $MAIN $pid"
rm -f $PID_FILE
kill $pid
wait $pid
}
#當前指令碼退出的時候也關閉啟動了的FastCGI進程
trap "term;exit" SIGINT SIGTERM
while true
do
#初次開機或者檔案改動後都需要關閉之前的進程
term
#以no fork方式啟動指令碼來調試,並將PID寫入到檔案
spawn-fcgi -n -a 127.0.0.1 -p 8184 -f ./$MAIN &
pid=$!
echo $pid > $PID_FILE
echo "My Perl service started, pid = $pid"
#監控檔案變化
files=`find . -name '*.pl' -o -name '*.pm' -o -name '*.html'`
md5=`md5sum $files|md5sum`
#wait for file change
while [[ `md5sum $files|md5sum` = "$md5" ]]
do
sleep 1
done
echo "File changes detected, restarting service"
done
該指令碼已在Mac OSX和Linux下測試通過
路由系統
做Web開發離不開路由實現,來對不同請求來做出特定的響應。
路由請求依賴HTTP Method和URI兩部分,因此主要就是需要這兩者來做指派。
在CGI中可以通過環境變數REQUEST_METHOD和REQUEST_URI來擷取要求方法和URI。
因此一個簡單的路由系統實際上可以分解為一個二級的map,註冊路由實際上就是往這個map裡放入規則對應的處理函數,而指派請求則是從這個map雷根據規則擷取對應的處理函數,一個簡單的例子:
複製代碼 代碼如下:
my %routers = ();
sub not_found
{
print "Status: 404\n";
print "Content-Type: text/html\n\n";
print<<EOF
<html>
<body>
<h1>404 Not found</h1>
Cannot find $ENV{REQUEST_PATH}.
</body>
</html>
EOF
}
sub add_rule
{
my ($method, $path, $callback) = @_;
my $handlers = $routers{$method};
$handlers = $routers{$method} = {} if not $handlers;
$handlers->{$path} = $callback;
}
sub dispatch
{
my $q = shift;
my $method = $ENV{REQUEST_METHOD};
my $uri = $ENV{REQUEST_URI};
$uri =~ s/\?.*$//;
my $handler = ($routers{$method} || {})->{$uri} || not_found;
eval
{
&$handler($q);
};
print STDERR "Failed to handle $method $uri: $@\n" if $@;
}
使用這個路由系統的例子:
複製代碼 代碼如下:
sub index
{
my ($q) = @_;
print $q->header('text/plain');
print "Hello World!";
}
router::add_rule('GET', '/', \&index);
模板系統
perl提供了大量的模板系統的實現,我個人最喜歡的是Template Toolkit,文檔也非常豐富,網站是 http://www.template-toolkit.org/ 。
將前面的index修改為使用模板的例子:
複製代碼 代碼如下:
use Template;
my $tt = new Template({INCLUDE_PATH => 'templates', INTERPOLATE => 1});
sub index
{
my ($q) = @_;
my $output = '';
print $q->header('text/html');
$tt->process('index.html', {world => 'World'}, $output) || die $tt->error();
print $output;
}
其中templates/index.html檔案內容如下:
複製代碼 代碼如下:
<html>
<head><title>Demo</title></head>
<body>
Hello ${world}
</body>
</html>
完!