C++文件服务器项目—FastCGI—4

  • 前言
  • 1. CGI 概念理解
  • 2. FastCGI 概念理解
  • 3. FastCGI和spawn-fcgi安装
  • 4. FastCGI和 Nginx的关系
  • 5. Nginx数据转发-修改配置文件
  • 6. spawn-fcgi如何启动
  • 7. FastCGI程序怎么写
    • 7.1 echo.c代码阅读与分析
    • 7.2 fastCGI接收与发送数据
    • 7.3 fastCGI程序编写流程与思路
    • 7.4 实现一个fastCGI程序并测试
  • 8. 部署一个能够上传文件的网页
    • 8.1 上传文件夹并修改Nginx的配置文件
    • 8.2 修改配置文件,实现数据转发
    • 8.3 reference & libfcgi.so.0 => not found 问题解决
  • 9. 其他知识点
    • 9.1 HTTP环境变量 -> fastcgi.conf
    • 9.2 常用的四种 Content-Type
  • 10. fastCGI总结

前言

  本文重点介绍FastCGI的概念、如何编写FastCGI程序,以及nginx如何配合fastCGI使用。源码地址:gopherWxf git

  本专栏知识点是通过零声教育的线上课学习,进行梳理总结写下文章,对c/c++linux课程感兴趣的读者,可以点击链接 C/C++后台高级服务器课程介绍 详细查看课程的服务。

1. CGI 概念理解

  CGI:通用网关接口Common Gateway Interface/CGI描述了客户端和服务器程序之间传输数据的一种标准,可以让一个客户端,从网页浏览器向执行在网络服务器上的程序请求数据。CGI 独立于任何语言的,CGI 程序可以用任何脚本语言或者是完全独立编程语言实现,只要这个语言可以在这个系统上运行

http://localhost/login?user=zhang3&passwd=123456&age=12&sex=man

  比如上面这个url请求,web nginx server是能够解析的,但是它不能处理,因为这里带了数据,是动态请求,而nginx只能处理静态请求。所以要把动态请求交给CGI去处理。

http://localhost/login?user=zhang3&passwd=123456&age=12&sex=man

  1. 用户通过浏览器访问服务器, 发送了一个请求, 请求的url如上
  2. 服务器接收数据, 对接收的数据进行解析
  3. nginx对于一些登录数据不知道如何处理, nginx将数据发送给了cgi程序
    • 服务器端会创建一个CGI进程
  4. CGI进程执行
    • 加载配置, 如果有需求加载配置文件获取数据
    • 连接其他服务器: 比如数据库
    • 逻辑处理:
    • 得到处理结果, 将结果发送给服务器
    • 退出
  5. 服务器将cgi处理结果发送给客户端
    • 每一个动态资源请求都有一个CGI进程
    • 在服务器端CGI进程会被频繁的创建销毁
    • 频繁的创建销毁CGI进程,服务器开销大, 效率低

2. FastCGI 概念理解

  fastCGI:快速通用网关接口(Fast Common Gateway Interface/FastCGI)是通用网关接口(CGI)的改进,描述了客户端和服务器程序之间传输数据的一种标准。FastCGI致力于减少Web服务器CGI程式之间互动的开销,从而使服务器可以同时处理更多的Web请求。与为每个请求创建一个新的进程不同,FastCGI使用持续的进程来处理一连串的请求。这些进程由FastCGI进程管理器管理,而不是web服务器。

fastCGI与CGI的区别: CGI 就是所谓的短生存期应用程序,FastCGI 就是所谓的长生存期应用程序。FastCGI像是一个常驻(long-live)型的CGI,它可以一直执行着,不会每次都要花费时间去fork一次。说人话:CGI是来一个请求就fork一个进程,而fastCGI只会fork一个进程,多个请求都使用同一个进程。

http://localhost/login?user=zhang3&passwd=123456&age=12&sex=man

  1. 用户通过浏览器访问服务器, 发送了一个请求, 请求的url如上
  2. 服务器接收数据, 对接收的数据进行解析
  3. nginx对于一些登录数据不知道如何处理, nginx将数据发送给了fastcgi程序
    • 通过本地套接字
    • 网络通信的方式
  4. fastCGI程序如何启动
    • 不是由web服务器直接启动
    • 通过一个fastCGI进程管理器启动
  5. fastcgi启动
    • 加载配置 - 可选
    • 连接服务器 - 数据库
    • 循环
      • 服务器有请求 -> 处理
      • 没有请求 -> 阻塞等待
  6. 服务器将fastCGI的处理结果发送给客户端
    • 通过本地套接字
    • 网络通信的方式

