HTTP过滤模块

参考资料<深入理解Nginx>

 

HTTP过滤模块也是一种HTTP模块,与普通HTTP处理模块不同在于:

1.一个请求仅由一个HTTP处理模块处理,而可以被任意个HTTP过滤模块处理

2.普通的HTTP模块倾向于完成请求的核心功能,而HTTP过滤模块所做的工作是对发送给用户的HTTP响应包做一些加工

 


 

HTTP过滤模块的简单例子

该过滤模块实现的功能是:用户的请求由static静态文件模块进行处理,根据URI返回磁盘中的文件给用户,然后该过滤模块就会在返回给用户的相应包体前添加一段字符串:"[my filter prefix]"

static静态文件模块处理完成后会调用ngx_http_send_header跟ngx_http_output_filter来调用过滤模块

1.编写config文件

  跟普通HTTP模块不同的是HTTP_MODULES变量要改为HTTP_FILTER_MODULES

ngx_addon_name=ngx_http_myfilter_module
HTTP_FILTER_MODULES="$HTTP_FILTER_MODULES ngx_http_myfilter_module"
NGX_ADDON_SRCS="$NGX_ADDON_SRCS $ngx_addon_dir/ngx_http_myfilter_module.c"

2.配置项与上下文

  该过滤模块希望在nginx.conf中由一个控制当前HTTP过滤模块是否生效的配置项,它的参数值是on或者off。首先建立如下结构来存储配置项

typedef struct {
    ngx_flag_t enable;
} ngx_http_myfilter_conf_t;

下面实现的ngx_http_myfilter_create_conf用于为存储配置项的结构体分配内存

static void *ngx_http_myfilter_create_conf(ngx_conf_t *cf)
{
    ngx_http_myfilter_conf_t *mycf;
    mycf=(ngx_http_myfilter_conf_t *)ngx_pcalloc(cf->pool,sizeof(ngx_http_myfilter_conf_t));
    if(mycf==NULL){
        return NULL;
    }
    mycf->enable=NGX_CONF_UNSET;
    return mycf;
}

因为Nginx的异步处理,一个HTTP包体处理方法在1个请求中可能被多次调用,而实际上我们只希望在包头加1次前缀。因此再建立一个HTTP上下文结构体来处理HTTP包体时是否添加前缀

typedef struct {
    ngx_int_t add_prefix;
} ngx_http_myfilter_ctx_t;

3.定义HTTP过滤模块,与普通的HTTP模块类似,不多解释。

定义ngx_http_myfilter_commands数组,出现配置项是使用Nginx预设的方法ngx_conf_set_flag_slot来处理

static ngx_command_t ngx_http_myfilter_commands[]={
    {
        //配置项名称
        ngx_string("add_prefix"),
        //配置项类型(可出现的位置,参数的个数)
        NGX_HTTP_LOC_CONF|NGX_HTTP_LMT_CONF|NGX_CONF_FLAG,
        //出现了name中指定的配置项后,将会调用该方法处理配置项的参数
        ngx_conf_set_flag_slot,
        NGX_HTTP_LOC_CONF_OFFSET,
        offsetof(ngx_http_myfilter_conf_t,enable),
        NULL
    },
    ngx_null_command
};

定义ngx_http_module_t,ngx_http_myfilter_init方法用于初始化HTTP过滤模块,下面有它的实现代码

static ngx_http_module_t ngx_http_myfilter_module_ctx={
    NULL,
    ngx_http_myfilter_init,
    NULL,
    NULL,
    NULL,
    NULL,
    ngx_http_myfilter_create_conf,
    NULL
};

定义ngx_module_t

ngx_module_t ngx_http_myfilter_module={
    NGX_MODULE_V1,
    //指向ngx_http_module_t结构体
    &ngx_http_myfilter_module_ctx,
    //用来处理nginx.conf中的配置项
    ngx_http_myfilter_commands,
    //表示该模块的类型
    NGX_HTTP_MODULE,
    NULL,
    NULL,
    NULL,
    NULL,
    NULL,
    NULL,
    NULL,
    NGX_MODULE_V1_PADDING
};

4.初始化HTTP过滤模块

一个请求可以由多个过滤模块处理,所以过滤模块是有调用顺序的,该方法用于构建过滤链表

//用于初始化HTTP过滤模块
static ngx_http_output_header_filter_pt ngx_http_next_header_filter;
static ngx_http_output_body_filter_pt ngx_http_next_body_filter;
static ngx_int_t ngx_http_myfilter_init(ngx_conf_t *cf)
{
    //插入到头部处理方法链表的首部
    ngx_http_next_header_filter=ngx_http_top_header_filter;
    ngx_http_top_header_filter=ngx_http_myfilter_header_filter;
    
    //插入包体处理方法链表的首部
    ngx_http_next_body_filter=ngx_http_top_body_filter;
    ngx_http_top_body_filter=ngx_http_myfilter_body_filter;
    
    return NGX_OK;
}

