nginx模块开发

开发方法參考淘宝的教程

这个模块的功能是向client发送一个文件,类似于网页上的另存为功能

#include <ngx_config.h>
#include <ngx_core.h>
#include <ngx_http.h>

static ngx_int_t ngx_http_file_init(ngx_conf_t *cf);
void* ngx_http_file_create_loc_conf(ngx_conf_t *cf);
static char* ngx_http_file_name(ngx_conf_t* cf,ngx_command_t* cmd,void* conf);

typedef struct
{
	ngx_str_t file_name;
}ngx_http_file_loc_conf_t;

static ngx_http_module_t ngx_http_file_module_ctx =
{
	NULL,
	ngx_http_file_init,				    /* postconfiguration */
	NULL,
	NULL,
	NULL,
	NULL,
	ngx_http_file_create_loc_conf,	    /*  create location configuration */
	NULL
};

static ngx_command_t ngx_http_file_commands[] = {
	{
		ngx_string("file_name"),
		NGX_HTTP_LOC_CONF|NGX_CONF_NOARGS|NGX_CONF_TAKE1,/* 接收0个或1个參数 */
		ngx_http_file_name,
		NGX_HTTP_LOC_CONF_OFFSET,						 /* 配置项的级别 */
		offsetof(ngx_http_file_loc_conf_t, file_name),	 /*file_name配置项在ngx_http_file_loc_conf_t中的偏移位置*/
		NULL,
	},
	ngx_null_command
};

/* 模块的定义 */
ngx_module_t ngx_http_file_module =
{
	NGX_MODULE_V1,
	&ngx_http_file_module_ctx,			/* 模块上下文,即一些回调函数 */
	ngx_http_file_commands,				/* 配置项解析 */
	NGX_HTTP_MODULE,
	NULL,
	NULL,
	NULL,
	NULL,
	NULL,
	NULL,
	NULL,
	NGX_MODULE_V1_PADDING
};

static ngx_int_t ngx_http_file_handler(ngx_http_request_t* r)
{
	ngx_http_core_loc_conf_t  *clcf = ngx_http_get_module_loc_conf(r,ngx_http_core_module);
	ngx_http_file_loc_conf_t* file_conf = ngx_http_get_module_loc_conf(r,ngx_http_file_module);//获得配置项结构体
	if(file_conf -> file_name.len == 0)
	{
		ngx_log_error(NGX_LOG_EMERG, r->connection->log, 0,"file_name is empty");
		return NGX_DECLINED;
	}

	ngx_open_file_info_t of;//文件信息结构体
	ngx_str_t path = file_conf->file_name;
	ngx_memzero(&of, sizeof(ngx_open_file_info_t));
	if (ngx_open_cached_file(clcf->open_file_cache, &path, &of, r->pool)!= NGX_OK)
	{
		ngx_log_error(NGX_LOG_ERR,r->connection->log,of.err, "%s \"%s\" failed", of.failed, path.data);
	}

	ngx_int_t rc = ngx_http_discard_request_body(r);//丢弃包体
	if(rc != NGX_OK)return rc;

	r -> headers_out.status =  NGX_HTTP_OK;
	r -> headers_out.content_length_n = of.size;
	r -> headers_out.last_modified_time = of.mtime;
	if(ngx_http_set_etag(r) != NGX_OK)return NGX_HTTP_INTERNAL_SERVER_ERROR;
	if(ngx_http_set_content_type(r) != NGX_OK)return NGX_HTTP_INTERNAL_SERVER_ERROR;

	r -> allow_ranges = 1;
	/* 先申请发送包体的缓冲区空间,申请成功后再发送包头 */
	ngx_buf_t* b = ngx_pcalloc(r -> pool,sizeof(ngx_buf_t));
	if(b == NULL)return NGX_HTTP_INTERNAL_SERVER_ERROR;
	b -> file = ngx_pcalloc(r -> pool,sizeof(ngx_file_t));
	if(b->file == NULL)return NGX_HTTP_INTERNAL_SERVER_ERROR;

	rc = ngx_http_send_header(r); /* 发送头部 */
	if (rc == NGX_ERROR || rc > NGX_OK || r->header_only) return rc;

	b -> file_pos = 0;
	b -> file_last = of.size;
	b -> in_file = b -> file_last ? 1 : 0;
	b -> last_buf = (r == r -> main) ? 1 : 0;
	b -> last_in_chain = 1;
	b -> file -> name = path;
	b -> file -> fd = of.fd;

	ngx_chain_t out;
	out.buf = b;
	out.next = NULL;

	return ngx_http_output_filter(r,&out); /*把产生的内容传递给兴许的filter去处理 */

}

