2017-08-22 22:29:23 +08:00
|
|
|
// This is a deprecated implementation! It is using the deprecated AwaitEvent, I'm changing it to use OnEvent. WIP
|
|
|
|
|
|
|
|
// #include <atomic>
|
|
|
|
// #include <chrono>
|
|
|
|
// #include <cstdlib>
|
|
|
|
// #include <iostream>
|
|
|
|
// #include <string>
|
|
|
|
// #include <thread>
|
|
|
|
// #include <vector>
|
|
|
|
|
|
|
|
// #include "reactors_distributed.hpp"
|
|
|
|
|
|
|
|
// const int NUM_WORKERS = 1;
|
|
|
|
|
2017-08-23 21:16:26 +08:00
|
|
|
// class Txn : public ReturnAddressMsg {
|
2017-08-22 22:29:23 +08:00
|
|
|
// public:
|
2017-08-23 21:16:26 +08:00
|
|
|
// Txn(std::string reactor, std::string channel, int64_t id) : ReturnAddressMsg(reactor, channel), id_(id) {}
|
2017-08-22 22:29:23 +08:00
|
|
|
// int64_t id() const { return id_; }
|
|
|
|
|
|
|
|
// template <class Archive>
|
|
|
|
// void serialize(Archive &archive) {
|
2017-08-23 21:16:26 +08:00
|
|
|
// archive(cereal::base_class<ReturnAddressMsg>(this), id_);
|
2017-08-22 22:29:23 +08:00
|
|
|
// }
|
|
|
|
|
|
|
|
// private:
|
|
|
|
// int64_t id_;
|
|
|
|
// };
|
|
|
|
|
|
|
|
// class CreateNodeTxn : public Txn {
|
|
|
|
// public:
|
|
|
|
// CreateNodeTxn(std::string reactor, std::string channel, int64_t id) : Txn(reactor, channel, id) {}
|
|
|
|
|
|
|
|
// template <class Archive>
|
|
|
|
// void serialize(Archive &archive) {
|
|
|
|
// archive(cereal::base_class<Txn>(this));
|
|
|
|
// }
|
|
|
|
// };
|
|
|
|
|
|
|
|
// class CountNodesTxn : public Txn {
|
|
|
|
// public:
|
|
|
|
// CountNodesTxn(std::string reactor, std::string channel, int64_t id) : Txn(reactor, channel, id) {}
|
|
|
|
|
|
|
|
// template <class Archive>
|
|
|
|
// void serialize(Archive &archive) {
|
|
|
|
// archive(cereal::base_class<Txn>(this));
|
|
|
|
// }
|
|
|
|
// };
|
|
|
|
|
|
|
|
// class CountNodesTxnResult : public Message {
|
|
|
|
// public:
|
|
|
|
// CountNodesTxnResult(int64_t count) : count_(count) {}
|
|
|
|
// int64_t count() const { return count_; }
|
|
|
|
|
|
|
|
// template <class Archive>
|
|
|
|
// void serialize(Archive &archive) {
|
|
|
|
// archive(count_);
|
|
|
|
// }
|
|
|
|
|
|
|
|
// private:
|
|
|
|
// int64_t count_;
|
|
|
|
// };
|
|
|
|
|
2017-08-23 21:16:26 +08:00
|
|
|
// class CommitRequest : public ReturnAddressMsg {
|
2017-08-22 22:29:23 +08:00
|
|
|
// public:
|
|
|
|
// CommitRequest(std::string reactor, std::string channel, int64_t worker_id)
|
2017-08-23 21:16:26 +08:00
|
|
|
// : ReturnAddressMsg(reactor, channel), worker_id_(worker_id) {}
|
2017-08-22 22:29:23 +08:00
|
|
|
// int64_t worker_id() { return worker_id_; }
|
|
|
|
|
|
|
|
// template <class Archive>
|
|
|
|
// void serialize(Archive &archive) {
|
2017-08-23 21:16:26 +08:00
|
|
|
// archive(cereal::base_class<ReturnAddressMsg>(this), worker_id_);
|
2017-08-22 22:29:23 +08:00
|
|
|
// }
|
|
|
|
|
|
|
|
// private:
|
|
|
|
// int64_t worker_id_;
|
|
|
|
// };
|
|
|
|
|
2017-08-23 21:16:26 +08:00
|
|
|
// class AbortRequest : public ReturnAddressMsg {
|
2017-08-22 22:29:23 +08:00
|
|
|
// public:
|
|
|
|
// AbortRequest(std::string reactor, std::string channel, int64_t worker_id)
|
2017-08-23 21:16:26 +08:00
|
|
|
// : ReturnAddressMsg(reactor, channel), worker_id_(worker_id) {}
|
2017-08-22 22:29:23 +08:00
|
|
|
// int64_t worker_id() { return worker_id_; }
|
|
|
|
|
|
|
|
// template <class Archive>
|
|
|
|
// void serialize(Archive &archive) {
|
2017-08-23 21:16:26 +08:00
|
|
|
// archive(cereal::base_class<ReturnAddressMsg>(this), worker_id_);
|
2017-08-22 22:29:23 +08:00
|
|
|
// }
|
|
|
|
|
|
|
|
// private:
|
|
|
|
// int64_t worker_id_;
|
|
|
|
// };
|
|
|
|
|
|
|
|
// class CommitDirective : public Message {
|
|
|
|
// template <class Archive>
|
|
|
|
// void serialize(Archive &archive) {
|
|
|
|
// archive(cereal::base_class<Message>(this));
|
|
|
|
// }
|
|
|
|
// };
|
|
|
|
|
|
|
|
// class AbortDirective : public Message {
|
|
|
|
// template <class Archive>
|
|
|
|
// void serialize(Archive &archive) {
|
|
|
|
// archive(cereal::base_class<Message>(this));
|
|
|
|
// }
|
|
|
|
// };
|
|
|
|
|
|
|
|
// class Query : public Message {
|
|
|
|
// public:
|
|
|
|
// Query(std::string query) : Message(), query_(query) {}
|
|
|
|
// std::string query() const { return query_; }
|
|
|
|
|
|
|
|
// template <class Archive>
|
|
|
|
// void serialize(Archive &archive) {
|
|
|
|
// archive(cereal::base_class<Message>(this), query_);
|
|
|
|
// }
|
|
|
|
|
|
|
|
// private:
|
|
|
|
// std::string query_;
|
|
|
|
// };
|
|
|
|
|
|
|
|
// class Quit : public Message {
|
|
|
|
// template <class Archive>
|
|
|
|
// void serialize(Archive &archive) {
|
|
|
|
// archive(cereal::base_class<Message>(this));
|
|
|
|
// }
|
|
|
|
// };
|
|
|
|
|
|
|
|
// class Master : public Reactor {
|
|
|
|
// public:
|
|
|
|
// Master(System *system, std::string name) : Reactor(system, name), next_xid_(1) {}
|
|
|
|
|
|
|
|
// virtual void Run() {
|
|
|
|
// auto stream = main_.first;
|
|
|
|
// FindWorkers();
|
|
|
|
|
|
|
|
// std::cout << "Master is active" << std::endl;
|
|
|
|
// while (true) {
|
|
|
|
// auto m = stream->AwaitEvent();
|
|
|
|
// if (Query *query = dynamic_cast<Query *>(m.get())) {
|
|
|
|
// ProcessQuery(query);
|
|
|
|
// break; // process only the first query
|
2017-08-23 21:16:26 +08:00
|
|
|
// } else if (ReturnAddressMsg *msg = dynamic_cast<ReturnAddressMsg *>(m.get())) {
|
|
|
|
// std::cout << "ReturnAddressMsg received!" << std::endl;
|
2017-08-22 22:29:23 +08:00
|
|
|
// std::cout << " Address: " << msg->Address() << std::endl;
|
|
|
|
// std::cout << " Port: " << msg->Port() << std::endl;
|
|
|
|
// std::cout << " Reactor: " << msg->ReactorName() << std::endl;
|
|
|
|
// std::cout << " Channel: " << msg->ChannelName() << std::endl;
|
|
|
|
// } else {
|
|
|
|
// std::cerr << "unknown message\n";
|
|
|
|
// exit(1);
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
|
|
|
|
// stream->OnEvent<Message>([this](const Message &msg, const EventStream::Subscription& subscription) {
|
|
|
|
// std::cout << "Processing Query via Callback" << std::endl;
|
|
|
|
// const Query &query =
|
|
|
|
// dynamic_cast<const Query &>(msg); // exception bad_cast
|
|
|
|
// ProcessQuery(&query);
|
|
|
|
// subscription.unsubscribe();
|
|
|
|
// });
|
|
|
|
// }
|
|
|
|
|
|
|
|
// private:
|
|
|
|
// void ProcessQuery(const Query *query) {
|
|
|
|
// if (query->query() == "create node") {
|
|
|
|
// PerformCreateNode();
|
|
|
|
// } else if (query->query() == "count nodes") {
|
|
|
|
// PerformCountNodes();
|
|
|
|
// } else {
|
|
|
|
// std::cout << "got query: " << query->query() << std::endl;
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
|
|
|
|
// void PerformCreateNode() {
|
|
|
|
// int worker_id = rand() % NUM_WORKERS;
|
|
|
|
// int64_t xid = GetTransactionId();
|
|
|
|
// std::string txn_channel_name = GetTxnName(xid);
|
2017-08-23 21:16:26 +08:00
|
|
|
// auto channel = Open(txn_channel_name);
|
|
|
|
// auto stream = channel.first;
|
2017-08-22 22:29:23 +08:00
|
|
|
|
|
|
|
// channels_[worker_id]->Send<CreateNodeTxn>("master", "main", xid);
|
|
|
|
// auto m = stream->AwaitEvent();
|
|
|
|
// if (CommitRequest *req = dynamic_cast<CommitRequest *>(m.get())) {
|
2017-08-23 21:16:26 +08:00
|
|
|
// req->GetReturnChannelWriter(system_)->Send<CommitDirective>();
|
2017-08-22 22:29:23 +08:00
|
|
|
// } else if (AbortRequest *req = dynamic_cast<AbortRequest *>(m.get())) {
|
2017-08-23 21:16:26 +08:00
|
|
|
// req->GetReturnChannelWriter(system_)->Send<AbortDirective>();
|
2017-08-22 22:29:23 +08:00
|
|
|
// } else {
|
|
|
|
// std::cerr << "unknown message\n";
|
|
|
|
// exit(1);
|
|
|
|
// }
|
2017-08-23 21:16:26 +08:00
|
|
|
// CloseChannel(txn_channel_name);
|
2017-08-22 22:29:23 +08:00
|
|
|
// }
|
|
|
|
|
|
|
|
// void PerformCountNodes() {
|
|
|
|
// int64_t xid = GetTransactionId();
|
|
|
|
// std::string txn_channel_name = GetTxnName(xid);
|
2017-08-23 21:16:26 +08:00
|
|
|
// auto channel = Open(txn_channel_name);
|
|
|
|
// auto stream = channel.first;
|
2017-08-22 22:29:23 +08:00
|
|
|
// for (int w_id = 0; w_id < NUM_WORKERS; ++w_id)
|
|
|
|
// channels_[w_id]->Send<CountNodesTxn>("master", "main", xid);
|
|
|
|
|
|
|
|
// std::vector<std::shared_ptr<Channel>> txn_channels;
|
|
|
|
// txn_channels.resize(NUM_WORKERS, nullptr);
|
|
|
|
// bool commit = true;
|
|
|
|
// for (int responds = 0; responds < NUM_WORKERS; ++responds) {
|
|
|
|
// auto m = stream->AwaitEvent();
|
|
|
|
// if (CommitRequest *req = dynamic_cast<CommitRequest *>(m.get())) {
|
2017-08-23 21:16:26 +08:00
|
|
|
// txn_channels[req->worker_id()] = req->GetReturnChannelWriter(system_);
|
2017-08-22 22:29:23 +08:00
|
|
|
// commit &= true;
|
|
|
|
// } else if (AbortRequest *req = dynamic_cast<AbortRequest *>(m.get())) {
|
2017-08-23 21:16:26 +08:00
|
|
|
// txn_channels[req->worker_id()] = req->GetReturnChannelWriter(system_);
|
2017-08-22 22:29:23 +08:00
|
|
|
// commit = false;
|
|
|
|
// } else {
|
|
|
|
// std::cerr << "unknown message\n";
|
|
|
|
// exit(1);
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
|
|
|
|
// if (commit) {
|
|
|
|
// for (int w_id = 0; w_id < NUM_WORKERS; ++w_id)
|
|
|
|
// txn_channels[w_id]->Send<CommitDirective>();
|
|
|
|
// } else {
|
|
|
|
// for (int w_id = 0; w_id < NUM_WORKERS; ++w_id)
|
|
|
|
// txn_channels[w_id]->Send<AbortDirective>();
|
|
|
|
// }
|
|
|
|
|
|
|
|
// int64_t count = 0;
|
|
|
|
// for (int responds = 0; responds < NUM_WORKERS; ++responds) {
|
|
|
|
// auto m = stream->AwaitEvent();
|
|
|
|
// if (CountNodesTxnResult *cnt =
|
|
|
|
// dynamic_cast<CountNodesTxnResult *>(m.get())) {
|
|
|
|
// count += cnt->count();
|
|
|
|
// } else {
|
|
|
|
// std::cerr << "unknown message\n";
|
|
|
|
// exit(1);
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
|
2017-08-23 21:16:26 +08:00
|
|
|
// CloseChannel(txn_channel_name);
|
2017-08-22 22:29:23 +08:00
|
|
|
// std::cout << "graph has " << count << " vertices" << std::endl;
|
|
|
|
// }
|
|
|
|
|
|
|
|
// int64_t GetTransactionId() { return next_xid_++; }
|
|
|
|
|
|
|
|
// std::string GetWorkerName(int worker_id) {
|
|
|
|
// return "worker" + std::to_string(worker_id);
|
|
|
|
// }
|
|
|
|
|
|
|
|
// std::string GetTxnName(int txn_id) { return "txn" + std::to_string(txn_id); }
|
|
|
|
|
|
|
|
// void FindWorkers() {
|
|
|
|
// channels_.resize(NUM_WORKERS, nullptr);
|
|
|
|
// int workers_found = 0;
|
|
|
|
// while (workers_found < NUM_WORKERS) {
|
|
|
|
// for (int64_t w_id = 0; w_id < NUM_WORKERS; ++w_id) {
|
|
|
|
// if (channels_[w_id] == nullptr) {
|
|
|
|
// // TODO: Resolve worker channel using the network service.
|
|
|
|
// channels_[w_id] = system_->FindChannel(GetWorkerName(w_id), "main");
|
|
|
|
// if (channels_[w_id] != nullptr) ++workers_found;
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// if (workers_found < NUM_WORKERS)
|
|
|
|
// std::this_thread::sleep_for(std::chrono::seconds(1));
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
|
|
|
|
// // TODO: Why is master atomic, it should be unique?
|
|
|
|
// std::atomic<int64_t> next_xid_;
|
|
|
|
// std::vector<std::shared_ptr<Channel>> channels_;
|
|
|
|
// };
|
|
|
|
|
|
|
|
// class Worker : public Reactor {
|
|
|
|
// public:
|
|
|
|
// Worker(System *system, std::string name, int64_t id) : Reactor(system, name),
|
|
|
|
// worker_id_(id) {}
|
|
|
|
|
|
|
|
// virtual void Run() {
|
|
|
|
// std::cout << "worker " << worker_id_ << " is active" << std::endl;
|
|
|
|
// auto stream = main_.first;
|
|
|
|
// FindMaster();
|
|
|
|
// while (true) {
|
|
|
|
// auto m = stream->AwaitEvent();
|
|
|
|
// if (Txn *txn = dynamic_cast<Txn *>(m.get())) {
|
|
|
|
// HandleTransaction(txn);
|
|
|
|
// } else {
|
|
|
|
// std::cerr << "unknown message\n";
|
|
|
|
// exit(1);
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
|
|
|
|
// private:
|
|
|
|
// void HandleTransaction(Txn *txn) {
|
|
|
|
// if (CreateNodeTxn *create_txn = dynamic_cast<CreateNodeTxn *>(txn)) {
|
|
|
|
// HandleCreateNode(create_txn);
|
|
|
|
// } else if (CountNodesTxn *cnt_txn = dynamic_cast<CountNodesTxn *>(txn)) {
|
|
|
|
// HandleCountNodes(cnt_txn);
|
|
|
|
// } else {
|
|
|
|
// std::cerr << "unknown transaction\n";
|
|
|
|
// exit(1);
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
|
|
|
|
// void HandleCreateNode(CreateNodeTxn *txn) {
|
2017-08-23 21:16:26 +08:00
|
|
|
// auto channel = Open(GetTxnChannelName(txn->id()));
|
|
|
|
// auto stream = channel.first;
|
|
|
|
// auto masterChannel = txn->GetReturnChannelWriter(system_);
|
2017-08-22 22:29:23 +08:00
|
|
|
// // TODO: Do the actual commit.
|
|
|
|
// masterChannel->Send<CommitRequest>("master", "main", worker_id_);
|
|
|
|
// auto m = stream->AwaitEvent();
|
|
|
|
// if (dynamic_cast<CommitDirective *>(m.get())) {
|
|
|
|
// // TODO: storage_.CreateNode();
|
|
|
|
// } else if (dynamic_cast<AbortDirective *>(m.get())) {
|
|
|
|
// // TODO: Rollback.
|
|
|
|
// } else {
|
|
|
|
// std::cerr << "unknown message\n";
|
|
|
|
// exit(1);
|
|
|
|
// }
|
2017-08-23 21:16:26 +08:00
|
|
|
// CloseChannel(GetTxnChannelName(txn->id()));
|
2017-08-22 22:29:23 +08:00
|
|
|
// }
|
|
|
|
|
|
|
|
// void HandleCountNodes(CountNodesTxn *txn) {
|
2017-08-23 21:16:26 +08:00
|
|
|
// auto channel = Open(GetTxnChannelName(txn->id()));
|
|
|
|
// auto stream = channel.first;
|
|
|
|
// auto masterChannel = txn->GetReturnChannelWriter(system_);
|
2017-08-22 22:29:23 +08:00
|
|
|
|
|
|
|
// // TODO: Fix this hack -- use the storage.
|
|
|
|
// int num = 123;
|
|
|
|
|
|
|
|
// masterChannel->Send<CommitRequest>("master", "main", worker_id_);
|
|
|
|
// auto m = stream->AwaitEvent();
|
|
|
|
// if (dynamic_cast<CommitDirective *>(m.get())) {
|
|
|
|
// masterChannel->Send<CountNodesTxnResult>(num);
|
|
|
|
// } else if (dynamic_cast<AbortDirective *>(m.get())) {
|
|
|
|
// // send nothing
|
|
|
|
// } else {
|
|
|
|
// std::cerr << "unknown message\n";
|
|
|
|
// exit(1);
|
|
|
|
// }
|
2017-08-23 21:16:26 +08:00
|
|
|
// CloseChannel(GetTxnChannelName(txn->id()));
|
2017-08-22 22:29:23 +08:00
|
|
|
// }
|
|
|
|
|
|
|
|
// // TODO: Don't repeat code from Master.
|
|
|
|
// std::string GetTxnChannelName(int64_t transaction_id) {
|
|
|
|
// return "txn" + std::to_string(transaction_id);
|
|
|
|
// }
|
|
|
|
|
|
|
|
// void FindMaster() {
|
|
|
|
// // TODO: Replace with network service and channel resolution.
|
|
|
|
// while (!(master_channel_ = system_->FindChannel("master", "main")))
|
|
|
|
// std::this_thread::sleep_for(std::chrono::seconds(1));
|
|
|
|
// }
|
|
|
|
|
|
|
|
// std::shared_ptr<Channel> master_channel_ = nullptr;
|
|
|
|
// int worker_id_;
|
|
|
|
// };
|
|
|
|
|
|
|
|
// void ClientMain(System *system) {
|
|
|
|
// std::shared_ptr<Channel> channel = nullptr;
|
|
|
|
// // TODO: Replace this with network channel resolution.
|
|
|
|
// while (!(channel = system->FindChannel("master", "main")))
|
|
|
|
// std::this_thread::sleep_for(std::chrono::seconds(1));
|
|
|
|
// std::cout << "I/O Client Main active" << std::endl;
|
|
|
|
|
|
|
|
// bool active = true;
|
|
|
|
// while (active) {
|
|
|
|
// std::string s;
|
|
|
|
// std::getline(std::cin, s);
|
|
|
|
// if (s == "quit") {
|
|
|
|
// active = false;
|
|
|
|
// channel->Send<Quit>();
|
|
|
|
// } else {
|
|
|
|
// channel->Send<Query>(s);
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
|
|
|
|
|
|
|
|
// int main(int argc, char *argv[]) {
|
|
|
|
// //google::InitGoogleLogging(argv[0]);
|
|
|
|
// gflags::ParseCommandLineFlags(&argc, &argv, true);
|
|
|
|
// System system;
|
|
|
|
// system.Spawn<Master>("master");
|
|
|
|
// std::thread client(ClientMain, &system);
|
|
|
|
// for (int i = 0; i < NUM_WORKERS; ++i)
|
|
|
|
// system.Spawn<Worker>("worker" + std::to_string(i), i);
|
|
|
|
// system.AwaitShutdown();
|
|
|
|
// return 0;
|
|
|
|
// }
|