2003-11-26 04:44:56 +08:00
|
|
|
|
2004-09-28 16:34:51 +08:00
|
|
|
/*
|
2004-09-30 00:00:49 +08:00
|
|
|
* Copyright (C) Igor Sysoev
|
2004-09-28 16:34:51 +08:00
|
|
|
*/
|
|
|
|
|
|
|
|
|
2003-11-26 04:44:56 +08:00
|
|
|
#include <ngx_config.h>
|
|
|
|
#include <ngx_core.h>
|
|
|
|
|
|
|
|
|
2005-03-23 00:02:46 +08:00
|
|
|
static void * ngx_libc_cdecl ngx_regex_malloc(size_t size);
|
|
|
|
static void ngx_libc_cdecl ngx_regex_free(void *p);
|
2003-11-26 04:44:56 +08:00
|
|
|
|
|
|
|
|
|
|
|
static ngx_pool_t *ngx_pcre_pool;
|
|
|
|
|
|
|
|
|
2005-03-23 00:02:46 +08:00
|
|
|
void
|
|
|
|
ngx_regex_init(void)
|
2003-11-26 04:44:56 +08:00
|
|
|
{
|
|
|
|
pcre_malloc = ngx_regex_malloc;
|
|
|
|
pcre_free = ngx_regex_free;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-11-16 20:19:02 +08:00
|
|
|
static ngx_inline void
|
|
|
|
ngx_regex_malloc_init(ngx_pool_t *pool)
|
2003-11-26 04:44:56 +08:00
|
|
|
{
|
2004-07-07 23:01:00 +08:00
|
|
|
#if (NGX_THREADS)
|
|
|
|
ngx_core_tls_t *tls;
|
|
|
|
|
|
|
|
if (ngx_threaded) {
|
|
|
|
tls = ngx_thread_get_tls(ngx_core_tls_key);
|
|
|
|
tls->pool = pool;
|
2009-11-16 20:19:02 +08:00
|
|
|
return;
|
2004-07-07 23:01:00 +08:00
|
|
|
}
|
|
|
|
|
2009-11-17 18:24:45 +08:00
|
|
|
#endif
|
2003-11-26 04:44:56 +08:00
|
|
|
|
|
|
|
ngx_pcre_pool = pool;
|
2009-11-16 20:19:02 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static ngx_inline void
|
|
|
|
ngx_regex_malloc_done(void)
|
|
|
|
{
|
|
|
|
#if (NGX_THREADS)
|
|
|
|
ngx_core_tls_t *tls;
|
|
|
|
|
|
|
|
if (ngx_threaded) {
|
|
|
|
tls = ngx_thread_get_tls(ngx_core_tls_key);
|
|
|
|
tls->pool = NULL;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-11-17 18:24:45 +08:00
|
|
|
#endif
|
2009-11-16 20:19:02 +08:00
|
|
|
|
|
|
|
ngx_pcre_pool = NULL;
|
|
|
|
}
|
2004-07-07 23:01:00 +08:00
|
|
|
|
2009-11-16 20:19:02 +08:00
|
|
|
|
|
|
|
ngx_int_t
|
|
|
|
ngx_regex_compile(ngx_regex_compile_t *rc)
|
|
|
|
{
|
|
|
|
int n, erroff;
|
|
|
|
char *p;
|
|
|
|
const char *errstr;
|
|
|
|
ngx_regex_t *re;
|
|
|
|
|
|
|
|
ngx_regex_malloc_init(rc->pool);
|
|
|
|
|
|
|
|
re = pcre_compile((const char *) rc->pattern.data, (int) rc->options,
|
2004-03-16 15:10:12 +08:00
|
|
|
&errstr, &erroff, NULL);
|
2003-11-26 04:44:56 +08:00
|
|
|
|
2009-11-16 20:19:02 +08:00
|
|
|
/* ensure that there is no current pool */
|
|
|
|
ngx_regex_malloc_done();
|
|
|
|
|
2003-11-26 04:44:56 +08:00
|
|
|
if (re == NULL) {
|
2009-11-16 20:19:02 +08:00
|
|
|
if ((size_t) erroff == rc->pattern.len) {
|
|
|
|
rc->err.len = ngx_snprintf(rc->err.data, rc->err.len,
|
|
|
|
"pcre_compile() failed: %s in \"%V\"",
|
|
|
|
errstr, &rc->pattern)
|
|
|
|
- rc->err.data;
|
|
|
|
|
2003-11-26 04:44:56 +08:00
|
|
|
} else {
|
2009-11-16 20:19:02 +08:00
|
|
|
rc->err.len = ngx_snprintf(rc->err.data, rc->err.len,
|
|
|
|
"pcre_compile() failed: %s in \"%V\" at \"%s\"",
|
|
|
|
errstr, &rc->pattern, rc->pattern.data + erroff)
|
|
|
|
- rc->err.data;
|
2003-11-26 04:44:56 +08:00
|
|
|
}
|
2009-11-16 20:19:02 +08:00
|
|
|
|
|
|
|
return NGX_ERROR;
|
2003-11-26 04:44:56 +08:00
|
|
|
}
|
|
|
|
|
2009-11-16 20:19:02 +08:00
|
|
|
rc->regex = re;
|
2004-03-23 14:01:52 +08:00
|
|
|
|
2009-11-16 20:19:02 +08:00
|
|
|
n = pcre_fullinfo(re, NULL, PCRE_INFO_CAPTURECOUNT, &rc->captures);
|
|
|
|
if (n < 0) {
|
|
|
|
p = "pcre_fullinfo(\"%V\", PCRE_INFO_CAPTURECOUNT) failed: %d";
|
|
|
|
goto failed;
|
2004-07-07 23:01:00 +08:00
|
|
|
}
|
2004-03-23 14:01:52 +08:00
|
|
|
|
2009-11-16 20:19:02 +08:00
|
|
|
if (rc->captures == 0) {
|
|
|
|
return NGX_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
n = pcre_fullinfo(re, NULL, PCRE_INFO_NAMECOUNT, &rc->named_captures);
|
|
|
|
if (n < 0) {
|
|
|
|
p = "pcre_fullinfo(\"%V\", PCRE_INFO_NAMECOUNT) failed: %d";
|
|
|
|
goto failed;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rc->named_captures == 0) {
|
|
|
|
return NGX_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
n = pcre_fullinfo(re, NULL, PCRE_INFO_NAMEENTRYSIZE, &rc->name_size);
|
|
|
|
if (n < 0) {
|
|
|
|
p = "pcre_fullinfo(\"%V\", PCRE_INFO_NAMEENTRYSIZE) failed: %d";
|
|
|
|
goto failed;
|
|
|
|
}
|
|
|
|
|
|
|
|
n = pcre_fullinfo(re, NULL, PCRE_INFO_NAMETABLE, &rc->names);
|
|
|
|
if (n < 0) {
|
|
|
|
p = "pcre_fullinfo(\"%V\", PCRE_INFO_NAMETABLE) failed: %d";
|
|
|
|
goto failed;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NGX_OK;
|
|
|
|
|
|
|
|
failed:
|
|
|
|
|
|
|
|
rc->err.len = ngx_snprintf(rc->err.data, rc->err.len, p, &rc->pattern, n)
|
|
|
|
- rc->err.data;
|
|
|
|
return NGX_OK;
|
2003-11-26 04:44:56 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-03-23 00:02:46 +08:00
|
|
|
ngx_int_t
|
|
|
|
ngx_regex_capture_count(ngx_regex_t *re)
|
2005-01-19 21:10:56 +08:00
|
|
|
{
|
|
|
|
int rc, n;
|
|
|
|
|
|
|
|
n = 0;
|
|
|
|
|
|
|
|
rc = pcre_fullinfo(re, NULL, PCRE_INFO_CAPTURECOUNT, &n);
|
|
|
|
|
2005-03-19 20:38:37 +08:00
|
|
|
if (rc < 0) {
|
|
|
|
return (ngx_int_t) rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (ngx_int_t) n;
|
2005-01-19 21:10:56 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-12-27 21:15:08 +08:00
|
|
|
ngx_int_t
|
|
|
|
ngx_regex_exec_array(ngx_array_t *a, ngx_str_t *s, ngx_log_t *log)
|
|
|
|
{
|
|
|
|
ngx_int_t n;
|
|
|
|
ngx_uint_t i;
|
|
|
|
ngx_regex_elt_t *re;
|
|
|
|
|
|
|
|
re = a->elts;
|
|
|
|
|
|
|
|
for (i = 0; i < a->nelts; i++) {
|
|
|
|
|
|
|
|
n = ngx_regex_exec(re[i].regex, s, NULL, 0);
|
|
|
|
|
|
|
|
if (n == NGX_REGEX_NO_MATCHED) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (n < 0) {
|
|
|
|
ngx_log_error(NGX_LOG_ALERT, log, 0,
|
2009-11-16 20:19:02 +08:00
|
|
|
ngx_regex_exec_n " failed: %i on \"%V\" using \"%s\"",
|
2007-12-27 21:15:08 +08:00
|
|
|
n, s, re[i].name);
|
|
|
|
return NGX_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* match */
|
|
|
|
|
|
|
|
return NGX_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NGX_DECLINED;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-03-23 00:02:46 +08:00
|
|
|
static void * ngx_libc_cdecl
|
|
|
|
ngx_regex_malloc(size_t size)
|
2003-11-26 04:44:56 +08:00
|
|
|
{
|
2004-07-07 23:01:00 +08:00
|
|
|
ngx_pool_t *pool;
|
|
|
|
#if (NGX_THREADS)
|
|
|
|
ngx_core_tls_t *tls;
|
|
|
|
|
|
|
|
if (ngx_threaded) {
|
|
|
|
tls = ngx_thread_get_tls(ngx_core_tls_key);
|
|
|
|
pool = tls->pool;
|
2009-11-16 20:19:02 +08:00
|
|
|
|
2004-07-07 23:01:00 +08:00
|
|
|
} else {
|
|
|
|
pool = ngx_pcre_pool;
|
|
|
|
}
|
2009-11-16 20:19:02 +08:00
|
|
|
|
2004-07-07 23:01:00 +08:00
|
|
|
#else
|
2009-11-16 20:19:02 +08:00
|
|
|
|
2004-07-07 23:01:00 +08:00
|
|
|
pool = ngx_pcre_pool;
|
2009-11-16 20:19:02 +08:00
|
|
|
|
2004-07-07 23:01:00 +08:00
|
|
|
#endif
|
|
|
|
|
|
|
|
if (pool) {
|
2008-06-17 23:00:30 +08:00
|
|
|
return ngx_palloc(pool, size);
|
2004-03-23 14:01:52 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
2003-11-26 04:44:56 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-03-23 00:02:46 +08:00
|
|
|
static void ngx_libc_cdecl
|
|
|
|
ngx_regex_free(void *p)
|
2003-11-26 04:44:56 +08:00
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|