mirror of
https://github.com/ClickHouse/ClickHouse.git
synced 2024-11-17 13:13:36 +00:00
154 lines
5.0 KiB
C++
154 lines
5.0 KiB
C++
#include "getStructureOfRemoteTable.h"
|
|
#include <Interpreters/Cluster.h>
|
|
#include <Interpreters/Context.h>
|
|
#include <Interpreters/ClusterProxy/executeQuery.h>
|
|
#include <Interpreters/InterpreterDescribeQuery.h>
|
|
#include <QueryPipeline/RemoteQueryExecutor.h>
|
|
#include <DataTypes/DataTypeFactory.h>
|
|
#include <DataTypes/DataTypeString.h>
|
|
#include <Columns/ColumnString.h>
|
|
#include <Storages/IStorage.h>
|
|
#include <Parsers/ExpressionListParsers.h>
|
|
#include <Parsers/parseQuery.h>
|
|
#include <Parsers/ASTFunction.h>
|
|
#include <Common/quoteString.h>
|
|
#include <Common/NetException.h>
|
|
#include <TableFunctions/TableFunctionFactory.h>
|
|
|
|
|
|
namespace DB
|
|
{
|
|
|
|
namespace ErrorCodes
|
|
{
|
|
extern const int NO_REMOTE_SHARD_AVAILABLE;
|
|
}
|
|
|
|
|
|
ColumnsDescription getStructureOfRemoteTableInShard(
|
|
const Cluster & cluster,
|
|
const Cluster::ShardInfo & shard_info,
|
|
const StorageID & table_id,
|
|
ContextPtr context,
|
|
const ASTPtr & table_func_ptr)
|
|
{
|
|
String query;
|
|
|
|
if (table_func_ptr)
|
|
{
|
|
if (shard_info.isLocal())
|
|
{
|
|
TableFunctionPtr table_function_ptr = TableFunctionFactory::instance().get(table_func_ptr, context);
|
|
return table_function_ptr->getActualTableStructure(context);
|
|
}
|
|
|
|
auto table_func_name = queryToString(table_func_ptr);
|
|
query = "DESC TABLE " + table_func_name;
|
|
}
|
|
else
|
|
{
|
|
if (shard_info.isLocal())
|
|
{
|
|
auto storage_ptr = DatabaseCatalog::instance().getTable(table_id, context);
|
|
return storage_ptr->getInMemoryMetadataPtr()->getColumns();
|
|
}
|
|
|
|
/// Request for a table description
|
|
query = "DESC TABLE " + table_id.getFullTableName();
|
|
}
|
|
|
|
ColumnsDescription res;
|
|
|
|
auto new_context = ClusterProxy::updateSettingsForCluster(cluster, context, context->getSettingsRef());
|
|
|
|
/// Expect only needed columns from the result of DESC TABLE. NOTE 'comment' column is ignored for compatibility reasons.
|
|
Block sample_block
|
|
{
|
|
{ ColumnString::create(), std::make_shared<DataTypeString>(), "name" },
|
|
{ ColumnString::create(), std::make_shared<DataTypeString>(), "type" },
|
|
{ ColumnString::create(), std::make_shared<DataTypeString>(), "default_type" },
|
|
{ ColumnString::create(), std::make_shared<DataTypeString>(), "default_expression" },
|
|
};
|
|
|
|
/// Execute remote query without restrictions (because it's not real user query, but part of implementation)
|
|
RemoteQueryExecutor executor(shard_info.pool, query, sample_block, new_context);
|
|
executor.setPoolMode(PoolMode::GET_ONE);
|
|
if (!table_func_ptr)
|
|
executor.setMainTable(table_id);
|
|
|
|
const DataTypeFactory & data_type_factory = DataTypeFactory::instance();
|
|
|
|
ParserExpression expr_parser;
|
|
|
|
while (Block current = executor.read())
|
|
{
|
|
ColumnPtr name = current.getByName("name").column;
|
|
ColumnPtr type = current.getByName("type").column;
|
|
ColumnPtr default_kind = current.getByName("default_type").column;
|
|
ColumnPtr default_expr = current.getByName("default_expression").column;
|
|
size_t size = name->size();
|
|
|
|
for (size_t i = 0; i < size; ++i)
|
|
{
|
|
ColumnDescription column;
|
|
|
|
column.name = (*name)[i].get<const String &>();
|
|
|
|
String data_type_name = (*type)[i].get<const String &>();
|
|
column.type = data_type_factory.get(data_type_name);
|
|
|
|
String kind_name = (*default_kind)[i].get<const String &>();
|
|
if (!kind_name.empty())
|
|
{
|
|
column.default_desc.kind = columnDefaultKindFromString(kind_name);
|
|
String expr_str = (*default_expr)[i].get<const String &>();
|
|
column.default_desc.expression = parseQuery(
|
|
expr_parser, expr_str.data(), expr_str.data() + expr_str.size(), "default expression", 0, context->getSettingsRef().max_parser_depth);
|
|
}
|
|
|
|
res.add(column);
|
|
}
|
|
}
|
|
|
|
executor.finish();
|
|
return res;
|
|
}
|
|
|
|
ColumnsDescription getStructureOfRemoteTable(
|
|
const Cluster & cluster,
|
|
const StorageID & table_id,
|
|
ContextPtr context,
|
|
const ASTPtr & table_func_ptr)
|
|
{
|
|
const auto & shards_info = cluster.getShardsInfo();
|
|
|
|
std::string fail_messages;
|
|
|
|
for (const auto & shard_info : shards_info)
|
|
{
|
|
try
|
|
{
|
|
const auto & res = getStructureOfRemoteTableInShard(cluster, shard_info, table_id, context, table_func_ptr);
|
|
|
|
/// Expect at least some columns.
|
|
/// This is a hack to handle the empty block case returned by Connection when skip_unavailable_shards is set.
|
|
if (res.empty())
|
|
continue;
|
|
|
|
return res;
|
|
}
|
|
catch (const NetException &)
|
|
{
|
|
std::string fail_message = getCurrentExceptionMessage(false);
|
|
fail_messages += fail_message + '\n';
|
|
continue;
|
|
}
|
|
}
|
|
|
|
throw NetException(
|
|
"All attempts to get table structure failed. Log: \n\n" + fail_messages + "\n",
|
|
ErrorCodes::NO_REMOTE_SHARD_AVAILABLE);
|
|
}
|
|
|
|
}
|