/* 创建配置项结构体 */
void* ngx_http_file_create_loc_conf(ngx_conf_t *cf)
{
	ngx_http_file_loc_conf_t* local_conf = ngx_pcalloc(cf->pool,sizeof(ngx_http_file_loc_conf_t));
	if(local_conf == NULL)return NULL;
	ngx_str_null(&local_conf->file_name);
	return local_conf;
}

/* cf: 该參数里面保存从配置文件读取到的原始字符串以及相关的一些信息
 * cmd: 这个配置指令相应的ngx_command_t结构,即上面定义的数组中的第一个元素
 * conf: 就是定义的存储这个配置值的结构体,即上面的ngx_http_file_loc_conf_t*/
static char* ngx_http_file_name(ngx_conf_t* cf,ngx_command_t* cmd,void* conf)
{
	/* cf表示已经解析好的參数信息,将当中的数据取出来赋给自定义的配置项结构体中的相应成员 */
	char* rv = ngx_conf_set_str_slot(cf,cmd,conf);
	return rv;
}

static ngx_int_t ngx_http_file_init(ngx_conf_t *cf)
{
	ngx_http_core_main_conf_t* cmcf = ngx_http_conf_get_module_main_conf(cf,ngx_http_core_module);//获得核心配置文件
	ngx_http_handler_pt* h = ngx_array_push(&cmcf->phases[NGX_HTTP_CONTENT_PHASE].handlers);//将handler增加handlers链表
	if(h == NULL)return NGX_ERROR;
	*h = ngx_http_file_handler;//对该handler方法进行赋值
	return NGX_OK;
}

用法:

编译:还须要在file_module目录中加入config文件

./configure --prefix=/home/fangjian/study/code/nginx-1.4.4/nginx --add-module=/home/fangjian/study/code/nginx-1.4.4/fangjian_module/file_module

make

make install

然后在nginx.conf中加入       location /file {file_name   index.html; }启动nginx,在浏览器中输入127.0.0.1/file

时间: 2024-10-12 07:52:26

nginx模块开发的相关文章

Nginx模块开发入门(转)

前言 Nginx是当前最流行的HTTP Server之一,根据W3Techs的统计,目前世界排名(根据Alexa)前100万的网站中,Nginx的占有率为6.8%.与Apache相比,Nginx在高并发情况下具有巨大的性能优势. Nginx属于典型的微内核设计,其内核非常简洁和优雅,同时具有非常高的可扩展性.Nginx最初仅仅主要被用于做反向代理,后来随着HTTP核心的成熟和各种HTTP扩展模块的丰富,Nginx越来越多被用来取代Apache而单独承担HTTP Server的责任,例如目前淘宝内

linux下nginx模块开发入门

本文模块编写参考http://blog.codinglabs.org/articles/intro-of-nginx-module-development.html 之前讲了nginx的安装,算是对nginx有了最初步的了解,在配置完之后,我们就可以进行简单的nginx模块开发了. 下面本文展示一个简单的Nginx模块开发全过程,我们开发一个叫echo的handler模块,这个模块功能非常简单,它接收“echo”指令,指令可指定一个字符串参数,模块会输出这个字符串作为HTTP响应.例如,对ngi

Nginx模块开发入门

前言 Nginx是当前最流行的HTTP Server之一,根据W3Techs的统计,目前世界排名(根据Alexa)前100万的网站中,Nginx的占有率为6.8%.与Apache相比,Nginx在高并发情况下具有巨大的性能优势. Nginx属于典型的微内核设计,其内核非常简洁和优雅,同时具有非常高的可扩展性.Nginx最初仅仅主要被用于做反向代理,后来随着HTTP核心的成熟和各种HTTP扩展模块的丰富,Nginx越来越多被用来取代Apache而单独承担HTTP Server的责任,例如目前淘宝内

