2014-04-29 00:28:18 +00:00
|
|
|
#include <unordered_set>
|
2012-11-30 05:24:24 +00:00
|
|
|
#include <sparsehash/dense_hash_map>
|
2014-03-09 17:36:01 +00:00
|
|
|
#include <sparsehash/dense_hash_set>
|
2017-04-01 09:19:00 +00:00
|
|
|
#include <Storages/ITableDeclaration.h>
|
|
|
|
#include <Parsers/ASTIdentifier.h>
|
|
|
|
#include <Parsers/ASTNameTypePair.h>
|
|
|
|
#include <Interpreters/Context.h>
|
2017-06-06 17:18:32 +00:00
|
|
|
#include <ext/map.h>
|
|
|
|
#include <ext/identity.h>
|
|
|
|
#include <ext/collection_cast.h>
|
2011-08-15 21:50:08 +00:00
|
|
|
|
2015-02-22 15:31:39 +00:00
|
|
|
|
2011-08-15 21:50:08 +00:00
|
|
|
namespace DB
|
|
|
|
{
|
|
|
|
|
2016-01-11 21:46:36 +00:00
|
|
|
namespace ErrorCodes
|
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
extern const int NO_SUCH_COLUMN_IN_TABLE;
|
|
|
|
extern const int EMPTY_LIST_OF_COLUMNS_QUERIED;
|
|
|
|
extern const int COLUMN_QUERIED_MORE_THAN_ONCE;
|
|
|
|
extern const int TYPE_MISMATCH;
|
|
|
|
extern const int DUPLICATE_COLUMN;
|
|
|
|
extern const int NOT_FOUND_COLUMN_IN_BLOCK;
|
2018-01-25 14:42:39 +00:00
|
|
|
extern const int EMPTY_LIST_OF_COLUMNS_PASSED;
|
2016-01-11 21:46:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-12-25 21:57:29 +00:00
|
|
|
NamesAndTypesList ITableDeclaration::getColumnsList() const
|
2014-10-10 15:45:43 +00:00
|
|
|
{
|
2017-12-25 21:57:29 +00:00
|
|
|
return ext::collection_cast<NamesAndTypesList>(getColumnsListRange());
|
2014-10-10 15:45:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-10-16 11:20:33 +00:00
|
|
|
ITableDeclaration::ColumnsListRange ITableDeclaration::getColumnsListRange() const
|
2015-02-22 15:31:39 +00:00
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
return boost::join(getColumnsListImpl(), materialized_columns);
|
2015-02-22 15:31:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-29 00:28:18 +00:00
|
|
|
bool ITableDeclaration::hasRealColumn(const String & column_name) const
|
2014-01-16 14:52:13 +00:00
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
for (auto & it : getColumnsListRange())
|
|
|
|
if (it.name == column_name)
|
|
|
|
return true;
|
|
|
|
return false;
|
2014-01-16 14:52:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-13 15:00:06 +00:00
|
|
|
Names ITableDeclaration::getColumnNamesList() const
|
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
return ext::map<Names>(getColumnsListRange(), [] (const auto & it) { return it.name; });
|
2014-03-13 15:00:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-12-25 21:57:29 +00:00
|
|
|
NameAndTypePair ITableDeclaration::getRealColumn(const String & column_name) const
|
2014-01-16 14:52:13 +00:00
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
for (auto & it : getColumnsListRange())
|
|
|
|
if (it.name == column_name)
|
|
|
|
return it;
|
|
|
|
throw Exception("There is no column " + column_name + " in table.", ErrorCodes::NO_SUCH_COLUMN_IN_TABLE);
|
2014-01-16 14:52:13 +00:00
|
|
|
}
|
|
|
|
|
2017-12-25 21:57:29 +00:00
|
|
|
NameAndTypePair ITableDeclaration::getMaterializedColumn(const String & column_name) const
|
2014-09-30 03:08:47 +00:00
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
for (auto & column : materialized_columns)
|
|
|
|
if (column.name == column_name)
|
|
|
|
return column;
|
2014-01-16 14:52:13 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
throw Exception("There is no column " + column_name + " in table.", ErrorCodes::NO_SUCH_COLUMN_IN_TABLE);
|
2014-09-30 03:08:47 +00:00
|
|
|
}
|
|
|
|
|
2014-10-03 15:30:10 +00:00
|
|
|
bool ITableDeclaration::hasMaterializedColumn(const String & column_name) const
|
2014-01-16 14:52:13 +00:00
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
for (auto & column : materialized_columns)
|
|
|
|
if (column.name == column_name)
|
|
|
|
return true;
|
2014-09-30 03:08:47 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
return false;
|
2014-01-16 14:52:13 +00:00
|
|
|
}
|
|
|
|
|
2014-04-29 00:28:18 +00:00
|
|
|
bool ITableDeclaration::hasColumn(const String & column_name) const
|
2014-01-16 14:52:13 +00:00
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
return hasRealColumn(column_name); /// By default, we assume that there are no virtual columns in the storage.
|
2014-01-16 14:52:13 +00:00
|
|
|
}
|
|
|
|
|
2017-12-25 21:57:29 +00:00
|
|
|
NameAndTypePair ITableDeclaration::getColumn(const String & column_name) const
|
2014-01-16 14:52:13 +00:00
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
return getRealColumn(column_name); /// By default, we assume that there are no virtual columns in the storage.
|
2014-01-16 14:52:13 +00:00
|
|
|
}
|
|
|
|
|
2011-11-01 17:12:11 +00:00
|
|
|
|
2014-03-19 10:45:13 +00:00
|
|
|
const DataTypePtr ITableDeclaration::getDataTypeByName(const String & column_name) const
|
2011-11-01 17:12:11 +00:00
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
for (const auto & column : getColumnsListRange())
|
|
|
|
if (column.name == column_name)
|
|
|
|
return column.type;
|
2014-03-09 17:36:01 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
throw Exception("There is no column " + column_name + " in table.", ErrorCodes::NO_SUCH_COLUMN_IN_TABLE);
|
2011-11-01 17:12:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-19 10:45:13 +00:00
|
|
|
Block ITableDeclaration::getSampleBlock() const
|
2011-10-31 06:37:12 +00:00
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
Block res;
|
2011-10-31 06:37:12 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
for (const auto & col : getColumnsListRange())
|
|
|
|
res.insert({ col.type->createColumn(), col.type, col.name });
|
2014-10-13 14:35:40 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
return res;
|
2014-10-13 14:35:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Block ITableDeclaration::getSampleBlockNonMaterialized() const
|
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
Block res;
|
2014-10-13 14:35:40 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
for (const auto & col : getColumnsListNonMaterialized())
|
|
|
|
res.insert({ col.type->createColumn(), col.type, col.name });
|
2014-03-09 17:36:01 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
return res;
|
2011-10-31 06:37:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-01-09 01:51:08 +00:00
|
|
|
Block ITableDeclaration::getSampleBlockForColumns(const Names & column_names) const
|
|
|
|
{
|
|
|
|
Block res;
|
|
|
|
|
|
|
|
for (const auto & name : column_names)
|
|
|
|
{
|
|
|
|
auto col = getColumn(name);
|
|
|
|
res.insert({ col.type->createColumn(), col.type, name });
|
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-12-25 21:57:29 +00:00
|
|
|
static std::string listOfColumns(const NamesAndTypesList & available_columns)
|
2012-11-30 05:24:24 +00:00
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
std::stringstream s;
|
2017-12-25 21:57:29 +00:00
|
|
|
for (NamesAndTypesList::const_iterator it = available_columns.begin(); it != available_columns.end(); ++it)
|
2017-04-01 07:20:54 +00:00
|
|
|
{
|
|
|
|
if (it != available_columns.begin())
|
|
|
|
s << ", ";
|
|
|
|
s << it->name;
|
|
|
|
}
|
|
|
|
return s.str();
|
2012-11-30 05:24:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-05-28 10:35:44 +00:00
|
|
|
using NamesAndTypesMap = google::dense_hash_map<StringRef, const IDataType *, StringRefHash>;
|
2012-11-30 05:24:24 +00:00
|
|
|
|
2014-10-03 15:30:10 +00:00
|
|
|
static NamesAndTypesMap & getColumnsMapImpl(NamesAndTypesMap & res) { return res; }
|
2012-11-30 05:24:24 +00:00
|
|
|
|
2014-10-03 15:30:10 +00:00
|
|
|
template <typename Arg, typename... Args>
|
|
|
|
static NamesAndTypesMap & getColumnsMapImpl(NamesAndTypesMap & res, const Arg & arg, const Args &... args)
|
|
|
|
{
|
2017-12-25 21:57:29 +00:00
|
|
|
static_assert(std::is_same_v<Arg, NamesAndTypesList>, "getColumnsMap requires arguments of type NamesAndTypesList");
|
2014-10-03 15:30:10 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
for (const auto & column : arg)
|
|
|
|
res.insert({column.name, column.type.get()});
|
2014-10-03 15:30:10 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
return getColumnsMapImpl(res, args...);
|
2014-10-03 15:30:10 +00:00
|
|
|
}
|
2012-11-30 05:24:24 +00:00
|
|
|
|
2014-10-03 15:30:10 +00:00
|
|
|
template <typename... Args>
|
|
|
|
static NamesAndTypesMap getColumnsMap(const Args &... args)
|
2012-11-30 05:24:24 +00:00
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
NamesAndTypesMap res;
|
|
|
|
res.set_empty_key(StringRef());
|
2012-11-30 05:24:24 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
return getColumnsMapImpl(res, args...);
|
2012-11-30 05:24:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-19 10:45:13 +00:00
|
|
|
void ITableDeclaration::check(const Names & column_names) const
|
2011-08-15 21:50:08 +00:00
|
|
|
{
|
2017-12-25 21:57:29 +00:00
|
|
|
const NamesAndTypesList & available_columns = getColumnsList();
|
2014-03-09 17:36:01 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
if (column_names.empty())
|
|
|
|
throw Exception("Empty list of columns queried. There are columns: " + listOfColumns(available_columns),
|
|
|
|
ErrorCodes::EMPTY_LIST_OF_COLUMNS_QUERIED);
|
2011-08-15 21:50:08 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
const auto columns_map = getColumnsMap(available_columns);
|
2012-11-30 05:24:24 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
using UniqueStrings = google::dense_hash_set<StringRef, StringRefHash>;
|
|
|
|
UniqueStrings unique_names;
|
|
|
|
unique_names.set_empty_key(StringRef());
|
2011-08-15 21:50:08 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
for (const auto & name : column_names)
|
|
|
|
{
|
|
|
|
if (columns_map.end() == columns_map.find(name))
|
|
|
|
throw Exception("There is no column with name " + name + " in table. There are columns: " + listOfColumns(available_columns),
|
|
|
|
ErrorCodes::NO_SUCH_COLUMN_IN_TABLE);
|
2011-08-15 21:50:08 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
if (unique_names.end() != unique_names.find(name))
|
|
|
|
throw Exception("Column " + name + " queried more than once",
|
|
|
|
ErrorCodes::COLUMN_QUERIED_MORE_THAN_ONCE);
|
|
|
|
unique_names.insert(name);
|
|
|
|
}
|
2011-08-15 21:50:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-12-25 21:57:29 +00:00
|
|
|
void ITableDeclaration::check(const NamesAndTypesList & columns) const
|
2014-07-14 14:07:47 +00:00
|
|
|
{
|
2017-12-25 21:57:29 +00:00
|
|
|
const NamesAndTypesList & available_columns = getColumnsList();
|
2017-04-01 07:20:54 +00:00
|
|
|
const auto columns_map = getColumnsMap(available_columns);
|
|
|
|
|
|
|
|
using UniqueStrings = google::dense_hash_set<StringRef, StringRefHash>;
|
|
|
|
UniqueStrings unique_names;
|
|
|
|
unique_names.set_empty_key(StringRef());
|
|
|
|
|
2017-12-25 21:57:29 +00:00
|
|
|
for (const NameAndTypePair & column : columns)
|
2017-04-01 07:20:54 +00:00
|
|
|
{
|
|
|
|
NamesAndTypesMap::const_iterator it = columns_map.find(column.name);
|
|
|
|
if (columns_map.end() == it)
|
|
|
|
throw Exception("There is no column with name " + column.name + ". There are columns: "
|
|
|
|
+ listOfColumns(available_columns), ErrorCodes::NO_SUCH_COLUMN_IN_TABLE);
|
|
|
|
|
2017-12-23 01:55:46 +00:00
|
|
|
if (!column.type->equals(*it->second))
|
2017-04-01 07:20:54 +00:00
|
|
|
throw Exception("Type mismatch for column " + column.name + ". Column has type "
|
|
|
|
+ it->second->getName() + ", got type " + column.type->getName(), ErrorCodes::TYPE_MISMATCH);
|
|
|
|
|
|
|
|
if (unique_names.end() != unique_names.find(column.name))
|
|
|
|
throw Exception("Column " + column.name + " queried more than once",
|
|
|
|
ErrorCodes::COLUMN_QUERIED_MORE_THAN_ONCE);
|
|
|
|
unique_names.insert(column.name);
|
|
|
|
}
|
2014-07-14 14:07:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-12-25 21:57:29 +00:00
|
|
|
void ITableDeclaration::check(const NamesAndTypesList & columns, const Names & column_names) const
|
2014-07-14 14:07:47 +00:00
|
|
|
{
|
2017-12-25 21:57:29 +00:00
|
|
|
const NamesAndTypesList & available_columns = getColumnsList();
|
2017-04-01 07:20:54 +00:00
|
|
|
const auto available_columns_map = getColumnsMap(available_columns);
|
|
|
|
const NamesAndTypesMap & provided_columns_map = getColumnsMap(columns);
|
|
|
|
|
|
|
|
if (column_names.empty())
|
|
|
|
throw Exception("Empty list of columns queried. There are columns: " + listOfColumns(available_columns),
|
|
|
|
ErrorCodes::EMPTY_LIST_OF_COLUMNS_QUERIED);
|
|
|
|
|
|
|
|
using UniqueStrings = google::dense_hash_set<StringRef, StringRefHash>;
|
|
|
|
UniqueStrings unique_names;
|
|
|
|
unique_names.set_empty_key(StringRef());
|
|
|
|
|
|
|
|
for (const String & name : column_names)
|
|
|
|
{
|
|
|
|
NamesAndTypesMap::const_iterator it = provided_columns_map.find(name);
|
|
|
|
if (provided_columns_map.end() == it)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
NamesAndTypesMap::const_iterator jt = available_columns_map.find(name);
|
|
|
|
if (available_columns_map.end() == jt)
|
|
|
|
throw Exception("There is no column with name " + name + ". There are columns: "
|
|
|
|
+ listOfColumns(available_columns), ErrorCodes::NO_SUCH_COLUMN_IN_TABLE);
|
|
|
|
|
|
|
|
if (it->second->getName() != jt->second->getName())
|
|
|
|
throw Exception("Type mismatch for column " + name + ". Column has type "
|
|
|
|
+ jt->second->getName() + ", got type " + it->second->getName(), ErrorCodes::TYPE_MISMATCH);
|
|
|
|
|
|
|
|
if (unique_names.end() != unique_names.find(name))
|
|
|
|
throw Exception("Column " + name + " queried more than once",
|
|
|
|
ErrorCodes::COLUMN_QUERIED_MORE_THAN_ONCE);
|
|
|
|
unique_names.insert(name);
|
|
|
|
}
|
2014-07-14 14:07:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-19 10:45:13 +00:00
|
|
|
void ITableDeclaration::check(const Block & block, bool need_all) const
|
2011-08-15 21:50:08 +00:00
|
|
|
{
|
2017-12-25 21:57:29 +00:00
|
|
|
const NamesAndTypesList & available_columns = getColumnsList();
|
2017-04-01 07:20:54 +00:00
|
|
|
const auto columns_map = getColumnsMap(available_columns);
|
|
|
|
|
|
|
|
using NameSet = std::unordered_set<String>;
|
|
|
|
NameSet names_in_block;
|
|
|
|
|
2018-01-23 18:49:15 +00:00
|
|
|
block.checkNumberOfRows();
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2018-01-23 18:49:15 +00:00
|
|
|
for (const auto & column : block)
|
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
if (names_in_block.count(column.name))
|
|
|
|
throw Exception("Duplicate column " + column.name + " in block",
|
|
|
|
ErrorCodes::DUPLICATE_COLUMN);
|
|
|
|
|
|
|
|
names_in_block.insert(column.name);
|
|
|
|
|
|
|
|
NamesAndTypesMap::const_iterator it = columns_map.find(column.name);
|
|
|
|
if (columns_map.end() == it)
|
|
|
|
throw Exception("There is no column with name " + column.name + ". There are columns: "
|
|
|
|
+ listOfColumns(available_columns), ErrorCodes::NO_SUCH_COLUMN_IN_TABLE);
|
|
|
|
|
2017-12-23 01:55:46 +00:00
|
|
|
if (!column.type->equals(*it->second))
|
2017-04-01 07:20:54 +00:00
|
|
|
throw Exception("Type mismatch for column " + column.name + ". Column has type "
|
|
|
|
+ it->second->getName() + ", got type " + column.type->getName(), ErrorCodes::TYPE_MISMATCH);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (need_all && names_in_block.size() < columns_map.size())
|
|
|
|
{
|
2017-12-25 21:57:29 +00:00
|
|
|
for (NamesAndTypesList::const_iterator it = available_columns.begin(); it != available_columns.end(); ++it)
|
2017-04-01 07:20:54 +00:00
|
|
|
{
|
|
|
|
if (!names_in_block.count(it->name))
|
|
|
|
throw Exception("Expected column " + it->name, ErrorCodes::NOT_FOUND_COLUMN_IN_BLOCK);
|
|
|
|
}
|
|
|
|
}
|
2011-08-15 21:50:08 +00:00
|
|
|
}
|
|
|
|
|
2018-01-25 14:42:39 +00:00
|
|
|
ITableDeclaration::ITableDeclaration(const NamesAndTypesList & columns, const NamesAndTypesList & materialized_columns,
|
|
|
|
const NamesAndTypesList & alias_columns, const ColumnDefaults & column_defaults)
|
|
|
|
: columns{columns},
|
|
|
|
materialized_columns{materialized_columns},
|
|
|
|
alias_columns{alias_columns},
|
|
|
|
column_defaults{column_defaults}
|
|
|
|
{
|
|
|
|
if (columns.empty())
|
|
|
|
throw Exception("Empty list of columns passed to storage constructor", ErrorCodes::EMPTY_LIST_OF_COLUMNS_PASSED);
|
|
|
|
}
|
|
|
|
|
2011-08-15 21:50:08 +00:00
|
|
|
}
|