From e85250e87f8f8cceb7eecd8fbbf1b492276be0a3 Mon Sep 17 00:00:00 2001 From: Jan Keukert <35369791+jankeu@users.noreply.github.com> Date: Wed, 13 Dec 2023 09:33:58 +0100 Subject: [PATCH 1/4] introduce feature flag for regex dep --- Cargo.toml | 5 ++++- src/lib.rs | 25 ++++++++++++++++--------- 2 files changed, 20 insertions(+), 10 deletions(-) diff --git a/Cargo.toml b/Cargo.toml index 9155a5f..4d1944d 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -8,11 +8,14 @@ edition = "2021" version = "0.9.3" license = "MIT OR Apache-2.0" +[features] +parser = ["dep:regex"] + [dependencies] chrono = "0.4.31" derive_more = "0.99.17" lazy_static = "1.4.0" -regex = "1.10.2" +regex = {version = "1.10.2", optional=true} serde = { version = "1.0.193", features = ["derive"], default-features = false } thiserror = "1.0.50" diff --git a/src/lib.rs b/src/lib.rs index 031343e..f8c1713 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -53,6 +53,7 @@ use std::ops::Mul; use std::ops::MulAssign; use std::ops::Sub; use std::ops::SubAssign; +#[cfg(feature = "parser")] use std::str::FromStr; use chrono::format::DelayedFormat; @@ -64,7 +65,9 @@ use derive_more::From; use derive_more::Into; use derive_more::Neg; use derive_more::Sum; +#[cfg(feature = "parser")] use lazy_static::lazy_static; +#[cfg(feature = "parser")] use regex::Regex; use serde::de::Visitor; use serde::Deserialize; @@ -912,6 +915,7 @@ impl Duration { } } +#[cfg(feature = "parser")] /// Allows deserializing from strings, unsigned integers, and signed integers. impl<'de> Deserialize<'de> for Duration { fn deserialize(deserializer: D) -> Result @@ -934,8 +938,9 @@ impl PartialOrd for Duration { } } +#[cfg(feature = "parser")] struct DurationVisitor; - +#[cfg(feature = "parser")] impl<'de> Visitor<'de> for DurationVisitor { type Value = Duration; @@ -1276,6 +1281,7 @@ impl From for Duration { } } +#[cfg(feature = "parser")] /// Parses Duration from str /// /// # Example @@ -1301,11 +1307,13 @@ impl From for Duration { /// ); /// ``` impl FromStr for Duration { + const REGEX: &str = r"^(?P-)?((?P\d+)h)?((?P\d+)m)?((?P\d+)s)?((?P\d+)ms)?$"; + type Err = DurationParseError; fn from_str(seconds: &str) -> Result { lazy_static! { - static ref RE: Regex = Regex::new(REGEX).unwrap(); + static ref RE: Regex = Regex::new(Self::REGEX).unwrap(); } let captures = RE .captures(seconds) @@ -1346,13 +1354,8 @@ impl Display for DurationParseError { } } -const REGEX: &str = r"^(?P-)?((?P\d+)h)?((?P\d+)m)?((?P\d+)s)?((?P\d+)ms)?$"; - #[cfg(test)] mod time_test { - use serde_test::assert_de_tokens; - use serde_test::Token; - use crate::Duration; use crate::Time; @@ -1485,8 +1488,11 @@ mod time_test { } } + #[cfg(feature = "parser")] #[test] fn deserialize_time() { + use serde_test::assert_de_tokens; + use serde_test::Token; // strings assert_de_tokens(&Time::seconds(7), &[Token::Str("1970-01-01T00:00:07Z")]); assert_de_tokens(&Time::seconds(7), &[Token::String("1970-01-01T00:00:07Z")]); @@ -1546,8 +1552,6 @@ mod time_test { #[cfg(test)] mod duration_test { - use serde_test::assert_de_tokens; - use serde_test::Token; use super::*; @@ -1701,8 +1705,11 @@ mod duration_test { assert_eq!(Duration::minutes(-4), time2 - time); } + #[cfg(feature = "parser")] #[test] fn deserialize_duration() { + use serde_test::assert_de_tokens; + use serde_test::Token; // strings assert_de_tokens(&Duration::minutes(7), &[Token::Str("7m")]); assert_de_tokens( From 7432e109c5bf1c8397c32ca340844e82acef4c5a Mon Sep 17 00:00:00 2001 From: Jan Keukert <35369791+jankeu@users.noreply.github.com> Date: Wed, 13 Dec 2023 09:36:14 +0100 Subject: [PATCH 2/4] include lazy_static in feature flag --- Cargo.toml | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/Cargo.toml b/Cargo.toml index 4d1944d..1dd70c3 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -9,13 +9,13 @@ version = "0.9.3" license = "MIT OR Apache-2.0" [features] -parser = ["dep:regex"] +parser = ["dep:regex", "dep:lazy_static"] [dependencies] chrono = "0.4.31" derive_more = "0.99.17" -lazy_static = "1.4.0" -regex = {version = "1.10.2", optional=true} +lazy_static = { version = "1.4.0", optional = true } +regex = { version = "1.10.2", optional = true } serde = { version = "1.0.193", features = ["derive"], default-features = false } thiserror = "1.0.50" From e80ddf5d55777fead0a2f74c1b449df4e498f38d Mon Sep 17 00:00:00 2001 From: Jan Keukert <35369791+jankeu@users.noreply.github.com> Date: Wed, 13 Dec 2023 09:56:02 +0100 Subject: [PATCH 3/4] add parser feature flag --- Cargo.toml | 5 +++-- src/lib.rs | 64 ++++++++++++++++++++++++++++-------------------------- 2 files changed, 36 insertions(+), 33 deletions(-) diff --git a/Cargo.toml b/Cargo.toml index 1dd70c3..fdf6b75 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -10,13 +10,14 @@ license = "MIT OR Apache-2.0" [features] parser = ["dep:regex", "dep:lazy_static"] +formater = ["dep:chrono", "dep:serde"] [dependencies] -chrono = "0.4.31" +chrono = { version = "0.4.31", optional = true } derive_more = "0.99.17" lazy_static = { version = "1.4.0", optional = true } regex = { version = "1.10.2", optional = true } -serde = { version = "1.0.193", features = ["derive"], default-features = false } +serde = { version = "1.0.193", features = ["derive"], default-features = false, optional=true } thiserror = "1.0.50" [dev-dependencies] diff --git a/src/lib.rs b/src/lib.rs index 552ffd9..153e018 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -57,10 +57,6 @@ use std::ops::SubAssign; use std::str::FromStr; use std::time::SystemTime; -use chrono::format::DelayedFormat; -use chrono::format::StrftimeItems; -use chrono::DateTime; -use chrono::NaiveDateTime; use derive_more::Deref; use derive_more::From; use derive_more::Into; @@ -70,17 +66,19 @@ use derive_more::Sum; use lazy_static::lazy_static; #[cfg(feature = "parser")] use regex::Regex; +#[cfg(feature = "formater")] use serde::de::Visitor; +#[cfg(feature = "formater")] use serde::Deserialize; +#[cfg(feature = "formater")] use serde::Serialize; use thiserror::Error; /// A point in time. /// /// Low overhead time representation. Internally represented as milliseconds. -#[derive( - Eq, PartialEq, Hash, Ord, PartialOrd, Copy, Clone, Default, Serialize, Deref, From, Into, -)] +#[derive(Eq, PartialEq, Hash, Ord, PartialOrd, Copy, Clone, Default, Deref, From, Into)] +#[cfg_attr(feature = "formater", derive(Serialize))] pub struct Time(i64); impl Time { @@ -107,6 +105,7 @@ impl Time { Time::millis(hours * Self::HOUR.0) } + #[cfg(feature = "formater")] /// Returns an RFC 3339 and ISO 8601 date and time string such as /// 1996-12-19T16:39:57+00:00. /// @@ -122,6 +121,7 @@ impl Time { self.format("%Y-%m-%dT%H:%M:%S+00:00").to_string() } + #[cfg(feature = "formater")] /// The function format string is forwarded to /// [`chrono::NaiveDateTime::format()`] /// @@ -133,7 +133,10 @@ impl Time { /// use tinytime::Time; /// assert_eq!("∞", Time::MAX.format("whatever").to_string()); /// ``` - pub fn format<'a>(&self, fmt: &'a str) -> DelayedFormat> { + pub fn format<'a>( + &self, + fmt: &'a str, + ) -> chrono::format::DelayedFormat> { let secs = self.0 / 1000; let nanos = (self.0 % 1000) * 1_000_000; // casting to u32 is safe here because it is guaranteed that the value is in @@ -145,13 +148,18 @@ impl Time { nanos.unsigned_abs() } as u32; - let t = NaiveDateTime::from_timestamp_opt(secs, nanos); + let t = chrono::NaiveDateTime::from_timestamp_opt(secs, nanos); match t { - None => DelayedFormat::new(None, None, StrftimeItems::new("∞")), + None => chrono::format::DelayedFormat::new( + None, + None, + chrono::format::StrftimeItems::new("∞"), + ), Some(v) => v.format(fmt), } } + #[cfg(feature = "formater")] /// Parses an RFC 3339 date and time string into a [Time] instance. /// /// The parsing is forwarded to [`chrono::DateTime::parse_from_rfc3339()`]. @@ -167,7 +175,7 @@ impl Time { /// ); /// ``` pub fn parse_from_rfc3339(s: &str) -> Result { - DateTime::parse_from_rfc3339(s) + chrono::DateTime::parse_from_rfc3339(s) .map(|chrono_datetime| Time::millis(chrono_datetime.timestamp_millis())) } @@ -260,6 +268,7 @@ impl Time { } } +#[cfg(feature = "formater")] impl Display for Time { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { let rfc3339_string = self.to_rfc3339(); @@ -278,6 +287,7 @@ impl TryFrom for Time { } } +#[cfg(feature = "formater")] /// Allows deserializing from RFC 3339 strings and unsigned integers. impl<'de> Deserialize<'de> for Time { fn deserialize(deserializer: D) -> Result @@ -288,8 +298,9 @@ impl<'de> Deserialize<'de> for Time { } } +#[cfg(feature = "formater")] struct TimeVisitor; - +#[cfg(feature = "formater")] impl<'de> Visitor<'de> for TimeVisitor { type Value = Time; @@ -385,7 +396,8 @@ pub enum TimeWindowError { /// Debug-asserts ensure that start <= end. /// If compiled in release mode, the invariant of start <= end is maintained, by /// correcting invalid use of the API (and setting end to start). -#[derive(Clone, Debug, Eq, PartialEq, Default, Copy, Serialize, Deserialize, From, Into, Hash)] +#[derive(Clone, Debug, Eq, PartialEq, Default, Copy, From, Into, Hash)] +#[cfg_attr(feature = "formater", derive(Serialize, Deserialize))] pub struct TimeWindow { start: Time, end: Time, @@ -831,22 +843,9 @@ impl TimeWindow { /// Duration can be negative. Internally duration is represented as /// milliseconds. #[derive( - Eq, - PartialEq, - Ord, - PartialOrd, - Copy, - Clone, - Debug, - Default, - Hash, - Serialize, - Deref, - From, - Into, - Sum, - Neg, + Eq, PartialEq, Ord, PartialOrd, Copy, Clone, Debug, Default, Hash, Deref, From, Into, Sum, Neg, )] +#[cfg_attr(feature = "formater", derive(Serialize))] pub struct Duration(i64); impl Duration { @@ -1320,13 +1319,11 @@ impl From for Duration { /// ); /// ``` impl FromStr for Duration { - const REGEX: &str = r"^(?P-)?((?P\d+)h)?((?P\d+)m)?((?P\d+)s)?((?P\d+)ms)?$"; - type Err = DurationParseError; fn from_str(seconds: &str) -> Result { lazy_static! { - static ref RE: Regex = Regex::new(Self::REGEX).unwrap(); + static ref RE: Regex = Regex::new(DURATION_REGEX).unwrap(); } let captures = RE .captures(seconds) @@ -1351,6 +1348,10 @@ impl FromStr for Duration { } } +#[cfg(feature = "parser")] +const DURATION_REGEX: &str = + r"^(?P-)?((?P\d+)h)?((?P\d+)m)?((?P\d+)s)?((?P\d+)ms)?$"; + #[derive(Debug)] pub enum DurationParseError { UnrecognizedFormat, @@ -1372,6 +1373,7 @@ mod time_test { use crate::Duration; use crate::Time; + #[cfg(feature = "formater")] #[test] fn test_display() { struct TestCase { From d9af637a3530169bfec0038b41bb87cfd7021c14 Mon Sep 17 00:00:00 2001 From: Jan Keukert <35369791+jankeu@users.noreply.github.com> Date: Wed, 13 Dec 2023 11:08:49 +0100 Subject: [PATCH 4/4] spelling --- Cargo.toml | 2 +- src/lib.rs | 28 ++++++++++++++-------------- 2 files changed, 15 insertions(+), 15 deletions(-) diff --git a/Cargo.toml b/Cargo.toml index 1a2a823..722db6b 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -10,7 +10,7 @@ license = "MIT OR Apache-2.0" [features] parser = ["dep:regex", "dep:lazy_static"] -formater = ["dep:chrono", "dep:serde"] +formatter = ["dep:chrono", "dep:serde"] [dependencies] chrono = { version = "0.4.31", optional = true } diff --git a/src/lib.rs b/src/lib.rs index 153e018..64f8de5 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -66,11 +66,11 @@ use derive_more::Sum; use lazy_static::lazy_static; #[cfg(feature = "parser")] use regex::Regex; -#[cfg(feature = "formater")] +#[cfg(feature = "formatter")] use serde::de::Visitor; -#[cfg(feature = "formater")] +#[cfg(feature = "formatter")] use serde::Deserialize; -#[cfg(feature = "formater")] +#[cfg(feature = "formatter")] use serde::Serialize; use thiserror::Error; @@ -78,7 +78,7 @@ use thiserror::Error; /// /// Low overhead time representation. Internally represented as milliseconds. #[derive(Eq, PartialEq, Hash, Ord, PartialOrd, Copy, Clone, Default, Deref, From, Into)] -#[cfg_attr(feature = "formater", derive(Serialize))] +#[cfg_attr(feature = "formatter", derive(Serialize))] pub struct Time(i64); impl Time { @@ -105,7 +105,7 @@ impl Time { Time::millis(hours * Self::HOUR.0) } - #[cfg(feature = "formater")] + #[cfg(feature = "formatter")] /// Returns an RFC 3339 and ISO 8601 date and time string such as /// 1996-12-19T16:39:57+00:00. /// @@ -121,7 +121,7 @@ impl Time { self.format("%Y-%m-%dT%H:%M:%S+00:00").to_string() } - #[cfg(feature = "formater")] + #[cfg(feature = "formatter")] /// The function format string is forwarded to /// [`chrono::NaiveDateTime::format()`] /// @@ -159,7 +159,7 @@ impl Time { } } - #[cfg(feature = "formater")] + #[cfg(feature = "formatter")] /// Parses an RFC 3339 date and time string into a [Time] instance. /// /// The parsing is forwarded to [`chrono::DateTime::parse_from_rfc3339()`]. @@ -268,7 +268,7 @@ impl Time { } } -#[cfg(feature = "formater")] +#[cfg(feature = "formatter")] impl Display for Time { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { let rfc3339_string = self.to_rfc3339(); @@ -287,7 +287,7 @@ impl TryFrom for Time { } } -#[cfg(feature = "formater")] +#[cfg(feature = "formatter")] /// Allows deserializing from RFC 3339 strings and unsigned integers. impl<'de> Deserialize<'de> for Time { fn deserialize(deserializer: D) -> Result @@ -298,9 +298,9 @@ impl<'de> Deserialize<'de> for Time { } } -#[cfg(feature = "formater")] +#[cfg(feature = "formatter")] struct TimeVisitor; -#[cfg(feature = "formater")] +#[cfg(feature = "formatter")] impl<'de> Visitor<'de> for TimeVisitor { type Value = Time; @@ -397,7 +397,7 @@ pub enum TimeWindowError { /// If compiled in release mode, the invariant of start <= end is maintained, by /// correcting invalid use of the API (and setting end to start). #[derive(Clone, Debug, Eq, PartialEq, Default, Copy, From, Into, Hash)] -#[cfg_attr(feature = "formater", derive(Serialize, Deserialize))] +#[cfg_attr(feature = "formatter", derive(Serialize, Deserialize))] pub struct TimeWindow { start: Time, end: Time, @@ -845,7 +845,7 @@ impl TimeWindow { #[derive( Eq, PartialEq, Ord, PartialOrd, Copy, Clone, Debug, Default, Hash, Deref, From, Into, Sum, Neg, )] -#[cfg_attr(feature = "formater", derive(Serialize))] +#[cfg_attr(feature = "formatter", derive(Serialize))] pub struct Duration(i64); impl Duration { @@ -1373,7 +1373,7 @@ mod time_test { use crate::Duration; use crate::Time; - #[cfg(feature = "formater")] + #[cfg(feature = "formatter")] #[test] fn test_display() { struct TestCase {