Nginx 本身不能执行外部程序,Nginx 处理 PHP 是通过 PHP 的 fastcgi 管理器(php-fpm)进行处理,然后 nginx 再将结果返回给用户;所以如果我们需要通过 cgi 程序(shell、perl、c/c++ 等)来编写网站后台的话,就需要使用 fcgiwrap 这个通用的 fastcgi 进程管理器来帮助 nginx 处理 cgi。

  • 对于 PHP:只建议使用 PHP-FPM,因为这是官方的解决方案,性能和稳定性肯定是最好的。
  • 对于其它 CGI 程序:如 Shell、Perl、C/C++、Python,推荐使用 fcgiwrap,这是一个通用的 FCGI 管理器。

写这篇博客的主要目的也是为了让 Nginx 执行 Shell、Perl、C/C++、Python 程序,因为作为一个生信出身的伪 IT 工作者,Shell、Perl、Python 永远都是我们最熟悉的,用这些语言来编写网站后台可以更加节省我们的时间,效率更高。

一、概念

1. CGI 与 FastCGI

CGI 全称是”公共网关接口”(Common Gateway Interface),HTTP 服务器与你的或其它机器上的程序进行 “交谈” 的一种工具,其程序须运行在网络服务器上。它和 FastCGI(快速通用网关接口)都是语言无关的协议。CGI 诞生已经非常久远了,由于它每次在处理一个请求(连接)时都要重新启动脚本(可执行文件),重新传递所有的环境变量(其中非常多是完全一样的),导致性能非常低下。虽然性能较低,但功不可没,后来出现了性能更高的 FastCGI

FastCGI(简称 FCGI)是 CGI 的增强版本,FCGI 可以简单的理解为 CGI + 多进程模型。FCGI 的工作模式有点类似于 Nginx,一个 Master 进程和多个 Worker 进程。Master 进程主要用来监控 Worker 进程的运行情况,当某个 Worker 进程意外退出时,Master 进程会随即启动一个新的 Worker 进程;Worker 进程则是真正干活的进程,用来执行 CGI 程序(传递环境变量、标准输入),获取 CGI 程序的标准输出,再将其返回为 Web 服务器(如 Apache、Nginx)。Worker 进程处理完请求后不会结束运行,而是继续等待下一个请求的到来,直到我们手动关闭它们。

2. Spawn-FCGI 与 FcgiWrap

Spawn-FCGI 是一个通用的 FastCGI 管理服务器,它是 lighttpd 中的一部份,很多人都用 Lighttpd 的 Spawn-FCGI 进行 FastCGI 模式下的管理工作。之前一直以为 Nginx 执行 CGI 程序需要 spawn-fcgi 和 fcgiwrap 两个东西(网上很多文档都是抄来抄去,搞得我也一头雾水,只好照做),但是实际上只需要 fcgiwrap,spawn-fcgi 的作用仅仅是启动和配置 fcgiwrap,这个工作完全可以由 fcgiwrap 自己来完成,所以 spawn-fcgi 不安装也不会影响 fcgiwrap 的使用。

Spawn-FCGI 目前已经独成为一个项目,更加稳定一些,也给很多 Web 站点的配置带来便利。已经有不少站点将它与 nginx 搭配来解决动态网页。目前 Spawn-FCGI 的下载地址是:http://redmine.lighttpd.net/projects/spawn-fcgi,最新版本是:http://download.lighttpd.net/spawn-fcgi/releases-1.6.x/spawn-fcgi-1.6.4.tar.gz

FcgiWrap:Simple FastCGI wrapper for CGI scripts。首先这个东西的作用。它为那些不支持直接运行 CGI 脚本的 Web 服务器提供一种运行 CGI 脚本的方式。NGINX 就是一个只支持 FastCGI,不支持 CGI 的 HTTP(Web)服务器之一。也是我用得最多最熟悉的 Web 服务器。虽然 Apache 支持直接跑 CGI,但从来没用过它的我对它并不感冒,这里也就不再讨论了。

