#include "gtest/gtest.h" #include #include #include #include #include #include #include #include "communication.hpp" TEST(SystemTest, ReturnWithoutThrowing) { struct Master : public Reactor { Master(System *system, std::string name) : Reactor(system, name) {} virtual void Run() { CloseConnector("main"); } }; System system; ASSERT_NO_THROW(system.StartServices()); ASSERT_NO_THROW(system.Spawn("master")); ASSERT_NO_THROW(system.AwaitShutdown()); } TEST(ChannelCreationTest, ThrowOnReusingChannelName) { struct Master : public Reactor { Master(System *system, std::string name) : Reactor(system, name) {} virtual void Run() { Open("channel"); ASSERT_THROW(Open("channel"), std::runtime_error); CloseConnector("main"); CloseConnector("channel"); } }; System system; system.Spawn("master"); system.AwaitShutdown(); } TEST(ConnectorSetUpTest, CheckMainChannelIsSet) { struct Master : public Reactor { Master(System *system, std::string name) : Reactor(system, name) {} virtual void Run() { std::shared_ptr channel; while (!(channel = system_->FindChannel("worker", "main"))) std::this_thread::sleep_for(std::chrono::milliseconds(300)); std::this_thread::sleep_for(std::chrono::milliseconds(300)); CloseConnector("main"); } }; struct Worker : public Reactor { Worker(System *system, std::string name) : Reactor(system, name) {} virtual void Run() { std::shared_ptr channel; while (!(channel = system_->FindChannel("master", "main"))) std::this_thread::sleep_for(std::chrono::milliseconds(300)); std::this_thread::sleep_for(std::chrono::milliseconds(300)); CloseConnector("main"); } }; System system; system.Spawn("master"); system.Spawn("worker"); system.AwaitShutdown(); } TEST(SimpleSendTest, OneSimpleSend) { struct MessageInt : public Message { MessageInt(int xx) : x(xx) {} int x; }; struct Master : public Reactor { Master(System *system, std::string name) : Reactor(system, name) {} virtual void Run() { std::shared_ptr channel; while (!(channel = system_->FindChannel("worker", "main"))) std::this_thread::sleep_for(std::chrono::milliseconds(300)); channel->Send(123); CloseConnector("main"); // Write-end doesn't need to be closed because it's in RAII. } }; struct Worker : public Reactor { Worker(System *system, std::string name) : Reactor(system, name) {} virtual void Run() { EventStream* stream = main_.first; std::unique_ptr m_uptr = stream->AwaitEvent(); CloseConnector("main"); MessageInt* msg = dynamic_cast(m_uptr.get()); ASSERT_NE(msg, nullptr); ASSERT_EQ(msg->x, 123); } }; System system; 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(System *system, std::string name) : Reactor(system, name) {} virtual void Run() { std::shared_ptr channel; while (!(channel = system_->FindChannel("worker", "main"))) std::this_thread::sleep_for(std::chrono::milliseconds(300)); channel->Send(888); CloseConnector("main"); } }; struct Worker : public Reactor { Worker(System *system, std::string name) : Reactor(system, name) {} virtual void Run() { EventStream* stream = main_.first; stream->OnEvent([this](const MessageInt& msg, const EventStream::Subscription&) { ASSERT_EQ(msg.x, 888); CloseConnector("main"); }); } }; System system; 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(System *system, std::string name) : Reactor(system, name) {} virtual void Run() { std::shared_ptr channel; while (!(channel = system_->FindChannel("worker", "main"))) std::this_thread::sleep_for(std::chrono::milliseconds(300)); channel->Send(101); channel->Send(102); std::this_thread::sleep_for(std::chrono::milliseconds(300)); channel->Send(103); // should be ignored channel->Send(104); // should be ignored CloseConnector("main"); // Write-end doesn't need to be closed because it's in RAII. } }; struct Worker : public Reactor { Worker(System *system, std::string name) : Reactor(system, name) {} virtual void Run() { EventStream* stream = main_.first; std::unique_ptr m_uptr = stream->AwaitEvent(); CloseConnector("main"); MessageInt* msg = dynamic_cast(m_uptr.get()); ASSERT_NE(msg, nullptr); ASSERT_EQ(msg->x, 101); } }; System system; system.Spawn("master"); system.Spawn("worker"); 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(System *system, std::string name) : Reactor(system, name) {} virtual void Run() { std::shared_ptr channel; while (!(channel = system_->FindChannel("worker", "main"))) std::this_thread::sleep_for(std::chrono::milliseconds(300)); channel->Send(55); channel->Send(66); channel->Send(77); channel->Send(88); std::this_thread::sleep_for(std::chrono::milliseconds(300)); channel->Send('a'); channel->Send('b'); channel->Send('c'); channel->Send('d'); CloseConnector("main"); } }; struct Worker : public Reactor { Worker(System *system, std::string name) : Reactor(system, name) {} int num_msgs_received = 0; virtual void Run() { EventStream* stream = main_.first; stream->OnEvent([this](const MessageInt& msgint, const EventStream::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 EventStream::Subscription& subscription) { char c = msgchar.x; ++num_msgs_received; ASSERT_TRUE(c == 'a' || c == 'b' || c == 'c'); if (num_msgs_received == 5) { subscription.unsubscribe(); CloseConnector("main"); } }); } }; System system; 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(System *system, std::string name) : Reactor(system, name) {} virtual void Run() { std::shared_ptr channel; while (!(channel = system_->FindChannel("worker", "main"))) std::this_thread::sleep_for(std::chrono::milliseconds(300)); channel->Send(101); channel->Send('a'); channel->Send(103); channel->Send('b'); CloseConnector("main"); } }; struct Worker : public Reactor { Worker(System *system, std::string name) : Reactor(system, 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 EventStream::Subscription&) { ASSERT_TRUE(msgint.x == 101 || msgint.x == 103); ++correct_vals; main_.second->Send(); }); stream->OnEvent([this](const MessageChar& msgchar, const EventStream::Subscription&) { ASSERT_TRUE(msgchar.x == 'a' || msgchar.x == 'b'); ++correct_vals; main_.second->Send(); }); stream->OnEvent([this](const EndMessage&, const EventStream::Subscription&) { ASSERT_LE(correct_vals, 4); if (correct_vals == 4) { CloseConnector("main"); } }); } }; System system; system.Spawn("master"); system.Spawn("worker"); system.AwaitShutdown(); } int main(int argc, char **argv) { ::testing::InitGoogleTest(&argc, argv); return RUN_ALL_TESTS(); }