Skip to content

Commit 9a56456

Browse files
committed
update original
1 parent 209d89b commit 9a56456

File tree

7 files changed

+92
-115
lines changed

7 files changed

+92
-115
lines changed

rust-cookbook/Cargo.toml

+27-27
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
[package]
22
name = "rust-cookbook"
3-
version = "1.0.0"
3+
version = "1.1.0"
44
authors = ["Brian Anderson <[email protected]>", "Andrew Gauger <[email protected]>"]
55
edition = "2018"
66
license = "MIT/Apache-2.0"
@@ -21,7 +21,6 @@ crossbeam-channel = "0.3.9"
2121
csv = "1.0"
2222
data-encoding = "2.1.0"
2323
env_logger = "0.11.3"
24-
error-chain = "0.12"
2524
flate2 = "1.0"
2625
glob = "0.3"
2726
image = "0.20"
@@ -30,42 +29,43 @@ log = "0.4"
3029
log4rs = "0.8"
3130
memmap = "0.7"
3231
mime = "0.3"
33-
nalgebra = { version = "0.16.12", features = ["serde-serialize"] }
34-
ndarray = { version = "0.13", features = ["approx"] }
35-
num = "0.2"
36-
num_cpus = "1.8"
37-
percent-encoding = "2.1"
38-
petgraph = "0.4"
39-
postgres = "0.17.2"
40-
rand = "0.8.0"
41-
rand_distr = "0.4.0"
42-
rayon = "1.0"
43-
regex = "1.0"
44-
reqwest = { version = "0.10", features = ["blocking", "json", "stream"] }
45-
ring = "0.16.11"
46-
rusqlite = { version = "0.25", features = ["chrono"] }
32+
nalgebra = { version = "0.33", features = ["serde-serialize"] }
33+
ndarray = { version = "0.16", features = ["approx"] }
34+
num = "0.4"
35+
num_cpus = "1.16"
36+
percent-encoding = "2.3"
37+
petgraph = "0.6"
38+
postgres = "0.19"
39+
rand = "0.8"
40+
rand_distr = "0.4"
41+
rayon = "1.10"
42+
regex = "1.11"
43+
reqwest = { version = "0.12", features = ["blocking", "json", "stream"] }
44+
ring = "0.17"
45+
rusqlite = { version = "0.32", features = ["chrono"] }
4746
same-file = "1.0"
48-
select = "0.4"
49-
semver = "0.9"
47+
select = "0.6"
48+
semver = "1.0"
5049
serde = { version = "1.0", features = ["derive"] }
5150
serde_derive = "1.0"
5251
serde_json = "1.0"
53-
tar = "0.4.12"
54-
tempfile = "3.1"
55-
threadpool = "1.6"
56-
toml = "0.4"
57-
tokio = { version = "0.2", features = ["full"] }
52+
tar = "0.4"
53+
tempfile = "3.14"
54+
thiserror = "2"
55+
threadpool = "1.8"
56+
toml = "0.8"
57+
tokio = { version = "1", features = ["full"] }
5858
unicode-segmentation = "1.2.1"
59-
url = "2.1"
60-
walkdir = "2.0"
59+
url = "2.5"
60+
walkdir = "2.5"
6161

6262
[target.'cfg(target_os = "linux")'.dependencies]
6363
syslog = "5.0"
6464

6565
[build-dependencies]
6666
skeptic = "0.13"
67-
walkdir = "2.0"
67+
walkdir = "2.5"
6868

6969
[dev-dependencies]
7070
skeptic = "0.13"
71-
walkdir = "2.0"
71+
walkdir = "2.5"

rust-cookbook/src/about.md

+11-71
Original file line numberDiff line numberDiff line change
@@ -96,78 +96,15 @@ documentation on [docs.rs], and is often the next documentation you
9696
should read after deciding which crate suites your purpose.
9797

9898
## A note about error handling
99+
Rust has [`std::error::Trait`] which is implemented to handle exceptions.
100+
Handling multiple types of these traits can be simplified using [`anyhow`]
101+
or specified with an `enum` which macros exist to make this easier within
102+
[`thiserror`] for library authors.
99103

