Mercurial > hg > nginx
annotate src/http/modules/ngx_http_limit_req_module.c @ 3183:b87542338ac3
make limit_req to conform to the leaky bucket algorithm
author | Igor Sysoev <igor@sysoev.ru> |
---|---|
date | Tue, 06 Oct 2009 09:37:18 +0000 |
parents | 70c8b2d28d1d |
children | 30570a5b9bb0 |
rev | line source |
---|---|
980 | 1 |
2 /* | |
3 * Copyright (C) Igor Sysoev | |
4 */ | |
5 | |
6 | |
7 #include <ngx_config.h> | |
8 #include <ngx_core.h> | |
9 #include <ngx_http.h> | |
10 | |
11 | |
12 typedef struct { | |
2720
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
13 u_char color; |
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
14 u_char dummy; |
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
15 u_short len; |
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
16 ngx_queue_t queue; |
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
17 ngx_msec_t last; |
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
18 /* integer value, 1 corresponds to 0.001 r/s */ |
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
19 ngx_uint_t excess; |
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
20 u_char data[1]; |
2294 | 21 } ngx_http_limit_req_node_t; |
980 | 22 |
23 | |
24 typedef struct { | |
2720
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
25 ngx_rbtree_t rbtree; |
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
26 ngx_rbtree_node_t sentinel; |
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
27 ngx_queue_t queue; |
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
28 } ngx_http_limit_req_shctx_t; |
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
29 |
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
30 |
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
31 typedef struct { |
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
32 ngx_http_limit_req_shctx_t *sh; |
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
33 ngx_slab_pool_t *shpool; |
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
34 /* integer value, 1 corresponds to 0.001 r/s */ |
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
35 ngx_uint_t rate; |
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
36 ngx_int_t index; |
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
37 ngx_str_t var; |
2294 | 38 } ngx_http_limit_req_ctx_t; |
987 | 39 |
40 | |
41 typedef struct { | |
2720
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
42 ngx_shm_zone_t *shm_zone; |
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
43 /* integer value, 1 corresponds to 0.001 r/s */ |
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
44 ngx_uint_t burst; |
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
45 ngx_uint_t nodelay;/* unsigned nodelay:1 */ |
2294 | 46 } ngx_http_limit_req_conf_t; |
980 | 47 |
48 | |
2294 | 49 static void ngx_http_limit_req_delay(ngx_http_request_t *r); |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
50 static ngx_int_t ngx_http_limit_req_lookup(ngx_http_limit_req_conf_t *lrcf, |
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
51 ngx_uint_t hash, u_char *data, size_t len, ngx_http_limit_req_node_t **lrp); |
2294 | 52 static void ngx_http_limit_req_expire(ngx_http_limit_req_ctx_t *ctx, |
53 ngx_uint_t n); | |
980 | 54 |
2294 | 55 static void *ngx_http_limit_req_create_conf(ngx_conf_t *cf); |
56 static char *ngx_http_limit_req_merge_conf(ngx_conf_t *cf, void *parent, | |
980 | 57 void *child); |
2294 | 58 static char *ngx_http_limit_req_zone(ngx_conf_t *cf, ngx_command_t *cmd, |
980 | 59 void *conf); |
2294 | 60 static char *ngx_http_limit_req(ngx_conf_t *cf, ngx_command_t *cmd, |
980 | 61 void *conf); |
2294 | 62 static ngx_int_t ngx_http_limit_req_init(ngx_conf_t *cf); |
980 | 63 |
64 | |
2294 | 65 static ngx_command_t ngx_http_limit_req_commands[] = { |
980 | 66 |
2294 | 67 { ngx_string("limit_req_zone"), |
987 | 68 NGX_HTTP_MAIN_CONF|NGX_CONF_TAKE3, |
2294 | 69 ngx_http_limit_req_zone, |
980 | 70 0, |
71 0, | |
72 NULL }, | |
73 | |
2294 | 74 { ngx_string("limit_req"), |
75 NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_TAKE123, | |
76 ngx_http_limit_req, | |
980 | 77 NGX_HTTP_LOC_CONF_OFFSET, |
78 0, | |
79 NULL }, | |
80 | |
81 ngx_null_command | |
82 }; | |
83 | |
84 | |
2294 | 85 static ngx_http_module_t ngx_http_limit_req_module_ctx = { |
980 | 86 NULL, /* preconfiguration */ |
2294 | 87 ngx_http_limit_req_init, /* postconfiguration */ |
980 | 88 |
89 NULL, /* create main configuration */ | |
90 NULL, /* init main configuration */ | |
91 | |
92 NULL, /* create server configuration */ | |
93 NULL, /* merge server configuration */ | |
94 | |
2294 | 95 ngx_http_limit_req_create_conf, /* create location configration */ |
96 ngx_http_limit_req_merge_conf /* merge location configration */ | |
980 | 97 }; |
98 | |
99 | |
2294 | 100 ngx_module_t ngx_http_limit_req_module = { |
980 | 101 NGX_MODULE_V1, |
2294 | 102 &ngx_http_limit_req_module_ctx, /* module context */ |
103 ngx_http_limit_req_commands, /* module directives */ | |
980 | 104 NGX_HTTP_MODULE, /* module type */ |
105 NULL, /* init master */ | |
106 NULL, /* init module */ | |
107 NULL, /* init process */ | |
108 NULL, /* init thread */ | |
109 NULL, /* exit thread */ | |
110 NULL, /* exit process */ | |
111 NULL, /* exit master */ | |
112 NGX_MODULE_V1_PADDING | |
113 }; | |
114 | |
115 | |
116 static ngx_int_t | |
2294 | 117 ngx_http_limit_req_handler(ngx_http_request_t *r) |
980 | 118 { |
2294 | 119 size_t len, n; |
120 uint32_t hash; | |
121 ngx_int_t rc; | |
2313 | 122 ngx_uint_t excess; |
2294 | 123 ngx_time_t *tp; |
124 ngx_rbtree_node_t *node; | |
125 ngx_http_variable_value_t *vv; | |
126 ngx_http_limit_req_ctx_t *ctx; | |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
127 ngx_http_limit_req_node_t *lr; |
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
128 ngx_http_limit_req_conf_t *lrcf; |
980 | 129 |
2294 | 130 if (r->main->limit_req_set) { |
984
dd128232e6ba
count connection once per request
Igor Sysoev <igor@sysoev.ru>
parents:
981
diff
changeset
|
131 return NGX_DECLINED; |
dd128232e6ba
count connection once per request
Igor Sysoev <igor@sysoev.ru>
parents:
981
diff
changeset
|
132 } |
dd128232e6ba
count connection once per request
Igor Sysoev <igor@sysoev.ru>
parents:
981
diff
changeset
|
133 |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
134 lrcf = ngx_http_get_module_loc_conf(r, ngx_http_limit_req_module); |
980 | 135 |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
136 if (lrcf->shm_zone == NULL) { |
980 | 137 return NGX_DECLINED; |
138 } | |
139 | |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
140 ctx = lrcf->shm_zone->data; |
987 | 141 |
142 vv = ngx_http_get_indexed_variable(r, ctx->index); | |
980 | 143 |
144 if (vv == NULL || vv->not_found) { | |
145 return NGX_DECLINED; | |
146 } | |
147 | |
1011
19118c44303f
test length of variable and number of connections
Igor Sysoev <igor@sysoev.ru>
parents:
1002
diff
changeset
|
148 len = vv->len; |
19118c44303f
test length of variable and number of connections
Igor Sysoev <igor@sysoev.ru>
parents:
1002
diff
changeset
|
149 |
19118c44303f
test length of variable and number of connections
Igor Sysoev <igor@sysoev.ru>
parents:
1002
diff
changeset
|
150 if (len == 0) { |
19118c44303f
test length of variable and number of connections
Igor Sysoev <igor@sysoev.ru>
parents:
1002
diff
changeset
|
151 return NGX_DECLINED; |
19118c44303f
test length of variable and number of connections
Igor Sysoev <igor@sysoev.ru>
parents:
1002
diff
changeset
|
152 } |
984
dd128232e6ba
count connection once per request
Igor Sysoev <igor@sysoev.ru>
parents:
981
diff
changeset
|
153 |
2294 | 154 if (len > 65535) { |
1011
19118c44303f
test length of variable and number of connections
Igor Sysoev <igor@sysoev.ru>
parents:
1002
diff
changeset
|
155 ngx_log_error(NGX_LOG_ERR, r->connection->log, 0, |
19118c44303f
test length of variable and number of connections
Igor Sysoev <igor@sysoev.ru>
parents:
1002
diff
changeset
|
156 "the value of the \"%V\" variable " |
2294 | 157 "is more than 65535 bytes: \"%v\"", |
1011
19118c44303f
test length of variable and number of connections
Igor Sysoev <igor@sysoev.ru>
parents:
1002
diff
changeset
|
158 &ctx->var, vv); |
19118c44303f
test length of variable and number of connections
Igor Sysoev <igor@sysoev.ru>
parents:
1002
diff
changeset
|
159 return NGX_DECLINED; |
19118c44303f
test length of variable and number of connections
Igor Sysoev <igor@sysoev.ru>
parents:
1002
diff
changeset
|
160 } |
19118c44303f
test length of variable and number of connections
Igor Sysoev <igor@sysoev.ru>
parents:
1002
diff
changeset
|
161 |
2294 | 162 r->main->limit_req_set = 1; |
980 | 163 |
164 hash = ngx_crc32_short(vv->data, len); | |
165 | |
2294 | 166 ngx_shmtx_lock(&ctx->shpool->mutex); |
167 | |
168 ngx_http_limit_req_expire(ctx, 1); | |
169 | |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
170 rc = ngx_http_limit_req_lookup(lrcf, hash, vv->data, len, &lr); |
2294 | 171 |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
172 if (lr) { |
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
173 ngx_queue_remove(&lr->queue); |
2294 | 174 |
2720
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
175 ngx_queue_insert_head(&ctx->sh->queue, &lr->queue); |
2294 | 176 |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
177 excess = lr->excess; |
2294 | 178 |
179 } else { | |
2313 | 180 excess = 0; |
980 | 181 } |
182 | |
2313 | 183 ngx_log_debug3(NGX_LOG_DEBUG_HTTP, r->connection->log, 0, |
2973 | 184 "limit_req: %i %ui.%03ui", rc, excess / 1000, excess % 1000); |
1026
38be15c1379a
fix duplicate rbtree keys case
Igor Sysoev <igor@sysoev.ru>
parents:
1012
diff
changeset
|
185 |
2294 | 186 if (rc == NGX_BUSY) { |
187 ngx_shmtx_unlock(&ctx->shpool->mutex); | |
1012
11ffb8e4753f
stop rbtree search early if equal hash was found
Igor Sysoev <igor@sysoev.ru>
parents:
1011
diff
changeset
|
188 |
2294 | 189 ngx_log_error(NGX_LOG_ERR, r->connection->log, 0, |
2376
29d89920a749
*) add zone name while logging 503 error reason
Igor Sysoev <igor@sysoev.ru>
parents:
2375
diff
changeset
|
190 "limiting requests, excess: %ui.%03ui by zone \"%V\"", |
2716
d5896f6608e8
move zone name from ngx_shm_zone_t to ngx_shm_t to use Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2612
diff
changeset
|
191 excess / 1000, excess % 1000, &lrcf->shm_zone->shm.name); |
980 | 192 |
193 return NGX_HTTP_SERVICE_UNAVAILABLE; | |
194 } | |
195 | |
2294 | 196 if (rc == NGX_AGAIN) { |
197 ngx_shmtx_unlock(&ctx->shpool->mutex); | |
198 | |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
199 if (lrcf->nodelay) { |
2300
159136c9808d
*) correct leaky bucket implementation
Igor Sysoev <igor@sysoev.ru>
parents:
2294
diff
changeset
|
200 return NGX_DECLINED; |
2294 | 201 } |
202 | |
2300
159136c9808d
*) correct leaky bucket implementation
Igor Sysoev <igor@sysoev.ru>
parents:
2294
diff
changeset
|
203 ngx_log_error(NGX_LOG_WARN, r->connection->log, 0, |
2376
29d89920a749
*) add zone name while logging 503 error reason
Igor Sysoev <igor@sysoev.ru>
parents:
2375
diff
changeset
|
204 "delaying request, excess: %ui.%03ui, by zone \"%V\"", |
2716
d5896f6608e8
move zone name from ngx_shm_zone_t to ngx_shm_t to use Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2612
diff
changeset
|
205 excess / 1000, excess % 1000, &lrcf->shm_zone->shm.name); |
2300
159136c9808d
*) correct leaky bucket implementation
Igor Sysoev <igor@sysoev.ru>
parents:
2294
diff
changeset
|
206 |
159136c9808d
*) correct leaky bucket implementation
Igor Sysoev <igor@sysoev.ru>
parents:
2294
diff
changeset
|
207 if (ngx_handle_read_event(r->connection->read, 0) != NGX_OK) { |
159136c9808d
*) correct leaky bucket implementation
Igor Sysoev <igor@sysoev.ru>
parents:
2294
diff
changeset
|
208 return NGX_HTTP_INTERNAL_SERVER_ERROR; |
159136c9808d
*) correct leaky bucket implementation
Igor Sysoev <igor@sysoev.ru>
parents:
2294
diff
changeset
|
209 } |
159136c9808d
*) correct leaky bucket implementation
Igor Sysoev <igor@sysoev.ru>
parents:
2294
diff
changeset
|
210 |
159136c9808d
*) correct leaky bucket implementation
Igor Sysoev <igor@sysoev.ru>
parents:
2294
diff
changeset
|
211 r->read_event_handler = ngx_http_test_reading; |
159136c9808d
*) correct leaky bucket implementation
Igor Sysoev <igor@sysoev.ru>
parents:
2294
diff
changeset
|
212 r->write_event_handler = ngx_http_limit_req_delay; |
2313 | 213 ngx_add_timer(r->connection->write, (ngx_msec_t) excess); |
2300
159136c9808d
*) correct leaky bucket implementation
Igor Sysoev <igor@sysoev.ru>
parents:
2294
diff
changeset
|
214 |
159136c9808d
*) correct leaky bucket implementation
Igor Sysoev <igor@sysoev.ru>
parents:
2294
diff
changeset
|
215 return NGX_AGAIN; |
2294 | 216 } |
217 | |
218 if (rc == NGX_OK) { | |
219 goto done; | |
220 } | |
221 | |
222 /* rc == NGX_DECLINED */ | |
223 | |
224 n = offsetof(ngx_rbtree_node_t, color) | |
225 + offsetof(ngx_http_limit_req_node_t, data) | |
226 + len; | |
227 | |
228 node = ngx_slab_alloc_locked(ctx->shpool, n); | |
229 if (node == NULL) { | |
230 | |
231 ngx_http_limit_req_expire(ctx, 0); | |
232 | |
233 node = ngx_slab_alloc_locked(ctx->shpool, n); | |
234 if (node == NULL) { | |
235 ngx_shmtx_unlock(&ctx->shpool->mutex); | |
236 return NGX_HTTP_SERVICE_UNAVAILABLE; | |
237 } | |
238 } | |
239 | |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
240 lr = (ngx_http_limit_req_node_t *) &node->color; |
980 | 241 |
242 node->key = hash; | |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
243 lr->len = (u_char) len; |
2294 | 244 |
245 tp = ngx_timeofday(); | |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
246 lr->last = (ngx_msec_t) (tp->sec * 1000 + tp->msec); |
2294 | 247 |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
248 lr->excess = 0; |
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
249 ngx_memcpy(lr->data, vv->data, len); |
980 | 250 |
2720
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
251 ngx_rbtree_insert(&ctx->sh->rbtree, node); |
980 | 252 |
2720
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
253 ngx_queue_insert_head(&ctx->sh->queue, &lr->queue); |
2294 | 254 |
980 | 255 done: |
256 | |
2294 | 257 ngx_shmtx_unlock(&ctx->shpool->mutex); |
980 | 258 |
259 return NGX_DECLINED; | |
260 } | |
261 | |
262 | |
263 static void | |
2294 | 264 ngx_http_limit_req_delay(ngx_http_request_t *r) |
265 { | |
2972
c5ad288f851d
fix client write event handling in ngx_http_limit_req_module
Igor Sysoev <igor@sysoev.ru>
parents:
2912
diff
changeset
|
266 ngx_event_t *wev; |
c5ad288f851d
fix client write event handling in ngx_http_limit_req_module
Igor Sysoev <igor@sysoev.ru>
parents:
2912
diff
changeset
|
267 |
2294 | 268 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, r->connection->log, 0, |
2973 | 269 "limit_req delay"); |
2294 | 270 |
2972
c5ad288f851d
fix client write event handling in ngx_http_limit_req_module
Igor Sysoev <igor@sysoev.ru>
parents:
2912
diff
changeset
|
271 wev = r->connection->write; |
c5ad288f851d
fix client write event handling in ngx_http_limit_req_module
Igor Sysoev <igor@sysoev.ru>
parents:
2912
diff
changeset
|
272 |
c5ad288f851d
fix client write event handling in ngx_http_limit_req_module
Igor Sysoev <igor@sysoev.ru>
parents:
2912
diff
changeset
|
273 if (!wev->timedout) { |
c5ad288f851d
fix client write event handling in ngx_http_limit_req_module
Igor Sysoev <igor@sysoev.ru>
parents:
2912
diff
changeset
|
274 |
c5ad288f851d
fix client write event handling in ngx_http_limit_req_module
Igor Sysoev <igor@sysoev.ru>
parents:
2912
diff
changeset
|
275 if (ngx_handle_write_event(wev, 0) != NGX_OK) { |
c5ad288f851d
fix client write event handling in ngx_http_limit_req_module
Igor Sysoev <igor@sysoev.ru>
parents:
2912
diff
changeset
|
276 ngx_http_finalize_request(r, NGX_HTTP_INTERNAL_SERVER_ERROR); |
c5ad288f851d
fix client write event handling in ngx_http_limit_req_module
Igor Sysoev <igor@sysoev.ru>
parents:
2912
diff
changeset
|
277 } |
c5ad288f851d
fix client write event handling in ngx_http_limit_req_module
Igor Sysoev <igor@sysoev.ru>
parents:
2912
diff
changeset
|
278 |
c5ad288f851d
fix client write event handling in ngx_http_limit_req_module
Igor Sysoev <igor@sysoev.ru>
parents:
2912
diff
changeset
|
279 return; |
c5ad288f851d
fix client write event handling in ngx_http_limit_req_module
Igor Sysoev <igor@sysoev.ru>
parents:
2912
diff
changeset
|
280 } |
c5ad288f851d
fix client write event handling in ngx_http_limit_req_module
Igor Sysoev <igor@sysoev.ru>
parents:
2912
diff
changeset
|
281 |
c5ad288f851d
fix client write event handling in ngx_http_limit_req_module
Igor Sysoev <igor@sysoev.ru>
parents:
2912
diff
changeset
|
282 wev->timedout = 0; |
c5ad288f851d
fix client write event handling in ngx_http_limit_req_module
Igor Sysoev <igor@sysoev.ru>
parents:
2912
diff
changeset
|
283 |
2294 | 284 if (ngx_handle_read_event(r->connection->read, 0) != NGX_OK) { |
285 ngx_http_finalize_request(r, NGX_HTTP_INTERNAL_SERVER_ERROR); | |
286 return; | |
287 } | |
288 | |
289 r->read_event_handler = ngx_http_block_reading; | |
290 r->write_event_handler = ngx_http_core_run_phases; | |
291 | |
292 ngx_http_core_run_phases(r); | |
293 } | |
294 | |
295 | |
296 static void | |
297 ngx_http_limit_req_rbtree_insert_value(ngx_rbtree_node_t *temp, | |
1026
38be15c1379a
fix duplicate rbtree keys case
Igor Sysoev <igor@sysoev.ru>
parents:
1012
diff
changeset
|
298 ngx_rbtree_node_t *node, ngx_rbtree_node_t *sentinel) |
38be15c1379a
fix duplicate rbtree keys case
Igor Sysoev <igor@sysoev.ru>
parents:
1012
diff
changeset
|
299 { |
2294 | 300 ngx_rbtree_node_t **p; |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
301 ngx_http_limit_req_node_t *lrn, *lrnt; |
1026
38be15c1379a
fix duplicate rbtree keys case
Igor Sysoev <igor@sysoev.ru>
parents:
1012
diff
changeset
|
302 |
38be15c1379a
fix duplicate rbtree keys case
Igor Sysoev <igor@sysoev.ru>
parents:
1012
diff
changeset
|
303 for ( ;; ) { |
38be15c1379a
fix duplicate rbtree keys case
Igor Sysoev <igor@sysoev.ru>
parents:
1012
diff
changeset
|
304 |
38be15c1379a
fix duplicate rbtree keys case
Igor Sysoev <igor@sysoev.ru>
parents:
1012
diff
changeset
|
305 if (node->key < temp->key) { |
38be15c1379a
fix duplicate rbtree keys case
Igor Sysoev <igor@sysoev.ru>
parents:
1012
diff
changeset
|
306 |
1743
4fc402c3ec73
optimize rbtree initialization and insert
Igor Sysoev <igor@sysoev.ru>
parents:
1406
diff
changeset
|
307 p = &temp->left; |
1026
38be15c1379a
fix duplicate rbtree keys case
Igor Sysoev <igor@sysoev.ru>
parents:
1012
diff
changeset
|
308 |
38be15c1379a
fix duplicate rbtree keys case
Igor Sysoev <igor@sysoev.ru>
parents:
1012
diff
changeset
|
309 } else if (node->key > temp->key) { |
38be15c1379a
fix duplicate rbtree keys case
Igor Sysoev <igor@sysoev.ru>
parents:
1012
diff
changeset
|
310 |
1743
4fc402c3ec73
optimize rbtree initialization and insert
Igor Sysoev <igor@sysoev.ru>
parents:
1406
diff
changeset
|
311 p = &temp->right; |
1026
38be15c1379a
fix duplicate rbtree keys case
Igor Sysoev <igor@sysoev.ru>
parents:
1012
diff
changeset
|
312 |
38be15c1379a
fix duplicate rbtree keys case
Igor Sysoev <igor@sysoev.ru>
parents:
1012
diff
changeset
|
313 } else { /* node->key == temp->key */ |
38be15c1379a
fix duplicate rbtree keys case
Igor Sysoev <igor@sysoev.ru>
parents:
1012
diff
changeset
|
314 |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
315 lrn = (ngx_http_limit_req_node_t *) &node->color; |
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
316 lrnt = (ngx_http_limit_req_node_t *) &temp->color; |
1026
38be15c1379a
fix duplicate rbtree keys case
Igor Sysoev <igor@sysoev.ru>
parents:
1012
diff
changeset
|
317 |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
318 p = (ngx_memn2cmp(lrn->data, lrnt->data, lrn->len, lrnt->len) < 0) |
1743
4fc402c3ec73
optimize rbtree initialization and insert
Igor Sysoev <igor@sysoev.ru>
parents:
1406
diff
changeset
|
319 ? &temp->left : &temp->right; |
4fc402c3ec73
optimize rbtree initialization and insert
Igor Sysoev <igor@sysoev.ru>
parents:
1406
diff
changeset
|
320 } |
1026
38be15c1379a
fix duplicate rbtree keys case
Igor Sysoev <igor@sysoev.ru>
parents:
1012
diff
changeset
|
321 |
1743
4fc402c3ec73
optimize rbtree initialization and insert
Igor Sysoev <igor@sysoev.ru>
parents:
1406
diff
changeset
|
322 if (*p == sentinel) { |
4fc402c3ec73
optimize rbtree initialization and insert
Igor Sysoev <igor@sysoev.ru>
parents:
1406
diff
changeset
|
323 break; |
4fc402c3ec73
optimize rbtree initialization and insert
Igor Sysoev <igor@sysoev.ru>
parents:
1406
diff
changeset
|
324 } |
1026
38be15c1379a
fix duplicate rbtree keys case
Igor Sysoev <igor@sysoev.ru>
parents:
1012
diff
changeset
|
325 |
1743
4fc402c3ec73
optimize rbtree initialization and insert
Igor Sysoev <igor@sysoev.ru>
parents:
1406
diff
changeset
|
326 temp = *p; |
1026
38be15c1379a
fix duplicate rbtree keys case
Igor Sysoev <igor@sysoev.ru>
parents:
1012
diff
changeset
|
327 } |
38be15c1379a
fix duplicate rbtree keys case
Igor Sysoev <igor@sysoev.ru>
parents:
1012
diff
changeset
|
328 |
1743
4fc402c3ec73
optimize rbtree initialization and insert
Igor Sysoev <igor@sysoev.ru>
parents:
1406
diff
changeset
|
329 *p = node; |
1026
38be15c1379a
fix duplicate rbtree keys case
Igor Sysoev <igor@sysoev.ru>
parents:
1012
diff
changeset
|
330 node->parent = temp; |
38be15c1379a
fix duplicate rbtree keys case
Igor Sysoev <igor@sysoev.ru>
parents:
1012
diff
changeset
|
331 node->left = sentinel; |
38be15c1379a
fix duplicate rbtree keys case
Igor Sysoev <igor@sysoev.ru>
parents:
1012
diff
changeset
|
332 node->right = sentinel; |
38be15c1379a
fix duplicate rbtree keys case
Igor Sysoev <igor@sysoev.ru>
parents:
1012
diff
changeset
|
333 ngx_rbt_red(node); |
38be15c1379a
fix duplicate rbtree keys case
Igor Sysoev <igor@sysoev.ru>
parents:
1012
diff
changeset
|
334 } |
38be15c1379a
fix duplicate rbtree keys case
Igor Sysoev <igor@sysoev.ru>
parents:
1012
diff
changeset
|
335 |
38be15c1379a
fix duplicate rbtree keys case
Igor Sysoev <igor@sysoev.ru>
parents:
1012
diff
changeset
|
336 |
2294 | 337 static ngx_int_t |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
338 ngx_http_limit_req_lookup(ngx_http_limit_req_conf_t *lrcf, ngx_uint_t hash, |
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
339 u_char *data, size_t len, ngx_http_limit_req_node_t **lrp) |
980 | 340 { |
2313 | 341 ngx_int_t rc, excess; |
2294 | 342 ngx_time_t *tp; |
343 ngx_msec_t now; | |
344 ngx_msec_int_t ms; | |
345 ngx_rbtree_node_t *node, *sentinel; | |
346 ngx_http_limit_req_ctx_t *ctx; | |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
347 ngx_http_limit_req_node_t *lr; |
2294 | 348 |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
349 ctx = lrcf->shm_zone->data; |
2294 | 350 |
2720
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
351 node = ctx->sh->rbtree.root; |
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
352 sentinel = ctx->sh->rbtree.sentinel; |
980 | 353 |
2294 | 354 while (node != sentinel) { |
355 | |
356 if (hash < node->key) { | |
357 node = node->left; | |
358 continue; | |
359 } | |
360 | |
361 if (hash > node->key) { | |
362 node = node->right; | |
363 continue; | |
364 } | |
365 | |
366 /* hash == node->key */ | |
367 | |
368 do { | |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
369 lr = (ngx_http_limit_req_node_t *) &node->color; |
980 | 370 |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
371 rc = ngx_memn2cmp(data, lr->data, len, (size_t) lr->len); |
2294 | 372 |
373 if (rc == 0) { | |
374 | |
375 tp = ngx_timeofday(); | |
980 | 376 |
2294 | 377 now = (ngx_msec_t) (tp->sec * 1000 + tp->msec); |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
378 ms = (ngx_msec_int_t) (now - lr->last); |
2294 | 379 |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
380 excess = lr->excess - ctx->rate * ngx_abs(ms) / 1000 + 1000; |
2294 | 381 |
3183
b87542338ac3
make limit_req to conform to the leaky bucket algorithm
Igor Sysoev <igor@sysoev.ru>
parents:
2973
diff
changeset
|
382 if ((ngx_uint_t) excess > lrcf->burst) { |
b87542338ac3
make limit_req to conform to the leaky bucket algorithm
Igor Sysoev <igor@sysoev.ru>
parents:
2973
diff
changeset
|
383 *lrp = lr; |
b87542338ac3
make limit_req to conform to the leaky bucket algorithm
Igor Sysoev <igor@sysoev.ru>
parents:
2973
diff
changeset
|
384 return NGX_BUSY; |
b87542338ac3
make limit_req to conform to the leaky bucket algorithm
Igor Sysoev <igor@sysoev.ru>
parents:
2973
diff
changeset
|
385 } |
b87542338ac3
make limit_req to conform to the leaky bucket algorithm
Igor Sysoev <igor@sysoev.ru>
parents:
2973
diff
changeset
|
386 |
2313 | 387 if (excess < 0) { |
388 excess = 0; | |
2294 | 389 } |
390 | |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
391 lr->excess = excess; |
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
392 lr->last = now; |
980 | 393 |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
394 *lrp = lr; |
2294 | 395 |
2313 | 396 if (excess) { |
2294 | 397 return NGX_AGAIN; |
398 } | |
980 | 399 |
2294 | 400 return NGX_OK; |
401 } | |
402 | |
403 node = (rc < 0) ? node->left : node->right; | |
404 | |
405 } while (node != sentinel && hash == node->key); | |
406 | |
407 break; | |
980 | 408 } |
409 | |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
410 *lrp = NULL; |
2294 | 411 |
412 return NGX_DECLINED; | |
413 } | |
414 | |
415 | |
416 static void | |
417 ngx_http_limit_req_expire(ngx_http_limit_req_ctx_t *ctx, ngx_uint_t n) | |
418 { | |
2313 | 419 ngx_int_t excess; |
2294 | 420 ngx_time_t *tp; |
421 ngx_msec_t now; | |
422 ngx_queue_t *q; | |
423 ngx_msec_int_t ms; | |
424 ngx_rbtree_node_t *node; | |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
425 ngx_http_limit_req_node_t *lr; |
2294 | 426 |
427 tp = ngx_timeofday(); | |
428 | |
429 now = (ngx_msec_t) (tp->sec * 1000 + tp->msec); | |
430 | |
431 /* | |
432 * n == 1 deletes one or two zero rate entries | |
433 * n == 0 deletes oldest entry by force | |
434 * and one or two zero rate entries | |
435 */ | |
436 | |
437 while (n < 3) { | |
438 | |
2720
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
439 if (ngx_queue_empty(&ctx->sh->queue)) { |
2294 | 440 return; |
441 } | |
442 | |
2720
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
443 q = ngx_queue_last(&ctx->sh->queue); |
2294 | 444 |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
445 lr = ngx_queue_data(q, ngx_http_limit_req_node_t, queue); |
2294 | 446 |
447 if (n++ != 0) { | |
448 | |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
449 ms = (ngx_msec_int_t) (now - lr->last); |
2294 | 450 ms = ngx_abs(ms); |
451 | |
452 if (ms < 60000) { | |
453 return; | |
454 } | |
455 | |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
456 excess = lr->excess - ctx->rate * ms / 1000; |
2294 | 457 |
2313 | 458 if (excess > 0) { |
2294 | 459 return; |
460 } | |
461 } | |
462 | |
463 ngx_queue_remove(q); | |
464 | |
465 node = (ngx_rbtree_node_t *) | |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
466 ((u_char *) lr - offsetof(ngx_rbtree_node_t, color)); |
2294 | 467 |
2720
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
468 ngx_rbtree_delete(&ctx->sh->rbtree, node); |
2294 | 469 |
470 ngx_slab_free_locked(ctx->shpool, node); | |
471 } | |
980 | 472 } |
473 | |
474 | |
475 static ngx_int_t | |
2294 | 476 ngx_http_limit_req_init_zone(ngx_shm_zone_t *shm_zone, void *data) |
980 | 477 { |
2294 | 478 ngx_http_limit_req_ctx_t *octx = data; |
993
1b9a4d92173f
pass the inherited shm_zone data
Igor Sysoev <igor@sysoev.ru>
parents:
987
diff
changeset
|
479 |
2611
2bce3f6416c6
improve ngx_slab_alloc() error logging
Igor Sysoev <igor@sysoev.ru>
parents:
2376
diff
changeset
|
480 size_t len; |
2294 | 481 ngx_http_limit_req_ctx_t *ctx; |
980 | 482 |
993
1b9a4d92173f
pass the inherited shm_zone data
Igor Sysoev <igor@sysoev.ru>
parents:
987
diff
changeset
|
483 ctx = shm_zone->data; |
1b9a4d92173f
pass the inherited shm_zone data
Igor Sysoev <igor@sysoev.ru>
parents:
987
diff
changeset
|
484 |
1b9a4d92173f
pass the inherited shm_zone data
Igor Sysoev <igor@sysoev.ru>
parents:
987
diff
changeset
|
485 if (octx) { |
1b9a4d92173f
pass the inherited shm_zone data
Igor Sysoev <igor@sysoev.ru>
parents:
987
diff
changeset
|
486 if (ngx_strcmp(ctx->var.data, octx->var.data) != 0) { |
1b9a4d92173f
pass the inherited shm_zone data
Igor Sysoev <igor@sysoev.ru>
parents:
987
diff
changeset
|
487 ngx_log_error(NGX_LOG_EMERG, shm_zone->shm.log, 0, |
2294 | 488 "limit_req \"%V\" uses the \"%V\" variable " |
993
1b9a4d92173f
pass the inherited shm_zone data
Igor Sysoev <igor@sysoev.ru>
parents:
987
diff
changeset
|
489 "while previously it used the \"%V\" variable", |
2716
d5896f6608e8
move zone name from ngx_shm_zone_t to ngx_shm_t to use Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2612
diff
changeset
|
490 &shm_zone->shm.name, &ctx->var, &octx->var); |
993
1b9a4d92173f
pass the inherited shm_zone data
Igor Sysoev <igor@sysoev.ru>
parents:
987
diff
changeset
|
491 return NGX_ERROR; |
1b9a4d92173f
pass the inherited shm_zone data
Igor Sysoev <igor@sysoev.ru>
parents:
987
diff
changeset
|
492 } |
1b9a4d92173f
pass the inherited shm_zone data
Igor Sysoev <igor@sysoev.ru>
parents:
987
diff
changeset
|
493 |
2720
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
494 ctx->sh = octx->sh; |
2294 | 495 ctx->shpool = octx->shpool; |
993
1b9a4d92173f
pass the inherited shm_zone data
Igor Sysoev <igor@sysoev.ru>
parents:
987
diff
changeset
|
496 |
1b9a4d92173f
pass the inherited shm_zone data
Igor Sysoev <igor@sysoev.ru>
parents:
987
diff
changeset
|
497 return NGX_OK; |
1b9a4d92173f
pass the inherited shm_zone data
Igor Sysoev <igor@sysoev.ru>
parents:
987
diff
changeset
|
498 } |
1b9a4d92173f
pass the inherited shm_zone data
Igor Sysoev <igor@sysoev.ru>
parents:
987
diff
changeset
|
499 |
2294 | 500 ctx->shpool = (ngx_slab_pool_t *) shm_zone->shm.addr; |
980 | 501 |
2720
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
502 if (shm_zone->shm.exists) { |
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
503 ctx->sh = ctx->shpool->data; |
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
504 |
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
505 return NGX_OK; |
980 | 506 } |
507 | |
2720
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
508 ctx->sh = ngx_slab_alloc(ctx->shpool, sizeof(ngx_http_limit_req_shctx_t)); |
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
509 if (ctx->sh == NULL) { |
980 | 510 return NGX_ERROR; |
511 } | |
512 | |
2720
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
513 ctx->shpool->data = ctx->sh; |
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
514 |
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
515 ngx_rbtree_init(&ctx->sh->rbtree, &ctx->sh->sentinel, |
2294 | 516 ngx_http_limit_req_rbtree_insert_value); |
517 | |
2720
b3b8c66bd520
support attaching to an existent Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2716
diff
changeset
|
518 ngx_queue_init(&ctx->sh->queue); |
980 | 519 |
2716
d5896f6608e8
move zone name from ngx_shm_zone_t to ngx_shm_t to use Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2612
diff
changeset
|
520 len = sizeof(" in limit_req zone \"\"") + shm_zone->shm.name.len; |
2611
2bce3f6416c6
improve ngx_slab_alloc() error logging
Igor Sysoev <igor@sysoev.ru>
parents:
2376
diff
changeset
|
521 |
2bce3f6416c6
improve ngx_slab_alloc() error logging
Igor Sysoev <igor@sysoev.ru>
parents:
2376
diff
changeset
|
522 ctx->shpool->log_ctx = ngx_slab_alloc(ctx->shpool, len); |
2bce3f6416c6
improve ngx_slab_alloc() error logging
Igor Sysoev <igor@sysoev.ru>
parents:
2376
diff
changeset
|
523 if (ctx->shpool->log_ctx == NULL) { |
2bce3f6416c6
improve ngx_slab_alloc() error logging
Igor Sysoev <igor@sysoev.ru>
parents:
2376
diff
changeset
|
524 return NGX_ERROR; |
2bce3f6416c6
improve ngx_slab_alloc() error logging
Igor Sysoev <igor@sysoev.ru>
parents:
2376
diff
changeset
|
525 } |
2bce3f6416c6
improve ngx_slab_alloc() error logging
Igor Sysoev <igor@sysoev.ru>
parents:
2376
diff
changeset
|
526 |
2bce3f6416c6
improve ngx_slab_alloc() error logging
Igor Sysoev <igor@sysoev.ru>
parents:
2376
diff
changeset
|
527 ngx_sprintf(ctx->shpool->log_ctx, " in limit_req zone \"%V\"%Z", |
2716
d5896f6608e8
move zone name from ngx_shm_zone_t to ngx_shm_t to use Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2612
diff
changeset
|
528 &shm_zone->shm.name); |
2611
2bce3f6416c6
improve ngx_slab_alloc() error logging
Igor Sysoev <igor@sysoev.ru>
parents:
2376
diff
changeset
|
529 |
980 | 530 return NGX_OK; |
531 } | |
532 | |
533 | |
534 static void * | |
2294 | 535 ngx_http_limit_req_create_conf(ngx_conf_t *cf) |
980 | 536 { |
2294 | 537 ngx_http_limit_req_conf_t *conf; |
980 | 538 |
2294 | 539 conf = ngx_pcalloc(cf->pool, sizeof(ngx_http_limit_req_conf_t)); |
980 | 540 if (conf == NULL) { |
2912
c7d57b539248
return NULL instead of NGX_CONF_ERROR on a create conf failure
Igor Sysoev <igor@sysoev.ru>
parents:
2720
diff
changeset
|
541 return NULL; |
980 | 542 } |
543 | |
544 /* | |
545 * set by ngx_pcalloc(): | |
546 * | |
547 * conf->shm_zone = NULL; | |
2375 | 548 * conf->burst = 0; |
2300
159136c9808d
*) correct leaky bucket implementation
Igor Sysoev <igor@sysoev.ru>
parents:
2294
diff
changeset
|
549 * conf->nodelay = 0; |
980 | 550 */ |
551 | |
552 return conf; | |
553 } | |
554 | |
555 | |
556 static char * | |
2294 | 557 ngx_http_limit_req_merge_conf(ngx_conf_t *cf, void *parent, void *child) |
980 | 558 { |
2294 | 559 ngx_http_limit_req_conf_t *prev = parent; |
560 ngx_http_limit_req_conf_t *conf = child; | |
980 | 561 |
562 if (conf->shm_zone == NULL) { | |
563 *conf = *prev; | |
564 } | |
565 | |
566 return NGX_CONF_OK; | |
567 } | |
568 | |
569 | |
570 static char * | |
2294 | 571 ngx_http_limit_req_zone(ngx_conf_t *cf, ngx_command_t *cmd, void *conf) |
980 | 572 { |
2294 | 573 u_char *p; |
574 size_t size, len; | |
575 ngx_str_t *value, name, s; | |
576 ngx_int_t rate, scale; | |
577 ngx_uint_t i; | |
578 ngx_shm_zone_t *shm_zone; | |
579 ngx_http_limit_req_ctx_t *ctx; | |
980 | 580 |
581 value = cf->args->elts; | |
582 | |
2294 | 583 ctx = NULL; |
584 size = 0; | |
585 rate = 1; | |
586 scale = 1; | |
587 name.len = 0; | |
588 | |
589 for (i = 1; i < cf->args->nelts; i++) { | |
590 | |
591 if (ngx_strncmp(value[i].data, "zone=", 5) == 0) { | |
592 | |
593 name.data = value[i].data + 5; | |
594 | |
595 p = (u_char *) ngx_strchr(name.data, ':'); | |
596 | |
597 if (p) { | |
2716
d5896f6608e8
move zone name from ngx_shm_zone_t to ngx_shm_t to use Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2612
diff
changeset
|
598 *p = '\0'; |
d5896f6608e8
move zone name from ngx_shm_zone_t to ngx_shm_t to use Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2612
diff
changeset
|
599 |
2294 | 600 name.len = p - name.data; |
601 | |
602 p++; | |
603 | |
604 s.len = value[i].data + value[i].len - p; | |
605 s.data = p; | |
606 | |
607 size = ngx_parse_size(&s); | |
608 if (size > 8191) { | |
609 continue; | |
610 } | |
611 } | |
612 | |
613 ngx_conf_log_error(NGX_LOG_EMERG, cf, 0, | |
614 "invalid zone size \"%V\"", &value[i]); | |
615 return NGX_CONF_ERROR; | |
616 } | |
617 | |
618 if (ngx_strncmp(value[i].data, "rate=", 5) == 0) { | |
619 | |
620 len = value[i].len; | |
621 p = value[i].data + len - 3; | |
987 | 622 |
2294 | 623 if (ngx_strncmp(p, "r/s", 3) == 0) { |
624 scale = 1; | |
625 len -= 3; | |
626 | |
627 } else if (ngx_strncmp(p, "r/m", 3) == 0) { | |
628 scale = 60; | |
629 len -= 3; | |
630 } | |
631 | |
632 rate = ngx_atoi(value[i].data + 5, len - 5); | |
633 if (rate <= NGX_ERROR) { | |
634 ngx_conf_log_error(NGX_LOG_EMERG, cf, 0, | |
635 "invalid rate \"%V\"", &value[i]); | |
636 return NGX_CONF_ERROR; | |
637 } | |
638 | |
639 continue; | |
640 } | |
641 | |
642 if (value[i].data[0] == '$') { | |
987 | 643 |
2294 | 644 value[i].len--; |
645 value[i].data++; | |
646 | |
647 ctx = ngx_pcalloc(cf->pool, sizeof(ngx_http_limit_req_ctx_t)); | |
648 if (ctx == NULL) { | |
649 return NGX_CONF_ERROR; | |
650 } | |
651 | |
652 ctx->index = ngx_http_get_variable_index(cf, &value[i]); | |
653 if (ctx->index == NGX_ERROR) { | |
654 return NGX_CONF_ERROR; | |
655 } | |
656 | |
657 ctx->var = value[i]; | |
658 | |
659 continue; | |
660 } | |
661 | |
662 ngx_conf_log_error(NGX_LOG_EMERG, cf, 0, | |
663 "invalid parameter \"%V\"", &value[i]); | |
987 | 664 return NGX_CONF_ERROR; |
665 } | |
666 | |
2294 | 667 if (name.len == 0 || size == 0) { |
668 ngx_conf_log_error(NGX_LOG_EMERG, cf, 0, | |
669 "\"%V\" must have \"zone\" parameter", | |
670 &cmd->name); | |
987 | 671 return NGX_CONF_ERROR; |
672 } | |
673 | |
2294 | 674 if (ctx == NULL) { |
980 | 675 ngx_conf_log_error(NGX_LOG_EMERG, cf, 0, |
2294 | 676 "no variable is defined for limit_req_zone \"%V\"", |
677 &cmd->name); | |
980 | 678 return NGX_CONF_ERROR; |
679 } | |
680 | |
2313 | 681 ctx->rate = rate * 1000 / scale; |
980 | 682 |
2294 | 683 shm_zone = ngx_shared_memory_add(cf, &name, size, |
684 &ngx_http_limit_req_module); | |
980 | 685 if (shm_zone == NULL) { |
686 return NGX_CONF_ERROR; | |
687 } | |
688 | |
987 | 689 if (shm_zone->data) { |
690 ctx = shm_zone->data; | |
691 | |
692 ngx_conf_log_error(NGX_LOG_EMERG, cf, 0, | |
2294 | 693 "limit_req_zone \"%V\" is already bound to variable \"%V\"", |
694 &value[1], &ctx->var); | |
987 | 695 return NGX_CONF_ERROR; |
696 } | |
697 | |
2294 | 698 shm_zone->init = ngx_http_limit_req_init_zone; |
987 | 699 shm_zone->data = ctx; |
980 | 700 |
701 return NGX_CONF_OK; | |
702 } | |
703 | |
704 | |
705 static char * | |
2294 | 706 ngx_http_limit_req(ngx_conf_t *cf, ngx_command_t *cmd, void *conf) |
980 | 707 { |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
708 ngx_http_limit_req_conf_t *lrcf = conf; |
980 | 709 |
2300
159136c9808d
*) correct leaky bucket implementation
Igor Sysoev <igor@sysoev.ru>
parents:
2294
diff
changeset
|
710 ngx_int_t burst; |
159136c9808d
*) correct leaky bucket implementation
Igor Sysoev <igor@sysoev.ru>
parents:
2294
diff
changeset
|
711 ngx_str_t *value, s; |
159136c9808d
*) correct leaky bucket implementation
Igor Sysoev <igor@sysoev.ru>
parents:
2294
diff
changeset
|
712 ngx_uint_t i; |
2294 | 713 |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
714 if (lrcf->shm_zone) { |
2294 | 715 return "is duplicate"; |
716 } | |
980 | 717 |
718 value = cf->args->elts; | |
719 | |
2294 | 720 burst = 0; |
721 | |
722 for (i = 1; i < cf->args->nelts; i++) { | |
723 | |
724 if (ngx_strncmp(value[i].data, "zone=", 5) == 0) { | |
725 | |
726 s.len = value[i].len - 5; | |
727 s.data = value[i].data + 5; | |
728 | |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
729 lrcf->shm_zone = ngx_shared_memory_add(cf, &s, 0, |
2294 | 730 &ngx_http_limit_req_module); |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
731 if (lrcf->shm_zone == NULL) { |
2294 | 732 return NGX_CONF_ERROR; |
733 } | |
734 | |
735 continue; | |
736 } | |
737 | |
738 if (ngx_strncmp(value[i].data, "burst=", 6) == 0) { | |
739 | |
2300
159136c9808d
*) correct leaky bucket implementation
Igor Sysoev <igor@sysoev.ru>
parents:
2294
diff
changeset
|
740 burst = ngx_atoi(value[i].data + 6, value[i].len - 6); |
2294 | 741 if (burst <= 0) { |
742 ngx_conf_log_error(NGX_LOG_EMERG, cf, 0, | |
743 "invalid burst rate \"%V\"", &value[i]); | |
744 return NGX_CONF_ERROR; | |
745 } | |
746 | |
747 continue; | |
748 } | |
749 | |
2300
159136c9808d
*) correct leaky bucket implementation
Igor Sysoev <igor@sysoev.ru>
parents:
2294
diff
changeset
|
750 if (ngx_strncmp(value[i].data, "nodelay", 7) == 0) { |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
751 lrcf->nodelay = 1; |
2294 | 752 continue; |
753 } | |
754 | |
755 ngx_conf_log_error(NGX_LOG_EMERG, cf, 0, | |
756 "invalid parameter \"%V\"", &value[i]); | |
980 | 757 return NGX_CONF_ERROR; |
758 } | |
759 | |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
760 if (lrcf->shm_zone == NULL) { |
980 | 761 ngx_conf_log_error(NGX_LOG_EMERG, cf, 0, |
2294 | 762 "\"%V\" must have \"zone\" parameter", |
763 &cmd->name); | |
980 | 764 return NGX_CONF_ERROR; |
765 } | |
766 | |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
767 if (lrcf->shm_zone->data == NULL) { |
1011
19118c44303f
test length of variable and number of connections
Igor Sysoev <igor@sysoev.ru>
parents:
1002
diff
changeset
|
768 ngx_conf_log_error(NGX_LOG_EMERG, cf, 0, |
2294 | 769 "unknown limit_req_zone \"%V\"", |
2716
d5896f6608e8
move zone name from ngx_shm_zone_t to ngx_shm_t to use Win32 shared memory
Igor Sysoev <igor@sysoev.ru>
parents:
2612
diff
changeset
|
770 &lrcf->shm_zone->shm.name); |
1011
19118c44303f
test length of variable and number of connections
Igor Sysoev <igor@sysoev.ru>
parents:
1002
diff
changeset
|
771 return NGX_CONF_ERROR; |
19118c44303f
test length of variable and number of connections
Igor Sysoev <igor@sysoev.ru>
parents:
1002
diff
changeset
|
772 } |
19118c44303f
test length of variable and number of connections
Igor Sysoev <igor@sysoev.ru>
parents:
1002
diff
changeset
|
773 |
2374
7b11f9a1bfe1
rename "lz" to "lr" in variable names
Igor Sysoev <igor@sysoev.ru>
parents:
2313
diff
changeset
|
774 lrcf->burst = burst * 1000; |
980 | 775 |
776 return NGX_CONF_OK; | |
777 } | |
778 | |
779 | |
780 static ngx_int_t | |
2294 | 781 ngx_http_limit_req_init(ngx_conf_t *cf) |
980 | 782 { |
783 ngx_http_handler_pt *h; | |
784 ngx_http_core_main_conf_t *cmcf; | |
785 | |
786 cmcf = ngx_http_conf_get_module_main_conf(cf, ngx_http_core_module); | |
787 | |
788 h = ngx_array_push(&cmcf->phases[NGX_HTTP_PREACCESS_PHASE].handlers); | |
789 if (h == NULL) { | |
790 return NGX_ERROR; | |
791 } | |
792 | |
2294 | 793 *h = ngx_http_limit_req_handler; |
980 | 794 |
795 return NGX_OK; | |
796 } |