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項目的最佳選擇?!禫eVb.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教程鄭重聲明:本文版權歸原作者所有,轉載文章僅為傳播更多信息之目的,如作者信息標記有誤,請第一時間聯系我們修改或刪除,多謝。
新聞熱點
疑難解答