// Licensed to the Apache Software Foundation (ASF) under one // or more contributor license agreements. See the NOTICE file // distributed with this work for additional information // regarding copyright ownership. The ASF licenses this file // to you under the Apache License, Version 2.0 (the // "License"); you may not use this file except in compliance // with the License. You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, // software distributed under the License is distributed on an // "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY // KIND, either express or implied. See the License for the // specific language governing permissions and limitations // under the License. // #include "util/timezone_utils.h" #include #include #include #include #include #include #include #include #include "common/exception.h" #include "common/logging.h" namespace doris { RE2 TimezoneUtils::time_zone_offset_format_reg("^[+-]{1}\\d{2}\\:\\d{2}$"); std::unordered_map TimezoneUtils::timezone_names_map_; bool TimezoneUtils::inited_ = false; const std::string TimezoneUtils::default_time_zone = "+08:00"; void TimezoneUtils::load_timezone_names() { if (inited_) { return; } inited_ = true; std::string path; const char* tzdir = "/usr/share/zoneinfo"; char* tzdir_env = std::getenv("TZDIR"); if (tzdir_env && *tzdir_env) { tzdir = tzdir_env; } path += tzdir; path += '/'; auto path_prefix_len = path.size(); for (auto const& dir_entry : std::filesystem::recursive_directory_iterator {path}) { if (dir_entry.is_regular_file()) { auto timezone_full_name = dir_entry.path().string().substr(path_prefix_len); timezone_names_map_[boost::algorithm::to_lower_copy(timezone_full_name)] = timezone_full_name; } } } bool TimezoneUtils::find_cctz_time_zone(const std::string& timezone, cctz::time_zone& ctz) { auto timezone_lower = boost::algorithm::to_lower_copy(timezone); re2::StringPiece value; // +08:00 if (time_zone_offset_format_reg.Match(timezone, 0, timezone.size(), RE2::UNANCHORED, &value, 1)) { bool positive = value[0] != '-'; //Regular expression guarantees hour and minute must be int int hour = std::stoi(value.substr(1, 2).as_string()); int minute = std::stoi(value.substr(4, 2).as_string()); // timezone offsets around the world extended from -12:00 to +14:00 if (!positive && hour > 12) { return false; } else if (positive && hour > 14) { return false; } int offset = hour * 60 * 60 + minute * 60; offset *= positive ? 1 : -1; ctz = cctz::fixed_time_zone(cctz::seconds(offset)); return true; } else { // not only offset, GMT or GMT+8 // split tz_name and offset int split = timezone_lower.find('+') != std::string::npos ? timezone_lower.find('+') : timezone_lower.find('-'); cctz::time_zone offset; bool have_both = split != std::string::npos && split + 1 < timezone_lower.length() && std::isdigit(timezone_lower[split + 1]); if (have_both) { auto offset_str = timezone_lower.substr(split); timezone_lower = timezone_lower.substr(0, split); int offset_hours = 0; try { offset_hours = std::stoi(offset_str); } catch ([[maybe_unused]] std::exception& e) { VLOG_DEBUG << "Unable to cast " << timezone << " as timezone"; return false; } offset = cctz::fixed_time_zone(cctz::seconds(offset_hours * 60 * 60)); } bool tz_parsed = false; if (timezone_lower == "cst") { // Supports offset and region timezone type, "CST" use here is compatibility purposes. ctz = cctz::fixed_time_zone(cctz::seconds(8 * 60 * 60)); tz_parsed = true; } else if (timezone_lower == "z") { ctz = cctz::utc_time_zone(); tz_parsed = true; } else { auto it = timezone_names_map_.find(timezone_lower); if (it == timezone_names_map_.end()) { return false; } tz_parsed = cctz::load_time_zone(it->second, &ctz); } if (tz_parsed) { if (!have_both) { // GMT only return true; } // GMT+8 auto tz = (cctz::convert(cctz::civil_second {}, ctz) - cctz::time_point()) - (cctz::convert(cctz::civil_second {}, offset) - cctz::time_point()); ctz = cctz::fixed_time_zone(std::chrono::duration_cast(tz)); return true; } } return false; } } // namespace doris