use crate::backend::backend_type::BackendType;
use crate::config::Settings;
use crate::registry::{REGISTRY, RegistryTool, tool_enabled};
use crate::ui::table::MiseTable;
use eyre::{Result, bail};
use itertools::Itertools;
use serde_derive::Serialize;
#[derive(Debug, clap::Args)]
#[clap(after_long_help = AFTER_LONG_HELP, verbatim_doc_comment)]
pub struct Registry {
name: Option<String>,
#[clap(short, long)]
backend: Option<BackendType>,
#[clap(long, hide = true)]
complete: bool,
#[clap(long)]
hide_aliased: bool,
#[clap(long, short = 'J')]
json: bool,
}
#[derive(Serialize)]
struct RegistryToolOutput {
short: String,
backends: Vec<String>,
#[serde(skip_serializing_if = "Option::is_none")]
description: Option<String>,
#[serde(skip_serializing_if = "Vec::is_empty")]
aliases: Vec<String>,
}
impl Registry {
pub async fn run(self) -> Result<()> {
if let Some(name) = &self.name {
if let Some(rt) = REGISTRY.get(name.as_str()) {
if self.json {
let tool = self.to_output(name, rt);
miseprintln!("{}", serde_json::to_string_pretty(&tool)?);
} else {
miseprintln!("{}", self.filter_backends(rt).join(" "));
}
} else {
bail!("tool not found in registry: {name}");
}
} else if self.complete {
self.complete()?;
} else if self.json {
self.display_json()?;
} else {
self.display_table()?;
}
Ok(())
}
fn filter_backends(&self, rt: &RegistryTool) -> Vec<&'static str> {
if let Some(backend) = &self.backend {
rt.backends()
.into_iter()
.filter(|full| full.starts_with(&format!("{backend}:")))
.collect()
} else {
rt.backends()
}
}
fn to_output(&self, short: &str, rt: &RegistryTool) -> RegistryToolOutput {
RegistryToolOutput {
short: short.to_string(),
backends: self
.filter_backends(rt)
.iter()
.map(|s| s.to_string())
.collect(),
description: rt.description.map(|s| s.to_string()),
aliases: rt.aliases.iter().map(|s| s.to_string()).collect(),
}
}
fn filtered_tools(&self) -> impl Iterator<Item = (&&'static str, &RegistryTool)> {
REGISTRY
.iter()
.filter(|(short, _)| filter_enabled(short))
.filter(|(short, rt)| !self.hide_aliased || **short == rt.short)
}
fn display_table(&self) -> Result<()> {
let mut table = MiseTable::new(false, &["Tool", "Backends"]);
let data = self
.filtered_tools()
.map(|(short, rt)| (short.to_string(), self.filter_backends(rt).join(" ")))
.filter(|(_, backends)| !backends.is_empty())
.sorted_by(|(a, _), (b, _)| a.cmp(b))
.map(|(short, backends)| vec![short, backends])
.collect_vec();
for row in data {
table.add_row(row);
}
table.print()
}
fn complete(&self) -> Result<()> {
self.filtered_tools()
.map(|(short, rt)| {
(
short.to_string(),
rt.description
.or(rt.backends().first().cloned())
.unwrap_or_default(),
)
})
.sorted_by(|(a, _), (b, _)| a.cmp(b))
.for_each(|(short, description)| {
println!(
"{}:{}",
short.replace(":", "\\:"),
description.replace(":", "\\:")
);
});
Ok(())
}
fn display_json(&self) -> Result<()> {
let tools: Vec<RegistryToolOutput> = self
.filtered_tools()
.map(|(short, rt)| self.to_output(short, rt))
.filter(|tool| !tool.backends.is_empty())
.sorted_by(|a, b| a.short.cmp(&b.short))
.collect();
miseprintln!("{}", serde_json::to_string_pretty(&tools)?);
Ok(())
}
}
static AFTER_LONG_HELP: &str = color_print::cstr!(
r#"<bold><underline>Examples:</underline></bold>
$ <bold>mise registry</bold>
node core:node
poetry asdf:mise-plugins/mise-poetry
ubi cargo:ubi-cli
$ <bold>mise registry poetry</bold>
asdf:mise-plugins/mise-poetry
"#
);
fn filter_enabled(short: &str) -> bool {
tool_enabled(
&Settings::get().enable_tools,
&Settings::get().disable_tools,
&short.to_string(),
)
}