fj 0.40.0

Early-stage, next-generation, code-first CAD application. Because the world needs another CAD program.
Documentation
use std::{
    fmt::Write,
    path::PathBuf,
    process::{Command, Output, Stdio},
};

fn main() -> anyhow::Result<()> {
    let version = Version::determine()?;

    println!("cargo:rustc-env=FJ_VERSION_PKG={}", version.pkg);
    println!("cargo:rustc-env=FJ_VERSION_FULL={}", version.full);

    // Make sure the build script doesn't run too often.
    println!("cargo:rerun-if-changed=Cargo.toml");

    Ok(())
}

struct Version {
    pkg: String,
    full: String,
}
impl Version {
    fn determine() -> anyhow::Result<Self> {
        let pkg = std::env::var("CARGO_PKG_VERSION").unwrap();
        let commit = git_description();

        let official_release =
            std::env::var("RELEASE_DETECTED").as_deref() == Ok("true");
        println!("cargo:rerun-if-env-changed=RELEASE_DETECTED");

        let mut full = format!("{pkg} (");

        if official_release {
            write!(full, "official release binary")?;
        } else {
            write!(full, "development build")?;
        }

        if let Some(commit) = commit {
            write!(full, "; {commit}")?;
        }

        writeln!(full, ")")?;

        Ok(Self { pkg, full })
    }
}

/// Try to get the current git commit.
///
/// This may fail if `git` isn't installed (unlikely) or if the `.git/` folder
/// isn't accessible (more likely than you think). This typically happens when
/// we're building just the `fj-app` crate in a Docker container or when
/// someone is installing from crates.io via `cargo install`.
fn git_description() -> Option<String> {
    let crate_dir = PathBuf::from(std::env::var("CARGO_MANIFEST_DIR").unwrap());

    let mut cmd = Command::new("git");
    cmd.args(["describe", "--always", "--tags"])
        .stdout(Stdio::piped())
        .stderr(Stdio::piped())
        .current_dir(&crate_dir);

    let Output {
        status,
        stdout,
        stderr,
    } = cmd.output().ok()?;

    let stdout = String::from_utf8_lossy(&stdout);

    if !status.success() {
        // Not sure if anyone will ever see this, but it could be helpful for
        // troubleshooting.
        eprintln!("Command failed: {cmd:?}");
        let stderr = String::from_utf8_lossy(&stderr);
        eprintln!("---- Stdout ----");
        eprintln!("{stdout}");
        eprintln!("---- Stderr ----");
        eprintln!("{stderr}");
        return None;
    }

    // Make sure we re-run whenever the current commit changes
    let project_root = crate_dir.ancestors().nth(2).unwrap();
    let head_file = project_root.join(".git").join("HEAD");
    println!("cargo:rerun-if-changed={}", head_file.display());

    if let Ok(contents) = std::fs::read_to_string(&head_file) {
        // Most of the time the HEAD file will be `ref: refs/heads/$branch`, but
        // when it's a detached head we'll only get the commit hash and can skip
        // the rerun-if-changed logic.

        if let Some((_, branch)) = contents.split_once(' ') {
            let commit_hash_file =
                project_root.join(".git").join(branch.trim());
            println!("cargo:rerun-if-changed={}", commit_hash_file.display());
        }
    }

    Some(stdout.trim().to_string())
}