use std::path::PathBuf;
#[derive(Debug, thiserror::Error)]
pub enum Error {
#[error("no functions matched {specs}{hint}")]
NoTargetsFound { specs: String, hint: String },
#[error("failed to parse {}: {source}", path.display())]
ParseError {
path: PathBuf,
#[source]
source: syn::Error,
},
#[error("build failed: {0}")]
BuildFailed(String),
#[error("run failed: {0}")]
RunFailed(String),
#[error("no instrumented binary found -- run `piano build` first")]
NoBinary,
#[error("no piano runs found -- run `piano profile` to generate one")]
NoRuns,
#[error("need at least 2 runs to diff -- run `piano profile` to generate one")]
NotEnoughRuns,
#[error(
"piano diff takes 0 or 2 arguments\n\n piano diff compare the two most recent runs\n piano diff <A> <B> compare two specific runs (path or tag)"
)]
DiffArgCount,
#[error("no run found for tag '{tag}' -- run piano tag to see available tags")]
RunNotFound { tag: String },
#[error("no Cargo.toml found in {0} or any parent directory")]
NoProjectFound(PathBuf),
#[error("failed to read run file {}: {source}", path.display())]
RunReadError {
path: PathBuf,
#[source]
source: std::io::Error,
},
#[error("invalid run data in {}: {reason}", path.display())]
InvalidRunData { path: PathBuf, reason: String },
#[error("{0}")]
InvalidTagName(String),
#[error("profiling data was not written -- check disk space and permissions for {}", .0.display())]
NoDataWritten(PathBuf),
#[error("failed to {operation} {}: {source}", path.display())]
IoWithContext {
operation: &'static str,
path: PathBuf,
#[source]
source: std::io::Error,
},
#[error("{0}")]
Io(#[from] std::io::Error),
}
pub fn io_context(
operation: &'static str,
path: impl Into<PathBuf>,
) -> impl FnOnce(std::io::Error) -> Error {
let path = path.into();
move |source| Error::IoWithContext {
operation,
path,
source,
}
}