二、安装与配置

2.1 安装 fcgiwrap

安装依赖:

  1. yum -y install autoconf automake libtool fcgi fcgi-devel spawn-fcgi

安装 fcgiwrap:

  1. $ git clone https://github.com/gnosek/fcgiwrap
  2. $ cd fcgiwrap
  3. $ autoreconf -i
  4. $ ./configure # fcgiwrap 默认安装到 /usr/local/sbin/fcgiwrap
  5. $ make
  6. $ make install

2.2 配置 spawn-fcgi

通过 yum install spawn-fcgi 方式安装的 spawn-fcgi 配置文件默认为:/etc/sysconfig/spawn-fcgi,编辑该文件:

  1. vi /etc/sysconfig/spawn-fcgi
  2. # You must set some working options before the "spawn-fcgi" service will work.
  3. # If SOCKET points to a file, then this file is cleaned up by the init script.
  4. #
  5. # See spawn-fcgi(1) for all possible options.
  6. #
  7. # Example :
  8. #SOCKET=/var/run/php-fcgi.sock
  9. #OPTIONS="-u apache -g apache -s $SOCKET -S -M 0600 -C 32 -F 1 -P /var/run/spawn-fcgi.pid -- /usr/bin/php-cgi"
  10. FCGI_SOCKET=/var/run/fcgiwrap.sock
  11. FCGI_PROGRAM=/usr/local/sbin/fcgiwrap
  12. FCGI_USER=nginx
  13. FCGI_GROUP=nginx
  14. FCGI_EXTRA_OPTIONS="-M 0777"
  15. OPTIONS="-u $FCGI_USER -g $FCGI_GROUP -s $FCGI_SOCKET -S $FCGI_EXTRA_OPTIONS -F 1 -P /var/run/spawn-fcgi.pid -- $FCGI_PROGRAM"

2.3 启动 spawn-fcgi 服务

  1. [root@ecs-steven conf]# systemctl enable spawn-fcgi # 添加开机启动(或者:chkconfig spawn-fcgi on)
  2. spawn-fcgi.service is not a native service, redirecting to /sbin/chkconfig.
  3. Executing /sbin/chkconfig spawn-fcgi on
  4. $ systemctl start spawn-fcgi # 启动 spawn-fcgi 服务(或者:service spawn-fcgi start)

spawn-fcgi 启动出现下面报错:

  1. [root@ecs-steven ~]# service spawn-fcgi start
  2. Starting spawn-fcgi (via systemctl): Job for spawn-fcgi.service failed because the control process exited with error code. See "systemctl status spawn-fcgi.service" and "journalctl -xe" for details.
  3. [FAILED]
  4. [root@ecs-steven ~]# systemctl status spawn-fcgi.service
  5. spawn-fcgi.service - LSB: Start and stop FastCGI processes
  6. Loaded: loaded (/etc/rc.d/init.d/spawn-fcgi; bad; vendor preset: disabled)
  7. Active: failed (Result: exit-code) since Fri 2019-04-26 08:31:51 CST; 9min ago
  8. Docs: man:systemd-sysv-generator(8)
  9. Process: 7069 ExecStart=/etc/rc.d/init.d/spawn-fcgi start (code=exited, status=1/FAILURE)
  10. Apr 26 08:31:51 ecs-steven systemd[1]: Starting LSB: Start and stop FastCGI processes...
  11. Apr 26 08:31:51 ecs-steven spawn-fcgi[7069]: Starting spawn-fcgi: spawn-fcgi: can't find user name nginx
  12. Apr 26 08:31:51 ecs-steven spawn-fcgi[7069]: [FAILED]
  13. Apr 26 08:31:51 ecs-steven systemd[1]: spawn-fcgi.service: control process exited, code=exited status=1
  14. Apr 26 08:31:51 ecs-steven systemd[1]: Failed to start LSB: Start and stop FastCGI processes.
  15. Apr 26 08:31:51 ecs-steven systemd[1]: Unit spawn-fcgi.service entered failed state.
  16. Apr 26 08:31:51 ecs-steven systemd[1]: spawn-fcgi.service failed.

