Unverified Commit de69fed2 authored by Kenton Varda's avatar Kenton Varda Committed by GitHub

Merge pull request #631 from capnproto/fix-client-reuse

Fix bug in HTTP client connection reuse.
parents c047c831 59da69a2
...@@ -2619,6 +2619,12 @@ KJ_TEST("HttpClient connection management") { ...@@ -2619,6 +2619,12 @@ KJ_TEST("HttpClient connection management") {
.wait(io.waitScope); .wait(io.waitScope);
KJ_EXPECT(count == 0); KJ_EXPECT(count == 0);
// Connections where we didn't even wait for the response headers are not reused.
doRequest().wait(io.waitScope);
KJ_EXPECT(count == 1);
client->request(HttpMethod::GET, kj::str("/foo"), HttpHeaders(headerTable));
KJ_EXPECT(count == 0);
// Connections where we failed to write the full request body are not reused. // Connections where we failed to write the full request body are not reused.
doRequest().wait(io.waitScope); doRequest().wait(io.waitScope);
KJ_EXPECT(count == 1); KJ_EXPECT(count == 1);
......
...@@ -993,7 +993,7 @@ public: ...@@ -993,7 +993,7 @@ public:
} }
bool canReuse() { bool canReuse() {
return !broken; return !broken && pendingMessageCount == 0;
} }
// --------------------------------------------------------------------------- // ---------------------------------------------------------------------------
...@@ -1005,6 +1005,7 @@ public: ...@@ -1005,6 +1005,7 @@ public:
KJ_REQUIRE_NONNULL(onMessageDone)->fulfill(); KJ_REQUIRE_NONNULL(onMessageDone)->fulfill();
onMessageDone = nullptr; onMessageDone = nullptr;
--pendingMessageCount;
} }
void abortRead() { void abortRead() {
...@@ -1066,6 +1067,7 @@ public: ...@@ -1066,6 +1067,7 @@ public:
} }
kj::Promise<kj::ArrayPtr<char>> readMessageHeaders() { kj::Promise<kj::ArrayPtr<char>> readMessageHeaders() {
++pendingMessageCount;
auto paf = kj::newPromiseAndFulfiller<void>(); auto paf = kj::newPromiseAndFulfiller<void>();
auto promise = messageReadQueue auto promise = messageReadQueue
...@@ -1194,6 +1196,9 @@ private: ...@@ -1194,6 +1196,9 @@ private:
bool broken = false; bool broken = false;
// Becomes true if the caller failed to read the whole entity-body before closing the stream. // Becomes true if the caller failed to read the whole entity-body before closing the stream.
uint pendingMessageCount = 0;
// Number of reads we have queued up.
kj::Promise<void> messageReadQueue = kj::READY_NOW; kj::Promise<void> messageReadQueue = kj::READY_NOW;
kj::Maybe<kj::Own<kj::PromiseFulfiller<void>>> onMessageDone; kj::Maybe<kj::Own<kj::PromiseFulfiller<void>>> onMessageDone;
...@@ -2413,7 +2418,8 @@ public: ...@@ -2413,7 +2418,8 @@ public:
settings(kj::mv(settings)) {} settings(kj::mv(settings)) {}
bool canReuse() { bool canReuse() {
// Returns true if we can reuse this HttpClient for another request. // Returns true if we can immediately reuse this HttpClient for another message (so all
// previous messages have been fully read).
return !upgraded && !closed && httpInput.canReuse() && httpOutput.canReuse(); return !upgraded && !closed && httpInput.canReuse() && httpOutput.canReuse();
} }
......
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