value_bag/internal/
error.rsuse crate::{
fill::Slot,
std::{any::Any, error},
ValueBag,
};
use super::Internal;
impl<'v> ValueBag<'v> {
pub fn capture_error<T>(value: &'v T) -> Self
where
T: error::Error + 'static,
{
ValueBag {
inner: Internal::Error(value),
}
}
#[inline]
pub const fn from_dyn_error(value: &'v (dyn error::Error + 'static)) -> Self {
ValueBag {
inner: Internal::AnonError(value),
}
}
#[inline]
pub fn to_borrowed_error(&self) -> Option<&'v (dyn Error + 'static)> {
match self.inner {
Internal::Error(value) => Some(value.as_super()),
Internal::AnonError(value) => Some(value),
_ => None,
}
}
}
#[cfg(feature = "error")]
pub(crate) trait DowncastError {
fn as_any(&self) -> &dyn Any;
fn as_super(&self) -> &(dyn error::Error + 'static);
}
#[cfg(feature = "error")]
impl<T: error::Error + 'static> DowncastError for T {
fn as_any(&self) -> &dyn Any {
self
}
fn as_super(&self) -> &(dyn error::Error + 'static) {
self
}
}
impl<'s, 'f> Slot<'s, 'f> {
pub fn fill_error<T>(self, value: T) -> Result<(), crate::Error>
where
T: error::Error + 'static,
{
self.fill(|visitor| visitor.error(&value))
}
pub fn fill_dyn_error(self, value: &(dyn error::Error + 'static)) -> Result<(), crate::Error> {
self.fill(|visitor| visitor.error(value))
}
}
pub use self::error::Error;
#[cfg(feature = "owned")]
pub(crate) mod owned {
use crate::std::{boxed::Box, error, fmt, string::ToString};
#[derive(Clone, Debug)]
pub(crate) struct OwnedError {
display: Box<str>,
source: Option<Box<OwnedError>>,
}
impl fmt::Display for OwnedError {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
fmt::Display::fmt(&self.display, f)
}
}
impl error::Error for OwnedError {
fn source(&self) -> Option<&(dyn error::Error + 'static)> {
if let Some(ref source) = self.source {
Some(&**source)
} else {
None
}
}
}
pub(crate) fn buffer(err: &(dyn error::Error + 'static)) -> OwnedError {
OwnedError {
display: err.to_string().into(),
source: err.source().map(buffer).map(Box::new),
}
}
}
impl<'v> From<&'v (dyn error::Error + 'static)> for ValueBag<'v> {
#[inline]
fn from(v: &'v (dyn error::Error + 'static)) -> Self {
ValueBag::from_dyn_error(v)
}
}
impl<'v> From<Option<&'v (dyn error::Error + 'static)>> for ValueBag<'v> {
#[inline]
fn from(v: Option<&'v (dyn error::Error + 'static)>) -> Self {
ValueBag::from_option(v)
}
}
impl<'v> TryFrom<ValueBag<'v>> for &'v (dyn error::Error + 'static) {
type Error = crate::Error;
#[inline]
fn try_from(v: ValueBag<'v>) -> Result<Self, Self::Error> {
v.to_borrowed_error()
.ok_or_else(|| Self::Error::msg("conversion failed"))
}
}
impl<'v, 'u> From<&'v &'u (dyn error::Error + 'static)> for ValueBag<'v>
where
'u: 'v,
{
#[inline]
fn from(v: &'v &'u (dyn error::Error + 'static)) -> Self {
ValueBag::from_dyn_error(*v)
}
}
#[cfg(test)]
mod tests {
#[cfg(target_arch = "wasm32")]
use wasm_bindgen_test::*;
use super::*;
use crate::{
std::{io, string::ToString},
test::*,
};
#[test]
#[cfg_attr(target_arch = "wasm32", wasm_bindgen_test)]
fn error_capture() {
let err = io::Error::from(io::ErrorKind::Other);
assert_eq!(
err.to_string(),
ValueBag::capture_error(&err)
.to_borrowed_error()
.expect("invalid value")
.to_string()
);
assert_eq!(
err.to_string(),
ValueBag::from_dyn_error(&err)
.to_borrowed_error()
.expect("invalid value")
.to_string()
);
}
#[test]
#[cfg_attr(target_arch = "wasm32", wasm_bindgen_test)]
fn error_downcast() {
let err = io::Error::from(io::ErrorKind::Other);
assert!(ValueBag::capture_error(&err)
.downcast_ref::<io::Error>()
.is_some());
}
#[test]
#[cfg_attr(target_arch = "wasm32", wasm_bindgen_test)]
fn error_visit() {
let err = io::Error::from(io::ErrorKind::Other);
ValueBag::from_dyn_error(&err)
.visit(TestVisit::default())
.expect("failed to visit value");
}
}