#pragma once #include "logging.hpp" #include #include #include #include #include #include #include #include #include // IWYU pragma: no_include // IWYU pragma: no_include namespace redfish { namespace time_utils { namespace details { // Creates a string from an integer in the most efficient way possible without // using std::locale. Adds an exact zero pad based on the pad input parameter. // Does not handle negative numbers. inline std::string padZeros(int64_t value, size_t pad) { std::string result(pad, '0'); for (int64_t val = value; pad > 0; pad--) { result[pad - 1] = static_cast('0' + val % 10); val /= 10; } return result; } } // namespace details /** * @brief Convert string that represents value in Duration Format to its numeric * equivalent. */ inline std::optional fromDurationString(std::string_view v) { std::chrono::milliseconds out = std::chrono::milliseconds::zero(); enum class ProcessingStage { // P1DT1H1M1.100S P, Days, T, Hours, Minutes, Seconds, Milliseconds, Done, }; ProcessingStage stage = ProcessingStage::P; while (!v.empty()) { if (stage == ProcessingStage::P) { if (v.front() != 'P') { return std::nullopt; } v.remove_prefix(1); stage = ProcessingStage::Days; continue; } if (stage == ProcessingStage::Days || stage == ProcessingStage::T) { if (v.front() == 'T') { if (stage == ProcessingStage::T) { return std::nullopt; } v.remove_prefix(1); stage = ProcessingStage::Hours; continue; } } uint64_t ticks = 0; auto [ptr, ec] = std::from_chars(v.begin(), v.end(), ticks); if (ec != std::errc()) { BMCWEB_LOG_ERROR("Failed to convert string \"{}\" to decimal", v); return std::nullopt; } size_t charactersRead = static_cast(ptr - v.data()); if (ptr >= v.end()) { BMCWEB_LOG_ERROR("Missing postfix"); return std::nullopt; } if (*ptr == 'D') { if (stage > ProcessingStage::Days) { return std::nullopt; } out += std::chrono::days(ticks); } else if (*ptr == 'H') { if (stage > ProcessingStage::Hours) { return std::nullopt; } out += std::chrono::hours(ticks); } else if (*ptr == 'M') { if (stage > ProcessingStage::Minutes) { return std::nullopt; } out += std::chrono::minutes(ticks); } else if (*ptr == '.') { if (stage > ProcessingStage::Seconds) { return std::nullopt; } out += std::chrono::seconds(ticks); stage = ProcessingStage::Milliseconds; } else if (*ptr == 'S') { // We could be seeing seconds for the first time, (as is the case in // 1S) or for the second time (in the case of 1.1S). if (stage <= ProcessingStage::Seconds) { out += std::chrono::seconds(ticks); stage = ProcessingStage::Milliseconds; } else if (stage > ProcessingStage::Milliseconds) { BMCWEB_LOG_ERROR("Got unexpected information at end of parse"); return std::nullopt; } else { // Seconds could be any form of (1S, 1.1S, 1.11S, 1.111S); // Handle them all milliseconds are after the decimal point, // so they need right padded. if (charactersRead == 1) { ticks *= 100; } else if (charactersRead == 2) { ticks *= 10; } out += std::chrono::milliseconds(ticks); stage = ProcessingStage::Milliseconds; } } else { BMCWEB_LOG_ERROR("Unknown postfix {}", *ptr); return std::nullopt; } v.remove_prefix(charactersRead + 1U); } return out; } /** * @brief Convert time value into duration format that is based on ISO 8601. * Example output: "P12DT1M5.5S" * Ref: Redfish Specification, Section 9.4.4. Duration values */ inline std::string toDurationString(std::chrono::milliseconds ms) { if (ms < std::chrono::milliseconds::zero()) { return ""; } std::string fmt; fmt.reserve(sizeof("PxxxxxxxxxxxxDTxxHxxMxx.xxxxxxS")); std::chrono::days days = std::chrono::floor(ms); ms -= days; std::chrono::hours hours = std::chrono::floor(ms); ms -= hours; std::chrono::minutes minutes = std::chrono::floor(ms); ms -= minutes; std::chrono::seconds seconds = std::chrono::floor(ms); ms -= seconds; fmt = "P"; if (days.count() > 0) { fmt += std::to_string(days.count()) + "D"; } fmt += "T"; if (hours.count() > 0) { fmt += std::to_string(hours.count()) + "H"; } if (minutes.count() > 0) { fmt += std::to_string(minutes.count()) + "M"; } if (seconds.count() != 0 || ms.count() != 0) { fmt += std::to_string(seconds.count()) + "."; fmt += details::padZeros(ms.count(), 3); fmt += "S"; } return fmt; } inline std::optional toDurationStringFromUint(const uint64_t timeMs) { static const uint64_t maxTimeMs = static_cast(std::chrono::milliseconds::max().count()); if (maxTimeMs < timeMs) { return std::nullopt; } std::string duration = toDurationString(std::chrono::milliseconds(timeMs)); if (duration.empty()) { return std::nullopt; } return std::make_optional(duration); } namespace details { // Returns year/month/day triple in civil calendar // Preconditions: z is number of days since 1970-01-01 and is in the range: // [numeric_limits::min(), // numeric_limits::max()-719468]. // Algorithm sourced from // https://howardhinnant.github.io/date_algorithms.html#civil_from_days // All constants are explained in the above template constexpr std::tuple civilFromDays(IntType z) noexcept { z += 719468; IntType era = (z >= 0 ? z : z - 146096) / 146097; unsigned doe = static_cast(z - era * 146097); // [0, 146096] unsigned yoe = (doe - doe / 1460 + doe / 36524 - doe / 146096) / 365; // [0, 399] IntType y = static_cast(yoe) + era * 400; unsigned doy = doe - (365 * yoe + yoe / 4 - yoe / 100); // [0, 365] unsigned mp = (5 * doy + 2) / 153; // [0, 11] unsigned d = doy - (153 * mp + 2) / 5 + 1; // [1, 31] unsigned m = mp < 10 ? mp + 3 : mp - 9; // [1, 12] return std::tuple(y + (m <= 2), m, d); } template std::string toISO8061ExtendedStr(std::chrono::duration t) { using seconds = std::chrono::duration; using minutes = std::chrono::duration>; using hours = std::chrono::duration>; using days = std::chrono::duration< IntType, std::ratio_multiply>>; // d is days since 1970-01-01 days d = std::chrono::duration_cast(t); // t is now time duration since midnight of day d t -= d; // break d down into year/month/day int year = 0; int month = 0; int day = 0; std::tie(year, month, day) = details::civilFromDays(d.count()); // Check against limits. Can't go above year 9999, and can't go below epoch // (1970) if (year >= 10000) { year = 9999; month = 12; day = 31; t = days(1) - std::chrono::duration(1); } else if (year < 1970) { year = 1970; month = 1; day = 1; t = std::chrono::duration::zero(); } std::string out; out += details::padZeros(year, 4); out += '-'; out += details::padZeros(month, 2); out += '-'; out += details::padZeros(day, 2); out += 'T'; hours hr = duration_cast(t); out += details::padZeros(hr.count(), 2); t -= hr; out += ':'; minutes mt = duration_cast(t); out += details::padZeros(mt.count(), 2); t -= mt; out += ':'; seconds se = duration_cast(t); out += details::padZeros(se.count(), 2); t -= se; if constexpr (std::is_same_v) { out += '.'; using MilliDuration = std::chrono::duration; MilliDuration subsec = duration_cast(t); out += details::padZeros(subsec.count(), 3); } else if constexpr (std::is_same_v) { out += '.'; using MicroDuration = std::chrono::duration; MicroDuration subsec = duration_cast(t); out += details::padZeros(subsec.count(), 6); } out += "+00:00"; return out; } } // namespace details // Returns the formatted date time string. // Note that the maximum supported date is 9999-12-31T23:59:59+00:00, if // the given |secondsSinceEpoch| is too large, we return the maximum supported // date. inline std::string getDateTimeUint(uint64_t secondsSinceEpoch) { using DurationType = std::chrono::duration; DurationType sinceEpoch(secondsSinceEpoch); return details::toISO8061ExtendedStr(sinceEpoch); } // Returns the formatted date time string with millisecond precision // Note that the maximum supported date is 9999-12-31T23:59:59+00:00, if // the given |secondsSinceEpoch| is too large, we return the maximum supported // date. inline std::string getDateTimeUintMs(uint64_t milliSecondsSinceEpoch) { using DurationType = std::chrono::duration; DurationType sinceEpoch(milliSecondsSinceEpoch); return details::toISO8061ExtendedStr(sinceEpoch); } // Returns the formatted date time string with microsecond precision inline std::string getDateTimeUintUs(uint64_t microSecondsSinceEpoch) { using DurationType = std::chrono::duration; DurationType sinceEpoch(microSecondsSinceEpoch); return details::toISO8061ExtendedStr(sinceEpoch); } inline std::string getDateTimeStdtime(std::time_t secondsSinceEpoch) { using DurationType = std::chrono::duration; DurationType sinceEpoch(secondsSinceEpoch); return details::toISO8061ExtendedStr(sinceEpoch); } /** * Returns the current Date, Time & the local Time Offset * information in a pair * * @param[in] None * * @return std::pair, which consist * of current DateTime & the TimeOffset strings respectively. */ inline std::pair getDateTimeOffsetNow() { std::time_t time = std::time(nullptr); std::string dateTime = getDateTimeStdtime(time); /* extract the local Time Offset value from the * received dateTime string. */ std::string timeOffset("Z00:00"); std::size_t lastPos = dateTime.size(); std::size_t len = timeOffset.size(); if (lastPos > len) { timeOffset = dateTime.substr(lastPos - len); } return std::make_pair(dateTime, timeOffset); } using usSinceEpoch = std::chrono::duration; std::optional dateStringToEpoch(std::string_view datetime); } // namespace time_utils } // namespace redfish