Files
oceanbase/src/sql/engine/expr/ob_expr_timestamp_add.cpp

268 lines
10 KiB
C++

/**
* Copyright (c) 2021 OceanBase
* OceanBase CE is licensed under Mulan PubL v2.
* You can use this software according to the terms and conditions of the Mulan PubL v2.
* You may obtain a copy of Mulan PubL v2 at:
* http://license.coscl.org.cn/MulanPubL-2.0
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND,
* EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT,
* MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE.
* See the Mulan PubL v2 for more details.
*/
#define USING_LOG_PREFIX SQL_ENG
#include "sql/engine/expr/ob_expr_timestamp_add.h"
#include "sql/engine/expr/ob_expr_add.h"
#include "sql/engine/expr/ob_expr_mul.h"
#include "sql/engine/expr/ob_datum_cast.h"
#include "sql/engine/ob_exec_context.h"
#include "lib/ob_date_unit_type.h"
#include "lib/ob_name_def.h"
#include "lib/timezone/ob_time_convert.h"
#include "share/object/ob_obj_cast.h"
#include "sql/session/ob_sql_session_info.h"
#include "common/sql_mode/ob_sql_mode_utils.h"
namespace oceanbase
{
using namespace common;
namespace sql
{
ObExprTimeStampAdd::ObExprTimeStampAdd(ObIAllocator &alloc)
: ObFuncExprOperator(alloc, T_FUN_SYS_TIME_STAMP_ADD, N_TIME_STAMP_ADD, 3, NOT_ROW_DIMENSION)
{
}
ObExprTimeStampAdd::~ObExprTimeStampAdd()
{
}
inline int ObExprTimeStampAdd::calc_result_type3(ObExprResType &type,
ObExprResType &unit,
ObExprResType &interval,
ObExprResType &timestamp,
common::ObExprTypeCtx &type_ctx) const
{
UNUSED(type_ctx);
UNUSED(unit);
UNUSED(interval);
UNUSED(timestamp);
int ret = OB_SUCCESS;
ObCompatibilityMode compat_mode = get_compatibility_mode();
//not timestamp. compatible with mysql.
type.set_varchar();
type.set_length(common::ObAccuracy::MAX_ACCURACY2[compat_mode][common::ObTimestampType].precision_
+ common::ObAccuracy::MAX_ACCURACY2[compat_mode][common::ObTimestampType].scale_ + 1);
type.set_collation_level(common::CS_LEVEL_IMPLICIT);
//not connection collation. compatible with mysql.
type.set_collation_type(common::ObCharset::get_default_collation(common::ObCharset::get_default_charset()));
unit.set_calc_type(ObIntType);
interval.set_calc_type(ObIntType);
return ret;
}
void ObExprTimeStampAdd::check_reset_status(const ObCastMode cast_mode,
int &ret,
ObObj &result)
{
if (OB_FAIL(ret)) {
if (CM_IS_WARN_ON_FAIL(cast_mode)) {
ret = OB_SUCCESS;
result.set_null();
}
}
}
int ObExprTimeStampAdd::calc(const int64_t unit_value,
ObTime &ot,
const int64_t ts,
const ObTimeConvertCtx &cvrt_ctx,
int64_t interval,
int64_t &value)
{
int ret = OB_SUCCESS;
static const int64_t USECS_PER_WEEK = (USECS_PER_DAY * DAYS_PER_WEEK);
static const int64_t USECS_PER_HOUR = (static_cast<int64_t>(USECS_PER_SEC) * SECS_PER_HOUR);
static const int64_t USECS_PER_MIN = (USECS_PER_SEC * SECS_PER_MIN);
static const int64_t MONTH_PER_QUARTER = 3;
int64_t quota = -1;
int64_t delta = 0;
switch(unit_value) {
case DATE_UNIT_MICROSECOND: {
quota = 1;
//fall through
}
case DATE_UNIT_SECOND: {
quota = (-1 == quota ? USECS_PER_SEC : quota);
//fall through
}
case DATE_UNIT_MINUTE: {
quota = (-1 == quota ? USECS_PER_MIN : quota);
//fall through
}
case DATE_UNIT_HOUR: {
quota = (-1 == quota ? USECS_PER_HOUR : quota);
//fall through
}
case DATE_UNIT_DAY: {
quota = (-1 == quota ? USECS_PER_DAY : quota);
//fall through
}
case DATE_UNIT_WEEK: {
quota = (-1 == quota ? USECS_PER_WEEK : quota);
if (ObExprMul::is_mul_out_of_range(interval, quota, delta)) {
ret = OB_DATETIME_FUNCTION_OVERFLOW;
LOG_WARN("timestamp value is out of range", K(ret), K(quota), K(interval), K(ts), K(delta), K(value));
} else {
if (ObExprAdd::is_add_out_of_range(ts, delta, value)) {
ret = OB_DATETIME_FUNCTION_OVERFLOW;
LOG_WARN("timestamp value is out of range", K(ret), K(quota), K(interval), K(ts), K(delta), K(value));
}
}
break;
}
case DATE_UNIT_MONTH: {
//select timestampadd(month, 3, "2010-08-01 11:11:11");
//diff_month = 3
//so, result == 2010-08-01 11:11:11 + 3 month == 2010-11-01 11:11:11
delta = interval;
int32_t month = static_cast<int32_t>((ot.parts_[DT_YEAR]) * (MONS_PER_YEAR) + ot.parts_[DT_MON] - 1);
if (OB_UNLIKELY(ObExprAdd::is_add_out_of_range(month, delta, month))) {
ret = OB_DATETIME_FUNCTION_OVERFLOW;
LOG_WARN("timestamp value is out of range", K(ret), K(ts), K(month), K(interval));
} else {
ot.parts_[DT_YEAR] = month / 12;
ot.parts_[DT_MON] = month % 12 + 1;
int32_t days = ObTimeConverter::get_days_of_month(ot.parts_[DT_YEAR], ot.parts_[DT_MON]);
if (ot.parts_[DT_MDAY] > days) {
ot.parts_[DT_MDAY] = days;
}
ot.parts_[DT_DATE] = ObTimeConverter::ob_time_to_date(ot);
if (OB_FAIL(ObTimeConverter::ob_time_to_datetime(ot, cvrt_ctx, value))) {
LOG_WARN("ob time to datetime failed", K(ret));
}
}
break;
}
case DATE_UNIT_QUARTER: {
//select timestampadd(quarter, -2, "2010-11-01 11:11:11");
//diff_month = -2 * MONTH_PER_QUARTER = -2 *3 = -6
//so, result == 2010-11-01 11:11:11 + (- 6 month) == 2010-05-01 11:11:11
int32_t month = static_cast<int32_t>((ot.parts_[DT_YEAR]) * MONS_PER_YEAR + (ot.parts_[DT_MON] - 1));
if (OB_UNLIKELY(ObExprMul::is_mul_out_of_range(interval, MONTH_PER_QUARTER, delta))) {
ret = OB_DATETIME_FUNCTION_OVERFLOW;
LOG_WARN("timestamp value is out of range", K(ret), K(ts), K(month), K(interval));
} else if (OB_UNLIKELY(ObExprAdd::is_add_out_of_range(month, delta, month))) {
ret = OB_DATETIME_FUNCTION_OVERFLOW;
LOG_WARN("timestamp value is out of range", K(ret), K(ts), K(month), K(interval));
} else {
//IMHO, no need to define 12 as MONTH_PER_YEAR here.
ot.parts_[DT_YEAR] = month / 12;
ot.parts_[DT_MON] = month % 12 + 1;
ot.parts_[DT_DATE] = ObTimeConverter::ob_time_to_date(ot);
if (OB_FAIL(ObTimeConverter::ob_time_to_datetime(ot, cvrt_ctx, value))) {
LOG_WARN("ob time to datetime failed", K(ret));
}
}
break;
}
case DATE_UNIT_YEAR: {
delta = interval;
if (OB_UNLIKELY(ObExprAdd::is_add_out_of_range(ot.parts_[DT_YEAR], delta, ot.parts_[DT_YEAR]))) {
ret = OB_DATETIME_FUNCTION_OVERFLOW;
LOG_WARN("timestamp value is out of range", K(ret), K(ts), K(ot.parts_[DT_YEAR]), K(interval));
} else {
ot.parts_[DT_DATE] = ObTimeConverter::ob_time_to_date(ot);
if (OB_FAIL(ObTimeConverter::ob_time_to_datetime(ot, cvrt_ctx, value))) {
LOG_WARN("ob time to datetime failed", K(ret));
}
}
break;
}
default: {
ret = OB_INVALID_ARGUMENT;
LOG_WARN("Invalid argument", K(ret), K(unit_value));
break;
}
}
return ret;
}
int calc_timestampadd_expr(const ObExpr &expr, ObEvalCtx &ctx, ObDatum &res_datum)
{
int ret = OB_SUCCESS;
const ObSQLSessionInfo *session = ctx.exec_ctx_.get_my_session();
ObDatum *unit_datum = NULL;
ObDatum *interval_datum = NULL;
ObDatum *timestamp_datum = NULL;
if (OB_ISNULL(session)) {
ret = OB_ERR_UNEXPECTED;
LOG_WARN("session is NULL", K(ret));
} else if (OB_FAIL(expr.args_[0]->eval(ctx, unit_datum)) ||
OB_FAIL(expr.args_[1]->eval(ctx, interval_datum)) ||
OB_FAIL(expr.args_[2]->eval(ctx, timestamp_datum))) {
LOG_WARN("eval arg failed", K(ret), KP(unit_datum), KP(interval_datum),
KP(timestamp_datum));
} else if (unit_datum->is_null() || interval_datum->is_null() ||
timestamp_datum->is_null()) {
res_datum.set_null();
} else {
int64_t ts = 0;
int64_t interval_int = interval_datum->get_int();
int64_t res = 0;
ObTime ot;
ObTimeConvertCtx cvrt_ctx(get_timezone_info(ctx.exec_ctx_.get_my_session()), true);
char *buf = NULL;
int64_t buf_len = OB_CAST_TO_VARCHAR_MAX_LENGTH;
int64_t out_len = 0;
if (OB_FAIL(ob_datum_to_ob_time_with_date(*timestamp_datum,
expr.args_[2]->datum_meta_.type_,cvrt_ctx.tz_info_, ot,
get_cur_time(ctx.exec_ctx_.get_physical_plan_ctx()), false, 0,
expr.args_[2]->obj_meta_.has_lob_header()))) {
LOG_WARN("cast to ob time failed", K(ret), K(*timestamp_datum));
} else if (OB_FAIL(ObTimeConverter::ob_time_to_datetime(ot, cvrt_ctx, ts))) {
LOG_WARN("ob time to datetime failed", K(ret));
} else if (OB_FAIL(ObExprTimeStampAdd::calc(unit_datum->get_int(), ot, ts, cvrt_ctx,
interval_int, res))) {
LOG_WARN("calc failed", K(ret), K(*unit_datum), K(ts), K(interval_int));
} else if (OB_ISNULL(buf = expr.get_str_res_mem(ctx, buf_len))) {
ret = OB_ALLOCATE_MEMORY_FAILED;
LOG_WARN("allocate memory failed", K(ret), K(buf_len));
} else if (OB_FAIL(common_datetime_string(ObTimestampType, ObVarcharType,
expr.args_[2]->datum_meta_.scale_, false,
res, ctx, buf, buf_len, out_len))) {
LOG_WARN("common_datetime_string failed", K(ret), K(res), K(expr));
} else {
res_datum.set_string(ObString(out_len, buf));
}
}
if (OB_FAIL(ret) && OB_NOT_NULL(session)) {
uint64_t cast_mode = 0;
if (OB_FAIL(ObSQLUtils::get_default_cast_mode(session->get_stmt_type(),
session, cast_mode))) {
LOG_WARN("get_default_cast_mode failed", K(ret), K(session->get_stmt_type()));
} else if (CM_IS_WARN_ON_FAIL(cast_mode)) {
ret = OB_SUCCESS;
res_datum.set_null();
}
}
return ret;
}
int ObExprTimeStampAdd::cg_expr(ObExprCGCtx &expr_cg_ctx, const ObRawExpr &raw_expr,
ObExpr &rt_expr) const
{
int ret = OB_SUCCESS;
UNUSED(expr_cg_ctx);
UNUSED(raw_expr);
rt_expr.eval_func_ = calc_timestampadd_expr;
return ret;
}
} //namespace sql
} //namespace oceanbase