blob: e52b96e9b1561374b2d919c32b00fee430a1dd8e [file] [log] [blame]
/*
* Copyright (C) Igor Sysoev
* Copyright (C) Nginx, Inc.
*/
#include <ngx_config.h>
#include <ngx_core.h>
#include <ngx_http.h>
#define NGX_HTTP_CHARSET_OFF -2
#define NGX_HTTP_NO_CHARSET -3
#define NGX_HTTP_CHARSET_VAR 0x10000
/* 1 byte length and up to 3 bytes for the UTF-8 encoding of the UCS-2 */
#define NGX_UTF_LEN 4
#define NGX_HTML_ENTITY_LEN (sizeof("&#1114111;") - 1)
typedef struct {
u_char **tables;
ngx_str_t name;
unsigned length:16;
unsigned utf8:1;
} ngx_http_charset_t;
typedef struct {
ngx_int_t src;
ngx_int_t dst;
} ngx_http_charset_recode_t;
typedef struct {
ngx_int_t src;
ngx_int_t dst;
u_char *src2dst;
u_char *dst2src;
} ngx_http_charset_tables_t;
typedef struct {
ngx_array_t charsets; /* ngx_http_charset_t */
ngx_array_t tables; /* ngx_http_charset_tables_t */
ngx_array_t recodes; /* ngx_http_charset_recode_t */
} ngx_http_charset_main_conf_t;
typedef struct {
ngx_int_t charset;
ngx_int_t source_charset;
ngx_flag_t override_charset;
ngx_hash_t types;
ngx_array_t *types_keys;
} ngx_http_charset_loc_conf_t;
typedef struct {
u_char *table;
ngx_int_t charset;
ngx_str_t charset_name;
ngx_chain_t *busy;
ngx_chain_t *free_bufs;
ngx_chain_t *free_buffers;
size_t saved_len;
u_char saved[NGX_UTF_LEN];
unsigned length:16;
unsigned from_utf8:1;
unsigned to_utf8:1;
} ngx_http_charset_ctx_t;
typedef struct {
ngx_http_charset_tables_t *table;
ngx_http_charset_t *charset;
ngx_uint_t characters;
} ngx_http_charset_conf_ctx_t;
static ngx_int_t ngx_http_destination_charset(ngx_http_request_t *r,
ngx_str_t *name);
static ngx_int_t ngx_http_main_request_charset(ngx_http_request_t *r,
ngx_str_t *name);
static ngx_int_t ngx_http_source_charset(ngx_http_request_t *r,
ngx_str_t *name);
static ngx_int_t ngx_http_get_charset(ngx_http_request_t *r, ngx_str_t *name);
static ngx_inline void ngx_http_set_charset(ngx_http_request_t *r,
ngx_str_t *charset);
static ngx_int_t ngx_http_charset_ctx(ngx_http_request_t *r,
ngx_http_charset_t *charsets, ngx_int_t charset, ngx_int_t source_charset);
static ngx_uint_t ngx_http_charset_recode(ngx_buf_t *b, u_char *table);
static ngx_chain_t *ngx_http_charset_recode_from_utf8(ngx_pool_t *pool,
ngx_buf_t *buf, ngx_http_charset_ctx_t *ctx);
static ngx_chain_t *ngx_http_charset_recode_to_utf8(ngx_pool_t *pool,
ngx_buf_t *buf, ngx_http_charset_ctx_t *ctx);
static ngx_chain_t *ngx_http_charset_get_buf(ngx_pool_t *pool,
ngx_http_charset_ctx_t *ctx);
static ngx_chain_t *ngx_http_charset_get_buffer(ngx_pool_t *pool,
ngx_http_charset_ctx_t *ctx, size_t size);
static char *ngx_http_charset_map_block(ngx_conf_t *cf, ngx_command_t *cmd,
void *conf);
static char *ngx_http_charset_map(ngx_conf_t *cf, ngx_command_t *dummy,
void *conf);
static char *ngx_http_set_charset_slot(ngx_conf_t *cf, ngx_command_t *cmd,
void *conf);
static ngx_int_t ngx_http_add_charset(ngx_array_t *charsets, ngx_str_t *name);
static void *ngx_http_charset_create_main_conf(ngx_conf_t *cf);
static void *ngx_http_charset_create_loc_conf(ngx_conf_t *cf);
static char *ngx_http_charset_merge_loc_conf(ngx_conf_t *cf,
void *parent, void *child);
static ngx_int_t ngx_http_charset_postconfiguration(ngx_conf_t *cf);
static ngx_str_t ngx_http_charset_default_types[] = {
ngx_string("text/html"),
ngx_string("text/xml"),
ngx_string("text/plain"),
ngx_string("text/vnd.wap.wml"),
ngx_string("application/javascript"),
ngx_string("application/rss+xml"),
ngx_null_string
};
static ngx_command_t ngx_http_charset_filter_commands[] = {
{ ngx_string("charset"),
NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF
|NGX_HTTP_LIF_CONF|NGX_CONF_TAKE1,
ngx_http_set_charset_slot,
NGX_HTTP_LOC_CONF_OFFSET,
offsetof(ngx_http_charset_loc_conf_t, charset),
NULL },
{ ngx_string("source_charset"),
NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF
|NGX_HTTP_LIF_CONF|NGX_CONF_TAKE1,
ngx_http_set_charset_slot,
NGX_HTTP_LOC_CONF_OFFSET,
offsetof(ngx_http_charset_loc_conf_t, source_charset),
NULL },
{ ngx_string("override_charset"),
NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF
|NGX_HTTP_LIF_CONF|NGX_CONF_FLAG,
ngx_conf_set_flag_slot,
NGX_HTTP_LOC_CONF_OFFSET,
offsetof(ngx_http_charset_loc_conf_t, override_charset),
NULL },
{ ngx_string("charset_types"),
NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_1MORE,
ngx_http_types_slot,
NGX_HTTP_LOC_CONF_OFFSET,
offsetof(ngx_http_charset_loc_conf_t, types_keys),
&ngx_http_charset_default_types[0] },
{ ngx_string("charset_map"),
NGX_HTTP_MAIN_CONF|NGX_CONF_BLOCK|NGX_CONF_TAKE2,
ngx_http_charset_map_block,
NGX_HTTP_MAIN_CONF_OFFSET,
0,
NULL },
ngx_null_command
};
static ngx_http_module_t ngx_http_charset_filter_module_ctx = {
NULL, /* preconfiguration */
ngx_http_charset_postconfiguration, /* postconfiguration */
ngx_http_charset_create_main_conf, /* create main configuration */
NULL, /* init main configuration */
NULL, /* create server configuration */
NULL, /* merge server configuration */
ngx_http_charset_create_loc_conf, /* create location configuration */
ngx_http_charset_merge_loc_conf /* merge location configuration */
};
ngx_module_t ngx_http_charset_filter_module = {
NGX_MODULE_V1,
&ngx_http_charset_filter_module_ctx, /* module context */
ngx_http_charset_filter_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_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_charset_header_filter(ngx_http_request_t *r)
{
ngx_int_t charset, source_charset;
ngx_str_t dst, src;
ngx_http_charset_t *charsets;
ngx_http_charset_main_conf_t *mcf;
if (r == r->main) {
charset = ngx_http_destination_charset(r, &dst);
} else {
charset = ngx_http_main_request_charset(r, &dst);
}
if (charset == NGX_ERROR) {
return NGX_ERROR;
}
if (charset == NGX_DECLINED) {
return ngx_http_next_header_filter(r);
}
/* charset: charset index or NGX_HTTP_NO_CHARSET */
source_charset = ngx_http_source_charset(r, &src);
if (source_charset == NGX_ERROR) {
return NGX_ERROR;
}
/*
* source_charset: charset index, NGX_HTTP_NO_CHARSET,
* or NGX_HTTP_CHARSET_OFF
*/
ngx_log_debug2(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
"charset: \"%V\" > \"%V\"", &src, &dst);
if (source_charset == NGX_HTTP_CHARSET_OFF) {
ngx_http_set_charset(r, &dst);
return ngx_http_next_header_filter(r);
}
if (charset == NGX_HTTP_NO_CHARSET
|| source_charset == NGX_HTTP_NO_CHARSET)
{
if (source_charset != charset
|| ngx_strncasecmp(dst.data, src.data, dst.len) != 0)
{
goto no_charset_map;
}
ngx_http_set_charset(r, &dst);
return ngx_http_next_header_filter(r);
}
if (source_charset == charset) {
r->headers_out.content_type.len = r->headers_out.content_type_len;
ngx_http_set_charset(r, &dst);
return ngx_http_next_header_filter(r);
}
/* source_charset != charset */
if (r->headers_out.content_encoding
&& r->headers_out.content_encoding->value.len)
{
return ngx_http_next_header_filter(r);
}
mcf = ngx_http_get_module_main_conf(r, ngx_http_charset_filter_module);
charsets = mcf->charsets.elts;
if (charsets[source_charset].tables == NULL
|| charsets[source_charset].tables[charset] == NULL)
{
goto no_charset_map;
}
r->headers_out.content_type.len = r->headers_out.content_type_len;
ngx_http_set_charset(r, &dst);
return ngx_http_charset_ctx(r, charsets, charset, source_charset);
no_charset_map:
ngx_log_error(NGX_LOG_ERR, r->connection->log, 0,
"no \"charset_map\" between the charsets \"%V\" and \"%V\"",
&src, &dst);
return ngx_http_next_header_filter(r);
}
static ngx_int_t
ngx_http_destination_charset(ngx_http_request_t *r, ngx_str_t *name)
{
ngx_int_t charset;
ngx_http_charset_t *charsets;
ngx_http_variable_value_t *vv;
ngx_http_charset_loc_conf_t *mlcf;
ngx_http_charset_main_conf_t *mcf;
if (r->headers_out.content_type.len == 0) {
return NGX_DECLINED;
}
if (r->headers_out.override_charset
&& r->headers_out.override_charset->len)
{
*name = *r->headers_out.override_charset;
charset = ngx_http_get_charset(r, name);
if (charset != NGX_HTTP_NO_CHARSET) {
return charset;
}
ngx_log_error(NGX_LOG_ERR, r->connection->log, 0,
"unknown charset \"%V\" to override", name);
return NGX_DECLINED;
}
mlcf = ngx_http_get_module_loc_conf(r, ngx_http_charset_filter_module);
charset = mlcf->charset;
if (charset == NGX_HTTP_CHARSET_OFF) {
return NGX_DECLINED;
}
if (r->headers_out.charset.len) {
if (mlcf->override_charset == 0) {
return NGX_DECLINED;
}
} else {
if (ngx_http_test_content_type(r, &mlcf->types) == NULL) {
return NGX_DECLINED;
}
}
if (charset < NGX_HTTP_CHARSET_VAR) {
mcf = ngx_http_get_module_main_conf(r, ngx_http_charset_filter_module);
charsets = mcf->charsets.elts;
*name = charsets[charset].name;
return charset;
}
vv = ngx_http_get_indexed_variable(r, charset - NGX_HTTP_CHARSET_VAR);
if (vv == NULL || vv->not_found) {
return NGX_ERROR;
}
name->len = vv->len;
name->data = vv->data;
return ngx_http_get_charset(r, name);
}
static ngx_int_t
ngx_http_main_request_charset(ngx_http_request_t *r, ngx_str_t *src)
{
ngx_int_t charset;
ngx_str_t *main_charset;
ngx_http_charset_ctx_t *ctx;
ctx = ngx_http_get_module_ctx(r->main, ngx_http_charset_filter_module);
if (ctx) {
*src = ctx->charset_name;
return ctx->charset;
}
main_charset = &r->main->headers_out.charset;
if (main_charset->len == 0) {
return NGX_DECLINED;
}
ctx = ngx_pcalloc(r->pool, sizeof(ngx_http_charset_ctx_t));
if (ctx == NULL) {
return NGX_ERROR;
}
ngx_http_set_ctx(r->main, ctx, ngx_http_charset_filter_module);
charset = ngx_http_get_charset(r, main_charset);
ctx->charset = charset;
ctx->charset_name = *main_charset;
*src = *main_charset;
return charset;
}
static ngx_int_t
ngx_http_source_charset(ngx_http_request_t *r, ngx_str_t *name)
{
ngx_int_t charset;
ngx_http_charset_t *charsets;
ngx_http_variable_value_t *vv;
ngx_http_charset_loc_conf_t *lcf;
ngx_http_charset_main_conf_t *mcf;
if (r->headers_out.charset.len) {
*name = r->headers_out.charset;
return ngx_http_get_charset(r, name);
}
lcf = ngx_http_get_module_loc_conf(r, ngx_http_charset_filter_module);
charset = lcf->source_charset;
if (charset == NGX_HTTP_CHARSET_OFF) {
name->len = 0;
return charset;
}
if (charset < NGX_HTTP_CHARSET_VAR) {
mcf = ngx_http_get_module_main_conf(r, ngx_http_charset_filter_module);
charsets = mcf->charsets.elts;
*name = charsets[charset].name;
return charset;
}
vv = ngx_http_get_indexed_variable(r, charset - NGX_HTTP_CHARSET_VAR);
if (vv == NULL || vv->not_found) {
return NGX_ERROR;
}
name->len = vv->len;
name->data = vv->data;
return ngx_http_get_charset(r, name);
}
static ngx_int_t
ngx_http_get_charset(ngx_http_request_t *r, ngx_str_t *name)
{
ngx_uint_t i, n;
ngx_http_charset_t *charset;
ngx_http_charset_main_conf_t *mcf;
mcf = ngx_http_get_module_main_conf(r, ngx_http_charset_filter_module);
charset = mcf->charsets.elts;
n = mcf->charsets.nelts;
for (i = 0; i < n; i++) {
if (charset[i].name.len != name->len) {
continue;
}
if (ngx_strncasecmp(charset[i].name.data, name->data, name->len) == 0) {
return i;
}
}
return NGX_HTTP_NO_CHARSET;
}
static ngx_inline void
ngx_http_set_charset(ngx_http_request_t *r, ngx_str_t *charset)
{
if (r != r->main) {
return;
}
if (r->headers_out.status == NGX_HTTP_MOVED_PERMANENTLY
|| r->headers_out.status == NGX_HTTP_MOVED_TEMPORARILY)
{
/*
* do not set charset for the redirect because NN 4.x
* use this charset instead of the next page charset
*/
r->headers_out.charset.len = 0;
return;
}
r->headers_out.charset = *charset;
}
static ngx_int_t
ngx_http_charset_ctx(ngx_http_request_t *r, ngx_http_charset_t *charsets,
ngx_int_t charset, ngx_int_t source_charset)
{
ngx_http_charset_ctx_t *ctx;
ctx = ngx_pcalloc(r->pool, sizeof(ngx_http_charset_ctx_t));
if (ctx == NULL) {
return NGX_ERROR;
}
ngx_http_set_ctx(r, ctx, ngx_http_charset_filter_module);
ctx->table = charsets[source_charset].tables[charset];
ctx->charset = charset;
ctx->charset_name = charsets[charset].name;
ctx->length = charsets[charset].length;
ctx->from_utf8 = charsets[source_charset].utf8;
ctx->to_utf8 = charsets[charset].utf8;
r->filter_need_in_memory = 1;
if ((ctx->to_utf8 || ctx->from_utf8) && r == r->main) {
ngx_http_clear_content_length(r);
} else {
r->filter_need_temporary = 1;
}
return ngx_http_next_header_filter(r);
}
static ngx_int_t
ngx_http_charset_body_filter(ngx_http_request_t *r, ngx_chain_t *in)
{
ngx_int_t rc;
ngx_buf_t *b;
ngx_chain_t *cl, *out, **ll;
ngx_http_charset_ctx_t *ctx;
ctx = ngx_http_get_module_ctx(r, ngx_http_charset_filter_module);
if (ctx == NULL || ctx->table == NULL) {
return ngx_http_next_body_filter(r, in);
}
if ((ctx->to_utf8 || ctx->from_utf8) || ctx->busy) {
out = NULL;
ll = &out;
for (cl = in; cl; cl = cl->next) {
b = cl->buf;
if (ngx_buf_size(b) == 0) {
*ll = ngx_alloc_chain_link(r->pool);
if (*ll == NULL) {
return NGX_ERROR;
}
(*ll)->buf = b;
(*ll)->next = NULL;
ll = &(*ll)->next;
continue;
}
if (ctx->to_utf8) {
*ll = ngx_http_charset_recode_to_utf8(r->pool, b, ctx);
} else {
*ll = ngx_http_charset_recode_from_utf8(r->pool, b, ctx);
}
if (*ll == NULL) {
return NGX_ERROR;
}
while (*ll) {
ll = &(*ll)->next;
}
}
rc = ngx_http_next_body_filter(r, out);
if (out) {
if (ctx->busy == NULL) {
ctx->busy = out;
} else {
for (cl = ctx->busy; cl->next; cl = cl->next) { /* void */ }
cl->next = out;
}
}
while (ctx->busy) {
cl = ctx->busy;
b = cl->buf;
if (ngx_buf_size(b) != 0) {
break;
}
ctx->busy = cl->next;
if (b->tag != (ngx_buf_tag_t) &ngx_http_charset_filter_module) {
continue;
}
if (b->shadow) {
b->shadow->pos = b->shadow->last;
}
if (b->pos) {
cl->next = ctx->free_buffers;
ctx->free_buffers = cl;
continue;
}
cl->next = ctx->free_bufs;
ctx->free_bufs = cl;
}
return rc;
}
for (cl = in; cl; cl = cl->next) {
(void) ngx_http_charset_recode(cl->buf, ctx->table);
}
return ngx_http_next_body_filter(r, in);
}
static ngx_uint_t
ngx_http_charset_recode(ngx_buf_t *b, u_char *table)
{
u_char *p, *last;
last = b->last;
for (p = b->pos; p < last; p++) {
if (*p != table[*p]) {
goto recode;
}
}
return 0;
recode:
do {
if (*p != table[*p]) {
*p = table[*p];
}
p++;
} while (p < last);
b->in_file = 0;
return 1;
}
static ngx_chain_t *
ngx_http_charset_recode_from_utf8(ngx_pool_t *pool, ngx_buf_t *buf,
ngx_http_charset_ctx_t *ctx)
{
size_t len, size;
u_char c, *p, *src, *dst, *saved, **table;
uint32_t n;
ngx_buf_t *b;
ngx_uint_t i;
ngx_chain_t *out, *cl, **ll;
src = buf->pos;
if (ctx->saved_len == 0) {
for ( /* void */ ; src < buf->last; src++) {
if (*src < 0x80) {
continue;
}
len = src - buf->pos;
if (len > 512) {
out = ngx_http_charset_get_buf(pool, ctx);
if (out == NULL) {
return NULL;
}
b = out->buf;
b->temporary = buf->temporary;
b->memory = buf->memory;
b->mmap = buf->mmap;
b->flush = buf->flush;
b->pos = buf->pos;
b->last = src;
out->buf = b;
out->next = NULL;
size = buf->last - src;
saved = src;
n = ngx_utf8_decode(&saved, size);
if (n == 0xfffffffe) {
/* incomplete UTF-8 symbol */
ngx_memcpy(ctx->saved, src, size);
ctx->saved_len = size;
b->shadow = buf;
return out;
}
} else {
out = NULL;
size = len + buf->last - src;
src = buf->pos;
}
if (size < NGX_HTML_ENTITY_LEN) {
size += NGX_HTML_ENTITY_LEN;
}
cl = ngx_http_charset_get_buffer(pool, ctx, size);
if (cl == NULL) {
return NULL;
}
if (out) {
out->next = cl;
} else {
out = cl;
}
b = cl->buf;
dst = b->pos;
goto recode;
}
out = ngx_alloc_chain_link(pool);
if (out == NULL) {
return NULL;
}
out->buf = buf;
out->next = NULL;
return out;
}
/* process incomplete UTF sequence from previous buffer */
ngx_log_debug1(NGX_LOG_DEBUG_HTTP, pool->log, 0,
"http charset utf saved: %z", ctx->saved_len);
p = src;
for (i = ctx->saved_len; i < NGX_UTF_LEN; i++) {
ctx->saved[i] = *p++;
if (p == buf->last) {
break;
}
}
saved = ctx->saved;
n = ngx_utf8_decode(&saved, i);
c = '\0';
if (n < 0x10000) {
table = (u_char **) ctx->table;
p = table[n >> 8];
if (p) {
c = p[n & 0xff];
}
} else if (n == 0xfffffffe) {
/* incomplete UTF-8 symbol */
if (i < NGX_UTF_LEN) {
out = ngx_http_charset_get_buf(pool, ctx);
if (out == NULL) {
return NULL;
}
b = out->buf;
b->pos = buf->pos;
b->last = buf->last;
b->sync = 1;
b->shadow = buf;
ngx_memcpy(&ctx->saved[ctx->saved_len], src, i);
ctx->saved_len += i;
return out;
}
}
size = buf->last - buf->pos;
if (size < NGX_HTML_ENTITY_LEN) {
size += NGX_HTML_ENTITY_LEN;
}
cl = ngx_http_charset_get_buffer(pool, ctx, size);
if (cl == NULL) {
return NULL;
}
out = cl;
b = cl->buf;
dst = b->pos;
if (c) {
*dst++ = c;
} else if (n == 0xfffffffe) {
*dst++ = '?';
ngx_log_debug0(NGX_LOG_DEBUG_HTTP, pool->log, 0,
"http charset invalid utf 0");
saved = &ctx->saved[NGX_UTF_LEN];
} else if (n > 0x10ffff) {
*dst++ = '?';
ngx_log_debug0(NGX_LOG_DEBUG_HTTP, pool->log, 0,
"http charset invalid utf 1");
} else {
dst = ngx_sprintf(dst, "&#%uD;", n);
}
src += (saved - ctx->saved) - ctx->saved_len;
ctx->saved_len = 0;
recode:
ll = &cl->next;
table = (u_char **) ctx->table;
while (src < buf->last) {
if ((size_t) (b->end - dst) < NGX_HTML_ENTITY_LEN) {
b->last = dst;
size = buf->last - src + NGX_HTML_ENTITY_LEN;
cl = ngx_http_charset_get_buffer(pool, ctx, size);
if (cl == NULL) {
return NULL;
}
*ll = cl;
ll = &cl->next;
b = cl->buf;
dst = b->pos;
}
if (*src < 0x80) {
*dst++ = *src++;
continue;
}
len = buf->last - src;
n = ngx_utf8_decode(&src, len);
if (n < 0x10000) {
p = table[n >> 8];
if (p) {
c = p[n & 0xff];
if (c) {
*dst++ = c;
continue;
}
}
dst = ngx_sprintf(dst, "&#%uD;", n);
continue;
}
if (n == 0xfffffffe) {
/* incomplete UTF-8 symbol */
ngx_memcpy(ctx->saved, src, len);
ctx->saved_len = len;
if (b->pos == dst) {
b->sync = 1;
b->temporary = 0;
}
break;
}
if (n > 0x10ffff) {
*dst++ = '?';
ngx_log_debug0(NGX_LOG_DEBUG_HTTP, pool->log, 0,
"http charset invalid utf 2");
continue;
}
/* n > 0xffff */
dst = ngx_sprintf(dst, "&#%uD;", n);
}
b->last = dst;
b->last_buf = buf->last_buf;
b->last_in_chain = buf->last_in_chain;
b->flush = buf->flush;
b->shadow = buf;
return out;
}
static ngx_chain_t *
ngx_http_charset_recode_to_utf8(ngx_pool_t *pool, ngx_buf_t *buf,
ngx_http_charset_ctx_t *ctx)
{
size_t len, size;
u_char *p, *src, *dst, *table;
ngx_buf_t *b;
ngx_chain_t *out, *cl, **ll;
table = ctx->table;
for (src = buf->pos; src < buf->last; src++) {
if (table[*src * NGX_UTF_LEN] == '\1') {
continue;
}
goto recode;
}
out = ngx_alloc_chain_link(pool);
if (out == NULL) {
return NULL;
}
out->buf = buf;
out->next = NULL;
return out;
recode:
/*
* we assume that there are about half of characters to be recoded,
* so we preallocate "size / 2 + size / 2 * ctx->length"
*/
len = src - buf->pos;
if (len > 512) {
out = ngx_http_charset_get_buf(pool, ctx);
if (out == NULL) {
return NULL;
}
b = out->buf;
b->temporary = buf->temporary;
b->memory = buf->memory;
b->mmap = buf->mmap;
b->flush = buf->flush;
b->pos = buf->pos;
b->last = src;
out->buf = b;
out->next = NULL;
size = buf->last - src;
size = size / 2 + size / 2 * ctx->length;
} else {
out = NULL;
size = buf->last - src;
size = len + size / 2 + size / 2 * ctx->length;
src = buf->pos;
}
cl = ngx_http_charset_get_buffer(pool, ctx, size);
if (cl == NULL) {
return NULL;
}
if (out) {
out->next = cl;
} else {
out = cl;
}
ll = &cl->next;
b = cl->buf;
dst = b->pos;
while (src < buf->last) {
p = &table[*src++ * NGX_UTF_LEN];
len = *p++;
if ((size_t) (b->end - dst) < len) {
b->last = dst;
size = buf->last - src;
size = len + size / 2 + size / 2 * ctx->length;
cl = ngx_http_charset_get_buffer(pool, ctx, size);
if (cl == NULL) {
return NULL;
}
*ll = cl;
ll = &cl->next;
b = cl->buf;
dst = b->pos;
}
while (len) {
*dst++ = *p++;
len--;
}
}
b->last = dst;
b->last_buf = buf->last_buf;
b->last_in_chain = buf->last_in_chain;
b->flush = buf->flush;
b->shadow = buf;
return out;
}
static ngx_chain_t *
ngx_http_charset_get_buf(ngx_pool_t *pool, ngx_http_charset_ctx_t *ctx)
{
ngx_chain_t *cl;
cl = ctx->free_bufs;
if (cl) {
ctx->free_bufs = cl->next;
cl->buf->shadow = NULL;
cl->next = NULL;
return cl;
}
cl = ngx_alloc_chain_link(pool);
if (cl == NULL) {
return NULL;
}
cl->buf = ngx_calloc_buf(pool);
if (cl->buf == NULL) {
return NULL;
}
cl->next = NULL;
cl->buf->tag = (ngx_buf_tag_t) &ngx_http_charset_filter_module;
return cl;
}
static ngx_chain_t *
ngx_http_charset_get_buffer(ngx_pool_t *pool, ngx_http_charset_ctx_t *ctx,
size_t size)
{
ngx_buf_t *b;
ngx_chain_t *cl, **ll;
for (ll = &ctx->free_buffers, cl = ctx->free_buffers;
cl;
ll = &cl->next, cl = cl->next)
{
b = cl->buf;
if ((size_t) (b->end - b->start) >= size) {
*ll = cl->next;
cl->next = NULL;
b->pos = b->start;
b->temporary = 1;
b->shadow = NULL;
return cl;
}
}
cl = ngx_alloc_chain_link(pool);
if (cl == NULL) {
return NULL;
}
cl->buf = ngx_create_temp_buf(pool, size);
if (cl->buf == NULL) {
return NULL;
}
cl->next = NULL;
cl->buf->temporary = 1;
cl->buf->tag = (ngx_buf_tag_t) &ngx_http_charset_filter_module;
return cl;
}
static char *
ngx_http_charset_map_block(ngx_conf_t *cf, ngx_command_t *cmd, void *conf)
{
ngx_http_charset_main_conf_t *mcf = conf;
char *rv;
u_char *p, *dst2src, **pp;
ngx_int_t src, dst;
ngx_uint_t i, n;
ngx_str_t *value;
ngx_conf_t pvcf;
ngx_http_charset_t *charset;
ngx_http_charset_tables_t *table;
ngx_http_charset_conf_ctx_t ctx;
value = cf->args->elts;
src = ngx_http_add_charset(&mcf->charsets, &value[1]);
if (src == NGX_ERROR) {
return NGX_CONF_ERROR;
}
dst = ngx_http_add_charset(&mcf->charsets, &value[2]);
if (dst == NGX_ERROR) {
return NGX_CONF_ERROR;
}
if (src == dst) {
ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,
"\"charset_map\" between the same charsets "
"\"%V\" and \"%V\"", &value[1], &value[2]);
return NGX_CONF_ERROR;
}
table = mcf->tables.elts;
for (i = 0; i < mcf->tables.nelts; i++) {
if ((src == table->src && dst == table->dst)
|| (src == table->dst && dst == table->src))
{
ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,
"duplicate \"charset_map\" between "
"\"%V\" and \"%V\"", &value[1], &value[2]);
return NGX_CONF_ERROR;
}
}
table = ngx_array_push(&mcf->tables);
if (table == NULL) {
return NGX_CONF_ERROR;
}
table->src = src;
table->dst = dst;
if (ngx_strcasecmp(value[2].data, (u_char *) "utf-8") == 0) {
table->src2dst = ngx_pcalloc(cf->pool, 256 * NGX_UTF_LEN);
if (table->src2dst == NULL) {
return NGX_CONF_ERROR;
}
table->dst2src = ngx_pcalloc(cf->pool, 256 * sizeof(void *));
if (table->dst2src == NULL) {
return NGX_CONF_ERROR;
}
dst2src = ngx_pcalloc(cf->pool, 256);
if (dst2src == NULL) {
return NGX_CONF_ERROR;
}
pp = (u_char **) &table->dst2src[0];
pp[0] = dst2src;
for (i = 0; i < 128; i++) {
p = &table->src2dst[i * NGX_UTF_LEN];
p[0] = '\1';
p[1] = (u_char) i;
dst2src[i] = (u_char) i;
}
for (/* void */; i < 256; i++) {
p = &table->src2dst[i * NGX_UTF_LEN];
p[0] = '\1';
p[1] = '?';
}
} else {
table->src2dst = ngx_palloc(cf->pool, 256);
if (table->src2dst == NULL) {
return NGX_CONF_ERROR;
}
table->dst2src = ngx_palloc(cf->pool, 256);
if (table->dst2src == NULL) {
return NGX_CONF_ERROR;
}
for (i = 0; i < 128; i++) {
table->src2dst[i] = (u_char) i;
table->dst2src[i] = (u_char) i;
}
for (/* void */; i < 256; i++) {
table->src2dst[i] = '?';
table->dst2src[i] = '?';
}
}
charset = mcf->charsets.elts;
ctx.table = table;
ctx.charset = &charset[dst];
ctx.characters = 0;
pvcf = *cf;
cf->ctx = &ctx;
cf->handler = ngx_http_charset_map;
cf->handler_conf = conf;
rv = ngx_conf_parse(cf, NULL);
*cf = pvcf;
if (ctx.characters) {
n = ctx.charset->length;
ctx.charset->length /= ctx.characters;
if (((n * 10) / ctx.characters) % 10 > 4) {
ctx.charset->length++;
}
}
return rv;
}
static char *
ngx_http_charset_map(ngx_conf_t *cf, ngx_command_t *dummy, void *conf)
{
u_char *p, *dst2src, **pp;
uint32_t n;
ngx_int_t src, dst;
ngx_str_t *value;
ngx_uint_t i;
ngx_http_charset_tables_t *table;
ngx_http_charset_conf_ctx_t *ctx;
if (cf->args->nelts != 2) {
ngx_conf_log_error(NGX_LOG_EMERG, cf, 0, "invalid parameters number");
return NGX_CONF_ERROR;
}
value = cf->args->elts;
src = ngx_hextoi(value[0].data, value[0].len);
if (src == NGX_ERROR || src > 255) {
ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,
"invalid value \"%V\"", &value[0]);
return NGX_CONF_ERROR;
}
ctx = cf->ctx;
table = ctx->table;
if (ctx->charset->utf8) {
p = &table->src2dst[src * NGX_UTF_LEN];
*p++ = (u_char) (value[1].len / 2);
for (i = 0; i < value[1].len; i += 2) {
dst = ngx_hextoi(&value[1].data[i], 2);
if (dst == NGX_ERROR || dst > 255) {
ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,
"invalid value \"%V\"", &value[1]);
return NGX_CONF_ERROR;
}
*p++ = (u_char) dst;
}
i /= 2;
ctx->charset->length += i;
ctx->characters++;
p = &table->src2dst[src * NGX_UTF_LEN] + 1;
n = ngx_utf8_decode(&p, i);
if (n > 0xffff) {
ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,
"invalid value \"%V\"", &value[1]);
return NGX_CONF_ERROR;
}
pp = (u_char **) &table->dst2src[0];
dst2src = pp[n >> 8];
if (dst2src == NULL) {
dst2src = ngx_pcalloc(cf->pool, 256);
if (dst2src == NULL) {
return NGX_CONF_ERROR;
}
pp[n >> 8] = dst2src;
}
dst2src[n & 0xff] = (u_char) src;
} else {
dst = ngx_hextoi(value[1].data, value[1].len);
if (dst == NGX_ERROR || dst > 255) {
ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,
"invalid value \"%V\"", &value[1]);
return NGX_CONF_ERROR;
}
table->src2dst[src] = (u_char) dst;
table->dst2src[dst] = (u_char) src;
}
return NGX_CONF_OK;
}
static char *
ngx_http_set_charset_slot(ngx_conf_t *cf, ngx_command_t *cmd, void *conf)
{
char *p = conf;
ngx_int_t *cp;
ngx_str_t *value, var;
ngx_http_charset_main_conf_t *mcf;
cp = (ngx_int_t *) (p + cmd->offset);
if (*cp != NGX_CONF_UNSET) {
return "is duplicate";
}
value = cf->args->elts;
if (cmd->offset == offsetof(ngx_http_charset_loc_conf_t, charset)
&& ngx_strcmp(value[1].data, "off") == 0)
{
*cp = NGX_HTTP_CHARSET_OFF;
return NGX_CONF_OK;
}
if (value[1].data[0] == '$') {
var.len = value[1].len - 1;
var.data = value[1].data + 1;
*cp = ngx_http_get_variable_index(cf, &var);
if (*cp == NGX_ERROR) {
return NGX_CONF_ERROR;
}
*cp += NGX_HTTP_CHARSET_VAR;
return NGX_CONF_OK;
}
mcf = ngx_http_conf_get_module_main_conf(cf,
ngx_http_charset_filter_module);
*cp = ngx_http_add_charset(&mcf->charsets, &value[1]);
if (*cp == NGX_ERROR) {
return NGX_CONF_ERROR;
}
return NGX_CONF_OK;
}
static ngx_int_t
ngx_http_add_charset(ngx_array_t *charsets, ngx_str_t *name)
{
ngx_uint_t i;
ngx_http_charset_t *c;
c = charsets->elts;
for (i = 0; i < charsets->nelts; i++) {
if (name->len != c[i].name.len) {
continue;
}
if (ngx_strcasecmp(name->data, c[i].name.data) == 0) {
break;
}
}
if (i < charsets->nelts) {
return i;
}
c = ngx_array_push(charsets);
if (c == NULL) {
return NGX_ERROR;
}
c->tables = NULL;
c->name = *name;
c->length = 0;
if (ngx_strcasecmp(name->data, (u_char *) "utf-8") == 0) {
c->utf8 = 1;
} else {
c->utf8 = 0;
}
return i;
}
static void *
ngx_http_charset_create_main_conf(ngx_conf_t *cf)
{
ngx_http_charset_main_conf_t *mcf;
mcf = ngx_pcalloc(cf->pool, sizeof(ngx_http_charset_main_conf_t));
if (mcf == NULL) {
return NULL;
}
if (ngx_array_init(&mcf->charsets, cf->pool, 2, sizeof(ngx_http_charset_t))
!= NGX_OK)
{
return NULL;
}
if (ngx_array_init(&mcf->tables, cf->pool, 1,
sizeof(ngx_http_charset_tables_t))
!= NGX_OK)
{
return NULL;
}
if (ngx_array_init(&mcf->recodes, cf->pool, 2,
sizeof(ngx_http_charset_recode_t))
!= NGX_OK)
{
return NULL;
}
return mcf;
}
static void *
ngx_http_charset_create_loc_conf(ngx_conf_t *cf)
{
ngx_http_charset_loc_conf_t *lcf;
lcf = ngx_pcalloc(cf->pool, sizeof(ngx_http_charset_loc_conf_t));
if (lcf == NULL) {
return NULL;
}
/*
* set by ngx_pcalloc():
*
* lcf->types = { NULL };
* lcf->types_keys = NULL;
*/
lcf->charset = NGX_CONF_UNSET;
lcf->source_charset = NGX_CONF_UNSET;
lcf->override_charset = NGX_CONF_UNSET;
return lcf;
}
static char *
ngx_http_charset_merge_loc_conf(ngx_conf_t *cf, void *parent, void *child)
{
ngx_http_charset_loc_conf_t *prev = parent;
ngx_http_charset_loc_conf_t *conf = child;
ngx_uint_t i;
ngx_http_charset_recode_t *recode;
ngx_http_charset_main_conf_t *mcf;
if (ngx_http_merge_types(cf, &conf->types_keys, &conf->types,
&prev->types_keys, &prev->types,
ngx_http_charset_default_types)
!= NGX_OK)
{
return NGX_CONF_ERROR;
}
ngx_conf_merge_value(conf->override_charset, prev->override_charset, 0);
ngx_conf_merge_value(conf->charset, prev->charset, NGX_HTTP_CHARSET_OFF);
ngx_conf_merge_value(conf->source_charset, prev->source_charset,
NGX_HTTP_CHARSET_OFF);
if (conf->charset == NGX_HTTP_CHARSET_OFF
|| conf->source_charset == NGX_HTTP_CHARSET_OFF
|| conf->charset == conf->source_charset)
{
return NGX_CONF_OK;
}
if (conf->source_charset >= NGX_HTTP_CHARSET_VAR
|| conf->charset >= NGX_HTTP_CHARSET_VAR)
{
return NGX_CONF_OK;
}
mcf = ngx_http_conf_get_module_main_conf(cf,
ngx_http_charset_filter_module);
recode = mcf->recodes.elts;
for (i = 0; i < mcf->recodes.nelts; i++) {
if (conf->source_charset == recode[i].src
&& conf->charset == recode[i].dst)
{
return NGX_CONF_OK;
}
}
recode = ngx_array_push(&mcf->recodes);
if (recode == NULL) {
return NGX_CONF_ERROR;
}
recode->src = conf->source_charset;
recode->dst = conf->charset;
return NGX_CONF_OK;
}
static ngx_int_t
ngx_http_charset_postconfiguration(ngx_conf_t *cf)
{
u_char **src, **dst;
ngx_int_t c;
ngx_uint_t i, t;
ngx_http_charset_t *charset;
ngx_http_charset_recode_t *recode;
ngx_http_charset_tables_t *tables;
ngx_http_charset_main_conf_t *mcf;
mcf = ngx_http_conf_get_module_main_conf(cf,
ngx_http_charset_filter_module);
recode = mcf->recodes.elts;
tables = mcf->tables.elts;
charset = mcf->charsets.elts;
for (i = 0; i < mcf->recodes.nelts; i++) {
c = recode[i].src;
for (t = 0; t < mcf->tables.nelts; t++) {
if (c == tables[t].src && recode[i].dst == tables[t].dst) {
goto next;
}
if (c == tables[t].dst && recode[i].dst == tables[t].src) {
goto next;
}
}
ngx_log_error(NGX_LOG_EMERG, cf->log, 0,
"no \"charset_map\" between the charsets \"%V\" and \"%V\"",
&charset[c].name, &charset[recode[i].dst].name);
return NGX_ERROR;
next:
continue;
}
for (t = 0; t < mcf->tables.nelts; t++) {
src = charset[tables[t].src].tables;
if (src == NULL) {
src = ngx_pcalloc(cf->pool, sizeof(u_char *) * mcf->charsets.nelts);
if (src == NULL) {
return NGX_ERROR;
}
charset[tables[t].src].tables = src;
}
dst = charset[tables[t].dst].tables;
if (dst == NULL) {
dst = ngx_pcalloc(cf->pool, sizeof(u_char *) * mcf->charsets.nelts);
if (dst == NULL) {
return NGX_ERROR;
}
charset[tables[t].dst].tables = dst;
}
src[tables[t].dst] = tables[t].src2dst;
dst[tables[t].src] = tables[t].dst2src;
}
ngx_http_next_header_filter = ngx_http_top_header_filter;
ngx_http_top_header_filter = ngx_http_charset_header_filter;
ngx_http_next_body_filter = ngx_http_top_body_filter;
ngx_http_top_body_filter = ngx_http_charset_body_filter;
return NGX_OK;
}