xref: /unit/src/nxt_http_proxy.c (revision 1505)
11270Sigor@sysoev.ru 
21270Sigor@sysoev.ru /*
31270Sigor@sysoev.ru  * Copyright (C) Igor Sysoev
41270Sigor@sysoev.ru  * Copyright (C) NGINX, Inc.
51270Sigor@sysoev.ru  */
61270Sigor@sysoev.ru 
71270Sigor@sysoev.ru #include <nxt_router.h>
81270Sigor@sysoev.ru #include <nxt_http.h>
91394Sigor@sysoev.ru #include <nxt_upstream.h>
101270Sigor@sysoev.ru 
111270Sigor@sysoev.ru 
121394Sigor@sysoev.ru struct nxt_upstream_proxy_s {
131394Sigor@sysoev.ru     nxt_sockaddr_t  *sockaddr;
141394Sigor@sysoev.ru     uint8_t         protocol;
151270Sigor@sysoev.ru };
161270Sigor@sysoev.ru 
171270Sigor@sysoev.ru 
181394Sigor@sysoev.ru static void nxt_http_proxy_server_get(nxt_task_t *task,
191394Sigor@sysoev.ru     nxt_upstream_server_t *us);
201394Sigor@sysoev.ru static void nxt_http_proxy_upstream_ready(nxt_task_t *task,
211394Sigor@sysoev.ru     nxt_upstream_server_t *us);
221394Sigor@sysoev.ru static void nxt_http_proxy_upstream_error(nxt_task_t *task,
231394Sigor@sysoev.ru     nxt_upstream_server_t *us);
241270Sigor@sysoev.ru static nxt_http_action_t *nxt_http_proxy_handler(nxt_task_t *task,
251270Sigor@sysoev.ru     nxt_http_request_t *r, nxt_http_action_t *action);
261270Sigor@sysoev.ru static void nxt_http_proxy_header_send(nxt_task_t *task, void *obj, void *data);
271270Sigor@sysoev.ru static void nxt_http_proxy_header_sent(nxt_task_t *task, void *obj, void *data);
281270Sigor@sysoev.ru static void nxt_http_proxy_header_read(nxt_task_t *task, void *obj, void *data);
291270Sigor@sysoev.ru static void nxt_http_proxy_send_body(nxt_task_t *task, void *obj, void *data);
301270Sigor@sysoev.ru static void nxt_http_proxy_read(nxt_task_t *task, void *obj, void *data);
311270Sigor@sysoev.ru static void nxt_http_proxy_buf_mem_completion(nxt_task_t *task, void *obj,
321270Sigor@sysoev.ru     void *data);
331270Sigor@sysoev.ru static void nxt_http_proxy_error(nxt_task_t *task, void *obj, void *data);
341270Sigor@sysoev.ru 
351270Sigor@sysoev.ru 
361270Sigor@sysoev.ru static const nxt_http_request_state_t  nxt_http_proxy_header_send_state;
371270Sigor@sysoev.ru static const nxt_http_request_state_t  nxt_http_proxy_header_sent_state;
381270Sigor@sysoev.ru static const nxt_http_request_state_t  nxt_http_proxy_header_read_state;
391270Sigor@sysoev.ru static const nxt_http_request_state_t  nxt_http_proxy_read_state;
401270Sigor@sysoev.ru 
411270Sigor@sysoev.ru 
421394Sigor@sysoev.ru static const nxt_upstream_server_proto_t  nxt_upstream_simple_proto = {
431394Sigor@sysoev.ru     .get = nxt_http_proxy_server_get,
441394Sigor@sysoev.ru };
451394Sigor@sysoev.ru 
461394Sigor@sysoev.ru 
471394Sigor@sysoev.ru static const nxt_upstream_peer_state_t  nxt_upstream_proxy_state = {
481394Sigor@sysoev.ru     .ready = nxt_http_proxy_upstream_ready,
491394Sigor@sysoev.ru     .error = nxt_http_proxy_upstream_error,
501394Sigor@sysoev.ru };
511394Sigor@sysoev.ru 
521394Sigor@sysoev.ru 
531270Sigor@sysoev.ru nxt_int_t
541270Sigor@sysoev.ru nxt_http_proxy_create(nxt_mp_t *mp, nxt_http_action_t *action)
551270Sigor@sysoev.ru {
561394Sigor@sysoev.ru     nxt_str_t             name;
571394Sigor@sysoev.ru     nxt_sockaddr_t        *sa;
581394Sigor@sysoev.ru     nxt_upstream_t        *up;
591394Sigor@sysoev.ru     nxt_upstream_proxy_t  *proxy;
601270Sigor@sysoev.ru 
611270Sigor@sysoev.ru     sa = NULL;
621270Sigor@sysoev.ru     name = action->name;
631270Sigor@sysoev.ru 
641270Sigor@sysoev.ru     if (nxt_str_start(&name, "http://", 7)) {
651270Sigor@sysoev.ru         name.length -= 7;
661270Sigor@sysoev.ru         name.start += 7;
671270Sigor@sysoev.ru 
681270Sigor@sysoev.ru         sa = nxt_sockaddr_parse(mp, &name);
691270Sigor@sysoev.ru         if (nxt_slow_path(sa == NULL)) {
701270Sigor@sysoev.ru             return NXT_ERROR;
711270Sigor@sysoev.ru         }
721270Sigor@sysoev.ru 
731270Sigor@sysoev.ru         sa->type = SOCK_STREAM;
741270Sigor@sysoev.ru     }
751270Sigor@sysoev.ru 
761270Sigor@sysoev.ru     if (sa != NULL) {
771394Sigor@sysoev.ru         up = nxt_mp_alloc(mp, sizeof(nxt_upstream_t));
781394Sigor@sysoev.ru         if (nxt_slow_path(up == NULL)) {
791270Sigor@sysoev.ru             return NXT_ERROR;
801270Sigor@sysoev.ru         }
811270Sigor@sysoev.ru 
821394Sigor@sysoev.ru         up->name.length = sa->length;
831394Sigor@sysoev.ru         up->name.start = nxt_sockaddr_start(sa);
841394Sigor@sysoev.ru         up->proto = &nxt_upstream_simple_proto;
851270Sigor@sysoev.ru 
861394Sigor@sysoev.ru         proxy = nxt_mp_alloc(mp, sizeof(nxt_upstream_proxy_t));
871394Sigor@sysoev.ru         if (nxt_slow_path(proxy == NULL)) {
881394Sigor@sysoev.ru             return NXT_ERROR;
891394Sigor@sysoev.ru         }
901394Sigor@sysoev.ru 
911394Sigor@sysoev.ru         proxy->sockaddr = sa;
921394Sigor@sysoev.ru         proxy->protocol = NXT_HTTP_PROTO_H1;
931394Sigor@sysoev.ru         up->type.proxy = proxy;
941394Sigor@sysoev.ru 
951394Sigor@sysoev.ru         action->u.upstream = up;
961270Sigor@sysoev.ru         action->handler = nxt_http_proxy_handler;
971270Sigor@sysoev.ru     }
981270Sigor@sysoev.ru 
991270Sigor@sysoev.ru     return NXT_OK;
1001270Sigor@sysoev.ru }
1011270Sigor@sysoev.ru 
1021270Sigor@sysoev.ru 
1031270Sigor@sysoev.ru static nxt_http_action_t *
1041270Sigor@sysoev.ru nxt_http_proxy_handler(nxt_task_t *task, nxt_http_request_t *r,
1051270Sigor@sysoev.ru     nxt_http_action_t *action)
1061270Sigor@sysoev.ru {
1071394Sigor@sysoev.ru     return nxt_upstream_proxy_handler(task, r, action->u.upstream);
1081394Sigor@sysoev.ru }
1091394Sigor@sysoev.ru 
1101394Sigor@sysoev.ru 
1111394Sigor@sysoev.ru nxt_http_action_t *
1121394Sigor@sysoev.ru nxt_upstream_proxy_handler(nxt_task_t *task, nxt_http_request_t *r,
1131394Sigor@sysoev.ru     nxt_upstream_t *upstream)
1141394Sigor@sysoev.ru {
1151394Sigor@sysoev.ru     nxt_http_peer_t        *peer;
1161394Sigor@sysoev.ru     nxt_upstream_server_t  *us;
1171394Sigor@sysoev.ru 
1181394Sigor@sysoev.ru     us = nxt_mp_zalloc(r->mem_pool, sizeof(nxt_upstream_server_t));
1191394Sigor@sysoev.ru     if (nxt_slow_path(us == NULL)) {
1201394Sigor@sysoev.ru         nxt_http_request_error(task, r, NXT_HTTP_INTERNAL_SERVER_ERROR);
1211394Sigor@sysoev.ru         return NULL;
1221394Sigor@sysoev.ru     }
1231270Sigor@sysoev.ru 
1241270Sigor@sysoev.ru     peer = nxt_mp_zalloc(r->mem_pool, sizeof(nxt_http_peer_t));
1251270Sigor@sysoev.ru     if (nxt_slow_path(peer == NULL)) {
1261270Sigor@sysoev.ru         nxt_http_request_error(task, r, NXT_HTTP_INTERNAL_SERVER_ERROR);
1271270Sigor@sysoev.ru         return NULL;
1281270Sigor@sysoev.ru     }
1291270Sigor@sysoev.ru 
1301270Sigor@sysoev.ru     peer->request = r;
1311270Sigor@sysoev.ru     r->peer = peer;
1321270Sigor@sysoev.ru 
1331270Sigor@sysoev.ru     nxt_mp_retain(r->mem_pool);
1341270Sigor@sysoev.ru 
1351394Sigor@sysoev.ru     us->state = &nxt_upstream_proxy_state;
1361394Sigor@sysoev.ru     us->peer.http = peer;
1371394Sigor@sysoev.ru     peer->server = us;
1381394Sigor@sysoev.ru 
1391394Sigor@sysoev.ru     us->upstream = upstream;
1401394Sigor@sysoev.ru     upstream->proto->get(task, us);
1411270Sigor@sysoev.ru 
1421270Sigor@sysoev.ru     return NULL;
1431270Sigor@sysoev.ru }
1441270Sigor@sysoev.ru 
1451270Sigor@sysoev.ru 
1461270Sigor@sysoev.ru static void
1471394Sigor@sysoev.ru nxt_http_proxy_server_get(nxt_task_t *task, nxt_upstream_server_t *us)
1481270Sigor@sysoev.ru {
1491394Sigor@sysoev.ru     nxt_upstream_proxy_t  *proxy;
1501394Sigor@sysoev.ru 
1511394Sigor@sysoev.ru     proxy = us->upstream->type.proxy;
1521394Sigor@sysoev.ru 
1531394Sigor@sysoev.ru     us->sockaddr = proxy->sockaddr;
1541394Sigor@sysoev.ru     us->protocol = proxy->protocol;
1551394Sigor@sysoev.ru 
1561394Sigor@sysoev.ru     us->state->ready(task, us);
1571394Sigor@sysoev.ru }
1581394Sigor@sysoev.ru 
1591394Sigor@sysoev.ru 
1601394Sigor@sysoev.ru static void
1611394Sigor@sysoev.ru nxt_http_proxy_upstream_ready(nxt_task_t *task, nxt_upstream_server_t *us)
1621394Sigor@sysoev.ru {
1631394Sigor@sysoev.ru     nxt_http_peer_t  *peer;
1641394Sigor@sysoev.ru 
1651394Sigor@sysoev.ru     peer = us->peer.http;
1661394Sigor@sysoev.ru 
1671394Sigor@sysoev.ru     peer->protocol = us->protocol;
1681270Sigor@sysoev.ru 
1691270Sigor@sysoev.ru     peer->request->state = &nxt_http_proxy_header_send_state;
1701270Sigor@sysoev.ru 
1711270Sigor@sysoev.ru     nxt_http_proto[peer->protocol].peer_connect(task, peer);
1721270Sigor@sysoev.ru }
1731270Sigor@sysoev.ru 
1741270Sigor@sysoev.ru 
1751394Sigor@sysoev.ru static void
1761394Sigor@sysoev.ru nxt_http_proxy_upstream_error(nxt_task_t *task, nxt_upstream_server_t *us)
1771394Sigor@sysoev.ru {
1781394Sigor@sysoev.ru     nxt_http_request_t  *r;
1791394Sigor@sysoev.ru 
1801394Sigor@sysoev.ru     r = us->peer.http->request;
1811394Sigor@sysoev.ru 
1821394Sigor@sysoev.ru     nxt_mp_release(r->mem_pool);
1831394Sigor@sysoev.ru 
1841394Sigor@sysoev.ru     nxt_http_request_error(task, r, NXT_HTTP_BAD_GATEWAY);
1851394Sigor@sysoev.ru }
1861394Sigor@sysoev.ru 
1871394Sigor@sysoev.ru 
1881270Sigor@sysoev.ru static const nxt_http_request_state_t  nxt_http_proxy_header_send_state
1891270Sigor@sysoev.ru     nxt_aligned(64) =
1901270Sigor@sysoev.ru {
1911270Sigor@sysoev.ru     .ready_handler = nxt_http_proxy_header_send,
1921270Sigor@sysoev.ru     .error_handler = nxt_http_proxy_error,
1931270Sigor@sysoev.ru };
1941270Sigor@sysoev.ru 
1951270Sigor@sysoev.ru 
1961270Sigor@sysoev.ru static void
1971270Sigor@sysoev.ru nxt_http_proxy_header_send(nxt_task_t *task, void *obj, void *data)
1981270Sigor@sysoev.ru {
1991270Sigor@sysoev.ru     nxt_http_peer_t     *peer;
2001270Sigor@sysoev.ru     nxt_http_request_t  *r;
2011270Sigor@sysoev.ru 
2021270Sigor@sysoev.ru     r = obj;
2031270Sigor@sysoev.ru     peer = data;
2041270Sigor@sysoev.ru     r->state = &nxt_http_proxy_header_sent_state;
2051270Sigor@sysoev.ru 
2061270Sigor@sysoev.ru     nxt_http_proto[peer->protocol].peer_header_send(task, peer);
2071270Sigor@sysoev.ru }
2081270Sigor@sysoev.ru 
2091270Sigor@sysoev.ru 
2101270Sigor@sysoev.ru static const nxt_http_request_state_t  nxt_http_proxy_header_sent_state
2111270Sigor@sysoev.ru     nxt_aligned(64) =
2121270Sigor@sysoev.ru {
2131270Sigor@sysoev.ru     .ready_handler = nxt_http_proxy_header_sent,
2141270Sigor@sysoev.ru     .error_handler = nxt_http_proxy_error,
2151270Sigor@sysoev.ru };
2161270Sigor@sysoev.ru 
2171270Sigor@sysoev.ru 
2181270Sigor@sysoev.ru static void
2191270Sigor@sysoev.ru nxt_http_proxy_header_sent(nxt_task_t *task, void *obj, void *data)
2201270Sigor@sysoev.ru {
2211270Sigor@sysoev.ru     nxt_http_peer_t     *peer;
2221270Sigor@sysoev.ru     nxt_http_request_t  *r;
2231270Sigor@sysoev.ru 
2241270Sigor@sysoev.ru     r = obj;
2251270Sigor@sysoev.ru     peer = data;
2261270Sigor@sysoev.ru     r->state = &nxt_http_proxy_header_read_state;
2271270Sigor@sysoev.ru 
2281270Sigor@sysoev.ru     nxt_http_proto[peer->protocol].peer_header_read(task, peer);
2291270Sigor@sysoev.ru }
2301270Sigor@sysoev.ru 
2311270Sigor@sysoev.ru 
2321270Sigor@sysoev.ru static const nxt_http_request_state_t  nxt_http_proxy_header_read_state
2331270Sigor@sysoev.ru     nxt_aligned(64) =
2341270Sigor@sysoev.ru {
2351270Sigor@sysoev.ru     .ready_handler = nxt_http_proxy_header_read,
2361270Sigor@sysoev.ru     .error_handler = nxt_http_proxy_error,
2371270Sigor@sysoev.ru };
2381270Sigor@sysoev.ru 
2391270Sigor@sysoev.ru 
2401270Sigor@sysoev.ru static void
2411270Sigor@sysoev.ru nxt_http_proxy_header_read(nxt_task_t *task, void *obj, void *data)
2421270Sigor@sysoev.ru {
2431270Sigor@sysoev.ru     nxt_http_peer_t     *peer;
2441270Sigor@sysoev.ru     nxt_http_field_t    *f, *field;
2451270Sigor@sysoev.ru     nxt_http_request_t  *r;
2461270Sigor@sysoev.ru 
2471270Sigor@sysoev.ru     r = obj;
2481270Sigor@sysoev.ru     peer = data;
2491270Sigor@sysoev.ru 
2501270Sigor@sysoev.ru     r->status = peer->status;
2511270Sigor@sysoev.ru 
2521270Sigor@sysoev.ru     nxt_debug(task, "http proxy status: %d", peer->status);
2531270Sigor@sysoev.ru 
2541270Sigor@sysoev.ru     nxt_list_each(field, peer->fields) {
2551270Sigor@sysoev.ru 
2561270Sigor@sysoev.ru         nxt_debug(task, "http proxy header: \"%*s: %*s\"",
2571270Sigor@sysoev.ru                   (size_t) field->name_length, field->name,
2581270Sigor@sysoev.ru                   (size_t) field->value_length, field->value);
2591270Sigor@sysoev.ru 
2601270Sigor@sysoev.ru         if (!field->skip) {
2611270Sigor@sysoev.ru             f = nxt_list_add(r->resp.fields);
2621270Sigor@sysoev.ru             if (nxt_slow_path(f == NULL)) {
2631270Sigor@sysoev.ru                 nxt_http_proxy_error(task, r, peer);
2641270Sigor@sysoev.ru                 return;
2651270Sigor@sysoev.ru             }
2661270Sigor@sysoev.ru 
2671270Sigor@sysoev.ru             *f = *field;
2681270Sigor@sysoev.ru         }
2691270Sigor@sysoev.ru 
2701270Sigor@sysoev.ru     } nxt_list_loop;
2711270Sigor@sysoev.ru 
272*1505Sigor@sysoev.ru     r->state = &nxt_http_proxy_read_state;
273*1505Sigor@sysoev.ru 
2741270Sigor@sysoev.ru     nxt_http_request_header_send(task, r, nxt_http_proxy_send_body, peer);
2751270Sigor@sysoev.ru }
2761270Sigor@sysoev.ru 
2771270Sigor@sysoev.ru 
2781270Sigor@sysoev.ru static void
2791270Sigor@sysoev.ru nxt_http_proxy_send_body(nxt_task_t *task, void *obj, void *data)
2801270Sigor@sysoev.ru {
2811270Sigor@sysoev.ru     nxt_buf_t           *out;
2821270Sigor@sysoev.ru     nxt_http_peer_t     *peer;
2831270Sigor@sysoev.ru     nxt_http_request_t  *r;
2841270Sigor@sysoev.ru 
2851270Sigor@sysoev.ru     r = obj;
2861270Sigor@sysoev.ru     peer = data;
2871270Sigor@sysoev.ru     out = peer->body;
2881270Sigor@sysoev.ru 
2891270Sigor@sysoev.ru     if (out != NULL) {
2901270Sigor@sysoev.ru         peer->body = NULL;
291*1505Sigor@sysoev.ru         nxt_http_request_send(task, r, out);
292*1505Sigor@sysoev.ru 
2931270Sigor@sysoev.ru     }
2941270Sigor@sysoev.ru 
295*1505Sigor@sysoev.ru     if (!peer->closed) {
296*1505Sigor@sysoev.ru         nxt_http_proto[peer->protocol].peer_read(task, peer);
2971271Sigor@sysoev.ru     }
2981271Sigor@sysoev.ru }
2991271Sigor@sysoev.ru 
3001271Sigor@sysoev.ru 
3011270Sigor@sysoev.ru static const nxt_http_request_state_t  nxt_http_proxy_read_state
3021270Sigor@sysoev.ru     nxt_aligned(64) =
3031270Sigor@sysoev.ru {
3041270Sigor@sysoev.ru     .ready_handler = nxt_http_proxy_read,
3051270Sigor@sysoev.ru     .error_handler = nxt_http_proxy_error,
3061270Sigor@sysoev.ru };
3071270Sigor@sysoev.ru 
3081270Sigor@sysoev.ru 
3091270Sigor@sysoev.ru static void
3101270Sigor@sysoev.ru nxt_http_proxy_read(nxt_task_t *task, void *obj, void *data)
3111270Sigor@sysoev.ru {
3121270Sigor@sysoev.ru     nxt_buf_t           *out;
3131270Sigor@sysoev.ru     nxt_http_peer_t     *peer;
3141270Sigor@sysoev.ru     nxt_http_request_t  *r;
3151270Sigor@sysoev.ru 
3161270Sigor@sysoev.ru     r = obj;
3171270Sigor@sysoev.ru     peer = data;
3181270Sigor@sysoev.ru     out = peer->body;
3191270Sigor@sysoev.ru     peer->body = NULL;
3201270Sigor@sysoev.ru 
321*1505Sigor@sysoev.ru     if (out != NULL) {
322*1505Sigor@sysoev.ru         nxt_http_request_send(task, r, out);
323*1505Sigor@sysoev.ru     }
3241270Sigor@sysoev.ru 
325*1505Sigor@sysoev.ru     if (!peer->closed) {
3261270Sigor@sysoev.ru         nxt_http_proto[peer->protocol].peer_read(task, peer);
3271270Sigor@sysoev.ru 
3281270Sigor@sysoev.ru     } else {
3291270Sigor@sysoev.ru         nxt_http_proto[peer->protocol].peer_close(task, peer);
3301270Sigor@sysoev.ru 
3311270Sigor@sysoev.ru         nxt_mp_release(r->mem_pool);
3321270Sigor@sysoev.ru     }
3331270Sigor@sysoev.ru }
3341270Sigor@sysoev.ru 
3351270Sigor@sysoev.ru 
3361270Sigor@sysoev.ru nxt_buf_t *
3371270Sigor@sysoev.ru nxt_http_proxy_buf_mem_alloc(nxt_task_t *task, nxt_http_request_t *r,
3381270Sigor@sysoev.ru     size_t size)
3391270Sigor@sysoev.ru {
3401270Sigor@sysoev.ru     nxt_buf_t  *b;
3411270Sigor@sysoev.ru 
3421270Sigor@sysoev.ru     b = nxt_event_engine_buf_mem_alloc(task->thread->engine, size);
3431270Sigor@sysoev.ru     if (nxt_fast_path(b != NULL)) {
3441270Sigor@sysoev.ru         b->completion_handler = nxt_http_proxy_buf_mem_completion;
3451270Sigor@sysoev.ru         b->parent = r;
3461270Sigor@sysoev.ru         nxt_mp_retain(r->mem_pool);
3471270Sigor@sysoev.ru 
3481270Sigor@sysoev.ru     } else {
3491270Sigor@sysoev.ru         nxt_http_request_error(task, r, NXT_HTTP_INTERNAL_SERVER_ERROR);
3501270Sigor@sysoev.ru     }
3511270Sigor@sysoev.ru 
3521270Sigor@sysoev.ru     return b;
3531270Sigor@sysoev.ru }
3541270Sigor@sysoev.ru 
3551270Sigor@sysoev.ru 
3561270Sigor@sysoev.ru static void
3571270Sigor@sysoev.ru nxt_http_proxy_buf_mem_completion(nxt_task_t *task, void *obj, void *data)
3581270Sigor@sysoev.ru {
3591270Sigor@sysoev.ru     nxt_buf_t           *b, *next;
3601270Sigor@sysoev.ru     nxt_http_peer_t     *peer;
3611270Sigor@sysoev.ru     nxt_http_request_t  *r;
3621270Sigor@sysoev.ru 
3631270Sigor@sysoev.ru     b = obj;
3641270Sigor@sysoev.ru     r = data;
3651270Sigor@sysoev.ru 
3661270Sigor@sysoev.ru     peer = r->peer;
3671270Sigor@sysoev.ru 
3681270Sigor@sysoev.ru     do {
3691270Sigor@sysoev.ru         next = b->next;
3701270Sigor@sysoev.ru 
3711270Sigor@sysoev.ru         nxt_http_proxy_buf_mem_free(task, r, b);
3721270Sigor@sysoev.ru 
3731270Sigor@sysoev.ru         b = next;
3741270Sigor@sysoev.ru     } while (b != NULL);
3751270Sigor@sysoev.ru 
3761270Sigor@sysoev.ru     if (!peer->closed) {
3771270Sigor@sysoev.ru         nxt_http_proto[peer->protocol].peer_read(task, peer);
3781270Sigor@sysoev.ru     }
3791270Sigor@sysoev.ru }
3801270Sigor@sysoev.ru 
3811270Sigor@sysoev.ru 
3821270Sigor@sysoev.ru void
3831270Sigor@sysoev.ru nxt_http_proxy_buf_mem_free(nxt_task_t *task, nxt_http_request_t *r,
3841270Sigor@sysoev.ru     nxt_buf_t *b)
3851270Sigor@sysoev.ru {
3861270Sigor@sysoev.ru     nxt_event_engine_buf_mem_free(task->thread->engine, b);
3871270Sigor@sysoev.ru 
3881270Sigor@sysoev.ru     nxt_mp_release(r->mem_pool);
3891270Sigor@sysoev.ru }
3901270Sigor@sysoev.ru 
3911270Sigor@sysoev.ru 
3921270Sigor@sysoev.ru static void
3931270Sigor@sysoev.ru nxt_http_proxy_error(nxt_task_t *task, void *obj, void *data)
3941270Sigor@sysoev.ru {
3951270Sigor@sysoev.ru     nxt_http_peer_t     *peer;
3961270Sigor@sysoev.ru     nxt_http_request_t  *r;
3971270Sigor@sysoev.ru 
3981270Sigor@sysoev.ru     r = obj;
3991270Sigor@sysoev.ru     peer = r->peer;
4001270Sigor@sysoev.ru 
4011270Sigor@sysoev.ru     nxt_http_proto[peer->protocol].peer_close(task, peer);
4021270Sigor@sysoev.ru 
4031270Sigor@sysoev.ru     nxt_mp_release(r->mem_pool);
4041270Sigor@sysoev.ru 
405*1505Sigor@sysoev.ru     nxt_http_request_error(&r->task, r, peer->status);
4061270Sigor@sysoev.ru }
4071270Sigor@sysoev.ru 
4081270Sigor@sysoev.ru 
4091270Sigor@sysoev.ru nxt_int_t
4101270Sigor@sysoev.ru nxt_http_proxy_date(void *ctx, nxt_http_field_t *field, uintptr_t data)
4111270Sigor@sysoev.ru {
4121270Sigor@sysoev.ru     nxt_http_request_t  *r;
4131270Sigor@sysoev.ru 
4141270Sigor@sysoev.ru     r = ctx;
4151270Sigor@sysoev.ru 
4161270Sigor@sysoev.ru     r->resp.date = field;
4171270Sigor@sysoev.ru 
4181270Sigor@sysoev.ru     return NXT_OK;
4191270Sigor@sysoev.ru }
4201270Sigor@sysoev.ru 
4211270Sigor@sysoev.ru 
4221270Sigor@sysoev.ru nxt_int_t
4231270Sigor@sysoev.ru nxt_http_proxy_content_length(void *ctx, nxt_http_field_t *field,
4241270Sigor@sysoev.ru     uintptr_t data)
4251270Sigor@sysoev.ru {
4261270Sigor@sysoev.ru     nxt_off_t           n;
4271270Sigor@sysoev.ru     nxt_http_request_t  *r;
4281270Sigor@sysoev.ru 
4291270Sigor@sysoev.ru     r = ctx;
4301270Sigor@sysoev.ru 
4311270Sigor@sysoev.ru     r->resp.content_length = field;
4321270Sigor@sysoev.ru 
4331270Sigor@sysoev.ru     n = nxt_off_t_parse(field->value, field->value_length);
4341270Sigor@sysoev.ru 
4351270Sigor@sysoev.ru     if (nxt_fast_path(n >= 0)) {
4361270Sigor@sysoev.ru         r->resp.content_length_n = n;
4371270Sigor@sysoev.ru     }
4381270Sigor@sysoev.ru 
4391270Sigor@sysoev.ru     return NXT_OK;
4401270Sigor@sysoev.ru }
4411270Sigor@sysoev.ru 
4421270Sigor@sysoev.ru 
4431270Sigor@sysoev.ru nxt_int_t
4441270Sigor@sysoev.ru nxt_http_proxy_skip(void *ctx, nxt_http_field_t *field, uintptr_t data)
4451270Sigor@sysoev.ru {
4461270Sigor@sysoev.ru     field->skip = 1;
4471270Sigor@sysoev.ru 
4481270Sigor@sysoev.ru     return NXT_OK;
4491270Sigor@sysoev.ru }
450