From c24e0fca7bbade43f83ece2fc7f5e983a751f6c0 Mon Sep 17 00:00:00 2001 From: Romain Ruetschi Date: Thu, 8 Aug 2024 09:32:57 +0200 Subject: [PATCH 1/2] Fix `TryFrom` impl to return a `DurationError` as per `prost-types` --- proto/src/google/protobuf/duration.rs | 60 +++++++++++++++++++++++---- 1 file changed, 53 insertions(+), 7 deletions(-) diff --git a/proto/src/google/protobuf/duration.rs b/proto/src/google/protobuf/duration.rs index b684abc32..0fad5ee81 100644 --- a/proto/src/google/protobuf/duration.rs +++ b/proto/src/google/protobuf/duration.rs @@ -5,6 +5,7 @@ // Copyright (c) 2020 InfluxData use core::convert::TryFrom; +use core::fmt; use prost::Name; @@ -94,6 +95,38 @@ impl Duration { } } +/// A duration handling error. +#[derive(Debug, PartialEq, Eq)] +#[non_exhaustive] +pub enum DurationError { + /// Indicates failure to convert a [`Duration`] to a [`core::time::Duration`] because + /// the duration is negative. The included [`core::time::Duration`] matches the magnitude of the + /// original negative [`Duration`]. + NegativeDuration(core::time::Duration), + + /// Indicates failure to convert a [`core::time::Duration`] to a [`Duration`]. + /// + /// Converting a [`core::time::Duration`] to a [`Duration`] fails if the magnitude + /// exceeds that representable by [`Duration`]. + OutOfRange, +} + +impl fmt::Display for DurationError { + fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { + match self { + DurationError::NegativeDuration(duration) => { + write!(f, "failed to convert negative duration: {duration:?}") + }, + DurationError::OutOfRange => { + write!(f, "failed to convert duration out of range") + }, + } + } +} + +#[cfg(feature = "std")] +impl std::error::Error for DurationError {} + /// Converts a `core::time::Duration` to a `Duration`. impl From for Duration { fn from(duration: core::time::Duration) -> Duration { @@ -116,26 +149,39 @@ impl From for Duration { } impl TryFrom for core::time::Duration { - type Error = core::time::Duration; + type Error = DurationError; - /// Converts a `Duration` to a result containing a positive (`Ok`) or negative (`Err`) - /// `std::time::Duration`. - fn try_from(mut duration: Duration) -> Result { + /// Converts a `Duration` to a `core::time::Duration`, failing if the duration is negative. + fn try_from(mut duration: Duration) -> Result { duration.normalize(); - if duration.seconds >= 0 { + if duration.seconds >= 0 && duration.nanos >= 0 { Ok(core::time::Duration::new( duration.seconds as u64, duration.nanos as u32, )) } else { - Err(core::time::Duration::new( + Err(DurationError::NegativeDuration(core::time::Duration::new( (-duration.seconds) as u64, (-duration.nanos) as u32, - )) + ))) } } } +impl TryFrom for Duration { + type Error = DurationError; + + /// Converts a `core::time::Duration` to a `Duration`, failing if the duration is too large. + fn try_from(duration: core::time::Duration) -> Result { + let seconds = i64::try_from(duration.as_secs()).map_err(|_| DurationError::OutOfRange)?; + let nanos = duration.subsec_nanos() as i32; + + let mut duration = Duration { seconds, nanos }; + duration.normalize(); + Ok(duration) + } +} + impl serde::Serialize for Duration { fn serialize(&self, serializer: S) -> Result where From cd8a3f77b0b4b1299af879c5824f4e183d4f812e Mon Sep 17 00:00:00 2001 From: Romain Ruetschi Date: Thu, 8 Aug 2024 09:38:29 +0200 Subject: [PATCH 2/2] Remove conflicting `From` impl --- proto/src/google/protobuf/duration.rs | 21 --------------------- 1 file changed, 21 deletions(-) diff --git a/proto/src/google/protobuf/duration.rs b/proto/src/google/protobuf/duration.rs index 0fad5ee81..c14957511 100644 --- a/proto/src/google/protobuf/duration.rs +++ b/proto/src/google/protobuf/duration.rs @@ -127,27 +127,6 @@ impl fmt::Display for DurationError { #[cfg(feature = "std")] impl std::error::Error for DurationError {} -/// Converts a `core::time::Duration` to a `Duration`. -impl From for Duration { - fn from(duration: core::time::Duration) -> Duration { - let seconds = duration.as_secs(); - let seconds = if seconds > i64::MAX as u64 { - i64::MAX - } else { - seconds as i64 - }; - let nanos = duration.subsec_nanos(); - let nanos = if nanos > i32::MAX as u32 { - i32::MAX - } else { - nanos as i32 - }; - let mut duration = Duration { seconds, nanos }; - duration.normalize(); - duration - } -} - impl TryFrom for core::time::Duration { type Error = DurationError;