use alef_core::config::{AlefConfig, Language};
use anyhow::Context as _;
use rayon::prelude::*;
use std::path::Path;
use tracing::{debug, info};
use crate::registry;
use super::helpers::{check_precondition, run_before, run_command, run_command_captured};
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
enum LintPhase {
Format,
Check,
Typecheck,
}
fn prepare_lint_languages(config: &AlefConfig, languages: &[Language]) -> anyhow::Result<Vec<Language>> {
let mut ready = Vec::with_capacity(languages.len());
for &lang in languages {
let lang_lint = config.lint_config_for_language(lang);
if !check_precondition(lang, lang_lint.precondition.as_deref()) {
continue;
}
run_before(lang, lang_lint.before.as_ref())?;
ready.push(lang);
}
Ok(ready)
}
fn run_phase(config: &AlefConfig, languages: &[Language], phase: LintPhase) -> anyhow::Result<()> {
let tasks: Vec<(&Language, String)> = languages
.iter()
.filter_map(|lang| {
let lang_lint = config.lint_config_for_language(*lang);
let cmds = match phase {
LintPhase::Format => lang_lint.format,
LintPhase::Check => lang_lint.check,
LintPhase::Typecheck => lang_lint.typecheck,
};
cmds.map(|cmd_list| {
cmd_list
.commands()
.into_iter()
.map(|c| (lang, c.to_string()))
.collect::<Vec<_>>()
})
})
.flatten()
.collect();
let results: Vec<anyhow::Result<(String, String, String)>> = tasks
.par_iter()
.map(|(_, cmd)| {
let (stdout, stderr) = run_command_captured(cmd)?;
Ok((cmd.clone(), stdout, stderr))
})
.collect();
let mut first_error: Option<anyhow::Error> = None;
for result in results {
match result {
Ok((cmd, stdout, stderr)) => {
if !stdout.is_empty() {
info!("[{cmd}] stdout:\n{stdout}");
}
if !stderr.is_empty() {
info!("[{cmd}] stderr:\n{stderr}");
}
}
Err(e) => {
if first_error.is_none() {
first_error = Some(e);
}
}
}
}
if let Some(e) = first_error {
return Err(e);
}
Ok(())
}
pub fn lint(config: &AlefConfig, languages: &[Language]) -> anyhow::Result<()> {
let ready = prepare_lint_languages(config, languages)?;
run_phase(config, &ready, LintPhase::Format)?;
run_phase(config, &ready, LintPhase::Check)?;
run_phase(config, &ready, LintPhase::Typecheck)?;
Ok(())
}
pub fn fmt(config: &AlefConfig, languages: &[Language]) -> anyhow::Result<()> {
let ready = prepare_lint_languages(config, languages)?;
run_phase(config, &ready, LintPhase::Format)
}
pub fn update(config: &AlefConfig, languages: &[Language], latest: bool) -> anyhow::Result<()> {
let results: Vec<anyhow::Result<Vec<(String, String, String)>>> = languages
.par_iter()
.map(|lang| {
let update_cfg = config.update_config_for_language(*lang);
if !check_precondition(*lang, update_cfg.precondition.as_deref()) {
return Ok(Vec::new());
}
run_before(*lang, update_cfg.before.as_ref())?;
let cmds = if latest {
update_cfg.upgrade.as_ref()
} else {
update_cfg.update.as_ref()
};
let mut outputs = Vec::new();
if let Some(cmd_list) = cmds {
for cmd in cmd_list.commands() {
let (stdout, stderr) = run_command_captured(cmd)?;
outputs.push((cmd.to_string(), stdout, stderr));
}
}
Ok(outputs)
})
.collect();
let mut first_error: Option<anyhow::Error> = None;
for result in results {
match result {
Ok(outputs) => {
for (cmd, stdout, stderr) in outputs {
if !stdout.is_empty() {
info!("[{cmd}] stdout:\n{stdout}");
}
if !stderr.is_empty() {
info!("[{cmd}] stderr:\n{stderr}");
}
}
}
Err(e) => {
if first_error.is_none() {
first_error = Some(e);
}
}
}
}
if let Some(e) = first_error {
return Err(e);
}
Ok(())
}
pub fn test(config: &AlefConfig, languages: &[Language], e2e: bool, coverage: bool) -> anyhow::Result<()> {
let results: Vec<anyhow::Result<Vec<(String, String, String)>>> = languages
.par_iter()
.map(|lang| {
let lang_test = config.test_config_for_language(*lang);
if !check_precondition(*lang, lang_test.precondition.as_deref()) {
return Ok(Vec::new());
}
run_before(*lang, lang_test.before.as_ref())?;
let mut outputs = Vec::new();
let test_cmds = if coverage {
lang_test.coverage.as_ref().or(lang_test.command.as_ref())
} else {
lang_test.command.as_ref()
};
if let Some(cmd_list) = test_cmds {
for cmd in cmd_list.commands() {
let (stdout, stderr) = run_command_captured(cmd)?;
outputs.push((cmd.to_string(), stdout, stderr));
}
}
if e2e {
if let Some(e2e_cmd_list) = &lang_test.e2e {
for cmd in e2e_cmd_list.commands() {
let (stdout, stderr) = run_command_captured(cmd)?;
outputs.push((cmd.to_string(), stdout, stderr));
}
}
}
Ok(outputs)
})
.collect();
let mut first_error: Option<anyhow::Error> = None;
for result in results {
match result {
Ok(outputs) => {
for (cmd, stdout, stderr) in outputs {
if !stdout.is_empty() {
info!("[{cmd}] stdout:\n{stdout}");
}
if !stderr.is_empty() {
info!("[{cmd}] stderr:\n{stderr}");
}
}
}
Err(e) => {
if first_error.is_none() {
first_error = Some(e);
}
}
}
}
if let Some(e) = first_error {
return Err(e);
}
Ok(())
}
pub fn setup(config: &AlefConfig, languages: &[Language]) -> anyhow::Result<()> {
let results: Vec<anyhow::Result<Vec<(String, String, String)>>> = languages
.par_iter()
.map(|lang| {
let setup_cfg = config.setup_config_for_language(*lang);
if !check_precondition(*lang, setup_cfg.precondition.as_deref()) {
return Ok(Vec::new());
}
run_before(*lang, setup_cfg.before.as_ref())?;
let mut outputs = Vec::new();
if let Some(cmd_list) = &setup_cfg.install {
for cmd in cmd_list.commands() {
let (stdout, stderr) = run_command_captured(cmd)?;
outputs.push((cmd.to_string(), stdout, stderr));
}
}
Ok(outputs)
})
.collect();
let mut first_error: Option<anyhow::Error> = None;
for result in results {
match result {
Ok(outputs) => {
for (cmd, stdout, stderr) in outputs {
if !stdout.is_empty() {
info!("[{cmd}] stdout:\n{stdout}");
}
if !stderr.is_empty() {
info!("[{cmd}] stderr:\n{stderr}");
}
}
}
Err(e) => {
if first_error.is_none() {
first_error = Some(e);
}
}
}
}
if let Some(e) = first_error {
return Err(e);
}
Ok(())
}
pub fn clean(config: &AlefConfig, languages: &[Language]) -> anyhow::Result<()> {
let results: Vec<anyhow::Result<Vec<(String, String, String)>>> = languages
.par_iter()
.map(|lang| {
let clean_cfg = config.clean_config_for_language(*lang);
if !check_precondition(*lang, clean_cfg.precondition.as_deref()) {
return Ok(Vec::new());
}
run_before(*lang, clean_cfg.before.as_ref())?;
let mut outputs = Vec::new();
if let Some(cmd_list) = &clean_cfg.clean {
for cmd in cmd_list.commands() {
let (stdout, stderr) = run_command_captured(cmd)?;
outputs.push((cmd.to_string(), stdout, stderr));
}
}
Ok(outputs)
})
.collect();
let mut first_error: Option<anyhow::Error> = None;
for result in results {
match result {
Ok(outputs) => {
for (cmd, stdout, stderr) in outputs {
if !stdout.is_empty() {
info!("[{cmd}] stdout:\n{stdout}");
}
if !stderr.is_empty() {
info!("[{cmd}] stderr:\n{stderr}");
}
}
}
Err(e) => {
if first_error.is_none() {
first_error = Some(e);
}
}
}
}
if let Some(e) = first_error {
return Err(e);
}
Ok(())
}
pub fn build(config: &AlefConfig, languages: &[Language], release: bool) -> anyhow::Result<()> {
let crate_name = &config.crate_config.name;
let base_dir = std::env::current_dir()?;
let mut independent = Vec::new();
let mut ffi_dependent = Vec::new();
let mut need_ffi = false;
let mut rust_langs: Vec<Language> = Vec::new();
for &lang in languages {
let build_cmd_cfg = config.build_command_config_for_language(lang);
if !check_precondition(lang, build_cmd_cfg.precondition.as_deref()) {
continue;
}
if lang == Language::Rust {
rust_langs.push(lang);
continue;
}
let backend = registry::get_backend(lang);
if let Some(bc) = backend.build_config() {
if bc.depends_on_ffi {
ffi_dependent.push((lang, bc));
need_ffi = true;
} else {
independent.push((lang, bc));
}
} else {
info!("No build config for {lang}, skipping");
}
}
for &lang in &rust_langs {
let build_cmd_cfg = config.build_command_config_for_language(lang);
run_before(lang, build_cmd_cfg.before.as_ref())?;
let cmds = if release {
build_cmd_cfg.build_release.as_ref()
} else {
build_cmd_cfg.build.as_ref()
};
if let Some(cmd_list) = cmds {
for cmd in cmd_list.commands() {
info!("Building {lang}: {cmd}");
run_command(cmd).with_context(|| format!("failed to build {lang}"))?;
}
}
}
if need_ffi
&& !independent
.iter()
.any(|(_, bc)| bc.tool == "cargo" && bc.crate_suffix == "-ffi")
{
let ffi_crate = output_path_for(Language::Ffi, config)
.map(resolve_crate_dir)
.and_then(|p| p.file_name())
.and_then(|n| n.to_str())
.unwrap_or_else(|| {
Box::leak(format!("{crate_name}-ffi").into_boxed_str())
});
info!("Building FFI crate: {ffi_crate}");
let mut cmd = format!("cargo build -p {ffi_crate}");
if release {
cmd.push_str(" --release");
}
run_command(&cmd).context("failed to build FFI crate")?;
}
for (lang, _) in &independent {
let build_cmd_cfg = config.build_command_config_for_language(*lang);
run_before(*lang, build_cmd_cfg.before.as_ref())?;
}
let build_results: Vec<anyhow::Result<(String, String)>> = independent
.par_iter()
.map(|(lang, bc)| {
info!("Building {lang} ({})...", bc.tool);
let build_cmd = build_command_for(*lang, bc, config, release);
run_command_captured(&build_cmd).with_context(|| format!("failed to build language bindings for {lang}"))
})
.collect();
for ((lang, bc), result) in independent.iter().zip(build_results) {
let (stdout, stderr) = result?;
if !stdout.is_empty() {
info!("[{lang} build] {stdout}");
}
if !stderr.is_empty() {
debug!("[{lang} build] {stderr}");
}
run_post_build(*lang, bc, config, &base_dir)
.with_context(|| format!("failed to run post-build steps for {lang}"))?;
}
for (lang, _) in &ffi_dependent {
let build_cmd_cfg = config.build_command_config_for_language(*lang);
run_before(*lang, build_cmd_cfg.before.as_ref())?;
}
let build_results: Vec<anyhow::Result<(String, String)>> = ffi_dependent
.par_iter()
.map(|(lang, bc)| {
info!("Building {lang} ({})...", bc.tool);
let build_cmd = build_command_for(*lang, bc, config, release);
run_command_captured(&build_cmd).with_context(|| format!("failed to build language bindings for {lang}"))
})
.collect();
for ((lang, bc), result) in ffi_dependent.iter().zip(build_results) {
let (stdout, stderr) = result?;
if !stdout.is_empty() {
info!("[{lang} build] {stdout}");
}
if !stderr.is_empty() {
debug!("[{lang} build] {stderr}");
}
run_post_build(*lang, bc, config, &base_dir)
.with_context(|| format!("failed to run post-build steps for {lang}"))?;
}
Ok(())
}
fn resolve_crate_dir(output_path: &Path) -> &Path {
if output_path.file_name().is_some_and(|n| n == "src") {
output_path.parent().unwrap_or(output_path)
} else {
output_path
}
}
fn output_path_for(lang: Language, config: &AlefConfig) -> Option<&Path> {
match lang {
Language::Python => config.output.python.as_deref(),
Language::Node => config.output.node.as_deref(),
Language::Ruby => config.output.ruby.as_deref(),
Language::Php => config.output.php.as_deref(),
Language::Ffi => config.output.ffi.as_deref(),
Language::Go => config.output.go.as_deref(),
Language::Java => config.output.java.as_deref(),
Language::Csharp => config.output.csharp.as_deref(),
Language::Wasm => config.output.wasm.as_deref(),
Language::Elixir => config.output.elixir.as_deref(),
Language::R => config.output.r.as_deref(),
Language::Rust => None,
}
}
fn build_command_for(
lang: Language,
bc: &alef_core::backend::BuildConfig,
config: &AlefConfig,
release: bool,
) -> String {
let release_flag = if release { " --release" } else { "" };
let crate_dir = output_path_for(lang, config)
.map(resolve_crate_dir)
.and_then(|p| p.to_str())
.unwrap_or("");
match bc.tool {
"maturin" => {
format!("maturin develop --manifest-path {crate_dir}/Cargo.toml{release_flag}")
}
"napi" => {
format!("napi build --platform --manifest-path {crate_dir}/Cargo.toml -o {crate_dir}{release_flag}")
}
"wasm-pack" => {
let profile = if release { "--release" } else { "--dev" };
format!("wasm-pack build {crate_dir} {profile} --target bundler")
}
"cargo" => {
let native_dir = Path::new(crate_dir).join("native");
let native_manifest = native_dir.join("Cargo.toml");
if native_manifest.exists() {
let dir = native_dir.display();
format!("cd {dir} && cargo build{release_flag}")
} else {
let crate_name = Path::new(crate_dir)
.file_name()
.and_then(|n| n.to_str())
.unwrap_or(crate_dir);
format!("cargo build -p {crate_name}{release_flag}")
}
}
"mix" => "mix compile".to_string(),
"mvn" => {
let dir = config
.output
.java
.as_ref()
.and_then(|p| p.to_str())
.unwrap_or("packages/java");
format!("cd {dir} && mvn package -DskipTests -q")
}
"dotnet" => {
let dir = config
.output
.csharp
.as_ref()
.and_then(|p| p.to_str())
.unwrap_or("packages/csharp");
let build_dir = {
let dir_path = std::path::Path::new(dir);
let has_direct = dir_path
.read_dir()
.ok()
.map(|entries| {
entries
.filter_map(|e| e.ok())
.any(|e| e.path().extension().is_some_and(|ext| ext == "csproj"))
})
.unwrap_or(false);
if has_direct {
dir.to_string()
} else {
dir_path
.read_dir()
.ok()
.and_then(|entries| {
entries
.filter_map(|e| e.ok())
.find(|e| {
e.path().is_dir()
&& e.path().read_dir().ok().is_some_and(|sub| {
sub.filter_map(|s| s.ok())
.any(|s| s.path().extension().is_some_and(|ext| ext == "csproj"))
})
})
.map(|e| e.path().to_string_lossy().to_string())
})
.unwrap_or_else(|| dir.to_string())
}
};
let dotnet_config = if release { "Release" } else { "Debug" };
format!("cd {build_dir} && dotnet build --configuration {dotnet_config} -q")
}
"go" => {
let dir = config
.output
.go
.as_ref()
.and_then(|p| p.to_str())
.unwrap_or("packages/go");
format!("cd {dir} && go build ./...")
}
other => format!("echo 'Unknown build tool: {other}'"),
}
}
fn run_post_build(
lang: Language,
bc: &alef_core::backend::BuildConfig,
config: &AlefConfig,
base_dir: &Path,
) -> anyhow::Result<()> {
use alef_core::backend::PostBuildStep;
let crate_dir = output_path_for(lang, config)
.map(resolve_crate_dir)
.unwrap_or(Path::new(""));
for step in &bc.post_build {
match step {
PostBuildStep::PatchFile { path, find, replace } => {
let file_path = base_dir.join(crate_dir).join(path);
if file_path.exists() {
let content = std::fs::read_to_string(&file_path)
.with_context(|| format!("failed to read post-build patch target {}", file_path.display()))?;
let patched = content.replace(find, replace);
if patched != content {
std::fs::write(&file_path, &patched)
.with_context(|| format!("failed to write patched file {}", file_path.display()))?;
info!("Patched {}: replaced '{}' → '{}'", file_path.display(), find, replace);
}
} else {
debug!("Post-build patch target not found: {}", file_path.display());
}
}
}
}
Ok(())
}