HTTP的处理请求流程我们从ngx_http_init_connection开始论述
里面注册了一个处理函数
rev->handler = ngx_http_wait_request_handler;
ngx_http_wait_request_handler的参数是ngx_event_t rev,一旦有请求到达,数据已经被复制到rev->data中,这时会调用ngx_http_process_request_line来处理请求数据
但这里存在一个keepalive的问题,如果浏览器端设置了keepalive头部,那么r->keepalive标志为真,这样ngx在调用ngx_http_finalize_connection的时候,会调用ngx_http_set_keepalive设置keepalive的回调ngx_http_keepalive_handler,并保留当前的ngx_connection_t,然后直接返回。这样,在超时时间内,当同一个url请求再次到来的时候,i/o框架调用的是ngx_http_keepalive_handler而不是ngx_http_wait_request_handler。
还要补充的是,在http2中,因为socket默认是keepalive状态,而且多流并发,ngx_http_finalize_connection关闭的只是这个连接上的stream,因此会直接调用ngx_http_close_request然后返回:
#if (NGX_HTTP_V2)
if (r->stream) {
ngx_http_close_request(r, 0);
return;
}
#endif
而在ngx_http_close_request中,也不会关闭连接,而是关闭当前流:
#if (NGX_HTTP_V2)
if (r->stream) {
ngx_http_v2_close_stream(r->stream, rc);
return;
}
#endif
ngx_http_process_request_line的主要代码如下:
rc = NGX_AGAIN;
for ( ;; ) {
if (rc == NGX_AGAIN) {
n = ngx_http_read_request_header(r);
if (n == NGX_AGAIN || n == NGX_ERROR) {
return;
}
}
rc = ngx_http_parse_request_line(r, r->header_in);
if (rc == NGX_OK) {
if (ngx_http_process_request_uri(r) != NGX_OK) {
return;
}
if (r->host_start && r-&g