FastCGI 是一种协议,它是建立在CGI/1.1基础之上的,把CGI/1.1里面的要传递的数据通过FastCGI协议定义的顺序和格式进行传递。为了更好理解PHP-FPM的工作,下面具体阐述一下FastCGI协议的内容。
1. 消息类型FastCGI协议分为了10种类型,具体定义如下:
typedef enum _fcgi_request_type { FCGI_BEGIN_REQUEST = 1, /* [in] */ FCGI_ABORT_REQUEST = 2, /* [in] (not supported) */ FCGI_END_REQUEST = 3, /* [out] */ FCGI_PARAMS = 4, /* [in] environment variables */ FCGI_STDIN = 5, /* [in] post data */ FCGI_STDOUT = 6, /* [out] response */ FCGI_STDERR = 7, /* [out] errors */ FCGI_DATA = 8, /* [in] filter data (not supported) */ FCGI_GET_VALUES = 9, /* [in] */ FCGI_GET_VALUES_RESULT = 10 /* [out] */} fcgi_request_type;
整个FastCGI是二进制连续传递的,定义了一个统一结构的消息头,用来读取每个消息的消息体,方便消息包的切割。一般情况下,最先发送的是FCGI_BEGIN_REQUEST类型的消息,然后是FCGI_PARAMS和FCGI_STDIN类型的消息,当FastCGI响应处理完后,将发送FCGI_STDOUT和FCGI_STDERR类型的消息,最后以FCGI_END_REQUEST表示请求的结束。FCGI_BEGIN_REQUEST和FCGI_END_REQUEST分别表示请求的开始和结束,与整个协议相关。
2. 消息头对于10种类型的消息,都是以一个消息头开始的,其结构体定义如下:
typedef struct _fcgi_header { unsigned char version; unsigned char type; unsigned char requestIdB1; unsigned char requestIdB0; unsigned char contentLengthB1; unsigned char contentLengthB0; unsigned char paddingLength; unsigned char reserved;} fcgi_header;
其中,
version标识FastCGI协议版本
type 标识FastCGI记录类型
requestId标识消息所属的FastCGI请求
requestId计算方式如下:
(requestIdB1 8) + requestIdB0
所以requestId的范围为0~2的16次方-1,也就是0~65535;
contentLength标识消息的contentData组件的字节数,计算方式跟requestId类似,范围同样是0~65535:
(contentLengthB1 8) | contentLengthB0
paddingLength标识消息的paddingData组件的字节数,范围是0~255;协议通过paddingData提供给发送者填充发送的记录的功能,并且方便接受者通过paddingLength快速的跳过paddingData。填充的目的是允许发送者为更有效地处理保持对齐的数据。如果内容的长度超过65535怎么办呢?答案是可以分成多个消息发送。
3. FCGI_BEGIN_REQUESTFCGI_BEGIN_REQUEST 的结构体定义如下:
typedef struct _fcgi_begin_request { unsigned char roleB1; unsigned char roleB0; unsigned char flags; unsigned char reserved[5];} fcgi_begin_request;
其中role代表的是Web服务器期望应用扮演的角色,计算方式是:
(roleB1 8) + roleB0
对于PHP7中,处理了三种角色,分别是FCGI_RESPONDER,FCGI_AUTHORIZER 和FCGI_FILTER。
flags FCGI_KEEP_CONN:如果为0,则在对本次请求响应后关闭链接。如果非0,在对本次请求响应后不会关闭链接。
4. 名-值对对于,type为FCGI_PARAMS类型,FastCGI协议中提供了名-值对来很好的满足读写可变长度的name和html' target='_blank'>value,格式如下:
nameLength+valueLength+name+value
为了节省空间,对于0~127长度的值,Length使用了一个char来表示,第一位为0,对于大于127的长度的值,Length使用了4个char来表示,第一位为1;如图所示
长度计算代码如下:
if (UNEXPECTED(name_len = 128)) { if (UNEXPECTED(p + 3 = end)) return 0; name_len = ((name_len 0x7f) 24); name_len |= (*p++ 16); name_len |= (*p++ 8); name_len |= *p++;}
这样最长可以表达0~2的31次方的长度。
5. 请求协议FastCGI协议的定义结构体如下:
typedef struct _fcgi_begin_request_rec { fcgi_header hdr; fcgi_begin_request body;} fcgi_begin_request_rec;
分析完FastCGI的协议,我们整体掌握了请求的FastCGI消息的内容,我们通过访问对应的接口,采用gdb抓取其中的内容:
首先我们修改php-fpm.conf的参数,保证只启动一个worker:
pm.max_children = 1
然后重新启动php-fpm:
./sbin/php-fpm -y etc/php-fpm.conf
然后对worker进行gdb:
ps aux | grep php-fpmroot 30014 0.0 0.0 142308 4724 ? Ss Nov26 0:03 php-fpm: master process (etc/php-fpm.conf)chenlei 30015 0.0 0.0 142508 5500 ? S Nov26 0:00 php-fpm: pool wwwgdb –p 30015(gdb) b fcgi_read_request
然后通过浏览器访问nginx,nginx转发到php-fpm的worker上,根据gdb可以打印出FastCGI消息的内容:
(gdb) b fcgi_read_request
对于第一个消息,内容如图:
其中type对应的是FCGI_BEGIN_REQUEST,requestid为1,长度为8, 恰好是fcgi_begin_request结构体的大小,内容如图:
role对应的是FCGI_RESPONDER。继续往下读,得到的消息内容如图:
其中type对应的是FCGI_PARAMS,requestid为1,长度为:
(contentLengthB1 8) | contentLengthB0 == 987
paddingLength=5,而987+5=992,恰好是8的倍数。根据contentLength+ paddingLength向后读取992长度的字节流,我们打印一下:
(gdb) p *p@987$1 = /017TSCRIPT_FILENAME/home/xiaoju/webroot/beatles/application/mis/mis/src/index.php/admin/operation/index/f/016QUERY_STRINGactivity_id=89/016/003REQUEST_METHODGET/f/000CONTENT_TYPE/016/000CONTENT_LENGTH/v SCRIPT_NAME/index.php/admin/operation/index/v%REQUEST_URI/admin/operation/index?activity_id=89/f DOCUMENT_URI/index.php/admin/operation/index/r4DOCUMENT_ROOT/home/xiaoju/webroot/beatles/application/mis/mis/src/017/bSERVER_PROTOCOLHTTP/1.1/021/aGATEWAY_INTERFACECGI/1.1/017/vSERVER_SOFTWAREnginx/1.2.5/v/rREMOTE_ADDR172.22.32.131/v/005REMOTE_PORT50973/v/fSERVER_ADDR10.94.98.116/v/004SERVER_PORT8085/v/000SERVER_NAME/017/003REDIRECT_STATUS200/t/021HTTP_HOST10.94.98.116:8085/017/nHTTP_CONNECTIONkeep-alive/017xHTTP_USER_AGENTMozilla/5.0 (Macintosh; Intel Mac OS X 10_11_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/62.0.3202.94 Safari/537.36/036/001HTTP_UPGRADE_INSECURE_REQUESTS1/vUHTTP_ACCEPTtext/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8/024/rHTTP_ACCEPT_ENCODINGgzip, deflate/024/027HTTP_ACCEPT_LANGUAGEzh-CN,zh;q=0.9,en;q=0.8
根据上一节我们讲到的名-值对的长度规则,我们可以看出,Fastcgi协议中封装了类似于http协议里面的键值对。读取完毕后,继续跟踪消息,打印可以得出,得到的消息如图所示。
其中type对应的是FCGI_PARAMS,requestid为1,长度为0,此时完成了FastCGI协议消息的读取过程。下面说一下处理完请求后返回给nginx的FastCGI协议的消息。
6. 响应协议在fcgi_finish_request中调用fcgi_flush,fcgi_flush中封装一个FCGI_END_REQUEST消息体,再通过safe_write写入 socket 连接的客户端描述符。
int fcgi_flush(fcgi_request *req, int close) int len;
独孤九贱(5)_ThinkPHP5视频教程ThinkPHP是国内最流行的中文PHP开发框架,也是您Web项目的最佳选择。《VeVb.com独孤九贱(5)-ThinkPHP5视频教程》课程以ThinkPHP5最新版本为例,从最基本的框架常识开始,将...
Peter-Zhu 2017-05-16 12:03:57
独孤九贱(4)_PHP视频教程江湖传言:PHP是世界上最好的编程语言。真的是这样吗?这个梗究竟是从哪来的?学会本课程,你就会明白了。PHP 出品的PHP入门系统教学视频,完全从初学者的角度出发,绝不玩虚的,一切以实用、有用...
Peter-Zhu 2017-03-20 22:47:17
独孤九贱(1)_HTML5视频教程《VeVb.com原创html5视频教程》课程特色:php 原创幽默段子系列课程,以恶搞,段子为主题风格的php视频教程!轻松的教学风格,简短的教学模式,让同学们在不知不觉中,学会了HTML知识。...
Peter-Zhu 2017-03-13 10:15:11
ThinkPHP5实战之[教学管理系统]本套教程,以一个真实的学校教学管理系统为案例,手把手教会您如何在一张白纸上,从零开始,一步一步的用ThinkPHP5框架快速开发出一个商业项目。
Peter-Zhu 2017-07-24 16:48:56
PHP入门视频教程之一周学会PHP所有计算机语言的学习都要从基础开始,《PHP入门视频教程之一周学会PHP》不仅是PHP的基础部分更主要的是PHP语言的核心技术,是学习PHP必须掌握的内容,任何PHP项目的实现都离不开这部分的内容,通...
PHP教程郑重声明:本文版权归原作者所有,转载文章仅为传播更多信息之目的,如作者信息标记有误,请第一时间联系我们修改或删除,多谢。
新闻热点
疑难解答