100-
Error handling in Rust is robust when done correctly, but in today's
101-
Rust it requires a fair bit of boilerplate. Because of this one often
102-
sees Rust examples filled with `unwrap` calls instead of proper error
103-
handling.
104-
105-
Since these recipes are intended to be reused as-is and encourage best
106-
practices, they set up error handling correctly when there are
107-
`Result` types involved.
108-
109-
The basic pattern we use is to have a `fn main() -> Result`.
110-
111-
The structure generally looks like:
112-
113-
```rust,edition2018
114-
use error_chain::error_chain;
115-
use std::net::IpAddr;
116-
use std::str;
117-
118-
error_chain! {
119-
foreign_links {
120-
Utf8(std::str::Utf8Error);
121-
AddrParse(std::net::AddrParseError);
122-
}
123-
}
124-
125-
fn main() -> Result<()> {
126-
let bytes = b"2001:db8::1";
127-
128-
// Bytes to string.
129-
let s = str::from_utf8(bytes)?;
130-
131-
// String to IP address.
132-
let addr: IpAddr = s.parse()?;
133-
134-
println!("{:?}", addr);
135-
Ok(())
136-
}
137-
138-
```
139-
140-
This is using the `error_chain!` macro to define a custom `Error` and
141-
`Result` type, along with automatic conversions from two standard
142-
library error types. The automatic conversions make the `?` operator
143-
work.
144-
145-
For the sake of readability error handling boilerplate is hidden by
146-
default like below. In order to read full contents click on the
147-
"Show hidden lines" (<i class="fa fa-eye"></i>) button located in the top
148-
right corner of the snippet.
149-
150-
```rust,edition2018
151-
# use error_chain::error_chain;
152-
153-
use url::{Url, Position};
154-
#
155-
# error_chain! {
156-
# foreign_links {
157-
# UrlParse(url::ParseError);
158-
# }
159-
# }
160-
161-
fn main() -> Result<()> {
162-
let parsed = Url::parse("https://httpbin.org/cookies/set?k2=v2&k1=v1")?;
163-
let cleaned: &str = &parsed[..Position::AfterPath];
164-
println!("cleaned: {}", cleaned);
165-
Ok(())
166-
}
167-
```
168-
169-
For more background on error handling in Rust, read [this page of the
170-
Rust book][error-docs] and [this blog post][error-blog].
104+
Error chain has been shown in this book for historical reasons before Rust
105+
`std` and crates represented macro use as a preference. For more background
106+
on error handling in Rust, read [this page of the Rust book][error-docs]
107+
and [this blog post][error-blog].
171108

172109
## A note about crate representation
173110

@@ -199,4 +136,7 @@ as are crates that are pending evaluation.
199136
[crates.io]: https://crates.io
200137
[docs.rs]: https://docs.rs
201138
[Cargo.toml]: http://doc.crates.io/manifest.html
139+
[`anyhow`]: https://docs.rs/anyhow/latest/anyhow/
202140
[`cargo-edit`]: https://github.com/killercup/cargo-edit
141+
[`std::error::Trait`]: https://doc.rust-lang.org/std/error/trait.Error.html
142+
[`thiserror`]: https://docs.rs/thiserror/latest/thiserror/

rust-cookbook/src/development_tools/build_tools/cc-bundled-static.md

+2-8
Original file line numberDiff line numberDiff line change
@@ -25,7 +25,7 @@ build = "build.rs"
2525
cc = "1"
2626

