#include #include #include #include #include #include "utils/algorithm.hpp" #include "utils/string.hpp" #include "utils/thread/sync.hpp" #include "utils/timer.hpp" #include "common.hpp" DEFINE_int32(num_workers, 1, "Number of workers"); DEFINE_string(input, "", "Input file"); DEFINE_string(output, "", "Output file"); DEFINE_string(address, "127.0.0.1", "Server address"); DEFINE_int32(port, 7687, "Server port"); DEFINE_string(username, "", "Username for the database"); DEFINE_string(password, "", "Password for the database"); DEFINE_bool(use_ssl, false, "Set to true to connect with SSL to the server."); using communication::bolt::Value; const int MAX_RETRIES = 50; void PrintJsonMetadata( std::ostream &os, const std::vector> &metadata) { os << "["; utils::PrintIterable(os, metadata, ", ", [](auto &stream, const auto &item) { PrintJsonValue(stream, item); }); os << "]"; } void PrintSummary(std::ostream &os, double duration, const std::vector> &metadata) { os << "{\"wall_time\": " << duration << ", " << "\"metadatas\": "; PrintJsonMetadata(os, metadata); os << "}\n"; } void ExecuteQueries(const std::vector &queries, std::ostream &ostream) { std::vector threads; utils::SpinLock spinlock; uint64_t last = 0; std::vector> metadata; metadata.resize(queries.size()); utils::Timer timer; for (int i = 0; i < FLAGS_num_workers; ++i) { threads.push_back(std::thread([&]() { Endpoint endpoint(FLAGS_address, FLAGS_port); ClientContext context(FLAGS_use_ssl); Client client(&context); if (!client.Connect(endpoint, FLAGS_username, FLAGS_password)) { LOG(FATAL) << "Couldn't connect to " << endpoint; } std::string str; while (true) { uint64_t pos; { std::lock_guard lock(spinlock); if (last == queries.size()) { break; } pos = last++; str = queries[pos]; } try { metadata[pos] = ExecuteNTimesTillSuccess(client, str, {}, MAX_RETRIES) .first.metadata; } catch (const utils::BasicException &e) { LOG(FATAL) << "Could not execute query '" << str << "' " << MAX_RETRIES << " times! Error message: " << e.what(); } } client.Close(); })); } for (int i = 0; i < FLAGS_num_workers; ++i) { threads[i].join(); } auto elapsed = timer.Elapsed(); double duration = elapsed.count(); PrintSummary(ostream, duration, metadata); } int main(int argc, char **argv) { gflags::ParseCommandLineFlags(&argc, &argv, true); google::InitGoogleLogging(argv[0]); communication::Init(); std::ifstream ifile; std::istream *istream{&std::cin}; std::ofstream ofile; std::ostream *ostream{&std::cout}; if (FLAGS_input != "") { ifile.open(FLAGS_input); istream = &ifile; } if (FLAGS_output != "") { ofile.open(FLAGS_output); ostream = &ofile; } while (!istream->eof()) { std::vector queries; std::string query; while (std::getline(*istream, query) && utils::Trim(query) != "" && utils::Trim(query) != ";") { queries.push_back(query); } ExecuteQueries(queries, *ostream); } return 0; }