blob: 0fa08bc87fde448a2805bd5e39dc11a3fb5dcf82 [file] [log] [blame]
Igor Sysoev7578ec92003-06-02 15:24:30 +00001
Igor Sysoevd90282d2004-09-28 08:34:51 +00002/*
Igor Sysoevff8da912004-09-29 16:00:49 +00003 * Copyright (C) Igor Sysoev
Igor Sysoevd90282d2004-09-28 08:34:51 +00004 */
5
6
Igor Sysoev7578ec92003-06-02 15:24:30 +00007#include <ngx_config.h>
8#include <ngx_core.h>
9#include <ngx_http.h>
10
11
Igor Sysoev899b44e2005-05-12 14:58:06 +000012#define ngx_http_script_exit (u_char *) &ngx_http_script_exit_code
13
14static uintptr_t ngx_http_script_exit_code = (uintptr_t) NULL;
15
16
17ngx_uint_t
18ngx_http_script_variables_count(ngx_str_t *value)
Igor Sysoev7578ec92003-06-02 15:24:30 +000019{
Igor Sysoev899b44e2005-05-12 14:58:06 +000020 ngx_uint_t i, n;
Igor Sysoev02025fd2005-01-18 13:03:58 +000021
Igor Sysoev899b44e2005-05-12 14:58:06 +000022 for (n = 0, i = 0; i < value->len; i++) {
23 if (value->data[i] == '$') {
24 n++;
25 }
Igor Sysoev02f742b2005-04-08 15:18:55 +000026 }
Igor Sysoev02025fd2005-01-18 13:03:58 +000027
Igor Sysoev899b44e2005-05-12 14:58:06 +000028 return n;
29}
30
31
32ngx_int_t
33ngx_http_script_compile(ngx_http_script_compile_t *sc)
34{
35 u_char ch;
36 size_t size;
Igor Sysoev09c684b2005-11-09 17:25:55 +000037 ngx_int_t index, *p;
Igor Sysoev899b44e2005-05-12 14:58:06 +000038 ngx_str_t name;
39 uintptr_t *code;
40 ngx_uint_t i, n, bracket;
41 ngx_http_script_var_code_t *var_code;
42 ngx_http_script_copy_code_t *copy;
43 ngx_http_script_copy_capture_code_t *copy_capture;
44
Igor Sysoev09c684b2005-11-09 17:25:55 +000045 if (sc->flushes && *sc->flushes == NULL) {
46 n = sc->variables ? sc->variables : 1;
47 *sc->flushes = ngx_array_create(sc->cf->pool, n, sizeof(ngx_uint_t));
48 if (*sc->flushes == NULL) {
49 return NGX_ERROR;
50 }
51 }
52
53
Igor Sysoev899b44e2005-05-12 14:58:06 +000054 if (*sc->lengths == NULL) {
55 n = sc->variables * (2 * sizeof(ngx_http_script_copy_code_t)
56 + sizeof(ngx_http_script_var_code_t))
57 + sizeof(uintptr_t);
58
59 *sc->lengths = ngx_array_create(sc->cf->pool, n, 1);
60 if (*sc->lengths == NULL) {
Igor Sysoev02f742b2005-04-08 15:18:55 +000061 return NGX_ERROR;
62 }
63 }
Igor Sysoev02025fd2005-01-18 13:03:58 +000064
Igor Sysoev899b44e2005-05-12 14:58:06 +000065
66 if (*sc->values == NULL) {
67 n = (sc->variables * (2 * sizeof(ngx_http_script_copy_code_t)
68 + sizeof(ngx_http_script_var_code_t))
69 + sizeof(uintptr_t)
70 + sc->source->len
71 + sizeof(uintptr_t) - 1)
72 & ~(sizeof(uintptr_t) - 1);
73
74 *sc->values = ngx_array_create(sc->cf->pool, n, 1);
75 if (*sc->values == NULL) {
Igor Sysoev02f742b2005-04-08 15:18:55 +000076 return NGX_ERROR;
77 }
78 }
79
Igor Sysoev899b44e2005-05-12 14:58:06 +000080 sc->variables = 0;
Igor Sysoev02f742b2005-04-08 15:18:55 +000081
Igor Sysoev899b44e2005-05-12 14:58:06 +000082 for (i = 0; i < sc->source->len; /* void */ ) {
83
84 name.len = 0;
85
86 if (sc->source->data[i] == '$') {
87
88 if (++i == sc->source->len) {
89 goto invalid_variable;
90 }
91
92 if (sc->source->data[i] >= '1' && sc->source->data[i] <= '9') {
93
Igor Sysoev7f7846d2006-04-26 09:52:47 +000094 n = sc->source->data[i] - '0';
95
96 if (sc->captures_mask & (1 << n)) {
97 sc->dup_capture = 1;
98 }
99
100 sc->captures_mask |= 1 << n;
101
Igor Sysoev899b44e2005-05-12 14:58:06 +0000102 copy_capture = ngx_http_script_add_code(*sc->lengths,
103 sizeof(ngx_http_script_copy_capture_code_t),
104 NULL);
105 if (copy_capture == NULL) {
106 return NGX_ERROR;
107 }
108
109 copy_capture->code = (ngx_http_script_code_pt)
110 ngx_http_script_copy_capture_len_code;
Igor Sysoev7f7846d2006-04-26 09:52:47 +0000111 copy_capture->n = 2 * n;
112
Igor Sysoev899b44e2005-05-12 14:58:06 +0000113
114 copy_capture = ngx_http_script_add_code(*sc->values,
115 sizeof(ngx_http_script_copy_capture_code_t),
116 &sc->main);
117 if (copy_capture == NULL) {
118 return NGX_ERROR;
119 }
120
121 copy_capture->code = ngx_http_script_copy_capture_code;
Igor Sysoev7f7846d2006-04-26 09:52:47 +0000122 copy_capture->n = 2 * n;
Igor Sysoev899b44e2005-05-12 14:58:06 +0000123
Igor Sysoev7f7846d2006-04-26 09:52:47 +0000124 if (sc->ncaptures < n) {
125 sc->ncaptures = n;
Igor Sysoev899b44e2005-05-12 14:58:06 +0000126 }
127
Igor Sysoev899b44e2005-05-12 14:58:06 +0000128 i++;
129
130 continue;
131 }
132
133 if (sc->source->data[i] == '{') {
134 bracket = 1;
135
136 if (++i == sc->source->len) {
137 goto invalid_variable;
138 }
139
140 name.data = &sc->source->data[i];
141
142 } else {
143 bracket = 0;
144 name.data = &sc->source->data[i];
145 }
146
147 for ( /* void */ ; i < sc->source->len; i++, name.len++) {
148 ch = sc->source->data[i];
149
150 if (ch == '}' && bracket) {
151 i++;
152 bracket = 0;
153 break;
154 }
155
156 if ((ch >= 'A' && ch <= 'Z')
157 || (ch >= 'a' && ch <= 'z')
158 || (ch >= '0' && ch <= '9')
159 || ch == '_')
160 {
161 continue;
162 }
163
164 break;
165 }
166
167 if (bracket) {
168 ngx_conf_log_error(NGX_LOG_EMERG, sc->cf, 0,
169 "the closing bracket in \"%V\" "
170 "variable is missing", &name);
Igor Sysoev02f742b2005-04-08 15:18:55 +0000171 return NGX_ERROR;
172 }
173
Igor Sysoev899b44e2005-05-12 14:58:06 +0000174 if (name.len == 0) {
175 goto invalid_variable;
176 }
Igor Sysoev02f742b2005-04-08 15:18:55 +0000177
Igor Sysoev899b44e2005-05-12 14:58:06 +0000178 sc->variables++;
Igor Sysoev02f742b2005-04-08 15:18:55 +0000179
Igor Sysoev899b44e2005-05-12 14:58:06 +0000180 index = ngx_http_get_variable_index(sc->cf, &name);
181
182 if (index == NGX_ERROR) {
Igor Sysoev02f742b2005-04-08 15:18:55 +0000183 return NGX_ERROR;
184 }
185
Igor Sysoev09c684b2005-11-09 17:25:55 +0000186 if (sc->flushes) {
187 p = ngx_array_push(*sc->flushes);
188 if (p == NULL) {
189 return NGX_ERROR;
190 }
191
192 *p = index;
193 }
194
Igor Sysoev899b44e2005-05-12 14:58:06 +0000195 var_code = ngx_http_script_add_code(*sc->lengths,
196 sizeof(ngx_http_script_var_code_t),
197 NULL);
Igor Sysoev02f742b2005-04-08 15:18:55 +0000198 if (var_code == NULL) {
199 return NGX_ERROR;
200 }
201
202 var_code->code = (ngx_http_script_code_pt)
Igor Sysoev899b44e2005-05-12 14:58:06 +0000203 ngx_http_script_copy_var_len_code;
204 var_code->index = (uintptr_t) index;
Igor Sysoev02f742b2005-04-08 15:18:55 +0000205
206
Igor Sysoev899b44e2005-05-12 14:58:06 +0000207 var_code = ngx_http_script_add_code(*sc->values,
208 sizeof(ngx_http_script_var_code_t),
209 &sc->main);
Igor Sysoev02f742b2005-04-08 15:18:55 +0000210 if (var_code == NULL) {
211 return NGX_ERROR;
212 }
213
Igor Sysoev899b44e2005-05-12 14:58:06 +0000214 var_code->code = ngx_http_script_copy_var_code;
215 var_code->index = (uintptr_t) index;
Igor Sysoev02f742b2005-04-08 15:18:55 +0000216
217 continue;
218 }
219
Igor Sysoev899b44e2005-05-12 14:58:06 +0000220 if (sc->source->data[i] == '?' && sc->compile_args) {
221 sc->args = 1;
222 sc->compile_args = 0;
223
224 code = ngx_http_script_add_code(*sc->values, sizeof(uintptr_t),
225 &sc->main);
226 if (code == NULL) {
227 return NGX_ERROR;
228 }
229
230 *code = (uintptr_t) ngx_http_script_start_args_code;
231
232 i++;
233
234 continue;
235 }
236
237 name.data = &sc->source->data[i];
238
239 while (i < sc->source->len
240 && sc->source->data[i] != '$'
241 && !(sc->source->data[i] == '?' && sc->compile_args))
242 {
243 i++;
244 name.len++;
245 }
246
247 sc->size += name.len;
248
249 copy = ngx_http_script_add_code(*sc->lengths,
250 sizeof(ngx_http_script_copy_code_t),
251 NULL);
252 if (copy == NULL) {
Igor Sysoev02f742b2005-04-08 15:18:55 +0000253 return NGX_ERROR;
254 }
Igor Sysoev899b44e2005-05-12 14:58:06 +0000255
256 copy->code = (ngx_http_script_code_pt) ngx_http_script_copy_len_code;
257 copy->len = name.len;
258
259 size = (sizeof(ngx_http_script_copy_code_t) + name.len
260 + sizeof(uintptr_t) - 1)
261 & ~(sizeof(uintptr_t) - 1);
262
263 copy = ngx_http_script_add_code(*sc->values, size, &sc->main);
264 if (copy == NULL) {
265 return NGX_ERROR;
266 }
267
268 copy->code = ngx_http_script_copy_code;
269 copy->len = name.len;
270
271 ngx_memcpy((u_char *) copy + sizeof(ngx_http_script_copy_code_t),
272 name.data, name.len);
Igor Sysoev02f742b2005-04-08 15:18:55 +0000273 }
274
Igor Sysoev899b44e2005-05-12 14:58:06 +0000275 if (sc->complete_lengths) {
276 code = ngx_http_script_add_code(*sc->lengths, sizeof(uintptr_t), NULL);
277 if (code == NULL) {
278 return NGX_ERROR;
279 }
280
281 *code = (uintptr_t) NULL;
Igor Sysoev02f742b2005-04-08 15:18:55 +0000282 }
283
Igor Sysoev899b44e2005-05-12 14:58:06 +0000284 if (sc->complete_values) {
285 code = ngx_http_script_add_code(*sc->values, sizeof(uintptr_t),
286 &sc->main);
287 if (code == NULL) {
288 return NGX_ERROR;
289 }
Igor Sysoev02f742b2005-04-08 15:18:55 +0000290
Igor Sysoev899b44e2005-05-12 14:58:06 +0000291 *code = (uintptr_t) NULL;
Igor Sysoev02f742b2005-04-08 15:18:55 +0000292 }
293
Igor Sysoev02f742b2005-04-08 15:18:55 +0000294 return NGX_OK;
Igor Sysoev899b44e2005-05-12 14:58:06 +0000295
296invalid_variable:
297
298 ngx_conf_log_error(NGX_LOG_EMERG, sc->cf, 0, "invalid variable name");
299
300 return NGX_ERROR;
Igor Sysoev7578ec92003-06-02 15:24:30 +0000301}
302
303
Igor Sysoev8fea8852006-03-15 09:53:04 +0000304u_char *
305ngx_http_script_run(ngx_http_request_t *r, ngx_str_t *value,
306 void *code_lengths, size_t len, void *code_values)
307{
Igor Sysoev2c8f0572007-03-30 19:00:34 +0000308 ngx_uint_t i;
309 ngx_http_script_code_pt code;
310 ngx_http_script_len_code_pt lcode;
311 ngx_http_script_engine_t e;
312 ngx_http_core_main_conf_t *cmcf;
313
314 cmcf = ngx_http_get_module_main_conf(r, ngx_http_core_module);
315
316 for (i = 0; i < cmcf->variables.nelts; i++) {
Igor Sysoev2d3f3f62007-10-14 18:56:15 +0000317 if (r->variables[i].no_cacheable) {
Igor Sysoev2c8f0572007-03-30 19:00:34 +0000318 r->variables[i].valid = 0;
319 r->variables[i].not_found = 0;
320 }
321 }
Igor Sysoev8fea8852006-03-15 09:53:04 +0000322
323 ngx_memzero(&e, sizeof(ngx_http_script_engine_t));
324
325 e.ip = code_lengths;
326 e.request = r;
327 e.flushed = 1;
328
329 while (*(uintptr_t *) e.ip) {
330 lcode = *(ngx_http_script_len_code_pt *) e.ip;
331 len += lcode(&e);
332 }
333
334
335 value->len = len;
336 value->data = ngx_palloc(r->pool, len);
337 if (value->data == NULL) {
338 return NULL;
339 }
340
341 e.ip = code_values;
342 e.pos = value->data;
343
344 while (*(uintptr_t *) e.ip) {
345 code = *(ngx_http_script_code_pt *) e.ip;
346 code((ngx_http_script_engine_t *) &e);
347 }
348
349 return e.pos;
350}
351
352
Igor Sysoev09c684b2005-11-09 17:25:55 +0000353void
Igor Sysoev2d3f3f62007-10-14 18:56:15 +0000354ngx_http_script_flush_no_cacheable_variables(ngx_http_request_t *r,
Igor Sysoev09c684b2005-11-09 17:25:55 +0000355 ngx_array_t *indices)
356{
357 ngx_uint_t n, *index;
358
359 if (indices) {
360 index = indices->elts;
361 for (n = 0; n < indices->nelts; n++) {
Igor Sysoev2d3f3f62007-10-14 18:56:15 +0000362 if (r->variables[index[n]].no_cacheable) {
Igor Sysoev09c684b2005-11-09 17:25:55 +0000363 r->variables[index[n]].valid = 0;
364 r->variables[index[n]].not_found = 0;
365 }
366 }
367 }
368}
369
370
Igor Sysoev899b44e2005-05-12 14:58:06 +0000371void *
Igor Sysoev02f742b2005-04-08 15:18:55 +0000372ngx_http_script_start_code(ngx_pool_t *pool, ngx_array_t **codes, size_t size)
Igor Sysoev7578ec92003-06-02 15:24:30 +0000373{
Igor Sysoev02f742b2005-04-08 15:18:55 +0000374 if (*codes == NULL) {
375 *codes = ngx_array_create(pool, 256, 1);
376 if (*codes == NULL) {
377 return NULL;
378 }
379 }
Igor Sysoev7578ec92003-06-02 15:24:30 +0000380
Igor Sysoev02f742b2005-04-08 15:18:55 +0000381 return ngx_array_push_n(*codes, size);
382}
Igor Sysoev7578ec92003-06-02 15:24:30 +0000383
Igor Sysoev899b44e2005-05-12 14:58:06 +0000384
385void *
386ngx_http_script_add_code(ngx_array_t *codes, size_t size, void *code)
387{
388 u_char *elts, **p;
389 void *new;
390
391 elts = codes->elts;
392
393 new = ngx_array_push_n(codes, size);
394 if (new == NULL) {
395 return NGX_CONF_ERROR;
396 }
397
398 if (code) {
399 if (elts != codes->elts) {
400 p = code;
401 *p += (u_char *) codes->elts - elts;
402 }
403 }
404
405 return new;
406}
Igor Sysoev02025fd2005-01-18 13:03:58 +0000407
Igor Sysoev02f742b2005-04-08 15:18:55 +0000408
409size_t
Igor Sysoev899b44e2005-05-12 14:58:06 +0000410ngx_http_script_copy_len_code(ngx_http_script_engine_t *e)
Igor Sysoev02f742b2005-04-08 15:18:55 +0000411{
412 ngx_http_script_copy_code_t *code;
413
Igor Sysoev899b44e2005-05-12 14:58:06 +0000414 code = (ngx_http_script_copy_code_t *) e->ip;
Igor Sysoev02f742b2005-04-08 15:18:55 +0000415
Igor Sysoev899b44e2005-05-12 14:58:06 +0000416 e->ip += sizeof(ngx_http_script_copy_code_t);
Igor Sysoev02f742b2005-04-08 15:18:55 +0000417
418 return code->len;
Igor Sysoev7578ec92003-06-02 15:24:30 +0000419}
420
421
Igor Sysoev02f742b2005-04-08 15:18:55 +0000422void
Igor Sysoev899b44e2005-05-12 14:58:06 +0000423ngx_http_script_copy_code(ngx_http_script_engine_t *e)
Igor Sysoev7578ec92003-06-02 15:24:30 +0000424{
Igor Sysoev02f742b2005-04-08 15:18:55 +0000425 ngx_http_script_copy_code_t *code;
426
Igor Sysoev899b44e2005-05-12 14:58:06 +0000427 code = (ngx_http_script_copy_code_t *) e->ip;
Igor Sysoev02f742b2005-04-08 15:18:55 +0000428
Igor Sysoev899b44e2005-05-12 14:58:06 +0000429 if (!e->skip) {
Igor Sysoev09c684b2005-11-09 17:25:55 +0000430 e->pos = ngx_copy(e->pos, e->ip + sizeof(ngx_http_script_copy_code_t),
431 code->len);
Igor Sysoev899b44e2005-05-12 14:58:06 +0000432 }
Igor Sysoev02f742b2005-04-08 15:18:55 +0000433
Igor Sysoev899b44e2005-05-12 14:58:06 +0000434 e->ip += sizeof(ngx_http_script_copy_code_t)
435 + ((code->len + sizeof(uintptr_t) - 1) & ~(sizeof(uintptr_t) - 1));
436
Igor Sysoev58364232006-11-14 12:45:03 +0000437 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, e->request->connection->log, 0,
438 "http script copy: \"%V\"", &e->buf);
Igor Sysoev7578ec92003-06-02 15:24:30 +0000439}
440
441
Igor Sysoev02f742b2005-04-08 15:18:55 +0000442size_t
Igor Sysoev899b44e2005-05-12 14:58:06 +0000443ngx_http_script_copy_var_len_code(ngx_http_script_engine_t *e)
Igor Sysoev7578ec92003-06-02 15:24:30 +0000444{
Igor Sysoev02f742b2005-04-08 15:18:55 +0000445 ngx_http_variable_value_t *value;
446 ngx_http_script_var_code_t *code;
447
Igor Sysoev899b44e2005-05-12 14:58:06 +0000448 code = (ngx_http_script_var_code_t *) e->ip;
Igor Sysoev02f742b2005-04-08 15:18:55 +0000449
Igor Sysoev899b44e2005-05-12 14:58:06 +0000450 e->ip += sizeof(ngx_http_script_var_code_t);
Igor Sysoev02f742b2005-04-08 15:18:55 +0000451
Igor Sysoev09c684b2005-11-09 17:25:55 +0000452 if (e->flushed) {
453 value = ngx_http_get_indexed_variable(e->request, code->index);
Igor Sysoev02f742b2005-04-08 15:18:55 +0000454
Igor Sysoev09c684b2005-11-09 17:25:55 +0000455 } else {
456 value = ngx_http_get_flushed_variable(e->request, code->index);
457 }
458
459 if (value && !value->not_found) {
460 return value->len;
Igor Sysoev02f742b2005-04-08 15:18:55 +0000461 }
462
Igor Sysoevf6e1fe32005-10-04 10:38:53 +0000463 return 0;
Igor Sysoev7578ec92003-06-02 15:24:30 +0000464}
465
466
Igor Sysoev02f742b2005-04-08 15:18:55 +0000467void
Igor Sysoev899b44e2005-05-12 14:58:06 +0000468ngx_http_script_copy_var_code(ngx_http_script_engine_t *e)
Igor Sysoev7578ec92003-06-02 15:24:30 +0000469{
Igor Sysoev02f742b2005-04-08 15:18:55 +0000470 ngx_http_variable_value_t *value;
471 ngx_http_script_var_code_t *code;
472
Igor Sysoev899b44e2005-05-12 14:58:06 +0000473 code = (ngx_http_script_var_code_t *) e->ip;
Igor Sysoev02f742b2005-04-08 15:18:55 +0000474
Igor Sysoev899b44e2005-05-12 14:58:06 +0000475 e->ip += sizeof(ngx_http_script_var_code_t);
Igor Sysoev02f742b2005-04-08 15:18:55 +0000476
Igor Sysoev899b44e2005-05-12 14:58:06 +0000477 if (!e->skip) {
Igor Sysoev02f742b2005-04-08 15:18:55 +0000478
Igor Sysoev09c684b2005-11-09 17:25:55 +0000479 if (e->flushed) {
480 value = ngx_http_get_indexed_variable(e->request, code->index);
481
482 } else {
483 value = ngx_http_get_flushed_variable(e->request, code->index);
484 }
485
486 if (value && !value->not_found) {
487 e->pos = ngx_copy(e->pos, value->data, value->len);
Igor Sysoev899b44e2005-05-12 14:58:06 +0000488
Igor Sysoev58364232006-11-14 12:45:03 +0000489 ngx_log_debug1(NGX_LOG_DEBUG_HTTP,
490 e->request->connection->log, 0,
491 "http script var: \"%V\"", &e->buf);
Igor Sysoev899b44e2005-05-12 14:58:06 +0000492 }
493 }
494}
495
496
497size_t
498ngx_http_script_copy_capture_len_code(ngx_http_script_engine_t *e)
499{
500 ngx_http_script_copy_capture_code_t *code;
501
502 code = (ngx_http_script_copy_capture_code_t *) e->ip;
503
504 e->ip += sizeof(ngx_http_script_copy_capture_code_t);
505
Igor Sysoeve31e90b2005-05-19 13:25:22 +0000506 if (code->n < e->ncaptures) {
507 if ((e->args || e->quote)
508 && (e->request->quoted_uri || e->request->plus_in_uri))
509 {
510 return e->captures[code->n + 1] - e->captures[code->n]
Igor Sysoevaf3b7ea2006-05-31 14:11:45 +0000511 + 2 * ngx_escape_uri(NULL,
Igor Sysoeve31e90b2005-05-19 13:25:22 +0000512 &e->line.data[e->captures[code->n]],
Igor Sysoev899b44e2005-05-12 14:58:06 +0000513 e->captures[code->n + 1] - e->captures[code->n],
514 NGX_ESCAPE_ARGS);
Igor Sysoeve31e90b2005-05-19 13:25:22 +0000515 } else {
516 return e->captures[code->n + 1] - e->captures[code->n];
517 }
Igor Sysoev899b44e2005-05-12 14:58:06 +0000518 }
Igor Sysoeve31e90b2005-05-19 13:25:22 +0000519
520 return 0;
Igor Sysoev899b44e2005-05-12 14:58:06 +0000521}
522
523
524void
525ngx_http_script_copy_capture_code(ngx_http_script_engine_t *e)
526{
527 ngx_http_script_copy_capture_code_t *code;
528
529 code = (ngx_http_script_copy_capture_code_t *) e->ip;
530
531 e->ip += sizeof(ngx_http_script_copy_capture_code_t);
532
Igor Sysoeve31e90b2005-05-19 13:25:22 +0000533 if (code->n < e->ncaptures) {
534 if ((e->args || e->quote)
535 && (e->request->quoted_uri || e->request->plus_in_uri))
536 {
537 e->pos = (u_char *) ngx_escape_uri(e->pos,
538 &e->line.data[e->captures[code->n]],
Igor Sysoev899b44e2005-05-12 14:58:06 +0000539 e->captures[code->n + 1] - e->captures[code->n],
540 NGX_ESCAPE_ARGS);
Igor Sysoeve31e90b2005-05-19 13:25:22 +0000541 } else {
Igor Sysoev09c684b2005-11-09 17:25:55 +0000542 e->pos = ngx_copy(e->pos,
543 &e->line.data[e->captures[code->n]],
544 e->captures[code->n + 1] - e->captures[code->n]);
Igor Sysoeve31e90b2005-05-19 13:25:22 +0000545 }
Igor Sysoev899b44e2005-05-12 14:58:06 +0000546 }
547
548 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, e->request->connection->log, 0,
549 "http script capture: \"%V\"", &e->buf);
550}
551
552
553void
554ngx_http_script_start_args_code(ngx_http_script_engine_t *e)
555{
556 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, e->request->connection->log, 0,
557 "http script args");
Igor Sysoev0e5dc5c2005-11-15 13:30:52 +0000558
Igor Sysoev899b44e2005-05-12 14:58:06 +0000559 e->args = e->pos;
560 e->ip += sizeof(uintptr_t);
561}
562
563
Igor Sysoev4959ec42005-05-23 12:07:45 +0000564
565#if (NGX_PCRE)
566
Igor Sysoev899b44e2005-05-12 14:58:06 +0000567void
568ngx_http_script_regex_start_code(ngx_http_script_engine_t *e)
569{
570 size_t len;
571 ngx_int_t rc;
572 ngx_uint_t n;
573 ngx_http_request_t *r;
574 ngx_http_script_engine_t le;
575 ngx_http_script_len_code_pt lcode;
576 ngx_http_script_regex_code_t *code;
577
578 code = (ngx_http_script_regex_code_t *) e->ip;
579
580 r = e->request;
581
582 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
583 "http script regex: \"%V\"", &code->name);
584
585 if (code->uri) {
Igor Sysoeve31e90b2005-05-19 13:25:22 +0000586 e->line = r->uri;
Igor Sysoev899b44e2005-05-12 14:58:06 +0000587 } else {
588 e->sp--;
Igor Sysoev09c684b2005-11-09 17:25:55 +0000589 e->line.len = e->sp->len;
590 e->line.data = e->sp->data;
Igor Sysoev899b44e2005-05-12 14:58:06 +0000591 }
592
Igor Sysoeve31e90b2005-05-19 13:25:22 +0000593 rc = ngx_regex_exec(code->regex, &e->line, e->captures, code->ncaptures);
Igor Sysoev899b44e2005-05-12 14:58:06 +0000594
595 if (rc == NGX_REGEX_NO_MATCHED) {
596 if (e->log) {
597 ngx_log_error(NGX_LOG_NOTICE, r->connection->log, 0,
Igor Sysoeve31e90b2005-05-19 13:25:22 +0000598 "\"%V\" does not match \"%V\"",
599 &code->name, &e->line);
Igor Sysoev899b44e2005-05-12 14:58:06 +0000600 }
601
Igor Sysoeve31e90b2005-05-19 13:25:22 +0000602 e->ncaptures = 0;
603
Igor Sysoev899b44e2005-05-12 14:58:06 +0000604 if (code->test) {
Igor Sysoev94e32ce2006-04-07 14:08:04 +0000605 if (code->negative_test) {
606 e->sp->len = 1;
607 e->sp->data = (u_char *) "1";
608
609 } else {
610 e->sp->len = 0;
611 e->sp->data = (u_char *) "";
612 }
613
Igor Sysoev899b44e2005-05-12 14:58:06 +0000614 e->sp++;
615
616 e->ip += sizeof(ngx_http_script_regex_code_t);
617 return;
618 }
619
620 e->ip += code->next;
Igor Sysoev02f742b2005-04-08 15:18:55 +0000621 return;
622 }
623
Igor Sysoev899b44e2005-05-12 14:58:06 +0000624 if (rc < 0) {
625 ngx_log_error(NGX_LOG_ALERT, r->connection->log, 0,
626 ngx_regex_exec_n " failed: %d on \"%V\" using \"%V\"",
Igor Sysoeve31e90b2005-05-19 13:25:22 +0000627 rc, &e->line, &code->name);
Igor Sysoev899b44e2005-05-12 14:58:06 +0000628
629 e->ip = ngx_http_script_exit;
630 e->status = NGX_HTTP_INTERNAL_SERVER_ERROR;
631 return;
632 }
633
634 if (e->log) {
635 ngx_log_error(NGX_LOG_NOTICE, r->connection->log, 0,
Igor Sysoeve31e90b2005-05-19 13:25:22 +0000636 "\"%V\" matches \"%V\"", &code->name, &e->line);
Igor Sysoev899b44e2005-05-12 14:58:06 +0000637 }
638
Igor Sysoeve31e90b2005-05-19 13:25:22 +0000639 e->ncaptures = code->ncaptures;
640
Igor Sysoev899b44e2005-05-12 14:58:06 +0000641 if (code->test) {
Igor Sysoev94e32ce2006-04-07 14:08:04 +0000642 if (code->negative_test) {
643 e->sp->len = 0;
644 e->sp->data = (u_char *) "";
645
646 } else {
647 e->sp->len = 1;
648 e->sp->data = (u_char *) "1";
649 }
650
Igor Sysoev899b44e2005-05-12 14:58:06 +0000651 e->sp++;
652
653 e->ip += sizeof(ngx_http_script_regex_code_t);
654 return;
655 }
656
657 if (code->status) {
658 e->status = code->status;
659
660 if (!code->redirect) {
661 e->ip = ngx_http_script_exit;
662 return;
663 }
664 }
665
666 if (code->uri) {
667 r->internal = 1;
668 r->valid_unparsed_uri = 0;
669
670 if (code->break_cycle) {
671 r->valid_location = 0;
Igor Sysoev5192b362005-07-08 14:34:20 +0000672 r->uri_changed = 0;
Igor Sysoev899b44e2005-05-12 14:58:06 +0000673
674 } else {
675 r->uri_changed = 1;
676 }
677 }
678
679 if (code->lengths == NULL) {
680 e->buf.len = code->size;
681
682 if (code->uri) {
683 if (rc && (r->quoted_uri || r->plus_in_uri)) {
684 e->buf.len += 2 * ngx_escape_uri(NULL, r->uri.data, r->uri.len,
685 NGX_ESCAPE_ARGS);
686 }
687 }
688
689 for (n = 1; n < (ngx_uint_t) rc; n++) {
690 e->buf.len += e->captures[2 * n + 1] - e->captures[2 * n];
691 }
692
693 } else {
694 ngx_memzero(&le, sizeof(ngx_http_script_engine_t));
695
696 le.ip = code->lengths->elts;
Igor Sysoev3f8dc592006-08-28 16:57:48 +0000697 le.line = e->line;
Igor Sysoev899b44e2005-05-12 14:58:06 +0000698 le.request = r;
699 le.captures = e->captures;
Igor Sysoeve31e90b2005-05-19 13:25:22 +0000700 le.ncaptures = e->ncaptures;
Igor Sysoev3f8dc592006-08-28 16:57:48 +0000701 le.quote = code->redirect;
Igor Sysoev899b44e2005-05-12 14:58:06 +0000702
703 len = 1; /* reserve 1 byte for possible "?" */
704
705 while (*(uintptr_t *) le.ip) {
706 lcode = *(ngx_http_script_len_code_pt *) le.ip;
707 len += lcode(&le);
708 }
709
710 e->buf.len = len;
711 }
712
Igor Sysoeve31e90b2005-05-19 13:25:22 +0000713 if (code->add_args && r->args.len) {
Igor Sysoev899b44e2005-05-12 14:58:06 +0000714 e->buf.len += r->args.len + 1;
715 }
716
717 e->buf.data = ngx_palloc(r->pool, e->buf.len);
718 if (e->buf.data == NULL) {
719 e->ip = ngx_http_script_exit;
720 e->status = NGX_HTTP_INTERNAL_SERVER_ERROR;
721 return;
722 }
723
724 e->quote = code->redirect;
725
726 e->pos = e->buf.data;
727
728 e->ip += sizeof(ngx_http_script_regex_code_t);
729}
730
731
732void
733ngx_http_script_regex_end_code(ngx_http_script_engine_t *e)
734{
Igor Sysoev0e5dc5c2005-11-15 13:30:52 +0000735 u_char *dst, *src;
Igor Sysoev899b44e2005-05-12 14:58:06 +0000736 ngx_http_request_t *r;
737 ngx_http_script_regex_end_code_t *code;
738
739 code = (ngx_http_script_regex_end_code_t *) e->ip;
740
741 r = e->request;
742
743 e->quote = 0;
744
745 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
746 "http script regex end");
747
748 if (code->redirect) {
749
Igor Sysoev0e5dc5c2005-11-15 13:30:52 +0000750 dst = e->buf.data;
751 src = e->buf.data;
752
Igor Sysoevae33d012006-01-17 20:04:32 +0000753 ngx_unescape_uri(&dst, &src, e->pos - e->buf.data, NGX_UNESCAPE_URI);
Igor Sysoev0e5dc5c2005-11-15 13:30:52 +0000754
755 if (src < e->pos) {
756 dst = ngx_copy(dst, src, e->pos - src);
757 }
758
759 e->pos = dst;
760
761 if (code->add_args && r->args.len) {
Igor Sysoev899b44e2005-05-12 14:58:06 +0000762 *e->pos++ = (u_char) (code->args ? '&' : '?');
Igor Sysoev09c684b2005-11-09 17:25:55 +0000763 e->pos = ngx_copy(e->pos, r->args.data, r->args.len);
Igor Sysoev899b44e2005-05-12 14:58:06 +0000764 }
765
766 e->buf.len = e->pos - e->buf.data;
767
768 if (e->log) {
769 ngx_log_error(NGX_LOG_NOTICE, r->connection->log, 0,
770 "rewritten redirect: \"%V\"", &e->buf);
771 }
772
773 r->headers_out.location = ngx_list_push(&r->headers_out.headers);
774 if (r->headers_out.location == NULL) {
775 e->ip = ngx_http_script_exit;
776 e->status = NGX_HTTP_INTERNAL_SERVER_ERROR;
777 return;
778 }
779
Igor Sysoev0e5dc5c2005-11-15 13:30:52 +0000780 r->headers_out.location->hash = 1;
Igor Sysoev899b44e2005-05-12 14:58:06 +0000781 r->headers_out.location->key.len = sizeof("Location") - 1;
782 r->headers_out.location->key.data = (u_char *) "Location";
783 r->headers_out.location->value = e->buf;
784
785 e->ip += sizeof(ngx_http_script_regex_end_code_t);
786 return;
787 }
788
Igor Sysoev0e5dc5c2005-11-15 13:30:52 +0000789 if (e->args) {
Igor Sysoev899b44e2005-05-12 14:58:06 +0000790 e->buf.len = e->args - e->buf.data;
791
792 if (code->add_args && r->args.len) {
793 *e->pos++ = '&';
Igor Sysoev09c684b2005-11-09 17:25:55 +0000794 e->pos = ngx_copy(e->pos, r->args.data, r->args.len);
Igor Sysoev899b44e2005-05-12 14:58:06 +0000795 }
796
797 r->args.len = e->pos - e->args;
798 r->args.data = e->args;
799
800 e->args = NULL;
801
802 } else {
803 e->buf.len = e->pos - e->buf.data;
Igor Sysoev08e63d42006-08-14 15:09:38 +0000804
805 if (!code->add_args) {
806 r->args.len = 0;
807 }
Igor Sysoev899b44e2005-05-12 14:58:06 +0000808 }
809
810 if (e->log) {
811 ngx_log_error(NGX_LOG_NOTICE, r->connection->log, 0,
812 "rewritten data: \"%V\", args: \"%V\"",
813 &e->buf, &r->args);
814 }
815
816 if (code->uri) {
817 r->uri = e->buf;
818
Igor Sysoevb85fd592005-08-23 15:36:54 +0000819 if (r->uri.len == 0) {
820 ngx_log_error(NGX_LOG_ERR, r->connection->log, 0,
821 "the rewritten URI has a zero length");
822 e->ip = ngx_http_script_exit;
823 e->status = NGX_HTTP_INTERNAL_SERVER_ERROR;
824 return;
825 }
826
Igor Sysoev899b44e2005-05-12 14:58:06 +0000827 if (ngx_http_set_exten(r) != NGX_OK) {
828 e->ip = ngx_http_script_exit;
829 e->status = NGX_HTTP_INTERNAL_SERVER_ERROR;
830 return;
831 }
832 }
833
834 e->ip += sizeof(ngx_http_script_regex_end_code_t);
835}
836
Igor Sysoev4959ec42005-05-23 12:07:45 +0000837#endif
838
Igor Sysoev899b44e2005-05-12 14:58:06 +0000839
840void
841ngx_http_script_return_code(ngx_http_script_engine_t *e)
Igor Sysoev0e5dc5c2005-11-15 13:30:52 +0000842{
Igor Sysoev899b44e2005-05-12 14:58:06 +0000843 ngx_http_script_return_code_t *code;
Igor Sysoev0e5dc5c2005-11-15 13:30:52 +0000844
Igor Sysoev899b44e2005-05-12 14:58:06 +0000845 code = (ngx_http_script_return_code_t *) e->ip;
846
847 e->status = code->status;
848
Igor Sysoevafd7ec52006-05-29 17:28:12 +0000849 if (code->status == NGX_HTTP_NO_CONTENT) {
850 e->request->header_only = 1;
Igor Sysoev8fd830a2006-10-02 10:22:51 +0000851 e->request->zero_body = 1;
Igor Sysoevafd7ec52006-05-29 17:28:12 +0000852 }
853
Igor Sysoev899b44e2005-05-12 14:58:06 +0000854 e->ip += sizeof(ngx_http_script_return_code_t) - sizeof(uintptr_t);
855}
856
857
858void
Igor Sysoev5192b362005-07-08 14:34:20 +0000859ngx_http_script_break_code(ngx_http_script_engine_t *e)
860{
861 e->request->uri_changed = 0;
862
863 e->ip = ngx_http_script_exit;
864}
865
866
867void
Igor Sysoev899b44e2005-05-12 14:58:06 +0000868ngx_http_script_if_code(ngx_http_script_engine_t *e)
869{
870 ngx_http_script_if_code_t *code;
871
872 code = (ngx_http_script_if_code_t *) e->ip;
873
874 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, e->request->connection->log, 0,
875 "http script if");
876
877 e->sp--;
878
Igor Sysoev09c684b2005-11-09 17:25:55 +0000879 if (e->sp->len && e->sp->data[0] != '0') {
Igor Sysoev899b44e2005-05-12 14:58:06 +0000880 if (code->loc_conf) {
881 e->request->loc_conf = code->loc_conf;
Igor Sysoevb85fd592005-08-23 15:36:54 +0000882 ngx_http_update_location_config(e->request);
Igor Sysoev899b44e2005-05-12 14:58:06 +0000883 }
884
885 e->ip += sizeof(ngx_http_script_if_code_t);
886 return;
887 }
888
889 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, e->request->connection->log, 0,
Igor Sysoev0e5dc5c2005-11-15 13:30:52 +0000890 "http script if: false");
Igor Sysoev899b44e2005-05-12 14:58:06 +0000891
892 e->ip += code->next;
893}
894
895
896void
Igor Sysoev0e5dc5c2005-11-15 13:30:52 +0000897ngx_http_script_equal_code(ngx_http_script_engine_t *e)
898{
899 ngx_http_variable_value_t *val, *res;
900
901 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, e->request->connection->log, 0,
902 "http script equal");
903
904 e->sp--;
905 val = e->sp;
906 res = e->sp - 1;
907
908 e->ip += sizeof(uintptr_t);
909
910 if (val->len == res->len && ngx_strncmp(val->data, res->data, res->len)
911 == 0)
912 {
913 *res = ngx_http_variable_true_value;
914 return;
915 }
916
917 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, e->request->connection->log, 0,
918 "http script equal: no");
919
920 *res = ngx_http_variable_null_value;
921}
922
923
924void
925ngx_http_script_not_equal_code(ngx_http_script_engine_t *e)
926{
927 ngx_http_variable_value_t *val, *res;
928
929 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, e->request->connection->log, 0,
930 "http script not equal");
931
932 e->sp--;
933 val = e->sp;
934 res = e->sp - 1;
935
936 e->ip += sizeof(uintptr_t);
937
938 if (val->len == res->len && ngx_strncmp(val->data, res->data, res->len)
939 == 0)
940 {
941 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, e->request->connection->log, 0,
942 "http script not equal: no");
943
944 *res = ngx_http_variable_null_value;
945 return;
946 }
947
948 *res = ngx_http_variable_true_value;
949}
950
951
952void
Igor Sysoev94e32ce2006-04-07 14:08:04 +0000953ngx_http_script_file_code(ngx_http_script_engine_t *e)
954{
Igor Sysoev140c7552007-09-01 12:12:48 +0000955 ngx_str_t path;
956 ngx_http_request_t *r;
957 ngx_open_file_info_t of;
958 ngx_http_core_loc_conf_t *clcf;
Igor Sysoev94e32ce2006-04-07 14:08:04 +0000959 ngx_http_variable_value_t *value;
960 ngx_http_script_file_code_t *code;
961
962 value = e->sp - 1;
963
964 code = (ngx_http_script_file_code_t *) e->ip;
965 e->ip += sizeof(ngx_http_script_file_code_t);
966
Igor Sysoev140c7552007-09-01 12:12:48 +0000967 path.len = value->len - 1;
968 path.data = value->data;
Igor Sysoev94e32ce2006-04-07 14:08:04 +0000969
Igor Sysoev140c7552007-09-01 12:12:48 +0000970 r = e->request;
Igor Sysoev94e32ce2006-04-07 14:08:04 +0000971
Igor Sysoev140c7552007-09-01 12:12:48 +0000972 ngx_log_debug2(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
973 "http script file op %p \"%V\"", code->op, &path);
974
975 clcf = ngx_http_get_module_loc_conf(r, ngx_http_core_module);
976
977 of.test_dir = 0;
978 of.retest = clcf->open_file_cache_retest;
979 of.errors = clcf->open_file_cache_errors;
Igor Sysoev9afd58f2007-09-03 08:41:42 +0000980 of.events = clcf->open_file_cache_events;
Igor Sysoev140c7552007-09-01 12:12:48 +0000981
982 if (ngx_open_cached_file(clcf->open_file_cache, &path, &of, r->pool)
983 == NGX_ERROR)
984 {
985 if (of.err != NGX_ENOENT && of.err != NGX_ENOTDIR) {
986 ngx_log_error(NGX_LOG_CRIT, r->connection->log, of.err,
Igor Sysoev94e32ce2006-04-07 14:08:04 +0000987 ngx_file_info_n " \"%s\" failed", value->data);
988 }
989
990 switch (code->op) {
Igor Sysoevc55a1042006-08-09 19:59:45 +0000991
Igor Sysoev94e32ce2006-04-07 14:08:04 +0000992 case ngx_http_script_file_plain:
Igor Sysoevb71c6902006-08-04 16:04:04 +0000993 case ngx_http_script_file_dir:
994 case ngx_http_script_file_exists:
995 case ngx_http_script_file_exec:
Igor Sysoev94e32ce2006-04-07 14:08:04 +0000996 goto false;
Igor Sysoevc55a1042006-08-09 19:59:45 +0000997
Igor Sysoev94e32ce2006-04-07 14:08:04 +0000998 case ngx_http_script_file_not_plain:
Igor Sysoevb71c6902006-08-04 16:04:04 +0000999 case ngx_http_script_file_not_dir:
Igor Sysoevc55a1042006-08-09 19:59:45 +00001000 case ngx_http_script_file_not_exists:
Igor Sysoevb71c6902006-08-04 16:04:04 +00001001 case ngx_http_script_file_not_exec:
Igor Sysoev94e32ce2006-04-07 14:08:04 +00001002 goto true;
1003 }
1004
1005 goto false;
1006 }
1007
1008 switch (code->op) {
1009 case ngx_http_script_file_plain:
Igor Sysoev140c7552007-09-01 12:12:48 +00001010 if (of.is_file) {
Igor Sysoev94e32ce2006-04-07 14:08:04 +00001011 goto true;
1012 }
1013 goto false;
1014
1015 case ngx_http_script_file_not_plain:
Igor Sysoev140c7552007-09-01 12:12:48 +00001016 if (of.is_file) {
Igor Sysoev94e32ce2006-04-07 14:08:04 +00001017 goto false;
1018 }
1019 goto true;
Igor Sysoevb71c6902006-08-04 16:04:04 +00001020
1021 case ngx_http_script_file_dir:
Igor Sysoev140c7552007-09-01 12:12:48 +00001022 if (of.is_dir) {
Igor Sysoevb71c6902006-08-04 16:04:04 +00001023 goto true;
1024 }
1025 goto false;
1026
1027 case ngx_http_script_file_not_dir:
Igor Sysoev140c7552007-09-01 12:12:48 +00001028 if (of.is_dir) {
Igor Sysoevb71c6902006-08-04 16:04:04 +00001029 goto false;
1030 }
1031 goto true;
1032
1033 case ngx_http_script_file_exists:
Igor Sysoev140c7552007-09-01 12:12:48 +00001034 if (of.is_file || of.is_dir || of.is_link) {
Igor Sysoevb71c6902006-08-04 16:04:04 +00001035 goto true;
1036 }
1037 goto false;
1038
1039 case ngx_http_script_file_not_exists:
Igor Sysoev140c7552007-09-01 12:12:48 +00001040 if (of.is_file || of.is_dir || of.is_link) {
Igor Sysoevb71c6902006-08-04 16:04:04 +00001041 goto false;
1042 }
1043 goto true;
1044
Igor Sysoevb71c6902006-08-04 16:04:04 +00001045 case ngx_http_script_file_exec:
Igor Sysoev140c7552007-09-01 12:12:48 +00001046 if (of.is_exec) {
Igor Sysoevb71c6902006-08-04 16:04:04 +00001047 goto true;
1048 }
1049 goto false;
1050
1051 case ngx_http_script_file_not_exec:
Igor Sysoev140c7552007-09-01 12:12:48 +00001052 if (of.is_exec) {
Igor Sysoevb71c6902006-08-04 16:04:04 +00001053 goto false;
1054 }
1055 goto true;
Igor Sysoev94e32ce2006-04-07 14:08:04 +00001056 }
1057
1058false:
1059
Igor Sysoev140c7552007-09-01 12:12:48 +00001060 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
Igor Sysoev94e32ce2006-04-07 14:08:04 +00001061 "http script file op false");
1062
1063 *value = ngx_http_variable_null_value;
1064 return;
1065
1066true:
1067
1068 *value = ngx_http_variable_true_value;
1069 return;
1070}
1071
1072
1073void
Igor Sysoeve31e90b2005-05-19 13:25:22 +00001074ngx_http_script_complex_value_code(ngx_http_script_engine_t *e)
1075{
1076 size_t len;
1077 ngx_http_script_engine_t le;
1078 ngx_http_script_len_code_pt lcode;
1079 ngx_http_script_complex_value_code_t *code;
1080
1081 code = (ngx_http_script_complex_value_code_t *) e->ip;
1082
1083 e->ip += sizeof(ngx_http_script_complex_value_code_t);
1084
1085 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, e->request->connection->log, 0,
1086 "http script complex value");
1087
1088 ngx_memzero(&le, sizeof(ngx_http_script_engine_t));
1089
1090 le.ip = code->lengths->elts;
Igor Sysoevaf3b7ea2006-05-31 14:11:45 +00001091 le.line = e->line;
Igor Sysoeve31e90b2005-05-19 13:25:22 +00001092 le.request = e->request;
1093 le.captures = e->captures;
1094 le.ncaptures = e->ncaptures;
Igor Sysoevaf3b7ea2006-05-31 14:11:45 +00001095 le.quote = e->quote;
Igor Sysoeve31e90b2005-05-19 13:25:22 +00001096
1097 for (len = 0; *(uintptr_t *) le.ip; len += lcode(&le)) {
1098 lcode = *(ngx_http_script_len_code_pt *) le.ip;
1099 }
1100
1101 e->buf.len = len;
1102 e->buf.data = ngx_palloc(e->request->pool, len);
1103 if (e->buf.data == NULL) {
1104 e->ip = ngx_http_script_exit;
1105 e->status = NGX_HTTP_INTERNAL_SERVER_ERROR;
1106 return;
1107 }
1108
1109 e->pos = e->buf.data;
1110
Igor Sysoev09c684b2005-11-09 17:25:55 +00001111 e->sp->len = e->buf.len;
Igor Sysoev0e5dc5c2005-11-15 13:30:52 +00001112 e->sp->data = e->buf.data;
Igor Sysoeve31e90b2005-05-19 13:25:22 +00001113 e->sp++;
1114}
1115
1116
1117void
Igor Sysoev899b44e2005-05-12 14:58:06 +00001118ngx_http_script_value_code(ngx_http_script_engine_t *e)
1119{
1120 ngx_http_script_value_code_t *code;
1121
1122 code = (ngx_http_script_value_code_t *) e->ip;
1123
1124 e->ip += sizeof(ngx_http_script_value_code_t);
1125
Igor Sysoev09c684b2005-11-09 17:25:55 +00001126 e->sp->len = code->text_len;
1127 e->sp->data = (u_char *) code->text_data;
Igor Sysoev0e5dc5c2005-11-15 13:30:52 +00001128
1129 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, e->request->connection->log, 0,
Igor Sysoev5ba67392007-10-09 18:44:59 +00001130 "http script value: \"%v\"", e->sp);
Igor Sysoev0e5dc5c2005-11-15 13:30:52 +00001131
Igor Sysoev899b44e2005-05-12 14:58:06 +00001132 e->sp++;
1133}
1134
1135
1136void
1137ngx_http_script_set_var_code(ngx_http_script_engine_t *e)
1138{
1139 ngx_http_request_t *r;
Igor Sysoev899b44e2005-05-12 14:58:06 +00001140 ngx_http_script_var_code_t *code;
1141
1142 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, e->request->connection->log, 0,
1143 "http script set var");
1144
1145 code = (ngx_http_script_var_code_t *) e->ip;
1146
1147 e->ip += sizeof(ngx_http_script_var_code_t);
1148
1149 r = e->request;
1150
Igor Sysoev899b44e2005-05-12 14:58:06 +00001151 e->sp--;
1152
Igor Sysoev09c684b2005-11-09 17:25:55 +00001153 r->variables[code->index].len = e->sp->len;
1154 r->variables[code->index].valid = 1;
Igor Sysoev2d3f3f62007-10-14 18:56:15 +00001155 r->variables[code->index].no_cacheable = 0;
Igor Sysoev09c684b2005-11-09 17:25:55 +00001156 r->variables[code->index].not_found = 0;
1157 r->variables[code->index].data = e->sp->data;
Igor Sysoev899b44e2005-05-12 14:58:06 +00001158}
1159
1160
1161void
Igor Sysoev7bdb7202006-04-19 15:30:56 +00001162ngx_http_script_var_set_handler_code(ngx_http_script_engine_t *e)
1163{
1164 ngx_http_script_var_handler_code_t *code;
1165
1166 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, e->request->connection->log, 0,
1167 "http script set var handler");
1168
1169 code = (ngx_http_script_var_handler_code_t *) e->ip;
1170
1171 e->ip += sizeof(ngx_http_script_var_handler_code_t);
1172
1173 e->sp--;
1174
1175 code->handler(e->request, e->sp, code->data);
1176}
1177
1178
1179void
Igor Sysoev899b44e2005-05-12 14:58:06 +00001180ngx_http_script_var_code(ngx_http_script_engine_t *e)
1181{
1182 ngx_http_variable_value_t *value;
1183 ngx_http_script_var_code_t *code;
1184
1185 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, e->request->connection->log, 0,
1186 "http script var");
1187
1188 code = (ngx_http_script_var_code_t *) e->ip;
1189
1190 e->ip += sizeof(ngx_http_script_var_code_t);
1191
Igor Sysoev09c684b2005-11-09 17:25:55 +00001192 value = ngx_http_get_flushed_variable(e->request, code->index);
Igor Sysoev899b44e2005-05-12 14:58:06 +00001193
Igor Sysoev09c684b2005-11-09 17:25:55 +00001194 if (value && !value->not_found) {
1195 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, e->request->connection->log, 0,
Igor Sysoev0d4b3722007-08-20 09:57:19 +00001196 "http script var: \"%v\"", value);
Igor Sysoev09c684b2005-11-09 17:25:55 +00001197
Igor Sysoevf6e1fe32005-10-04 10:38:53 +00001198 *e->sp = *value;
Igor Sysoev899b44e2005-05-12 14:58:06 +00001199 e->sp++;
1200
1201 return;
1202 }
1203
Igor Sysoev0e5dc5c2005-11-15 13:30:52 +00001204 *e->sp = ngx_http_variable_null_value;
Igor Sysoev899b44e2005-05-12 14:58:06 +00001205 e->sp++;
1206}
1207
1208
1209void
1210ngx_http_script_nop_code(ngx_http_script_engine_t *e)
1211{
1212 e->ip += sizeof(uintptr_t);
Igor Sysoev7578ec92003-06-02 15:24:30 +00001213}