3. FastCGI和spawn-fcgi安装

  fastCGI是一个框架,它给我们提供了api,它内部遵循cgi协议,以及与服务器通信的细节隐藏了。我们只需要遵循fastCGI的api接口去写程序,就可以与nginx配合使用了。

  spawn-fcgi是FastCGI的进程管理器,也就是说我们编写的FastCGI程序,是由spawn-fcgi去启动,而不是由nginx web服务器去启动。

  spawn-fcgi-1.6.4.tar.gzfcgi-2.4.1-SNAP-0910052249.tar.gz都在前言的git源码处,需要的可以进git hub下载。

  1. 安装fastCGI
tar zxvf fcgi-2.4.1-SNAP-0910052249.tar.gz
cd fcgi-2.4.1-SNAP-0910052249/
./configure
make
make install
ldconfig
make
# make的时候如果出现EOF的错误,是因为这个cpp文件没有添加头文件
- fcgio.cpp:50:14: error: 'EOF' was not declared in this scope
- 没有包含对应的头文件:- stdio.h - c- cstdio -> c++
vi ./libfcgi/fcgio.cpp
#include<cstdio>
  1. 安装spawn-fcgi
tar -zxvf spawn-fcgi-1.6.4.tar.gz
cd spawn-fcgi-1.6.4/
./configure
make
make install
ldconfig

4. FastCGI和 Nginx的关系

  nginx 不能像apache那样直接执行外部可执行程序,但nginx可以作为代理服务器,将请求转发给后端服务器,这也是nginx的主要作用之一。其中nginx就支持FastCGI代理,接收客户端的请求,然后将请求转发给后端fastcgi进程。后文会介绍如何使用C/C++编写cgi/fastcgi,并部署到nginx中。

  通过前面的介绍知道,fastcgi进程由FastCGI进程管理器管理,而不是nginx。这样就需要一个FastCGI管理,管理我们编写fastcgi程序。我们使用spawn-fcgi作为FastCGI进程管理器。

  spawn-fcgi是一个通用的FastCGI进程管理器,简单小巧,原先是属于lighttpd的一部分,后来由于使用比较广泛,所以就迁移出来作为独立项目了。spawn-fcgi使用pre-fork 模型,功能主要是打开监听端口,绑定地址,然后fork-and-exec创建我们编写的fastcgi应用程序进程,退出完成工作。fastcgi应用程序初始化,然后进入死循环侦听socket的连接请求。

http://localhost/login?user=zhang3&passwd=123456&age=12&sex=man

  • 客户端访问, 发送请求
  • nginx web服务器, 无法处理用户提交的数据,将数据转发给 spawn-fcgi
  • spawn-fcgi - 通信过程中的服务器角色
    • 被动接收数据
    • 在spawn-fcgi启动的时候给其绑定IP和端口
  • fastCGI程序
    • 程序猿写的 -> login.c -> 可执行程序login
    • 使用 spawn-fcgi 进程管理器启动 login 程序, 得到一进程
      • login进程处理请求数据
      • 处理完后数据发送给nginx
      • 阻塞等待下一个请求的到来

5. Nginx数据转发-修改配置文件

  nginx的数据转发,需要修改nginx的配置文件 nginx.conf(/usr/local/nginx/conf)

通过请求的url http://localhost/login?user=zhang3&passwd=123456&age=12&sex=man 转换为一个指令:- 去掉协议- 去掉域名/IP + 端口- 如果尾部有文件名 去掉- 去掉 ? + 后边的字符串- 剩下的就是服务器要处理的指令: /login
location /login
{# 转发这个数据, fastCGI进程fastcgi_pass 地址信息:端口;# fastcgi.conf 和nginx.conf在同一级目录: /usr/local/nginx/conf# 这个文件中定义了一些http通信的时候用到环境变量, nginx赋值的include fastcgi.conf;
}
地址信息: - localhost- 127.0.0.1- 192.168.1.100
端口: 找一个空闲的没有被占用的端口即可

6. spawn-fcgi如何启动

# 前提条件: 程序猿的fastCGI程序已经编写完毕 -> 可执行文件 login
spawn-fcgi -a IP地址 -p 端口 -f ./fastcgi可执行程序login- IP地址: 应该和nginx的 fastcgi_pass 配置项对应- nginx: localhost       ->   IP: 127.0.0.1- nginx: 127.0.0.1      ->   IP: 127.0.0.1- nginx: 192.168.109.101 ->   IP: 192.168.109.101
- 端口:应该和nginx的 fastcgi_pass 中的端口一致

