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

Global variables defined

Data types defined

Functions defined

Macros 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. #define NGX_HTTP_LIMIT_REQ_PASSED            1
  9. #define NGX_HTTP_LIMIT_REQ_DELAYED           2
  10. #define NGX_HTTP_LIMIT_REQ_REJECTED          3
  11. #define NGX_HTTP_LIMIT_REQ_DELAYED_DRY_RUN   4
  12. #define NGX_HTTP_LIMIT_REQ_REJECTED_DRY_RUN  5


  13. typedef struct {
  14.     u_char                       color;
  15.     u_char                       dummy;
  16.     u_short                      len;
  17.     ngx_queue_t                  queue;
  18.     ngx_msec_t                   last;
  19.     /* integer value, 1 corresponds to 0.001 r/s */
  20.     ngx_uint_t                   excess;
  21.     ngx_uint_t                   count;
  22.     u_char                       data[1];
  23. } ngx_http_limit_req_node_t;


  24. typedef struct {
  25.     ngx_rbtree_t                  rbtree;
  26.     ngx_rbtree_node_t             sentinel;
  27.     ngx_queue_t                   queue;
  28. } ngx_http_limit_req_shctx_t;


  29. typedef struct {
  30.     ngx_http_limit_req_shctx_t  *sh;
  31.     ngx_slab_pool_t             *shpool;
  32.     /* integer value, 1 corresponds to 0.001 r/s */
  33.     ngx_uint_t                   rate;
  34.     ngx_http_complex_value_t     key;
  35.     ngx_http_limit_req_node_t   *node;
  36. } ngx_http_limit_req_ctx_t;


  37. typedef struct {
  38.     ngx_shm_zone_t              *shm_zone;
  39.     /* integer value, 1 corresponds to 0.001 r/s */
  40.     ngx_uint_t                   burst;
  41.     ngx_uint_t                   delay;
  42. } ngx_http_limit_req_limit_t;


  43. typedef struct {
  44.     ngx_array_t                  limits;
  45.     ngx_uint_t                   limit_log_level;
  46.     ngx_uint_t                   delay_log_level;
  47.     ngx_uint_t                   status_code;
  48.     ngx_flag_t                   dry_run;
  49. } ngx_http_limit_req_conf_t;


  50. static void ngx_http_limit_req_delay(ngx_http_request_t *r);
  51. static ngx_int_t ngx_http_limit_req_lookup(ngx_http_limit_req_limit_t *limit,
  52.     ngx_uint_t hash, ngx_str_t *key, ngx_uint_t *ep, ngx_uint_t account);
  53. static ngx_msec_t ngx_http_limit_req_account(ngx_http_limit_req_limit_t *limits,
  54.     ngx_uint_t n, ngx_uint_t *ep, ngx_http_limit_req_limit_t **limit);
  55. static void ngx_http_limit_req_unlock(ngx_http_limit_req_limit_t *limits,
  56.     ngx_uint_t n);
  57. static void ngx_http_limit_req_expire(ngx_http_limit_req_ctx_t *ctx,
  58.     ngx_uint_t n);

  59. static ngx_int_t ngx_http_limit_req_status_variable(ngx_http_request_t *r,
  60.     ngx_http_variable_value_t *v, uintptr_t data);
  61. static void *ngx_http_limit_req_create_conf(ngx_conf_t *cf);
  62. static char *ngx_http_limit_req_merge_conf(ngx_conf_t *cf, void *parent,
  63.     void *child);
  64. static char *ngx_http_limit_req_zone(ngx_conf_t *cf, ngx_command_t *cmd,
  65.     void *conf);
  66. static char *ngx_http_limit_req(ngx_conf_t *cf, ngx_command_t *cmd,
  67.     void *conf);
  68. static ngx_int_t ngx_http_limit_req_add_variables(ngx_conf_t *cf);
  69. static ngx_int_t ngx_http_limit_req_init(ngx_conf_t *cf);


  70. static ngx_conf_enum_t  ngx_http_limit_req_log_levels[] = {
  71.     { ngx_string("info"), NGX_LOG_INFO },
  72.     { ngx_string("notice"), NGX_LOG_NOTICE },
  73.     { ngx_string("warn"), NGX_LOG_WARN },
  74.     { ngx_string("error"), NGX_LOG_ERR },
  75.     { ngx_null_string, 0 }
  76. };


  77. static ngx_conf_num_bounds_t  ngx_http_limit_req_status_bounds = {
  78.     ngx_conf_check_num_bounds, 400, 599
  79. };


  80. static ngx_command_t  ngx_http_limit_req_commands[] = {

  81.     { ngx_string("limit_req_zone"),
  82.       NGX_HTTP_MAIN_CONF|NGX_CONF_TAKE3,
  83.       ngx_http_limit_req_zone,
  84.       0,
  85.       0,
  86.       NULL },

  87.     { ngx_string("limit_req"),
  88.       NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_TAKE123,
  89.       ngx_http_limit_req,
  90.       NGX_HTTP_LOC_CONF_OFFSET,
  91.       0,
  92.       NULL },

  93.     { ngx_string("limit_req_log_level"),
  94.       NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_TAKE1,
  95.       ngx_conf_set_enum_slot,
  96.       NGX_HTTP_LOC_CONF_OFFSET,
  97.       offsetof(ngx_http_limit_req_conf_t, limit_log_level),
  98.       &ngx_http_limit_req_log_levels },

  99.     { ngx_string("limit_req_status"),
  100.       NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_TAKE1,
  101.       ngx_conf_set_num_slot,
  102.       NGX_HTTP_LOC_CONF_OFFSET,
  103.       offsetof(ngx_http_limit_req_conf_t, status_code),
  104.       &ngx_http_limit_req_status_bounds },

  105.     { ngx_string("limit_req_dry_run"),
  106.       NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_FLAG,
  107.       ngx_conf_set_flag_slot,
  108.       NGX_HTTP_LOC_CONF_OFFSET,
  109.       offsetof(ngx_http_limit_req_conf_t, dry_run),
  110.       NULL },

  111.       ngx_null_command
  112. };


  113. static ngx_http_module_t  ngx_http_limit_req_module_ctx = {
  114.     ngx_http_limit_req_add_variables,      /* preconfiguration */
  115.     ngx_http_limit_req_init,               /* postconfiguration */

  116.     NULL,                                  /* create main configuration */
  117.     NULL,                                  /* init main configuration */

  118.     NULL,                                  /* create server configuration */
  119.     NULL,                                  /* merge server configuration */

  120.     ngx_http_limit_req_create_conf,        /* create location configuration */
  121.     ngx_http_limit_req_merge_conf          /* merge location configuration */
  122. };


  123. ngx_module_t  ngx_http_limit_req_module = {
  124.     NGX_MODULE_V1,
  125.     &ngx_http_limit_req_module_ctx,        /* module context */
  126.     ngx_http_limit_req_commands,           /* module directives */
  127.     NGX_HTTP_MODULE,                       /* module type */
  128.     NULL,                                  /* init master */
  129.     NULL,                                  /* init module */
  130.     NULL,                                  /* init process */
  131.     NULL,                                  /* init thread */
  132.     NULL,                                  /* exit thread */
  133.     NULL,                                  /* exit process */
  134.     NULL,                                  /* exit master */
  135.     NGX_MODULE_V1_PADDING
  136. };


  137. static ngx_http_variable_t  ngx_http_limit_req_vars[] = {

  138.     { ngx_string("limit_req_status"), NULL,
  139.       ngx_http_limit_req_status_variable, 0, NGX_HTTP_VAR_NOCACHEABLE, 0 },

  140.       ngx_http_null_variable
  141. };


  142. static ngx_str_t  ngx_http_limit_req_status[] = {
  143.     ngx_string("PASSED"),
  144.     ngx_string("DELAYED"),
  145.     ngx_string("REJECTED"),
  146.     ngx_string("DELAYED_DRY_RUN"),
  147.     ngx_string("REJECTED_DRY_RUN")
  148. };


  149. static ngx_int_t
  150. ngx_http_limit_req_handler(ngx_http_request_t *r)
  151. {
  152.     uint32_t                     hash;
  153.     ngx_str_t                    key;
  154.     ngx_int_t                    rc;
  155.     ngx_uint_t                   n, excess;
  156.     ngx_msec_t                   delay;
  157.     ngx_http_limit_req_ctx_t    *ctx;
  158.     ngx_http_limit_req_conf_t   *lrcf;
  159.     ngx_http_limit_req_limit_t  *limit, *limits;

  160.     if (r->main->limit_req_status) {
  161.         return NGX_DECLINED;
  162.     }

  163.     lrcf = ngx_http_get_module_loc_conf(r, ngx_http_limit_req_module);
  164.     limits = lrcf->limits.elts;

  165.     excess = 0;

  166.     rc = NGX_DECLINED;

  167. #if (NGX_SUPPRESS_WARN)
  168.     limit = NULL;
  169. #endif

  170.     for (n = 0; n < lrcf->limits.nelts; n++) {

  171.         limit = &limits[n];

  172.         ctx = limit->shm_zone->data;

  173.         if (ngx_http_complex_value(r, &ctx->key, &key) != NGX_OK) {
  174.             ngx_http_limit_req_unlock(limits, n);
  175.             return NGX_HTTP_INTERNAL_SERVER_ERROR;
  176.         }

  177.         if (key.len == 0) {
  178.             continue;
  179.         }

  180.         if (key.len > 65535) {
  181.             ngx_log_error(NGX_LOG_ERR, r->connection->log, 0,
  182.                           "the value of the \"%V\" key "
  183.                           "is more than 65535 bytes: \"%V\"",
  184.                           &ctx->key.value, &key);
  185.             continue;
  186.         }

  187.         hash = ngx_crc32_short(key.data, key.len);

  188.         ngx_shmtx_lock(&ctx->shpool->mutex);

  189.         rc = ngx_http_limit_req_lookup(limit, hash, &key, &excess,
  190.                                        (n == lrcf->limits.nelts - 1));

  191.         ngx_shmtx_unlock(&ctx->shpool->mutex);

  192.         ngx_log_debug4(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
  193.                        "limit_req[%ui]: %i %ui.%03ui",
  194.                        n, rc, excess / 1000, excess % 1000);

  195.         if (rc != NGX_AGAIN) {
  196.             break;
  197.         }
  198.     }

  199.     if (rc == NGX_DECLINED) {
  200.         return NGX_DECLINED;
  201.     }

  202.     if (rc == NGX_BUSY || rc == NGX_ERROR) {

  203.         if (rc == NGX_BUSY) {
  204.             ngx_log_error(lrcf->limit_log_level, r->connection->log, 0,
  205.                         "limiting requests%s, excess: %ui.%03ui by zone \"%V\"",
  206.                         lrcf->dry_run ? ", dry run" : "",
  207.                         excess / 1000, excess % 1000,
  208.                         &limit->shm_zone->shm.name);
  209.         }

  210.         ngx_http_limit_req_unlock(limits, n);

  211.         if (lrcf->dry_run) {
  212.             r->main->limit_req_status = NGX_HTTP_LIMIT_REQ_REJECTED_DRY_RUN;
  213.             return NGX_DECLINED;
  214.         }

  215.         r->main->limit_req_status = NGX_HTTP_LIMIT_REQ_REJECTED;

  216.         return lrcf->status_code;
  217.     }

  218.     /* rc == NGX_AGAIN || rc == NGX_OK */

  219.     if (rc == NGX_AGAIN) {
  220.         excess = 0;
  221.     }

  222.     delay = ngx_http_limit_req_account(limits, n, &excess, &limit);

  223.     if (!delay) {
  224.         r->main->limit_req_status = NGX_HTTP_LIMIT_REQ_PASSED;
  225.         return NGX_DECLINED;
  226.     }

  227.     ngx_log_error(lrcf->delay_log_level, r->connection->log, 0,
  228.                   "delaying request%s, excess: %ui.%03ui, by zone \"%V\"",
  229.                   lrcf->dry_run ? ", dry run" : "",
  230.                   excess / 1000, excess % 1000, &limit->shm_zone->shm.name);

  231.     if (lrcf->dry_run) {
  232.         r->main->limit_req_status = NGX_HTTP_LIMIT_REQ_DELAYED_DRY_RUN;
  233.         return NGX_DECLINED;
  234.     }

  235.     r->main->limit_req_status = NGX_HTTP_LIMIT_REQ_DELAYED;

  236.     if (r->connection->read->ready) {
  237.         ngx_post_event(r->connection->read, &ngx_posted_events);

  238.     } else {
  239.         if (ngx_handle_read_event(r->connection->read, 0) != NGX_OK) {
  240.             return NGX_HTTP_INTERNAL_SERVER_ERROR;
  241.         }
  242.     }

  243.     r->read_event_handler = ngx_http_test_reading;
  244.     r->write_event_handler = ngx_http_limit_req_delay;

  245.     r->connection->write->delayed = 1;
  246.     ngx_add_timer(r->connection->write, delay);

  247.     return NGX_AGAIN;
  248. }


  249. static void
  250. ngx_http_limit_req_delay(ngx_http_request_t *r)
  251. {
  252.     ngx_event_t  *wev;

  253.     ngx_log_debug0(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
  254.                    "limit_req delay");

  255.     wev = r->connection->write;

  256.     if (wev->delayed) {

  257.         if (ngx_handle_write_event(wev, 0) != NGX_OK) {
  258.             ngx_http_finalize_request(r, NGX_HTTP_INTERNAL_SERVER_ERROR);
  259.         }

  260.         return;
  261.     }

  262.     if (ngx_handle_read_event(r->connection->read, 0) != NGX_OK) {
  263.         ngx_http_finalize_request(r, NGX_HTTP_INTERNAL_SERVER_ERROR);
  264.         return;
  265.     }

  266.     r->read_event_handler = ngx_http_block_reading;
  267.     r->write_event_handler = ngx_http_core_run_phases;

  268.     ngx_http_core_run_phases(r);
  269. }


  270. static void
  271. ngx_http_limit_req_rbtree_insert_value(ngx_rbtree_node_t *temp,
  272.     ngx_rbtree_node_t *node, ngx_rbtree_node_t *sentinel)
  273. {
  274.     ngx_rbtree_node_t          **p;
  275.     ngx_http_limit_req_node_t   *lrn, *lrnt;

  276.     for ( ;; ) {

  277.         if (node->key < temp->key) {

  278.             p = &temp->left;

  279.         } else if (node->key > temp->key) {

  280.             p = &temp->right;

  281.         } else { /* node->key == temp->key */

  282.             lrn = (ngx_http_limit_req_node_t *) &node->color;
  283.             lrnt = (ngx_http_limit_req_node_t *) &temp->color;

  284.             p = (ngx_memn2cmp(lrn->data, lrnt->data, lrn->len, lrnt->len) < 0)
  285.                 ? &temp->left : &temp->right;
  286.         }

  287.         if (*p == sentinel) {
  288.             break;
  289.         }

  290.         temp = *p;
  291.     }

  292.     *p = node;
  293.     node->parent = temp;
  294.     node->left = sentinel;
  295.     node->right = sentinel;
  296.     ngx_rbt_red(node);
  297. }


  298. static ngx_int_t
  299. ngx_http_limit_req_lookup(ngx_http_limit_req_limit_t *limit, ngx_uint_t hash,
  300.     ngx_str_t *key, ngx_uint_t *ep, ngx_uint_t account)
  301. {
  302.     size_t                      size;
  303.     ngx_int_t                   rc, excess;
  304.     ngx_msec_t                  now;
  305.     ngx_msec_int_t              ms;
  306.     ngx_rbtree_node_t          *node, *sentinel;
  307.     ngx_http_limit_req_ctx_t   *ctx;
  308.     ngx_http_limit_req_node_t  *lr;

  309.     now = ngx_current_msec;

  310.     ctx = limit->shm_zone->data;

  311.     node = ctx->sh->rbtree.root;
  312.     sentinel = ctx->sh->rbtree.sentinel;

  313.     while (node != sentinel) {

  314.         if (hash < node->key) {
  315.             node = node->left;
  316.             continue;
  317.         }

  318.         if (hash > node->key) {
  319.             node = node->right;
  320.             continue;
  321.         }

  322.         /* hash == node->key */

  323.         lr = (ngx_http_limit_req_node_t *) &node->color;

  324.         rc = ngx_memn2cmp(key->data, lr->data, key->len, (size_t) lr->len);

  325.         if (rc == 0) {
  326.             ngx_queue_remove(&lr->queue);
  327.             ngx_queue_insert_head(&ctx->sh->queue, &lr->queue);

  328.             ms = (ngx_msec_int_t) (now - lr->last);

  329.             if (ms < -60000) {
  330.                 ms = 1;

  331.             } else if (ms < 0) {
  332.                 ms = 0;
  333.             }

  334.             excess = lr->excess - ctx->rate * ms / 1000 + 1000;

  335.             if (excess < 0) {
  336.                 excess = 0;
  337.             }

  338.             *ep = excess;

  339.             if ((ngx_uint_t) excess > limit->burst) {
  340.                 return NGX_BUSY;
  341.             }

  342.             if (account) {
  343.                 lr->excess = excess;

  344.                 if (ms) {
  345.                     lr->last = now;
  346.                 }

  347.                 return NGX_OK;
  348.             }

  349.             lr->count++;

  350.             ctx->node = lr;

  351.             return NGX_AGAIN;
  352.         }

  353.         node = (rc < 0) ? node->left : node->right;
  354.     }

  355.     *ep = 0;

  356.     size = offsetof(ngx_rbtree_node_t, color)
  357.            + offsetof(ngx_http_limit_req_node_t, data)
  358.            + key->len;

  359.     ngx_http_limit_req_expire(ctx, 1);

  360.     node = ngx_slab_alloc_locked(ctx->shpool, size);

  361.     if (node == NULL) {
  362.         ngx_http_limit_req_expire(ctx, 0);

  363.         node = ngx_slab_alloc_locked(ctx->shpool, size);
  364.         if (node == NULL) {
  365.             ngx_log_error(NGX_LOG_ALERT, ngx_cycle->log, 0,
  366.                           "could not allocate node%s", ctx->shpool->log_ctx);
  367.             return NGX_ERROR;
  368.         }
  369.     }

  370.     node->key = hash;

  371.     lr = (ngx_http_limit_req_node_t *) &node->color;

  372.     lr->len = (u_short) key->len;
  373.     lr->excess = 0;

  374.     ngx_memcpy(lr->data, key->data, key->len);

  375.     ngx_rbtree_insert(&ctx->sh->rbtree, node);

  376.     ngx_queue_insert_head(&ctx->sh->queue, &lr->queue);

  377.     if (account) {
  378.         lr->last = now;
  379.         lr->count = 0;
  380.         return NGX_OK;
  381.     }

  382.     lr->last = 0;
  383.     lr->count = 1;

  384.     ctx->node = lr;

  385.     return NGX_AGAIN;
  386. }


  387. static ngx_msec_t
  388. ngx_http_limit_req_account(ngx_http_limit_req_limit_t *limits, ngx_uint_t n,
  389.     ngx_uint_t *ep, ngx_http_limit_req_limit_t **limit)
  390. {
  391.     ngx_int_t                   excess;
  392.     ngx_msec_t                  now, delay, max_delay;
  393.     ngx_msec_int_t              ms;
  394.     ngx_http_limit_req_ctx_t   *ctx;
  395.     ngx_http_limit_req_node_t  *lr;

  396.     excess = *ep;

  397.     if ((ngx_uint_t) excess <= (*limit)->delay) {
  398.         max_delay = 0;

  399.     } else {
  400.         ctx = (*limit)->shm_zone->data;
  401.         max_delay = (excess - (*limit)->delay) * 1000 / ctx->rate;
  402.     }

  403.     while (n--) {
  404.         ctx = limits[n].shm_zone->data;
  405.         lr = ctx->node;

  406.         if (lr == NULL) {
  407.             continue;
  408.         }

  409.         ngx_shmtx_lock(&ctx->shpool->mutex);

  410.         now = ngx_current_msec;
  411.         ms = (ngx_msec_int_t) (now - lr->last);

  412.         if (ms < -60000) {
  413.             ms = 1;

  414.         } else if (ms < 0) {
  415.             ms = 0;
  416.         }

  417.         excess = lr->excess - ctx->rate * ms / 1000 + 1000;

  418.         if (excess < 0) {
  419.             excess = 0;
  420.         }

  421.         if (ms) {
  422.             lr->last = now;
  423.         }

  424.         lr->excess = excess;
  425.         lr->count--;

  426.         ngx_shmtx_unlock(&ctx->shpool->mutex);

  427.         ctx->node = NULL;

  428.         if ((ngx_uint_t) excess <= limits[n].delay) {
  429.             continue;
  430.         }

  431.         delay = (excess - limits[n].delay) * 1000 / ctx->rate;

  432.         if (delay > max_delay) {
  433.             max_delay = delay;
  434.             *ep = excess;
  435.             *limit = &limits[n];
  436.         }
  437.     }

  438.     return max_delay;
  439. }


  440. static void
  441. ngx_http_limit_req_unlock(ngx_http_limit_req_limit_t *limits, ngx_uint_t n)
  442. {
  443.     ngx_http_limit_req_ctx_t  *ctx;

  444.     while (n--) {
  445.         ctx = limits[n].shm_zone->data;

  446.         if (ctx->node == NULL) {
  447.             continue;
  448.         }

  449.         ngx_shmtx_lock(&ctx->shpool->mutex);

  450.         ctx->node->count--;

  451.         ngx_shmtx_unlock(&ctx->shpool->mutex);

  452.         ctx->node = NULL;
  453.     }
  454. }


  455. static void
  456. ngx_http_limit_req_expire(ngx_http_limit_req_ctx_t *ctx, ngx_uint_t n)
  457. {
  458.     ngx_int_t                   excess;
  459.     ngx_msec_t                  now;
  460.     ngx_queue_t                *q;
  461.     ngx_msec_int_t              ms;
  462.     ngx_rbtree_node_t          *node;
  463.     ngx_http_limit_req_node_t  *lr;

  464.     now = ngx_current_msec;

  465.     /*
  466.      * n == 1 deletes one or two zero rate entries
  467.      * n == 0 deletes oldest entry by force
  468.      *        and one or two zero rate entries
  469.      */

  470.     while (n < 3) {

  471.         if (ngx_queue_empty(&ctx->sh->queue)) {
  472.             return;
  473.         }

  474.         q = ngx_queue_last(&ctx->sh->queue);

  475.         lr = ngx_queue_data(q, ngx_http_limit_req_node_t, queue);

  476.         if (lr->count) {

  477.             /*
  478.              * There is not much sense in looking further,
  479.              * because we bump nodes on the lookup stage.
  480.              */

  481.             return;
  482.         }

  483.         if (n++ != 0) {

  484.             ms = (ngx_msec_int_t) (now - lr->last);
  485.             ms = ngx_abs(ms);

  486.             if (ms < 60000) {
  487.                 return;
  488.             }

  489.             excess = lr->excess - ctx->rate * ms / 1000;

  490.             if (excess > 0) {
  491.                 return;
  492.             }
  493.         }

  494.         ngx_queue_remove(q);

  495.         node = (ngx_rbtree_node_t *)
  496.                    ((u_char *) lr - offsetof(ngx_rbtree_node_t, color));

  497.         ngx_rbtree_delete(&ctx->sh->rbtree, node);

  498.         ngx_slab_free_locked(ctx->shpool, node);
  499.     }
  500. }


  501. static ngx_int_t
  502. ngx_http_limit_req_init_zone(ngx_shm_zone_t *shm_zone, void *data)
  503. {
  504.     ngx_http_limit_req_ctx_t  *octx = data;

  505.     size_t                     len;
  506.     ngx_http_limit_req_ctx_t  *ctx;

  507.     ctx = shm_zone->data;

  508.     if (octx) {
  509.         if (ctx->key.value.len != octx->key.value.len
  510.             || ngx_strncmp(ctx->key.value.data, octx->key.value.data,
  511.                            ctx->key.value.len)
  512.                != 0)
  513.         {
  514.             ngx_log_error(NGX_LOG_EMERG, shm_zone->shm.log, 0,
  515.                           "limit_req \"%V\" uses the \"%V\" key "
  516.                           "while previously it used the \"%V\" key",
  517.                           &shm_zone->shm.name, &ctx->key.value,
  518.                           &octx->key.value);
  519.             return NGX_ERROR;
  520.         }

  521.         ctx->sh = octx->sh;
  522.         ctx->shpool = octx->shpool;

  523.         return NGX_OK;
  524.     }

  525.     ctx->shpool = (ngx_slab_pool_t *) shm_zone->shm.addr;

  526.     if (shm_zone->shm.exists) {
  527.         ctx->sh = ctx->shpool->data;

  528.         return NGX_OK;
  529.     }

  530.     ctx->sh = ngx_slab_alloc(ctx->shpool, sizeof(ngx_http_limit_req_shctx_t));
  531.     if (ctx->sh == NULL) {
  532.         return NGX_ERROR;
  533.     }

  534.     ctx->shpool->data = ctx->sh;

  535.     ngx_rbtree_init(&ctx->sh->rbtree, &ctx->sh->sentinel,
  536.                     ngx_http_limit_req_rbtree_insert_value);

  537.     ngx_queue_init(&ctx->sh->queue);

  538.     len = sizeof(" in limit_req zone \"\"") + shm_zone->shm.name.len;

  539.     ctx->shpool->log_ctx = ngx_slab_alloc(ctx->shpool, len);
  540.     if (ctx->shpool->log_ctx == NULL) {
  541.         return NGX_ERROR;
  542.     }

  543.     ngx_sprintf(ctx->shpool->log_ctx, " in limit_req zone \"%V\"%Z",
  544.                 &shm_zone->shm.name);

  545.     ctx->shpool->log_nomem = 0;

  546.     return NGX_OK;
  547. }


  548. static ngx_int_t
  549. ngx_http_limit_req_status_variable(ngx_http_request_t *r,
  550.     ngx_http_variable_value_t *v, uintptr_t data)
  551. {
  552.     if (r->main->limit_req_status == 0) {
  553.         v->not_found = 1;
  554.         return NGX_OK;
  555.     }

  556.     v->valid = 1;
  557.     v->no_cacheable = 0;
  558.     v->not_found = 0;
  559.     v->len = ngx_http_limit_req_status[r->main->limit_req_status - 1].len;
  560.     v->data = ngx_http_limit_req_status[r->main->limit_req_status - 1].data;

  561.     return NGX_OK;
  562. }


  563. static void *
  564. ngx_http_limit_req_create_conf(ngx_conf_t *cf)
  565. {
  566.     ngx_http_limit_req_conf_t  *conf;

  567.     conf = ngx_pcalloc(cf->pool, sizeof(ngx_http_limit_req_conf_t));
  568.     if (conf == NULL) {
  569.         return NULL;
  570.     }

  571.     /*
  572.      * set by ngx_pcalloc():
  573.      *
  574.      *     conf->limits.elts = NULL;
  575.      */

  576.     conf->limit_log_level = NGX_CONF_UNSET_UINT;
  577.     conf->status_code = NGX_CONF_UNSET_UINT;
  578.     conf->dry_run = NGX_CONF_UNSET;

  579.     return conf;
  580. }


  581. static char *
  582. ngx_http_limit_req_merge_conf(ngx_conf_t *cf, void *parent, void *child)
  583. {
  584.     ngx_http_limit_req_conf_t *prev = parent;
  585.     ngx_http_limit_req_conf_t *conf = child;

  586.     if (conf->limits.elts == NULL) {
  587.         conf->limits = prev->limits;
  588.     }

  589.     ngx_conf_merge_uint_value(conf->limit_log_level, prev->limit_log_level,
  590.                               NGX_LOG_ERR);

  591.     conf->delay_log_level = (conf->limit_log_level == NGX_LOG_INFO) ?
  592.                                 NGX_LOG_INFO : conf->limit_log_level + 1;

  593.     ngx_conf_merge_uint_value(conf->status_code, prev->status_code,
  594.                               NGX_HTTP_SERVICE_UNAVAILABLE);

  595.     ngx_conf_merge_value(conf->dry_run, prev->dry_run, 0);

  596.     return NGX_CONF_OK;
  597. }


  598. static char *
  599. ngx_http_limit_req_zone(ngx_conf_t *cf, ngx_command_t *cmd, void *conf)
  600. {
  601.     u_char                            *p;
  602.     size_t                             len;
  603.     ssize_t                            size;
  604.     ngx_str_t                         *value, name, s;
  605.     ngx_int_t                          rate, scale;
  606.     ngx_uint_t                         i;
  607.     ngx_shm_zone_t                    *shm_zone;
  608.     ngx_http_limit_req_ctx_t          *ctx;
  609.     ngx_http_compile_complex_value_t   ccv;

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

  611.     ctx = ngx_pcalloc(cf->pool, sizeof(ngx_http_limit_req_ctx_t));
  612.     if (ctx == NULL) {
  613.         return NGX_CONF_ERROR;
  614.     }

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

  616.     ccv.cf = cf;
  617.     ccv.value = &value[1];
  618.     ccv.complex_value = &ctx->key;

  619.     if (ngx_http_compile_complex_value(&ccv) != NGX_OK) {
  620.         return NGX_CONF_ERROR;
  621.     }

  622.     size = 0;
  623.     rate = 1;
  624.     scale = 1;
  625.     name.len = 0;

  626.     for (i = 2; i < cf->args->nelts; i++) {

  627.         if (ngx_strncmp(value[i].data, "zone=", 5) == 0) {

  628.             name.data = value[i].data + 5;

  629.             p = (u_char *) ngx_strchr(name.data, ':');

  630.             if (p == NULL) {
  631.                 ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,
  632.                                    "invalid zone size \"%V\"", &value[i]);
  633.                 return NGX_CONF_ERROR;
  634.             }

  635.             name.len = p - name.data;

  636.             s.data = p + 1;
  637.             s.len = value[i].data + value[i].len - s.data;

  638.             size = ngx_parse_size(&s);

  639.             if (size == NGX_ERROR) {
  640.                 ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,
  641.                                    "invalid zone size \"%V\"", &value[i]);
  642.                 return NGX_CONF_ERROR;
  643.             }

  644.             if (size < (ssize_t) (8 * ngx_pagesize)) {
  645.                 ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,
  646.                                    "zone \"%V\" is too small", &value[i]);
  647.                 return NGX_CONF_ERROR;
  648.             }

  649.             continue;
  650.         }

  651.         if (ngx_strncmp(value[i].data, "rate=", 5) == 0) {

  652.             len = value[i].len;
  653.             p = value[i].data + len - 3;

  654.             if (ngx_strncmp(p, "r/s", 3) == 0) {
  655.                 scale = 1;
  656.                 len -= 3;

  657.             } else if (ngx_strncmp(p, "r/m", 3) == 0) {
  658.                 scale = 60;
  659.                 len -= 3;
  660.             }

  661.             rate = ngx_atoi(value[i].data + 5, len - 5);
  662.             if (rate <= 0) {
  663.                 ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,
  664.                                    "invalid rate \"%V\"", &value[i]);
  665.                 return NGX_CONF_ERROR;
  666.             }

  667.             continue;
  668.         }

  669.         ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,
  670.                            "invalid parameter \"%V\"", &value[i]);
  671.         return NGX_CONF_ERROR;
  672.     }

  673.     if (name.len == 0) {
  674.         ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,
  675.                            "\"%V\" must have \"zone\" parameter",
  676.                            &cmd->name);
  677.         return NGX_CONF_ERROR;
  678.     }

  679.     ctx->rate = rate * 1000 / scale;

  680.     shm_zone = ngx_shared_memory_add(cf, &name, size,
  681.                                      &ngx_http_limit_req_module);
  682.     if (shm_zone == NULL) {
  683.         return NGX_CONF_ERROR;
  684.     }

  685.     if (shm_zone->data) {
  686.         ctx = shm_zone->data;

  687.         ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,
  688.                            "%V \"%V\" is already bound to key \"%V\"",
  689.                            &cmd->name, &name, &ctx->key.value);
  690.         return NGX_CONF_ERROR;
  691.     }

  692.     shm_zone->init = ngx_http_limit_req_init_zone;
  693.     shm_zone->data = ctx;

  694.     return NGX_CONF_OK;
  695. }


  696. static char *
  697. ngx_http_limit_req(ngx_conf_t *cf, ngx_command_t *cmd, void *conf)
  698. {
  699.     ngx_http_limit_req_conf_t  *lrcf = conf;

  700.     ngx_int_t                    burst, delay;
  701.     ngx_str_t                   *value, s;
  702.     ngx_uint_t                   i;
  703.     ngx_shm_zone_t              *shm_zone;
  704.     ngx_http_limit_req_limit_t  *limit, *limits;

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

  706.     shm_zone = NULL;
  707.     burst = 0;
  708.     delay = 0;

  709.     for (i = 1; i < cf->args->nelts; i++) {

  710.         if (ngx_strncmp(value[i].data, "zone=", 5) == 0) {

  711.             s.len = value[i].len - 5;
  712.             s.data = value[i].data + 5;

  713.             shm_zone = ngx_shared_memory_add(cf, &s, 0,
  714.                                              &ngx_http_limit_req_module);
  715.             if (shm_zone == NULL) {
  716.                 return NGX_CONF_ERROR;
  717.             }

  718.             continue;
  719.         }

  720.         if (ngx_strncmp(value[i].data, "burst=", 6) == 0) {

  721.             burst = ngx_atoi(value[i].data + 6, value[i].len - 6);
  722.             if (burst <= 0) {
  723.                 ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,
  724.                                    "invalid burst value \"%V\"", &value[i]);
  725.                 return NGX_CONF_ERROR;
  726.             }

  727.             continue;
  728.         }

  729.         if (ngx_strncmp(value[i].data, "delay=", 6) == 0) {

  730.             delay = ngx_atoi(value[i].data + 6, value[i].len - 6);
  731.             if (delay <= 0) {
  732.                 ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,
  733.                                    "invalid delay value \"%V\"", &value[i]);
  734.                 return NGX_CONF_ERROR;
  735.             }

  736.             continue;
  737.         }

  738.         if (ngx_strcmp(value[i].data, "nodelay") == 0) {
  739.             delay = NGX_MAX_INT_T_VALUE / 1000;
  740.             continue;
  741.         }

  742.         ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,
  743.                            "invalid parameter \"%V\"", &value[i]);
  744.         return NGX_CONF_ERROR;
  745.     }

  746.     if (shm_zone == NULL) {
  747.         ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,
  748.                            "\"%V\" must have \"zone\" parameter",
  749.                            &cmd->name);
  750.         return NGX_CONF_ERROR;
  751.     }

  752.     limits = lrcf->limits.elts;

  753.     if (limits == NULL) {
  754.         if (ngx_array_init(&lrcf->limits, cf->pool, 1,
  755.                            sizeof(ngx_http_limit_req_limit_t))
  756.             != NGX_OK)
  757.         {
  758.             return NGX_CONF_ERROR;
  759.         }
  760.     }

  761.     for (i = 0; i < lrcf->limits.nelts; i++) {
  762.         if (shm_zone == limits[i].shm_zone) {
  763.             return "is duplicate";
  764.         }
  765.     }

  766.     limit = ngx_array_push(&lrcf->limits);
  767.     if (limit == NULL) {
  768.         return NGX_CONF_ERROR;
  769.     }

  770.     limit->shm_zone = shm_zone;
  771.     limit->burst = burst * 1000;
  772.     limit->delay = delay * 1000;

  773.     return NGX_CONF_OK;
  774. }


  775. static ngx_int_t
  776. ngx_http_limit_req_add_variables(ngx_conf_t *cf)
  777. {
  778.     ngx_http_variable_t  *var, *v;

  779.     for (v = ngx_http_limit_req_vars; v->name.len; v++) {
  780.         var = ngx_http_add_variable(cf, &v->name, v->flags);
  781.         if (var == NULL) {
  782.             return NGX_ERROR;
  783.         }

  784.         var->get_handler = v->get_handler;
  785.         var->data = v->data;
  786.     }

  787.     return NGX_OK;
  788. }


  789. static ngx_int_t
  790. ngx_http_limit_req_init(ngx_conf_t *cf)
  791. {
  792.     ngx_http_handler_pt        *h;
  793.     ngx_http_core_main_conf_t  *cmcf;

  794.     cmcf = ngx_http_conf_get_module_main_conf(cf, ngx_http_core_module);

  795.     h = ngx_array_push(&cmcf->phases[NGX_HTTP_PREACCESS_PHASE].handlers);
  796.     if (h == NULL) {
  797.         return NGX_ERROR;
  798.     }

  799.     *h = ngx_http_limit_req_handler;

  800.     return NGX_OK;
  801. }