The two commits
5162e3b0ee7bd1d0fd6e75e1ca7993a1834b5291
"allow request handlers to disable chunked reponses"
and
618493e378e2239f0d30902e47adfa134e649fdc
"file: disable chunked encoding for file responses"
broke the chunked transfer encoding handling for proc responses in keep-alive
connections that followed a file response with http status 204 or 304.
The effect of this bug is that cgi responses following a 204 or 304 one where
sent neither in chunked encoding nor with a content-length header, causing
browsers to stall until the keep alive timeout was reached.
Fix the logic flaw by inverting the chunk prevention flag in the client state
and by testing the chunked encoding preconditions every time instead of
once upon client (re-)initialization.
Signed-off-by: Jo-Philipp Wich <jow@openwrt.org>
cl->http_code = code;
- if (!cl->request.respond_chunked)
+ if (!uh_use_chunked(cl))
enc = "";
if (r->connection_close)
!conf.http_keepalive)
req->connection_close = true;
- req->respond_chunked = uh_use_chunked(cl);
-
return CLIENT_STATE_HEADER;
}
else if ((ptr = strchr(host, ':')) != NULL)
*ptr = 0;
- cl->request.respond_chunked = false;
+ cl->request.disable_chunked = true;
cl->request.connection_close = true;
uh_http_header(cl, 307, "Temporary Redirect");
if (fd < 0)
goto error;
- req->respond_chunked = false;
+ req->disable_chunked = true;
cl->dispatch.file.hdr = tb;
uh_file_data(cl, pi, fd);
cl->dispatch.file.hdr = NULL;
int content_length;
bool expect_cont;
bool connection_close;
- bool respond_chunked;
+ bool disable_chunked;
uint8_t transfer_chunked;
const struct auth_realm *realm;
};
if (cl->http_code == 204 || cl->http_code == 304)
return false;
- return true;
+ return !cl->request.disable_chunked;
}
void uh_chunk_write(struct client *cl, const void *data, int len)
{
- bool chunked = cl->request.respond_chunked;
+ bool chunked = uh_use_chunked(cl);
if (cl->state == CLIENT_STATE_CLEANUP)
return;
return;
uloop_timeout_set(&cl->timeout, conf.network_timeout * 1000);
- if (!cl->request.respond_chunked) {
+ if (!uh_use_chunked(cl)) {
ustream_vprintf(cl->us, format, arg);
return;
}
void uh_chunk_eof(struct client *cl)
{
- if (!cl->request.respond_chunked)
+ if (!uh_use_chunked(cl))
return;
if (cl->state == CLIENT_STATE_CLEANUP)