7. FastCGI程序怎么写

7.1 echo.c代码阅读与分析

  进入FastCGI源码目录下的example目录,看echo.c是如何编写的

root@wxf:/source_code_dir/fcgi-2.4.1-SNAP-0910052249/examples# pwd
/source_code_dir/fcgi-2.4.1-SNAP-0910052249/examples
root@wxf:/source_code_dir/fcgi-2.4.1-SNAP-0910052249/examples# ls
authorizer      echo.c        echo.mak   echo-x.o    Makefile.am  size.o
authorizer.c    echo-cpp      echo.o     log-dump    Makefile.in  threaded
authorizer.mak  echo-cpp.cpp  echo-x     log-dump.c  size         threaded.c
authorizer.o    echo-cpp.mak  echo-x.c   log-dump.o  size.c       threaded-threaded.o
echo            echo-cpp.o    echox.mak  Makefile    size.mak
  • echo.c
#ifndef lint
static const char rcsid[] = "$Id: echo.c,v 1.5 1999/07/28 00:29:37 roberts Exp $";
#endif /* not lint */#include "fcgi_config.h"
#include <stdlib.h>#ifdef HAVE_UNISTD_H
#include <unistd.h>
#endif#ifdef _WIN32
#include <process.h>
#else
extern char **environ;
#endif#include "fcgi_stdio.h"static void PrintEnv(char *label, char **envp) {printf("%s:<br>\n<pre>\n", label);for (; *envp != NULL; envp++) {printf("%s\n", *envp);}printf("</pre><p>\n");
}int main() {char **initialEnv = environ;int count = 0;while (FCGI_Accept() >= 0) {char *contentLength = getenv("CONTENT_LENGTH");int len;printf("Content-type: text/html\r\n""\r\n""<title>FastCGI echo</title>""<h1>FastCGI echo</h1>\n""Request number %d,  Process ID: %d<p>\n", ++count, getpid());if (contentLength != NULL) {len = strtol(contentLength, NULL, 10);}else {len = 0;}if (len <= 0) {printf("No data from standard input.<p>\n");}else {int i, ch;printf("Standard input:<br>\n<pre>\n");for (i = 0; i < len; i++) {if ((ch = getchar()) < 0) {printf("Error: Not enough bytes received on standard input<p>\n");break;}putchar(ch);}printf("\n</pre><p>\n");}PrintEnv("Request environment", environ);PrintEnv("Initial environment", initialEnv);} /* while */return 0;
}

  char **environ是一个全局变量,在#include <unistd.h>中,它其实就是存储了linux bash中输入env打印出来的环境变量。所以PrintEnv这个函数就是把所有的键值对打印出来。

  程序显示往标准输出里面打印了这个kv

printf("Content-type: text/html\r\n");

  while (FCGI_Accept() >= 0)符合上文的分析,如果有请求来了那么就会进入循环。

  通过环境变量,获取http请求报文头中这个字段的值。CONTENT_LENGTH其实就在fastcgi.conf中,在上文修改配置文件的时候,被include了。

char *contentLength = getenv("CONTENT_LENGTH");

  然后从标准输入读这个长度的数据,再写入标准输出。

for (i = 0; i < len; i++) {if ((ch = getchar()) < 0) {printf("Error: Not enough bytes received on standard input<p>\n");break;}putchar(ch);
}

7.2 fastCGI接收与发送数据

  从上文我们发现,该程序从标准输入读数据,往标准输出写数据。很明显,这里是做了重定向的。表面操作的是终端,实际被重定向到了内部被隐藏的fd。

dup2(fd,STDIN_FILENO)
dup2(fd,STDOUT_FILENO)

7.3 fastCGI程序编写流程与思路