这种情况我们需要创建 nginx 用户,然后启动 spawn-fcgi.service:

  1. [root@ecs-steven conf]# /usr/sbin/useradd nginx -s /bin/false
  2. [root@ecs-steven conf]# service spawn-fcgi start
  3. Starting spawn-fcgi (via systemctl): [ OK ]

2.4 配置 Nginx

  1. [root@ecs-steven conf]# vim nginx.conf
  2. user nginx nginx;
  3. worker_processes 1;
  4. error_log logs/error.log;
  5. pid logs/nginx.pid;
  6. events {
  7. worker_connections 1024;
  8. }
  9. http {
  10. include mime.types;
  11. default_type application/octet-stream;
  12. log_format main '$remote_addr - $remote_user [$time_local] "$request" '
  13. '$status $body_bytes_sent "$http_referer" '
  14. '"$http_user_agent" "$http_x_forwarded_for"';
  15. access_log logs/access.log main;
  16. sendfile on;
  17. #tcp_nopush on;
  18. client_max_body_size 50m;
  19. keepalive_timeout 65;
  20. gzip on;
  21. server {
  22. listen 80;
  23. server_name localhost;
  24. location / {
  25. root html;
  26. index index.html index.htm;
  27. }
  28. error_page 500 502 503 504 /50x.html;
  29. location = /50x.html {
  30. root html;
  31. }
  32. }
  33. include sites-available/*.conf;
  34. }

在 Nginx 目录下添加 sites-available/fcgi.conf:

  1. [root@ecs-steven conf]# cat sites-available/fcgi.conf
  2. server {
  3. listen 80;
  4. access_log logs/fcgi_access.log;
  5. error_log logs/fcgi_error.log debug;
  6. # 开启 SSL 配置
  7. #listen 443;
  8. #ssl on;
  9. #server_name tools.shenweiyan.com;
  10. #ssl_certificate ../cert/tools/tools.shenweiyan.com.pem;
  11. #ssl_certificate_key ../cert/tools/tools.shenweiyan.com.key;
  12. #ssl_session_timeout 5m;
  13. #ssl_ciphers ECDHE-RSA-AES128-GCM-SHA256:ECDHE:ECDH:AES:HIGH:!NULL:!aNULL:!MD5:!ADH:!RC4;
  14. #ssl_protocols TLSv1 TLSv1.1 TLSv1.2;
  15. #ssl_prefer_server_ciphers on;
  16. root /apps/tools;
  17. location / {
  18. index index.html index.htm;
  19. }
  20. location ~ .*\.(pl|py|cgi)?$ {
  21. include /usr/local/software/nginx/conf/fastcgi_params;
  22. fastcgi_pass unix:/var/run/fcgiwrap.sock;
  23. fastcgi_index index.cgi;
  24. fastcgi_param SCRIPT_FILENAME /apps/tools/$fastcgi_script_name;
  25. }
  26. }

最后,重启 Nginx:

  1. $ service nginx reload
  2. 或者:
  3. $ systemctl restart nginx

三、添加 CGI 程序

编写 CGI 测试程序:

  1. $ vim /apps/tools/test.cgi
  2. #!/usr/bin/perl -w
  3. print "Content-type: text/html\n\n";
  4. print "<html><head><title>Hello World!</title></head>\n";
  5. print "<body><h1>Hello world, CGI work!</h1>
  6. </body></html>\n";

设定权限:

  1. $chmod 0755 /apps/tools/test.cgi
  2. $ chown nginx.nginx /apps/tools/test.cgi

最后,利用 firefox/chrome 测试!http://192.168.xxx.xxx/test.cgi 访问出现 “Hello world, CGI work!“ 即说明配置部署成功。

四、参考资料