#[macro_export]
macro_rules! assert_err {
($cond:expr $(,)?) => {
match $cond {
::core::result::Result::Err(e) => e,
::core::result::Result::Ok(t) => ::core::panic!("assertion failed, expected Err(_), got Ok({:?})", t),
}
};
($cond:expr, $($arg:tt)+) => {
match $cond {
::core::result::Result::Err(e) => e,
::core::result::Result::Ok(t) => ::core::panic!("assertion failed, expected Err(_), got Ok({:?}): {}", t, ::core::format_args!($($arg)+)),
}
};
}
#[macro_export]
macro_rules! debug_assert_err {
($($arg:tt)*) => {
#[cfg(debug_assertions)]
$crate::assert_err!($($arg)*);
}
}
#[cfg(test)]
mod tests {
#[test]
fn err() {
assert_err!(Err::<(), _>(()));
}
#[test]
#[should_panic(expected = "assertion failed, expected Err(_), got Ok(())")]
fn not_err() {
assert_err!(Ok::<_, ()>(()));
}
#[test]
#[should_panic(expected = "assertion failed, expected Err(_), got Ok(()): foo")]
fn not_err_custom_message() {
assert_err!(Ok::<_, ()>(()), "foo");
}
#[test]
fn err_value_returned() {
let value = assert_err!(Err::<(), _>(42));
assert_eq!(value, 42);
}
#[test]
#[cfg_attr(not(debug_assertions), ignore = "only run in debug mode")]
fn debug_err() {
debug_assert_err!(Err::<(), _>(()));
}
#[test]
#[cfg_attr(not(debug_assertions), ignore = "only run in debug mode")]
#[should_panic(expected = "assertion failed, expected Err(_), got Ok(())")]
fn debug_not_err() {
debug_assert_err!(Ok::<_, ()>(()));
}
#[test]
#[cfg_attr(not(debug_assertions), ignore = "only run in debug mode")]
#[should_panic(expected = "assertion failed, expected Err(_), got Ok(()): foo")]
fn debug_not_err_custom_message() {
debug_assert_err!(Ok::<_, ()>(()), "foo");
}
#[test]
#[cfg_attr(debug_assertions, ignore = "only run in release mode")]
fn debug_release_not_err() {
debug_assert_err!(Ok::<_, ()>(()));
}
#[test]
fn does_not_require_err_to_impl_debug() {
enum Foo {
Bar,
}
assert_err!(Err::<(), _>(Foo::Bar));
}
#[test]
fn debug_does_not_require_err_to_impl_debug() {
#[allow(dead_code)]
enum Foo {
Bar,
}
debug_assert_err!(Err::<(), _>(Foo::Bar));
}
#[test]
fn does_not_require_err_to_impl_debug_custom_message() {
enum Foo {
Bar,
}
assert_err!(Err::<(), _>(Foo::Bar), "foo");
}
#[test]
fn debug_does_not_require_err_to_impl_debug_custom_message() {
#[allow(dead_code)]
enum Foo {
Bar,
}
debug_assert_err!(Err::<(), _>(Foo::Bar), "foo");
}
}