forked from mirror/async-std
verbose errors feature
This adds a new "verbose-errors" feature flag to async-std that enables wrapping certain errors in structures with more context. As an example, we use it in `fs::File::{open,create}` to add the given path to the error message (something that is lacking in std to annoyance of many).
This commit is contained in:
parent
355e2eded8
commit
8ce3e78952
7 changed files with 98 additions and 2 deletions
6
.github/workflows/ci.yml
vendored
6
.github/workflows/ci.yml
vendored
|
@ -64,6 +64,12 @@ jobs:
|
|||
command: test
|
||||
args: --all --features unstable
|
||||
|
||||
- name: tests with verbose errors
|
||||
uses: actions-rs/cargo@v1
|
||||
with:
|
||||
command: test
|
||||
args: --all --features 'unstable verbose-errors'
|
||||
|
||||
check_fmt_and_docs:
|
||||
name: Checking fmt and docs
|
||||
runs-on: ubuntu-latest
|
||||
|
|
|
@ -48,6 +48,7 @@ std = [
|
|||
"pin-utils",
|
||||
"slab",
|
||||
]
|
||||
verbose-errors = []
|
||||
|
||||
[dependencies]
|
||||
async-attributes = { version = "1.1.1", optional = true }
|
||||
|
|
|
@ -9,6 +9,7 @@ use std::sync::{Arc, Mutex};
|
|||
|
||||
use crate::fs::{Metadata, Permissions};
|
||||
use crate::future;
|
||||
use crate::utils::VerboseErrorExt;
|
||||
use crate::io::{self, Read, Seek, SeekFrom, Write};
|
||||
use crate::path::Path;
|
||||
use crate::prelude::*;
|
||||
|
@ -112,7 +113,11 @@ impl File {
|
|||
/// ```
|
||||
pub async fn open<P: AsRef<Path>>(path: P) -> io::Result<File> {
|
||||
let path = path.as_ref().to_owned();
|
||||
let file = spawn_blocking(move || std::fs::File::open(&path)).await?;
|
||||
let file = spawn_blocking(move || {
|
||||
std::fs::File::open(&path)
|
||||
.verbose_context(|| format!("Could not open {}", path.display()))
|
||||
})
|
||||
.await?;
|
||||
Ok(File::new(file, true))
|
||||
}
|
||||
|
||||
|
@ -147,7 +152,11 @@ impl File {
|
|||
/// ```
|
||||
pub async fn create<P: AsRef<Path>>(path: P) -> io::Result<File> {
|
||||
let path = path.as_ref().to_owned();
|
||||
let file = spawn_blocking(move || std::fs::File::create(&path)).await?;
|
||||
let file = spawn_blocking(move || {
|
||||
std::fs::File::create(&path)
|
||||
.verbose_context(|| format!("Could not create {}", path.display()))
|
||||
})
|
||||
.await?;
|
||||
Ok(File::new(file, true))
|
||||
}
|
||||
|
||||
|
|
|
@ -291,6 +291,7 @@ cfg_std! {
|
|||
pub(crate) mod read;
|
||||
pub(crate) mod seek;
|
||||
pub(crate) mod write;
|
||||
pub(crate) mod utils;
|
||||
|
||||
mod buf_reader;
|
||||
mod buf_writer;
|
||||
|
|
51
src/io/utils.rs
Normal file
51
src/io/utils.rs
Normal file
|
@ -0,0 +1,51 @@
|
|||
use std::{error::Error, fmt, io};
|
||||
use crate::utils::VerboseErrorExt;
|
||||
|
||||
/// Wrap `std::io::Error` with additional message
|
||||
///
|
||||
/// *Note* Only active when `verbose-errors` feature is enabled for this crate!
|
||||
///
|
||||
/// Keeps the original error kind and stores the original I/O error as `source`.
|
||||
impl<T> VerboseErrorExt for Result<T, io::Error> {
|
||||
fn verbose_context(self, message: impl Fn() -> String) -> Self {
|
||||
if cfg!(feature = "verbose-errors") {
|
||||
self.map_err(|e| VerboseError::wrap(e, message()))
|
||||
} else {
|
||||
self
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Debug)]
|
||||
struct VerboseError {
|
||||
source: io::Error,
|
||||
message: String,
|
||||
}
|
||||
|
||||
impl VerboseError {
|
||||
fn wrap(source: io::Error, message: impl Into<String>) -> io::Error {
|
||||
io::Error::new(
|
||||
source.kind(),
|
||||
VerboseError {
|
||||
source,
|
||||
message: message.into(),
|
||||
},
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
impl fmt::Display for VerboseError {
|
||||
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||
write!(f, "{}", self.message)
|
||||
}
|
||||
}
|
||||
|
||||
impl Error for VerboseError {
|
||||
fn description(&self) -> &str {
|
||||
self.source.description()
|
||||
}
|
||||
|
||||
fn source(&self) -> Option<&(dyn Error + 'static)> {
|
||||
Some(&self.source)
|
||||
}
|
||||
}
|
|
@ -52,6 +52,14 @@ pub fn random(n: u32) -> u32 {
|
|||
})
|
||||
}
|
||||
|
||||
/// Add additional context to errors
|
||||
///
|
||||
/// *Note for implementors:* The given closure must only be executed when
|
||||
/// `verbose-errors` feature is enabled for this crate!
|
||||
pub(crate) trait VerboseErrorExt {
|
||||
fn verbose_context(self, message: impl Fn() -> String) -> Self;
|
||||
}
|
||||
|
||||
/// Defers evaluation of a block of code until the end of the scope.
|
||||
#[cfg(feature = "default")]
|
||||
#[doc(hidden)]
|
||||
|
|
20
tests/verbose_errors.rs
Normal file
20
tests/verbose_errors.rs
Normal file
|
@ -0,0 +1,20 @@
|
|||
#[cfg(feature = "verbose-errors")]
|
||||
mod verbose_tests {
|
||||
use async_std::{fs, task};
|
||||
|
||||
#[test]
|
||||
fn open_file() {
|
||||
task::block_on(async {
|
||||
let non_existing_file =
|
||||
"/ashjudlkahasdasdsikdhajik/asdasdasdasdasdasd/fjuiklashdbflasas";
|
||||
let res = fs::File::open(non_existing_file).await;
|
||||
match res {
|
||||
Ok(_) => panic!("Found file with random name: We live in a simulation"),
|
||||
Err(e) => assert_eq!(
|
||||
"Could not open /ashjudlkahasdasdsikdhajik/asdasdasdasdasdasd/fjuiklashdbflasas",
|
||||
&format!("{}", e)
|
||||
),
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
Loading…
Reference in a new issue