polardbxengine/plugin/polarx_rpc/sql_query/query_formatter.h

183 lines
5.6 KiB
C++

/*
* Copyright (c) 2015, 2018, Oracle and/or its affiliates. All rights reserved.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License, version 2.0,
* as published by the Free Software Foundation.
*
* This program is also distributed with certain software (including
* but not limited to OpenSSL) that is licensed under separate terms,
* as designated in a particular file or component or in included license
* documentation. The authors of MySQL hereby grant you an additional
* permission to link the program and your derivative works with the
* separately licensed software that they have included with MySQL.
*
* 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, version 2.0, 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 St, Fifth Floor, Boston, MA 02110-1301 USA
*/
#pragma once
#include <cstdint>
#include <cstdlib>
#include <cstring>
#include <iomanip>
#include <iostream>
#include <limits>
#include <memory>
#include <sstream>
#include <stdexcept>
#include <string>
#include <vector>
#include "m_ctype.h"
#include "../helper/to_string.h"
#include "identifier.h"
#include "raw_binary.h"
#include "raw_string.h"
namespace polarx_rpc {
class Query_formatter {
public:
Query_formatter(std::string &query, CHARSET_INFO &charser);
template <typename Value_type> class No_escape {
public:
explicit No_escape(const Value_type &value) : m_value(value) {}
const Value_type &m_value;
};
Query_formatter &operator%(const char *value);
Query_formatter &operator%(const No_escape<const char *> &value);
Query_formatter &operator%(const std::string &value);
Query_formatter &operator%(const No_escape<std::string> &value);
Query_formatter &operator%(const Identifier &identifier);
Query_formatter &operator%(const RawBinary &raw_binary);
Query_formatter &operator%(const RawString &raw_string);
template <typename Value_type>
Query_formatter &operator%(const Value_type &value) {
return put(value);
}
std::size_t count_tags() const;
private:
template <typename Value_type> Query_formatter &put(const Value_type &value) {
validate_next_tag();
std::string string_value = to_string(value);
put_value(string_value.c_str(), string_value.length());
return *this;
}
Query_formatter &put(const bool value) {
validate_next_tag();
std::string string_value = value ? "true" : "false";
put_value(string_value.c_str(), string_value.length());
return *this;
}
template <typename Value_type>
Query_formatter &put_fp(const Value_type &value) {
std::stringstream stream;
validate_next_tag();
stream << std::setprecision(std::numeric_limits<Value_type>::max_digits10)
<< value;
std::string string_value = stream.str();
put_value(string_value.c_str(), string_value.length());
return *this;
}
void put_value(const char *value, std::size_t length);
void put_value_and_escape(const char *value, std::size_t length);
void validate_next_tag();
void put_ident_and_escape(const char *value, std::size_t length);
std::string &m_query;
CHARSET_INFO &m_charset;
std::size_t m_last_tag_position;
std::size_t m_prev_start_position{0};
/** Fast query string builder buffer. */
std::vector<std::unique_ptr<std::string>> m_bufs;
static constexpr size_t BLOCK_BUF_SIZE = 0x10000; // 64KB
inline std::string &reserve_buf(size_t size) {
if (m_bufs.empty()) {
std::unique_ptr<std::string> block(new std::string);
block->reserve(size > BLOCK_BUF_SIZE ? size : BLOCK_BUF_SIZE);
m_bufs.emplace_back(std::move(block));
return *m_bufs.back();
}
if (size <= BLOCK_BUF_SIZE) {
// Allocate new if now not enough.
auto &now = *m_bufs.back();
if (now.size() + size <= BLOCK_BUF_SIZE)
return now;
// Allocate new one.
std::unique_ptr<std::string> block(new std::string);
block->reserve(BLOCK_BUF_SIZE);
m_bufs.emplace_back(std::move(block));
return *m_bufs.back();
}
// Too big for single block. Just allocate big block.
std::unique_ptr<std::string> block(new std::string);
block->reserve(size);
m_bufs.emplace_back(std::move(block));
return *m_bufs.back();
}
public:
inline void finalize() {
// Concat last one.
if (m_prev_start_position < m_query.size())
reserve_buf(0).append(m_query.data() + m_prev_start_position,
m_query.size() - m_prev_start_position);
// Form full stmt.
size_t sz = 0;
for (const auto &block : m_bufs)
sz += block->size();
m_query.resize(sz);
auto ptr = const_cast<char *>(m_query.data());
for (const auto &block : m_bufs) {
::memcpy(ptr, block->data(), block->size());
ptr += block->size();
}
m_bufs.clear();
}
};
template <>
inline Query_formatter &Query_formatter::operator%
<double>(const double &value) {
return put_fp(value);
}
template <>
inline Query_formatter &Query_formatter::operator%<float>(const float &value) {
return put_fp(value);
}
template <>
inline Query_formatter &Query_formatter::put<bool>(const bool &value) {
validate_next_tag();
const std::string string_value = value ? "true" : "false";
put_value(string_value.c_str(), string_value.length());
return *this;
}
} // namespace polarx_rpc