use std::sync::Arc;
use console::style;
use eyre::{Result, bail, eyre};
use itertools::Itertools;
use crate::backend::Backend;
use crate::cli::args::ToolArg;
use crate::config::Config;
use crate::toolset::{ToolRequest, ToolSource, ToolVersion, ToolsetBuilder};
use crate::ui::multi_progress_report::MultiProgressReport;
use crate::{config, dirs, exit, file};
#[derive(Debug, clap::Args)]
#[clap(verbatim_doc_comment, after_long_help = AFTER_LONG_HELP)]
pub struct Uninstall {
#[clap(value_name = "INSTALLED_TOOL@VERSION", required_unless_present = "all")]
installed_tool: Vec<ToolArg>,
#[clap(long, short)]
all: bool,
#[clap(long, short = 'n')]
dry_run: bool,
#[clap(long, verbatim_doc_comment)]
dry_run_code: bool,
}
impl Uninstall {
fn is_dry_run(&self) -> bool {
self.dry_run || self.dry_run_code
}
pub async fn run(self) -> Result<()> {
let config = Config::get().await?;
let tool_versions = if self.installed_tool.is_empty() && self.all {
self.get_all_tool_versions(&config).await?
} else {
self.get_requested_tool_versions(&config).await?
};
let tool_versions = tool_versions
.into_iter()
.unique_by(|(_, tv)| (tv.request.ba().short.clone(), tv.version.clone()))
.collect::<Vec<_>>();
if !self.all && tool_versions.len() > self.installed_tool.len() {
bail!("multiple tools specified, use --all to uninstall all versions");
}
let mpr = MultiProgressReport::get();
let mut has_work = false;
for (plugin, tv) in tool_versions {
if !plugin.is_version_installed(&config, &tv, true) {
warn!("{} is not installed", tv.style());
continue;
}
has_work = true;
let pr = mpr.add(&tv.style());
if let Err(err) = plugin
.uninstall_version(&config, &tv, pr.as_ref(), self.is_dry_run())
.await
{
error!("{err}");
return Err(eyre!(err).wrap_err(format!("failed to uninstall {tv}")));
}
if self.is_dry_run() {
pr.finish_with_message("uninstalled (dry-run)".into());
} else {
pr.finish_with_message("uninstalled".into());
}
}
if self.is_dry_run() {
if self.dry_run_code && has_work {
exit::exit(1);
}
return Ok(());
}
file::touch_dir(&dirs::DATA)?;
let config = Config::reset().await?;
let ts = config.get_toolset().await?;
config::rebuild_shims_and_runtime_symlinks(&config, ts, &[]).await?;
Ok(())
}
async fn get_all_tool_versions(
&self,
config: &Arc<Config>,
) -> Result<Vec<(Arc<dyn Backend>, ToolVersion)>> {
let ts = ToolsetBuilder::new().build(config).await?;
let tool_versions = ts
.list_installed_versions(config)
.await?
.into_iter()
.collect::<Vec<_>>();
Ok(tool_versions)
}
async fn get_requested_tool_versions(
&self,
config: &Arc<Config>,
) -> Result<Vec<(Arc<dyn Backend>, ToolVersion)>> {
let runtimes = ToolArg::double_tool_condition(&self.installed_tool)?;
let mut tool_versions = Vec::new();
for ta in runtimes {
let backend = ta.ba.backend()?;
let query = ta.tvr.as_ref().map(|tvr| tvr.version()).unwrap_or_default();
let installed_versions = backend.list_installed_versions();
let exact_match = installed_versions.iter().find(|v| v == &&query);
let matches = match exact_match {
Some(m) => vec![m],
None => installed_versions
.iter()
.filter(|v| v.starts_with(&query))
.collect_vec(),
};
let mut tvs = matches
.into_iter()
.map(|v| {
let tvr = ToolRequest::new(backend.ba().clone(), v, ToolSource::Unknown)?;
let tv = ToolVersion::new(tvr, v.into());
Ok((backend.clone(), tv))
})
.collect::<Result<Vec<_>>>()?;
if let Some(tvr) = &ta.tvr {
tvs.push((
backend.clone(),
tvr.resolve(config, &Default::default()).await?,
));
}
if tvs.is_empty() {
warn!(
"no versions found for {}",
style(&backend).blue().for_stderr()
);
}
tool_versions.extend(tvs);
}
Ok(tool_versions)
}
}
static AFTER_LONG_HELP: &str = color_print::cstr!(
r#"<bold><underline>Examples:</underline></bold>
# will uninstall specific version
$ <bold>mise uninstall node@18.0.0</bold>
# will uninstall the current node version (if only one version is installed)
$ <bold>mise uninstall node</bold>
# will uninstall all installed versions of node
$ <bold>mise uninstall --all node@18.0.0</bold> # will uninstall all node versions
"#
);