Unverified Commit 0497297c authored by Kenton Varda's avatar Kenton Varda Committed by GitHub

Merge pull request #912 from capnproto/kenton/http-over-capnp-set-content-length

Override Content-Length and Transfer-Encoding in http-over-capnp.
parents b2fe9c54 0123e171
...@@ -236,8 +236,9 @@ class HttpOverCapnpFactory::ClientRequestContextImpl final ...@@ -236,8 +236,9 @@ class HttpOverCapnpFactory::ClientRequestContextImpl final
public: public:
ClientRequestContextImpl(HttpOverCapnpFactory& factory, ClientRequestContextImpl(HttpOverCapnpFactory& factory,
kj::Own<RequestState> state, kj::Own<RequestState> state,
kj::HttpMethod method,
kj::HttpService::Response& kjResponse) kj::HttpService::Response& kjResponse)
: factory(factory), state(kj::mv(state)), kjResponse(kjResponse) {} : factory(factory), state(kj::mv(state)), method(method), kjResponse(kjResponse) {}
~ClientRequestContextImpl() noexcept(false) { ~ClientRequestContextImpl() noexcept(false) {
// Note this implicitly cancels the upstream pump task. // Note this implicitly cancels the upstream pump task.
...@@ -260,8 +261,25 @@ public: ...@@ -260,8 +261,25 @@ public:
hasBody = size > 0; hasBody = size > 0;
} }
auto headers = factory.headersToKj(rpcResponse.getHeaders());
// Some apps rely on content-length and transfer-encoding actually reflecting the underlying
// stream, so overwrite them here to reflect what the RPC layer says.
if (method == kj::HttpMethod::HEAD) {
// On a HEAD response, there is no body, but the server is allowed to send arbitrary
// values for content-length and transfer-encoding.
} else KJ_IF_MAYBE(s, expectedSize) {
// Known body size, set content-length.
headers.set(kj::HttpHeaderId::CONTENT_LENGTH, kj::str(*s));
headers.unset(kj::HttpHeaderId::TRANSFER_ENCODING);
} else {
// Unknown body size, set transfer-encoding: chunked.
headers.set(kj::HttpHeaderId::TRANSFER_ENCODING, "chunked"_kj);
headers.unset(kj::HttpHeaderId::CONTENT_LENGTH);
}
auto bodyStream = kjResponse.send(rpcResponse.getStatusCode(), rpcResponse.getStatusText(), auto bodyStream = kjResponse.send(rpcResponse.getStatusCode(), rpcResponse.getStatusText(),
factory.headersToKj(rpcResponse.getHeaders()), expectedSize); headers, expectedSize);
auto results = context.getResults(MessageSize { 16, 1 }); auto results = context.getResults(MessageSize { 16, 1 });
if (hasBody) { if (hasBody) {
...@@ -309,6 +327,7 @@ public: ...@@ -309,6 +327,7 @@ public:
private: private:
HttpOverCapnpFactory& factory; HttpOverCapnpFactory& factory;
kj::Own<RequestState> state; kj::Own<RequestState> state;
kj::HttpMethod method;
bool sent = false; bool sent = false;
kj::HttpService::Response& kjResponse; kj::HttpService::Response& kjResponse;
...@@ -355,7 +374,7 @@ public: ...@@ -355,7 +374,7 @@ public:
}); });
rpcRequest.setContext( rpcRequest.setContext(
kj::heap<ClientRequestContextImpl>(factory, kj::addRef(*state), kjResponse)); kj::heap<ClientRequestContextImpl>(factory, kj::addRef(*state), method, kjResponse));
auto pipeline = rpcRequest.send(); auto pipeline = rpcRequest.send();
...@@ -444,13 +463,14 @@ class HttpOverCapnpFactory::ServerRequestContextImpl final ...@@ -444,13 +463,14 @@ class HttpOverCapnpFactory::ServerRequestContextImpl final
public: public:
ServerRequestContextImpl(HttpOverCapnpFactory& factory, ServerRequestContextImpl(HttpOverCapnpFactory& factory,
capnp::HttpRequest::Reader request, capnp::HttpRequest::Reader request,
kj::HttpHeaders&& headersParam,
capnp::HttpService::ClientRequestContext::Client clientContext, capnp::HttpService::ClientRequestContext::Client clientContext,
kj::Own<kj::AsyncInputStream> requestBodyIn, kj::Own<kj::AsyncInputStream> requestBodyIn,
kj::HttpService& kjService) kj::HttpService& kjService)
: factory(factory), : factory(factory),
method(validateMethod(request.getMethod())), method(validateMethod(request.getMethod())),
url(kj::str(request.getUrl())), url(kj::str(request.getUrl())),
headers(factory.headersToKj(request.getHeaders()).clone()), headers(kj::mv(headersParam)),
clientContext(kj::mv(clientContext)), clientContext(kj::mv(clientContext)),
// Note we attach `requestBodyIn` to `task` so that we will implicitly cancel reading // Note we attach `requestBodyIn` to `task` so that we will implicitly cancel reading
// the request body as soon as the service returns. This is important in particular when // the request body as soon as the service returns. This is important in particular when
...@@ -584,8 +604,27 @@ public: ...@@ -584,8 +604,27 @@ public:
} else { } else {
requestBody = kj::heap<NullInputStream>(); requestBody = kj::heap<NullInputStream>();
} }
// Some apps rely on content-length and transfer-encoding actually reflecting the underlying
// stream, so overwrite them here to reflect what the RPC layer says.
kj::HttpHeaders headers = factory.headersToKj(metadata.getHeaders()).clone();
auto method = metadata.getMethod();
if ((method == HttpMethod::HEAD || method == HttpMethod::GET) && !hasBody) {
// When a GET or HEAD request has no body, we set neither header.
headers.unset(kj::HttpHeaderId::CONTENT_LENGTH);
headers.unset(kj::HttpHeaderId::TRANSFER_ENCODING);
} else KJ_IF_MAYBE(s, expectedSize) {
// Known body size, set content-length.
headers.set(kj::HttpHeaderId::CONTENT_LENGTH, kj::str(*s));
headers.unset(kj::HttpHeaderId::TRANSFER_ENCODING);
} else {
// Unknown body size, set transfer-encoding: chunked.
headers.set(kj::HttpHeaderId::TRANSFER_ENCODING, "chunked"_kj);
headers.unset(kj::HttpHeaderId::CONTENT_LENGTH);
}
results.setContext(kj::heap<ServerRequestContextImpl>( results.setContext(kj::heap<ServerRequestContextImpl>(
factory, metadata, params.getContext(), kj::mv(requestBody), *inner)); factory, metadata, kj::mv(headers), params.getContext(), kj::mv(requestBody), *inner));
return kj::READY_NOW; return kj::READY_NOW;
} }
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment