src/http/modules/ngx_http_map_module.c - nginx source code

Global variables defined

Data types defined

Functions defined

Source code


  1. /*
  2. * Copyright (C) Igor Sysoev
  3. * Copyright (C) Nginx, Inc.
  4. */


  5. #include <ngx_config.h>
  6. #include <ngx_core.h>
  7. #include <ngx_http.h>


  8. typedef struct {
  9.     ngx_uint_t                  hash_max_size;
  10.     ngx_uint_t                  hash_bucket_size;
  11. } ngx_http_map_conf_t;


  12. typedef struct {
  13.     ngx_hash_keys_arrays_t      keys;

  14.     ngx_array_t                *values_hash;
  15. #if (NGX_PCRE)
  16.     ngx_array_t                 regexes;
  17. #endif

  18.     ngx_http_variable_value_t  *default_value;
  19.     ngx_conf_t                 *cf;
  20.     unsigned                    hostnames:1;
  21.     unsigned                    no_cacheable:1;
  22. } ngx_http_map_conf_ctx_t;


  23. typedef struct {
  24.     ngx_http_map_t              map;
  25.     ngx_http_complex_value_t    value;
  26.     ngx_http_variable_value_t  *default_value;
  27.     ngx_uint_t                  hostnames;      /* unsigned  hostnames:1 */
  28. } ngx_http_map_ctx_t;


  29. static int ngx_libc_cdecl ngx_http_map_cmp_dns_wildcards(const void *one,
  30.     const void *two);
  31. static void *ngx_http_map_create_conf(ngx_conf_t *cf);
  32. static char *ngx_http_map_block(ngx_conf_t *cf, ngx_command_t *cmd, void *conf);
  33. static char *ngx_http_map(ngx_conf_t *cf, ngx_command_t *dummy, void *conf);


  34. static ngx_command_t  ngx_http_map_commands[] = {

  35.     { ngx_string("map"),
  36.       NGX_HTTP_MAIN_CONF|NGX_CONF_BLOCK|NGX_CONF_TAKE2,
  37.       ngx_http_map_block,
  38.       NGX_HTTP_MAIN_CONF_OFFSET,
  39.       0,
  40.       NULL },

  41.     { ngx_string("map_hash_max_size"),
  42.       NGX_HTTP_MAIN_CONF|NGX_CONF_TAKE1,
  43.       ngx_conf_set_num_slot,
  44.       NGX_HTTP_MAIN_CONF_OFFSET,
  45.       offsetof(ngx_http_map_conf_t, hash_max_size),
  46.       NULL },

  47.     { ngx_string("map_hash_bucket_size"),
  48.       NGX_HTTP_MAIN_CONF|NGX_CONF_TAKE1,
  49.       ngx_conf_set_num_slot,
  50.       NGX_HTTP_MAIN_CONF_OFFSET,
  51.       offsetof(ngx_http_map_conf_t, hash_bucket_size),
  52.       NULL },

  53.       ngx_null_command
  54. };


  55. static ngx_http_module_t  ngx_http_map_module_ctx = {
  56.     NULL,                                  /* preconfiguration */
  57.     NULL,                                  /* postconfiguration */

  58.     ngx_http_map_create_conf,              /* create main configuration */
  59.     NULL,                                  /* init main configuration */

  60.     NULL,                                  /* create server configuration */
  61.     NULL,                                  /* merge server configuration */

  62.     NULL,                                  /* create location configuration */
  63.     NULL                                   /* merge location configuration */
  64. };


  65. ngx_module_t  ngx_http_map_module = {
  66.     NGX_MODULE_V1,
  67.     &ngx_http_map_module_ctx,              /* module context */
  68.     ngx_http_map_commands,                 /* module directives */
  69.     NGX_HTTP_MODULE,                       /* module type */
  70.     NULL,                                  /* init master */
  71.     NULL,                                  /* init module */
  72.     NULL,                                  /* init process */
  73.     NULL,                                  /* init thread */
  74.     NULL,                                  /* exit thread */
  75.     NULL,                                  /* exit process */
  76.     NULL,                                  /* exit master */
  77.     NGX_MODULE_V1_PADDING
  78. };


  79. static ngx_int_t
  80. ngx_http_map_variable(ngx_http_request_t *r, ngx_http_variable_value_t *v,
  81.     uintptr_t data)
  82. {
  83.     ngx_http_map_ctx_t  *map = (ngx_http_map_ctx_t *) data;

  84.     ngx_str_t                   val, str;
  85.     ngx_http_complex_value_t   *cv;
  86.     ngx_http_variable_value_t  *value;

  87.     ngx_log_debug0(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
  88.                    "http map started");

  89.     if (ngx_http_complex_value(r, &map->value, &val) != NGX_OK) {
  90.         return NGX_ERROR;
  91.     }

  92.     if (map->hostnames && val.len > 0 && val.data[val.len - 1] == '.') {
  93.         val.len--;
  94.     }

  95.     value = ngx_http_map_find(r, &map->map, &val);

  96.     if (value == NULL) {
  97.         value = map->default_value;
  98.     }

  99.     if (!value->valid) {
  100.         cv = (ngx_http_complex_value_t *) value->data;

  101.         if (ngx_http_complex_value(r, cv, &str) != NGX_OK) {
  102.             return NGX_ERROR;
  103.         }

  104.         v->valid = 1;
  105.         v->no_cacheable = 0;
  106.         v->not_found = 0;
  107.         v->len = str.len;
  108.         v->data = str.data;

  109.     } else {
  110.         *v = *value;
  111.     }

  112.     ngx_log_debug2(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
  113.                    "http map: \"%V\" \"%v\"", &val, v);

  114.     return NGX_OK;
  115. }


  116. static void *
  117. ngx_http_map_create_conf(ngx_conf_t *cf)
  118. {
  119.     ngx_http_map_conf_t  *mcf;

  120.     mcf = ngx_palloc(cf->pool, sizeof(ngx_http_map_conf_t));
  121.     if (mcf == NULL) {
  122.         return NULL;
  123.     }

  124.     mcf->hash_max_size = NGX_CONF_UNSET_UINT;
  125.     mcf->hash_bucket_size = NGX_CONF_UNSET_UINT;

  126.     return mcf;
  127. }


  128. static char *
  129. ngx_http_map_block(ngx_conf_t *cf, ngx_command_t *cmd, void *conf)
  130. {
  131.     ngx_http_map_conf_t  *mcf = conf;

  132.     char                              *rv;
  133.     ngx_str_t                         *value, name;
  134.     ngx_conf_t                         save;
  135.     ngx_pool_t                        *pool;
  136.     ngx_hash_init_t                    hash;
  137.     ngx_http_map_ctx_t                *map;
  138.     ngx_http_variable_t               *var;
  139.     ngx_http_map_conf_ctx_t            ctx;
  140.     ngx_http_compile_complex_value_t   ccv;

  141.     if (mcf->hash_max_size == NGX_CONF_UNSET_UINT) {
  142.         mcf->hash_max_size = 2048;
  143.     }

  144.     if (mcf->hash_bucket_size == NGX_CONF_UNSET_UINT) {
  145.         mcf->hash_bucket_size = ngx_cacheline_size;

  146.     } else {
  147.         mcf->hash_bucket_size = ngx_align(mcf->hash_bucket_size,
  148.                                           ngx_cacheline_size);
  149.     }

  150.     map = ngx_pcalloc(cf->pool, sizeof(ngx_http_map_ctx_t));
  151.     if (map == NULL) {
  152.         return NGX_CONF_ERROR;
  153.     }

  154.     value = cf->args->elts;

  155.     ngx_memzero(&ccv, sizeof(ngx_http_compile_complex_value_t));

  156.     ccv.cf = cf;
  157.     ccv.value = &value[1];
  158.     ccv.complex_value = &map->value;

  159.     if (ngx_http_compile_complex_value(&ccv) != NGX_OK) {
  160.         return NGX_CONF_ERROR;
  161.     }

  162.     name = value[2];

  163.     if (name.data[0] != '$') {
  164.         ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,
  165.                            "invalid variable name \"%V\"", &name);
  166.         return NGX_CONF_ERROR;
  167.     }

  168.     name.len--;
  169.     name.data++;

  170.     var = ngx_http_add_variable(cf, &name, NGX_HTTP_VAR_CHANGEABLE);
  171.     if (var == NULL) {
  172.         return NGX_CONF_ERROR;
  173.     }

  174.     var->get_handler = ngx_http_map_variable;
  175.     var->data = (uintptr_t) map;

  176.     pool = ngx_create_pool(NGX_DEFAULT_POOL_SIZE, cf->log);
  177.     if (pool == NULL) {
  178.         return NGX_CONF_ERROR;
  179.     }

  180.     ctx.keys.pool = cf->pool;
  181.     ctx.keys.temp_pool = pool;

  182.     if (ngx_hash_keys_array_init(&ctx.keys, NGX_HASH_LARGE) != NGX_OK) {
  183.         ngx_destroy_pool(pool);
  184.         return NGX_CONF_ERROR;
  185.     }

  186.     ctx.values_hash = ngx_pcalloc(pool, sizeof(ngx_array_t) * ctx.keys.hsize);
  187.     if (ctx.values_hash == NULL) {
  188.         ngx_destroy_pool(pool);
  189.         return NGX_CONF_ERROR;
  190.     }

  191. #if (NGX_PCRE)
  192.     if (ngx_array_init(&ctx.regexes, cf->pool, 2, sizeof(ngx_http_map_regex_t))
  193.         != NGX_OK)
  194.     {
  195.         ngx_destroy_pool(pool);
  196.         return NGX_CONF_ERROR;
  197.     }
  198. #endif

  199.     ctx.default_value = NULL;
  200.     ctx.cf = &save;
  201.     ctx.hostnames = 0;
  202.     ctx.no_cacheable = 0;

  203.     save = *cf;
  204.     cf->pool = pool;
  205.     cf->ctx = &ctx;
  206.     cf->handler = ngx_http_map;
  207.     cf->handler_conf = conf;

  208.     rv = ngx_conf_parse(cf, NULL);

  209.     *cf = save;

  210.     if (rv != NGX_CONF_OK) {
  211.         ngx_destroy_pool(pool);
  212.         return rv;
  213.     }

  214.     if (ctx.no_cacheable) {
  215.         var->flags |= NGX_HTTP_VAR_NOCACHEABLE;
  216.     }

  217.     map->default_value = ctx.default_value ? ctx.default_value:
  218.                                              &ngx_http_variable_null_value;

  219.     map->hostnames = ctx.hostnames;

  220.     hash.key = ngx_hash_key_lc;
  221.     hash.max_size = mcf->hash_max_size;
  222.     hash.bucket_size = mcf->hash_bucket_size;
  223.     hash.name = "map_hash";
  224.     hash.pool = cf->pool;

  225.     if (ctx.keys.keys.nelts) {
  226.         hash.hash = &map->map.hash.hash;
  227.         hash.temp_pool = NULL;

  228.         if (ngx_hash_init(&hash, ctx.keys.keys.elts, ctx.keys.keys.nelts)
  229.             != NGX_OK)
  230.         {
  231.             ngx_destroy_pool(pool);
  232.             return NGX_CONF_ERROR;
  233.         }
  234.     }

  235.     if (ctx.keys.dns_wc_head.nelts) {

  236.         ngx_qsort(ctx.keys.dns_wc_head.elts,
  237.                   (size_t) ctx.keys.dns_wc_head.nelts,
  238.                   sizeof(ngx_hash_key_t), ngx_http_map_cmp_dns_wildcards);

  239.         hash.hash = NULL;
  240.         hash.temp_pool = pool;

  241.         if (ngx_hash_wildcard_init(&hash, ctx.keys.dns_wc_head.elts,
  242.                                    ctx.keys.dns_wc_head.nelts)
  243.             != NGX_OK)
  244.         {
  245.             ngx_destroy_pool(pool);
  246.             return NGX_CONF_ERROR;
  247.         }

  248.         map->map.hash.wc_head = (ngx_hash_wildcard_t *) hash.hash;
  249.     }

  250.     if (ctx.keys.dns_wc_tail.nelts) {

  251.         ngx_qsort(ctx.keys.dns_wc_tail.elts,
  252.                   (size_t) ctx.keys.dns_wc_tail.nelts,
  253.                   sizeof(ngx_hash_key_t), ngx_http_map_cmp_dns_wildcards);

  254.         hash.hash = NULL;
  255.         hash.temp_pool = pool;

  256.         if (ngx_hash_wildcard_init(&hash, ctx.keys.dns_wc_tail.elts,
  257.                                    ctx.keys.dns_wc_tail.nelts)
  258.             != NGX_OK)
  259.         {
  260.             ngx_destroy_pool(pool);
  261.             return NGX_CONF_ERROR;
  262.         }

  263.         map->map.hash.wc_tail = (ngx_hash_wildcard_t *) hash.hash;
  264.     }

  265. #if (NGX_PCRE)

  266.     if (ctx.regexes.nelts) {
  267.         map->map.regex = ctx.regexes.elts;
  268.         map->map.nregex = ctx.regexes.nelts;
  269.     }

  270. #endif

  271.     ngx_destroy_pool(pool);

  272.     return rv;
  273. }


  274. static int ngx_libc_cdecl
  275. ngx_http_map_cmp_dns_wildcards(const void *one, const void *two)
  276. {
  277.     ngx_hash_key_t  *first, *second;

  278.     first = (ngx_hash_key_t *) one;
  279.     second = (ngx_hash_key_t *) two;

  280.     return ngx_dns_strcmp(first->key.data, second->key.data);
  281. }


  282. static char *
  283. ngx_http_map(ngx_conf_t *cf, ngx_command_t *dummy, void *conf)
  284. {
  285.     u_char                            *data;
  286.     size_t                             len;
  287.     ngx_int_t                          rv;
  288.     ngx_str_t                         *value, v;
  289.     ngx_uint_t                         i, key;
  290.     ngx_http_map_conf_ctx_t           *ctx;
  291.     ngx_http_complex_value_t           cv, *cvp;
  292.     ngx_http_variable_value_t         *var, **vp;
  293.     ngx_http_compile_complex_value_t   ccv;

  294.     ctx = cf->ctx;

  295.     value = cf->args->elts;

  296.     if (cf->args->nelts == 1
  297.         && ngx_strcmp(value[0].data, "hostnames") == 0)
  298.     {
  299.         ctx->hostnames = 1;
  300.         return NGX_CONF_OK;
  301.     }

  302.     if (cf->args->nelts == 1
  303.         && ngx_strcmp(value[0].data, "volatile") == 0)
  304.     {
  305.         ctx->no_cacheable = 1;
  306.         return NGX_CONF_OK;
  307.     }

  308.     if (cf->args->nelts != 2) {
  309.         ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,
  310.                            "invalid number of the map parameters");
  311.         return NGX_CONF_ERROR;
  312.     }

  313.     if (ngx_strcmp(value[0].data, "include") == 0) {
  314.         return ngx_conf_include(cf, dummy, conf);
  315.     }

  316.     key = 0;

  317.     for (i = 0; i < value[1].len; i++) {
  318.         key = ngx_hash(key, value[1].data[i]);
  319.     }

  320.     key %= ctx->keys.hsize;

  321.     vp = ctx->values_hash[key].elts;

  322.     if (vp) {
  323.         for (i = 0; i < ctx->values_hash[key].nelts; i++) {

  324.             if (vp[i]->valid) {
  325.                 data = vp[i]->data;
  326.                 len = vp[i]->len;

  327.             } else {
  328.                 cvp = (ngx_http_complex_value_t *) vp[i]->data;
  329.                 data = cvp->value.data;
  330.                 len = cvp->value.len;
  331.             }

  332.             if (value[1].len != len) {
  333.                 continue;
  334.             }

  335.             if (ngx_strncmp(value[1].data, data, len) == 0) {
  336.                 var = vp[i];
  337.                 goto found;
  338.             }
  339.         }

  340.     } else {
  341.         if (ngx_array_init(&ctx->values_hash[key], cf->pool, 4,
  342.                            sizeof(ngx_http_variable_value_t *))
  343.             != NGX_OK)
  344.         {
  345.             return NGX_CONF_ERROR;
  346.         }
  347.     }

  348.     var = ngx_palloc(ctx->keys.pool, sizeof(ngx_http_variable_value_t));
  349.     if (var == NULL) {
  350.         return NGX_CONF_ERROR;
  351.     }

  352.     v.len = value[1].len;
  353.     v.data = ngx_pstrdup(ctx->keys.pool, &value[1]);
  354.     if (v.data == NULL) {
  355.         return NGX_CONF_ERROR;
  356.     }

  357.     ngx_memzero(&ccv, sizeof(ngx_http_compile_complex_value_t));

  358.     ccv.cf = ctx->cf;
  359.     ccv.value = &v;
  360.     ccv.complex_value = &cv;

  361.     if (ngx_http_compile_complex_value(&ccv) != NGX_OK) {
  362.         return NGX_CONF_ERROR;
  363.     }

  364.     if (cv.lengths != NULL) {
  365.         cvp = ngx_palloc(ctx->keys.pool, sizeof(ngx_http_complex_value_t));
  366.         if (cvp == NULL) {
  367.             return NGX_CONF_ERROR;
  368.         }

  369.         *cvp = cv;

  370.         var->len = 0;
  371.         var->data = (u_char *) cvp;
  372.         var->valid = 0;

  373.     } else {
  374.         var->len = v.len;
  375.         var->data = v.data;
  376.         var->valid = 1;
  377.     }

  378.     var->no_cacheable = 0;
  379.     var->not_found = 0;

  380.     vp = ngx_array_push(&ctx->values_hash[key]);
  381.     if (vp == NULL) {
  382.         return NGX_CONF_ERROR;
  383.     }

  384.     *vp = var;

  385. found:

  386.     if (ngx_strcmp(value[0].data, "default") == 0) {

  387.         if (ctx->default_value) {
  388.             ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,
  389.                                "duplicate default map parameter");
  390.             return NGX_CONF_ERROR;
  391.         }

  392.         ctx->default_value = var;

  393.         return NGX_CONF_OK;
  394.     }

  395. #if (NGX_PCRE)

  396.     if (value[0].len && value[0].data[0] == '~') {
  397.         ngx_regex_compile_t    rc;
  398.         ngx_http_map_regex_t  *regex;
  399.         u_char                 errstr[NGX_MAX_CONF_ERRSTR];

  400.         regex = ngx_array_push(&ctx->regexes);
  401.         if (regex == NULL) {
  402.             return NGX_CONF_ERROR;
  403.         }

  404.         value[0].len--;
  405.         value[0].data++;

  406.         ngx_memzero(&rc, sizeof(ngx_regex_compile_t));

  407.         if (value[0].data[0] == '*') {
  408.             value[0].len--;
  409.             value[0].data++;
  410.             rc.options = NGX_REGEX_CASELESS;
  411.         }

  412.         rc.pattern = value[0];
  413.         rc.err.len = NGX_MAX_CONF_ERRSTR;
  414.         rc.err.data = errstr;

  415.         regex->regex = ngx_http_regex_compile(ctx->cf, &rc);
  416.         if (regex->regex == NULL) {
  417.             return NGX_CONF_ERROR;
  418.         }

  419.         regex->value = var;

  420.         return NGX_CONF_OK;
  421.     }

  422. #endif

  423.     if (value[0].len && value[0].data[0] == '\\') {
  424.         value[0].len--;
  425.         value[0].data++;
  426.     }

  427.     rv = ngx_hash_add_key(&ctx->keys, &value[0], var,
  428.                           (ctx->hostnames) ? NGX_HASH_WILDCARD_KEY : 0);

  429.     if (rv == NGX_OK) {
  430.         return NGX_CONF_OK;
  431.     }

  432.     if (rv == NGX_DECLINED) {
  433.         ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,
  434.                            "invalid hostname or wildcard \"%V\"", &value[0]);
  435.     }

  436.     if (rv == NGX_BUSY) {
  437.         ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,
  438.                            "conflicting parameter \"%V\"", &value[0]);
  439.     }

  440.     return NGX_CONF_ERROR;
  441. }