2023-09-08 13:27:24 +00:00
|
|
|
#include <Columns/ColumnsNumber.h>
|
|
|
|
#include <DataTypes/DataTypeDate.h>
|
|
|
|
#include <DataTypes/DataTypeDate32.h>
|
|
|
|
#include <DataTypes/DataTypesNumber.h>
|
|
|
|
#include <Functions/DateTimeTransforms.h>
|
|
|
|
#include <Functions/FunctionFactory.h>
|
|
|
|
#include <Functions/FunctionHelpers.h>
|
|
|
|
#include <Functions/IFunction.h>
|
2023-09-20 15:59:37 +00:00
|
|
|
#include "DataTypes/IDataType.h"
|
|
|
|
#include "Functions/TransformDateTime64.h"
|
2023-09-08 13:27:24 +00:00
|
|
|
|
|
|
|
|
|
|
|
namespace DB
|
|
|
|
{
|
|
|
|
|
|
|
|
namespace ErrorCodes
|
|
|
|
{
|
2023-09-21 14:16:03 +00:00
|
|
|
extern const int ILLEGAL_TYPE_OF_ARGUMENT;
|
2023-09-08 13:27:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
namespace
|
|
|
|
{
|
|
|
|
|
2023-09-10 12:55:15 +00:00
|
|
|
/** Returns number of days passed since 0000-01-01 */
|
2023-09-08 13:27:24 +00:00
|
|
|
class FunctionToDaysSinceYearZero : public IFunction
|
|
|
|
{
|
|
|
|
using ResultType = DataTypeUInt32;
|
2023-09-20 15:59:37 +00:00
|
|
|
using Transformer = TransformDateTime64<ToDaysSinceYearZeroImpl>;
|
2023-09-21 14:16:03 +00:00
|
|
|
|
2023-09-08 13:27:24 +00:00
|
|
|
public:
|
|
|
|
static constexpr auto name = "toDaysSinceYearZero";
|
|
|
|
static FunctionPtr create(ContextPtr context) { return std::make_shared<FunctionToDaysSinceYearZero>(context); }
|
|
|
|
|
2023-09-21 14:16:03 +00:00
|
|
|
explicit FunctionToDaysSinceYearZero(ContextPtr /*context*/) { }
|
2023-09-08 13:27:24 +00:00
|
|
|
|
|
|
|
String getName() const override { return name; }
|
|
|
|
size_t getNumberOfArguments() const override { return 1; }
|
|
|
|
bool isSuitableForShortCircuitArgumentsExecution(const DataTypesWithConstInfo & /*arguments*/) const override { return true; }
|
|
|
|
bool useDefaultImplementationForConstants() const override { return true; }
|
|
|
|
|
|
|
|
DataTypePtr getReturnTypeImpl(const ColumnsWithTypeAndName & arguments) const override
|
|
|
|
{
|
|
|
|
FunctionArgumentDescriptors mandatory_args{
|
2023-09-20 15:59:37 +00:00
|
|
|
{"date",
|
2023-09-21 14:16:03 +00:00
|
|
|
[](const IDataType & dt) { return isDateOrDate32<IDataType>(dt) || isDateTime<IDataType>(dt) || isDateTime64<IDataType>(dt); },
|
2023-09-20 15:59:37 +00:00
|
|
|
nullptr,
|
|
|
|
"Date, Date32, DateTime or DateTime64"}};
|
2023-09-08 13:27:24 +00:00
|
|
|
|
|
|
|
validateFunctionArgumentTypes(*this, arguments, mandatory_args);
|
|
|
|
|
|
|
|
return std::make_shared<DataTypeUInt32>();
|
|
|
|
}
|
|
|
|
|
|
|
|
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())
|
2023-09-21 14:16:03 +00:00
|
|
|
return DateTimeTransformImpl<DataTypeDate, ResultType, ToDaysSinceYearZeroImpl>::execute(
|
|
|
|
arguments, result_type, input_rows_count);
|
2023-09-08 13:27:24 +00:00
|
|
|
else if (which.isDate32())
|
2023-09-21 14:16:03 +00:00
|
|
|
return DateTimeTransformImpl<DataTypeDate32, ResultType, ToDaysSinceYearZeroImpl>::execute(
|
|
|
|
arguments, result_type, input_rows_count);
|
2023-09-20 15:59:37 +00:00
|
|
|
else if (which.isDateTime())
|
2023-09-21 14:16:03 +00:00
|
|
|
return DateTimeTransformImpl<DataTypeDateTime, ResultType, ToDaysSinceYearZeroImpl>::execute(
|
|
|
|
arguments, result_type, input_rows_count);
|
2023-09-20 15:59:37 +00:00
|
|
|
else if (which.isDateTime64())
|
|
|
|
{
|
|
|
|
const auto scale = static_cast<const DataTypeDateTime64 *>(from_type)->getScale();
|
|
|
|
const Transformer transformer(scale);
|
2023-09-21 14:16:03 +00:00
|
|
|
return DateTimeTransformImpl<DataTypeDateTime64, ResultType, Transformer>::execute(
|
|
|
|
arguments, result_type, input_rows_count, transformer);
|
2023-09-20 15:59:37 +00:00
|
|
|
}
|
2023-09-08 13:27:24 +00:00
|
|
|
|
2023-09-21 14:16:03 +00:00
|
|
|
throw Exception(
|
|
|
|
ErrorCodes::ILLEGAL_TYPE_OF_ARGUMENT,
|
2023-09-08 13:27:24 +00:00
|
|
|
"Illegal type {} of argument of function {}",
|
2023-09-21 14:16:03 +00:00
|
|
|
arguments[0].type->getName(),
|
|
|
|
this->getName());
|
2023-09-08 13:27:24 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
REGISTER_FUNCTION(ToDaysSinceYearZero)
|
|
|
|
{
|
2023-09-21 14:16:03 +00:00
|
|
|
factory.registerFunction<FunctionToDaysSinceYearZero>(FunctionDocumentation{
|
|
|
|
.description = R"(
|
2023-09-10 12:55:15 +00:00
|
|
|
Returns for a given date, the number of days passed since 1 January 0000 in the proleptic Gregorian calendar defined by ISO 8601.
|
|
|
|
The calculation is the same as in MySQL's TO_DAYS() function.
|
|
|
|
)",
|
2023-09-21 14:16:03 +00:00
|
|
|
.examples{{"typical", "SELECT toDaysSinceYearZero(toDate('2023-09-08'))", "713569"}},
|
|
|
|
.categories{"Dates and Times"}});
|
2023-09-10 12:55:15 +00:00
|
|
|
|
2023-09-08 13:27:24 +00:00
|
|
|
/// MySQL compatibility alias.
|
|
|
|
factory.registerAlias("TO_DAYS", FunctionToDaysSinceYearZero::name, FunctionFactory::CaseInsensitive);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|