use std::str::Utf8Error;
use crate::spec::ElementType;
#[derive(Debug, PartialEq, Clone)]
#[non_exhaustive]
pub struct Error {
pub kind: ErrorKind,
pub(crate) key: Option<String>,
}
impl Error {
pub(crate) fn new_with_key(key: impl Into<String>, kind: ErrorKind) -> Self {
Self {
kind,
key: Some(key.into()),
}
}
pub(crate) fn new_without_key(kind: ErrorKind) -> Self {
Self { key: None, kind }
}
pub(crate) fn with_key(mut self, key: impl AsRef<str>) -> Self {
self.key = Some(key.as_ref().to_string());
self
}
pub fn key(&self) -> Option<&str> {
self.key.as_deref()
}
}
#[derive(Clone, Debug, PartialEq)]
#[non_exhaustive]
pub enum ErrorKind {
#[non_exhaustive]
MalformedValue { message: String },
Utf8EncodingError(Utf8Error),
}
impl std::fmt::Display for Error {
fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
let p = self
.key
.as_ref()
.map(|k| format!("error at key \"{}\": ", k));
let prefix = p.as_ref().map_or("", |p| p.as_str());
match &self.kind {
ErrorKind::MalformedValue { message } => {
write!(f, "{}malformed value: {:?}", prefix, message)
}
ErrorKind::Utf8EncodingError(e) => write!(f, "{}utf-8 encoding error: {}", prefix, e),
}
}
}
impl std::error::Error for Error {}
pub type Result<T> = std::result::Result<T, Error>;
pub(crate) fn try_with_key<G, F: FnOnce() -> Result<G>>(key: impl AsRef<str>, f: F) -> Result<G> {
f().map_err(|e| e.with_key(key))
}
pub type ValueAccessResult<T> = std::result::Result<T, ValueAccessError>;
#[derive(Debug, PartialEq, Clone)]
#[non_exhaustive]
pub struct ValueAccessError {
pub kind: ValueAccessErrorKind,
pub(crate) key: String,
}
impl ValueAccessError {
pub fn key(&self) -> &str {
self.key.as_str()
}
}
#[derive(Debug, PartialEq, Clone)]
#[non_exhaustive]
pub enum ValueAccessErrorKind {
NotPresent,
#[non_exhaustive]
UnexpectedType {
expected: ElementType,
actual: ElementType,
},
InvalidBson(super::Error),
}
impl std::fmt::Display for ValueAccessError {
fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
let prefix = format!("error at key: \"{}\": ", self.key);
match &self.kind {
ValueAccessErrorKind::UnexpectedType { actual, expected } => write!(
f,
"{} unexpected element type: {:?}, expected: {:?}",
prefix, actual, expected
),
ValueAccessErrorKind::InvalidBson(error) => {
write!(f, "{}: {}", prefix, error)
}
ValueAccessErrorKind::NotPresent => write!(f, "{}value not present", prefix),
}
}
}
impl std::error::Error for ValueAccessError {}