blob: b638f86fcd66d00bc9bca46203a20f19cee9b014 [file] [log] [blame]
Igor Sysoev0c331d92002-08-15 17:20:26 +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 Sysoev0c331d92002-08-15 17:20:26 +00007#include <ngx_config.h>
Igor Sysoev016b8522002-08-29 16:59:54 +00008#include <ngx_core.h>
Igor Sysoev0c331d92002-08-15 17:20:26 +00009#include <ngx_http.h>
10
Igor Sysoev59cf56c2004-09-07 15:29:22 +000011
Igor Sysoevbc808a72006-10-31 12:27:32 +000012static uint32_t usual[] = {
Igor Sysoevb5c75dc2006-10-28 14:36:44 +000013 0xffffdbfe, /* 1111 1111 1111 1111 1101 1011 1111 1110 */
Igor Sysoev0359ba82006-10-28 12:04:43 +000014
Igor Sysoevb5c75dc2006-10-28 14:36:44 +000015 /* ?>=< ;:98 7654 3210 /.-, +*)( '&%$ #"! */
16 0x7fff37d6, /* 0111 1111 1111 1111 0011 0111 1101 0110 */
Igor Sysoev0359ba82006-10-28 12:04:43 +000017
Igor Sysoevb5c75dc2006-10-28 14:36:44 +000018 /* _^]\ [ZYX WVUT SRQP ONML KJIH GFED CBA@ */
Igor Sysoevaddd3c82006-10-30 20:36:54 +000019#if (NGX_WIN32)
Igor Sysoevb5c75dc2006-10-28 14:36:44 +000020 0xefffffff, /* 1110 1111 1111 1111 1111 1111 1111 1111 */
Igor Sysoevaddd3c82006-10-30 20:36:54 +000021#else
22 0xffffffff, /* 1111 1111 1111 1111 1111 1111 1111 1111 */
23#endif
Igor Sysoev0359ba82006-10-28 12:04:43 +000024
Igor Sysoevb5c75dc2006-10-28 14:36:44 +000025 /* ~}| {zyx wvut srqp onml kjih gfed cba` */
26 0xffffffff, /* 1111 1111 1111 1111 1111 1111 1111 1111 */
Igor Sysoev0359ba82006-10-28 12:04:43 +000027
Igor Sysoevb5c75dc2006-10-28 14:36:44 +000028 0xffffffff, /* 1111 1111 1111 1111 1111 1111 1111 1111 */
29 0xffffffff, /* 1111 1111 1111 1111 1111 1111 1111 1111 */
30 0xffffffff, /* 1111 1111 1111 1111 1111 1111 1111 1111 */
31 0xffffffff /* 1111 1111 1111 1111 1111 1111 1111 1111 */
32};
Igor Sysoev0359ba82006-10-28 12:04:43 +000033
34
Igor Sysoev544e9f12007-05-07 06:27:14 +000035#if (NGX_HAVE_LITTLE_ENDIAN && NGX_HAVE_NONALIGNED)
36
37#define ngx_str3_cmp(m, c0, c1, c2, c3) \
38 *(uint32_t *) m == ((c3 << 24) | (c2 << 16) | (c1 << 8) | c0)
39
40#define ngx_str3Ocmp(m, c0, c1, c2, c3) \
41 *(uint32_t *) m == ((c3 << 24) | (c2 << 16) | (c1 << 8) | c0)
42
43#define ngx_str4cmp(m, c0, c1, c2, c3) \
44 *(uint32_t *) m == ((c3 << 24) | (c2 << 16) | (c1 << 8) | c0)
45
46#define ngx_str5cmp(m, c0, c1, c2, c3, c4) \
47 *(uint32_t *) m == ((c3 << 24) | (c2 << 16) | (c1 << 8) | c0) \
48 && m[4] == c4
49
50#define ngx_str6cmp(m, c0, c1, c2, c3, c4, c5) \
51 *(uint32_t *) m == ((c3 << 24) | (c2 << 16) | (c1 << 8) | c0) \
52 && (((uint32_t *) m)[1] & 0xffff) == ((c5 << 8) | c4)
53
54#define ngx_str7_cmp(m, c0, c1, c2, c3, c4, c5, c6, c7) \
55 *(uint32_t *) m == ((c3 << 24) | (c2 << 16) | (c1 << 8) | c0) \
56 && ((uint32_t *) m)[1] == ((c7 << 24) | (c6 << 16) | (c5 << 8) | c4)
57
58#define ngx_str8cmp(m, c0, c1, c2, c3, c4, c5, c6, c7) \
59 *(uint32_t *) m == ((c3 << 24) | (c2 << 16) | (c1 << 8) | c0) \
60 && ((uint32_t *) m)[1] == ((c7 << 24) | (c6 << 16) | (c5 << 8) | c4)
61
62#define ngx_str9cmp(m, c0, c1, c2, c3, c4, c5, c6, c7, c8) \
63 *(uint32_t *) m == ((c3 << 24) | (c2 << 16) | (c1 << 8) | c0) \
64 && ((uint32_t *) m)[1] == ((c7 << 24) | (c6 << 16) | (c5 << 8) | c4) \
65 && m[8] == c8
66
67#else /* !(NGX_HAVE_LITTLE_ENDIAN && NGX_HAVE_NONALIGNED) */
68
69#define ngx_str3_cmp(m, c0, c1, c2, c3) \
70 m[0] == c0 && m[1] == c1 && m[2] == c2
71
72#define ngx_str3Ocmp(m, c0, c1, c2, c3) \
73 m[0] == c0 && m[2] == c2 && m[3] == c3
74
75#define ngx_str4cmp(m, c0, c1, c2, c3) \
76 m[0] == c0 && m[1] == c1 && m[2] == c2 && m[3] == c3
77
78#define ngx_str5cmp(m, c0, c1, c2, c3, c4) \
79 m[0] == c0 && m[1] == c1 && m[2] == c2 && m[3] == c3 && m[4] == c4
80
81#define ngx_str6cmp(m, c0, c1, c2, c3, c4, c5) \
82 m[0] == c0 && m[1] == c1 && m[2] == c2 && m[3] == c3 \
83 && m[4] == c4 && m[5] == c5
84
85#define ngx_str7_cmp(m, c0, c1, c2, c3, c4, c5, c6, c7) \
86 m[0] == c0 && m[1] == c1 && m[2] == c2 && m[3] == c3 \
87 && m[4] == c4 && m[5] == c5 && m[6] == c6
88
89#define ngx_str8cmp(m, c0, c1, c2, c3, c4, c5, c6, c7) \
90 m[0] == c0 && m[1] == c1 && m[2] == c2 && m[3] == c3 \
91 && m[4] == c4 && m[5] == c5 && m[6] == c6 && m[7] == c7
92
93#define ngx_str9cmp(m, c0, c1, c2, c3, c4, c5, c6, c7, c8) \
94 m[0] == c0 && m[1] == c1 && m[2] == c2 && m[3] == c3 \
95 && m[4] == c4 && m[5] == c5 && m[6] == c6 && m[7] == c7 && m[8] == c8
96
97#endif
98
99
Igor Sysoev3338cfd2006-05-11 14:43:47 +0000100/* gcc, icc, msvc and others compile these switches as an jump table */
101
Igor Sysoev899b44e2005-05-12 14:58:06 +0000102ngx_int_t
103ngx_http_parse_request_line(ngx_http_request_t *r, ngx_buf_t *b)
Igor Sysoev0c331d92002-08-15 17:20:26 +0000104{
Igor Sysoev02f742b2005-04-08 15:18:55 +0000105 u_char c, ch, *p, *m;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000106 enum {
Igor Sysoev016b8522002-08-29 16:59:54 +0000107 sw_start = 0,
Igor Sysoeva3677242004-04-14 05:57:36 +0000108 sw_method,
Igor Sysoev016b8522002-08-29 16:59:54 +0000109 sw_spaces_before_uri,
Igor Sysoeva3677242004-04-14 05:57:36 +0000110 sw_schema,
111 sw_schema_slash,
112 sw_schema_slash_slash,
113 sw_host,
114 sw_port,
Igor Sysoev016b8522002-08-29 16:59:54 +0000115 sw_after_slash_in_uri,
116 sw_check_uri,
117 sw_uri,
118 sw_http_09,
Igor Sysoeva9830112003-05-19 16:39:14 +0000119 sw_http_H,
120 sw_http_HT,
121 sw_http_HTT,
122 sw_http_HTTP,
Igor Sysoev016b8522002-08-29 16:59:54 +0000123 sw_first_major_digit,
124 sw_major_digit,
125 sw_first_minor_digit,
126 sw_minor_digit,
Igor Sysoev81924e82007-10-26 11:34:10 +0000127 sw_spaces_after_digit,
Igor Sysoev02025fd2005-01-18 13:03:58 +0000128 sw_almost_done
Igor Sysoev481b6432002-12-04 16:29:40 +0000129 } state;
130
131 state = r->state;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000132
Igor Sysoev02025fd2005-01-18 13:03:58 +0000133 for (p = b->pos; p < b->last; p++) {
134 ch = *p;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000135
Igor Sysoev0c331d92002-08-15 17:20:26 +0000136 switch (state) {
137
138 /* HTTP methods: GET, HEAD, POST */
Igor Sysoev016b8522002-08-29 16:59:54 +0000139 case sw_start:
Igor Sysoev02025fd2005-01-18 13:03:58 +0000140 r->request_start = p;
Igor Sysoev3d062ad2003-03-05 06:37:42 +0000141
Igor Sysoev732a2712004-04-21 18:54:33 +0000142 if (ch == CR || ch == LF) {
143 break;
144 }
145
Igor Sysoevdf4b2302009-07-13 09:33:34 +0000146 if ((ch < 'A' || ch > 'Z') && ch != '_') {
Igor Sysoev1af7c822002-09-13 14:47:42 +0000147 return NGX_HTTP_PARSE_INVALID_METHOD;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000148 }
Igor Sysoeva3677242004-04-14 05:57:36 +0000149
150 state = sw_method;
Igor Sysoeva9830112003-05-19 16:39:14 +0000151 break;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000152
Igor Sysoeva3677242004-04-14 05:57:36 +0000153 case sw_method:
154 if (ch == ' ') {
Igor Sysoev899b44e2005-05-12 14:58:06 +0000155 r->method_end = p - 1;
Igor Sysoeva3677242004-04-14 05:57:36 +0000156 m = r->request_start;
Igor Sysoeva9830112003-05-19 16:39:14 +0000157
Igor Sysoev4ecb4d72006-04-21 12:06:44 +0000158 switch (p - m) {
Igor Sysoeva9830112003-05-19 16:39:14 +0000159
Igor Sysoev4ecb4d72006-04-21 12:06:44 +0000160 case 3:
Igor Sysoev544e9f12007-05-07 06:27:14 +0000161 if (ngx_str3_cmp(m, 'G', 'E', 'T', ' ')) {
Igor Sysoeva3677242004-04-14 05:57:36 +0000162 r->method = NGX_HTTP_GET;
Igor Sysoev8365f732006-11-14 12:43:48 +0000163 break;
Igor Sysoeva3677242004-04-14 05:57:36 +0000164 }
Igor Sysoev8365f732006-11-14 12:43:48 +0000165
Igor Sysoev544e9f12007-05-07 06:27:14 +0000166 if (ngx_str3_cmp(m, 'P', 'U', 'T', ' ')) {
Igor Sysoev8365f732006-11-14 12:43:48 +0000167 r->method = NGX_HTTP_PUT;
168 break;
169 }
170
Igor Sysoev4ecb4d72006-04-21 12:06:44 +0000171 break;
Igor Sysoeva9830112003-05-19 16:39:14 +0000172
Igor Sysoev4ecb4d72006-04-21 12:06:44 +0000173 case 4:
Igor Sysoev8365f732006-11-14 12:43:48 +0000174 if (m[1] == 'O') {
Igor Sysoeva9830112003-05-19 16:39:14 +0000175
Igor Sysoev544e9f12007-05-07 06:27:14 +0000176 if (ngx_str3Ocmp(m, 'P', 'O', 'S', 'T')) {
Igor Sysoev8365f732006-11-14 12:43:48 +0000177 r->method = NGX_HTTP_POST;
178 break;
179 }
180
Igor Sysoev544e9f12007-05-07 06:27:14 +0000181 if (ngx_str3Ocmp(m, 'C', 'O', 'P', 'Y')) {
Igor Sysoev8365f732006-11-14 12:43:48 +0000182 r->method = NGX_HTTP_COPY;
183 break;
184 }
185
Igor Sysoev544e9f12007-05-07 06:27:14 +0000186 if (ngx_str3Ocmp(m, 'M', 'O', 'V', 'E')) {
Igor Sysoev8365f732006-11-14 12:43:48 +0000187 r->method = NGX_HTTP_MOVE;
188 break;
189 }
190
Igor Sysoev544e9f12007-05-07 06:27:14 +0000191 if (ngx_str3Ocmp(m, 'L', 'O', 'C', 'K')) {
Igor Sysoev8365f732006-11-14 12:43:48 +0000192 r->method = NGX_HTTP_LOCK;
193 break;
194 }
195
196 } else {
197
Igor Sysoev544e9f12007-05-07 06:27:14 +0000198 if (ngx_str4cmp(m, 'H', 'E', 'A', 'D')) {
Igor Sysoev8365f732006-11-14 12:43:48 +0000199 r->method = NGX_HTTP_HEAD;
200 break;
201 }
Igor Sysoeva3677242004-04-14 05:57:36 +0000202 }
Igor Sysoev8365f732006-11-14 12:43:48 +0000203
Igor Sysoev4ecb4d72006-04-21 12:06:44 +0000204 break;
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000205
Igor Sysoev4ecb4d72006-04-21 12:06:44 +0000206 case 5:
Igor Sysoev544e9f12007-05-07 06:27:14 +0000207 if (ngx_str5cmp(m, 'M', 'K', 'C', 'O', 'L')) {
Igor Sysoev7bdb7202006-04-19 15:30:56 +0000208 r->method = NGX_HTTP_MKCOL;
209 }
Igor Sysoev8365f732006-11-14 12:43:48 +0000210
Igor Sysoev544e9f12007-05-07 06:27:14 +0000211 if (ngx_str5cmp(m, 'T', 'R', 'A', 'C', 'E')) {
Igor Sysoev3e933d22007-03-30 18:59:26 +0000212 r->method = NGX_HTTP_TRACE;
213 }
214
Igor Sysoev4ecb4d72006-04-21 12:06:44 +0000215 break;
Igor Sysoev7bdb7202006-04-19 15:30:56 +0000216
Igor Sysoev4ecb4d72006-04-21 12:06:44 +0000217 case 6:
Igor Sysoev544e9f12007-05-07 06:27:14 +0000218 if (ngx_str6cmp(m, 'D', 'E', 'L', 'E', 'T', 'E')) {
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000219 r->method = NGX_HTTP_DELETE;
Igor Sysoev8365f732006-11-14 12:43:48 +0000220 break;
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000221 }
Igor Sysoev8365f732006-11-14 12:43:48 +0000222
Igor Sysoev544e9f12007-05-07 06:27:14 +0000223 if (ngx_str6cmp(m, 'U', 'N', 'L', 'O', 'C', 'K')) {
Igor Sysoev8365f732006-11-14 12:43:48 +0000224 r->method = NGX_HTTP_UNLOCK;
225 break;
226 }
227
228 break;
229
230 case 7:
Igor Sysoev544e9f12007-05-07 06:27:14 +0000231 if (ngx_str7_cmp(m, 'O', 'P', 'T', 'I', 'O', 'N', 'S', ' '))
Igor Sysoev8365f732006-11-14 12:43:48 +0000232 {
233 r->method = NGX_HTTP_OPTIONS;
234 }
235
236 break;
237
238 case 8:
Igor Sysoev544e9f12007-05-07 06:27:14 +0000239 if (ngx_str8cmp(m, 'P', 'R', 'O', 'P', 'F', 'I', 'N', 'D'))
Igor Sysoev8365f732006-11-14 12:43:48 +0000240 {
241 r->method = NGX_HTTP_PROPFIND;
242 }
243
244 break;
245
246 case 9:
Igor Sysoev544e9f12007-05-07 06:27:14 +0000247 if (ngx_str9cmp(m,
248 'P', 'R', 'O', 'P', 'P', 'A', 'T', 'C', 'H'))
Igor Sysoev8365f732006-11-14 12:43:48 +0000249 {
250 r->method = NGX_HTTP_PROPPATCH;
251 }
252
Igor Sysoev4ecb4d72006-04-21 12:06:44 +0000253 break;
Igor Sysoeva3677242004-04-14 05:57:36 +0000254 }
Igor Sysoeva9830112003-05-19 16:39:14 +0000255
Igor Sysoeva3677242004-04-14 05:57:36 +0000256 state = sw_spaces_before_uri;
Igor Sysoeva9830112003-05-19 16:39:14 +0000257 break;
Igor Sysoeva9830112003-05-19 16:39:14 +0000258 }
Igor Sysoeva9830112003-05-19 16:39:14 +0000259
Igor Sysoevdf4b2302009-07-13 09:33:34 +0000260 if ((ch < 'A' || ch > 'Z') && ch != '_') {
Igor Sysoeva9830112003-05-19 16:39:14 +0000261 return NGX_HTTP_PARSE_INVALID_METHOD;
262 }
Igor Sysoeva3677242004-04-14 05:57:36 +0000263
Igor Sysoev0c331d92002-08-15 17:20:26 +0000264 break;
265
Igor Sysoev0c331d92002-08-15 17:20:26 +0000266 /* space* before URI */
Igor Sysoev016b8522002-08-29 16:59:54 +0000267 case sw_spaces_before_uri:
Igor Sysoev02f742b2005-04-08 15:18:55 +0000268
Igor Sysoeve23b4842006-10-28 10:47:11 +0000269 if (ch == '/' ){
270 r->uri_start = p;
271 state = sw_after_slash_in_uri;
272 break;
273 }
274
Igor Sysoev02f742b2005-04-08 15:18:55 +0000275 c = (u_char) (ch | 0x20);
Igor Sysoev7b190b42005-06-07 15:56:31 +0000276 if (c >= 'a' && c <= 'z') {
Igor Sysoev1ebfead2005-02-16 13:40:36 +0000277 r->schema_start = p;
278 state = sw_schema;
279 break;
280 }
281
Igor Sysoev0c331d92002-08-15 17:20:26 +0000282 switch (ch) {
Igor Sysoev0c331d92002-08-15 17:20:26 +0000283 case ' ':
284 break;
285 default:
Igor Sysoeva3677242004-04-14 05:57:36 +0000286 return NGX_HTTP_PARSE_INVALID_REQUEST;
287 }
288 break;
289
290 case sw_schema:
Igor Sysoev02f742b2005-04-08 15:18:55 +0000291
292 c = (u_char) (ch | 0x20);
Igor Sysoev7b190b42005-06-07 15:56:31 +0000293 if (c >= 'a' && c <= 'z') {
Igor Sysoev1ebfead2005-02-16 13:40:36 +0000294 break;
295 }
296
Igor Sysoeva3677242004-04-14 05:57:36 +0000297 switch (ch) {
298 case ':':
Igor Sysoev02025fd2005-01-18 13:03:58 +0000299 r->schema_end = p;
Igor Sysoeva3677242004-04-14 05:57:36 +0000300 state = sw_schema_slash;
301 break;
302 default:
Igor Sysoeva3677242004-04-14 05:57:36 +0000303 return NGX_HTTP_PARSE_INVALID_REQUEST;
304 }
305 break;
306
307 case sw_schema_slash:
308 switch (ch) {
309 case '/':
310 state = sw_schema_slash_slash;
311 break;
312 default:
313 return NGX_HTTP_PARSE_INVALID_REQUEST;
314 }
315 break;
316
317 case sw_schema_slash_slash:
318 switch (ch) {
319 case '/':
Igor Sysoev95ead462006-11-23 20:20:23 +0000320 r->host_start = p + 1;
Igor Sysoeva3677242004-04-14 05:57:36 +0000321 state = sw_host;
322 break;
323 default:
324 return NGX_HTTP_PARSE_INVALID_REQUEST;
325 }
326 break;
327
328 case sw_host:
Igor Sysoev02f742b2005-04-08 15:18:55 +0000329
330 c = (u_char) (ch | 0x20);
Igor Sysoev7b190b42005-06-07 15:56:31 +0000331 if (c >= 'a' && c <= 'z') {
Igor Sysoev02f742b2005-04-08 15:18:55 +0000332 break;
333 }
334
Igor Sysoeve9b78092006-10-31 12:30:24 +0000335 if ((ch >= '0' && ch <= '9') || ch == '.' || ch == '-') {
Igor Sysoev1ebfead2005-02-16 13:40:36 +0000336 break;
337 }
338
Igor Sysoev4c5207f2007-10-18 11:33:31 +0000339 r->host_end = p;
340
Igor Sysoeva3677242004-04-14 05:57:36 +0000341 switch (ch) {
342 case ':':
Igor Sysoeva3677242004-04-14 05:57:36 +0000343 state = sw_port;
344 break;
345 case '/':
Igor Sysoev02025fd2005-01-18 13:03:58 +0000346 r->uri_start = p;
Igor Sysoeva3677242004-04-14 05:57:36 +0000347 state = sw_after_slash_in_uri;
348 break;
Igor Sysoev4c5207f2007-10-18 11:33:31 +0000349 case ' ':
350 /*
351 * use single "/" from request line to preserve pointers,
352 * if request line will be copied to large client buffer
353 */
354 r->uri_start = r->schema_end + 1;
355 r->uri_end = r->schema_end + 2;
356 state = sw_http_09;
357 break;
Igor Sysoeva3677242004-04-14 05:57:36 +0000358 default:
Igor Sysoeva3677242004-04-14 05:57:36 +0000359 return NGX_HTTP_PARSE_INVALID_REQUEST;
360 }
361 break;
362
363 case sw_port:
Igor Sysoev1ebfead2005-02-16 13:40:36 +0000364 if (ch >= '0' && ch <= '9') {
365 break;
366 }
367
Igor Sysoeva3677242004-04-14 05:57:36 +0000368 switch (ch) {
369 case '/':
Igor Sysoev02025fd2005-01-18 13:03:58 +0000370 r->port_end = p;
371 r->uri_start = p;
Igor Sysoeva3677242004-04-14 05:57:36 +0000372 state = sw_after_slash_in_uri;
373 break;
Igor Sysoev4c5207f2007-10-18 11:33:31 +0000374 case ' ':
375 r->port_end = p;
376 /*
377 * use single "/" from request line to preserve pointers,
378 * if request line will be copied to large client buffer
379 */
380 r->uri_start = r->schema_end + 1;
381 r->uri_end = r->schema_end + 2;
382 state = sw_http_09;
383 break;
Igor Sysoeva3677242004-04-14 05:57:36 +0000384 default:
Igor Sysoev1ebfead2005-02-16 13:40:36 +0000385 return NGX_HTTP_PARSE_INVALID_REQUEST;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000386 }
387 break;
388
Igor Sysoev1b735832004-11-11 14:07:14 +0000389 /* check "/.", "//", "%", and "\" (Win32) in URI */
Igor Sysoev016b8522002-08-29 16:59:54 +0000390 case sw_after_slash_in_uri:
Igor Sysoev1ebfead2005-02-16 13:40:36 +0000391
Igor Sysoev0359ba82006-10-28 12:04:43 +0000392 if (usual[ch >> 5] & (1 << (ch & 0x1f))) {
Igor Sysoev1ebfead2005-02-16 13:40:36 +0000393 state = sw_check_uri;
394 break;
395 }
396
Igor Sysoev0c331d92002-08-15 17:20:26 +0000397 switch (ch) {
Igor Sysoev1ebfead2005-02-16 13:40:36 +0000398 case ' ':
399 r->uri_end = p;
400 state = sw_http_09;
401 break;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000402 case CR:
Igor Sysoev02025fd2005-01-18 13:03:58 +0000403 r->uri_end = p;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000404 r->http_minor = 9;
Igor Sysoev016b8522002-08-29 16:59:54 +0000405 state = sw_almost_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000406 break;
407 case LF:
Igor Sysoev02025fd2005-01-18 13:03:58 +0000408 r->uri_end = p;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000409 r->http_minor = 9;
Igor Sysoev02025fd2005-01-18 13:03:58 +0000410 goto done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000411 case '.':
412 r->complex_uri = 1;
Igor Sysoev016b8522002-08-29 16:59:54 +0000413 state = sw_uri;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000414 break;
Igor Sysoev924bd792004-10-11 15:07:03 +0000415 case '%':
416 r->quoted_uri = 1;
417 state = sw_uri;
418 break;
Igor Sysoev1ebfead2005-02-16 13:40:36 +0000419 case '/':
420 r->complex_uri = 1;
421 state = sw_uri;
422 break;
Igor Sysoev1b735832004-11-11 14:07:14 +0000423#if (NGX_WIN32)
424 case '\\':
425 r->complex_uri = 1;
Igor Sysoev1ebfead2005-02-16 13:40:36 +0000426 state = sw_uri;
Igor Sysoev1b735832004-11-11 14:07:14 +0000427 break;
428#endif
Igor Sysoev1ebfead2005-02-16 13:40:36 +0000429 case '?':
430 r->args_start = p + 1;
431 state = sw_uri;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000432 break;
Igor Sysoevb80a7f42006-10-28 10:15:31 +0000433 case '#':
434 r->complex_uri = 1;
435 state = sw_uri;
436 break;
Igor Sysoev805d9db2005-02-03 19:33:37 +0000437 case '+':
438 r->plus_in_uri = 1;
439 break;
Igor Sysoev1ebfead2005-02-16 13:40:36 +0000440 case '\0':
441 r->zero_in_uri = 1;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000442 break;
443 default:
Igor Sysoev016b8522002-08-29 16:59:54 +0000444 state = sw_check_uri;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000445 break;
446 }
447 break;
448
Igor Sysoev1b735832004-11-11 14:07:14 +0000449 /* check "/", "%" and "\" (Win32) in URI */
Igor Sysoev016b8522002-08-29 16:59:54 +0000450 case sw_check_uri:
Igor Sysoev1ebfead2005-02-16 13:40:36 +0000451
Igor Sysoev0359ba82006-10-28 12:04:43 +0000452 if (usual[ch >> 5] & (1 << (ch & 0x1f))) {
Igor Sysoev1ebfead2005-02-16 13:40:36 +0000453 break;
454 }
455
Igor Sysoev0c331d92002-08-15 17:20:26 +0000456 switch (ch) {
Igor Sysoev1ebfead2005-02-16 13:40:36 +0000457 case '/':
458 r->uri_ext = NULL;
459 state = sw_after_slash_in_uri;
460 break;
461 case '.':
462 r->uri_ext = p + 1;
463 break;
464 case ' ':
465 r->uri_end = p;
466 state = sw_http_09;
467 break;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000468 case CR:
Igor Sysoev02025fd2005-01-18 13:03:58 +0000469 r->uri_end = p;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000470 r->http_minor = 9;
Igor Sysoev016b8522002-08-29 16:59:54 +0000471 state = sw_almost_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000472 break;
473 case LF:
Igor Sysoev02025fd2005-01-18 13:03:58 +0000474 r->uri_end = p;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000475 r->http_minor = 9;
Igor Sysoev02025fd2005-01-18 13:03:58 +0000476 goto done;
Igor Sysoev1b735832004-11-11 14:07:14 +0000477#if (NGX_WIN32)
478 case '\\':
479 r->complex_uri = 1;
480 state = sw_after_slash_in_uri;
481 break;
482#endif
Igor Sysoev7578ec92003-06-02 15:24:30 +0000483 case '%':
Igor Sysoev924bd792004-10-11 15:07:03 +0000484 r->quoted_uri = 1;
Igor Sysoev7578ec92003-06-02 15:24:30 +0000485 state = sw_uri;
486 break;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000487 case '?':
Igor Sysoev02025fd2005-01-18 13:03:58 +0000488 r->args_start = p + 1;
Igor Sysoev016b8522002-08-29 16:59:54 +0000489 state = sw_uri;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000490 break;
Igor Sysoevb80a7f42006-10-28 10:15:31 +0000491 case '#':
492 r->complex_uri = 1;
493 state = sw_uri;
494 break;
Igor Sysoevef809b82006-06-28 16:00:26 +0000495 case '+':
496 r->plus_in_uri = 1;
497 break;
Igor Sysoev1ebfead2005-02-16 13:40:36 +0000498 case '\0':
499 r->zero_in_uri = 1;
500 break;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000501 }
502 break;
503
504 /* URI */
Igor Sysoev016b8522002-08-29 16:59:54 +0000505 case sw_uri:
Igor Sysoev0359ba82006-10-28 12:04:43 +0000506
507 if (usual[ch >> 5] & (1 << (ch & 0x1f))) {
508 break;
509 }
510
Igor Sysoev0c331d92002-08-15 17:20:26 +0000511 switch (ch) {
Igor Sysoev1ebfead2005-02-16 13:40:36 +0000512 case ' ':
513 r->uri_end = p;
514 state = sw_http_09;
515 break;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000516 case CR:
Igor Sysoev02025fd2005-01-18 13:03:58 +0000517 r->uri_end = p;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000518 r->http_minor = 9;
Igor Sysoev016b8522002-08-29 16:59:54 +0000519 state = sw_almost_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000520 break;
521 case LF:
Igor Sysoev02025fd2005-01-18 13:03:58 +0000522 r->uri_end = p;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000523 r->http_minor = 9;
Igor Sysoev02025fd2005-01-18 13:03:58 +0000524 goto done;
Igor Sysoevb80a7f42006-10-28 10:15:31 +0000525 case '#':
526 r->complex_uri = 1;
527 break;
Igor Sysoev1ebfead2005-02-16 13:40:36 +0000528 case '\0':
529 r->zero_in_uri = 1;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000530 break;
531 }
532 break;
533
534 /* space+ after URI */
Igor Sysoev016b8522002-08-29 16:59:54 +0000535 case sw_http_09:
Igor Sysoev0c331d92002-08-15 17:20:26 +0000536 switch (ch) {
537 case ' ':
538 break;
539 case CR:
540 r->http_minor = 9;
Igor Sysoev016b8522002-08-29 16:59:54 +0000541 state = sw_almost_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000542 break;
543 case LF:
544 r->http_minor = 9;
Igor Sysoev02025fd2005-01-18 13:03:58 +0000545 goto done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000546 case 'H':
Igor Sysoev02025fd2005-01-18 13:03:58 +0000547 r->http_protocol.data = p;
Igor Sysoeva9830112003-05-19 16:39:14 +0000548 state = sw_http_H;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000549 break;
550 default:
Igor Sysoev1af7c822002-09-13 14:47:42 +0000551 return NGX_HTTP_PARSE_INVALID_REQUEST;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000552 }
553 break;
554
Igor Sysoeva9830112003-05-19 16:39:14 +0000555 case sw_http_H:
556 switch (ch) {
557 case 'T':
558 state = sw_http_HT;
559 break;
560 default:
Igor Sysoev1af7c822002-09-13 14:47:42 +0000561 return NGX_HTTP_PARSE_INVALID_REQUEST;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000562 }
Igor Sysoeva9830112003-05-19 16:39:14 +0000563 break;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000564
Igor Sysoeva9830112003-05-19 16:39:14 +0000565 case sw_http_HT:
566 switch (ch) {
567 case 'T':
568 state = sw_http_HTT;
569 break;
570 default:
571 return NGX_HTTP_PARSE_INVALID_REQUEST;
572 }
573 break;
574
575 case sw_http_HTT:
576 switch (ch) {
577 case 'P':
578 state = sw_http_HTTP;
579 break;
580 default:
581 return NGX_HTTP_PARSE_INVALID_REQUEST;
582 }
583 break;
584
585 case sw_http_HTTP:
586 switch (ch) {
587 case '/':
588 state = sw_first_major_digit;
589 break;
590 default:
591 return NGX_HTTP_PARSE_INVALID_REQUEST;
592 }
Igor Sysoev0c331d92002-08-15 17:20:26 +0000593 break;
594
595 /* first digit of major HTTP version */
Igor Sysoev016b8522002-08-29 16:59:54 +0000596 case sw_first_major_digit:
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000597 if (ch < '1' || ch > '9') {
Igor Sysoev1af7c822002-09-13 14:47:42 +0000598 return NGX_HTTP_PARSE_INVALID_REQUEST;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000599 }
Igor Sysoev0c331d92002-08-15 17:20:26 +0000600
601 r->http_major = ch - '0';
Igor Sysoev016b8522002-08-29 16:59:54 +0000602 state = sw_major_digit;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000603 break;
604
605 /* major HTTP version or dot */
Igor Sysoev016b8522002-08-29 16:59:54 +0000606 case sw_major_digit:
Igor Sysoev0c331d92002-08-15 17:20:26 +0000607 if (ch == '.') {
Igor Sysoev016b8522002-08-29 16:59:54 +0000608 state = sw_first_minor_digit;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000609 break;
610 }
611
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000612 if (ch < '0' || ch > '9') {
Igor Sysoev1af7c822002-09-13 14:47:42 +0000613 return NGX_HTTP_PARSE_INVALID_REQUEST;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000614 }
Igor Sysoev0c331d92002-08-15 17:20:26 +0000615
616 r->http_major = r->http_major * 10 + ch - '0';
617 break;
618
619 /* first digit of minor HTTP version */
Igor Sysoev016b8522002-08-29 16:59:54 +0000620 case sw_first_minor_digit:
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000621 if (ch < '0' || ch > '9') {
Igor Sysoev1af7c822002-09-13 14:47:42 +0000622 return NGX_HTTP_PARSE_INVALID_REQUEST;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000623 }
Igor Sysoev0c331d92002-08-15 17:20:26 +0000624
625 r->http_minor = ch - '0';
Igor Sysoev016b8522002-08-29 16:59:54 +0000626 state = sw_minor_digit;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000627 break;
628
629 /* minor HTTP version or end of request line */
Igor Sysoev016b8522002-08-29 16:59:54 +0000630 case sw_minor_digit:
Igor Sysoev0c331d92002-08-15 17:20:26 +0000631 if (ch == CR) {
Igor Sysoev016b8522002-08-29 16:59:54 +0000632 state = sw_almost_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000633 break;
634 }
635
636 if (ch == LF) {
Igor Sysoev02025fd2005-01-18 13:03:58 +0000637 goto done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000638 }
639
Igor Sysoev81924e82007-10-26 11:34:10 +0000640 if (ch == ' ') {
641 state = sw_spaces_after_digit;
642 break;
643 }
644
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000645 if (ch < '0' || ch > '9') {
Igor Sysoev1af7c822002-09-13 14:47:42 +0000646 return NGX_HTTP_PARSE_INVALID_REQUEST;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000647 }
Igor Sysoev0c331d92002-08-15 17:20:26 +0000648
649 r->http_minor = r->http_minor * 10 + ch - '0';
650 break;
651
Igor Sysoev81924e82007-10-26 11:34:10 +0000652 case sw_spaces_after_digit:
653 switch (ch) {
654 case ' ':
655 break;
656 case CR:
657 state = sw_almost_done;
658 break;
659 case LF:
660 goto done;
661 default:
662 return NGX_HTTP_PARSE_INVALID_REQUEST;
663 }
664 break;
665
Igor Sysoev0c331d92002-08-15 17:20:26 +0000666 /* end of request line */
Igor Sysoev016b8522002-08-29 16:59:54 +0000667 case sw_almost_done:
Igor Sysoev02025fd2005-01-18 13:03:58 +0000668 r->request_end = p - 1;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000669 switch (ch) {
670 case LF:
Igor Sysoev02025fd2005-01-18 13:03:58 +0000671 goto done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000672 default:
Igor Sysoev1af7c822002-09-13 14:47:42 +0000673 return NGX_HTTP_PARSE_INVALID_REQUEST;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000674 }
Igor Sysoev0c331d92002-08-15 17:20:26 +0000675 }
676 }
677
Igor Sysoevdd888c42004-09-21 05:38:28 +0000678 b->pos = p;
Igor Sysoev02025fd2005-01-18 13:03:58 +0000679 r->state = state;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000680
Igor Sysoev02025fd2005-01-18 13:03:58 +0000681 return NGX_AGAIN;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000682
Igor Sysoev02025fd2005-01-18 13:03:58 +0000683done:
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000684
Igor Sysoev02025fd2005-01-18 13:03:58 +0000685 b->pos = p + 1;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000686
Igor Sysoev02025fd2005-01-18 13:03:58 +0000687 if (r->request_end == NULL) {
688 r->request_end = p;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000689 }
Igor Sysoev02025fd2005-01-18 13:03:58 +0000690
691 r->http_version = r->http_major * 1000 + r->http_minor;
692 r->state = sw_start;
693
694 if (r->http_version == 9 && r->method != NGX_HTTP_GET) {
695 return NGX_HTTP_PARSE_INVALID_09_METHOD;
696 }
697
698 return NGX_OK;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000699}
700
Igor Sysoevc2bba092003-11-28 17:41:47 +0000701
Igor Sysoev899b44e2005-05-12 14:58:06 +0000702ngx_int_t
Igor Sysoev753792e2008-09-24 14:02:50 +0000703ngx_http_parse_header_line(ngx_http_request_t *r, ngx_buf_t *b,
704 ngx_uint_t allow_underscores)
Igor Sysoev0c331d92002-08-15 17:20:26 +0000705{
Igor Sysoev02f742b2005-04-08 15:18:55 +0000706 u_char c, ch, *p;
Igor Sysoev3338cfd2006-05-11 14:43:47 +0000707 ngx_uint_t hash, i;
Igor Sysoevc2bba092003-11-28 17:41:47 +0000708 enum {
Igor Sysoev016b8522002-08-29 16:59:54 +0000709 sw_start = 0,
710 sw_name,
711 sw_space_before_value,
712 sw_value,
713 sw_space_after_value,
Igor Sysoev4d656dc2005-03-22 16:02:46 +0000714 sw_ignore_line,
Igor Sysoev016b8522002-08-29 16:59:54 +0000715 sw_almost_done,
Igor Sysoev4d656dc2005-03-22 16:02:46 +0000716 sw_header_almost_done
Igor Sysoev481b6432002-12-04 16:29:40 +0000717 } state;
718
Igor Sysoevbb8bbb72006-10-17 12:47:14 +0000719 /* the last '\0' is not needed because string is zero terminated */
720
Igor Sysoev3338cfd2006-05-11 14:43:47 +0000721 static u_char lowcase[] =
722 "\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0"
723 "\0\0\0\0\0\0\0\0\0\0\0\0\0-\0\0" "0123456789\0\0\0\0\0\0"
Igor Sysoev753792e2008-09-24 14:02:50 +0000724 "\0abcdefghijklmnopqrstuvwxyz\0\0\0\0\0"
Igor Sysoev3338cfd2006-05-11 14:43:47 +0000725 "\0abcdefghijklmnopqrstuvwxyz\0\0\0\0\0"
726 "\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0"
727 "\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0"
728 "\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0"
Igor Sysoevbb8bbb72006-10-17 12:47:14 +0000729 "\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0";
Igor Sysoev3338cfd2006-05-11 14:43:47 +0000730
Igor Sysoev481b6432002-12-04 16:29:40 +0000731 state = r->state;
Igor Sysoev02f742b2005-04-08 15:18:55 +0000732 hash = r->header_hash;
Igor Sysoev3338cfd2006-05-11 14:43:47 +0000733 i = r->lowcase_index;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000734
Igor Sysoev02025fd2005-01-18 13:03:58 +0000735 for (p = b->pos; p < b->last; p++) {
736 ch = *p;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000737
Igor Sysoev0c331d92002-08-15 17:20:26 +0000738 switch (state) {
739
740 /* first char */
Igor Sysoev016b8522002-08-29 16:59:54 +0000741 case sw_start:
Igor Sysoev63494b42009-09-02 07:02:49 +0000742 r->header_name_start = p;
Igor Sysoev3362b8d2005-05-14 18:42:03 +0000743 r->invalid_header = 0;
744
Igor Sysoev0c331d92002-08-15 17:20:26 +0000745 switch (ch) {
746 case CR:
Igor Sysoev02025fd2005-01-18 13:03:58 +0000747 r->header_end = p;
Igor Sysoev016b8522002-08-29 16:59:54 +0000748 state = sw_header_almost_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000749 break;
750 case LF:
Igor Sysoev02025fd2005-01-18 13:03:58 +0000751 r->header_end = p;
752 goto header_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000753 default:
Igor Sysoev016b8522002-08-29 16:59:54 +0000754 state = sw_name;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000755
Igor Sysoev3338cfd2006-05-11 14:43:47 +0000756 c = lowcase[ch];
Igor Sysoev0c331d92002-08-15 17:20:26 +0000757
Igor Sysoev3338cfd2006-05-11 14:43:47 +0000758 if (c) {
759 hash = ngx_hash(0, c);
760 r->lowcase_header[0] = c;
761 i = 1;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000762 break;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000763 }
Igor Sysoev0c331d92002-08-15 17:20:26 +0000764
Igor Sysoev3362b8d2005-05-14 18:42:03 +0000765 r->invalid_header = 1;
766
Igor Sysoev4d656dc2005-03-22 16:02:46 +0000767 break;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000768
769 }
770 break;
771
772 /* header name */
Igor Sysoev016b8522002-08-29 16:59:54 +0000773 case sw_name:
Igor Sysoev3338cfd2006-05-11 14:43:47 +0000774 c = lowcase[ch];
775
776 if (c) {
777 hash = ngx_hash(hash, c);
778 r->lowcase_header[i++] = c;
Igor Sysoev8c8a6e52008-03-16 13:29:49 +0000779 i &= (NGX_HTTP_LC_HEADER_LEN - 1);
Igor Sysoev0c331d92002-08-15 17:20:26 +0000780 break;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000781 }
Igor Sysoev0c331d92002-08-15 17:20:26 +0000782
Igor Sysoev753792e2008-09-24 14:02:50 +0000783 if (ch == '_') {
784 if (allow_underscores) {
785 hash = ngx_hash(hash, ch);
786 r->lowcase_header[i++] = ch;
787 i &= (NGX_HTTP_LC_HEADER_LEN - 1);
788
789 } else {
790 r->invalid_header = 1;
791 }
792
793 break;
794 }
795
Igor Sysoev0c331d92002-08-15 17:20:26 +0000796 if (ch == ':') {
Igor Sysoev02025fd2005-01-18 13:03:58 +0000797 r->header_name_end = p;
Igor Sysoev016b8522002-08-29 16:59:54 +0000798 state = sw_space_before_value;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000799 break;
800 }
801
Igor Sysoev899b44e2005-05-12 14:58:06 +0000802 if (ch == CR) {
803 r->header_name_end = p;
804 r->header_start = p;
805 r->header_end = p;
806 state = sw_almost_done;
807 break;
808 }
809
810 if (ch == LF) {
811 r->header_name_end = p;
812 r->header_start = p;
813 r->header_end = p;
814 goto done;
815 }
816
Igor Sysoev924bd792004-10-11 15:07:03 +0000817 /* IIS may send the duplicate "HTTP/1.1 ..." lines */
Igor Sysoev183f9a62003-04-09 15:42:08 +0000818 if (ch == '/'
Igor Sysoev899b44e2005-05-12 14:58:06 +0000819 && r->upstream
820 && p - r->header_name_start == 4
821 && ngx_strncmp(r->header_name_start, "HTTP", 4) == 0)
Igor Sysoev183f9a62003-04-09 15:42:08 +0000822 {
823 state = sw_ignore_line;
824 break;
Igor Sysoeve2a31542003-04-08 15:40:10 +0000825 }
826
Igor Sysoev3362b8d2005-05-14 18:42:03 +0000827 r->invalid_header = 1;
828
Igor Sysoev4d656dc2005-03-22 16:02:46 +0000829 break;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000830
831 /* space* before header value */
Igor Sysoev016b8522002-08-29 16:59:54 +0000832 case sw_space_before_value:
Igor Sysoev0c331d92002-08-15 17:20:26 +0000833 switch (ch) {
834 case ' ':
835 break;
836 case CR:
Igor Sysoev899b44e2005-05-12 14:58:06 +0000837 r->header_start = p;
838 r->header_end = p;
Igor Sysoev016b8522002-08-29 16:59:54 +0000839 state = sw_almost_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000840 break;
841 case LF:
Igor Sysoev899b44e2005-05-12 14:58:06 +0000842 r->header_start = p;
843 r->header_end = p;
Igor Sysoev02025fd2005-01-18 13:03:58 +0000844 goto done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000845 default:
Igor Sysoev02025fd2005-01-18 13:03:58 +0000846 r->header_start = p;
Igor Sysoev016b8522002-08-29 16:59:54 +0000847 state = sw_value;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000848 break;
849 }
850 break;
851
852 /* header value */
Igor Sysoev016b8522002-08-29 16:59:54 +0000853 case sw_value:
Igor Sysoev0c331d92002-08-15 17:20:26 +0000854 switch (ch) {
855 case ' ':
Igor Sysoev02025fd2005-01-18 13:03:58 +0000856 r->header_end = p;
Igor Sysoev016b8522002-08-29 16:59:54 +0000857 state = sw_space_after_value;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000858 break;
859 case CR:
Igor Sysoev02025fd2005-01-18 13:03:58 +0000860 r->header_end = p;
Igor Sysoev016b8522002-08-29 16:59:54 +0000861 state = sw_almost_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000862 break;
863 case LF:
Igor Sysoev02025fd2005-01-18 13:03:58 +0000864 r->header_end = p;
865 goto done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000866 }
867 break;
868
869 /* space* before end of header line */
Igor Sysoev016b8522002-08-29 16:59:54 +0000870 case sw_space_after_value:
Igor Sysoev0c331d92002-08-15 17:20:26 +0000871 switch (ch) {
872 case ' ':
873 break;
874 case CR:
Igor Sysoev016b8522002-08-29 16:59:54 +0000875 state = sw_almost_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000876 break;
877 case LF:
Igor Sysoev02025fd2005-01-18 13:03:58 +0000878 goto done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000879 default:
Igor Sysoev016b8522002-08-29 16:59:54 +0000880 state = sw_value;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000881 break;
882 }
883 break;
884
Igor Sysoeve2a31542003-04-08 15:40:10 +0000885 /* ignore header line */
886 case sw_ignore_line:
887 switch (ch) {
888 case LF:
889 state = sw_start;
890 break;
891 default:
892 break;
893 }
894 break;
895
Igor Sysoev0c331d92002-08-15 17:20:26 +0000896 /* end of header line */
Igor Sysoev016b8522002-08-29 16:59:54 +0000897 case sw_almost_done:
Igor Sysoev0c331d92002-08-15 17:20:26 +0000898 switch (ch) {
899 case LF:
Igor Sysoev02025fd2005-01-18 13:03:58 +0000900 goto done;
Igor Sysoev70d09612008-03-16 16:47:16 +0000901 case CR:
902 break;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000903 default:
Igor Sysoev1af7c822002-09-13 14:47:42 +0000904 return NGX_HTTP_PARSE_INVALID_HEADER;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000905 }
Igor Sysoev455a7fc2006-03-21 08:20:41 +0000906 break;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000907
908 /* end of header */
Igor Sysoev016b8522002-08-29 16:59:54 +0000909 case sw_header_almost_done:
Igor Sysoev0c331d92002-08-15 17:20:26 +0000910 switch (ch) {
911 case LF:
Igor Sysoev02025fd2005-01-18 13:03:58 +0000912 goto header_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000913 default:
Igor Sysoev1af7c822002-09-13 14:47:42 +0000914 return NGX_HTTP_PARSE_INVALID_HEADER;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000915 }
Igor Sysoev0c331d92002-08-15 17:20:26 +0000916 }
917 }
918
Igor Sysoev369145c2004-05-28 15:49:23 +0000919 b->pos = p;
Igor Sysoev02025fd2005-01-18 13:03:58 +0000920 r->state = state;
Igor Sysoev02f742b2005-04-08 15:18:55 +0000921 r->header_hash = hash;
Igor Sysoev3338cfd2006-05-11 14:43:47 +0000922 r->lowcase_index = i;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000923
Igor Sysoev02025fd2005-01-18 13:03:58 +0000924 return NGX_AGAIN;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000925
Igor Sysoev02025fd2005-01-18 13:03:58 +0000926done:
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000927
Igor Sysoev02025fd2005-01-18 13:03:58 +0000928 b->pos = p + 1;
929 r->state = sw_start;
Igor Sysoev02f742b2005-04-08 15:18:55 +0000930 r->header_hash = hash;
Igor Sysoev3338cfd2006-05-11 14:43:47 +0000931 r->lowcase_index = i;
Igor Sysoev02025fd2005-01-18 13:03:58 +0000932
933 return NGX_OK;
934
935header_done:
936
937 b->pos = p + 1;
938 r->state = sw_start;
939
940 return NGX_HTTP_PARSE_HEADER_DONE;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000941}
Igor Sysoevc2bba092003-11-28 17:41:47 +0000942
943
Igor Sysoev899b44e2005-05-12 14:58:06 +0000944ngx_int_t
Igor Sysoev8decab32007-10-18 11:36:58 +0000945ngx_http_parse_complex_uri(ngx_http_request_t *r, ngx_uint_t merge_slashes)
Igor Sysoevc2bba092003-11-28 17:41:47 +0000946{
Igor Sysoev10a543a2004-03-16 07:10:12 +0000947 u_char c, ch, decoded, *p, *u;
Igor Sysoevc2bba092003-11-28 17:41:47 +0000948 enum {
949 sw_usual = 0,
950 sw_slash,
951 sw_dot,
952 sw_dot_dot,
Igor Sysoevc2bba092003-11-28 17:41:47 +0000953 sw_quoted,
954 sw_quoted_second
955 } state, quoted_state;
956
Igor Sysoev02025fd2005-01-18 13:03:58 +0000957#if (NGX_SUPPRESS_WARN)
Igor Sysoevc2bba092003-11-28 17:41:47 +0000958 decoded = '\0';
959 quoted_state = sw_usual;
Igor Sysoev02025fd2005-01-18 13:03:58 +0000960#endif
Igor Sysoevc2bba092003-11-28 17:41:47 +0000961
962 state = sw_usual;
963 p = r->uri_start;
964 u = r->uri.data;
Igor Sysoev865c1502003-11-30 20:03:18 +0000965 r->uri_ext = NULL;
Igor Sysoev1b735832004-11-11 14:07:14 +0000966 r->args_start = NULL;
Igor Sysoevc2bba092003-11-28 17:41:47 +0000967
968 ch = *p++;
969
Igor Sysoev02025fd2005-01-18 13:03:58 +0000970 while (p <= r->uri_end) {
Igor Sysoev1b735832004-11-11 14:07:14 +0000971
972 /*
Igor Sysoev02025fd2005-01-18 13:03:58 +0000973 * we use "ch = *p++" inside the cycle, but this operation is safe,
Igor Sysoev1b735832004-11-11 14:07:14 +0000974 * because after the URI there is always at least one charcter:
975 * the line feed
976 */
Igor Sysoevc2bba092003-11-28 17:41:47 +0000977
Igor Sysoev54498db2004-02-11 17:08:49 +0000978 ngx_log_debug4(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
Igor Sysoev1b735832004-11-11 14:07:14 +0000979 "s:%d in:'%Xd:%c', out:'%c'", state, ch, ch, *u);
Igor Sysoevc2bba092003-11-28 17:41:47 +0000980
981 switch (state) {
Igor Sysoev09c684b2005-11-09 17:25:55 +0000982
Igor Sysoevc2bba092003-11-28 17:41:47 +0000983 case sw_usual:
Igor Sysoev0359ba82006-10-28 12:04:43 +0000984
985 if (usual[ch >> 5] & (1 << (ch & 0x1f))) {
986 *u++ = ch;
987 ch = *p++;
988 break;
989 }
990
Igor Sysoevc2bba092003-11-28 17:41:47 +0000991 switch(ch) {
Igor Sysoev1b735832004-11-11 14:07:14 +0000992#if (NGX_WIN32)
993 case '\\':
994 r->uri_ext = NULL;
995
996 if (p == r->uri_start + r->uri.len) {
997
998 /*
999 * we omit the last "\" to cause redirect because
1000 * the browsers do not treat "\" as "/" in relative URL path
1001 */
1002
1003 break;
1004 }
1005
1006 state = sw_slash;
1007 *u++ = '/';
1008 break;
1009#endif
Igor Sysoevc2bba092003-11-28 17:41:47 +00001010 case '/':
Igor Sysoev865c1502003-11-30 20:03:18 +00001011 r->uri_ext = NULL;
Igor Sysoevc2bba092003-11-28 17:41:47 +00001012 state = sw_slash;
1013 *u++ = ch;
1014 break;
1015 case '%':
1016 quoted_state = state;
1017 state = sw_quoted;
1018 break;
Igor Sysoev1b735832004-11-11 14:07:14 +00001019 case '?':
1020 r->args_start = p;
Igor Sysoevb80a7f42006-10-28 10:15:31 +00001021 goto args;
1022 case '#':
Igor Sysoev02025fd2005-01-18 13:03:58 +00001023 goto done;
Igor Sysoev865c1502003-11-30 20:03:18 +00001024 case '.':
1025 r->uri_ext = u + 1;
Igor Sysoev42b12b32004-12-02 18:40:46 +00001026 *u++ = ch;
1027 break;
Igor Sysoevef809b82006-06-28 16:00:26 +00001028 case '+':
1029 r->plus_in_uri = 1;
Igor Sysoevc2bba092003-11-28 17:41:47 +00001030 default:
1031 *u++ = ch;
1032 break;
1033 }
Igor Sysoev0359ba82006-10-28 12:04:43 +00001034
Igor Sysoevc2bba092003-11-28 17:41:47 +00001035 ch = *p++;
1036 break;
1037
1038 case sw_slash:
Igor Sysoev0359ba82006-10-28 12:04:43 +00001039
1040 if (usual[ch >> 5] & (1 << (ch & 0x1f))) {
1041 state = sw_usual;
1042 *u++ = ch;
1043 ch = *p++;
1044 break;
1045 }
1046
Igor Sysoevc2bba092003-11-28 17:41:47 +00001047 switch(ch) {
Igor Sysoev1b735832004-11-11 14:07:14 +00001048#if (NGX_WIN32)
1049 case '\\':
Igor Sysoev8decab32007-10-18 11:36:58 +00001050 break;
Igor Sysoev1b735832004-11-11 14:07:14 +00001051#endif
Igor Sysoevc2bba092003-11-28 17:41:47 +00001052 case '/':
Igor Sysoevb862cf42007-12-14 14:33:00 +00001053 if (!merge_slashes) {
Igor Sysoev8decab32007-10-18 11:36:58 +00001054 *u++ = ch;
1055 }
Igor Sysoevc2bba092003-11-28 17:41:47 +00001056 break;
1057 case '.':
1058 state = sw_dot;
1059 *u++ = ch;
1060 break;
1061 case '%':
1062 quoted_state = state;
1063 state = sw_quoted;
1064 break;
Igor Sysoev02025fd2005-01-18 13:03:58 +00001065 case '?':
1066 r->args_start = p;
Igor Sysoevb80a7f42006-10-28 10:15:31 +00001067 goto args;
1068 case '#':
Igor Sysoev02025fd2005-01-18 13:03:58 +00001069 goto done;
Igor Sysoevef809b82006-06-28 16:00:26 +00001070 case '+':
1071 r->plus_in_uri = 1;
Igor Sysoevc2bba092003-11-28 17:41:47 +00001072 default:
1073 state = sw_usual;
1074 *u++ = ch;
1075 break;
1076 }
Igor Sysoev0359ba82006-10-28 12:04:43 +00001077
Igor Sysoevc2bba092003-11-28 17:41:47 +00001078 ch = *p++;
1079 break;
1080
1081 case sw_dot:
Igor Sysoev0359ba82006-10-28 12:04:43 +00001082
1083 if (usual[ch >> 5] & (1 << (ch & 0x1f))) {
1084 state = sw_usual;
1085 *u++ = ch;
1086 ch = *p++;
1087 break;
1088 }
1089
Igor Sysoevc2bba092003-11-28 17:41:47 +00001090 switch(ch) {
Igor Sysoev1b735832004-11-11 14:07:14 +00001091#if (NGX_WIN32)
1092 case '\\':
Igor Sysoev1b735832004-11-11 14:07:14 +00001093#endif
Igor Sysoevc2bba092003-11-28 17:41:47 +00001094 case '/':
1095 state = sw_slash;
1096 u--;
1097 break;
1098 case '.':
1099 state = sw_dot_dot;
1100 *u++ = ch;
1101 break;
1102 case '%':
1103 quoted_state = state;
1104 state = sw_quoted;
1105 break;
Igor Sysoev02025fd2005-01-18 13:03:58 +00001106 case '?':
1107 r->args_start = p;
Igor Sysoevb80a7f42006-10-28 10:15:31 +00001108 goto args;
1109 case '#':
Igor Sysoev02025fd2005-01-18 13:03:58 +00001110 goto done;
Igor Sysoevef809b82006-06-28 16:00:26 +00001111 case '+':
1112 r->plus_in_uri = 1;
Igor Sysoevc2bba092003-11-28 17:41:47 +00001113 default:
1114 state = sw_usual;
1115 *u++ = ch;
1116 break;
1117 }
Igor Sysoev0359ba82006-10-28 12:04:43 +00001118
Igor Sysoevc2bba092003-11-28 17:41:47 +00001119 ch = *p++;
1120 break;
1121
1122 case sw_dot_dot:
Igor Sysoev0359ba82006-10-28 12:04:43 +00001123
1124 if (usual[ch >> 5] & (1 << (ch & 0x1f))) {
1125 state = sw_usual;
1126 *u++ = ch;
1127 ch = *p++;
1128 break;
1129 }
1130
Igor Sysoevc2bba092003-11-28 17:41:47 +00001131 switch(ch) {
Igor Sysoev1b735832004-11-11 14:07:14 +00001132#if (NGX_WIN32)
1133 case '\\':
Igor Sysoev1b735832004-11-11 14:07:14 +00001134#endif
Igor Sysoevc2bba092003-11-28 17:41:47 +00001135 case '/':
1136 state = sw_slash;
Igor Sysoev97aa4c82009-09-14 07:42:01 +00001137 u -= 5;
1138 for ( ;; ) {
1139 if (u < r->uri.data) {
1140 return NGX_HTTP_PARSE_INVALID_REQUEST;
1141 }
1142 if (*u == '/') {
1143 u++;
1144 break;
1145 }
Igor Sysoevc2bba092003-11-28 17:41:47 +00001146 u--;
1147 }
1148 break;
1149 case '%':
1150 quoted_state = state;
1151 state = sw_quoted;
1152 break;
Igor Sysoev02025fd2005-01-18 13:03:58 +00001153 case '?':
1154 r->args_start = p;
Igor Sysoevb80a7f42006-10-28 10:15:31 +00001155 goto args;
1156 case '#':
Igor Sysoev02025fd2005-01-18 13:03:58 +00001157 goto done;
Igor Sysoevef809b82006-06-28 16:00:26 +00001158 case '+':
1159 r->plus_in_uri = 1;
Igor Sysoevc2bba092003-11-28 17:41:47 +00001160 default:
1161 state = sw_usual;
1162 *u++ = ch;
1163 break;
1164 }
Igor Sysoev0359ba82006-10-28 12:04:43 +00001165
Igor Sysoevc2bba092003-11-28 17:41:47 +00001166 ch = *p++;
1167 break;
1168
Igor Sysoevc2bba092003-11-28 17:41:47 +00001169 case sw_quoted:
Igor Sysoev8f125582006-07-28 15:16:17 +00001170 r->quoted_uri = 1;
1171
Igor Sysoevc2bba092003-11-28 17:41:47 +00001172 if (ch >= '0' && ch <= '9') {
Igor Sysoev9c610952004-03-16 13:35:20 +00001173 decoded = (u_char) (ch - '0');
Igor Sysoevc2bba092003-11-28 17:41:47 +00001174 state = sw_quoted_second;
1175 ch = *p++;
1176 break;
1177 }
1178
Igor Sysoev9c610952004-03-16 13:35:20 +00001179 c = (u_char) (ch | 0x20);
Igor Sysoevc2bba092003-11-28 17:41:47 +00001180 if (c >= 'a' && c <= 'f') {
Igor Sysoev9c610952004-03-16 13:35:20 +00001181 decoded = (u_char) (c - 'a' + 10);
Igor Sysoevc2bba092003-11-28 17:41:47 +00001182 state = sw_quoted_second;
1183 ch = *p++;
1184 break;
1185 }
1186
1187 return NGX_HTTP_PARSE_INVALID_REQUEST;
1188
1189 case sw_quoted_second:
1190 if (ch >= '0' && ch <= '9') {
Igor Sysoev9c610952004-03-16 13:35:20 +00001191 ch = (u_char) ((decoded << 4) + ch - '0');
Igor Sysoev1ebfead2005-02-16 13:40:36 +00001192
Igor Sysoev7adb8c02003-12-02 16:57:05 +00001193 if (ch == '%') {
1194 state = sw_usual;
1195 *u++ = ch;
1196 ch = *p++;
1197 break;
1198 }
Igor Sysoev1ebfead2005-02-16 13:40:36 +00001199
Igor Sysoevb80a7f42006-10-28 10:15:31 +00001200 if (ch == '#') {
1201 *u++ = ch;
1202 ch = *p++;
1203
1204 } else if (ch == '\0') {
Igor Sysoev1ebfead2005-02-16 13:40:36 +00001205 r->zero_in_uri = 1;
Igor Sysoev1ebfead2005-02-16 13:40:36 +00001206 }
1207
Igor Sysoevc2bba092003-11-28 17:41:47 +00001208 state = quoted_state;
1209 break;
1210 }
1211
Igor Sysoev9c610952004-03-16 13:35:20 +00001212 c = (u_char) (ch | 0x20);
Igor Sysoevc2bba092003-11-28 17:41:47 +00001213 if (c >= 'a' && c <= 'f') {
Igor Sysoev9c610952004-03-16 13:35:20 +00001214 ch = (u_char) ((decoded << 4) + c - 'a' + 10);
Igor Sysoevef809b82006-06-28 16:00:26 +00001215
Igor Sysoev02025fd2005-01-18 13:03:58 +00001216 if (ch == '?') {
Igor Sysoev7adb8c02003-12-02 16:57:05 +00001217 *u++ = ch;
1218 ch = *p++;
Igor Sysoevef809b82006-06-28 16:00:26 +00001219
1220 } else if (ch == '+') {
1221 r->plus_in_uri = 1;
Igor Sysoev7adb8c02003-12-02 16:57:05 +00001222 }
Igor Sysoevef809b82006-06-28 16:00:26 +00001223
Igor Sysoevc2bba092003-11-28 17:41:47 +00001224 state = quoted_state;
1225 break;
1226 }
1227
1228 return NGX_HTTP_PARSE_INVALID_REQUEST;
1229 }
1230 }
1231
Igor Sysoev02025fd2005-01-18 13:03:58 +00001232done:
1233
Igor Sysoevc2bba092003-11-28 17:41:47 +00001234 r->uri.len = u - r->uri.data;
Igor Sysoevc2bba092003-11-28 17:41:47 +00001235
Igor Sysoev865c1502003-11-30 20:03:18 +00001236 if (r->uri_ext) {
1237 r->exten.len = u - r->uri_ext;
Igor Sysoev1b735832004-11-11 14:07:14 +00001238 r->exten.data = r->uri_ext;
Igor Sysoev865c1502003-11-30 20:03:18 +00001239 }
1240
1241 r->uri_ext = NULL;
1242
Igor Sysoevc2bba092003-11-28 17:41:47 +00001243 return NGX_OK;
Igor Sysoevb80a7f42006-10-28 10:15:31 +00001244
1245args:
1246
1247 while (p < r->uri_end) {
1248 if (*p++ != '#') {
1249 continue;
1250 }
1251
1252 r->args.len = p - 1 - r->args_start;
1253 r->args.data = r->args_start;
1254 r->args_start = NULL;
1255
1256 break;
1257 }
1258
1259 r->uri.len = u - r->uri.data;
1260
1261 if (r->uri_ext) {
1262 r->exten.len = u - r->uri_ext;
1263 r->exten.data = r->uri_ext;
1264 }
1265
1266 r->uri_ext = NULL;
1267
1268 return NGX_OK;
Igor Sysoevc2bba092003-11-28 17:41:47 +00001269}
Igor Sysoev899b44e2005-05-12 14:58:06 +00001270
1271
1272ngx_int_t
Igor Sysoev09c684b2005-11-09 17:25:55 +00001273ngx_http_parse_unsafe_uri(ngx_http_request_t *r, ngx_str_t *uri,
1274 ngx_str_t *args, ngx_uint_t *flags)
1275{
1276 u_char ch, *p;
1277 size_t len;
1278
1279 len = uri->len;
1280 p = uri->data;
1281
1282 if (len == 0 || p[0] == '?') {
1283 goto unsafe;
1284 }
1285
Igor Sysoevbf14b002009-04-23 16:38:59 +00001286 if (p[0] == '.' && len == 3 && p[1] == '.' && (ngx_path_separator(p[2]))) {
Igor Sysoev09c684b2005-11-09 17:25:55 +00001287 goto unsafe;
1288 }
1289
1290 for ( /* void */ ; len; len--) {
1291
1292 ch = *p++;
1293
Igor Sysoev0359ba82006-10-28 12:04:43 +00001294 if (usual[ch >> 5] & (1 << (ch & 0x1f))) {
1295 continue;
1296 }
1297
Igor Sysoev09c684b2005-11-09 17:25:55 +00001298 if (ch == '?') {
1299 args->len = len - 1;
1300 args->data = p;
1301 uri->len -= len;
1302
1303 return NGX_OK;
1304 }
1305
1306 if (ch == '\0') {
1307 *flags |= NGX_HTTP_ZERO_IN_URI;
1308 continue;
1309 }
1310
Igor Sysoevbf14b002009-04-23 16:38:59 +00001311 if (ngx_path_separator(ch) && len > 2) {
1312
Igor Sysoev09c684b2005-11-09 17:25:55 +00001313 /* detect "/../" */
1314
Igor Sysoevbf14b002009-04-23 16:38:59 +00001315 if (p[0] == '.' && p[1] == '.' && ngx_path_separator(p[2])) {
Igor Sysoev09c684b2005-11-09 17:25:55 +00001316 goto unsafe;
1317 }
Igor Sysoev09c684b2005-11-09 17:25:55 +00001318 }
1319 }
1320
1321 return NGX_OK;
1322
1323unsafe:
1324
Igor Sysoeva552ab42009-09-25 09:30:06 +00001325 if (*flags & NGX_HTTP_LOG_UNSAFE) {
1326 ngx_log_error(NGX_LOG_ERR, r->connection->log, 0,
1327 "unsafe URI \"%V\" was detected", uri);
1328 }
Igor Sysoev09c684b2005-11-09 17:25:55 +00001329
1330 return NGX_ERROR;
1331}
1332
1333
1334ngx_int_t
Igor Sysoev899b44e2005-05-12 14:58:06 +00001335ngx_http_parse_multi_header_lines(ngx_array_t *headers, ngx_str_t *name,
1336 ngx_str_t *value)
1337{
1338 ngx_uint_t i;
1339 u_char *start, *last, *end, ch;
1340 ngx_table_elt_t **h;
1341
1342 h = headers->elts;
1343
1344 for (i = 0; i < headers->nelts; i++) {
1345
1346 ngx_log_debug2(NGX_LOG_DEBUG_HTTP, headers->pool->log, 0,
1347 "parse header: \"%V: %V\"", &h[i]->key, &h[i]->value);
1348
1349 if (name->len > h[i]->value.len) {
1350 continue;
1351 }
Igor Sysoev0e5dc5c2005-11-15 13:30:52 +00001352
Igor Sysoev899b44e2005-05-12 14:58:06 +00001353 start = h[i]->value.data;
1354 end = h[i]->value.data + h[i]->value.len;
1355
1356 while (start < end) {
1357
1358 if (ngx_strncasecmp(start, name->data, name->len) != 0) {
1359 goto skip;
1360 }
1361
1362 for (start += name->len; start < end && *start == ' '; start++) {
1363 /* void */
1364 }
1365
1366 if (value == NULL) {
1367 if (start == end || *start == ',') {
1368 return i;
1369 }
1370
1371 goto skip;
1372 }
1373
1374 if (start == end || *start++ != '=') {
1375 /* the invalid header value */
1376 goto skip;
1377 }
1378
1379 while (start < end && *start == ' ') { start++; }
1380
1381 for (last = start; last < end && *last != ';'; last++) {
1382 /* void */
1383 }
1384
1385 value->len = last - start;
1386 value->data = start;
1387
1388 return i;
1389
1390 skip:
Igor Sysoev09c684b2005-11-09 17:25:55 +00001391
Igor Sysoev899b44e2005-05-12 14:58:06 +00001392 while (start < end) {
1393 ch = *start++;
1394 if (ch == ';' || ch == ',') {
1395 break;
1396 }
1397 }
1398
1399 while (start < end && *start == ' ') { start++; }
1400 }
1401 }
1402
1403 return NGX_DECLINED;
1404}
Igor Sysoev84d17bb2008-12-22 12:02:05 +00001405
1406
1407ngx_int_t
1408ngx_http_arg(ngx_http_request_t *r, u_char *name, size_t len, ngx_str_t *value)
1409{
Igor Sysoev1dcaa972009-04-04 17:51:38 +00001410 u_char *p, *last;
Igor Sysoev84d17bb2008-12-22 12:02:05 +00001411
1412 if (r->args.len == 0) {
1413 return NGX_DECLINED;
1414 }
1415
Igor Sysoev1dcaa972009-04-04 17:51:38 +00001416 p = r->args.data;
1417 last = p + r->args.len;
Igor Sysoev84d17bb2008-12-22 12:02:05 +00001418
Igor Sysoev1dcaa972009-04-04 17:51:38 +00001419 for ( /* void */ ; p < last; p++) {
Igor Sysoev84d17bb2008-12-22 12:02:05 +00001420
Igor Sysoev1dcaa972009-04-04 17:51:38 +00001421 /* we need '=' after name, so drop one char from last */
1422
1423 p = ngx_strlcasestrn(p, last - 1, name, len - 1);
Igor Sysoev84d17bb2008-12-22 12:02:05 +00001424
1425 if (p == NULL) {
1426 return NGX_DECLINED;
1427 }
1428
1429 if ((p == r->args.data || *(p - 1) == '&') && *(p + len) == '=') {
1430
1431 value->data = p + len + 1;
1432
Igor Sysoev1dcaa972009-04-04 17:51:38 +00001433 p = ngx_strlchr(p, last, '&');
Igor Sysoev84d17bb2008-12-22 12:02:05 +00001434
1435 if (p == NULL) {
1436 p = r->args.data + r->args.len;
1437 }
1438
1439 value->len = p - value->data;
1440
1441 return NGX_OK;
1442 }
1443 }
1444
1445 return NGX_DECLINED;
1446}
Igor Sysoev2c7cb552009-03-19 13:41:29 +00001447
1448
1449void
1450ngx_http_split_args(ngx_http_request_t *r, ngx_str_t *uri, ngx_str_t *args)
1451{
1452 u_char ch, *p, *last;
1453
1454 p = uri->data;
1455
1456 last = p + uri->len;
1457
Igor Sysoevf072a022009-03-30 14:15:47 +00001458 args->len = 0;
1459
Igor Sysoev2c7cb552009-03-19 13:41:29 +00001460 while (p < last) {
1461
Igor Sysoev9bc41a42009-03-22 15:50:07 +00001462 ch = *p++;
Igor Sysoev2c7cb552009-03-19 13:41:29 +00001463
Igor Sysoev9bc41a42009-03-22 15:50:07 +00001464 if (ch == '?') {
1465 args->len = last - p;
1466 args->data = p;
Igor Sysoev2c7cb552009-03-19 13:41:29 +00001467
Igor Sysoev9bc41a42009-03-22 15:50:07 +00001468 uri->len = p - 1 - uri->data;
Igor Sysoev2c7cb552009-03-19 13:41:29 +00001469
Igor Sysoev9bc41a42009-03-22 15:50:07 +00001470 if (ngx_strlchr(p, last, '\0') != NULL) {
1471 r->zero_in_uri = 1;
1472 }
Igor Sysoev2c7cb552009-03-19 13:41:29 +00001473
Igor Sysoev9bc41a42009-03-22 15:50:07 +00001474 return;
1475 }
Igor Sysoev2c7cb552009-03-19 13:41:29 +00001476
Igor Sysoev9bc41a42009-03-22 15:50:07 +00001477 if (ch == '\0') {
1478 r->zero_in_uri = 1;
1479 continue;
1480 }
Igor Sysoev2c7cb552009-03-19 13:41:29 +00001481 }
1482}