In the previous article, we explained how nginx reads the data of the request line and parses the request line. In this article, we will mainly explain how nginx reads the request header data sent by the client and parses this data. Essentially, the data reading process of the request line and the request header is basically the same, because both face the problem of how to read data from intermittent data streams and how to process the data. 1. Request header reading main process Before introducing the request header reading process, we first show an example of an http request message: POST /web/book/read HTTP/1.1 Host: localhost Connection: keep-alive Content-Length: 365 Accept: application/json, text/plain, */* The first line of data in the example is the request line, and the following lines are request headers. Each request header is assembled in the name: value format, and each request header occupies one line. In the previous article introducing the request line reading process, we mentioned that once the request line is read, nginx will modify the callback function of the current read event to the ngx_http_process_request_headers() method and directly call this method to try to read the request header data. This method is the main process of reading the request line data. The following is the source code of this method: /** * Parse the header data sent by the client*/ 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) { // If there is no remaining space in the current header buffer, apply for new space if (r->header_in->pos == r->header_in->end) { // Apply for new space rv = ngx_http_alloc_large_header_buffer(r, 0); if (rv == NGX_ERROR) { ngx_http_close_request(r, NGX_HTTP_INTERNAL_SERVER_ERROR); return; } // The header sent by the client is too long and exceeds the maximum size specified by 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; } } // Try to read the newly sent data from the connected client 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); // Here we mainly convert the read data rc = ngx_http_parse_header_line(r, r->header_in, cscf->underscores_in_headers); // NGX_OK indicates successful parsing and obtaining a header data if (rc == NGX_OK) { r->request_length += r->header_in->pos - r->header_name_start; // Filter invalid headers if (r->invalid_header && cscf->ignore_invalid_headers) { continue; } // Create a structure to store headers 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; // Use the header name as the key of the hash table 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'; // Use the header value as the hash table 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 stores all headers. Here we check whether the header sent by the current client is a valid header. hh = ngx_hash_find(&cmcf->headers_in_hash, h->hash, h->lowcase_key, h->key.len); // The handler here is the processing method defined for each header in ngx_http_headers_in. After being processed by the // handler() method of each header, the header sent by the client is converted to the various attributes in the r->headers_in structure if (hh && hh->handler(r, h, hh->offset) != NGX_OK) { return; } continue; } // NGX_HTTP_PARSE_HEADER_DONE means that all headers have been processed 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; // Check the legitimacy of the header data sent by the client rc = ngx_http_process_request_header(r); if (rc != NGX_OK) { return; } ngx_http_process_request(r); return; } // NGX_AGAIN indicates that the read header row data is incomplete and needs to be read further 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; } } The reading of the request header here is mainly divided into the following steps:
2. Reading request header data As you can see, there are two main methods for reading the request header: ngx_http_read_request_header() and ngx_http_parse_header_line(). The second method here is relatively long, but its logic is very simple. It mainly parses the read data to see if it can form a complete request header (in the form of name: value and takes up one line). If so, it returns NGX_OK, otherwise it returns NGX_AGAIN in the hope of continuing to read data. We will not explain this method here. Readers can read the source code by themselves. We will mainly explain how the ngx_http_read_request_header() method reads the request header data sent by the client: 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; // Calculate how much data is not processed yet n = r->header_in->last - r->header_in->pos; // If n is greater than 0, it means that there is still unprocessed data, then return n directly if (n > 0) { return n; } // Going here means that the currently read data has been processed, so a judgment will be made here. If the ready parameter of the current event is 1, // This means that the handle of the current connection stores unread data, so the c->recv() method is called to read the data. Otherwise, the current event is added to the // event queue, and the read event of the current connection handle is continued to be monitored if (rev->ready) { // Read data on the connection file descriptor n = c->recv(c, r->header_in->last, r->header_in->end - r->header_in->last); } else { n = NGX_AGAIN; } // If n is NGX_AGAIN, add the current event to the event listener and continue to listen to the read event of the current epoll handle 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; } // If n is 0, it means the client has closed the connection if (n == 0) { ngx_log_error(NGX_LOG_INFO, c->log, 0, "client prematurely closed connection"); } // If the client closes the connection or reads abnormally, recycle the current request structure 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; } // Update the currently read data pointer r->header_in->last += n; return n; } The reading of request header data here is mainly divided into the following steps:
3. Summary This article mainly explains how nginx reads and parses the request header, and focuses on the main process code for reading data and the detailed steps of reading. This is the end of this article about the detailed explanation of the nginx request header data reading process. For more relevant nginx request header data reading content, please search for previous articles on 123WORDPRESS.COM or continue to browse the following related articles. I hope everyone will support 123WORDPRESS.COM in the future! You may also be interested in:
|
<<: React Class component life cycle and execution order
>>: Detailed tutorial for installing winx64 under mysql8.0.18 (with pictures and text)
Table of contents this Method In the object Hidde...
The scroll-view of WeChat applet has more bugs wh...
Open the connection tool. I use MobaXterm_Persona...
The server reports an error 502 when synchronizin...
Problem: The partition where MySQL stores data fi...
This article example shares the specific code of ...
To replace a string, we need to use the following...
When you see a class, what information do you wan...
Docker tag detailed explanation The use of the do...
To remove the underline of a hyperlink, you need t...
Table of contents 1. Comments on MySQL primary ke...
Ubuntu is a free and open source desktop PC opera...
If you often use FTP server in your study or work...
illustrate In front-end development, you often en...
Assuming business: View the salary information of...