#pragma once #include "logging.hpp" #include #include #include #include #include #include #include #include #include #include #include namespace redfish { namespace time_utils { namespace details { constexpr intmax_t dayDuration = static_cast(24 * 60 * 60); using Days = std::chrono::duration>; inline void leftZeroPadding(std::string& str, const std::size_t padding) { if (str.size() < padding) { str.insert(0, padding - str.size(), '0'); } } template bool fromDurationItem(std::string_view& fmt, const char postfix, std::chrono::milliseconds& out) { const size_t pos = fmt.find(postfix); if (pos == std::string::npos) { return true; } if ((pos + 1U) > fmt.size()) { return false; } const char* end = nullptr; std::chrono::milliseconds::rep ticks = 0; if constexpr (std::is_same_v) { end = fmt.data() + std::min(pos, 3U); } else { end = fmt.data() + pos; } auto [ptr, ec] = std::from_chars(fmt.data(), end, ticks); if (ptr != end || ec != std::errc()) { BMCWEB_LOG_ERROR << "Failed to convert string to decimal with err: " << static_cast(ec) << "(" << std::make_error_code(ec).message() << "), ptr{" << static_cast(ptr) << "} != end{" << static_cast(end) << "})"; return false; } if constexpr (std::is_same_v) { ticks *= static_cast( std::pow(10, 3 - std::min(pos, 3U))); } if (ticks < 0) { return false; } out += FromTime(ticks); const auto maxConversionRange = std::chrono::duration_cast(std::chrono::milliseconds::max()) .count(); if (out < FromTime(ticks) || maxConversionRange < ticks) { return false; } fmt.remove_prefix(pos + 1U); return true; } } // namespace details /** * @brief Convert string that represents value in Duration Format to its numeric * equivalent. */ inline std::optional fromDurationString(const std::string& str) { std::chrono::milliseconds out = std::chrono::milliseconds::zero(); std::string_view v = str; if (v.empty()) { return out; } if (v.front() != 'P') { BMCWEB_LOG_ERROR << "Invalid duration format: " << str; return std::nullopt; } v.remove_prefix(1); if (!details::fromDurationItem(v, 'D', out)) { BMCWEB_LOG_ERROR << "Invalid duration format: " << str; return std::nullopt; } if (v.empty()) { return out; } if (v.front() != 'T') { BMCWEB_LOG_ERROR << "Invalid duration format: " << str; return std::nullopt; } v.remove_prefix(1); if (!details::fromDurationItem(v, 'H', out) || !details::fromDurationItem(v, 'M', out)) { BMCWEB_LOG_ERROR << "Invalid duration format: " << str; return std::nullopt; } if (v.find('.') != std::string::npos && v.find('S') != std::string::npos) { if (!details::fromDurationItem(v, '.', out) || !details::fromDurationItem(v, 'S', out)) { BMCWEB_LOG_ERROR << "Invalid duration format: " << str; return std::nullopt; } } else if (!details::fromDurationItem(v, 'S', out)) { BMCWEB_LOG_ERROR << "Invalid duration format: " << str; return std::nullopt; } if (!v.empty()) { BMCWEB_LOG_ERROR << "Invalid duration format: " << str; return std::nullopt; } 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")); details::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()) + "."; std::string msStr = std::to_string(ms.count()); details::leftZeroPadding(msStr, 3); fmt += msStr + "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 time_utils } // namespace redfish