Skip to content

Commit d3976cd

Browse files
committed
Show source by default, add feature to hide it
close andrewhickman#59 Changes the default behavior so that the output in the readme now matches the output a user would see when they use the library. Specifically, it now includes the original `std::io::Error` in the output. It introduces a feature `anyhow`. By default: - By default: fs-err will include `std::io::Error` in the Display output, and return `None` from `Error::source()` - With `anyhow` fs-err will not include std::io::Error in the Display output, and return `Some(std::io::Error)` from `Error::source()` This is based on the guidance from andrewhickman#51. That discussion links to this 2020 discussion rust-lang/project-error-handling#23 that suggests that you should either print the source and return `None` from `Error::source()` (https://doc.rust-lang.org/std/error/trait.Error.html#method.source) or not print anything and return `Some(E)`. This allows users of anyhow (or similar) libraries to configure fs-err for the behavior they desire, while not hiding information by default from unsuspecting adopters that are not using those libraries. It optimizes for the case of accidentally showing extra information that a user can then investigate and disable, rather than hiding information that the user might not realize is missing.
1 parent 25ac421 commit d3976cd

File tree

5 files changed

+40
-2
lines changed

5 files changed

+40
-2
lines changed

.github/workflows/ci.yml

+7
Original file line numberDiff line numberDiff line change
@@ -67,3 +67,10 @@ jobs:
6767
command: check
6868
args: --features tokio
6969
if: ${{ matrix.rust == 'stable' || matrix.rust == 'beta' }}
70+
71+
- name: cargo check --features anyhow
72+
uses: actions-rs/cargo@v1
73+
with:
74+
command: check
75+
args: --features anyhow
76+
if: ${{ matrix.rust == 'stable' || matrix.rust == 'beta' }}

CHANGELOG.md

+2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,7 @@
11
# fs-err Changelog
22

3+
* Change errors to output original `std::io::Error` information Display by default. This functionality can be disabled for [anyhow](https://docs.rs/anyhow/latest/anyhow/) users by using the new feature `anyhow` ([]()).
4+
35
## 2.11.0
46

57
* Added the first line of the standard library documentation to each function's rustdocs, to make them more useful in IDEs ([#50](https://github.com/andrewhickman/fs-err/issues/45))

Cargo.toml

+3
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,9 @@ serde_json = "1.0.64"
2323
[features]
2424
# Adds I/O safety traits, introduced in Rust 1.63
2525
io_safety = []
26+
# Removes the original error text from Display and relies on a wrapper library
27+
# (such as anyhow) to emit them via `Error::source()`.
28+
anyhow = []
2629

2730
[package.metadata.release]
2831
tag-name = "{{version}}"

src/errors.rs

+24-2
Original file line numberDiff line numberDiff line change
@@ -99,7 +99,12 @@ impl fmt::Display for Error {
9999
E::ReadAt => write!(formatter, "failed to read with offset from `{}`", path),
100100
#[cfg(unix)]
101101
E::WriteAt => write!(formatter, "failed to write with offset to `{}`", path),
102-
}
102+
}?;
103+
104+
#[cfg(not(feature = "anyhow"))]
105+
write!(formatter, " caused by: {}", self.source)?;
106+
107+
Ok(())
103108
}
104109
}
105110

@@ -108,6 +113,12 @@ impl StdError for Error {
108113
self.source()
109114
}
110115

116+
#[cfg(not(feature = "anyhow"))]
117+
fn source(&self) -> Option<&(dyn StdError + 'static)> {
118+
None
119+
}
120+
121+
#[cfg(feature = "anyhow")]
111122
fn source(&self) -> Option<&(dyn StdError + 'static)> {
112123
Some(&self.source)
113124
}
@@ -188,7 +199,12 @@ impl fmt::Display for SourceDestError {
188199
SourceDestErrorKind::SymlinkDir => {
189200
write!(formatter, "failed to symlink dir from {} to {}", from, to)
190201
}
191-
}
202+
}?;
203+
204+
#[cfg(not(feature = "anyhow"))]
205+
write!(formatter, " caused by: {}", self.source)?;
206+
207+
Ok(())
192208
}
193209
}
194210

@@ -197,6 +213,12 @@ impl StdError for SourceDestError {
197213
self.source()
198214
}
199215

216+
#[cfg(not(feature = "anyhow"))]
217+
fn source(&self) -> Option<&(dyn StdError + 'static)> {
218+
None
219+
}
220+
221+
#[cfg(feature = "anyhow")]
200222
fn source(&self) -> Option<&(dyn StdError + 'static)> {
201223
Some(&self.source)
202224
}

src/lib.rs

+4
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,10 @@ failed to open file `does not exist.txt`
2626
caused by: The system cannot find the file specified. (os error 2)
2727
```
2828
29+
> Note: To bypass displaying the original error message you can enable the `anyhow` feature.
30+
> When the `anyhow` feature is enabled `Error::source()` will return `Some` and the original
31+
> error will not be `Display`-ed via fs-err.
32+
2933
# Usage
3034
3135
fs-err's API is the same as [`std::fs`][std::fs], so migrating code to use it is easy.

0 commit comments

Comments
 (0)