2
0
Fork 1
mirror of https://github.com/async-rs/async-std.git synced 2025-10-20 17:16:35 +00:00

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:
Pascal Hertleif 2019-11-17 21:54:44 +01:00
parent 355e2eded8
commit 8ce3e78952
7 changed files with 98 additions and 2 deletions

View file

@ -64,6 +64,12 @@ jobs:
command: test command: test
args: --all --features unstable 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: check_fmt_and_docs:
name: Checking fmt and docs name: Checking fmt and docs
runs-on: ubuntu-latest runs-on: ubuntu-latest

View file

@ -48,6 +48,7 @@ std = [
"pin-utils", "pin-utils",
"slab", "slab",
] ]
verbose-errors = []
[dependencies] [dependencies]
async-attributes = { version = "1.1.1", optional = true } async-attributes = { version = "1.1.1", optional = true }

View file

@ -9,6 +9,7 @@ use std::sync::{Arc, Mutex};
use crate::fs::{Metadata, Permissions}; use crate::fs::{Metadata, Permissions};
use crate::future; use crate::future;
use crate::utils::VerboseErrorExt;
use crate::io::{self, Read, Seek, SeekFrom, Write}; use crate::io::{self, Read, Seek, SeekFrom, Write};
use crate::path::Path; use crate::path::Path;
use crate::prelude::*; use crate::prelude::*;
@ -112,7 +113,11 @@ impl File {
/// ``` /// ```
pub async fn open<P: AsRef<Path>>(path: P) -> io::Result<File> { pub async fn open<P: AsRef<Path>>(path: P) -> io::Result<File> {
let path = path.as_ref().to_owned(); 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)) Ok(File::new(file, true))
} }
@ -147,7 +152,11 @@ impl File {
/// ``` /// ```
pub async fn create<P: AsRef<Path>>(path: P) -> io::Result<File> { pub async fn create<P: AsRef<Path>>(path: P) -> io::Result<File> {
let path = path.as_ref().to_owned(); 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)) Ok(File::new(file, true))
} }

View file

@ -291,6 +291,7 @@ cfg_std! {
pub(crate) mod read; pub(crate) mod read;
pub(crate) mod seek; pub(crate) mod seek;
pub(crate) mod write; pub(crate) mod write;
pub(crate) mod utils;
mod buf_reader; mod buf_reader;
mod buf_writer; mod buf_writer;

51
src/io/utils.rs Normal file
View 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)
}
}

View file

@ -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. /// Defers evaluation of a block of code until the end of the scope.
#[cfg(feature = "default")] #[cfg(feature = "default")]
#[doc(hidden)] #[doc(hidden)]

20
tests/verbose_errors.rs Normal file
View 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)
),
}
})
}
}