#pragma once #include namespace DB { namespace ErrorCodes { extern const int ILLEGAL_TYPE_OF_ARGUMENT; } template class FunctionDateOrDateTimeToDateOrDate32 : public IFunctionDateOrDateTime { private: const bool enable_extended_results_for_datetime_functions = false; public: static FunctionPtr create(ContextPtr context_) { return std::make_shared(context_); } explicit FunctionDateOrDateTimeToDateOrDate32(ContextPtr context_) : enable_extended_results_for_datetime_functions(context_->getSettingsRef().enable_extended_results_for_datetime_functions) { } DataTypePtr getReturnTypeImpl(const ColumnsWithTypeAndName & arguments) const override { this->checkArguments(arguments, /*is_result_type_date_or_date32*/ true); const IDataType * from_type = arguments[0].type.get(); WhichDataType which(from_type); /// If the time zone is specified but empty, throw an exception. /// only validate the time_zone part if the number of arguments is 2. if ((which.isDateTime() || which.isDateTime64()) && arguments.size() == 2 && extractTimeZoneNameFromFunctionArguments(arguments, 1, 0).empty()) throw Exception(ErrorCodes::ILLEGAL_TYPE_OF_ARGUMENT, "Function {} supports a 2nd argument (optional) that must be a valid time zone", this->getName()); if ((which.isDate32() || which.isDateTime64()) && enable_extended_results_for_datetime_functions) return std::make_shared(); else return std::make_shared(); } ColumnPtr executeImpl(const ColumnsWithTypeAndName & arguments, const DataTypePtr & result_type, size_t input_rows_count) const override { const IDataType * from_type = arguments[0].type.get(); WhichDataType which(from_type); if (which.isDate()) return DateTimeTransformImpl::execute(arguments, result_type, input_rows_count); else if (which.isDate32()) { if (enable_extended_results_for_datetime_functions) return DateTimeTransformImpl::execute(arguments, result_type, input_rows_count); else return DateTimeTransformImpl::execute(arguments, result_type, input_rows_count); } else if (which.isDateTime()) return DateTimeTransformImpl::execute(arguments, result_type, input_rows_count); else if (which.isDateTime64()) { const auto scale = static_cast(from_type)->getScale(); const TransformDateTime64 transformer(scale); if (enable_extended_results_for_datetime_functions) return DateTimeTransformImpl::execute(arguments, result_type, input_rows_count, transformer); else return DateTimeTransformImpl::execute(arguments, result_type, input_rows_count, transformer); } else throw Exception(ErrorCodes::ILLEGAL_TYPE_OF_ARGUMENT, "Illegal type {} of argument of function {}", arguments[0].type->getName(), this->getName()); } }; }