// http://localhost/login?user=zhang3&passwd=123456&age=12&sex=man
// 要包含的头文件
#include "fcgi_config.h" // 可选
#include "fcgi_stdio.h" // 必须的, 编译的时候找不到这个头文件, find->path , gcc -I
// 编写代码的流程
int main()
{// FCGI_Accept()是一个阻塞函数, nginx给fastcgi程序发送数据的时候解除阻塞while (FCGI_Accept() >= 0) {// 1. 接收数据// 1.1 get方式提交数据 - 数据在请求行的第二部分,QUERY_STRING直接获取数据// user=zhang3&passwd=123456&age=12&sex=manchar *text = getenv("QUERY_STRING"); // 1.2 post方式提交数据,只能获取数据长度,需要手动读出来char *contentLength = getenv("CONTENT_LENGTH");// 根据长度大小判断是否需要循环读-read// 2. 按照业务流程进行处理......// 3. 将处理结果发送给nginx// 3.1 数据回发的时候, 需要告诉nginx处理结果的格式 - 假设是html格式。只有Content-type需要指定,别的头部字段自动,不用我们管// 要放在处理结果之前发送该kv字段//Content-type后续会介绍printf("Content-type: text/html\r\n");// 注意\r\n ,别忘了空行!printf("\r\n");// 3.2 再将处理结果回发printf("<html>处理结果</html>");}
}

7.4 实现一个fastCGI程序并测试

  • 第一步:修改Nginx数据转发的配置文件
    location /upload/UploadAction {fastcgi_pass 192.168.109.101:10000;include fastcgi.conf;}
  • 第二步:编写fastCGI程序
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include "fcgi_stdio.h"int main(int argc, char *argv[]) {int count = 0;while (FCGI_Accept() >= 0) {printf("Content-type: text/html\r\n");printf("\r\n");printf("<title>Fast CGI Hello WXF!</title>");printf("<h1>Fast CGI Hello WXF!</h1>");printf("Request number %d running on host <i>%s</i>\n", ++count, getenv("SERVER_NAME"));}return 0;
}
  • 第三步:启动fastcgi进程管理器
root@wxf:/source_code_dir# gcc -o test fcgi.c -lfcgi
root@wxf:/source_code_dir# spawn-fcgi -a 192.168.109.101 -p 10000 -f ./test
spawn-fcgi: child spawned successfully: PID: 14487
  • 第四步:测试

8. 部署一个能够上传文件的网页

8.1 上传文件夹并修改Nginx的配置文件

  至于为什么放在/usr/local/nginx/目录下不再赘述,不懂的读者可以翻阅我上一篇nginx的文章。

location / {root   zyFile2;index  index.html index.htm;
}


8.2 修改配置文件,实现数据转发

  为什么是404 Not Found?通过html的源码,我们发现发送的是post请求,并且路径是/upload/UploadAction。这意味着带数据的动态请求,nginx是处理不了的,所以这个时候我们需要配置数据转发。

location /upload/UploadAction {fastcgi_pass 192.168.109.101:10000;include fastcgi.conf;
}


  An error occurred是因为虽然我们配置了数据转发,但是我们并没有启动fastcgi程序去处理这个请求。

  现在我们使用之前介绍过的echo.c当作fastcgi程序,实现回发的功能看看。

root@wxf:/source_code_dir# gcc -o echo echo.c -lfcgi
root@wxf:/source_code_dir# spawn-fcgi -a 192.168.109.101 -p 10000 -f ./echo
spawn-fcgi: child spawned successfully: PID: 14762

  注意这里不要上传太大的图片,因为我们回发的类型是Content-type: text/html,并不符合图片的类型。可以看到这里我们上传的流程成功了,我们并没有编写上传的代码,只是做了个echo而已。

8.3 reference & libfcgi.so.0 => not found 问题解决

没有对应的动态库,那么加上即可

root@wxf:/source_code_dir# gcc -o echo echo.c
/tmp/ccghTkS5.o: In function `PrintEnv':
echo.c:(.text+0x24): undefined reference to `FCGI_printf'
echo.c:(.text+0x41): undefined reference to `FCGI_printf'
echo.c:(.text+0x63): undefined reference to `FCGI_printf'
/tmp/ccghTkS5.o: In function `main':
echo.c:(.text+0xb6): undefined reference to `FCGI_printf'
echo.c:(.text+0xf6): undefined reference to `FCGI_printf'
/tmp/ccghTkS5.o:echo.c:(.text+0x109): more undefined references to `FCGI_printf' follow
/tmp/ccghTkS5.o: In function `main':
echo.c:(.text+0x117): undefined reference to `FCGI_getchar'
echo.c:(.text+0x131): undefined reference to `FCGI_printf'
echo.c:(.text+0x13d): undefined reference to `FCGI_putchar'
echo.c:(.text+0x15a): undefined reference to `FCGI_printf'
echo.c:(.text+0x188): undefined reference to `FCGI_Accept'
collect2: error: ld returned 1 exit status
root@wxf:/source_code_dir# cd fcgi-2.4.1-SNAP-0910052249/
root@wxf:/source_code_dir/fcgi-2.4.1-SNAP-0910052249# find ./ -name "lib*.so"
./libfcgi/.libs/libfcgi++.so
./libfcgi/.libs/libfcgi.so
root@wxf:/source_code_dir/fcgi-2.4.1-SNAP-0910052249# cd ..
root@wxf:/source_code_dir# gcc -o echo echo.c -lfcgi

    ldd echo查看程序启动之后需要加载的动态库,如果发现not found,那么就有问题了。


  使用find查找这个动态库的位置

