blob: bfdc8e4430be356b7690bbdd13b44ade92bfb834 [file] [log] [blame]
Igor Sysoeva09f08d2003-04-25 14:43:13 +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 Sysoev1d8d9ee2003-04-28 15:06:39 +00007#include <ngx_config.h>
8#include <ngx_core.h>
Igor Sysoev10fc9ef2003-10-27 08:53:49 +00009#include <ngx_event.h>
Igor Sysoev1d8d9ee2003-04-28 15:06:39 +000010#include <ngx_http.h>
Igor Sysoeva09f08d2003-04-25 14:43:13 +000011
12
Igor Sysoev899b44e2005-05-12 14:58:06 +000013static void ngx_http_read_client_request_body_handler(ngx_http_request_t *r);
14static ngx_int_t ngx_http_do_read_client_request_body(ngx_http_request_t *r);
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +000015static ngx_int_t ngx_http_write_request_body(ngx_http_request_t *r,
16 ngx_chain_t *body);
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +000017static void ngx_http_read_discarded_body_handler(ngx_http_request_t *r);
18static ngx_int_t ngx_http_read_discarded_body(ngx_http_request_t *r);
19
Igor Sysoev10fc9ef2003-10-27 08:53:49 +000020
Igor Sysoev02025fd2005-01-18 13:03:58 +000021/*
22 * on completion ngx_http_read_client_request_body() adds to
23 * r->request_body->bufs one or two bufs:
24 * *) one memory buf that was preread in r->header_in;
25 * *) one memory or file buf that contains the rest of the body
26 */
Igor Sysoev10fc9ef2003-10-27 08:53:49 +000027
Igor Sysoevc1571722005-03-19 12:38:37 +000028ngx_int_t
29ngx_http_read_client_request_body(ngx_http_request_t *r,
30 ngx_http_client_body_handler_pt post_handler)
Igor Sysoeva09f08d2003-04-25 14:43:13 +000031{
Igor Sysoev1765f472006-07-07 16:33:19 +000032 size_t preread;
33 ssize_t size;
Igor Sysoev369145c2004-05-28 15:49:23 +000034 ngx_buf_t *b;
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +000035 ngx_chain_t *cl, **next;
Igor Sysoev4c7f5112006-10-01 07:17:01 +000036 ngx_temp_file_t *tf;
Igor Sysoev02025fd2005-01-18 13:03:58 +000037 ngx_http_request_body_t *rb;
Igor Sysoevdbb27762004-04-01 16:20:53 +000038 ngx_http_core_loc_conf_t *clcf;
Igor Sysoev11dbe972004-03-29 17:43:58 +000039
Igor Sysoeve5035392005-08-30 10:55:07 +000040 if (r->request_body || r->discard_body) {
Igor Sysoev899b44e2005-05-12 14:58:06 +000041 post_handler(r);
42 return NGX_OK;
43 }
44
Igor Sysoevc1571722005-03-19 12:38:37 +000045 rb = ngx_pcalloc(r->pool, sizeof(ngx_http_request_body_t));
46 if (rb == NULL) {
Igor Sysoev02025fd2005-01-18 13:03:58 +000047 return NGX_HTTP_INTERNAL_SERVER_ERROR;
48 }
49
50 r->request_body = rb;
51
Igor Sysoev4c7f5112006-10-01 07:17:01 +000052 if (r->headers_in.content_length_n < 0) {
53 post_handler(r);
54 return NGX_OK;
55 }
56
57 clcf = ngx_http_get_module_loc_conf(r, ngx_http_core_module);
58
59 if (r->headers_in.content_length_n == 0) {
60
61 if (r->request_body_in_file_only) {
62 tf = ngx_pcalloc(r->pool, sizeof(ngx_temp_file_t));
63 if (tf == NULL) {
64 return NGX_HTTP_INTERNAL_SERVER_ERROR;
65 }
66
67 tf->file.fd = NGX_INVALID_FILE;
68 tf->file.log = r->connection->log;
69 tf->path = clcf->client_body_temp_path;
70 tf->pool = r->pool;
71 tf->warn = "a client request body is buffered to a temporary file";
72 tf->log_level = r->request_body_file_log_level;
73 tf->persistent = r->request_body_in_persistent_file;
Igor Sysoevcd5b99a2007-01-25 08:45:04 +000074 tf->clean = r->request_body_in_clean_file;
Igor Sysoev4c7f5112006-10-01 07:17:01 +000075
76 if (r->request_body_file_group_access) {
Igor Sysoevfe1cb8c2007-01-18 19:52:18 +000077 tf->access = 0660;
Igor Sysoev4c7f5112006-10-01 07:17:01 +000078 }
79
80 rb->temp_file = tf;
81
82 if (ngx_create_temp_file(&tf->file, tf->path, tf->pool,
Igor Sysoevcd5b99a2007-01-25 08:45:04 +000083 tf->persistent, tf->clean, tf->access)
Igor Sysoev4c7f5112006-10-01 07:17:01 +000084 != NGX_OK)
85 {
86 return NGX_HTTP_INTERNAL_SERVER_ERROR;
87 }
88 }
89
Igor Sysoev02025fd2005-01-18 13:03:58 +000090 post_handler(r);
91 return NGX_OK;
92 }
93
94 rb->post_handler = post_handler;
95
96 /*
97 * set by ngx_pcalloc():
98 *
99 * rb->bufs = NULL;
100 * rb->buf = NULL;
101 * rb->rest = 0;
102 */
103
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000104 preread = r->header_in->last - r->header_in->pos;
Igor Sysoeva09f08d2003-04-25 14:43:13 +0000105
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000106 if (preread) {
Igor Sysoev14f02ed2004-03-26 16:13:01 +0000107
108 /* there is the pre-read part of the request body */
109
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000110 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
111 "http client request body preread %uz", preread);
112
Igor Sysoevc1571722005-03-19 12:38:37 +0000113 b = ngx_calloc_buf(r->pool);
114 if (b == NULL) {
Igor Sysoev02025fd2005-01-18 13:03:58 +0000115 return NGX_HTTP_INTERNAL_SERVER_ERROR;
116 }
Igor Sysoeva09f08d2003-04-25 14:43:13 +0000117
Igor Sysoev369145c2004-05-28 15:49:23 +0000118 b->temporary = 1;
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000119 b->start = r->header_in->pos;
120 b->pos = r->header_in->pos;
121 b->last = r->header_in->last;
122 b->end = r->header_in->end;
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000123
Igor Sysoevc1571722005-03-19 12:38:37 +0000124 rb->bufs = ngx_alloc_chain_link(r->pool);
125 if (rb->bufs == NULL) {
Igor Sysoev02025fd2005-01-18 13:03:58 +0000126 return NGX_HTTP_INTERNAL_SERVER_ERROR;
127 }
128
129 rb->bufs->buf = b;
130 rb->bufs->next = NULL;
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000131
Igor Sysoev8f125582006-07-28 15:16:17 +0000132 rb->buf = b;
133
Igor Sysoevbb28b6d2006-07-11 13:20:19 +0000134 if ((off_t) preread >= r->headers_in.content_length_n) {
Igor Sysoev2b0c76c2003-10-27 21:01:00 +0000135
Igor Sysoev14f02ed2004-03-26 16:13:01 +0000136 /* the whole request body was pre-read */
137
Igor Sysoev1765f472006-07-07 16:33:19 +0000138 r->header_in->pos += (size_t) r->headers_in.content_length_n;
Igor Sysoev6a12fc92004-12-06 14:45:08 +0000139 r->request_length += r->headers_in.content_length_n;
Igor Sysoev6707ba92004-03-30 15:59:50 +0000140
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000141 if (r->request_body_in_file_only) {
142 if (ngx_http_write_request_body(r, rb->bufs) != NGX_OK) {
143 return NGX_HTTP_INTERNAL_SERVER_ERROR;
144 }
145 }
146
Igor Sysoev02025fd2005-01-18 13:03:58 +0000147 post_handler(r);
Igor Sysoev6707ba92004-03-30 15:59:50 +0000148
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000149 return NGX_OK;
150 }
Igor Sysoev2b0c76c2003-10-27 21:01:00 +0000151
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000152 /*
153 * to not consider the body as pipelined request in
154 * ngx_http_set_keepalive()
155 */
Igor Sysoev2b0c76c2003-10-27 21:01:00 +0000156 r->header_in->pos = r->header_in->last;
Igor Sysoev1d8d9ee2003-04-28 15:06:39 +0000157
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000158 r->request_length += preread;
159
160 rb->rest = r->headers_in.content_length_n - preread;
161
Igor Sysoevbb28b6d2006-07-11 13:20:19 +0000162 if (rb->rest <= (off_t) (b->end - b->last)) {
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000163
164 /* the whole request body may be placed in r->header_in */
165
Igor Sysoev36588962007-01-19 16:13:15 +0000166 rb->to_write = rb->bufs;
167
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000168 r->read_event_handler = ngx_http_read_client_request_body_handler;
169
Igor Sysoevcd5b99a2007-01-25 08:45:04 +0000170 return ngx_http_do_read_client_request_body(r);
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000171 }
172
173 next = &rb->bufs->next;
174
175 } else {
176 b = NULL;
177 rb->rest = r->headers_in.content_length_n;
178 next = &rb->bufs;
179 }
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000180
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000181 size = clcf->client_body_buffer_size;
182 size += size >> 2;
Igor Sysoev14f02ed2004-03-26 16:13:01 +0000183
Igor Sysoev1765f472006-07-07 16:33:19 +0000184 if (rb->rest < size) {
185 size = (ssize_t) rb->rest;
Igor Sysoeva09f08d2003-04-25 14:43:13 +0000186
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000187 if (r->request_body_in_single_buf) {
188 size += preread;
189 }
190
Igor Sysoeva09f08d2003-04-25 14:43:13 +0000191 } else {
Igor Sysoevdbb27762004-04-01 16:20:53 +0000192 size = clcf->client_body_buffer_size;
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000193
194 /* disable copying buffer for r->request_body_in_single_buf */
195 b = NULL;
Igor Sysoeva09f08d2003-04-25 14:43:13 +0000196 }
197
Igor Sysoevc1571722005-03-19 12:38:37 +0000198 rb->buf = ngx_create_temp_buf(r->pool, size);
199 if (rb->buf == NULL) {
Igor Sysoev02025fd2005-01-18 13:03:58 +0000200 return NGX_HTTP_INTERNAL_SERVER_ERROR;
201 }
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000202
Igor Sysoevc1571722005-03-19 12:38:37 +0000203 cl = ngx_alloc_chain_link(r->pool);
204 if (cl == NULL) {
Igor Sysoev02025fd2005-01-18 13:03:58 +0000205 return NGX_HTTP_INTERNAL_SERVER_ERROR;
206 }
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000207
Igor Sysoev02025fd2005-01-18 13:03:58 +0000208 cl->buf = rb->buf;
209 cl->next = NULL;
210
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000211 if (b && r->request_body_in_single_buf) {
212 size = b->last - b->pos;
213 ngx_memcpy(rb->buf->pos, b->pos, size);
214 rb->buf->last += size;
215
216 next = &rb->bufs;
217 }
218
219 *next = cl;
220
221 if (r->request_body_in_file_only || r->request_body_in_single_buf) {
222 rb->to_write = rb->bufs;
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000223
224 } else {
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000225 rb->to_write = rb->bufs->next ? rb->bufs->next : rb->bufs;
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000226 }
227
Igor Sysoev899b44e2005-05-12 14:58:06 +0000228 r->read_event_handler = ngx_http_read_client_request_body_handler;
Igor Sysoeva09f08d2003-04-25 14:43:13 +0000229
Igor Sysoevcd5b99a2007-01-25 08:45:04 +0000230 return ngx_http_do_read_client_request_body(r);
Igor Sysoeva09f08d2003-04-25 14:43:13 +0000231}
232
233
Igor Sysoevc1571722005-03-19 12:38:37 +0000234static void
Igor Sysoev899b44e2005-05-12 14:58:06 +0000235ngx_http_read_client_request_body_handler(ngx_http_request_t *r)
Igor Sysoeva09f08d2003-04-25 14:43:13 +0000236{
Igor Sysoev899b44e2005-05-12 14:58:06 +0000237 ngx_int_t rc;
Igor Sysoevae02c192004-03-19 05:25:53 +0000238
Igor Sysoev899b44e2005-05-12 14:58:06 +0000239 if (r->connection->read->timedout) {
Igor Sysoev0e5dc5c2005-11-15 13:30:52 +0000240 r->connection->timedout = 1;
Igor Sysoevcd5b99a2007-01-25 08:45:04 +0000241 ngx_http_finalize_request(r, NGX_HTTP_REQUEST_TIME_OUT);
Igor Sysoev6707ba92004-03-30 15:59:50 +0000242 return;
243 }
244
Igor Sysoev899b44e2005-05-12 14:58:06 +0000245 rc = ngx_http_do_read_client_request_body(r);
Igor Sysoevae02c192004-03-19 05:25:53 +0000246
247 if (rc >= NGX_HTTP_SPECIAL_RESPONSE) {
Igor Sysoevcd5b99a2007-01-25 08:45:04 +0000248 ngx_http_finalize_request(r, rc);
Igor Sysoevae02c192004-03-19 05:25:53 +0000249 }
250}
251
252
Igor Sysoevc1571722005-03-19 12:38:37 +0000253static ngx_int_t
Igor Sysoev899b44e2005-05-12 14:58:06 +0000254ngx_http_do_read_client_request_body(ngx_http_request_t *r)
Igor Sysoevae02c192004-03-19 05:25:53 +0000255{
Igor Sysoev10a543a2004-03-16 07:10:12 +0000256 size_t size;
257 ssize_t n;
Igor Sysoev369145c2004-05-28 15:49:23 +0000258 ngx_buf_t *b;
Igor Sysoev899b44e2005-05-12 14:58:06 +0000259 ngx_connection_t *c;
Igor Sysoev02025fd2005-01-18 13:03:58 +0000260 ngx_http_request_body_t *rb;
Igor Sysoev2b0c76c2003-10-27 21:01:00 +0000261 ngx_http_core_loc_conf_t *clcf;
Igor Sysoev1d8d9ee2003-04-28 15:06:39 +0000262
Igor Sysoev899b44e2005-05-12 14:58:06 +0000263 c = r->connection;
Igor Sysoev02025fd2005-01-18 13:03:58 +0000264 rb = r->request_body;
Igor Sysoeva09f08d2003-04-25 14:43:13 +0000265
Igor Sysoevda85f7f2004-03-16 21:26:01 +0000266 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0,
267 "http read client request body");
268
269 for ( ;; ) {
Igor Sysoev12c94ae2006-11-13 20:40:17 +0000270 for ( ;; ) {
271 if (rb->buf->last == rb->buf->end) {
Igor Sysoev02025fd2005-01-18 13:03:58 +0000272
Igor Sysoev12c94ae2006-11-13 20:40:17 +0000273 if (ngx_http_write_request_body(r, rb->to_write) != NGX_OK) {
274 return NGX_HTTP_INTERNAL_SERVER_ERROR;
275 }
276
277 rb->to_write = rb->bufs->next ? rb->bufs->next : rb->bufs;
278 rb->buf->last = rb->buf->start;
Igor Sysoevda85f7f2004-03-16 21:26:01 +0000279 }
280
Igor Sysoev12c94ae2006-11-13 20:40:17 +0000281 size = rb->buf->end - rb->buf->last;
282
283 if ((off_t) size > rb->rest) {
284 size = (size_t) rb->rest;
285 }
286
287 n = c->recv(c, rb->buf->last, size);
288
289 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, c->log, 0,
290 "http client request body recv %z", n);
291
292 if (n == NGX_AGAIN) {
293 break;
294 }
295
296 if (n == 0) {
297 ngx_log_error(NGX_LOG_INFO, c->log, 0,
298 "client closed prematurely connection");
299 }
300
301 if (n == 0 || n == NGX_ERROR) {
302 c->error = 1;
303 return NGX_HTTP_BAD_REQUEST;
304 }
305
306 rb->buf->last += n;
307 rb->rest -= n;
308 r->request_length += n;
309
310 if (rb->rest == 0) {
311 break;
312 }
313
314 if (rb->buf->last < rb->buf->end) {
315 break;
316 }
Igor Sysoevda85f7f2004-03-16 21:26:01 +0000317 }
318
Igor Sysoevae02c192004-03-19 05:25:53 +0000319 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, c->log, 0,
Igor Sysoevcf815022007-02-15 14:13:24 +0000320 "http client request body rest %O", rb->rest);
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000321
Igor Sysoev02025fd2005-01-18 13:03:58 +0000322 if (rb->rest == 0) {
Igor Sysoev89690bf2004-03-23 06:01:52 +0000323 break;
324 }
325
Igor Sysoev12c94ae2006-11-13 20:40:17 +0000326 if (!c->read->ready) {
327 clcf = ngx_http_get_module_loc_conf(r, ngx_http_core_module);
328 ngx_add_timer(c->read, clcf->client_body_timeout);
329
330 if (ngx_handle_read_event(c->read, 0) == NGX_ERROR) {
331 return NGX_HTTP_INTERNAL_SERVER_ERROR;
332 }
333
334 return NGX_AGAIN;
Igor Sysoevda85f7f2004-03-16 21:26:01 +0000335 }
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000336 }
337
Igor Sysoev09c684b2005-11-09 17:25:55 +0000338 if (c->read->timer_set) {
339 ngx_del_timer(c->read);
340 }
341
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000342 if (rb->temp_file || r->request_body_in_file_only) {
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000343
344 /* save the last part */
Igor Sysoev02025fd2005-01-18 13:03:58 +0000345
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000346 if (ngx_http_write_request_body(r, rb->to_write) != NGX_OK) {
Igor Sysoevae02c192004-03-19 05:25:53 +0000347 return NGX_HTTP_INTERNAL_SERVER_ERROR;
Igor Sysoev2b0c76c2003-10-27 21:01:00 +0000348 }
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000349
Igor Sysoevc1571722005-03-19 12:38:37 +0000350 b = ngx_calloc_buf(r->pool);
351 if (b == NULL) {
Igor Sysoevae02c192004-03-19 05:25:53 +0000352 return NGX_HTTP_INTERNAL_SERVER_ERROR;
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000353 }
354
Igor Sysoev369145c2004-05-28 15:49:23 +0000355 b->in_file = 1;
356 b->file_pos = 0;
Igor Sysoev02025fd2005-01-18 13:03:58 +0000357 b->file_last = rb->temp_file->file.offset;
358 b->file = &rb->temp_file->file;
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000359
Igor Sysoev02025fd2005-01-18 13:03:58 +0000360 if (rb->bufs->next) {
361 rb->bufs->next->buf = b;
Igor Sysoeva09f08d2003-04-25 14:43:13 +0000362
363 } else {
Igor Sysoev02025fd2005-01-18 13:03:58 +0000364 rb->bufs->buf = b;
Igor Sysoeva09f08d2003-04-25 14:43:13 +0000365 }
Igor Sysoeva09f08d2003-04-25 14:43:13 +0000366 }
367
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000368 if (r->request_body_in_file_only && rb->bufs->next) {
369 rb->bufs = rb->bufs->next;
370 }
371
Igor Sysoev02025fd2005-01-18 13:03:58 +0000372 rb->post_handler(r);
Igor Sysoevae02c192004-03-19 05:25:53 +0000373
374 return NGX_OK;
Igor Sysoeva09f08d2003-04-25 14:43:13 +0000375}
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000376
377
378static ngx_int_t
379ngx_http_write_request_body(ngx_http_request_t *r, ngx_chain_t *body)
380{
381 ssize_t n;
382 ngx_temp_file_t *tf;
383 ngx_http_request_body_t *rb;
384 ngx_http_core_loc_conf_t *clcf;
385
386 rb = r->request_body;
387
388 if (rb->temp_file == NULL) {
389 tf = ngx_pcalloc(r->pool, sizeof(ngx_temp_file_t));
390 if (tf == NULL) {
391 return NGX_ERROR;
392 }
393
394 clcf = ngx_http_get_module_loc_conf(r, ngx_http_core_module);
395
396 tf->file.fd = NGX_INVALID_FILE;
397 tf->file.log = r->connection->log;
398 tf->path = clcf->client_body_temp_path;
399 tf->pool = r->pool;
400 tf->warn = "a client request body is buffered to a temporary file";
Igor Sysoev7bdb7202006-04-19 15:30:56 +0000401 tf->log_level = r->request_body_file_log_level;
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000402 tf->persistent = r->request_body_in_persistent_file;
Igor Sysoevcd5b99a2007-01-25 08:45:04 +0000403 tf->clean = r->request_body_in_clean_file;
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000404
405 if (r->request_body_file_group_access) {
Igor Sysoevfe1cb8c2007-01-18 19:52:18 +0000406 tf->access = 0660;
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000407 }
408
409 rb->temp_file = tf;
410 }
411
412 n = ngx_write_chain_to_temp_file(rb->temp_file, body);
413
414 /* TODO: n == 0 or not complete and level event */
415
416 if (n == NGX_ERROR) {
417 return NGX_ERROR;
418 }
419
420 rb->temp_file->offset += n;
421
422 return NGX_OK;
423}
424
425
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000426ngx_int_t
427ngx_http_discard_body(ngx_http_request_t *r)
428{
429 ssize_t size;
430 ngx_event_t *rev;
431
Igor Sysoev08e63d42006-08-14 15:09:38 +0000432 if (r != r->main || r->discard_body) {
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000433 return NGX_OK;
434 }
435
436 rev = r->connection->read;
437
438 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, rev->log, 0, "http set discard body");
439
440 if (rev->timer_set) {
441 ngx_del_timer(rev);
442 }
443
Igor Sysoevda173ab2006-08-30 10:39:17 +0000444 r->discard_body = 1;
445
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000446 if (r->headers_in.content_length_n <= 0) {
447 return NGX_OK;
448 }
449
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000450 size = r->header_in->last - r->header_in->pos;
451
452 if (size) {
453 if (r->headers_in.content_length_n > size) {
454 r->headers_in.content_length_n -= size;
455
456 } else {
Igor Sysoev1765f472006-07-07 16:33:19 +0000457 r->header_in->pos += (size_t) r->headers_in.content_length_n;
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000458 r->headers_in.content_length_n = 0;
459 return NGX_OK;
460 }
461 }
462
463 r->read_event_handler = ngx_http_read_discarded_body_handler;
464
465 if (ngx_handle_read_event(rev, 0) == NGX_ERROR) {
466 return NGX_HTTP_INTERNAL_SERVER_ERROR;
467 }
468
469 return ngx_http_read_discarded_body(r);
470}
471
472
473static void
474ngx_http_read_discarded_body_handler(ngx_http_request_t *r)
475{
476 ngx_int_t rc;
477
478 rc = ngx_http_read_discarded_body(r);
479
480 if (rc == NGX_AGAIN) {
481 if (ngx_handle_read_event(r->connection->read, 0) == NGX_ERROR) {
482 ngx_http_finalize_request(r, rc);
483 return;
484 }
485 }
486
487 if (rc != NGX_OK) {
488 ngx_http_finalize_request(r, rc);
489 }
490}
491
492
493static ngx_int_t
494ngx_http_read_discarded_body(ngx_http_request_t *r)
495{
Igor Sysoev1765f472006-07-07 16:33:19 +0000496 size_t size;
497 ssize_t n;
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000498 u_char buffer[NGX_HTTP_DISCARD_BUFFER_SIZE];
499
500 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
501 "http read discarded body");
502
503 if (r->headers_in.content_length_n == 0) {
504 return NGX_OK;
505 }
506
Igor Sysoev1765f472006-07-07 16:33:19 +0000507 size = (r->headers_in.content_length_n > NGX_HTTP_DISCARD_BUFFER_SIZE) ?
508 NGX_HTTP_DISCARD_BUFFER_SIZE:
509 (size_t) r->headers_in.content_length_n;
Igor Sysoev8a2b2fb2006-04-14 09:53:38 +0000510
511 n = r->connection->recv(r->connection, buffer, size);
512
513 if (n == NGX_ERROR) {
514
515 r->connection->error = 1;
516
517 /*
518 * if a client request body is discarded then we already set
519 * some HTTP response code for client and we can ignore the error
520 */
521
522 return NGX_OK;
523 }
524
525 if (n == NGX_AGAIN) {
526 return NGX_AGAIN;
527 }
528
529 r->headers_in.content_length_n -= n;
530
531 return NGX_OK;
532}