use core::fmt;
pub type Result<T> = core::result::Result<T, Error>;
#[derive(Debug)]
#[non_exhaustive]
pub enum Error {
Io(std::io::Error),
InvalidConfig(&'static str),
NotImplemented,
}
impl fmt::Display for Error {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
Self::Io(err) => write!(f, "iqdb: io error ({})", err.kind()),
Self::InvalidConfig(msg) => write!(f, "iqdb: invalid configuration ({msg})"),
Self::NotImplemented => f.write_str("iqdb: not implemented"),
}
}
}
impl std::error::Error for Error {}
impl From<std::io::Error> for Error {
fn from(value: std::io::Error) -> Self {
Self::Io(value)
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn error_implements_std_error() {
fn assert_error<E: std::error::Error>() {}
assert_error::<Error>();
}
#[test]
fn io_error_display_does_not_leak_payload() {
let err = Error::Io(std::io::Error::new(
std::io::ErrorKind::PermissionDenied,
"secret",
));
let msg = format!("{err}");
assert!(msg.contains("permission denied") || msg.contains("PermissionDenied"));
assert!(!msg.contains("secret"));
}
#[test]
fn invalid_config_display_is_stable() {
let msg = format!("{}", Error::InvalidConfig("bad path"));
assert!(msg.contains("invalid configuration"));
assert!(msg.contains("bad path"));
}
#[test]
fn not_implemented_display_is_stable() {
let msg = format!("{}", Error::NotImplemented);
assert!(msg.contains("not implemented"));
}
#[test]
fn from_io_maps_to_io_variant() {
let err: Error = std::io::Error::new(std::io::ErrorKind::NotFound, "missing").into();
assert!(matches!(err, Error::Io(_)));
}
}