static ngx_str_t filter_prefix = ngx_string("[filter_prefix]");

5.处理请求的HTTP头部

filebeat 配置nginx模块到es多个索引 nginx filter 模块_配置项

filebeat 配置nginx模块到es多个索引 nginx filter 模块_字符串_02

1 static ngx_int_t
 2 ngx_http_myfilter_header_filter(ngx_http_request_t *r)
 3 {
 4     ngx_http_myfilter_ctx_t *ctx;
 5     ngx_http_myfilter_conf_t *conf;
 6     //如果返回失败,则直接交由下一个过滤模块处理
 7     if(r->headers_out.status!=NGX_HTTP_OK){
 8         return ngx_http_next_header_filter(r);
 9     }
10     //获取HTTP上下文
11     ctx=ngx_http_get_module_ctx(r,ngx_http_myfilter_module);
12     if(ctx){
13         //如果请求的上下文已经存在,说明ngx_http_myfilter_header_filter已经被调用过1次
14         return ngx_http_next_header_filter(r);
15     }
16     //获取存储配置项的ngx_http_myfilter_conf结构体
17     conf=ngx_http_get_module_loc_conf(r,ngx_http_myfilter_module);
18     //如果enable成员为0,则直接交给下一个
19     if(conf->enable==0){
20         return ngx_http_next_header_filter(r);
21     }
22     //构造上下文结构体ngx_http_myfilter_ctx_t
23     ctx=ngx_pcalloc(r->pool,sizeof(ngx_http_myfilter_ctx_t));
24     if(ctx==NULL){
25         return NGX_ERROR;
26     }
27     //add_prefix为0表示不加前缀
28     ctx->add_prefix=0;
29     //将构造的上下文设置到当前请求
30     ngx_http_set_ctx(r,ctx,ngx_http_myfilter_module);
31     //myfilter过滤模块只处理Content-Type是"text-plain"类型的HTTP响应
32     if(r->headers_out.content_type.len>=sizeof("text/plain")-1
33        && ngx_strncasecmp(r->headers_out.content_type.data,(u_char *)"text/plain",sizeof("text/plain")-1)==0)
34     {
35         //设置为1表示需要再HTTP包体前加入前缀
36         ctx->add_prefix=1;
37         //把前缀字符串的长度加入Content-length中
38         if(r->headers_out.content_length_n>0){
39             r->headers_out.content_length_n+=filter_prefix.len;
40         }
41     }
42     //交由下一个过滤模块继续处理
43     return ngx_http_next_header_filter(r);
44 }

View Code

6.处理请求的HTTP包体

filebeat 配置nginx模块到es多个索引 nginx filter 模块_配置项

filebeat 配置nginx模块到es多个索引 nginx filter 模块_字符串_02

1 //处理请求中的HTTP包体
 2 static ngx_int_t
 3 ngx_http_myfilter_body_filter(ngx_http_request_t *r,ngx_chain_t *in)
 4 {
 5     ngx_http_myfilter_ctx_t *ctx;
 6     ctx=ngx_http_get_module_ctx(r,ngx_http_myfilter_module);
 7     //如果获取不到上下文,或者上下文结构体中add_prefix为0或2时,都不会添加前缀
 8     if(ctx==NULL||ctx->add_prefix!=1){
 9         return ngx_http_next_body_filter(r,in);
10     }
11     //将add_prefix设置为2,防止重复添加前缀
12     ctx->add_prefix=2;
13     //分配内存用于存储字符串前缀
14     ngx_buf_t *b=ngx_create_temp_buf(r->pool,filter_prefix.len);
15     //将ngx_buf_t中的指针指向filter_prefix字符串
16     b->start=b->pos=filter_prefix.data;
17     b->last=b->pos+filter_prefix.len;
18     //生成要发送的ngx_chain_t链表,加上刚分配的ngx_buf_t成员
19     ngx_chain_t *cl=ngx_alloc_chain_link(r->pool);
20     cl->buf=b;
21     cl->next=in;
22     //调用下一个过滤模块
23     return ngx_http_next_body_filter(r,cl);
24 }

View Code

 7.配置nginx.conf文件

 添加如下配置,在nginx安装目录下添加txt/test.txt文件,访问localhost/test.txt。

location / {
    root txt;
    add_prefix on;
}