Nginx模块开发入门(转)

前言 Nginx是当前最流行的HTTP Server之一,根据W3Techs的统计,目前世界排名(根据Alexa)前100万的网站中,Nginx的占有率为6.8%.与Apache相比,Nginx在高并发情况下具有巨大的性能优势. Nginx属于典型的微内核设计,其内核非常简洁和优雅,同时具有非常高的可扩展性.Nginx最初仅仅主要被用于做反向代理,后来随着HTTP核心的成熟和各种HTTP扩展模块的丰富,Nginx越来越多被用来取代Apache而单独承担HTTP Server的责任,例如目前淘宝内

NGINX模块开发 之 验证URL参数

作者:邹祁峰 邮箱:[email protected] 博客:http://blog.csdn.net/qifengzou 日期:2014.05.26 16:45 转载请注明来自"祁峰"的CSDN博客 要求在浏览器地址栏中输入"localhost/login?user=qifeng&passwd=123456",并在浏览器上显示验证结果(Success 或 Failed).以下是在NGINX中添加一个LOGIN模块的整个处理过程. 1 修改配置 修改配置文件

[转] Nginx模块开发入门

前言 Nginx是当前最流行的HTTP Server之一,根据W3Techs的统计,目前世界排名(根据Alexa)前100万的网站中,Nginx的占有率为6.8%.与Apache相比,Nginx在高并发情况下具有巨大的性能优势. Nginx属于典型的微内核设计,其内核非常简洁和优雅,同时具有非常高的可扩展性.Nginx最初仅仅主要被用于做反向代理,后来随着HTTP核 心的成熟和各种HTTP扩展模块的丰富,Nginx越来越多被用来取代Apache而单独承担HTTP Server的责任,例如目前淘宝

FW: Nginx模块开发入门

前言 Nginx是当前最流行的HTTP Server之一,根据W3Techs的统计,目前世界排名(根据Alexa)前100万的网站中,Nginx的占有率为6.8%.与Apache相比,Nginx在高并发情况下具有巨大的性能优势. Nginx属于典型的微内核设计,其内核非常简洁和优雅,同时具有非常高的可扩展性.Nginx最初仅仅主要被用于做反向代理,后来随着HTTP核心的成熟和各种HTTP扩展模块的丰富,Nginx越来越多被用来取代Apache而单独承担HTTP Server的责任,例如目前淘宝内

Nginx模块开发—Nginx代码规范

1.简介 基本上,Nginx所采用的是一种类似BSD的C代码风格,很规范.也很清晰.建议我们的Nginx模块开发也采用Nginx的编码风格. 2.命名方式 除宏定义外,字母均为小写,单词间用下划线(_)间隔. 3.对齐方式 代码方式是K&R的对齐方式,需要注意以下几点: (1)用空格而不是tab对齐,空格数目为4个 : if (ngx_process == NGX_PROCESS_SINGLE) { ngx_single_process_cycle(cycle); } else { ngx_ma

解剖Nginx&#183;模块开发篇(1)跑起你的 Hello World 模块!

1 学习 Nginx 模块开发需要有哪些准备? 需要的预备知识不多,有如下几点: 有过一些 C 语言的编程经历: 知道 Nginx 是干嘛的,并有过编写或改写 Nginx 的配置文件的经历. OK,就这两点就够了 :) 好了,那就开始吧~ 2 我们的 HelloWorld 的目标是什么? 我们的目标,就是你在浏览器里输入http://localhost/hello_world时,显示: hello world 当然,为了能够更加自定义一些,我们尝试在hello world后面再显示一个字符串,比

Nginx 模块开发

Nginx 模块概述 Nginx 模块有三种角色: 处理请求并产生输出的 Handler 模块 : 处理由  Handler  产生的输出的 Filter (滤波器)模块: 当出现多个后台 服务器时, Load-balancer (负载均衡器)模块负责选择其中一个后台服务器发送请求: 通常,服务器启动时,任何 Handler 模块都有可能去处理配置文件中的  location  定义.若出现多个 Handler 模块被配置成需要处理某一特定的  location 时,最终只有其中一个 Handl