blob: d1dc290fc0782c7e3ca1c538ff9456b8249f89fb [file] [log] [blame]
Igor Sysoev6de5c2c2002-08-06 16:39:45 +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 Sysoev6de5c2c2002-08-06 16:39:45 +00007#include <ngx_config.h>
Igor Sysoev0ad17c02002-08-26 15:18:19 +00008#include <ngx_core.h>
Igor Sysoev6de5c2c2002-08-06 16:39:45 +00009#include <ngx_http.h>
10
Igor Sysoeva58e3ca2002-09-02 14:48:24 +000011
Igor Sysoev6b863e32003-05-12 15:52:24 +000012static void ngx_http_init_request(ngx_event_t *ev);
13static void ngx_http_process_request_line(ngx_event_t *rev);
14static void ngx_http_process_request_headers(ngx_event_t *rev);
Igor Sysoev6b863e32003-05-12 15:52:24 +000015static ssize_t ngx_http_read_request_header(ngx_http_request_t *r);
Igor Sysoevf7abd722004-09-23 06:32:00 +000016static ngx_int_t ngx_http_alloc_large_header_buffer(ngx_http_request_t *r,
Igor Sysoevc1571722005-03-19 12:38:37 +000017 ngx_uint_t request_line);
Igor Sysoev899b44e2005-05-12 14:58:06 +000018
19static ngx_int_t ngx_http_process_header_line(ngx_http_request_t *r,
20 ngx_table_elt_t *h, ngx_uint_t offset);
Igor Sysoevb145b062005-06-15 18:33:41 +000021static ngx_int_t ngx_http_process_unique_header_line(ngx_http_request_t *r,
22 ngx_table_elt_t *h, ngx_uint_t offset);
Igor Sysoev5fd09312008-05-15 14:44:47 +000023static ngx_int_t ngx_http_process_host(ngx_http_request_t *r,
24 ngx_table_elt_t *h, ngx_uint_t offset);
Igor Sysoevec67b192007-09-09 18:28:49 +000025static ngx_int_t ngx_http_process_connection(ngx_http_request_t *r,
26 ngx_table_elt_t *h, ngx_uint_t offset);
Igor Sysoevbc5fb292008-05-13 09:18:58 +000027static ngx_int_t ngx_http_process_user_agent(ngx_http_request_t *r,
28 ngx_table_elt_t *h, ngx_uint_t offset);
Igor Sysoev899b44e2005-05-12 14:58:06 +000029static ngx_int_t ngx_http_process_cookie(ngx_http_request_t *r,
30 ngx_table_elt_t *h, ngx_uint_t offset);
31
Igor Sysoev2f657222004-06-16 15:32:11 +000032static ngx_int_t ngx_http_process_request_header(ngx_http_request_t *r);
Igor Sysoev62864d12007-06-04 20:40:03 +000033static void ngx_http_process_request(ngx_http_request_t *r);
Igor Sysoev5fd09312008-05-15 14:44:47 +000034static ssize_t ngx_http_validate_host(u_char *host, size_t len);
35static ngx_int_t ngx_http_find_virtual_server(ngx_http_request_t *r,
36 u_char *host, size_t len);
Igor Sysoev6b863e32003-05-12 15:52:24 +000037
Igor Sysoev899b44e2005-05-12 14:58:06 +000038static void ngx_http_request_handler(ngx_event_t *ev);
Igor Sysoev85ef94b2005-06-23 13:41:06 +000039static ngx_int_t ngx_http_set_write_handler(ngx_http_request_t *r);
Igor Sysoev899b44e2005-05-12 14:58:06 +000040static void ngx_http_writer(ngx_http_request_t *r);
Igor Sysoev851cd732008-12-08 14:23:20 +000041static void ngx_http_request_finalizer(ngx_http_request_t *r);
Igor Sysoev6b863e32003-05-12 15:52:24 +000042
Igor Sysoevd581fd52003-05-13 16:02:32 +000043static void ngx_http_set_keepalive(ngx_http_request_t *r);
44static void ngx_http_keepalive_handler(ngx_event_t *ev);
45static void ngx_http_set_lingering_close(ngx_http_request_t *r);
46static void ngx_http_lingering_close_handler(ngx_event_t *ev);
Igor Sysoevc2807ec2006-02-16 15:26:46 +000047static ngx_int_t ngx_http_post_action(ngx_http_request_t *r);
Igor Sysoev90c08142005-07-25 09:41:38 +000048static void ngx_http_close_request(ngx_http_request_t *r, ngx_int_t error);
Igor Sysoev9ac946b2005-10-24 15:09:41 +000049static void ngx_http_request_done(ngx_http_request_t *r, ngx_int_t error);
Igor Sysoev2eec1e12008-09-27 15:08:02 +000050static void ngx_http_log_request(ngx_http_request_t *r);
Igor Sysoev90c08142005-07-25 09:41:38 +000051static void ngx_http_close_connection(ngx_connection_t *c);
Igor Sysoevd581fd52003-05-13 16:02:32 +000052
Igor Sysoeve5a222c2005-01-25 12:27:35 +000053static u_char *ngx_http_log_error(ngx_log_t *log, u_char *buf, size_t len);
Igor Sysoevbb28b6d2006-07-11 13:20:19 +000054static u_char *ngx_http_log_error_handler(ngx_http_request_t *r,
55 ngx_http_request_t *sr, u_char *buf, size_t len);
Igor Sysoev0ad17c02002-08-26 15:18:19 +000056
Igor Sysoevceb99292005-09-06 16:09:32 +000057#if (NGX_HTTP_SSL)
58static void ngx_http_ssl_handshake(ngx_event_t *rev);
Igor Sysoev9fa5a822005-09-30 14:41:25 +000059static void ngx_http_ssl_handshake_handler(ngx_connection_t *c);
Igor Sysoevceb99292005-09-06 16:09:32 +000060#endif
61
Igor Sysoev6de5c2c2002-08-06 16:39:45 +000062
Igor Sysoevc04deca2005-03-28 14:43:02 +000063static char *ngx_http_client_errors[] = {
Igor Sysoev0ad17c02002-08-26 15:18:19 +000064
Igor Sysoevc04deca2005-03-28 14:43:02 +000065 /* NGX_HTTP_PARSE_INVALID_METHOD */
66 "client sent invalid method",
Igor Sysoev6a644c62003-03-04 06:33:48 +000067
Igor Sysoevc04deca2005-03-28 14:43:02 +000068 /* NGX_HTTP_PARSE_INVALID_REQUEST */
69 "client sent invalid request",
70
71 /* NGX_HTTP_PARSE_INVALID_09_METHOD */
72 "client sent invalid method in HTTP/0.9 request"
Igor Sysoev016b8522002-08-29 16:59:54 +000073};
74
75
Igor Sysoevaab4d8c2004-09-06 18:45:00 +000076ngx_http_header_t ngx_http_headers_in[] = {
Igor Sysoevcbc7c612008-08-19 19:40:45 +000077 { ngx_string("Host"), offsetof(ngx_http_headers_in_t, host),
78 ngx_http_process_host },
Igor Sysoevaab4d8c2004-09-06 18:45:00 +000079
Igor Sysoevcbc7c612008-08-19 19:40:45 +000080 { ngx_string("Connection"), offsetof(ngx_http_headers_in_t, connection),
81 ngx_http_process_connection },
Igor Sysoev899b44e2005-05-12 14:58:06 +000082
83 { ngx_string("If-Modified-Since"),
84 offsetof(ngx_http_headers_in_t, if_modified_since),
Igor Sysoev4f62b732007-08-29 15:32:49 +000085 ngx_http_process_unique_header_line },
Igor Sysoev899b44e2005-05-12 14:58:06 +000086
Igor Sysoevcbc7c612008-08-19 19:40:45 +000087 { ngx_string("User-Agent"), offsetof(ngx_http_headers_in_t, user_agent),
88 ngx_http_process_user_agent },
Igor Sysoev899b44e2005-05-12 14:58:06 +000089
90 { ngx_string("Referer"), offsetof(ngx_http_headers_in_t, referer),
91 ngx_http_process_header_line },
92
93 { ngx_string("Content-Length"),
94 offsetof(ngx_http_headers_in_t, content_length),
Igor Sysoevb145b062005-06-15 18:33:41 +000095 ngx_http_process_unique_header_line },
Igor Sysoev899b44e2005-05-12 14:58:06 +000096
97 { ngx_string("Content-Type"),
98 offsetof(ngx_http_headers_in_t, content_type),
99 ngx_http_process_header_line },
100
101 { ngx_string("Range"), offsetof(ngx_http_headers_in_t, range),
102 ngx_http_process_header_line },
103
Igor Sysoev6c2f0522007-11-09 15:41:35 +0000104 { ngx_string("If-Range"),
105 offsetof(ngx_http_headers_in_t, if_range),
106 ngx_http_process_unique_header_line },
107
Igor Sysoevc31a9bb2005-11-26 10:11:11 +0000108 { ngx_string("Transfer-Encoding"),
109 offsetof(ngx_http_headers_in_t, transfer_encoding),
110 ngx_http_process_header_line },
111
Igor Sysoev1e1f4c82008-06-26 13:00:39 +0000112 { ngx_string("Expect"),
113 offsetof(ngx_http_headers_in_t, expect),
114 ngx_http_process_unique_header_line },
115
Igor Sysoevaab4d8c2004-09-06 18:45:00 +0000116#if (NGX_HTTP_GZIP)
117 { ngx_string("Accept-Encoding"),
Igor Sysoev899b44e2005-05-12 14:58:06 +0000118 offsetof(ngx_http_headers_in_t, accept_encoding),
119 ngx_http_process_header_line },
120
121 { ngx_string("Via"), offsetof(ngx_http_headers_in_t, via),
122 ngx_http_process_header_line },
Igor Sysoevaab4d8c2004-09-06 18:45:00 +0000123#endif
124
125 { ngx_string("Authorization"),
Igor Sysoev899b44e2005-05-12 14:58:06 +0000126 offsetof(ngx_http_headers_in_t, authorization),
Igor Sysoevb145b062005-06-15 18:33:41 +0000127 ngx_http_process_unique_header_line },
Igor Sysoevaab4d8c2004-09-06 18:45:00 +0000128
Igor Sysoev899b44e2005-05-12 14:58:06 +0000129 { ngx_string("Keep-Alive"), offsetof(ngx_http_headers_in_t, keep_alive),
130 ngx_http_process_header_line },
Igor Sysoevaab4d8c2004-09-06 18:45:00 +0000131
Igor Sysoev09c684b2005-11-09 17:25:55 +0000132#if (NGX_HTTP_PROXY || NGX_HTTP_REALIP)
Igor Sysoevaab4d8c2004-09-06 18:45:00 +0000133 { ngx_string("X-Forwarded-For"),
Igor Sysoev899b44e2005-05-12 14:58:06 +0000134 offsetof(ngx_http_headers_in_t, x_forwarded_for),
135 ngx_http_process_header_line },
Igor Sysoevaab4d8c2004-09-06 18:45:00 +0000136#endif
Igor Sysoev02025fd2005-01-18 13:03:58 +0000137
Igor Sysoev09c684b2005-11-09 17:25:55 +0000138#if (NGX_HTTP_REALIP)
139 { ngx_string("X-Real-IP"),
140 offsetof(ngx_http_headers_in_t, x_real_ip),
141 ngx_http_process_header_line },
142#endif
143
Igor Sysoev1ebfead2005-02-16 13:40:36 +0000144#if (NGX_HTTP_HEADERS)
Igor Sysoev899b44e2005-05-12 14:58:06 +0000145 { ngx_string("Accept"), offsetof(ngx_http_headers_in_t, accept),
146 ngx_http_process_header_line },
147
Igor Sysoev1ebfead2005-02-16 13:40:36 +0000148 { ngx_string("Accept-Language"),
Igor Sysoev899b44e2005-05-12 14:58:06 +0000149 offsetof(ngx_http_headers_in_t, accept_language),
150 ngx_http_process_header_line },
Igor Sysoev1ebfead2005-02-16 13:40:36 +0000151#endif
152
Igor Sysoev7bdb7202006-04-19 15:30:56 +0000153#if (NGX_HTTP_DAV)
154 { ngx_string("Depth"), offsetof(ngx_http_headers_in_t, depth),
155 ngx_http_process_header_line },
156
157 { ngx_string("Destination"), offsetof(ngx_http_headers_in_t, destination),
158 ngx_http_process_header_line },
Igor Sysoevda173ab2006-08-30 10:39:17 +0000159
Igor Sysoev56331ba2007-01-18 21:11:23 +0000160 { ngx_string("Overwrite"), offsetof(ngx_http_headers_in_t, overwrite),
161 ngx_http_process_header_line },
162
Igor Sysoevda173ab2006-08-30 10:39:17 +0000163 { ngx_string("Date"), offsetof(ngx_http_headers_in_t, date),
164 ngx_http_process_header_line },
Igor Sysoev7bdb7202006-04-19 15:30:56 +0000165#endif
166
Igor Sysoev899b44e2005-05-12 14:58:06 +0000167 { ngx_string("Cookie"), 0, ngx_http_process_cookie },
168
169 { ngx_null_string, 0, NULL }
Igor Sysoevaab4d8c2004-09-06 18:45:00 +0000170};
171
172
Igor Sysoevc1571722005-03-19 12:38:37 +0000173void
174ngx_http_init_connection(ngx_connection_t *c)
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000175{
Igor Sysoev6253ca12003-05-27 12:18:54 +0000176 ngx_event_t *rev;
Igor Sysoev160d7742003-11-19 16:26:41 +0000177 ngx_http_log_ctx_t *ctx;
Igor Sysoev6b863e32003-05-12 15:52:24 +0000178
Igor Sysoevc1571722005-03-19 12:38:37 +0000179 ctx = ngx_palloc(c->pool, sizeof(ngx_http_log_ctx_t));
180 if (ctx == NULL) {
Igor Sysoev6b863e32003-05-12 15:52:24 +0000181 ngx_http_close_connection(c);
182 return;
183 }
184
Igor Sysoevcc595672007-12-30 08:01:50 +0000185 ctx->connection = c;
Igor Sysoev1b735832004-11-11 14:07:14 +0000186 ctx->request = NULL;
Igor Sysoevbb28b6d2006-07-11 13:20:19 +0000187 ctx->current_request = NULL;
Igor Sysoeve5a222c2005-01-25 12:27:35 +0000188
189 c->log->connection = c->number;
Igor Sysoev6b863e32003-05-12 15:52:24 +0000190 c->log->handler = ngx_http_log_error;
Igor Sysoeve5a222c2005-01-25 12:27:35 +0000191 c->log->data = ctx;
192 c->log->action = "reading client request line";
193
Igor Sysoevc7a2f682004-02-10 16:23:38 +0000194 c->log_error = NGX_ERROR_INFO;
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000195
Igor Sysoevdc479b42003-03-20 16:09:44 +0000196 rev = c->read;
Igor Sysoev899b44e2005-05-12 14:58:06 +0000197 rev->handler = ngx_http_init_request;
Igor Sysoev9ac946b2005-10-24 15:09:41 +0000198 c->write->handler = ngx_http_empty_handler;
Igor Sysoev7af6b162004-02-09 07:46:43 +0000199
Igor Sysoev055951d2005-10-21 19:12:18 +0000200#if (NGX_STAT_STUB)
201 ngx_atomic_fetch_add(ngx_stat_reading, 1);
202#endif
203
Igor Sysoevdc479b42003-03-20 16:09:44 +0000204 if (rev->ready) {
Igor Sysoev0ab91b92004-06-06 19:49:18 +0000205 /* the deferred accept(), rtsig, aio, iocp */
Igor Sysoev709405b2004-03-31 15:26:46 +0000206
Igor Sysoevffe71442006-02-08 15:33:12 +0000207 if (ngx_use_accept_mutex) {
Igor Sysoevc2068d02005-10-19 12:33:58 +0000208 ngx_post_event(rev, &ngx_posted_events);
Igor Sysoev709405b2004-03-31 15:26:46 +0000209 return;
210 }
211
Igor Sysoev6b863e32003-05-12 15:52:24 +0000212 ngx_http_init_request(rev);
213 return;
Igor Sysoev016b8522002-08-29 16:59:54 +0000214 }
Igor Sysoev73009772003-02-06 17:21:13 +0000215
Igor Sysoev239baac2003-06-11 15:28:34 +0000216 ngx_add_timer(rev, c->listening->post_accept_timeout);
Igor Sysoev9b25d692003-01-26 21:08:14 +0000217
Igor Sysoevc9aae142008-12-09 17:27:48 +0000218 if (ngx_handle_read_event(rev, 0) != NGX_OK) {
Igor Sysoev055951d2005-10-21 19:12:18 +0000219#if (NGX_STAT_STUB)
220 ngx_atomic_fetch_add(ngx_stat_reading, -1);
221#endif
Igor Sysoevb3e73d82003-10-10 15:10:50 +0000222 ngx_http_close_connection(c);
223 return;
224 }
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000225}
226
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000227
Igor Sysoev7f7846d2006-04-26 09:52:47 +0000228static void
229ngx_http_init_request(ngx_event_t *rev)
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000230{
Igor Sysoevb4ccb9f2007-04-21 07:50:19 +0000231 ngx_time_t *tp;
Igor Sysoev7f7846d2006-04-26 09:52:47 +0000232 ngx_uint_t i;
Igor Sysoev09c684b2005-11-09 17:25:55 +0000233 ngx_connection_t *c;
234 ngx_http_request_t *r;
Igor Sysoeva35eacc2009-02-21 07:02:02 +0000235 struct sockaddr_in *sin;
236 ngx_http_port_t *port;
237 ngx_http_in_addr_t *addr;
Igor Sysoev09c684b2005-11-09 17:25:55 +0000238 ngx_http_log_ctx_t *ctx;
Igor Sysoeva35eacc2009-02-21 07:02:02 +0000239 ngx_http_addr_conf_t *addr_conf;
Igor Sysoev09c684b2005-11-09 17:25:55 +0000240 ngx_http_connection_t *hc;
Igor Sysoev09c684b2005-11-09 17:25:55 +0000241 ngx_http_core_srv_conf_t *cscf;
242 ngx_http_core_loc_conf_t *clcf;
243 ngx_http_core_main_conf_t *cmcf;
Igor Sysoeva35eacc2009-02-21 07:02:02 +0000244#if (NGX_HAVE_INET6)
245 struct sockaddr_in6 *sin6;
246 ngx_http_in6_addr_t *addr6;
247#endif
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000248
Igor Sysoev055951d2005-10-21 19:12:18 +0000249#if (NGX_STAT_STUB)
250 ngx_atomic_fetch_add(ngx_stat_reading, -1);
251#endif
252
Igor Sysoeva9830112003-05-19 16:39:14 +0000253 c = rev->data;
Igor Sysoeva9830112003-05-19 16:39:14 +0000254
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000255 if (rev->timedout) {
256 ngx_log_error(NGX_LOG_INFO, c->log, NGX_ETIMEDOUT, "client timed out");
Igor Sysoev85cccfb2004-09-15 16:00:43 +0000257
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000258 ngx_http_close_connection(c);
259 return;
260 }
261
Igor Sysoevf7abd722004-09-23 06:32:00 +0000262 hc = c->data;
Igor Sysoevdd888c42004-09-21 05:38:28 +0000263
Igor Sysoev055951d2005-10-21 19:12:18 +0000264 if (hc == NULL) {
Igor Sysoevc1571722005-03-19 12:38:37 +0000265 hc = ngx_pcalloc(c->pool, sizeof(ngx_http_connection_t));
266 if (hc == NULL) {
Igor Sysoevdd888c42004-09-21 05:38:28 +0000267 ngx_http_close_connection(c);
268 return;
269 }
Igor Sysoevf7abd722004-09-23 06:32:00 +0000270 }
Igor Sysoevdd888c42004-09-21 05:38:28 +0000271
Igor Sysoevf7abd722004-09-23 06:32:00 +0000272 r = hc->request;
273
274 if (r) {
275 ngx_memzero(r, sizeof(ngx_http_request_t));
276
277 r->pipeline = hc->pipeline;
278
279 if (hc->nbusy) {
280 r->header_in = hc->busy[0];
281 }
282
Igor Sysoevf7abd722004-09-23 06:32:00 +0000283 } else {
Igor Sysoevc1571722005-03-19 12:38:37 +0000284 r = ngx_pcalloc(c->pool, sizeof(ngx_http_request_t));
285 if (r == NULL) {
Igor Sysoev0a280a32003-10-12 16:49:16 +0000286 ngx_http_close_connection(c);
287 return;
288 }
Igor Sysoev59f3aa32004-06-24 16:07:04 +0000289
Igor Sysoevdd888c42004-09-21 05:38:28 +0000290 hc->request = r;
Igor Sysoev6253ca12003-05-27 12:18:54 +0000291 }
292
Igor Sysoevdd888c42004-09-21 05:38:28 +0000293 c->data = r;
294 r->http_connection = hc;
295
Igor Sysoev59f3aa32004-06-24 16:07:04 +0000296 c->sent = 0;
Igor Sysoevc83f6832004-06-24 07:53:37 +0000297 r->signature = NGX_HTTP_MODULE;
Igor Sysoev3b30a902003-12-25 20:26:58 +0000298
Igor Sysoev6253ca12003-05-27 12:18:54 +0000299 /* find the server configuration for the address:port */
300
Igor Sysoeva35eacc2009-02-21 07:02:02 +0000301 port = c->listening->servers;
Igor Sysoev6253ca12003-05-27 12:18:54 +0000302
Igor Sysoevc7a9b7a2007-12-29 16:00:34 +0000303 r->connection = c;
304
Igor Sysoeva35eacc2009-02-21 07:02:02 +0000305 if (port->naddrs > 1) {
Igor Sysoev6253ca12003-05-27 12:18:54 +0000306
Igor Sysoev239baac2003-06-11 15:28:34 +0000307 /*
Igor Sysoeva35eacc2009-02-21 07:02:02 +0000308 * there are several addresses on this port and one of them
309 * is an "*:port" wildcard so getsockname() in ngx_http_server_addr()
310 * is required to determine a server address
Igor Sysoev239baac2003-06-11 15:28:34 +0000311 */
Igor Sysoev6253ca12003-05-27 12:18:54 +0000312
Igor Sysoeva35eacc2009-02-21 07:02:02 +0000313 c->local_sockaddr = NULL;
Igor Sysoev6253ca12003-05-27 12:18:54 +0000314
Igor Sysoeva35eacc2009-02-21 07:02:02 +0000315 if (ngx_http_server_addr(r, NULL) != NGX_OK) {
316 ngx_http_close_connection(c);
317 return;
Igor Sysoev6253ca12003-05-27 12:18:54 +0000318 }
319
Igor Sysoeva35eacc2009-02-21 07:02:02 +0000320 switch (c->local_sockaddr->sa_family) {
Igor Sysoev6253ca12003-05-27 12:18:54 +0000321
Igor Sysoeva35eacc2009-02-21 07:02:02 +0000322#if (NGX_HAVE_INET6)
323 case AF_INET6:
324 sin6 = (struct sockaddr_in6 *) c->local_sockaddr;
325
326 addr6 = (ngx_http_in6_addr_t *) port->addrs;
327
328 /* the last address is "*" */
329
330 for (i = 0; i < port->naddrs - 1; i++) {
331 if (ngx_memcmp(&addr6[i].addr6, &sin6->sin6_addr, 16) == 0) {
332 break;
333 }
Igor Sysoev6253ca12003-05-27 12:18:54 +0000334 }
Igor Sysoeva35eacc2009-02-21 07:02:02 +0000335
336 addr_conf = &addr6[i].conf;
337
338 break;
339#endif
340
341 default: /* AF_INET */
342 sin = (struct sockaddr_in *) c->local_sockaddr;
343
344 addr = port->addrs;
345
346 /* the last address is "*" */
347
348 for (i = 0; i < port->naddrs - 1; i++) {
349 if (addr[i].addr == sin->sin_addr.s_addr) {
350 break;
351 }
352 }
353
354 addr_conf = &addr[i].conf;
355
356 break;
Igor Sysoev6253ca12003-05-27 12:18:54 +0000357 }
358
359 } else {
Igor Sysoeva35eacc2009-02-21 07:02:02 +0000360
361 switch (c->local_sockaddr->sa_family) {
362
363#if (NGX_HAVE_INET6)
364 case AF_INET6:
365 addr6 = (ngx_http_in6_addr_t *) port->addrs;
366 addr_conf = &addr6[0].conf;
367 break;
368#endif
369
370 default: /* AF_INET */
371 addr = port->addrs;
372 addr_conf = &addr[0].conf;
Igor Sysoeva35eacc2009-02-21 07:02:02 +0000373 break;
374 }
Igor Sysoev6253ca12003-05-27 12:18:54 +0000375 }
376
Igor Sysoeva35eacc2009-02-21 07:02:02 +0000377 r->virtual_names = addr_conf->virtual_names;
Igor Sysoev6253ca12003-05-27 12:18:54 +0000378
379 /* the default server configuration for the address:port */
Igor Sysoeva35eacc2009-02-21 07:02:02 +0000380 cscf = addr_conf->core_srv_conf;
Igor Sysoev6253ca12003-05-27 12:18:54 +0000381
382 r->main_conf = cscf->ctx->main_conf;
383 r->srv_conf = cscf->ctx->srv_conf;
384 r->loc_conf = cscf->ctx->loc_conf;
385
Igor Sysoev899b44e2005-05-12 14:58:06 +0000386 rev->handler = ngx_http_process_request_line;
Igor Sysoevf38e0462004-07-16 17:11:43 +0000387
Igor Sysoev1c3567e2004-07-15 16:35:51 +0000388#if (NGX_HTTP_SSL)
389
Igor Sysoev86ef6aa2007-12-10 12:09:51 +0000390 {
391 ngx_http_ssl_srv_conf_t *sscf;
392
Igor Sysoevf38e0462004-07-16 17:11:43 +0000393 sscf = ngx_http_get_module_srv_conf(r, ngx_http_ssl_module);
Igor Sysoeva35eacc2009-02-21 07:02:02 +0000394 if (sscf->enable || addr_conf->ssl) {
Igor Sysoev74a5ddb2004-07-18 19:11:20 +0000395
396 if (c->ssl == NULL) {
Igor Sysoevf100c782008-09-01 14:19:01 +0000397
398 c->log->action = "SSL handshaking";
399
Igor Sysoeva35eacc2009-02-21 07:02:02 +0000400 if (addr_conf->ssl && sscf->ssl.ctx == NULL) {
Igor Sysoevf100c782008-09-01 14:19:01 +0000401 ngx_log_error(NGX_LOG_ERR, c->log, 0,
402 "no \"ssl_certificate\" is defined "
403 "in server listening on SSL port");
404 ngx_http_close_connection(c);
405 return;
406 }
407
Igor Sysoev9fa5a822005-09-30 14:41:25 +0000408 if (ngx_ssl_create_connection(&sscf->ssl, c, NGX_SSL_BUFFER)
Igor Sysoev02f742b2005-04-08 15:18:55 +0000409 == NGX_ERROR)
Igor Sysoev74a5ddb2004-07-18 19:11:20 +0000410 {
411 ngx_http_close_connection(c);
412 return;
413 }
414
Igor Sysoev899b44e2005-05-12 14:58:06 +0000415 rev->handler = ngx_http_ssl_handshake;
Igor Sysoev1c3567e2004-07-15 16:35:51 +0000416 }
417
Igor Sysoevd52477f2005-05-16 13:53:20 +0000418 r->main_filter_need_in_memory = 1;
Igor Sysoev1c3567e2004-07-15 16:35:51 +0000419 }
Igor Sysoev86ef6aa2007-12-10 12:09:51 +0000420 }
Igor Sysoev1c3567e2004-07-15 16:35:51 +0000421
Igor Sysoev4aa88882004-07-14 20:07:58 +0000422#endif
423
Igor Sysoev890fc962003-07-20 21:15:59 +0000424 clcf = ngx_http_get_module_loc_conf(r, ngx_http_core_module);
425 c->log->file = clcf->err_log->file;
Igor Sysoevfff32322004-04-08 15:58:25 +0000426 if (!(c->log->log_level & NGX_LOG_DEBUG_CONNECTION)) {
427 c->log->log_level = clcf->err_log->log_level;
428 }
Igor Sysoev890fc962003-07-20 21:15:59 +0000429
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000430 if (c->buffer == NULL) {
Igor Sysoev369145c2004-05-28 15:49:23 +0000431 c->buffer = ngx_create_temp_buf(c->pool,
432 cscf->client_header_buffer_size);
Igor Sysoev6b863e32003-05-12 15:52:24 +0000433 if (c->buffer == NULL) {
434 ngx_http_close_connection(c);
435 return;
436 }
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000437 }
438
Igor Sysoevf7abd722004-09-23 06:32:00 +0000439 if (r->header_in == NULL) {
440 r->header_in = c->buffer;
441 }
442
Igor Sysoevc1571722005-03-19 12:38:37 +0000443 r->pool = ngx_create_pool(cscf->request_pool_size, c->log);
444 if (r->pool == NULL) {
Igor Sysoev6b863e32003-05-12 15:52:24 +0000445 ngx_http_close_connection(c);
446 return;
447 }
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000448
Igor Sysoev89690bf2004-03-23 06:01:52 +0000449
Igor Sysoevaab4d8c2004-09-06 18:45:00 +0000450 if (ngx_list_init(&r->headers_out.headers, r->pool, 20,
Igor Sysoevc31a9bb2005-11-26 10:11:11 +0000451 sizeof(ngx_table_elt_t))
452 == NGX_ERROR)
Igor Sysoev980a9242004-09-05 19:54:02 +0000453 {
454 ngx_http_close_request(r, NGX_HTTP_INTERNAL_SERVER_ERROR);
Igor Sysoev980a9242004-09-05 19:54:02 +0000455 return;
456 }
457
Igor Sysoev6b863e32003-05-12 15:52:24 +0000458 r->ctx = ngx_pcalloc(r->pool, sizeof(void *) * ngx_http_max_module);
459 if (r->ctx == NULL) {
460 ngx_http_close_request(r, NGX_HTTP_INTERNAL_SERVER_ERROR);
Igor Sysoev6b863e32003-05-12 15:52:24 +0000461 return;
462 }
Igor Sysoeve0268b92002-09-11 15:18:33 +0000463
Igor Sysoev09c684b2005-11-09 17:25:55 +0000464 cmcf = ngx_http_get_module_main_conf(r, ngx_http_core_module);
465
466 r->variables = ngx_pcalloc(r->pool, cmcf->variables.nelts
467 * sizeof(ngx_http_variable_value_t));
468 if (r->variables == NULL) {
469 ngx_http_close_request(r, NGX_HTTP_INTERNAL_SERVER_ERROR);
470 return;
471 }
472
Igor Sysoevc78c41c2004-07-07 06:15:04 +0000473 c->single_connection = 1;
Igor Sysoevd3283ff2005-12-05 13:18:09 +0000474 c->destroyed = 0;
475
Igor Sysoevf6e1fe32005-10-04 10:38:53 +0000476 r->main = r;
477
Igor Sysoevb4ccb9f2007-04-21 07:50:19 +0000478 tp = ngx_timeofday();
479 r->start_sec = tp->sec;
480 r->start_msec = tp->msec;
Igor Sysoev11d75322005-03-01 15:20:36 +0000481
Igor Sysoev94e32ce2006-04-07 14:08:04 +0000482 r->method = NGX_HTTP_UNKNOWN;
483
Igor Sysoeva09f08d2003-04-25 14:43:13 +0000484 r->headers_in.content_length_n = -1;
Igor Sysoev419f9ac2003-10-21 16:49:56 +0000485 r->headers_in.keep_alive_n = -1;
486 r->headers_out.content_length_n = -1;
Igor Sysoev42feecb2002-12-15 06:25:09 +0000487 r->headers_out.last_modified_time = -1;
488
Igor Sysoevef316432006-08-16 13:09:33 +0000489 r->uri_changes = NGX_HTTP_MAX_URI_CHANGES + 1;
490 r->subrequests = NGX_HTTP_MAX_SUBREQUESTS + 1;
491
Igor Sysoev59f3aa32004-06-24 16:07:04 +0000492 r->http_state = NGX_HTTP_READING_REQUEST_STATE;
493
Igor Sysoevc04deca2005-03-28 14:43:02 +0000494 ctx = c->log->data;
495 ctx->request = r;
Igor Sysoevbb28b6d2006-07-11 13:20:19 +0000496 ctx->current_request = r;
Igor Sysoev899b44e2005-05-12 14:58:06 +0000497 r->log_handler = ngx_http_log_error_handler;
Igor Sysoevc04deca2005-03-28 14:43:02 +0000498
Igor Sysoevdc3b2a72004-09-14 19:39:54 +0000499#if (NGX_STAT_STUB)
Igor Sysoev78452232005-10-12 13:50:36 +0000500 ngx_atomic_fetch_add(ngx_stat_reading, 1);
Igor Sysoevb1dfe472004-12-21 12:30:30 +0000501 r->stat_reading = 1;
Igor Sysoev78452232005-10-12 13:50:36 +0000502 ngx_atomic_fetch_add(ngx_stat_requests, 1);
Igor Sysoevdc3b2a72004-09-14 19:39:54 +0000503#endif
504
Igor Sysoev899b44e2005-05-12 14:58:06 +0000505 rev->handler(rev);
Igor Sysoevf38e0462004-07-16 17:11:43 +0000506}
507
508
509#if (NGX_HTTP_SSL)
510
Igor Sysoevc1571722005-03-19 12:38:37 +0000511static void
512ngx_http_ssl_handshake(ngx_event_t *rev)
Igor Sysoevf38e0462004-07-16 17:11:43 +0000513{
Igor Sysoevf38e0462004-07-16 17:11:43 +0000514 u_char buf[1];
Igor Sysoev1b735832004-11-11 14:07:14 +0000515 ssize_t n;
516 ngx_int_t rc;
Igor Sysoevf38e0462004-07-16 17:11:43 +0000517 ngx_connection_t *c;
518 ngx_http_request_t *r;
519
520 c = rev->data;
521 r = c->data;
522
523 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, rev->log, 0,
524 "http check ssl handshake");
525
526 if (rev->timedout) {
Igor Sysoevc04deca2005-03-28 14:43:02 +0000527 ngx_log_error(NGX_LOG_INFO, c->log, NGX_ETIMEDOUT, "client timed out");
528 c->timedout = 1;
529 ngx_http_close_request(r, NGX_HTTP_REQUEST_TIME_OUT);
Igor Sysoevf38e0462004-07-16 17:11:43 +0000530 return;
531 }
532
Igor Sysoevd3283ff2005-12-05 13:18:09 +0000533 n = recv(c->fd, (char *) buf, 1, MSG_PEEK);
Igor Sysoevf38e0462004-07-16 17:11:43 +0000534
535 if (n == -1 && ngx_socket_errno == NGX_EAGAIN) {
Igor Sysoevbd68d512008-01-08 20:51:06 +0000536
537 if (!rev->timer_set) {
538 ngx_add_timer(rev, c->listening->post_accept_timeout);
539 }
540
Igor Sysoevc9aae142008-12-09 17:27:48 +0000541 if (ngx_handle_read_event(rev, 0) != NGX_OK) {
Igor Sysoev57b088d2008-01-09 08:21:57 +0000542 ngx_http_close_request(r, NGX_HTTP_INTERNAL_SERVER_ERROR);
543 }
544
Igor Sysoevf38e0462004-07-16 17:11:43 +0000545 return;
546 }
547
548 if (n == 1) {
549 if (buf[0] == 0x80 /* SSLv2 */ || buf[0] == 0x16 /* SSLv3/TLSv1 */) {
550 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, rev->log, 0,
Igor Sysoev1b735832004-11-11 14:07:14 +0000551 "https ssl handshake: 0x%02Xd", buf[0]);
Igor Sysoevf38e0462004-07-16 17:11:43 +0000552
Igor Sysoev74a5ddb2004-07-18 19:11:20 +0000553 rc = ngx_ssl_handshake(c);
554
Igor Sysoev9fa5a822005-09-30 14:41:25 +0000555 if (rc == NGX_AGAIN) {
Igor Sysoev7b8ed422008-01-04 09:32:12 +0000556
557 if (!rev->timer_set) {
558 ngx_add_timer(rev, c->listening->post_accept_timeout);
559 }
560
Igor Sysoev9fa5a822005-09-30 14:41:25 +0000561 c->ssl->handler = ngx_http_ssl_handshake_handler;
Igor Sysoev74a5ddb2004-07-18 19:11:20 +0000562 return;
563 }
564
Igor Sysoev9fa5a822005-09-30 14:41:25 +0000565 ngx_http_ssl_handshake_handler(c);
566
567 return;
Igor Sysoevf38e0462004-07-16 17:11:43 +0000568
569 } else {
570 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, rev->log, 0,
571 "plain http");
572
573 r->plain_http = 1;
574 }
575 }
576
Igor Sysoevf100c782008-09-01 14:19:01 +0000577 c->log->action = "reading client request line";
578
Igor Sysoev899b44e2005-05-12 14:58:06 +0000579 rev->handler = ngx_http_process_request_line;
Igor Sysoev6b863e32003-05-12 15:52:24 +0000580 ngx_http_process_request_line(rev);
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000581}
582
Igor Sysoev9fa5a822005-09-30 14:41:25 +0000583
584static void
585ngx_http_ssl_handshake_handler(ngx_connection_t *c)
586{
587 ngx_http_request_t *r;
588
589 if (c->ssl->handshaked) {
590
591 /*
592 * The majority of browsers do not send the "close notify" alert.
593 * Among them are MSIE, old Mozilla, Netscape 4, Konqueror,
594 * and Links. And what is more, MSIE ignores the server's alert.
595 *
596 * Opera and recent Mozilla send the alert.
597 */
598
599 c->ssl->no_wait_shutdown = 1;
600
601 c->read->handler = ngx_http_process_request_line;
602 /* STUB: epoll edge */ c->write->handler = ngx_http_empty_handler;
603
604 ngx_http_process_request_line(c->read);
605
606 return;
607 }
608
609 r = c->data;
610
611 ngx_http_close_request(r, NGX_HTTP_BAD_REQUEST);
Igor Sysoev9fa5a822005-09-30 14:41:25 +0000612
613 return;
614}
615
Igor Sysoeve60303c2007-05-29 15:21:09 +0000616#ifdef SSL_CTRL_SET_TLSEXT_HOSTNAME
617
618int
619ngx_http_ssl_servername(ngx_ssl_conn_t *ssl_conn, int *ad, void *arg)
620{
Igor Sysoevb29426d2008-08-21 12:56:10 +0000621 size_t len;
Igor Sysoeve60303c2007-05-29 15:21:09 +0000622 const char *servername;
623 ngx_connection_t *c;
624 ngx_http_request_t *r;
625 ngx_http_ssl_srv_conf_t *sscf;
626
627 servername = SSL_get_servername(ssl_conn, TLSEXT_NAMETYPE_host_name);
628
629 if (servername == NULL) {
630 return SSL_TLSEXT_ERR_NOACK;
631 }
632
633 c = ngx_ssl_get_connection(ssl_conn);
634
635 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, c->log, 0,
636 "SSL server name: \"%s\"", servername);
637
Igor Sysoevb29426d2008-08-21 12:56:10 +0000638 len = ngx_strlen(servername);
639
640 if (len == 0) {
641 return SSL_TLSEXT_ERR_NOACK;
642 }
643
Igor Sysoeve60303c2007-05-29 15:21:09 +0000644 r = c->data;
645
Igor Sysoevb29426d2008-08-21 12:56:10 +0000646 if (ngx_http_find_virtual_server(r, (u_char *) servername, len) != NGX_OK) {
Igor Sysoeve60303c2007-05-29 15:21:09 +0000647 return SSL_TLSEXT_ERR_NOACK;
648 }
649
Igor Sysoeve60303c2007-05-29 15:21:09 +0000650 sscf = ngx_http_get_module_srv_conf(r, ngx_http_ssl_module);
651
652 SSL_set_SSL_CTX(ssl_conn, sscf->ssl.ctx);
653
654 return SSL_TLSEXT_ERR_OK;
655}
656
657#endif
Igor Sysoev9fa5a822005-09-30 14:41:25 +0000658
Igor Sysoevf38e0462004-07-16 17:11:43 +0000659#endif
660
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000661
Igor Sysoevc1571722005-03-19 12:38:37 +0000662static void
663ngx_http_process_request_line(ngx_event_t *rev)
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000664{
Igor Sysoev8decab32007-10-18 11:36:58 +0000665 ssize_t n;
666 ngx_int_t rc, rv;
667 ngx_connection_t *c;
668 ngx_http_request_t *r;
669 ngx_http_core_srv_conf_t *cscf;
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000670
Igor Sysoeva9830112003-05-19 16:39:14 +0000671 c = rev->data;
672 r = c->data;
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000673
Igor Sysoev2b58fbf2003-12-09 15:08:11 +0000674 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, rev->log, 0,
675 "http process request line");
Igor Sysoevdc479b42003-03-20 16:09:44 +0000676
677 if (rev->timedout) {
Igor Sysoevc04deca2005-03-28 14:43:02 +0000678 ngx_log_error(NGX_LOG_INFO, c->log, NGX_ETIMEDOUT, "client timed out");
679 c->timedout = 1;
680 ngx_http_close_request(r, NGX_HTTP_REQUEST_TIME_OUT);
Igor Sysoev6b863e32003-05-12 15:52:24 +0000681 return;
Igor Sysoevdc479b42003-03-20 16:09:44 +0000682 }
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000683
Igor Sysoevf7abd722004-09-23 06:32:00 +0000684 rc = NGX_AGAIN;
Igor Sysoev016b8522002-08-29 16:59:54 +0000685
Igor Sysoevf7abd722004-09-23 06:32:00 +0000686 for ( ;; ) {
Igor Sysoevdc479b42003-03-20 16:09:44 +0000687
Igor Sysoevf7abd722004-09-23 06:32:00 +0000688 if (rc == NGX_AGAIN) {
689 n = ngx_http_read_request_header(r);
Igor Sysoevdc479b42003-03-20 16:09:44 +0000690
Igor Sysoevf7abd722004-09-23 06:32:00 +0000691 if (n == NGX_AGAIN || n == NGX_ERROR) {
Igor Sysoev865c1502003-11-30 20:03:18 +0000692 return;
693 }
Igor Sysoeva9830112003-05-19 16:39:14 +0000694 }
695
Igor Sysoevf7abd722004-09-23 06:32:00 +0000696 rc = ngx_http_parse_request_line(r, r->header_in);
Igor Sysoeva9830112003-05-19 16:39:14 +0000697
Igor Sysoevf7abd722004-09-23 06:32:00 +0000698 if (rc == NGX_OK) {
Igor Sysoevdc479b42003-03-20 16:09:44 +0000699
Igor Sysoevf7abd722004-09-23 06:32:00 +0000700 /* the request line has been parsed successfully */
Igor Sysoev6b863e32003-05-12 15:52:24 +0000701
Igor Sysoev1b735832004-11-11 14:07:14 +0000702 r->request_line.len = r->request_end - r->request_start;
703 r->request_line.data = r->request_start;
Igor Sysoev42511c02008-08-04 11:10:52 +0000704 *r->request_end = '\0';
Igor Sysoevbb6ec8c2003-11-20 07:05:50 +0000705
Igor Sysoevbb6ec8c2003-11-20 07:05:50 +0000706
Igor Sysoevdc479b42003-03-20 16:09:44 +0000707 if (r->args_start) {
Igor Sysoevf7abd722004-09-23 06:32:00 +0000708 r->uri.len = r->args_start - 1 - r->uri_start;
Igor Sysoevdc479b42003-03-20 16:09:44 +0000709 } else {
Igor Sysoevf7abd722004-09-23 06:32:00 +0000710 r->uri.len = r->uri_end - r->uri_start;
Igor Sysoevdc479b42003-03-20 16:09:44 +0000711 }
712
Igor Sysoevf6906042004-11-25 16:17:31 +0000713
Igor Sysoev924bd792004-10-11 15:07:03 +0000714 if (r->complex_uri || r->quoted_uri) {
Igor Sysoev1b735832004-11-11 14:07:14 +0000715
Igor Sysoev7f6b2ff2008-06-17 15:00:30 +0000716 r->uri.data = ngx_pnalloc(r->pool, r->uri.len + 1);
Igor Sysoevc1571722005-03-19 12:38:37 +0000717 if (r->uri.data == NULL) {
Igor Sysoev1b735832004-11-11 14:07:14 +0000718 ngx_http_close_request(r, NGX_HTTP_INTERNAL_SERVER_ERROR);
Igor Sysoev1b735832004-11-11 14:07:14 +0000719 return;
720 }
721
Igor Sysoev8decab32007-10-18 11:36:58 +0000722 cscf = ngx_http_get_module_srv_conf(r, ngx_http_core_module);
723
724 rc = ngx_http_parse_complex_uri(r, cscf->merge_slashes);
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000725
Igor Sysoevc04deca2005-03-28 14:43:02 +0000726 if (rc == NGX_HTTP_PARSE_INVALID_REQUEST) {
727 ngx_log_error(NGX_LOG_INFO, c->log, 0,
728 "client sent invalid request");
729 ngx_http_finalize_request(r, NGX_HTTP_BAD_REQUEST);
Igor Sysoevf7abd722004-09-23 06:32:00 +0000730 return;
731 }
Igor Sysoevdc479b42003-03-20 16:09:44 +0000732
Igor Sysoevf7abd722004-09-23 06:32:00 +0000733 } else {
Igor Sysoev1b735832004-11-11 14:07:14 +0000734 r->uri.data = r->uri_start;
Igor Sysoevf7abd722004-09-23 06:32:00 +0000735 }
736
Igor Sysoevf6906042004-11-25 16:17:31 +0000737
Igor Sysoev1b735832004-11-11 14:07:14 +0000738 r->unparsed_uri.len = r->uri_end - r->uri_start;
739 r->unparsed_uri.data = r->uri_start;
Igor Sysoevf7abd722004-09-23 06:32:00 +0000740
Igor Sysoevf7abd722004-09-23 06:32:00 +0000741
Igor Sysoev899b44e2005-05-12 14:58:06 +0000742 r->method_name.len = r->method_end - r->request_start + 1;
743 r->method_name.data = r->request_line.data;
Igor Sysoevf7abd722004-09-23 06:32:00 +0000744
Igor Sysoev1b735832004-11-11 14:07:14 +0000745
Igor Sysoev02025fd2005-01-18 13:03:58 +0000746 if (r->http_protocol.data) {
747 r->http_protocol.len = r->request_end - r->http_protocol.data;
748 }
749
750
Igor Sysoevf7abd722004-09-23 06:32:00 +0000751 if (r->uri_ext) {
Igor Sysoevf7abd722004-09-23 06:32:00 +0000752 if (r->args_start) {
753 r->exten.len = r->args_start - 1 - r->uri_ext;
754 } else {
755 r->exten.len = r->uri_end - r->uri_ext;
756 }
757
Igor Sysoev1b735832004-11-11 14:07:14 +0000758 r->exten.data = r->uri_ext;
Igor Sysoevf7abd722004-09-23 06:32:00 +0000759 }
760
Igor Sysoev1b735832004-11-11 14:07:14 +0000761
Igor Sysoevf7abd722004-09-23 06:32:00 +0000762 if (r->args_start && r->uri_end > r->args_start) {
Igor Sysoevf7abd722004-09-23 06:32:00 +0000763 r->args.len = r->uri_end - r->args_start;
Igor Sysoev1b735832004-11-11 14:07:14 +0000764 r->args.data = r->args_start;
Igor Sysoevf7abd722004-09-23 06:32:00 +0000765 }
766
Igor Sysoevf7abd722004-09-23 06:32:00 +0000767
768 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, c->log, 0,
Igor Sysoev1b735832004-11-11 14:07:14 +0000769 "http request line: \"%V\"", &r->request_line);
Igor Sysoevf7abd722004-09-23 06:32:00 +0000770
771 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, c->log, 0,
Igor Sysoev1b735832004-11-11 14:07:14 +0000772 "http uri: \"%V\"", &r->uri);
Igor Sysoevf7abd722004-09-23 06:32:00 +0000773
774 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, c->log, 0,
Igor Sysoev1b735832004-11-11 14:07:14 +0000775 "http args: \"%V\"", &r->args);
776
777 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, c->log, 0,
778 "http exten: \"%V\"", &r->exten);
779
Igor Sysoev5fd09312008-05-15 14:44:47 +0000780 if (r->host_start && r->host_end) {
781 n = ngx_http_validate_host(r->host_start,
782 r->host_end - r->host_start);
783
784 if (n <= 0) {
785 ngx_log_error(NGX_LOG_INFO, c->log, 0,
786 "client sent invalid host in request line");
787 ngx_http_finalize_request(r, NGX_HTTP_BAD_REQUEST);
788 return;
789 }
790
791 r->headers_in.server.len = n;
792 r->headers_in.server.data = r->host_start;
793 }
794
Igor Sysoevf7abd722004-09-23 06:32:00 +0000795 if (r->http_version < NGX_HTTP_VERSION_10) {
Igor Sysoev5fd09312008-05-15 14:44:47 +0000796
797 if (ngx_http_find_virtual_server(r, r->headers_in.server.data,
798 r->headers_in.server.len)
799 == NGX_ERROR)
800 {
801 ngx_http_close_request(r, NGX_HTTP_INTERNAL_SERVER_ERROR);
802 return;
803 }
804
Igor Sysoev62864d12007-06-04 20:40:03 +0000805 ngx_http_process_request(r);
Igor Sysoev6b863e32003-05-12 15:52:24 +0000806 return;
807 }
Igor Sysoevdc479b42003-03-20 16:09:44 +0000808
Igor Sysoevdc479b42003-03-20 16:09:44 +0000809
Igor Sysoevf7abd722004-09-23 06:32:00 +0000810 if (ngx_list_init(&r->headers_in.headers, r->pool, 20,
Igor Sysoevc31a9bb2005-11-26 10:11:11 +0000811 sizeof(ngx_table_elt_t))
812 == NGX_ERROR)
Igor Sysoevf7abd722004-09-23 06:32:00 +0000813 {
814 ngx_http_close_request(r, NGX_HTTP_INTERNAL_SERVER_ERROR);
Igor Sysoevf7abd722004-09-23 06:32:00 +0000815 return;
Igor Sysoev3d58f8c2004-01-08 08:47:17 +0000816 }
Igor Sysoevf7abd722004-09-23 06:32:00 +0000817
818
Igor Sysoev8184d1b2005-03-04 14:06:57 +0000819 if (ngx_array_init(&r->headers_in.cookies, r->pool, 2,
Igor Sysoevc31a9bb2005-11-26 10:11:11 +0000820 sizeof(ngx_table_elt_t *))
821 == NGX_ERROR)
Igor Sysoevf7abd722004-09-23 06:32:00 +0000822 {
823 ngx_http_close_request(r, NGX_HTTP_INTERNAL_SERVER_ERROR);
Igor Sysoevf7abd722004-09-23 06:32:00 +0000824 return;
825 }
826
Igor Sysoeve5a222c2005-01-25 12:27:35 +0000827 c->log->action = "reading client request headers";
Igor Sysoevf7abd722004-09-23 06:32:00 +0000828
Igor Sysoev899b44e2005-05-12 14:58:06 +0000829 rev->handler = ngx_http_process_request_headers;
Igor Sysoevf7abd722004-09-23 06:32:00 +0000830 ngx_http_process_request_headers(rev);
831
832 return;
Igor Sysoevc04deca2005-03-28 14:43:02 +0000833 }
Igor Sysoevf7abd722004-09-23 06:32:00 +0000834
Igor Sysoevc04deca2005-03-28 14:43:02 +0000835 if (rc != NGX_AGAIN) {
Igor Sysoevf7abd722004-09-23 06:32:00 +0000836
837 /* there was error while a request line parsing */
838
Igor Sysoevc04deca2005-03-28 14:43:02 +0000839 ngx_log_error(NGX_LOG_INFO, c->log, 0,
840 ngx_http_client_errors[rc - NGX_HTTP_CLIENT_ERROR]);
841 ngx_http_finalize_request(r, NGX_HTTP_BAD_REQUEST);
Igor Sysoevf7abd722004-09-23 06:32:00 +0000842 return;
Igor Sysoev3d58f8c2004-01-08 08:47:17 +0000843 }
844
Igor Sysoevf7abd722004-09-23 06:32:00 +0000845 /* NGX_AGAIN: a request line parsing is still incomplete */
Igor Sysoev3d58f8c2004-01-08 08:47:17 +0000846
Igor Sysoev236e0452004-09-23 16:39:34 +0000847 if (r->header_in->pos == r->header_in->end) {
Igor Sysoeva1c8a922004-01-19 18:09:14 +0000848
Igor Sysoevf7abd722004-09-23 06:32:00 +0000849 rv = ngx_http_alloc_large_header_buffer(r, 1);
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000850
Igor Sysoevf7abd722004-09-23 06:32:00 +0000851 if (rv == NGX_ERROR) {
852 ngx_http_close_request(r, NGX_HTTP_INTERNAL_SERVER_ERROR);
Igor Sysoevf7abd722004-09-23 06:32:00 +0000853 return;
854 }
Igor Sysoev0dad6292003-03-05 17:30:51 +0000855
Igor Sysoevf7abd722004-09-23 06:32:00 +0000856 if (rv == NGX_DECLINED) {
Igor Sysoevc04deca2005-03-28 14:43:02 +0000857 r->request_line.len = r->header_in->end - r->request_start;
858 r->request_line.data = r->request_start;
859
860 ngx_log_error(NGX_LOG_INFO, c->log, 0,
861 "client sent too long URI");
862 ngx_http_finalize_request(r, NGX_HTTP_REQUEST_URI_TOO_LARGE);
Igor Sysoev6b863e32003-05-12 15:52:24 +0000863 return;
Igor Sysoev0dad6292003-03-05 17:30:51 +0000864 }
Igor Sysoev0dad6292003-03-05 17:30:51 +0000865 }
Igor Sysoev1af7c822002-09-13 14:47:42 +0000866 }
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000867}
868
Igor Sysoev1af7c822002-09-13 14:47:42 +0000869
Igor Sysoevc1571722005-03-19 12:38:37 +0000870static void
871ngx_http_process_request_headers(ngx_event_t *rev)
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000872{
Igor Sysoev02f742b2005-04-08 15:18:55 +0000873 ssize_t n;
874 ngx_int_t rc, rv;
Igor Sysoev02f742b2005-04-08 15:18:55 +0000875 ngx_str_t header;
Igor Sysoev899b44e2005-05-12 14:58:06 +0000876 ngx_table_elt_t *h;
Igor Sysoev02f742b2005-04-08 15:18:55 +0000877 ngx_connection_t *c;
878 ngx_http_header_t *hh;
879 ngx_http_request_t *r;
Igor Sysoev3362b8d2005-05-14 18:42:03 +0000880 ngx_http_core_srv_conf_t *cscf;
Igor Sysoev02f742b2005-04-08 15:18:55 +0000881 ngx_http_core_main_conf_t *cmcf;
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000882
Igor Sysoeva9830112003-05-19 16:39:14 +0000883 c = rev->data;
884 r = c->data;
Igor Sysoevd581fd52003-05-13 16:02:32 +0000885
Igor Sysoev865c1502003-11-30 20:03:18 +0000886 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, rev->log, 0,
887 "http process request header line");
Igor Sysoevd581fd52003-05-13 16:02:32 +0000888
Igor Sysoev6b863e32003-05-12 15:52:24 +0000889 if (rev->timedout) {
Igor Sysoevc04deca2005-03-28 14:43:02 +0000890 ngx_log_error(NGX_LOG_INFO, c->log, NGX_ETIMEDOUT, "client timed out");
891 c->timedout = 1;
892 ngx_http_close_request(r, NGX_HTTP_REQUEST_TIME_OUT);
Igor Sysoev6b863e32003-05-12 15:52:24 +0000893 return;
894 }
895
Igor Sysoev02f742b2005-04-08 15:18:55 +0000896 cmcf = ngx_http_get_module_main_conf(r, ngx_http_core_module);
Igor Sysoev3362b8d2005-05-14 18:42:03 +0000897 cscf = ngx_http_get_module_srv_conf(r, ngx_http_core_module);
Igor Sysoev02f742b2005-04-08 15:18:55 +0000898
Igor Sysoevd581fd52003-05-13 16:02:32 +0000899 rc = NGX_AGAIN;
Igor Sysoev6b863e32003-05-12 15:52:24 +0000900
Igor Sysoev016b8522002-08-29 16:59:54 +0000901 for ( ;; ) {
Igor Sysoevf7abd722004-09-23 06:32:00 +0000902
Igor Sysoevd581fd52003-05-13 16:02:32 +0000903 if (rc == NGX_AGAIN) {
Igor Sysoevf7abd722004-09-23 06:32:00 +0000904
Igor Sysoev236e0452004-09-23 16:39:34 +0000905 if (r->header_in->pos == r->header_in->end) {
Igor Sysoevf7abd722004-09-23 06:32:00 +0000906
907 rv = ngx_http_alloc_large_header_buffer(r, 0);
908
909 if (rv == NGX_ERROR) {
910 ngx_http_close_request(r, NGX_HTTP_INTERNAL_SERVER_ERROR);
Igor Sysoevf7abd722004-09-23 06:32:00 +0000911 return;
912 }
913
914 if (rv == NGX_DECLINED) {
Igor Sysoevc04deca2005-03-28 14:43:02 +0000915 header.len = r->header_in->end - r->header_name_start;
916 header.data = r->header_name_start;
Igor Sysoev90c08142005-07-25 09:41:38 +0000917
918 if (header.len > NGX_MAX_ERROR_STR - 300) {
919 header.len = NGX_MAX_ERROR_STR - 300;
920 header.data[header.len++] = '.';
921 header.data[header.len++] = '.';
922 header.data[header.len++] = '.';
923 }
924
Igor Sysoevc04deca2005-03-28 14:43:02 +0000925 ngx_log_error(NGX_LOG_INFO, c->log, 0,
926 "client sent too long header line: \"%V\"",
927 &header);
Igor Sysoev543a6ae2007-08-27 20:52:40 +0000928 ngx_http_finalize_request(r, NGX_HTTP_BAD_REQUEST);
Igor Sysoevf7abd722004-09-23 06:32:00 +0000929 return;
930 }
931 }
932
Igor Sysoevd581fd52003-05-13 16:02:32 +0000933 n = ngx_http_read_request_header(r);
934
935 if (n == NGX_AGAIN || n == NGX_ERROR) {
936 return;
937 }
938 }
939
Igor Sysoev753792e2008-09-24 14:02:50 +0000940 rc = ngx_http_parse_header_line(r, r->header_in,
941 cscf->underscores_in_headers);
Igor Sysoev016b8522002-08-29 16:59:54 +0000942
Igor Sysoev0dad6292003-03-05 17:30:51 +0000943 if (rc == NGX_OK) {
Igor Sysoevd581fd52003-05-13 16:02:32 +0000944
Igor Sysoev3362b8d2005-05-14 18:42:03 +0000945 if (r->invalid_header && cscf->ignore_invalid_headers) {
Igor Sysoev4d656dc2005-03-22 16:02:46 +0000946
947 /* there was error while a header line parsing */
948
949 header.len = r->header_end - r->header_name_start;
950 header.data = r->header_name_start;
951
Igor Sysoevc04deca2005-03-28 14:43:02 +0000952 ngx_log_error(NGX_LOG_INFO, c->log, 0,
953 "client sent invalid header line: \"%V\"",
Igor Sysoev4d656dc2005-03-22 16:02:46 +0000954 &header);
955 continue;
956 }
957
Igor Sysoevd581fd52003-05-13 16:02:32 +0000958 /* a header line has been parsed successfully */
959
Igor Sysoevc1571722005-03-19 12:38:37 +0000960 h = ngx_list_push(&r->headers_in.headers);
961 if (h == NULL) {
Igor Sysoevd581fd52003-05-13 16:02:32 +0000962 ngx_http_close_request(r, NGX_HTTP_INTERNAL_SERVER_ERROR);
Igor Sysoevd581fd52003-05-13 16:02:32 +0000963 return;
Igor Sysoevdc479b42003-03-20 16:09:44 +0000964 }
965
Igor Sysoev02f742b2005-04-08 15:18:55 +0000966 h->hash = r->header_hash;
967
Igor Sysoevd581fd52003-05-13 16:02:32 +0000968 h->key.len = r->header_name_end - r->header_name_start;
Igor Sysoevf7abd722004-09-23 06:32:00 +0000969 h->key.data = r->header_name_start;
970 h->key.data[h->key.len] = '\0';
971
Igor Sysoevd581fd52003-05-13 16:02:32 +0000972 h->value.len = r->header_end - r->header_start;
Igor Sysoevf7abd722004-09-23 06:32:00 +0000973 h->value.data = r->header_start;
974 h->value.data[h->value.len] = '\0';
Igor Sysoevd581fd52003-05-13 16:02:32 +0000975
Igor Sysoev7f6b2ff2008-06-17 15:00:30 +0000976 h->lowcase_key = ngx_pnalloc(r->pool, h->key.len);
Igor Sysoev3338cfd2006-05-11 14:43:47 +0000977 if (h->lowcase_key == NULL) {
978 ngx_http_close_request(r, NGX_HTTP_INTERNAL_SERVER_ERROR);
979 return;
980 }
Igor Sysoev899b44e2005-05-12 14:58:06 +0000981
Igor Sysoev3338cfd2006-05-11 14:43:47 +0000982 if (h->key.len == r->lowcase_index) {
983 ngx_memcpy(h->lowcase_key, r->lowcase_header, h->key.len);
984
985 } else {
Igor Sysoev777b0192008-08-04 10:07:00 +0000986 ngx_strlow(h->lowcase_key, h->key.data, h->key.len);
Igor Sysoevd581fd52003-05-13 16:02:32 +0000987 }
988
Igor Sysoev3338cfd2006-05-11 14:43:47 +0000989 hh = ngx_hash_find(&cmcf->headers_in_hash, h->hash,
990 h->lowcase_key, h->key.len);
991
992 if (hh && hh->handler(r, h, hh->offset) != NGX_OK) {
993 return;
994 }
995
Igor Sysoevdc867cd2003-12-14 20:10:27 +0000996 ngx_log_debug2(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
Igor Sysoev1b735832004-11-11 14:07:14 +0000997 "http header: \"%V: %V\"",
998 &h->key, &h->value);
Igor Sysoevd581fd52003-05-13 16:02:32 +0000999
Igor Sysoev6253ca12003-05-27 12:18:54 +00001000 continue;
Igor Sysoevc04deca2005-03-28 14:43:02 +00001001 }
Igor Sysoev6253ca12003-05-27 12:18:54 +00001002
Igor Sysoevc04deca2005-03-28 14:43:02 +00001003 if (rc == NGX_HTTP_PARSE_HEADER_DONE) {
Igor Sysoevd581fd52003-05-13 16:02:32 +00001004
1005 /* a whole header has been parsed successfully */
1006
Igor Sysoevdc867cd2003-12-14 20:10:27 +00001007 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
1008 "http header done");
Igor Sysoev42feecb2002-12-15 06:25:09 +00001009
Igor Sysoev6a12fc92004-12-06 14:45:08 +00001010 r->request_length += r->header_in->pos - r->header_in->start;
1011
Igor Sysoev59f3aa32004-06-24 16:07:04 +00001012 r->http_state = NGX_HTTP_PROCESS_REQUEST_STATE;
1013
Igor Sysoev419f9ac2003-10-21 16:49:56 +00001014 rc = ngx_http_process_request_header(r);
Igor Sysoeva19a85e2003-01-28 15:56:37 +00001015
Igor Sysoev419f9ac2003-10-21 16:49:56 +00001016 if (rc != NGX_OK) {
Igor Sysoev419f9ac2003-10-21 16:49:56 +00001017 return;
Igor Sysoeva09f08d2003-04-25 14:43:13 +00001018 }
1019
Igor Sysoev62864d12007-06-04 20:40:03 +00001020 ngx_http_process_request(r);
Igor Sysoeve5035392005-08-30 10:55:07 +00001021
Igor Sysoevd581fd52003-05-13 16:02:32 +00001022 return;
Igor Sysoev0dad6292003-03-05 17:30:51 +00001023 }
1024
Igor Sysoevc04deca2005-03-28 14:43:02 +00001025 if (rc == NGX_AGAIN) {
Igor Sysoev0dad6292003-03-05 17:30:51 +00001026
Igor Sysoevc04deca2005-03-28 14:43:02 +00001027 /* a header line parsing is still not complete */
1028
1029 continue;
1030 }
1031
1032 /* rc == NGX_HTTP_PARSE_INVALID_HEADER: "\r" is not followed by "\n" */
1033
1034 header.len = r->header_end - r->header_name_start;
1035 header.data = r->header_name_start;
1036 ngx_log_error(NGX_LOG_INFO, c->log, 0,
1037 "client sent invalid header line: \"%V\\r...\"",
1038 &header);
Igor Sysoev543a6ae2007-08-27 20:52:40 +00001039 ngx_http_finalize_request(r, NGX_HTTP_BAD_REQUEST);
Igor Sysoevc04deca2005-03-28 14:43:02 +00001040 return;
Igor Sysoeva0bb31f2002-12-02 16:09:40 +00001041 }
Igor Sysoev0ad17c02002-08-26 15:18:19 +00001042}
1043
Igor Sysoevb0869052002-12-10 18:05:12 +00001044
Igor Sysoevc1571722005-03-19 12:38:37 +00001045static ssize_t
1046ngx_http_read_request_header(ngx_http_request_t *r)
Igor Sysoev6b863e32003-05-12 15:52:24 +00001047{
Igor Sysoev187fcd82003-05-23 11:53:01 +00001048 ssize_t n;
1049 ngx_event_t *rev;
Igor Sysoevda173ab2006-08-30 10:39:17 +00001050 ngx_connection_t *c;
Igor Sysoev187fcd82003-05-23 11:53:01 +00001051 ngx_http_core_srv_conf_t *cscf;
Igor Sysoev6b863e32003-05-12 15:52:24 +00001052
Igor Sysoevda173ab2006-08-30 10:39:17 +00001053 c = r->connection;
1054 rev = c->read;
Igor Sysoev239baac2003-06-11 15:28:34 +00001055
Igor Sysoev6b863e32003-05-12 15:52:24 +00001056 n = r->header_in->last - r->header_in->pos;
1057
1058 if (n > 0) {
1059 return n;
1060 }
1061
Igor Sysoevb145b062005-06-15 18:33:41 +00001062 if (rev->ready) {
Igor Sysoevda173ab2006-08-30 10:39:17 +00001063 n = c->recv(c, r->header_in->last,
1064 r->header_in->end - r->header_in->last);
Igor Sysoevb145b062005-06-15 18:33:41 +00001065 } else {
1066 n = NGX_AGAIN;
Igor Sysoev222a2ad2003-11-18 16:49:00 +00001067 }
1068
Igor Sysoev6b863e32003-05-12 15:52:24 +00001069 if (n == NGX_AGAIN) {
Igor Sysoev1bc3a932008-01-08 20:55:27 +00001070 if (!rev->timer_set) {
Igor Sysoevf38e0462004-07-16 17:11:43 +00001071 cscf = ngx_http_get_module_srv_conf(r, ngx_http_core_module);
Igor Sysoev187fcd82003-05-23 11:53:01 +00001072 ngx_add_timer(rev, cscf->client_header_timeout);
Igor Sysoev6b863e32003-05-12 15:52:24 +00001073 }
1074
Igor Sysoevc9aae142008-12-09 17:27:48 +00001075 if (ngx_handle_read_event(rev, 0) != NGX_OK) {
Igor Sysoevb3e73d82003-10-10 15:10:50 +00001076 ngx_http_close_request(r, NGX_HTTP_INTERNAL_SERVER_ERROR);
Igor Sysoevb3e73d82003-10-10 15:10:50 +00001077 return NGX_ERROR;
1078 }
1079
Igor Sysoev6b863e32003-05-12 15:52:24 +00001080 return NGX_AGAIN;
1081 }
1082
1083 if (n == 0) {
Igor Sysoevda173ab2006-08-30 10:39:17 +00001084 ngx_log_error(NGX_LOG_INFO, c->log, 0,
Igor Sysoev6b863e32003-05-12 15:52:24 +00001085 "client closed prematurely connection");
Igor Sysoevd581fd52003-05-13 16:02:32 +00001086 }
Igor Sysoev6b863e32003-05-12 15:52:24 +00001087
1088 if (n == 0 || n == NGX_ERROR) {
Igor Sysoev23169362007-01-08 18:22:06 +00001089 c->error = 1;
Igor Sysoev1691aaf2007-01-08 18:23:38 +00001090 c->log->action = "reading client request headers";
Igor Sysoevda173ab2006-08-30 10:39:17 +00001091
Igor Sysoev23169362007-01-08 18:22:06 +00001092 ngx_http_finalize_request(r, NGX_HTTP_BAD_REQUEST);
Igor Sysoev6b863e32003-05-12 15:52:24 +00001093 return NGX_ERROR;
1094 }
1095
1096 r->header_in->last += n;
1097
1098 return n;
1099}
1100
1101
Igor Sysoevc1571722005-03-19 12:38:37 +00001102static ngx_int_t
1103ngx_http_alloc_large_header_buffer(ngx_http_request_t *r,
1104 ngx_uint_t request_line)
Igor Sysoev85080d02004-09-22 16:18:21 +00001105{
1106 u_char *old, *new;
Igor Sysoev85080d02004-09-22 16:18:21 +00001107 ngx_buf_t *b;
1108 ngx_http_connection_t *hc;
1109 ngx_http_core_srv_conf_t *cscf;
1110
1111 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
1112 "http alloc large header buffer");
1113
Igor Sysoevf7abd722004-09-23 06:32:00 +00001114 if (request_line && r->state == 0) {
1115
1116 /* the client fills up the buffer with "\r\n" */
1117
Igor Sysoev6a12fc92004-12-06 14:45:08 +00001118 r->request_length += r->header_in->end - r->header_in->start;
1119
Igor Sysoevf7abd722004-09-23 06:32:00 +00001120 r->header_in->pos = r->header_in->start;
1121 r->header_in->last = r->header_in->start;
1122
1123 return NGX_OK;
1124 }
1125
Igor Sysoev85080d02004-09-22 16:18:21 +00001126 old = request_line ? r->request_start : r->header_name_start;
1127
1128 cscf = ngx_http_get_module_srv_conf(r, ngx_http_core_module);
1129
Igor Sysoevf7abd722004-09-23 06:32:00 +00001130 if (r->state != 0
1131 && (size_t) (r->header_in->pos - old)
1132 >= cscf->large_client_header_buffers.size)
1133 {
Igor Sysoev85080d02004-09-22 16:18:21 +00001134 return NGX_DECLINED;
1135 }
1136
1137 hc = r->http_connection;
1138
1139 if (hc->nfree) {
1140 b = hc->free[--hc->nfree];
1141
Igor Sysoev236e0452004-09-23 16:39:34 +00001142 ngx_log_debug2(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
Igor Sysoev1b735832004-11-11 14:07:14 +00001143 "http large header free: %p %uz",
Igor Sysoev236e0452004-09-23 16:39:34 +00001144 b->pos, b->end - b->last);
1145
Igor Sysoevf7abd722004-09-23 06:32:00 +00001146 } else if (hc->nbusy < cscf->large_client_header_buffers.num) {
Igor Sysoev85080d02004-09-22 16:18:21 +00001147
1148 if (hc->busy == NULL) {
1149 hc->busy = ngx_palloc(r->connection->pool,
Igor Sysoevf7abd722004-09-23 06:32:00 +00001150 cscf->large_client_header_buffers.num * sizeof(ngx_buf_t *));
Igor Sysoev85080d02004-09-22 16:18:21 +00001151 if (hc->busy == NULL) {
1152 return NGX_ERROR;
1153 }
1154 }
1155
1156 b = ngx_create_temp_buf(r->connection->pool,
Igor Sysoevf7abd722004-09-23 06:32:00 +00001157 cscf->large_client_header_buffers.size);
Igor Sysoev85080d02004-09-22 16:18:21 +00001158 if (b == NULL) {
1159 return NGX_ERROR;
1160 }
1161
Igor Sysoev236e0452004-09-23 16:39:34 +00001162 ngx_log_debug2(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
Igor Sysoev1b735832004-11-11 14:07:14 +00001163 "http large header alloc: %p %uz",
Igor Sysoev236e0452004-09-23 16:39:34 +00001164 b->pos, b->end - b->last);
1165
Igor Sysoev85080d02004-09-22 16:18:21 +00001166 } else {
1167 return NGX_DECLINED;
1168 }
1169
1170 hc->busy[hc->nbusy++] = b;
1171
Igor Sysoevf7abd722004-09-23 06:32:00 +00001172 if (r->state == 0) {
1173 /*
1174 * r->state == 0 means that a header line was parsed successfully
1175 * and we do not need to copy incomplete header line and
1176 * to relocate the parser header pointers
1177 */
1178
Igor Sysoev6a12fc92004-12-06 14:45:08 +00001179 r->request_length += r->header_in->end - r->header_in->start;
1180
Igor Sysoevf7abd722004-09-23 06:32:00 +00001181 r->header_in = b;
1182
1183 return NGX_OK;
1184 }
1185
Igor Sysoev236e0452004-09-23 16:39:34 +00001186 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
1187 "http large header copy: %d", r->header_in->pos - old);
1188
Igor Sysoev6a12fc92004-12-06 14:45:08 +00001189 r->request_length += old - r->header_in->start;
1190
Igor Sysoev85080d02004-09-22 16:18:21 +00001191 new = b->start;
1192
Igor Sysoev236e0452004-09-23 16:39:34 +00001193 ngx_memcpy(new, old, r->header_in->pos - old);
Igor Sysoev85080d02004-09-22 16:18:21 +00001194
1195 b->pos = new + (r->header_in->pos - old);
Igor Sysoev236e0452004-09-23 16:39:34 +00001196 b->last = new + (r->header_in->pos - old);
Igor Sysoev85080d02004-09-22 16:18:21 +00001197
1198 if (request_line) {
1199 r->request_start = new;
Igor Sysoevf7abd722004-09-23 06:32:00 +00001200
1201 if (r->request_end) {
1202 r->request_end = new + (r->request_end - old);
1203 }
1204
1205 r->method_end = new + (r->method_end - old);
Igor Sysoev85080d02004-09-22 16:18:21 +00001206
1207 r->uri_start = new + (r->uri_start - old);
1208 r->uri_end = new + (r->uri_end - old);
1209
Igor Sysoevf7abd722004-09-23 06:32:00 +00001210 if (r->schema_start) {
1211 r->schema_start = new + (r->schema_start - old);
1212 r->schema_end = new + (r->schema_end - old);
1213 }
1214
1215 if (r->host_start) {
1216 r->host_start = new + (r->host_start - old);
Igor Sysoev03011fa2006-12-15 10:24:57 +00001217 if (r->host_end) {
1218 r->host_end = new + (r->host_end - old);
1219 }
Igor Sysoevf7abd722004-09-23 06:32:00 +00001220 }
1221
1222 if (r->port_start) {
1223 r->port_start = new + (r->port_start - old);
1224 r->port_end = new + (r->port_end - old);
1225 }
1226
Igor Sysoev85080d02004-09-22 16:18:21 +00001227 if (r->uri_ext) {
1228 r->uri_ext = new + (r->uri_ext - old);
1229 }
1230
1231 if (r->args_start) {
1232 r->args_start = new + (r->args_start - old);
1233 }
1234
Igor Sysoev58ea0c12007-12-01 12:22:48 +00001235 if (r->http_protocol.data) {
1236 r->http_protocol.data = new + (r->http_protocol.data - old);
1237 }
1238
Igor Sysoev85080d02004-09-22 16:18:21 +00001239 } else {
1240 r->header_name_start = new;
1241 r->header_name_end = new + (r->header_name_end - old);
1242 r->header_start = new + (r->header_start - old);
1243 r->header_end = new + (r->header_end - old);
1244 }
1245
1246 r->header_in = b;
1247
1248 return NGX_OK;
1249}
1250
Igor Sysoev85080d02004-09-22 16:18:21 +00001251
Igor Sysoevc1571722005-03-19 12:38:37 +00001252static ngx_int_t
Igor Sysoev899b44e2005-05-12 14:58:06 +00001253ngx_http_process_header_line(ngx_http_request_t *r, ngx_table_elt_t *h,
1254 ngx_uint_t offset)
1255{
1256 ngx_table_elt_t **ph;
1257
1258 ph = (ngx_table_elt_t **) ((char *) &r->headers_in + offset);
1259
1260 if (*ph == NULL) {
1261 *ph = h;
1262 }
1263
1264 return NGX_OK;
1265}
1266
1267
1268static ngx_int_t
Igor Sysoevb145b062005-06-15 18:33:41 +00001269ngx_http_process_unique_header_line(ngx_http_request_t *r, ngx_table_elt_t *h,
1270 ngx_uint_t offset)
1271{
1272 ngx_table_elt_t **ph;
1273
1274 ph = (ngx_table_elt_t **) ((char *) &r->headers_in + offset);
1275
1276 if (*ph == NULL) {
1277 *ph = h;
1278 return NGX_OK;
1279 }
1280
1281 ngx_log_error(NGX_LOG_INFO, r->connection->log, 0,
Igor Sysoevda173ab2006-08-30 10:39:17 +00001282 "client sent duplicate header line: \"%V: %V\", "
1283 "previous value: \"%V: %V\"",
1284 &h->key, &h->value, &(*ph)->key, &(*ph)->value);
Igor Sysoevb145b062005-06-15 18:33:41 +00001285
1286 ngx_http_finalize_request(r, NGX_HTTP_BAD_REQUEST);
1287
1288 return NGX_ERROR;
1289}
1290
1291
1292static ngx_int_t
Igor Sysoev5fd09312008-05-15 14:44:47 +00001293ngx_http_process_host(ngx_http_request_t *r, ngx_table_elt_t *h,
1294 ngx_uint_t offset)
1295{
1296 ssize_t len;
1297
1298 if (r->headers_in.host == NULL) {
1299 r->headers_in.host = h;
1300 }
1301
1302 len = ngx_http_validate_host(h->value.data, h->value.len);
1303
1304 if (len <= 0) {
1305 ngx_log_error(NGX_LOG_INFO, r->connection->log, 0,
1306 "client sent invalid host header");
1307 ngx_http_finalize_request(r, NGX_HTTP_BAD_REQUEST);
1308 return NGX_ERROR;
1309 }
1310
1311 if (r->headers_in.server.len) {
1312 return NGX_OK;
1313 }
1314
1315 r->headers_in.server.len = len;
1316 r->headers_in.server.data = h->value.data;
1317
1318 return NGX_OK;
1319}
1320
1321
1322static ngx_int_t
Igor Sysoevec67b192007-09-09 18:28:49 +00001323ngx_http_process_connection(ngx_http_request_t *r, ngx_table_elt_t *h,
1324 ngx_uint_t offset)
1325{
Igor Sysoevf5329412007-10-01 12:57:47 +00001326 if (ngx_strcasestrn(h->value.data, "close", 5 - 1)) {
Igor Sysoevec67b192007-09-09 18:28:49 +00001327 r->headers_in.connection_type = NGX_HTTP_CONNECTION_CLOSE;
1328
Igor Sysoevf5329412007-10-01 12:57:47 +00001329 } else if (ngx_strcasestrn(h->value.data, "keep-alive", 10 - 1)) {
Igor Sysoevec67b192007-09-09 18:28:49 +00001330 r->headers_in.connection_type = NGX_HTTP_CONNECTION_KEEP_ALIVE;
1331 }
1332
1333 return NGX_OK;
1334}
1335
1336
1337static ngx_int_t
Igor Sysoevbc5fb292008-05-13 09:18:58 +00001338ngx_http_process_user_agent(ngx_http_request_t *r, ngx_table_elt_t *h,
1339 ngx_uint_t offset)
1340{
Igor Sysoev54477e42008-08-26 15:09:28 +00001341 u_char *user_agent, *msie;
Igor Sysoevbc5fb292008-05-13 09:18:58 +00001342
1343 if (r->headers_in.user_agent) {
1344 return NGX_OK;
1345 }
1346
1347 r->headers_in.user_agent = h;
1348
1349 /* check some widespread browsers while the header is in CPU cache */
1350
1351 user_agent = h->value.data;
1352
Igor Sysoev54477e42008-08-26 15:09:28 +00001353 msie = ngx_strstrn(user_agent, "MSIE ", 5 - 1);
Igor Sysoevbc5fb292008-05-13 09:18:58 +00001354
Igor Sysoev54477e42008-08-26 15:09:28 +00001355 if (msie && msie + 7 < user_agent + h->value.len) {
Igor Sysoevbc5fb292008-05-13 09:18:58 +00001356
1357 r->headers_in.msie = 1;
1358
Igor Sysoev54477e42008-08-26 15:09:28 +00001359 if (msie[6] == '.') {
1360
1361 switch (msie[5]) {
1362 case '4':
1363 r->headers_in.msie4 = 1;
1364 /* fall through */
1365 case '5':
1366 case '6':
1367 r->headers_in.msie6 = 1;
1368 }
Igor Sysoevbc5fb292008-05-13 09:18:58 +00001369 }
1370
1371#if 0
1372 /* MSIE ignores the SSL "close notify" alert */
1373 if (c->ssl) {
1374 c->ssl->no_send_shutdown = 1;
1375 }
1376#endif
1377 }
1378
1379 if (ngx_strstrn(user_agent, "Opera", 5 - 1)) {
1380 r->headers_in.opera = 1;
1381 r->headers_in.msie = 0;
1382 r->headers_in.msie4 = 0;
Igor Sysoev54477e42008-08-26 15:09:28 +00001383 r->headers_in.msie6 = 0;
Igor Sysoevbc5fb292008-05-13 09:18:58 +00001384 }
1385
1386 if (!r->headers_in.msie && !r->headers_in.opera) {
1387
1388 if (ngx_strstrn(user_agent, "Gecko/", 6 - 1)) {
1389 r->headers_in.gecko = 1;
1390
1391 } else if (ngx_strstrn(user_agent, "Konqueror", 9 - 1)) {
1392 r->headers_in.konqueror = 1;
1393 }
1394 }
1395
1396 return NGX_OK;
1397}
1398
1399
1400static ngx_int_t
Igor Sysoev5fd09312008-05-15 14:44:47 +00001401ngx_http_process_cookie(ngx_http_request_t *r, ngx_table_elt_t *h,
1402 ngx_uint_t offset)
1403{
1404 ngx_table_elt_t **cookie;
1405
1406 cookie = ngx_array_push(&r->headers_in.cookies);
1407 if (cookie) {
1408 *cookie = h;
1409 return NGX_OK;
1410 }
1411
1412 ngx_http_close_request(r, NGX_HTTP_INTERNAL_SERVER_ERROR);
1413
1414 return NGX_ERROR;
1415}
1416
1417
1418static ngx_int_t
Igor Sysoevc1571722005-03-19 12:38:37 +00001419ngx_http_process_request_header(ngx_http_request_t *r)
Igor Sysoev419f9ac2003-10-21 16:49:56 +00001420{
Igor Sysoev5fd09312008-05-15 14:44:47 +00001421 if (ngx_http_find_virtual_server(r, r->headers_in.server.data,
1422 r->headers_in.server.len)
1423 == NGX_ERROR)
1424 {
1425 ngx_http_finalize_request(r, NGX_HTTP_INTERNAL_SERVER_ERROR);
1426 return NGX_ERROR;
1427 }
Igor Sysoev419f9ac2003-10-21 16:49:56 +00001428
Igor Sysoev5fd09312008-05-15 14:44:47 +00001429 if (r->headers_in.host == NULL && r->http_version > NGX_HTTP_VERSION_10) {
1430 ngx_log_error(NGX_LOG_INFO, r->connection->log, 0,
1431 "client sent HTTP/1.1 request without \"Host\" header");
1432 ngx_http_finalize_request(r, NGX_HTTP_BAD_REQUEST);
1433 return NGX_ERROR;
Igor Sysoev419f9ac2003-10-21 16:49:56 +00001434 }
1435
1436 if (r->headers_in.content_length) {
1437 r->headers_in.content_length_n =
Igor Sysoev1765f472006-07-07 16:33:19 +00001438 ngx_atoof(r->headers_in.content_length->value.data,
Igor Sysoev419f9ac2003-10-21 16:49:56 +00001439 r->headers_in.content_length->value.len);
1440
1441 if (r->headers_in.content_length_n == NGX_ERROR) {
Igor Sysoevc04deca2005-03-28 14:43:02 +00001442 ngx_log_error(NGX_LOG_INFO, r->connection->log, 0,
1443 "client sent invalid \"Content-Length\" header");
Igor Sysoevc31a9bb2005-11-26 10:11:11 +00001444 ngx_http_finalize_request(r, NGX_HTTP_LENGTH_REQUIRED);
Igor Sysoevc04deca2005-03-28 14:43:02 +00001445 return NGX_ERROR;
Igor Sysoev419f9ac2003-10-21 16:49:56 +00001446 }
1447 }
1448
Igor Sysoev5bf20872008-12-08 14:04:31 +00001449 if (r->method & NGX_HTTP_PUT && r->headers_in.content_length_n == -1) {
Igor Sysoevc04deca2005-03-28 14:43:02 +00001450 ngx_log_error(NGX_LOG_INFO, r->connection->log, 0,
Igor Sysoev0d68f272007-12-30 09:11:19 +00001451 "client sent %V method without \"Content-Length\" header",
1452 &r->method_name);
Igor Sysoevc31a9bb2005-11-26 10:11:11 +00001453 ngx_http_finalize_request(r, NGX_HTTP_LENGTH_REQUIRED);
1454 return NGX_ERROR;
1455 }
1456
Igor Sysoev589fc742007-09-07 09:29:08 +00001457 if (r->method & NGX_HTTP_TRACE) {
Igor Sysoev3e933d22007-03-30 18:59:26 +00001458 ngx_log_error(NGX_LOG_INFO, r->connection->log, 0,
1459 "client sent TRACE method");
1460 ngx_http_finalize_request(r, NGX_HTTP_NOT_ALLOWED);
1461 return NGX_ERROR;
1462 }
1463
Igor Sysoevc31a9bb2005-11-26 10:11:11 +00001464 if (r->headers_in.transfer_encoding
Igor Sysoev06597202007-09-26 19:26:14 +00001465 && ngx_strcasestrn(r->headers_in.transfer_encoding->value.data,
Igor Sysoevf5329412007-10-01 12:57:47 +00001466 "chunked", 7 - 1))
Igor Sysoevc31a9bb2005-11-26 10:11:11 +00001467 {
1468 ngx_log_error(NGX_LOG_INFO, r->connection->log, 0,
1469 "client sent \"Transfer-Encoding: chunked\" header");
1470 ngx_http_finalize_request(r, NGX_HTTP_LENGTH_REQUIRED);
Igor Sysoevc04deca2005-03-28 14:43:02 +00001471 return NGX_ERROR;
Igor Sysoev6881bfb2004-03-30 06:27:36 +00001472 }
1473
Igor Sysoevec67b192007-09-09 18:28:49 +00001474 if (r->headers_in.connection_type == NGX_HTTP_CONNECTION_KEEP_ALIVE) {
1475 if (r->headers_in.keep_alive) {
1476 r->headers_in.keep_alive_n =
1477 ngx_atotm(r->headers_in.keep_alive->value.data,
1478 r->headers_in.keep_alive->value.len);
Igor Sysoev419f9ac2003-10-21 16:49:56 +00001479 }
1480 }
1481
Igor Sysoev62864d12007-06-04 20:40:03 +00001482 return NGX_OK;
1483}
1484
1485
1486static void
1487ngx_http_process_request(ngx_http_request_t *r)
1488{
Igor Sysoev86ef6aa2007-12-10 12:09:51 +00001489 ngx_connection_t *c;
Igor Sysoev62864d12007-06-04 20:40:03 +00001490
1491 c = r->connection;
1492
Igor Sysoev1d4e9b32007-12-01 11:46:04 +00001493 if (r->plain_http) {
1494 ngx_log_error(NGX_LOG_INFO, c->log, 0,
1495 "client sent plain HTTP request to HTTPS port");
1496 ngx_http_finalize_request(r, NGX_HTTP_TO_HTTPS);
1497 return;
1498 }
1499
Igor Sysoevb71c6902006-08-04 16:04:04 +00001500#if (NGX_HTTP_SSL)
1501
Igor Sysoev62864d12007-06-04 20:40:03 +00001502 if (c->ssl) {
Igor Sysoev86ef6aa2007-12-10 12:09:51 +00001503 long rc;
Igor Sysoev439e2882008-04-23 18:57:25 +00001504 X509 *cert;
Igor Sysoev86ef6aa2007-12-10 12:09:51 +00001505 ngx_http_ssl_srv_conf_t *sscf;
1506
Igor Sysoevb71c6902006-08-04 16:04:04 +00001507 sscf = ngx_http_get_module_srv_conf(r, ngx_http_ssl_module);
1508
Igor Sysoev49ed6f32008-07-29 14:29:02 +00001509 if (sscf->verify == 1) {
Igor Sysoev62864d12007-06-04 20:40:03 +00001510 rc = SSL_get_verify_result(c->ssl->connection);
Igor Sysoevb71c6902006-08-04 16:04:04 +00001511
1512 if (rc != X509_V_OK) {
Igor Sysoev62864d12007-06-04 20:40:03 +00001513 ngx_log_error(NGX_LOG_INFO, c->log, 0,
Igor Sysoevc55a1042006-08-09 19:59:45 +00001514 "client SSL certificate verify error: (%l:%s)",
Igor Sysoevb71c6902006-08-04 16:04:04 +00001515 rc, X509_verify_cert_error_string(rc));
Igor Sysoev472233d2008-03-10 14:47:07 +00001516
1517 ngx_ssl_remove_cached_session(sscf->ssl.ctx,
1518 (SSL_get0_session(c->ssl->connection)));
1519
Igor Sysoevb71c6902006-08-04 16:04:04 +00001520 ngx_http_finalize_request(r, NGX_HTTPS_CERT_ERROR);
Igor Sysoev62864d12007-06-04 20:40:03 +00001521 return;
Igor Sysoevb71c6902006-08-04 16:04:04 +00001522 }
1523
Igor Sysoev439e2882008-04-23 18:57:25 +00001524 cert = SSL_get_peer_certificate(c->ssl->connection);
1525
1526 if (cert == NULL) {
Igor Sysoev62864d12007-06-04 20:40:03 +00001527 ngx_log_error(NGX_LOG_INFO, c->log, 0,
Igor Sysoevb71c6902006-08-04 16:04:04 +00001528 "client sent no required SSL certificate");
Igor Sysoev472233d2008-03-10 14:47:07 +00001529
1530 ngx_ssl_remove_cached_session(sscf->ssl.ctx,
1531 (SSL_get0_session(c->ssl->connection)));
1532
Igor Sysoevb71c6902006-08-04 16:04:04 +00001533 ngx_http_finalize_request(r, NGX_HTTPS_NO_CERT);
Igor Sysoev62864d12007-06-04 20:40:03 +00001534 return;
Igor Sysoevb71c6902006-08-04 16:04:04 +00001535 }
Igor Sysoev439e2882008-04-23 18:57:25 +00001536
1537 X509_free(cert);
Igor Sysoevb71c6902006-08-04 16:04:04 +00001538 }
1539 }
1540
1541#endif
1542
Igor Sysoev62864d12007-06-04 20:40:03 +00001543 if (c->read->timer_set) {
1544 ngx_del_timer(c->read);
1545 }
1546
1547#if (NGX_STAT_STUB)
1548 ngx_atomic_fetch_add(ngx_stat_reading, -1);
1549 r->stat_reading = 0;
1550 ngx_atomic_fetch_add(ngx_stat_writing, 1);
1551 r->stat_writing = 1;
1552#endif
1553
1554 c->read->handler = ngx_http_request_handler;
1555 c->write->handler = ngx_http_request_handler;
Igor Sysoev7ab4d9c2007-08-06 15:22:40 +00001556 r->read_event_handler = ngx_http_block_reading;
Igor Sysoev62864d12007-06-04 20:40:03 +00001557
1558 ngx_http_handler(r);
1559
Igor Sysoev851cd732008-12-08 14:23:20 +00001560 ngx_http_run_posted_requests(c);
Igor Sysoev419f9ac2003-10-21 16:49:56 +00001561}
1562
1563
Igor Sysoev5fd09312008-05-15 14:44:47 +00001564static ssize_t
1565ngx_http_validate_host(u_char *host, size_t len)
Igor Sysoevb1dfe472004-12-21 12:30:30 +00001566{
Igor Sysoev5fd09312008-05-15 14:44:47 +00001567 u_char ch;
1568 size_t i, last;
1569 ngx_uint_t dot;
1570
1571 last = len;
1572 dot = 0;
1573
1574 for (i = 0; i < len; i++) {
1575 ch = host[i];
1576
1577 if (ch == '.') {
1578 if (dot) {
1579 return -1;
1580 }
1581
1582 dot = 1;
1583 continue;
1584 }
1585
1586 dot = 0;
1587
1588 if (ch == ':') {
1589 last = i;
1590 continue;
1591 }
1592
1593 if (ch == '/' || ch == '\0') {
1594 return -1;
1595 }
1596
1597#if (NGX_WIN32)
1598 if (ch == '\\') {
1599 return -1;
1600 }
1601#endif
1602 }
1603
1604 if (dot) {
1605 last--;
1606 }
1607
1608 return last;
1609}
1610
1611
1612static ngx_int_t
1613ngx_http_find_virtual_server(ngx_http_request_t *r, u_char *host, size_t len)
1614{
Igor Sysoev6a078332008-08-04 10:18:36 +00001615 u_char *server;
1616 ngx_uint_t hash;
Igor Sysoev305a9d82005-12-26 17:07:48 +00001617 ngx_http_core_loc_conf_t *clcf;
1618 ngx_http_core_srv_conf_t *cscf;
Igor Sysoev5fd09312008-05-15 14:44:47 +00001619 u_char buf[32];
Igor Sysoevb1dfe472004-12-21 12:30:30 +00001620
Igor Sysoevb29426d2008-08-21 12:56:10 +00001621 if (r->virtual_names == NULL) {
Igor Sysoev5fd09312008-05-15 14:44:47 +00001622 return NGX_DECLINED;
1623 }
1624
1625 if (len <= 32) {
1626 server = buf;
1627
1628 } else {
Igor Sysoev7f6b2ff2008-06-17 15:00:30 +00001629 server = ngx_pnalloc(r->pool, len);
Igor Sysoev5fd09312008-05-15 14:44:47 +00001630 if (server == NULL) {
1631 return NGX_ERROR;
1632 }
1633 }
1634
Igor Sysoev6a078332008-08-04 10:18:36 +00001635 hash = ngx_hash_strlow(server, host, len);
Igor Sysoev5fd09312008-05-15 14:44:47 +00001636
1637 cscf = ngx_hash_find_combined(&r->virtual_names->names, hash, server, len);
Igor Sysoevb1dfe472004-12-21 12:30:30 +00001638
Igor Sysoev9d8a75c2007-06-11 19:49:22 +00001639 if (cscf) {
1640 goto found;
Igor Sysoevb1dfe472004-12-21 12:30:30 +00001641 }
1642
Igor Sysoev43c3bf42007-08-12 19:48:12 +00001643#if (NGX_PCRE)
1644
1645 if (r->virtual_names->nregex) {
Igor Sysoev86ef6aa2007-12-10 12:09:51 +00001646 ngx_int_t n;
1647 ngx_uint_t i;
1648 ngx_str_t name;
1649 ngx_http_server_name_t *sn;
Igor Sysoev43c3bf42007-08-12 19:48:12 +00001650
1651 name.len = len;
Igor Sysoev5fd09312008-05-15 14:44:47 +00001652 name.data = server;
Igor Sysoev43c3bf42007-08-12 19:48:12 +00001653
1654 sn = r->virtual_names->regex;
1655
1656 for (i = 0; i < r->virtual_names->nregex; i++) {
1657
1658 n = ngx_regex_exec(sn[i].regex, &name, NULL, 0);
1659
1660 if (n == NGX_REGEX_NO_MATCHED) {
1661 continue;
1662 }
1663
1664 if (n < 0) {
1665 ngx_log_error(NGX_LOG_ALERT, r->connection->log, 0,
1666 ngx_regex_exec_n
1667 " failed: %d on \"%V\" using \"%V\"",
1668 n, &name, &sn[i].name);
Igor Sysoev5fd09312008-05-15 14:44:47 +00001669 return NGX_ERROR;
Igor Sysoev43c3bf42007-08-12 19:48:12 +00001670 }
1671
1672 /* match */
1673
1674 cscf = sn[i].core_srv_conf;
1675
1676 goto found;
1677 }
1678 }
1679
1680#endif
1681
Igor Sysoev5fd09312008-05-15 14:44:47 +00001682 return NGX_OK;
Igor Sysoevb1dfe472004-12-21 12:30:30 +00001683
Igor Sysoev899b44e2005-05-12 14:58:06 +00001684found:
1685
Igor Sysoev305a9d82005-12-26 17:07:48 +00001686 r->srv_conf = cscf->ctx->srv_conf;
1687 r->loc_conf = cscf->ctx->loc_conf;
Igor Sysoev899b44e2005-05-12 14:58:06 +00001688
1689 clcf = ngx_http_get_module_loc_conf(r, ngx_http_core_module);
1690 r->connection->log->file = clcf->err_log->file;
1691
1692 if (!(r->connection->log->log_level & NGX_LOG_DEBUG_CONNECTION)) {
1693 r->connection->log->log_level = clcf->err_log->log_level;
Igor Sysoevb1dfe472004-12-21 12:30:30 +00001694 }
1695
Igor Sysoev5fd09312008-05-15 14:44:47 +00001696 return NGX_OK;
Igor Sysoevb1dfe472004-12-21 12:30:30 +00001697}
1698
1699
Igor Sysoev899b44e2005-05-12 14:58:06 +00001700static void
1701ngx_http_request_handler(ngx_event_t *ev)
Igor Sysoevd581fd52003-05-13 16:02:32 +00001702{
Igor Sysoev899b44e2005-05-12 14:58:06 +00001703 ngx_connection_t *c;
1704 ngx_http_request_t *r;
Igor Sysoevbb28b6d2006-07-11 13:20:19 +00001705 ngx_http_log_ctx_t *ctx;
Igor Sysoev899b44e2005-05-12 14:58:06 +00001706
1707 c = ev->data;
1708 r = c->data;
1709
Igor Sysoevbb28b6d2006-07-11 13:20:19 +00001710 ctx = c->log->data;
1711 ctx->current_request = r;
1712
Igor Sysoev851cd732008-12-08 14:23:20 +00001713 ngx_log_debug2(NGX_LOG_DEBUG_HTTP, c->log, 0,
1714 "http run request: \"%V?%V\"", &r->uri, &r->args);
1715
Igor Sysoev899b44e2005-05-12 14:58:06 +00001716 if (ev->write) {
1717 r->write_event_handler(r);
1718
1719 } else {
1720 r->read_event_handler(r);
1721 }
Igor Sysoev851cd732008-12-08 14:23:20 +00001722
1723 ngx_http_run_posted_requests(c);
1724}
1725
1726
1727void
1728ngx_http_run_posted_requests(ngx_connection_t *c)
1729{
1730 ngx_http_request_t *r;
1731 ngx_http_log_ctx_t *ctx;
1732 ngx_http_posted_request_t *pr;
1733
1734 for ( ;; ) {
1735
1736 if (c->destroyed) {
1737 return;
1738 }
1739
1740 r = c->data;
1741 pr = r->main->posted_requests;
1742
1743 if (pr == NULL) {
1744 return;
1745 }
1746
1747 r->main->posted_requests = pr->next;
1748
1749 r = pr->request;
1750
1751 ctx = c->log->data;
1752 ctx->current_request = r;
1753
1754 ngx_log_debug2(NGX_LOG_DEBUG_HTTP, c->log, 0,
1755 "http posted request: \"%V?%V\"", &r->uri, &r->args);
1756
1757 r->write_event_handler(r);
1758 }
1759}
1760
1761
1762ngx_int_t
1763ngx_http_post_request(ngx_http_request_t *r)
1764{
1765 ngx_http_posted_request_t *pr, **p;
1766
1767 pr = ngx_palloc(r->pool, sizeof(ngx_http_posted_request_t));
1768 if (pr == NULL) {
1769 return NGX_ERROR;
1770 }
1771
1772 pr->request = r;
1773 pr->next = NULL;
1774
1775 for (p = &r->main->posted_requests; *p; p = &(*p)->next) { /* void */ }
1776
1777 *p = pr;
1778
1779 return NGX_OK;
Igor Sysoev899b44e2005-05-12 14:58:06 +00001780}
1781
1782
1783void
1784ngx_http_finalize_request(ngx_http_request_t *r, ngx_int_t rc)
1785{
Igor Sysoevae0347a2007-08-27 15:53:00 +00001786 ngx_connection_t *c;
Igor Sysoev899b44e2005-05-12 14:58:06 +00001787 ngx_http_request_t *pr;
Igor Sysoevdc867cd2003-12-14 20:10:27 +00001788 ngx_http_core_loc_conf_t *clcf;
1789
Igor Sysoev899b44e2005-05-12 14:58:06 +00001790 if (rc == NGX_DONE) {
Igor Sysoev9ac946b2005-10-24 15:09:41 +00001791 /* the request pool may be already destroyed */
Igor Sysoev1342d9c2003-10-09 07:00:45 +00001792 return;
1793 }
1794
Igor Sysoevae0347a2007-08-27 15:53:00 +00001795 c = r->connection;
1796
Igor Sysoev851cd732008-12-08 14:23:20 +00001797 ngx_log_debug4(NGX_LOG_DEBUG_HTTP, c->log, 0,
1798 "http finalize request: %d, \"%V?%V\" %d",
1799 rc, &r->uri, &r->args, r == c->data);
Igor Sysoev899b44e2005-05-12 14:58:06 +00001800
Igor Sysoevda173ab2006-08-30 10:39:17 +00001801 if (rc == NGX_DECLINED) {
1802 r->content_handler = NULL;
1803 r->write_event_handler = ngx_http_core_run_phases;
1804 ngx_http_core_run_phases(r);
1805 return;
1806 }
1807
Igor Sysoev960100e2006-10-13 15:20:10 +00001808 if (r != r->main && r->post_subrequest) {
1809 rc = r->post_subrequest->handler(r, r->post_subrequest->data, rc);
Igor Sysoev8f125582006-07-28 15:16:17 +00001810 }
1811
Igor Sysoeve87afba2007-12-27 13:13:34 +00001812 if (rc == NGX_ERROR
1813 || rc == NGX_HTTP_REQUEST_TIME_OUT
1814 || rc == NGX_HTTP_CLIENT_CLOSED_REQUEST
1815 || c->error)
1816 {
Igor Sysoev8b57df92007-01-29 11:53:23 +00001817 if (rc > 0 && r->headers_out.status == 0) {
1818 r->headers_out.status = rc;
Igor Sysoev7f7846d2006-04-26 09:52:47 +00001819 }
1820
Igor Sysoevc2807ec2006-02-16 15:26:46 +00001821 if (ngx_http_post_action(r) == NGX_OK) {
1822 return;
1823 }
1824
Igor Sysoev9ac946b2005-10-24 15:09:41 +00001825 ngx_http_close_request(r, 0);
Igor Sysoev899b44e2005-05-12 14:58:06 +00001826 return;
1827 }
1828
Igor Sysoev7bdb7202006-04-19 15:30:56 +00001829 if (rc >= NGX_HTTP_SPECIAL_RESPONSE
1830 || rc == NGX_HTTP_CREATED
1831 || rc == NGX_HTTP_NO_CONTENT)
1832 {
Igor Sysoevcdc46302005-12-07 14:51:31 +00001833 if (rc == NGX_HTTP_CLOSE) {
1834 ngx_http_close_request(r, rc);
1835 return;
1836 }
1837
Igor Sysoevdf3254a2006-01-11 15:26:57 +00001838 if (r == r->main) {
Igor Sysoevae0347a2007-08-27 15:53:00 +00001839 if (c->read->timer_set) {
1840 ngx_del_timer(c->read);
Igor Sysoev9ac946b2005-10-24 15:09:41 +00001841 }
1842
Igor Sysoevae0347a2007-08-27 15:53:00 +00001843 if (c->write->timer_set) {
1844 ngx_del_timer(c->write);
Igor Sysoev9ac946b2005-10-24 15:09:41 +00001845 }
1846 }
1847
Igor Sysoev05a76562008-09-15 16:44:37 +00001848 c->read->handler = ngx_http_request_handler;
1849 c->write->handler = ngx_http_request_handler;
1850
Igor Sysoev9ac946b2005-10-24 15:09:41 +00001851 ngx_http_finalize_request(r, ngx_http_special_response_handler(r, rc));
Igor Sysoev9ac946b2005-10-24 15:09:41 +00001852 return;
1853 }
1854
Igor Sysoev851cd732008-12-08 14:23:20 +00001855 if (r != r->main) {
Igor Sysoeveb62fa02008-09-29 13:18:41 +00001856
Igor Sysoev851cd732008-12-08 14:23:20 +00001857 if (r->buffered || r->postponed) {
Igor Sysoeveb62fa02008-09-29 13:18:41 +00001858
Igor Sysoev851cd732008-12-08 14:23:20 +00001859 if (ngx_http_set_write_handler(r) != NGX_OK) {
1860 ngx_http_close_request(r->main, 0);
1861 }
Igor Sysoeveb62fa02008-09-29 13:18:41 +00001862
Igor Sysoev85ef94b2005-06-23 13:41:06 +00001863 return;
1864 }
Igor Sysoev899b44e2005-05-12 14:58:06 +00001865
Igor Sysoev851cd732008-12-08 14:23:20 +00001866#if (NGX_DEBUG)
1867 if (r != c->data) {
1868 ngx_log_debug2(NGX_LOG_DEBUG_HTTP, c->log, 0,
1869 "http finalize non-active request: \"%V?%V\"",
1870 &r->uri, &r->args);
1871 }
1872#endif
Igor Sysoev899b44e2005-05-12 14:58:06 +00001873
1874 pr = r->parent;
1875
Igor Sysoev851cd732008-12-08 14:23:20 +00001876 if (r == c->data) {
Igor Sysoev5192b362005-07-08 14:34:20 +00001877
Igor Sysoev851cd732008-12-08 14:23:20 +00001878 if (!r->logged) {
Igor Sysoev899b44e2005-05-12 14:58:06 +00001879
Igor Sysoev851cd732008-12-08 14:23:20 +00001880 clcf = ngx_http_get_module_loc_conf(r, ngx_http_core_module);
Igor Sysoevbb28b6d2006-07-11 13:20:19 +00001881
Igor Sysoev851cd732008-12-08 14:23:20 +00001882 if (clcf->log_subrequest) {
1883 ngx_http_log_request(r);
1884 }
Igor Sysoev899b44e2005-05-12 14:58:06 +00001885
Igor Sysoev851cd732008-12-08 14:23:20 +00001886 r->logged = 1;
Igor Sysoev5192b362005-07-08 14:34:20 +00001887
Igor Sysoev851cd732008-12-08 14:23:20 +00001888 } else {
1889 ngx_log_error(NGX_LOG_ALERT, c->log, 0,
1890 "subrequest: \"%V?%V\" logged again",
1891 &r->uri, &r->args);
1892 }
1893
1894 r->done = 1;
1895
1896 if (pr->postponed && pr->postponed->request == r) {
Igor Sysoev899b44e2005-05-12 14:58:06 +00001897 pr->postponed = pr->postponed->next;
Igor Sysoev899b44e2005-05-12 14:58:06 +00001898 }
1899
Igor Sysoev851cd732008-12-08 14:23:20 +00001900 c->data = pr;
Igor Sysoevac72bd12006-05-04 15:32:46 +00001901
Igor Sysoev851cd732008-12-08 14:23:20 +00001902 } else {
Igor Sysoevac72bd12006-05-04 15:32:46 +00001903
Igor Sysoev851cd732008-12-08 14:23:20 +00001904 r->write_event_handler = ngx_http_request_finalizer;
1905
1906 if (r->waited) {
1907 r->done = 1;
Igor Sysoevdf3254a2006-01-11 15:26:57 +00001908 }
Igor Sysoev851cd732008-12-08 14:23:20 +00001909 }
Igor Sysoevdf3254a2006-01-11 15:26:57 +00001910
Igor Sysoev851cd732008-12-08 14:23:20 +00001911 if (ngx_http_post_request(pr) != NGX_OK) {
1912 ngx_http_close_request(r->main, 0);
1913 return;
1914 }
Igor Sysoevdf3254a2006-01-11 15:26:57 +00001915
Igor Sysoev851cd732008-12-08 14:23:20 +00001916 ngx_log_debug2(NGX_LOG_DEBUG_HTTP, c->log, 0,
1917 "http wake parent request: \"%V?%V\"",
1918 &pr->uri, &pr->args);
1919
Igor Sysoev851cd732008-12-08 14:23:20 +00001920 return;
1921 }
1922
1923 if (r->buffered || c->buffered || r->postponed) {
1924
1925 if (ngx_http_set_write_handler(r) != NGX_OK) {
1926 ngx_http_close_request(r, 0);
Igor Sysoev2eec1e12008-09-27 15:08:02 +00001927 }
1928
Igor Sysoev899b44e2005-05-12 14:58:06 +00001929 return;
1930 }
Igor Sysoev10fc9ef2003-10-27 08:53:49 +00001931
Igor Sysoev851cd732008-12-08 14:23:20 +00001932 if (r != c->data) {
1933 ngx_log_error(NGX_LOG_ALERT, c->log, 0,
1934 "http finalize non-active request: \"%V?%V\"",
1935 &r->uri, &r->args);
Igor Sysoev1342d9c2003-10-09 07:00:45 +00001936 return;
Igor Sysoev5192b362005-07-08 14:34:20 +00001937 }
Igor Sysoev1342d9c2003-10-09 07:00:45 +00001938
Igor Sysoev851cd732008-12-08 14:23:20 +00001939 r->done = 1;
Igor Sysoevd581fd52003-05-13 16:02:32 +00001940
Igor Sysoev6d16e1e2006-04-05 13:40:54 +00001941 if (!r->post_action) {
1942 r->request_complete = 1;
1943 }
1944
Igor Sysoevc2807ec2006-02-16 15:26:46 +00001945 if (ngx_http_post_action(r) == NGX_OK) {
Igor Sysoev0e5dc5c2005-11-15 13:30:52 +00001946 return;
1947 }
1948
Igor Sysoevae0347a2007-08-27 15:53:00 +00001949 if (c->read->timer_set) {
1950 ngx_del_timer(c->read);
Igor Sysoevd581fd52003-05-13 16:02:32 +00001951 }
1952
Igor Sysoevae0347a2007-08-27 15:53:00 +00001953 if (c->write->timer_set) {
1954 c->write->delayed = 0;
1955 ngx_del_timer(c->write);
Igor Sysoevd581fd52003-05-13 16:02:32 +00001956 }
1957
Igor Sysoevae0347a2007-08-27 15:53:00 +00001958 if (c->destroyed) {
Igor Sysoev24025022005-12-16 15:07:08 +00001959 return;
1960 }
1961
Igor Sysoevae0347a2007-08-27 15:53:00 +00001962 if (c->read->eof) {
Igor Sysoevdb7b1262007-08-27 15:40:19 +00001963 ngx_http_close_request(r, 0);
1964 return;
1965 }
1966
Igor Sysoev851cd732008-12-08 14:23:20 +00001967 clcf = ngx_http_get_module_loc_conf(r, ngx_http_core_module);
1968
Igor Sysoevc5991982004-01-16 06:15:48 +00001969 if (!ngx_terminate
Igor Sysoev3f4685f2004-04-25 20:13:21 +00001970 && !ngx_exiting
Igor Sysoevcdd43542008-05-21 17:39:51 +00001971 && r->keepalive
Igor Sysoevc5991982004-01-16 06:15:48 +00001972 && clcf->keepalive_timeout > 0)
1973 {
Igor Sysoevd581fd52003-05-13 16:02:32 +00001974 ngx_http_set_keepalive(r);
Igor Sysoevc5991982004-01-16 06:15:48 +00001975 return;
Igor Sysoevd581fd52003-05-13 16:02:32 +00001976
Igor Sysoevdc867cd2003-12-14 20:10:27 +00001977 } else if (r->lingering_close && clcf->lingering_timeout > 0) {
Igor Sysoevd581fd52003-05-13 16:02:32 +00001978 ngx_http_set_lingering_close(r);
Igor Sysoevc5991982004-01-16 06:15:48 +00001979 return;
Igor Sysoevd581fd52003-05-13 16:02:32 +00001980 }
Igor Sysoev1342d9c2003-10-09 07:00:45 +00001981
Igor Sysoevc5991982004-01-16 06:15:48 +00001982 ngx_http_close_request(r, 0);
Igor Sysoevd581fd52003-05-13 16:02:32 +00001983}
1984
1985
Igor Sysoev85ef94b2005-06-23 13:41:06 +00001986static ngx_int_t
Igor Sysoevc1571722005-03-19 12:38:37 +00001987ngx_http_set_write_handler(ngx_http_request_t *r)
Igor Sysoevd581fd52003-05-13 16:02:32 +00001988{
Igor Sysoevd581fd52003-05-13 16:02:32 +00001989 ngx_event_t *wev;
Igor Sysoeva9830112003-05-19 16:39:14 +00001990 ngx_http_core_loc_conf_t *clcf;
Igor Sysoevd581fd52003-05-13 16:02:32 +00001991
Igor Sysoevb1af9bb2004-06-25 14:42:03 +00001992 r->http_state = NGX_HTTP_WRITING_REQUEST_STATE;
1993
Igor Sysoev7ab4d9c2007-08-06 15:22:40 +00001994 r->read_event_handler = ngx_http_test_reading;
Igor Sysoev85ef94b2005-06-23 13:41:06 +00001995 r->write_event_handler = ngx_http_writer;
1996
Igor Sysoev899b44e2005-05-12 14:58:06 +00001997 wev = r->connection->write;
1998
Igor Sysoevef066482004-06-21 15:59:32 +00001999 if (wev->ready && wev->delayed) {
Igor Sysoev85ef94b2005-06-23 13:41:06 +00002000 return NGX_OK;
Igor Sysoevd581fd52003-05-13 16:02:32 +00002001 }
2002
Igor Sysoev899b44e2005-05-12 14:58:06 +00002003 clcf = ngx_http_get_module_loc_conf(r, ngx_http_core_module);
Igor Sysoevef066482004-06-21 15:59:32 +00002004 if (!wev->delayed) {
Igor Sysoev73a73b52004-06-20 19:54:15 +00002005 ngx_add_timer(wev, clcf->send_timeout);
2006 }
Igor Sysoevd581fd52003-05-13 16:02:32 +00002007
Igor Sysoevc9aae142008-12-09 17:27:48 +00002008 if (ngx_handle_write_event(wev, clcf->send_lowat) != NGX_OK) {
Igor Sysoevb3e73d82003-10-10 15:10:50 +00002009 ngx_http_close_request(r, 0);
Igor Sysoev85ef94b2005-06-23 13:41:06 +00002010 return NGX_ERROR;
Igor Sysoevb3e73d82003-10-10 15:10:50 +00002011 }
Igor Sysoev85ef94b2005-06-23 13:41:06 +00002012
2013 return NGX_OK;
Igor Sysoevd581fd52003-05-13 16:02:32 +00002014}
2015
2016
Igor Sysoev899b44e2005-05-12 14:58:06 +00002017static void
2018ngx_http_writer(ngx_http_request_t *r)
Igor Sysoevd581fd52003-05-13 16:02:32 +00002019{
2020 int rc;
Igor Sysoev899b44e2005-05-12 14:58:06 +00002021 ngx_event_t *wev;
Igor Sysoevd581fd52003-05-13 16:02:32 +00002022 ngx_connection_t *c;
Igor Sysoev6253ca12003-05-27 12:18:54 +00002023 ngx_http_core_loc_conf_t *clcf;
Igor Sysoevd581fd52003-05-13 16:02:32 +00002024
Igor Sysoev899b44e2005-05-12 14:58:06 +00002025 c = r->connection;
2026 wev = c->write;
Igor Sysoev160d7742003-11-19 16:26:41 +00002027
Igor Sysoevd3283ff2005-12-05 13:18:09 +00002028 ngx_log_debug2(NGX_LOG_DEBUG_HTTP, wev->log, 0,
2029 "http writer handler: \"%V?%V\"", &r->uri, &r->args);
Igor Sysoevd581fd52003-05-13 16:02:32 +00002030
Igor Sysoev851cd732008-12-08 14:23:20 +00002031 clcf = ngx_http_get_module_loc_conf(r->main, ngx_http_core_module);
2032
Igor Sysoev10fc9ef2003-10-27 08:53:49 +00002033 if (wev->timedout) {
Igor Sysoevef066482004-06-21 15:59:32 +00002034 if (!wev->delayed) {
Igor Sysoevc04deca2005-03-28 14:43:02 +00002035 ngx_log_error(NGX_LOG_INFO, c->log, NGX_ETIMEDOUT,
2036 "client timed out");
2037 c->timedout = 1;
Igor Sysoev055951d2005-10-21 19:12:18 +00002038
Igor Sysoevc2807ec2006-02-16 15:26:46 +00002039 ngx_http_finalize_request(r, NGX_HTTP_REQUEST_TIME_OUT);
Igor Sysoev73a73b52004-06-20 19:54:15 +00002040 return;
2041 }
2042
2043 wev->timedout = 0;
Igor Sysoevef066482004-06-21 15:59:32 +00002044 wev->delayed = 0;
Igor Sysoev73a73b52004-06-20 19:54:15 +00002045
2046 if (!wev->ready) {
Igor Sysoev73a73b52004-06-20 19:54:15 +00002047 ngx_add_timer(wev, clcf->send_timeout);
2048
Igor Sysoevc9aae142008-12-09 17:27:48 +00002049 if (ngx_handle_write_event(wev, clcf->send_lowat) != NGX_OK) {
Igor Sysoev9ac946b2005-10-24 15:09:41 +00002050 ngx_http_close_request(r, 0);
Igor Sysoev73a73b52004-06-20 19:54:15 +00002051 }
2052
2053 return;
2054 }
2055
2056 } else {
Igor Sysoevef066482004-06-21 15:59:32 +00002057 if (wev->delayed) {
Igor Sysoev73a73b52004-06-20 19:54:15 +00002058 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, wev->log, 0,
2059 "http writer delayed");
Igor Sysoevef066482004-06-21 15:59:32 +00002060
Igor Sysoevc9aae142008-12-09 17:27:48 +00002061 if (ngx_handle_write_event(wev, clcf->send_lowat) != NGX_OK) {
Igor Sysoev9ac946b2005-10-24 15:09:41 +00002062 ngx_http_close_request(r, 0);
Igor Sysoevef066482004-06-21 15:59:32 +00002063 }
2064
Igor Sysoev73a73b52004-06-20 19:54:15 +00002065 return;
2066 }
Igor Sysoev10fc9ef2003-10-27 08:53:49 +00002067 }
2068
Igor Sysoevd3283ff2005-12-05 13:18:09 +00002069 rc = ngx_http_output_filter(r, NULL);
Igor Sysoevd581fd52003-05-13 16:02:32 +00002070
Igor Sysoevd3283ff2005-12-05 13:18:09 +00002071 if (c->destroyed) {
2072 return;
Igor Sysoev899b44e2005-05-12 14:58:06 +00002073 }
2074
Igor Sysoevd3283ff2005-12-05 13:18:09 +00002075 ngx_log_debug3(NGX_LOG_DEBUG_HTTP, c->log, 0,
2076 "http writer output filter: %d, \"%V?%V\"",
2077 rc, &r->uri, &r->args);
Igor Sysoevd581fd52003-05-13 16:02:32 +00002078
Igor Sysoev851cd732008-12-08 14:23:20 +00002079 if (r->buffered || r->postponed || (r == r->main && c->buffered)) {
2080
Igor Sysoevef066482004-06-21 15:59:32 +00002081 if (!wev->ready && !wev->delayed) {
Igor Sysoev0a280a32003-10-12 16:49:16 +00002082 ngx_add_timer(wev, clcf->send_timeout);
Igor Sysoevd581fd52003-05-13 16:02:32 +00002083 }
2084
Igor Sysoevc9aae142008-12-09 17:27:48 +00002085 if (ngx_handle_write_event(wev, clcf->send_lowat) != NGX_OK) {
Igor Sysoev9ac946b2005-10-24 15:09:41 +00002086 ngx_http_close_request(r, 0);
Igor Sysoev0a280a32003-10-12 16:49:16 +00002087 }
Igor Sysoevd581fd52003-05-13 16:02:32 +00002088
Igor Sysoev851cd732008-12-08 14:23:20 +00002089 return;
Igor Sysoevd581fd52003-05-13 16:02:32 +00002090 }
2091
Igor Sysoevd3283ff2005-12-05 13:18:09 +00002092 ngx_log_debug2(NGX_LOG_DEBUG_HTTP, wev->log, 0,
2093 "http writer done: \"%V?%V\"", &r->uri, &r->args);
Igor Sysoevd581fd52003-05-13 16:02:32 +00002094
Igor Sysoev0a280a32003-10-12 16:49:16 +00002095 ngx_http_finalize_request(r, rc);
Igor Sysoevd581fd52003-05-13 16:02:32 +00002096}
2097
2098
Igor Sysoev851cd732008-12-08 14:23:20 +00002099static void
2100ngx_http_request_finalizer(ngx_http_request_t *r)
2101{
2102 ngx_log_debug2(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
2103 "http finalizer done: \"%V?%V\"", &r->uri, &r->args);
2104
2105 ngx_http_finalize_request(r, 0);
2106}
2107
2108
Igor Sysoev7ab4d9c2007-08-06 15:22:40 +00002109void
2110ngx_http_block_reading(ngx_http_request_t *r)
Igor Sysoev899b44e2005-05-12 14:58:06 +00002111{
Igor Sysoevf53d0f32007-07-10 20:53:45 +00002112 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
Igor Sysoev7ab4d9c2007-08-06 15:22:40 +00002113 "http reading blocked");
Igor Sysoevf53d0f32007-07-10 20:53:45 +00002114
2115 /* aio does not call this handler */
2116
2117 if ((ngx_event_flags & NGX_USE_LEVEL_EVENT)
2118 && r->connection->read->active)
2119 {
Igor Sysoevc9aae142008-12-09 17:27:48 +00002120 if (ngx_del_event(r->connection->read, NGX_READ_EVENT, 0) != NGX_OK) {
Igor Sysoevf53d0f32007-07-10 20:53:45 +00002121 ngx_http_close_request(r, 0);
2122 }
2123 }
2124}
2125
2126
Igor Sysoevabe378e2008-11-10 15:22:33 +00002127void
Igor Sysoev7ab4d9c2007-08-06 15:22:40 +00002128ngx_http_test_reading(ngx_http_request_t *r)
Igor Sysoevf53d0f32007-07-10 20:53:45 +00002129{
Igor Sysoev8d6de942007-01-19 08:10:06 +00002130 int n;
2131 char buf[1];
2132 ngx_err_t err;
2133 ngx_event_t *rev;
2134 ngx_connection_t *c;
2135
2136 c = r->connection;
2137 rev = c->read;
2138
Igor Sysoev7ab4d9c2007-08-06 15:22:40 +00002139 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0, "http test reading");
Igor Sysoev8d6de942007-01-19 08:10:06 +00002140
2141#if (NGX_HAVE_KQUEUE)
2142
2143 if (ngx_event_flags & NGX_USE_KQUEUE_EVENT) {
2144
2145 if (!rev->pending_eof) {
2146 return;
2147 }
2148
2149 rev->eof = 1;
2150 c->error = 1;
2151 err = rev->kq_errno;
2152
2153 goto closed;
2154 }
2155
2156#endif
2157
2158 n = recv(c->fd, buf, 1, MSG_PEEK);
2159
2160 if (n == 0) {
2161 rev->eof = 1;
2162 c->error = 1;
2163 err = 0;
2164
2165 goto closed;
2166
2167 } else if (n == -1) {
2168 err = ngx_socket_errno;
2169
2170 if (err != NGX_EAGAIN) {
2171 rev->eof = 1;
2172 c->error = 1;
2173
2174 goto closed;
2175 }
2176 }
Igor Sysoev42feecb2002-12-15 06:25:09 +00002177
Igor Sysoev73009772003-02-06 17:21:13 +00002178 /* aio does not call this handler */
2179
Igor Sysoev8d6de942007-01-19 08:10:06 +00002180 if ((ngx_event_flags & NGX_USE_LEVEL_EVENT) && rev->active) {
2181
Igor Sysoevc9aae142008-12-09 17:27:48 +00002182 if (ngx_del_event(rev, NGX_READ_EVENT, 0) != NGX_OK) {
Igor Sysoev6ddfbf02003-05-15 15:42:53 +00002183 ngx_http_close_request(r, 0);
Igor Sysoev6ddfbf02003-05-15 15:42:53 +00002184 }
Igor Sysoev73009772003-02-06 17:21:13 +00002185 }
Igor Sysoev8d6de942007-01-19 08:10:06 +00002186
2187 return;
2188
2189closed:
2190
2191 if (err) {
2192 rev->error = 1;
2193 }
2194
2195 ngx_log_error(NGX_LOG_INFO, c->log, err,
2196 "client closed prematurely connection");
2197
Igor Sysoevd74b66c2007-02-15 18:53:48 +00002198 ngx_http_finalize_request(r, 0);
Igor Sysoev42feecb2002-12-15 06:25:09 +00002199}
2200
2201
Igor Sysoevc1571722005-03-19 12:38:37 +00002202static void
2203ngx_http_set_keepalive(ngx_http_request_t *r)
Igor Sysoev1e7ec9d2003-02-11 07:14:40 +00002204{
Igor Sysoev924bd792004-10-11 15:07:03 +00002205 int tcp_nodelay;
Igor Sysoevf7abd722004-09-23 06:32:00 +00002206 ngx_int_t i;
Igor Sysoevf7abd722004-09-23 06:32:00 +00002207 ngx_buf_t *b, *f;
Igor Sysoev187fcd82003-05-23 11:53:01 +00002208 ngx_event_t *rev, *wev;
2209 ngx_connection_t *c;
Igor Sysoevdd888c42004-09-21 05:38:28 +00002210 ngx_http_connection_t *hc;
Igor Sysoev187fcd82003-05-23 11:53:01 +00002211 ngx_http_core_srv_conf_t *cscf;
2212 ngx_http_core_loc_conf_t *clcf;
Igor Sysoev1e7ec9d2003-02-11 07:14:40 +00002213
Igor Sysoev6253ca12003-05-27 12:18:54 +00002214 c = r->connection;
Igor Sysoev3d062ad2003-03-05 06:37:42 +00002215 rev = c->read;
Igor Sysoev1e7ec9d2003-02-11 07:14:40 +00002216
Igor Sysoev4fbd8682007-08-07 10:53:27 +00002217 clcf = ngx_http_get_module_loc_conf(r, ngx_http_core_module);
2218
Igor Sysoeve04084c2004-01-26 08:52:49 +00002219 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0, "set http keepalive handler");
Igor Sysoev6a1cc902003-05-22 15:23:47 +00002220
Igor Sysoev4fbd8682007-08-07 10:53:27 +00002221 if (r->discard_body) {
Igor Sysoev32856e12009-01-31 20:44:30 +00002222 r->write_event_handler = ngx_http_request_empty_handler;
Igor Sysoev4fbd8682007-08-07 10:53:27 +00002223 r->lingering_time = ngx_time() + (time_t) (clcf->lingering_time / 1000);
2224 ngx_add_timer(rev, clcf->lingering_timeout);
2225 return;
2226 }
2227
Igor Sysoeve5a222c2005-01-25 12:27:35 +00002228 c->log->action = "closing request";
Igor Sysoevdd888c42004-09-21 05:38:28 +00002229
2230 hc = r->http_connection;
Igor Sysoevf7abd722004-09-23 06:32:00 +00002231 b = r->header_in;
2232
Igor Sysoev236e0452004-09-23 16:39:34 +00002233 if (b->pos < b->last) {
Igor Sysoevf7abd722004-09-23 06:32:00 +00002234
Igor Sysoev236e0452004-09-23 16:39:34 +00002235 /* the pipelined request */
Igor Sysoevf7abd722004-09-23 06:32:00 +00002236
2237 if (b != c->buffer) {
2238
Igor Sysoevc0edbcc2004-10-21 15:34:38 +00002239 /*
2240 * If the large header buffers were allocated while the previous
2241 * request processing then we do not use c->buffer for
2242 * the pipelined request (see ngx_http_init_request()).
Igor Sysoev0e5dc5c2005-11-15 13:30:52 +00002243 *
Igor Sysoevc0edbcc2004-10-21 15:34:38 +00002244 * Now we would move the large header buffers to the free list.
2245 */
Igor Sysoevf7abd722004-09-23 06:32:00 +00002246
2247 cscf = ngx_http_get_module_srv_conf(r, ngx_http_core_module);
2248
2249 if (hc->free == NULL) {
2250 hc->free = ngx_palloc(c->pool,
2251 cscf->large_client_header_buffers.num * sizeof(ngx_buf_t *));
Igor Sysoev236e0452004-09-23 16:39:34 +00002252
Igor Sysoevf7abd722004-09-23 06:32:00 +00002253 if (hc->free == NULL) {
Igor Sysoevb1dfe472004-12-21 12:30:30 +00002254 ngx_http_close_request(r, 0);
Igor Sysoevf7abd722004-09-23 06:32:00 +00002255 return;
2256 }
2257 }
2258
2259 for (i = 0; i < hc->nbusy - 1; i++) {
2260 f = hc->busy[i];
2261 hc->free[hc->nfree++] = f;
2262 f->pos = f->start;
2263 f->last = f->start;
2264 }
2265
2266 hc->busy[0] = b;
2267 hc->nbusy = 1;
2268 }
2269 }
2270
Igor Sysoev9ac946b2005-10-24 15:09:41 +00002271 ngx_http_request_done(r, 0);
2272
Igor Sysoevdd888c42004-09-21 05:38:28 +00002273 c->data = hc;
Igor Sysoev1e7ec9d2003-02-11 07:14:40 +00002274
Igor Sysoevfa73aac2003-05-21 13:28:21 +00002275 ngx_add_timer(rev, clcf->keepalive_timeout);
2276
Igor Sysoevc9aae142008-12-09 17:27:48 +00002277 if (ngx_handle_read_event(rev, 0) != NGX_OK) {
Igor Sysoev0a280a32003-10-12 16:49:16 +00002278 ngx_http_close_connection(c);
2279 return;
Igor Sysoevb7387572003-03-11 20:38:13 +00002280 }
2281
Igor Sysoev7b6062a2004-02-12 20:57:10 +00002282 wev = c->write;
Igor Sysoev899b44e2005-05-12 14:58:06 +00002283 wev->handler = ngx_http_empty_handler;
Igor Sysoev3d062ad2003-03-05 06:37:42 +00002284
Igor Sysoev369145c2004-05-28 15:49:23 +00002285 if (b->pos < b->last) {
Igor Sysoev3d062ad2003-03-05 06:37:42 +00002286
Igor Sysoeve04084c2004-01-26 08:52:49 +00002287 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0, "pipelined request");
Igor Sysoev6253ca12003-05-27 12:18:54 +00002288
Igor Sysoev055951d2005-10-21 19:12:18 +00002289#if (NGX_STAT_STUB)
2290 ngx_atomic_fetch_add(ngx_stat_reading, 1);
2291#endif
2292
Igor Sysoevdd888c42004-09-21 05:38:28 +00002293 hc->pipeline = 1;
Igor Sysoeve5a222c2005-01-25 12:27:35 +00002294 c->log->action = "reading client pipelined request line";
Igor Sysoev055951d2005-10-21 19:12:18 +00002295
Igor Sysoev48242f42008-05-19 10:24:22 +00002296 rev->handler = ngx_http_init_request;
2297 ngx_post_event(rev, &ngx_posted_events);
Igor Sysoevd581fd52003-05-13 16:02:32 +00002298 return;
Igor Sysoev3d062ad2003-03-05 06:37:42 +00002299 }
2300
Igor Sysoevdd888c42004-09-21 05:38:28 +00002301 hc->pipeline = 0;
Igor Sysoev3d062ad2003-03-05 06:37:42 +00002302
Igor Sysoevc0edbcc2004-10-21 15:34:38 +00002303 /*
2304 * To keep a memory footprint as small as possible for an idle
2305 * keepalive connection we try to free the ngx_http_request_t and
2306 * c->buffer's memory if they were allocated outside the c->pool.
2307 * The large header buffers are always allocated outside the c->pool and
2308 * are freed too.
2309 */
2310
Igor Sysoev236e0452004-09-23 16:39:34 +00002311 if (ngx_pfree(c->pool, r) == NGX_OK) {
2312 hc->request = NULL;
2313 }
Igor Sysoevf7abd722004-09-23 06:32:00 +00002314
Igor Sysoev236e0452004-09-23 16:39:34 +00002315 b = c->buffer;
Igor Sysoevf7abd722004-09-23 06:32:00 +00002316
Igor Sysoev236e0452004-09-23 16:39:34 +00002317 if (ngx_pfree(c->pool, b->start) == NGX_OK) {
Igor Sysoevc0edbcc2004-10-21 15:34:38 +00002318
2319 /*
2320 * the special note for ngx_http_keepalive_handler() that
2321 * c->buffer's memory was freed
2322 */
2323
Igor Sysoev236e0452004-09-23 16:39:34 +00002324 b->pos = NULL;
2325
2326 } else {
2327 b->pos = b->start;
2328 b->last = b->start;
2329 }
2330
Igor Sysoev1b735832004-11-11 14:07:14 +00002331 ngx_log_debug2(NGX_LOG_DEBUG_HTTP, c->log, 0, "hc free: %p %d",
Igor Sysoev236e0452004-09-23 16:39:34 +00002332 hc->free, hc->nfree);
2333
2334 if (hc->free) {
2335 for (i = 0; i < hc->nfree; i++) {
Igor Sysoevacb2f5d2007-12-12 15:42:39 +00002336 ngx_pfree(c->pool, hc->free[i]->start);
Igor Sysoev236e0452004-09-23 16:39:34 +00002337 hc->free[i] = NULL;
Igor Sysoevf7abd722004-09-23 06:32:00 +00002338 }
2339
Igor Sysoev236e0452004-09-23 16:39:34 +00002340 hc->nfree = 0;
2341 }
2342
Igor Sysoev1b735832004-11-11 14:07:14 +00002343 ngx_log_debug2(NGX_LOG_DEBUG_HTTP, c->log, 0, "hc busy: %p %d",
Igor Sysoev236e0452004-09-23 16:39:34 +00002344 hc->busy, hc->nbusy);
2345
2346 if (hc->busy) {
2347 for (i = 0; i < hc->nbusy; i++) {
Igor Sysoevacb2f5d2007-12-12 15:42:39 +00002348 ngx_pfree(c->pool, hc->busy[i]->start);
Igor Sysoev236e0452004-09-23 16:39:34 +00002349 hc->busy[i] = NULL;
Igor Sysoevf7abd722004-09-23 06:32:00 +00002350 }
2351
Igor Sysoev236e0452004-09-23 16:39:34 +00002352 hc->nbusy = 0;
Igor Sysoevf7abd722004-09-23 06:32:00 +00002353 }
2354
Igor Sysoevcd2aa8e2007-12-26 21:07:30 +00002355#if (NGX_HTTP_SSL)
2356 if (c->ssl) {
2357 ngx_ssl_free_buffer(c);
2358 }
2359#endif
2360
Igor Sysoev899b44e2005-05-12 14:58:06 +00002361 rev->handler = ngx_http_keepalive_handler;
Igor Sysoev3d062ad2003-03-05 06:37:42 +00002362
Igor Sysoevd3283ff2005-12-05 13:18:09 +00002363 if (wev->active && (ngx_event_flags & NGX_USE_LEVEL_EVENT)) {
Igor Sysoevc9aae142008-12-09 17:27:48 +00002364 if (ngx_del_event(wev, NGX_WRITE_EVENT, 0) != NGX_OK) {
Igor Sysoevd3283ff2005-12-05 13:18:09 +00002365 ngx_http_close_connection(c);
2366 return;
Igor Sysoev3d062ad2003-03-05 06:37:42 +00002367 }
2368 }
2369
Igor Sysoeve5a222c2005-01-25 12:27:35 +00002370 c->log->action = "keepalive";
Igor Sysoev6a644c62003-03-04 06:33:48 +00002371
Igor Sysoev67f450d2004-06-01 06:04:46 +00002372 if (c->tcp_nopush == NGX_TCP_NOPUSH_SET) {
Igor Sysoevc1571722005-03-19 12:38:37 +00002373 if (ngx_tcp_push(c->fd) == -1) {
Igor Sysoev4cec79f2004-04-28 06:14:50 +00002374 ngx_connection_error(c, ngx_socket_errno, ngx_tcp_push_n " failed");
2375 ngx_http_close_connection(c);
2376 return;
Igor Sysoev9cf78302003-06-04 17:28:33 +00002377 }
Igor Sysoev924bd792004-10-11 15:07:03 +00002378
Igor Sysoev67f450d2004-06-01 06:04:46 +00002379 c->tcp_nopush = NGX_TCP_NOPUSH_UNSET;
Igor Sysoev42b12b32004-12-02 18:40:46 +00002380 tcp_nodelay = ngx_tcp_nodelay_and_tcp_nopush ? 1 : 0;
Igor Sysoev924bd792004-10-11 15:07:03 +00002381
2382 } else {
Igor Sysoev42b12b32004-12-02 18:40:46 +00002383 tcp_nodelay = 1;
2384 }
Igor Sysoev924bd792004-10-11 15:07:03 +00002385
Igor Sysoeve5a222c2005-01-25 12:27:35 +00002386 if (tcp_nodelay
2387 && clcf->tcp_nodelay
2388 && c->tcp_nodelay == NGX_TCP_NODELAY_UNSET)
2389 {
Igor Sysoev42b12b32004-12-02 18:40:46 +00002390 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0, "tcp_nodelay");
Igor Sysoev924bd792004-10-11 15:07:03 +00002391
Igor Sysoev42b12b32004-12-02 18:40:46 +00002392 if (setsockopt(c->fd, IPPROTO_TCP, TCP_NODELAY,
Igor Sysoevcdf609a2006-04-17 19:55:41 +00002393 (const void *) &tcp_nodelay, sizeof(int))
2394 == -1)
Igor Sysoev42b12b32004-12-02 18:40:46 +00002395 {
2396 ngx_connection_error(c, ngx_socket_errno,
2397 "setsockopt(TCP_NODELAY) failed");
2398 ngx_http_close_connection(c);
2399 return;
Igor Sysoev924bd792004-10-11 15:07:03 +00002400 }
Igor Sysoev42b12b32004-12-02 18:40:46 +00002401
Igor Sysoeve5a222c2005-01-25 12:27:35 +00002402 c->tcp_nodelay = NGX_TCP_NODELAY_SET;
Igor Sysoev9cf78302003-06-04 17:28:33 +00002403 }
2404
Igor Sysoevf7abd722004-09-23 06:32:00 +00002405#if 0
Igor Sysoevc0edbcc2004-10-21 15:34:38 +00002406 /* if ngx_http_request_t was freed then we need some other place */
Igor Sysoevc83f6832004-06-24 07:53:37 +00002407 r->http_state = NGX_HTTP_KEEPALIVE_STATE;
Igor Sysoevf7abd722004-09-23 06:32:00 +00002408#endif
Igor Sysoevc83f6832004-06-24 07:53:37 +00002409
Igor Sysoev4ddeff42007-03-19 13:20:15 +00002410 c->idle = 1;
2411
Igor Sysoevb5faed22003-10-29 08:30:44 +00002412 if (rev->ready) {
Igor Sysoev48242f42008-05-19 10:24:22 +00002413 ngx_post_event(rev, &ngx_posted_events);
Igor Sysoevb7387572003-03-11 20:38:13 +00002414 }
Igor Sysoev1e7ec9d2003-02-11 07:14:40 +00002415}
2416
2417
Igor Sysoevc1571722005-03-19 12:38:37 +00002418static void
2419ngx_http_keepalive_handler(ngx_event_t *rev)
Igor Sysoev0ad17c02002-08-26 15:18:19 +00002420{
Igor Sysoevc1571722005-03-19 12:38:37 +00002421 size_t size;
2422 ssize_t n;
2423 ngx_buf_t *b;
2424 ngx_connection_t *c;
Igor Sysoev0ad17c02002-08-26 15:18:19 +00002425
Igor Sysoev0ad25372004-07-16 06:33:35 +00002426 c = rev->data;
Igor Sysoev682bf8e2002-09-16 15:01:44 +00002427
Igor Sysoeve04084c2004-01-26 08:52:49 +00002428 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0, "http keepalive handler");
Igor Sysoev1af7c822002-09-13 14:47:42 +00002429
Igor Sysoev4ddeff42007-03-19 13:20:15 +00002430 if (rev->timedout || c->close) {
Igor Sysoevd581fd52003-05-13 16:02:32 +00002431 ngx_http_close_connection(c);
2432 return;
Igor Sysoev90ace682003-03-12 17:32:22 +00002433 }
Igor Sysoev1af7c822002-09-13 14:47:42 +00002434
Igor Sysoevf6906042004-11-25 16:17:31 +00002435#if (NGX_HAVE_KQUEUE)
Igor Sysoev236e0452004-09-23 16:39:34 +00002436
Igor Sysoevc0edbcc2004-10-21 15:34:38 +00002437 if (ngx_event_flags & NGX_USE_KQUEUE_EVENT) {
Igor Sysoevcdc46302005-12-07 14:51:31 +00002438 if (rev->pending_eof) {
Igor Sysoeve5a222c2005-01-25 12:27:35 +00002439 c->log->handler = NULL;
Igor Sysoev236e0452004-09-23 16:39:34 +00002440 ngx_log_error(NGX_LOG_INFO, c->log, rev->kq_errno,
Igor Sysoev1b735832004-11-11 14:07:14 +00002441 "kevent() reported that client %V closed "
Igor Sysoeve5a222c2005-01-25 12:27:35 +00002442 "keepalive connection", &c->addr_text);
Igor Sysoev42b12b32004-12-02 18:40:46 +00002443#if (NGX_HTTP_SSL)
2444 if (c->ssl) {
Igor Sysoev9fa5a822005-09-30 14:41:25 +00002445 c->ssl->no_send_shutdown = 1;
Igor Sysoev42b12b32004-12-02 18:40:46 +00002446 }
2447#endif
Igor Sysoev236e0452004-09-23 16:39:34 +00002448 ngx_http_close_connection(c);
2449 return;
2450 }
2451 }
2452
2453#endif
2454
Igor Sysoev236e0452004-09-23 16:39:34 +00002455 b = c->buffer;
2456 size = b->end - b->start;
2457
2458 if (b->pos == NULL) {
Igor Sysoevc0edbcc2004-10-21 15:34:38 +00002459
2460 /*
2461 * The c->buffer's memory was freed by ngx_http_set_keepalive().
2462 * However, the c->buffer->start and c->buffer->end were not changed
2463 * to keep the buffer size.
2464 */
2465
Igor Sysoevc1571722005-03-19 12:38:37 +00002466 b->pos = ngx_palloc(c->pool, size);
2467 if (b->pos == NULL) {
Igor Sysoev236e0452004-09-23 16:39:34 +00002468 ngx_http_close_connection(c);
2469 return;
2470 }
2471
2472 b->start = b->pos;
2473 b->last = b->pos;
2474 b->end = b->pos + size;
2475 }
Igor Sysoevf7abd722004-09-23 06:32:00 +00002476
Igor Sysoev0a280a32003-10-12 16:49:16 +00002477 /*
2478 * MSIE closes a keepalive connection with RST flag
2479 * so we ignore ECONNRESET here.
2480 */
Igor Sysoevb7387572003-03-11 20:38:13 +00002481
Igor Sysoevc7a2f682004-02-10 16:23:38 +00002482 c->log_error = NGX_ERROR_IGNORE_ECONNRESET;
Igor Sysoevb7387572003-03-11 20:38:13 +00002483 ngx_set_socket_errno(0);
Igor Sysoev74a5ddb2004-07-18 19:11:20 +00002484
Igor Sysoev236e0452004-09-23 16:39:34 +00002485 n = c->recv(c, b->last, size);
Igor Sysoevc7a2f682004-02-10 16:23:38 +00002486 c->log_error = NGX_ERROR_INFO;
Igor Sysoev0ad17c02002-08-26 15:18:19 +00002487
Igor Sysoevd581fd52003-05-13 16:02:32 +00002488 if (n == NGX_AGAIN) {
Igor Sysoevc9aae142008-12-09 17:27:48 +00002489 if (ngx_handle_read_event(rev, 0) != NGX_OK) {
Igor Sysoev6a12fc92004-12-06 14:45:08 +00002490 ngx_http_close_connection(c);
2491 }
2492
Igor Sysoevd581fd52003-05-13 16:02:32 +00002493 return;
2494 }
2495
2496 if (n == NGX_ERROR) {
2497 ngx_http_close_connection(c);
2498 return;
Igor Sysoev90ace682003-03-12 17:32:22 +00002499 }
Igor Sysoev0ad17c02002-08-26 15:18:19 +00002500
Igor Sysoeve5a222c2005-01-25 12:27:35 +00002501 c->log->handler = NULL;
Igor Sysoev682bf8e2002-09-16 15:01:44 +00002502
2503 if (n == 0) {
Igor Sysoevdc479b42003-03-20 16:09:44 +00002504 ngx_log_error(NGX_LOG_INFO, c->log, ngx_socket_errno,
Igor Sysoeve5a222c2005-01-25 12:27:35 +00002505 "client %V closed keepalive connection", &c->addr_text);
Igor Sysoevd581fd52003-05-13 16:02:32 +00002506 ngx_http_close_connection(c);
2507 return;
Igor Sysoev682bf8e2002-09-16 15:01:44 +00002508 }
2509
Igor Sysoevf7abd722004-09-23 06:32:00 +00002510 b->last += n;
Igor Sysoeve5a222c2005-01-25 12:27:35 +00002511
Igor Sysoev055951d2005-10-21 19:12:18 +00002512#if (NGX_STAT_STUB)
2513 ngx_atomic_fetch_add(ngx_stat_reading, 1);
2514#endif
2515
Igor Sysoeve5a222c2005-01-25 12:27:35 +00002516 c->log->handler = ngx_http_log_error;
2517 c->log->action = "reading client request line";
Igor Sysoev0ad17c02002-08-26 15:18:19 +00002518
Igor Sysoev4ddeff42007-03-19 13:20:15 +00002519 c->idle = 0;
2520
Igor Sysoevd581fd52003-05-13 16:02:32 +00002521 ngx_http_init_request(rev);
Igor Sysoev0ad17c02002-08-26 15:18:19 +00002522}
2523
Igor Sysoev0ad17c02002-08-26 15:18:19 +00002524
Igor Sysoevc1571722005-03-19 12:38:37 +00002525static void
2526ngx_http_set_lingering_close(ngx_http_request_t *r)
Igor Sysoev0e5dc5c2005-11-15 13:30:52 +00002527{
Igor Sysoev0a280a32003-10-12 16:49:16 +00002528 ngx_event_t *rev, *wev;
Igor Sysoevb7387572003-03-11 20:38:13 +00002529 ngx_connection_t *c;
Igor Sysoev6253ca12003-05-27 12:18:54 +00002530 ngx_http_core_loc_conf_t *clcf;
Igor Sysoev4e9393a2003-01-09 05:36:00 +00002531
Igor Sysoev1e7ec9d2003-02-11 07:14:40 +00002532 c = r->connection;
Igor Sysoev9b25d692003-01-26 21:08:14 +00002533
Igor Sysoev6253ca12003-05-27 12:18:54 +00002534 clcf = ngx_http_get_module_loc_conf(r, ngx_http_core_module);
Igor Sysoev4e9393a2003-01-09 05:36:00 +00002535
Igor Sysoev0a280a32003-10-12 16:49:16 +00002536 rev = c->read;
Igor Sysoev899b44e2005-05-12 14:58:06 +00002537 rev->handler = ngx_http_lingering_close_handler;
Igor Sysoev42feecb2002-12-15 06:25:09 +00002538
Igor Sysoev208eed22005-10-07 13:30:52 +00002539 r->lingering_time = ngx_time() + (time_t) (clcf->lingering_time / 1000);
Igor Sysoev6253ca12003-05-27 12:18:54 +00002540 ngx_add_timer(rev, clcf->lingering_timeout);
Igor Sysoev9b25d692003-01-26 21:08:14 +00002541
Igor Sysoevc9aae142008-12-09 17:27:48 +00002542 if (ngx_handle_read_event(rev, 0) != NGX_OK) {
Igor Sysoev0a280a32003-10-12 16:49:16 +00002543 ngx_http_close_request(r, 0);
Igor Sysoev0a280a32003-10-12 16:49:16 +00002544 return;
2545 }
2546
2547 wev = c->write;
Igor Sysoev899b44e2005-05-12 14:58:06 +00002548 wev->handler = ngx_http_empty_handler;
Igor Sysoev0a280a32003-10-12 16:49:16 +00002549
Igor Sysoevd3283ff2005-12-05 13:18:09 +00002550 if (wev->active && (ngx_event_flags & NGX_USE_LEVEL_EVENT)) {
Igor Sysoevc9aae142008-12-09 17:27:48 +00002551 if (ngx_del_event(wev, NGX_WRITE_EVENT, 0) != NGX_OK) {
Igor Sysoevd3283ff2005-12-05 13:18:09 +00002552 ngx_http_close_request(r, 0);
2553 return;
Igor Sysoevb7387572003-03-11 20:38:13 +00002554 }
Igor Sysoev42feecb2002-12-15 06:25:09 +00002555 }
2556
Igor Sysoevb7387572003-03-11 20:38:13 +00002557 if (ngx_shutdown_socket(c->fd, NGX_WRITE_SHUTDOWN) == -1) {
Igor Sysoev4cec79f2004-04-28 06:14:50 +00002558 ngx_connection_error(c, ngx_socket_errno,
2559 ngx_shutdown_socket_n " failed");
Igor Sysoevd581fd52003-05-13 16:02:32 +00002560 ngx_http_close_request(r, 0);
Igor Sysoevd581fd52003-05-13 16:02:32 +00002561 return;
Igor Sysoev42feecb2002-12-15 06:25:09 +00002562 }
2563
Igor Sysoevb5faed22003-10-29 08:30:44 +00002564 if (rev->ready) {
Igor Sysoevd581fd52003-05-13 16:02:32 +00002565 ngx_http_lingering_close_handler(rev);
Igor Sysoevb7387572003-03-11 20:38:13 +00002566 }
Igor Sysoev42feecb2002-12-15 06:25:09 +00002567}
2568
2569
Igor Sysoevc1571722005-03-19 12:38:37 +00002570static void
2571ngx_http_lingering_close_handler(ngx_event_t *rev)
Igor Sysoev42feecb2002-12-15 06:25:09 +00002572{
Igor Sysoevdc479b42003-03-20 16:09:44 +00002573 ssize_t n;
2574 ngx_msec_t timer;
2575 ngx_connection_t *c;
2576 ngx_http_request_t *r;
Igor Sysoev6253ca12003-05-27 12:18:54 +00002577 ngx_http_core_loc_conf_t *clcf;
Igor Sysoevf7abd722004-09-23 06:32:00 +00002578 u_char buffer[NGX_HTTP_LINGERING_BUFFER_SIZE];
Igor Sysoev1af7c822002-09-13 14:47:42 +00002579
Igor Sysoev6253ca12003-05-27 12:18:54 +00002580 c = rev->data;
2581 r = c->data;
Igor Sysoev86de4cb2003-01-30 07:28:09 +00002582
Igor Sysoeve04084c2004-01-26 08:52:49 +00002583 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0,
2584 "http lingering close handler");
Igor Sysoevdc479b42003-03-20 16:09:44 +00002585
2586 if (rev->timedout) {
Igor Sysoev0e5dc5c2005-11-15 13:30:52 +00002587 c->timedout = 1;
Igor Sysoevd581fd52003-05-13 16:02:32 +00002588 ngx_http_close_request(r, 0);
Igor Sysoevd581fd52003-05-13 16:02:32 +00002589 return;
Igor Sysoev0d2bda52002-12-24 07:09:57 +00002590 }
Igor Sysoev1af7c822002-09-13 14:47:42 +00002591
Igor Sysoev1d04b142007-11-15 14:26:36 +00002592 timer = (ngx_msec_t) (r->lingering_time - ngx_time());
Igor Sysoev0d2bda52002-12-24 07:09:57 +00002593 if (timer <= 0) {
Igor Sysoevd581fd52003-05-13 16:02:32 +00002594 ngx_http_close_request(r, 0);
Igor Sysoevd581fd52003-05-13 16:02:32 +00002595 return;
Igor Sysoev0d2bda52002-12-24 07:09:57 +00002596 }
Igor Sysoev1af7c822002-09-13 14:47:42 +00002597
Igor Sysoevdc479b42003-03-20 16:09:44 +00002598 do {
Igor Sysoevf7abd722004-09-23 06:32:00 +00002599 n = c->recv(c, buffer, NGX_HTTP_LINGERING_BUFFER_SIZE);
Igor Sysoev1af7c822002-09-13 14:47:42 +00002600
Igor Sysoeve04084c2004-01-26 08:52:49 +00002601 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, c->log, 0, "lingering read: %d", n);
Igor Sysoev1af7c822002-09-13 14:47:42 +00002602
Igor Sysoevdc479b42003-03-20 16:09:44 +00002603 if (n == NGX_ERROR || n == 0) {
Igor Sysoevd581fd52003-05-13 16:02:32 +00002604 ngx_http_close_request(r, 0);
Igor Sysoevd581fd52003-05-13 16:02:32 +00002605 return;
Igor Sysoevdc479b42003-03-20 16:09:44 +00002606 }
2607
2608 } while (rev->ready);
Igor Sysoev1af7c822002-09-13 14:47:42 +00002609
Igor Sysoevc9aae142008-12-09 17:27:48 +00002610 if (ngx_handle_read_event(rev, 0) != NGX_OK) {
Igor Sysoevb1dfe472004-12-21 12:30:30 +00002611 ngx_http_close_request(r, 0);
Igor Sysoev6a12fc92004-12-06 14:45:08 +00002612 return;
2613 }
2614
Igor Sysoevf7abd722004-09-23 06:32:00 +00002615 clcf = ngx_http_get_module_loc_conf(r, ngx_http_core_module);
2616
Igor Sysoev682bf8e2002-09-16 15:01:44 +00002617 timer *= 1000;
Igor Sysoevf7abd722004-09-23 06:32:00 +00002618
Igor Sysoev6253ca12003-05-27 12:18:54 +00002619 if (timer > clcf->lingering_timeout) {
2620 timer = clcf->lingering_timeout;
Igor Sysoev0d2bda52002-12-24 07:09:57 +00002621 }
Igor Sysoev1af7c822002-09-13 14:47:42 +00002622
Igor Sysoevdc479b42003-03-20 16:09:44 +00002623 ngx_add_timer(rev, timer);
Igor Sysoev1af7c822002-09-13 14:47:42 +00002624}
2625
Igor Sysoev2b542382002-08-20 14:48:28 +00002626
Igor Sysoevc1571722005-03-19 12:38:37 +00002627void
2628ngx_http_empty_handler(ngx_event_t *wev)
Igor Sysoev6a1cc902003-05-22 15:23:47 +00002629{
Igor Sysoeve04084c2004-01-26 08:52:49 +00002630 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, wev->log, 0, "http empty handler");
Igor Sysoev6a1cc902003-05-22 15:23:47 +00002631
2632 return;
2633}
2634
2635
Igor Sysoev899b44e2005-05-12 14:58:06 +00002636void
2637ngx_http_request_empty_handler(ngx_http_request_t *r)
2638{
2639 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
Igor Sysoevd3283ff2005-12-05 13:18:09 +00002640 "http request empty handler");
Igor Sysoev899b44e2005-05-12 14:58:06 +00002641
2642 return;
2643}
2644
2645
Igor Sysoevc1571722005-03-19 12:38:37 +00002646ngx_int_t
Igor Sysoevc31a9bb2005-11-26 10:11:11 +00002647ngx_http_send_special(ngx_http_request_t *r, ngx_uint_t flags)
Igor Sysoev1342d9c2003-10-09 07:00:45 +00002648{
Igor Sysoev369145c2004-05-28 15:49:23 +00002649 ngx_buf_t *b;
Igor Sysoev419f9ac2003-10-21 16:49:56 +00002650 ngx_chain_t out;
Igor Sysoev1342d9c2003-10-09 07:00:45 +00002651
Igor Sysoevc1571722005-03-19 12:38:37 +00002652 b = ngx_calloc_buf(r->pool);
2653 if (b == NULL) {
Igor Sysoev369145c2004-05-28 15:49:23 +00002654 return NGX_ERROR;
2655 }
2656
Igor Sysoevc31a9bb2005-11-26 10:11:11 +00002657 if (flags & NGX_HTTP_LAST) {
2658 b->last_buf = 1;
2659 }
2660
2661 if (flags & NGX_HTTP_FLUSH) {
2662 b->flush = 1;
2663 }
2664
Igor Sysoev369145c2004-05-28 15:49:23 +00002665 out.buf = b;
Igor Sysoev419f9ac2003-10-21 16:49:56 +00002666 out.next = NULL;
Igor Sysoev1342d9c2003-10-09 07:00:45 +00002667
Igor Sysoev419f9ac2003-10-21 16:49:56 +00002668 return ngx_http_output_filter(r, &out);
Igor Sysoev1342d9c2003-10-09 07:00:45 +00002669}
2670
2671
Igor Sysoevc2807ec2006-02-16 15:26:46 +00002672static ngx_int_t
2673ngx_http_post_action(ngx_http_request_t *r)
2674{
2675 ngx_http_core_loc_conf_t *clcf;
2676
2677 clcf = ngx_http_get_module_loc_conf(r, ngx_http_core_module);
2678
2679 if (clcf->post_action.data == NULL) {
2680 return NGX_DECLINED;
2681 }
2682
Igor Sysoev8fea8852006-03-15 09:53:04 +00002683 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
2684 "post action: \"%V\"", &clcf->post_action);
2685
Igor Sysoevc2807ec2006-02-16 15:26:46 +00002686 r->http_version = NGX_HTTP_VERSION_9;
2687 r->header_only = 1;
Igor Sysoev6d16e1e2006-04-05 13:40:54 +00002688 r->post_action = 1;
Igor Sysoevc2807ec2006-02-16 15:26:46 +00002689
Igor Sysoev7ab4d9c2007-08-06 15:22:40 +00002690 r->read_event_handler = ngx_http_block_reading;
Igor Sysoevcf100a72006-12-11 08:31:54 +00002691
Igor Sysoevce307652008-02-15 12:46:40 +00002692 if (clcf->post_action.data[0] == '/') {
2693 ngx_http_internal_redirect(r, &clcf->post_action, NULL);
2694
2695 } else {
2696 ngx_http_named_location(r, &clcf->post_action);
2697 }
Igor Sysoevc2807ec2006-02-16 15:26:46 +00002698
2699 return NGX_OK;
2700}
2701
2702
2703static void
Igor Sysoev899b44e2005-05-12 14:58:06 +00002704ngx_http_close_request(ngx_http_request_t *r, ngx_int_t error)
Igor Sysoevd581fd52003-05-13 16:02:32 +00002705{
Igor Sysoev4ecb4d72006-04-21 12:06:44 +00002706 ngx_connection_t *c;
Igor Sysoev9ac946b2005-10-24 15:09:41 +00002707
2708 c = r->connection;
Igor Sysoev9ac946b2005-10-24 15:09:41 +00002709
Igor Sysoev4ecb4d72006-04-21 12:06:44 +00002710 ngx_http_request_done(r->main, error);
Igor Sysoev9ac946b2005-10-24 15:09:41 +00002711 ngx_http_close_connection(c);
2712}
2713
2714
Igor Sysoevc2807ec2006-02-16 15:26:46 +00002715static void
Igor Sysoev9ac946b2005-10-24 15:09:41 +00002716ngx_http_request_done(ngx_http_request_t *r, ngx_int_t error)
2717{
Igor Sysoev2eec1e12008-09-27 15:08:02 +00002718 ngx_log_t *log;
2719 struct linger linger;
2720 ngx_http_cleanup_t *cln;
2721 ngx_http_log_ctx_t *ctx;
2722 ngx_http_core_loc_conf_t *clcf;
Igor Sysoevd581fd52003-05-13 16:02:32 +00002723
Igor Sysoev865c1502003-11-30 20:03:18 +00002724 log = r->connection->log;
2725
2726 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, log, 0, "http close request");
Igor Sysoevd581fd52003-05-13 16:02:32 +00002727
2728 if (r->pool == NULL) {
Igor Sysoev90c08142005-07-25 09:41:38 +00002729 ngx_log_error(NGX_LOG_ALERT, log, 0, "http request already closed");
Igor Sysoevd581fd52003-05-13 16:02:32 +00002730 return;
2731 }
2732
Igor Sysoev4ecb4d72006-04-21 12:06:44 +00002733 for (cln = r->cleanup; cln; cln = cln->next) {
2734 if (cln->handler) {
2735 cln->handler(cln->data);
2736 }
2737 }
2738
Igor Sysoev85cccfb2004-09-15 16:00:43 +00002739#if (NGX_STAT_STUB)
Igor Sysoev055951d2005-10-21 19:12:18 +00002740
Igor Sysoev85cccfb2004-09-15 16:00:43 +00002741 if (r->stat_reading) {
Igor Sysoev78452232005-10-12 13:50:36 +00002742 ngx_atomic_fetch_add(ngx_stat_reading, -1);
Igor Sysoev85cccfb2004-09-15 16:00:43 +00002743 }
2744
2745 if (r->stat_writing) {
Igor Sysoev78452232005-10-12 13:50:36 +00002746 ngx_atomic_fetch_add(ngx_stat_writing, -1);
Igor Sysoev85cccfb2004-09-15 16:00:43 +00002747 }
Igor Sysoev055951d2005-10-21 19:12:18 +00002748
Igor Sysoev85cccfb2004-09-15 16:00:43 +00002749#endif
2750
Igor Sysoev669e3312003-12-22 09:40:48 +00002751 if (error && r->headers_out.status == 0) {
Igor Sysoevd581fd52003-05-13 16:02:32 +00002752 r->headers_out.status = error;
2753 }
2754
Igor Sysoev504efff2008-06-30 15:32:57 +00002755 log->action = "logging request";
2756
Igor Sysoev2eec1e12008-09-27 15:08:02 +00002757 ngx_http_log_request(r);
Igor Sysoevd581fd52003-05-13 16:02:32 +00002758
Igor Sysoev504efff2008-06-30 15:32:57 +00002759 log->action = "closing request";
2760
Igor Sysoev0ab91b92004-06-06 19:49:18 +00002761 if (r->connection->timedout) {
2762 clcf = ngx_http_get_module_loc_conf(r, ngx_http_core_module);
2763
2764 if (clcf->reset_timedout_connection) {
Igor Sysoev899b44e2005-05-12 14:58:06 +00002765 linger.l_onoff = 1;
2766 linger.l_linger = 0;
Igor Sysoev0ab91b92004-06-06 19:49:18 +00002767
2768 if (setsockopt(r->connection->fd, SOL_SOCKET, SO_LINGER,
Igor Sysoev899b44e2005-05-12 14:58:06 +00002769 (const void *) &linger, sizeof(struct linger)) == -1)
Igor Sysoev0ab91b92004-06-06 19:49:18 +00002770 {
2771 ngx_log_error(NGX_LOG_ALERT, log, ngx_socket_errno,
2772 "setsockopt(SO_LINGER) failed");
2773 }
2774 }
2775 }
2776
Igor Sysoev899b44e2005-05-12 14:58:06 +00002777 /* the various request strings were allocated from r->pool */
Igor Sysoev865c1502003-11-30 20:03:18 +00002778 ctx = log->data;
Igor Sysoev1b735832004-11-11 14:07:14 +00002779 ctx->request = NULL;
Igor Sysoevd581fd52003-05-13 16:02:32 +00002780
Igor Sysoevc83f6832004-06-24 07:53:37 +00002781 r->request_line.len = 0;
2782
Igor Sysoevd3283ff2005-12-05 13:18:09 +00002783 r->connection->destroyed = 1;
2784
Igor Sysoevd581fd52003-05-13 16:02:32 +00002785 ngx_destroy_pool(r->pool);
2786}
2787
2788
Igor Sysoev90c08142005-07-25 09:41:38 +00002789static void
Igor Sysoev2eec1e12008-09-27 15:08:02 +00002790ngx_http_log_request(ngx_http_request_t *r)
2791{
2792 ngx_uint_t i, n;
2793 ngx_http_handler_pt *log_handler;
2794 ngx_http_core_main_conf_t *cmcf;
2795
2796 cmcf = ngx_http_get_module_main_conf(r, ngx_http_core_module);
2797
2798 log_handler = cmcf->phases[NGX_HTTP_LOG_PHASE].handlers.elts;
2799 n = cmcf->phases[NGX_HTTP_LOG_PHASE].handlers.nelts;
2800
2801 for (i = 0; i < n; i++) {
2802 log_handler[i](r);
2803 }
2804}
2805
2806
2807static void
Igor Sysoevc1571722005-03-19 12:38:37 +00002808ngx_http_close_connection(ngx_connection_t *c)
Igor Sysoev6b863e32003-05-12 15:52:24 +00002809{
Igor Sysoev02025fd2005-01-18 13:03:58 +00002810 ngx_pool_t *pool;
2811
Igor Sysoeve04084c2004-01-26 08:52:49 +00002812 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, c->log, 0,
Igor Sysoev54498db2004-02-11 17:08:49 +00002813 "close http connection: %d", c->fd);
Igor Sysoev6b863e32003-05-12 15:52:24 +00002814
Igor Sysoevceb99292005-09-06 16:09:32 +00002815#if (NGX_HTTP_SSL)
Igor Sysoev924bd792004-10-11 15:07:03 +00002816
2817 if (c->ssl) {
2818 if (ngx_ssl_shutdown(c) == NGX_AGAIN) {
Igor Sysoev0e5dc5c2005-11-15 13:30:52 +00002819 c->ssl->handler = ngx_http_close_connection;
Igor Sysoev924bd792004-10-11 15:07:03 +00002820 return;
2821 }
2822 }
2823
2824#endif
2825
Igor Sysoevdc3b2a72004-09-14 19:39:54 +00002826#if (NGX_STAT_STUB)
Igor Sysoev78452232005-10-12 13:50:36 +00002827 ngx_atomic_fetch_add(ngx_stat_active, -1);
Igor Sysoevdc3b2a72004-09-14 19:39:54 +00002828#endif
2829
Igor Sysoevd3283ff2005-12-05 13:18:09 +00002830 c->destroyed = 1;
2831
Igor Sysoev02025fd2005-01-18 13:03:58 +00002832 pool = c->pool;
2833
Igor Sysoev59cf56c2004-09-07 15:29:22 +00002834 ngx_close_connection(c);
Igor Sysoev02025fd2005-01-18 13:03:58 +00002835
Igor Sysoevc1571722005-03-19 12:38:37 +00002836 ngx_destroy_pool(pool);
Igor Sysoev2ba1ee02002-10-04 17:58:04 +00002837}
2838
2839
Igor Sysoevc1571722005-03-19 12:38:37 +00002840static u_char *
2841ngx_http_log_error(ngx_log_t *log, u_char *buf, size_t len)
Igor Sysoev0ad17c02002-08-26 15:18:19 +00002842{
Igor Sysoeve5a222c2005-01-25 12:27:35 +00002843 u_char *p;
Igor Sysoevc04deca2005-03-28 14:43:02 +00002844 ngx_http_request_t *r;
Igor Sysoeve5a222c2005-01-25 12:27:35 +00002845 ngx_http_log_ctx_t *ctx;
Igor Sysoev7578ec92003-06-02 15:24:30 +00002846
Igor Sysoeve5a222c2005-01-25 12:27:35 +00002847 if (log->action) {
Igor Sysoev899b44e2005-05-12 14:58:06 +00002848 p = ngx_snprintf(buf, len, " while %s", log->action);
Igor Sysoev1b735832004-11-11 14:07:14 +00002849 len -= p - buf;
Igor Sysoevc04deca2005-03-28 14:43:02 +00002850 buf = p;
Igor Sysoevdc479b42003-03-20 16:09:44 +00002851 }
Igor Sysoev1b735832004-11-11 14:07:14 +00002852
Igor Sysoev899b44e2005-05-12 14:58:06 +00002853 ctx = log->data;
2854
Igor Sysoevcc595672007-12-30 08:01:50 +00002855 p = ngx_snprintf(buf, len, ", client: %V", &ctx->connection->addr_text);
Igor Sysoev899b44e2005-05-12 14:58:06 +00002856 len -= p - buf;
Igor Sysoev1b735832004-11-11 14:07:14 +00002857
Igor Sysoevc04deca2005-03-28 14:43:02 +00002858 r = ctx->request;
2859
Igor Sysoev899b44e2005-05-12 14:58:06 +00002860 if (r) {
Igor Sysoevbb28b6d2006-07-11 13:20:19 +00002861 return r->log_handler(r, ctx->current_request, p, len);
Igor Sysoevcc595672007-12-30 08:01:50 +00002862
2863 } else {
2864 p = ngx_snprintf(p, len, ", server: %V",
2865 &ctx->connection->listening->addr_text);
Igor Sysoev1b735832004-11-11 14:07:14 +00002866 }
2867
Igor Sysoev899b44e2005-05-12 14:58:06 +00002868 return p;
2869}
2870
2871
2872static u_char *
Igor Sysoevbb28b6d2006-07-11 13:20:19 +00002873ngx_http_log_error_handler(ngx_http_request_t *r, ngx_http_request_t *sr,
2874 u_char *buf, size_t len)
Igor Sysoev899b44e2005-05-12 14:58:06 +00002875{
Igor Sysoevfeee7262007-12-30 08:15:27 +00002876 char *uri_separator;
2877 u_char *p;
2878 ngx_http_upstream_t *u;
2879 ngx_http_core_srv_conf_t *cscf;
Igor Sysoev1b735832004-11-11 14:07:14 +00002880
Igor Sysoevfeee7262007-12-30 08:15:27 +00002881 cscf = ngx_http_get_module_srv_conf(r, ngx_http_core_module);
2882
2883 p = ngx_snprintf(buf, len, ", server: %V", &cscf->server_name);
2884 len -= p - buf;
2885 buf = p;
Igor Sysoev4d656dc2005-03-22 16:02:46 +00002886
Igor Sysoeva6b59572007-11-27 12:22:01 +00002887 if (r->request_line.data == NULL && r->request_start) {
2888 for (p = r->request_start; p < r->header_in->last; p++) {
2889 if (*p == CR || *p == LF) {
2890 break;
2891 }
2892 }
2893
2894 r->request_line.len = p - r->request_start;
2895 r->request_line.data = r->request_start;
2896 }
2897
2898 if (r->request_line.len) {
2899 p = ngx_snprintf(buf, len, ", request: \"%V\"", &r->request_line);
Igor Sysoevc04deca2005-03-28 14:43:02 +00002900 len -= p - buf;
2901 buf = p;
Igor Sysoev1b735832004-11-11 14:07:14 +00002902 }
2903
Igor Sysoevbb28b6d2006-07-11 13:20:19 +00002904 if (r != sr) {
2905 p = ngx_snprintf(buf, len, ", subrequest: \"%V\"", &sr->uri);
2906 len -= p - buf;
2907 buf = p;
2908 }
2909
2910 u = sr->upstream;
2911
Igor Sysoev3d2fd182006-12-04 16:46:13 +00002912 if (u && u->peer.name) {
2913
2914 uri_separator = "";
2915
2916#if (NGX_HAVE_UNIX_DOMAIN)
2917 if (u->peer.sockaddr && u->peer.sockaddr->sa_family == AF_UNIX) {
2918 uri_separator = ":";
2919 }
2920#endif
Igor Sysoevbb28b6d2006-07-11 13:20:19 +00002921
2922 p = ngx_snprintf(buf, len, ", upstream: \"%V%V%s%V\"",
Igor Sysoev96dd8af2007-11-27 13:34:13 +00002923 &u->schema, u->peer.name,
Igor Sysoev3d2fd182006-12-04 16:46:13 +00002924 uri_separator, &u->uri);
Igor Sysoevbb28b6d2006-07-11 13:20:19 +00002925 len -= p - buf;
2926 buf = p;
2927 }
2928
Igor Sysoevc04deca2005-03-28 14:43:02 +00002929 if (r->headers_in.host) {
2930 p = ngx_snprintf(buf, len, ", host: \"%V\"",
2931 &r->headers_in.host->value);
2932 len -= p - buf;
2933 buf = p;
2934 }
2935
2936 if (r->headers_in.referer) {
2937 p = ngx_snprintf(buf, len, ", referrer: \"%V\"",
2938 &r->headers_in.referer->value);
2939 buf = p;
2940 }
2941
2942 return buf;
Igor Sysoev6de5c2c2002-08-06 16:39:45 +00002943}