tesohh-bricks 1.0.4

build system and package manager for C/C++
Documentation
pub mod compile;
pub mod compile_commands;
pub mod include;
pub mod link;
pub mod tools;

use std::{
    fs::{self},
    path::{Path, PathBuf},
    process::Command,
};

use anyhow::Result;
use compile_commands::CompileDatabase;

use crate::{
    cli::{args::BuildCommand, pretty},
    config::{brick::BrickKind, overrides::OverrideDatabase, Config},
};

pub fn build(
    config: &Config,
    build_command: BuildCommand,
    override_cmd: Option<String>,
) -> Result<Option<PathBuf>> {
    if let Some(override_cmd) = override_cmd {
        pretty::msg("build", &override_cmd);

        let mut cmd = match std::env::consts::OS {
            "windows" => {
                let mut cmd = Command::new("powershell.exe");
                cmd.args(["-NoLogo", "-NonInteractive", "-NoProfile"]);
                cmd.arg("/C");
                cmd
            }
            _ => {
                let mut cmd = Command::new("sh");
                cmd.arg("-c");
                cmd
            }
        };
        cmd.current_dir(build_command.path);
        cmd.arg(override_cmd);
        cmd.status()?;

        return Ok(None);
    }

    let mut compile_paths = vec![];

    let src_path = Path::new(&build_command.path).join("src");

    let override_path = Path::new(&build_command.path)
        .join("build")
        .join("overrides.json");
    let override_file = match fs::read_to_string(override_path) {
        Ok(v) => v,
        Err(_) => {
            pretty::warning(format!(
                "no `build/overrides.json` found in {}. defaulting to no overrides.",
                &build_command.path
            ));
            "{}".to_string()
        }
    };
    let override_db: OverrideDatabase = serde_json::from_str(&override_file)?;

    let mut compile_db = CompileDatabase::new();

    for entry in walkdir::WalkDir::new(&src_path).follow_links(true) {
        let Some((path, compile_cmd)) = compile::compile(
            config,
            entry,
            &override_db,
            build_command.force,
            build_command.silent,
        )?
        else {
            continue;
        };
        compile_paths.push(path);
        compile_db.push(compile_cmd);
    }

    let mut ldflags = config.brick.ldflags.to_string();
    ldflags += &match config.brick.platform() {
        Some(platform) => match &platform.ldflags {
            Some(v) => format!(" {}", v),
            None => "".to_string(),
        },
        None => "".to_string(),
    };

    let build_result = match config.brick.kind {
        BrickKind::Binary => link::binary(
            &config.libs,
            &compile_paths,
            &Path::new(&build_command.path)
                .join("build")
                .join(&config.brick.name),
            &override_db,
            &ldflags,
            config.brick.lang,
            build_command.silent,
        ),
        BrickKind::Library => link::library(
            &config.libs,
            &compile_paths,
            &Path::new(&build_command.path)
                .join("build")
                .join("lib")
                .join(String::from("lib") + &config.brick.name + ".a"),
            &override_db,
            &ldflags,
            config.brick.lang,
            build_command.silent,
        ),
    };

    if build_command.emit_compile_commands {
        if !build_command.silent {
            pretty::msg("emit", "build/compile_commands.json");
        }
        let comp_path = Path::new(&build_command.path)
            .join("build")
            .join("compile_commands.json");
        let comp_file = fs::File::create(comp_path)?;
        serde_json::to_writer(comp_file, &compile_db)?;
    };

    include::copy_headers(&src_path, &config.brick.name)?;

    build_result
}

#[cfg(test)]
mod tests {
    use crate::build::compile::src_to_build_path;
    use std::path::Path;

    #[test]
    fn check_src_to_build_path() {
        assert_eq!(
            src_to_build_path(Path::new("./src/main.c")),
            Path::new("./build/main.o")
        );
        assert_eq!(
            src_to_build_path(Path::new("./src/utils/crazy/main.c")),
            Path::new("./build/utils/crazy/main.o")
        );
        assert_eq!(
            src_to_build_path(Path::new("./main.c")),
            Path::new("./main.o")
        );
    }
}