blob: a5386965e3cffb1bee05955be917f4e5a6ff7069 [file] [log] [blame]
Igor Sysoev6de5c2c2002-08-06 16:39:45 +00001
2#include <ngx_config.h>
Igor Sysoev0ad17c02002-08-26 15:18:19 +00003#include <ngx_core.h>
4#include <ngx_string.h>
Igor Sysoeva58e3ca2002-09-02 14:48:24 +00005#include <ngx_files.h>
Igor Sysoev6de5c2c2002-08-06 16:39:45 +00006#include <ngx_log.h>
7#include <ngx_alloc.h>
Igor Sysoeva0bb31f2002-12-02 16:09:40 +00008#include <ngx_array.h>
9#include <ngx_table.h>
Igor Sysoev6de5c2c2002-08-06 16:39:45 +000010#include <ngx_hunk.h>
11#include <ngx_connection.h>
Igor Sysoev42feecb2002-12-15 06:25:09 +000012#include <ngx_inet.h>
Igor Sysoev6de5c2c2002-08-06 16:39:45 +000013#include <ngx_http.h>
Igor Sysoeve0268b92002-09-11 15:18:33 +000014#include <ngx_http_config.h>
Igor Sysoev4e9393a2003-01-09 05:36:00 +000015#include <ngx_http_core_module.h>
Igor Sysoev6de5c2c2002-08-06 16:39:45 +000016
Igor Sysoeva58e3ca2002-09-02 14:48:24 +000017/* STUB */
18#include <ngx_http_output_filter.h>
Igor Sysoeva58e3ca2002-09-02 14:48:24 +000019int ngx_http_static_handler(ngx_http_request_t *r);
Igor Sysoeve0268b92002-09-11 15:18:33 +000020int ngx_http_index_handler(ngx_http_request_t *r);
Igor Sysoeva0bb31f2002-12-02 16:09:40 +000021int ngx_http_proxy_handler(ngx_http_request_t *r);
Igor Sysoevb0869052002-12-10 18:05:12 +000022/**/
Igor Sysoeva58e3ca2002-09-02 14:48:24 +000023
Igor Sysoev0ad17c02002-08-26 15:18:19 +000024int ngx_http_init_connection(ngx_connection_t *c);
Igor Sysoev6de5c2c2002-08-06 16:39:45 +000025
Igor Sysoev6de5c2c2002-08-06 16:39:45 +000026static int ngx_http_init_request(ngx_event_t *ev);
Igor Sysoevb0869052002-12-10 18:05:12 +000027static int ngx_http_process_request_header(ngx_event_t *ev);
Igor Sysoev6de5c2c2002-08-06 16:39:45 +000028
Igor Sysoev016b8522002-08-29 16:59:54 +000029static int ngx_http_process_request_line(ngx_http_request_t *r);
Igor Sysoevb0869052002-12-10 18:05:12 +000030static int ngx_http_process_request_headers(ngx_http_request_t *r);
Igor Sysoeva58e3ca2002-09-02 14:48:24 +000031static int ngx_http_process_request_header_line(ngx_http_request_t *r);
Igor Sysoev6de5c2c2002-08-06 16:39:45 +000032
Igor Sysoev42feecb2002-12-15 06:25:09 +000033static int ngx_http_event_request_handler(ngx_http_request_t *r);
Igor Sysoeva58e3ca2002-09-02 14:48:24 +000034
35static int ngx_http_writer(ngx_event_t *ev);
Igor Sysoev42feecb2002-12-15 06:25:09 +000036static int ngx_http_block_read(ngx_event_t *ev);
37static int ngx_http_read_discarded_body(ngx_event_t *ev);
Igor Sysoev1af7c822002-09-13 14:47:42 +000038static int ngx_http_keepalive_handler(ngx_event_t *ev);
Igor Sysoev42feecb2002-12-15 06:25:09 +000039static int ngx_http_set_lingering_close(ngx_http_request_t *r);
40static int ngx_http_lingering_close_handler(ngx_event_t *ev);
Igor Sysoev3a40d482002-09-12 14:42:29 +000041
Igor Sysoevb0869052002-12-10 18:05:12 +000042#if 0
43int ngx_http_special_response(ngx_http_request_t *r, int error);
44int ngx_http_redirect(ngx_http_request_t *r, int redirect);
45int ngx_http_error(ngx_http_request_t *r, int error);
46int ngx_http_close_request(ngx_http_request_t *r);
47#endif
Igor Sysoev6de5c2c2002-08-06 16:39:45 +000048
Igor Sysoev2ba1ee02002-10-04 17:58:04 +000049static int ngx_http_close_connection(ngx_event_t *ev);
Igor Sysoev0ad17c02002-08-26 15:18:19 +000050static size_t ngx_http_log_error(void *data, char *buf, size_t len);
51
Igor Sysoev6de5c2c2002-08-06 16:39:45 +000052
Igor Sysoev0ad17c02002-08-26 15:18:19 +000053
Igor Sysoev016b8522002-08-29 16:59:54 +000054static char *header_errors[] = {
55 "client %s sent invalid method",
56 "client %s sent invalid request",
Igor Sysoev1af7c822002-09-13 14:47:42 +000057 "client %s sent too long URI",
Igor Sysoev016b8522002-08-29 16:59:54 +000058 "client %s sent HEAD method in HTTP/0.9 request"
59};
60
61
Igor Sysoeva0bb31f2002-12-02 16:09:40 +000062static ngx_http_header_t headers_in[] = {
63 { 4, "Host", offsetof(ngx_http_headers_in_t, host) },
64 { 10, "Connection", offsetof(ngx_http_headers_in_t, connection) },
Igor Sysoev42feecb2002-12-15 06:25:09 +000065 { 17, "If-Modified-Since",
66 offsetof(ngx_http_headers_in_t,if_modified_since) },
Igor Sysoeva0bb31f2002-12-02 16:09:40 +000067
68 { 10, "User-Agent", offsetof(ngx_http_headers_in_t, user_agent) },
69
70 { 0, NULL, 0 }
71};
72
Igor Sysoev6de5c2c2002-08-06 16:39:45 +000073
74int ngx_http_init_connection(ngx_connection_t *c)
75{
Igor Sysoev0ad17c02002-08-26 15:18:19 +000076 ngx_event_t *ev;
77 struct sockaddr *addr;
78 ngx_http_log_ctx_t *ctx;
Igor Sysoev6de5c2c2002-08-06 16:39:45 +000079
80 ev = c->read;
Igor Sysoev6de5c2c2002-08-06 16:39:45 +000081 ev->event_handler = ngx_http_init_request;
Igor Sysoev2ba1ee02002-10-04 17:58:04 +000082
Igor Sysoev1af7c822002-09-13 14:47:42 +000083 ngx_test_null(c->pool,
Igor Sysoev4e9393a2003-01-09 05:36:00 +000084 ngx_create_pool(ngx_http_connection_pool_size, ev->log),
Igor Sysoev1af7c822002-09-13 14:47:42 +000085 NGX_ERROR);
Igor Sysoev6de5c2c2002-08-06 16:39:45 +000086
Igor Sysoev2ba1ee02002-10-04 17:58:04 +000087 ngx_test_null(c->requests, ngx_create_array(c->pool, 10, sizeof(char *)),
88 NGX_ERROR);
89
90 ev->close_handler = ngx_http_close_connection;
91 c->write->close_handler = ngx_http_close_connection;
92
Igor Sysoev0ad17c02002-08-26 15:18:19 +000093 ngx_test_null(addr, ngx_palloc(c->pool, c->socklen), NGX_ERROR);
94 ngx_memcpy(addr, c->sockaddr, c->socklen);
95 c->sockaddr = addr;
96
Igor Sysoev4e9393a2003-01-09 05:36:00 +000097 ngx_test_null(c->addr_text.data, ngx_palloc(c->pool, c->addr_text_max_len),
Igor Sysoev0ad17c02002-08-26 15:18:19 +000098 NGX_ERROR);
Igor Sysoevb0869052002-12-10 18:05:12 +000099
Igor Sysoev5518aba2002-12-17 15:48:27 +0000100 c->addr_text.len = ngx_inet_ntop(c->family,
101 (char *)c->sockaddr + c->addr,
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000102 c->addr_text.data, c->addr_text_max_len);
Igor Sysoev88092572002-12-19 07:08:55 +0000103 if (c->addr_text.len == 0)
Igor Sysoev5518aba2002-12-17 15:48:27 +0000104 return NGX_ERROR;
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000105
106 ngx_test_null(ctx, ngx_pcalloc(c->pool, sizeof(ngx_http_log_ctx_t)),
107 NGX_ERROR);
Igor Sysoevb0869052002-12-10 18:05:12 +0000108 ctx->client = c->addr_text.data;
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000109 ctx->action = "reading client request line";
110 c->log->data = ctx;
111 c->log->handler = ngx_http_log_error;
112
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000113#if (HAVE_DEFERRED_ACCEPT)
Igor Sysoev016b8522002-08-29 16:59:54 +0000114 if (ev->ready) {
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000115 return ngx_http_init_request(ev);
Igor Sysoev016b8522002-08-29 16:59:54 +0000116 } else {
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000117#endif
Igor Sysoev016b8522002-08-29 16:59:54 +0000118 ngx_add_timer(ev, c->post_accept_timeout);
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000119#if (USE_KQUEUE)
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000120 return ngx_add_event(ev, NGX_READ_EVENT, NGX_CLEAR_EVENT);
121#else
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000122#if (HAVE_AIO_EVENT)
123 if (ngx_event_type == NGX_AIO_EVENT)
124 return ngx_http_init_request(ev);
125 else
126#endif
127#if (HAVE_CLEAR_EVENT)
128 if (ngx_event_type == NGX_KQUEUE_EVENT)
129 return ngx_add_event(ev, NGX_READ_EVENT, NGX_CLEAR_EVENT);
130 else
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000131#endif
Igor Sysoev016b8522002-08-29 16:59:54 +0000132 return ngx_add_event(ev, NGX_READ_EVENT, NGX_LEVEL_EVENT);
133#endif /* USE_KQUEUE */
134#if (HAVE_DEFERRED_ACCEPT)
135 }
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000136#endif
137}
138
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000139
Igor Sysoev1af7c822002-09-13 14:47:42 +0000140static int ngx_http_init_request(ngx_event_t *ev)
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000141{
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000142 ngx_connection_t *c;
143 ngx_http_request_t *r;
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000144
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000145 c = (ngx_connection_t *) ev->data;
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000146
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000147 ngx_test_null(r, ngx_pcalloc(c->pool, sizeof(ngx_http_request_t)),
148 NGX_ERROR);
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000149
150 c->data = r;
151 r->connection = c;
Igor Sysoev42feecb2002-12-15 06:25:09 +0000152 r->file.fd = NGX_INVALID_FILE;
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000153
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000154 if (c->buffer == NULL) {
155 ngx_test_null(c->buffer,
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000156 ngx_create_temp_hunk(c->pool,
157 ngx_http_client_header_buffer_size,
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000158 0, 0),
159 NGX_ERROR);
160 } else {
161 r->header_read = 1;
162 }
163
164 r->header_in = c->buffer;
165
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000166 ngx_test_null(r->pool, ngx_create_pool(ngx_http_request_pool_size, ev->log),
Igor Sysoev016b8522002-08-29 16:59:54 +0000167 ngx_http_close_request(r));
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000168
Igor Sysoev207ed5a2002-12-26 16:26:23 +0000169 ngx_test_null(r->ctx,
170 ngx_pcalloc(r->pool, sizeof(void *) * ngx_http_max_module),
Igor Sysoeve0268b92002-09-11 15:18:33 +0000171 ngx_http_close_request(r));
172
Igor Sysoev42feecb2002-12-15 06:25:09 +0000173 r->headers_out.headers = ngx_create_table(r->pool, 10);
174 r->headers_out.content_length = -1;
175 r->headers_out.last_modified_time = -1;
176
Igor Sysoevb0869052002-12-10 18:05:12 +0000177 ev->event_handler = ngx_http_process_request_header;
Igor Sysoev016b8522002-08-29 16:59:54 +0000178 r->state_handler = ngx_http_process_request_line;
Igor Sysoev1af7c822002-09-13 14:47:42 +0000179 r->header_timeout = 1;
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000180
Igor Sysoevb0869052002-12-10 18:05:12 +0000181 return ngx_http_process_request_header(ev);
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000182}
183
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000184
Igor Sysoevb0869052002-12-10 18:05:12 +0000185static int ngx_http_process_request_header(ngx_event_t *ev)
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000186{
Igor Sysoev016b8522002-08-29 16:59:54 +0000187 int n, rc;
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000188 ngx_connection_t *c ;
189 ngx_http_request_t *r;
Igor Sysoev1af7c822002-09-13 14:47:42 +0000190 ngx_http_log_ctx_t *ctx;
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000191
192 c = (ngx_connection_t *) ev->data;
193 r = (ngx_http_request_t *) c->data;
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000194
195 ngx_log_debug(ev->log, "http process request");
196
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000197 if (r->header_read) {
198 r->header_read = 0;
199 ngx_log_debug(ev->log, "http preread %d" _
200 r->header_in->last.mem - r->header_in->pos.mem);
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000201
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000202 } else {
203 n = ngx_event_recv(c, r->header_in->last.mem,
204 r->header_in->end - r->header_in->last.mem);
205
206 if (n == NGX_AGAIN) {
207 if (r->header_timeout) {
208 r->header_timeout = 0;
209 ngx_del_timer(ev);
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000210 ngx_add_timer(ev, ngx_http_client_header_timeout);
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000211 }
212 return NGX_AGAIN;
Igor Sysoev016b8522002-08-29 16:59:54 +0000213 }
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000214
215 if (n == NGX_ERROR)
216 return ngx_http_close_request(r);
217
218 ngx_log_debug(ev->log, "http read %d" _ n);
219
220 if (n == 0) {
Igor Sysoevb0869052002-12-10 18:05:12 +0000221 ngx_log_error(NGX_LOG_INFO, c->log, 0,
222 "client has prematurely closed connection");
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000223 return ngx_http_close_request(r);
224 }
225
226 r->header_in->last.mem += n;
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000227 }
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000228
Igor Sysoev016b8522002-08-29 16:59:54 +0000229 /* state_handlers are called in following order:
230 ngx_http_process_request_line(r)
Igor Sysoevb0869052002-12-10 18:05:12 +0000231 ngx_http_process_request_headers(r) */
Igor Sysoev016b8522002-08-29 16:59:54 +0000232
233 do {
234 rc = (r->state_handler)(r);
235
236 if (rc == NGX_ERROR)
237 return rc;
238
Igor Sysoevb0869052002-12-10 18:05:12 +0000239 } while (rc == NGX_AGAIN && r->header_in->pos.mem < r->header_in->last.mem);
Igor Sysoeva58e3ca2002-09-02 14:48:24 +0000240
Igor Sysoev1af7c822002-09-13 14:47:42 +0000241 if (r->header_timeout) {
242 r->header_timeout = 0;
Igor Sysoev016b8522002-08-29 16:59:54 +0000243 ngx_del_timer(ev);
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000244 ngx_add_timer(ev, ngx_http_client_header_timeout);
Igor Sysoev016b8522002-08-29 16:59:54 +0000245 }
246
Igor Sysoevb0869052002-12-10 18:05:12 +0000247 if (rc == NGX_OK)
Igor Sysoev42feecb2002-12-15 06:25:09 +0000248 return ngx_http_event_request_handler(r);
Igor Sysoevb0869052002-12-10 18:05:12 +0000249 else
250 return rc;
Igor Sysoev016b8522002-08-29 16:59:54 +0000251}
252
Igor Sysoev1af7c822002-09-13 14:47:42 +0000253
Igor Sysoev016b8522002-08-29 16:59:54 +0000254static int ngx_http_process_request_line(ngx_http_request_t *r)
255{
Igor Sysoev2ba1ee02002-10-04 17:58:04 +0000256 int rc, len;
257 char **request;
258 ngx_connection_t *c;
Igor Sysoev016b8522002-08-29 16:59:54 +0000259 ngx_http_log_ctx_t *ctx;
260
261 rc = ngx_read_http_request_line(r);
262
Igor Sysoev2ba1ee02002-10-04 17:58:04 +0000263 c = r->connection;
264
Igor Sysoev016b8522002-08-29 16:59:54 +0000265 if (rc == NGX_OK) {
Igor Sysoev42feecb2002-12-15 06:25:09 +0000266 r->uri.len = (r->args_start ? r->args_start - 1 : r->uri_end)
267 - r->uri_start;
Igor Sysoevb0869052002-12-10 18:05:12 +0000268 ngx_test_null(r->uri.data, ngx_palloc(r->pool, r->uri.len + 1),
Igor Sysoev016b8522002-08-29 16:59:54 +0000269 ngx_http_close_request(r));
Igor Sysoevb0869052002-12-10 18:05:12 +0000270 ngx_cpystrn(r->uri.data, r->uri_start, r->uri.len + 1);
Igor Sysoev016b8522002-08-29 16:59:54 +0000271
Igor Sysoev2d0d9092002-12-03 15:45:38 +0000272 r->request_line.len = r->request_end - r->header_in->start;
273 ngx_test_null(r->request_line.data,
274 ngx_palloc(r->pool, r->request_line.len + 1),
275 ngx_http_close_request(r));
276 ngx_cpystrn(r->request_line.data, r->header_in->start,
277 r->request_line.len + 1);
278
279 /* TEMP */
Igor Sysoev2ba1ee02002-10-04 17:58:04 +0000280 ngx_test_null(request, ngx_push_array(c->requests),
281 ngx_http_close_request(r));
282
283 if (r->request_end)
284 len = r->request_end - r->header_in->start + 1;
285 else
286 len = 1;
287 c->requests_len += len;
288 ngx_test_null(*request, ngx_palloc(c->pool, len),
289 ngx_http_close_request(r));
290 ngx_cpystrn(*request, r->header_in->start, len);
291
292 ngx_log_debug(c->log, "REQ: '%s'" _ *request);
Igor Sysoev2d0d9092002-12-03 15:45:38 +0000293 /* */
Igor Sysoev2ba1ee02002-10-04 17:58:04 +0000294
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000295 if (r->uri_ext) {
Igor Sysoev42feecb2002-12-15 06:25:09 +0000296 r->exten.len = (r->args_start ? r->args_start - 1 : r->uri_end)
297 - r->uri_ext;
Igor Sysoevb0869052002-12-10 18:05:12 +0000298 ngx_test_null(r->exten.data,
299 ngx_palloc(r->pool, r->exten.len + 1),
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000300 ngx_http_close_request(r));
Igor Sysoevb0869052002-12-10 18:05:12 +0000301 ngx_cpystrn(r->exten.data, r->uri_ext, r->exten.len + 1);
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000302 }
303
Igor Sysoev295bb632002-12-23 18:22:18 +0000304#if 0
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000305 ngx_log_debug(r->connection->log, "HTTP: %d, %d, %s %s" _
Igor Sysoevb0869052002-12-10 18:05:12 +0000306 r->method _ r->http_version _
307 r->uri.data _ r->exten.data);
Igor Sysoev295bb632002-12-23 18:22:18 +0000308#endif
Igor Sysoev016b8522002-08-29 16:59:54 +0000309
Igor Sysoeva58e3ca2002-09-02 14:48:24 +0000310 if (r->http_version == 9)
Igor Sysoevb0869052002-12-10 18:05:12 +0000311 return NGX_OK;
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000312
Igor Sysoev1af7c822002-09-13 14:47:42 +0000313 /* TODO: check too long URI - no space for header, compact buffer */
314
Igor Sysoeva0bb31f2002-12-02 16:09:40 +0000315 r->headers_in.headers = ngx_create_table(r->pool, 10);
316
Igor Sysoevb0869052002-12-10 18:05:12 +0000317 r->state_handler = ngx_http_process_request_headers;
Igor Sysoeva58e3ca2002-09-02 14:48:24 +0000318 ctx = r->connection->log->data;
319 ctx->action = "reading client request headers";
Igor Sysoev016b8522002-08-29 16:59:54 +0000320
Igor Sysoevb0869052002-12-10 18:05:12 +0000321 return NGX_AGAIN;
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000322 }
323
Igor Sysoev1af7c822002-09-13 14:47:42 +0000324 if (r->header_in->last.mem >= r->header_in->end) {
Igor Sysoevef259d12002-12-06 16:32:33 +0000325 rc = NGX_HTTP_PARSE_TOO_LONG_URI;
Igor Sysoev1af7c822002-09-13 14:47:42 +0000326
327 } else if (rc == NGX_AGAIN) {
328 return NGX_AGAIN;
329 }
330
Igor Sysoev016b8522002-08-29 16:59:54 +0000331 ctx = r->connection->log->data;
Igor Sysoeva58e3ca2002-09-02 14:48:24 +0000332 r->connection->log->handler = NULL;
Igor Sysoev1af7c822002-09-13 14:47:42 +0000333 ngx_log_error(NGX_LOG_ERR, r->connection->log, 0,
334 header_errors[rc - NGX_HTTP_PARSE_INVALID_METHOD],
335 ctx->client);
Igor Sysoev016b8522002-08-29 16:59:54 +0000336 r->connection->log->handler = ngx_http_log_error;
337
Igor Sysoev1af7c822002-09-13 14:47:42 +0000338 return ngx_http_error(r, (rc == NGX_HTTP_PARSE_TOO_LONG_URI) ?
339 NGX_HTTP_REQUEST_URI_TOO_LARGE:
340 NGX_HTTP_BAD_REQUEST);
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000341}
342
Igor Sysoev1af7c822002-09-13 14:47:42 +0000343
Igor Sysoevb0869052002-12-10 18:05:12 +0000344static int ngx_http_process_request_headers(ngx_http_request_t *r)
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000345{
Igor Sysoev016b8522002-08-29 16:59:54 +0000346 int rc;
347 ngx_http_log_ctx_t *ctx;
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000348
Igor Sysoev016b8522002-08-29 16:59:54 +0000349 for ( ;; ) {
Igor Sysoevb0869052002-12-10 18:05:12 +0000350 rc = ngx_read_http_header_line(r, r->header_in);
Igor Sysoev016b8522002-08-29 16:59:54 +0000351
Igor Sysoev1af7c822002-09-13 14:47:42 +0000352 /* TODO: check too long header, compact buffer */
353
Igor Sysoev016b8522002-08-29 16:59:54 +0000354 if (rc == NGX_OK) {
Igor Sysoeva58e3ca2002-09-02 14:48:24 +0000355 if (ngx_http_process_request_header_line(r) == NGX_ERROR)
356 return ngx_http_error(r, NGX_HTTP_BAD_REQUEST);
357
Igor Sysoev1af7c822002-09-13 14:47:42 +0000358 } else if (rc == NGX_HTTP_PARSE_HEADER_DONE) {
Igor Sysoeva58e3ca2002-09-02 14:48:24 +0000359 ngx_log_debug(r->connection->log, "HTTP header done");
Igor Sysoev42feecb2002-12-15 06:25:09 +0000360
361 if (r->http_version > NGX_HTTP_VERSION_10
362 && r->headers_in.host == NULL)
363 {
364 return ngx_http_error(r, NGX_HTTP_BAD_REQUEST);
365 } else {
366 return NGX_OK;
367 }
Igor Sysoev016b8522002-08-29 16:59:54 +0000368
369 } else if (rc == NGX_AGAIN) {
370 return NGX_AGAIN;
371
Igor Sysoev1af7c822002-09-13 14:47:42 +0000372 } else if (rc == NGX_HTTP_PARSE_INVALID_HEADER) {
Igor Sysoev016b8522002-08-29 16:59:54 +0000373 ctx = r->connection->log->data;
Igor Sysoeva58e3ca2002-09-02 14:48:24 +0000374 r->connection->log->handler = NULL;
Igor Sysoev1af7c822002-09-13 14:47:42 +0000375 ngx_log_error(NGX_LOG_ERR, r->connection->log, 0,
Igor Sysoev016b8522002-08-29 16:59:54 +0000376 "client %s sent invalid header", ctx->client);
377 r->connection->log->handler = ngx_http_log_error;
378
Igor Sysoeva58e3ca2002-09-02 14:48:24 +0000379 return ngx_http_error(r, NGX_HTTP_BAD_REQUEST);
Igor Sysoev016b8522002-08-29 16:59:54 +0000380 }
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000381 }
Igor Sysoeva58e3ca2002-09-02 14:48:24 +0000382}
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000383
Igor Sysoev1af7c822002-09-13 14:47:42 +0000384
Igor Sysoeva58e3ca2002-09-02 14:48:24 +0000385static int ngx_http_process_request_header_line(ngx_http_request_t *r)
386{
Igor Sysoeva0bb31f2002-12-02 16:09:40 +0000387 int i;
388 ngx_table_elt_t *h;
389
Igor Sysoev2d0d9092002-12-03 15:45:38 +0000390 ngx_test_null(h, ngx_push_table(r->headers_in.headers), NGX_ERROR);
Igor Sysoeva0bb31f2002-12-02 16:09:40 +0000391
392 h->key.len = r->header_name_end - r->header_name_start;
393 ngx_test_null(h->key.data, ngx_palloc(r->pool, h->key.len + 1), NGX_ERROR);
394 ngx_cpystrn(h->key.data, r->header_name_start, h->key.len + 1);
395
396 h->value.len = r->header_end - r->header_start;
397 ngx_test_null(h->value.data, ngx_palloc(r->pool, h->value.len + 1),
398 NGX_ERROR);
399 ngx_cpystrn(h->value.data, r->header_start, h->value.len + 1);
400
401 for (i = 0; headers_in[i].len != 0; i++) {
402 if (headers_in[i].len == h->key.len) {
403 if (strcasecmp(headers_in[i].data, h->key.data) == 0) {
404 *((ngx_table_elt_t **)
405 ((char *) &r->headers_in + headers_in[i].offset)) = h;
406 }
407 }
408 }
409
Igor Sysoeva58e3ca2002-09-02 14:48:24 +0000410 ngx_log_debug(r->connection->log, "HTTP header: '%s: %s'" _
Igor Sysoeva0bb31f2002-12-02 16:09:40 +0000411 h->key.data _ h->value.data);
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000412
Igor Sysoeva58e3ca2002-09-02 14:48:24 +0000413 return NGX_OK;
414}
415
Igor Sysoeva58e3ca2002-09-02 14:48:24 +0000416
Igor Sysoev42feecb2002-12-15 06:25:09 +0000417static int ngx_http_event_request_handler(ngx_http_request_t *r)
Igor Sysoeva58e3ca2002-09-02 14:48:24 +0000418{
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000419 int rc;
Igor Sysoev96f83772002-09-07 10:14:25 +0000420 ngx_msec_t timeout;
Igor Sysoeva58e3ca2002-09-02 14:48:24 +0000421
Igor Sysoevb0869052002-12-10 18:05:12 +0000422 ngx_del_timer(r->connection->read);
423 r->header_timeout = 0;
424
425 r->state_handler = NULL;
426 r->connection->read->event_handler = ngx_http_block_read;
Igor Sysoeva0bb31f2002-12-02 16:09:40 +0000427
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000428 rc = ngx_http_handler(r);
Igor Sysoeva58e3ca2002-09-02 14:48:24 +0000429
Igor Sysoevef259d12002-12-06 16:32:33 +0000430 /* handler is still busy */
Igor Sysoev6ed07e42002-12-05 16:21:24 +0000431 if (rc == NGX_WAITING)
432 return rc;
433
Igor Sysoevef259d12002-12-06 16:32:33 +0000434 /* handler has done its work but transfer is not completed */
Igor Sysoeva58e3ca2002-09-02 14:48:24 +0000435 if (rc == NGX_AGAIN) {
436#if (HAVE_CLEAR_EVENT)
437 if (ngx_add_event(r->connection->write, NGX_WRITE_EVENT,
438 NGX_CLEAR_EVENT) == NGX_ERROR) {
439#else
440 if (ngx_add_event(r->connection->write, NGX_WRITE_EVENT,
441 NGX_ONESHOT_EVENT) == NGX_ERROR) {
442#endif
Igor Sysoeva58e3ca2002-09-02 14:48:24 +0000443 return ngx_http_close_request(r);
444 }
445
Igor Sysoev96f83772002-09-07 10:14:25 +0000446 if (r->connection->sent > 0) {
447 ngx_log_debug(r->connection->log, "sent: " QD_FMT _
448 r->connection->sent);
449 timeout = (ngx_msec_t) (r->connection->sent * 10);
450 ngx_log_debug(r->connection->log, "timeout: %d" _ timeout);
451 ngx_add_timer(r->connection->write, timeout);
452
453 } else {
454 ngx_add_timer(r->connection->write, 10000);
455 }
Igor Sysoeva58e3ca2002-09-02 14:48:24 +0000456
457 r->connection->write->event_handler = ngx_http_writer;
458 return rc;
459 }
460
461 if (rc == NGX_ERROR) {
462 /* log http request */
463 return ngx_http_close_request(r);
464 }
465
466 if (rc >= NGX_HTTP_SPECIAL_RESPONSE)
467 return ngx_http_special_response(r, rc);
468
469 /* rc == NGX_OK */
470
Igor Sysoev3a40d482002-09-12 14:42:29 +0000471 if (!r->keepalive) {
472 if (r->lingering_close) {
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000473 return ngx_http_set_lingering_close(r);
474
Igor Sysoev3a40d482002-09-12 14:42:29 +0000475 } else {
476 return ngx_http_close_request(r);
477 }
478 }
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000479
480 /* keepalive */
481
482 ngx_http_close_request(r);
483 r->connection->buffer->pos.mem = r->connection->buffer->last.mem
484 = r->connection->buffer->start;
485 r->connection->read->event_handler = ngx_http_keepalive_handler;
486}
487
Igor Sysoevb0869052002-12-10 18:05:12 +0000488
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000489static int ngx_http_writer(ngx_event_t *ev)
490{
491 int rc;
Igor Sysoeva0bb31f2002-12-02 16:09:40 +0000492 ngx_msec_t timeout;
493 ngx_connection_t *c;
494 ngx_http_request_t *r;
495 ngx_http_core_loc_conf_t *conf;
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000496
Igor Sysoeva58e3ca2002-09-02 14:48:24 +0000497 c = (ngx_connection_t *) ev->data;
498 r = (ngx_http_request_t *) c->data;
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000499
Igor Sysoev96f83772002-09-07 10:14:25 +0000500 c->sent = 0;
501
Igor Sysoeva58e3ca2002-09-02 14:48:24 +0000502 rc = ngx_http_output_filter(r, NULL);
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000503
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000504 ngx_log_debug(ev->log, "output filter in writer: %d" _ rc);
Igor Sysoeva58e3ca2002-09-02 14:48:24 +0000505
506 if (rc == NGX_AGAIN) {
507
508 if (c->sent > 0) {
Igor Sysoeva0bb31f2002-12-02 16:09:40 +0000509 conf = (ngx_http_core_loc_conf_t *)
Igor Sysoev207ed5a2002-12-26 16:26:23 +0000510 ngx_http_get_module_loc_conf(r->main ? r->main : r,
511 ngx_http_core_module_ctx);
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000512
513 timeout = (ngx_msec_t) (c->sent * conf->send_timeout);
514
Igor Sysoeva58e3ca2002-09-02 14:48:24 +0000515 ngx_log_debug(ev->log, "sent: " QD_FMT _ c->sent);
Igor Sysoeva58e3ca2002-09-02 14:48:24 +0000516 ngx_log_debug(ev->log, "timeout: %d" _ timeout);
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000517
Igor Sysoeva58e3ca2002-09-02 14:48:24 +0000518 ngx_del_timer(ev);
519 ngx_add_timer(ev, timeout);
520 }
521
522 if (ev->oneshot)
523 if (ngx_add_event(r->connection->write, NGX_WRITE_EVENT,
524 NGX_ONESHOT_EVENT) == NGX_ERROR) {
Igor Sysoeva58e3ca2002-09-02 14:48:24 +0000525 return ngx_http_close_request(r);
526 }
527
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000528 return rc;
Igor Sysoeva58e3ca2002-09-02 14:48:24 +0000529 }
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000530
531 if (rc == NGX_ERROR)
532 return rc;
533
534 /* rc == NGX_OK */
535
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000536 ngx_log_debug(ev->log, "http writer done");
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000537
Igor Sysoev3a40d482002-09-12 14:42:29 +0000538 if (!r->keepalive) {
539 if (r->lingering_close) {
Igor Sysoev3a17f242002-12-24 17:30:59 +0000540 return ngx_http_set_lingering_close(r);
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000541
Igor Sysoev3a40d482002-09-12 14:42:29 +0000542 } else {
543 return ngx_http_close_request(r);
544 }
545 }
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000546
Igor Sysoev3a40d482002-09-12 14:42:29 +0000547 /* keepalive */
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000548
Igor Sysoev3a40d482002-09-12 14:42:29 +0000549 ngx_http_close_request(r);
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000550 c->buffer->pos.mem = c->buffer->last.mem = c->buffer->start;
551 c->read->event_handler = ngx_http_keepalive_handler;
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000552}
553
Igor Sysoevb0869052002-12-10 18:05:12 +0000554
Igor Sysoev42feecb2002-12-15 06:25:09 +0000555static int ngx_http_block_read(ngx_event_t *ev)
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000556{
Igor Sysoev42feecb2002-12-15 06:25:09 +0000557 ngx_log_debug(ev->log, "http read blocked");
558
559 ev->blocked = 1;
560 return ngx_del_event(ev, NGX_READ_EVENT, 0);
561}
562
563
564int ngx_http_discard_body(ngx_http_request_t *r)
565{
566 ngx_log_debug(r->connection->log, "set discard body");
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000567
568 ngx_del_timer(r->connection->read);
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000569
Igor Sysoev42feecb2002-12-15 06:25:09 +0000570 if (r->client_content_length)
571 r->connection->read->event_handler = ngx_http_read_discarded_body;
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000572
573 return NGX_OK;
574}
575
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000576
Igor Sysoev42feecb2002-12-15 06:25:09 +0000577static int ngx_http_read_discarded_body(ngx_event_t *ev)
578{
579 size_t size;
580 ssize_t n;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000581 ngx_connection_t *c;
582 ngx_http_request_t *r;
583 ngx_http_core_loc_conf_t *lcf;
Igor Sysoev42feecb2002-12-15 06:25:09 +0000584
585 ngx_log_debug(ev->log, "http read discarded body");
586
587 if (ev->timedout)
588 return NGX_ERROR;
589
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000590 c = (ngx_connection_t *) ev->data;
591 r = (ngx_http_request_t *) c->data;
592
593 lcf = (ngx_http_core_loc_conf_t *)
594 ngx_http_get_module_loc_conf(r, ngx_http_core_module_ctx);
595
Igor Sysoev42feecb2002-12-15 06:25:09 +0000596 if (r->discarded_buffer == NULL)
597 ngx_test_null(r->discarded_buffer,
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000598 ngx_palloc(r->pool, lcf->discarded_buffer_size),
Igor Sysoev42feecb2002-12-15 06:25:09 +0000599 NGX_ERROR);
600
601 size = r->client_content_length;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000602 if (size > lcf->discarded_buffer_size)
603 size = lcf->discarded_buffer_size;
Igor Sysoev42feecb2002-12-15 06:25:09 +0000604
605 n = ngx_event_recv(c, r->discarded_buffer, size);
606 if (n == NGX_ERROR)
607 return NGX_ERROR;
608
609 if (n == NGX_AGAIN)
610 return NGX_OK;
611
612 r->client_content_length -= n;
613 /* XXX: what if r->client_content_length == 0 ? */
614 return NGX_OK;
615}
616
617
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000618static int ngx_http_keepalive_handler(ngx_event_t *ev)
619{
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000620 ssize_t n;
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000621 ngx_connection_t *c;
622 ngx_http_log_ctx_t *ctx;
623
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000624 c = (ngx_connection_t *) ev->data;
625
Igor Sysoev42feecb2002-12-15 06:25:09 +0000626 ngx_log_debug(ev->log, "http keepalive handler");
Igor Sysoev1af7c822002-09-13 14:47:42 +0000627
628 if (ev->timedout)
629 return NGX_DONE;
630
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000631 n = ngx_event_recv(c, c->buffer->last.mem,
632 c->buffer->end - c->buffer->last.mem);
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000633
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000634 if (n == NGX_AGAIN || n == NGX_ERROR)
635 return n;
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000636
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000637 ctx = (ngx_http_log_ctx_t *) ev->log->data;
638 ev->log->handler = NULL;
639
640 if (n == 0) {
641 ngx_log_error(NGX_LOG_INFO, ev->log, 0,
642 "client %s closed keepalive connection", ctx->client);
643 return NGX_DONE;
644 }
645
646 c->buffer->last.mem += n;
647 ev->log->handler = ngx_http_log_error;
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000648 ctx->action = "reading client request line";
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000649
650 return ngx_http_init_request(ev);
651}
652
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000653
Igor Sysoev42feecb2002-12-15 06:25:09 +0000654static int ngx_http_set_lingering_close(ngx_http_request_t *r)
Igor Sysoev1af7c822002-09-13 14:47:42 +0000655{
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000656 ngx_http_core_loc_conf_t *lcf;
657
658 lcf = (ngx_http_core_loc_conf_t *)
659 ngx_http_get_module_loc_conf(r, ngx_http_core_module_ctx);
660
661 r->lingering_time = ngx_time() + lcf->lingering_time;
Igor Sysoev42feecb2002-12-15 06:25:09 +0000662 r->connection->read->event_handler = ngx_http_lingering_close_handler;
663
664 ngx_del_timer(r->connection->read);
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000665 ngx_add_timer(r->connection->read, lcf->lingering_timeout);
Igor Sysoev42feecb2002-12-15 06:25:09 +0000666
Igor Sysoev0d2bda52002-12-24 07:09:57 +0000667 if (r->connection->read->blocked) {
668 if (ngx_add_event(r->connection->read, NGX_READ_EVENT,
Igor Sysoev42feecb2002-12-15 06:25:09 +0000669#if (HAVE_CLEAR_EVENT)
Igor Sysoev0d2bda52002-12-24 07:09:57 +0000670 NGX_CLEAR_EVENT) == NGX_ERROR)
Igor Sysoev42feecb2002-12-15 06:25:09 +0000671#else
Igor Sysoev0d2bda52002-12-24 07:09:57 +0000672 NGX_ONESHOT_EVENT) == NGX_ERROR)
Igor Sysoev42feecb2002-12-15 06:25:09 +0000673#endif
Igor Sysoev0d2bda52002-12-24 07:09:57 +0000674 {
675 return ngx_http_close_request(r);
676 }
Igor Sysoev42feecb2002-12-15 06:25:09 +0000677 }
678
679 if (ngx_shutdown_socket(r->connection->fd, NGX_WRITE_SHUTDOWN) == -1)
680 {
681 ngx_log_error(NGX_LOG_ERR, r->connection->log, ngx_socket_errno,
682 ngx_shutdown_socket_n " failed");
683 return ngx_http_close_request(r);
684 }
685
686 return NGX_OK;
687}
688
689
690static int ngx_http_lingering_close_handler(ngx_event_t *ev)
691{
692 ssize_t n;
693 ngx_msec_t timer;
Igor Sysoev1af7c822002-09-13 14:47:42 +0000694 ngx_connection_t *c;
695 ngx_http_request_t *r;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000696 ngx_http_core_loc_conf_t *lcf;
Igor Sysoev1af7c822002-09-13 14:47:42 +0000697
Igor Sysoev42feecb2002-12-15 06:25:09 +0000698 ngx_log_debug(ev->log, "http lingering close handler");
Igor Sysoev1af7c822002-09-13 14:47:42 +0000699
Igor Sysoev0d2bda52002-12-24 07:09:57 +0000700 if (ev->timedout) {
701 return ngx_http_close_request(r);
702 }
Igor Sysoev1af7c822002-09-13 14:47:42 +0000703
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000704 c = (ngx_connection_t *) ev->data;
705 r = (ngx_http_request_t *) c->data;
706
Igor Sysoev1af7c822002-09-13 14:47:42 +0000707 timer = r->lingering_time - ngx_time();
Igor Sysoev0d2bda52002-12-24 07:09:57 +0000708 if (timer <= 0) {
709 return ngx_http_close_request(r);
710 }
Igor Sysoev1af7c822002-09-13 14:47:42 +0000711
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000712 lcf = (ngx_http_core_loc_conf_t *)
713 ngx_http_get_module_loc_conf(r, ngx_http_core_module_ctx);
714
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000715 if (r->discarded_buffer == NULL) {
716 if (r->header_in->end - r->header_in->last.mem
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000717 >= lcf->discarded_buffer_size) {
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000718 r->discarded_buffer = r->header_in->last.mem;
719
720 } else {
721 ngx_test_null(r->discarded_buffer,
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000722 ngx_palloc(c->pool, lcf->discarded_buffer_size),
Igor Sysoev0d2bda52002-12-24 07:09:57 +0000723 ngx_http_close_request(r));
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000724 }
725 }
Igor Sysoev1af7c822002-09-13 14:47:42 +0000726
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000727 n = ngx_event_recv(c, r->discarded_buffer, lcf->discarded_buffer_size);
Igor Sysoev1af7c822002-09-13 14:47:42 +0000728
Igor Sysoev0d2bda52002-12-24 07:09:57 +0000729 ngx_log_debug(ev->log, "lingering read: %d" _ n);
Igor Sysoev1af7c822002-09-13 14:47:42 +0000730
Igor Sysoev0d2bda52002-12-24 07:09:57 +0000731 if (n == NGX_ERROR || n == 0) {
732 return ngx_http_close_request(r);
733 }
Igor Sysoev1af7c822002-09-13 14:47:42 +0000734
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000735 timer *= 1000;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000736 if (timer > lcf->lingering_timeout) {
737 timer = lcf->lingering_timeout;
Igor Sysoev0d2bda52002-12-24 07:09:57 +0000738 }
Igor Sysoev1af7c822002-09-13 14:47:42 +0000739
740 ngx_del_timer(ev);
Igor Sysoev682bf8e2002-09-16 15:01:44 +0000741 ngx_add_timer(ev, timer);
Igor Sysoev1af7c822002-09-13 14:47:42 +0000742
743 return NGX_OK;
744}
745
Igor Sysoev2b542382002-08-20 14:48:28 +0000746
Igor Sysoev2ba1ee02002-10-04 17:58:04 +0000747static int ngx_http_close_connection(ngx_event_t *ev)
748{
749 int i, len;
750 char **requests, *requests_line, *prev, *new;
751 ngx_connection_t *c = (ngx_connection_t *) ev->data;
752
753 if (c->requests->nelts > 1) {
754 len = c->requests_len + c->requests->nelts * 2 - 1;
755
756 ngx_test_null(requests_line, ngx_palloc(c->pool, len),
757 ngx_event_close_connection(ev));
758
759 requests = (char **) c->requests->elts;
760 prev = requests_line;
761 new = ngx_cpystrn(prev, requests[0], len);
762 len -= new - prev;
763 prev = new;
764
765 for (i = 1; i < c->requests->nelts; i++) {
766 new = ngx_cpystrn(prev, ", ", len);
767 new = ngx_cpystrn(new, requests[i], len);
768 len -= new - prev;
769 prev = new;
770 }
771
772 } else {
773 requests_line = * (char **) c->requests->elts;
774 }
775
776 ngx_log_error(NGX_LOG_INFO, c->log, 0,
777 "REQUESTS: %d, '%s'", c->requests->nelts, requests_line);
778
779 return ngx_event_close_connection(ev);
780}
781
782
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000783static size_t ngx_http_log_error(void *data, char *buf, size_t len)
784{
785 ngx_http_log_ctx_t *ctx = (ngx_http_log_ctx_t *) data;
Igor Sysoeva58e3ca2002-09-02 14:48:24 +0000786
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000787 if (ctx->url)
788 return ngx_snprintf(buf, len, " while %s, client: %s, URL: %s",
789 ctx->action, ctx->client, ctx->url);
790 else
791 return ngx_snprintf(buf, len, " while %s, client: %s",
792 ctx->action, ctx->client);
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000793}