2016-10-06 19:24:09 +02:00
|
|
|
/*
|
|
|
|
Copyright (c) 2014, SkySQL Ab
|
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation; version 2 of the License.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program; if not, write to the Free Software
|
|
|
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
|
|
|
|
|
|
|
|
#ifdef USE_PRAGMA_IMPLEMENTATION
|
2017-02-06 18:39:08 +01:00
|
|
|
#pragma implementation // gcc: Class implementation
|
2016-10-06 19:24:09 +02:00
|
|
|
#endif
|
|
|
|
|
2016-10-09 14:36:58 +02:00
|
|
|
#include <my_config.h>
|
|
|
|
|
2016-10-06 19:24:09 +02:00
|
|
|
/* This C++ files header file */
|
|
|
|
#include "./rdb_cf_options.h"
|
|
|
|
|
|
|
|
/* C++ system header files */
|
|
|
|
#include <string>
|
|
|
|
|
|
|
|
/* MySQL header files */
|
|
|
|
#include "./log.h"
|
|
|
|
|
|
|
|
/* RocksDB header files */
|
|
|
|
#include "rocksdb/utilities/convenience.h"
|
|
|
|
|
|
|
|
/* MyRocks header files */
|
|
|
|
#include "./ha_rocksdb.h"
|
|
|
|
#include "./rdb_cf_manager.h"
|
|
|
|
#include "./rdb_compact_filter.h"
|
|
|
|
|
|
|
|
namespace myrocks {
|
|
|
|
|
|
|
|
Rdb_pk_comparator Rdb_cf_options::s_pk_comparator;
|
|
|
|
Rdb_rev_comparator Rdb_cf_options::s_rev_pk_comparator;
|
|
|
|
|
|
|
|
bool Rdb_cf_options::init(
|
2017-02-06 18:39:08 +01:00
|
|
|
const rocksdb::BlockBasedTableOptions &table_options,
|
|
|
|
std::shared_ptr<rocksdb::TablePropertiesCollectorFactory> prop_coll_factory,
|
|
|
|
const char *const default_cf_options,
|
|
|
|
const char *const override_cf_options) {
|
2016-12-31 21:30:09 +01:00
|
|
|
DBUG_ASSERT(default_cf_options != nullptr);
|
|
|
|
DBUG_ASSERT(override_cf_options != nullptr);
|
|
|
|
|
2016-10-06 19:24:09 +02:00
|
|
|
m_default_cf_opts.comparator = &s_pk_comparator;
|
|
|
|
m_default_cf_opts.compaction_filter_factory.reset(
|
2017-02-06 18:39:08 +01:00
|
|
|
new Rdb_compact_filter_factory);
|
2016-10-06 19:24:09 +02:00
|
|
|
|
|
|
|
m_default_cf_opts.table_factory.reset(
|
2017-02-06 18:39:08 +01:00
|
|
|
rocksdb::NewBlockBasedTableFactory(table_options));
|
2016-10-06 19:24:09 +02:00
|
|
|
|
|
|
|
if (prop_coll_factory) {
|
|
|
|
m_default_cf_opts.table_properties_collector_factories.push_back(
|
2017-02-06 18:39:08 +01:00
|
|
|
prop_coll_factory);
|
2016-10-06 19:24:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!set_default(std::string(default_cf_options)) ||
|
|
|
|
!set_override(std::string(override_cf_options))) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Rdb_cf_options::get(const std::string &cf_name,
|
2017-02-06 18:39:08 +01:00
|
|
|
rocksdb::ColumnFamilyOptions *const opts) {
|
2016-10-06 19:24:09 +02:00
|
|
|
DBUG_ASSERT(opts != nullptr);
|
|
|
|
|
2017-07-28 19:52:07 +02:00
|
|
|
// Get defaults.
|
2017-02-06 18:39:08 +01:00
|
|
|
rocksdb::GetColumnFamilyOptionsFromString(*opts, m_default_config, opts);
|
2016-10-06 19:24:09 +02:00
|
|
|
|
2017-07-28 19:52:07 +02:00
|
|
|
// Get a custom confguration if we have one.
|
2016-10-06 19:24:09 +02:00
|
|
|
Name_to_config_t::iterator it = m_name_map.find(cf_name);
|
2017-07-28 19:52:07 +02:00
|
|
|
|
2016-10-06 19:24:09 +02:00
|
|
|
if (it != m_name_map.end()) {
|
2017-02-06 18:39:08 +01:00
|
|
|
rocksdb::GetColumnFamilyOptionsFromString(*opts, it->second, opts);
|
2016-10-06 19:24:09 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-28 19:52:07 +02:00
|
|
|
void Rdb_cf_options::update(const std::string &cf_name,
|
|
|
|
const std::string &cf_options) {
|
|
|
|
DBUG_ASSERT(!cf_name.empty());
|
|
|
|
DBUG_ASSERT(!cf_options.empty());
|
|
|
|
|
|
|
|
// Always update. If we didn't have an entry before then add it.
|
|
|
|
m_name_map[cf_name] = cf_options;
|
|
|
|
|
|
|
|
DBUG_ASSERT(!m_name_map.empty());
|
|
|
|
}
|
|
|
|
|
2017-02-06 18:39:08 +01:00
|
|
|
bool Rdb_cf_options::set_default(const std::string &default_config) {
|
2016-10-06 19:24:09 +02:00
|
|
|
rocksdb::ColumnFamilyOptions options;
|
|
|
|
|
|
|
|
if (!default_config.empty() &&
|
2017-02-06 18:39:08 +01:00
|
|
|
!rocksdb::GetColumnFamilyOptionsFromString(options, default_config,
|
|
|
|
&options)
|
|
|
|
.ok()) {
|
|
|
|
fprintf(stderr, "Invalid default column family config: %s\n",
|
2016-10-06 19:24:09 +02:00
|
|
|
default_config.c_str());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_default_config = default_config;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Skip over any spaces in the input string.
|
2017-02-06 18:39:08 +01:00
|
|
|
void Rdb_cf_options::skip_spaces(const std::string &input, size_t *const pos) {
|
2016-10-06 19:24:09 +02:00
|
|
|
DBUG_ASSERT(pos != nullptr);
|
|
|
|
|
|
|
|
while (*pos < input.size() && isspace(input[*pos]))
|
|
|
|
++(*pos);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Find a valid column family name. Note that all characters except a
|
|
|
|
// semicolon are valid (should this change?) and all spaces are trimmed from
|
|
|
|
// the beginning and end but are not removed between other characters.
|
2017-02-06 18:39:08 +01:00
|
|
|
bool Rdb_cf_options::find_column_family(const std::string &input,
|
|
|
|
size_t *const pos,
|
|
|
|
std::string *const key) {
|
2016-10-06 19:24:09 +02:00
|
|
|
DBUG_ASSERT(pos != nullptr);
|
|
|
|
DBUG_ASSERT(key != nullptr);
|
|
|
|
|
2016-12-31 21:30:09 +01:00
|
|
|
const size_t beg_pos = *pos;
|
2016-10-06 19:24:09 +02:00
|
|
|
size_t end_pos = *pos - 1;
|
|
|
|
|
|
|
|
// Loop through the characters in the string until we see a '='.
|
2017-02-06 18:39:08 +01:00
|
|
|
for (; *pos < input.size() && input[*pos] != '='; ++(*pos)) {
|
2016-10-06 19:24:09 +02:00
|
|
|
// If this is not a space, move the end position to the current position.
|
|
|
|
if (input[*pos] != ' ')
|
|
|
|
end_pos = *pos;
|
|
|
|
}
|
|
|
|
|
2017-02-06 18:39:08 +01:00
|
|
|
if (end_pos == beg_pos - 1) {
|
2016-10-06 19:24:09 +02:00
|
|
|
// NO_LINT_DEBUG
|
|
|
|
sql_print_warning("No column family found (options: %s)", input.c_str());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
*key = input.substr(beg_pos, end_pos - beg_pos + 1);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Find a valid options portion. Everything is deemed valid within the options
|
|
|
|
// portion until we hit as many close curly braces as we have seen open curly
|
|
|
|
// braces.
|
2017-02-06 18:39:08 +01:00
|
|
|
bool Rdb_cf_options::find_options(const std::string &input, size_t *const pos,
|
|
|
|
std::string *const options) {
|
2016-10-06 19:24:09 +02:00
|
|
|
DBUG_ASSERT(pos != nullptr);
|
|
|
|
DBUG_ASSERT(options != nullptr);
|
|
|
|
|
|
|
|
// Make sure we have an open curly brace at the current position.
|
2017-02-06 18:39:08 +01:00
|
|
|
if (*pos < input.size() && input[*pos] != '{') {
|
2016-10-06 19:24:09 +02:00
|
|
|
// NO_LINT_DEBUG
|
|
|
|
sql_print_warning("Invalid cf options, '{' expected (options: %s)",
|
2017-02-06 18:39:08 +01:00
|
|
|
input.c_str());
|
2016-10-06 19:24:09 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Skip the open curly brace and any spaces.
|
|
|
|
++(*pos);
|
|
|
|
skip_spaces(input, pos);
|
|
|
|
|
|
|
|
// Set up our brace_count, the begin position and current end position.
|
|
|
|
size_t brace_count = 1;
|
2016-12-31 21:30:09 +01:00
|
|
|
const size_t beg_pos = *pos;
|
2016-10-06 19:24:09 +02:00
|
|
|
|
|
|
|
// Loop through the characters in the string until we find the appropriate
|
|
|
|
// number of closing curly braces.
|
2017-02-06 18:39:08 +01:00
|
|
|
while (*pos < input.size()) {
|
|
|
|
switch (input[*pos]) {
|
|
|
|
case '}':
|
|
|
|
// If this is a closing curly brace and we bring the count down to zero
|
|
|
|
// we can exit the loop with a valid options string.
|
|
|
|
if (--brace_count == 0) {
|
|
|
|
*options = input.substr(beg_pos, *pos - beg_pos);
|
|
|
|
++(*pos); // Move past the last closing curly brace
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case '{':
|
|
|
|
// If this is an open curly brace increment the count.
|
|
|
|
++brace_count;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
2016-10-06 19:24:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Move to the next character.
|
|
|
|
++(*pos);
|
|
|
|
}
|
|
|
|
|
|
|
|
// We never found the correct number of closing curly braces.
|
|
|
|
// Generate an error.
|
|
|
|
// NO_LINT_DEBUG
|
|
|
|
sql_print_warning("Mismatched cf options, '}' expected (options: %s)",
|
2017-02-06 18:39:08 +01:00
|
|
|
input.c_str());
|
2016-10-06 19:24:09 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-02-06 18:39:08 +01:00
|
|
|
bool Rdb_cf_options::find_cf_options_pair(const std::string &input,
|
|
|
|
size_t *const pos,
|
|
|
|
std::string *const cf,
|
|
|
|
std::string *const opt_str) {
|
2016-10-06 19:24:09 +02:00
|
|
|
DBUG_ASSERT(pos != nullptr);
|
|
|
|
DBUG_ASSERT(cf != nullptr);
|
|
|
|
DBUG_ASSERT(opt_str != nullptr);
|
|
|
|
|
|
|
|
// Skip any spaces.
|
|
|
|
skip_spaces(input, pos);
|
|
|
|
|
|
|
|
// We should now have a column family name.
|
|
|
|
if (!find_column_family(input, pos, cf))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// If we are at the end of the input then we generate an error.
|
2017-02-06 18:39:08 +01:00
|
|
|
if (*pos == input.size()) {
|
2016-10-06 19:24:09 +02:00
|
|
|
// NO_LINT_DEBUG
|
|
|
|
sql_print_warning("Invalid cf options, '=' expected (options: %s)",
|
2017-02-06 18:39:08 +01:00
|
|
|
input.c_str());
|
2016-10-06 19:24:09 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Skip equal sign and any spaces after it
|
|
|
|
++(*pos);
|
|
|
|
skip_spaces(input, pos);
|
|
|
|
|
|
|
|
// Find the options for this column family. This should be in the format
|
|
|
|
// {<options>} where <options> may contain embedded pairs of curly braces.
|
|
|
|
if (!find_options(input, pos, opt_str))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Skip any trailing spaces after the option string.
|
|
|
|
skip_spaces(input, pos);
|
|
|
|
|
|
|
|
// We should either be at the end of the input string or at a semicolon.
|
2017-02-06 18:39:08 +01:00
|
|
|
if (*pos < input.size()) {
|
|
|
|
if (input[*pos] != ';') {
|
2016-10-06 19:24:09 +02:00
|
|
|
// NO_LINT_DEBUG
|
|
|
|
sql_print_warning("Invalid cf options, ';' expected (options: %s)",
|
2017-02-06 18:39:08 +01:00
|
|
|
input.c_str());
|
2016-10-06 19:24:09 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
++(*pos);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-07-28 19:52:07 +02:00
|
|
|
bool Rdb_cf_options::parse_cf_options(const std::string &cf_options,
|
|
|
|
Name_to_config_t *option_map) {
|
2016-10-06 19:24:09 +02:00
|
|
|
std::string cf;
|
|
|
|
std::string opt_str;
|
|
|
|
rocksdb::ColumnFamilyOptions options;
|
2017-07-28 19:52:07 +02:00
|
|
|
|
|
|
|
DBUG_ASSERT(option_map != nullptr);
|
|
|
|
DBUG_ASSERT(option_map->empty());
|
2016-10-06 19:24:09 +02:00
|
|
|
|
|
|
|
// Loop through the characters of the string until we reach the end.
|
|
|
|
size_t pos = 0;
|
2017-07-28 19:52:07 +02:00
|
|
|
|
|
|
|
while (pos < cf_options.size()) {
|
2016-10-06 19:24:09 +02:00
|
|
|
// Attempt to find <cf>={<opt_str>}.
|
2017-07-28 19:52:07 +02:00
|
|
|
if (!find_cf_options_pair(cf_options, &pos, &cf, &opt_str)) {
|
2016-10-06 19:24:09 +02:00
|
|
|
return false;
|
2017-07-28 19:52:07 +02:00
|
|
|
}
|
2016-10-06 19:24:09 +02:00
|
|
|
|
|
|
|
// Generate an error if we have already seen this column family.
|
2017-07-28 19:52:07 +02:00
|
|
|
if (option_map->find(cf) != option_map->end()) {
|
2016-10-06 19:24:09 +02:00
|
|
|
// NO_LINT_DEBUG
|
|
|
|
sql_print_warning(
|
|
|
|
"Duplicate entry for %s in override options (options: %s)",
|
2017-07-28 19:52:07 +02:00
|
|
|
cf.c_str(), cf_options.c_str());
|
2016-10-06 19:24:09 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Generate an error if the <opt_str> is not valid according to RocksDB.
|
2017-02-06 18:39:08 +01:00
|
|
|
if (!rocksdb::GetColumnFamilyOptionsFromString(options, opt_str, &options)
|
|
|
|
.ok()) {
|
2016-10-06 19:24:09 +02:00
|
|
|
// NO_LINT_DEBUG
|
|
|
|
sql_print_warning(
|
|
|
|
"Invalid cf config for %s in override options (options: %s)",
|
2017-07-28 19:52:07 +02:00
|
|
|
cf.c_str(), cf_options.c_str());
|
2016-10-06 19:24:09 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If everything is good, add this cf/opt_str pair to the map.
|
2017-07-28 19:52:07 +02:00
|
|
|
(*option_map)[cf] = opt_str;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Rdb_cf_options::set_override(const std::string &override_config) {
|
|
|
|
Name_to_config_t configs;
|
|
|
|
|
|
|
|
if (!parse_cf_options(override_config, &configs)) {
|
|
|
|
return false;
|
2016-10-06 19:24:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Everything checked out - make the map live
|
|
|
|
m_name_map = configs;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-02-06 18:39:08 +01:00
|
|
|
const rocksdb::Comparator *
|
|
|
|
Rdb_cf_options::get_cf_comparator(const std::string &cf_name) {
|
|
|
|
if (Rdb_cf_manager::is_cf_name_reverse(cf_name.c_str())) {
|
2016-10-06 19:24:09 +02:00
|
|
|
return &s_rev_pk_comparator;
|
2017-02-06 18:39:08 +01:00
|
|
|
} else {
|
2016-10-06 19:24:09 +02:00
|
|
|
return &s_pk_comparator;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Rdb_cf_options::get_cf_options(const std::string &cf_name,
|
2017-02-06 18:39:08 +01:00
|
|
|
rocksdb::ColumnFamilyOptions *const opts) {
|
2016-10-06 19:24:09 +02:00
|
|
|
DBUG_ASSERT(opts != nullptr);
|
|
|
|
|
|
|
|
*opts = m_default_cf_opts;
|
|
|
|
get(cf_name, opts);
|
|
|
|
|
|
|
|
// Set the comparator according to 'rev:'
|
2017-02-06 18:39:08 +01:00
|
|
|
opts->comparator = get_cf_comparator(cf_name);
|
2016-10-06 19:24:09 +02:00
|
|
|
}
|
|
|
|
|
2017-02-06 18:39:08 +01:00
|
|
|
} // namespace myrocks
|