xref: /unit/src/nxt_http_request.c (revision 2448:243735980417)
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