use crate::alloc::borrow::Cow;
#[cfg(feature = "chain-error")]
use crate::alloc::boxed::Box;
#[derive(PartialEq, Eq, Clone, Debug)]
pub struct Error {
	#[cfg(feature = "chain-error")]
	cause: Option<Box<Error>>,
	#[cfg(feature = "chain-error")]
	desc: Cow<'static, str>,
}
impl Error {
	pub fn chain(self, desc: impl Into<Cow<'static, str>>) -> Self {
		#[cfg(feature = "chain-error")]
		{
			Self { desc: desc.into(), cause: Some(Box::new(self)) }
		}
		#[cfg(not(feature = "chain-error"))]
		{
			let _ = desc;
			self
		}
	}
	#[cfg(feature = "chain-error")]
	fn display_with_indent(&self, indent: u32, f: &mut core::fmt::Formatter) -> core::fmt::Result {
		for _ in 0..indent {
			f.write_str("\t")?;
		}
		f.write_str(&self.desc)?;
		if let Some(cause) = &self.cause {
			f.write_str(":")?;
			f.write_str("\n")?;
			cause.display_with_indent(indent + 1, f)
		} else {
			if indent != 0 {
				f.write_str("\n")?;
			}
			Ok(())
		}
	}
}
impl core::fmt::Display for Error {
	fn fmt(&self, f: &mut core::fmt::Formatter) -> core::fmt::Result {
		#[cfg(feature = "chain-error")]
		{
			self.display_with_indent(0, f)
		}
		#[cfg(not(feature = "chain-error"))]
		{
			f.write_str("Codec error")
		}
	}
}
impl From<&'static str> for Error {
	fn from(desc: &'static str) -> Error {
		#[cfg(feature = "chain-error")]
		{
			Error { desc: desc.into(), cause: None }
		}
		#[cfg(not(feature = "chain-error"))]
		{
			let _ = desc;
			Error {}
		}
	}
}
#[cfg(feature = "std")]
impl std::error::Error for Error {
	fn source(&self) -> Option<&(dyn std::error::Error + 'static)> {
		#[cfg(feature = "chain-error")]
		{
			self.cause.as_ref().map(|e| e as &(dyn std::error::Error + 'static))
		}
		#[cfg(not(feature = "chain-error"))]
		{
			None
		}
	}
}
#[cfg(test)]
mod tests {
	use crate::Error;
	#[test]
	fn test_full_error() {
		let msg: &str = "final type:\n\twrap cause:\n\t\troot cause\n";
		let error = Error::from("root cause").chain("wrap cause").chain("final type");
		assert_eq!(&error.to_string(), msg);
	}
	#[test]
	fn impl_std_error() {
		use std::error::Error as _;
		let error = Error::from("root cause").chain("wrap cause").chain("final type");
		let s = error.source().unwrap();
		assert_eq!(&s.to_string(), "wrap cause:\n\troot cause\n");
	}
}