2017-08-14 17:44:20 +08:00
|
|
|
// command to run:
|
|
|
|
// gnome-terminal --tab -e './network_chat --port 10000 --minloglevel 2' --tab -e './network_chat --port 10001 --minloglevel 2'
|
|
|
|
|
2017-08-17 21:32:12 +08:00
|
|
|
#include "reactors_distributed.hpp"
|
2017-08-07 22:04:24 +08:00
|
|
|
|
2017-08-23 21:16:26 +08:00
|
|
|
class ChatMessage : public ReturnAddressMsg {
|
2017-08-07 22:04:24 +08:00
|
|
|
public:
|
2017-08-23 21:16:26 +08:00
|
|
|
ChatMessage() : ReturnAddressMsg(), message_("") {}
|
2017-08-07 22:04:24 +08:00
|
|
|
|
|
|
|
ChatMessage(std::string reactor, std::string channel, std::string message)
|
2017-08-23 21:16:26 +08:00
|
|
|
: ReturnAddressMsg(reactor, channel), message_(message) {}
|
2017-08-07 22:04:24 +08:00
|
|
|
|
|
|
|
std::string Message() const { return message_; }
|
|
|
|
|
|
|
|
template <class Archive>
|
|
|
|
void serialize(Archive &ar) {
|
2017-08-23 21:16:26 +08:00
|
|
|
ar(cereal::base_class<ReturnAddressMsg>(this), message_);
|
2017-08-07 22:04:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
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 <class Archive>
|
|
|
|
void serialize(Archive &ar) {
|
|
|
|
ar(cereal::base_class<ChatMessage>(this));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
CEREAL_REGISTER_TYPE(ChatACK);
|
|
|
|
|
2017-08-22 23:13:12 +08:00
|
|
|
class ChatServer : public Reactor {
|
2017-08-07 22:04:24 +08:00
|
|
|
public:
|
2017-08-22 23:13:12 +08:00
|
|
|
ChatServer(std::string name)
|
|
|
|
: Reactor(name) {}
|
2017-08-07 22:04:24 +08:00
|
|
|
|
|
|
|
virtual void Run() {
|
|
|
|
std::cout << "ChatServer is active" << std::endl;
|
|
|
|
|
|
|
|
auto chat = Open("chat").first;
|
|
|
|
|
2017-08-14 17:44:20 +08:00
|
|
|
chat->OnEvent<ChatACK>([](const ChatACK& ack, const EventStream::Subscription&) {
|
|
|
|
std::cout << "Received ACK from " << ack.Address() << ":"
|
|
|
|
<< ack.Port() << " -> '" << ack.Message() << "'"
|
2017-08-07 22:04:24 +08:00
|
|
|
<< std::endl;
|
2017-08-14 17:44:20 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
chat->OnEvent<ChatMessage>([this](const ChatMessage& msg, const EventStream::Subscription&) {
|
|
|
|
std::cout << "Received message from " << msg.Address() << ":"
|
|
|
|
<< msg.Port() << " -> '" << msg.Message() << "'"
|
2017-08-07 22:04:24 +08:00
|
|
|
<< std::endl;
|
2017-08-23 21:16:26 +08:00
|
|
|
auto channel = msg.GetReturnChannelWriter();
|
2017-08-07 22:04:24 +08:00
|
|
|
if (channel != nullptr) {
|
2017-08-14 17:44:20 +08:00
|
|
|
channel->Send<ChatACK>("server", "chat", msg.Message());
|
2017-08-07 22:04:24 +08:00
|
|
|
}
|
2017-08-14 17:44:20 +08:00
|
|
|
});
|
2017-08-07 22:04:24 +08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2017-08-22 23:13:12 +08:00
|
|
|
class ChatClient : public Reactor {
|
2017-08-07 22:04:24 +08:00
|
|
|
public:
|
2017-08-22 23:13:12 +08:00
|
|
|
ChatClient(std::string name)
|
|
|
|
: Reactor(name) {}
|
2017-08-07 22:04:24 +08:00
|
|
|
|
|
|
|
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 =
|
2017-08-22 23:13:12 +08:00
|
|
|
Distributed::GetInstance().network().Resolve(address, port, "server", "chat");
|
2017-08-07 22:04:24 +08:00
|
|
|
if (channel != nullptr) {
|
2017-08-08 23:36:07 +08:00
|
|
|
channel->Send<ChatMessage>("server", "chat", message);
|
2017-08-07 22:04:24 +08:00
|
|
|
} else {
|
|
|
|
std::cerr << "Couldn't resolve that server!" << std::endl;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
int main(int argc, char *argv[]) {
|
|
|
|
gflags::ParseCommandLineFlags(&argc, &argv, true);
|
2017-08-22 22:29:23 +08:00
|
|
|
System& system = System::GetInstance();
|
2017-08-22 23:13:12 +08:00
|
|
|
Distributed &distributed = Distributed::GetInstance();
|
2017-08-17 20:46:34 +08:00
|
|
|
distributed.StartServices();
|
2017-08-22 23:13:12 +08:00
|
|
|
system.Spawn<ChatServer>("server");
|
|
|
|
system.Spawn<ChatClient>("client");
|
2017-08-07 22:04:24 +08:00
|
|
|
system.AwaitShutdown();
|
2017-08-17 20:46:34 +08:00
|
|
|
distributed.StopServices();
|
2017-08-07 22:04:24 +08:00
|
|
|
return 0;
|
|
|
|
}
|