小知识:Linux下用Nginx作Perl程序服务器及其中Perl模块的配置

perl + fastcgi + nginx搭建

nginx + fastcgi是php下最流行的一套环境了,那perl会不会也有fastcgi呢,当然有,今天来搭建下nginx下perl的fastcgi.性能方面也不亚于php,但是现在web程序php的流行程度perl无法比拟了,性能再好也枉然,但是部分小功能可以考虑使用perl的fastcgi来搞定.进入正题.

1. 准备软件环境:

nginx:http://www.nginx.org

perl:系统自带

fastcgi:http://www.cpan.org/modules/by-module/FCGI/

1.1 nginx安装

这里就不再详细介绍了~

1.2 perl安装 一般linux都有自带perl,可以不用安装,如果确实没有,请执行:
?
1
# yum install perl

1.3 perl-fastcgi安装

?
1
2
3
4
5
6
7
# 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虚拟主机配置

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
server {
listen  80;
server_name test.<font style=”background-color: #efd200″>tuohang.net</font>;
#access_log /data/logs/nginx/test.<font style=”background-color: #efd200″>tuohang.net</font>.access.log main;
index index.html index.php index.html;
root /data/site/test.<font style=”background-color: #efd200″>tuohang.net</font>;
location /
{
}
location ~ \.pl$
{
include fastcgi_params;
fastcgi_pass 127.0.0.1:8999;
#fastcgi_pass unix:/var/run/<font style=”background-color: #efd200″>tuohang.net</font>.perl.sock;
fastcgi_index index.pl;
}
}

如果想把tcp/ip方式改为socket方式,可以修改fastcgi-wrapper.pl.

?
1
$socket = FCGI::OpenSocket( “127.0.0.1:8999”, 10 ); #use IP sockets

改为

?
1
$socket = FCGI::OpenSocket( “/var/run/tuohang.net.perl.sock”, 10 ); #use IP sockets

3. 配置脚本

3.1 fastcgi监听脚本 文件路径:/usr/bin/fastcgi-wrapper.pl

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
#!/usr/bin/perl
use FCGI;
use Socket;
use POSIX qw(setsid);
require syscall.ph;
&daemonize;
#this keeps the program alive or something after execing 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 “Cant chdir to /: $!”;
defined(my $pid = fork) or die “Cant fork: $!”;
exit if $pid;
setsid     or die “Cant 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 scripts 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

 
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
#!/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
esac

3.3 设置脚本权限

?
1
2
# chmod a+x /usr/bin/fastcgi-wrapper.pl
# chmod a+x /etc/rc.d/init.d/perl-fastcgi

4. FastCGI测试

4.1 启动nginx与fastcgi
?
1
2
# /usr/local/nginx-1.4.2/sbin/nginx
# /etc/init.d/perl-fastcgi start

4.2 perl测试文件: 文件路径/data/site/test.tuohang.net/test.pl

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
#!/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.tuohang.net/test.pl,出现内容表示OK.

6. 简单压力测试:

6.1 使用tcp/ip方式
?
1
ab -n 1000 -c 10 http://test.tuohang.net/test.pl

他是在是太慢了,只好用10个并发,共计100个请求来测试.

%小知识:Linux下用Nginx作Perl程序服务器及其中Perl模块的配置-猿站网-插图

6.2 使用socket方式:

?
1
ab -n 100000 -c 500 http://test.tuohang.net/test.pl

%小知识:Linux下用Nginx作Perl程序服务器及其中Perl模块的配置-1猿站网-插图

很奇怪,使用tcp/ip方式,每秒就140多个请求,而使用socket方式却有5800个请求/秒。差距不是一般的大。顺便测试了一下php的fastcgi,大概请求在3000(tcp/ip方式),4800(socket方式)。

perl模块的使用 如果对于一个绝大部分内容是静态的网站,只有极少数的地方需要动态显示,碰巧你又了解一点perl知识,那么nginx + perl的结合就能很好解决问题。要想nginx支持perl脚本,在编译nginx时候需要如下参数:

?
1
./configure –with-http_perl_module

如果make时候出现如下类似错误:

?
1
Cant 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搞定,例如:

?
1
yum -y install perl-devel perl-ExtUtils-Embed

nginx中使用perl有两种方法,一种是直接在配置文件写,还有一种是把perl脚本写在外部文件中,下面主要介绍一下第二种用法。

假设nginx的根目录为/usr/local/nginx,perl脚本存放的目录为nginx的根目录下的perl/lib下,脚本名字为test.pm,nginx配置为:
?
1
2
3
4
5
6
7
8
#位于http配置中
perl_modules perl/lib;
perl_require test.pm;
#位于server配置中
location /user/ {
perl pkg_name::process;
}

上述配置是把所有来自http://servername/user/下的请求交由test.pm脚本中定义的process方法来处理。

test.pm脚本的内容如下:
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
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,你应该可以在网页上看到:

?
1
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]) – 设置变量的值
声明: 猿站网有关资源均来自网络搜集与网友提供,任何涉及商业盈利目的的均不得使用,否则产生的一切后果将由您自己承担! 本平台资源仅供个人学习交流、测试使用 所有内容请在下载后24小时内删除,制止非法恶意传播,不对任何下载或转载者造成的危害负任何法律责任!也请大家支持、购置正版! 。本站一律禁止以任何方式发布或转载任何违法的相关信息访客发现请向站长举报,会员发帖仅代表会员个人观点,并不代表本站赞同其观点和对其真实性负责。本网站的资源部分来源于网络,如有侵权烦请发送邮件至:2697268773@qq.com进行处理。
建站知识

小知识:9个Linux 常用查看系统硬件信息命令(实例详解)

2023-5-1 2:11:46

建站知识

小知识:选择 Linux 的五大好处

2023-5-1 2:20:05

0 条回复 A文章作者 M管理员
    暂无讨论,说说你的看法吧
个人中心
购物车
优惠劵
今日签到
有新私信 私信列表
搜索