2021-07-18 04:24:38 +08:00
|
|
|
/*
|
|
|
|
* IncludeCommand.actor.cpp
|
|
|
|
*
|
|
|
|
* This source file is part of the FoundationDB open source project
|
|
|
|
*
|
2022-03-22 04:36:23 +08:00
|
|
|
* Copyright 2013-2022 Apple Inc. and the FoundationDB project authors
|
2021-07-18 04:24:38 +08:00
|
|
|
*
|
|
|
|
* 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 "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 {
|
|
|
|
|
|
|
|
// Remove the given localities from the exclusion list.
|
|
|
|
// include localities by clearing the keys.
|
|
|
|
ACTOR Future<Void> includeLocalities(Reference<IDatabase> db,
|
2021-09-17 08:42:34 +08:00
|
|
|
std::vector<std::string> localities,
|
2021-07-18 04:24:38 +08:00
|
|
|
bool failed,
|
|
|
|
bool includeAll) {
|
|
|
|
state std::string versionKey = deterministicRandom()->randomUniqueID().toString();
|
|
|
|
state Reference<ITransaction> tr = db->createTransaction();
|
|
|
|
loop {
|
|
|
|
tr->setOption(FDBTransactionOptions::SPECIAL_KEY_SPACE_ENABLE_WRITES);
|
|
|
|
try {
|
|
|
|
|
|
|
|
if (includeAll) {
|
|
|
|
if (failed) {
|
|
|
|
tr->clear(fdb_cli::failedLocalitySpecialKeyRange);
|
|
|
|
} else {
|
|
|
|
tr->clear(fdb_cli::excludedLocalitySpecialKeyRange);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for (const auto& l : localities) {
|
|
|
|
Key locality = failed ? fdb_cli::failedLocalitySpecialKeyRange.begin.withSuffix(l)
|
|
|
|
: fdb_cli::excludedLocalitySpecialKeyRange.begin.withSuffix(l);
|
|
|
|
tr->clear(locality);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
wait(safeThreadFutureToFuture(tr->commit()));
|
|
|
|
return Void();
|
|
|
|
} catch (Error& e) {
|
2022-02-25 04:25:52 +08:00
|
|
|
TraceEvent("IncludeLocalitiesError").errorUnsuppressed(e);
|
2021-07-18 04:24:38 +08:00
|
|
|
wait(safeThreadFutureToFuture(tr->onError(e)));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-09-17 08:42:34 +08:00
|
|
|
ACTOR Future<Void> includeServers(Reference<IDatabase> db, std::vector<AddressExclusion> servers, bool failed) {
|
2021-07-18 04:24:38 +08:00
|
|
|
state std::string versionKey = deterministicRandom()->randomUniqueID().toString();
|
|
|
|
state Reference<ITransaction> tr = db->createTransaction();
|
|
|
|
loop {
|
|
|
|
tr->setOption(FDBTransactionOptions::SPECIAL_KEY_SPACE_ENABLE_WRITES);
|
|
|
|
try {
|
|
|
|
for (auto& s : servers) {
|
|
|
|
// include all, just clear the whole key range
|
|
|
|
if (!s.isValid()) {
|
|
|
|
if (failed) {
|
|
|
|
tr->clear(fdb_cli::failedServersSpecialKeyRange);
|
|
|
|
} else {
|
|
|
|
tr->clear(fdb_cli::excludedServersSpecialKeyRange);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
Key addr = failed ? fdb_cli::failedServersSpecialKeyRange.begin.withSuffix(s.toString())
|
|
|
|
: fdb_cli::excludedServersSpecialKeyRange.begin.withSuffix(s.toString());
|
|
|
|
tr->clear(addr);
|
|
|
|
// Eliminate both any ip-level exclusion (1.2.3.4) and any
|
|
|
|
// port-level exclusions (1.2.3.4:5)
|
|
|
|
// The range ['IP', 'IP;'] was originally deleted. ';' is
|
|
|
|
// char(':' + 1). This does not work, as other for all
|
|
|
|
// x between 0 and 9, 'IPx' will also be in this range.
|
|
|
|
//
|
|
|
|
// This is why we now make two clears: first only of the ip
|
|
|
|
// address, the second will delete all ports.
|
|
|
|
if (s.isWholeMachine())
|
2022-09-20 02:35:58 +08:00
|
|
|
tr->clear(KeyRangeRef(addr.withSuffix(":"_sr), addr.withSuffix(";"_sr)));
|
2021-07-18 04:24:38 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
wait(safeThreadFutureToFuture(tr->commit()));
|
|
|
|
return Void();
|
|
|
|
} catch (Error& e) {
|
2022-02-25 04:25:52 +08:00
|
|
|
TraceEvent("IncludeServersError").errorUnsuppressed(e);
|
2021-07-18 04:24:38 +08:00
|
|
|
wait(safeThreadFutureToFuture(tr->onError(e)));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Includes the servers that could be IP addresses or localities back to the cluster.
|
|
|
|
ACTOR Future<bool> include(Reference<IDatabase> db, std::vector<StringRef> tokens) {
|
|
|
|
state std::vector<AddressExclusion> addresses;
|
|
|
|
state std::vector<std::string> localities;
|
|
|
|
state bool failed = false;
|
|
|
|
state bool all = false;
|
|
|
|
for (auto t = tokens.begin() + 1; t != tokens.end(); ++t) {
|
2022-09-20 02:35:58 +08:00
|
|
|
if (*t == "all"_sr) {
|
2021-07-18 04:24:38 +08:00
|
|
|
all = true;
|
2022-09-20 02:35:58 +08:00
|
|
|
} else if (*t == "failed"_sr) {
|
2021-07-18 04:24:38 +08:00
|
|
|
failed = true;
|
|
|
|
} else if (t->startsWith(LocalityData::ExcludeLocalityPrefix) && t->toString().find(':') != std::string::npos) {
|
|
|
|
// if the token starts with 'locality_' prefix.
|
|
|
|
localities.push_back(t->toString());
|
|
|
|
} else {
|
|
|
|
auto a = AddressExclusion::parse(*t);
|
|
|
|
if (!a.isValid()) {
|
|
|
|
fprintf(stderr,
|
|
|
|
"ERROR: '%s' is neither a valid network endpoint address nor a locality\n",
|
|
|
|
t->toString().c_str());
|
|
|
|
if (t->toString().find(":tls") != std::string::npos)
|
|
|
|
printf(" Do not include the `:tls' suffix when naming a process\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
addresses.push_back(a);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (all) {
|
|
|
|
std::vector<AddressExclusion> includeAll;
|
|
|
|
includeAll.push_back(AddressExclusion());
|
|
|
|
wait(includeServers(db, includeAll, failed));
|
|
|
|
wait(includeLocalities(db, localities, failed, all));
|
|
|
|
} else {
|
|
|
|
if (!addresses.empty()) {
|
|
|
|
wait(includeServers(db, addresses, failed));
|
|
|
|
}
|
|
|
|
if (!localities.empty()) {
|
|
|
|
// include the servers that belong to given localities.
|
|
|
|
wait(includeLocalities(db, localities, failed, all));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
namespace fdb_cli {
|
|
|
|
|
|
|
|
ACTOR Future<bool> includeCommandActor(Reference<IDatabase> db, std::vector<StringRef> tokens) {
|
|
|
|
if (tokens.size() < 2) {
|
|
|
|
printUsage(tokens[0]);
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
bool result = wait(include(db, tokens));
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CommandFactory includeFactory(
|
|
|
|
"include",
|
|
|
|
CommandHelp(
|
|
|
|
"include all|[<ADDRESS...>] [locality_dcid:<excludedcid>] [locality_zoneid:<excludezoneid>] "
|
|
|
|
"[locality_machineid:<excludemachineid>] [locality_processid:<excludeprocessid>] or any locality data",
|
|
|
|
"permit previously-excluded servers and localities to rejoin the database",
|
|
|
|
"If `all' is specified, the excluded servers and localities list is cleared.\n\nFor each IP address or IP:port "
|
|
|
|
"pair in <ADDRESS...> or any LocalityData (like dcid, zoneid, machineid, processid), removes any "
|
|
|
|
"matching exclusions from the excluded servers and localities list. "
|
|
|
|
"(A specified IP will match all IP:* exclusion entries)"));
|
|
|
|
} // namespace fdb_cli
|