#pragma once #include #include #include namespace DB { /** Time window functions: * * tumble(time_attr, interval [, timezone]) * * tumbleStart(window_id) * * tumbleStart(time_attr, interval [, timezone]) * * tumbleEnd(window_id) * * tumbleEnd(time_attr, interval [, timezone]) * * hop(time_attr, hop_interval, window_interval [, timezone]) * * hopStart(window_id) * * hopStart(time_attr, hop_interval, window_interval [, timezone]) * * hopEnd(window_id) * * hopEnd(time_attr, hop_interval, window_interval [, timezone]) * */ enum TimeWindowFunctionName { TUMBLE, TUMBLE_START, TUMBLE_END, HOP, HOP_START, HOP_END, WINDOW_ID }; template struct ToStartOfTransform; #define TRANSFORM_DATE(INTERVAL_KIND) \ template <> \ struct ToStartOfTransform \ { \ static auto execute(UInt32 t, UInt64 delta, const DateLUTImpl & time_zone) \ { \ return time_zone.toStartOf##INTERVAL_KIND##Interval(time_zone.toDayNum(t), delta); \ } \ }; TRANSFORM_DATE(Year) TRANSFORM_DATE(Quarter) TRANSFORM_DATE(Month) TRANSFORM_DATE(Week) #undef TRANSFORM_DATE template <> struct ToStartOfTransform { static UInt32 execute(UInt32 t, UInt64 delta, const DateLUTImpl & time_zone) { return time_zone.toStartOfDayInterval(time_zone.toDayNum(t), delta); } }; #define TRANSFORM_TIME(INTERVAL_KIND) \ template <> \ struct ToStartOfTransform \ { \ static UInt32 execute(UInt32 t, UInt64 delta, const DateLUTImpl & time_zone) \ { \ return time_zone.toStartOf##INTERVAL_KIND##Interval(t, delta); \ } \ }; TRANSFORM_TIME(Hour) TRANSFORM_TIME(Minute) TRANSFORM_TIME(Second) #undef TRANSFORM_TIME #define TRANSFORM_SUBSECONDS(INTERVAL_KIND, DEF_SCALE) \ template<> \ struct ToStartOfTransform \ { \ static Int64 execute(Int64 t, UInt64 delta, const UInt32 scale) \ { \ if (scale <= DEF_SCALE) \ { \ auto val = t * DecimalUtils::scaleMultiplier(DEF_SCALE - scale); \ if (delta == 1) \ return val; \ else \ return val - (val % delta); \ } \ else \ { \ return t - (t % (delta * DecimalUtils::scaleMultiplier(scale - DEF_SCALE))) ; \ } \ } \ }; TRANSFORM_SUBSECONDS(Millisecond, 3) TRANSFORM_SUBSECONDS(Microsecond, 6) TRANSFORM_SUBSECONDS(Nanosecond, 9) #undef TRANSFORM_SUBSECONDS template struct AddTime; #define ADD_DATE(INTERVAL_KIND) \ template <> \ struct AddTime \ { \ static inline auto execute(UInt16 d, UInt64 delta, const DateLUTImpl & time_zone) \ { \ return time_zone.add##INTERVAL_KIND##s(ExtendedDayNum(d), delta); \ } \ }; ADD_DATE(Year) ADD_DATE(Quarter) ADD_DATE(Month) #undef ADD_DATE template <> struct AddTime { static inline NO_SANITIZE_UNDEFINED ExtendedDayNum execute(UInt16 d, UInt64 delta, const DateLUTImpl &) { return ExtendedDayNum(d + delta * 7);} }; #define ADD_TIME(INTERVAL_KIND, INTERVAL) \ template <> \ struct AddTime \ { \ static inline NO_SANITIZE_UNDEFINED UInt32 execute(UInt32 t, Int64 delta, const DateLUTImpl &) { return t + delta * INTERVAL; } \ }; ADD_TIME(Day, 86400) ADD_TIME(Hour, 3600) ADD_TIME(Minute, 60) ADD_TIME(Second, 1) #undef ADD_TIME #define ADD_SUBSECONDS(INTERVAL_KIND, DEF_SCALE) \ template <> \ struct AddTime \ { \ static inline NO_SANITIZE_UNDEFINED Int64 execute(Int64 t, UInt64 delta, const UInt32 scale) \ { \ if (scale < DEF_SCALE) \ { \ return t + delta * DecimalUtils::scaleMultiplier(DEF_SCALE - scale); \ } \ else \ return t + delta * DecimalUtils::scaleMultiplier(scale - DEF_SCALE); \ } \ }; ADD_SUBSECONDS(Millisecond, 3) ADD_SUBSECONDS(Microsecond, 6) ADD_SUBSECONDS(Nanosecond, 9) #undef ADD_SUBSECONDS template struct TimeWindowImpl { static constexpr auto name = "UNKNOWN"; static DataTypePtr getReturnType(const ColumnsWithTypeAndName & arguments, const String & function_name); static ColumnPtr dispatchForColumns(const ColumnsWithTypeAndName & arguments, const String & function_name); }; template class FunctionTimeWindow : public IFunction { public: static constexpr auto name = TimeWindowImpl::name; static FunctionPtr create(ContextPtr) { return std::make_shared(); } String getName() const override { return name; } bool isVariadic() const override { return true; } size_t getNumberOfArguments() const override { return 0; } bool useDefaultImplementationForConstants() const override { return true; } ColumnNumbers getArgumentsThatAreAlwaysConstant() const override { return {1, 2, 3}; } bool isSuitableForShortCircuitArgumentsExecution(const DataTypesWithConstInfo &) const override { return true; } DataTypePtr getReturnTypeImpl(const ColumnsWithTypeAndName & arguments) const override; ColumnPtr executeImpl(const ColumnsWithTypeAndName & arguments, const DataTypePtr & /*result_type*/, size_t /*input_rows_count*/) const override; }; using FunctionTumble = FunctionTimeWindow; using FunctionTumbleStart = FunctionTimeWindow; using FunctionTumbleEnd = FunctionTimeWindow; using FunctionHop = FunctionTimeWindow; using FunctionWindowId = FunctionTimeWindow; using FunctionHopStart = FunctionTimeWindow; using FunctionHopEnd = FunctionTimeWindow; }