tomato/test/entt/process/scheduler.cpp

155 lines
3.6 KiB
C++
Raw Normal View History

#include <functional>
#include <utility>
#include <gtest/gtest.h>
#include <entt/process/process.hpp>
#include <entt/process/scheduler.hpp>
struct foo_process: entt::process<foo_process, int> {
foo_process(std::function<void()> upd, std::function<void()> abort)
: on_update{upd}, on_aborted{abort} {}
void update(delta_type, void *) {
on_update();
}
void aborted() {
on_aborted();
}
std::function<void()> on_update;
std::function<void()> on_aborted;
};
struct succeeded_process: entt::process<succeeded_process, int> {
void update(delta_type, void *) {
++invoked;
succeed();
}
static inline unsigned int invoked;
};
struct failed_process: entt::process<failed_process, int> {
void update(delta_type, void *) {
++invoked;
fail();
}
static inline unsigned int invoked;
};
struct Scheduler: ::testing::Test {
void SetUp() override {
succeeded_process::invoked = 0u;
failed_process::invoked = 0u;
}
};
TEST_F(Scheduler, Functionalities) {
entt::scheduler<int> scheduler{};
bool updated = false;
bool aborted = false;
ASSERT_EQ(scheduler.size(), 0u);
ASSERT_TRUE(scheduler.empty());
scheduler.attach<foo_process>(
[&updated]() { updated = true; },
[&aborted]() { aborted = true; });
ASSERT_NE(scheduler.size(), 0u);
ASSERT_FALSE(scheduler.empty());
scheduler.update(0);
scheduler.abort(true);
ASSERT_TRUE(updated);
ASSERT_TRUE(aborted);
ASSERT_NE(scheduler.size(), 0u);
ASSERT_FALSE(scheduler.empty());
scheduler.clear();
ASSERT_EQ(scheduler.size(), 0u);
ASSERT_TRUE(scheduler.empty());
}
TEST_F(Scheduler, Then) {
entt::scheduler<int> scheduler;
// failing process with successor
scheduler.attach<succeeded_process>()
.then<succeeded_process>()
.then<failed_process>()
.then<succeeded_process>();
// failing process without successor
scheduler.attach<succeeded_process>()
.then<succeeded_process>()
.then<failed_process>();
// non-failing process
scheduler.attach<succeeded_process>()
.then<succeeded_process>();
ASSERT_EQ(succeeded_process::invoked, 0u);
ASSERT_EQ(failed_process::invoked, 0u);
while(!scheduler.empty()) {
scheduler.update(0);
}
ASSERT_EQ(succeeded_process::invoked, 6u);
ASSERT_EQ(failed_process::invoked, 2u);
}
TEST_F(Scheduler, Functor) {
entt::scheduler<int> scheduler;
bool first_functor = false;
bool second_functor = false;
auto attach = [&first_functor](auto, void *, auto resolve, auto) {
ASSERT_FALSE(first_functor);
first_functor = true;
resolve();
};
auto then = [&second_functor](auto, void *, auto, auto reject) {
ASSERT_FALSE(second_functor);
second_functor = true;
reject();
};
scheduler.attach(std::move(attach)).then(std::move(then)).then([](auto...) { FAIL(); });
while(!scheduler.empty()) {
scheduler.update(0);
}
ASSERT_TRUE(first_functor);
ASSERT_TRUE(second_functor);
ASSERT_TRUE(scheduler.empty());
}
TEST_F(Scheduler, SpawningProcess) {
entt::scheduler<int> scheduler;
scheduler.attach([&scheduler](auto, void *, auto resolve, auto) {
scheduler.attach<succeeded_process>().then<failed_process>();
resolve();
});
ASSERT_EQ(succeeded_process::invoked, 0u);
ASSERT_EQ(failed_process::invoked, 0u);
while(!scheduler.empty()) {
scheduler.update(0);
}
ASSERT_EQ(succeeded_process::invoked, 1u);
ASSERT_EQ(failed_process::invoked, 1u);
}