本篇內容介紹了“Linux下如何用Nginx作Perl程序服務器及其中Perl模塊”的有關知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領大家學習一下如何處理這些情況吧!希望大家仔細閱讀,能夠學有所成!
perl + fastcgi + nginx搭建
nginx + fastcgi是php下最流行的一套環境了,那perl會不會也有fastcgi呢,當然有,今天來搭建下nginx下perl的fastcgi.性能方面也不亞于php,但是現在web程序php的流行程度perl無法比擬了,性能再好也枉然,但是部分小功能可以考慮使用perl的fastcgi來搞定.進入正題.
1. 準備軟件環境:
nginx
perl:系統自帶
fastcgi
1.2 perl安裝
一般linux都有自帶perl,可以不用安裝,如果確實沒有,請執行:
# yum install perl
1.3 perl-fastcgi安裝
# cd /usr/local/src # wget http://www.cpan.org/modules/by-module/fcgi/fcgi-0.74.tar.gz # tar -xzvf fcgi-0.74.tar.gz # cd fcgi-0.74 # perl makefile.pl # make # make install
2. nginx虛擬主機配置
server {
listen 80;
server_name test.jb51.net;
#access_log /data/logs/nginx/test.jb51.net.access.log main;
index index.html index.php index.html;
root /data/site/test.jb51.net;
location /
{
}
location ~ \.pl$
{
include fastcgi_params;
fastcgi_pass 127.0.0.1:8999;
#fastcgi_pass unix:/var/run/jb51.net.perl.sock;
fastcgi_index index.pl;
}
}如果想把tcp/ip方式改為socket方式,可以修改fastcgi-wrapper.pl.
$socket = fcgi::opensocket( "127.0.0.1:8999", 10 ); #use ip sockets
改為
$socket = fcgi::opensocket( "/var/run/jb51.net.perl.sock", 10 ); #use ip sockets
3. 配置腳本
3.1 fastcgi監聽腳本
文件路徑:/usr/bin/fastcgi-wrapper.pl
#!/usr/bin/perl
use fcgi;
use socket;
use posix qw(setsid);
require 'syscall.ph';
&daemonize;
#this keeps the program alive or something after exec'ing perl scripts
end() { } begin() { }
*core::global::exit = sub { die "fakeexit\nrc=".shift()."\n"; };
eval q{exit};
if ($@) {
exit unless $@ =~ /^fakeexit/;
};
&main;
sub daemonize() {
chdir '/' or die "can't chdir to /: $!";
defined(my $pid = fork) or die "can't fork: $!";
exit if $pid;
setsid or die "can't start a new session: $!";
umask 0;
}
sub main {
$socket = fcgi::opensocket( "127.0.0.1:8999", 10 ); #use ip sockets
$request = fcgi::request( \*stdin, \*stdout, \*stderr, \%req_params, $socket );
if ($request) { request_loop()};
fcgi::closesocket( $socket );
}
sub request_loop {
while( $request->accept() >= 0 ) {
#processing any stdin input from webserver (for cgi-post actions)
$stdin_passthrough ='';
$req_len = 0 + $req_params{'content_length'};
if (($req_params{'request_method'} eq 'post') && ($req_len != 0) ){
my $bytes_read = 0;
while ($bytes_read < $req_len) {
my $data = '';
my $bytes = read(stdin, $data, ($req_len - $bytes_read));
last if ($bytes == 0 || !defined($bytes));
$stdin_passthrough .= $data;
$bytes_read += $bytes;
}
}
#running the cgi app
if ( (-x $req_params{script_filename}) && #can i execute this?
(-s $req_params{script_filename}) && #is this file empty?
(-r $req_params{script_filename}) #can i read this file?
){
pipe(child_rd, parent_wr);
my $pid = open(kid_to_read, "-|");
unless(defined($pid)) {
print("content-type: text/plain\r\n\r\n");
print "error: cgi app returned no output - ";
print "executing $req_params{script_filename} failed !\n";
next;
}
if ($pid > 0) {
close(child_rd);
print parent_wr $stdin_passthrough;
close(parent_wr);
while(my $s = <kid_to_read>) { print $s; }
close kid_to_read;
waitpid($pid, 0);
} else {
foreach $key ( keys %req_params){
$env{$key} = $req_params{$key};
}
# cd to the script's local directory
if ($req_params{script_filename} =~ /^(.*)\/[^\/]+$/) {
chdir $1;
}
close(parent_wr);
close(stdin);
#fcntl(child_rd, f_dupfd, 0);
syscall(&sys_dup2, fileno(child_rd), 0);
#open(stdin, "<&child_rd");
exec($req_params{script_filename});
die("exec failed");
}
}
else {
print("content-type: text/plain\r\n\r\n");
print "error: no such cgi app - $req_params{script_filename} may not ";
print "exist or is not executable by this process.\n";
}
}
}3.2 fastcgi自啟動服務腳本:
文件路徑:/etc/rc.d/init.d/perl-fastcgi
文件路徑:/etc/rc.d/init.d/perl-fastcgi
#!/bin/sh
#
# nginx – this script starts and stops the nginx daemon
#
# chkconfig: - 85 15
# description: nginx is an http(s) server, http(s) reverse \
# proxy and imap/pop3 proxy server
# processname: nginx
# config: /opt/nginx/conf/nginx.conf
# pidfile: /opt/nginx/logs/nginx.pid
# source function library.
. /etc/rc.d/init.d/functions
# source networking configuration.
. /etc/sysconfig/network
# check that networking is up.
[ "$networking" = "no" ] && exit 0
perlfastcgi="/usr/bin/fastcgi-wrapper.pl"
prog=$(basename perl)
lockfile=/var/lock/subsys/perl-fastcgi
start() {
[ -x $perlfastcgi ] || exit 5
echo -n $"starting $prog: "
daemon $perlfastcgi
retval=$?
echo
[ $retval -eq 0 ] && touch $lockfile
return $retval
}
stop() {
echo -n $"stopping $prog: "
killproc $prog -quit
retval=$?
echo
[ $retval -eq 0 ] && rm -f $lockfile
return $retval
}
restart() {
stop
start
}
reload() {
echo -n $”reloading $prog: ”
killproc $nginx -hup
retval=$?
echo
}
force_reload() {
restart
}
rh_status() {
status $prog
}
rh_status_q() {
rh_status >/dev/null 2>&1
}
case "$1" in
start)
rh_status_q && exit 0
$1
;;
stop)
rh_status_q || exit 0
$1
;;
restart)
$1
;;
reload)
rh_status_q || exit 7
$1
;;
force-reload)
force_reload
;;
status)
rh_status
;;
condrestart|try-restart)
rh_status_q || exit 0
;;
*)
echo $"usage: $0 {start|stop|status|restart|condrestart|try-restart|reload|force-reload}"
exit 2
esac3.3 設置腳本權限
# chmod a+x /usr/bin/fastcgi-wrapper.pl # chmod a+x /etc/rc.d/init.d/perl-fastcgi
4. fastcgi測試
4.1 啟動nginx與fastcgi
# /usr/local/nginx-1.4.2/sbin/nginx # /etc/init.d/perl-fastcgi start
4.2 perl測試文件:
文件路徑/data/site/test.jb51.net/test.pl
#!/usr/bin/perl
print "content-type:text/html\n\n";
print <<endofhtml;
<html><head><title>perl environment variables</title></head>
<body>
<h1>perl environment variables</h1>
endofhtml
foreach $key (sort(keys %env)) {
print "$key = $env{$key}<br>\n";
}
print "</body></html>";5. 訪問測試
5.1 訪問
http://http:test.jb51.net/test.pl,出現內容表示ok.
6. 簡單壓力測試:
6.1 使用tcp/ip方式
ab -n 1000 -c 10 http://test.jb51.net/test.pl
他是在是太慢了,只好用10個并發,共計100個請求來測試.

