#include "gtest/gtest.h" #include "reactors_local.hpp" #include #include #include #include #include #include #include #include TEST(SystemTest, ReturnWithoutThrowing) { struct Master : public Reactor { Master(std::string name) : Reactor(name) {} virtual void Run() { CloseChannel("main"); } }; System &system = System::GetInstance(); ASSERT_NO_THROW(system.Spawn("master")); ASSERT_NO_THROW(system.AwaitShutdown()); } TEST(ChannelCreationTest, ThrowOnReusingChannelName) { struct Master : public Reactor { Master(std::string name) : Reactor(name) {} virtual void Run() { Open("channel"); ASSERT_THROW(Open("channel"), std::runtime_error); CloseChannel("main"); CloseChannel("channel"); } }; System &system = System::GetInstance(); system.Spawn("master"); system.AwaitShutdown(); } TEST(ChannelSetUpTest, CheckMainChannelIsSet) { struct Master : public Reactor { Master(std::string name) : Reactor(name) {} virtual void Run() { std::shared_ptr channel_writer; while (!(channel_writer = System::GetInstance().FindChannel("worker", "main"))) std::this_thread::sleep_for(std::chrono::milliseconds(300)); std::this_thread::sleep_for(std::chrono::milliseconds(300)); CloseChannel("main"); } }; struct Worker : public Reactor { Worker(std::string name) : Reactor(name) {} virtual void Run() { std::shared_ptr channel_writer; while (!(channel_writer = System::GetInstance().FindChannel("master", "main"))) std::this_thread::sleep_for(std::chrono::milliseconds(300)); std::this_thread::sleep_for(std::chrono::milliseconds(300)); CloseChannel("main"); } }; System &system = System::GetInstance(); system.Spawn("master"); system.Spawn("worker"); system.AwaitShutdown(); } TEST(SimpleSendTest, OneCallback) { struct MessageInt : public Message { MessageInt(int xx) : x(xx) {} int x; }; struct Master : public Reactor { Master(std::string name) : Reactor(name) {} virtual void Run() { std::shared_ptr channel_writer; while (!(channel_writer = System::GetInstance().FindChannel("worker", "main"))) std::this_thread::sleep_for(std::chrono::milliseconds(300)); channel_writer->Send(888); CloseChannel("main"); } }; struct Worker : public Reactor { Worker(std::string name) : Reactor(name) {} virtual void Run() { EventStream* stream = main_.first; stream->OnEvent([this](const MessageInt &msg, const Subscription&) { ASSERT_EQ(msg.x, 888); CloseChannel("main"); }); } }; System &system = System::GetInstance(); system.Spawn("master"); system.Spawn("worker"); system.AwaitShutdown(); } TEST(SimpleSendTest, IgnoreAfterClose) { struct MessageInt : public Message { MessageInt(int xx) : x(xx) {} int x; }; struct Master : public Reactor { Master(std::string name) : Reactor(name) {} virtual void Run() { std::shared_ptr channel_writer; while (!(channel_writer = System::GetInstance().FindChannel("worker", "main"))) std::this_thread::sleep_for(std::chrono::milliseconds(300)); channel_writer->Send(101); channel_writer->Send(102); // should be ignored std::this_thread::sleep_for(std::chrono::milliseconds(300)); channel_writer->Send(103); // should be ignored channel_writer->Send(104); // should be ignored CloseChannel("main"); // Write-end doesn't need to be closed because it's in RAII. } }; struct Worker : public Reactor { Worker(std::string name) : Reactor(name) {} virtual void Run() { EventStream* stream = main_.first; stream->OnEvent([this](const MessageInt& msg, const Subscription&) { CloseChannel("main"); ASSERT_EQ(msg.x, 101); }); } }; System &system = System::GetInstance(); system.Spawn("master"); system.Spawn("worker"); system.AwaitShutdown(); } TEST(SimpleSendTest, DuringFirstEvent) { struct MessageInt : public Message { MessageInt(int xx) : x(xx) {} int x; }; struct Master : public Reactor { Master(std::string name, std::promise p) : Reactor(name), p_(std::move(p)) {} virtual void Run() { EventStream* stream = main_.first; stream->OnEvent([this](const Message &msg, const Subscription &subscription) { const MessageInt &msgint = dynamic_cast(msg); if (msgint.x == 101) FindChannel("main")->Send(102); if (msgint.x == 102) { subscription.Unsubscribe(); CloseChannel("main"); p_.set_value(777); } }); std::shared_ptr channel_writer = FindChannel("main"); channel_writer->Send(101); } std::promise p_; }; System &system = System::GetInstance(); std::promise p; auto f = p.get_future(); system.Spawn("master", std::move(p)); f.wait(); ASSERT_EQ(f.get(), 777); system.AwaitShutdown(); } TEST(MultipleSendTest, UnsubscribeService) { struct MessageInt : public Message { MessageInt(int xx) : x(xx) {} int x; }; struct MessageChar : public Message { MessageChar(char xx) : x(xx) {} char x; }; struct Master : public Reactor { Master(std::string name) : Reactor(name) {} virtual void Run() { std::shared_ptr channel_writer; while (!(channel_writer = System::GetInstance().FindChannel("worker", "main"))) std::this_thread::sleep_for(std::chrono::milliseconds(300)); channel_writer->Send(55); channel_writer->Send(66); channel_writer->Send(77); channel_writer->Send(88); std::this_thread::sleep_for(std::chrono::milliseconds(300)); channel_writer->Send('a'); channel_writer->Send('b'); channel_writer->Send('c'); channel_writer->Send('d'); CloseChannel("main"); } }; struct Worker : public Reactor { Worker(std::string name) : Reactor(name) {} int num_msgs_received = 0; virtual void Run() { EventStream* stream = main_.first; stream->OnEvent([this](const MessageInt &msgint, const Subscription &subscription) { ASSERT_TRUE(msgint.x == 55 || msgint.x == 66); ++num_msgs_received; if (msgint.x == 66) { subscription.Unsubscribe(); // receive only two of them } }); stream->OnEvent([this](const MessageChar &msgchar, const Subscription &subscription) { char c = msgchar.x; ++num_msgs_received; ASSERT_TRUE(c == 'a' || c == 'b' || c == 'c'); if (num_msgs_received == 5) { subscription.Unsubscribe(); CloseChannel("main"); } }); } }; System &system = System::GetInstance(); system.Spawn("master"); system.Spawn("worker"); system.AwaitShutdown(); } TEST(MultipleSendTest, OnEvent) { struct MessageInt : public Message { MessageInt(int xx) : x(xx) {} int x; }; struct MessageChar : public Message { MessageChar(char xx) : x(xx) {} char x; }; struct Master : public Reactor { Master(std::string name) : Reactor(name) {} virtual void Run() { std::shared_ptr channel_writer; while (!(channel_writer = System::GetInstance().FindChannel("worker", "main"))) std::this_thread::sleep_for(std::chrono::milliseconds(300)); channel_writer->Send(101); channel_writer->Send('a'); channel_writer->Send(103); channel_writer->Send('b'); CloseChannel("main"); } }; struct Worker : public Reactor { Worker(std::string name) : Reactor(name) {} struct EndMessage : Message {}; int correct_vals = 0; virtual void Run() { EventStream* stream = main_.first; correct_vals = 0; stream->OnEvent([this](const MessageInt &msgint, const Subscription&) { ASSERT_TRUE(msgint.x == 101 || msgint.x == 103); ++correct_vals; main_.second->Send(); }); stream->OnEvent([this](const MessageChar &msgchar, const Subscription&) { ASSERT_TRUE(msgchar.x == 'a' || msgchar.x == 'b'); ++correct_vals; main_.second->Send(); }); stream->OnEvent([this](const EndMessage&, const Subscription&) { ASSERT_LE(correct_vals, 4); if (correct_vals == 4) { CloseChannel("main"); } }); } }; System &system = System::GetInstance(); system.Spawn("master"); system.Spawn("worker"); system.AwaitShutdown(); } TEST(MultipleSendTest, Chaining) { struct MessageInt : public Message { MessageInt(int xx) : x(xx) {} int x; }; struct Master : public Reactor { Master(std::string name) : Reactor(name) {} virtual void Run() { std::shared_ptr channel_writer; while (!(channel_writer = System::GetInstance().FindChannel("worker", "main"))) std::this_thread::sleep_for(std::chrono::milliseconds(300)); channel_writer->Send(55); channel_writer->Send(66); channel_writer->Send(77); CloseChannel("main"); } }; struct Worker : public Reactor { Worker(std::string name) : Reactor(name) {} virtual void Run() { EventStream* stream = main_.first; stream->OnEventOnce() .ChainOnce([this](const MessageInt &msg, const Subscription&) { ASSERT_EQ(msg.x, 55); }) .ChainOnce([](const MessageInt &msg, const Subscription&) { ASSERT_EQ(msg.x, 66); }) .ChainOnce([this](const MessageInt &msg, const Subscription&) { ASSERT_EQ(msg.x, 77); CloseChannel("main"); }); } }; System &system = System::GetInstance(); system.Spawn("master"); system.Spawn("worker"); system.AwaitShutdown(); } TEST(MultipleSendTest, ChainingInRightOrder) { struct MessageInt : public Message { MessageInt(int xx) : x(xx) {} int x; }; struct MessageChar : public Message { MessageChar(char xx) : x(xx) {} char x; }; struct Master : public Reactor { Master(std::string name) : Reactor(name) {} virtual void Run() { std::shared_ptr channel_writer; while (!(channel_writer = System::GetInstance().FindChannel("worker", "main"))) std::this_thread::sleep_for(std::chrono::milliseconds(300)); channel_writer->Send('a'); channel_writer->Send(55); channel_writer->Send('b'); channel_writer->Send(77); CloseChannel("main"); } }; struct Worker : public Reactor { Worker(std::string name) : Reactor(name) {} virtual void Run() { EventStream* stream = main_.first; stream->OnEventOnce() .ChainOnce([this](const MessageInt &msg, const Subscription&) { ASSERT_EQ(msg.x, 55); }) .ChainOnce([](const MessageChar &msg, const Subscription&) { ASSERT_EQ(msg.x, 'b'); }) .ChainOnce([this](const MessageInt &msg, const Subscription&) { ASSERT_EQ(msg.x, 77); CloseChannel("main"); }); } }; System &system = System::GetInstance(); system.Spawn("master"); system.Spawn("worker"); system.AwaitShutdown(); } TEST(MultipleSendTest, ProcessManyMessages) { const static int num_tests = 100; struct MessageInt : public Message { MessageInt(int xx) : x(xx) {} int x; }; struct Master : public Reactor { Master(std::string name) : Reactor(name) {} virtual void Run() { std::shared_ptr channel_writer; while (!(channel_writer = System::GetInstance().FindChannel("worker", "main"))) std::this_thread::sleep_for(std::chrono::milliseconds(300)); std::this_thread::sleep_for(std::chrono::milliseconds(rand() % 100)); for (int i = 0; i < num_tests; ++i) { channel_writer->Send(rand()); std::this_thread::sleep_for(std::chrono::milliseconds(rand() % 5)); } CloseChannel("main"); } }; struct Worker : public Reactor { Worker(std::string name) : Reactor(name) {} struct EndMessage : Message {}; int vals = 0; virtual void Run() { EventStream* stream = main_.first; vals = 0; stream->OnEvent([this](const Message&, const Subscription&) { ++vals; main_.second->Send(); }); stream->OnEvent([this](const Message&, const Subscription&) { ASSERT_LE(vals, num_tests); if (vals == num_tests) { CloseChannel("main"); } }); } }; System &system = System::GetInstance(); system.Spawn("master"); system.Spawn("worker"); system.AwaitShutdown(); } int main(int argc, char **argv) { ::testing::InitGoogleTest(&argc, argv); return RUN_ALL_TESTS(); }