Add Struct Type to OptionsTypeInfo (#6425)
Summary: Added code for generically handing structs to OptionTypeInfo. A struct is a collection of variables handled by their own map of OptionTypeInfos. Examples of structs include Compaction and Cache options. Pull Request resolved: https://github.com/facebook/rocksdb/pull/6425 Reviewed By: siying Differential Revision: D21668789 Pulled By: zhichao-cao fbshipit-source-id: 064b110de39dadf82361ed4663f7ac1a535b0b07
This commit is contained in:
parent
c7aedf1b48
commit
38be686160
35
cache/cache.cc
vendored
35
cache/cache.cc
vendored
@ -14,7 +14,30 @@
|
|||||||
#include "util/string_util.h"
|
#include "util/string_util.h"
|
||||||
|
|
||||||
namespace ROCKSDB_NAMESPACE {
|
namespace ROCKSDB_NAMESPACE {
|
||||||
Status Cache::CreateFromString(const ConfigOptions& /*opts*/,
|
#ifndef ROCKSDB_LITE
|
||||||
|
static std::unordered_map<std::string, OptionTypeInfo>
|
||||||
|
lru_cache_options_type_info = {
|
||||||
|
{"capacity",
|
||||||
|
{offsetof(struct LRUCacheOptions, capacity), OptionType::kSizeT,
|
||||||
|
OptionVerificationType::kNormal, OptionTypeFlags::kMutable,
|
||||||
|
offsetof(struct LRUCacheOptions, capacity)}},
|
||||||
|
{"num_shard_bits",
|
||||||
|
{offsetof(struct LRUCacheOptions, num_shard_bits), OptionType::kInt,
|
||||||
|
OptionVerificationType::kNormal, OptionTypeFlags::kMutable,
|
||||||
|
offsetof(struct LRUCacheOptions, num_shard_bits)}},
|
||||||
|
{"strict_capacity_limit",
|
||||||
|
{offsetof(struct LRUCacheOptions, strict_capacity_limit),
|
||||||
|
OptionType::kBoolean, OptionVerificationType::kNormal,
|
||||||
|
OptionTypeFlags::kMutable,
|
||||||
|
offsetof(struct LRUCacheOptions, strict_capacity_limit)}},
|
||||||
|
{"high_pri_pool_ratio",
|
||||||
|
{offsetof(struct LRUCacheOptions, high_pri_pool_ratio),
|
||||||
|
OptionType::kDouble, OptionVerificationType::kNormal,
|
||||||
|
OptionTypeFlags::kMutable,
|
||||||
|
offsetof(struct LRUCacheOptions, high_pri_pool_ratio)}}};
|
||||||
|
#endif // ROCKSDB_LITE
|
||||||
|
|
||||||
|
Status Cache::CreateFromString(const ConfigOptions& config_options,
|
||||||
const std::string& value,
|
const std::string& value,
|
||||||
std::shared_ptr<Cache>* result) {
|
std::shared_ptr<Cache>* result) {
|
||||||
Status status;
|
Status status;
|
||||||
@ -24,12 +47,14 @@ Status Cache::CreateFromString(const ConfigOptions& /*opts*/,
|
|||||||
} else {
|
} else {
|
||||||
#ifndef ROCKSDB_LITE
|
#ifndef ROCKSDB_LITE
|
||||||
LRUCacheOptions cache_opts;
|
LRUCacheOptions cache_opts;
|
||||||
if (!ParseOptionHelper(reinterpret_cast<char*>(&cache_opts),
|
status = OptionTypeInfo::ParseStruct(
|
||||||
OptionType::kLRUCacheOptions, value)) {
|
config_options, "", &lru_cache_options_type_info, "", value,
|
||||||
status = Status::InvalidArgument("Invalid cache options");
|
reinterpret_cast<char*>(&cache_opts));
|
||||||
}
|
if (status.ok()) {
|
||||||
cache = NewLRUCache(cache_opts);
|
cache = NewLRUCache(cache_opts);
|
||||||
|
}
|
||||||
#else
|
#else
|
||||||
|
(void)config_options;
|
||||||
status = Status::NotSupported("Cannot load cache in LITE mode ", value);
|
status = Status::NotSupported("Cannot load cache in LITE mode ", value);
|
||||||
#endif //! ROCKSDB_LITE
|
#endif //! ROCKSDB_LITE
|
||||||
}
|
}
|
||||||
|
@ -114,6 +114,63 @@ static Status ParseCompressionOptions(const std::string& value,
|
|||||||
const std::string kOptNameBMCompOpts = "bottommost_compression_opts";
|
const std::string kOptNameBMCompOpts = "bottommost_compression_opts";
|
||||||
const std::string kOptNameCompOpts = "compression_opts";
|
const std::string kOptNameCompOpts = "compression_opts";
|
||||||
|
|
||||||
|
static std::unordered_map<std::string, OptionTypeInfo>
|
||||||
|
fifo_compaction_options_type_info = {
|
||||||
|
{"max_table_files_size",
|
||||||
|
{offsetof(struct CompactionOptionsFIFO, max_table_files_size),
|
||||||
|
OptionType::kUInt64T, OptionVerificationType::kNormal,
|
||||||
|
OptionTypeFlags::kMutable,
|
||||||
|
offsetof(struct CompactionOptionsFIFO, max_table_files_size)}},
|
||||||
|
{"ttl",
|
||||||
|
{0, OptionType::kUInt64T, OptionVerificationType::kDeprecated,
|
||||||
|
OptionTypeFlags::kNone, 0}},
|
||||||
|
{"allow_compaction",
|
||||||
|
{offsetof(struct CompactionOptionsFIFO, allow_compaction),
|
||||||
|
OptionType::kBoolean, OptionVerificationType::kNormal,
|
||||||
|
OptionTypeFlags::kMutable,
|
||||||
|
offsetof(struct CompactionOptionsFIFO, allow_compaction)}}};
|
||||||
|
|
||||||
|
static std::unordered_map<std::string, OptionTypeInfo>
|
||||||
|
universal_compaction_options_type_info = {
|
||||||
|
{"size_ratio",
|
||||||
|
{offsetof(class CompactionOptionsUniversal, size_ratio),
|
||||||
|
OptionType::kUInt, OptionVerificationType::kNormal,
|
||||||
|
OptionTypeFlags::kMutable,
|
||||||
|
offsetof(class CompactionOptionsUniversal, size_ratio)}},
|
||||||
|
{"min_merge_width",
|
||||||
|
{offsetof(class CompactionOptionsUniversal, min_merge_width),
|
||||||
|
OptionType::kUInt, OptionVerificationType::kNormal,
|
||||||
|
OptionTypeFlags::kMutable,
|
||||||
|
offsetof(class CompactionOptionsUniversal, min_merge_width)}},
|
||||||
|
{"max_merge_width",
|
||||||
|
{offsetof(class CompactionOptionsUniversal, max_merge_width),
|
||||||
|
OptionType::kUInt, OptionVerificationType::kNormal,
|
||||||
|
OptionTypeFlags::kMutable,
|
||||||
|
offsetof(class CompactionOptionsUniversal, max_merge_width)}},
|
||||||
|
{"max_size_amplification_percent",
|
||||||
|
{offsetof(class CompactionOptionsUniversal,
|
||||||
|
max_size_amplification_percent),
|
||||||
|
OptionType::kUInt, OptionVerificationType::kNormal,
|
||||||
|
OptionTypeFlags::kMutable,
|
||||||
|
offsetof(class CompactionOptionsUniversal,
|
||||||
|
max_size_amplification_percent)}},
|
||||||
|
{"compression_size_percent",
|
||||||
|
{offsetof(class CompactionOptionsUniversal, compression_size_percent),
|
||||||
|
OptionType::kInt, OptionVerificationType::kNormal,
|
||||||
|
OptionTypeFlags::kMutable,
|
||||||
|
offsetof(class CompactionOptionsUniversal,
|
||||||
|
compression_size_percent)}},
|
||||||
|
{"stop_style",
|
||||||
|
{offsetof(class CompactionOptionsUniversal, stop_style),
|
||||||
|
OptionType::kCompactionStopStyle, OptionVerificationType::kNormal,
|
||||||
|
OptionTypeFlags::kMutable,
|
||||||
|
offsetof(class CompactionOptionsUniversal, stop_style)}},
|
||||||
|
{"allow_trivial_move",
|
||||||
|
{offsetof(class CompactionOptionsUniversal, allow_trivial_move),
|
||||||
|
OptionType::kBoolean, OptionVerificationType::kNormal,
|
||||||
|
OptionTypeFlags::kMutable,
|
||||||
|
offsetof(class CompactionOptionsUniversal, allow_trivial_move)}}};
|
||||||
|
|
||||||
std::unordered_map<std::string, OptionTypeInfo>
|
std::unordered_map<std::string, OptionTypeInfo>
|
||||||
OptionsHelper::cf_options_type_info = {
|
OptionsHelper::cf_options_type_info = {
|
||||||
/* not yet supported
|
/* not yet supported
|
||||||
@ -473,15 +530,36 @@ std::unordered_map<std::string, OptionTypeInfo>
|
|||||||
OptionType::kCompactionPri, OptionVerificationType::kNormal,
|
OptionType::kCompactionPri, OptionVerificationType::kNormal,
|
||||||
OptionTypeFlags::kNone, 0}},
|
OptionTypeFlags::kNone, 0}},
|
||||||
{"compaction_options_fifo",
|
{"compaction_options_fifo",
|
||||||
{offset_of(&ColumnFamilyOptions::compaction_options_fifo),
|
OptionTypeInfo::Struct(
|
||||||
OptionType::kCompactionOptionsFIFO, OptionVerificationType::kNormal,
|
"compaction_options_fifo", &fifo_compaction_options_type_info,
|
||||||
OptionTypeFlags::kMutable,
|
offset_of(&ColumnFamilyOptions::compaction_options_fifo),
|
||||||
offsetof(struct MutableCFOptions, compaction_options_fifo)}},
|
|
||||||
{"compaction_options_universal",
|
|
||||||
{offset_of(&ColumnFamilyOptions::compaction_options_universal),
|
|
||||||
OptionType::kCompactionOptionsUniversal,
|
|
||||||
OptionVerificationType::kNormal, OptionTypeFlags::kMutable,
|
OptionVerificationType::kNormal, OptionTypeFlags::kMutable,
|
||||||
offsetof(struct MutableCFOptions, compaction_options_universal)}},
|
offsetof(struct MutableCFOptions, compaction_options_fifo),
|
||||||
|
[](const ConfigOptions& opts, const std::string& name,
|
||||||
|
const std::string& value, char* addr) {
|
||||||
|
// This is to handle backward compatibility, where
|
||||||
|
// compaction_options_fifo could be assigned a single scalar
|
||||||
|
// value, say, like "23", which would be assigned to
|
||||||
|
// max_table_files_size.
|
||||||
|
if (name == "compaction_options_fifo" &&
|
||||||
|
value.find("=") == std::string::npos) {
|
||||||
|
// Old format. Parse just a single uint64_t value.
|
||||||
|
auto options = reinterpret_cast<CompactionOptionsFIFO*>(addr);
|
||||||
|
options->max_table_files_size = ParseUint64(value);
|
||||||
|
return Status::OK();
|
||||||
|
} else {
|
||||||
|
return OptionTypeInfo::ParseStruct(
|
||||||
|
opts, "compaction_options_fifo",
|
||||||
|
&fifo_compaction_options_type_info, name, value, addr);
|
||||||
|
}
|
||||||
|
})},
|
||||||
|
{"compaction_options_universal",
|
||||||
|
OptionTypeInfo::Struct(
|
||||||
|
"compaction_options_universal",
|
||||||
|
&universal_compaction_options_type_info,
|
||||||
|
offset_of(&ColumnFamilyOptions::compaction_options_universal),
|
||||||
|
OptionVerificationType::kNormal, OptionTypeFlags::kMutable,
|
||||||
|
offsetof(struct MutableCFOptions, compaction_options_universal))},
|
||||||
{"ttl",
|
{"ttl",
|
||||||
{offset_of(&ColumnFamilyOptions::ttl), OptionType::kUInt64T,
|
{offset_of(&ColumnFamilyOptions::ttl), OptionType::kUInt64T,
|
||||||
OptionVerificationType::kNormal, OptionTypeFlags::kMutable,
|
OptionVerificationType::kNormal, OptionTypeFlags::kMutable,
|
||||||
@ -534,14 +612,16 @@ Status ParseColumnFamilyOption(const ConfigOptions& config_options,
|
|||||||
? UnescapeOptionString(org_value)
|
? UnescapeOptionString(org_value)
|
||||||
: org_value;
|
: org_value;
|
||||||
try {
|
try {
|
||||||
auto iter = cf_options_type_info.find(name);
|
std::string elem;
|
||||||
if (iter == cf_options_type_info.end()) {
|
const auto opt_info =
|
||||||
|
OptionTypeInfo::Find(name, cf_options_type_info, &elem);
|
||||||
|
if (opt_info != nullptr) {
|
||||||
|
return opt_info->Parse(
|
||||||
|
config_options, elem, value,
|
||||||
|
reinterpret_cast<char*>(new_options) + opt_info->offset_);
|
||||||
|
} else {
|
||||||
return Status::InvalidArgument(
|
return Status::InvalidArgument(
|
||||||
"Unable to parse the specified CF option " + name);
|
"Unable to parse the specified CF option " + name);
|
||||||
} else {
|
|
||||||
return iter->second.ParseOption(
|
|
||||||
config_options, name, value,
|
|
||||||
reinterpret_cast<char*>(new_options) + iter->second.offset);
|
|
||||||
}
|
}
|
||||||
} catch (const std::exception&) {
|
} catch (const std::exception&) {
|
||||||
return Status::InvalidArgument("unable to parse the specified option " +
|
return Status::InvalidArgument("unable to parse the specified option " +
|
||||||
|
@ -322,80 +322,6 @@ bool ParseVectorCompressionType(
|
|||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
// This is to handle backward compatibility, where compaction_options_fifo
|
|
||||||
// could be assigned a single scalar value, say, like "23", which would be
|
|
||||||
// assigned to max_table_files_size.
|
|
||||||
bool FIFOCompactionOptionsSpecialCase(const std::string& opt_str,
|
|
||||||
CompactionOptionsFIFO* options) {
|
|
||||||
if (opt_str.find("=") != std::string::npos) {
|
|
||||||
// New format. Go do your new parsing using ParseStructOptions.
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Old format. Parse just a single uint64_t value.
|
|
||||||
options->max_table_files_size = ParseUint64(opt_str);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool SerializeStruct(
|
|
||||||
const void* const opt_ptr, std::string* value,
|
|
||||||
const std::unordered_map<std::string, OptionTypeInfo>& type_info_map) {
|
|
||||||
ConfigOptions config_options;
|
|
||||||
config_options.delimiter = ";";
|
|
||||||
|
|
||||||
std::string opt_str;
|
|
||||||
Status s =
|
|
||||||
GetStringFromStruct(config_options, opt_ptr, type_info_map, &opt_str);
|
|
||||||
if (!s.ok()) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
*value = "{" + opt_str + "}";
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool ParseSingleStructOption(
|
|
||||||
const ConfigOptions& config_options, const std::string& opt_val_str,
|
|
||||||
void* options,
|
|
||||||
const std::unordered_map<std::string, OptionTypeInfo>& type_info_map) {
|
|
||||||
size_t end = opt_val_str.find('=');
|
|
||||||
std::string key = opt_val_str.substr(0, end);
|
|
||||||
std::string value = opt_val_str.substr(end + 1);
|
|
||||||
auto iter = type_info_map.find(key);
|
|
||||||
if (iter == type_info_map.end()) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
const auto& opt_info = iter->second;
|
|
||||||
Status s = opt_info.ParseOption(
|
|
||||||
config_options, key, value,
|
|
||||||
reinterpret_cast<char*>(options) + opt_info.mutable_offset);
|
|
||||||
return s.ok();
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool ParseStructOptions(
|
|
||||||
const std::string& opt_str, void* options,
|
|
||||||
const std::unordered_map<std::string, OptionTypeInfo>& type_info_map) {
|
|
||||||
assert(!opt_str.empty());
|
|
||||||
ConfigOptions config_options;
|
|
||||||
|
|
||||||
size_t start = 0;
|
|
||||||
if (opt_str[0] == '{') {
|
|
||||||
start++;
|
|
||||||
}
|
|
||||||
while ((start != std::string::npos) && (start < opt_str.size())) {
|
|
||||||
if (opt_str[start] == '}') {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
size_t end = opt_str.find(';', start);
|
|
||||||
size_t len = (end == std::string::npos) ? end : end - start;
|
|
||||||
if (!ParseSingleStructOption(config_options, opt_str.substr(start, len),
|
|
||||||
options, type_info_map)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
start = (end == std::string::npos) ? end : end + 1;
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
} // anonymouse namespace
|
} // anonymouse namespace
|
||||||
|
|
||||||
bool ParseSliceTransformHelper(
|
bool ParseSliceTransformHelper(
|
||||||
@ -516,21 +442,6 @@ bool ParseOptionHelper(char* opt_address, const OptionType& opt_type,
|
|||||||
return ParseEnum<EncodingType>(
|
return ParseEnum<EncodingType>(
|
||||||
encoding_type_string_map, value,
|
encoding_type_string_map, value,
|
||||||
reinterpret_cast<EncodingType*>(opt_address));
|
reinterpret_cast<EncodingType*>(opt_address));
|
||||||
case OptionType::kCompactionOptionsFIFO: {
|
|
||||||
if (!FIFOCompactionOptionsSpecialCase(
|
|
||||||
value, reinterpret_cast<CompactionOptionsFIFO*>(opt_address))) {
|
|
||||||
return ParseStructOptions(value, opt_address,
|
|
||||||
fifo_compaction_options_type_info);
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
case OptionType::kLRUCacheOptions: {
|
|
||||||
return ParseStructOptions(value, opt_address,
|
|
||||||
lru_cache_options_type_info);
|
|
||||||
}
|
|
||||||
case OptionType::kCompactionOptionsUniversal:
|
|
||||||
return ParseStructOptions(value, opt_address,
|
|
||||||
universal_compaction_options_type_info);
|
|
||||||
case OptionType::kCompactionStopStyle:
|
case OptionType::kCompactionStopStyle:
|
||||||
return ParseEnum<CompactionStopStyle>(
|
return ParseEnum<CompactionStopStyle>(
|
||||||
compaction_stop_style_string_map, value,
|
compaction_stop_style_string_map, value,
|
||||||
@ -691,12 +602,6 @@ bool SerializeSingleOptionHelper(const char* opt_address,
|
|||||||
return SerializeEnum<EncodingType>(
|
return SerializeEnum<EncodingType>(
|
||||||
encoding_type_string_map,
|
encoding_type_string_map,
|
||||||
*reinterpret_cast<const EncodingType*>(opt_address), value);
|
*reinterpret_cast<const EncodingType*>(opt_address), value);
|
||||||
case OptionType::kCompactionOptionsFIFO:
|
|
||||||
return SerializeStruct(opt_address, value,
|
|
||||||
fifo_compaction_options_type_info);
|
|
||||||
case OptionType::kCompactionOptionsUniversal:
|
|
||||||
return SerializeStruct(opt_address, value,
|
|
||||||
universal_compaction_options_type_info);
|
|
||||||
case OptionType::kCompactionStopStyle:
|
case OptionType::kCompactionStopStyle:
|
||||||
return SerializeEnum<CompactionStopStyle>(
|
return SerializeEnum<CompactionStopStyle>(
|
||||||
compaction_stop_style_string_map,
|
compaction_stop_style_string_map,
|
||||||
@ -715,28 +620,27 @@ Status GetMutableOptionsFromStrings(
|
|||||||
*new_options = base_options;
|
*new_options = base_options;
|
||||||
ConfigOptions config_options;
|
ConfigOptions config_options;
|
||||||
for (const auto& o : options_map) {
|
for (const auto& o : options_map) {
|
||||||
auto iter = cf_options_type_info.find(o.first);
|
std::string elem;
|
||||||
if (iter == cf_options_type_info.end()) {
|
const auto opt_info =
|
||||||
|
OptionTypeInfo::Find(o.first, cf_options_type_info, &elem);
|
||||||
|
if (opt_info == nullptr) {
|
||||||
return Status::InvalidArgument("Unrecognized option: " + o.first);
|
return Status::InvalidArgument("Unrecognized option: " + o.first);
|
||||||
}
|
} else if (!opt_info->IsMutable()) {
|
||||||
const auto& opt_info = iter->second;
|
|
||||||
if (!opt_info.IsMutable()) {
|
|
||||||
return Status::InvalidArgument("Option not changeable: " + o.first);
|
return Status::InvalidArgument("Option not changeable: " + o.first);
|
||||||
}
|
} else if (opt_info->IsDeprecated()) {
|
||||||
if (opt_info.IsDeprecated()) {
|
|
||||||
// log warning when user tries to set a deprecated option but don't fail
|
// log warning when user tries to set a deprecated option but don't fail
|
||||||
// the call for compatibility.
|
// the call for compatibility.
|
||||||
ROCKS_LOG_WARN(info_log, "%s is a deprecated option and cannot be set",
|
ROCKS_LOG_WARN(info_log, "%s is a deprecated option and cannot be set",
|
||||||
o.first.c_str());
|
o.first.c_str());
|
||||||
continue;
|
} else {
|
||||||
}
|
Status s = opt_info->Parse(
|
||||||
Status s = opt_info.ParseOption(
|
config_options, elem, o.second,
|
||||||
config_options, o.first, o.second,
|
reinterpret_cast<char*>(new_options) + opt_info->mutable_offset_);
|
||||||
reinterpret_cast<char*>(new_options) + opt_info.mutable_offset);
|
|
||||||
if (!s.ok()) {
|
if (!s.ok()) {
|
||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
return Status::OK();
|
return Status::OK();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -750,20 +654,21 @@ Status GetMutableDBOptionsFromStrings(
|
|||||||
|
|
||||||
for (const auto& o : options_map) {
|
for (const auto& o : options_map) {
|
||||||
try {
|
try {
|
||||||
auto iter = db_options_type_info.find(o.first);
|
std::string elem;
|
||||||
if (iter == db_options_type_info.end()) {
|
const auto opt_info =
|
||||||
|
OptionTypeInfo::Find(o.first, db_options_type_info, &elem);
|
||||||
|
if (opt_info == nullptr) {
|
||||||
return Status::InvalidArgument("Unrecognized option: " + o.first);
|
return Status::InvalidArgument("Unrecognized option: " + o.first);
|
||||||
}
|
} else if (!opt_info->IsMutable()) {
|
||||||
const auto& opt_info = iter->second;
|
|
||||||
if (!opt_info.IsMutable()) {
|
|
||||||
return Status::InvalidArgument("Option not changeable: " + o.first);
|
return Status::InvalidArgument("Option not changeable: " + o.first);
|
||||||
}
|
} else {
|
||||||
Status s = opt_info.ParseOption(
|
Status s = opt_info->Parse(
|
||||||
config_options, o.first, o.second,
|
config_options, elem, o.second,
|
||||||
reinterpret_cast<char*>(new_options) + opt_info.mutable_offset);
|
reinterpret_cast<char*>(new_options) + opt_info->mutable_offset_);
|
||||||
if (!s.ok()) {
|
if (!s.ok()) {
|
||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
} catch (std::exception& e) {
|
} catch (std::exception& e) {
|
||||||
return Status::InvalidArgument("Error parsing " + o.first + ":" +
|
return Status::InvalidArgument("Error parsing " + o.first + ":" +
|
||||||
std::string(e.what()));
|
std::string(e.what()));
|
||||||
@ -780,6 +685,11 @@ Status StringToMap(const std::string& opts_str,
|
|||||||
// "nested_opt={opt1=1;opt2=2};max_bytes_for_level_base=100"
|
// "nested_opt={opt1=1;opt2=2};max_bytes_for_level_base=100"
|
||||||
size_t pos = 0;
|
size_t pos = 0;
|
||||||
std::string opts = trim(opts_str);
|
std::string opts = trim(opts_str);
|
||||||
|
// If the input string starts and ends with "{...}", strip off the brackets
|
||||||
|
while (opts.size() > 2 && opts[0] == '{' && opts[opts.size() - 1] == '}') {
|
||||||
|
opts = trim(opts.substr(1, opts.size() - 2));
|
||||||
|
}
|
||||||
|
|
||||||
while (pos < opts.size()) {
|
while (pos < opts.size()) {
|
||||||
size_t eq_pos = opts.find('=', pos);
|
size_t eq_pos = opts.find('=', pos);
|
||||||
if (eq_pos == std::string::npos) {
|
if (eq_pos == std::string::npos) {
|
||||||
@ -860,10 +770,10 @@ Status GetStringFromStruct(
|
|||||||
// we skip it in the serialization.
|
// we skip it in the serialization.
|
||||||
if (opt_info.ShouldSerialize()) {
|
if (opt_info.ShouldSerialize()) {
|
||||||
const char* opt_addr =
|
const char* opt_addr =
|
||||||
reinterpret_cast<const char*>(opt_ptr) + opt_info.offset;
|
reinterpret_cast<const char*>(opt_ptr) + opt_info.offset_;
|
||||||
std::string value;
|
std::string value;
|
||||||
Status s = opt_info.SerializeOption(config_options, iter.first, opt_addr,
|
Status s =
|
||||||
&value);
|
opt_info.Serialize(config_options, iter.first, opt_addr, &value);
|
||||||
if (s.ok()) {
|
if (s.ok()) {
|
||||||
opt_string->append(iter.first + "=" + value + config_options.delimiter);
|
opt_string->append(iter.first + "=" + value + config_options.delimiter);
|
||||||
} else {
|
} else {
|
||||||
@ -923,13 +833,14 @@ static Status ParseDBOption(const ConfigOptions& config_options,
|
|||||||
const std::string& value = config_options.input_strings_escaped
|
const std::string& value = config_options.input_strings_escaped
|
||||||
? UnescapeOptionString(org_value)
|
? UnescapeOptionString(org_value)
|
||||||
: org_value;
|
: org_value;
|
||||||
auto iter = db_options_type_info.find(name);
|
std::string elem;
|
||||||
if (iter == db_options_type_info.end()) {
|
const auto opt_info = OptionTypeInfo::Find(name, db_options_type_info, &elem);
|
||||||
|
if (opt_info == nullptr) {
|
||||||
return Status::InvalidArgument("Unrecognized option DBOptions:", name);
|
return Status::InvalidArgument("Unrecognized option DBOptions:", name);
|
||||||
} else {
|
} else {
|
||||||
return iter->second.ParseOption(
|
return opt_info->Parse(
|
||||||
config_options, name, value,
|
config_options, elem, value,
|
||||||
reinterpret_cast<char*>(new_options) + iter->second.offset);
|
reinterpret_cast<char*>(new_options) + opt_info->offset_);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1170,109 +1081,12 @@ std::unordered_map<std::string, CompactionPri>
|
|||||||
{"kOldestSmallestSeqFirst", kOldestSmallestSeqFirst},
|
{"kOldestSmallestSeqFirst", kOldestSmallestSeqFirst},
|
||||||
{"kMinOverlappingRatio", kMinOverlappingRatio}};
|
{"kMinOverlappingRatio", kMinOverlappingRatio}};
|
||||||
|
|
||||||
LRUCacheOptions OptionsHelper::dummy_lru_cache_options;
|
|
||||||
CompactionOptionsUniversal OptionsHelper::dummy_comp_options_universal;
|
|
||||||
CompactionOptionsFIFO OptionsHelper::dummy_comp_options;
|
|
||||||
|
|
||||||
template <typename T1>
|
|
||||||
int offset_of(T1 LRUCacheOptions::*member) {
|
|
||||||
return int(size_t(&(OptionsHelper::dummy_lru_cache_options.*member)) -
|
|
||||||
size_t(&OptionsHelper::dummy_lru_cache_options));
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename T1>
|
|
||||||
int offset_of(T1 CompactionOptionsFIFO::*member) {
|
|
||||||
return int(size_t(&(OptionsHelper::dummy_comp_options.*member)) -
|
|
||||||
size_t(&OptionsHelper::dummy_comp_options));
|
|
||||||
}
|
|
||||||
template <typename T1>
|
|
||||||
int offset_of(T1 CompactionOptionsUniversal::*member) {
|
|
||||||
return int(size_t(&(OptionsHelper::dummy_comp_options_universal.*member)) -
|
|
||||||
size_t(&OptionsHelper::dummy_comp_options_universal));
|
|
||||||
}
|
|
||||||
|
|
||||||
std::unordered_map<std::string, OptionTypeInfo>
|
|
||||||
OptionsHelper::fifo_compaction_options_type_info = {
|
|
||||||
{"max_table_files_size",
|
|
||||||
{offset_of(&CompactionOptionsFIFO::max_table_files_size),
|
|
||||||
OptionType::kUInt64T, OptionVerificationType::kNormal,
|
|
||||||
OptionTypeFlags::kMutable,
|
|
||||||
offsetof(struct CompactionOptionsFIFO, max_table_files_size)}},
|
|
||||||
{"ttl",
|
|
||||||
{0, OptionType::kUInt64T, OptionVerificationType::kDeprecated,
|
|
||||||
OptionTypeFlags::kNone, 0}},
|
|
||||||
{"allow_compaction",
|
|
||||||
{offset_of(&CompactionOptionsFIFO::allow_compaction),
|
|
||||||
OptionType::kBoolean, OptionVerificationType::kNormal,
|
|
||||||
OptionTypeFlags::kMutable,
|
|
||||||
offsetof(struct CompactionOptionsFIFO, allow_compaction)}}};
|
|
||||||
|
|
||||||
std::unordered_map<std::string, OptionTypeInfo>
|
|
||||||
OptionsHelper::universal_compaction_options_type_info = {
|
|
||||||
{"size_ratio",
|
|
||||||
{offset_of(&CompactionOptionsUniversal::size_ratio), OptionType::kUInt,
|
|
||||||
OptionVerificationType::kNormal, OptionTypeFlags::kMutable,
|
|
||||||
offsetof(class CompactionOptionsUniversal, size_ratio)}},
|
|
||||||
{"min_merge_width",
|
|
||||||
{offset_of(&CompactionOptionsUniversal::min_merge_width),
|
|
||||||
OptionType::kUInt, OptionVerificationType::kNormal,
|
|
||||||
OptionTypeFlags::kMutable,
|
|
||||||
offsetof(class CompactionOptionsUniversal, min_merge_width)}},
|
|
||||||
{"max_merge_width",
|
|
||||||
{offset_of(&CompactionOptionsUniversal::max_merge_width),
|
|
||||||
OptionType::kUInt, OptionVerificationType::kNormal,
|
|
||||||
OptionTypeFlags::kMutable,
|
|
||||||
offsetof(class CompactionOptionsUniversal, max_merge_width)}},
|
|
||||||
{"max_size_amplification_percent",
|
|
||||||
{offset_of(
|
|
||||||
&CompactionOptionsUniversal::max_size_amplification_percent),
|
|
||||||
OptionType::kUInt, OptionVerificationType::kNormal,
|
|
||||||
OptionTypeFlags::kMutable,
|
|
||||||
offsetof(class CompactionOptionsUniversal,
|
|
||||||
max_size_amplification_percent)}},
|
|
||||||
{"compression_size_percent",
|
|
||||||
{offset_of(&CompactionOptionsUniversal::compression_size_percent),
|
|
||||||
OptionType::kInt, OptionVerificationType::kNormal,
|
|
||||||
OptionTypeFlags::kMutable,
|
|
||||||
offsetof(class CompactionOptionsUniversal,
|
|
||||||
compression_size_percent)}},
|
|
||||||
{"stop_style",
|
|
||||||
{offset_of(&CompactionOptionsUniversal::stop_style),
|
|
||||||
OptionType::kCompactionStopStyle, OptionVerificationType::kNormal,
|
|
||||||
OptionTypeFlags::kMutable,
|
|
||||||
offsetof(class CompactionOptionsUniversal, stop_style)}},
|
|
||||||
{"allow_trivial_move",
|
|
||||||
{offset_of(&CompactionOptionsUniversal::allow_trivial_move),
|
|
||||||
OptionType::kBoolean, OptionVerificationType::kNormal,
|
|
||||||
OptionTypeFlags::kMutable,
|
|
||||||
offsetof(class CompactionOptionsUniversal, allow_trivial_move)}}};
|
|
||||||
|
|
||||||
std::unordered_map<std::string, CompactionStopStyle>
|
std::unordered_map<std::string, CompactionStopStyle>
|
||||||
OptionsHelper::compaction_stop_style_string_map = {
|
OptionsHelper::compaction_stop_style_string_map = {
|
||||||
{"kCompactionStopStyleSimilarSize", kCompactionStopStyleSimilarSize},
|
{"kCompactionStopStyleSimilarSize", kCompactionStopStyleSimilarSize},
|
||||||
{"kCompactionStopStyleTotalSize", kCompactionStopStyleTotalSize}};
|
{"kCompactionStopStyleTotalSize", kCompactionStopStyleTotalSize}};
|
||||||
|
|
||||||
std::unordered_map<std::string, OptionTypeInfo>
|
Status OptionTypeInfo::Parse(const ConfigOptions& config_options,
|
||||||
OptionsHelper::lru_cache_options_type_info = {
|
|
||||||
{"capacity",
|
|
||||||
{offset_of(&LRUCacheOptions::capacity), OptionType::kSizeT,
|
|
||||||
OptionVerificationType::kNormal, OptionTypeFlags::kMutable,
|
|
||||||
offsetof(struct LRUCacheOptions, capacity)}},
|
|
||||||
{"num_shard_bits",
|
|
||||||
{offset_of(&LRUCacheOptions::num_shard_bits), OptionType::kInt,
|
|
||||||
OptionVerificationType::kNormal, OptionTypeFlags::kMutable,
|
|
||||||
offsetof(struct LRUCacheOptions, num_shard_bits)}},
|
|
||||||
{"strict_capacity_limit",
|
|
||||||
{offset_of(&LRUCacheOptions::strict_capacity_limit),
|
|
||||||
OptionType::kBoolean, OptionVerificationType::kNormal,
|
|
||||||
OptionTypeFlags::kMutable,
|
|
||||||
offsetof(struct LRUCacheOptions, strict_capacity_limit)}},
|
|
||||||
{"high_pri_pool_ratio",
|
|
||||||
{offset_of(&LRUCacheOptions::high_pri_pool_ratio), OptionType::kDouble,
|
|
||||||
OptionVerificationType::kNormal, OptionTypeFlags::kMutable,
|
|
||||||
offsetof(struct LRUCacheOptions, high_pri_pool_ratio)}}};
|
|
||||||
|
|
||||||
Status OptionTypeInfo::ParseOption(const ConfigOptions& config_options,
|
|
||||||
const std::string& opt_name,
|
const std::string& opt_name,
|
||||||
const std::string& opt_value,
|
const std::string& opt_value,
|
||||||
char* opt_addr) const {
|
char* opt_addr) const {
|
||||||
@ -1282,9 +1096,9 @@ Status OptionTypeInfo::ParseOption(const ConfigOptions& config_options,
|
|||||||
try {
|
try {
|
||||||
if (opt_addr == nullptr) {
|
if (opt_addr == nullptr) {
|
||||||
return Status::NotFound("Could not find option: ", opt_name);
|
return Status::NotFound("Could not find option: ", opt_name);
|
||||||
} else if (parser_func != nullptr) {
|
} else if (parse_func_ != nullptr) {
|
||||||
return parser_func(config_options, opt_name, opt_value, opt_addr);
|
return parse_func_(config_options, opt_name, opt_value, opt_addr);
|
||||||
} else if (ParseOptionHelper(opt_addr, type, opt_value)) {
|
} else if (ParseOptionHelper(opt_addr, type_, opt_value)) {
|
||||||
return Status::OK();
|
return Status::OK();
|
||||||
} else if (IsByName()) {
|
} else if (IsByName()) {
|
||||||
return Status::NotSupported("Deserializing the option " + opt_name +
|
return Status::NotSupported("Deserializing the option " + opt_name +
|
||||||
@ -1298,23 +1112,122 @@ Status OptionTypeInfo::ParseOption(const ConfigOptions& config_options,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Status OptionTypeInfo::SerializeOption(const ConfigOptions& config_options,
|
Status OptionTypeInfo::ParseStruct(
|
||||||
|
const ConfigOptions& config_options, const std::string& struct_name,
|
||||||
|
const std::unordered_map<std::string, OptionTypeInfo>* struct_map,
|
||||||
|
const std::string& opt_name, const std::string& opt_value, char* opt_addr) {
|
||||||
|
assert(struct_map);
|
||||||
|
Status status;
|
||||||
|
if (opt_name == struct_name || EndsWith(opt_name, "." + struct_name)) {
|
||||||
|
// This option represents the entire struct
|
||||||
|
std::unordered_map<std::string, std::string> opt_map;
|
||||||
|
status = StringToMap(opt_value, &opt_map);
|
||||||
|
if (status.ok()) {
|
||||||
|
for (const auto& map_iter : opt_map) {
|
||||||
|
const auto iter = struct_map->find(map_iter.first);
|
||||||
|
if (iter != struct_map->end()) {
|
||||||
|
status = iter->second.Parse(config_options, map_iter.first,
|
||||||
|
map_iter.second,
|
||||||
|
opt_addr + iter->second.offset_);
|
||||||
|
} else {
|
||||||
|
return Status::InvalidArgument("Unrecognized option: ",
|
||||||
|
struct_name + "." + map_iter.first);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if (StartsWith(opt_name, struct_name + ".")) {
|
||||||
|
// This option represents a nested field in the struct (e.g, struct.field)
|
||||||
|
std::string elem_name;
|
||||||
|
const auto opt_info =
|
||||||
|
Find(opt_name.substr(struct_name.size() + 1), *struct_map, &elem_name);
|
||||||
|
if (opt_info != nullptr) {
|
||||||
|
status = opt_info->Parse(config_options, elem_name, opt_value,
|
||||||
|
opt_addr + opt_info->offset_);
|
||||||
|
} else {
|
||||||
|
status = Status::InvalidArgument("Unrecognized option: ", opt_name);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// This option represents a field in the struct (e.g. field)
|
||||||
|
std::string elem_name;
|
||||||
|
const auto opt_info = Find(opt_name, *struct_map, &elem_name);
|
||||||
|
if (opt_info != nullptr) {
|
||||||
|
status = opt_info->Parse(config_options, elem_name, opt_value,
|
||||||
|
opt_addr + opt_info->offset_);
|
||||||
|
} else {
|
||||||
|
status = Status::InvalidArgument("Unrecognized option: ",
|
||||||
|
struct_name + "." + opt_name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
|
||||||
|
Status OptionTypeInfo::Serialize(const ConfigOptions& config_options,
|
||||||
const std::string& opt_name,
|
const std::string& opt_name,
|
||||||
const char* opt_addr,
|
const char* opt_addr,
|
||||||
std::string* opt_value) const {
|
std::string* opt_value) const {
|
||||||
// If the option is no longer used in rocksdb and marked as deprecated,
|
// If the option is no longer used in rocksdb and marked as deprecated,
|
||||||
// we skip it in the serialization.
|
// we skip it in the serialization.
|
||||||
Status s;
|
if (opt_addr != nullptr && ShouldSerialize()) {
|
||||||
if (opt_addr == nullptr || IsDeprecated()) {
|
if (serialize_func_ != nullptr) {
|
||||||
s = Status::OK();
|
return serialize_func_(config_options, opt_name, opt_addr, opt_value);
|
||||||
} else if (string_func != nullptr) {
|
} else if (!SerializeSingleOptionHelper(opt_addr, type_, opt_value)) {
|
||||||
s = string_func(config_options, opt_name, opt_addr, opt_value);
|
return Status::InvalidArgument("Cannot serialize option: ", opt_name);
|
||||||
} else if (SerializeSingleOptionHelper(opt_addr, type, opt_value)) {
|
|
||||||
s = Status::OK();
|
|
||||||
} else {
|
|
||||||
s = Status::InvalidArgument("Cannot serialize option: ", opt_name);
|
|
||||||
}
|
}
|
||||||
return s;
|
}
|
||||||
|
return Status::OK();
|
||||||
|
}
|
||||||
|
|
||||||
|
Status OptionTypeInfo::SerializeStruct(
|
||||||
|
const ConfigOptions& config_options, const std::string& struct_name,
|
||||||
|
const std::unordered_map<std::string, OptionTypeInfo>* struct_map,
|
||||||
|
const std::string& opt_name, const char* opt_addr, std::string* value) {
|
||||||
|
assert(struct_map);
|
||||||
|
Status status;
|
||||||
|
if (EndsWith(opt_name, struct_name)) {
|
||||||
|
// We are going to write the struct as "{ prop1=value1; prop2=value2;}.
|
||||||
|
// Set the delimiter to ";" so that the everything will be on one line.
|
||||||
|
ConfigOptions embedded = config_options;
|
||||||
|
embedded.delimiter = ";";
|
||||||
|
|
||||||
|
// This option represents the entire struct
|
||||||
|
std::string result;
|
||||||
|
for (const auto& iter : *struct_map) {
|
||||||
|
std::string single;
|
||||||
|
const auto& opt_info = iter.second;
|
||||||
|
if (opt_info.ShouldSerialize()) {
|
||||||
|
status = opt_info.Serialize(embedded, iter.first,
|
||||||
|
opt_addr + opt_info.offset_, &single);
|
||||||
|
if (!status.ok()) {
|
||||||
|
return status;
|
||||||
|
} else {
|
||||||
|
result.append(iter.first + "=" + single + embedded.delimiter);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
*value = "{" + result + "}";
|
||||||
|
} else if (StartsWith(opt_name, struct_name + ".")) {
|
||||||
|
// This option represents a nested field in the struct (e.g, struct.field)
|
||||||
|
std::string elem_name;
|
||||||
|
const auto opt_info =
|
||||||
|
Find(opt_name.substr(struct_name.size() + 1), *struct_map, &elem_name);
|
||||||
|
if (opt_info != nullptr) {
|
||||||
|
status = opt_info->Serialize(config_options, elem_name,
|
||||||
|
opt_addr + opt_info->offset_, value);
|
||||||
|
} else {
|
||||||
|
status = Status::InvalidArgument("Unrecognized option: ", opt_name);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// This option represents a field in the struct (e.g. field)
|
||||||
|
std::string elem_name;
|
||||||
|
const auto opt_info = Find(opt_name, *struct_map, &elem_name);
|
||||||
|
if (opt_info == nullptr) {
|
||||||
|
return Status::InvalidArgument("Unrecognized option: ", opt_name);
|
||||||
|
} else if (opt_info->ShouldSerialize()) {
|
||||||
|
return opt_info->Serialize(config_options, opt_name + "." + elem_name,
|
||||||
|
opt_addr + opt_info->offset_, value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return Status::OK();
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
@ -1380,43 +1293,14 @@ static bool AreOptionsEqual(OptionType type, const char* this_offset,
|
|||||||
return IsOptionEqual<ChecksumType>(this_offset, that_offset);
|
return IsOptionEqual<ChecksumType>(this_offset, that_offset);
|
||||||
case OptionType::kEncodingType:
|
case OptionType::kEncodingType:
|
||||||
return IsOptionEqual<EncodingType>(this_offset, that_offset);
|
return IsOptionEqual<EncodingType>(this_offset, that_offset);
|
||||||
case OptionType::kCompactionOptionsFIFO: {
|
|
||||||
CompactionOptionsFIFO lhs =
|
|
||||||
*reinterpret_cast<const CompactionOptionsFIFO*>(this_offset);
|
|
||||||
CompactionOptionsFIFO rhs =
|
|
||||||
*reinterpret_cast<const CompactionOptionsFIFO*>(that_offset);
|
|
||||||
if (lhs.max_table_files_size == rhs.max_table_files_size &&
|
|
||||||
lhs.allow_compaction == rhs.allow_compaction) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
case OptionType::kCompactionOptionsUniversal: {
|
|
||||||
CompactionOptionsUniversal lhs =
|
|
||||||
*reinterpret_cast<const CompactionOptionsUniversal*>(this_offset);
|
|
||||||
CompactionOptionsUniversal rhs =
|
|
||||||
*reinterpret_cast<const CompactionOptionsUniversal*>(that_offset);
|
|
||||||
if (lhs.size_ratio == rhs.size_ratio &&
|
|
||||||
lhs.min_merge_width == rhs.min_merge_width &&
|
|
||||||
lhs.max_merge_width == rhs.max_merge_width &&
|
|
||||||
lhs.max_size_amplification_percent ==
|
|
||||||
rhs.max_size_amplification_percent &&
|
|
||||||
lhs.compression_size_percent == rhs.compression_size_percent &&
|
|
||||||
lhs.stop_style == rhs.stop_style &&
|
|
||||||
lhs.allow_trivial_move == rhs.allow_trivial_move) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
default:
|
default:
|
||||||
return false;
|
return false;
|
||||||
} // End switch
|
} // End switch
|
||||||
}
|
}
|
||||||
|
|
||||||
bool OptionTypeInfo::MatchesOption(const ConfigOptions& config_options,
|
bool OptionTypeInfo::AreEqual(const ConfigOptions& config_options,
|
||||||
const std::string& opt_name,
|
const std::string& opt_name,
|
||||||
const char* this_addr, const char* that_addr,
|
const char* this_addr, const char* that_addr,
|
||||||
|
|
||||||
std::string* mismatch) const {
|
std::string* mismatch) const {
|
||||||
if (!config_options.IsCheckEnabled(GetSanityLevel())) {
|
if (!config_options.IsCheckEnabled(GetSanityLevel())) {
|
||||||
return true; // If the sanity level is not being checked, skip it
|
return true; // If the sanity level is not being checked, skip it
|
||||||
@ -1425,11 +1309,12 @@ bool OptionTypeInfo::MatchesOption(const ConfigOptions& config_options,
|
|||||||
if (this_addr == that_addr) {
|
if (this_addr == that_addr) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
} else if (equals_func != nullptr) {
|
} else if (equals_func_ != nullptr) {
|
||||||
if (equals_func(config_options, opt_name, this_addr, that_addr, mismatch)) {
|
if (equals_func_(config_options, opt_name, this_addr, that_addr,
|
||||||
|
mismatch)) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
} else if (AreOptionsEqual(type, this_addr, that_addr)) {
|
} else if (AreOptionsEqual(type_, this_addr, that_addr)) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (mismatch->empty()) {
|
if (mismatch->empty()) {
|
||||||
@ -1438,29 +1323,82 @@ bool OptionTypeInfo::MatchesOption(const ConfigOptions& config_options,
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool OptionTypeInfo::MatchesByName(const ConfigOptions& config_options,
|
bool OptionTypeInfo::StructsAreEqual(
|
||||||
|
const ConfigOptions& config_options, const std::string& struct_name,
|
||||||
|
const std::unordered_map<std::string, OptionTypeInfo>* struct_map,
|
||||||
|
const std::string& opt_name, const char* this_addr, const char* that_addr,
|
||||||
|
std::string* mismatch) {
|
||||||
|
assert(struct_map);
|
||||||
|
Status status;
|
||||||
|
bool matches = true;
|
||||||
|
std::string result;
|
||||||
|
if (EndsWith(opt_name, struct_name)) {
|
||||||
|
// This option represents the entire struct
|
||||||
|
for (const auto& iter : *struct_map) {
|
||||||
|
const auto& opt_info = iter.second;
|
||||||
|
|
||||||
|
matches = opt_info.AreEqual(config_options, iter.first,
|
||||||
|
this_addr + opt_info.offset_,
|
||||||
|
that_addr + opt_info.offset_, &result);
|
||||||
|
if (!matches) {
|
||||||
|
*mismatch = struct_name + "." + result;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if (StartsWith(opt_name, struct_name + ".")) {
|
||||||
|
// This option represents a nested field in the struct (e.g, struct.field)
|
||||||
|
std::string elem_name;
|
||||||
|
const auto opt_info =
|
||||||
|
Find(opt_name.substr(struct_name.size() + 1), *struct_map, &elem_name);
|
||||||
|
assert(opt_info);
|
||||||
|
if (opt_info == nullptr) {
|
||||||
|
*mismatch = opt_name;
|
||||||
|
matches = false;
|
||||||
|
} else if (!opt_info->AreEqual(config_options, elem_name,
|
||||||
|
this_addr + opt_info->offset_,
|
||||||
|
that_addr + opt_info->offset_, &result)) {
|
||||||
|
matches = false;
|
||||||
|
*mismatch = struct_name + "." + result;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// This option represents a field in the struct (e.g. field)
|
||||||
|
std::string elem_name;
|
||||||
|
const auto opt_info = Find(opt_name, *struct_map, &elem_name);
|
||||||
|
assert(opt_info);
|
||||||
|
if (opt_info == nullptr) {
|
||||||
|
*mismatch = struct_name + "." + opt_name;
|
||||||
|
matches = false;
|
||||||
|
} else if (!opt_info->AreEqual(config_options, elem_name,
|
||||||
|
this_addr + opt_info->offset_,
|
||||||
|
that_addr + opt_info->offset_, &result)) {
|
||||||
|
matches = false;
|
||||||
|
*mismatch = struct_name + "." + result;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return matches;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool OptionTypeInfo::AreEqualByName(const ConfigOptions& config_options,
|
||||||
const std::string& opt_name,
|
const std::string& opt_name,
|
||||||
const char* this_addr,
|
const char* this_addr,
|
||||||
const char* that_addr) const {
|
const char* that_addr) const {
|
||||||
if (IsByName()) {
|
if (IsByName()) {
|
||||||
std::string that_value;
|
std::string that_value;
|
||||||
if (SerializeOption(config_options, opt_name, that_addr, &that_value)
|
if (Serialize(config_options, opt_name, that_addr, &that_value).ok()) {
|
||||||
.ok()) {
|
return AreEqualByName(config_options, opt_name, this_addr, that_value);
|
||||||
return MatchesByName(config_options, opt_name, this_addr, that_value);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool OptionTypeInfo::MatchesByName(const ConfigOptions& config_options,
|
bool OptionTypeInfo::AreEqualByName(const ConfigOptions& config_options,
|
||||||
const std::string& opt_name,
|
const std::string& opt_name,
|
||||||
const char* opt_addr,
|
const char* opt_addr,
|
||||||
const std::string& that_value) const {
|
const std::string& that_value) const {
|
||||||
std::string this_value;
|
std::string this_value;
|
||||||
if (!IsByName()) {
|
if (!IsByName()) {
|
||||||
return false;
|
return false;
|
||||||
} else if (!SerializeOption(config_options, opt_name, opt_addr, &this_value)
|
} else if (!Serialize(config_options, opt_name, opt_addr, &this_value).ok()) {
|
||||||
.ok()) {
|
|
||||||
return false;
|
return false;
|
||||||
} else if (IsEnabled(OptionVerificationType::kByNameAllowFromNull)) {
|
} else if (IsEnabled(OptionVerificationType::kByNameAllowFromNull)) {
|
||||||
if (that_value == kNullptrString) {
|
if (that_value == kNullptrString) {
|
||||||
@ -1473,6 +1411,30 @@ bool OptionTypeInfo::MatchesByName(const ConfigOptions& config_options,
|
|||||||
}
|
}
|
||||||
return (this_value == that_value);
|
return (this_value == that_value);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const OptionTypeInfo* OptionTypeInfo::Find(
|
||||||
|
const std::string& opt_name,
|
||||||
|
const std::unordered_map<std::string, OptionTypeInfo>& opt_map,
|
||||||
|
std::string* elem_name) {
|
||||||
|
const auto iter = opt_map.find(opt_name); // Look up the value in the map
|
||||||
|
if (iter != opt_map.end()) { // Found the option in the map
|
||||||
|
*elem_name = opt_name; // Return the name
|
||||||
|
return &(iter->second); // Return the contents of the iterator
|
||||||
|
} else {
|
||||||
|
auto idx = opt_name.find("."); // Look for a separator
|
||||||
|
if (idx > 0 && idx != std::string::npos) { // We found a separator
|
||||||
|
auto siter =
|
||||||
|
opt_map.find(opt_name.substr(0, idx)); // Look for the short name
|
||||||
|
if (siter != opt_map.end()) { // We found the short name
|
||||||
|
if (siter->second.IsStruct()) { // If the object is a struct
|
||||||
|
*elem_name = opt_name.substr(idx + 1); // Return the rest
|
||||||
|
return &(siter->second); // Return the contents of the iterator
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
#endif // !ROCKSDB_LITE
|
#endif // !ROCKSDB_LITE
|
||||||
|
|
||||||
} // namespace ROCKSDB_NAMESPACE
|
} // namespace ROCKSDB_NAMESPACE
|
||||||
|
@ -98,24 +98,15 @@ struct OptionsHelper {
|
|||||||
compression_type_string_map;
|
compression_type_string_map;
|
||||||
#ifndef ROCKSDB_LITE
|
#ifndef ROCKSDB_LITE
|
||||||
static std::unordered_map<std::string, OptionTypeInfo> cf_options_type_info;
|
static std::unordered_map<std::string, OptionTypeInfo> cf_options_type_info;
|
||||||
static std::unordered_map<std::string, OptionTypeInfo>
|
|
||||||
fifo_compaction_options_type_info;
|
|
||||||
static std::unordered_map<std::string, OptionTypeInfo>
|
|
||||||
universal_compaction_options_type_info;
|
|
||||||
static std::unordered_map<std::string, CompactionStopStyle>
|
static std::unordered_map<std::string, CompactionStopStyle>
|
||||||
compaction_stop_style_string_map;
|
compaction_stop_style_string_map;
|
||||||
static std::unordered_map<std::string, OptionTypeInfo> db_options_type_info;
|
static std::unordered_map<std::string, OptionTypeInfo> db_options_type_info;
|
||||||
static std::unordered_map<std::string, OptionTypeInfo>
|
|
||||||
lru_cache_options_type_info;
|
|
||||||
static std::unordered_map<std::string, EncodingType> encoding_type_string_map;
|
static std::unordered_map<std::string, EncodingType> encoding_type_string_map;
|
||||||
static std::unordered_map<std::string, CompactionStyle>
|
static std::unordered_map<std::string, CompactionStyle>
|
||||||
compaction_style_string_map;
|
compaction_style_string_map;
|
||||||
static std::unordered_map<std::string, CompactionPri>
|
static std::unordered_map<std::string, CompactionPri>
|
||||||
compaction_pri_string_map;
|
compaction_pri_string_map;
|
||||||
static ColumnFamilyOptions dummy_cf_options;
|
static ColumnFamilyOptions dummy_cf_options;
|
||||||
static CompactionOptionsFIFO dummy_comp_options;
|
|
||||||
static LRUCacheOptions dummy_lru_cache_options;
|
|
||||||
static CompactionOptionsUniversal dummy_comp_options_universal;
|
|
||||||
#endif // !ROCKSDB_LITE
|
#endif // !ROCKSDB_LITE
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -128,15 +119,9 @@ static auto& compaction_stop_style_to_string =
|
|||||||
static auto& checksum_type_string_map = OptionsHelper::checksum_type_string_map;
|
static auto& checksum_type_string_map = OptionsHelper::checksum_type_string_map;
|
||||||
#ifndef ROCKSDB_LITE
|
#ifndef ROCKSDB_LITE
|
||||||
static auto& cf_options_type_info = OptionsHelper::cf_options_type_info;
|
static auto& cf_options_type_info = OptionsHelper::cf_options_type_info;
|
||||||
static auto& fifo_compaction_options_type_info =
|
|
||||||
OptionsHelper::fifo_compaction_options_type_info;
|
|
||||||
static auto& universal_compaction_options_type_info =
|
|
||||||
OptionsHelper::universal_compaction_options_type_info;
|
|
||||||
static auto& compaction_stop_style_string_map =
|
static auto& compaction_stop_style_string_map =
|
||||||
OptionsHelper::compaction_stop_style_string_map;
|
OptionsHelper::compaction_stop_style_string_map;
|
||||||
static auto& db_options_type_info = OptionsHelper::db_options_type_info;
|
static auto& db_options_type_info = OptionsHelper::db_options_type_info;
|
||||||
static auto& lru_cache_options_type_info =
|
|
||||||
OptionsHelper::lru_cache_options_type_info;
|
|
||||||
static auto& compression_type_string_map =
|
static auto& compression_type_string_map =
|
||||||
OptionsHelper::compression_type_string_map;
|
OptionsHelper::compression_type_string_map;
|
||||||
static auto& encoding_type_string_map = OptionsHelper::encoding_type_string_map;
|
static auto& encoding_type_string_map = OptionsHelper::encoding_type_string_map;
|
||||||
|
@ -610,13 +610,13 @@ Status RocksDBOptionsParser::VerifyDBOptions(
|
|||||||
const auto& opt_info = pair.second;
|
const auto& opt_info = pair.second;
|
||||||
if (config_options.IsCheckEnabled(opt_info.GetSanityLevel())) {
|
if (config_options.IsCheckEnabled(opt_info.GetSanityLevel())) {
|
||||||
const char* base_addr =
|
const char* base_addr =
|
||||||
reinterpret_cast<const char*>(&base_opt) + opt_info.offset;
|
reinterpret_cast<const char*>(&base_opt) + opt_info.offset_;
|
||||||
const char* file_addr =
|
const char* file_addr =
|
||||||
reinterpret_cast<const char*>(&file_opt) + opt_info.offset;
|
reinterpret_cast<const char*>(&file_opt) + opt_info.offset_;
|
||||||
std::string mismatch;
|
std::string mismatch;
|
||||||
if (!opt_info.MatchesOption(config_options, pair.first, base_addr,
|
if (!opt_info.AreEqual(config_options, pair.first, base_addr, file_addr,
|
||||||
file_addr, &mismatch) &&
|
&mismatch) &&
|
||||||
!opt_info.MatchesByName(config_options, pair.first, base_addr,
|
!opt_info.AreEqualByName(config_options, pair.first, base_addr,
|
||||||
file_addr)) {
|
file_addr)) {
|
||||||
const size_t kBufferSize = 2048;
|
const size_t kBufferSize = 2048;
|
||||||
char buffer[kBufferSize];
|
char buffer[kBufferSize];
|
||||||
@ -627,10 +627,10 @@ Status RocksDBOptionsParser::VerifyDBOptions(
|
|||||||
"[RocksDBOptionsParser]: "
|
"[RocksDBOptionsParser]: "
|
||||||
"failed the verification on ColumnFamilyOptions::%s",
|
"failed the verification on ColumnFamilyOptions::%s",
|
||||||
pair.first.c_str());
|
pair.first.c_str());
|
||||||
Status s = opt_info.SerializeOption(config_options, pair.first,
|
Status s = opt_info.Serialize(config_options, pair.first, base_addr,
|
||||||
base_addr, &base_value);
|
&base_value);
|
||||||
if (s.ok()) {
|
if (s.ok()) {
|
||||||
s = opt_info.SerializeOption(config_options, pair.first, file_addr,
|
s = opt_info.Serialize(config_options, pair.first, file_addr,
|
||||||
&file_value);
|
&file_value);
|
||||||
}
|
}
|
||||||
snprintf(buffer, sizeof(buffer),
|
snprintf(buffer, sizeof(buffer),
|
||||||
@ -668,11 +668,11 @@ Status RocksDBOptionsParser::VerifyCFOptions(
|
|||||||
if (config_options.IsCheckEnabled(opt_info.GetSanityLevel())) {
|
if (config_options.IsCheckEnabled(opt_info.GetSanityLevel())) {
|
||||||
std::string mismatch;
|
std::string mismatch;
|
||||||
const char* base_addr =
|
const char* base_addr =
|
||||||
reinterpret_cast<const char*>(&base_opt) + opt_info.offset;
|
reinterpret_cast<const char*>(&base_opt) + opt_info.offset_;
|
||||||
const char* file_addr =
|
const char* file_addr =
|
||||||
reinterpret_cast<const char*>(&file_opt) + opt_info.offset;
|
reinterpret_cast<const char*>(&file_opt) + opt_info.offset_;
|
||||||
bool matches = opt_info.MatchesOption(config_options, pair.first,
|
bool matches = opt_info.AreEqual(config_options, pair.first, base_addr,
|
||||||
base_addr, file_addr, &mismatch);
|
file_addr, &mismatch);
|
||||||
if (!matches && opt_info.IsByName()) {
|
if (!matches && opt_info.IsByName()) {
|
||||||
if (opt_map == nullptr) {
|
if (opt_map == nullptr) {
|
||||||
matches = true;
|
matches = true;
|
||||||
@ -681,7 +681,7 @@ Status RocksDBOptionsParser::VerifyCFOptions(
|
|||||||
if (iter == opt_map->end()) {
|
if (iter == opt_map->end()) {
|
||||||
matches = true;
|
matches = true;
|
||||||
} else {
|
} else {
|
||||||
matches = opt_info.MatchesByName(config_options, pair.first,
|
matches = opt_info.AreEqualByName(config_options, pair.first,
|
||||||
base_addr, iter->second);
|
base_addr, iter->second);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -692,10 +692,10 @@ Status RocksDBOptionsParser::VerifyCFOptions(
|
|||||||
char buffer[kBufferSize];
|
char buffer[kBufferSize];
|
||||||
std::string base_value;
|
std::string base_value;
|
||||||
std::string file_value;
|
std::string file_value;
|
||||||
Status s = opt_info.SerializeOption(config_options, pair.first,
|
Status s = opt_info.Serialize(config_options, pair.first, base_addr,
|
||||||
base_addr, &base_value);
|
&base_value);
|
||||||
if (s.ok()) {
|
if (s.ok()) {
|
||||||
s = opt_info.SerializeOption(config_options, pair.first, file_addr,
|
s = opt_info.Serialize(config_options, pair.first, file_addr,
|
||||||
&file_value);
|
&file_value);
|
||||||
}
|
}
|
||||||
int offset =
|
int offset =
|
||||||
|
@ -2999,13 +2999,12 @@ static void TestAndCompareOption(const ConfigOptions& config_options,
|
|||||||
const std::string& opt_name, void* base_ptr,
|
const std::string& opt_name, void* base_ptr,
|
||||||
void* comp_ptr) {
|
void* comp_ptr) {
|
||||||
std::string result, mismatch;
|
std::string result, mismatch;
|
||||||
char* base_addr = reinterpret_cast<char*>(base_ptr) + opt_info.offset;
|
char* base_addr = reinterpret_cast<char*>(base_ptr) + opt_info.offset_;
|
||||||
char* comp_addr = reinterpret_cast<char*>(comp_ptr) + opt_info.offset;
|
char* comp_addr = reinterpret_cast<char*>(comp_ptr) + opt_info.offset_;
|
||||||
ASSERT_OK(
|
ASSERT_OK(opt_info.Serialize(config_options, opt_name, base_addr, &result));
|
||||||
opt_info.SerializeOption(config_options, opt_name, base_addr, &result));
|
ASSERT_OK(opt_info.Parse(config_options, opt_name, result, comp_addr));
|
||||||
ASSERT_OK(opt_info.ParseOption(config_options, opt_name, result, comp_addr));
|
ASSERT_TRUE(opt_info.AreEqual(config_options, opt_name, base_addr, comp_addr,
|
||||||
ASSERT_TRUE(opt_info.MatchesOption(config_options, opt_name, base_addr,
|
&mismatch));
|
||||||
comp_addr, &mismatch));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void TestAndCompareOption(const ConfigOptions& config_options,
|
static void TestAndCompareOption(const ConfigOptions& config_options,
|
||||||
@ -3013,9 +3012,8 @@ static void TestAndCompareOption(const ConfigOptions& config_options,
|
|||||||
const std::string& opt_name,
|
const std::string& opt_name,
|
||||||
const std::string& opt_value, void* base_ptr,
|
const std::string& opt_value, void* base_ptr,
|
||||||
void* comp_ptr) {
|
void* comp_ptr) {
|
||||||
char* base_addr = reinterpret_cast<char*>(base_ptr) + opt_info.offset;
|
char* base_addr = reinterpret_cast<char*>(base_ptr) + opt_info.offset_;
|
||||||
ASSERT_OK(
|
ASSERT_OK(opt_info.Parse(config_options, opt_name, opt_value, base_addr));
|
||||||
opt_info.ParseOption(config_options, opt_name, opt_value, base_addr));
|
|
||||||
TestAndCompareOption(config_options, opt_info, opt_name, base_ptr, comp_ptr);
|
TestAndCompareOption(config_options, opt_info, opt_name, base_ptr, comp_ptr);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -3026,8 +3024,8 @@ void TestOptInfo(const ConfigOptions& config_options, OptionType opt_type,
|
|||||||
OptionTypeInfo opt_info(0, opt_type);
|
OptionTypeInfo opt_info(0, opt_type);
|
||||||
char* base_addr = reinterpret_cast<char*>(base);
|
char* base_addr = reinterpret_cast<char*>(base);
|
||||||
char* comp_addr = reinterpret_cast<char*>(comp);
|
char* comp_addr = reinterpret_cast<char*>(comp);
|
||||||
ASSERT_FALSE(opt_info.MatchesOption(config_options, "base", base_addr,
|
ASSERT_FALSE(
|
||||||
comp_addr, &result));
|
opt_info.AreEqual(config_options, "base", base_addr, comp_addr, &result));
|
||||||
ASSERT_EQ(result, "base");
|
ASSERT_EQ(result, "base");
|
||||||
ASSERT_NE(*base, *comp);
|
ASSERT_NE(*base, *comp);
|
||||||
TestAndCompareOption(config_options, opt_info, "base", base_addr, comp_addr);
|
TestAndCompareOption(config_options, opt_info, "base", base_addr, comp_addr);
|
||||||
@ -3092,38 +3090,33 @@ TEST_F(OptionTypeInfoTest, TestInvalidArgs) {
|
|||||||
size_t sz;
|
size_t sz;
|
||||||
double d;
|
double d;
|
||||||
|
|
||||||
|
ASSERT_NOK(OptionTypeInfo(0, OptionType::kBoolean)
|
||||||
|
.Parse(config_options, "b", "x", reinterpret_cast<char*>(&b)));
|
||||||
|
ASSERT_NOK(OptionTypeInfo(0, OptionType::kInt)
|
||||||
|
.Parse(config_options, "b", "x", reinterpret_cast<char*>(&i)));
|
||||||
ASSERT_NOK(
|
ASSERT_NOK(
|
||||||
OptionTypeInfo(0, OptionType::kBoolean)
|
OptionTypeInfo(0, OptionType::kInt32T)
|
||||||
.ParseOption(config_options, "b", "x", reinterpret_cast<char*>(&b)));
|
.Parse(config_options, "b", "x", reinterpret_cast<char*>(&i32)));
|
||||||
ASSERT_NOK(
|
ASSERT_NOK(
|
||||||
OptionTypeInfo(0, OptionType::kInt)
|
OptionTypeInfo(0, OptionType::kInt64T)
|
||||||
.ParseOption(config_options, "b", "x", reinterpret_cast<char*>(&i)));
|
.Parse(config_options, "b", "x", reinterpret_cast<char*>(&i64)));
|
||||||
ASSERT_NOK(OptionTypeInfo(0, OptionType::kInt32T)
|
ASSERT_NOK(OptionTypeInfo(0, OptionType::kUInt)
|
||||||
.ParseOption(config_options, "b", "x",
|
.Parse(config_options, "b", "x", reinterpret_cast<char*>(&u)));
|
||||||
reinterpret_cast<char*>(&i32)));
|
|
||||||
ASSERT_NOK(OptionTypeInfo(0, OptionType::kInt64T)
|
|
||||||
.ParseOption(config_options, "b", "x",
|
|
||||||
reinterpret_cast<char*>(&i64)));
|
|
||||||
ASSERT_NOK(
|
ASSERT_NOK(
|
||||||
OptionTypeInfo(0, OptionType::kUInt)
|
OptionTypeInfo(0, OptionType::kUInt32T)
|
||||||
.ParseOption(config_options, "b", "x", reinterpret_cast<char*>(&u)));
|
.Parse(config_options, "b", "x", reinterpret_cast<char*>(&u32)));
|
||||||
ASSERT_NOK(OptionTypeInfo(0, OptionType::kUInt32T)
|
ASSERT_NOK(
|
||||||
.ParseOption(config_options, "b", "x",
|
OptionTypeInfo(0, OptionType::kUInt64T)
|
||||||
reinterpret_cast<char*>(&u32)));
|
.Parse(config_options, "b", "x", reinterpret_cast<char*>(&u64)));
|
||||||
ASSERT_NOK(OptionTypeInfo(0, OptionType::kUInt64T)
|
|
||||||
.ParseOption(config_options, "b", "x",
|
|
||||||
reinterpret_cast<char*>(&u64)));
|
|
||||||
ASSERT_NOK(
|
ASSERT_NOK(
|
||||||
OptionTypeInfo(0, OptionType::kSizeT)
|
OptionTypeInfo(0, OptionType::kSizeT)
|
||||||
.ParseOption(config_options, "b", "x", reinterpret_cast<char*>(&sz)));
|
.Parse(config_options, "b", "x", reinterpret_cast<char*>(&sz)));
|
||||||
ASSERT_NOK(
|
ASSERT_NOK(OptionTypeInfo(0, OptionType::kDouble)
|
||||||
OptionTypeInfo(0, OptionType::kDouble)
|
.Parse(config_options, "b", "x", reinterpret_cast<char*>(&d)));
|
||||||
.ParseOption(config_options, "b", "x", reinterpret_cast<char*>(&d)));
|
|
||||||
|
|
||||||
// Don't know how to convert Unknowns to anything else
|
// Don't know how to convert Unknowns to anything else
|
||||||
ASSERT_NOK(
|
ASSERT_NOK(OptionTypeInfo(0, OptionType::kUnknown)
|
||||||
OptionTypeInfo(0, OptionType::kUnknown)
|
.Parse(config_options, "b", "x", reinterpret_cast<char*>(&d)));
|
||||||
.ParseOption(config_options, "b", "x", reinterpret_cast<char*>(&d)));
|
|
||||||
|
|
||||||
// Verify that if the parse function throws an exception, it is also trapped
|
// Verify that if the parse function throws an exception, it is also trapped
|
||||||
OptionTypeInfo func_info(0, OptionType::kUnknown,
|
OptionTypeInfo func_info(0, OptionType::kUnknown,
|
||||||
@ -3135,10 +3128,10 @@ TEST_F(OptionTypeInfoTest, TestInvalidArgs) {
|
|||||||
*ptr = ParseInt(value);
|
*ptr = ParseInt(value);
|
||||||
return Status::OK();
|
return Status::OK();
|
||||||
});
|
});
|
||||||
ASSERT_OK(func_info.ParseOption(config_options, "b", "1",
|
ASSERT_OK(
|
||||||
reinterpret_cast<char*>(&i)));
|
func_info.Parse(config_options, "b", "1", reinterpret_cast<char*>(&i)));
|
||||||
ASSERT_NOK(func_info.ParseOption(config_options, "b", "x",
|
ASSERT_NOK(
|
||||||
reinterpret_cast<char*>(&i)));
|
func_info.Parse(config_options, "b", "x", reinterpret_cast<char*>(&i)));
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(OptionTypeInfoTest, TestParseFunc) {
|
TEST_F(OptionTypeInfoTest, TestParseFunc) {
|
||||||
@ -3157,10 +3150,10 @@ TEST_F(OptionTypeInfoTest, TestParseFunc) {
|
|||||||
});
|
});
|
||||||
ConfigOptions config_options;
|
ConfigOptions config_options;
|
||||||
std::string base;
|
std::string base;
|
||||||
ASSERT_OK(opt_info.ParseOption(config_options, "World", "Hello",
|
ASSERT_OK(opt_info.Parse(config_options, "World", "Hello",
|
||||||
reinterpret_cast<char*>(&base)));
|
reinterpret_cast<char*>(&base)));
|
||||||
ASSERT_EQ(base, "Hello World");
|
ASSERT_EQ(base, "Hello World");
|
||||||
ASSERT_NOK(opt_info.ParseOption(config_options, "Oops", "Hello",
|
ASSERT_NOK(opt_info.Parse(config_options, "Oops", "Hello",
|
||||||
reinterpret_cast<char*>(&base)));
|
reinterpret_cast<char*>(&base)));
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -3181,10 +3174,10 @@ TEST_F(OptionTypeInfoTest, TestSerializeFunc) {
|
|||||||
ConfigOptions config_options;
|
ConfigOptions config_options;
|
||||||
std::string base;
|
std::string base;
|
||||||
std::string value;
|
std::string value;
|
||||||
ASSERT_OK(opt_info.SerializeOption(config_options, "Hello",
|
ASSERT_OK(opt_info.Serialize(config_options, "Hello",
|
||||||
reinterpret_cast<char*>(&base), &value));
|
reinterpret_cast<char*>(&base), &value));
|
||||||
ASSERT_EQ(value, "Hello");
|
ASSERT_EQ(value, "Hello");
|
||||||
ASSERT_NOK(opt_info.SerializeOption(config_options, "Oops",
|
ASSERT_NOK(opt_info.Serialize(config_options, "Oops",
|
||||||
reinterpret_cast<char*>(&base), &value));
|
reinterpret_cast<char*>(&base), &value));
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -3212,16 +3205,16 @@ TEST_F(OptionTypeInfoTest, TestEqualsFunc) {
|
|||||||
int int1 = 100;
|
int int1 = 100;
|
||||||
int int2 = 200;
|
int int2 = 200;
|
||||||
std::string mismatch;
|
std::string mismatch;
|
||||||
ASSERT_TRUE(opt_info.MatchesOption(
|
ASSERT_TRUE(opt_info.AreEqual(
|
||||||
config_options, "LT", reinterpret_cast<const char*>(&int1),
|
config_options, "LT", reinterpret_cast<const char*>(&int1),
|
||||||
reinterpret_cast<const char*>(&int2), &mismatch));
|
reinterpret_cast<const char*>(&int2), &mismatch));
|
||||||
ASSERT_EQ(mismatch, "");
|
ASSERT_EQ(mismatch, "");
|
||||||
ASSERT_FALSE(opt_info.MatchesOption(
|
ASSERT_FALSE(opt_info.AreEqual(config_options, "GT",
|
||||||
config_options, "GT", reinterpret_cast<char*>(&int1),
|
reinterpret_cast<char*>(&int1),
|
||||||
reinterpret_cast<char*>(&int2), &mismatch));
|
reinterpret_cast<char*>(&int2), &mismatch));
|
||||||
ASSERT_EQ(mismatch, "GT");
|
ASSERT_EQ(mismatch, "GT");
|
||||||
ASSERT_FALSE(opt_info.MatchesOption(
|
ASSERT_FALSE(opt_info.AreEqual(config_options, "NO",
|
||||||
config_options, "NO", reinterpret_cast<char*>(&int1),
|
reinterpret_cast<char*>(&int1),
|
||||||
reinterpret_cast<char*>(&int2), &mismatch));
|
reinterpret_cast<char*>(&int2), &mismatch));
|
||||||
ASSERT_EQ(mismatch, "NO???");
|
ASSERT_EQ(mismatch, "NO???");
|
||||||
}
|
}
|
||||||
@ -3244,36 +3237,36 @@ TEST_F(OptionTypeInfoTest, TestOptionFlags) {
|
|||||||
std::string comp = "comp";
|
std::string comp = "comp";
|
||||||
|
|
||||||
// If marked string none, the serialization returns okay but does nothing
|
// If marked string none, the serialization returns okay but does nothing
|
||||||
ASSERT_OK(opt_none.SerializeOption(config_options, "None",
|
ASSERT_OK(opt_none.Serialize(config_options, "None",
|
||||||
reinterpret_cast<char*>(&base), &base));
|
reinterpret_cast<char*>(&base), &base));
|
||||||
// If marked never compare, they match even when they do not
|
// If marked never compare, they match even when they do not
|
||||||
ASSERT_TRUE(opt_never.MatchesOption(config_options, "Never",
|
ASSERT_TRUE(opt_never.AreEqual(config_options, "Never",
|
||||||
reinterpret_cast<char*>(&base),
|
reinterpret_cast<char*>(&base),
|
||||||
reinterpret_cast<char*>(&comp), &base));
|
reinterpret_cast<char*>(&comp), &base));
|
||||||
ASSERT_FALSE(opt_none.MatchesOption(config_options, "Never",
|
ASSERT_FALSE(opt_none.AreEqual(config_options, "Never",
|
||||||
reinterpret_cast<char*>(&base),
|
reinterpret_cast<char*>(&base),
|
||||||
reinterpret_cast<char*>(&comp), &base));
|
reinterpret_cast<char*>(&comp), &base));
|
||||||
|
|
||||||
// An alias can change the value via parse, but does nothing on serialize on
|
// An alias can change the value via parse, but does nothing on serialize on
|
||||||
// match
|
// match
|
||||||
std::string result;
|
std::string result;
|
||||||
ASSERT_OK(opt_alias.ParseOption(config_options, "Alias", "Alias",
|
ASSERT_OK(opt_alias.Parse(config_options, "Alias", "Alias",
|
||||||
reinterpret_cast<char*>(&base)));
|
reinterpret_cast<char*>(&base)));
|
||||||
ASSERT_OK(opt_alias.SerializeOption(config_options, "Alias",
|
ASSERT_OK(opt_alias.Serialize(config_options, "Alias",
|
||||||
reinterpret_cast<char*>(&base), &result));
|
reinterpret_cast<char*>(&base), &result));
|
||||||
ASSERT_TRUE(opt_alias.MatchesOption(config_options, "Alias",
|
ASSERT_TRUE(opt_alias.AreEqual(config_options, "Alias",
|
||||||
reinterpret_cast<char*>(&base),
|
reinterpret_cast<char*>(&base),
|
||||||
reinterpret_cast<char*>(&comp), &result));
|
reinterpret_cast<char*>(&comp), &result));
|
||||||
ASSERT_EQ(base, "Alias");
|
ASSERT_EQ(base, "Alias");
|
||||||
ASSERT_NE(base, comp);
|
ASSERT_NE(base, comp);
|
||||||
|
|
||||||
// Deprecated options do nothing on any of the commands
|
// Deprecated options do nothing on any of the commands
|
||||||
ASSERT_OK(opt_deprecated.ParseOption(config_options, "Alias", "Deprecated",
|
ASSERT_OK(opt_deprecated.Parse(config_options, "Alias", "Deprecated",
|
||||||
reinterpret_cast<char*>(&base)));
|
reinterpret_cast<char*>(&base)));
|
||||||
ASSERT_OK(opt_deprecated.SerializeOption(
|
ASSERT_OK(opt_deprecated.Serialize(config_options, "Alias",
|
||||||
config_options, "Alias", reinterpret_cast<char*>(&base), &result));
|
reinterpret_cast<char*>(&base), &result));
|
||||||
ASSERT_TRUE(opt_deprecated.MatchesOption(
|
ASSERT_TRUE(opt_deprecated.AreEqual(config_options, "Alias",
|
||||||
config_options, "Alias", reinterpret_cast<char*>(&base),
|
reinterpret_cast<char*>(&base),
|
||||||
reinterpret_cast<char*>(&comp), &result));
|
reinterpret_cast<char*>(&comp), &result));
|
||||||
ASSERT_EQ(base, "Alias");
|
ASSERT_EQ(base, "Alias");
|
||||||
ASSERT_NE(base, comp);
|
ASSERT_NE(base, comp);
|
||||||
@ -3293,14 +3286,14 @@ TEST_F(OptionTypeInfoTest, TestCustomEnum) {
|
|||||||
|
|
||||||
e2 = TestEnum::kA;
|
e2 = TestEnum::kA;
|
||||||
|
|
||||||
ASSERT_OK(opt_info.ParseOption(config_options, "", "B",
|
ASSERT_OK(
|
||||||
reinterpret_cast<char*>(&e1)));
|
opt_info.Parse(config_options, "", "B", reinterpret_cast<char*>(&e1)));
|
||||||
ASSERT_OK(opt_info.SerializeOption(config_options, "",
|
ASSERT_OK(opt_info.Serialize(config_options, "", reinterpret_cast<char*>(&e1),
|
||||||
reinterpret_cast<char*>(&e1), &result));
|
&result));
|
||||||
ASSERT_EQ(e1, TestEnum::kB);
|
ASSERT_EQ(e1, TestEnum::kB);
|
||||||
ASSERT_EQ(result, "B");
|
ASSERT_EQ(result, "B");
|
||||||
|
|
||||||
ASSERT_FALSE(opt_info.MatchesOption(config_options, "Enum",
|
ASSERT_FALSE(opt_info.AreEqual(config_options, "Enum",
|
||||||
reinterpret_cast<char*>(&e1),
|
reinterpret_cast<char*>(&e1),
|
||||||
reinterpret_cast<char*>(&e2), &mismatch));
|
reinterpret_cast<char*>(&e2), &mismatch));
|
||||||
ASSERT_EQ(mismatch, "Enum");
|
ASSERT_EQ(mismatch, "Enum");
|
||||||
@ -3310,8 +3303,8 @@ TEST_F(OptionTypeInfoTest, TestCustomEnum) {
|
|||||||
reinterpret_cast<char*>(&e2));
|
reinterpret_cast<char*>(&e2));
|
||||||
ASSERT_EQ(e2, TestEnum::kC);
|
ASSERT_EQ(e2, TestEnum::kC);
|
||||||
|
|
||||||
ASSERT_NOK(opt_info.ParseOption(config_options, "", "D",
|
ASSERT_NOK(
|
||||||
reinterpret_cast<char*>(&e1)));
|
opt_info.Parse(config_options, "", "D", reinterpret_cast<char*>(&e1)));
|
||||||
ASSERT_EQ(e1, TestEnum::kC);
|
ASSERT_EQ(e1, TestEnum::kC);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -3360,6 +3353,94 @@ TEST_F(OptionTypeInfoTest, TestBuiltinEnum) {
|
|||||||
ASSERT_EQ(e1, iter.second);
|
ASSERT_EQ(e1, iter.second);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST_F(OptionTypeInfoTest, TestStruct) {
|
||||||
|
struct Basic {
|
||||||
|
int i = 42;
|
||||||
|
std::string s = "Hello";
|
||||||
|
};
|
||||||
|
|
||||||
|
struct Extended {
|
||||||
|
int j = 11;
|
||||||
|
Basic b;
|
||||||
|
};
|
||||||
|
|
||||||
|
std::unordered_map<std::string, OptionTypeInfo> basic_type_map = {
|
||||||
|
{"i", {offsetof(struct Basic, i), OptionType::kInt}},
|
||||||
|
{"s", {offsetof(struct Basic, s), OptionType::kString}},
|
||||||
|
};
|
||||||
|
OptionTypeInfo basic_info = OptionTypeInfo::Struct(
|
||||||
|
"b", &basic_type_map, 0, OptionVerificationType::kNormal,
|
||||||
|
OptionTypeFlags::kMutable, 0);
|
||||||
|
|
||||||
|
std::unordered_map<std::string, OptionTypeInfo> extended_type_map = {
|
||||||
|
{"j", {offsetof(struct Extended, j), OptionType::kInt}},
|
||||||
|
{"b", OptionTypeInfo::Struct(
|
||||||
|
"b", &basic_type_map, offsetof(struct Extended, b),
|
||||||
|
OptionVerificationType::kNormal, OptionTypeFlags::kNone, 0)},
|
||||||
|
{"m", OptionTypeInfo::Struct(
|
||||||
|
"m", &basic_type_map, offsetof(struct Extended, b),
|
||||||
|
OptionVerificationType::kNormal, OptionTypeFlags::kMutable,
|
||||||
|
offsetof(struct Extended, b))},
|
||||||
|
};
|
||||||
|
OptionTypeInfo extended_info = OptionTypeInfo::Struct(
|
||||||
|
"e", &extended_type_map, 0, OptionVerificationType::kNormal,
|
||||||
|
OptionTypeFlags::kMutable, 0);
|
||||||
|
Extended e1, e2;
|
||||||
|
ConfigOptions config_options;
|
||||||
|
std::string mismatch;
|
||||||
|
TestAndCompareOption(config_options, basic_info, "b", "{i=33;s=33}", &e1.b,
|
||||||
|
&e2.b);
|
||||||
|
ASSERT_EQ(e1.b.i, 33);
|
||||||
|
ASSERT_EQ(e1.b.s, "33");
|
||||||
|
|
||||||
|
TestAndCompareOption(config_options, basic_info, "b.i", "44", &e1.b, &e2.b);
|
||||||
|
ASSERT_EQ(e1.b.i, 44);
|
||||||
|
|
||||||
|
TestAndCompareOption(config_options, basic_info, "i", "55", &e1.b, &e2.b);
|
||||||
|
ASSERT_EQ(e1.b.i, 55);
|
||||||
|
|
||||||
|
e1.b.i = 0;
|
||||||
|
auto e1bc = reinterpret_cast<char*>(&e1.b);
|
||||||
|
auto e2bc = reinterpret_cast<char*>(&e2.b);
|
||||||
|
|
||||||
|
ASSERT_FALSE(basic_info.AreEqual(config_options, "b", e1bc, e2bc, &mismatch));
|
||||||
|
ASSERT_EQ(mismatch, "b.i");
|
||||||
|
mismatch.clear();
|
||||||
|
ASSERT_FALSE(
|
||||||
|
basic_info.AreEqual(config_options, "b.i", e1bc, e2bc, &mismatch));
|
||||||
|
ASSERT_EQ(mismatch, "b.i");
|
||||||
|
mismatch.clear();
|
||||||
|
ASSERT_FALSE(basic_info.AreEqual(config_options, "i", e1bc, e2bc, &mismatch));
|
||||||
|
ASSERT_EQ(mismatch, "b.i");
|
||||||
|
mismatch.clear();
|
||||||
|
|
||||||
|
e1 = e2;
|
||||||
|
ASSERT_NOK(basic_info.Parse(config_options, "b", "{i=33;s=33;j=44}", e1bc));
|
||||||
|
ASSERT_TRUE(
|
||||||
|
basic_info.AreEqual(config_options, "b.i", e1bc, e2bc, &mismatch));
|
||||||
|
ASSERT_NOK(basic_info.Parse(config_options, "b.j", "44", e1bc));
|
||||||
|
ASSERT_TRUE(
|
||||||
|
basic_info.AreEqual(config_options, "b.i", e1bc, e2bc, &mismatch));
|
||||||
|
ASSERT_NOK(basic_info.Parse(config_options, "j", "44", e1bc));
|
||||||
|
ASSERT_TRUE(
|
||||||
|
basic_info.AreEqual(config_options, "b.i", e1bc, e2bc, &mismatch));
|
||||||
|
|
||||||
|
TestAndCompareOption(config_options, extended_info, "e",
|
||||||
|
"b={i=55;s=55}; j=22;", &e1, &e2);
|
||||||
|
ASSERT_EQ(e1.b.i, 55);
|
||||||
|
ASSERT_EQ(e1.j, 22);
|
||||||
|
ASSERT_EQ(e1.b.s, "55");
|
||||||
|
TestAndCompareOption(config_options, extended_info, "e.b", "{i=66;s=66;}",
|
||||||
|
&e1, &e2);
|
||||||
|
ASSERT_EQ(e1.b.i, 66);
|
||||||
|
ASSERT_EQ(e1.j, 22);
|
||||||
|
ASSERT_EQ(e1.b.s, "66");
|
||||||
|
TestAndCompareOption(config_options, extended_info, "e.b.i", "77", &e1, &e2);
|
||||||
|
ASSERT_EQ(e1.b.i, 77);
|
||||||
|
ASSERT_EQ(e1.j, 22);
|
||||||
|
ASSERT_EQ(e1.b.s, "66");
|
||||||
|
}
|
||||||
#endif // !ROCKSDB_LITE
|
#endif // !ROCKSDB_LITE
|
||||||
} // namespace ROCKSDB_NAMESPACE
|
} // namespace ROCKSDB_NAMESPACE
|
||||||
|
|
||||||
|
@ -36,8 +36,6 @@ enum class OptionType {
|
|||||||
kComparator,
|
kComparator,
|
||||||
kCompactionFilter,
|
kCompactionFilter,
|
||||||
kCompactionFilterFactory,
|
kCompactionFilterFactory,
|
||||||
kCompactionOptionsFIFO,
|
|
||||||
kCompactionOptionsUniversal,
|
|
||||||
kCompactionStopStyle,
|
kCompactionStopStyle,
|
||||||
kMergeOperator,
|
kMergeOperator,
|
||||||
kMemTableRepFactory,
|
kMemTableRepFactory,
|
||||||
@ -45,9 +43,9 @@ enum class OptionType {
|
|||||||
kFlushBlockPolicyFactory,
|
kFlushBlockPolicyFactory,
|
||||||
kChecksumType,
|
kChecksumType,
|
||||||
kEncodingType,
|
kEncodingType,
|
||||||
kLRUCacheOptions,
|
|
||||||
kEnv,
|
kEnv,
|
||||||
kEnum,
|
kEnum,
|
||||||
|
kStruct,
|
||||||
kUnknown,
|
kUnknown,
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -133,7 +131,7 @@ bool SerializeEnum(const std::unordered_map<std::string, T>& type_map,
|
|||||||
// @param name The name of the options being parsed
|
// @param name The name of the options being parsed
|
||||||
// @param value The string representation of the option
|
// @param value The string representation of the option
|
||||||
// @param addr Pointer to the object
|
// @param addr Pointer to the object
|
||||||
using ParserFunc = std::function<Status(
|
using ParseFunc = std::function<Status(
|
||||||
const ConfigOptions& /*opts*/, const std::string& /*name*/,
|
const ConfigOptions& /*opts*/, const std::string& /*name*/,
|
||||||
const std::string& /*value*/, char* /*addr*/)>;
|
const std::string& /*value*/, char* /*addr*/)>;
|
||||||
|
|
||||||
@ -144,7 +142,7 @@ using ParserFunc = std::function<Status(
|
|||||||
// @param name The name of the options being serialized
|
// @param name The name of the options being serialized
|
||||||
// @param addr Pointer to the value being serialized
|
// @param addr Pointer to the value being serialized
|
||||||
// @param value The result of the serialization.
|
// @param value The result of the serialization.
|
||||||
using StringFunc = std::function<Status(
|
using SerializeFunc = std::function<Status(
|
||||||
const ConfigOptions& /*opts*/, const std::string& /*name*/,
|
const ConfigOptions& /*opts*/, const std::string& /*name*/,
|
||||||
const char* /*addr*/, std::string* /*value*/)>;
|
const char* /*addr*/, std::string* /*value*/)>;
|
||||||
|
|
||||||
@ -164,67 +162,68 @@ using EqualsFunc = std::function<bool(
|
|||||||
// option type, and offset.
|
// option type, and offset.
|
||||||
class OptionTypeInfo {
|
class OptionTypeInfo {
|
||||||
public:
|
public:
|
||||||
int offset;
|
int offset_;
|
||||||
int mutable_offset;
|
int mutable_offset_;
|
||||||
|
|
||||||
// A simple "normal", non-mutable Type "_type" at _offset
|
// A simple "normal", non-mutable Type "type" at offset
|
||||||
OptionTypeInfo(int _offset, OptionType _type)
|
OptionTypeInfo(int offset, OptionType type)
|
||||||
: offset(_offset),
|
: offset_(offset),
|
||||||
mutable_offset(0),
|
mutable_offset_(0),
|
||||||
parser_func(nullptr),
|
parse_func_(nullptr),
|
||||||
string_func(nullptr),
|
serialize_func_(nullptr),
|
||||||
equals_func(nullptr),
|
equals_func_(nullptr),
|
||||||
type(_type),
|
type_(type),
|
||||||
verification(OptionVerificationType::kNormal),
|
verification_(OptionVerificationType::kNormal),
|
||||||
flags(OptionTypeFlags::kNone) {}
|
flags_(OptionTypeFlags::kNone) {}
|
||||||
|
|
||||||
// A simple "normal", mutable Type "_type" at _offset
|
// A simple "normal", mutable Type "type" at offset
|
||||||
OptionTypeInfo(int _offset, OptionType _type, int _mutable_offset)
|
OptionTypeInfo(int offset, OptionType type, int mutable_offset)
|
||||||
: offset(_offset),
|
: offset_(offset),
|
||||||
mutable_offset(_mutable_offset),
|
mutable_offset_(mutable_offset),
|
||||||
parser_func(nullptr),
|
parse_func_(nullptr),
|
||||||
string_func(nullptr),
|
serialize_func_(nullptr),
|
||||||
equals_func(nullptr),
|
equals_func_(nullptr),
|
||||||
type(_type),
|
type_(type),
|
||||||
verification(OptionVerificationType::kNormal),
|
verification_(OptionVerificationType::kNormal),
|
||||||
flags(OptionTypeFlags::kMutable) {}
|
flags_(OptionTypeFlags::kMutable) {}
|
||||||
|
|
||||||
OptionTypeInfo(int _offset, OptionType _type,
|
OptionTypeInfo(int offset, OptionType type,
|
||||||
OptionVerificationType _verification, OptionTypeFlags _flags,
|
OptionVerificationType verification, OptionTypeFlags flags,
|
||||||
int _mutable_offset)
|
int mutable_offset)
|
||||||
: offset(_offset),
|
: offset_(offset),
|
||||||
mutable_offset(_mutable_offset),
|
mutable_offset_(mutable_offset),
|
||||||
parser_func(nullptr),
|
parse_func_(nullptr),
|
||||||
string_func(nullptr),
|
serialize_func_(nullptr),
|
||||||
equals_func(nullptr),
|
equals_func_(nullptr),
|
||||||
type(_type),
|
type_(type),
|
||||||
verification(_verification),
|
verification_(verification),
|
||||||
flags(_flags) {}
|
flags_(flags) {}
|
||||||
|
|
||||||
OptionTypeInfo(int _offset, OptionType _type,
|
OptionTypeInfo(int offset, OptionType type,
|
||||||
OptionVerificationType _verification, OptionTypeFlags _flags,
|
OptionVerificationType verification, OptionTypeFlags flags,
|
||||||
int _mutable_offset, const ParserFunc& _pfunc)
|
int mutable_offset, const ParseFunc& parse_func)
|
||||||
: offset(_offset),
|
: offset_(offset),
|
||||||
mutable_offset(_mutable_offset),
|
mutable_offset_(mutable_offset),
|
||||||
parser_func(_pfunc),
|
parse_func_(parse_func),
|
||||||
string_func(nullptr),
|
serialize_func_(nullptr),
|
||||||
equals_func(nullptr),
|
equals_func_(nullptr),
|
||||||
type(_type),
|
type_(type),
|
||||||
verification(_verification),
|
verification_(verification),
|
||||||
flags(_flags) {}
|
flags_(flags) {}
|
||||||
|
|
||||||
OptionTypeInfo(int _offset, OptionType _type,
|
OptionTypeInfo(int offset, OptionType type,
|
||||||
OptionVerificationType _verification, OptionTypeFlags _flags,
|
OptionVerificationType verification, OptionTypeFlags flags,
|
||||||
int _mutable_offset, const ParserFunc& _pfunc,
|
int mutable_offset, const ParseFunc& parse_func,
|
||||||
const StringFunc& _sfunc, const EqualsFunc& _efunc)
|
const SerializeFunc& serialize_func,
|
||||||
: offset(_offset),
|
const EqualsFunc& equals_func)
|
||||||
mutable_offset(_mutable_offset),
|
: offset_(offset),
|
||||||
parser_func(_pfunc),
|
mutable_offset_(mutable_offset),
|
||||||
string_func(_sfunc),
|
parse_func_(parse_func),
|
||||||
equals_func(_efunc),
|
serialize_func_(serialize_func),
|
||||||
type(_type),
|
equals_func_(equals_func),
|
||||||
verification(_verification),
|
type_(type),
|
||||||
flags(_flags) {}
|
verification_(verification),
|
||||||
|
flags_(flags) {}
|
||||||
|
|
||||||
// Creates an OptionTypeInfo for an enum type. Enums use an additional
|
// Creates an OptionTypeInfo for an enum type. Enums use an additional
|
||||||
// map to convert the enums to/from their string representation.
|
// map to convert the enums to/from their string representation.
|
||||||
@ -235,13 +234,13 @@ class OptionTypeInfo {
|
|||||||
// other changes -- the returned object handles parsing, serialiation, and
|
// other changes -- the returned object handles parsing, serialiation, and
|
||||||
// comparisons.
|
// comparisons.
|
||||||
//
|
//
|
||||||
// @param _offset The offset in the option object for this enum
|
// @param offset The offset in the option object for this enum
|
||||||
// @param map The string to enum mapping for this enum
|
// @param map The string to enum mapping for this enum
|
||||||
template <typename T>
|
template <typename T>
|
||||||
static OptionTypeInfo Enum(
|
static OptionTypeInfo Enum(
|
||||||
int _offset, const std::unordered_map<std::string, T>* const map) {
|
int offset, const std::unordered_map<std::string, T>* const map) {
|
||||||
return OptionTypeInfo(
|
return OptionTypeInfo(
|
||||||
_offset, OptionType::kEnum, OptionVerificationType::kNormal,
|
offset, OptionType::kEnum, OptionVerificationType::kNormal,
|
||||||
OptionTypeFlags::kNone, 0,
|
OptionTypeFlags::kNone, 0,
|
||||||
// Uses the map argument to convert the input string into
|
// Uses the map argument to convert the input string into
|
||||||
// its corresponding enum value. If value is found in the map,
|
// its corresponding enum value. If value is found in the map,
|
||||||
@ -283,7 +282,79 @@ class OptionTypeInfo {
|
|||||||
});
|
});
|
||||||
} // End OptionTypeInfo::Enum
|
} // End OptionTypeInfo::Enum
|
||||||
|
|
||||||
bool IsEnabled(OptionTypeFlags otf) const { return (flags & otf) == otf; }
|
// Creates an OptionTypeInfo for a Struct type. Structs have a
|
||||||
|
// map of string-OptionTypeInfo associated with them that describes how
|
||||||
|
// to process the object for parsing, serializing, and matching.
|
||||||
|
// Structs also have a struct_name, which is the name of the object
|
||||||
|
// as registered in the parent map.
|
||||||
|
// When processing a struct, the option name can be specified as:
|
||||||
|
// - <struct_name> Meaning to process the entire struct.
|
||||||
|
// - <struct_name.field> Meaning to process the single field
|
||||||
|
// - <field> Process the single fields
|
||||||
|
// The CompactionOptionsFIFO, CompactionOptionsUniversal, and LRUCacheOptions
|
||||||
|
// are all examples of Struct options.
|
||||||
|
//
|
||||||
|
// To create an OptionTypeInfo that is a Struct, one should:
|
||||||
|
// - Create a static map of string-OptionTypeInfo corresponding to the
|
||||||
|
// properties of the object that can be set via the options.
|
||||||
|
// - Call this method passing the name and map in as parameters.
|
||||||
|
// Note that it is not necessary to add a new OptionType or make any
|
||||||
|
// other changes -- the returned object handles parsing, serialization, and
|
||||||
|
// comparisons.
|
||||||
|
//
|
||||||
|
// @param offset The offset in the option object for this enum
|
||||||
|
// @param map The string to enum mapping for this enum
|
||||||
|
static OptionTypeInfo Struct(
|
||||||
|
const std::string& struct_name,
|
||||||
|
const std::unordered_map<std::string, OptionTypeInfo>* struct_map,
|
||||||
|
int offset, OptionVerificationType verification, OptionTypeFlags flags,
|
||||||
|
int mutable_offset) {
|
||||||
|
return OptionTypeInfo(
|
||||||
|
offset, OptionType::kStruct, verification, flags, mutable_offset,
|
||||||
|
// Parses the struct and updates the fields at addr
|
||||||
|
[struct_name, struct_map](const ConfigOptions& opts,
|
||||||
|
const std::string& name,
|
||||||
|
const std::string& value, char* addr) {
|
||||||
|
return ParseStruct(opts, struct_name, struct_map, name, value, addr);
|
||||||
|
},
|
||||||
|
// Serializes the struct options into value
|
||||||
|
[struct_name, struct_map](const ConfigOptions& opts,
|
||||||
|
const std::string& name, const char* addr,
|
||||||
|
std::string* value) {
|
||||||
|
return SerializeStruct(opts, struct_name, struct_map, name, addr,
|
||||||
|
value);
|
||||||
|
},
|
||||||
|
// Compares the struct fields of addr1 and addr2 for equality
|
||||||
|
[struct_name, struct_map](const ConfigOptions& opts,
|
||||||
|
const std::string& name, const char* addr1,
|
||||||
|
const char* addr2, std::string* mismatch) {
|
||||||
|
return StructsAreEqual(opts, struct_name, struct_map, name, addr1,
|
||||||
|
addr2, mismatch);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
static OptionTypeInfo Struct(
|
||||||
|
const std::string& struct_name,
|
||||||
|
const std::unordered_map<std::string, OptionTypeInfo>* struct_map,
|
||||||
|
int offset, OptionVerificationType verification, OptionTypeFlags flags,
|
||||||
|
int mutable_offset, const ParseFunc& parse_func) {
|
||||||
|
return OptionTypeInfo(
|
||||||
|
offset, OptionType::kStruct, verification, flags, mutable_offset,
|
||||||
|
parse_func,
|
||||||
|
[struct_name, struct_map](const ConfigOptions& opts,
|
||||||
|
const std::string& name, const char* addr,
|
||||||
|
std::string* value) {
|
||||||
|
return SerializeStruct(opts, struct_name, struct_map, name, addr,
|
||||||
|
value);
|
||||||
|
},
|
||||||
|
[struct_name, struct_map](const ConfigOptions& opts,
|
||||||
|
const std::string& name, const char* addr1,
|
||||||
|
const char* addr2, std::string* mismatch) {
|
||||||
|
return StructsAreEqual(opts, struct_name, struct_map, name, addr1,
|
||||||
|
addr2, mismatch);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
bool IsEnabled(OptionTypeFlags otf) const { return (flags_ & otf) == otf; }
|
||||||
|
|
||||||
bool IsMutable() const { return IsEnabled(OptionTypeFlags::kMutable); }
|
bool IsMutable() const { return IsEnabled(OptionTypeFlags::kMutable); }
|
||||||
|
|
||||||
@ -297,7 +368,7 @@ class OptionTypeInfo {
|
|||||||
bool IsAlias() const { return IsEnabled(OptionVerificationType::kAlias); }
|
bool IsAlias() const { return IsEnabled(OptionVerificationType::kAlias); }
|
||||||
|
|
||||||
bool IsEnabled(OptionVerificationType ovf) const {
|
bool IsEnabled(OptionVerificationType ovf) const {
|
||||||
return verification == ovf;
|
return verification_ == ovf;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Returns the sanity level for comparing the option.
|
// Returns the sanity level for comparing the option.
|
||||||
@ -308,7 +379,7 @@ class OptionTypeInfo {
|
|||||||
if (IsDeprecated() || IsAlias()) {
|
if (IsDeprecated() || IsAlias()) {
|
||||||
return ConfigOptions::SanityLevel::kSanityLevelNone;
|
return ConfigOptions::SanityLevel::kSanityLevelNone;
|
||||||
} else {
|
} else {
|
||||||
auto match = (flags & OptionTypeFlags::kCompareExact);
|
auto match = (flags_ & OptionTypeFlags::kCompareExact);
|
||||||
if (match == OptionTypeFlags::kCompareDefault) {
|
if (match == OptionTypeFlags::kCompareDefault) {
|
||||||
return ConfigOptions::SanityLevel::kSanityLevelExactMatch;
|
return ConfigOptions::SanityLevel::kSanityLevelExactMatch;
|
||||||
} else {
|
} else {
|
||||||
@ -331,54 +402,99 @@ class OptionTypeInfo {
|
|||||||
}
|
}
|
||||||
|
|
||||||
bool IsByName() const {
|
bool IsByName() const {
|
||||||
return (verification == OptionVerificationType::kByName ||
|
return (verification_ == OptionVerificationType::kByName ||
|
||||||
verification == OptionVerificationType::kByNameAllowNull ||
|
verification_ == OptionVerificationType::kByNameAllowNull ||
|
||||||
verification == OptionVerificationType::kByNameAllowFromNull);
|
verification_ == OptionVerificationType::kByNameAllowFromNull);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool IsStruct() const { return (type_ == OptionType::kStruct); }
|
||||||
|
|
||||||
// Parses the option in "opt_value" according to the rules of this class
|
// Parses the option in "opt_value" according to the rules of this class
|
||||||
// and updates the value at "opt_addr".
|
// and updates the value at "opt_addr".
|
||||||
// On success, Status::OK() is returned. On failure:
|
// On success, Status::OK() is returned. On failure:
|
||||||
// NotFound means the opt_name is not valid for this option
|
// NotFound means the opt_name is not valid for this option
|
||||||
// NotSupported means we do not know how to parse the value for this option
|
// NotSupported means we do not know how to parse the value for this option
|
||||||
// InvalidArgument means the opt_value is not valid for this option.
|
// InvalidArgument means the opt_value is not valid for this option.
|
||||||
Status ParseOption(const ConfigOptions& config_options,
|
Status Parse(const ConfigOptions& config_options, const std::string& opt_name,
|
||||||
const std::string& opt_name, const std::string& opt_value,
|
const std::string& opt_value, char* opt_addr) const;
|
||||||
char* opt_addr) const;
|
|
||||||
|
|
||||||
// Serializes the option in "opt_addr" according to the rules of this class
|
// Serializes the option in "opt_addr" according to the rules of this class
|
||||||
// into the value at "opt_value".
|
// into the value at "opt_value".
|
||||||
Status SerializeOption(const ConfigOptions& config_options,
|
Status Serialize(const ConfigOptions& config_options,
|
||||||
const std::string& opt_name, const char* opt_addr,
|
const std::string& opt_name, const char* opt_addr,
|
||||||
std::string* opt_value) const;
|
std::string* opt_value) const;
|
||||||
|
|
||||||
// Compares the "addr1" and "addr2" values according to the rules of this
|
// Compares the "addr1" and "addr2" values according to the rules of this
|
||||||
// class and returns true if they match. On a failed match, mismatch is the
|
// class and returns true if they match. On a failed match, mismatch is the
|
||||||
// name of the option that failed to match.
|
// name of the option that failed to match.
|
||||||
bool MatchesOption(const ConfigOptions& config_options,
|
bool AreEqual(const ConfigOptions& config_options,
|
||||||
const std::string& opt_name, const char* addr1,
|
const std::string& opt_name, const char* addr1,
|
||||||
const char* addr2, std::string* mismatch) const;
|
const char* addr2, std::string* mismatch) const;
|
||||||
|
|
||||||
// Used to override the match rules for "ByName" options.
|
// Used to override the match rules for "ByName" options.
|
||||||
bool MatchesByName(const ConfigOptions& config_options,
|
bool AreEqualByName(const ConfigOptions& config_options,
|
||||||
const std::string& opt_name, const char* this_offset,
|
const std::string& opt_name, const char* this_offset,
|
||||||
const char* that_offset) const;
|
const char* that_offset) const;
|
||||||
bool MatchesByName(const ConfigOptions& config_options,
|
bool AreEqualByName(const ConfigOptions& config_options,
|
||||||
const std::string& opt_name, const char* this_ptr,
|
const std::string& opt_name, const char* this_ptr,
|
||||||
const std::string& that_value) const;
|
const std::string& that_value) const;
|
||||||
|
|
||||||
|
// Parses the input value according to the map for the struct at opt_addr
|
||||||
|
// struct_name is the name of the struct option as registered
|
||||||
|
// opt_name is the name of the option being evaluated. This may
|
||||||
|
// be the whole struct or a sub-element of it, based on struct_name and
|
||||||
|
// opt_name.
|
||||||
|
static Status ParseStruct(
|
||||||
|
const ConfigOptions& config_options, const std::string& struct_name,
|
||||||
|
const std::unordered_map<std::string, OptionTypeInfo>* map,
|
||||||
|
const std::string& opt_name, const std::string& value, char* opt_addr);
|
||||||
|
|
||||||
|
// Serializes the input addr according to the map for the struct to value.
|
||||||
|
// struct_name is the name of the struct option as registered
|
||||||
|
// opt_name is the name of the option being evaluated. This may
|
||||||
|
// be the whole struct or a sub-element of it
|
||||||
|
static Status SerializeStruct(
|
||||||
|
const ConfigOptions& config_options, const std::string& struct_name,
|
||||||
|
const std::unordered_map<std::string, OptionTypeInfo>* map,
|
||||||
|
const std::string& opt_name, const char* opt_addr, std::string* value);
|
||||||
|
|
||||||
|
// Compares the input offsets according to the map for the struct and returns
|
||||||
|
// true if they are equivalent, false otherwise.
|
||||||
|
// struct_name is the name of the struct option as registered
|
||||||
|
// opt_name is the name of the option being evaluated. This may
|
||||||
|
// be the whole struct or a sub-element of it
|
||||||
|
static bool StructsAreEqual(
|
||||||
|
const ConfigOptions& config_options, const std::string& struct_name,
|
||||||
|
const std::unordered_map<std::string, OptionTypeInfo>* map,
|
||||||
|
const std::string& opt_name, const char* this_offset,
|
||||||
|
const char* that_offset, std::string* mismatch);
|
||||||
|
|
||||||
|
// Finds the entry for the opt_name in the opt_map, returning
|
||||||
|
// nullptr if not found.
|
||||||
|
// If found, elem_name will be the name of option to find.
|
||||||
|
// This may be opt_name, or a substring of opt_name.
|
||||||
|
// For "simple" options, opt_name will be equal to elem_name. Given the
|
||||||
|
// opt_name "opt", elem_name will equal "opt".
|
||||||
|
// For "embedded" options (like structs), elem_name may be opt_name
|
||||||
|
// or a field within the opt_name. For example, given the struct "struct",
|
||||||
|
// and opt_name of "struct.field", elem_name will be "field"
|
||||||
|
static const OptionTypeInfo* Find(
|
||||||
|
const std::string& opt_name,
|
||||||
|
const std::unordered_map<std::string, OptionTypeInfo>& opt_map,
|
||||||
|
std::string* elem_name);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
// The optional function to convert a string to its representation
|
// The optional function to convert a string to its representation
|
||||||
ParserFunc parser_func;
|
ParseFunc parse_func_;
|
||||||
|
|
||||||
// The optional function to convert a value to its string representation
|
// The optional function to convert a value to its string representation
|
||||||
StringFunc string_func;
|
SerializeFunc serialize_func_;
|
||||||
|
|
||||||
// The optional function to convert a match to option values
|
// The optional function to match two option values
|
||||||
EqualsFunc equals_func;
|
EqualsFunc equals_func_;
|
||||||
|
|
||||||
OptionType type;
|
OptionType type_;
|
||||||
OptionVerificationType verification;
|
OptionVerificationType verification_;
|
||||||
OptionTypeFlags flags;
|
OptionTypeFlags flags_;
|
||||||
};
|
};
|
||||||
} // namespace ROCKSDB_NAMESPACE
|
} // namespace ROCKSDB_NAMESPACE
|
||||||
|
@ -661,9 +661,9 @@ std::string ParseBlockBasedTableOption(const ConfigOptions& config_options,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
const auto& opt_info = iter->second;
|
const auto& opt_info = iter->second;
|
||||||
Status s = opt_info.ParseOption(
|
Status s =
|
||||||
config_options, iter->first, value,
|
opt_info.Parse(config_options, iter->first, value,
|
||||||
reinterpret_cast<char*>(new_options) + opt_info.offset);
|
reinterpret_cast<char*>(new_options) + opt_info.offset_);
|
||||||
if (s.ok()) {
|
if (s.ok()) {
|
||||||
return "";
|
return "";
|
||||||
} else {
|
} else {
|
||||||
@ -757,13 +757,13 @@ Status VerifyBlockBasedTableFactory(const ConfigOptions& config_options,
|
|||||||
// contain random values since they might not be initialized
|
// contain random values since they might not be initialized
|
||||||
if (config_options.IsCheckEnabled(pair.second.GetSanityLevel())) {
|
if (config_options.IsCheckEnabled(pair.second.GetSanityLevel())) {
|
||||||
const char* base_addr =
|
const char* base_addr =
|
||||||
reinterpret_cast<const char*>(&base_opt) + pair.second.offset;
|
reinterpret_cast<const char*>(&base_opt) + pair.second.offset_;
|
||||||
const char* file_addr =
|
const char* file_addr =
|
||||||
reinterpret_cast<const char*>(&file_opt) + pair.second.offset;
|
reinterpret_cast<const char*>(&file_opt) + pair.second.offset_;
|
||||||
|
|
||||||
if (!pair.second.MatchesOption(config_options, pair.first, base_addr,
|
if (!pair.second.AreEqual(config_options, pair.first, base_addr,
|
||||||
file_addr, &mismatch) &&
|
file_addr, &mismatch) &&
|
||||||
!pair.second.MatchesByName(config_options, pair.first, base_addr,
|
!pair.second.AreEqualByName(config_options, pair.first, base_addr,
|
||||||
file_addr)) {
|
file_addr)) {
|
||||||
return Status::Corruption(
|
return Status::Corruption(
|
||||||
"[RocksDBOptionsParser]: "
|
"[RocksDBOptionsParser]: "
|
||||||
|
@ -220,9 +220,9 @@ std::string ParsePlainTableOptions(const ConfigOptions& config_options,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
const auto& opt_info = iter->second;
|
const auto& opt_info = iter->second;
|
||||||
Status s = opt_info.ParseOption(
|
Status s =
|
||||||
config_options, name, value,
|
opt_info.Parse(config_options, name, value,
|
||||||
reinterpret_cast<char*>(new_options) + opt_info.offset);
|
reinterpret_cast<char*>(new_options) + opt_info.offset_);
|
||||||
if (s.ok()) {
|
if (s.ok()) {
|
||||||
return "";
|
return "";
|
||||||
} else {
|
} else {
|
||||||
|
@ -263,6 +263,20 @@ std::string trim(const std::string& str) {
|
|||||||
return std::string();
|
return std::string();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool EndsWith(const std::string& string, const std::string& pattern) {
|
||||||
|
size_t plen = pattern.size();
|
||||||
|
size_t slen = string.size();
|
||||||
|
if (plen <= slen) {
|
||||||
|
return string.compare(slen - plen, plen, pattern) == 0;
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool StartsWith(const std::string& string, const std::string& pattern) {
|
||||||
|
return string.compare(0, pattern.size(), pattern) == 0;
|
||||||
|
}
|
||||||
|
|
||||||
#ifndef ROCKSDB_LITE
|
#ifndef ROCKSDB_LITE
|
||||||
|
|
||||||
bool ParseBoolean(const std::string& type, const std::string& value) {
|
bool ParseBoolean(const std::string& type, const std::string& value) {
|
||||||
|
@ -111,6 +111,12 @@ std::string UnescapeOptionString(const std::string& escaped_string);
|
|||||||
|
|
||||||
std::string trim(const std::string& str);
|
std::string trim(const std::string& str);
|
||||||
|
|
||||||
|
// Returns true if "string" ends with "pattern"
|
||||||
|
bool EndsWith(const std::string& string, const std::string& pattern);
|
||||||
|
|
||||||
|
// Returns true if "string" starts with "pattern"
|
||||||
|
bool StartsWith(const std::string& string, const std::string& pattern);
|
||||||
|
|
||||||
#ifndef ROCKSDB_LITE
|
#ifndef ROCKSDB_LITE
|
||||||
bool ParseBoolean(const std::string& type, const std::string& value);
|
bool ParseBoolean(const std::string& type, const std::string& value);
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user