use std::collections::HashMap;
use std::sync::OnceLock;
use crate::registry::manifest::{BundleDefinition, BundlesFile, ExtensionManifest};
const EMBEDDED_CATALOG: &str = include_str!(concat!(env!("OUT_DIR"), "/embedded_catalog.json"));
#[derive(serde::Deserialize)]
struct EmbeddedCatalogRaw {
#[serde(default)]
tools: Vec<ExtensionManifest>,
#[serde(default)]
channels: Vec<ExtensionManifest>,
#[serde(default)]
mcp_servers: Vec<ExtensionManifest>,
#[serde(default)]
bundles: BundlesFile,
}
struct ParsedCatalog {
manifests: HashMap<String, ExtensionManifest>,
bundles: HashMap<String, BundleDefinition>,
}
fn parsed_catalog() -> &'static ParsedCatalog {
static CACHE: OnceLock<ParsedCatalog> = OnceLock::new();
CACHE.get_or_init(|| {
let raw: EmbeddedCatalogRaw = match serde_json::from_str(EMBEDDED_CATALOG) {
Ok(v) => v,
Err(e) => {
tracing::warn!("Failed to parse embedded catalog: {}", e);
return ParsedCatalog {
manifests: HashMap::new(),
bundles: HashMap::new(),
};
}
};
let mut manifests = HashMap::new();
for m in raw.tools {
let key = format!("tools/{}", m.name);
manifests.insert(key, m);
}
for m in raw.channels {
let key = format!("channels/{}", m.name);
manifests.insert(key, m);
}
for m in raw.mcp_servers {
let key = format!("mcp-servers/{}", m.name);
manifests.insert(key, m);
}
ParsedCatalog {
manifests,
bundles: raw.bundles.bundles,
}
})
}
pub fn load_embedded() -> HashMap<String, ExtensionManifest> {
parsed_catalog().manifests.clone()
}
pub fn load_embedded_bundles() -> HashMap<String, BundleDefinition> {
parsed_catalog().bundles.clone()
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_load_embedded_parses() {
let manifests = load_embedded();
assert!(
manifests.is_empty() || manifests.contains_key("tools/github"),
"Expected either empty catalog or github tool, got {} entries",
manifests.len()
);
}
#[test]
fn test_load_embedded_bundles_parses() {
let bundles = load_embedded_bundles();
assert!(
bundles.is_empty() || bundles.contains_key("default"),
"Expected either empty bundles or 'default' bundle"
);
}
}