#pragma once #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "FunctionsMiscellaneous.h" namespace DB { /** Функции преобразования типов. * toType - преобразование "естественным образом"; */ /** Преобразование чисел друг в друга, перечислений в числа, дат/дат-с-временем в числа и наоборот: делается обычным присваиванием. * (дата внутри хранится как количество дней с какого-то, дата-с-временем - как unix timestamp) */ template struct ConvertImpl { typedef typename FromDataType::FieldType FromFieldType; typedef typename ToDataType::FieldType ToFieldType; static void execute(Block & block, const ColumnNumbers & arguments, size_t result) { if (const ColumnVector * col_from = typeid_cast *>(&*block.getByPosition(arguments[0]).column)) { ColumnVector * col_to = new ColumnVector; block.getByPosition(result).column = col_to; const typename ColumnVector::Container_t & vec_from = col_from->getData(); typename ColumnVector::Container_t & vec_to = col_to->getData(); size_t size = vec_from.size(); vec_to.resize(size); for (size_t i = 0; i < size; ++i) vec_to[i] = vec_from[i]; } else if (const ColumnConst * col_from = typeid_cast *>(&*block.getByPosition(arguments[0]).column)) { block.getByPosition(result).column = new ColumnConst(col_from->size(), col_from->getData()); } else throw Exception("Illegal column " + block.getByPosition(arguments[0]).column->getName() + " of first argument of function " + Name::name, ErrorCodes::ILLEGAL_COLUMN); } }; /** Преобразование даты в дату-с-временем: добавление нулевого времени. */ template struct ConvertImpl { typedef DataTypeDate::FieldType FromFieldType; typedef DataTypeDateTime::FieldType ToFieldType; static void execute(Block & block, const ColumnNumbers & arguments, size_t result) { typedef DataTypeDate::FieldType FromFieldType; const auto & date_lut = DateLUT::instance(); if (const ColumnVector * col_from = typeid_cast *>(&*block.getByPosition(arguments[0]).column)) { ColumnVector * col_to = new ColumnVector; block.getByPosition(result).column = col_to; const typename ColumnVector::Container_t & vec_from = col_from->getData(); typename ColumnVector::Container_t & vec_to = col_to->getData(); size_t size = vec_from.size(); vec_to.resize(size); for (size_t i = 0; i < size; ++i) { vec_to[i] = date_lut.fromDayNum(DayNum_t(vec_from[i])); } } else if (const ColumnConst * col_from = typeid_cast *>(&*block.getByPosition(arguments[0]).column)) { block.getByPosition(result).column = new ColumnConst(col_from->size(), date_lut.fromDayNum(DayNum_t(col_from->getData()))); } else throw Exception("Illegal column " + block.getByPosition(arguments[0]).column->getName() + " of first argument of function " + Name::name, ErrorCodes::ILLEGAL_COLUMN); } }; /// Реализация функции toDate. namespace details { template class Transformation> class Transformer { private: using Op = Transformation; public: static void vector_vector(const PODArray & vec_from, const ColumnString::Chars_t & data, const ColumnString::Offsets_t & offsets, PODArray & vec_to) { ColumnString::Offset_t prev_offset = 0; for (size_t i = 0; i < vec_from.size(); ++i) { ColumnString::Offset_t cur_offset = offsets[i]; const std::string time_zone(reinterpret_cast(&data[prev_offset]), cur_offset - prev_offset - 1); const auto & remote_date_lut = DateLUT::instance(time_zone); vec_to[i] = Op::execute(vec_from[i], remote_date_lut); prev_offset = cur_offset; } } static void vector_constant(const PODArray & vec_from, const std::string & data, PODArray & vec_to) { const auto & remote_date_lut = DateLUT::instance(data); for (size_t i = 0; i < vec_from.size(); ++i) vec_to[i] = Op::execute(vec_from[i], remote_date_lut); } static void vector_constant(const PODArray & vec_from, PODArray & vec_to) { const auto & local_date_lut = DateLUT::instance(); for (size_t i = 0; i < vec_from.size(); ++i) vec_to[i] = Op::execute(vec_from[i], local_date_lut); } static void constant_vector(const FromType & from, const ColumnString::Chars_t & data, const ColumnString::Offsets_t & offsets, PODArray & vec_to) { ColumnString::Offset_t prev_offset = 0; for (size_t i = 0; i < offsets.size(); ++i) { ColumnString::Offset_t cur_offset = offsets[i]; const std::string time_zone(reinterpret_cast(&data[prev_offset]), cur_offset - prev_offset - 1); const auto & remote_date_lut = DateLUT::instance(time_zone); vec_to[i] = Op::execute(from, remote_date_lut); prev_offset = cur_offset; } } static void constant_constant(const FromType & from, const std::string & data, ToType & to) { const auto & remote_date_lut = DateLUT::instance(data); to = Op::execute(from, remote_date_lut); } static void constant_constant(const FromType & from, ToType & to) { const auto & local_date_lut = DateLUT::instance(); to = Op::execute(from, local_date_lut); } }; template class Transformation, typename Name> class ToDateConverter { private: using FromFieldType = typename FromType::FieldType; using ToFieldType = typename DataTypeDate::FieldType; using Op = Transformer; public: static void execute(Block & block, const ColumnNumbers & arguments, size_t result) { const ColumnPtr source_col = block.getByPosition(arguments[0]).column; const auto * sources = typeid_cast *>(&*source_col); const auto * const_source = typeid_cast *>(&*source_col); if (arguments.size() == 1) { if (sources) { auto * col_to = new ColumnVector; block.getByPosition(result).column = col_to; const auto & vec_from = sources->getData(); auto & vec_to = col_to->getData(); size_t size = vec_from.size(); vec_to.resize(size); Op::vector_constant(vec_from, vec_to); } else if (const_source) { ToFieldType res; Op::constant_constant(const_source->getData(), res); block.getByPosition(result).column = new ColumnConst(const_source->size(), res); } else throw Exception("Illegal column " + block.getByPosition(arguments[0]).column->getName() + " of argument of function " + Name::name, ErrorCodes::ILLEGAL_COLUMN); } else if (arguments.size() == 2) { const ColumnPtr time_zone_col = block.getByPosition(arguments[1]).column; const auto * time_zones = typeid_cast(&*time_zone_col); const auto * const_time_zone = typeid_cast(&*time_zone_col); if (sources) { auto * col_to = new ColumnVector; block.getByPosition(result).column = col_to; auto & vec_from = sources->getData(); auto & vec_to = col_to->getData(); vec_to.resize(vec_from.size()); if (time_zones) Op::vector_vector(vec_from, time_zones->getChars(), time_zones->getOffsets(), vec_to); else if (const_time_zone) Op::vector_constant(vec_from, const_time_zone->getData(), vec_to); else throw Exception("Illegal column " + block.getByPosition(arguments[1]).column->getName() + " of second argument of function " + Name::name, ErrorCodes::ILLEGAL_COLUMN); } else if (const_source) { if (time_zones) { auto * col_to = new ColumnVector; block.getByPosition(result).column = col_to; auto & vec_to = col_to->getData(); vec_to.resize(time_zones->getOffsets().size()); Op::constant_vector(const_source->getData(), time_zones->getChars(), time_zones->getOffsets(), vec_to); } else if (const_time_zone) { ToFieldType res; Op::constant_constant(const_source->getData(), const_time_zone->getData(), res); block.getByPosition(result).column = new ColumnConst(const_source->size(), res); } else throw Exception("Illegal column " + block.getByPosition(arguments[1]).column->getName() + " of second argument of function " + Name::name, ErrorCodes::ILLEGAL_COLUMN); } else throw Exception("Illegal column " + block.getByPosition(arguments[0]).column->getName() + " of first argument of function " + Name::name, ErrorCodes::ILLEGAL_COLUMN); } else throw Exception("FunctionsConversion: Internal error", ErrorCodes::LOGICAL_ERROR); } }; template struct ToDateTransform { static inline ToType execute(const FromType & from, const DateLUTImpl & date_lut) { return date_lut.toDayNum(from); } }; template struct ToDateTransform32Or64 { static inline ToType execute(const FromType & from, const DateLUTImpl & date_lut) { return (from < 0xFFFF) ? from : date_lut.toDayNum(from); } }; } /** Преобразование даты-с-временем в дату: отбрасывание времени. */ template struct ConvertImpl : details::ToDateConverter {}; /** Отдельный случай для преобразования (U)Int32 или (U)Int64 в Date. * Если число меньше 65536, то оно понимается, как DayNum, а если больше или равно - как unix timestamp. * Немного нелогично, что мы, по сути, помещаем две разные функции в одну. * Но зато это позволяет поддержать распространённый случай, * когда пользователь пишет toDate(UInt32), ожидая, что это - перевод unix timestamp в дату * (иначе такое использование было бы распространённой ошибкой). */ template struct ConvertImpl : details::ToDateConverter {}; template struct ConvertImpl : details::ToDateConverter {}; template struct ConvertImpl : details::ToDateConverter {}; template struct ConvertImpl : details::ToDateConverter {}; /** Преобразование чисел, дат, дат-с-временем в строки: через форматирование. */ template struct FormatImpl { static void execute(const typename DataType::FieldType x, WriteBuffer & wb, const DataType & type = DataType{}) { writeText(x, wb); } }; template <> struct FormatImpl { static void execute(const DataTypeDate::FieldType x, WriteBuffer & wb, const DataTypeDate & type = DataTypeDate{}) { writeDateText(DayNum_t(x), wb); } }; template <> struct FormatImpl { static void execute(const DataTypeDateTime::FieldType x, WriteBuffer & wb, const DataTypeDateTime &type = DataTypeDateTime{}) { writeDateTimeText(x, wb); } }; template struct FormatImpl> { static void execute(const FieldType x, WriteBuffer & wb, const DataTypeEnum & type) { writeText(type.getNameForValue(x), wb); } }; template struct ConvertImpl { typedef typename FromDataType::FieldType FromFieldType; static void execute(Block & block, const ColumnNumbers & arguments, size_t result) { const auto & col_with_name_and_type = block.getByPosition(arguments[0]); const auto & type = static_cast(*col_with_name_and_type.type); if (const auto col_from = typeid_cast *>(&*col_with_name_and_type.column)) { ColumnString * col_to = new ColumnString; block.getByPosition(result).column = col_to; const typename ColumnVector::Container_t & vec_from = col_from->getData(); ColumnString::Chars_t & data_to = col_to->getChars(); ColumnString::Offsets_t & offsets_to = col_to->getOffsets(); size_t size = vec_from.size(); data_to.resize(size * 2); offsets_to.resize(size); WriteBufferFromVector write_buffer(data_to); for (size_t i = 0; i < size; ++i) { FormatImpl::execute(vec_from[i], write_buffer, type); writeChar(0, write_buffer); offsets_to[i] = write_buffer.count(); } data_to.resize(write_buffer.count()); } else if (const auto col_from = typeid_cast *>(&*col_with_name_and_type.column)) { std::vector buf; WriteBufferFromVector > write_buffer(buf); FormatImpl::execute(col_from->getData(), write_buffer, type); block.getByPosition(result).column = new ColumnConstString(col_from->size(), std::string(&buf[0], write_buffer.count())); } else throw Exception("Illegal column " + block.getByPosition(arguments[0]).column->getName() + " of first argument of function " + Name::name, ErrorCodes::ILLEGAL_COLUMN); } }; namespace details { namespace { /** Пусть source_timestamp представляет дату и время в исходном часовом поясе соответствующем * объекту from_date_lut. Эта функция возвращает timestamp представлящий те же дату и время * в часовом поясе соответствующем объекту to_date_lut. */ time_t convertTimestamp(time_t source_timestamp, const DateLUTImpl & from_date_lut, const DateLUTImpl & to_date_lut) { if (&from_date_lut == &to_date_lut) return source_timestamp; else { const auto & values = from_date_lut.getValues(source_timestamp); return to_date_lut.makeDateTime(values.year, values.month, values.day_of_month, from_date_lut.toHourInaccurate(source_timestamp), from_date_lut.toMinuteInaccurate(source_timestamp), from_date_lut.toSecondInaccurate(source_timestamp)); } } /** Функции для преобразования даты + времени в строку. */ struct DateTimeToStringConverter { using FromFieldType = typename DataTypeDateTime::FieldType; static void vector_vector(const PODArray & vec_from, const ColumnString::Chars_t & data, const ColumnString::Offsets_t & offsets, ColumnString & vec_to) { const auto & local_date_lut = DateLUT::instance(); ColumnString::Chars_t & data_to = vec_to.getChars(); ColumnString::Offsets_t & offsets_to = vec_to.getOffsets(); size_t size = vec_from.size(); data_to.resize(size * 2); offsets_to.resize(size); WriteBufferFromVector write_buffer(data_to); ColumnString::Offset_t prev_offset = 0; for (size_t i = 0; i < size; ++i) { ColumnString::Offset_t cur_offset = offsets[i]; const std::string time_zone(reinterpret_cast(&data[prev_offset]), cur_offset - prev_offset - 1); const auto & remote_date_lut = DateLUT::instance(time_zone); auto ti = convertTimestamp(vec_from[i], remote_date_lut, local_date_lut); FormatImpl::execute(ti, write_buffer); writeChar(0, write_buffer); offsets_to[i] = write_buffer.count(); prev_offset = cur_offset; } data_to.resize(write_buffer.count()); } static void vector_constant(const PODArray & vec_from, const std::string & data, ColumnString & vec_to) { const auto & local_date_lut = DateLUT::instance(); const auto & remote_date_lut = DateLUT::instance(data); ColumnString::Chars_t & data_to = vec_to.getChars(); ColumnString::Offsets_t & offsets_to = vec_to.getOffsets(); size_t size = vec_from.size(); data_to.resize(size * 2); offsets_to.resize(size); WriteBufferFromVector write_buffer(data_to); for (size_t i = 0; i < size; ++i) { auto ti = convertTimestamp(vec_from[i], remote_date_lut, local_date_lut); FormatImpl::execute(ti, write_buffer); writeChar(0, write_buffer); offsets_to[i] = write_buffer.count(); } data_to.resize(write_buffer.count()); } static void vector_constant(const PODArray & vec_from, ColumnString & vec_to) { ColumnString::Chars_t & data_to = vec_to.getChars(); ColumnString::Offsets_t & offsets_to = vec_to.getOffsets(); size_t size = vec_from.size(); data_to.resize(size * 2); offsets_to.resize(size); WriteBufferFromVector write_buffer(data_to); for (size_t i = 0; i < size; ++i) { FormatImpl::execute(vec_from[i], write_buffer); writeChar(0, write_buffer); offsets_to[i] = write_buffer.count(); } data_to.resize(write_buffer.count()); } static void constant_vector(FromFieldType from, const ColumnString::Chars_t & data, const ColumnString::Offsets_t & offsets, ColumnString & vec_to) { const auto & local_date_lut = DateLUT::instance(); ColumnString::Chars_t & data_to = vec_to.getChars(); ColumnString::Offsets_t & offsets_to = vec_to.getOffsets(); size_t size = offsets.size(); data_to.resize(size * 2); offsets_to.resize(size); WriteBufferFromVector write_buffer(data_to); ColumnString::Offset_t prev_offset = 0; for (size_t i = 0; i < size; ++i) { ColumnString::Offset_t cur_offset = offsets[i]; const std::string time_zone(reinterpret_cast(&data[prev_offset]), cur_offset - prev_offset - 1); const auto & remote_date_lut = DateLUT::instance(time_zone); auto ti = convertTimestamp(from, remote_date_lut, local_date_lut); FormatImpl::execute(ti, write_buffer); writeChar(0, write_buffer); offsets_to[i] = write_buffer.count(); prev_offset = cur_offset; } data_to.resize(write_buffer.count()); } static void constant_constant(FromFieldType from, const std::string & data, std::string & to) { const auto & local_date_lut = DateLUT::instance(); const auto & remote_date_lut = DateLUT::instance(data); std::vector buf; WriteBufferFromVector > write_buffer(buf); auto ti = convertTimestamp(from, remote_date_lut, local_date_lut); FormatImpl::execute(ti, write_buffer); to = std::string(&buf[0], write_buffer.count()); } static void constant_constant(FromFieldType from, std::string & to) { std::vector buf; WriteBufferFromVector > write_buffer(buf); FormatImpl::execute(from, write_buffer); to = std::string(&buf[0], write_buffer.count()); } }; }} template struct ConvertImpl { using Op = details::DateTimeToStringConverter; using FromFieldType = Op::FromFieldType; static void execute(Block & block, const ColumnNumbers & arguments, size_t result) { const ColumnPtr source_col = block.getByPosition(arguments[0]).column; const auto * sources = typeid_cast *>(&*source_col); const auto * const_source = typeid_cast *>(&*source_col); if (arguments.size() == 1) { if (sources) { ColumnString * col_to = new ColumnString; block.getByPosition(result).column = col_to; auto & vec_from = sources->getData(); auto & vec_to = *col_to; Op::vector_constant(vec_from, vec_to); } else if (const_source) { std::string res; Op::constant_constant(const_source->getData(), res); block.getByPosition(result).column = new ColumnConstString(const_source->size(), res); } else { throw Exception("Illegal column " + block.getByPosition(arguments[0]).column->getName() + " of first argument of function " + Name::name, ErrorCodes::ILLEGAL_COLUMN); } } else if (arguments.size() == 2) { const ColumnPtr time_zone_col = block.getByPosition(arguments[1]).column; const auto * time_zones = typeid_cast(&*time_zone_col); const auto * const_time_zone = typeid_cast(&*time_zone_col); if (sources) { ColumnString * col_to = new ColumnString; block.getByPosition(result).column = col_to; auto & vec_from = sources->getData(); auto & vec_to = *col_to; if (time_zones) Op::vector_vector(vec_from, time_zones->getChars(), time_zones->getOffsets(), vec_to); else if (const_time_zone) Op::vector_constant(vec_from, const_time_zone->getData(), vec_to); else throw Exception("Illegal column " + block.getByPosition(arguments[1]).column->getName() + " of second argument of function " + Name::name, ErrorCodes::ILLEGAL_COLUMN); } else if (const_source) { if (time_zones) { ColumnString * col_to = new ColumnString; block.getByPosition(result).column = col_to; auto & vec_to = *col_to; Op::constant_vector(const_source->getData(), time_zones->getChars(), time_zones->getOffsets(), vec_to); } else if (const_time_zone) { std::string res; Op::constant_constant(const_source->getData(), const_time_zone->getData(), res); block.getByPosition(result).column = new ColumnConstString(const_source->size(), res); } else throw Exception("Illegal column " + block.getByPosition(arguments[1]).column->getName() + " of second argument of function " + Name::name, ErrorCodes::ILLEGAL_COLUMN); } else throw Exception("Illegal column " + block.getByPosition(arguments[0]).column->getName() + " of first argument of function " + Name::name, ErrorCodes::ILLEGAL_COLUMN); } else throw Exception("Internal error.", ErrorCodes::LOGICAL_ERROR); } }; /** Преобразование строк в числа, даты, даты-с-временем: через парсинг. */ template void parseImpl(typename DataType::FieldType & x, ReadBuffer & rb) { readText(x,rb); } template <> inline void parseImpl(DataTypeDate::FieldType & x, ReadBuffer & rb) { DayNum_t tmp(0); readDateText(tmp, rb); x = tmp; } template <> inline void parseImpl(DataTypeDateTime::FieldType & x, ReadBuffer & rb) { time_t tmp = 0; readDateTimeText(tmp, rb); x = tmp; } template struct ConvertImpl { typedef typename ToDataType::FieldType ToFieldType; static void execute(Block & block, const ColumnNumbers & arguments, size_t result) { if (const ColumnString * col_from = typeid_cast(&*block.getByPosition(arguments[0]).column)) { ColumnVector * col_to = new ColumnVector; block.getByPosition(result).column = col_to; const ColumnString::Chars_t & data_from = col_from->getChars(); typename ColumnVector::Container_t & vec_to = col_to->getData(); size_t size = col_from->size(); vec_to.resize(size); ReadBuffer read_buffer(const_cast(reinterpret_cast(&data_from[0])), data_from.size(), 0); char zero = 0; for (size_t i = 0; i < size; ++i) { parseImpl(vec_to[i], read_buffer); readChar(zero, read_buffer); if (zero != 0) throw Exception("Cannot parse from string.", ErrorCodes::CANNOT_PARSE_NUMBER); } } else if (const ColumnConstString * col_from = typeid_cast(&*block.getByPosition(arguments[0]).column)) { const String & s = col_from->getData(); ReadBufferFromString read_buffer(s); ToFieldType x = 0; parseImpl(x, read_buffer); block.getByPosition(result).column = new ColumnConst(col_from->size(), x); } else throw Exception("Illegal column " + block.getByPosition(arguments[0]).column->getName() + " of first argument of function " + Name::name, ErrorCodes::ILLEGAL_COLUMN); } }; namespace details { namespace { /** Функции для преобразования строк в timestamp. */ struct StringToTimestampConverter { using ToFieldType = typename DataTypeInt32::FieldType; static void vector_vector(const ColumnString::Chars_t & vec_from, const ColumnString::Chars_t & data, const ColumnString::Offsets_t & offsets, PODArray & vec_to) { const auto & local_date_lut = DateLUT::instance(); ReadBuffer read_buffer(const_cast(reinterpret_cast(&vec_from[0])), vec_from.size(), 0); ColumnString::Offset_t prev_offset = 0; char zero = 0; for (size_t i = 0; i < vec_to.size(); ++i) { DataTypeDateTime::FieldType x = 0; parseImpl(x, read_buffer); ColumnString::Offset_t cur_offset = offsets[i]; const std::string time_zone(reinterpret_cast(&data[prev_offset]), cur_offset - prev_offset - 1); const auto & remote_date_lut = DateLUT::instance(time_zone); auto ti = convertTimestamp(x, local_date_lut, remote_date_lut); vec_to[i] = ti; readChar(zero, read_buffer); if (zero != 0) throw Exception("Cannot parse from string.", ErrorCodes::CANNOT_PARSE_NUMBER); prev_offset = cur_offset; } } static void vector_constant(const ColumnString::Chars_t & vec_from, const std::string & data, PODArray & vec_to) { const auto & local_date_lut = DateLUT::instance(); const auto & remote_date_lut = DateLUT::instance(data); ReadBuffer read_buffer(const_cast(reinterpret_cast(&vec_from[0])), vec_from.size(), 0); char zero = 0; for (size_t i = 0; i < vec_to.size(); ++i) { DataTypeDateTime::FieldType x = 0; parseImpl(x, read_buffer); auto ti = convertTimestamp(x, local_date_lut, remote_date_lut); vec_to[i] = ti; readChar(zero, read_buffer); if (zero != 0) throw Exception("Cannot parse from string.", ErrorCodes::CANNOT_PARSE_NUMBER); } } static void vector_constant(const ColumnString::Chars_t & vec_from, PODArray & vec_to) { ReadBuffer read_buffer(const_cast(reinterpret_cast(&vec_from[0])), vec_from.size(), 0); char zero = 0; for (size_t i = 0; i < vec_to.size(); ++i) { DataTypeDateTime::FieldType x = 0; parseImpl(x, read_buffer); vec_to[i] = x; readChar(zero, read_buffer); if (zero != 0) throw Exception("Cannot parse from string.", ErrorCodes::CANNOT_PARSE_NUMBER); } } static void constant_vector(const std::string & from, const ColumnString::Chars_t & data, const ColumnString::Offsets_t & offsets, PODArray & vec_to) { const auto & local_date_lut = DateLUT::instance(); ReadBufferFromString read_buffer(from); DataTypeDateTime::FieldType x = 0; parseImpl(x, read_buffer); ColumnString::Offset_t prev_offset = 0; for (size_t i = 0; i < offsets.size(); ++i) { ColumnString::Offset_t cur_offset = offsets[i]; const std::string time_zone(reinterpret_cast(&data[prev_offset]), cur_offset - prev_offset - 1); const auto & remote_date_lut = DateLUT::instance(time_zone); auto ti = convertTimestamp(x, local_date_lut, remote_date_lut); vec_to[i] = ti; prev_offset = cur_offset; } } static void constant_constant(const std::string & from, const std::string & data, ToFieldType & to) { const auto & local_date_lut = DateLUT::instance(); const auto & remote_date_lut = DateLUT::instance(data); ReadBufferFromString read_buffer(from); DataTypeDateTime::FieldType x = 0; parseImpl(x, read_buffer); to = convertTimestamp(x, local_date_lut, remote_date_lut); } static void constant_constant(const std::string & from, ToFieldType & to) { ReadBufferFromString read_buffer(from); DataTypeDateTime::FieldType x = 0; parseImpl(x, read_buffer); to = x; } }; }} struct NameToUnixTimestamp { static constexpr auto name = "toUnixTimestamp"; }; template<> struct ConvertImpl { using Op = details::StringToTimestampConverter; using ToFieldType = Op::ToFieldType; static void execute(Block & block, const ColumnNumbers & arguments, size_t result) { const ColumnPtr source_col = block.getByPosition(arguments[0]).column; const auto * sources = typeid_cast(&*source_col); const auto * const_source = typeid_cast(&*source_col); if (arguments.size() == 1) { if (sources) { auto * col_to = new ColumnVector; block.getByPosition(result).column = col_to; auto & vec_from = sources->getChars(); auto & vec_to = col_to->getData(); size_t size = sources->size(); vec_to.resize(size); Op::vector_constant(vec_from, vec_to); } else if (const_source) { ToFieldType res; Op::constant_constant(const_source->getData(), res); block.getByPosition(result).column = new ColumnConst(const_source->size(), res); } else { throw Exception("Illegal column " + block.getByPosition(arguments[0]).column->getName() + " of first argument of function " + NameToUnixTimestamp::name, ErrorCodes::ILLEGAL_COLUMN); } } else if (arguments.size() == 2) { const ColumnPtr time_zone_col = block.getByPosition(arguments[1]).column; const auto * time_zones = typeid_cast(&*time_zone_col); const auto * const_time_zone = typeid_cast(&*time_zone_col); if (sources) { auto * col_to = new ColumnVector; block.getByPosition(result).column = col_to; auto & vec_from = sources->getChars(); auto & vec_to = col_to->getData(); size_t size = sources->size(); vec_to.resize(size); if (time_zones) Op::vector_vector(vec_from, time_zones->getChars(), time_zones->getOffsets(), vec_to); else if (const_time_zone) Op::vector_constant(vec_from, const_time_zone->getData(), vec_to); else throw Exception("Illegal column " + block.getByPosition(arguments[1]).column->getName() + " of second argument of function " + NameToUnixTimestamp::name, ErrorCodes::ILLEGAL_COLUMN); } else if (const_source) { if (time_zones) { auto * col_to = new ColumnVector; block.getByPosition(result).column = col_to; auto & vec_to = col_to->getData(); vec_to.resize(time_zones->getOffsets().size()); Op::constant_vector(const_source->getData(), time_zones->getChars(), time_zones->getOffsets(), vec_to); } else if (const_time_zone) { ToFieldType res; Op::constant_constant(const_source->getData(), const_time_zone->getData(), res); block.getByPosition(result).column = new ColumnConst(const_source->size(), res); } else throw Exception("Illegal column " + block.getByPosition(arguments[1]).column->getName() + " of second argument of function " + NameToUnixTimestamp::name, ErrorCodes::ILLEGAL_COLUMN); } else throw Exception("Illegal column " + block.getByPosition(arguments[0]).column->getName() + " of first argument of function " + NameToUnixTimestamp::name, ErrorCodes::ILLEGAL_COLUMN); } else throw Exception("Internal error.", ErrorCodes::LOGICAL_ERROR); } }; /** Если типы совпадают - просто скопируем ссылку на столбец. */ template struct ConvertImpl { static void execute(Block & block, const ColumnNumbers & arguments, size_t result) { block.getByPosition(result).column = block.getByPosition(arguments[0]).column; } }; /** Преобразование из FixedString. */ template struct ConvertImpl { typedef typename ToDataType::FieldType ToFieldType; static void execute(Block & block, const ColumnNumbers & arguments, size_t result) { if (const ColumnFixedString * col_from = typeid_cast(&*block.getByPosition(arguments[0]).column)) { ColumnVector * col_to = new ColumnVector; block.getByPosition(result).column = col_to; const ColumnFixedString::Chars_t & data_from = col_from->getChars(); size_t n = col_from->getN(); typename ColumnVector::Container_t & vec_to = col_to->getData(); size_t size = col_from->size(); vec_to.resize(size); for (size_t i = 0; i < size; ++i) { char * begin = const_cast(reinterpret_cast(&data_from[i * n])); char * end = begin + n; ReadBuffer read_buffer(begin, n, 0); parseImpl(vec_to[i], read_buffer); if (!read_buffer.eof()) { while (read_buffer.position() < end && *read_buffer.position() == 0) ++read_buffer.position(); if (read_buffer.position() < end) throw Exception("Cannot parse from fixed string.", ErrorCodes::CANNOT_PARSE_NUMBER); } } } else if (typeid_cast(&*block.getByPosition(arguments[0]).column)) { ConvertImpl::execute(block, arguments, result); } else throw Exception("Illegal column " + block.getByPosition(arguments[0]).column->getName() + " of first argument of function " + Name::name, ErrorCodes::ILLEGAL_COLUMN); } }; /** Преобразование из FixedString в String. * При этом, вырезаются последовательности нулевых байт с конца строк. */ template struct ConvertImpl { static void execute(Block & block, const ColumnNumbers & arguments, size_t result) { if (const ColumnFixedString * col_from = typeid_cast(&*block.getByPosition(arguments[0]).column)) { ColumnString * col_to = new ColumnString; block.getByPosition(result).column = col_to; const ColumnFixedString::Chars_t & data_from = col_from->getChars(); ColumnString::Chars_t & data_to = col_to->getChars(); ColumnString::Offsets_t & offsets_to = col_to->getOffsets(); size_t size = col_from->size(); size_t n = col_from->getN(); data_to.resize(size * (n + 1)); /// + 1 - нулевой байт offsets_to.resize(size); size_t offset_from = 0; size_t offset_to = 0; for (size_t i = 0; i < size; ++i) { size_t bytes_to_copy = n; while (bytes_to_copy > 0 && data_from[offset_from + bytes_to_copy - 1] == 0) --bytes_to_copy; memcpy(&data_to[offset_to], &data_from[offset_from], bytes_to_copy); offset_from += n; offset_to += bytes_to_copy; data_to[offset_to] = 0; ++offset_to; offsets_to[i] = offset_to; } data_to.resize(offset_to); } else if (const ColumnConstString * col_from = typeid_cast(&*block.getByPosition(arguments[0]).column)) { const String & s = col_from->getData(); size_t bytes_to_copy = s.size(); while (bytes_to_copy > 0 && s[bytes_to_copy - 1] == 0) --bytes_to_copy; block.getByPosition(result).column = new ColumnConstString(col_from->size(), s.substr(0, bytes_to_copy)); } else throw Exception("Illegal column " + block.getByPosition(arguments[0]).column->getName() + " of first argument of function " + Name::name, ErrorCodes::ILLEGAL_COLUMN); } }; /// Предварительное объявление. struct NameToDate { static constexpr auto name = "toDate"; }; template class FunctionConvert : public IFunction { public: static constexpr auto name = Name::name; static IFunction * create(const Context & context) { return new FunctionConvert; } /// Получить имя функции. String getName() const override { return name; } /// Получить тип результата по типам аргументов. Если функция неприменима для данных аргументов - кинуть исключение. DataTypePtr getReturnType(const DataTypes & arguments) const override { return getReturnTypeImpl(arguments); } /// Выполнить функцию над блоком. void execute(Block & block, const ColumnNumbers & arguments, size_t result) override { IDataType * from_type = &*block.getByPosition(arguments[0]).type; if (typeid_cast(from_type)) ConvertImpl::execute(block, arguments, result); else if (typeid_cast(from_type)) ConvertImpl::execute(block, arguments, result); else if (typeid_cast(from_type)) ConvertImpl::execute(block, arguments, result); else if (typeid_cast(from_type)) ConvertImpl::execute(block, arguments, result); else if (typeid_cast(from_type)) ConvertImpl::execute(block, arguments, result); else if (typeid_cast(from_type)) ConvertImpl::execute(block, arguments, result); else if (typeid_cast(from_type)) ConvertImpl::execute(block, arguments, result); else if (typeid_cast(from_type)) ConvertImpl::execute(block, arguments, result); else if (typeid_cast(from_type)) ConvertImpl::execute(block, arguments, result); else if (typeid_cast(from_type)) ConvertImpl::execute(block, arguments, result); else if (typeid_cast(from_type)) ConvertImpl::execute(block, arguments, result); else if (typeid_cast(from_type)) ConvertImpl::execute(block, arguments, result); else if (typeid_cast(from_type)) ConvertImpl::execute(block, arguments, result); else if (typeid_cast(from_type)) ConvertImpl::execute(block, arguments, result); else if (typeid_cast(from_type)) ConvertImpl::execute(block, arguments, result); else if (typeid_cast(from_type)) ConvertImpl::execute(block, arguments, result); else throw Exception("Illegal type " + block.getByPosition(arguments[0]).type->getName() + " of argument of function " + getName(), ErrorCodes::ILLEGAL_TYPE_OF_ARGUMENT); } bool hasInformationAboutMonotonicity() const override { return Monotonic::has(); } Monotonicity getMonotonicityForRange(const IDataType & type, const Field & left, const Field & right) const override { return Monotonic::get(type, left, right); } private: template DataTypePtr getReturnTypeImpl(const DataTypes & arguments, typename std::enable_if::value || std::is_same::value || std::is_same::value)>::type * = nullptr) const { if (arguments.size() != 1) throw Exception("Number of arguments for function " + getName() + " doesn't match: passed " + toString(arguments.size()) + ", should be 1.", ErrorCodes::NUMBER_OF_ARGUMENTS_DOESNT_MATCH); return new ToDataType; } template DataTypePtr getReturnTypeImpl(const DataTypes & arguments, typename std::enable_if::value>::type * = nullptr) const { if ((arguments.size() < 1) || (arguments.size() > 2)) throw Exception("Number of arguments for function " + getName() + " doesn't match: passed " + toString(arguments.size()) + ", should be 1 or 2.", ErrorCodes::NUMBER_OF_ARGUMENTS_DOESNT_MATCH); if (typeid_cast(&*arguments[0]) == nullptr) { if (arguments.size() != 1) throw Exception("Number of arguments for function " + getName() + " doesn't match: passed " + toString(arguments.size()) + ", should be 1.", ErrorCodes::NUMBER_OF_ARGUMENTS_DOESNT_MATCH); } else if ((arguments.size() == 2) && (typeid_cast(&*arguments[1]) == nullptr)) { throw Exception{ "Illegal type " + arguments[1]->getName() + " of argument of function " + getName(), ErrorCodes::ILLEGAL_TYPE_OF_ARGUMENT }; } return new ToDataType2; } template DataTypePtr getReturnTypeImpl(const DataTypes & arguments, typename std::enable_if::value, void>::type * = nullptr) const { if ((arguments.size() < 1) || (arguments.size() > 2)) throw Exception("Number of arguments for function " + getName() + " doesn't match: passed " + toString(arguments.size()) + ", should be 1 or 2.", ErrorCodes::NUMBER_OF_ARGUMENTS_DOESNT_MATCH); if (typeid_cast(&*arguments[0]) == nullptr) { if (arguments.size() != 1) throw Exception("Number of arguments for function " + getName() + " doesn't match: passed " + toString(arguments.size()) + ", should be 1.", ErrorCodes::NUMBER_OF_ARGUMENTS_DOESNT_MATCH); } else if ((arguments.size() == 2) && (typeid_cast(&*arguments[1]) == nullptr)) {F throw Exception{ "Illegal type " + arguments[1]->getName() + " of argument of function " + getName(), ErrorCodes::ILLEGAL_TYPE_OF_ARGUMENT }; } return new ToDataType2; } template DataTypePtr getReturnTypeImpl(const DataTypes & arguments, typename std::enable_if::value>::type * = nullptr) const { if ((arguments.size() < 1) || (arguments.size() > 2)) throw Exception("Number of arguments for function " + getName() + " doesn't match: passed " + toString(arguments.size()) + ", should be 1 or 2.", ErrorCodes::NUMBER_OF_ARGUMENTS_DOESNT_MATCH); if ((arguments.size() == 2) && (typeid_cast(&*arguments[1]) == nullptr)) { throw Exception{ "Illegal type " + arguments[1]->getName() + " of 2nd argument of function " + getName(), ErrorCodes::ILLEGAL_TYPE_OF_ARGUMENT }; } return new ToDataType2; } }; /** Преобразование в строку фиксированной длины реализовано только из строк. */ class FunctionToFixedString : public IFunction { public: static constexpr auto name = "toFixedString"; static IFunction * create(const Context & context) { return new FunctionToFixedString; }; /// Получить имя функции. String getName() const override { return name; } /** Получить тип результата по типам аргументов и значениям константных аргументов. * Если функция неприменима для данных аргументов - кинуть исключение. * Для неконстантных столбцов arguments[i].column = nullptr. */ void getReturnTypeAndPrerequisites(const ColumnsWithTypeAndName & arguments, DataTypePtr & out_return_type, std::vector & out_prerequisites) override { if (arguments.size() != 2) throw Exception("Number of arguments for function " + getName() + " doesn't match: passed " + toString(arguments.size()) + ", should be 2.", ErrorCodes::NUMBER_OF_ARGUMENTS_DOESNT_MATCH); if (!arguments[1].column) throw Exception("Second argument for function " + getName() + " must be constant", ErrorCodes::ILLEGAL_COLUMN); if (!typeid_cast(arguments[0].type.get()) && !typeid_cast(arguments[0].type.get())) throw Exception(getName() + " is only implemented for types String and FixedString", ErrorCodes::NOT_IMPLEMENTED); const size_t n = getSize(arguments[1]); out_return_type = new DataTypeFixedString(n); } /// Выполнить функцию над блоком. void execute(Block & block, const ColumnNumbers & arguments, const size_t result) override { const auto n = getSize(block.getByPosition(arguments[1])); return execute(block, arguments, result, n); } static void execute(Block & block, const ColumnNumbers & arguments, const size_t result, const size_t n) { const auto & column = block.getByPosition(arguments[0]).column; if (const auto column_const = typeid_cast(&*column)) { if (column_const->getData().size() > n) throw Exception("String too long for type FixedString(" + toString(n) + ")", ErrorCodes::TOO_LARGE_STRING_SIZE); auto resized_string = column_const->getData(); resized_string.resize(n); block.getByPosition(result).column = new ColumnConst{ column_const->size(), std::move(resized_string), new DataTypeFixedString(n) }; } else if (const auto column_string = typeid_cast(&*column)) { const auto column_fixed = new ColumnFixedString(n); ColumnPtr result_ptr = column_fixed; auto & out_chars = column_fixed->getChars(); const auto & in_chars = column_string->getChars(); const auto & in_offsets = column_string->getOffsets(); out_chars.resize_fill(in_offsets.size() * n); for (size_t i = 0; i < in_offsets.size(); ++i) { const size_t off = i ? in_offsets[i - 1] : 0; const size_t len = in_offsets[i] - off - 1; if (len > n) throw Exception("String too long for type FixedString(" + toString(n) + ")", ErrorCodes::TOO_LARGE_STRING_SIZE); memcpy(&out_chars[i * n], &in_chars[off], len); } block.getByPosition(result).column = result_ptr; } else if (const auto column_fixed_string = typeid_cast(column.get())) { const auto src_n = column_fixed_string->getN(); if (src_n > n) throw Exception{ "String too long for type FixedString(" + toString(n) + ")", ErrorCodes::TOO_LARGE_STRING_SIZE }; const auto column_fixed = new ColumnFixedString{n}; block.getByPosition(result).column = column_fixed; auto & out_chars = column_fixed->getChars(); const auto & in_chars = column_fixed_string->getChars(); const auto size = column_fixed_string->size(); out_chars.resize_fill(size * n); for (const auto i : ext::range(0, size)) memcpy(&out_chars[i * n], &in_chars[i * src_n], src_n); } else throw Exception("Unexpected column: " + column->getName(), ErrorCodes::ILLEGAL_COLUMN); } private: template bool getSizeTyped(const ColumnWithTypeAndName & column, size_t & out_size) { if (!typeid_cast::Type *>(&*column.type)) return false; const ColumnConst * column_const = typeid_cast *>(&*column.column); if (!column_const) throw Exception("Unexpected type of column for FixedString length: " + column.column->getName(), ErrorCodes::ILLEGAL_COLUMN); T s = column_const->getData(); if (s <= 0) throw Exception("FixedString length must be positive (unlike " + toString(s) + ")", ErrorCodes::ILLEGAL_COLUMN); out_size = static_cast(s); return true; } size_t getSize(const ColumnWithTypeAndName & column) { size_t res; if (getSizeTyped(column, res) || getSizeTyped(column, res) || getSizeTyped(column, res) || getSizeTyped(column, res) || getSizeTyped< Int8 >(column, res) || getSizeTyped< Int16>(column, res) || getSizeTyped< Int32>(column, res) || getSizeTyped< Int64>(column, res)) return res; throw Exception("Length of FixedString must be integer; got " + column.type->getName(), ErrorCodes::ILLEGAL_TYPE_OF_ARGUMENT); } }; /// Монотонность. struct PositiveMonotonicity { static bool has() { return true; } static IFunction::Monotonicity get(const IDataType & type, const Field & left, const Field & right) { return { true }; } }; template struct ToIntMonotonicity { static bool has() { return true; } template static UInt64 divideByRangeOfType(typename std::enable_if_t x) { return x >> (sizeof(T) * 8); }; template static UInt64 divideByRangeOfType(typename std::enable_if_t x) { return 0; }; static IFunction::Monotonicity get(const IDataType & type, const Field & left, const Field & right) { size_t size_of_type = type.getSizeOfField(); /// Если тип расширяется, то функция монотонна. if (sizeof(T) > size_of_type) return { true }; /// Если тип совпадает - тоже. if (typeid_cast::Type *>(&type)) return { true }; /// В других случаях, для неограниченного диапазона не знаем, будет ли функция монотонной. if (left.isNull() || right.isNull()) return {}; /// Если преобразуем из float, то аргументы должны помещаться в тип результата. if (typeid_cast(&type) || typeid_cast(&type)) { Float64 left_float = left.get(); Float64 right_float = right.get(); if (left_float >= std::numeric_limits::min() && left_float <= std::numeric_limits::max() && right_float >= std::numeric_limits::min() && right_float <= std::numeric_limits::max()) return { true }; return {}; } /// Если меняем знаковость типа или преобразуем из даты, даты-времени, то аргумент должен быть из одной половинки. /// На всякий случай, в остальных случаях тоже будем этого требовать. if ((left.get() >= 0) != (right.get() >= 0)) return {}; /// Если уменьшаем тип, то все биты кроме тех, которые в него помещаются, должны совпадать. if (divideByRangeOfType(left.get()) != divideByRangeOfType(right.get())) return {}; return { true }; } }; /** Монотонность для функции toString определяем, в основном, для тестовых целей. * Всерьёз вряд ли кто-нибудь рассчитывает на оптимизацию запросов с условиями toString(CounterID) = 34. */ struct ToStringMonotonicity { static bool has() { return true; } static IFunction::Monotonicity get(const IDataType & type, const Field & left, const Field & right) { IFunction::Monotonicity positive = { .is_monotonic = true, .is_positive = true }; IFunction::Monotonicity not_monotonic; /// Функция toString монотонна, если аргумент - Date или DateTime, или неотрицательные числа с одинаковым количеством знаков. if (typeid_cast(&type) || typeid_cast(&type)) return positive; if (left.isNull() || right.isNull()) return {}; if (left.getType() == Field::Types::UInt64 && right.getType() == Field::Types::UInt64) { return (left.get() == 0 && right.get() == 0) || (floor(log10(left.get())) == floor(log10(right.get()))) ? positive : not_monotonic; } if (left.getType() == Field::Types::Int64 && right.getType() == Field::Types::Int64) { return (left.get() == 0 && right.get() == 0) || (left.get() > 0 && right.get() > 0 && floor(log10(left.get())) == floor(log10(right.get()))) ? positive : not_monotonic; } return not_monotonic; } }; struct NameToUInt8 { static constexpr auto name = "toUInt8"; }; struct NameToUInt16 { static constexpr auto name = "toUInt16"; }; struct NameToUInt32 { static constexpr auto name = "toUInt32"; }; struct NameToUInt64 { static constexpr auto name = "toUInt64"; }; struct NameToInt8 { static constexpr auto name = "toInt8"; }; struct NameToInt16 { static constexpr auto name = "toInt16"; }; struct NameToInt32 { static constexpr auto name = "toInt32"; }; struct NameToInt64 { static constexpr auto name = "toInt64"; }; struct NameToFloat32 { static constexpr auto name = "toFloat32"; }; struct NameToFloat64 { static constexpr auto name = "toFloat64"; }; struct NameToDateTime { static constexpr auto name = "toDateTime"; }; struct NameToString { static constexpr auto name = "toString"; }; typedef FunctionConvert> FunctionToUInt8; typedef FunctionConvert> FunctionToUInt16; typedef FunctionConvert> FunctionToUInt32; typedef FunctionConvert> FunctionToUInt64; typedef FunctionConvert> FunctionToInt8; typedef FunctionConvert> FunctionToInt16; typedef FunctionConvert> FunctionToInt32; typedef FunctionConvert> FunctionToInt64; typedef FunctionConvert FunctionToFloat32; typedef FunctionConvert FunctionToFloat64; typedef FunctionConvert> FunctionToDate; typedef FunctionConvert> FunctionToDateTime; typedef FunctionConvert FunctionToString; typedef FunctionConvert> FunctionToUnixTimestamp; template struct ToPrimitiveType; template <> struct ToPrimitiveType { using Type = FunctionToUInt8; }; template <> struct ToPrimitiveType { using Type = FunctionToUInt16; }; template <> struct ToPrimitiveType { using Type = FunctionToUInt32; }; template <> struct ToPrimitiveType { using Type = FunctionToUInt64; }; template <> struct ToPrimitiveType { using Type = FunctionToInt8; }; template <> struct ToPrimitiveType { using Type = FunctionToInt16; }; template <> struct ToPrimitiveType { using Type = FunctionToInt32; }; template <> struct ToPrimitiveType { using Type = FunctionToInt64; }; template <> struct ToPrimitiveType { using Type = FunctionToFloat32; }; template <> struct ToPrimitiveType { using Type = FunctionToFloat64; }; template <> struct ToPrimitiveType { using Type = FunctionToString; }; class FunctionCast final : public IFunction { using WrapperType = std::function; const Context & context; WrapperType wrapper_function; FunctionCast(const Context & context) : context(context) {} template auto createWrapper(const DataTypePtr & from_type) { std::shared_ptr function{static_cast(FunctionType::create(context))}; /// Check conversion using underlying function (void) function->getReturnType({ from_type }); return [function] (Block & block, const ColumnNumbers & arguments, const size_t result) { function->execute(block, arguments, result); }; } static auto createFixedStringWrapper(const DataTypePtr & from_type, const size_t N) { if (!typeid_cast(from_type.get()) && !typeid_cast(from_type.get())) throw Exception{ "CAST AS FixedString is only implemented for types String and FixedString", ErrorCodes::NOT_IMPLEMENTED }; return [N] (Block & block, const ColumnNumbers & arguments, const size_t result) { FunctionToFixedString::execute(block, arguments, result, N); }; } auto createArrayWrapper(const DataTypePtr & from_type_untyped, const DataTypeArray * to_type) { DataTypePtr from_nested_type, to_nested_type; auto from_type = typeid_cast(from_type_untyped.get()); /// get the most nested type while (from_type && to_type) { from_nested_type = from_type->getNestedType(); to_nested_type = to_type->getNestedType(); from_type = typeid_cast(from_nested_type.get()); to_type = typeid_cast(to_nested_type.get()); } /// both from_type and to_type should be nullptr now is array types had same dimensions if (from_type || to_type) throw Exception{ "CAST AS Array can only be performed between same-dimensional array types", ErrorCodes::TYPE_MISMATCH }; /// Prepare nested type conversion const auto nested_function = prepare(from_nested_type, to_nested_type.get()); return [nested_function, from_nested_type, to_nested_type] ( Block & block, const ColumnNumbers & arguments, const size_t result) { auto array_arg = block.getByPosition(arguments.front()); /// @todo add const variant which retains array constness if (const auto col_const_array = typeid_cast(array_arg.column.get())) array_arg.column = col_const_array->convertToFullColumn(); if (auto col_array = typeid_cast(array_arg.column.get())) { auto res = new ColumnArray{nullptr, col_array->getOffsetsColumn()}; block.getByPosition(result).column = res; /// get the most nested column while (const auto nested_col_array = typeid_cast(col_array->getDataPtr().get())) { /// create new level of array, copy offsets res->getDataPtr() = new ColumnArray{nullptr, nested_col_array->getOffsetsColumn()}; res = static_cast(res->getDataPtr().get()); col_array = nested_col_array; } /// create block for converting nested column containing original and result columns Block nested_block{ { col_array->getDataPtr(), from_nested_type, "" }, { nullptr, to_nested_type, "" } }; const auto nested_result = 1; /// convert nested column nested_function(nested_block, {0 }, nested_result); /// set converted nested column to result res->getDataPtr() = nested_block.getByPosition(nested_result).column; } else throw Exception{ "Illegal column " + array_arg.column->getName() + " for function CAST AS Array", ErrorCodes::LOGICAL_ERROR }; }; } auto createTupleWrapper(const DataTypePtr & from_type_untyped, const DataTypeTuple * to_type) { const auto from_type = typeid_cast(from_type_untyped.get()); if (!from_type) throw Exception{ "CAST AS Tuple can only be performed between tuple types.\nLeft type: " + from_type_untyped->getName() + ", right type: " + to_type->getName(), ErrorCodes::TYPE_MISMATCH }; if (from_type->getElements().size() != to_type->getElements().size()) throw Exception{ "CAST AS Tuple can only be performed between tuple types with the same number of elements.\n" "Left type: " + from_type->getName() + ", right type: " + to_type->getName(), ErrorCodes::TYPE_MISMATCH }; const auto & from_element_types = from_type->getElements(); const auto & to_element_types = to_type->getElements(); std::vector element_wrappers; element_wrappers.reserve(from_element_types.size()); /// Create conversion wrapper for each element in tuple for (const auto & idx_type : ext::enumerate(from_type->getElements())) element_wrappers.push_back(prepare(idx_type.second, to_element_types[idx_type.first].get())); std::shared_ptr function_tuple{static_cast(FunctionTuple::create(context))}; return [element_wrappers, function_tuple, from_element_types, to_element_types] (Block & block, const ColumnNumbers & arguments, const size_t result) { const auto col = block.getByPosition(arguments.front()).column.get(); /// copy tuple elements to a separate block Block element_block; /// @todo retain constness if (const auto column_tuple = typeid_cast(col)) element_block = column_tuple->getData(); else if (const auto column_const_tuple = typeid_cast(col)) element_block = static_cast(*column_const_tuple->convertToFullColumn()).getData(); /// create columns for converted elements for (const auto & to_element_type : to_element_types) element_block.insert({ nullptr, to_element_type, "" }); /// store position for converted tuple const auto converted_tuple_pos = element_block.columns(); /// insert column for converted tuple element_block.insert({ nullptr, new DataTypeTuple{to_element_types}, "" }); const auto converted_element_offset = from_element_types.size(); /// invoke conversion for each element for (const auto & idx_element_wrapper : ext::enumerate(element_wrappers)) idx_element_wrapper.second(element_block, { idx_element_wrapper.first }, converted_element_offset + idx_element_wrapper.first); /// form tuple from converted elements using FunctionTuple function_tuple->execute(element_block, ext::collection_cast(ext::range(converted_element_offset, 2 * converted_element_offset)), converted_tuple_pos); /// copy FunctionTuple's result from element_block to resulting block block.getByPosition(result).column = element_block.getByPosition(converted_tuple_pos).column; }; } template WrapperType createEnumWrapper(const DataTypePtr & from_type, const DataTypeEnum * to_type) { using EnumType = DataTypeEnum; if (const auto from_enum8 = typeid_cast(from_type.get())) checkEnumToEnumConversion(from_enum8, to_type); else if (const auto from_enum16 = typeid_cast(from_type.get())) checkEnumToEnumConversion(from_enum16, to_type); if (typeid_cast(from_type.get())) return createStringToEnumWrapper(); else if (typeid_cast(from_type.get())) return createStringToEnumWrapper(); else if (from_type->behavesAsNumber()) { std::shared_ptr function{static_cast(Function::create(context))}; /// Check conversion using underlying function (void) function->getReturnType({ from_type }); return [function] (Block & block, const ColumnNumbers & arguments, const size_t result) { function->execute(block, arguments, result); }; } else throw Exception{ "Conversion from " + from_type->getName() + " to " + to_type->getName() + " is not supported", ErrorCodes::CANNOT_CONVERT_TYPE }; } template void checkEnumToEnumConversion(const EnumTypeFrom * const from_type, const EnumTypeTo * const to_type) { const auto & from_values = from_type->getValues(); const auto & to_values = to_type->getValues(); using ValueType = std::common_type_t; using NameValuePair = std::pair; struct CompareNameValuePairSecond final { bool operator()(const NameValuePair & lhs, const NameValuePair & rhs) const { return lhs.second < rhs.second; } }; std::set isection; std::set_intersection(std::begin(from_values), std::end(from_values), std::begin(to_values), std::end(to_values), std::inserter(isection, std::end(isection)), CompareNameValuePairSecond{}); if (!isection.empty()) for (const auto & name_value : isection) { const auto & old_name = from_type->getNameForValue(name_value.second); const auto & new_name = to_type->getNameForValue(name_value.second); if (old_name != new_name) throw Exception{ "Enum conversion changes name for value " + toString(name_value.second) + " from " + old_name + " to " + new_name, ErrorCodes::CANNOT_CONVERT_TYPE }; } }; template auto createStringToEnumWrapper() { return [] (Block & block, const ColumnNumbers & arguments, const size_t result) { const auto first_col = block.getByPosition(arguments.front()).column.get(); auto & col_with_type_and_name = block.getByPosition(result); auto & result_col = col_with_type_and_name.column; const auto & result_type = typeid_cast(*col_with_type_and_name.type); if (const auto col = typeid_cast(first_col)) { const auto size = col->size(); const auto res = result_type.createColumn(); auto & out_data = static_cast(*result_col).getData(); out_data.resize(size); for (const auto i : ext::range(0, size)) out_data[i] = result_type.getValue(col->getDataAt(i).toString()); result_col = res; } else if (const auto const_col = typeid_cast(first_col)) { result_col = result_type.createConstColumn(const_col->size(), nearestFieldType(result_type.getValue(const_col->getData()))); } else throw Exception{ "Unexpected column " + first_col->getName() + " as first argument of function " + name, ErrorCodes::LOGICAL_ERROR }; }; } WrapperType prepare(const DataTypePtr & from_type, const IDataType * to_type) { if (typeid_cast(to_type)) return createWrapper(from_type); else if (typeid_cast(to_type)) return createWrapper(from_type); else if (typeid_cast(to_type)) return createWrapper(from_type); else if (typeid_cast(to_type)) return createWrapper(from_type); else if (typeid_cast(to_type)) return createWrapper(from_type); else if (typeid_cast(to_type)) return createWrapper(from_type); else if (typeid_cast(to_type)) return createWrapper(from_type); else if (typeid_cast(to_type)) return createWrapper(from_type); else if (typeid_cast(to_type)) return createWrapper(from_type); else if (typeid_cast(to_type)) return createWrapper(from_type); else if (typeid_cast(to_type)) return createWrapper(from_type); else if (typeid_cast(to_type)) return createWrapper(from_type); else if (typeid_cast(to_type)) return createWrapper(from_type); else if (const auto type_fixed_string = typeid_cast(to_type)) return createFixedStringWrapper(from_type, type_fixed_string->getN()); else if (const auto type_array = typeid_cast(to_type)) return createArrayWrapper(from_type, type_array); else if (const auto type_tuple = typeid_cast(to_type)) return createTupleWrapper(from_type, type_tuple); else if (const auto type_enum = typeid_cast(to_type)) return createEnumWrapper::Type>(from_type, type_enum); else if (const auto type_enum = typeid_cast(to_type)) return createEnumWrapper::Type>(from_type, type_enum); throw Exception{ "Conversion from " + from_type->getName() + " to " + to_type->getName() + " is not supported", ErrorCodes::CANNOT_CONVERT_TYPE }; } public: static constexpr auto name = "CAST"; static IFunction * create(const Context & context) { return new FunctionCast{context}; } String getName() const override { return name; } void getReturnTypeAndPrerequisites( const ColumnsWithTypeAndName & arguments, DataTypePtr & out_return_type, std::vector & out_prerequisites) override { if (arguments.size() != 2) throw Exception("Number of arguments for function " + getName() + " doesn't match: passed " + toString(arguments.size()) + ", should be 2.", ErrorCodes::NUMBER_OF_ARGUMENTS_DOESNT_MATCH); const auto type_col = typeid_cast(arguments.back().column.get()); if (!type_col) throw Exception("Second argument to " + getName() + " must be a constant string describing type", ErrorCodes::ILLEGAL_TYPE_OF_ARGUMENT); out_return_type = DataTypeFactory::instance().get(type_col->getData()); wrapper_function = prepare(arguments.front().type, out_return_type.get()); } void execute(Block & block, const ColumnNumbers & arguments, const size_t result) override { /// drop second argument, pass others ColumnNumbers new_arguments{arguments.front()}; if (arguments.size() > 2) new_arguments.insert(std::end(new_arguments), std::next(std::begin(arguments), 2), std::end(arguments)); wrapper_function(block, new_arguments, result); } }; }