ClickHouse/src/Storages/StorageSnapshot.cpp

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

245 lines
8.3 KiB
C++
Raw Normal View History

#include <Storages/StorageSnapshot.h>
2022-07-21 19:50:19 +00:00
#include <Storages/LightweightDeleteDescription.h>
#include <Storages/IStorage.h>
#include <DataTypes/ObjectUtils.h>
2021-07-21 14:45:19 +00:00
#include <DataTypes/NestedUtils.h>
#include <sparsehash/dense_hash_set>
namespace DB
{
namespace ErrorCodes
{
extern const int NOT_FOUND_COLUMN_IN_BLOCK;
extern const int EMPTY_LIST_OF_COLUMNS_QUERIED;
extern const int NO_SUCH_COLUMN_IN_TABLE;
extern const int COLUMN_QUERIED_MORE_THAN_ONCE;
}
2021-07-21 14:45:19 +00:00
void StorageSnapshot::init()
{
2021-07-21 14:45:19 +00:00
for (const auto & [name, type] : storage.getVirtuals())
virtual_columns[name] = type;
2022-07-21 19:50:19 +00:00
if (storage.hasLightweightDeletedMask())
2022-07-25 14:15:15 +00:00
system_columns[LightweightDeleteDescription::FILTER_COLUMN.name] = LightweightDeleteDescription::FILTER_COLUMN.type;
}
2021-07-21 14:45:19 +00:00
NamesAndTypesList StorageSnapshot::getColumns(const GetColumnsOptions & options) const
{
2021-07-21 14:45:19 +00:00
auto all_columns = getMetadataForQuery()->getColumns().get(options);
if (options.with_extended_objects)
extendObjectColumns(all_columns, object_columns, options.with_subcolumns);
2021-07-21 14:45:19 +00:00
2022-07-21 19:50:19 +00:00
NameSet column_names;
if (options.with_virtuals)
{
/// Virtual columns must be appended after ordinary,
/// because user can override them.
2021-12-01 02:58:24 +00:00
if (!virtual_columns.empty())
{
2021-07-21 14:45:19 +00:00
for (const auto & column : all_columns)
column_names.insert(column.name);
2021-12-01 02:58:24 +00:00
for (const auto & [name, type] : virtual_columns)
if (!column_names.contains(name))
2021-12-01 02:58:24 +00:00
all_columns.emplace_back(name, type);
}
}
2022-07-21 19:50:19 +00:00
if (options.with_system_columns)
{
if (!system_columns.empty() && column_names.empty())
{
for (const auto & column : all_columns)
column_names.insert(column.name);
}
for (const auto & [name, type] : system_columns)
if (!column_names.contains(name))
all_columns.emplace_back(name, type);
}
2021-07-21 14:45:19 +00:00
return all_columns;
}
2021-07-21 14:45:19 +00:00
NamesAndTypesList StorageSnapshot::getColumnsByNames(const GetColumnsOptions & options, const Names & names) const
{
NamesAndTypesList res;
for (const auto & name : names)
2022-03-28 17:21:47 +00:00
res.push_back(getColumn(options, name));
return res;
}
2021-07-21 14:45:19 +00:00
2022-03-28 17:21:47 +00:00
std::optional<NameAndTypePair> StorageSnapshot::tryGetColumn(const GetColumnsOptions & options, const String & column_name) const
{
const auto & columns = getMetadataForQuery()->getColumns();
auto column = columns.tryGetColumn(options, column_name);
2022-05-06 14:44:00 +00:00
if (column && (!column->type->hasDynamicSubcolumns() || !options.with_extended_objects))
2022-03-28 17:21:47 +00:00
return column;
2021-07-21 14:45:19 +00:00
2022-03-28 17:21:47 +00:00
if (options.with_extended_objects)
{
auto object_column = object_columns.tryGetColumn(options, column_name);
if (object_column)
return object_column;
}
2021-07-21 14:45:19 +00:00
2022-03-28 17:21:47 +00:00
if (options.with_virtuals)
{
auto it = virtual_columns.find(column_name);
if (it != virtual_columns.end())
return NameAndTypePair(column_name, it->second);
2021-07-21 14:45:19 +00:00
}
2022-07-21 19:50:19 +00:00
if (options.with_system_columns)
{
auto it = system_columns.find(column_name);
if (it != system_columns.end())
return NameAndTypePair(column_name, it->second);
}
2022-03-28 17:21:47 +00:00
return {};
}
NameAndTypePair StorageSnapshot::getColumn(const GetColumnsOptions & options, const String & column_name) const
{
auto column = tryGetColumn(options, column_name);
if (!column)
throw Exception(ErrorCodes::NO_SUCH_COLUMN_IN_TABLE, "There is no column {} in table", column_name);
return *column;
}
Block StorageSnapshot::getSampleBlockForColumns(const Names & column_names,const NameToNameMap & parameter_values) const
{
Block res;
const auto & columns = getMetadataForQuery()->getColumns();
for (const auto & column_name : column_names)
{
/// substituted_column_name is used for parameterized view (which are created using query parameters
/// and SELECT is used with substitution of these query parameters )
std::string substituted_column_name = column_name;
std::string::size_type pos = 0u;
for (const auto & parameter : parameter_values)
{
if ((pos = substituted_column_name.find("_CAST(" + parameter.second)) != std::string::npos)
{
substituted_column_name = substituted_column_name.substr(0,pos) + parameter.first + ")";
break;
}
}
auto column = columns.tryGetColumnOrSubcolumn(GetColumnsOptions::All, substituted_column_name);
auto object_column = object_columns.tryGetColumnOrSubcolumn(GetColumnsOptions::All, substituted_column_name);
2021-09-09 18:17:54 +00:00
if (column && !object_column)
{
res.insert({column->type->createColumn(), column->type, column_name});
}
2021-09-09 18:17:54 +00:00
else if (object_column)
2021-07-21 14:45:19 +00:00
{
res.insert({object_column->type->createColumn(), object_column->type, column_name});
2021-07-21 14:45:19 +00:00
}
else if (auto it = virtual_columns.find(column_name); it != virtual_columns.end())
{
2021-07-21 14:45:19 +00:00
/// Virtual columns must be appended after ordinary, because user can
/// override them.
const auto & type = it->second;
res.insert({type->createColumn(), type, column_name});
}
else
{
throw Exception(ErrorCodes::NOT_FOUND_COLUMN_IN_BLOCK,
"Column {} not found in table {}", backQuote(substituted_column_name), storage.getStorageID().getNameForLogs());
}
}
return res;
}
ColumnsDescription StorageSnapshot::getDescriptionForColumns(const Names & column_names) const
{
ColumnsDescription res;
const auto & columns = getMetadataForQuery()->getColumns();
for (const auto & name : column_names)
{
auto column = columns.tryGetColumnOrSubcolumnDescription(GetColumnsOptions::All, name);
2022-05-20 17:48:24 +00:00
auto object_column = object_columns.tryGetColumnOrSubcolumnDescription(GetColumnsOptions::All, name);
if (column && !object_column)
{
2022-05-20 17:48:24 +00:00
res.add(*column, "", false, false);
}
else if (object_column)
{
2022-05-20 17:48:24 +00:00
res.add(*object_column, "", false, false);
}
else if (auto it = virtual_columns.find(name); it != virtual_columns.end())
{
/// Virtual columns must be appended after ordinary, because user can
/// override them.
const auto & type = it->second;
res.add({name, type});
}
else
{
throw Exception(ErrorCodes::NOT_FOUND_COLUMN_IN_BLOCK,
"Column {} not found in table {}", backQuote(name), storage.getStorageID().getNameForLogs());
}
}
return res;
}
namespace
{
using DenseHashSet = google::dense_hash_set<StringRef, StringRefHash>;
}
void StorageSnapshot::check(const Names & column_names) const
{
const auto & columns = getMetadataForQuery()->getColumns();
auto options = GetColumnsOptions(GetColumnsOptions::AllPhysical).withSubcolumns();
if (column_names.empty())
{
auto list_of_columns = listOfColumns(columns.get(options));
throw Exception(ErrorCodes::EMPTY_LIST_OF_COLUMNS_QUERIED,
"Empty list of columns queried. There are columns: {}", list_of_columns);
}
DenseHashSet unique_names;
unique_names.set_empty_key(StringRef());
for (const auto & name : column_names)
{
2021-07-21 14:45:19 +00:00
bool has_column = columns.hasColumnOrSubcolumn(GetColumnsOptions::AllPhysical, name)
|| object_columns.hasColumnOrSubcolumn(GetColumnsOptions::AllPhysical, name)
|| virtual_columns.contains(name);
if (!has_column)
{
auto list_of_columns = listOfColumns(columns.get(options));
throw Exception(ErrorCodes::NO_SUCH_COLUMN_IN_TABLE,
"There is no column with name {} in table {}. There are columns: {}",
backQuote(name), storage.getStorageID().getNameForLogs(), list_of_columns);
}
if (unique_names.count(name))
throw Exception(ErrorCodes::COLUMN_QUERIED_MORE_THAN_ONCE, "Column {} queried more than once", name);
unique_names.insert(name);
}
}
DataTypePtr StorageSnapshot::getConcreteType(const String & column_name) const
{
auto object_column = object_columns.tryGetColumnOrSubcolumn(GetColumnsOptions::All, column_name);
if (object_column)
return object_column->type;
return metadata->getColumns().get(column_name).type;
}
}