本文我們則主要講解nginx是如何讀取客戶(hù)端發(fā)送來(lái)的請(qǐng)求頭的數(shù)據(jù),并且解析這些數(shù)據(jù)的。本質(zhì)上來(lái)講,請(qǐng)求行的數(shù)據(jù)和請(qǐng)求頭的數(shù)據(jù)讀取流程是基本一致的,因?yàn)槠涠济媾R著如何從間斷的數(shù)據(jù)流中讀取到數(shù)據(jù),也面臨著如何對(duì)數(shù)據(jù)進(jìn)行處理的問(wèn)題。 1. 請(qǐng)求頭讀取主流程 在介紹請(qǐng)求頭的讀取流程之前,我們首先展示一個(gè)http請(qǐng)求報(bào)文的示例: POST /web/book/read HTTP/1.1 Host: localhost Connection: keep-alive Content-Length: 365 Accept: application/json, text/plain, */* 示例中的第一行數(shù)據(jù)就是請(qǐng)求行,而后面的幾行都是請(qǐng)求頭。每一個(gè)請(qǐng)求頭都是以name: value的格式組裝的,并且每一個(gè)請(qǐng)求頭都占用一行。 在上一篇介紹請(qǐng)求行讀取流程的文章中,我們講到,一旦請(qǐng)求行讀取完成,nginx就會(huì)將當(dāng)前讀取事件的回調(diào)函數(shù)修改為ngx_http_process_request_headers()方法,并且直接調(diào)用該方法嘗試讀取請(qǐng)求頭數(shù)據(jù)。這個(gè)方法就是讀取請(qǐng)求行數(shù)據(jù)的主流程,如下是該方法的源碼: /** * 解析客戶(hù)端發(fā)送來(lái)的header數(shù)據(jù) */ static void ngx_http_process_request_headers(ngx_event_t *rev) { u_char *p; size_t len; ssize_t n; ngx_int_t rc, rv; ngx_table_elt_t *h; ngx_connection_t *c; ngx_http_header_t *hh; ngx_http_request_t *r; ngx_http_core_srv_conf_t *cscf; ngx_http_core_main_conf_t *cmcf; c = rev->data; r = c->data; if (rev->timedout) { ngx_log_error(NGX_LOG_INFO, c->log, NGX_ETIMEDOUT, "client timed out"); c->timedout = 1; ngx_http_close_request(r, NGX_HTTP_REQUEST_TIME_OUT); return; } cmcf = ngx_http_get_module_main_conf(r, ngx_http_core_module); rc = NGX_AGAIN; for (;;) { if (rc == NGX_AGAIN) { // 如果當(dāng)前header緩沖區(qū)中沒(méi)有剩余空間,則申請(qǐng)新的空間 if (r->header_in->pos == r->header_in->end) { // 申請(qǐng)新的空間 rv = ngx_http_alloc_large_header_buffer(r, 0); if (rv == NGX_ERROR) { ngx_http_close_request(r, NGX_HTTP_INTERNAL_SERVER_ERROR); return; } // 客戶(hù)端發(fā)送的header太長(zhǎng),超出了large_client_header_buffers指定的最大大小 if (rv == NGX_DECLINED) { p = r->header_name_start; r->lingering_close = 1; if (p == NULL) { ngx_log_error(NGX_LOG_INFO, c->log, 0, "client sent too large request"); ngx_http_finalize_request(r, NGX_HTTP_REQUEST_HEADER_TOO_LARGE); return; } len = r->header_in->end - p; if (len > NGX_MAX_ERROR_STR - 300) { len = NGX_MAX_ERROR_STR - 300; } ngx_http_finalize_request(r, NGX_HTTP_REQUEST_HEADER_TOO_LARGE); return; } } // 嘗試讀取連接上客戶(hù)端新發(fā)送來(lái)的數(shù)據(jù) n = ngx_http_read_request_header(r); if (n == NGX_AGAIN || n == NGX_ERROR) { return; } } cscf = ngx_http_get_module_srv_conf(r, ngx_http_core_module); // 這里主要是對(duì)讀取到的數(shù)據(jù)進(jìn)行轉(zhuǎn)換 rc = ngx_http_parse_header_line(r, r->header_in, cscf->underscores_in_headers); // NGX_OK表示成功解析得到了一個(gè)header數(shù)據(jù) if (rc == NGX_OK) { r->request_length += r->header_in->pos - r->header_name_start; // 過(guò)濾無(wú)效的header if (r->invalid_header && cscf->ignore_invalid_headers) { continue; } // 創(chuàng)建一個(gè)存儲(chǔ)header的結(jié)構(gòu)體 h = ngx_list_push(&r->headers_in.headers); if (h == NULL) { ngx_http_close_request(r, NGX_HTTP_INTERNAL_SERVER_ERROR); return; } h->hash = r->header_hash; // 把header的name作為hash表的key h->key.len = r->header_name_end - r->header_name_start; h->key.data = r->header_name_start; h->key.data[h->key.len] = '\0'; // 把header的value作為hash表的value h->value.len = r->header_end - r->header_start; h->value.data = r->header_start; h->value.data[h->value.len] = '\0'; h->lowcase_key = ngx_pnalloc(r->pool, h->key.len); if (h->lowcase_key == NULL) { ngx_http_close_request(r, NGX_HTTP_INTERNAL_SERVER_ERROR); return; } if (h->key.len == r->lowcase_index) { ngx_memcpy(h->lowcase_key, r->lowcase_header, h->key.len); } else { ngx_strlow(h->lowcase_key, h->key.data, h->key.len); } // headers_in_hash中存儲(chǔ)了所有的header,這里是查找當(dāng)前客戶(hù)端傳的header是否為有效的header hh = ngx_hash_find(&cmcf->headers_in_hash, h->hash, h->lowcase_key, h->key.len); // 這里的handler是在ngx_http_headers_in中為每一個(gè)header定義的處理方法,經(jīng)過(guò)各個(gè)header的 // handler()方法處理后,客戶(hù)端傳來(lái)的header就都轉(zhuǎn)換到r->headers_in結(jié)構(gòu)體中的各個(gè)屬性中了 if (hh && hh->handler(r, h, hh->offset) != NGX_OK) { return; } continue; } // NGX_HTTP_PARSE_HEADER_DONE表示已經(jīng)將所有的header都處理完成了 if (rc == NGX_HTTP_PARSE_HEADER_DONE) { r->request_length += r->header_in->pos - r->header_name_start; r->http_state = NGX_HTTP_PROCESS_REQUEST_STATE; // 檢查客戶(hù)端發(fā)送來(lái)的header數(shù)據(jù)的合法性 rc = ngx_http_process_request_header(r); if (rc != NGX_OK) { return; } ngx_http_process_request(r); return; } // NGX_AGAIN表示讀取到的header行數(shù)據(jù)不完全,還需要繼續(xù)讀取 if (rc == NGX_AGAIN) { continue; } ngx_log_error(NGX_LOG_INFO, c->log, 0, "client sent invalid header line"); ngx_http_finalize_request(r, NGX_HTTP_BAD_REQUEST); return; } } 這里請(qǐng)求頭的讀取主要分為如下幾個(gè)步驟:
2. 請(qǐng)求頭數(shù)據(jù)的讀取 可以看到,對(duì)請(qǐng)求頭的讀取主要有兩個(gè)方法:ngx_http_read_request_header()和ngx_http_parse_header_line()。這里的第二個(gè)方法比較長(zhǎng),但是其邏輯非常的簡(jiǎn)單,主要就是解析讀取到的數(shù)據(jù)是否能組成一個(gè)完整的請(qǐng)求頭(name: value的形式,并且占用一行),如果是,則返回NGX_OK,否則返回NGX_AGAIN以期待繼續(xù)讀取數(shù)據(jù)。對(duì)于這個(gè)方法,我們這里不進(jìn)行講解,讀者可自行閱讀源碼,我們主要講解ngx_http_read_request_header()方法是如何讀取客戶(hù)端發(fā)送來(lái)的請(qǐng)求頭數(shù)據(jù)的: static ssize_t ngx_http_read_request_header(ngx_http_request_t *r) { ssize_t n; ngx_event_t *rev; ngx_connection_t *c; ngx_http_core_srv_conf_t *cscf; c = r->connection; rev = c->read; // 計(jì)算當(dāng)前還有多少數(shù)據(jù)未處理 n = r->header_in->last - r->header_in->pos; // 如果n大于0,說(shuō)明還有讀取到的數(shù)據(jù)未處理,則直接返回n if (n > 0) { return n; } // 走到這里,說(shuō)明當(dāng)前讀取到的數(shù)據(jù)都已經(jīng)處理完了,因而這里會(huì)進(jìn)行判斷,如果當(dāng)前事件的ready參數(shù)為1, // 則表示當(dāng)前連接的句柄上存儲(chǔ)還未讀取的數(shù)據(jù),因而調(diào)用c->recv()方法讀取數(shù)據(jù),否則繼續(xù)將當(dāng)前事件添加到 // 事件隊(duì)列中,并且繼續(xù)監(jiān)聽(tīng)當(dāng)前連接句柄的讀事件 if (rev->ready) { // 在連接文件描述符上讀取數(shù)據(jù) n = c->recv(c, r->header_in->last, r->header_in->end - r->header_in->last); } else { n = NGX_AGAIN; } // 如果n為NGX_AGAIN,則將當(dāng)前事件添加到事件監(jiān)聽(tīng)器中,并且繼續(xù)監(jiān)聽(tīng)當(dāng)前epoll句柄的讀事件 if (n == NGX_AGAIN) { if (!rev->timer_set) { cscf = ngx_http_get_module_srv_conf(r, ngx_http_core_module); ngx_add_timer(rev, cscf->client_header_timeout); } if (ngx_handle_read_event(rev, 0) != NGX_OK) { ngx_http_close_request(r, NGX_HTTP_INTERNAL_SERVER_ERROR); return NGX_ERROR; } return NGX_AGAIN; } // 如果n為0,說(shuō)明客戶(hù)端關(guān)閉了連接 if (n == 0) { ngx_log_error(NGX_LOG_INFO, c->log, 0, "client prematurely closed connection"); } // 如果客戶(hù)端關(guān)閉了連接或者讀取異常,則回收當(dāng)前的request結(jié)構(gòu)體 if (n == 0 || n == NGX_ERROR) { c->error = 1; c->log->action = "reading client request headers"; ngx_http_finalize_request(r, NGX_HTTP_BAD_REQUEST); return NGX_ERROR; } // 更新當(dāng)前讀取到的數(shù)據(jù)指針 r->header_in->last += n; return n; } 這里請(qǐng)求頭數(shù)據(jù)的讀取主要分為如下幾個(gè)步驟:
|
免責(zé)聲明:本站部分文章和圖片均來(lái)自用戶(hù)投稿和網(wǎng)絡(luò)收集,旨在傳播知識(shí),文章和圖片版權(quán)歸原作者及原出處所有,僅供學(xué)習(xí)與參考,請(qǐng)勿用于商業(yè)用途,如果損害了您的權(quán)利,請(qǐng)聯(lián)系我們及時(shí)修正或刪除。謝謝!
始終以前瞻性的眼光聚焦站長(zhǎng)、創(chuàng)業(yè)、互聯(lián)網(wǎng)等領(lǐng)域,為您提供最新最全的互聯(lián)網(wǎng)資訊,幫助站長(zhǎng)轉(zhuǎn)型升級(jí),為互聯(lián)網(wǎng)創(chuàng)業(yè)者提供更加優(yōu)質(zhì)的創(chuàng)業(yè)信息和品牌營(yíng)銷(xiāo)服務(wù),與站長(zhǎng)一起進(jìn)步!讓互聯(lián)網(wǎng)創(chuàng)業(yè)者不再孤獨(dú)!
掃一掃,關(guān)注站長(zhǎng)網(wǎng)微信
當(dāng)我們?cè)诠蚕砭W(wǎng)絡(luò)訪(fǎng)問(wèn)的時(shí)候,可能會(huì)遇到提示指定的網(wǎng)絡(luò)名不再可用的問(wèn)題,這可能是由于我們的共享網(wǎng)絡(luò)出現(xiàn)了錯(cuò)誤,也可能是被共享的對(duì)象所拒絕了。指定的網(wǎng)絡(luò)名 ......
1、首先進(jìn)入到“百度”軟件中, 2、然后在其中輸入“百度識(shí)圖”, 3、之后點(diǎn)擊圖中的“開(kāi)始使用”按鈕, 4、緊接著點(diǎn)擊右下角的“相冊(cè)”功能, 5、在相冊(cè)下 ......
文/曹楊 原標(biāo)題:誰(shuí)還看電視? 爸爸戴一副老花鏡,媽媽戴一副近視鏡,一人坐在沙發(fā),一人躺在床上,各自刷著自己關(guān)注的博主更新的短視頻。電視也許開(kāi)著,但只是背景。 這樣的畫(huà)面,幾乎成了洛奇家的常 ...
一、軟件沖突1、首先確認(rèn)是否是應(yīng)用程序沖突導(dǎo)致的。2、查看是否只有特定幾個(gè)游戲或應(yīng)用會(huì)導(dǎo)致該問(wèn)題。3、如果是應(yīng)用沖突,那么只要卸載這些app就可以解決了。二 ......
圖片來(lái)源于簡(jiǎn)書(shū) 文/郭開(kāi)森 楊帆 陸玖財(cái)經(jīng)準(zhǔn)備開(kāi)新欄目了,每周一創(chuàng)始人郭開(kāi)森和楊帆合體郭德帆,對(duì)行業(yè)進(jìn)行一些觀(guān)察和評(píng)論,第一篇我們?nèi)允谴蛩銓?xiě)社區(qū)團(tuán)購(gòu),這是當(dāng)下最火的話(huà)題。 來(lái)過(guò)陸玖財(cái)經(jīng)做客的朋友們...
電腦端:1、大家可以點(diǎn)擊右邊鏈接進(jìn)入網(wǎng)頁(yè)版的百度網(wǎng)盤(pán),進(jìn)入之后點(diǎn)擊“去登錄”。https://pan.baidu.com/2、之后正確的輸入賬號(hào)密碼進(jìn)行登錄就好啦。手機(jī)端:1 ......
一、N100對(duì)比intel i3 1、N100的跑分達(dá)到了147210分,這個(gè)數(shù)據(jù)可以達(dá)到i3的七代級(jí)別。 2、在跑分上也是超越了大部分的I3七代CPU,不過(guò)比I3八代要弱勢(shì)一些。 3 ......
8月15日消息 上周,有媒體報(bào)道前身為百度圖片的“榴蓮”APP含有大量不雅視頻內(nèi)容被用戶(hù)舉報(bào)。對(duì)此,百度圖片官方進(jìn)行了回應(yīng),百度圖片表示已經(jīng)對(duì)報(bào)道中所涉及的“生吃旋風(fēng)哥”等爭(zhēng)議內(nèi)容進(jìn)行了下線(xiàn)處理。 此外,百度...
在填寫(xiě)一些項(xiàng)目申請(qǐng)書(shū)中,總是免不了要選擇一些數(shù)字,但是在方框中如何插入數(shù)字,該怎么辦呢?那么下面就由學(xué)習(xí)啦小編給大家分享下word在方框里輸入數(shù)字的技巧, ......
WPS Office手機(jī)版怎么加橫線(xiàn)?很多用戶(hù)還不知道WPS Office手機(jī)版怎么加橫線(xiàn),WPS Office手機(jī)版怎么加橫線(xiàn),WPS Office手機(jī)版怎么打橫線(xiàn),WPS Office手機(jī)版怎么弄 ......
迅雷前綴是什么 答:迅雷前綴是(magnet:?xt=urn:btih:)括號(hào)里的就是了。 我們只要在這段文字之后輸入后續(xù)的內(nèi)容,就可以創(chuàng)建下載鏈接了。 1、磁力鏈接不基于文 ......
一、內(nèi)容特權(quán)。 1、半價(jià)點(diǎn)播。 許多站內(nèi)視頻都需要付費(fèi)觀(guān)看,而大會(huì)員用戶(hù)可以直接半價(jià)享受; 購(gòu)買(mǎi)成功后的48h內(nèi)無(wú)限次觀(guān)看。有部分的內(nèi)容是只限在中國(guó)大陸內(nèi)觀(guān) ......
1、首先打開(kāi)小米運(yùn)動(dòng)的“實(shí)驗(yàn)室功能”。 2、接著點(diǎn)擊“門(mén)卡模擬”。 3、然后點(diǎn)擊“我知道了”。 4、最后貼近就可以刷卡成功了。...
1、打開(kāi)手機(jī)輕顏相機(jī)app,點(diǎn)擊“我的”,點(diǎn)擊“設(shè)置”,2、點(diǎn)擊“幫助與反饋”,3、點(diǎn)擊右下角“在線(xiàn)咨詢(xún)”即可聯(lián)系客服,詢(xún)問(wèn)自己的問(wèn)題啦!...
答:華為P系列: 華為p40,華為p40plus,華為p50,華為p50e,華為p60 華為mate系列: 華為mate40,華為mate50,華為mate50e,華為mate60 華為nova系列: 華為n ......
近期有用戶(hù)反映,電腦在更新Windows 11 Insider Preview 25252.1000后,出現(xiàn)了應(yīng)用和已壓縮的文件點(diǎn)擊毫無(wú)反應(yīng),拖拽都不行,只能從開(kāi)始菜單打開(kāi)的情況,這是怎 ......
可見(jiàn)單元格就是不包括隱藏或者篩選篩選后隱藏起來(lái)的單元格區(qū)域。方法:篩選或隱藏?cái)?shù)據(jù),復(fù)制需要粘貼的值,在目標(biāo)單元格區(qū)域左上角的第一個(gè)單元格處右擊,選擇【 ......
答:驍龍8+更好。 驍龍7+gen2實(shí)際上就是驍龍8+的低配版本。 在一些其他的核心架構(gòu)方面都是保持一致的,比如說(shuō)CPU的架構(gòu)、GPU的架構(gòu)等等。 驍龍7+和驍龍8+具體 ......
文/黎明 一場(chǎng)針對(duì)中國(guó)互聯(lián)網(wǎng)巨頭的反壟斷風(fēng)暴正在醞釀,而且這次動(dòng)真格了。 11月10日,國(guó)家市場(chǎng)監(jiān)管總局發(fā)布《關(guān)于平臺(tái)經(jīng)濟(jì)領(lǐng)域的反壟斷指南(征求意見(jiàn)稿)》,要加大對(duì)互聯(lián)網(wǎng)巨頭涉嫌壟斷的調(diào)查和監(jiān)管。 ...
文件被win10系統(tǒng)誤報(bào)病毒自動(dòng)刪除了如何進(jìn)行恢復(fù)?有用戶(hù)下載了某些破解軟件卻被Win10系統(tǒng)誤認(rèn)為是病毒文件而自動(dòng)刪除,當(dāng)然系統(tǒng)自帶殺毒軟件其實(shí)挺不錯(cuò)的,就是有時(shí)候會(huì)誤報(bào),大家遇到這種情況的時(shí)候就希望把誤刪的...
win11系統(tǒng)如何釋放掉系統(tǒng)默認(rèn)保留的存儲(chǔ)空間?一般情況下,Windows會(huì)保留一些存儲(chǔ)空間,以便設(shè)備獲得良好性能和成功更新。但是當(dāng)出現(xiàn)系統(tǒng)盤(pán)儲(chǔ)存空間不足時(shí),我們會(huì)將幾個(gè)G的保留空間釋放出來(lái),以解燃眉之急。本期教...
1、先打開(kāi)機(jī)頂盒進(jìn)入主界面,并且使用遙控器打開(kāi)設(shè)置。 2、然后選擇“賬號(hào)與安全”,并且進(jìn)入。 3、最后往下面翻就可以看到“ADB調(diào)試”的選項(xiàng),直接開(kāi)啟就行了 ......
答:在3DMark壓力測(cè)試當(dāng)中,顯卡需要超高97%才能夠算合格,證明顯卡的穩(wěn)定性是過(guò)關(guān)的。 1、一般的默認(rèn)情況下在2500~3000分就算很正常的了。 2、分?jǐn)?shù)越高說(shuō)明顯卡 ......
羅技g304dpi燈顏色代表什么:1、藍(lán)色:這種情況是正常工作的顯示,如果說(shuō)是常亮或者閃爍,那都沒(méi)有問(wèn)題這是在正常工作呢。2、紅色:如果說(shuō)是紅燈閃爍的話(huà)那就是 ......
我們經(jīng)常用WPS的時(shí)候,如果需要輸入波浪號(hào)~,會(huì)發(fā)現(xiàn)鍵盤(pán)上的波浪號(hào)輸入之后在最上面,但是為了美觀(guān),我們希望波浪號(hào)顯示在中間。這里總結(jié)了三個(gè)方法分享給大家 ......
win11系統(tǒng)快速跳過(guò)聯(lián)網(wǎng)創(chuàng)建本地管理賬戶(hù)3種方法?現(xiàn)在市面上銷(xiāo)售的品牌筆記本和臺(tái)式機(jī)基本上都預(yù)裝Windows11家庭中文版正版操作系統(tǒng),聯(lián)網(wǎng)后系統(tǒng)會(huì)自動(dòng)激活。當(dāng)用戶(hù)拿到新機(jī)器后還需要按照cortana(小娜)的提示一步...
答:中高端水平 i513500hx在處理器當(dāng)中是處于一個(gè)中高端的水平。 i513500hx是第十一代酷睿處理器系列的一員,基礎(chǔ)頻率為2.4GHz,表現(xiàn)十分的不錯(cuò)。 i513500hx介 ......
打開(kāi)軟件,直接填寫(xiě)就可以。 1、下載安裝easyconnect軟件, 2、打開(kāi)easyconnect應(yīng)用,在如圖所示的“服務(wù)器地址”一欄輸入如圖所示網(wǎng)址,點(diǎn)擊“連接”, 3、等 ......
相信有非常多使用過(guò)筆記本的用戶(hù)都聽(tīng)說(shuō)過(guò)獨(dú)顯直連這個(gè)詞,但很多用戶(hù)并不了解獨(dú)顯直連是什么,又有什么用處,那么下面就和小編一起來(lái)看看什么是獨(dú)顯直連和開(kāi)啟這 ......
win11系統(tǒng)開(kāi)機(jī)總是自動(dòng)登錄OneDrive如何關(guān)閉?win11系統(tǒng)開(kāi)機(jī)的時(shí)候,會(huì)自動(dòng)啟動(dòng)OneDrive,不想要啟動(dòng),該怎么操作呢?下面我們就來(lái)看看詳細(xì)的教程。 在OneDrive界面點(diǎn)小齒輪按鈕,下拉菜單中點(diǎn)【設(shè)置】。 單擊【...