Commit b62c247b authored by Kenton Varda's avatar Kenton Varda

Add TaskSet.onEmpty() to wait until all tasks have completed.

Fixes #286.
parent 5cb6cf76
...@@ -620,6 +620,30 @@ TEST(Async, TaskSet) { ...@@ -620,6 +620,30 @@ TEST(Async, TaskSet) {
EXPECT_EQ(1u, errorHandler.exceptionCount); EXPECT_EQ(1u, errorHandler.exceptionCount);
} }
TEST(Async, TaskSetOnEmpty) {
EventLoop loop;
WaitScope waitScope(loop);
ErrorHandlerImpl errorHandler;
TaskSet tasks(errorHandler);
KJ_EXPECT(tasks.isEmpty());
auto paf = newPromiseAndFulfiller<void>();
tasks.add(kj::mv(paf.promise));
tasks.add(evalLater([]() {}));
KJ_EXPECT(!tasks.isEmpty());
auto promise = tasks.onEmpty();
KJ_EXPECT(!promise.poll(waitScope));
KJ_EXPECT(!tasks.isEmpty());
paf.fulfiller->fulfill();
KJ_ASSERT(promise.poll(waitScope));
KJ_EXPECT(tasks.isEmpty());
promise.wait(waitScope);
}
class DestructorDetector { class DestructorDetector {
public: public:
DestructorDetector(bool& setTrue): setTrue(setTrue) {} DestructorDetector(bool& setTrue): setTrue(setTrue) {}
......
...@@ -127,6 +127,14 @@ protected: ...@@ -127,6 +127,14 @@ protected:
*prev = kj::mv(next); *prev = kj::mv(next);
next = nullptr; next = nullptr;
prev = nullptr; prev = nullptr;
KJ_IF_MAYBE(f, taskSet.emptyFulfiller) {
if (taskSet.tasks == nullptr) {
f->get()->fulfill();
taskSet.emptyFulfiller = nullptr;
}
}
return mv(self); return mv(self);
} }
...@@ -165,6 +173,18 @@ kj::String TaskSet::trace() { ...@@ -165,6 +173,18 @@ kj::String TaskSet::trace() {
return kj::strArray(traces, "\n============================================\n"); return kj::strArray(traces, "\n============================================\n");
} }
Promise<void> TaskSet::onEmpty() {
KJ_REQUIRE(emptyFulfiller == nullptr, "onEmpty() can only be called once at a time");
if (tasks == nullptr) {
return READY_NOW;
} else {
auto paf = newPromiseAndFulfiller<void>();
emptyFulfiller = kj::mv(paf.fulfiller);
return kj::mv(paf.promise);
}
}
namespace _ { // private namespace _ { // private
class LoggingErrorHandler: public TaskSet::ErrorHandler { class LoggingErrorHandler: public TaskSet::ErrorHandler {
......
...@@ -532,11 +532,19 @@ public: ...@@ -532,11 +532,19 @@ public:
kj::String trace(); kj::String trace();
// Return debug info about all promises currently in the TaskSet. // Return debug info about all promises currently in the TaskSet.
bool isEmpty() { return tasks == nullptr; }
// Check if any tasks are running.
Promise<void> onEmpty();
// Returns a promise that fulfills the next time the TaskSet is empty. Only one such promise can
// exist at a time.
private: private:
class Task; class Task;
TaskSet::ErrorHandler& errorHandler; TaskSet::ErrorHandler& errorHandler;
Maybe<Own<Task>> tasks; Maybe<Own<Task>> tasks;
Maybe<Own<PromiseFulfiller<void>>> emptyFulfiller;
}; };
// ======================================================================================= // =======================================================================================
......
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