use serde::Deserialize;
use std::collections::HashMap;
use std::path::{Path, PathBuf};
#[derive(Debug, Deserialize)]
pub struct Manifest {
pub package: Option<PackageMeta>,
#[serde(default)]
pub dependencies: HashMap<String, Dependency>,
}
#[derive(Debug, Deserialize)]
pub struct PackageMeta {
pub name: String,
#[serde(default)]
pub version: String,
#[serde(default)]
pub description: Option<String>,
#[serde(default)]
pub registry: Option<String>,
}
#[derive(Debug, Deserialize)]
#[serde(untagged)]
pub enum Dependency {
Path { path: String },
Git {
git: String,
#[serde(default)]
branch: Option<String>,
#[serde(default)]
tag: Option<String>,
#[serde(default)]
rev: Option<String>,
},
Registry { registry: String, version: String },
}
impl Dependency {
pub fn validate(&self) -> Result<(), String> {
if let Dependency::Git { branch, tag, rev, .. } = self {
let count = [branch, tag, rev].iter().filter(|o| o.is_some()).count();
if count > 1 {
return Err("at most one of `branch`, `tag`, `rev` may be set on a git dependency".into());
}
}
Ok(())
}
}
impl Manifest {
pub fn load(toml_path: &Path) -> Result<Self, String> {
let src = std::fs::read_to_string(toml_path)
.map_err(|e| format!("reading {}: {e}", toml_path.display()))?;
toml::from_str(&src)
.map_err(|e| format!("parsing {}: {e}", toml_path.display()))
}
}
pub fn find_manifest(start: &Path) -> Option<(PathBuf, PathBuf)> {
let mut dir = if start.is_dir() {
start.to_path_buf()
} else {
start.parent()?.to_path_buf()
};
loop {
let candidate = dir.join("lex.toml");
if candidate.exists() {
return Some((candidate, dir));
}
match dir.parent() {
Some(p) if p != dir => dir = p.to_path_buf(),
_ => return None,
}
}
}
pub fn resolve_package_import(
importer: &Path,
pkg_name: &str,
module_path: &str,
) -> Result<PathBuf, PackageError> {
let (toml_path, toml_dir) = find_manifest(importer).ok_or_else(|| {
PackageError::NoManifest {
reference: format!("{pkg_name}/{module_path}"),
searched_from: importer.display().to_string(),
}
})?;
let manifest = Manifest::load(&toml_path)
.map_err(|e| PackageError::ManifestParse { path: toml_path.display().to_string(), detail: e })?;
let dep = manifest.dependencies.get(pkg_name).ok_or_else(|| {
PackageError::UnknownPackage {
name: pkg_name.to_string(),
manifest: toml_path.display().to_string(),
}
})?;
let pkg_root = match dep {
Dependency::Path { path } => {
let raw = toml_dir.join(path);
raw.canonicalize().map_err(|e| PackageError::Io {
path: raw.display().to_string(),
detail: e.to_string(),
})?
}
Dependency::Git { git, branch, tag, rev } => {
dep.validate().map_err(|e| PackageError::ManifestParse {
path: toml_path.display().to_string(),
detail: e,
})?;
let git_ref = GitRef::from(branch.as_deref(), tag.as_deref(), rev.as_deref());
git_ensure_cached(pkg_name, git, &git_ref)?
}
Dependency::Registry { registry, version } => {
registry_ensure_cached(pkg_name, registry, version)?
}
};
find_module_file(&pkg_root, module_path).ok_or_else(|| PackageError::ModuleNotFound {
pkg: pkg_name.to_string(),
module: module_path.to_string(),
pkg_root: pkg_root.display().to_string(),
})
}
fn find_module_file(pkg_root: &Path, module_path: &str) -> Option<PathBuf> {
let rel = PathBuf::from(module_path).with_extension("lex");
let in_src = pkg_root.join("src").join(&rel);
if in_src.exists() {
return Some(in_src);
}
let at_root = pkg_root.join(&rel);
if at_root.exists() {
return Some(at_root);
}
None
}
#[derive(Debug)]
enum GitRef<'a> {
Branch(&'a str),
Tag(&'a str),
Rev(&'a str),
DefaultBranch,
}
impl<'a> GitRef<'a> {
fn from(branch: Option<&'a str>, tag: Option<&'a str>, rev: Option<&'a str>) -> Self {
if let Some(b) = branch { return GitRef::Branch(b); }
if let Some(t) = tag { return GitRef::Tag(t); }
if let Some(r) = rev { return GitRef::Rev(r); }
GitRef::DefaultBranch
}
fn cache_slug(&self) -> String {
match self {
GitRef::Branch(b) => format!("@branch-{}", sanitize_ref(b)),
GitRef::Tag(t) => format!("@tag-{}", sanitize_ref(t)),
GitRef::Rev(r) => format!("@rev-{}", &r[..r.len().min(12)]),
GitRef::DefaultBranch => String::new(),
}
}
}
fn sanitize_ref(r: &str) -> String {
r.chars().map(|c| if c.is_alphanumeric() || c == '-' || c == '.' { c } else { '_' }).collect()
}
fn git_ensure_cached(pkg_name: &str, url: &str, git_ref: &GitRef<'_>) -> Result<PathBuf, PackageError> {
let cache_root = packages_cache_dir()?;
let dir_name = format!("{}{}", pkg_name, git_ref.cache_slug());
let pkg_dir = cache_root.join(&dir_name);
if pkg_dir.exists() {
return Ok(pkg_dir);
}
std::fs::create_dir_all(&cache_root).map_err(|e| PackageError::Io {
path: cache_root.display().to_string(),
detail: e.to_string(),
})?;
let dest = pkg_dir.to_str().unwrap_or(&dir_name);
let status = match git_ref {
GitRef::Rev(rev) => {
let s = run_git(&["clone", "--quiet", url, dest], url)?;
if s {
run_git(&["-C", dest, "checkout", "--quiet", rev], url)?;
true
} else {
false
}
}
GitRef::Tag(tag) => run_git(&["clone", "--quiet", "--depth=1", "--branch", tag, url, dest], url)?,
GitRef::Branch(branch) => run_git(&["clone", "--quiet", "--depth=1", "--branch", branch, url, dest], url)?,
GitRef::DefaultBranch => run_git(&["clone", "--quiet", "--depth=1", url, dest], url)?,
};
if !status {
let _ = std::fs::remove_dir_all(&pkg_dir);
return Err(PackageError::GitFailed {
url: url.to_string(),
detail: "`git` exited with non-zero status".into(),
});
}
pkg_dir.canonicalize().map_err(|e| PackageError::Io {
path: pkg_dir.display().to_string(),
detail: e.to_string(),
})
}
fn run_git(args: &[&str], url: &str) -> Result<bool, PackageError> {
let status = std::process::Command::new("git")
.args(args)
.status()
.map_err(|e| PackageError::GitFailed {
url: url.to_string(),
detail: format!("could not run `git`: {e}"),
})?;
Ok(status.success())
}
fn registry_ensure_cached(
pkg_name: &str,
registry: &str,
version: &str,
) -> Result<PathBuf, PackageError> {
let cache_root = packages_cache_dir()?;
let pkg_dir = cache_root.join(format!("{pkg_name}-{version}"));
if pkg_dir.exists() {
return Ok(pkg_dir);
}
std::fs::create_dir_all(&cache_root).map_err(|e| PackageError::Io {
path: cache_root.display().to_string(),
detail: e.to_string(),
})?;
let url = format!(
"{}/v1/pkg/{}/{}/archive",
registry.trim_end_matches('/'),
pkg_name,
version,
);
let response = ureq::get(&url).call().map_err(|e| PackageError::RegistryFailed {
name: pkg_name.to_string(),
registry: registry.to_string(),
version: version.to_string(),
detail: format!("GET {url}: {e}"),
})?;
if response.status() != 200 {
return Err(PackageError::RegistryFailed {
name: pkg_name.to_string(),
registry: registry.to_string(),
version: version.to_string(),
detail: format!("GET {url} returned HTTP {}", response.status()),
});
}
let archive_bytes = response
.into_body()
.read_to_vec()
.map_err(|e| PackageError::RegistryFailed {
name: pkg_name.to_string(),
registry: registry.to_string(),
version: version.to_string(),
detail: format!("reading response body: {e}"),
})?;
let gz = flate2::read::GzDecoder::new(std::io::Cursor::new(&archive_bytes));
let mut ar = tar::Archive::new(gz);
ar.unpack(&pkg_dir).map_err(|e| PackageError::RegistryFailed {
name: pkg_name.to_string(),
registry: registry.to_string(),
version: version.to_string(),
detail: format!("extracting archive: {e}"),
})?;
pkg_dir.canonicalize().map_err(|e| PackageError::Io {
path: pkg_dir.display().to_string(),
detail: e.to_string(),
})
}
fn packages_cache_dir() -> Result<PathBuf, PackageError> {
if let Ok(dir) = std::env::var("LEX_PACKAGES_DIR") {
return Ok(PathBuf::from(dir));
}
let home = std::env::var("HOME")
.or_else(|_| std::env::var("USERPROFILE"))
.map_err(|_| PackageError::Io {
path: "~/.lex/packages".into(),
detail: "could not determine home directory (set LEX_PACKAGES_DIR)".into(),
})?;
Ok(PathBuf::from(home).join(".lex").join("packages"))
}
#[derive(Debug, thiserror::Error)]
pub enum PackageError {
#[error("no lex.toml found searching up from {searched_from} (needed to resolve \"{reference}\")")]
NoManifest { reference: String, searched_from: String },
#[error("failed to parse {path}: {detail}")]
ManifestParse { path: String, detail: String },
#[error("package \"{name}\" not found in {manifest}")]
UnknownPackage { name: String, manifest: String },
#[error("module \"{module}\" not found in package \"{pkg}\" (looked in {pkg_root}/src/ and {pkg_root}/)")]
ModuleNotFound { pkg: String, module: String, pkg_root: String },
#[error("git clone of {url} failed: {detail}")]
GitFailed { url: String, detail: String },
#[error("registry fetch of {name}@{version} from {registry} failed: {detail}")]
RegistryFailed { name: String, registry: String, version: String, detail: String },
#[error("I/O error at {path}: {detail}")]
Io { path: String, detail: String },
}