/* Copyright (C) 2021 MariaDB Corporation This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; version 2 of the License. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include #include using namespace std; #include "functor_dtm.h" #include "funchelpers.h" #include "functioncolumn.h" #include "intervalcolumn.h" #include "rowgroup.h" using namespace execplan; #include "dataconvert.h" using namespace dataconvert; #include "bytestream.h" using namespace messageqcpp; using namespace funcexp; #include "errorcodes.h" #include "idberrorinfo.h" #include "errorids.h" using namespace logging; namespace funcexp { // need my_tz_find in dataconvert CalpontSystemCatalog::ColType Func_convert_tz::operationType(FunctionParm& fp, CalpontSystemCatalog::ColType& resultType) { return resultType; } int64_t Func_convert_tz::getDatetimeIntVal(rowgroup::Row& row, FunctionParm& parm, bool& isNull, execplan::CalpontSystemCatalog::ColType& ct) { return dataconvert::DataConvert::intToDatetime(getIntVal(row, parm, isNull, ct)); } int64_t Func_convert_tz::getIntVal(rowgroup::Row& row, FunctionParm& parm, bool& isNull, CalpontSystemCatalog::ColType& op_ct) { messageqcpp::ByteStream bs; bool bFromTz = false; bool bToTz = false; dataconvert::TIME_ZONE_INFO tzinfo; std::vector ats; std::vector types; std::vector ttis; #ifdef ABBR_ARE_USED std::vector chars; #endif std::vector lsis; std::vector revts; std::vector revtis; std::vector fallback_tti; int64_t seconds = 0; uint32_t err_code = 0; int64_t datetime_start = parm[0]->data()->getDatetimeIntVal(row, isNull); int64_t rtn = 0; if (isNull) return 0; // Adding a zero date to a time is always NULL if (datetime_start == 0) { isNull = true; return -1; } const auto& from_tz = parm[1]->data()->getStrVal(row, isNull); if (isNull) { return 0; } const auto& to_tz = parm[2]->data()->getStrVal(row, isNull); if (isNull) { return 0; } cout << "from " << from_tz.safeString("") << endl; cout << "to " << to_tz.safeString("") << endl; const auto& serialized_from_tzinfo = parm[3]->data()->getStrVal(row, isNull); const auto& serialized_to_tzinfo = parm[4]->data()->getStrVal(row, isNull); if (!serialized_from_tzinfo.isNull()) { bs.append((uint8_t*)serialized_from_tzinfo.str(), serialized_from_tzinfo.length()); dataconvert::unserializeTimezoneInfo(bs, &tzinfo); deserializeInlineVector(bs, ats); tzinfo.ats = ats.data(); deserializeInlineVector(bs, types); tzinfo.types = types.data(); deserializeInlineVector(bs, ttis); tzinfo.ttis = ttis.data(); #ifdef ABBR_ARE_USED deserializeInlineVector(bs_fromTZ, chars); tzinfo.chars = chars.data(); #endif deserializeInlineVector(bs, lsis); tzinfo.lsis = lsis.data(); deserializeInlineVector(bs, revts); tzinfo.revts = revts.data(); deserializeInlineVector(bs, revtis); tzinfo.revtis = revtis.data(); deserializeInlineVector(bs, fallback_tti); tzinfo.fallback_tti = fallback_tti.data(); bFromTz = true; } dataconvert::DateTime datetime(datetime_start); if (!dataconvert::DataConvert::isColumnDateTimeValid(datetime_start)) return datetime_start; // int64_t result_datetime; bool valid = true; MySQLTime my_time_tmp, my_start_time; my_time_tmp.reset(); my_start_time.year = datetime.year; my_start_time.month = datetime.month; my_start_time.day = datetime.day; my_start_time.hour = datetime.hour; my_start_time.minute = datetime.minute; my_start_time.second = datetime.second; my_start_time.second_part = datetime.msecond; if (bFromTz) { seconds = dataconvert::TIME_to_gmt_sec(my_start_time, &tzinfo, &err_code); if (err_code) { return datetime.convertToMySQLint(); } } else { long from_tz_offset; dataconvert::timeZoneToOffset(from_tz.str(), from_tz.length(), &from_tz_offset); seconds = dataconvert::mySQLTimeToGmtSec(my_start_time, from_tz_offset, valid); if (!valid) { if (seconds != 0) isNull = true; return datetime.convertToMySQLint(); } } if (!serialized_to_tzinfo.isNull()) { bs.reset(); bs.append((uint8_t*)serialized_to_tzinfo.str(), serialized_to_tzinfo.length()); dataconvert::unserializeTimezoneInfo(bs, &tzinfo); deserializeInlineVector(bs, ats); tzinfo.ats = ats.data(); deserializeInlineVector(bs, types); tzinfo.types = types.data(); deserializeInlineVector(bs, ttis); tzinfo.ttis = ttis.data(); #ifdef ABBR_ARE_USED deserializeInlineVector(bs_fromTZ, chars); tzinfo.chars = chars.data(); #endif deserializeInlineVector(bs, lsis); tzinfo.lsis = lsis.data(); deserializeInlineVector(bs, revts); tzinfo.revts = revts.data(); deserializeInlineVector(bs, revtis); tzinfo.revtis = revtis.data(); deserializeInlineVector(bs, fallback_tti); tzinfo.fallback_tti = fallback_tti.data(); bToTz = true; } if (bToTz) { dataconvert::gmt_sec_to_TIME(&my_time_tmp, seconds, &tzinfo); if (my_time_tmp.second == 60 || my_time_tmp.second == 61) my_time_tmp.second = 59; } else { long to_tz_offset; dataconvert::timeZoneToOffset(to_tz.str(), to_tz.length(), &to_tz_offset); dataconvert::gmtSecToMySQLTime(seconds, my_time_tmp, to_tz_offset); } dataconvert::DateTime result_datetime(my_time_tmp.year, my_time_tmp.month, my_time_tmp.day, my_time_tmp.hour, my_time_tmp.minute, my_time_tmp.second, my_time_tmp.second_part); if ((rtn = result_datetime.convertToMySQLint()) == 0) isNull = true; return rtn; } string Func_convert_tz::getStrVal(rowgroup::Row& row, FunctionParm& parm, bool& isNull, CalpontSystemCatalog::ColType& ct) { return dataconvert::DataConvert::datetimeToString(getIntVal(row, parm, isNull, ct)); } } // namespace funcexp