// command to run: // gnome-terminal --tab -e './network_chat --port 10000 --minloglevel 2' --tab -e './network_chat --port 10001 --minloglevel 2' #include "reactors_distributed.hpp" class ChatMessage : public SenderMessage { public: ChatMessage() : SenderMessage(), message_("") {} ChatMessage(std::string reactor, std::string channel, std::string message) : SenderMessage(reactor, channel), message_(message) {} std::string Message() const { return message_; } template void serialize(Archive &ar) { ar(cereal::base_class(this), message_); } private: std::string message_; }; CEREAL_REGISTER_TYPE(ChatMessage); class ChatACK : public ChatMessage { public: ChatACK() : ChatMessage() {} ChatACK(std::string reactor, std::string channel, std::string message) : ChatMessage(reactor, channel, message) {} template void serialize(Archive &ar) { ar(cereal::base_class(this)); } }; CEREAL_REGISTER_TYPE(ChatACK); class ChatServer : public DistributedReactor { public: ChatServer(System *system, std::string name, Distributed &distributed) : DistributedReactor(system, name, distributed) {} virtual void Run() { std::cout << "ChatServer is active" << std::endl; auto chat = Open("chat").first; chat->OnEvent([](const ChatACK& ack, const EventStream::Subscription&) { std::cout << "Received ACK from " << ack.Address() << ":" << ack.Port() << " -> '" << ack.Message() << "'" << std::endl; }); chat->OnEvent([this](const ChatMessage& msg, const EventStream::Subscription&) { std::cout << "Received message from " << msg.Address() << ":" << msg.Port() << " -> '" << msg.Message() << "'" << std::endl; auto channel = msg.GetChannelToSender(system_); if (channel != nullptr) { channel->Send("server", "chat", msg.Message()); } }); } }; class ChatClient : public DistributedReactor { public: ChatClient(System *system, std::string name, Distributed &distributed) : DistributedReactor(system, name, distributed) {} virtual void Run() { std::cout << "ChatClient is active" << std::endl; std::string address, message; uint16_t port; while (true) { std::cout << "Enter IP, port and message to send." << std::endl; std::cin >> address >> port >> message; auto channel = distributed_.network().Resolve(address, port, "server", "chat"); if (channel != nullptr) { channel->Send("server", "chat", message); } else { std::cerr << "Couldn't resolve that server!" << std::endl; } } } }; int main(int argc, char *argv[]) { gflags::ParseCommandLineFlags(&argc, &argv, true); System system; Distributed distributed(system); distributed.StartServices(); system.Spawn("server", distributed); system.Spawn("client", distributed); system.AwaitShutdown(); distributed.StopServices(); return 0; }