1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
//! Error-handling macros for the `abscissa` framework
//!
//! This crate defines two error handling macros designed to produce formatted
//! error messages from error kind enums that implement the `Fail` trait:
//!
//! * `err!(kind, description)` creates a new `Error<Kind>` with the given
//!   description. If additional parameters are given, `description` is treated as
//!   a format string, e.g. `err!(kind, "something went wrong: {}", &wrongness)`.
//! * `fail!(kind, description)` creates a new `Error<kind>` and returns it.

/// Create a new error (of a given kind) with a formatted message
#[macro_export]
macro_rules! err {
    ($kind:path, $msg:expr) => {
        $crate::error::Error::new($crate::error::Context::new($kind), Some($msg.to_string()))
    };
    ($kind:path, $fmt:expr, $($arg:tt)+) => {
        err!($kind, &format!($fmt, $($arg)+))
    };
}

/// Create and return an error with a formatted message
#[macro_export]
macro_rules! fail {
    ($kind:path, $msg:expr) => {
        return Err(err!($kind, $msg).into());
    };
    ($kind:path, $fmt:expr, $($arg:tt)+) => {
        fail!($kind, &format!($fmt, $($arg)+));
    };
}

/// Ensure a condition holds, returning an error if it doesn't (ala assert)
#[macro_export]
macro_rules! ensure {
    ($cond:expr, $kind:path, $msg:expr) => {
        if !($cond) {
            return Err(err!($kind, $msg).into());
        }
    };
    ($cond:expr, $kind:path, $fmt:expr, $($arg:tt)+) => {
        ensure!($cond, $kind, format!($fmt, $($arg)+))
    };
}

/// Terminate the application with a fatal error, running Abscissa's shutdown hooks.
///
/// This macro is useful in cases where you don't have a particular error type
/// you'd like to use when exiting but would like to have a formatted error
/// message. If you do have a suitable error type, use `fatal_error!()` instead.
///
/// Takes the same arguments as `format!()`.
#[macro_export]
macro_rules! fatal {
    ($app:expr, $msg:expr) => {
        fatal_error!($app, ::failure::err_msg($smg))
    };
    ($app:expr, $fmt:expr, $($arg:tt)+) => {
        fatal!($app, format!($fmt, $($arg:tt)+))
    };
}

/// Terminate the application with a fatal error, running Abscissa's shutdown hooks.
#[macro_export]
macro_rules! fatal_error {
    ($app:expr, $err:expr) => {{
        let err: ::failure::Error = $err.into();
        $crate::application::exit::fatal_error($app, &err)
    }};
}