[BUG] http: automatically close response if req is aborted

Latest BF_READ_ATTACHED fix has unveiled a nice issue with the way
HTTP requests and responses are forwarded. The case where the request
aborts after the response has responded (POST with early response)
forgot to re-enable auto-close on the response. In fact it still
worked thanks to a side effect as long as BF_READ_ATTACHED was there
to force the states to be resynced (and the flags). Since last fix,
the missing auto-close causes CLOSE_WAIT connections when the client
aborts too late during a data transfer.

The right fix consists in considering the situation where the client
experiences an error and to explicitly abort the transfer. There is
no need to wake the response analysers up for that since they'd have
no added value and the analysers flags are cleared. However for a
future usage, that might help (eg: stickiness, ...).

This fix should be backported to 1.4 if the previous one is backported
too. After all the non-reg tests, the risks to see a problem arise
without both patches seems low, and both patches touch sensible areas
of the code. So there's no hurry.
This commit is contained in:
Willy Tarreau 2010-06-07 22:27:41 +02:00
parent a6eebb372d
commit 4fe4190278

View File

@ -4064,6 +4064,7 @@ int http_resync_states(struct session *s)
} }
else if (txn->rsp.msg_state == HTTP_MSG_CLOSED || else if (txn->rsp.msg_state == HTTP_MSG_CLOSED ||
txn->rsp.msg_state == HTTP_MSG_ERROR || txn->rsp.msg_state == HTTP_MSG_ERROR ||
txn->req.msg_state == HTTP_MSG_ERROR ||
(s->rep->flags & BF_SHUTW)) { (s->rep->flags & BF_SHUTW)) {
s->rep->analysers = 0; s->rep->analysers = 0;
buffer_auto_close(s->rep); buffer_auto_close(s->rep);
@ -4109,15 +4110,16 @@ int http_request_forward_body(struct session *s, struct buffer *req, int an_bit)
if ((req->flags & (BF_READ_ERROR|BF_READ_TIMEOUT|BF_WRITE_ERROR|BF_WRITE_TIMEOUT)) || if ((req->flags & (BF_READ_ERROR|BF_READ_TIMEOUT|BF_WRITE_ERROR|BF_WRITE_TIMEOUT)) ||
((req->flags & BF_SHUTW) && (req->to_forward || req->send_max))) { ((req->flags & BF_SHUTW) && (req->to_forward || req->send_max))) {
/* Output closed while we were sending data. We must abort. */ /* Output closed while we were sending data. We must abort and
buffer_ignore(req, req->l - req->send_max); * wake the other side up.
buffer_auto_read(req); */
buffer_auto_close(req); msg->msg_state = HTTP_MSG_ERROR;
req->analysers &= ~an_bit; http_resync_states(s);
return 1; return 1;
} }
buffer_dont_close(req); /* in most states, we should abort in case of early close */
buffer_auto_close(req);
/* Note that we don't have to send 100-continue back because we don't /* Note that we don't have to send 100-continue back because we don't
* need the data to complete our job, and it's up to the server to * need the data to complete our job, and it's up to the server to
@ -4202,6 +4204,8 @@ int http_request_forward_body(struct session *s, struct buffer *req, int an_bit)
} }
else { else {
/* other states, DONE...TUNNEL */ /* other states, DONE...TUNNEL */
/* for keep-alive we don't want to forward closes on DONE */
buffer_dont_close(req);
if (http_resync_states(s)) { if (http_resync_states(s)) {
/* some state changes occurred, maybe the analyser /* some state changes occurred, maybe the analyser
* was disabled too. * was disabled too.
@ -5007,15 +5011,16 @@ int http_response_forward_body(struct session *s, struct buffer *res, int an_bit
if ((res->flags & (BF_READ_ERROR|BF_READ_TIMEOUT|BF_WRITE_ERROR|BF_WRITE_TIMEOUT)) || if ((res->flags & (BF_READ_ERROR|BF_READ_TIMEOUT|BF_WRITE_ERROR|BF_WRITE_TIMEOUT)) ||
((res->flags & BF_SHUTW) && (res->to_forward || res->send_max)) || ((res->flags & BF_SHUTW) && (res->to_forward || res->send_max)) ||
!s->req->analysers) { !s->req->analysers) {
/* in case of error or if the other analyser went away, we can't analyse HTTP anymore */ /* Output closed while we were sending data. We must abort and
buffer_ignore(res, res->l - res->send_max); * wake the other side up.
buffer_auto_read(res); */
buffer_auto_close(res); msg->msg_state = HTTP_MSG_ERROR;
res->analysers &= ~an_bit; http_resync_states(s);
return 1; return 1;
} }
buffer_dont_close(res); /* in most states, we should abort in case of early close */
buffer_auto_close(res);
if (msg->msg_state < HTTP_MSG_CHUNK_SIZE) { if (msg->msg_state < HTTP_MSG_CHUNK_SIZE) {
/* we have msg->col and msg->sov which both point to the first /* we have msg->col and msg->sov which both point to the first
@ -5093,6 +5098,8 @@ int http_response_forward_body(struct session *s, struct buffer *res, int an_bit
} }
else { else {
/* other states, DONE...TUNNEL */ /* other states, DONE...TUNNEL */
/* for keep-alive we don't want to forward closes on DONE */
buffer_dont_close(res);
if (http_resync_states(s)) { if (http_resync_states(s)) {
http_silent_debug(__LINE__, s); http_silent_debug(__LINE__, s);
/* some state changes occurred, maybe the analyser /* some state changes occurred, maybe the analyser