ClickHouse/dbms/src/Dictionaries/ExternalQueryBuilder.cpp

373 lines
8.5 KiB
C++
Raw Normal View History

2017-06-06 17:18:32 +00:00
#include <ext/range.h>
2016-12-08 02:49:04 +00:00
#include <boost/range/join.hpp>
#include <IO/WriteBuffer.h>
#include <IO/WriteBufferFromString.h>
#include <IO/WriteHelpers.h>
#include <Dictionaries/writeParenthesisedString.h>
#include <Dictionaries/DictionaryStructure.h>
#include <Dictionaries/ExternalQueryBuilder.h>
2016-12-08 02:49:04 +00:00
namespace DB
{
namespace ErrorCodes
{
extern const int UNSUPPORTED_METHOD;
2016-12-08 02:49:04 +00:00
}
ExternalQueryBuilder::ExternalQueryBuilder(
const DictionaryStructure & dict_struct,
const std::string & db,
const std::string & table,
const std::string & where,
IdentifierQuotingStyle quoting_style)
: dict_struct(dict_struct), db(db), table(table), where(where), quoting_style(quoting_style)
2016-12-08 02:49:04 +00:00
{
}
void ExternalQueryBuilder::writeQuoted(const std::string & s, WriteBuffer & out) const
{
switch (quoting_style)
{
case IdentifierQuotingStyle::None:
writeString(s, out);
break;
case IdentifierQuotingStyle::Backticks:
writeBackQuotedString(s, out);
break;
case IdentifierQuotingStyle::DoubleQuotes:
writeDoubleQuotedString(s, out);
break;
}
}
2016-12-08 02:49:04 +00:00
std::string ExternalQueryBuilder::composeLoadAllQuery() const
{
2017-07-31 21:39:24 +00:00
WriteBufferFromOwnString out;
writeString("SELECT ", out);
2017-07-31 21:39:24 +00:00
if (dict_struct.id)
{
if (!dict_struct.id->expression.empty())
2017-07-31 21:39:24 +00:00
{
writeParenthesisedString(dict_struct.id->expression, out);
2017-07-31 21:39:24 +00:00
writeString(" AS ", out);
}
writeQuoted(dict_struct.id->name, out);
2017-07-31 21:39:24 +00:00
if (dict_struct.range_min && dict_struct.range_max)
{
2017-07-31 21:39:24 +00:00
writeString(", ", out);
if (!dict_struct.range_min->expression.empty())
{
writeParenthesisedString(dict_struct.range_min->expression, out);
writeString(" AS ", out);
}
writeQuoted(dict_struct.range_min->name, out);
2017-07-31 21:39:24 +00:00
writeString(", ", out);
if (!dict_struct.range_max->expression.empty())
{
writeParenthesisedString(dict_struct.range_max->expression, out);
2017-07-31 21:39:24 +00:00
writeString(" AS ", out);
}
writeQuoted(dict_struct.range_max->name, out);
2017-07-31 21:39:24 +00:00
}
}
else if (dict_struct.key)
{
auto first = true;
for (const auto & key : *dict_struct.key)
{
2017-07-31 21:39:24 +00:00
if (!first)
writeString(", ", out);
first = false;
2017-07-31 21:39:24 +00:00
if (!key.expression.empty())
{
2017-07-31 21:39:24 +00:00
writeParenthesisedString(key.expression, out);
writeString(" AS ", out);
}
2017-07-31 21:39:24 +00:00
writeQuoted(key.name, out);
}
2017-07-31 21:39:24 +00:00
}
2017-07-31 21:39:24 +00:00
for (const auto & attr : dict_struct.attributes)
{
writeString(", ", out);
2017-07-31 21:39:24 +00:00
if (!attr.expression.empty())
{
2017-07-31 21:39:24 +00:00
writeParenthesisedString(attr.expression, out);
writeString(" AS ", out);
}
2017-07-31 21:39:24 +00:00
writeQuoted(attr.name, out);
}
2017-07-31 21:39:24 +00:00
writeString(" FROM ", out);
if (!db.empty())
{
writeQuoted(db, out);
writeChar('.', out);
}
writeQuoted(table, out);
if (!where.empty())
{
writeString(" WHERE ", out);
writeString(where, out);
}
writeChar(';', out);
return out.str();
2016-12-08 02:49:04 +00:00
}
std::string ExternalQueryBuilder::composeUpdateQuery(const std::string & update_field, const std::string & time_point) const
{
std::string out = composeLoadAllQuery();
std::string update_query;
if (!where.empty())
update_query = " AND " + update_field + " >= '" + time_point + "'";
else
update_query = " WHERE " + update_field + " >= '" + time_point + "'";
return out.insert(out.size()-1, update_query); ///This is done to insert "update_query" before "out"'s semicolon
}
2016-12-08 02:49:04 +00:00
std::string ExternalQueryBuilder::composeLoadIdsQuery(const std::vector<UInt64> & ids)
{
if (!dict_struct.id)
throw Exception{"Simple key required for method", ErrorCodes::UNSUPPORTED_METHOD};
2016-12-08 02:49:04 +00:00
2017-07-31 21:39:24 +00:00
WriteBufferFromOwnString out;
writeString("SELECT ", out);
2016-12-08 02:49:04 +00:00
if (!dict_struct.id->expression.empty())
{
writeParenthesisedString(dict_struct.id->expression, out);
2017-07-31 21:39:24 +00:00
writeString(" AS ", out);
}
2016-12-08 02:49:04 +00:00
writeQuoted(dict_struct.id->name, out);
2016-12-08 02:49:04 +00:00
2017-07-31 21:39:24 +00:00
for (const auto & attr : dict_struct.attributes)
{
writeString(", ", out);
2016-12-08 02:49:04 +00:00
2017-07-31 21:39:24 +00:00
if (!attr.expression.empty())
{
2017-07-31 21:39:24 +00:00
writeParenthesisedString(attr.expression, out);
writeString(" AS ", out);
}
2016-12-08 02:49:04 +00:00
2017-07-31 21:39:24 +00:00
writeQuoted(attr.name, out);
}
2016-12-08 02:49:04 +00:00
2017-07-31 21:39:24 +00:00
writeString(" FROM ", out);
if (!db.empty())
{
writeQuoted(db, out);
writeChar('.', out);
}
writeQuoted(table, out);
2016-12-08 02:49:04 +00:00
2017-07-31 21:39:24 +00:00
writeString(" WHERE ", out);
2016-12-08 02:49:04 +00:00
2017-07-31 21:39:24 +00:00
if (!where.empty())
{
writeString(where, out);
writeString(" AND ", out);
}
2016-12-08 02:49:04 +00:00
writeQuoted(dict_struct.id->name, out);
2017-07-31 21:39:24 +00:00
writeString(" IN (", out);
2016-12-08 02:49:04 +00:00
2017-07-31 21:39:24 +00:00
auto first = true;
for (const auto id : ids)
{
if (!first)
writeString(", ", out);
2016-12-08 02:49:04 +00:00
2017-07-31 21:39:24 +00:00
first = false;
writeString(DB::toString(id), out);
}
2016-12-08 02:49:04 +00:00
2017-07-31 21:39:24 +00:00
writeString(");", out);
return out.str();
2016-12-08 02:49:04 +00:00
}
std::string ExternalQueryBuilder::composeLoadKeysQuery(
2017-05-25 19:52:05 +00:00
const Columns & key_columns,
ColumnConst unification (#1011) * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * Fixed error in ColumnArray::replicateGeneric [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150].
2017-07-21 06:35:58 +00:00
const std::vector<size_t> & requested_rows,
LoadKeysMethod method)
2016-12-08 02:49:04 +00:00
{
if (!dict_struct.key)
throw Exception{"Composite key required for method", ErrorCodes::UNSUPPORTED_METHOD};
2017-07-31 21:39:24 +00:00
WriteBufferFromOwnString out;
writeString("SELECT ", out);
2017-07-31 21:39:24 +00:00
auto first = true;
for (const auto & key_or_attribute : boost::join(*dict_struct.key, dict_struct.attributes))
{
2017-07-31 21:39:24 +00:00
if (!first)
writeString(", ", out);
2017-07-31 21:39:24 +00:00
first = false;
2017-07-31 21:39:24 +00:00
if (!key_or_attribute.expression.empty())
{
2017-07-31 21:39:24 +00:00
writeParenthesisedString(key_or_attribute.expression, out);
writeString(" AS ", out);
}
2017-07-31 21:39:24 +00:00
writeQuoted(key_or_attribute.name, out);
}
2017-07-31 21:39:24 +00:00
writeString(" FROM ", out);
if (!db.empty())
{
writeQuoted(db, out);
writeChar('.', out);
}
writeQuoted(table, out);
2017-07-31 21:39:24 +00:00
writeString(" WHERE ", out);
2017-07-31 21:39:24 +00:00
if (!where.empty())
{
writeString("(", out);
writeString(where, out);
writeString(") AND (", out);
}
2017-07-31 21:39:24 +00:00
if (method == AND_OR_CHAIN)
{
first = true;
for (const auto row : requested_rows)
{
if (!first)
writeString(" OR ", out);
2017-07-31 21:39:24 +00:00
first = false;
composeKeyCondition(key_columns, row, out);
}
2017-07-31 21:39:24 +00:00
}
else if (method == IN_WITH_TUPLES)
{
writeString(composeKeyTupleDefinition(), out);
writeString(" IN (", out);
2017-07-31 21:39:24 +00:00
first = true;
for (const auto row : requested_rows)
{
2017-07-31 21:39:24 +00:00
if (!first)
writeString(", ", out);
first = false;
composeKeyTuple(key_columns, row, out);
}
2017-07-31 21:39:24 +00:00
writeString(")", out);
}
2017-07-31 21:39:24 +00:00
if (!where.empty())
{
writeString(")", out);
}
writeString(";", out);
return out.str();
2016-12-08 02:49:04 +00:00
}
ColumnConst unification (#1011) * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * Fixed error in ColumnArray::replicateGeneric [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150].
2017-07-21 06:35:58 +00:00
void ExternalQueryBuilder::composeKeyCondition(const Columns & key_columns, const size_t row, WriteBuffer & out) const
2016-12-08 02:49:04 +00:00
{
writeString("(", out);
2016-12-08 02:49:04 +00:00
const auto keys_size = key_columns.size();
auto first = true;
for (const auto i : ext::range(0, keys_size))
{
if (!first)
writeString(" AND ", out);
2016-12-08 02:49:04 +00:00
first = false;
2016-12-08 02:49:04 +00:00
const auto & key_description = (*dict_struct.key)[i];
2016-12-08 02:49:04 +00:00
/// key_i=value_i
writeString(key_description.name, out);
writeString("=", out);
key_description.type->serializeTextQuoted(*key_columns[i], row, out, format_settings);
}
2016-12-08 02:49:04 +00:00
writeString(")", out);
2016-12-08 02:49:04 +00:00
}
std::string ExternalQueryBuilder::composeKeyTupleDefinition() const
{
if (!dict_struct.key)
throw Exception{"Composite key required for method", ErrorCodes::UNSUPPORTED_METHOD};
2016-12-08 02:49:04 +00:00
std::string result{"("};
2016-12-08 02:49:04 +00:00
auto first = true;
for (const auto & key : *dict_struct.key)
{
if (!first)
result += ", ";
2016-12-08 02:49:04 +00:00
first = false;
result += key.name;
}
2016-12-08 02:49:04 +00:00
result += ")";
2016-12-08 02:49:04 +00:00
return result;
2016-12-08 02:49:04 +00:00
}
ColumnConst unification (#1011) * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * Fixed error in ColumnArray::replicateGeneric [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150]. * ColumnConst: unification (incomplete) [#CLICKHOUSE-3150].
2017-07-21 06:35:58 +00:00
void ExternalQueryBuilder::composeKeyTuple(const Columns & key_columns, const size_t row, WriteBuffer & out) const
2016-12-08 02:49:04 +00:00
{
writeString("(", out);
2016-12-08 02:49:04 +00:00
const auto keys_size = key_columns.size();
auto first = true;
for (const auto i : ext::range(0, keys_size))
{
if (!first)
writeString(", ", out);
2016-12-08 02:49:04 +00:00
first = false;
(*dict_struct.key)[i].type->serializeTextQuoted(*key_columns[i], row, out, format_settings);
}
2016-12-08 02:49:04 +00:00
writeString(")", out);
2016-12-08 02:49:04 +00:00
}
}