use std::{env, format, path::PathBuf, sync::LazyLock};
use proc_macro::TokenStream;
use toml_edit::{DocumentMut, Item};
pub struct ObelManifest {
manifest: DocumentMut,
}
const OBEL: &str = "obel";
const OBEL_API: &str = "obel_api";
impl ObelManifest {
pub fn shared() -> &'static LazyLock<Self> {
static LAZY_SELF: LazyLock<ObelManifest> = LazyLock::new(|| ObelManifest {
manifest: env::var_os("CARGO_MANIFEST_DIR")
.map(PathBuf::from)
.map(|mut path| {
path.push("Cargo.toml");
if !path.exists() {
panic!("No Cargo manifest found for crate. Expected: {}", path.display());
}
let manifest = std::fs::read_to_string(path.clone()).unwrap_or_else(|_| {
panic!("Unable to read cargo manifest: {}", path.display())
});
manifest.parse::<DocumentMut>().unwrap_or_else(|_| {
panic!("Failed to parse cargo manifest: {}", path.display())
})
})
.expect("CARGO_MANIFEST_DIR is not defined."),
});
&LAZY_SELF
}
pub fn maybe_get_path(&self, name: &str) -> Option<syn::Path> {
fn dep_package(dep: &Item) -> Option<&str> {
if dep.as_str().is_some() {
None
} else {
dep.get("package").map(|name| name.as_str().unwrap())
}
}
let find_in_deps = |deps: &Item| -> Option<syn::Path> {
let package = if let Some(dep) = deps.get(name) {
return Some(Self::parse_str(dep_package(dep).unwrap_or(name)));
} else if let Some(dep) = deps.get(OBEL) {
dep_package(dep).unwrap_or(OBEL)
} else if let Some(dep) = deps.get(OBEL_API) {
dep_package(dep).unwrap_or(OBEL_API)
} else {
return None;
};
let mut path = Self::parse_str::<syn::Path>(package);
if let Some(module) = name.strip_prefix("bevy_") {
path.segments.push(Self::parse_str(module));
}
Some(path)
};
let deps = self.manifest.get("dependencies");
let deps_dev = self.manifest.get("dev-dependencies");
deps.and_then(find_in_deps).or_else(|| deps_dev.and_then(find_in_deps))
}
pub fn get_path(&self, name: &str) -> syn::Path {
self.maybe_get_path(name).unwrap_or_else(|| Self::parse_str(name))
}
pub fn try_parse_str<T: syn::parse::Parse>(path: &str) -> Option<T> {
syn::parse(path.parse::<TokenStream>().ok()?).ok()
}
pub fn parse_str<T: syn::parse::Parse>(path: &str) -> T {
Self::try_parse_str(path).unwrap()
}
pub fn get_subcrate(&self, subcrate: &str) -> Option<syn::Path> {
self.maybe_get_path(OBEL)
.map(|bevy_path| {
let mut segments = bevy_path.segments;
segments.push(ObelManifest::parse_str(subcrate));
syn::Path {
leading_colon: None,
segments,
}
})
.or_else(|| self.maybe_get_path(&format!("bevy_{subcrate}")))
}
}