#![allow(clippy::needless_late_init, clippy::uninlined_format_args)] use core::fmt::{self, Debug, Display}; use core::str::FromStr; use thiserror::Error; pub struct NoFormat; #[derive(Debug)] pub struct DebugOnly; pub struct DisplayOnly; impl Display for DisplayOnly { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { f.write_str("display only") } } #[derive(Debug)] pub struct DebugAndDisplay; impl Display for DebugAndDisplay { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { f.write_str("debug and display") } } // Should expand to: // // impl Display for EnumDebugField // where // E: Debug; // // impl Error for EnumDebugField // where // Self: Debug + Display; // #[derive(Error, Debug)] pub enum EnumDebugGeneric { #[error("{0:?}")] FatalError(E), } // Should expand to: // // impl Display for EnumFromGeneric; // // impl Error for EnumFromGeneric // where // EnumDebugGeneric: Error + 'static, // Self: Debug + Display; // #[derive(Error, Debug)] pub enum EnumFromGeneric { #[error("enum from generic")] Source(#[from] EnumDebugGeneric), } // Should expand to: // // impl Display // for EnumCompound // where // HasDisplay: Display, // HasDebug: Debug; // // impl Error // for EnumCompound // where // Self: Debug + Display; // #[derive(Error)] pub enum EnumCompound { #[error("{0} {1:?}")] DisplayDebug(HasDisplay, HasDebug), #[error("{0}")] Display(HasDisplay, HasNeither), #[error("{1:?}")] Debug(HasNeither, HasDebug), } impl Debug for EnumCompound { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { f.write_str("EnumCompound") } } #[test] fn test_display_enum_compound() { let mut instance: EnumCompound; instance = EnumCompound::DisplayDebug(DisplayOnly, DebugOnly); assert_eq!(format!("{}", instance), "display only DebugOnly"); instance = EnumCompound::Display(DisplayOnly, NoFormat); assert_eq!(format!("{}", instance), "display only"); instance = EnumCompound::Debug(NoFormat, DebugOnly); assert_eq!(format!("{}", instance), "DebugOnly"); } // Should expand to: // // impl Display for EnumTransparentGeneric // where // E: Display; // // impl Error for EnumTransparentGeneric // where // E: Error, // Self: Debug + Display; // #[derive(Error, Debug)] pub enum EnumTransparentGeneric { #[error(transparent)] Other(E), } // Should expand to: // // impl Display for StructDebugGeneric // where // E: Debug; // // impl Error for StructDebugGeneric // where // Self: Debug + Display; // #[derive(Error, Debug)] #[error("{underlying:?}")] pub struct StructDebugGeneric { pub underlying: E, } // Should expand to: // // impl Error for StructFromGeneric // where // StructDebugGeneric: Error + 'static, // Self: Debug + Display; // #[derive(Error, Debug)] pub struct StructFromGeneric { #[from] pub source: StructDebugGeneric, } // Should expand to: // // impl Display for StructTransparentGeneric // where // E: Display; // // impl Error for StructTransparentGeneric // where // E: Error, // Self: Debug + Display; // #[derive(Error, Debug)] #[error(transparent)] pub struct StructTransparentGeneric(pub E); // Should expand to: // // impl Display for AssociatedTypeError // where // T::Err: Display; // // impl Error for AssociatedTypeError // where // Self: Debug + Display; // #[derive(Error, Debug)] pub enum AssociatedTypeError { #[error("couldn't parse matrix")] Other, #[error("couldn't parse entry: {0}")] EntryParseError(T::Err), } // Regression test for https://github.com/dtolnay/thiserror/issues/345 #[test] fn test_no_bound_on_named_fmt() { #[derive(Error, Debug)] #[error("{thing}", thing = "...")] struct Error { thing: T, } let error = Error { thing: DebugOnly }; assert_eq!(error.to_string(), "..."); } #[test] fn test_multiple_bound() { #[derive(Error, Debug)] #[error("0x{thing:x} 0x{thing:X}")] pub struct Error { thing: T, } let error = Error { thing: 0xFFi32 }; assert_eq!(error.to_string(), "0xff 0xFF"); }