2021-06-15 19:55:21 +00:00
|
|
|
#include <common/map.h>
|
2021-03-09 14:10:28 +00:00
|
|
|
#include <Common/StringUtils/StringUtils.h>
|
|
|
|
#include <Columns/ColumnMap.h>
|
|
|
|
#include <Columns/ColumnArray.h>
|
|
|
|
#include <Core/Field.h>
|
|
|
|
#include <Formats/FormatSettings.h>
|
|
|
|
#include <DataTypes/DataTypeMap.h>
|
|
|
|
#include <DataTypes/DataTypeArray.h>
|
|
|
|
#include <DataTypes/DataTypeTuple.h>
|
|
|
|
#include <DataTypes/DataTypeFactory.h>
|
|
|
|
#include <DataTypes/Serializations/SerializationMap.h>
|
|
|
|
#include <Parsers/IAST.h>
|
|
|
|
#include <Parsers/ASTNameTypePair.h>
|
|
|
|
#include <Common/typeid_cast.h>
|
|
|
|
#include <Common/assert_cast.h>
|
|
|
|
#include <Common/quoteString.h>
|
|
|
|
#include <IO/WriteHelpers.h>
|
|
|
|
#include <IO/ReadHelpers.h>
|
|
|
|
#include <IO/WriteBufferFromString.h>
|
|
|
|
#include <IO/ReadBufferFromString.h>
|
|
|
|
#include <IO/Operators.h>
|
|
|
|
|
|
|
|
|
|
|
|
namespace DB
|
|
|
|
{
|
|
|
|
|
|
|
|
namespace ErrorCodes
|
|
|
|
{
|
|
|
|
extern const int NUMBER_OF_ARGUMENTS_DOESNT_MATCH;
|
2021-02-17 22:05:31 +00:00
|
|
|
extern const int BAD_ARGUMENTS;
|
2021-03-09 14:10:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
DataTypeMap::DataTypeMap(const DataTypes & elems_)
|
|
|
|
{
|
|
|
|
assert(elems_.size() == 2);
|
|
|
|
key_type = elems_[0];
|
|
|
|
value_type = elems_[1];
|
|
|
|
|
2021-02-17 22:05:31 +00:00
|
|
|
assertKeyType();
|
|
|
|
|
2021-03-09 14:10:28 +00:00
|
|
|
nested = std::make_shared<DataTypeArray>(
|
|
|
|
std::make_shared<DataTypeTuple>(DataTypes{key_type, value_type}, Names{"keys", "values"}));
|
|
|
|
}
|
|
|
|
|
|
|
|
DataTypeMap::DataTypeMap(const DataTypePtr & key_type_, const DataTypePtr & value_type_)
|
|
|
|
: key_type(key_type_), value_type(value_type_)
|
|
|
|
, nested(std::make_shared<DataTypeArray>(
|
2021-02-17 22:05:31 +00:00
|
|
|
std::make_shared<DataTypeTuple>(DataTypes{key_type_, value_type_}, Names{"keys", "values"})))
|
|
|
|
{
|
|
|
|
assertKeyType();
|
|
|
|
}
|
|
|
|
|
2021-02-24 18:13:26 +00:00
|
|
|
void DataTypeMap::assertKeyType() const
|
2021-02-17 22:05:31 +00:00
|
|
|
{
|
2021-05-07 00:24:58 +00:00
|
|
|
if (!key_type->isValueRepresentedByInteger()
|
|
|
|
&& !isStringOrFixedString(*key_type)
|
|
|
|
&& !WhichDataType(key_type).isNothing()
|
|
|
|
&& !WhichDataType(key_type).isUUID())
|
2021-02-17 22:05:31 +00:00
|
|
|
throw Exception(ErrorCodes::BAD_ARGUMENTS,
|
2021-05-07 00:24:58 +00:00
|
|
|
"Type of Map key must be a type, that can be represented by integer or string or UUID,"
|
2021-02-17 22:05:31 +00:00
|
|
|
" but {} given", key_type->getName());
|
|
|
|
}
|
|
|
|
|
2021-03-09 14:10:28 +00:00
|
|
|
|
|
|
|
std::string DataTypeMap::doGetName() const
|
|
|
|
{
|
|
|
|
WriteBufferFromOwnString s;
|
|
|
|
s << "Map(" << key_type->getName() << "," << value_type->getName() << ")";
|
|
|
|
|
|
|
|
return s.str();
|
|
|
|
}
|
|
|
|
|
|
|
|
static const IColumn & extractNestedColumn(const IColumn & column)
|
|
|
|
{
|
|
|
|
return assert_cast<const ColumnMap &>(column).getNestedColumn();
|
|
|
|
}
|
|
|
|
|
|
|
|
DataTypePtr DataTypeMap::tryGetSubcolumnType(const String & subcolumn_name) const
|
|
|
|
{
|
|
|
|
return nested->tryGetSubcolumnType(subcolumn_name);
|
|
|
|
}
|
|
|
|
|
|
|
|
ColumnPtr DataTypeMap::getSubcolumn(const String & subcolumn_name, const IColumn & column) const
|
|
|
|
{
|
|
|
|
return nested->getSubcolumn(subcolumn_name, extractNestedColumn(column));
|
|
|
|
}
|
|
|
|
|
|
|
|
SerializationPtr DataTypeMap::getSubcolumnSerialization(
|
|
|
|
const String & subcolumn_name, const BaseSerializationGetter & base_serialization_getter) const
|
|
|
|
{
|
|
|
|
return nested->getSubcolumnSerialization(subcolumn_name, base_serialization_getter);
|
|
|
|
}
|
|
|
|
|
|
|
|
MutableColumnPtr DataTypeMap::createColumn() const
|
|
|
|
{
|
|
|
|
return ColumnMap::create(nested->createColumn());
|
|
|
|
}
|
|
|
|
|
|
|
|
Field DataTypeMap::getDefault() const
|
|
|
|
{
|
|
|
|
return Map();
|
|
|
|
}
|
|
|
|
|
|
|
|
SerializationPtr DataTypeMap::doGetDefaultSerialization() const
|
|
|
|
{
|
|
|
|
return std::make_shared<SerializationMap>(
|
|
|
|
key_type->getDefaultSerialization(),
|
|
|
|
value_type->getDefaultSerialization(),
|
|
|
|
nested->getDefaultSerialization());
|
|
|
|
}
|
|
|
|
|
|
|
|
bool DataTypeMap::equals(const IDataType & rhs) const
|
|
|
|
{
|
|
|
|
if (typeid(rhs) != typeid(*this))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
const DataTypeMap & rhs_map = static_cast<const DataTypeMap &>(rhs);
|
|
|
|
return nested->equals(*rhs_map.nested);
|
|
|
|
}
|
|
|
|
|
|
|
|
static DataTypePtr create(const ASTPtr & arguments)
|
|
|
|
{
|
|
|
|
if (!arguments || arguments->children.size() != 2)
|
|
|
|
throw Exception("Map data type family must have two arguments: key and value types", ErrorCodes::NUMBER_OF_ARGUMENTS_DOESNT_MATCH);
|
|
|
|
|
|
|
|
DataTypes nested_types;
|
|
|
|
nested_types.reserve(arguments->children.size());
|
|
|
|
|
|
|
|
for (const ASTPtr & child : arguments->children)
|
|
|
|
nested_types.emplace_back(DataTypeFactory::instance().get(child));
|
|
|
|
|
|
|
|
return std::make_shared<DataTypeMap>(nested_types);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void registerDataTypeMap(DataTypeFactory & factory)
|
|
|
|
{
|
|
|
|
factory.registerDataType("Map", create);
|
|
|
|
}
|
|
|
|
}
|