mirror of
https://github.com/MariaDB/server.git
synced 2025-01-16 12:02:42 +01:00
369 lines
9.3 KiB
C++
369 lines
9.3 KiB
C++
/*
|
|
Copyright (c) 2016, Facebook, Inc.
|
|
|
|
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., 51 Franklin Street, Fifth Floor, Boston, MA 02111-1301 USA */
|
|
|
|
#include <my_global.h>
|
|
|
|
/* This C++ file's header */
|
|
#include "./rdb_utils.h"
|
|
|
|
/* C++ standard header files */
|
|
#include <array>
|
|
#include <sstream>
|
|
#include <string>
|
|
#include <vector>
|
|
|
|
/* C standard header files */
|
|
#include <ctype.h>
|
|
|
|
/* MyRocks header files */
|
|
#include "./ha_rocksdb.h"
|
|
|
|
/*
|
|
Both innobase/include/ut0counter.h and rocksdb/port/port_posix.h define
|
|
CACHE_LINE_SIZE.
|
|
*/
|
|
#ifdef CACHE_LINE_SIZE
|
|
# undef CACHE_LINE_SIZE
|
|
#endif
|
|
|
|
/* RocksDB header files */
|
|
#include "util/compression.h"
|
|
|
|
namespace myrocks {
|
|
|
|
/*
|
|
Skip past any spaces in the input
|
|
*/
|
|
const char *rdb_skip_spaces(const struct charset_info_st *const cs,
|
|
const char *str) {
|
|
while (my_isspace(cs, *str)) {
|
|
str++;
|
|
}
|
|
|
|
return str;
|
|
}
|
|
|
|
/*
|
|
Compare (ignoring case) to see if str2 is the next data in str1.
|
|
Note that str1 can be longer but we only compare up to the number
|
|
of characters in str2.
|
|
*/
|
|
bool rdb_compare_strings_ic(const char *const str1, const char *const str2) {
|
|
// Scan through the strings
|
|
size_t ii;
|
|
for (ii = 0; str2[ii]; ii++) {
|
|
if (toupper(static_cast<int>(str1[ii])) !=
|
|
toupper(static_cast<int>(str2[ii]))) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
Scan through an input string looking for pattern, ignoring case
|
|
and skipping all data enclosed in quotes.
|
|
*/
|
|
const char *rdb_find_in_string(const char *str, const char *pattern,
|
|
bool *const succeeded) {
|
|
char quote = '\0';
|
|
bool escape = false;
|
|
|
|
*succeeded = false;
|
|
|
|
for (; *str; str++) {
|
|
/* If we found a our starting quote character */
|
|
if (*str == quote) {
|
|
/* If it was escaped ignore it */
|
|
if (escape) {
|
|
escape = false;
|
|
}
|
|
/* Otherwise we are now outside of the quoted string */
|
|
else {
|
|
quote = '\0';
|
|
}
|
|
}
|
|
/* Else if we are currently inside a quoted string? */
|
|
else if (quote != '\0') {
|
|
/* If so, check for the escape character */
|
|
escape = !escape && *str == '\\';
|
|
}
|
|
/* Else if we found a quote we are starting a quoted string */
|
|
else if (*str == '"' || *str == '\'' || *str == '`') {
|
|
quote = *str;
|
|
}
|
|
/* Else we are outside of a quoted string - look for our pattern */
|
|
else {
|
|
if (rdb_compare_strings_ic(str, pattern)) {
|
|
*succeeded = true;
|
|
return str;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Return the character after the found pattern or the null terminateor
|
|
// if the pattern wasn't found.
|
|
return str;
|
|
}
|
|
|
|
/*
|
|
See if the next valid token matches the specified string
|
|
*/
|
|
const char *rdb_check_next_token(const struct charset_info_st *const cs,
|
|
const char *str, const char *const pattern,
|
|
bool *const succeeded) {
|
|
// Move past any spaces
|
|
str = rdb_skip_spaces(cs, str);
|
|
|
|
// See if the next characters match the pattern
|
|
if (rdb_compare_strings_ic(str, pattern)) {
|
|
*succeeded = true;
|
|
return str + strlen(pattern);
|
|
}
|
|
|
|
*succeeded = false;
|
|
return str;
|
|
}
|
|
|
|
/*
|
|
Parse id
|
|
*/
|
|
const char *rdb_parse_id(const struct charset_info_st *const cs,
|
|
const char *str, std::string *const id) {
|
|
// Move past any spaces
|
|
str = rdb_skip_spaces(cs, str);
|
|
|
|
if (*str == '\0') {
|
|
return str;
|
|
}
|
|
|
|
char quote = '\0';
|
|
if (*str == '`' || *str == '"') {
|
|
quote = *str++;
|
|
}
|
|
|
|
size_t len = 0;
|
|
const char *start = str;
|
|
|
|
if (quote != '\0') {
|
|
for (;;) {
|
|
if (*str == '\0') {
|
|
return str;
|
|
}
|
|
|
|
if (*str == quote) {
|
|
str++;
|
|
if (*str != quote) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
str++;
|
|
len++;
|
|
}
|
|
} else {
|
|
while (!my_isspace(cs, *str) && *str != '(' && *str != ')' && *str != '.' &&
|
|
*str != ',' && *str != '\0') {
|
|
str++;
|
|
len++;
|
|
}
|
|
}
|
|
|
|
// If the user requested the id create it and return it
|
|
if (id != nullptr) {
|
|
*id = std::string("");
|
|
id->reserve(len);
|
|
while (len--) {
|
|
*id += *start;
|
|
if (*start++ == quote) {
|
|
start++;
|
|
}
|
|
}
|
|
}
|
|
|
|
return str;
|
|
}
|
|
|
|
/*
|
|
Skip id
|
|
*/
|
|
const char *rdb_skip_id(const struct charset_info_st *const cs,
|
|
const char *str) {
|
|
return rdb_parse_id(cs, str, nullptr);
|
|
}
|
|
|
|
/*
|
|
Parses a given string into tokens (if any) separated by a specific delimiter.
|
|
*/
|
|
const std::vector<std::string> parse_into_tokens(const std::string &s,
|
|
const char delim) {
|
|
std::vector<std::string> tokens;
|
|
std::string t;
|
|
std::stringstream ss(s);
|
|
|
|
while (getline(ss, t, delim)) {
|
|
tokens.push_back(t);
|
|
}
|
|
|
|
return tokens;
|
|
}
|
|
|
|
static const std::size_t rdb_hex_bytes_per_char = 2;
|
|
static const std::array<char, 16> rdb_hexdigit = {{'0', '1', '2', '3', '4', '5',
|
|
'6', '7', '8', '9', 'a', 'b',
|
|
'c', 'd', 'e', 'f'}};
|
|
|
|
/*
|
|
Convert data into a hex string with optional maximum length.
|
|
If the data is larger than the maximum length trancate it and append "..".
|
|
*/
|
|
std::string rdb_hexdump(const char *data, const std::size_t data_len,
|
|
const std::size_t maxsize) {
|
|
// Count the elements in the string
|
|
std::size_t elems = data_len;
|
|
// Calculate the amount of output needed
|
|
std::size_t len = elems * rdb_hex_bytes_per_char;
|
|
std::string str;
|
|
|
|
if (maxsize != 0 && len > maxsize) {
|
|
// If the amount of output is too large adjust the settings
|
|
// and leave room for the ".." at the end
|
|
elems = (maxsize - 2) / rdb_hex_bytes_per_char;
|
|
len = elems * rdb_hex_bytes_per_char + 2;
|
|
}
|
|
|
|
// Reserve sufficient space to avoid reallocations
|
|
str.reserve(len);
|
|
|
|
// Loop through the input data and build the output string
|
|
for (std::size_t ii = 0; ii < elems; ii++, data++) {
|
|
uint8_t ch = (uint8_t)*data;
|
|
str += rdb_hexdigit[ch >> 4];
|
|
str += rdb_hexdigit[ch & 0x0F];
|
|
}
|
|
|
|
// If we can't fit it all add the ".."
|
|
if (elems != data_len) {
|
|
str += "..";
|
|
}
|
|
|
|
return str;
|
|
}
|
|
|
|
/*
|
|
Attempt to access the database subdirectory to see if it exists
|
|
*/
|
|
bool rdb_database_exists(const std::string &db_name) {
|
|
const std::string dir =
|
|
std::string(mysql_real_data_home) + FN_DIRSEP + db_name;
|
|
struct st_my_dir *const dir_info =
|
|
my_dir(dir.c_str(), MYF(MY_DONT_SORT | MY_WANT_STAT));
|
|
if (dir_info == nullptr) {
|
|
return false;
|
|
}
|
|
|
|
my_dirend(dir_info);
|
|
return true;
|
|
}
|
|
|
|
void rdb_log_status_error(const rocksdb::Status &s, const char *msg) {
|
|
if (msg == nullptr) {
|
|
// NO_LINT_DEBUG
|
|
sql_print_error("RocksDB: status error, code: %d, error message: %s",
|
|
s.code(), s.ToString().c_str());
|
|
return;
|
|
}
|
|
|
|
// NO_LINT_DEBUG
|
|
sql_print_error("RocksDB: %s, Status Code: %d, Status: %s", msg, s.code(),
|
|
s.ToString().c_str());
|
|
}
|
|
|
|
/*
|
|
@brief
|
|
Return a comma-separated string with compiled-in compression types.
|
|
Not thread-safe.
|
|
*/
|
|
const char *get_rocksdb_supported_compression_types()
|
|
{
|
|
static std::string compression_methods_buf;
|
|
static bool inited=false;
|
|
if (!inited)
|
|
{
|
|
inited= true;
|
|
std::vector<rocksdb::CompressionType> known_types=
|
|
{
|
|
rocksdb::kSnappyCompression,
|
|
rocksdb::kZlibCompression,
|
|
rocksdb::kBZip2Compression,
|
|
rocksdb::kLZ4Compression,
|
|
rocksdb::kLZ4HCCompression,
|
|
rocksdb::kXpressCompression,
|
|
rocksdb::kZSTDNotFinalCompression
|
|
};
|
|
|
|
for (auto typ : known_types)
|
|
{
|
|
if (CompressionTypeSupported(typ))
|
|
{
|
|
if (compression_methods_buf.size())
|
|
compression_methods_buf.append(",");
|
|
compression_methods_buf.append(CompressionTypeToString(typ));
|
|
}
|
|
}
|
|
}
|
|
return compression_methods_buf.c_str();
|
|
}
|
|
|
|
bool rdb_check_rocksdb_corruption() {
|
|
return !my_access(myrocks::rdb_corruption_marker_file_name().c_str(), F_OK);
|
|
}
|
|
|
|
void rdb_persist_corruption_marker() {
|
|
const std::string &fileName(myrocks::rdb_corruption_marker_file_name());
|
|
/* O_SYNC is not supported on windows */
|
|
int fd = my_open(fileName.c_str(), O_CREAT | IF_WIN(0, O_SYNC), MYF(MY_WME));
|
|
if (fd < 0) {
|
|
// NO_LINT_DEBUG
|
|
sql_print_error(
|
|
"RocksDB: Can't create file %s to mark rocksdb as "
|
|
"corrupted.",
|
|
fileName.c_str());
|
|
} else {
|
|
// NO_LINT_DEBUG
|
|
sql_print_information(
|
|
"RocksDB: Creating the file %s to abort mysqld "
|
|
"restarts. Remove this file from the data directory "
|
|
"after fixing the corruption to recover. ",
|
|
fileName.c_str());
|
|
}
|
|
|
|
#ifdef _WIN32
|
|
/* A replacement for O_SYNC flag above */
|
|
if (fd >= 0)
|
|
my_sync(fd, MYF(0));
|
|
#endif
|
|
|
|
int ret = my_close(fd, MYF(MY_WME));
|
|
if (ret) {
|
|
// NO_LINT_DEBUG
|
|
sql_print_error("RocksDB: Error (%d) closing the file %s", ret,
|
|
fileName.c_str());
|
|
}
|
|
}
|
|
|
|
} // namespace myrocks
|