1. 框架简介

Nginx的HTTP框架是由core模块ngx_http_module 和 http模块ngx_http_core_module共同定义的。ngx_http_module定义了指令 http,保存和管理各个层次里所有http模块的配置数据(我们已经在第6章了解了它的上作原理);而ngx_http_core_module则是http模块的“核心”模块,它定义了listen、server、location等 http核心指令,搭建了Nginx的HTTP处理框架。

1.1 模块分类

http模块处理现今应用得最广泛的HTTP协议,也是目前Nginx里数量最多的模块,按照功能分类可以分为四种:

handler   直接处理客户端的请求,产生响应数据,是最常用的模块;
filter     对handler模块产生的数据做各种加工过滤处理;
upstream   实现反向代理功能,转发请求到上游服务器,从后端获取响应数据再发回给客户端;
load-balance 不直接处理数据,而是实现负载均衡算法,从upstream块的配置里选择一个合适的上游服务器。

1.2 处理流程
1.2.1 通用的处理流程

本节简要介绍Nginx的HTTP工作流程,这是理解Nginx HTTP框架的基础。
通用的处理流程:          
web 服务器的HTTP请求处理流程可以粗略地叙述为如下的顺序步骤:
(1)监听端口,接受客户端的连接;
(2)读取请求头,包括请求行(request line)和请求头(headers);
(3)读取或者丢弃请求体(body);
(4)生成并发送响应头;
(5)生成并发送响应体.

上面只是最简单的HTTP处理流程,Nginx使用了操作系统提供的事件驱动模型(epoll、kqueue等)来实
现异步并发,并且还增加了很多其他功能, 如权限检查、重定向、缓存、记录访问日志等,使HTTP处理
流程变得十分复杂.

1.2.2 Nginx的处理流程

如果我们忽略Nginx的异步机制,把注意力集中在处理逻辑上,那么可以看到Nginx
的主处理流程与基本的HTTP处理流程区别并不大,它的工作流程如下:
(1)监听端口,设置回调为ngx_http_init_connection();
(2)接受客户端的连接,调用ngx_http_wait_request_handler();
(3)调用ngx_http_create_request()创建请求对象;
(4)接收数据,调用ngx_http_process_request_line()解析请求行;
(5)请求行接收完毕,继续接收数据,调用ngx_http_process_request_headers()解析请求头;
(6)请求头接收完毕,调用ngx_http_process_request()设置异步读写事件. 
   注意:为了提高运行效率,Nginx不会主动处理请求体.
(7)调用ngx_http_handler()开始真正处理请求;
(8)调用ngx_http_core_run_phases()按阶段处理请求. 这是HTTP框架的核心部分,大部分http模块都在
   这里运行,最终产生响应数据. 
(9)调用ngx_http_send_header()发送响应头,从函数指针ngx_http_top_header_filter开始,
   通过header filter模块链过滤处理,最终发送处理过的响应头.
(10)调用ngx_http_output_filter()发送响应体,从函数指针ngx_http_top_body_filter开始,
   通过body filter模块链过滤处理,最终发送处理过的响应体.
(11)处理完毕,记录访问日志.

ngx_http_init_connection 

ngx_module_t  ngx_http_module = {
    NGX_MODULE_V1,
    &ngx_http_module_ctx,                  /* module context */
    ngx_http_commands,                     /* module directives */
    NGX_CORE_MODULE,                       /* module type */
    NULL,                                  /* init master */
    NULL,                                  /* init module */
    NULL,                                  /* init process */
    NULL,                                  /* init thread */
    NULL,                                  /* exit thread */
    NULL,                                  /* exit process */
    NULL,                                  /* exit master */
    NGX_MODULE_V1_PADDING
};


static ngx_command_t  ngx_http_commands[] = {

    { ngx_string("http"),
      NGX_MAIN_CONF|NGX_CONF_BLOCK|NGX_CONF_NOARGS,
      ngx_http_block,
      0,
      0,
      NULL },

      ngx_null_command
};

static char *ngx_http_block(ngx_conf_t *cf, ngx_command_t *cmd, void *conf)
{
    // ...
    if (ngx_http_optimize_servers(cf, cmcf, cmcf->ports) != NGX_OK) 
	{
        return NGX_CONF_ERROR;
    }
	// ...
}


static ngx_int_t 
ngx_http_optimize_servers(ngx_conf_t *cf, ngx_http_core_main_conf_t *cmcf,ngx_array_t *ports)
{
    // ...
	ngx_http_init_listening
	// ...
}


ngx_http_init_listening(ngx_conf_t *cf, ngx_http_conf_port_t *port)
{
  //...
  ls = ngx_http_add_listening(cf, &addr[i]);
  //...
}

ngx_http_add_listening()
{
  // ...
  ls->handler = ngx_http_init_connection;
  // ...
}

Logo

为开发者提供学习成长、分享交流、生态实践、资源工具等服务,帮助开发者快速成长。

更多推荐