From a454a0d17e1aae216f1bb01112133df4a6103d80 Mon Sep 17 00:00:00 2001 From: Dominik Spicher Date: Tue, 19 Oct 2021 13:24:05 +0200 Subject: [PATCH] Add TryFrom<&str> to EnumString The implementation simply delegates to `std::str::FromStr`. To maintain compatibility with Rust 1.32, a new dependency is introduced which allows us to emit the `TryFrom<&str>` implementation only for Rust 1.34 and above when `std::convert::TryFrom` was stabilized. --- strum_macros/Cargo.toml | 1 + .../src/macros/strings/from_string.rs | 45 ++++++++- strum_tests/Cargo.toml | 5 +- strum_tests/tests/from_str.rs | 94 ++++++++++--------- 4 files changed, 101 insertions(+), 44 deletions(-) diff --git a/strum_macros/Cargo.toml b/strum_macros/Cargo.toml index ee84210a..fcf643c6 100644 --- a/strum_macros/Cargo.toml +++ b/strum_macros/Cargo.toml @@ -22,6 +22,7 @@ name = "strum_macros" heck = "0.3" proc-macro2 = "1.0" quote = "1.0" +rustversion = "1.0" syn = { version = "1.0", features = ["parsing", "extra-traits"] } [dev-dependencies] diff --git a/strum_macros/src/macros/strings/from_string.rs b/strum_macros/src/macros/strings/from_string.rs index 2977fd19..a42ff4de 100644 --- a/strum_macros/src/macros/strings/from_string.rs +++ b/strum_macros/src/macros/strings/from_string.rs @@ -87,7 +87,7 @@ pub fn from_string_inner(ast: &DeriveInput) -> syn::Result { arms.push(default); - Ok(quote! { + let from_str = quote! { #[allow(clippy::use_self)] impl #impl_generics ::std::str::FromStr for #name #ty_generics #where_clause { type Err = #strum_module_path::ParseError; @@ -97,5 +97,48 @@ pub fn from_string_inner(ast: &DeriveInput) -> syn::Result { } } } + }; + + let try_from_str = try_from_str( + name, + impl_generics, + ty_generics, + where_clause, + strum_module_path, + ); + + Ok(quote! { + #from_str + #try_from_str }) } + +#[rustversion::before(1.34)] +fn try_from_str( + _name: &proc_macro2::Ident, + _impl_generics: syn::ImplGenerics, + _ty_generics: syn::TypeGenerics, + _where_clause: Option<&syn::WhereClause>, + _strum_module_path: syn::Path, +) -> TokenStream { + Default::default() +} + +#[rustversion::since(1.34)] +fn try_from_str( + name: &proc_macro2::Ident, + impl_generics: syn::ImplGenerics, + ty_generics: syn::TypeGenerics, + where_clause: Option<&syn::WhereClause>, + strum_module_path: syn::Path, +) -> TokenStream { + quote! { + #[allow(clippy::use_self)] + impl #impl_generics ::std::convert::TryFrom<&str> for #name #ty_generics #where_clause { + type Error = #strum_module_path::ParseError; + fn try_from(s: &str) -> ::std::result::Result< #name #ty_generics , Self::Error> { + ::std::str::FromStr::from_str(s) + } + } + } +} diff --git a/strum_tests/Cargo.toml b/strum_tests/Cargo.toml index a4b3470e..28664a28 100644 --- a/strum_tests/Cargo.toml +++ b/strum_tests/Cargo.toml @@ -13,4 +13,7 @@ structopt = "0.2.18" bitflags = "=1.2" [build-dependencies] -version_check = "0.9.2" \ No newline at end of file +version_check = "0.9.2" + +[dev-dependencies] +rustversion = "1.0" diff --git a/strum_tests/tests/from_str.rs b/strum_tests/tests/from_str.rs index 77922e9d..496fa47a 100644 --- a/strum_tests/tests/from_str.rs +++ b/strum_tests/tests/from_str.rs @@ -17,39 +17,54 @@ enum Color { Black, } +#[rustversion::since(1.34)] +macro_rules! assert_from_str { + ($t:ty, $expected:expr, $lit:literal) => { + assert_eq!($expected, <$t as FromStr>::from_str($lit).unwrap()); + assert_eq!( + $expected, + <$t as std::convert::TryFrom<&str>>::try_from($lit).unwrap() + ); + }; +} + +#[rustversion::before(1.34)] +macro_rules! assert_from_str { + ($t:ty, $expected:expr, $lit:literal) => { + assert_eq!($expected, <$t as FromStr>::from_str($lit).unwrap()); + }; +} + #[test] fn color_simple() { - assert_eq!(Color::Red, Color::from_str("Red").unwrap()); + assert_from_str!(Color, Color::Red, "Red"); } #[test] fn color_value() { - assert_eq!(Color::Blue { hue: 0 }, Color::from_str("Blue").unwrap()); + assert_from_str!(Color, Color::Blue { hue: 0 }, "Blue"); } #[test] fn color_serialize() { - assert_eq!(Color::Yellow, Color::from_str("y").unwrap()); - assert_eq!(Color::Yellow, Color::from_str("yellow").unwrap()); + assert_from_str!(Color, Color::Yellow, "y"); + assert_from_str!(Color, Color::Yellow, "yellow"); } #[test] fn color_to_string() { - assert_eq!(Color::Purple, Color::from_str("purp").unwrap()); + assert_from_str!(Color, Color::Purple, "purp"); } #[test] fn color_default() { - assert_eq!( - Color::Green(String::from("not found")), - Color::from_str("not found").unwrap() - ); + assert_from_str!(Color, Color::Green(String::from("not found")), "not found"); } #[test] fn color_ascii_case_insensitive() { - assert_eq!(Color::Black, Color::from_str("BLK").unwrap()); - assert_eq!(Color::Black, Color::from_str("bLaCk").unwrap()); + assert_from_str!(Color, Color::Black, "BLK"); + assert_from_str!(Color, Color::Black, "bLaCk"); } #[derive(Debug, Eq, PartialEq, EnumString)] @@ -65,18 +80,9 @@ enum Brightness { #[test] fn brightness_serialize_all() { - assert_eq!( - Brightness::DarkBlack, - Brightness::from_str("dark_black").unwrap() - ); - assert_eq!( - Brightness::Dim { glow: 0 }, - Brightness::from_str("dim").unwrap() - ); - assert_eq!( - Brightness::BrightWhite, - Brightness::from_str("Bright").unwrap() - ); + assert_from_str!(Brightness, Brightness::DarkBlack, "dark_black"); + assert_from_str!(Brightness, Brightness::Dim { glow: 0 }, "dim"); + assert_from_str!(Brightness, Brightness::BrightWhite, "Bright"); } #[derive(Debug, Eq, PartialEq, EnumString)] @@ -100,13 +106,13 @@ fn week_not_found() { #[test] fn week_found() { - assert_eq!(Result::Ok(Week::Sunday), Week::from_str("Sunday")); - assert_eq!(Result::Ok(Week::Monday), Week::from_str("Monday")); - assert_eq!(Result::Ok(Week::Tuesday), Week::from_str("Tuesday")); - assert_eq!(Result::Ok(Week::Wednesday), Week::from_str("Wednesday")); - assert_eq!(Result::Ok(Week::Thursday), Week::from_str("Thursday")); - assert_eq!(Result::Ok(Week::Friday), Week::from_str("Friday")); - assert_eq!(Result::Ok(Week::Saturday), Week::from_str("Saturday")); + assert_from_str!(Week, Week::Sunday, "Sunday"); + assert_from_str!(Week, Week::Monday, "Monday"); + assert_from_str!(Week, Week::Tuesday, "Tuesday"); + assert_from_str!(Week, Week::Wednesday, "Wednesday"); + assert_from_str!(Week, Week::Thursday, "Thursday"); + assert_from_str!(Week, Week::Friday, "Friday"); + assert_from_str!(Week, Week::Saturday, "Saturday"); } #[derive(Debug, Eq, PartialEq, EnumString)] @@ -117,7 +123,7 @@ enum Lifetime<'a> { #[test] fn lifetime_test() { - assert_eq!(Lifetime::Life(""), Lifetime::from_str("Life").unwrap()); + assert_from_str!(Lifetime, Lifetime::Life(""), "Life"); } #[derive(Debug, Eq, PartialEq, EnumString)] @@ -128,7 +134,7 @@ enum Generic { #[test] fn generic_test() { - assert_eq!(Generic::Gen(""), Generic::from_str("Gen").unwrap()); + assert_from_str!(Generic<_>, Generic::Gen(""), "Gen"); } #[derive(Debug, Eq, PartialEq, EnumString)] @@ -143,29 +149,33 @@ enum CaseInsensitiveEnum { #[test] fn case_insensitive_enum_no_attr() { - assert_eq!( - CaseInsensitiveEnum::NoAttr, - CaseInsensitiveEnum::from_str("noattr").unwrap() - ); + assert_from_str!(CaseInsensitiveEnum, CaseInsensitiveEnum::NoAttr, "noattr"); } #[test] fn case_insensitive_enum_no_case_insensitive() { - assert_eq!( + assert_from_str!( + CaseInsensitiveEnum, CaseInsensitiveEnum::NoCaseInsensitive, - CaseInsensitiveEnum::from_str("NoCaseInsensitive").unwrap(), + "NoCaseInsensitive" ); assert!(CaseInsensitiveEnum::from_str("nocaseinsensitive").is_err()); + assert!( + >::try_from("nocaseinsensitive") + .is_err() + ); } #[test] fn case_insensitive_enum_case_insensitive() { - assert_eq!( + assert_from_str!( + CaseInsensitiveEnum, CaseInsensitiveEnum::CaseInsensitive, - CaseInsensitiveEnum::from_str("CaseInsensitive").unwrap(), + "CaseInsensitive" ); - assert_eq!( + assert_from_str!( + CaseInsensitiveEnum, CaseInsensitiveEnum::CaseInsensitive, - CaseInsensitiveEnum::from_str("caseinsensitive").unwrap(), + "caseinsensitive" ); }