// This is a deprecated implementation! It is using the deprecated AwaitEvent, I'm changing it to use OnEvent. WIP // #include // #include // #include // #include // #include // #include // #include // #include "reactors_distributed.hpp" // const int NUM_WORKERS = 1; // class Txn : public ReturnAddressMsg { // public: // Txn(std::string reactor, std::string channel, int64_t id) : ReturnAddressMsg(reactor, channel), id_(id) {} // int64_t id() const { return id_; } // template // void serialize(Archive &archive) { // archive(cereal::base_class(this), id_); // } // private: // int64_t id_; // }; // class CreateNodeTxn : public Txn { // public: // CreateNodeTxn(std::string reactor, std::string channel, int64_t id) : Txn(reactor, channel, id) {} // template // void serialize(Archive &archive) { // archive(cereal::base_class(this)); // } // }; // class CountNodesTxn : public Txn { // public: // CountNodesTxn(std::string reactor, std::string channel, int64_t id) : Txn(reactor, channel, id) {} // template // void serialize(Archive &archive) { // archive(cereal::base_class(this)); // } // }; // class CountNodesTxnResult : public Message { // public: // CountNodesTxnResult(int64_t count) : count_(count) {} // int64_t count() const { return count_; } // template // void serialize(Archive &archive) { // archive(count_); // } // private: // int64_t count_; // }; // class CommitRequest : public ReturnAddressMsg { // public: // CommitRequest(std::string reactor, std::string channel, int64_t worker_id) // : ReturnAddressMsg(reactor, channel), worker_id_(worker_id) {} // int64_t worker_id() { return worker_id_; } // template // void serialize(Archive &archive) { // archive(cereal::base_class(this), worker_id_); // } // private: // int64_t worker_id_; // }; // class AbortRequest : public ReturnAddressMsg { // public: // AbortRequest(std::string reactor, std::string channel, int64_t worker_id) // : ReturnAddressMsg(reactor, channel), worker_id_(worker_id) {} // int64_t worker_id() { return worker_id_; } // template // void serialize(Archive &archive) { // archive(cereal::base_class(this), worker_id_); // } // private: // int64_t worker_id_; // }; // class CommitDirective : public Message { // template // void serialize(Archive &archive) { // archive(cereal::base_class(this)); // } // }; // class AbortDirective : public Message { // template // void serialize(Archive &archive) { // archive(cereal::base_class(this)); // } // }; // class Query : public Message { // public: // Query(std::string query) : Message(), query_(query) {} // std::string query() const { return query_; } // template // void serialize(Archive &archive) { // archive(cereal::base_class(this), query_); // } // private: // std::string query_; // }; // class Quit : public Message { // template // void serialize(Archive &archive) { // archive(cereal::base_class(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(m.get())) { // ProcessQuery(query); // break; // process only the first query // } else if (ReturnAddressMsg *msg = dynamic_cast(m.get())) { // std::cout << "ReturnAddressMsg received!" << std::endl; // 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([this](const Message &msg, const Subscription& subscription) { // std::cout << "Processing Query via Callback" << std::endl; // const Query &query = // dynamic_cast(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); // auto channel = Open(txn_channel_name); // auto stream = channel.first; // channels_[worker_id]->Send("master", "main", xid); // auto m = stream->AwaitEvent(); // if (CommitRequest *req = dynamic_cast(m.get())) { // req->GetReturnChannelWriter(system_)->Send(); // } else if (AbortRequest *req = dynamic_cast(m.get())) { // req->GetReturnChannelWriter(system_)->Send(); // } else { // std::cerr << "unknown message\n"; // exit(1); // } // CloseChannel(txn_channel_name); // } // void PerformCountNodes() { // int64_t xid = GetTransactionId(); // std::string txn_channel_name = GetTxnName(xid); // auto channel = Open(txn_channel_name); // auto stream = channel.first; // for (int w_id = 0; w_id < NUM_WORKERS; ++w_id) // channels_[w_id]->Send("master", "main", xid); // std::vector> 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(m.get())) { // txn_channels[req->worker_id()] = req->GetReturnChannelWriter(system_); // commit &= true; // } else if (AbortRequest *req = dynamic_cast(m.get())) { // txn_channels[req->worker_id()] = req->GetReturnChannelWriter(system_); // 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(); // } else { // for (int w_id = 0; w_id < NUM_WORKERS; ++w_id) // txn_channels[w_id]->Send(); // } // int64_t count = 0; // for (int responds = 0; responds < NUM_WORKERS; ++responds) { // auto m = stream->AwaitEvent(); // if (CountNodesTxnResult *cnt = // dynamic_cast(m.get())) { // count += cnt->count(); // } else { // std::cerr << "unknown message\n"; // exit(1); // } // } // CloseChannel(txn_channel_name); // 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 next_xid_; // std::vector> 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(m.get())) { // HandleTransaction(txn); // } else { // std::cerr << "unknown message\n"; // exit(1); // } // } // } // private: // void HandleTransaction(Txn *txn) { // if (CreateNodeTxn *create_txn = dynamic_cast(txn)) { // HandleCreateNode(create_txn); // } else if (CountNodesTxn *cnt_txn = dynamic_cast(txn)) { // HandleCountNodes(cnt_txn); // } else { // std::cerr << "unknown transaction\n"; // exit(1); // } // } // void HandleCreateNode(CreateNodeTxn *txn) { // auto channel = Open(GetTxnChannelName(txn->id())); // auto stream = channel.first; // auto masterChannel = txn->GetReturnChannelWriter(system_); // // TODO: Do the actual commit. // masterChannel->Send("master", "main", worker_id_); // auto m = stream->AwaitEvent(); // if (dynamic_cast(m.get())) { // // TODO: storage_.CreateNode(); // } else if (dynamic_cast(m.get())) { // // TODO: Rollback. // } else { // std::cerr << "unknown message\n"; // exit(1); // } // CloseChannel(GetTxnChannelName(txn->id())); // } // void HandleCountNodes(CountNodesTxn *txn) { // auto channel = Open(GetTxnChannelName(txn->id())); // auto stream = channel.first; // auto masterChannel = txn->GetReturnChannelWriter(system_); // // TODO: Fix this hack -- use the storage. // int num = 123; // masterChannel->Send("master", "main", worker_id_); // auto m = stream->AwaitEvent(); // if (dynamic_cast(m.get())) { // masterChannel->Send(num); // } else if (dynamic_cast(m.get())) { // // send nothing // } else { // std::cerr << "unknown message\n"; // exit(1); // } // CloseChannel(GetTxnChannelName(txn->id())); // } // // 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 master_channel_ = nullptr; // int worker_id_; // }; // void ClientMain(System *system) { // std::shared_ptr 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(); // } else { // channel->Send(s); // } // } // } // int main(int argc, char *argv[]) { // //google::InitGoogleLogging(argv[0]); // gflags::ParseCommandLineFlags(&argc, &argv, true); // System system; // system.Spawn("master"); // std::thread client(ClientMain, &system); // for (int i = 0; i < NUM_WORKERS; ++i) // system.Spawn("worker" + std::to_string(i), i); // system.AwaitShutdown(); // return 0; // }