1
2 /*
3 * Copyright (C) Igor Sysoev
4 * Copyright (C) NGINX, Inc.
5 */
6
7 #include <nxt_router.h>
8 #include <nxt_http.h>
9
10
11 static nxt_int_t nxt_http_validate_host(nxt_str_t *host, nxt_mp_t *mp);
12 static void nxt_http_request_start(nxt_task_t *task, void *obj, void *data);
13 static nxt_int_t nxt_http_request_forward(nxt_task_t *task,
14 nxt_http_request_t *r, nxt_http_forward_t *forward);
15 static void nxt_http_request_forward_client_ip(nxt_http_request_t *r,
16 nxt_http_forward_t *forward, nxt_array_t *fields);
17 static nxt_sockaddr_t *nxt_http_request_client_ip_sockaddr(
18 nxt_http_request_t *r, u_char *start, size_t len);
19 static void nxt_http_request_forward_protocol(nxt_http_request_t *r,
20 nxt_http_field_t *field);
21 static void nxt_http_request_ready(nxt_task_t *task, void *obj, void *data);
22 static void nxt_http_request_proto_info(nxt_task_t *task,
23 nxt_http_request_t *r);
24 static void nxt_http_request_mem_buf_completion(nxt_task_t *task, void *obj,
25 void *data);
26 static void nxt_http_request_done(nxt_task_t *task, void *obj, void *data);
27
28 static u_char *nxt_http_date_cache_handler(u_char *buf, nxt_realtime_t *now,
29 struct tm *tm, size_t size, const char *format);
30
31 static nxt_http_name_value_t *nxt_http_argument(nxt_array_t *array,
32 u_char *name, size_t name_length, uint32_t hash, u_char *start,
33 const u_char *end);
34 static nxt_int_t nxt_http_cookie_parse(nxt_array_t *cookies, u_char *start,
35 const u_char *end);
36 static nxt_http_name_value_t *nxt_http_cookie(nxt_array_t *array, u_char *name,
37 size_t name_length, u_char *start, const u_char *end);
38
39
40 #define NXT_HTTP_COOKIE_HASH \
41 (nxt_http_field_hash_end( \
42 nxt_http_field_hash_char( \
43 nxt_http_field_hash_char( \
44 nxt_http_field_hash_char( \
45 nxt_http_field_hash_char( \
46 nxt_http_field_hash_char( \
47 nxt_http_field_hash_char(NXT_HTTP_FIELD_HASH_INIT, \
48 'c'), 'o'), 'o'), 'k'), 'i'), 'e')) & 0xFFFF)
49
50
51 static const nxt_http_request_state_t nxt_http_request_init_state;
52 static const nxt_http_request_state_t nxt_http_request_body_state;
53
54
55 nxt_time_string_t nxt_http_date_cache = {
56 (nxt_atomic_uint_t) -1,
57 nxt_http_date_cache_handler,
58 NULL,
59 NXT_HTTP_DATE_LEN,
60 NXT_THREAD_TIME_GMT,
61 NXT_THREAD_TIME_SEC,
62 };
63
64
65 nxt_int_t
nxt_http_init(nxt_task_t * task)66 nxt_http_init(nxt_task_t *task)
67 {
68 nxt_int_t ret;
69
70 ret = nxt_h1p_init(task);
71
72 if (ret != NXT_OK) {
73 return ret;
74 }
75
76 return nxt_http_response_hash_init(task);
77 }
78
79
80 nxt_int_t
nxt_http_request_host(void * ctx,nxt_http_field_t * field,uintptr_t data)81 nxt_http_request_host(void *ctx, nxt_http_field_t *field, uintptr_t data)
82 {
83 nxt_int_t ret;
84 nxt_str_t host;
85 nxt_http_request_t *r;
86
87 r = ctx;
88
89 if (nxt_slow_path(r->host.start != NULL)) {
90 return NXT_HTTP_BAD_REQUEST;
91 }
92
93 host.length = field->value_length;
94 host.start = field->value;
95
96 ret = nxt_http_validate_host(&host, r->mem_pool);
97
98 if (nxt_fast_path(ret == NXT_OK)) {
99 r->host = host;
100 }
101
102 return ret;
103 }
104
105
106 static nxt_int_t
nxt_http_validate_host(nxt_str_t * host,nxt_mp_t * mp)107 nxt_http_validate_host(nxt_str_t *host, nxt_mp_t *mp)
108 {
109 u_char *h, ch;
110 size_t i, dot_pos, host_length;
111 nxt_bool_t lowcase;
112
113 enum {
114 sw_usual,
115 sw_literal,
116 sw_rest
117 } state;
118
119 dot_pos = host->length;
120 host_length = host->length;
121
122 h = host->start;
123
124 lowcase = 0;
125 state = sw_usual;
126
127 for (i = 0; i < host->length; i++) {
128 ch = h[i];
129
130 if (ch > ']') {
131 /* Short path. */
132 continue;
133 }
134
135 switch (ch) {
136
137 case '.':
138 if (dot_pos == i - 1) {
139 return NXT_HTTP_BAD_REQUEST;
140 }
141
142 dot_pos = i;
143 break;
144
145 case ':':
146 if (state == sw_usual) {
147 host_length = i;
148 state = sw_rest;
149 }
150
151 break;
152
153 case '[':
154 if (i == 0) {
155 state = sw_literal;
156 }
157
158 break;
159
160 case ']':
161 if (state == sw_literal) {
162 host_length = i + 1;
163 state = sw_rest;
164 }
165
166 break;
167
168 case '/':
169 return NXT_HTTP_BAD_REQUEST;
170
171 default:
172 if (ch >= 'A' && ch <= 'Z') {
173 lowcase = 1;
174 }
175
176 break;
177 }
178 }
179
180 if (dot_pos == host_length - 1) {
181 host_length--;
182 }
183
184 host->length = host_length;
185
186 if (lowcase) {
187 host->start = nxt_mp_nget(mp, host_length);
188 if (nxt_slow_path(host->start == NULL)) {
189 return NXT_HTTP_INTERNAL_SERVER_ERROR;
190 }
191
192 nxt_memcpy_lowcase(host->start, h, host_length);
193 }
194
195 return NXT_OK;
196 }
197
198
199 nxt_int_t
nxt_http_request_field(void * ctx,nxt_http_field_t * field,uintptr_t offset)200 nxt_http_request_field(void *ctx, nxt_http_field_t *field, uintptr_t offset)
201 {
202 nxt_http_request_t *r;
203
204 r = ctx;
205
206 nxt_value_at(nxt_http_field_t *, r, offset) = field;
207
208 return NXT_OK;
209 }
210
211
212 nxt_int_t
nxt_http_request_content_length(void * ctx,nxt_http_field_t * field,uintptr_t data)213 nxt_http_request_content_length(void *ctx, nxt_http_field_t *field,
214 uintptr_t data)
215 {
216 nxt_off_t n, max_body_size;
217 nxt_http_request_t *r;
218
219 r = ctx;
220
221 if (nxt_fast_path(r->content_length == NULL)) {
222 r->content_length = field;
223
224 n = nxt_off_t_parse(field->value, field->value_length);
225
226 if (nxt_fast_path(n >= 0)) {
227 r->content_length_n = n;
228
229 max_body_size = r->conf->socket_conf->max_body_size;
230
231 if (nxt_slow_path(n > max_body_size)) {
232 return NXT_HTTP_PAYLOAD_TOO_LARGE;
233 }
234
235 return NXT_OK;
236 }
237 }
238
239 return NXT_HTTP_BAD_REQUEST;
240 }
241
242
243 nxt_http_request_t *
nxt_http_request_create(nxt_task_t * task)244 nxt_http_request_create(nxt_task_t *task)
245 {
246 nxt_mp_t *mp;
247 nxt_buf_t *last;
248 nxt_http_request_t *r;
249
250 mp = nxt_mp_create(4096, 128, 512, 32);
251 if (nxt_slow_path(mp == NULL)) {
252 return NULL;
253 }
254
255 r = nxt_mp_zget(mp, sizeof(nxt_http_request_t));
256 if (nxt_slow_path(r == NULL)) {
257 goto fail;
258 }
259
260 r->resp.fields = nxt_list_create(mp, 8, sizeof(nxt_http_field_t));
261 if (nxt_slow_path(r->resp.fields == NULL)) {
262 goto fail;
263 }
264
265 last = nxt_mp_zget(mp, NXT_BUF_SYNC_SIZE);
266 if (nxt_slow_path(last == NULL)) {
267 goto fail;
268 }
269
270 nxt_buf_set_sync(last);
271 nxt_buf_set_last(last);
272 last->completion_handler = nxt_http_request_done;
273 last->parent = r;
274 r->last = last;
275
276 r->mem_pool = mp;
277 r->content_length_n = -1;
278 r->resp.content_length_n = -1;
279 r->state = &nxt_http_request_init_state;
280
281 r->start_time = nxt_thread_monotonic_time(task->thread);
282
283 task->thread->engine->requests_cnt++;
284
285 r->tstr_cache.var.pool = mp;
286
287 return r;
288
289 fail:
290
291 nxt_mp_release(mp);
292
293 return NULL;
294 }
295
296
297 static const nxt_http_request_state_t nxt_http_request_init_state
298 nxt_aligned(64) =
299 {
300 .ready_handler = nxt_http_request_start,
301 .error_handler = nxt_http_request_close_handler,
302 };
303
304
305 static void
nxt_http_request_start(nxt_task_t * task,void * obj,void * data)306 nxt_http_request_start(nxt_task_t *task, void *obj, void *data)
307 {
308 nxt_int_t ret;
309 nxt_socket_conf_t *skcf;
310 nxt_http_request_t *r;
311
312 r = obj;
313
314 r->state = &nxt_http_request_body_state;
315
316 skcf = r->conf->socket_conf;
317
318 if (skcf->forwarded != NULL) {
319 ret = nxt_http_request_forward(task, r, skcf->forwarded);
320 if (nxt_slow_path(ret != NXT_OK)) {
321 goto fail;
322 }
323 }
324
325 if (skcf->client_ip != NULL) {
326 ret = nxt_http_request_forward(task, r, skcf->client_ip);
327 if (nxt_slow_path(ret != NXT_OK)) {
328 goto fail;
329 }
330 }
331
332 nxt_http_request_read_body(task, r);
333
334 return;
335
336 fail:
337 nxt_http_request_error(task, r, NXT_HTTP_INTERNAL_SERVER_ERROR);
338 }
339
340
341 static nxt_int_t
nxt_http_request_forward(nxt_task_t * task,nxt_http_request_t * r,nxt_http_forward_t * forward)342 nxt_http_request_forward(nxt_task_t *task, nxt_http_request_t *r,
343 nxt_http_forward_t *forward)
344 {
345 nxt_int_t ret;
346 nxt_array_t *client_ip_fields;
347 nxt_http_field_t *f, **fields, *protocol_field;
348 nxt_http_forward_header_t *client_ip, *protocol;
349
350 ret = nxt_http_route_addr_rule(r, forward->source, r->remote);
351 if (ret <= 0) {
352 return NXT_OK;
353 }
354
355 client_ip = &forward->client_ip;
356 protocol = &forward->protocol;
357
358 if (client_ip->header != NULL) {
359 client_ip_fields = nxt_array_create(r->mem_pool, 1,
360 sizeof(nxt_http_field_t *));
361 if (nxt_slow_path(client_ip_fields == NULL)) {
362 return NXT_ERROR;
363 }
364
365 } else {
366 client_ip_fields = NULL;
367 }
368
369 protocol_field = NULL;
370
371 nxt_list_each(f, r->fields) {
372 if (client_ip_fields != NULL
373 && f->hash == client_ip->header_hash
374 && f->value_length > 0
375 && f->name_length == client_ip->header->length
376 && nxt_memcasecmp(f->name, client_ip->header->start,
377 client_ip->header->length) == 0)
378 {
379 fields = nxt_array_add(client_ip_fields);
380 if (nxt_slow_path(fields == NULL)) {
381 return NXT_ERROR;
382 }
383
384 *fields = f;
385 }
386
387 if (protocol->header != NULL
388 && protocol_field == NULL
389 && f->hash == protocol->header_hash
390 && f->value_length > 0
391 && f->name_length == protocol->header->length
392 && nxt_memcasecmp(f->name, protocol->header->start,
393 protocol->header->length) == 0)
394 {
395 protocol_field = f;
396 }
397 } nxt_list_loop;
398
399 if (client_ip_fields != NULL) {
400 nxt_http_request_forward_client_ip(r, forward, client_ip_fields);
401 }
402
403 if (protocol_field != NULL) {
404 nxt_http_request_forward_protocol(r, protocol_field);
405 }
406
407 return NXT_OK;
408 }
409
410
411 static void
nxt_http_request_forward_client_ip(nxt_http_request_t * r,nxt_http_forward_t * forward,nxt_array_t * fields)412 nxt_http_request_forward_client_ip(nxt_http_request_t *r,
413 nxt_http_forward_t *forward, nxt_array_t *fields)
414 {
415 u_char *start, *p;
416 nxt_int_t ret, i, len;
417 nxt_sockaddr_t *sa, *prev_sa;
418 nxt_http_field_t **f;
419
420 prev_sa = r->remote;
421 f = (nxt_http_field_t **) fields->elts;
422
423 i = fields->nelts;
424
425 while (i-- > 0) {
426 start = f[i]->value;
427 len = f[i]->value_length;
428
429 do {
430 for (p = start + len - 1; p > start; p--, len--) {
431 if (*p != ' ' && *p != ',') {
432 break;
433 }
434 }
435
436 for (/* void */; p > start; p--) {
437 if (*p == ' ' || *p == ',') {
438 p++;
439 break;
440 }
441 }
442
443 sa = nxt_http_request_client_ip_sockaddr(r, p, len - (p - start));
444 if (nxt_slow_path(sa == NULL)) {
445 if (prev_sa != NULL) {
446 r->remote = prev_sa;
447 }
448
449 return;
450 }
451
452 if (!forward->recursive) {
453 r->remote = sa;
454 return;
455 }
456
457 ret = nxt_http_route_addr_rule(r, forward->source, sa);
458 if (ret <= 0 || (i == 0 && p == start)) {
459 r->remote = sa;
460 return;
461 }
462
463 prev_sa = sa;
464 len = p - 1 - start;
465
466 } while (len > 0);
467 }
468 }
469
470
471 static nxt_sockaddr_t *
nxt_http_request_client_ip_sockaddr(nxt_http_request_t * r,u_char * start,size_t len)472 nxt_http_request_client_ip_sockaddr(nxt_http_request_t *r, u_char *start,
473 size_t len)
474 {
475 nxt_str_t addr;
476 nxt_sockaddr_t *sa;
477
478 addr.start = start;
479 addr.length = len;
480
481 sa = nxt_sockaddr_parse_optport(r->mem_pool, &addr);
482 if (nxt_slow_path(sa == NULL)) {
483 return NULL;
484 }
485
486 switch (sa->u.sockaddr.sa_family) {
487 case AF_INET:
488 if (sa->u.sockaddr_in.sin_addr.s_addr == INADDR_ANY) {
489 return NULL;
490 }
491
492 break;
493
494 #if (NXT_INET6)
495 case AF_INET6:
496 if (IN6_IS_ADDR_UNSPECIFIED(&sa->u.sockaddr_in6.sin6_addr)) {
497 return NULL;
498 }
499
500 break;
501 #endif /* NXT_INET6 */
502
503 default:
504 return NULL;
505 }
506
507 return sa;
508 }
509
510
511 static void
nxt_http_request_forward_protocol(nxt_http_request_t * r,nxt_http_field_t * field)512 nxt_http_request_forward_protocol(nxt_http_request_t *r,
513 nxt_http_field_t *field)
514 {
515 if (field->value_length == 4) {
516 if (nxt_memcasecmp(field->value, "http", 4) == 0) {
517 r->tls = 0;
518 }
519
520 } else if (field->value_length == 5) {
521 if (nxt_memcasecmp(field->value, "https", 5) == 0) {
522 r->tls = 1;
523 }
524
525 } else if (field->value_length == 2) {
526 if (nxt_memcasecmp(field->value, "on", 2) == 0) {
527 r->tls = 1;
528 }
529 }
530 }
531
532
533 static const nxt_http_request_state_t nxt_http_request_body_state
534 nxt_aligned(64) =
535 {
536 .ready_handler = nxt_http_request_ready,
537 .error_handler = nxt_http_request_close_handler,
538 };
539
540
541 static void
nxt_http_request_ready(nxt_task_t * task,void * obj,void * data)542 nxt_http_request_ready(nxt_task_t *task, void *obj, void *data)
543 {
544 nxt_http_action_t *action;
545 nxt_http_request_t *r;
546
547 r = obj;
548 action = r->conf->socket_conf->action;
549
550 nxt_http_request_action(task, r, action);
551 }
552
553
554 void
nxt_http_request_action(nxt_task_t * task,nxt_http_request_t * r,nxt_http_action_t * action)555 nxt_http_request_action(nxt_task_t *task, nxt_http_request_t *r,
556 nxt_http_action_t *action)
557 {
558 nxt_int_t ret;
559
560 if (nxt_fast_path(action != NULL)) {
561
562 do {
563 if (action->rewrite != NULL) {
564 ret = nxt_http_rewrite(task, r, action);
565 if (nxt_slow_path(ret != NXT_OK)) {
566 break;
567 }
568 }
569
570 action = action->handler(task, r, action);
571
572 if (action == NULL) {
573 return;
574 }
575
576 if (action == NXT_HTTP_ACTION_ERROR) {
577 break;
578 }
579
580 } while (r->pass_count++ < 255);
581 }
582
583 nxt_http_request_error(task, r, NXT_HTTP_INTERNAL_SERVER_ERROR);
584 }
585
586
587 nxt_http_action_t *
nxt_http_application_handler(nxt_task_t * task,nxt_http_request_t * r,nxt_http_action_t * action)588 nxt_http_application_handler(nxt_task_t *task, nxt_http_request_t *r,
589 nxt_http_action_t *action)
590 {
591 nxt_debug(task, "http application handler");
592
593 /*
594 * TODO: need an application flag to get local address
595 * required by "SERVER_ADDR" in Pyhton and PHP. Not used in Go.
596 */
597 nxt_http_request_proto_info(task, r);
598
599 if (r->host.length != 0) {
600 r->server_name = r->host;
601
602 } else {
603 nxt_str_set(&r->server_name, "localhost");
604 }
605
606 nxt_router_process_http_request(task, r, action);
607
608 return NULL;
609 }
610
611
612 static void
nxt_http_request_proto_info(nxt_task_t * task,nxt_http_request_t * r)613 nxt_http_request_proto_info(nxt_task_t *task, nxt_http_request_t *r)
614 {
615 if (nxt_fast_path(r->proto.any != NULL)) {
616 nxt_http_proto[r->protocol].local_addr(task, r);
617 }
618 }
619
620
621 void
nxt_http_request_read_body(nxt_task_t * task,nxt_http_request_t * r)622 nxt_http_request_read_body(nxt_task_t *task, nxt_http_request_t *r)
623 {
624 if (nxt_fast_path(r->proto.any != NULL)) {
625 nxt_http_proto[r->protocol].body_read(task, r);
626 }
627 }
628
629
630 void
nxt_http_request_header_send(nxt_task_t * task,nxt_http_request_t * r,nxt_work_handler_t body_handler,void * data)631 nxt_http_request_header_send(nxt_task_t *task, nxt_http_request_t *r,
632 nxt_work_handler_t body_handler, void *data)
633 {
634 u_char *p, *end, *server_string;
635 nxt_http_field_t *server, *date, *content_length;
636 nxt_socket_conf_t *skcf;
637
638 /*
639 * TODO: "Server", "Date", and "Content-Length" processing should be moved
640 * to the last header filter.
641 */
642
643 server = nxt_list_zero_add(r->resp.fields);
644 if (nxt_slow_path(server == NULL)) {
645 goto fail;
646 }
647
648 skcf = r->conf->socket_conf;
649 server_string = (u_char *) (skcf->server_version ? NXT_SERVER : NXT_NAME);
650
651 nxt_http_field_name_set(server, "Server");
652 server->value = server_string;
653 server->value_length = nxt_strlen(server_string);
654
655 if (r->resp.date == NULL) {
656 date = nxt_list_zero_add(r->resp.fields);
657 if (nxt_slow_path(date == NULL)) {
658 goto fail;
659 }
660
661 nxt_http_field_name_set(date, "Date");
662
663 p = nxt_mp_nget(r->mem_pool, nxt_http_date_cache.size);
664 if (nxt_slow_path(p == NULL)) {
665 goto fail;
666 }
667
668 (void) nxt_thread_time_string(task->thread, &nxt_http_date_cache, p);
669
670 date->value = p;
671 date->value_length = nxt_http_date_cache.size;
672
673 r->resp.date = date;
674 }
675
676 if (r->resp.content_length_n != -1
677 && (r->resp.content_length == NULL || r->resp.content_length->skip))
678 {
679 content_length = nxt_list_zero_add(r->resp.fields);
680 if (nxt_slow_path(content_length == NULL)) {
681 goto fail;
682 }
683
684 nxt_http_field_name_set(content_length, "Content-Length");
685
686 p = nxt_mp_nget(r->mem_pool, NXT_OFF_T_LEN);
687 if (nxt_slow_path(p == NULL)) {
688 goto fail;
689 }
690
691 content_length->value = p;
692 end = nxt_sprintf(p, p + NXT_OFF_T_LEN, "%O", r->resp.content_length_n);
693 content_length->value_length = end - p;
694
695 r->resp.content_length = content_length;
696 }
697
698 if (nxt_fast_path(r->proto.any != NULL)) {
699 nxt_http_proto[r->protocol].header_send(task, r, body_handler, data);
700 }
701
702 return;
703
704 fail:
705
706 nxt_http_request_error(task, r, NXT_HTTP_INTERNAL_SERVER_ERROR);
707 }
708
709
710 void
nxt_http_request_ws_frame_start(nxt_task_t * task,nxt_http_request_t * r,nxt_buf_t * ws_frame)711 nxt_http_request_ws_frame_start(nxt_task_t *task, nxt_http_request_t *r,
712 nxt_buf_t *ws_frame)
713 {
714 if (r->proto.any != NULL) {
715 nxt_http_proto[r->protocol].ws_frame_start(task, r, ws_frame);
716 }
717 }
718
719
720 void
nxt_http_request_send(nxt_task_t * task,nxt_http_request_t * r,nxt_buf_t * out)721 nxt_http_request_send(nxt_task_t *task, nxt_http_request_t *r, nxt_buf_t *out)
722 {
723 if (nxt_fast_path(r->proto.any != NULL)) {
724 nxt_http_proto[r->protocol].send(task, r, out);
725 }
726 }
727
728
729 nxt_buf_t *
nxt_http_buf_mem(nxt_task_t * task,nxt_http_request_t * r,size_t size)730 nxt_http_buf_mem(nxt_task_t *task, nxt_http_request_t *r, size_t size)
731 {
732 nxt_buf_t *b;
733
734 b = nxt_buf_mem_alloc(r->mem_pool, size, 0);
735 if (nxt_fast_path(b != NULL)) {
736 b->completion_handler = nxt_http_request_mem_buf_completion;
737 b->parent = r;
738 nxt_mp_retain(r->mem_pool);
739
740 } else {
741 nxt_http_request_error(task, r, NXT_HTTP_INTERNAL_SERVER_ERROR);
742 }
743
744 return b;
745 }
746
747
748 static void
nxt_http_request_mem_buf_completion(nxt_task_t * task,void * obj,void * data)749 nxt_http_request_mem_buf_completion(nxt_task_t *task, void *obj, void *data)
750 {
751 nxt_buf_t *b, *next;
752 nxt_http_request_t *r;
753
754 b = obj;
755 r = data;
756
757 do {
758 next = b->next;
759
760 nxt_mp_free(r->mem_pool, b);
761 nxt_mp_release(r->mem_pool);
762
763 b = next;
764 } while (b != NULL);
765 }
766
767
768 nxt_buf_t *
nxt_http_buf_last(nxt_http_request_t * r)769 nxt_http_buf_last(nxt_http_request_t *r)
770 {
771 nxt_buf_t *last;
772
773 last = r->last;
774 r->last = NULL;
775
776 return last;
777 }
778
779
780 static void
nxt_http_request_done(nxt_task_t * task,void * obj,void * data)781 nxt_http_request_done(nxt_task_t *task, void *obj, void *data)
782 {
783 nxt_http_request_t *r;
784
785 r = data;
786
787 nxt_debug(task, "http request done");
788
789 nxt_http_request_close_handler(task, r, r->proto.any);
790 }
791
792
793 void
nxt_http_request_error_handler(nxt_task_t * task,void * obj,void * data)794 nxt_http_request_error_handler(nxt_task_t *task, void *obj, void *data)
795 {
796 nxt_http_proto_t proto;
797 nxt_http_request_t *r;
798
799 r = obj;
800 proto.any = data;
801
802 nxt_debug(task, "http request error handler");
803
804 r->error = 1;
805
806 if (nxt_fast_path(proto.any != NULL)) {
807 nxt_http_proto[r->protocol].discard(task, r, nxt_http_buf_last(r));
808 }
809 }
810
811
812 void
nxt_http_request_close_handler(nxt_task_t * task,void * obj,void * data)813 nxt_http_request_close_handler(nxt_task_t *task, void *obj, void *data)
814 {
815 nxt_tstr_t *log_format;
816 nxt_http_proto_t proto;
817 nxt_http_request_t *r;
818 nxt_http_protocol_t protocol;
819 nxt_socket_conf_joint_t *conf;
820 nxt_router_access_log_t *access_log;
821
822 r = obj;
823 proto.any = data;
824
825 conf = r->conf;
826
827 if (!r->logged) {
828 r->logged = 1;
829
830 access_log = conf->socket_conf->router_conf->access_log;
831 log_format = conf->socket_conf->router_conf->log_format;
832
833 if (access_log != NULL) {
834 access_log->handler(task, r, access_log, log_format);
835 return;
836 }
837 }
838
839 nxt_debug(task, "http request close handler");
840
841 r->proto.any = NULL;
842
843 if (r->body != NULL && nxt_buf_is_file(r->body)
844 && r->body->file->fd != -1)
845 {
846 nxt_fd_close(r->body->file->fd);
847
848 r->body->file->fd = -1;
849 }
850
851 if (r->tstr_query != NULL) {
852 nxt_tstr_query_release(r->tstr_query);
853 }
854
855 if (nxt_fast_path(proto.any != NULL)) {
856 protocol = r->protocol;
857
858 nxt_http_proto[protocol].close(task, proto, conf);
859
860 nxt_mp_release(r->mem_pool);
861 }
862 }
863
864
865 static u_char *
nxt_http_date_cache_handler(u_char * buf,nxt_realtime_t * now,struct tm * tm,size_t size,const char * format)866 nxt_http_date_cache_handler(u_char *buf, nxt_realtime_t *now, struct tm *tm,
867 size_t size, const char *format)
868 {
869 return nxt_http_date(buf, tm);
870 }
871
872
873 nxt_array_t *
nxt_http_arguments_parse(nxt_http_request_t * r)874 nxt_http_arguments_parse(nxt_http_request_t *r)
875 {
876 size_t name_length;
877 u_char *p, *dst, *dst_start, *start, *end, *name;
878 uint8_t d0, d1;
879 uint32_t hash;
880 nxt_array_t *args;
881 nxt_http_name_value_t *nv;
882
883 if (r->arguments != NULL) {
884 return r->arguments;
885 }
886
887 args = nxt_array_create(r->mem_pool, 2, sizeof(nxt_http_name_value_t));
888 if (nxt_slow_path(args == NULL)) {
889 return NULL;
890 }
891
892 hash = NXT_HTTP_FIELD_HASH_INIT;
893 name = NULL;
894 name_length = 0;
895
896 dst_start = nxt_mp_nget(r->mem_pool, r->args->length);
897 if (nxt_slow_path(dst_start == NULL)) {
898 return NULL;
899 }
900
901 r->args_decoded.start = dst_start;
902
903 start = r->args->start;
904 end = start + r->args->length;
905
906 for (p = start, dst = dst_start; p < end; p++, dst++) {
907 *dst = *p;
908
909 switch (*p) {
910 case '=':
911 if (name == NULL) {
912 name_length = dst - dst_start;
913 name = dst_start;
914 dst_start = dst + 1;
915 }
916
917 continue;
918
919 case '&':
920 if (name_length != 0 || dst != dst_start) {
921 nv = nxt_http_argument(args, name, name_length, hash, dst_start,
922 dst);
923 if (nxt_slow_path(nv == NULL)) {
924 return NULL;
925 }
926 }
927
928 hash = NXT_HTTP_FIELD_HASH_INIT;
929 name_length = 0;
930 name = NULL;
931 dst_start = dst + 1;
932
933 continue;
934
935 case '+':
936 *dst = ' ';
937
938 break;
939
940 case '%':
941 if (nxt_slow_path(end - p <= 2)) {
942 break;
943 }
944
945 d0 = nxt_hex2int[p[1]];
946 d1 = nxt_hex2int[p[2]];
947
948 if (nxt_slow_path((d0 | d1) >= 16)) {
949 break;
950 }
951
952 p += 2;
953 *dst = (d0 << 4) + d1;
954
955 break;
956 }
957
958 if (name == NULL) {
959 hash = nxt_http_field_hash_char(hash, *dst);
960 }
961 }
962
963 r->args_decoded.length = dst - r->args_decoded.start;
964
965 if (name_length != 0 || dst != dst_start) {
966 nv = nxt_http_argument(args, name, name_length, hash, dst_start, dst);
967 if (nxt_slow_path(nv == NULL)) {
968 return NULL;
969 }
970 }
971
972 r->arguments = args;
973
974 return args;
975 }
976
977
978 static nxt_http_name_value_t *
nxt_http_argument(nxt_array_t * array,u_char * name,size_t name_length,uint32_t hash,u_char * start,const u_char * end)979 nxt_http_argument(nxt_array_t *array, u_char *name, size_t name_length,
980 uint32_t hash, u_char *start, const u_char *end)
981 {
982 size_t length;
983 nxt_http_name_value_t *nv;
984
985 nv = nxt_array_add(array);
986 if (nxt_slow_path(nv == NULL)) {
987 return NULL;
988 }
989
990 nv->hash = nxt_http_field_hash_end(hash) & 0xFFFF;
991
992 length = end - start;
993
994 if (name == NULL) {
995 name_length = length;
996 name = start;
997 length = 0;
998 }
999
1000 nv->name_length = name_length;
1001 nv->value_length = length;
1002 nv->name = name;
1003 nv->value = start;
1004
1005 return nv;
1006 }
1007
1008
1009 nxt_array_t *
nxt_http_cookies_parse(nxt_http_request_t * r)1010 nxt_http_cookies_parse(nxt_http_request_t *r)
1011 {
1012 nxt_int_t ret;
1013 nxt_array_t *cookies;
1014 nxt_http_field_t *f;
1015
1016 if (r->cookies != NULL) {
1017 return r->cookies;
1018 }
1019
1020 cookies = nxt_array_create(r->mem_pool, 2, sizeof(nxt_http_name_value_t));
1021 if (nxt_slow_path(cookies == NULL)) {
1022 return NULL;
1023 }
1024
1025 nxt_list_each(f, r->fields) {
1026
1027 if (f->hash != NXT_HTTP_COOKIE_HASH
1028 || f->name_length != 6
1029 || nxt_strncasecmp(f->name, (u_char *) "Cookie", 6) != 0)
1030 {
1031 continue;
1032 }
1033
1034 ret = nxt_http_cookie_parse(cookies, f->value,
1035 f->value + f->value_length);
1036 if (ret != NXT_OK) {
1037 return NULL;
1038 }
1039
1040 } nxt_list_loop;
1041
1042 r->cookies = cookies;
1043
1044 return cookies;
1045 }
1046
1047
1048 static nxt_int_t
nxt_http_cookie_parse(nxt_array_t * cookies,u_char * start,const u_char * end)1049 nxt_http_cookie_parse(nxt_array_t *cookies, u_char *start, const u_char *end)
1050 {
1051 size_t name_length;
1052 u_char c, *p, *name;
1053 nxt_http_name_value_t *nv;
1054
1055 name = NULL;
1056 name_length = 0;
1057
1058 for (p = start; p < end; p++) {
1059 c = *p;
1060
1061 if (c == '=' && name == NULL) {
1062 while (start[0] == ' ') { start++; }
1063
1064 name_length = p - start;
1065 name = start;
1066
1067 start = p + 1;
1068
1069 } else if (c == ';') {
1070 if (name != NULL) {
1071 nv = nxt_http_cookie(cookies, name, name_length, start, p);
1072 if (nxt_slow_path(nv == NULL)) {
1073 return NXT_ERROR;
1074 }
1075 }
1076
1077 name = NULL;
1078 start = p + 1;
1079 }
1080 }
1081
1082 if (name != NULL) {
1083 nv = nxt_http_cookie(cookies, name, name_length, start, p);
1084 if (nxt_slow_path(nv == NULL)) {
1085 return NXT_ERROR;
1086 }
1087 }
1088
1089 return NXT_OK;
1090 }
1091
1092
1093 static nxt_http_name_value_t *
nxt_http_cookie(nxt_array_t * array,u_char * name,size_t name_length,u_char * start,const u_char * end)1094 nxt_http_cookie(nxt_array_t *array, u_char *name, size_t name_length,
1095 u_char *start, const u_char *end)
1096 {
1097 u_char c, *p;
1098 uint32_t hash;
1099 nxt_http_name_value_t *nv;
1100
1101 nv = nxt_array_add(array);
1102 if (nxt_slow_path(nv == NULL)) {
1103 return NULL;
1104 }
1105
1106 nv->name_length = name_length;
1107 nv->name = name;
1108
1109 hash = NXT_HTTP_FIELD_HASH_INIT;
1110
1111 for (p = name; p < name + name_length; p++) {
1112 c = *p;
1113 hash = nxt_http_field_hash_char(hash, c);
1114 }
1115
1116 nv->hash = nxt_http_field_hash_end(hash) & 0xFFFF;
1117
1118 while (start < end && end[-1] == ' ') { end--; }
1119
1120 nv->value_length = end - start;
1121 nv->value = start;
1122
1123 return nv;
1124 }
1125
1126
1127 int64_t
nxt_http_field_hash(nxt_mp_t * mp,nxt_str_t * name,nxt_bool_t case_sensitive,uint8_t encoding)1128 nxt_http_field_hash(nxt_mp_t *mp, nxt_str_t *name, nxt_bool_t case_sensitive,
1129 uint8_t encoding)
1130 {
1131 u_char c, *p, *src, *start, *end, plus;
1132 uint8_t d0, d1;
1133 uint32_t hash;
1134 nxt_str_t str;
1135 nxt_uint_t i;
1136
1137 str.length = name->length;
1138
1139 str.start = nxt_mp_nget(mp, str.length);
1140 if (nxt_slow_path(str.start == NULL)) {
1141 return -1;
1142 }
1143
1144 p = str.start;
1145
1146 hash = NXT_HTTP_FIELD_HASH_INIT;
1147
1148 if (encoding == NXT_HTTP_URI_ENCODING_NONE) {
1149 for (i = 0; i < name->length; i++) {
1150 c = name->start[i];
1151 *p++ = c;
1152
1153 c = case_sensitive ? c : nxt_lowcase(c);
1154 hash = nxt_http_field_hash_char(hash, c);
1155 }
1156
1157 goto end;
1158 }
1159
1160 plus = (encoding == NXT_HTTP_URI_ENCODING_PLUS) ? ' ' : '+';
1161
1162 start = name->start;
1163 end = start + name->length;
1164
1165 for (src = start; src < end; src++) {
1166 c = *src;
1167
1168 switch (c) {
1169 case '%':
1170 if (nxt_slow_path(end - src <= 2)) {
1171 return -1;
1172 }
1173
1174 d0 = nxt_hex2int[src[1]];
1175 d1 = nxt_hex2int[src[2]];
1176 src += 2;
1177
1178 if (nxt_slow_path((d0 | d1) >= 16)) {
1179 return -1;
1180 }
1181
1182 c = (d0 << 4) + d1;
1183 *p++ = c;
1184 break;
1185
1186 case '+':
1187 c = plus;
1188 *p++ = c;
1189 break;
1190
1191 default:
1192 *p++ = c;
1193 break;
1194 }
1195
1196 c = case_sensitive ? c : nxt_lowcase(c);
1197 hash = nxt_http_field_hash_char(hash, c);
1198 }
1199
1200 str.length = p - str.start;
1201
1202 end:
1203
1204 *name = str;
1205
1206 return nxt_http_field_hash_end(hash) & 0xFFFF;
1207 }
1208
1209
1210 int64_t
nxt_http_argument_hash(nxt_mp_t * mp,nxt_str_t * name)1211 nxt_http_argument_hash(nxt_mp_t *mp, nxt_str_t *name)
1212 {
1213 return nxt_http_field_hash(mp, name, 1, NXT_HTTP_URI_ENCODING_PLUS);
1214 }
1215
1216
1217 int64_t
nxt_http_header_hash(nxt_mp_t * mp,nxt_str_t * name)1218 nxt_http_header_hash(nxt_mp_t *mp, nxt_str_t *name)
1219 {
1220 u_char c, *p;
1221 uint32_t i, hash;
1222 nxt_str_t str;
1223
1224 str.length = name->length;
1225
1226 str.start = nxt_mp_nget(mp, str.length);
1227 if (nxt_slow_path(str.start == NULL)) {
1228 return -1;
1229 }
1230
1231 p = str.start;
1232 hash = NXT_HTTP_FIELD_HASH_INIT;
1233
1234 for (i = 0; i < name->length; i++) {
1235 c = name->start[i];
1236
1237 if (c >= 'A' && c <= 'Z') {
1238 *p = c | 0x20;
1239
1240 } else if (c == '_') {
1241 *p = '-';
1242
1243 } else {
1244 *p = c;
1245 }
1246
1247 hash = nxt_http_field_hash_char(hash, *p);
1248 p++;
1249 }
1250
1251 *name = str;
1252
1253 return nxt_http_field_hash_end(hash) & 0xFFFF;
1254 }
1255
1256
1257 int64_t
nxt_http_cookie_hash(nxt_mp_t * mp,nxt_str_t * name)1258 nxt_http_cookie_hash(nxt_mp_t *mp, nxt_str_t *name)
1259 {
1260 return nxt_http_field_hash(mp, name, 1, NXT_HTTP_URI_ENCODING_NONE);
1261 }
1262