root@wxf:/source_code_dir# find / -name "libfcgi.so"
/usr/local/lib/libfcgi.so
/source_code_dir/fcgi-2.4.1-SNAP-0910052249/libfcgi/.libs/libfcgi.so

  我们需要让程序能够连接到这个动态库,就把这个/usr/local/lib这个目录放到下面的配置文件,并用ldconfig更新。

root@wxf:/source_code_dir# vi /etc/ld.so.conf
...
root@wxf:/source_code_dir# ldconfig

  上述问题发生的原因是因为,手动安装这些源码make install之后,没有输入ldconfig导致的。

9. 其他知识点

9.1 HTTP环境变量 -> fastcgi.conf

  上文测试的echo程序回发了两个环境变量,Request environmentInitial environment。分别是HTTP请求的环境变量和fastcgi系统的环境变量。

  Request environment中的环境变量都在fastcgi.conf中记录着。

  Initial environment中的环境变量都在bash: env 打印出来的一样。

Request number 2, Process ID: 14762
Standard input:
------WebKitFormBoundaryM3hhJcyUYWxGGdPb
Content-Disposition: form-data; name="file"; filename="100k.png"
Content-Type: image/png�PNGIHDRdN)IDAT8c���?Y��,] M�:  �h��$FS�hJ�)�5��IEND�B`�
------WebKitFormBoundaryM3hhJcyUYWxGGdPb
Content-Disposition: form-data; name="tailor"false
------WebKitFormBoundaryM3hhJcyUYWxGGdPb--Request environment:
FCGI_ROLE=RESPONDER
SCRIPT_FILENAME=/usr/local/nginx/html/upload/UploadAction
QUERY_STRING=
REQUEST_METHOD=POST
CONTENT_TYPE=multipart/form-data; boundary=----WebKitFormBoundaryM3hhJcyUYWxGGdPb
CONTENT_LENGTH=377
SCRIPT_NAME=/upload/UploadAction
REQUEST_URI=/upload/UploadAction
DOCUMENT_URI=/upload/UploadAction
DOCUMENT_ROOT=/usr/local/nginx/html
SERVER_PROTOCOL=HTTP/1.1
REQUEST_SCHEME=http
GATEWAY_INTERFACE=CGI/1.1
SERVER_SOFTWARE=nginx/1.16.1
REMOTE_ADDR=192.168.109.1
REMOTE_PORT=55644
SERVER_ADDR=192.168.109.101
SERVER_PORT=80
SERVER_NAME=localhost
REDIRECT_STATUS=200
HTTP_HOST=192.168.109.101
HTTP_CONNECTION=keep-alive
HTTP_CONTENT_LENGTH=377
HTTP_USER_AGENT=Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/104.0.0.0 Safari/537.36
HTTP_CONTENT_TYPE=multipart/form-data; boundary=----WebKitFormBoundaryM3hhJcyUYWxGGdPb
HTTP_ACCEPT=*/*
HTTP_ORIGIN=http://192.168.109.101
HTTP_REFERER=http://192.168.109.101/demo.html
HTTP_ACCEPT_ENCODING=gzip, deflate
HTTP_ACCEPT_LANGUAGE=zh-CN,zh;q=0.9
Initial environment:
LESSOPEN=| /usr/bin/lesspipe %s
MAIL=/var/mail/root
USER=root
SSH_CLIENT=192.168.109.1 61011 22
SHLVL=1
OLDPWD=/usr/local/nginx/conf
HOME=/root
SSH_TTY=/dev/pts/0
DBUS_SESSION_BUS_ADDRESS=unix:path=/run/user/0/bus
LOGNAME=root
_=/usr/local/bin/spawn-fcgi
XDG_SESSION_ID=1
TERM=xterm
PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:/usr/games:/usr/local/games:/snap/bin
XDG_RUNTIME_DIR=/run/user/0
DISPLAY=localhost:10.0
LANG=en_US.UTF-8
LS_COLORS=rs=0:di=01;34:ln=01;36:mh=00:pi=40;33:so=01;35:do=01;35:bd=40;33;01:cd=40;33;01:or=40;31;01:mi=00:su=37;41:sg=30;43:ca=30;41:tw=30;42:ow=34;42:st=37;44:ex=01;32:*.tar=01;31:*.tgz=01;31:*.arc=01;31:*.arj=01;31:*.taz=01;31:*.lha=01;31:*.lz4=01;31:*.lzh=01;31:*.lzma=01;31:*.tlz=01;31:*.txz=01;31:*.tzo=01;31:*.t7z=01;31:*.zip=01;31:*.z=01;31:*.Z=01;31:*.dz=01;31:*.gz=01;31:*.lrz=01;31:*.lz=01;31:*.lzo=01;31:*.xz=01;31:*.zst=01;31:*.tzst=01;31:*.bz2=01;31:*.bz=01;31:*.tbz=01;31:*.tbz2=01;31:*.tz=01;31:*.deb=01;31:*.rpm=01;31:*.jar=01;31:*.war=01;31:*.ear=01;31:*.sar=01;31:*.rar=01;31:*.alz=01;31:*.ace=01;31:*.zoo=01;31:*.cpio=01;31:*.7z=01;31:*.rz=01;31:*.cab=01;31:*.wim=01;31:*.swm=01;31:*.dwm=01;31:*.esd=01;31:*.jpg=01;35:*.jpeg=01;35:*.mjpg=01;35:*.mjpeg=01;35:*.gif=01;35:*.bmp=01;35:*.pbm=01;35:*.pgm=01;35:*.ppm=01;35:*.tga=01;35:*.xbm=01;35:*.xpm=01;35:*.tif=01;35:*.tiff=01;35:*.png=01;35:*.svg=01;35:*.svgz=01;35:*.mng=01;35:*.pcx=01;35:*.mov=01;35:*.mpg=01;35:*.mpeg=01;35:*.m2v=01;35:*.mkv=01;35:*.webm=01;35:*.ogm=01;35:*.mp4=01;35:*.m4v=01;35:*.mp4v=01;35:*.vob=01;35:*.qt=01;35:*.nuv=01;35:*.wmv=01;35:*.asf=01;35:*.rm=01;35:*.rmvb=01;35:*.flc=01;35:*.avi=01;35:*.fli=01;35:*.flv=01;35:*.gl=01;35:*.dl=01;35:*.xcf=01;35:*.xwd=01;35:*.yuv=01;35:*.cgm=01;35:*.emf=01;35:*.ogv=01;35:*.ogx=01;35:*.aac=00;36:*.au=00;36:*.flac=00;36:*.m4a=00;36:*.mid=00;36:*.midi=00;36:*.mka=00;36:*.mp3=00;36:*.mpc=00;36:*.ogg=00;36:*.ra=00;36:*.wav=00;36:*.oga=00;36:*.opus=00;36:*.spx=00;36:*.xspf=00;36:
SHELL=/bin/bash
LESSCLOSE=/usr/bin/lesspipe %s %s
PWD=/source_code_dir
SSH_CONNECTION=192.168.109.1 61011 192.168.109.101 22
XDG_DATA_DIRS=/usr/local/share:/usr/share:/var/lib/snapd/desktop

  最重要的就是下面四个。如果是get请求,则不会有CONTENT_LENGTH字段,参数在QUERY_STRING中。如果是post请求,则有CONTENT_LENGTH字段。

QUERY_STRING
REQUEST_METHOD
CONTENT_TYPE
CONTENT_LENGTH

  • fastCGI环境变量 - fastcgi.conf

    环境变量 说明
    SCRIPT_FILENAME 脚本文件请求的路径
    QUERY_STRING 请求的参数;如?app=123
    REQUEST_METHOD 请求的动作(GET,POST)
    CONTENT_TYPE 请求头中的Content-Type字段
    CONTENT_LENGTH 请求头中的Content-length字段
    SCRIPT_NAME 脚本名称
    REQUEST_URI 请求的地址不带参数
    DOCUMENT_URI 与$uri相同
    DOCUMENT_ROOT 网站的根目录。在server配置中root指令中指定的值
    SERVER_PROTOCOL 请求使用的协议,通常是HTTP/1.0或HTTP/1.1
    GATEWAY_INTERFACE cgi 版本
    SERVER_SOFTWARE nginx 版本号,可修改、隐藏
    REMOTE_ADDR 客户端IP
    REMOTE_PORT 客户端端口
    SERVER_ADDR 服务器IP地址
    SERVER_PORT 服务器端口
    SERVER_NAME 服务器名,域名在server配置中指定的server_name

9.2 常用的四种 Content-Type

  • application/x-www-form-urlencoded
# 请求行
POST http://www.example.com HTTP/1.1
# 请求头
Content-Type: application/x-www-form-urlencoded;charset=utf-8
# 空行
# 请求数据(向服务器提交的数据),用&做间隔,就是application/x-www-form-urlencoded格式
title=test&user=kevin&passwd=32222
  • application/json
# 请求行
POST / HTTP/1.1
# 请求头
Content-Type: application/json;charset=utf-8
# 空行
# 请求数据,如果是json格式就必须是application/json
{"title":"test","sub":[1,2,3]}
  • text/xml
# 请求行
POST / HTTP/1.1
# 请求头
Content-Type: text/xml
# 空行
<?xml version="1.0" encoding="utf8"?>
# 请求数据
<methodcall><methodname color="red">examples.getStateName</methodname><params><value><i4>41</i4></value></params>
</methodcall>
  • multipart/form-data

  multipart/form-data是传输大文件常用的一种数据格式,在数据刚开始的时候有一个分界线,这个分界线是随机生成的,在结束的时候也有一个分界线。

------WebKitFormBoundaryPpL3BfPQ4cHShsBz

  在分界线下面还有一个Content-DispositionContent-Type,这是对文件属性的描述。文件内容在两个分界线中间,那么用这种格式我们就可以上传多个文件了,一个文件对应一个数据块。不同的数据块都是相同的格式(Content-Disposition + Content-Type + 空行 + 文件内容);当然也可以上传一个大文件,分为多个小数据块。

  文件的Content-Type不需要我们去记后面写什么,用到的时候直接查表即可:https://tool.oschina.net/commons

# 请求行
POST / HTTP/1.1
# 请求头
Content-Type: multipart/form-data
# 空行
# 发送的数据
------WebKitFormBoundaryPpL3BfPQ4cHShsBz \r\n
Content-Disposition: form-data; name="file"; filename="1.png"
Content-Type: image/png\r\n; md5="xxxxxxxxxx"
\r\n
.............文件内容................
.............文件内容................
------WebKitFormBoundaryPpL3BfPQ4cHShsBz
Content-Disposition: form-data; name="file"; filename="2.png"
\r\n
.............文件内容................
.............文件内容................
------WebKitFormBoundaryPpL3BfPQ4cHShsBz
Content-Disposition: form-data; name="tailor"
\r\n
false
------WebKitFormBoundaryPpL3BfPQ4cHShsBz--

10. fastCGI总结

  1. fastCGI是什么?

    • 运行在服务器端的代码, 帮助服务器处理客户端提交的动态请求
  2. fastCGI干什么?

    • nginx服务器处理不了动态请求,fastCGI帮助服务器处理客户端提交的动态请求
  3. fastCGI怎么用?

    • 前提条件fastCGI和nginx部署在同一台机器上

    • nginx如何转发数据

      # 分析出客户端请求对应的指令 -- /test
      location /test
      {# 转发出去fastcgi_pass 地址:端口;include fastcgi.conf;
      }
      
    • fastcgi如何接收数据

      # 启动, 通过spawn-fcgi启动
      spawn-fcgi -a IP -p port -f ./fcgi
      # 编写fastCGI程序的时候- 接收数据: 调用读终端的函数就是接收数据- 发送数据: 调用写终端的函数就是发送数据
      
    • fastcgi如何处理数据

      // 编写登录的fastCgI程序
      int main()
      {while(FCGI_Accept() >= 0){// 1. 接收登录信息 -> 环境变量中// post -> 读数据块的长度 CONTENT-LENGTH// get -> 从请求行的第二部分读 QUEERY_STRING// 2. 处理数据// 3. 回发结果 -> 格式假设是jsonprintf("Content-type: application/json\r\n");printf("\r\n");printf("{\"status\":\"OK\"}")}
      }
      

C++文件服务器项目—FastCGI—4相关推荐

  1. C++文件服务器项目—FastDFS—1

    C++文件服务器项目-FastDFS-1 前言 1. 项目架构 2. 分布式文件系统 2.1 传统文件系统 2.2 分布式文件系统 3. FastDFS介绍 3.1 fdfs概述 3.2 fdfs框架 ...

  2. C++文件服务器项目—数据库表设计 与 后端接口设计—6

    C++文件服务器项目-数据库表的设计-6 前言 1. 数据库建表 1.1 用户信息表 user_info 1.2 文件信息表 file_info 1.3 用户文件列表表 user_file_list ...

  3. C++文件服务器项目—项目总结与反向代理—7

    C++文件服务器项目-项目总结与反向代理-7 1. 项目总结 2. 项目提炼 3. web服务器的反向代理 4. 存储节点的反向代理   组件介绍基本写完了,后续进行深入.   本专栏知识点是通过零声 ...

  4. C++文件服务器项目—Nginx—3

    C++文件服务器项目-Nginx-3 前言 1. Nginx一些基本概念 1.1 Nginx初步认识 1.2 正向代理概念理解 1.3 反向代理概念理解 2. Nginx的安装与配置 2.1 Ngin ...

  5. c语言实现fastcgi

    作者 QQ群:852283276 微信:arm80x86 微信公众号:青儿创客基地 B站:主页 https://space.bilibili.com/208826118 参考 fastcgi官网 网关 ...

  6. 开源应用框架BitAdminCore:更新日志20180605

    索引 NET Core应用框架之BitAdminCore框架应用篇系列 框架演示:http://bit.bitdao.cn 框架源码:https://github.com/chenyinxin/coo ...

  7. tomcat上传文件到不同服务器,使用SpringMVC进行跨服务器上传文件出现的那些坑

    [1] 405 Method Not Allowed 原因是因为Servlet默认是只读的,也就是写不进去(文件上传不过来) 所以要在文件服务器的 web.xml 对 Servlet 进行配置 def ...

  8. 春招总结(作业帮、华为、腾讯、美团、京东面经)

    在WP上发博客没人看( ̄▽ ̄)",最终还是选择CSDN上总结一下自己春招的经历.问题整理得不太及时,但已经是尽力回想了,希望能帮到需要的童鞋. 2018.05.27更新 已在美团点评云计算部 ...

  9. Springboot项目集成Minio文件服务器(下)

    Springboot项目集成Minio文件服务器(下) 1.配置依赖 在pom文件里面配置Minio的相关依赖. <!--添加minio的依赖--><dependency>&l ...

  10. 服务器项目文件,文件服务器的项目

    文件服务器的项目 内容精选 换一换 表1列出了弹性文件服务的常用功能.在使用弹性文件服务之前,建议您先通过常用概念介绍了解NFS.CIFS等基本概念,以便更好地理解弹性文件服务提供的功能.√:表示该类 ...

最新文章

  1. Android Eclipse之Git插件安装、配置、提交、修改、冲突、查看历史、覆盖。
  2. Fiddler之弱网测试(Web)
  3. linux 初学体验 20155317 王新玮
  4. 【leetcode】109. Convert Sorted List to Binary Search Tree
  5. JDK1.6官方下载_JDK6官方下载地址:http://www.java.net/download
  6. vue input 赋值无效
  7. windows server 2008R2 部署WSUS 服务
  8. sybase相关的知识
  9. pdg转pdf的正确方法!!
  10. win10、win7安装SQLServer2000最详细教程
  11. 服务器系统更新后找不到硬盘,重装win10系统找不到硬盘完美解决方法
  12. html 字幕文件怎么打开,srt文件怎么打开
  13. openCV+VS2010结合+正确运行图片
  14. 认识物联网系列——物联网架构
  15. Linux正则表达式和文本处理工具(gred、awk、sed)
  16. 孩子花掉父母两年工资,游戏你到底是“天使”还是“恶魔”?
  17. pandas取出特定行列数据
  18. 「首席架构师推荐」精选数据分析软件列表
  19. MybatisPlus多表联查分页多条件查
  20. Hive数据去重、多变一与一变多等实现

热门文章

  1. 提升领导力这7个法则,你不得不知道
  2. [HCTF 2018]WarmUp1PHP 代码审计
  3. 华为防火墙笔记-安全策略
  4. 魔兽对战平台官网地图《小兵合成》 最强攻略+合成表+隐藏合成+隐藏英雄+
  5. Reflex仓库管理系统(WMS)简介
  6. 植物大战僵尸 修改存档和金钱
  7. smbrun - smbd和外部程序间的接口程序。
  8. netbeans莫明其妙的报错
  9. Linux命令查看Linux服务器内存、CPU、显卡、硬盘使用情况
  10. 巨杉数据库 java,巨杉Tech|SequoiaDB 巨杉数据库高可用容灾测试