use crate::backend::backend_type::BackendType;
use crate::cli::args::BackendArg;
use crate::config::SETTINGS;
use std::collections::{BTreeMap, HashMap, HashSet};
use std::env::consts::{ARCH, OS};
use std::fmt::Display;
use std::iter::Iterator;
use std::sync::LazyLock as Lazy;
use strum::IntoEnumIterator;
use url::Url;
pub static REGISTRY: Lazy<BTreeMap<&'static str, RegistryTool>> =
Lazy::new(|| include!(concat!(env!("OUT_DIR"), "/registry.rs")));
#[derive(Debug, Clone)]
pub struct RegistryTool {
pub short: &'static str,
pub description: Option<&'static str>,
pub backends: &'static [RegistryBackend],
#[allow(unused)]
pub aliases: &'static [&'static str],
pub test: &'static Option<(&'static str, &'static str)>,
pub os: &'static [&'static str],
pub depends: &'static [&'static str],
pub idiomatic_files: &'static [&'static str],
}
#[derive(Debug, Clone)]
pub struct RegistryBackend {
pub full: &'static str,
pub platforms: &'static [&'static str],
}
impl RegistryTool {
pub fn backends(&self) -> Vec<&'static str> {
static BACKEND_TYPES: Lazy<HashSet<String>> = Lazy::new(|| {
let mut backend_types = BackendType::iter()
.map(|b| b.to_string())
.collect::<HashSet<_>>();
time!("disable_backends");
for backend in &SETTINGS.disable_backends {
backend_types.remove(backend);
}
time!("disable_backends");
if cfg!(windows) {
backend_types.remove("asdf");
}
backend_types
});
let platform = format!("{OS}-{ARCH}");
self.backends
.iter()
.filter(|rb| {
rb.platforms.is_empty()
|| rb.platforms.contains(&OS)
|| rb.platforms.contains(&ARCH)
|| rb.platforms.contains(&&*platform)
})
.map(|rb| rb.full)
.filter(|full| {
full.split(':')
.next()
.is_some_and(|b| BACKEND_TYPES.contains(b))
})
.collect()
}
pub fn is_supported_os(&self) -> bool {
self.os.is_empty() || self.os.contains(&OS)
}
pub fn ba(&self) -> Option<BackendArg> {
self.backends()
.first()
.map(|f| BackendArg::new(self.short.to_string(), Some(f.to_string())))
}
}
pub fn shorts_for_full(full: &str) -> &'static Vec<&'static str> {
static EMPTY: Vec<&'static str> = vec![];
static FULL_TO_SHORT: Lazy<HashMap<&'static str, Vec<&'static str>>> = Lazy::new(|| {
let mut map: HashMap<&'static str, Vec<&'static str>> = HashMap::new();
for (short, rt) in REGISTRY.iter() {
for full in rt.backends() {
map.entry(full).or_default().push(short);
}
}
map
});
FULL_TO_SHORT.get(full).unwrap_or(&EMPTY)
}
pub fn is_trusted_plugin(name: &str, remote: &str) -> bool {
let normalized_url = normalize_remote(remote).unwrap_or("INVALID_URL".into());
let is_shorthand = REGISTRY
.get(name)
.and_then(|tool| tool.backends().first().copied())
.map(full_to_url)
.is_some_and(|s| normalize_remote(&s).unwrap_or_default() == normalized_url);
let is_mise_url = normalized_url.starts_with("github.com/mise-plugins/");
!is_shorthand || is_mise_url
}
fn normalize_remote(remote: &str) -> eyre::Result<String> {
let url = Url::parse(remote)?;
let host = url.host_str().unwrap();
let path = url.path().trim_end_matches(".git");
Ok(format!("{host}{path}"))
}
pub fn full_to_url(full: &str) -> String {
if url_like(full) {
return full.to_string();
}
let (_backend, url) = full.split_once(':').unwrap_or(("", full));
if url_like(url) {
url.to_string()
} else {
format!("https://github.com/{url}.git")
}
}
fn url_like(s: &str) -> bool {
s.starts_with("https://")
|| s.starts_with("http://")
|| s.starts_with("git@")
|| s.starts_with("ssh://")
|| s.starts_with("git://")
}
impl Display for RegistryTool {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(f, "{}", self.short)
}
}