Commit 0c482f58 authored by Kenton Varda's avatar Kenton Varda Committed by GitHub

Merge pull request #557 from capnproto/http-client

Implement HttpClient that automatically manages connections. 
parents 99858df2 480e1f48
...@@ -136,8 +136,12 @@ class PromiseNode { ...@@ -136,8 +136,12 @@ class PromiseNode {
// internal implementation details. // internal implementation details.
public: public:
virtual void onReady(Event& event) noexcept = 0; virtual void onReady(Event* event) noexcept = 0;
// Arms the given event when ready. // Arms the given event when ready.
//
// May be called multiple times. If called again before the event was armed, the old event will
// never be armed, only the new one. If called again after the event was armed, the new event
// will be armed immediately. Can be called with nullptr to un-register the existing event.
virtual void setSelfPointer(Own<PromiseNode>* selfPtr) noexcept; virtual void setSelfPointer(Own<PromiseNode>* selfPtr) noexcept;
// Tells the node that `selfPtr` is the pointer that owns this node, and will continue to own // Tells the node that `selfPtr` is the pointer that owns this node, and will continue to own
...@@ -159,12 +163,11 @@ protected: ...@@ -159,12 +163,11 @@ protected:
// Helper class for implementing onReady(). // Helper class for implementing onReady().
public: public:
void init(Event& newEvent); void init(Event* newEvent);
// Returns true if arm() was already called.
void arm(); void arm();
// Arms the event if init() has already been called and makes future calls to init() return // Arms the event if init() has already been called and makes future calls to init()
// true. // automatically arm the event.
private: private:
Event* event = nullptr; Event* event = nullptr;
...@@ -178,7 +181,7 @@ public: ...@@ -178,7 +181,7 @@ public:
ImmediatePromiseNodeBase(); ImmediatePromiseNodeBase();
~ImmediatePromiseNodeBase() noexcept(false); ~ImmediatePromiseNodeBase() noexcept(false);
void onReady(Event& event) noexcept override; void onReady(Event* event) noexcept override;
}; };
template <typename T> template <typename T>
...@@ -212,7 +215,7 @@ class AttachmentPromiseNodeBase: public PromiseNode { ...@@ -212,7 +215,7 @@ class AttachmentPromiseNodeBase: public PromiseNode {
public: public:
AttachmentPromiseNodeBase(Own<PromiseNode>&& dependency); AttachmentPromiseNodeBase(Own<PromiseNode>&& dependency);
void onReady(Event& event) noexcept override; void onReady(Event* event) noexcept override;
void get(ExceptionOrValue& output) noexcept override; void get(ExceptionOrValue& output) noexcept override;
PromiseNode* getInnerForTrace() override; PromiseNode* getInnerForTrace() override;
...@@ -338,7 +341,7 @@ class TransformPromiseNodeBase: public PromiseNode { ...@@ -338,7 +341,7 @@ class TransformPromiseNodeBase: public PromiseNode {
public: public:
TransformPromiseNodeBase(Own<PromiseNode>&& dependency, void* continuationTracePtr); TransformPromiseNodeBase(Own<PromiseNode>&& dependency, void* continuationTracePtr);
void onReady(Event& event) noexcept override; void onReady(Event* event) noexcept override;
void get(ExceptionOrValue& output) noexcept override; void get(ExceptionOrValue& output) noexcept override;
PromiseNode* getInnerForTrace() override; PromiseNode* getInnerForTrace() override;
...@@ -410,7 +413,7 @@ public: ...@@ -410,7 +413,7 @@ public:
// Called by the hub to indicate that it is ready. // Called by the hub to indicate that it is ready.
// implements PromiseNode ------------------------------------------ // implements PromiseNode ------------------------------------------
void onReady(Event& event) noexcept override; void onReady(Event* event) noexcept override;
PromiseNode* getInnerForTrace() override; PromiseNode* getInnerForTrace() override;
protected: protected:
...@@ -545,7 +548,7 @@ public: ...@@ -545,7 +548,7 @@ public:
explicit ChainPromiseNode(Own<PromiseNode> inner); explicit ChainPromiseNode(Own<PromiseNode> inner);
~ChainPromiseNode() noexcept(false); ~ChainPromiseNode() noexcept(false);
void onReady(Event& event) noexcept override; void onReady(Event* event) noexcept override;
void setSelfPointer(Own<PromiseNode>* selfPtr) noexcept override; void setSelfPointer(Own<PromiseNode>* selfPtr) noexcept override;
void get(ExceptionOrValue& output) noexcept override; void get(ExceptionOrValue& output) noexcept override;
PromiseNode* getInnerForTrace() override; PromiseNode* getInnerForTrace() override;
...@@ -585,7 +588,7 @@ public: ...@@ -585,7 +588,7 @@ public:
ExclusiveJoinPromiseNode(Own<PromiseNode> left, Own<PromiseNode> right); ExclusiveJoinPromiseNode(Own<PromiseNode> left, Own<PromiseNode> right);
~ExclusiveJoinPromiseNode() noexcept(false); ~ExclusiveJoinPromiseNode() noexcept(false);
void onReady(Event& event) noexcept override; void onReady(Event* event) noexcept override;
void get(ExceptionOrValue& output) noexcept override; void get(ExceptionOrValue& output) noexcept override;
PromiseNode* getInnerForTrace() override; PromiseNode* getInnerForTrace() override;
...@@ -619,7 +622,7 @@ public: ...@@ -619,7 +622,7 @@ public:
ExceptionOrValue* resultParts, size_t partSize); ExceptionOrValue* resultParts, size_t partSize);
~ArrayJoinPromiseNodeBase() noexcept(false); ~ArrayJoinPromiseNodeBase() noexcept(false);
void onReady(Event& event) noexcept override final; void onReady(Event* event) noexcept override final;
void get(ExceptionOrValue& output) noexcept override final; void get(ExceptionOrValue& output) noexcept override final;
PromiseNode* getInnerForTrace() override final; PromiseNode* getInnerForTrace() override final;
...@@ -698,7 +701,7 @@ class EagerPromiseNodeBase: public PromiseNode, protected Event { ...@@ -698,7 +701,7 @@ class EagerPromiseNodeBase: public PromiseNode, protected Event {
public: public:
EagerPromiseNodeBase(Own<PromiseNode>&& dependency, ExceptionOrValue& resultRef); EagerPromiseNodeBase(Own<PromiseNode>&& dependency, ExceptionOrValue& resultRef);
void onReady(Event& event) noexcept override; void onReady(Event* event) noexcept override;
PromiseNode* getInnerForTrace() override; PromiseNode* getInnerForTrace() override;
private: private:
...@@ -735,7 +738,7 @@ Own<PromiseNode> spark(Own<PromiseNode>&& node) { ...@@ -735,7 +738,7 @@ Own<PromiseNode> spark(Own<PromiseNode>&& node) {
class AdapterPromiseNodeBase: public PromiseNode { class AdapterPromiseNodeBase: public PromiseNode {
public: public:
void onReady(Event& event) noexcept override; void onReady(Event* event) noexcept override;
protected: protected:
inline void setReady() { inline void setReady() {
...@@ -854,7 +857,7 @@ template <typename T> ...@@ -854,7 +857,7 @@ template <typename T>
T Promise<T>::wait(WaitScope& waitScope) { T Promise<T>::wait(WaitScope& waitScope) {
_::ExceptionOr<_::FixVoid<T>> result; _::ExceptionOr<_::FixVoid<T>> result;
waitImpl(kj::mv(node), result, waitScope); _::waitImpl(kj::mv(node), result, waitScope);
KJ_IF_MAYBE(value, result.value) { KJ_IF_MAYBE(value, result.value) {
KJ_IF_MAYBE(exception, result.exception) { KJ_IF_MAYBE(exception, result.exception) {
...@@ -875,7 +878,7 @@ inline void Promise<void>::wait(WaitScope& waitScope) { ...@@ -875,7 +878,7 @@ inline void Promise<void>::wait(WaitScope& waitScope) {
_::ExceptionOr<_::Void> result; _::ExceptionOr<_::Void> result;
waitImpl(kj::mv(node), result, waitScope); _::waitImpl(kj::mv(node), result, waitScope);
if (result.value != nullptr) { if (result.value != nullptr) {
KJ_IF_MAYBE(exception, result.exception) { KJ_IF_MAYBE(exception, result.exception) {
...@@ -889,6 +892,11 @@ inline void Promise<void>::wait(WaitScope& waitScope) { ...@@ -889,6 +892,11 @@ inline void Promise<void>::wait(WaitScope& waitScope) {
} }
} }
template <typename T>
bool Promise<T>::poll(WaitScope& waitScope) {
return _::pollImpl(*node, waitScope);
}
template <typename T> template <typename T>
ForkedPromise<T> Promise<T>::fork() { ForkedPromise<T> Promise<T>::fork() {
return ForkedPromise<T>(false, refcounted<_::ForkHub<_::FixVoid<T>>>(kj::mv(node))); return ForkedPromise<T>(false, refcounted<_::ForkHub<_::FixVoid<T>>>(kj::mv(node)));
......
...@@ -199,6 +199,7 @@ private: ...@@ -199,6 +199,7 @@ private:
void detach(kj::Promise<void>&& promise); void detach(kj::Promise<void>&& promise);
void waitImpl(Own<_::PromiseNode>&& node, _::ExceptionOrValue& result, WaitScope& waitScope); void waitImpl(Own<_::PromiseNode>&& node, _::ExceptionOrValue& result, WaitScope& waitScope);
bool pollImpl(_::PromiseNode& node, WaitScope& waitScope);
Promise<void> yield(); Promise<void> yield();
Own<PromiseNode> neverDone(); Own<PromiseNode> neverDone();
......
...@@ -748,5 +748,16 @@ TEST(Async, SetRunnable) { ...@@ -748,5 +748,16 @@ TEST(Async, SetRunnable) {
} }
} }
TEST(Async, Poll) {
EventLoop loop;
WaitScope waitScope(loop);
auto paf = newPromiseAndFulfiller<void>();
KJ_ASSERT(!paf.promise.poll(waitScope));
paf.fulfiller->fulfill();
KJ_ASSERT(paf.promise.poll(waitScope));
paf.promise.wait(waitScope);
}
} // namespace } // namespace
} // namespace kj } // namespace kj
...@@ -66,8 +66,8 @@ public: ...@@ -66,8 +66,8 @@ public:
class YieldPromiseNode final: public _::PromiseNode { class YieldPromiseNode final: public _::PromiseNode {
public: public:
void onReady(_::Event& event) noexcept override { void onReady(_::Event* event) noexcept override {
event.armBreadthFirst(); if (event) event->armBreadthFirst();
} }
void get(_::ExceptionOrValue& output) noexcept override { void get(_::ExceptionOrValue& output) noexcept override {
output.as<_::Void>() = _::Void(); output.as<_::Void>() = _::Void();
...@@ -76,7 +76,7 @@ public: ...@@ -76,7 +76,7 @@ public:
class NeverDonePromiseNode final: public _::PromiseNode { class NeverDonePromiseNode final: public _::PromiseNode {
public: public:
void onReady(_::Event& event) noexcept override { void onReady(_::Event* event) noexcept override {
// ignore // ignore
} }
void get(_::ExceptionOrValue& output) noexcept override { void get(_::ExceptionOrValue& output) noexcept override {
...@@ -108,7 +108,7 @@ public: ...@@ -108,7 +108,7 @@ public:
Task(TaskSetImpl& taskSet, Own<_::PromiseNode>&& nodeParam) Task(TaskSetImpl& taskSet, Own<_::PromiseNode>&& nodeParam)
: taskSet(taskSet), node(kj::mv(nodeParam)) { : taskSet(taskSet), node(kj::mv(nodeParam)) {
node->setSelfPointer(&node); node->setSelfPointer(&node);
node->onReady(*this); node->onReady(this);
} }
protected: protected:
...@@ -312,6 +312,26 @@ void EventLoop::leaveScope() { ...@@ -312,6 +312,26 @@ void EventLoop::leaveScope() {
threadLocalEventLoop = nullptr; threadLocalEventLoop = nullptr;
} }
void WaitScope::poll() {
KJ_REQUIRE(&loop == threadLocalEventLoop, "WaitScope not valid for this thread.");
KJ_REQUIRE(!loop.running, "poll() is not allowed from within event callbacks.");
loop.running = true;
KJ_DEFER(loop.running = false);
for (;;) {
if (!loop.turn()) {
// No events in the queue. Poll for I/O.
loop.port.poll();
if (!loop.isRunnable()) {
// Still no events in the queue. We're done.
return;
}
}
}
}
namespace _ { // private namespace _ { // private
void waitImpl(Own<_::PromiseNode>&& node, _::ExceptionOrValue& result, WaitScope& waitScope) { void waitImpl(Own<_::PromiseNode>&& node, _::ExceptionOrValue& result, WaitScope& waitScope) {
...@@ -321,7 +341,7 @@ void waitImpl(Own<_::PromiseNode>&& node, _::ExceptionOrValue& result, WaitScope ...@@ -321,7 +341,7 @@ void waitImpl(Own<_::PromiseNode>&& node, _::ExceptionOrValue& result, WaitScope
BoolEvent doneEvent; BoolEvent doneEvent;
node->setSelfPointer(&node); node->setSelfPointer(&node);
node->onReady(doneEvent); node->onReady(&doneEvent);
loop.running = true; loop.running = true;
KJ_DEFER(loop.running = false); KJ_DEFER(loop.running = false);
...@@ -343,6 +363,35 @@ void waitImpl(Own<_::PromiseNode>&& node, _::ExceptionOrValue& result, WaitScope ...@@ -343,6 +363,35 @@ void waitImpl(Own<_::PromiseNode>&& node, _::ExceptionOrValue& result, WaitScope
} }
} }
bool pollImpl(_::PromiseNode& node, WaitScope& waitScope) {
EventLoop& loop = waitScope.loop;
KJ_REQUIRE(&loop == threadLocalEventLoop, "WaitScope not valid for this thread.");
KJ_REQUIRE(!loop.running, "poll() is not allowed from within event callbacks.");
BoolEvent doneEvent;
node.onReady(&doneEvent);
loop.running = true;
KJ_DEFER(loop.running = false);
while (!doneEvent.fired) {
if (!loop.turn()) {
// No events in the queue. Poll for I/O.
loop.port.poll();
if (!doneEvent.fired && !loop.isRunnable()) {
// No progress. Give up.
node.onReady(nullptr);
loop.setRunnable(false);
return false;
}
}
}
loop.setRunnable(loop.isRunnable());
return true;
}
Promise<void> yield() { Promise<void> yield() {
return Promise<void>(false, kj::heap<YieldPromiseNode>()); return Promise<void>(false, kj::heap<YieldPromiseNode>());
} }
...@@ -498,26 +547,28 @@ void PromiseNode::setSelfPointer(Own<PromiseNode>* selfPtr) noexcept {} ...@@ -498,26 +547,28 @@ void PromiseNode::setSelfPointer(Own<PromiseNode>* selfPtr) noexcept {}
PromiseNode* PromiseNode::getInnerForTrace() { return nullptr; } PromiseNode* PromiseNode::getInnerForTrace() { return nullptr; }
void PromiseNode::OnReadyEvent::init(Event& newEvent) { void PromiseNode::OnReadyEvent::init(Event* newEvent) {
if (event == _kJ_ALREADY_READY) { if (event == _kJ_ALREADY_READY) {
// A new continuation was added to a promise that was already ready. In this case, we schedule // A new continuation was added to a promise that was already ready. In this case, we schedule
// breadth-first, to make it difficult for applications to accidentally starve the event loop // breadth-first, to make it difficult for applications to accidentally starve the event loop
// by repeatedly waiting on immediate promises. // by repeatedly waiting on immediate promises.
newEvent.armBreadthFirst(); if (newEvent) newEvent->armBreadthFirst();
} else { } else {
event = &newEvent; event = newEvent;
} }
} }
void PromiseNode::OnReadyEvent::arm() { void PromiseNode::OnReadyEvent::arm() {
if (event == nullptr) { KJ_ASSERT(event != _kJ_ALREADY_READY, "arm() should only be called once");
event = _kJ_ALREADY_READY;
} else { if (event != nullptr) {
// A promise resolved and an event is already waiting on it. In this case, arm in depth-first // A promise resolved and an event is already waiting on it. In this case, arm in depth-first
// order so that the event runs immediately after the current one. This way, chained promises // order so that the event runs immediately after the current one. This way, chained promises
// execute together for better cache locality and lower latency. // execute together for better cache locality and lower latency.
event->armDepthFirst(); event->armDepthFirst();
} }
event = _kJ_ALREADY_READY;
} }
// ------------------------------------------------------------------- // -------------------------------------------------------------------
...@@ -525,8 +576,8 @@ void PromiseNode::OnReadyEvent::arm() { ...@@ -525,8 +576,8 @@ void PromiseNode::OnReadyEvent::arm() {
ImmediatePromiseNodeBase::ImmediatePromiseNodeBase() {} ImmediatePromiseNodeBase::ImmediatePromiseNodeBase() {}
ImmediatePromiseNodeBase::~ImmediatePromiseNodeBase() noexcept(false) {} ImmediatePromiseNodeBase::~ImmediatePromiseNodeBase() noexcept(false) {}
void ImmediatePromiseNodeBase::onReady(Event& event) noexcept { void ImmediatePromiseNodeBase::onReady(Event* event) noexcept {
event.armBreadthFirst(); if (event) event->armBreadthFirst();
} }
ImmediateBrokenPromiseNode::ImmediateBrokenPromiseNode(Exception&& exception) ImmediateBrokenPromiseNode::ImmediateBrokenPromiseNode(Exception&& exception)
...@@ -543,7 +594,7 @@ AttachmentPromiseNodeBase::AttachmentPromiseNodeBase(Own<PromiseNode>&& dependen ...@@ -543,7 +594,7 @@ AttachmentPromiseNodeBase::AttachmentPromiseNodeBase(Own<PromiseNode>&& dependen
dependency->setSelfPointer(&dependency); dependency->setSelfPointer(&dependency);
} }
void AttachmentPromiseNodeBase::onReady(Event& event) noexcept { void AttachmentPromiseNodeBase::onReady(Event* event) noexcept {
dependency->onReady(event); dependency->onReady(event);
} }
...@@ -567,7 +618,7 @@ TransformPromiseNodeBase::TransformPromiseNodeBase( ...@@ -567,7 +618,7 @@ TransformPromiseNodeBase::TransformPromiseNodeBase(
dependency->setSelfPointer(&dependency); dependency->setSelfPointer(&dependency);
} }
void TransformPromiseNodeBase::onReady(Event& event) noexcept { void TransformPromiseNodeBase::onReady(Event* event) noexcept {
dependency->onReady(event); dependency->onReady(event);
} }
...@@ -635,7 +686,7 @@ void ForkBranchBase::releaseHub(ExceptionOrValue& output) { ...@@ -635,7 +686,7 @@ void ForkBranchBase::releaseHub(ExceptionOrValue& output) {
} }
} }
void ForkBranchBase::onReady(Event& event) noexcept { void ForkBranchBase::onReady(Event* event) noexcept {
onReadyEvent.init(event); onReadyEvent.init(event);
} }
...@@ -648,7 +699,7 @@ PromiseNode* ForkBranchBase::getInnerForTrace() { ...@@ -648,7 +699,7 @@ PromiseNode* ForkBranchBase::getInnerForTrace() {
ForkHubBase::ForkHubBase(Own<PromiseNode>&& innerParam, ExceptionOrValue& resultRef) ForkHubBase::ForkHubBase(Own<PromiseNode>&& innerParam, ExceptionOrValue& resultRef)
: inner(kj::mv(innerParam)), resultRef(resultRef) { : inner(kj::mv(innerParam)), resultRef(resultRef) {
inner->setSelfPointer(&inner); inner->setSelfPointer(&inner);
inner->onReady(*this); inner->onReady(this);
} }
Maybe<Own<Event>> ForkHubBase::fire() { Maybe<Own<Event>> ForkHubBase::fire() {
...@@ -682,16 +733,16 @@ _::PromiseNode* ForkHubBase::getInnerForTrace() { ...@@ -682,16 +733,16 @@ _::PromiseNode* ForkHubBase::getInnerForTrace() {
ChainPromiseNode::ChainPromiseNode(Own<PromiseNode> innerParam) ChainPromiseNode::ChainPromiseNode(Own<PromiseNode> innerParam)
: state(STEP1), inner(kj::mv(innerParam)) { : state(STEP1), inner(kj::mv(innerParam)) {
inner->setSelfPointer(&inner); inner->setSelfPointer(&inner);
inner->onReady(*this); inner->onReady(this);
} }
ChainPromiseNode::~ChainPromiseNode() noexcept(false) {} ChainPromiseNode::~ChainPromiseNode() noexcept(false) {}
void ChainPromiseNode::onReady(Event& event) noexcept { void ChainPromiseNode::onReady(Event* event) noexcept {
switch (state) { switch (state) {
case STEP1: case STEP1:
KJ_REQUIRE(onReadyEvent == nullptr, "onReady() can only be called once."); KJ_REQUIRE(onReadyEvent == nullptr, "onReady() can only be called once.");
onReadyEvent = &event; onReadyEvent = event;
return; return;
case STEP2: case STEP2:
inner->onReady(event); inner->onReady(event);
...@@ -755,7 +806,7 @@ Maybe<Own<Event>> ChainPromiseNode::fire() { ...@@ -755,7 +806,7 @@ Maybe<Own<Event>> ChainPromiseNode::fire() {
*selfPtr = kj::mv(inner); *selfPtr = kj::mv(inner);
selfPtr->get()->setSelfPointer(selfPtr); selfPtr->get()->setSelfPointer(selfPtr);
if (onReadyEvent != nullptr) { if (onReadyEvent != nullptr) {
selfPtr->get()->onReady(*onReadyEvent); selfPtr->get()->onReady(onReadyEvent);
} }
// Return our self-pointer so that the caller takes care of deleting it. // Return our self-pointer so that the caller takes care of deleting it.
...@@ -763,7 +814,7 @@ Maybe<Own<Event>> ChainPromiseNode::fire() { ...@@ -763,7 +814,7 @@ Maybe<Own<Event>> ChainPromiseNode::fire() {
} else { } else {
inner->setSelfPointer(&inner); inner->setSelfPointer(&inner);
if (onReadyEvent != nullptr) { if (onReadyEvent != nullptr) {
inner->onReady(*onReadyEvent); inner->onReady(onReadyEvent);
} }
return nullptr; return nullptr;
...@@ -777,7 +828,7 @@ ExclusiveJoinPromiseNode::ExclusiveJoinPromiseNode(Own<PromiseNode> left, Own<Pr ...@@ -777,7 +828,7 @@ ExclusiveJoinPromiseNode::ExclusiveJoinPromiseNode(Own<PromiseNode> left, Own<Pr
ExclusiveJoinPromiseNode::~ExclusiveJoinPromiseNode() noexcept(false) {} ExclusiveJoinPromiseNode::~ExclusiveJoinPromiseNode() noexcept(false) {}
void ExclusiveJoinPromiseNode::onReady(Event& event) noexcept { void ExclusiveJoinPromiseNode::onReady(Event* event) noexcept {
onReadyEvent.init(event); onReadyEvent.init(event);
} }
...@@ -797,7 +848,7 @@ ExclusiveJoinPromiseNode::Branch::Branch( ...@@ -797,7 +848,7 @@ ExclusiveJoinPromiseNode::Branch::Branch(
ExclusiveJoinPromiseNode& joinNode, Own<PromiseNode> dependencyParam) ExclusiveJoinPromiseNode& joinNode, Own<PromiseNode> dependencyParam)
: joinNode(joinNode), dependency(kj::mv(dependencyParam)) { : joinNode(joinNode), dependency(kj::mv(dependencyParam)) {
dependency->setSelfPointer(&dependency); dependency->setSelfPointer(&dependency);
dependency->onReady(*this); dependency->onReady(this);
} }
ExclusiveJoinPromiseNode::Branch::~Branch() noexcept(false) {} ExclusiveJoinPromiseNode::Branch::~Branch() noexcept(false) {}
...@@ -847,7 +898,7 @@ ArrayJoinPromiseNodeBase::ArrayJoinPromiseNodeBase( ...@@ -847,7 +898,7 @@ ArrayJoinPromiseNodeBase::ArrayJoinPromiseNodeBase(
} }
ArrayJoinPromiseNodeBase::~ArrayJoinPromiseNodeBase() noexcept(false) {} ArrayJoinPromiseNodeBase::~ArrayJoinPromiseNodeBase() noexcept(false) {}
void ArrayJoinPromiseNodeBase::onReady(Event& event) noexcept { void ArrayJoinPromiseNodeBase::onReady(Event* event) noexcept {
onReadyEvent.init(event); onReadyEvent.init(event);
} }
...@@ -873,7 +924,7 @@ ArrayJoinPromiseNodeBase::Branch::Branch( ...@@ -873,7 +924,7 @@ ArrayJoinPromiseNodeBase::Branch::Branch(
ArrayJoinPromiseNodeBase& joinNode, Own<PromiseNode> dependencyParam, ExceptionOrValue& output) ArrayJoinPromiseNodeBase& joinNode, Own<PromiseNode> dependencyParam, ExceptionOrValue& output)
: joinNode(joinNode), dependency(kj::mv(dependencyParam)), output(output) { : joinNode(joinNode), dependency(kj::mv(dependencyParam)), output(output) {
dependency->setSelfPointer(&dependency); dependency->setSelfPointer(&dependency);
dependency->onReady(*this); dependency->onReady(this);
} }
ArrayJoinPromiseNodeBase::Branch::~Branch() noexcept(false) {} ArrayJoinPromiseNodeBase::Branch::~Branch() noexcept(false) {}
...@@ -921,10 +972,10 @@ EagerPromiseNodeBase::EagerPromiseNodeBase( ...@@ -921,10 +972,10 @@ EagerPromiseNodeBase::EagerPromiseNodeBase(
Own<PromiseNode>&& dependencyParam, ExceptionOrValue& resultRef) Own<PromiseNode>&& dependencyParam, ExceptionOrValue& resultRef)
: dependency(kj::mv(dependencyParam)), resultRef(resultRef) { : dependency(kj::mv(dependencyParam)), resultRef(resultRef) {
dependency->setSelfPointer(&dependency); dependency->setSelfPointer(&dependency);
dependency->onReady(*this); dependency->onReady(this);
} }
void EagerPromiseNodeBase::onReady(Event& event) noexcept { void EagerPromiseNodeBase::onReady(Event* event) noexcept {
onReadyEvent.init(event); onReadyEvent.init(event);
} }
...@@ -946,7 +997,7 @@ Maybe<Own<Event>> EagerPromiseNodeBase::fire() { ...@@ -946,7 +997,7 @@ Maybe<Own<Event>> EagerPromiseNodeBase::fire() {
// ------------------------------------------------------------------- // -------------------------------------------------------------------
void AdapterPromiseNodeBase::onReady(Event& event) noexcept { void AdapterPromiseNodeBase::onReady(Event* event) noexcept {
onReadyEvent.init(event); onReadyEvent.init(event);
} }
......
...@@ -235,6 +235,19 @@ public: ...@@ -235,6 +235,19 @@ public:
// TODO(someday): Implement fibers, and let them call wait() even when they are handling an // TODO(someday): Implement fibers, and let them call wait() even when they are handling an
// event. // event.
bool poll(WaitScope& waitScope);
// Returns true if a call to wait() would complete without blocking, false if it would block.
//
// If the promise is not yet resolved, poll() will pump the event loop and poll for I/O in an
// attempt to resolve it. Only when there is nothing left to do will it return false.
//
// Generally, poll() is most useful in tests. Often, you may want to verify that a promise does
// not resolve until some specific event occurs. To do so, poll() the promise before the event to
// verify it isn't resolved, then trigger the event, then poll() again to verify that it resolves.
// The first poll() verifies that the promise doesn't resolve early, which would otherwise be
// hard to do deterministically. The second poll() allows you to check that the promise has
// resolved and avoid a wait() that might deadlock in the case that it hasn't.
ForkedPromise<T> fork() KJ_WARN_UNUSED_RESULT; ForkedPromise<T> fork() KJ_WARN_UNUSED_RESULT;
// Forks the promise, so that multiple different clients can independently wait on the result. // Forks the promise, so that multiple different clients can independently wait on the result.
// `T` must be copy-constructable for this to work. Or, in the special case where `T` is // `T` must be copy-constructable for this to work. Or, in the special case where `T` is
...@@ -650,6 +663,7 @@ private: ...@@ -650,6 +663,7 @@ private:
friend void _::detach(kj::Promise<void>&& promise); friend void _::detach(kj::Promise<void>&& promise);
friend void _::waitImpl(Own<_::PromiseNode>&& node, _::ExceptionOrValue& result, friend void _::waitImpl(Own<_::PromiseNode>&& node, _::ExceptionOrValue& result,
WaitScope& waitScope); WaitScope& waitScope);
friend bool _::pollImpl(_::PromiseNode& node, WaitScope& waitScope);
friend class _::Event; friend class _::Event;
friend class WaitScope; friend class WaitScope;
}; };
...@@ -668,11 +682,15 @@ public: ...@@ -668,11 +682,15 @@ public:
inline ~WaitScope() { loop.leaveScope(); } inline ~WaitScope() { loop.leaveScope(); }
KJ_DISALLOW_COPY(WaitScope); KJ_DISALLOW_COPY(WaitScope);
void poll();
// Pumps the event queue and polls for I/O until there's nothing left to do (without blocking).
private: private:
EventLoop& loop; EventLoop& loop;
friend class EventLoop; friend class EventLoop;
friend void _::waitImpl(Own<_::PromiseNode>&& node, _::ExceptionOrValue& result, friend void _::waitImpl(Own<_::PromiseNode>&& node, _::ExceptionOrValue& result,
WaitScope& waitScope); WaitScope& waitScope);
friend bool _::pollImpl(_::PromiseNode& node, WaitScope& waitScope);
}; };
} // namespace kj } // namespace kj
......
This diff is collapsed.
This diff is collapsed.
...@@ -562,26 +562,53 @@ public: ...@@ -562,26 +562,53 @@ public:
// UNIMPLEMENTED. // UNIMPLEMENTED.
}; };
kj::Own<HttpClient> newHttpClient(HttpHeaderTable& responseHeaderTable, kj::Network& network, struct HttpClientSettings {
kj::Maybe<kj::Network&> tlsNetwork = nullptr, kj::Duration idleTimout = 5 * kj::SECONDS;
kj::Maybe<EntropySource&> entropySource = nullptr); // For clients which automatically create new connections, any connection idle for at least this
// Creates a proxy HttpClient that connects to hosts over the given network. // long will be closed.
kj::Maybe<EntropySource&> entropySource = nullptr;
// Must be provided in order to use `openWebSocket`. If you don't need WebSockets, this can be
// omitted. The WebSocket protocol uses random values to avoid triggering flaws (including
// security flaws) in certain HTTP proxy software. Specifically, entropy is used to generate the
// `Sec-WebSocket-Key` header and to generate frame masks. If you know that there are no broken
// or vulnerable proxies between you and the server, you can provide a dummy entropy source that
// doesn't generate real entropy (e.g. returning the same value every time). Otherwise, you must
// provide a cryptographically-random entropy source.
};
kj::Own<HttpClient> newHttpClient(kj::Timer& timer, HttpHeaderTable& responseHeaderTable,
kj::Network& network, kj::Maybe<kj::Network&> tlsNetwork,
HttpClientSettings settings = HttpClientSettings());
// Creates a proxy HttpClient that connects to hosts over the given network. The URL must always
// be an absolute URL; the host is parsed from the URL. This implementation will automatically
// add an appropriate Host header (and convert the URL to just a path) once it has connected.
//
// Note that if you wish to route traffic through an HTTP proxy server rather than connect to
// remote hosts directly, you should use the form of newHttpClient() that takes a NetworkAddress,
// and supply the proxy's address.
// //
// `responseHeaderTable` is used when parsing HTTP responses. Requests can use any header table. // `responseHeaderTable` is used when parsing HTTP responses. Requests can use any header table.
// //
// `tlsNetwork` is required to support HTTPS destination URLs. Otherwise, only HTTP URLs can be // `tlsNetwork` is required to support HTTPS destination URLs. If null, only HTTP URLs can be
// fetched. // fetched.
kj::Own<HttpClient> newHttpClient(kj::Timer& timer, HttpHeaderTable& responseHeaderTable,
kj::NetworkAddress& addr,
HttpClientSettings settings = HttpClientSettings());
// Creates an HttpClient that always connects to the given address no matter what URL is requested.
// The client will open and close connections as needed. It will attempt to reuse connections for
// multiple requests but will not send a new request before the previous response on the same
// connection has completed, as doing so can result in head-of-line blocking issues. The client may
// be used as a proxy client or a host client depending on whether the peer is operating as
// a proxy. (Hint: This is the best kind of client to use when routing traffic through an HTTP
// proxy. `addr` should be the address of the proxy, and the proxy itself will resolve remote hosts
// based on the URLs passed to it.)
// //
// `entropySource` must be provided in order to use `openWebSocket`. If you don't need WebSockets, // `responseHeaderTable` is used when parsing HTTP responses. Requests can use any header table.
// `entropySource` can be omitted. The WebSocket protocol uses random values to avoid triggering
// flaws (including security flaws) in certain HTTP proxy software. Specifically, entropy is used
// to generate the `Sec-WebSocket-Key` header and to generate frame masks. If you know that there
// are no broken or vulnerable proxies between you and the server, you can provide an dummy entropy
// source that doesn't generate real entropy (e.g. returning the same value every time). Otherwise,
// you must provide a cryptographically-random entropy source.
kj::Own<HttpClient> newHttpClient(HttpHeaderTable& responseHeaderTable, kj::AsyncIoStream& stream, kj::Own<HttpClient> newHttpClient(HttpHeaderTable& responseHeaderTable, kj::AsyncIoStream& stream,
kj::Maybe<EntropySource&> entropySource = nullptr); HttpClientSettings settings = HttpClientSettings());
// Creates an HttpClient that speaks over the given pre-established connection. The client may // Creates an HttpClient that speaks over the given pre-established connection. The client may
// be used as a proxy client or a host client depending on whether the peer is operating as // be used as a proxy client or a host client depending on whether the peer is operating as
// a proxy. // a proxy.
...@@ -591,14 +618,12 @@ kj::Own<HttpClient> newHttpClient(HttpHeaderTable& responseHeaderTable, kj::Asyn ...@@ -591,14 +618,12 @@ kj::Own<HttpClient> newHttpClient(HttpHeaderTable& responseHeaderTable, kj::Asyn
// fail as well. If the destination server chooses to close the connection after a response, // fail as well. If the destination server chooses to close the connection after a response,
// subsequent requests will fail. If a response takes a long time, it blocks subsequent responses. // subsequent requests will fail. If a response takes a long time, it blocks subsequent responses.
// If a WebSocket is opened successfully, all subsequent requests fail. // If a WebSocket is opened successfully, all subsequent requests fail.
//
// `entropySource` must be provided in order to use `openWebSocket`. If you don't need WebSockets, kj::Own<HttpClient> newHttpClient(
// `entropySource` can be omitted. The WebSocket protocol uses random values to avoid triggering HttpHeaderTable& responseHeaderTable, kj::AsyncIoStream& stream,
// flaws (including security flaws) in certain HTTP proxy software. Specifically, entropy is used kj::Maybe<EntropySource&> entropySource) KJ_DEPRECATED("use HttpClientSettings");
// to generate the `Sec-WebSocket-Key` header and to generate frame masks. If you know that there // Temporary for backwards-compatibilty.
// are no broken or vulnerable proxies between you and the server, you can provide an dummy entropy // TODO(soon): Remove this before next release.
// source that doesn't generate real entropy (e.g. returning the same value every time). Otherwise,
// you must provide a cryptographically-random entropy source.
kj::Own<HttpClient> newHttpClient(HttpService& service); kj::Own<HttpClient> newHttpClient(HttpService& service);
kj::Own<HttpService> newHttpService(HttpClient& client); kj::Own<HttpService> newHttpService(HttpClient& client);
...@@ -726,6 +751,14 @@ inline void HttpHeaders::forEach(Func&& func) const { ...@@ -726,6 +751,14 @@ inline void HttpHeaders::forEach(Func&& func) const {
} }
} }
inline kj::Own<HttpClient> newHttpClient(
HttpHeaderTable& responseHeaderTable, kj::AsyncIoStream& stream,
kj::Maybe<EntropySource&> entropySource) {
HttpClientSettings settings;
settings.entropySource = entropySource;
return newHttpClient(responseHeaderTable, stream, kj::mv(settings));
}
} // namespace kj } // namespace kj
#endif // KJ_COMPAT_HTTP_H_ #endif // KJ_COMPAT_HTTP_H_
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