summaryrefslogtreecommitdiffhomepage
path: root/src/http/modules
diff options
context:
space:
mode:
Diffstat (limited to 'src/http/modules')
-rw-r--r--src/http/modules/ngx_http_gzip_filter_module.c235
-rw-r--r--src/http/modules/ngx_http_gzip_static_module.c302
2 files changed, 327 insertions, 210 deletions
diff --git a/src/http/modules/ngx_http_gzip_filter_module.c b/src/http/modules/ngx_http_gzip_filter_module.c
index 42dfa5021..1587c009d 100644
--- a/src/http/modules/ngx_http_gzip_filter_module.c
+++ b/src/http/modules/ngx_http_gzip_filter_module.c
@@ -14,15 +14,11 @@
typedef struct {
ngx_flag_t enable;
ngx_flag_t no_buffer;
- ngx_flag_t vary;
ngx_array_t *types; /* array of ngx_str_t */
ngx_bufs_t bufs;
- ngx_uint_t http_version;
- ngx_uint_t proxied;
-
ngx_int_t level;
size_t wbits;
size_t memlevel;
@@ -30,17 +26,6 @@ typedef struct {
} ngx_http_gzip_conf_t;
-#define NGX_HTTP_GZIP_PROXIED_OFF 0x0002
-#define NGX_HTTP_GZIP_PROXIED_EXPIRED 0x0004
-#define NGX_HTTP_GZIP_PROXIED_NO_CACHE 0x0008
-#define NGX_HTTP_GZIP_PROXIED_NO_STORE 0x0010
-#define NGX_HTTP_GZIP_PROXIED_PRIVATE 0x0020
-#define NGX_HTTP_GZIP_PROXIED_NO_LM 0x0040
-#define NGX_HTTP_GZIP_PROXIED_NO_ETAG 0x0080
-#define NGX_HTTP_GZIP_PROXIED_AUTH 0x0100
-#define NGX_HTTP_GZIP_PROXIED_ANY 0x0200
-
-
typedef struct {
ngx_chain_t *in;
ngx_chain_t *free;
@@ -70,8 +55,6 @@ typedef struct {
} ngx_http_gzip_ctx_t;
-static ngx_int_t ngx_http_gzip_proxied(ngx_http_request_t *r,
- ngx_http_gzip_conf_t *conf);
static void *ngx_http_gzip_filter_alloc(void *opaque, u_int items,
u_int size);
static void ngx_http_gzip_filter_free(void *opaque, void *address);
@@ -99,27 +82,6 @@ static ngx_conf_post_handler_pt ngx_http_gzip_window_p = ngx_http_gzip_window;
static ngx_conf_post_handler_pt ngx_http_gzip_hash_p = ngx_http_gzip_hash;
-static ngx_conf_enum_t ngx_http_gzip_http_version[] = {
- { ngx_string("1.0"), NGX_HTTP_VERSION_10 },
- { ngx_string("1.1"), NGX_HTTP_VERSION_11 },
- { ngx_null_string, 0 }
-};
-
-
-static ngx_conf_bitmask_t ngx_http_gzip_proxied_mask[] = {
- { ngx_string("off"), NGX_HTTP_GZIP_PROXIED_OFF },
- { ngx_string("expired"), NGX_HTTP_GZIP_PROXIED_EXPIRED },
- { ngx_string("no-cache"), NGX_HTTP_GZIP_PROXIED_NO_CACHE },
- { ngx_string("no-store"), NGX_HTTP_GZIP_PROXIED_NO_STORE },
- { ngx_string("private"), NGX_HTTP_GZIP_PROXIED_PRIVATE },
- { ngx_string("no_last_modified"), NGX_HTTP_GZIP_PROXIED_NO_LM },
- { ngx_string("no_etag"), NGX_HTTP_GZIP_PROXIED_NO_ETAG },
- { ngx_string("auth"), NGX_HTTP_GZIP_PROXIED_AUTH },
- { ngx_string("any"), NGX_HTTP_GZIP_PROXIED_ANY },
- { ngx_null_string, 0 }
-};
-
-
static ngx_command_t ngx_http_gzip_filter_commands[] = {
{ ngx_string("gzip"),
@@ -172,20 +134,6 @@ static ngx_command_t ngx_http_gzip_filter_commands[] = {
offsetof(ngx_http_gzip_conf_t, no_buffer),
NULL },
- { ngx_string("gzip_http_version"),
- NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_TAKE1,
- ngx_conf_set_enum_slot,
- NGX_HTTP_LOC_CONF_OFFSET,
- offsetof(ngx_http_gzip_conf_t, http_version),
- &ngx_http_gzip_http_version },
-
- { ngx_string("gzip_proxied"),
- NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_1MORE,
- ngx_conf_set_bitmask_slot,
- NGX_HTTP_LOC_CONF_OFFSET,
- offsetof(ngx_http_gzip_conf_t, proxied),
- &ngx_http_gzip_proxied_mask },
-
{ ngx_string("gzip_min_length"),
NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_TAKE1,
ngx_conf_set_size_slot,
@@ -193,13 +141,6 @@ static ngx_command_t ngx_http_gzip_filter_commands[] = {
offsetof(ngx_http_gzip_conf_t, min_length),
NULL },
- { ngx_string("gzip_vary"),
- NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_FLAG,
- ngx_conf_set_flag_slot,
- NGX_HTTP_LOC_CONF_OFFSET,
- offsetof(ngx_http_gzip_conf_t, vary),
- NULL },
-
ngx_null_command
};
@@ -255,10 +196,6 @@ struct gztrailer {
static ngx_str_t ngx_http_gzip_ratio = ngx_string("gzip_ratio");
-static ngx_str_t ngx_http_gzip_no_cache = ngx_string("no-cache");
-static ngx_str_t ngx_http_gzip_no_store = ngx_string("no-store");
-static ngx_str_t ngx_http_gzip_private = ngx_string("private");
-
static ngx_http_output_header_filter_pt ngx_http_next_header_filter;
static ngx_http_output_body_filter_pt ngx_http_next_body_filter;
@@ -267,11 +204,12 @@ static ngx_http_output_body_filter_pt ngx_http_next_body_filter;
static ngx_int_t
ngx_http_gzip_header_filter(ngx_http_request_t *r)
{
- ngx_str_t *type;
- ngx_uint_t i;
- ngx_table_elt_t *header;
- ngx_http_gzip_ctx_t *ctx;
- ngx_http_gzip_conf_t *conf;
+ ngx_str_t *type;
+ ngx_uint_t i;
+ ngx_table_elt_t *h;
+ ngx_http_gzip_ctx_t *ctx;
+ ngx_http_gzip_conf_t *conf;
+ ngx_http_core_loc_conf_t *clcf;
conf = ngx_http_get_module_loc_conf(r, ngx_http_gzip_filter_module);
@@ -280,23 +218,16 @@ ngx_http_gzip_header_filter(ngx_http_request_t *r)
&& r->headers_out.status != NGX_HTTP_FORBIDDEN
&& r->headers_out.status != NGX_HTTP_NOT_FOUND)
|| r->header_only
- || r != r->main
- || r->http_version < conf->http_version
|| r->headers_out.content_type.len == 0
|| (r->headers_out.content_encoding
&& r->headers_out.content_encoding->value.len)
- || r->headers_in.accept_encoding == NULL
|| (r->headers_out.content_length_n != -1
&& r->headers_out.content_length_n < conf->min_length)
- || ngx_strcasestrn(r->headers_in.accept_encoding->value.data,
- "gzip", 4 - 1)
- == NULL
- )
+ || ngx_http_gzip_ok(r) != NGX_OK)
{
return ngx_http_next_header_filter(r);
}
-
type = conf->types->elts;
for (i = 0; i < conf->types->nelts; i++) {
if (r->headers_out.content_type.len >= type[i].len
@@ -309,32 +240,8 @@ ngx_http_gzip_header_filter(ngx_http_request_t *r)
return ngx_http_next_header_filter(r);
-
found:
- if (r->headers_in.via) {
- if (conf->proxied & NGX_HTTP_GZIP_PROXIED_OFF) {
- return ngx_http_next_header_filter(r);
- }
-
- if (!(conf->proxied & NGX_HTTP_GZIP_PROXIED_ANY)
- && ngx_http_gzip_proxied(r, conf) == NGX_DECLINED)
- {
- return ngx_http_next_header_filter(r);
- }
- }
-
-
- /*
- * if the URL (without the "http://" prefix) is longer than 253 bytes
- * then MSIE 4.x can not handle the compressed stream - it waits too long,
- * hangs up or crashes
- */
-
- if (r->headers_in.msie4 && r->unparsed_uri.len > 200) {
- return ngx_http_next_header_filter(r);
- }
-
ctx = ngx_pcalloc(r->pool, sizeof(ngx_http_gzip_ctx_t));
if (ctx == NULL) {
return NGX_ERROR;
@@ -342,33 +249,34 @@ found:
ngx_http_set_ctx(r, ctx, ngx_http_gzip_filter_module);
-
ctx->request = r;
- header = ngx_list_push(&r->headers_out.headers);
- if (header == NULL) {
+ h = ngx_list_push(&r->headers_out.headers);
+ if (h == NULL) {
return NGX_ERROR;
}
- header->hash = 1;
- header->key.len = sizeof("Content-Encoding") - 1;
- header->key.data = (u_char *) "Content-Encoding";
- header->value.len = sizeof("gzip") - 1;
- header->value.data = (u_char *) "gzip";
+ h->hash = 1;
+ h->key.len = sizeof("Content-Encoding") - 1;
+ h->key.data = (u_char *) "Content-Encoding";
+ h->value.len = sizeof("gzip") - 1;
+ h->value.data = (u_char *) "gzip";
+
+ r->headers_out.content_encoding = h;
- r->headers_out.content_encoding = header;
+ clcf = ngx_http_get_module_loc_conf(r, ngx_http_core_module);
- if (conf->vary) {
- header = ngx_list_push(&r->headers_out.headers);
- if (header == NULL) {
+ if (clcf->gzip_vary) {
+ h = ngx_list_push(&r->headers_out.headers);
+ if (h == NULL) {
return NGX_ERROR;
}
- header->hash = 1;
- header->key.len = sizeof("Vary") - 1;
- header->key.data = (u_char *) "Vary";
- header->value.len = sizeof("Accept-Encoding") - 1;
- header->value.data = (u_char *) "Accept-Encoding";
+ h->hash = 1;
+ h->key.len = sizeof("Vary") - 1;
+ h->key.data = (u_char *) "Vary";
+ h->value.len = sizeof("Accept-Encoding") - 1;
+ h->value.data = (u_char *) "Accept-Encoding";
}
ctx->length = r->headers_out.content_length_n;
@@ -383,89 +291,6 @@ found:
static ngx_int_t
-ngx_http_gzip_proxied(ngx_http_request_t *r, ngx_http_gzip_conf_t *conf)
-{
- time_t date, expires;
-
- if (r->headers_in.authorization
- && (conf->proxied & NGX_HTTP_GZIP_PROXIED_AUTH))
- {
- return NGX_OK;
- }
-
- if (r->headers_out.expires) {
-
- if (!(conf->proxied & NGX_HTTP_GZIP_PROXIED_EXPIRED)) {
- return NGX_DECLINED;
- }
-
- expires = ngx_http_parse_time(r->headers_out.expires->value.data,
- r->headers_out.expires->value.len);
- if (expires == NGX_ERROR) {
- return NGX_DECLINED;
- }
-
- if (r->headers_out.date) {
- date = ngx_http_parse_time(r->headers_out.date->value.data,
- r->headers_out.date->value.len);
- if (date == NGX_ERROR) {
- return NGX_DECLINED;
- }
-
- } else {
- date = ngx_time();
- }
-
- if (expires < date) {
- return NGX_OK;
- }
-
- return NGX_DECLINED;
- }
-
- if (r->headers_out.cache_control.elts) {
-
- if ((conf->proxied & NGX_HTTP_GZIP_PROXIED_NO_CACHE)
- && ngx_http_parse_multi_header_lines(&r->headers_out.cache_control,
- &ngx_http_gzip_no_cache, NULL) >= 0)
- {
- return NGX_OK;
- }
-
- if ((conf->proxied & NGX_HTTP_GZIP_PROXIED_NO_STORE)
- && ngx_http_parse_multi_header_lines(&r->headers_out.cache_control,
- &ngx_http_gzip_no_store, NULL) >= 0)
- {
- return NGX_OK;
- }
-
- if ((conf->proxied & NGX_HTTP_GZIP_PROXIED_PRIVATE)
- && ngx_http_parse_multi_header_lines(&r->headers_out.cache_control,
- &ngx_http_gzip_private, NULL) >= 0)
- {
- return NGX_OK;
- }
-
- return NGX_DECLINED;
- }
-
- if ((conf->proxied & NGX_HTTP_GZIP_PROXIED_NO_LM)
- && r->headers_out.last_modified)
- {
- return NGX_DECLINED;
- }
-
- if ((conf->proxied & NGX_HTTP_GZIP_PROXIED_NO_ETAG)
- && r->headers_out.etag)
- {
- return NGX_DECLINED;
- }
-
- return NGX_OK;
-}
-
-
-static ngx_int_t
ngx_http_gzip_body_filter(ngx_http_request_t *r, ngx_chain_t *in)
{
int rc, wbits, memlevel;
@@ -1017,15 +842,11 @@ ngx_http_gzip_create_conf(ngx_conf_t *cf)
* set by ngx_pcalloc():
*
* conf->bufs.num = 0;
- * conf->proxied = 0;
* conf->types = NULL;
*/
conf->enable = NGX_CONF_UNSET;
conf->no_buffer = NGX_CONF_UNSET;
- conf->vary = NGX_CONF_UNSET;
-
- conf->http_version = NGX_CONF_UNSET_UINT;
conf->level = NGX_CONF_UNSET;
conf->wbits = (size_t) NGX_CONF_UNSET;
@@ -1048,18 +869,12 @@ ngx_http_gzip_merge_conf(ngx_conf_t *cf, void *parent, void *child)
ngx_conf_merge_bufs_value(conf->bufs, prev->bufs, 4, ngx_pagesize);
- ngx_conf_merge_uint_value(conf->http_version, prev->http_version,
- NGX_HTTP_VERSION_11);
- ngx_conf_merge_bitmask_value(conf->proxied, prev->proxied,
- (NGX_CONF_BITMASK_SET|NGX_HTTP_GZIP_PROXIED_OFF));
-
ngx_conf_merge_value(conf->level, prev->level, 1);
ngx_conf_merge_size_value(conf->wbits, prev->wbits, MAX_WBITS);
ngx_conf_merge_size_value(conf->memlevel, prev->memlevel,
MAX_MEM_LEVEL - 1);
ngx_conf_merge_value(conf->min_length, prev->min_length, 20);
ngx_conf_merge_value(conf->no_buffer, prev->no_buffer, 0);
- ngx_conf_merge_value(conf->vary, prev->vary, 0);
if (conf->types == NULL) {
if (prev->types == NULL) {
diff --git a/src/http/modules/ngx_http_gzip_static_module.c b/src/http/modules/ngx_http_gzip_static_module.c
new file mode 100644
index 000000000..e4aaeb345
--- /dev/null
+++ b/src/http/modules/ngx_http_gzip_static_module.c
@@ -0,0 +1,302 @@
+
+/*
+ * Copyright (C) Igor Sysoev
+ */
+
+
+#include <ngx_config.h>
+#include <ngx_core.h>
+#include <ngx_http.h>
+
+
+typedef struct {
+ ngx_flag_t enable;
+} ngx_http_gzip_static_conf_t;
+
+
+static ngx_int_t ngx_http_gzip_static_handler(ngx_http_request_t *r);
+static void *ngx_http_gzip_static_create_conf(ngx_conf_t *cf);
+static char *ngx_http_gzip_static_merge_conf(ngx_conf_t *cf, void *parent,
+ void *child);
+static ngx_int_t ngx_http_gzip_static_init(ngx_conf_t *cf);
+
+
+static ngx_command_t ngx_http_gzip_static_commands[] = {
+
+ { ngx_string("gzip_static"),
+ NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_FLAG,
+ ngx_conf_set_flag_slot,
+ NGX_HTTP_LOC_CONF_OFFSET,
+ offsetof(ngx_http_gzip_static_conf_t, enable),
+ NULL },
+
+ ngx_null_command
+};
+
+
+ngx_http_module_t ngx_http_gzip_static_module_ctx = {
+ NULL, /* preconfiguration */
+ ngx_http_gzip_static_init, /* postconfiguration */
+
+ NULL, /* create main configuration */
+ NULL, /* init main configuration */
+
+ NULL, /* create server configuration */
+ NULL, /* merge server configuration */
+
+ ngx_http_gzip_static_create_conf, /* create location configuration */
+ ngx_http_gzip_static_merge_conf /* merge location configuration */
+};
+
+
+ngx_module_t ngx_http_gzip_static_module = {
+ NGX_MODULE_V1,
+ &ngx_http_gzip_static_module_ctx, /* module context */
+ ngx_http_gzip_static_commands, /* module directives */
+ NGX_HTTP_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_int_t
+ngx_http_gzip_static_handler(ngx_http_request_t *r)
+{
+ u_char *p;
+ size_t root;
+ ngx_str_t path;
+ ngx_int_t rc;
+ ngx_uint_t level;
+ ngx_log_t *log;
+ ngx_buf_t *b;
+ ngx_chain_t out;
+ ngx_table_elt_t *h;
+ ngx_open_file_info_t of;
+ ngx_http_core_loc_conf_t *clcf;
+ ngx_http_gzip_static_conf_t *gzcf;
+
+ if (!(r->method & (NGX_HTTP_GET|NGX_HTTP_HEAD))) {
+ return NGX_HTTP_NOT_ALLOWED;
+ }
+
+ if (r->uri.data[r->uri.len - 1] == '/') {
+ return NGX_DECLINED;
+ }
+
+ /* TODO: Win32 */
+ if (r->zero_in_uri) {
+ return NGX_DECLINED;
+ }
+
+ gzcf = ngx_http_get_module_loc_conf(r, ngx_http_gzip_static_module);
+
+ if (!gzcf->enable || ngx_http_gzip_ok(r) != NGX_OK) {
+ return NGX_DECLINED;
+ }
+
+ log = r->connection->log;
+
+ p = ngx_http_map_uri_to_path(r, &path, &root, sizeof(".gz") - 1);
+ if (p == NULL) {
+ return NGX_HTTP_INTERNAL_SERVER_ERROR;
+ }
+
+ *p++ = '.';
+ *p++ = 'g';
+ *p++ = 'z';
+ *p = '\0';
+
+ path.len = p - path.data;
+
+ ngx_log_debug1(NGX_LOG_DEBUG_HTTP, log, 0,
+ "http filename: \"%s\"", path.data);
+
+ clcf = ngx_http_get_module_loc_conf(r, ngx_http_core_module);
+
+ of.test_dir = 0;
+ of.valid = clcf->open_file_cache_valid;
+ of.min_uses = clcf->open_file_cache_min_uses;
+ of.errors = clcf->open_file_cache_errors;
+ of.events = clcf->open_file_cache_events;
+
+ rc = ngx_open_cached_file(clcf->open_file_cache, &path, &of, r->pool);
+
+ if (rc == NGX_ERROR) {
+
+ switch (of.err) {
+
+ case 0:
+ return NGX_HTTP_INTERNAL_SERVER_ERROR;
+
+ case NGX_ENOENT:
+ case NGX_ENOTDIR:
+ case NGX_ENAMETOOLONG:
+
+ return NGX_DECLINED;
+
+ case NGX_EACCES:
+
+ level = NGX_LOG_ERR;
+ rc = NGX_DECLINED;
+ break;
+
+ default:
+
+ level = NGX_LOG_CRIT;
+ rc = NGX_DECLINED;
+ break;
+ }
+
+ ngx_log_error(level, log, of.err,
+ ngx_open_file_n " \"%s\" failed", path.data);
+
+ return rc;
+ }
+
+ ngx_log_debug1(NGX_LOG_DEBUG_HTTP, log, 0, "http static fd: %d", of.fd);
+
+ if (of.is_dir) {
+ ngx_log_debug0(NGX_LOG_DEBUG_HTTP, log, 0, "http dir");
+ return NGX_DECLINED;
+ }
+
+#if !(NGX_WIN32) /* the not regular files are probably Unix specific */
+
+ if (!of.is_file) {
+ ngx_log_error(NGX_LOG_CRIT, log, ngx_errno,
+ "\"%s\" is not a regular file", path.data);
+
+ return NGX_HTTP_NOT_FOUND;
+ }
+
+#endif
+
+ rc = ngx_http_discard_request_body(r);
+
+ if (rc != NGX_OK) {
+ return rc;
+ }
+
+ log->action = "sending response to client";
+
+ 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_content_type(r) != NGX_OK) {
+ return NGX_HTTP_INTERNAL_SERVER_ERROR;
+ }
+
+ h = ngx_list_push(&r->headers_out.headers);
+ if (h == NULL) {
+ return NGX_ERROR;
+ }
+
+ h->hash = 1;
+ h->key.len = sizeof("Content-Encoding") - 1;
+ h->key.data = (u_char *) "Content-Encoding";
+ h->value.len = sizeof("gzip") - 1;
+ h->value.data = (u_char *) "gzip";
+
+ r->headers_out.content_encoding = h;
+
+ if (clcf->gzip_vary) {
+ h = ngx_list_push(&r->headers_out.headers);
+ if (h == NULL) {
+ return NGX_ERROR;
+ }
+
+ h->hash = 1;
+ h->key.len = sizeof("Vary") - 1;
+ h->key.data = (u_char *) "Vary";
+ h->value.len = sizeof("Accept-Encoding") - 1;
+ h->value.data = (u_char *) "Accept-Encoding";
+ }
+
+ /* we need to allocate all before the header would be sent */
+
+ 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 = 1;
+ b->last_in_chain = 1;
+
+ b->file->fd = of.fd;
+ b->file->name = path;
+ b->file->log = log;
+
+ out.buf = b;
+ out.next = NULL;
+
+ return ngx_http_output_filter(r, &out);
+}
+
+
+static void *
+ngx_http_gzip_static_create_conf(ngx_conf_t *cf)
+{
+ ngx_http_gzip_static_conf_t *conf;
+
+ conf = ngx_palloc(cf->pool, sizeof(ngx_http_gzip_static_conf_t));
+ if (conf == NULL) {
+ return NGX_CONF_ERROR;
+ }
+
+ conf->enable = NGX_CONF_UNSET;
+
+ return conf;
+}
+
+
+static char *
+ngx_http_gzip_static_merge_conf(ngx_conf_t *cf, void *parent, void *child)
+{
+ ngx_http_gzip_static_conf_t *prev = parent;
+ ngx_http_gzip_static_conf_t *conf = child;
+
+ ngx_conf_merge_value(conf->enable, prev->enable, 0);
+
+ return NGX_CONF_OK;
+}
+
+
+static ngx_int_t
+ngx_http_gzip_static_init(ngx_conf_t *cf)
+{
+ ngx_http_handler_pt *h;
+ ngx_http_core_main_conf_t *cmcf;
+
+ cmcf = ngx_http_conf_get_module_main_conf(cf, ngx_http_core_module);
+
+ h = ngx_array_push(&cmcf->phases[NGX_HTTP_CONTENT_PHASE].handlers);
+ if (h == NULL) {
+ return NGX_ERROR;
+ }
+
+ *h = ngx_http_gzip_static_handler;
+
+ return NGX_OK;
+}