#include "Commands.h" #include #include "KeeperClient.h" #include "Parsers/CommonParsers.h" namespace DB { namespace ErrorCodes { extern const int KEEPER_EXCEPTION; extern const int UNEXPECTED_ZOOKEEPER_ERROR; } bool LSCommand::parse(IParser::Pos & pos, std::shared_ptr & node, Expected & expected) const { String path; if (!parseKeeperPath(pos, expected, path)) return true; node->args.push_back(std::move(path)); return true; } void LSCommand::execute(const ASTKeeperQuery * query, KeeperClient * client) const { String path; if (!query->args.empty()) path = client->getAbsolutePath(query->args[0].safeGet()); else path = client->cwd; auto children = client->zookeeper->getChildren(path); std::sort(children.begin(), children.end()); bool need_space = false; for (const auto & child : children) { if (std::exchange(need_space, true)) std::cout << " "; std::cout << child; } std::cout << "\n"; } bool CDCommand::parse(IParser::Pos & pos, std::shared_ptr & node, Expected & expected) const { String path; if (!parseKeeperPath(pos, expected, path)) return true; node->args.push_back(std::move(path)); return true; } void CDCommand::execute(const ASTKeeperQuery * query, KeeperClient * client) const { if (query->args.empty()) return; auto new_path = client->getAbsolutePath(query->args[0].safeGet()); if (!client->zookeeper->exists(new_path)) std::cerr << "Path " << new_path << " does not exists\n"; else client->cwd = new_path; } bool SetCommand::parse(IParser::Pos & pos, std::shared_ptr & node, Expected & expected) const { String path; if (!parseKeeperPath(pos, expected, path)) return false; node->args.push_back(std::move(path)); String arg; if (!parseKeeperArg(pos, expected, arg)) return false; node->args.push_back(std::move(arg)); ASTPtr version; if (ParserNumber{}.parse(pos, version, expected)) node->args.push_back(version->as().value); return true; } void SetCommand::execute(const ASTKeeperQuery * query, KeeperClient * client) const { if (query->args.size() == 2) client->zookeeper->set(client->getAbsolutePath(query->args[0].safeGet()), query->args[1].safeGet()); else client->zookeeper->set( client->getAbsolutePath(query->args[0].safeGet()), query->args[1].safeGet(), static_cast(query->args[2].get())); } bool CreateCommand::parse(IParser::Pos & pos, std::shared_ptr & node, Expected & expected) const { String path; if (!parseKeeperPath(pos, expected, path)) return false; node->args.push_back(std::move(path)); String arg; if (!parseKeeperArg(pos, expected, arg)) return false; node->args.push_back(std::move(arg)); int mode = zkutil::CreateMode::Persistent; if (ParserKeyword(Keyword::PERSISTENT).ignore(pos, expected)) mode = zkutil::CreateMode::Persistent; else if (ParserKeyword(Keyword::EPHEMERAL).ignore(pos, expected)) mode = zkutil::CreateMode::Ephemeral; else if (ParserKeyword(Keyword::EPHEMERAL_SEQUENTIAL).ignore(pos, expected)) mode = zkutil::CreateMode::EphemeralSequential; else if (ParserKeyword(Keyword::PERSISTENT_SEQUENTIAL).ignore(pos, expected)) mode = zkutil::CreateMode::PersistentSequential; node->args.push_back(std::move(mode)); return true; } void CreateCommand::execute(const ASTKeeperQuery * query, KeeperClient * client) const { client->zookeeper->create( client->getAbsolutePath(query->args[0].safeGet()), query->args[1].safeGet(), static_cast(query->args[2].safeGet())); } bool TouchCommand::parse(IParser::Pos & pos, std::shared_ptr & node, Expected & expected) const { String arg; if (!parseKeeperPath(pos, expected, arg)) return false; node->args.push_back(std::move(arg)); return true; } void TouchCommand::execute(const ASTKeeperQuery * query, KeeperClient * client) const { client->zookeeper->createIfNotExists(client->getAbsolutePath(query->args[0].safeGet()), ""); } bool GetCommand::parse(IParser::Pos & pos, std::shared_ptr & node, Expected & expected) const { String path; if (!parseKeeperPath(pos, expected, path)) return false; node->args.push_back(std::move(path)); return true; } void GetCommand::execute(const ASTKeeperQuery * query, KeeperClient * client) const { std::cout << client->zookeeper->get(client->getAbsolutePath(query->args[0].safeGet())) << "\n"; } bool ExistsCommand::parse(IParser::Pos & pos, std::shared_ptr & node, DB::Expected & expected) const { String path; if (!parseKeeperPath(pos, expected, path)) return false; node->args.push_back(std::move(path)); return true; } void ExistsCommand::execute(const DB::ASTKeeperQuery * query, DB::KeeperClient * client) const { std::cout << client->zookeeper->exists(client->getAbsolutePath(query->args[0].safeGet())) << "\n"; } bool GetStatCommand::parse(IParser::Pos & pos, std::shared_ptr & node, Expected & expected) const { String path; if (!parseKeeperPath(pos, expected, path)) return true; node->args.push_back(std::move(path)); return true; } void GetStatCommand::execute(const ASTKeeperQuery * query, KeeperClient * client) const { Coordination::Stat stat; String path; if (!query->args.empty()) path = client->getAbsolutePath(query->args[0].safeGet()); else path = client->cwd; client->zookeeper->get(path, &stat); std::cout << "cZxid = " << stat.czxid << "\n"; std::cout << "mZxid = " << stat.mzxid << "\n"; std::cout << "pZxid = " << stat.pzxid << "\n"; std::cout << "ctime = " << stat.ctime << "\n"; std::cout << "mtime = " << stat.mtime << "\n"; std::cout << "version = " << stat.version << "\n"; std::cout << "cversion = " << stat.cversion << "\n"; std::cout << "aversion = " << stat.aversion << "\n"; std::cout << "ephemeralOwner = " << stat.ephemeralOwner << "\n"; std::cout << "dataLength = " << stat.dataLength << "\n"; std::cout << "numChildren = " << stat.numChildren << "\n"; } bool FindSuperNodes::parse(IParser::Pos & pos, std::shared_ptr & node, Expected & expected) const { ASTPtr threshold; if (!ParserUnsignedInteger{}.parse(pos, threshold, expected)) return false; node->args.push_back(threshold->as().value); ParserToken{TokenType::Whitespace}.ignore(pos); String path; if (!parseKeeperPath(pos, expected, path)) path = "."; node->args.push_back(std::move(path)); return true; } void FindSuperNodes::execute(const ASTKeeperQuery * query, KeeperClient * client) const { auto threshold = query->args[0].safeGet(); auto path = client->getAbsolutePath(query->args[1].safeGet()); Coordination::Stat stat; if (!client->zookeeper->exists(path, &stat)) return; /// It is ok if node was deleted meanwhile if (stat.numChildren >= static_cast(threshold)) std::cout << static_cast(path) << "\t" << stat.numChildren << "\n"; Strings children; auto status = client->zookeeper->tryGetChildren(path, children); if (status == Coordination::Error::ZNONODE) return; /// It is ok if node was deleted meanwhile else if (status != Coordination::Error::ZOK) throw DB::Exception(DB::ErrorCodes::KEEPER_EXCEPTION, "Error {} while getting children of {}", status, path.string()); std::sort(children.begin(), children.end()); auto next_query = *query; for (const auto & child : children) { next_query.args[1] = DB::Field(path / child); execute(&next_query, client); } } bool DeleteStaleBackups::parse(IParser::Pos & /* pos */, std::shared_ptr & /* node */, Expected & /* expected */) const { return true; } void DeleteStaleBackups::execute(const ASTKeeperQuery * /* query */, KeeperClient * client) const { client->askConfirmation( "You are going to delete all inactive backups in /clickhouse/backups.", [client] { fs::path backup_root = "/clickhouse/backups"; auto backups = client->zookeeper->getChildren(backup_root); std::sort(backups.begin(), backups.end()); for (const auto & child : backups) { auto backup_path = backup_root / child; std::cout << "Found backup " << backup_path << ", checking if it's active\n"; String stage_path = backup_path / "stage"; auto stages = client->zookeeper->getChildren(stage_path); bool is_active = false; for (const auto & stage : stages) { if (startsWith(stage, "alive")) { is_active = true; break; } } if (is_active) { std::cout << "Backup " << backup_path << " is active, not going to delete\n"; continue; } std::cout << "Backup " << backup_path << " is not active, deleting it\n"; client->zookeeper->removeRecursive(backup_path); } }); } bool FindBigFamily::parse(IParser::Pos & pos, std::shared_ptr & node, Expected & expected) const { String path; if (!parseKeeperPath(pos, expected, path)) path = "."; node->args.push_back(std::move(path)); ASTPtr count; if (ParserUnsignedInteger{}.parse(pos, count, expected)) node->args.push_back(count->as().value); else node->args.push_back(UInt64(10)); return true; } /// DFS the subtree and return the number of nodes in the subtree static Int64 traverse(const fs::path & path, KeeperClient * client, std::vector> & result) { Int64 nodes_in_subtree = 1; Strings children; auto status = client->zookeeper->tryGetChildren(path, children); if (status == Coordination::Error::ZNONODE) return 0; else if (status != Coordination::Error::ZOK) throw DB::Exception(DB::ErrorCodes::KEEPER_EXCEPTION, "Error {} while getting children of {}", status, path.string()); for (auto & child : children) nodes_in_subtree += traverse(path / child, client, result); result.emplace_back(nodes_in_subtree, path.string()); return nodes_in_subtree; } void FindBigFamily::execute(const ASTKeeperQuery * query, KeeperClient * client) const { auto path = client->getAbsolutePath(query->args[0].safeGet()); auto n = query->args[1].safeGet(); std::vector> result; traverse(path, client, result); std::sort(result.begin(), result.end(), std::greater()); for (UInt64 i = 0; i < std::min(result.size(), static_cast(n)); ++i) std::cout << std::get<1>(result[i]) << "\t" << std::get<0>(result[i]) << "\n"; } bool RMCommand::parse(IParser::Pos & pos, std::shared_ptr & node, Expected & expected) const { String path; if (!parseKeeperPath(pos, expected, path)) return false; node->args.push_back(std::move(path)); ASTPtr version; if (ParserNumber{}.parse(pos, version, expected)) node->args.push_back(version->as().value); return true; } void RMCommand::execute(const ASTKeeperQuery * query, KeeperClient * client) const { Int32 version{-1}; if (query->args.size() == 2) version = static_cast(query->args[1].get()); client->zookeeper->remove(client->getAbsolutePath(query->args[0].safeGet()), version); } bool RMRCommand::parse(IParser::Pos & pos, std::shared_ptr & node, Expected & expected) const { String path; if (!parseKeeperPath(pos, expected, path)) return false; node->args.push_back(std::move(path)); return true; } void RMRCommand::execute(const ASTKeeperQuery * query, KeeperClient * client) const { String path = client->getAbsolutePath(query->args[0].safeGet()); client->askConfirmation( "You are going to recursively delete path " + path, [client, path] { client->zookeeper->removeRecursive(path); }); } bool ReconfigCommand::parse(IParser::Pos & pos, std::shared_ptr & node, DB::Expected & expected) const { ParserKeyword s_add(Keyword::ADD); ParserKeyword s_remove(Keyword::REMOVE); ParserKeyword s_set(Keyword::SET); ReconfigCommand::Operation operation; if (s_add.ignore(pos, expected)) operation = ReconfigCommand::Operation::ADD; else if (s_remove.ignore(pos, expected)) operation = ReconfigCommand::Operation::REMOVE; else if (s_set.ignore(pos, expected)) operation = ReconfigCommand::Operation::SET; else return false; node->args.push_back(operation); ParserToken{TokenType::Whitespace}.ignore(pos); String arg; if (!parseKeeperArg(pos, expected, arg)) return false; node->args.push_back(std::move(arg)); return true; } void ReconfigCommand::execute(const DB::ASTKeeperQuery * query, DB::KeeperClient * client) const { String joining; String leaving; String new_members; auto operation = query->args[0].get(); switch (operation) { case static_cast(ReconfigCommand::Operation::ADD): joining = query->args[1].safeGet(); break; case static_cast(ReconfigCommand::Operation::REMOVE): leaving = query->args[1].safeGet(); break; case static_cast(ReconfigCommand::Operation::SET): new_members = query->args[1].safeGet(); break; default: throw Exception(ErrorCodes::UNEXPECTED_ZOOKEEPER_ERROR, "Unexpected operation: {}", operation); } auto response = client->zookeeper->reconfig(joining, leaving, new_members); std::cout << response.value << '\n'; } bool SyncCommand::parse(IParser::Pos & pos, std::shared_ptr & node, DB::Expected & expected) const { String path; if (!parseKeeperPath(pos, expected, path)) return false; node->args.push_back(std::move(path)); return true; } void SyncCommand::execute(const DB::ASTKeeperQuery * query, DB::KeeperClient * client) const { std::cout << client->zookeeper->sync(client->getAbsolutePath(query->args[0].safeGet())) << "\n"; } bool HelpCommand::parse(IParser::Pos & /* pos */, std::shared_ptr & /* node */, Expected & /* expected */) const { return true; } void HelpCommand::execute(const ASTKeeperQuery * /* query */, KeeperClient * /* client */) const { for (const auto & pair : KeeperClient::commands) std::cout << pair.second->generateHelpString() << "\n"; } bool FourLetterWordCommand::parse(IParser::Pos & pos, std::shared_ptr & node, Expected & expected) const { expected.add(pos, "four-letter-word command"); if (pos->type != TokenType::BareWord) return false; String cmd(pos->begin, pos->end); if (cmd.size() != 4) return false; ++pos; node->args.push_back(std::move(cmd)); return true; } void FourLetterWordCommand::execute(const ASTKeeperQuery * query, KeeperClient * client) const { std::cout << client->executeFourLetterCommand(query->args[0].safeGet()) << "\n"; } bool GetDirectChildrenNumberCommand::parse(IParser::Pos & pos, std::shared_ptr & node, Expected & expected) const { String path; if (!parseKeeperPath(pos, expected, path)) path = "."; node->args.push_back(std::move(path)); return true; } void GetDirectChildrenNumberCommand::execute(const ASTKeeperQuery * query, KeeperClient * client) const { auto path = client->getAbsolutePath(query->args[0].safeGet()); Coordination::Stat stat; client->zookeeper->get(path, &stat); std::cout << stat.numChildren << "\n"; } bool GetAllChildrenNumberCommand::parse(IParser::Pos & pos, std::shared_ptr & node, Expected & expected) const { String path; if (!parseKeeperPath(pos, expected, path)) path = "."; node->args.push_back(std::move(path)); return true; } void GetAllChildrenNumberCommand::execute(const ASTKeeperQuery * query, KeeperClient * client) const { auto path = client->getAbsolutePath(query->args[0].safeGet()); std::queue queue; queue.push(path); Coordination::Stat stat; client->zookeeper->get(path, &stat); int totalNumChildren = stat.numChildren; while (!queue.empty()) { auto next_path = queue.front(); queue.pop(); auto children = client->zookeeper->getChildren(next_path); for (auto & child : children) child = next_path / child; auto response = client->zookeeper->get(children); for (size_t i = 0; i < response.size(); ++i) { totalNumChildren += response[i].stat.numChildren; queue.push(children[i]); } } std::cout << totalNumChildren << "\n"; } }