foundationdb/fdbclient/Tenant.cpp

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

250 lines
7.8 KiB
C++
Raw Normal View History

2022-02-24 00:06:39 +08:00
/*
* Tenant.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 "fdbclient/NativeAPI.actor.h"
2022-02-24 00:06:39 +08:00
#include "fdbclient/SystemData.h"
#include "fdbclient/Tenant.h"
#include "fdbrpc/TenantInfo.h"
#include "flow/BooleanParam.h"
#include "flow/IRandom.h"
#include "libb64/decode.h"
#include "libb64/encode.h"
#include "flow/ApiVersion.h"
2022-02-24 00:06:39 +08:00
#include "flow/UnitTest.h"
FDB_DEFINE_BOOLEAN_PARAM(EnforceValidTenantId);
namespace TenantAPI {
KeyRef idToPrefix(Arena& p, int64_t id) {
2022-05-27 03:49:22 +08:00
int64_t swapped = bigEndian64(id);
return StringRef(p, reinterpret_cast<const uint8_t*>(&swapped), TenantAPI::PREFIX_SIZE);
}
Key idToPrefix(int64_t id) {
Arena p(TenantAPI::PREFIX_SIZE);
return Key(idToPrefix(p, id), p);
2022-05-27 03:49:22 +08:00
}
int64_t prefixToId(KeyRef prefix, EnforceValidTenantId enforceValidTenantId) {
ASSERT(prefix.size() == TenantAPI::PREFIX_SIZE);
2022-05-27 03:49:22 +08:00
int64_t id = *reinterpret_cast<const int64_t*>(prefix.begin());
id = bigEndian64(id);
if (enforceValidTenantId) {
ASSERT(id >= 0);
} else if (id < 0) {
return TenantInfo::INVALID_TENANT;
}
2022-05-27 03:49:22 +08:00
return id;
}
}; // namespace TenantAPI
2022-05-27 03:49:22 +08:00
std::string TenantMapEntry::tenantStateToString(TenantState tenantState) {
switch (tenantState) {
case TenantState::REGISTERING:
return "registering";
case TenantState::READY:
return "ready";
case TenantState::REMOVING:
return "removing";
case TenantState::UPDATING_CONFIGURATION:
return "updating configuration";
case TenantState::RENAMING:
return "renaming";
2022-05-27 03:49:22 +08:00
case TenantState::ERROR:
return "error";
default:
UNREACHABLE();
2022-05-27 03:49:22 +08:00
}
}
TenantState TenantMapEntry::stringToTenantState(std::string stateStr) {
std::transform(stateStr.begin(), stateStr.end(), stateStr.begin(), [](unsigned char c) { return std::tolower(c); });
2022-05-27 03:49:22 +08:00
if (stateStr == "registering") {
return TenantState::REGISTERING;
} else if (stateStr == "ready") {
return TenantState::READY;
} else if (stateStr == "removing") {
return TenantState::REMOVING;
} else if (stateStr == "updating configuration") {
return TenantState::UPDATING_CONFIGURATION;
} else if (stateStr == "renaming") {
return TenantState::RENAMING;
2022-05-27 03:49:22 +08:00
} else if (stateStr == "error") {
return TenantState::ERROR;
}
throw invalid_option();
2022-05-27 03:49:22 +08:00
}
std::string TenantMapEntry::tenantLockStateToString(TenantLockState tenantState) {
switch (tenantState) {
case TenantLockState::UNLOCKED:
return "unlocked";
case TenantLockState::READ_ONLY:
return "read only";
case TenantLockState::LOCKED:
return "locked";
default:
UNREACHABLE();
}
}
TenantLockState TenantMapEntry::stringToTenantLockState(std::string stateStr) {
std::transform(stateStr.begin(), stateStr.end(), stateStr.begin(), [](unsigned char c) { return std::tolower(c); });
if (stateStr == "unlocked") {
return TenantLockState::UNLOCKED;
} else if (stateStr == "read only") {
return TenantLockState::READ_ONLY;
} else if (stateStr == "locked") {
return TenantLockState::LOCKED;
}
UNREACHABLE();
}
json_spirit::mObject binaryToJson(StringRef bytes) {
json_spirit::mObject obj;
std::string encodedBytes = base64::encoder::from_string(bytes.toString());
// Remove trailing newline
encodedBytes.resize(encodedBytes.size() - 1);
obj["base64"] = encodedBytes;
obj["printable"] = printable(bytes);
return obj;
}
2022-05-27 03:49:22 +08:00
TenantMapEntry::TenantMapEntry() {}
TenantMapEntry::TenantMapEntry(int64_t id, TenantName tenantName, TenantState tenantState)
: tenantName(tenantName), tenantState(tenantState) {
2022-07-21 03:56:00 +08:00
setId(id);
}
2022-07-26 08:19:14 +08:00
TenantMapEntry::TenantMapEntry(int64_t id,
TenantName tenantName,
2022-07-26 08:19:14 +08:00
TenantState tenantState,
Optional<TenantGroupName> tenantGroup)
: tenantName(tenantName), tenantState(tenantState), tenantGroup(tenantGroup) {
2022-07-21 03:56:00 +08:00
setId(id);
}
void TenantMapEntry::setId(int64_t id) {
ASSERT(id >= 0);
this->id = id;
prefix = TenantAPI::idToPrefix(id);
2022-07-21 03:56:00 +08:00
}
2022-05-27 03:49:22 +08:00
std::string TenantMapEntry::toJson() const {
json_spirit::mObject tenantEntry;
tenantEntry["id"] = id;
tenantEntry["name"] = binaryToJson(tenantName);
tenantEntry["prefix"] = binaryToJson(prefix);
tenantEntry["tenant_state"] = TenantMapEntry::tenantStateToString(tenantState);
if (assignedCluster.present()) {
tenantEntry["assigned_cluster"] = binaryToJson(assignedCluster.get());
}
if (tenantGroup.present()) {
tenantEntry["tenant_group"] = binaryToJson(tenantGroup.get());
}
return json_spirit::write_string(json_spirit::mValue(tenantEntry));
}
2022-07-09 06:56:22 +08:00
bool TenantMapEntry::matchesConfiguration(TenantMapEntry const& other) const {
return tenantGroup == other.tenantGroup;
}
void TenantMapEntry::configure(Standalone<StringRef> parameter, Optional<Value> value) {
if (parameter == "tenant_group"_sr) {
tenantGroup = value;
} else if (parameter == "assigned_cluster"_sr) {
assignedCluster = value;
} else {
TraceEvent(SevWarnAlways, "UnknownTenantConfigurationParameter").detail("Parameter", parameter);
throw invalid_tenant_configuration();
}
}
json_spirit::mObject TenantGroupEntry::toJson() const {
json_spirit::mObject tenantGroupEntry;
if (assignedCluster.present()) {
tenantGroupEntry["assigned_cluster"] = binaryToJson(assignedCluster.get());
}
return tenantGroupEntry;
}
TenantMetadataSpecification& TenantMetadata::instance() {
static TenantMetadataSpecification _instance = TenantMetadataSpecification("\xff/"_sr);
return _instance;
}
Key TenantMetadata::tenantMapPrivatePrefix() {
static Key _prefix = "\xff"_sr.withSuffix(tenantMap().subspace.begin);
return _prefix;
}
TEST_CASE("/fdbclient/libb64/base64decoder") {
Standalone<StringRef> buf = makeString(100);
for (int i = 0; i < 1000; ++i) {
int length = deterministicRandom()->randomInt(0, 100);
deterministicRandom()->randomBytes(mutateString(buf), length);
StringRef str = buf.substr(0, length);
std::string encodedStr = base64::encoder::from_string(str.toString());
// Remove trailing newline
encodedStr.resize(encodedStr.size() - 1);
std::string decodedStr = base64::decoder::from_string(encodedStr);
ASSERT(decodedStr == str.toString());
}
return Void();
}
2022-02-24 00:06:39 +08:00
TEST_CASE("/fdbclient/TenantMapEntry/Serialization") {
TenantMapEntry entry1(1, "name"_sr, TenantState::READY);
2022-02-24 00:06:39 +08:00
ASSERT(entry1.prefix == "\x00\x00\x00\x00\x00\x00\x00\x01"_sr);
TenantMapEntry entry2 = TenantMapEntry::decode(entry1.encode());
2022-02-24 00:06:39 +08:00
ASSERT(entry1.id == entry2.id && entry1.prefix == entry2.prefix);
TenantMapEntry entry3(std::numeric_limits<int64_t>::max(), "name"_sr, TenantState::READY);
ASSERT(entry3.prefix == "\x7f\xff\xff\xff\xff\xff\xff\xff"_sr);
TenantMapEntry entry4 = TenantMapEntry::decode(entry3.encode());
2022-02-24 00:06:39 +08:00
ASSERT(entry3.id == entry4.id && entry3.prefix == entry4.prefix);
for (int i = 0; i < 100; ++i) {
int bits = deterministicRandom()->randomInt(1, 64);
int64_t min = bits == 1 ? 0 : (UINT64_C(1) << (bits - 1));
int64_t maxPlusOne = std::min<uint64_t>(UINT64_C(1) << bits, std::numeric_limits<int64_t>::max());
2022-02-24 00:06:39 +08:00
int64_t id = deterministicRandom()->randomInt64(min, maxPlusOne);
TenantMapEntry entry(id, "name"_sr, TenantState::READY);
2022-02-24 00:06:39 +08:00
int64_t bigEndianId = bigEndian64(id);
ASSERT(entry.id == id && entry.prefix == StringRef(reinterpret_cast<uint8_t*>(&bigEndianId), 8));
2022-02-24 00:06:39 +08:00
TenantMapEntry decodedEntry = TenantMapEntry::decode(entry.encode());
ASSERT(decodedEntry.id == entry.id && decodedEntry.prefix == entry.prefix);
2022-02-24 00:06:39 +08:00
}
return Void();
}