2023-03-26 14:36:38 +00:00
|
|
|
#include <Interpreters/InterpreterShowColumnsQuery.h>
|
|
|
|
|
2023-05-05 11:03:47 +00:00
|
|
|
#include <Common/quoteString.h>
|
|
|
|
#include <IO/Operators.h>
|
2023-03-26 14:36:38 +00:00
|
|
|
#include <IO/WriteBufferFromString.h>
|
|
|
|
#include <Parsers/ASTShowColumnsQuery.h>
|
|
|
|
#include <Parsers/formatAST.h>
|
|
|
|
#include <Interpreters/Context.h>
|
|
|
|
#include <Interpreters/executeQuery.h>
|
|
|
|
|
|
|
|
|
|
|
|
namespace DB
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
|
|
InterpreterShowColumnsQuery::InterpreterShowColumnsQuery(const ASTPtr & query_ptr_, ContextMutablePtr context_)
|
|
|
|
: WithMutableContext(context_)
|
|
|
|
, query_ptr(query_ptr_)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
String InterpreterShowColumnsQuery::getRewrittenQuery()
|
|
|
|
{
|
|
|
|
const auto & query = query_ptr->as<ASTShowColumnsQuery &>();
|
|
|
|
|
2023-10-11 15:49:16 +00:00
|
|
|
const bool use_mysql_types = getContext()->getSettingsRef().use_mysql_types_in_show_columns;
|
2023-10-10 21:44:08 +00:00
|
|
|
|
2023-05-05 11:03:47 +00:00
|
|
|
WriteBufferFromOwnString buf_database;
|
|
|
|
String resolved_database = getContext()->resolveDatabase(query.database);
|
|
|
|
writeEscapedString(resolved_database, buf_database);
|
|
|
|
String database = buf_database.str();
|
|
|
|
|
|
|
|
WriteBufferFromOwnString buf_table;
|
|
|
|
writeEscapedString(query.table, buf_table);
|
|
|
|
String table = buf_table.str();
|
2023-03-26 14:36:38 +00:00
|
|
|
|
2023-10-10 21:44:08 +00:00
|
|
|
String rewritten_query;
|
|
|
|
if (use_mysql_types)
|
2023-10-11 12:11:44 +00:00
|
|
|
/// Cheapskate SQL-based mapping from native types to MySQL types, see https://dev.mysql.com/doc/refman/8.0/en/data-types.html
|
2023-10-10 21:44:08 +00:00
|
|
|
/// Known issues:
|
|
|
|
/// - Enums are translated to TEXT
|
|
|
|
rewritten_query += R"(
|
|
|
|
WITH map(
|
|
|
|
'Int8', 'TINYINT',
|
|
|
|
'Int16', 'SMALLINT',
|
|
|
|
'Int32', 'INTEGER',
|
|
|
|
'Int64', 'BIGINT',
|
|
|
|
'UInt8', 'TINYINT UNSIGNED',
|
|
|
|
'UInt16', 'SMALLINT UNSIGNED',
|
|
|
|
'UInt32', 'INTEGER UNSIGNED',
|
|
|
|
'UInt64', 'BIGINT UNSIGNED',
|
|
|
|
'Float32', 'FLOAT',
|
|
|
|
'Float64', 'DOUBLE',
|
|
|
|
'String', 'BLOB',
|
|
|
|
'UUID', 'CHAR',
|
|
|
|
'Bool', 'TINYINT',
|
|
|
|
'Date', 'DATE',
|
|
|
|
'Date32', 'DATE',
|
|
|
|
'DateTime', 'DATETIME',
|
|
|
|
'DateTime64', 'DATETIME',
|
|
|
|
'Map', 'JSON',
|
|
|
|
'Tuple', 'JSON',
|
|
|
|
'Object', 'JSON') AS native_to_mysql_mapping,
|
2023-10-11 12:11:44 +00:00
|
|
|
splitByRegexp('\(|\)', type_) AS split,
|
|
|
|
multiIf(startsWith(type_, 'LowCardinality(Nullable'), split[3],
|
|
|
|
startsWith(type_, 'LowCardinality'), split[2],
|
|
|
|
startsWith(type_, 'Nullable'), split[2],
|
2023-10-10 21:44:08 +00:00
|
|
|
split[1]) AS inner_type,
|
2023-10-11 14:32:42 +00:00
|
|
|
if (length(split) > 1, splitByString(', ', split[2]), []) AS decimal_scale_and_precision,
|
2023-10-10 21:44:08 +00:00
|
|
|
multiIf(inner_type = 'Decimal' AND toInt8(decimal_scale_and_precision[1]) <= 65 AND toInt8(decimal_scale_and_precision[2]) <= 30, concat('DECIMAL(', decimal_scale_and_precision[1], ', ', decimal_scale_and_precision[2], ')'),
|
|
|
|
mapContains(native_to_mysql_mapping, inner_type) = true, native_to_mysql_mapping[inner_type],
|
|
|
|
'TEXT') AS mysql_type
|
|
|
|
)";
|
|
|
|
|
|
|
|
rewritten_query += R"(
|
2023-05-02 12:14:17 +00:00
|
|
|
SELECT
|
2023-10-11 12:11:44 +00:00
|
|
|
name_ AS field,
|
2023-10-10 21:44:08 +00:00
|
|
|
)";
|
|
|
|
|
|
|
|
if (use_mysql_types)
|
|
|
|
rewritten_query += R"(
|
|
|
|
mysql_type AS type,
|
|
|
|
)";
|
|
|
|
else
|
|
|
|
rewritten_query += R"(
|
2023-10-11 12:11:44 +00:00
|
|
|
type_ AS type,
|
2023-10-10 21:44:08 +00:00
|
|
|
)";
|
|
|
|
|
|
|
|
rewritten_query += R"(
|
2023-10-11 12:11:44 +00:00
|
|
|
multiIf(startsWith(type_, 'Nullable('), 'YES', startsWith(type_, 'LowCardinality(Nullable('), 'YES', 'NO') AS `null`,
|
|
|
|
trim(concatWithSeparator(' ', if (is_in_primary_key_, 'PRI', ''), if (is_in_sorting_key_, 'SOR', ''))) AS key,
|
|
|
|
if (default_kind_ IN ('ALIAS', 'DEFAULT', 'MATERIALIZED'), default_expression_, NULL) AS default,
|
2023-05-02 12:14:17 +00:00
|
|
|
'' AS extra )";
|
2023-03-26 14:36:38 +00:00
|
|
|
|
|
|
|
// TODO Interpret query.extended. It is supposed to show internal/virtual columns. Need to fetch virtual column names, see
|
|
|
|
// IStorage::getVirtuals(). We can't easily do that via SQL.
|
|
|
|
|
|
|
|
if (query.full)
|
|
|
|
{
|
|
|
|
/// "Full" mode is mostly for MySQL compat
|
|
|
|
/// - collation: no such thing in ClickHouse
|
|
|
|
/// - comment
|
|
|
|
/// - privileges: <not implemented, TODO ask system.grants>
|
2023-05-02 12:14:17 +00:00
|
|
|
rewritten_query += R"(,
|
|
|
|
NULL AS collation,
|
2023-10-11 12:11:44 +00:00
|
|
|
comment_ AS comment,
|
2023-05-02 12:14:17 +00:00
|
|
|
'' AS privileges )";
|
2023-03-26 14:36:38 +00:00
|
|
|
}
|
|
|
|
|
2023-05-02 12:14:17 +00:00
|
|
|
rewritten_query += fmt::format(R"(
|
2023-10-11 12:11:44 +00:00
|
|
|
-- need to rename columns of the base table to avoid "CYCLIC_ALIASES" errors
|
|
|
|
FROM (SELECT name AS name_,
|
|
|
|
database AS database_,
|
|
|
|
table AS table_,
|
|
|
|
type AS type_,
|
|
|
|
is_in_primary_key AS is_in_primary_key_,
|
|
|
|
is_in_sorting_key AS is_in_sorting_key_,
|
|
|
|
default_kind AS default_kind_,
|
|
|
|
default_expression AS default_expression_,
|
|
|
|
comment AS comment_
|
|
|
|
FROM system.columns)
|
2023-05-05 11:03:47 +00:00
|
|
|
WHERE
|
2023-10-11 12:11:44 +00:00
|
|
|
database_ = '{}'
|
|
|
|
AND table_ = '{}' )", database, table);
|
2023-03-26 14:36:38 +00:00
|
|
|
|
|
|
|
if (!query.like.empty())
|
2023-05-02 12:14:17 +00:00
|
|
|
{
|
2023-10-11 12:11:44 +00:00
|
|
|
rewritten_query += " AND field ";
|
2023-05-02 12:14:17 +00:00
|
|
|
if (query.not_like)
|
|
|
|
rewritten_query += "NOT ";
|
|
|
|
if (query.case_insensitive_like)
|
|
|
|
rewritten_query += "ILIKE ";
|
|
|
|
else
|
|
|
|
rewritten_query += "LIKE ";
|
|
|
|
rewritten_query += fmt::format("'{}'", query.like);
|
|
|
|
}
|
2023-03-26 14:36:38 +00:00
|
|
|
else if (query.where_expression)
|
2023-05-02 12:14:17 +00:00
|
|
|
rewritten_query += fmt::format(" AND ({})", query.where_expression);
|
2023-03-26 14:36:38 +00:00
|
|
|
|
2023-05-02 12:14:17 +00:00
|
|
|
rewritten_query += " ORDER BY field, type, null, key, default, extra";
|
2023-03-26 14:36:38 +00:00
|
|
|
|
|
|
|
if (query.limit_length)
|
2023-05-02 12:14:17 +00:00
|
|
|
rewritten_query += fmt::format(" LIMIT {}", query.limit_length);
|
2023-03-26 14:36:38 +00:00
|
|
|
|
2023-05-02 12:14:17 +00:00
|
|
|
return rewritten_query;
|
2023-03-26 14:36:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
BlockIO InterpreterShowColumnsQuery::execute()
|
|
|
|
{
|
|
|
|
return executeQuery(getRewrittenQuery(), getContext(), true);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
}
|