6.2 使用socket方式:
ab -n 100000 -c 500 http://test.jb51.net/test.pl

很奇怪,使用tcp/ip方式,每秒就140多個請求,而使用socket方式卻有5800個請求/秒。差距不是一般的大。順便測試了一下php的fastcgi,大概請求在3000(tcp/ip方式),4800(socket方式)。
perl模塊的使用
如果對于一個絕大部分內容是靜態的網站,只有極少數的地方需要動態顯示,碰巧你又了解一點perl知識,那么nginx + perl的結合就能很好解決問題。要想nginx支持perl腳本,在編譯nginx時候需要如下參數:
./configure --with-http_perl_module
如果make時候出現如下類似錯誤:
can't locate extutils/embed.pm in @inc (@inc contains: /usr/lib/perl5/5.10.0/i386-linux-thread-multi /usr/lib/perl5/5.10.0 /usr/local/lib/perl5/site_perl/5.10.0/i386-linux-thread-multi /usr/local/lib/perl5/site_perl/5.10.0 /usr/lib/perl5/vendor_perl/5.10.0/i386-linux-thread-multi /usr/lib/perl5/vendor_perl/5.10.0 /usr/lib/perl5/vendor_perl /usr/local/lib/perl5/site_perl .)
你的機器上可能需要安裝perl-devel perl-extutils-embed,對于centos系統,直接使用yum搞定,例如:
yum -y install perl-devel perl-extutils-embed
nginx中使用perl有兩種方法,一種是直接在配置文件寫,還有一種是把perl腳本寫在外部文件中,下面主要介紹一下第二種用法。
假設nginx的根目錄為/usr/local/nginx,perl腳本存放的目錄為nginx的根目錄下的perl/lib下,腳本名字為test.pm,nginx配置為:
#位于http配置中
perl_modules perl/lib;
perl_require test.pm;
#位于server配置中
location /user/ {
perl pkg_name::process;
}上述配置是把所有來自http://servername/user/下的請求交由test.pm腳本中定義的process方法來處理。
test.pm腳本的內容如下:
package pkg_name;
use time::local;
use nginx;
sub process {
my $r = shift;
$r->send_http_header('text/html; charset=utf-8');
my @arr = split('/', $r->uri);
my $username = @arr[2];
if (!$username || ($username eq "")) {
$username = "anonymous";
}
$r->print('hello, you name is : <strong>' . $username . '</strong>');
$r->rflush();
return;
}
1;
__end__當你訪問http://servername/user/netingcn,你應該可以在網頁上看到:
hello, you name is : netingcn
另外:當使用 use nginx 時,會有如下的對象可以調用,可以看到上面 shift 一個對象到 $r 上,然后就可以用 $r 調用那些對象了:
$r->args – 請求的參數 .
$r->discard_request_body – 這個參數是讓 nginx 放棄 request 的 body 的內容.
$r->filename – 返回合適的請求文件的名字
$r->has_request_body(function) – 如果沒有請求主體,返回0,但是如果請求主體存在,那么建立傳遞的函數并返回1,在程序的最后,nginx將調用指定的處理器.
$r->header_in(header) – 查找請求頭的信息
$r->header_only – 如果我們只要返回一個響應的頭
$r->header_out(header, value) – 設置響應的頭
$r->internal_redirect(uri) – 使內部重定向到指定的uri,重定向僅在完成perl腳本后發生.可以使用 header_out(location….的方法來讓瀏覽器自己重定向
$r->print(args, …) – 發送數據給客戶端
$r->request_body – 得到客戶端提交過來的內容 (body 的參數,可能需要修改 nginx 的 client_body_buffer_size. )
$r->request_body_file —給客戶的 body 存成文件,并返回文件名
$r->request_method — 得到請求 http method.
$r->remote_addr – 得到客戶端的 ip 地址.
$r->rflush – 立即傳送數據給客戶端
$r->sendfile(file [, displacement [, length ] ) – 傳送給客戶端指定文件的內容,可選的參數表明只傳送數據的偏移量與長度,精確的傳遞僅在perl腳本執行完畢后生效.這可是所謂的高級功能啊
$r->send_http_header(type) – 添加一個回應的 http 頭的信息
$r->sleep(milliseconds, handler) – 設置為請求在指定的時間使用指定的處理方法和停止處理,在此期間nginx將繼續處理其他的請求,超過指定的時間后,nginx將運行安裝的處理方法,注意你需要為處理方法通過一個reference,在處理器間轉發數據你可以使用$r->variable().
$r->status(code) – 設置 http 的響應碼
$r->unescape(text) – 使用 http 方法加密內容如 %xx
$r->uri – 得到請求的 url.
$r->variable(name[, value]) – 設置變量的值
“Linux下如何用Nginx作Perl程序服務器及其中Perl模塊”的內容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業相關的知識可以關注億速云網站,小編將為大家輸出更多高質量的實用文章!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。