nginx-0.3.20-RELEASE import

*) Bugfix: in SSI handling.

    *) Bugfix: the ngx_http_memcached_module did not support the keys in
       the "/usr?args" form.
This commit is contained in:
Igor Sysoev 2006-01-11 15:26:57 +00:00
parent 04ebcd9429
commit df3254aa49
35 changed files with 485 additions and 252 deletions

View File

@ -110,6 +110,7 @@ fi
if [ $HTTP_SSI = YES ]; then if [ $HTTP_SSI = YES ]; then
have=NGX_HTTP_SSI . auto/have have=NGX_HTTP_SSI . auto/have
HTTP_FILTER_MODULES="$HTTP_FILTER_MODULES $HTTP_SSI_FILTER_MODULE" HTTP_FILTER_MODULES="$HTTP_FILTER_MODULES $HTTP_SSI_FILTER_MODULE"
HTTP_DEPS="$HTTP_DEPS $HTTP_SSI_DEPS"
HTTP_SRCS="$HTTP_SRCS $HTTP_SSI_SRCS" HTTP_SRCS="$HTTP_SRCS $HTTP_SSI_SRCS"
fi fi

View File

@ -303,6 +303,7 @@ HTTP_GZIP_SRCS=src/http/modules/ngx_http_gzip_filter_module.c
HTTP_SSI_FILTER_MODULE=ngx_http_ssi_filter_module HTTP_SSI_FILTER_MODULE=ngx_http_ssi_filter_module
HTTP_SSI_DEPS=src/http/modules/ngx_http_ssi_filter_module.h
HTTP_SSI_SRCS=src/http/modules/ngx_http_ssi_filter_module.c HTTP_SSI_SRCS=src/http/modules/ngx_http_ssi_filter_module.c

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (C) 2002-2005 Igor Sysoev * Copyright (C) 2002-2006 Igor Sysoev
* *
* Redistribution and use in source and binary forms, with or without * Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions * modification, are permitted provided that the following conditions

View File

@ -9,6 +9,28 @@
<title lang="en">nginx changelog</title> <title lang="en">nginx changelog</title>
<changes ver="0.3.20" date="11.01.2006">
<change type="bugfix">
<para lang="ru">
ÏÛÉÂËÉ × ÏÂÒÁÂÏÔËÅ SSI.
</para>
<para lang="en">
in SSI handling.
</para>
</change>
<change type="bugfix">
<para lang="ru">
ÍÏÄÕÌØ ngx_http_memcached_module ÎÅ ÐÏÄÄÅÒÖÉ×ÁÌ ËÌÀÞÉ × ×ÉÄÅ /uri?args.
</para>
<para lang="en">
the ngx_http_memcached_module did not support the keys in the "/usr?args" form.
</para>
</change>
</changes>
<changes ver="0.3.19" date="28.12.2005"> <changes ver="0.3.19" date="28.12.2005">
<change type="feature"> <change type="feature">

View File

@ -8,7 +8,7 @@
#define _NGINX_H_INCLUDED_ #define _NGINX_H_INCLUDED_
#define NGINX_VER "nginx/0.3.19" #define NGINX_VER "nginx/0.3.20"
#define NGINX_VAR "NGINX" #define NGINX_VAR "NGINX"
#define NGX_OLDPID_EXT ".oldbin" #define NGX_OLDPID_EXT ".oldbin"

View File

