Motivation

Emulating checked exceptions in Rust.

Prefer smaller enum that holds errors that may haven been encountered in the function, rather than a big enum that holds the various possible errors that may have been encountered at any point anywhere in the crate.

Usage

  1. Add this crate to Cargo.toml, and enable any features you want

Cargo.toml:

toml [dependencies.cex] version = "0.4" features = ["log","pretty_log"]

src/lib.rs:

rust use cex::*;

  1. Use Result!() to enumerate the possible error types

```rust

[cex] fn throws_never() -> Result!(i32) {/**/}

struct SomeError;

[cex] fn foo() -> Result!( i32 throws String, &'static str, SomeError ) {/**/}

```

```rust

[cex] fn foo() {

let _f = #[cex] || -> Result!( i32 throws String ) {/**/}

} ```

```rust

[cex] fn foo() {

#[cex] let v: Result!( i32 throws String ) = try {/**/};

} ```

  1. Use ret!()/throw!() to return a result.

```rust

[cex] fn foo() -> Result!( i32 throws String ) {

ret!( 42 ); // Ok-wrapping

} ```

```rust

[cex] fn foo() -> Result!( i32 throws String ) {

ret!( Ok( 42 ));
// or ret!( Err( String::from( "oops" )))

} ```

```rust

[cex] fn foo() -> Result!( i32 throws String, SomeError ) {

throw!( String::from( "oops" ))
// or throw!( SomeError )

}

[cex] fn bar() -> Result!( i32 throws String, &'static str, SomeError ) {

match foo() {
    Ok(v) => ret!(v),
    Err(e) => throw!(e), // all errors in foo()'s throws are in bar()'s
}

}

// equivalent to bar()

[cex] fn baz() -> Result!( i32 throws String, &'static str, SomeError ) {

ret!( foo()? ) // of course you can use `?` to propagate errors

} ```

  1. Use #[ty_pat] match to map errors returned by #[cex] functions or closures.

```rust

[cex] fn foo() -> Result!( () throws String, SomeError ) {/**/}

[cex] fn bar() {

if let Err( err ) = foo() {
    #[ty_pat] match err {
        String( s ) => println!( "foo's error:{}", s ),
        SomeError => println!( "foo's error: SomeError" ),
    }
}

} ```

```rust

[cex] fn foo() -> Result!( i32 throws &'static str, SomeError ) {/**/}

[cex] fn bar() {

if let Err( err ) = foo() {
    #[ty_pat] match err {
        TyPat::<&'static str>( s ) => println!( "foo's error:{}", s ),
        SomeError => println!( "foo's error: SomeError" ),
    }
}

} ```

```rust

[cex] fn foo() -> Result!( i32 throws String, SomeError ) {/**/}

[cex] fn bar() -> Result!( i32 throws String ) {

foo().or_else( |err| #[ty_pat(gen_throws)] match err {
    SomeError => ret!(0),
    // generated arm: String(s) => throw!(s),
})

} ```

```rust

[cex] fn foo() -> Result!( i32 throws String, SomeError ) {/**/}

[cex] fn bar() -> Result!( i32 throws String ) {

foo().or_else( |err| #[ty_pat(gen String)] match err {
    SomeError => ret!(0),
    // generated arm: String(s) => throw!(s),
})

} ```

Backtrace

Backtrace is disabled by default. When enabled, locations of error propagation by ret!(), throw!() and ? operator will be stored in the Err variant.

toml [dependencies.cex] version = "0.4" features = ["env_log"]

```toml [dependencies.cex] version = "0.4" features = ["log","pretty_log"]

or features = ["envlog","prettylog"]

```

```rust use cex::*;

[cex]

pub fn foo() -> Result!( () throws () ) { throw!( () ); }

[cex]

pub fn bar() -> Result!( () throws () ) { ret!( foo()? ); }

fn main() { bar().unwrap(); } ```

The output is similar as follows:

thread 'main' panicked at 'called `Result::unwrap()` on an `Err` value: _0(Log { error: (), agent: [ Frame { module: "my_program", file: "src/main.rs", line: 5, column: 13, info: Some( "throw!(())", ), }, Frame { module: "my_program", file: "src/main.rs", line: 10, column: 11, info: Some( "foo()", ), }, ], })', src/main.rs:14:5

toml [features] log = ["cex/log"] env_log = ["cex/env_log"] pretty_log = ["cex/pretty_log"]

rust ret!( expr, || frame!( "expect an ok value" )); throw!( expr, || frame!( "oops" ));

Even if backtrace is disabled at compile time, these will compile. The second argument just has no effect.

Summary

A Result!() macro, translating Result!( OkType throws A,B,.. ) into Result<OkType, Enum!(A,B,..)>. The Enum!(A,B,..) denotes an enum the variants of which are A,B,.. , etc. It is usually the return type of functions, closures or try blocks.

A throw!( expr ) macro invocation exits the function with Result::Err, the value of which is converted from expr.

A ret!( expr ) macro invocation does early return with the value of Ok( expr ), or expr. In other words, it provides Ok-wrapping as needed.

A #[ty_pat] match provides "type as pattern matching" feature in the match's arms, to consume the result returned by functions/closures/try blocks with #[cex] attributes.

The valid forms of #[ty_pat] are:

The #[cex] attribute on a function/closure/local let-bingding defines a scope that may hold a Result!() type. The #[cex] scopes could be nested. The inner's Result!() overrides the outer's.

Notice

All the features provided by this crate work with stable Rust, including #[cex] closures/let-bindings and #[ty_pat] match.

License

Under MIT.