mirror of
http://git.haproxy.org/git/haproxy.git/
synced 2024-12-15 07:54:33 +00:00
[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:
parent
a6eebb372d
commit
4fe4190278
@ -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
|
||||||
|
Loading…
Reference in New Issue
Block a user