@ -93,6 +93,10 @@ typedef enum {
} ngx_connection_tcp_nopush_e; } ngx_connection_tcp_nopush_e;
#define NGX_LOWLEVEL_BUFFERED 0x0000000f
#define NGX_SSL_BUFFERED 0x00000001
struct ngx_connection_s { struct ngx_connection_s {
void *data; void *data;
ngx_event_t *read; ngx_event_t *read;
@ -134,9 +138,10 @@ struct ngx_connection_s {
ngx_atomic_uint_t number; ngx_atomic_uint_t number;
ngx_uint_t buffered;
unsigned log_error:2; /* ngx_connection_log_error_e */ unsigned log_error:2; /* ngx_connection_log_error_e */
unsigned buffered:1;
unsigned single_connection:1; unsigned single_connection:1;
unsigned unexpected_eof:1; unsigned unexpected_eof:1;
unsigned timedout:1; unsigned timedout:1;

View File

@ -753,7 +753,9 @@ ngx_hash_add_key(ngx_hash_keys_arrays_t *ha, ngx_str_t *key, void *value,
k = 0; k = 0;
for (i = 0; i < key->len; i++) { for (i = 0; i < key->len; i++) {
key->data[i] = ngx_tolower(key->data[i]); if (!(flags & NGX_HASH_READONLY_KEY)) {
key->data[i] = ngx_tolower(key->data[i]);
}
k = ngx_hash(k, key->data[i]); k = ngx_hash(k, key->data[i]);
} }

View File

@ -61,6 +61,7 @@ typedef struct {
#define NGX_HASH_LARGE_HSIZE 10007 #define NGX_HASH_LARGE_HSIZE 10007
#define NGX_HASH_WILDCARD_KEY 1 #define NGX_HASH_WILDCARD_KEY 1
#define NGX_HASH_READONLY_KEY 2
typedef struct { typedef struct {
@ -114,6 +115,7 @@ ngx_int_t ngx_hash_add_key(ngx_hash_keys_arrays_t *ha, ngx_str_t *key,
void *value, ngx_uint_t flags); void *value, ngx_uint_t flags);
#define ngx_hash0(key, c) key + c
ngx_int_t ngx_hash0_init(ngx_hash0_t *hash, ngx_pool_t *pool, void *names, ngx_int_t ngx_hash0_init(ngx_hash0_t *hash, ngx_pool_t *pool, void *names,
ngx_uint_t nelts); ngx_uint_t nelts);

View File

@ -28,9 +28,10 @@ static ngx_int_t ngx_output_chain_copy_buf(ngx_buf_t *dst, ngx_buf_t *src,
ngx_int_t ngx_int_t
ngx_output_chain(ngx_output_chain_ctx_t *ctx, ngx_chain_t *in) ngx_output_chain(ngx_output_chain_ctx_t *ctx, ngx_chain_t *in)
{ {
int rc, last;
off_t bsize; off_t bsize;
size_t size; size_t size;
ngx_int_t rc, last;
ngx_uint_t recycled;
ngx_chain_t *cl, *out, **last_out; ngx_chain_t *cl, *out, **last_out;
if (ctx->in == NULL && ctx->busy == NULL) { if (ctx->in == NULL && ctx->busy == NULL) {
@ -131,6 +132,7 @@ ngx_output_chain(ngx_output_chain_ctx_t *ctx, ngx_chain_t *in)
} else { } else {
size = ctx->bufs.size; size = ctx->bufs.size;
recycled = 1;
if (ctx->in->buf->last_in_chain) { if (ctx->in->buf->last_in_chain) {
@ -142,6 +144,7 @@ ngx_output_chain(ngx_output_chain_ctx_t *ctx, ngx_chain_t *in)
*/ */
size = (size_t) bsize; size = (size_t) bsize;
recycled = 0;
} else if (ctx->bufs.num == 1 } else if (ctx->bufs.num == 1
&& (bsize < (off_t) (ctx->bufs.size && (bsize < (off_t) (ctx->bufs.size
@ -154,6 +157,7 @@ ngx_output_chain(ngx_output_chain_ctx_t *ctx, ngx_chain_t *in)
*/ */
size = (size_t) bsize; size = (size_t) bsize;
recycled = 0;
} }
} }
@ -163,7 +167,7 @@ ngx_output_chain(ngx_output_chain_ctx_t *ctx, ngx_chain_t *in)
} }
ctx->buf->tag = ctx->tag; ctx->buf->tag = ctx->tag;
ctx->buf->recycled = 1; ctx->buf->recycled = recycled;
ctx->allocated++; ctx->allocated++;
} }
} }

View File

@ -592,7 +592,9 @@ ngx_ssl_send_chain(ngx_connection_t *c, ngx_chain_t *in, off_t limit)
ssize_t send, size; ssize_t send, size;
ngx_buf_t *buf; ngx_buf_t *buf;
if (!c->ssl->buffer || (in && in->next == NULL && !c->buffered)) { if (!c->ssl->buffer
|| (in && in->next == NULL && !(c->buffered & NGX_SSL_BUFFERED)))
{
/* /*
* we avoid a buffer copy if * we avoid a buffer copy if
@ -613,7 +615,7 @@ ngx_ssl_send_chain(ngx_connection_t *c, ngx_chain_t *in, off_t limit)
} }
if (n == NGX_AGAIN) { if (n == NGX_AGAIN) {
c->buffered = 1; c->buffered |= NGX_SSL_BUFFERED;
return in; return in;
} }
@ -689,7 +691,7 @@ ngx_ssl_send_chain(ngx_connection_t *c, ngx_chain_t *in, off_t limit)
} }
if (n == NGX_AGAIN) { if (n == NGX_AGAIN) {
c->buffered = 1; c->buffered |= NGX_SSL_BUFFERED;
return in; return in;
} }
@ -711,7 +713,12 @@ ngx_ssl_send_chain(ngx_connection_t *c, ngx_chain_t *in, off_t limit)
} }
} }
c->buffered = (buf->pos < buf->last) ? 1 : 0; if (buf->pos < buf->last) {
c->buffered |= NGX_SSL_BUFFERED;
} else {
c->buffered &= ~NGX_SSL_BUFFERED;
}
return in; return in;
} }

View File

@ -503,7 +503,7 @@ ngx_http_autoindex_handler(ngx_http_request_t *r)
b->last = ngx_cpymem(b->last, tail, sizeof(tail) - 1); b->last = ngx_cpymem(b->last, tail, sizeof(tail) - 1);
if (r->main == r) { if (r == r->main) {
b->last_buf = 1; b->last_buf = 1;
} }

View File

@ -162,7 +162,7 @@ ngx_http_charset_header_filter(ngx_http_request_t *r)
return ngx_http_next_header_filter(r); return ngx_http_next_header_filter(r);
} }
if (r->main == r if (r == r->main
&& ngx_strstr(r->headers_out.content_type.data, "charset") != NULL) && ngx_strstr(r->headers_out.content_type.data, "charset") != NULL)
{ {
return ngx_http_next_header_filter(r); return ngx_http_next_header_filter(r);

View File

@ -50,7 +50,7 @@ static ngx_http_output_body_filter_pt ngx_http_next_body_filter;
static ngx_int_t static ngx_int_t
ngx_http_chunked_header_filter(ngx_http_request_t *r) ngx_http_chunked_header_filter(ngx_http_request_t *r)
{ {
if (r->headers_out.status == NGX_HTTP_NOT_MODIFIED || r->main != r) { if (r->headers_out.status == NGX_HTTP_NOT_MODIFIED || r != r->main) {
return ngx_http_next_header_filter(r); return ngx_http_next_header_filter(r);
} }

View File

@ -101,7 +101,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_post_handler_pt ngx_http_gzip_hash_p = ngx_http_gzip_hash;
static ngx_conf_enum_t ngx_http_gzip_http_version[] = { static ngx_conf_enum_t ngx_http_gzip_http_version[] = {
{ ngx_string("1.0"), NGX_HTTP_VERSION_10 }, { ngx_string("1.0"), NGX_HTTP_VERSION_10 },
{ ngx_string("1.1"), NGX_HTTP_VERSION_11 }, { ngx_string("1.1"), NGX_HTTP_VERSION_11 },
@ -283,7 +282,7 @@ ngx_http_gzip_header_filter(ngx_http_request_t *r)
&& r->headers_out.status != NGX_HTTP_FORBIDDEN && r->headers_out.status != NGX_HTTP_FORBIDDEN
&& r->headers_out.status != NGX_HTTP_NOT_FOUND) && r->headers_out.status != NGX_HTTP_NOT_FOUND)
|| r->header_only || r->header_only
|| r->main != r || r != r->main
|| r->http_version < conf->http_version || r->http_version < conf->http_version
|| r->headers_out.content_type.len == 0 || r->headers_out.content_type.len == 0
|| (r->headers_out.content_encoding || (r->headers_out.content_encoding
@ -544,6 +543,8 @@ ngx_http_gzip_body_filter(ngx_http_request_t *r, ngx_chain_t *in)
return NGX_ERROR; return NGX_ERROR;
} }
r->connection->buffered |= NGX_HTTP_GZIP_BUFFERED;
ctx->last_out = &ctx->out; ctx->last_out = &ctx->out;
ctx->crc32 = crc32(0L, Z_NULL, 0); ctx->crc32 = crc32(0L, Z_NULL, 0);
@ -799,6 +800,8 @@ ngx_http_gzip_body_filter(ngx_http_request_t *r, ngx_chain_t *in)
ctx->done = 1; ctx->done = 1;
r->connection->buffered &= ~NGX_HTTP_GZIP_BUFFERED;
break; break;
} }

View File

@ -97,7 +97,7 @@ ngx_http_headers_filter(ngx_http_request_t *r)
if ((r->headers_out.status != NGX_HTTP_OK if ((r->headers_out.status != NGX_HTTP_OK
&& r->headers_out.status != NGX_HTTP_NOT_MODIFIED) && r->headers_out.status != NGX_HTTP_NOT_MODIFIED)
|| r->main != r) || r != r->main)
{ {
return ngx_http_next_header_filter(r); return ngx_http_next_header_filter(r);
} }

View File

@ -19,6 +19,7 @@ typedef struct {
typedef struct { typedef struct {
size_t rest; size_t rest;
ngx_http_request_t *request; ngx_http_request_t *request;
ngx_str_t key;
} ngx_http_memcached_ctx_t; } ngx_http_memcached_ctx_t;
@ -202,6 +203,8 @@ ngx_http_memcached_handler(ngx_http_request_t *r)
ctx->rest = NGX_HTTP_MEMCACHED_END; ctx->rest = NGX_HTTP_MEMCACHED_END;
ctx->request = r; ctx->request = r;
ngx_http_set_ctx(r, ctx, ngx_http_memcached_module);
u->input_filter_init = ngx_http_memcached_filter_init; u->input_filter_init = ngx_http_memcached_filter_init;
u->input_filter = ngx_http_memcached_filter; u->input_filter = ngx_http_memcached_filter;
u->input_filter_ctx = ctx; u->input_filter_ctx = ctx;
@ -215,13 +218,14 @@ ngx_http_memcached_handler(ngx_http_request_t *r)
static ngx_int_t static ngx_int_t
ngx_http_memcached_create_request(ngx_http_request_t *r) ngx_http_memcached_create_request(ngx_http_request_t *r)
{ {
size_t len; size_t len;
ngx_buf_t *b; ngx_buf_t *b;
ngx_chain_t *cl; ngx_chain_t *cl;
ngx_http_memcached_ctx_t *ctx;
len = sizeof("get ") - 1 + r->uri.len + sizeof(" " CRLF) - 1; len = sizeof("get ") - 1 + r->uri.len + sizeof(" " CRLF) - 1;
if (r->args.len) { if (r->args.len) {
len += 1+ r->args.len; len += 1 + r->args.len;
} }
b = ngx_create_temp_buf(r->pool, len); b = ngx_create_temp_buf(r->pool, len);
@ -241,6 +245,10 @@ ngx_http_memcached_create_request(ngx_http_request_t *r)
*b->last++ = 'g'; *b->last++ = 'e'; *b->last++ = 't'; *b->last++ = ' '; *b->last++ = 'g'; *b->last++ = 'e'; *b->last++ = 't'; *b->last++ = ' ';
ctx = ngx_http_get_module_ctx(r, ngx_http_memcached_module);
ctx->key.data = b->last;
b->last = ngx_copy(b->last, r->uri.data, r->uri.len); b->last = ngx_copy(b->last, r->uri.data, r->uri.len);
if (r->args.len) { if (r->args.len) {
@ -248,16 +256,10 @@ ngx_http_memcached_create_request(ngx_http_request_t *r)
b->last = ngx_copy(b->last, r->args.data, r->args.len); b->last = ngx_copy(b->last, r->args.data, r->args.len);
} }
#if (NGX_DEBUG) ctx->key.len = b->last - ctx->key.data;
{
ngx_str_t s;
s.len = b->last - b->pos;
s.data = b->pos;
ngx_log_debug1(NGX_LOG_DEBUG_HTTP, r->connection->log, 0, ngx_log_debug1(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
"http memcached request: \"%V\"", &s); "http memcached request: \"%V\"", &ctx->key);
}
#endif
*b->last++ = ' '; *b->last++ = CR; *b->last++ = LF; *b->last++ = ' '; *b->last++ = CR; *b->last++ = LF;
@ -275,9 +277,10 @@ ngx_http_memcached_reinit_request(ngx_http_request_t *r)
static ngx_int_t static ngx_int_t
ngx_http_memcached_process_header(ngx_http_request_t *r) ngx_http_memcached_process_header(ngx_http_request_t *r)
{ {
u_char *p, *len; u_char *p, *len;
ngx_str_t line; ngx_str_t line;
ngx_http_upstream_t *u; ngx_http_upstream_t *u;
ngx_http_memcached_ctx_t *ctx;
u = r->upstream; u = r->upstream;
@ -301,20 +304,22 @@ found:
p = u->buffer.pos; p = u->buffer.pos;
ctx = ngx_http_get_module_ctx(r, ngx_http_memcached_module);
if (ngx_strncmp(p, "VALUE ", sizeof("VALUE ") - 1) == 0) { if (ngx_strncmp(p, "VALUE ", sizeof("VALUE ") - 1) == 0) {
p += sizeof("VALUE ") - 1; p += sizeof("VALUE ") - 1;
if (ngx_strncmp(p, r->uri.data, r->uri.len) != 0) { if (ngx_strncmp(p, ctx->key.data, ctx->key.len) != 0) {
ngx_log_error(NGX_LOG_ERR, r->connection->log, 0, ngx_log_error(NGX_LOG_ERR, r->connection->log, 0,
"memcached sent invalid key in response \"%V\" " "memcached sent invalid key in response \"%V\" "
"for key \"%V\"", "for key \"%V\"",
&line, &r->uri); &line, &ctx->key);
return NGX_HTTP_UPSTREAM_INVALID_HEADER; return NGX_HTTP_UPSTREAM_INVALID_HEADER;
} }
p += r->uri.len; p += ctx->key.len;
if (*p++ != ' ') { if (*p++ != ' ') {
goto no_valid; goto no_valid;
@ -341,7 +346,7 @@ found:
ngx_log_error(NGX_LOG_ERR, r->connection->log, 0, ngx_log_error(NGX_LOG_ERR, r->connection->log, 0,
"memcached sent invalid length in response \"%V\" " "memcached sent invalid length in response \"%V\" "
"for key \"%V\"", "for key \"%V\"",
&line, &r->uri); &line, &ctx->key);
return NGX_HTTP_UPSTREAM_INVALID_HEADER; return NGX_HTTP_UPSTREAM_INVALID_HEADER;
} }
@ -353,7 +358,7 @@ found:
if (ngx_strcmp(p, "END\x0d") == 0) { if (ngx_strcmp(p, "END\x0d") == 0) {
ngx_log_error(NGX_LOG_INFO, r->connection->log, 0, ngx_log_error(NGX_LOG_INFO, r->connection->log, 0,
"key: \"%V\" was not found by memcached", &r->uri); "key: \"%V\" was not found by memcached", &ctx->key);
u->headers_in.status_n = 404; u->headers_in.status_n = 404;

View File

@ -52,7 +52,7 @@ static ngx_int_t ngx_http_not_modified_header_filter(ngx_http_request_t *r)
time_t ims; time_t ims;
if (r->headers_out.status != NGX_HTTP_OK if (r->headers_out.status != NGX_HTTP_OK
|| r->main != r || r != r->main
|| r->headers_in.if_modified_since == NULL || r->headers_in.if_modified_since == NULL
|| r->headers_out.last_modified_time == -1) || r->headers_out.last_modified_time == -1)
{ {

View File

@ -134,7 +134,7 @@ ngx_http_range_header_filter(ngx_http_request_t *r)
if (r->http_version < NGX_HTTP_VERSION_10 if (r->http_version < NGX_HTTP_VERSION_10
|| r->headers_out.status != NGX_HTTP_OK || r->headers_out.status != NGX_HTTP_OK
|| r->main != r || r != r->main
|| r->headers_out.content_length_n == -1 || r->headers_out.content_length_n == -1
|| !r->allow_ranges) || !r->allow_ranges)
{ {

View File

@ -8,94 +8,31 @@
#include <ngx_core.h> #include <ngx_core.h>
#include <ngx_http.h> #include <ngx_http.h>
#define NGX_HTTP_SSI_MAX_PARAMS 16
#define NGX_HTTP_SSI_COMMAND_LEN 31
#define NGX_HTTP_SSI_PARAM_LEN 31
#define NGX_HTTP_SSI_PARAMS_N 4
#define NGX_HTTP_SSI_ERROR 1 #define NGX_HTTP_SSI_ERROR 1
#define NGX_HTTP_SSI_DATE_LEN 2048 #define NGX_HTTP_SSI_DATE_LEN 2048
#define NGX_HTTP_SSI_ADD_PREFIX 1 #define NGX_HTTP_SSI_ADD_PREFIX 1
typedef struct { typedef struct {
ngx_flag_t enable; ngx_flag_t enable;
ngx_flag_t silent_errors; ngx_flag_t silent_errors;
ngx_flag_t ignore_recycled_buffers; ngx_flag_t ignore_recycled_buffers;
ngx_array_t *types; /* array of ngx_str_t */ ngx_array_t *types; /* array of ngx_str_t */
size_t min_file_chunk; size_t min_file_chunk;
size_t value_len; size_t value_len;
} ngx_http_ssi_conf_t; } ngx_http_ssi_loc_conf_t;
typedef struct { typedef struct {
ngx_str_t name; ngx_str_t name;
ngx_str_t value; ngx_str_t value;
} ngx_http_ssi_var_t; } ngx_http_ssi_var_t;
typedef struct {
ngx_buf_t *buf;
u_char *pos;
u_char *copy_start;
u_char *copy_end;
ngx_str_t command;
ngx_array_t params;
ngx_table_elt_t *param;
ngx_table_elt_t params_array[NGX_HTTP_SSI_PARAMS_N];
ngx_chain_t *in;
ngx_chain_t *out;
ngx_chain_t **last_out;
ngx_chain_t *busy;
ngx_chain_t *free;
ngx_uint_t state;
ngx_uint_t saved_state;
size_t saved;
size_t looked;
size_t value_len;
ngx_array_t variables;
ngx_uint_t output; /* unsigned output:1; */
ngx_str_t timefmt;
ngx_str_t errmsg;
} ngx_http_ssi_ctx_t;
typedef ngx_int_t (*ngx_http_ssi_command_pt) (ngx_http_request_t *r,
ngx_http_ssi_ctx_t *ctx, ngx_str_t **);
typedef struct {
ngx_str_t name;
ngx_uint_t index;
ngx_uint_t mandatory;
} ngx_http_ssi_param_t;
typedef struct {
ngx_str_t name;
ngx_http_ssi_command_pt handler;
ngx_http_ssi_param_t *params;
unsigned conditional:1;
unsigned flush:1;
} ngx_http_ssi_command_t;
typedef enum { typedef enum {
ssi_start_state = 0, ssi_start_state = 0,
ssi_tag_state, ssi_tag_state,
@ -149,7 +86,9 @@ static ngx_int_t ngx_http_ssi_date_gmt_local_variable(ngx_http_request_t *r,
static char *ngx_http_ssi_types(ngx_conf_t *cf, ngx_command_t *cmd, void *conf); static char *ngx_http_ssi_types(ngx_conf_t *cf, ngx_command_t *cmd, void *conf);
static ngx_int_t ngx_http_ssi_add_variables(ngx_conf_t *cf); static ngx_int_t ngx_http_ssi_preconfiguration(ngx_conf_t *cf);
static void *ngx_http_ssi_create_main_conf(ngx_conf_t *cf);
static char *ngx_http_ssi_init_main_conf(ngx_conf_t *cf, void *conf);
static void *ngx_http_ssi_create_conf(ngx_conf_t *cf); static void *ngx_http_ssi_create_conf(ngx_conf_t *cf);
static char *ngx_http_ssi_merge_conf(ngx_conf_t *cf, static char *ngx_http_ssi_merge_conf(ngx_conf_t *cf,
void *parent, void *child); void *parent, void *child);
@ -162,35 +101,35 @@ static ngx_command_t ngx_http_ssi_filter_commands[] = {
NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_FLAG, NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_FLAG,
ngx_conf_set_flag_slot, ngx_conf_set_flag_slot,
NGX_HTTP_LOC_CONF_OFFSET, NGX_HTTP_LOC_CONF_OFFSET,
offsetof(ngx_http_ssi_conf_t, enable), offsetof(ngx_http_ssi_loc_conf_t, enable),
NULL }, NULL },
{ ngx_string("ssi_silent_errors"), { ngx_string("ssi_silent_errors"),
NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_FLAG, NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_FLAG,
ngx_conf_set_flag_slot, ngx_conf_set_flag_slot,
NGX_HTTP_LOC_CONF_OFFSET, NGX_HTTP_LOC_CONF_OFFSET,
offsetof(ngx_http_ssi_conf_t, silent_errors), offsetof(ngx_http_ssi_loc_conf_t, silent_errors),
NULL }, NULL },
{ ngx_string("ssi_ignore_recycled_buffers"), { ngx_string("ssi_ignore_recycled_buffers"),
NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_FLAG, NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_FLAG,
ngx_conf_set_flag_slot, ngx_conf_set_flag_slot,
NGX_HTTP_LOC_CONF_OFFSET, NGX_HTTP_LOC_CONF_OFFSET,
offsetof(ngx_http_ssi_conf_t, ignore_recycled_buffers), offsetof(ngx_http_ssi_loc_conf_t, ignore_recycled_buffers),
NULL }, NULL },
{ ngx_string("ssi_min_file_chunk"), { ngx_string("ssi_min_file_chunk"),
NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_FLAG, NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_FLAG,
ngx_conf_set_size_slot, ngx_conf_set_size_slot,
NGX_HTTP_LOC_CONF_OFFSET, NGX_HTTP_LOC_CONF_OFFSET,
offsetof(ngx_http_ssi_conf_t, min_file_chunk), offsetof(ngx_http_ssi_loc_conf_t, min_file_chunk),
NULL }, NULL },
{ ngx_string("ssi_value_length"), { ngx_string("ssi_value_length"),
NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_FLAG, NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_FLAG,
ngx_conf_set_size_slot, ngx_conf_set_size_slot,
NGX_HTTP_LOC_CONF_OFFSET, NGX_HTTP_LOC_CONF_OFFSET,
offsetof(ngx_http_ssi_conf_t, value_len), offsetof(ngx_http_ssi_loc_conf_t, value_len),
NULL }, NULL },
{ ngx_string("ssi_types"), { ngx_string("ssi_types"),
@ -206,11 +145,11 @@ static ngx_command_t ngx_http_ssi_filter_commands[] = {
static ngx_http_module_t ngx_http_ssi_filter_module_ctx = { static ngx_http_module_t ngx_http_ssi_filter_module_ctx = {
ngx_http_ssi_add_variables, /* preconfiguration */ ngx_http_ssi_preconfiguration, /* preconfiguration */
NULL, /* postconfiguration */ NULL, /* postconfiguration */
NULL, /* create main configuration */ ngx_http_ssi_create_main_conf, /* create main configuration */
NULL, /* init main configuration */ ngx_http_ssi_init_main_conf, /* init main configuration */
NULL, /* create server configuration */ NULL, /* create server configuration */
NULL, /* merge server configuration */ NULL, /* merge server configuration */
@ -260,41 +199,41 @@ static ngx_str_t ngx_http_ssi_none = ngx_string("(none)");
static ngx_http_ssi_param_t ngx_http_ssi_include_params[] = { static ngx_http_ssi_param_t ngx_http_ssi_include_params[] = {
{ ngx_string("virtual"), NGX_HTTP_SSI_INCLUDE_VIRTUAL, 0 }, { ngx_string("virtual"), NGX_HTTP_SSI_INCLUDE_VIRTUAL, 0, 0 },
{ ngx_string("file"), NGX_HTTP_SSI_INCLUDE_FILE, 0 }, { ngx_string("file"), NGX_HTTP_SSI_INCLUDE_FILE, 0, 0 },
{ ngx_null_string, 0, 0 } { ngx_null_string, 0, 0, 0 }
}; };
static ngx_http_ssi_param_t ngx_http_ssi_echo_params[] = { static ngx_http_ssi_param_t ngx_http_ssi_echo_params[] = {
{ ngx_string("var"), NGX_HTTP_SSI_ECHO_VAR, 1 }, { ngx_string("var"), NGX_HTTP_SSI_ECHO_VAR, 1, 0 },
{ ngx_string("default"), NGX_HTTP_SSI_ECHO_DEFAULT, 0 }, { ngx_string("default"), NGX_HTTP_SSI_ECHO_DEFAULT, 0, 0 },
{ ngx_null_string, 0, 0 } { ngx_null_string, 0, 0, 0 }
}; };
static ngx_http_ssi_param_t ngx_http_ssi_config_params[] = { static ngx_http_ssi_param_t ngx_http_ssi_config_params[] = {
{ ngx_string("errmsg"), NGX_HTTP_SSI_CONFIG_ERRMSG, 0 }, { ngx_string("errmsg"), NGX_HTTP_SSI_CONFIG_ERRMSG, 0, 0 },
{ ngx_string("timefmt"), NGX_HTTP_SSI_CONFIG_TIMEFMT, 0 }, { ngx_string("timefmt"), NGX_HTTP_SSI_CONFIG_TIMEFMT, 0, 0 },
{ ngx_null_string, 0, 0 } { ngx_null_string, 0, 0, 0 }
}; };
static ngx_http_ssi_param_t ngx_http_ssi_set_params[] = { static ngx_http_ssi_param_t ngx_http_ssi_set_params[] = {
{ ngx_string("var"), NGX_HTTP_SSI_SET_VAR, 1 }, { ngx_string("var"), NGX_HTTP_SSI_SET_VAR, 1, 0 },
{ ngx_string("value"), NGX_HTTP_SSI_SET_VALUE, 1 }, { ngx_string("value"), NGX_HTTP_SSI_SET_VALUE, 1, 0 },
{ ngx_null_string, 0, 0 } { ngx_null_string, 0, 0, 0 }
}; };
static ngx_http_ssi_param_t ngx_http_ssi_if_params[] = { static ngx_http_ssi_param_t ngx_http_ssi_if_params[] = {
{ ngx_string("expr"), NGX_HTTP_SSI_IF_EXPR, 1 }, { ngx_string("expr"), NGX_HTTP_SSI_IF_EXPR, 1, 0 },
{ ngx_null_string, 0, 0 } { ngx_null_string, 0, 0, 0 }
}; };
static ngx_http_ssi_param_t ngx_http_ssi_no_params[] = { static ngx_http_ssi_param_t ngx_http_ssi_no_params[] = {
{ ngx_null_string, 0, 0 } { ngx_null_string, 0, 0, 0 }
}; };
@ -330,14 +269,14 @@ static ngx_http_variable_t ngx_http_ssi_vars[] = {
static ngx_int_t static ngx_int_t
ngx_http_ssi_header_filter(ngx_http_request_t *r) ngx_http_ssi_header_filter(ngx_http_request_t *r)
{ {
ngx_uint_t i; ngx_uint_t i;
ngx_str_t *type; ngx_str_t *type;
ngx_http_ssi_ctx_t *ctx; ngx_http_ssi_ctx_t *ctx;
ngx_http_ssi_conf_t *conf; ngx_http_ssi_loc_conf_t *slcf;
conf = ngx_http_get_module_loc_conf(r, ngx_http_ssi_filter_module); slcf = ngx_http_get_module_loc_conf(r, ngx_http_ssi_filter_module);
if (!conf->enable if (!slcf->enable
|| r->headers_out.content_type.len == 0 || r->headers_out.content_type.len == 0
|| r->headers_out.content_length_n == 0) || r->headers_out.content_length_n == 0)
{ {
@ -345,8 +284,8 @@ ngx_http_ssi_header_filter(ngx_http_request_t *r)
} }
type = conf->types->elts; type = slcf->types->elts;
for (i = 0; i < conf->types->nelts; i++) { for (i = 0; i < slcf->types->nelts; i++) {
if (r->headers_out.content_type.len >= type[i].len if (r->headers_out.content_type.len >= type[i].len
&& ngx_strncasecmp(r->headers_out.content_type.data, && ngx_strncasecmp(r->headers_out.content_type.data,
type[i].data, type[i].len) == 0) type[i].data, type[i].len) == 0)
@ -368,7 +307,7 @@ found:
ngx_http_set_ctx(r, ctx, ngx_http_ssi_filter_module); ngx_http_set_ctx(r, ctx, ngx_http_ssi_filter_module);
ctx->value_len = conf->value_len; ctx->value_len = slcf->value_len;
ctx->last_out = &ctx->out; ctx->last_out = &ctx->out;
ctx->output = 1; ctx->output = 1;
@ -388,7 +327,7 @@ found:
r->filter_need_in_memory = 1; r->filter_need_in_memory = 1;
if (r->main == r) { if (r == r->main) {
ngx_http_clear_content_length(r); ngx_http_clear_content_length(r);
ngx_http_clear_last_modified(r); ngx_http_clear_last_modified(r);
} }
@ -400,17 +339,18 @@ found:
static ngx_int_t static ngx_int_t
ngx_http_ssi_body_filter(ngx_http_request_t *r, ngx_chain_t *in) ngx_http_ssi_body_filter(ngx_http_request_t *r, ngx_chain_t *in)
{ {
ngx_int_t rc; ngx_int_t rc;
ngx_uint_t i; ngx_buf_t *b;
ngx_buf_t *b; ngx_uint_t i, index;
ngx_chain_t *cl; ngx_chain_t *cl;
ngx_table_elt_t *param; ngx_table_elt_t *param;
ngx_connection_t *c; ngx_connection_t *c;
ngx_http_ssi_ctx_t *ctx; ngx_http_ssi_ctx_t *ctx;
ngx_http_ssi_conf_t *conf; ngx_http_ssi_param_t *prm;
ngx_http_ssi_param_t *prm; ngx_http_ssi_command_t *cmd;
ngx_http_ssi_command_t *cmd; ngx_http_ssi_loc_conf_t *slcf;
ngx_str_t *params[NGX_HTTP_SSI_MAX_PARAMS]; ngx_http_ssi_main_conf_t *smcf;
ngx_str_t *params[NGX_HTTP_SSI_MAX_PARAMS];
ctx = ngx_http_get_module_ctx(r, ngx_http_ssi_filter_module); ctx = ngx_http_get_module_ctx(r, ngx_http_ssi_filter_module);
@ -426,7 +366,7 @@ ngx_http_ssi_body_filter(ngx_http_request_t *r, ngx_chain_t *in)
} }
} }
conf = ngx_http_get_module_loc_conf(r, ngx_http_ssi_filter_module); slcf = ngx_http_get_module_loc_conf(r, ngx_http_ssi_filter_module);
ngx_log_debug1(NGX_LOG_DEBUG_HTTP, r->connection->log, 0, ngx_log_debug1(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
"http ssi filter \"%V\"", &r->uri); "http ssi filter \"%V\"", &r->uri);
@ -528,7 +468,7 @@ ngx_http_ssi_body_filter(ngx_http_request_t *r, ngx_chain_t *in)
b->recycled = 0; b->recycled = 0;
if (b->in_file) { if (b->in_file) {
if (conf->min_file_chunk < (size_t) (b->last - b->pos)) if (slcf->min_file_chunk < (size_t) (b->last - b->pos))
{ {
b->file_last = b->file_pos + (b->last - b->start); b->file_last = b->file_pos + (b->last - b->start);
b->file_pos += b->pos - b->start; b->file_pos += b->pos - b->start;
@ -565,37 +505,39 @@ ngx_http_ssi_body_filter(ngx_http_request_t *r, ngx_chain_t *in)
if (rc == NGX_OK) { if (rc == NGX_OK) {
for (cmd = ngx_http_ssi_commands; cmd->handler; cmd++) { smcf = ngx_http_get_module_main_conf(r,
if (cmd->name.len == 0) { ngx_http_ssi_filter_module);
cmd = (ngx_http_ssi_command_t *) cmd->handler;
cmd = ngx_hash_find(&smcf->hash, ctx->key, ctx->command.data,
ctx->command.len);
if (cmd == NULL) {
if (ctx->output) {
ngx_log_error(NGX_LOG_ERR, r->connection->log, 0,
"invalid SSI command: \"%V\"",
&ctx->command);
goto ssi_error;
} }
if (cmd->name.len != ctx->command.len continue;
|| ngx_strncmp(cmd->name.data, ctx->command.data,
ctx->command.len) != 0)
{
continue;
}
break;
}
if (cmd->name.len == 0 && ctx->output) {
ngx_log_error(NGX_LOG_ERR, r->connection->log, 0,
"invalid SSI command: \"%V\"", &ctx->command);
goto ssi_error;
} }
if (!ctx->output && !cmd->conditional) { if (!ctx->output && !cmd->conditional) {
continue; continue;
} }
if (ctx->params.nelts > NGX_HTTP_SSI_MAX_PARAMS) {
ngx_log_error(NGX_LOG_ERR, r->connection->log, 0,
"too many SSI command paramters: \"%V\"",
&ctx->command);
goto ssi_error;
}
ngx_memzero(params, ngx_memzero(params,
NGX_HTTP_SSI_MAX_PARAMS * sizeof(ngx_str_t *)); NGX_HTTP_SSI_MAX_PARAMS * sizeof(ngx_str_t *));
param = ctx->params.elts; param = ctx->params.elts;
for (i = 0; i < ctx->params.nelts; i++) { for (i = 0; i < ctx->params.nelts; i++) {
for (prm = cmd->params; prm->name.len; prm++) { for (prm = cmd->params; prm->name.len; prm++) {
@ -607,17 +549,27 @@ ngx_http_ssi_body_filter(ngx_http_request_t *r, ngx_chain_t *in)
continue; continue;
} }
if (params[prm->index]) { if (!prm->multiple) {
ngx_log_error(NGX_LOG_ERR, if (params[prm->index]) {
r->connection->log, 0, ngx_log_error(NGX_LOG_ERR,
"duplicate \"%V\" parameter " r->connection->log, 0,
"in \"%V\" SSI command", "duplicate \"%V\" parameter "
&param[i].key, &ctx->command); "in \"%V\" SSI command",
&param[i].key, &ctx->command);
goto ssi_error; goto ssi_error;
}
params[prm->index] = &param[i].value;
break;
} }
params[prm->index] = &param[i].value; for (index = prm->index; params[index]; index++) {
/* void */
}
params[index] = &param[i].value;
break; break;
} }
@ -673,7 +625,7 @@ ngx_http_ssi_body_filter(ngx_http_request_t *r, ngx_chain_t *in)
ssi_error: ssi_error:
if (conf->silent_errors) { if (slcf->silent_errors) {
continue; continue;
} }
@ -709,7 +661,6 @@ ngx_http_ssi_body_filter(ngx_http_request_t *r, ngx_chain_t *in)
} }
if (ctx->buf->last_buf || ctx->buf->recycled) { if (ctx->buf->last_buf || ctx->buf->recycled) {
if (b == NULL) { if (b == NULL) {
if (ctx->free) { if (ctx->free) {
cl = ctx->free; cl = ctx->free;
@ -741,7 +692,7 @@ ngx_http_ssi_body_filter(ngx_http_request_t *r, ngx_chain_t *in)
b->last_buf = ctx->buf->last_buf; b->last_buf = ctx->buf->last_buf;
b->shadow = ctx->buf; b->shadow = ctx->buf;
if (conf->ignore_recycled_buffers == 0) { if (slcf->ignore_recycled_buffers == 0) {
b->recycled = ctx->buf->recycled; b->recycled = ctx->buf->recycled;
} }
} }
@ -969,6 +920,9 @@ ngx_http_ssi_parse(ngx_http_request_t *r, ngx_http_ssi_ctx_t *ctx)
} }
ctx->command.data[0] = ch; ctx->command.data[0] = ch;
ctx->key = 0;
ctx->key = ngx_hash(ctx->key, ch);
ctx->params.nelts = 0; ctx->params.nelts = 0;
state = ssi_command_state; state = ssi_command_state;
break; break;
@ -991,6 +945,7 @@ ngx_http_ssi_parse(ngx_http_request_t *r, ngx_http_ssi_ctx_t *ctx)
default: default:
ctx->command.data[ctx->command.len++] = ch; ctx->command.data[ctx->command.len++] = ch;
ctx->key = ngx_hash(ctx->key, ch);
if (ctx->command.len == NGX_HTTP_SSI_COMMAND_LEN) { if (ctx->command.len == NGX_HTTP_SSI_COMMAND_LEN) {
ngx_log_error(NGX_LOG_ERR, r->connection->log, 0, ngx_log_error(NGX_LOG_ERR, r->connection->log, 0,
@ -2039,18 +1994,18 @@ ngx_http_ssi_date_gmt_local_variable(ngx_http_request_t *r,
static char * static char *
ngx_http_ssi_types(ngx_conf_t *cf, ngx_command_t *cmd, void *conf) ngx_http_ssi_types(ngx_conf_t *cf, ngx_command_t *cmd, void *conf)
{ {
ngx_http_ssi_conf_t *scf = conf; ngx_http_ssi_loc_conf_t *slcf = conf;
ngx_str_t *value, *type; ngx_str_t *value, *type;
ngx_uint_t i; ngx_uint_t i;
if (scf->types == NULL) { if (slcf->types == NULL) {
scf->types = ngx_array_create(cf->pool, 4, sizeof(ngx_str_t)); slcf->types = ngx_array_create(cf->pool, 4, sizeof(ngx_str_t));
if (scf->types == NULL) { if (slcf->types == NULL) {
return NGX_CONF_ERROR; return NGX_CONF_ERROR;
} }
type = ngx_array_push(scf->types); type = ngx_array_push(slcf->types);
if (type == NULL) { if (type == NULL) {
return NGX_CONF_ERROR; return NGX_CONF_ERROR;
} }
@ -2067,7 +2022,7 @@ ngx_http_ssi_types(ngx_conf_t *cf, ngx_command_t *cmd, void *conf)
continue; continue;
} }
type = ngx_array_push(scf->types); type = ngx_array_push(slcf->types);
if (type == NULL) { if (type == NULL) {
return NGX_CONF_ERROR; return NGX_CONF_ERROR;
} }
@ -2087,9 +2042,12 @@ ngx_http_ssi_types(ngx_conf_t *cf, ngx_command_t *cmd, void *conf)
static ngx_int_t static ngx_int_t
ngx_http_ssi_add_variables(ngx_conf_t *cf) ngx_http_ssi_preconfiguration(ngx_conf_t *cf)
{ {
ngx_http_variable_t *var, *v; ngx_int_t rc;
ngx_http_variable_t *var, *v;
ngx_http_ssi_command_t *cmd;
ngx_http_ssi_main_conf_t *smcf;
for (v = ngx_http_ssi_vars; v->name.len; v++) { for (v = ngx_http_ssi_vars; v->name.len; v++) {
var = ngx_http_add_variable(cf, &v->name, v->flags); var = ngx_http_add_variable(cf, &v->name, v->flags);
@ -2101,17 +2059,82 @@ ngx_http_ssi_add_variables(ngx_conf_t *cf)
var->data = v->data; var->data = v->data;
} }
smcf = ngx_http_conf_get_module_main_conf(cf, ngx_http_ssi_filter_module);
for (cmd = ngx_http_ssi_commands; cmd->name.len; cmd++) {
rc = ngx_hash_add_key(&smcf->commands, &cmd->name, cmd,
NGX_HASH_READONLY_KEY);
if (rc == NGX_OK) {
continue;
}
if (rc == NGX_BUSY) {
ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,
"conflicting SSI command \"%V\"", &cmd->name);
}
return NGX_ERROR;
}
return NGX_OK; return NGX_OK;
} }
static void *
ngx_http_ssi_create_main_conf(ngx_conf_t *cf)
{
ngx_http_ssi_main_conf_t *smcf;
smcf = ngx_pcalloc(cf->pool, sizeof(ngx_http_ssi_main_conf_t));
if (smcf == NULL) {
return NGX_CONF_ERROR;
}
smcf->commands.pool = cf->pool;
smcf->commands.temp_pool = cf->temp_pool;
if (ngx_hash_keys_array_init(&smcf->commands, NGX_HASH_SMALL) != NGX_OK) {
return NGX_CONF_ERROR;
}
return smcf;
}
static char *
ngx_http_ssi_init_main_conf(ngx_conf_t *cf, void *conf)
{
ngx_http_ssi_main_conf_t *smcf = conf;
ngx_hash_init_t hash;
hash.hash = &smcf->hash;
hash.key = ngx_hash_key;
hash.max_size = 1024;
hash.bucket_size = ngx_cacheline_size;
hash.name = "ssi_command_hash";
hash.pool = cf->pool;
hash.temp_pool = NULL;
if (ngx_hash_init(&hash, smcf->commands.keys.elts,
smcf->commands.keys.nelts)
!= NGX_OK)
{
return NGX_CONF_ERROR;
}
return NGX_CONF_OK;
}
static void * static void *
ngx_http_ssi_create_conf(ngx_conf_t *cf) ngx_http_ssi_create_conf(ngx_conf_t *cf)
{ {
ngx_http_ssi_conf_t *conf; ngx_http_ssi_loc_conf_t *slcf;
conf = ngx_pcalloc(cf->pool, sizeof(ngx_http_ssi_conf_t)); slcf = ngx_pcalloc(cf->pool, sizeof(ngx_http_ssi_loc_conf_t));
if (conf == NULL) { if (slcf == NULL) {
return NGX_CONF_ERROR; return NGX_CONF_ERROR;
} }
@ -2121,22 +2144,22 @@ ngx_http_ssi_create_conf(ngx_conf_t *cf)
* conf->types = NULL; * conf->types = NULL;
*/ */
conf->enable = NGX_CONF_UNSET; slcf->enable = NGX_CONF_UNSET;
conf->silent_errors = NGX_CONF_UNSET; slcf->silent_errors = NGX_CONF_UNSET;
conf->ignore_recycled_buffers = NGX_CONF_UNSET; slcf->ignore_recycled_buffers = NGX_CONF_UNSET;
conf->min_file_chunk = NGX_CONF_UNSET_SIZE; slcf->min_file_chunk = NGX_CONF_UNSET_SIZE;
conf->value_len = NGX_CONF_UNSET_SIZE; slcf->value_len = NGX_CONF_UNSET_SIZE;
return conf; return slcf;
} }
static char * static char *
ngx_http_ssi_merge_conf(ngx_conf_t *cf, void *parent, void *child) ngx_http_ssi_merge_conf(ngx_conf_t *cf, void *parent, void *child)
{ {
ngx_http_ssi_conf_t *prev = parent; ngx_http_ssi_loc_conf_t *prev = parent;
ngx_http_ssi_conf_t *conf = child; ngx_http_ssi_loc_conf_t *conf = child;
ngx_str_t *type; ngx_str_t *type;

View File

@ -0,0 +1,90 @@
/*
* Copyright (C) Igor Sysoev
*/
#ifndef _NGX_HTTP_SSI_FILTER_H_INCLUDED_
#define _NGX_HTTP_SSI_FILTER_H_INCLUDED_
#include <ngx_config.h>
#include <ngx_core.h>
#include <ngx_http.h>
#define NGX_HTTP_SSI_MAX_PARAMS 16
#define NGX_HTTP_SSI_COMMAND_LEN 31
#define NGX_HTTP_SSI_PARAM_LEN 31
#define NGX_HTTP_SSI_PARAMS_N 4
typedef struct {
ngx_hash_t hash;
ngx_hash_keys_arrays_t commands;
} ngx_http_ssi_main_conf_t;
typedef struct {
ngx_buf_t *buf;
u_char *pos;
u_char *copy_start;
u_char *copy_end;
ngx_uint_t key;
ngx_str_t command;
ngx_array_t params;
ngx_table_elt_t *param;
ngx_table_elt_t params_array[NGX_HTTP_SSI_PARAMS_N];
ngx_chain_t *in;
ngx_chain_t *out;
ngx_chain_t **last_out;
ngx_chain_t *busy;
ngx_chain_t *free;
ngx_uint_t state;
ngx_uint_t saved_state;
size_t saved;
size_t looked;
size_t value_len;
ngx_array_t variables;
ngx_uint_t output; /* unsigned output:1; */
ngx_str_t timefmt;
ngx_str_t errmsg;
} ngx_http_ssi_ctx_t;
typedef ngx_int_t (*ngx_http_ssi_command_pt) (ngx_http_request_t *r,
ngx_http_ssi_ctx_t *ctx, ngx_str_t **);
typedef struct {
ngx_str_t name;
ngx_uint_t index;
unsigned mandatory:1;
unsigned multiple:1;
} ngx_http_ssi_param_t;
typedef struct {
ngx_str_t name;
ngx_http_ssi_command_pt handler;
ngx_http_ssi_param_t *params;
unsigned conditional:1;
unsigned flush:1;
} ngx_http_ssi_command_t;
extern ngx_module_t ngx_http_ssi_filter_module;
#endif /* _NGX_HTTP_SSI_FILTER_H_INCLUDED_ */

View File

@ -244,7 +244,7 @@ ngx_http_static_handler(ngx_http_request_t *r)
return NGX_HTTP_INTERNAL_SERVER_ERROR; return NGX_HTTP_INTERNAL_SERVER_ERROR;
} }
if (r->main != r && ngx_file_size(&fi) == 0) { if (r != r->main && ngx_file_size(&fi) == 0) {
return ngx_http_send_header(r); return ngx_http_send_header(r);
} }
@ -272,7 +272,7 @@ ngx_http_static_handler(ngx_http_request_t *r)
b->file_last = ngx_file_size(&fi); b->file_last = ngx_file_size(&fi);
b->in_file = b->file_last ? 1: 0; b->in_file = b->file_last ? 1: 0;
b->last_buf = (r->main == r) ? 1: 0; b->last_buf = (r == r->main) ? 1: 0;
b->last_in_chain = 1; b->last_in_chain = 1;
b->file->fd = fd; b->file->fd = fd;

View File

@ -213,7 +213,7 @@ ngx_http_userid_filter(ngx_http_request_t *r)
ngx_http_userid_ctx_t *ctx; ngx_http_userid_ctx_t *ctx;
ngx_http_userid_conf_t *conf; ngx_http_userid_conf_t *conf;
if (r->main != r) { if (r != r->main) {
return ngx_http_next_header_filter(r); return ngx_http_next_header_filter(r);
} }

View File

@ -38,6 +38,9 @@ typedef u_char *(*ngx_http_log_handler_pt)(ngx_http_request_t *r, u_char *buf,
#include <ngx_http_core_module.h> #include <ngx_http_core_module.h>
#include <ngx_http_script.h> #include <ngx_http_script.h>
#if (NGX_HTTP_SSI)
#include <ngx_http_ssi_filter_module.h>
#endif
#if (NGX_HTTP_SSL) #if (NGX_HTTP_SSL)
#include <ngx_http_ssl_module.h> #include <ngx_http_ssl_module.h>
#endif #endif

View File

@ -469,7 +469,7 @@ ngx_http_handler(ngx_http_request_t *r)
r->uri_changed = 1; r->uri_changed = 1;
r->uri_changes = NGX_HTTP_MAX_REWRITE_CYCLES + 1; r->uri_changes = NGX_HTTP_MAX_REWRITE_CYCLES + 1;
r->phase = (r->main == r) ? NGX_HTTP_POST_READ_PHASE: r->phase = (r == r->main) ? NGX_HTTP_POST_READ_PHASE:
NGX_HTTP_SERVER_REWRITE_PHASE; NGX_HTTP_SERVER_REWRITE_PHASE;
r->phase_handler = 0; r->phase_handler = 0;
@ -516,7 +516,7 @@ ngx_http_core_run_phases(ngx_http_request_t *r)
r->phase = NGX_HTTP_FIND_CONFIG_PHASE; r->phase = NGX_HTTP_FIND_CONFIG_PHASE;
} }
if (r->phase == NGX_HTTP_ACCESS_PHASE && r->main != r) { if (r->phase == NGX_HTTP_ACCESS_PHASE && r != r->main) {
continue; continue;
} }
@ -1229,18 +1229,19 @@ ngx_http_subrequest(ngx_http_request_t *r,
} }
sr->internal = 1; sr->internal = 1;
sr->fast_subrequest = 1;
sr->discard_body = r->discard_body; sr->discard_body = r->discard_body;
sr->main_filter_need_in_memory = r->main_filter_need_in_memory; sr->main_filter_need_in_memory = r->main_filter_need_in_memory;
ngx_http_handler(sr); ngx_http_handler(sr);
#if (NGX_LOG_DEBUG)
if (!c->destroyed) { if (!c->destroyed) {
sr->fast_subrequest = 0;
ngx_log_debug2(NGX_LOG_DEBUG_HTTP, c->log, 0, ngx_log_debug2(NGX_LOG_DEBUG_HTTP, c->log, 0,
"http subrequest done \"%V?%V\"", uri, &sr->args); "http subrequest done \"%V?%V\"", uri, &sr->args);
} }
#endif
return NGX_OK; return NGX_OK;
} }

View File

@ -159,7 +159,7 @@ ngx_http_header_filter(ngx_http_request_t *r)
ngx_table_elt_t *header; ngx_table_elt_t *header;
ngx_http_core_loc_conf_t *clcf; ngx_http_core_loc_conf_t *clcf;
if (r->main != r) { if (r != r->main) {
return NGX_OK; return NGX_OK;
} }

View File

@ -96,7 +96,9 @@ ngx_http_postpone_filter(ngx_http_request_t *r, ngx_chain_t *in)
if (r->postponed) { if (r->postponed) {
out = r->postponed->out; out = r->postponed->out;
r->postponed = r->postponed->next; if (out) {
r->postponed = r->postponed->next;
}
} else { } else {
out = in; out = in;
@ -104,7 +106,10 @@ ngx_http_postpone_filter(ngx_http_request_t *r, ngx_chain_t *in)
rc = NGX_OK; rc = NGX_OK;
if (out || r->main->out || r->main->connection->buffered) { if (out
|| (r->connection->buffered
& (NGX_HTTP_LOWLEVEL_BUFFERED|NGX_LOWLEVEL_BUFFERED)))
{
ngx_log_debug2(NGX_LOG_DEBUG_HTTP, r->connection->log, 0, ngx_log_debug2(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
"http postpone filter out \"%V?%V\"", &r->uri, &r->args); "http postpone filter out \"%V?%V\"", &r->uri, &r->args);
@ -170,6 +175,10 @@ ngx_http_postpone_filter_output_postponed_request(ngx_http_request_t *r)
pr = r->postponed; pr = r->postponed;
} }
if (pr == NULL) {
return NGX_OK;
}
out = pr->out; out = pr->out;
if (out) { if (out) {

View File

@ -1423,7 +1423,7 @@ ngx_http_finalize_request(ngx_http_request_t *r, ngx_int_t rc)
return; return;
} }
if (r->main == r) { if (r == r->main) {
if (r->connection->read->timer_set) { if (r->connection->read->timer_set) {
ngx_del_timer(r->connection->read); ngx_del_timer(r->connection->read);
} }
@ -1438,7 +1438,7 @@ ngx_http_finalize_request(ngx_http_request_t *r, ngx_int_t rc)
return; return;
} }
if (r->main != r || rc == NGX_AGAIN) { if (r != r->main || rc == NGX_AGAIN) {
if (ngx_http_set_write_handler(r) != NGX_OK) { if (ngx_http_set_write_handler(r) != NGX_OK) {
return; return;
} }
@ -1453,7 +1453,7 @@ ngx_http_finalize_request(ngx_http_request_t *r, ngx_int_t rc)
return; return;
} }
if (r->main != r) { if (r != r->main) {
pr = r->parent; pr = r->parent;
@ -1461,7 +1461,7 @@ ngx_http_finalize_request(ngx_http_request_t *r, ngx_int_t rc)
"http parent request: \"%V?%V\"", &pr->uri, &pr->args); "http parent request: \"%V?%V\"", &pr->uri, &pr->args);
if (rc != NGX_AGAIN) { if (rc != NGX_AGAIN) {
pr->connection->data = pr; r->connection->data = pr;
} }
if (pr->postponed) { if (pr->postponed) {
@ -1472,12 +1472,22 @@ ngx_http_finalize_request(ngx_http_request_t *r, ngx_int_t rc)
if (rc != NGX_AGAIN && pr->postponed->request == r) { if (rc != NGX_AGAIN && pr->postponed->request == r) {
pr->postponed = pr->postponed->next; pr->postponed = pr->postponed->next;
if (pr->postponed == NULL) {
return;
}
} }
if (r->fast_subrequest) {
ngx_log_debug2(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
"http fast subrequest: \"%V?%V\" done",
&r->uri, &r->args);
return;
}
ngx_log_debug2(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
"http wake parent request: \"%V?%V\"",
&pr->uri, &pr->args);
pr->write_event_handler(pr);
#if 0
ngx_log_debug2(NGX_LOG_DEBUG_HTTP, r->connection->log, 0, ngx_log_debug2(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
"http request: \"%V?%V\" still has postponed", "http request: \"%V?%V\" still has postponed",
&pr->uri, &pr->args); &pr->uri, &pr->args);
@ -1489,6 +1499,8 @@ ngx_http_finalize_request(ngx_http_request_t *r, ngx_int_t rc)
pr->write_event_handler(pr); pr->write_event_handler(pr);
} }
#endif
} }
return; return;
@ -1498,7 +1510,7 @@ ngx_http_finalize_request(ngx_http_request_t *r, ngx_int_t rc)
return; return;
} }
if (r->out) { if (r->connection->buffered) {
(void) ngx_http_set_write_handler(r); (void) ngx_http_set_write_handler(r);
return; return;
} }
@ -1655,7 +1667,11 @@ ngx_http_writer(ngx_http_request_t *r)
ngx_http_close_request(r, 0); ngx_http_close_request(r, 0);
} }
return; if (r == r->main) {
return;
}
rc = NGX_OK;
} }
ngx_log_debug2(NGX_LOG_DEBUG_HTTP, wev->log, 0, ngx_log_debug2(NGX_LOG_DEBUG_HTTP, wev->log, 0,
@ -1691,7 +1707,7 @@ ngx_http_discard_body(ngx_http_request_t *r)
ssize_t size; ssize_t size;
ngx_event_t *rev; ngx_event_t *rev;
if (r->main != r) { if (r != r->main) {
return NGX_OK; return NGX_OK;
} }

View File

@ -99,6 +99,13 @@
#define NGX_HTTP_GATEWAY_TIME_OUT 504 #define NGX_HTTP_GATEWAY_TIME_OUT 504
#define NGX_HTTP_LOWLEVEL_BUFFERED 0x000000f0
#define NGX_HTTP_WRITE_BUFFERED 0x00000010
#define NGX_HTTP_GZIP_BUFFERED 0x00000020
#define NGX_HTTP_SSI_BUFFERED 0x00000100
#define NGX_HTTP_COPY_BUFFERED 0x00000200
typedef enum { typedef enum {
NGX_HTTP_RESTRICT_HOST_OFF = 0, NGX_HTTP_RESTRICT_HOST_OFF = 0,
NGX_HTTP_RESTRICT_HOST_ON, NGX_HTTP_RESTRICT_HOST_ON,
@ -379,6 +386,8 @@ struct ngx_http_request_s {
unsigned uri_changed:1; unsigned uri_changed:1;
unsigned uri_changes:4; unsigned uri_changes:4;
unsigned fast_subrequest:1;
unsigned low_case_exten:1; unsigned low_case_exten:1;
unsigned header_timeout_set:1; unsigned header_timeout_set:1;

View File

@ -447,7 +447,7 @@ ngx_http_special_response_handler(ngx_http_request_t *r, ngx_int_t error)
cl->buf = b; cl->buf = b;
} }
if (r->main == r) { if (r == r->main) {
b->last_buf = 1; b->last_buf = 1;
} }

View File

@ -558,7 +558,7 @@ ngx_http_upstream_connect(ngx_http_request_t *r, ngx_http_upstream_t *u)
} }
} }
if (r->request_body && r->request_body->temp_file && r->main == r) { if (r->request_body && r->request_body->temp_file && r == r->main) {
/* /*
* the r->request_body->buf can be reused for one request only, * the r->request_body->buf can be reused for one request only,
@ -695,7 +695,7 @@ ngx_http_upstream_reinit(ngx_http_request_t *r, ngx_http_upstream_t *u)
/* reinit the subrequest's ngx_output_chain() context */ /* reinit the subrequest's ngx_output_chain() context */
if (r->request_body && r->request_body->temp_file if (r->request_body && r->request_body->temp_file
&& r->main != r && u->output.buf) && r != r->main && u->output.buf)
{ {
u->output.free = ngx_alloc_chain_link(r->pool); u->output.free = ngx_alloc_chain_link(r->pool);
if (u->output.free == NULL) { if (u->output.free == NULL) {
@ -1031,12 +1031,20 @@ ngx_http_upstream_process_header(ngx_event_t *rev)
#endif #endif
} }
if (u->headers_in.status_n == NGX_HTTP_NOT_FOUND if (u->headers_in.status_n == NGX_HTTP_NOT_FOUND) {
&& u->peer.tries > 1
&& u->conf->next_upstream & NGX_HTTP_UPSTREAM_FT_HTTP_404) if (u->peer.tries > 1
{ && u->conf->next_upstream & NGX_HTTP_UPSTREAM_FT_HTTP_404)
ngx_http_upstream_next(r, u, NGX_HTTP_UPSTREAM_FT_HTTP_404); {
return; ngx_http_upstream_next(r, u, NGX_HTTP_UPSTREAM_FT_HTTP_404);
return;
}
if (u->conf->redirect_404) {
rc = (r->err_ctx == NULL) ? 404 : 204;
ngx_http_upstream_finalize_request(r, u, rc);
return;
}
} }
@ -1044,13 +1052,6 @@ ngx_http_upstream_process_header(ngx_event_t *rev)
&& u->conf->redirect_errors && u->conf->redirect_errors
&& r->err_ctx == NULL) && r->err_ctx == NULL)
{ {
if (u->headers_in.status_n == NGX_HTTP_NOT_FOUND
&& u->conf->redirect_404)
{
ngx_http_upstream_finalize_request(r, u, 404);
return;
}
clcf = ngx_http_get_module_loc_conf(r, ngx_http_core_module); clcf = ngx_http_get_module_loc_conf(r, ngx_http_core_module);
if (clcf->error_pages) { if (clcf->error_pages) {
@ -1947,7 +1948,7 @@ ngx_http_upstream_finalize_request(ngx_http_request_t *r,
r->connection->log->action = "sending to client"; r->connection->log->action = "sending to client";
if (rc == 0 && r->main == r) { if (rc == 0 && r == r->main) {
rc = ngx_http_send_special(r, NGX_HTTP_LAST); rc = ngx_http_send_special(r, NGX_HTTP_LAST);
} }

View File

@ -179,20 +179,24 @@ ngx_http_write_filter(ngx_http_request_t *r, ngx_chain_t *in)
} }
if (c->write->delayed) { if (c->write->delayed) {
c->buffered |= NGX_HTTP_WRITE_BUFFERED;
return NGX_AGAIN; return NGX_AGAIN;
} }
if (size == 0 && !c->buffered) { if (size == 0 && !(c->buffered & NGX_LOWLEVEL_BUFFERED)) {
if (last) { if (last) {
r->out = NULL; r->out = NULL;
c->buffered &= ~NGX_HTTP_WRITE_BUFFERED;
return NGX_OK; return NGX_OK;
} }
if (flush) { if (flush) {
do { do {
r->out = r->out->next; r->out = r->out->next;
} } while (r->out);
while (r->out);
c->buffered &= ~NGX_HTTP_WRITE_BUFFERED;
return NGX_OK; return NGX_OK;
} }
@ -232,7 +236,14 @@ ngx_http_write_filter(ngx_http_request_t *r, ngx_chain_t *in)
r->out = chain; r->out = chain;
if (chain || (c->buffered && r->postponed == NULL)) { if (chain) {
c->buffered |= NGX_HTTP_WRITE_BUFFERED;
return NGX_AGAIN;
}
c->buffered &= ~NGX_HTTP_WRITE_BUFFERED;
if ((c->buffered & NGX_LOWLEVEL_BUFFERED) && r->postponed == NULL) {
return NGX_AGAIN; return NGX_AGAIN;
} }

View File

@ -28,8 +28,13 @@
*/ */
#define NGX_HEADERS 8 #if (IOV_MAX > 64)
#define NGX_TRAILERS 8 #define NGX_HEADERS 64
#define NGX_TRAILERS 64
#else
#define NGX_HEADERS IOV_MAX
#define NGX_TRAILERS IOV_MAX
#endif
ngx_chain_t * ngx_chain_t *

View File

@ -20,7 +20,11 @@
*/ */
#define NGX_HEADERS 8 #if (IOV_MAX > 64)
#define NGX_HEADERS 64
#else
#define NGX_HEADERS IOV_MAX
#endif
ngx_chain_t * ngx_chain_t *

View File

@ -31,7 +31,12 @@ static ssize_t sendfilev(int fd, const struct sendfilevec *vec,
#endif #endif
#define NGX_SENDFILEVECS 16 #if (IOV_MAX > 64)
#define NGX_SENDFILEVECS 64
#else
#define NGX_SENDFILEVECS IOV_MAX
#endif
ngx_chain_t * ngx_chain_t *

View File

@ -9,7 +9,11 @@
#include <ngx_event.h> #include <ngx_event.h>
#define NGX_IOVS 16 #if (IOV_MAX > 64)
#define NGX_IOVS 64
#else
#define NGX_IOVS IOV_MAX
#endif
ngx_chain_t * ngx_chain_t *