2727
[dependencies]
28-
error-chain = "0.11"
28+
anyhow = "1"
2929
```
3030

3131
### `build.rs`
@@ -56,16 +56,10 @@ void greet(const char* name) {
5656
### `src/main.rs`
5757
5858
```rust,edition2018,ignore
59-
use error_chain::error_chain;
59+
use anyhow::Result;
6060
use std::ffi::CString;
6161
use std::os::raw::c_char;
6262
63-
error_chain! {
64-
foreign_links {
65-
NulError(::std::ffi::NulError);
66-
Io(::std::io::Error);
67-
}
68-
}
6963
fn prompt(s: &str) -> Result<String> {
7064
use std::io::Write;
7165
print!("{}", s);

rust-cookbook/src/errors.md

+4-4
Original file line numberDiff line numberDiff line change
@@ -2,12 +2,12 @@
22

33
| Recipe | Crates | Categories |
44
|--------|--------|------------|
5+
| [Composing errors with an enum][ex-thiserror] | [![thiserror-badge]][thiserror] | [![cat-rust-patterns-badge]][cat-rust-patterns] |
6+
| [Dynamic errors with anyhow][ex-anyhow] | [![anyhow-badge]][anyhow] | [![cat-rust-patterns-badge]][cat-rust-patterns] |
57
| [Handle errors correctly in main][ex-error-chain-simple-error-handling] | [![error-chain-badge]][error-chain] | [![cat-rust-patterns-badge]][cat-rust-patterns] |
6-
| [Avoid discarding errors during error conversions][ex-error-chain-avoid-discarding] | [![error-chain-badge]][error-chain] | [![cat-rust-patterns-badge]][cat-rust-patterns] |
7-
| [Obtain backtrace of complex error scenarios][ex-error-chain-backtrace] | [![error-chain-badge]][error-chain] | [![cat-rust-patterns-badge]][cat-rust-patterns] |
88

9+
[ex-thiserror]: errors/handle.html#thiserror
10+
[ex-anyhow]: errors/handle.html#anyhow
911
[ex-error-chain-simple-error-handling]: errors/handle.html#handle-errors-correctly-in-main
10-
[ex-error-chain-avoid-discarding]: errors/handle.html#avoid-discarding-errors-during-error-conversions
11-
[ex-error-chain-backtrace]: errors/handle.html#obtain-backtrace-of-complex-error-scenarios
1212

1313
{{#include links.md}}

rust-cookbook/src/errors/handle.md

-4
Original file line numberDiff line numberDiff line change
@@ -2,8 +2,4 @@
22

33
{{#include handle/main.md}}
44

5-
{{#include handle/retain.md}}
6-
7-
{{#include handle/backtrace.md}}
8-
95
{{#include ../links.md}}

rust-cookbook/src/errors/handle/main.md

+44-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,46 @@
11
## Handle errors correctly in main
22

3+
[![anyhow-badge]][anyhow] [![cat-rust-patterns-badge]][cat-rust-patterns]
4+
[![thiserror-badge]][thiserror] [![cat-rust-patterns-badge]][cat-rust-patterns]
35
[![error-chain-badge]][error-chain] [![cat-rust-patterns-badge]][cat-rust-patterns]
46

7+
# Error Strategies (2024)
8+
9+
As recommended in Rust by Example, [`Box`ing errors] is seen as an easy
10+
strategy for getting started.
11+
12+
```rust,edition2018
13+
Box<dyn Error>
14+
````
15+
16+
To understand what kind of error handling may be required study [Designing
17+
error types in Rust] and consider [`thiserror`] for libraries or [`anyhow`] as
18+
a maintained error aggregation option.
19+
20+
```rust,edition2018
21+
use thiserror::Error;
22+
23+
#[derive(Error,Debug)]
24+
pub enum MultiError {
25+
#[error("🦀 got {0}")]
26+
ErrorClass(String),
27+
}
28+
```
29+
30+
Application authors can compose enums using `anyhow` can import the `Result`
31+
type from the crate to provide auto-`Box`ing behavior
32+
33+
```rust,edition2018,should_panic
34+
use anyhow::Result;
35+
36+
fn main() -> Result<()> {
37+
let my_string = "yellow".to_string();
38+
let _my_int = my_string.parse::<i32>()?;
39+
Ok(())
40+
}
41+
```
42+
43+
# Error Chain (2015-2018)
544
Handles error that occur when trying to open a file that does not
645
exist. It is achieved by using [error-chain], a library that takes
746
care of a lot of boilerplate code needed in order to [handle errors in Rust].
@@ -17,7 +56,7 @@ first number. Returns uptime unless there is an error.
1756
Other recipes in this book will hide the [error-chain] boilerplate, and can be
1857
seen by expanding the code with the ⤢ button.
1958

20-
```rust,edition2018
59+
```rust,edition2018,ignore
2160
use error_chain::error_chain;
2261
2362
use std::fs::File;
@@ -49,9 +88,13 @@ fn main() {
4988
}
5089
```
5190

91+
[`anyhow`]: https://docs.rs/anyhow/latest/anyhow/
5292
[`error_chain!`]: https://docs.rs/error-chain/*/error_chain/macro.error_chain.html
5393
[`Error`]: https://doc.rust-lang.org/std/error/trait.Error.html
5494
[`foreign_links`]: https://docs.rs/error-chain/*/error_chain/#foreign-links
5595
[`std::io::Error`]: https://doc.rust-lang.org/std/io/struct.Error.html
96+
[`thiserror`]: https://docs.rs/thiserror/latest/thiserror/
5697

5798
[handle errors in Rust]: https://doc.rust-lang.org/book/second-edition/ch09-00-error-handling.html
99+
[`Box`ing errors]: https://doc.rust-lang.org/rust-by-example/error/multiple_error_types/boxing_errors.html
100+
[Designing error types in Rust]: https://mmapped.blog/posts/12-rust-error-handling

rust-cookbook/src/links.md

+4
Original file line numberDiff line numberDiff line change
@@ -52,6 +52,8 @@ Keep lines sorted.
5252

5353
[ansi_term-badge]: https://badge-cache.kominick.com/crates/v/ansi_term.svg?label=ansi_term
5454
[ansi_term]: https://docs.rs/ansi_term/
55+
[anyhow-badge]: https://badge-cache.kominick.com/crates/v/anyhow.svg?label=anyhow
56+
[anyhow]: https://docs.rs/anyhow/
5557
[base64-badge]: https://badge-cache.kominick.com/crates/v/base64.svg?label=base64
5658
[base64]: https://docs.rs/base64/
5759
[bitflags-badge]: https://badge-cache.kominick.com/crates/v/bitflags.svg?label=bitflags
@@ -135,6 +137,8 @@ Keep lines sorted.
135137
[tar]: https://docs.rs/tar/
136138
[tempfile-badge]: https://badge-cache.kominick.com/crates/v/tempfile.svg?label=tempfile
137139
[tempfile]: https://docs.rs/tempfile/
140+
[thiserror-badge]: https://badge-cache.kominick.com/crates/v/thiserror.svg?label=thiserror
141+
[thiserror]: https://docs.rs/thiserror/
138142
[threadpool-badge]: https://badge-cache.kominick.com/crates/v/threadpool.svg?label=threadpool
139143
[threadpool]: https://docs.rs/threadpool/
140144
[toml-badge]: https://badge-cache.kominick.com/crates/v/toml.svg?label=toml

0 commit comments

Comments
 (0)