You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
183 lines
6.9 KiB
183 lines
6.9 KiB
Anyhow ¯\\\_(°ペ)\_/¯
|
|
==========================
|
|
|
|
[<img alt="github" src="https://img.shields.io/badge/github-dtolnay/anyhow-8da0cb?style=for-the-badge&labelColor=555555&logo=github" height="20">](https://github.com/dtolnay/anyhow)
|
|
[<img alt="crates.io" src="https://img.shields.io/crates/v/anyhow.svg?style=for-the-badge&color=fc8d62&logo=rust" height="20">](https://crates.io/crates/anyhow)
|
|
[<img alt="docs.rs" src="https://img.shields.io/badge/docs.rs-anyhow-66c2a5?style=for-the-badge&labelColor=555555&logoColor=white&logo=" height="20">](https://docs.rs/anyhow)
|
|
[<img alt="build status" src="https://img.shields.io/github/workflow/status/dtolnay/anyhow/CI/master?style=for-the-badge" height="20">](https://github.com/dtolnay/anyhow/actions?query=branch%3Amaster)
|
|
|
|
This library provides [`anyhow::Error`][Error], a trait object based error type
|
|
for easy idiomatic error handling in Rust applications.
|
|
|
|
[Error]: https://docs.rs/anyhow/1.0/anyhow/struct.Error.html
|
|
|
|
```toml
|
|
[dependencies]
|
|
anyhow = "1.0"
|
|
```
|
|
|
|
*Compiler support: requires rustc 1.34+*
|
|
|
|
<br>
|
|
|
|
## Details
|
|
|
|
- Use `Result<T, anyhow::Error>`, or equivalently `anyhow::Result<T>`, as the
|
|
return type of any fallible function.
|
|
|
|
Within the function, use `?` to easily propagate any error that implements the
|
|
`std::error::Error` trait.
|
|
|
|
```rust
|
|
use anyhow::Result;
|
|
|
|
fn get_cluster_info() -> Result<ClusterMap> {
|
|
let config = std::fs::read_to_string("cluster.json")?;
|
|
let map: ClusterMap = serde_json::from_str(&config)?;
|
|
Ok(map)
|
|
}
|
|
```
|
|
|
|
- Attach context to help the person troubleshooting the error understand where
|
|
things went wrong. A low-level error like "No such file or directory" can be
|
|
annoying to debug without more context about what higher level step the
|
|
application was in the middle of.
|
|
|
|
```rust
|
|
use anyhow::{Context, Result};
|
|
|
|
fn main() -> Result<()> {
|
|
...
|
|
it.detach().context("Failed to detach the important thing")?;
|
|
|
|
let content = std::fs::read(path)
|
|
.with_context(|| format!("Failed to read instrs from {}", path))?;
|
|
...
|
|
}
|
|
```
|
|
|
|
```console
|
|
Error: Failed to read instrs from ./path/to/instrs.json
|
|
|
|
Caused by:
|
|
No such file or directory (os error 2)
|
|
```
|
|
|
|
- Downcasting is supported and can be by value, by shared reference, or by
|
|
mutable reference as needed.
|
|
|
|
```rust
|
|
// If the error was caused by redaction, then return a
|
|
// tombstone instead of the content.
|
|
match root_cause.downcast_ref::<DataStoreError>() {
|
|
Some(DataStoreError::Censored(_)) => Ok(Poll::Ready(REDACTED_CONTENT)),
|
|
None => Err(error),
|
|
}
|
|
```
|
|
|
|
- If using the nightly channel, a backtrace is captured and printed with the
|
|
error if the underlying error type does not already provide its own. In order
|
|
to see backtraces, they must be enabled through the environment variables
|
|
described in [`std::backtrace`]:
|
|
|
|
- If you want panics and errors to both have backtraces, set
|
|
`RUST_BACKTRACE=1`;
|
|
- If you want only errors to have backtraces, set `RUST_LIB_BACKTRACE=1`;
|
|
- If you want only panics to have backtraces, set `RUST_BACKTRACE=1` and
|
|
`RUST_LIB_BACKTRACE=0`.
|
|
|
|
The tracking issue for this feature is [rust-lang/rust#53487].
|
|
|
|
[`std::backtrace`]: https://doc.rust-lang.org/std/backtrace/index.html#environment-variables
|
|
[rust-lang/rust#53487]: https://github.com/rust-lang/rust/issues/53487
|
|
|
|
- Anyhow works with any error type that has an impl of `std::error::Error`,
|
|
including ones defined in your crate. We do not bundle a `derive(Error)` macro
|
|
but you can write the impls yourself or use a standalone macro like
|
|
[thiserror].
|
|
|
|
```rust
|
|
use thiserror::Error;
|
|
|
|
#[derive(Error, Debug)]
|
|
pub enum FormatError {
|
|
#[error("Invalid header (expected {expected:?}, got {found:?})")]
|
|
InvalidHeader {
|
|
expected: String,
|
|
found: String,
|
|
},
|
|
#[error("Missing attribute: {0}")]
|
|
MissingAttribute(String),
|
|
}
|
|
```
|
|
|
|
- One-off error messages can be constructed using the `anyhow!` macro, which
|
|
supports string interpolation and produces an `anyhow::Error`.
|
|
|
|
```rust
|
|
return Err(anyhow!("Missing attribute: {}", missing));
|
|
```
|
|
|
|
A `bail!` macro is provided as a shorthand for the same early return.
|
|
|
|
```rust
|
|
bail!("Missing attribute: {}", missing);
|
|
```
|
|
|
|
<br>
|
|
|
|
## No-std support
|
|
|
|
In no_std mode, the same API is almost all available and works the same way. To
|
|
depend on Anyhow in no_std mode, disable our default enabled "std" feature in
|
|
Cargo.toml. A global allocator is required.
|
|
|
|
```toml
|
|
[dependencies]
|
|
anyhow = { version = "1.0", default-features = false }
|
|
```
|
|
|
|
Since the `?`-based error conversions would normally rely on the
|
|
`std::error::Error` trait which is only available through std, no_std mode will
|
|
require an explicit `.map_err(Error::msg)` when working with a non-Anyhow error
|
|
type inside a function that returns Anyhow's error type.
|
|
|
|
<br>
|
|
|
|
## Comparison to failure
|
|
|
|
The `anyhow::Error` type works something like `failure::Error`, but unlike
|
|
failure ours is built around the standard library's `std::error::Error` trait
|
|
rather than a separate trait `failure::Fail`. The standard library has adopted
|
|
the necessary improvements for this to be possible as part of [RFC 2504].
|
|
|
|
[RFC 2504]: https://github.com/rust-lang/rfcs/blob/master/text/2504-fix-error.md
|
|
|
|
<br>
|
|
|
|
## Comparison to thiserror
|
|
|
|
Use Anyhow if you don't care what error type your functions return, you just
|
|
want it to be easy. This is common in application code. Use [thiserror] if you
|
|
are a library that wants to design your own dedicated error type(s) so that on
|
|
failures the caller gets exactly the information that you choose.
|
|
|
|
[thiserror]: https://github.com/dtolnay/thiserror
|
|
|
|
<br>
|
|
|
|
#### License
|
|
|
|
<sup>
|
|
Licensed under either of <a href="LICENSE-APACHE">Apache License, Version
|
|
2.0</a> or <a href="LICENSE-MIT">MIT license</a> at your option.
|
|
</sup>
|
|
|
|
<br>
|
|
|
|
<sub>
|
|
Unless you explicitly state otherwise, any contribution intentionally submitted
|
|
for inclusion in this crate by you, as defined in the Apache-2.0 license, shall
|
|
be dual licensed as above, without any additional terms or conditions.
|
|
</sub>
|