10#include <google/protobuf/duration.pb.h>
11#include <google/protobuf/timestamp.pb.h>
12#include <google/type/date.pb.h>
14#include <userver/formats/json_fwd.hpp>
15#include <userver/formats/parse/to.hpp>
16#include <userver/formats/serialize/to.hpp>
17#include <userver/utils/assert.hpp>
18#include <userver/utils/datetime/cpp_20_calendar.hpp>
19#include <userver/utils/datetime/date.hpp>
21USERVER_NAMESPACE_BEGIN
29bool IsValid(
const google::protobuf::Timestamp& grpc_ts);
34 using std::overflow_error::overflow_error;
40template <
class Duration>
42 const std::chrono::time_point<std::chrono::system_clock, Duration>& system_tp
44 const auto seconds = std::chrono::duration_cast<std::chrono::seconds>(system_tp.time_since_epoch());
45 const auto nanos = std::chrono::duration_cast<std::chrono::nanoseconds>(system_tp.time_since_epoch() - seconds);
47 google::protobuf::Timestamp timestamp;
48 timestamp.set_seconds(seconds.count());
49 timestamp.set_nanos(nanos.count());
60template <
class Duration = std::chrono::system_clock::duration>
61std::chrono::time_point<std::chrono::system_clock, Duration>
ToTimePoint(
const google::protobuf::Timestamp& grpc_ts) {
66 const std::chrono::seconds seconds(grpc_ts.seconds());
69 if (seconds >= std::chrono::duration_cast<std::chrono::seconds>(Duration::max())) {
73 return std::chrono::time_point<std::chrono::system_clock, Duration>(std::chrono::duration_cast<Duration>(
74 seconds + std::chrono::duration_cast<Duration>(std::chrono::nanoseconds(grpc_ts.nanos()))
86bool IsValid(
const google::type::Date& grpc_date);
91 using std::overflow_error::overflow_error;
94#if __cpp_lib_chrono
>= 201907L
99google::type::Date ToProtoDate(
const std::chrono::year_month_day& system_date);
104std::chrono::year_month_day ToYearMonthDay(
const google::type::Date& grpc_date);
121template <
class Duration>
122google::type::Date
ToProtoDate(
const std::chrono::time_point<std::chrono::system_clock, Duration>& system_tp) {
130 const google::type::Date& grpc_date
142bool IsValid(
const google::protobuf::Duration& grpc_duration);
147 using std::overflow_error::overflow_error;
153template <
class Duration = std::chrono::microseconds>
154Duration
ToDuration(
const google::protobuf::Duration& grpc_duration) {
159 const std::chrono::seconds seconds(grpc_duration.seconds());
162 if (seconds >= std::chrono::duration_cast<std::chrono::seconds>(Duration::max())) {
165 return std::chrono::duration_cast<Duration>(
166 seconds + std::chrono::duration_cast<Duration>(std::chrono::nanoseconds(grpc_duration.nanos()))
173template <
class Rep,
class Period>
174google::protobuf::Duration
ToProtoDuration(
const std::chrono::duration<Rep, Period>& duration) {
175 const auto seconds = std::chrono::duration_cast<std::chrono::seconds>(duration);
176 const auto nanos = std::chrono::duration_cast<std::chrono::nanoseconds>(duration - seconds);
178 google::protobuf::Duration result;
179 result.set_seconds(seconds.count());
180 result.set_nanos(nanos.count());