/* * KillCommand.actor.cpp * * This source file is part of the FoundationDB open source project * * Copyright 2013-2022 Apple Inc. and the FoundationDB project authors * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "boost/algorithm/string.hpp" #include "fdbcli/fdbcli.actor.h" #include "fdbclient/FDBOptions.g.h" #include "fdbclient/IClientApi.h" #include "fdbclient/Knobs.h" #include "flow/Arena.h" #include "flow/FastRef.h" #include "flow/ThreadHelper.actor.h" #include "flow/actorcompiler.h" // This must be the last #include. namespace fdb_cli { ACTOR Future killCommandActor(Reference db, Reference tr, std::vector tokens, std::map>* address_interface) { ASSERT(tokens.size() >= 1); state bool result = true; state std::string addressesStr; if (tokens.size() == 1) { // initialize worker interfaces address_interface->clear(); wait(getWorkerInterfaces(tr, address_interface)); } if (tokens.size() == 1 || tokencmp(tokens[1], "list")) { if (address_interface->size() == 0) { printf("\nNo addresses can be killed.\n"); } else if (address_interface->size() == 1) { printf("\nThe following address can be killed:\n"); } else { printf("\nThe following %zu addresses can be killed:\n", address_interface->size()); } for (auto it : *address_interface) { printf("%s\n", printable(it.first).c_str()); } printf("\n"); } else if (tokencmp(tokens[1], "all")) { if (address_interface->size() == 0) { result = false; fprintf(stderr, "ERROR: no processes to kill. You must run the `kill’ command before " "running `kill all’.\n"); } else { std::vector addressesVec; for (const auto& [address, _] : *address_interface) { addressesVec.push_back(address.toString()); } addressesStr = boost::algorithm::join(addressesVec, ","); // make sure we only call the interface once to send requests in parallel int64_t killRequestsSent = wait(safeThreadFutureToFuture(db->rebootWorker(addressesStr, false, 0))); if (!killRequestsSent) { result = false; fprintf(stderr, "ERROR: failed to send requests to all processes, please run the `kill’ command again to fetch " "latest addresses.\n"); } else { printf("Attempted to kill %zu processes\n", address_interface->size()); } } } else { state int i; for (i = 1; i < tokens.size(); i++) { if (!address_interface->count(tokens[i])) { fprintf(stderr, "ERROR: process `%s' not recognized.\n", printable(tokens[i]).c_str()); result = false; break; } } if (result) { std::vector addressesVec; for (i = 1; i < tokens.size(); i++) { addressesVec.push_back(tokens[i].toString()); } addressesStr = boost::algorithm::join(addressesVec, ","); int64_t killRequestsSent = wait(safeThreadFutureToFuture(db->rebootWorker(addressesStr, false, 0))); if (!killRequestsSent) { result = false; fprintf(stderr, "ERROR: failed to send requests to kill processes `%s', please run the `kill’ command again to " "fetch latest addresses.\n", addressesStr.c_str()); } else { printf("Attempted to kill %zu processes\n", tokens.size() - 1); } } } return result; } void killGenerator(const char* text, const char* line, std::vector& lc, std::vector const& tokens) { const char* opts[] = { "all", "list", nullptr }; arrayGenerator(text, line, opts, lc); } CommandFactory killFactory( "kill", CommandHelp( "kill all|list|", "attempts to kill one or more processes in the cluster", "If no addresses are specified, populates the list of processes which can be killed. Processes cannot be " "killed before this list has been populated.\n\nIf `all' is specified, attempts to kill all known " "processes.\n\nIf `list' is specified, displays all known processes. This is only useful when the database is " "unresponsive.\n\nFor each IP:port pair in
, attempt to kill the specified process."), &killGenerator); } // namespace fdb_cli