use std::path::PathBuf;
use std::process::ExitStatus;
use crate::cli::args::BackendArg;
use crate::file::display_path;
use crate::toolset::{ToolRequest, ToolSource, ToolVersion};
use eyre::Report;
use thiserror::Error;
#[derive(Debug, Error)]
pub enum Error {
#[error("[{ts}] {tr}: {source:#}")]
FailedToResolveVersion {
tr: Box<ToolRequest>,
ts: ToolSource,
source: Report,
},
#[error("[{0}] plugin not installed")]
PluginNotInstalled(String),
#[error("{0}@{1} not installed")]
VersionNotInstalled(Box<BackendArg>, String),
#[error("{} exited with non-zero status: {}", .0, render_exit_status(.1))]
ScriptFailed(String, Option<ExitStatus>),
#[error(
"Config files in {} are not trusted.\nTrust them with `mise trust`. See https://mise.jdx.dev/cli/trust.html for more information.",
display_path(.0)
)]
UntrustedConfig(PathBuf),
#[error("{}", format_install_failures(.failed_installations))]
InstallFailed {
successful_installations: Vec<ToolVersion>,
failed_installations: Vec<(ToolRequest, Report)>,
},
}
fn render_exit_status(exit_status: &Option<ExitStatus>) -> String {
match exit_status.and_then(|s| s.code()) {
Some(exit_status) => format!("exit code {exit_status}"),
None => "no exit status".into(),
}
}
fn format_install_failures(failed_installations: &[(ToolRequest, Report)]) -> String {
if failed_installations.is_empty() {
return "Installation failed".to_string();
}
if failed_installations.len() == 1 {
let (tr, error) = &failed_installations[0];
return format!(
"Failed to install {}@{}: {:#}",
tr.ba().full(),
tr.version(),
error
);
}
let mut sorted_failures: Vec<_> = failed_installations
.iter()
.map(|(tr, err)| (format!("{}@{}", tr.ba().full(), tr.version()), err))
.collect();
sorted_failures.sort_by(|a, b| a.0.cmp(&b.0));
let mut output = vec![];
let failed_tools: Vec<&str> = sorted_failures
.iter()
.map(|(name, _)| name.as_str())
.collect();
output.push(format!(
"Failed to install tools: {}",
failed_tools.join(", ")
));
for (name, error) in sorted_failures.iter() {
output.push(format!("\n{}: {:#}", name, error));
}
output.join("\n")
}
impl Error {
pub fn get_exit_status(err: &Report) -> Option<i32> {
if let Some(Error::ScriptFailed(_, Some(status))) = err.downcast_ref::<Error>() {
status.code()
} else {
None
}
}
pub fn is_argument_err(err: &Report) -> bool {
err.downcast_ref::<Error>()
.map(|e| {
matches!(
e,
Error::FailedToResolveVersion {
ts: ToolSource::Argument,
..
}
)
})
.unwrap_or(false)
}
}