#pragma once #include #include namespace DB { /** Transform-type wrapper for DateTime64, simplifies DateTime64 support for given Transform. * * Depending on what overloads of Transform::execute() are available, when called with DateTime64 value, * invokes Transform::execute() with either: * * whole part of DateTime64 value, discarding fractional part (1) * * DateTime64 value and scale factor (2) * * DateTime64 broken down to components, result of execute is then re-assembled back into DateTime64 value (3) * * Suitable Transform-types are commonly used in Date/DateTime manipulation functions, * and should implement static (or const) function with following signatures: * 1: * R execute(Int64 whole_value, ... ) * 2: * R execute(DateTime64 value, Int64 scale_multiplier, ... ) * 3: * R execute(DecimalUtils::DecimalComponents components, ... ) * * Where R could be of arbitrary type, in case of (3) if R is DecimalUtils::DecimalComponents, result is re-assembed back into DateTime64. */ template class TransformDateTime64 { private: // Detect if Transform::execute is const or static method // with signature defined by template args (ignoring result type). template struct TransformHasExecuteOverload : std::false_type {}; template struct TransformHasExecuteOverload().execute(std::declval()...))>, Args...> : std::true_type {}; template static constexpr bool TransformHasExecuteOverload_v = TransformHasExecuteOverload::value; public: static constexpr auto name = Transform::name; // non-explicit constructor to allow creating from scale value (or with no scale at all), indispensable in some contexts. TransformDateTime64(UInt32 scale_ = 0) /// NOLINT : scale_multiplier(DecimalUtils::scaleMultiplier(scale_)) {} TransformDateTime64(DateTime64::NativeType scale_multiplier_ = 1) /// NOLINT(google-explicit-constructor) : scale_multiplier(scale_multiplier_) {} template auto NO_SANITIZE_UNDEFINED execute(const DateTime64 & t, Args && ... args) const { /// Type conversion from float to integer may be required. /// We are Ok with implementation specific result for out of range and denormals conversion. if constexpr (TransformHasExecuteOverload_v) { return wrapped_transform.execute(t, scale_multiplier, std::forward(args)...); } else if constexpr (TransformHasExecuteOverload_v, Args...>) { auto components = DecimalUtils::splitWithScaleMultiplier(t, scale_multiplier); const auto result = wrapped_transform.execute(components, std::forward(args)...); using ResultType = std::decay_t; if constexpr (std::is_same_v, ResultType>) { return DecimalUtils::decimalFromComponentsWithMultiplier(result, scale_multiplier); } else { return result; } } else { auto components = DecimalUtils::splitWithScaleMultiplier(t, scale_multiplier); if (t.value < 0 && components.fractional) --components.whole; return wrapped_transform.execute(static_cast(components.whole), std::forward(args)...); } } template requires(!std::same_as) auto execute(const T & t, Args &&... args) const { return wrapped_transform.execute(t, std::forward(args)...); } template auto NO_SANITIZE_UNDEFINED executeExtendedResult(const DateTime64 & t, Args && ... args) const { /// Type conversion from float to integer may be required. /// We are Ok with implementation specific result for out of range and denormals conversion. if constexpr (TransformHasExecuteOverload_v) { return wrapped_transform.executeExtendedResult(t, scale_multiplier, std::forward(args)...); } else if constexpr (TransformHasExecuteOverload_v, Args...>) { auto components = DecimalUtils::splitWithScaleMultiplier(t, scale_multiplier); const auto result = wrapped_transform.executeExtendedResult(components, std::forward(args)...); using ResultType = std::decay_t; if constexpr (std::is_same_v, ResultType>) { return DecimalUtils::decimalFromComponentsWithMultiplier(result, scale_multiplier); } else { return result; } } else { const auto components = DecimalUtils::splitWithScaleMultiplier(t, scale_multiplier); return wrapped_transform.executeExtendedResult(static_cast(components.whole), std::forward(args)...); } } template requires (!std::same_as) auto executeExtendedResult(const T & t, Args && ... args) const { return wrapped_transform.executeExtendedResult(t, std::forward(args)...); } DateTime64::NativeType getScaleMultiplier() const { return scale_multiplier; } private: DateTime64::NativeType scale_multiplier = 1; Transform wrapped_transform = {}; }; }