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头部
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包体
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;
}