nested_workspace 0.7.4

Run Cargo commands on workspaces in workspaces
Documentation
use anyhow::{Result, ensure};
use assert_cmd::{assert::OutputAssertExt, output::OutputError};
use cargo_metadata::MetadataCommand;
use elaborate::std::{path::PathContext, process::CommandContext};
use std::{
    env::remove_var, ffi::OsStr, path::Path, path::PathBuf, process::Command, sync::LazyLock,
};
use walkdir::WalkDir;

mod util;
use util::Timestamps;

#[ctor::ctor]
fn initialize() {
    unsafe {
        remove_var("CARGO_TERM_COLOR");
    }
}

static CARGO_NESTED: LazyLock<PathBuf> = LazyLock::new(|| {
    Command::new("cargo")
        .args(["build", "--package", "cargo-nested"])
        .assert()
        .success();
    let metadata = MetadataCommand::new().no_deps().exec().unwrap();
    metadata
        .target_directory
        .join("debug/cargo-nested")
        .into_std_path_buf()
});

#[test]
fn build_script_always_runs() {
    let mut failures = Vec::new();
    for result in WalkDir::new(Path::new(env!("CARGO_MANIFEST_DIR"))) {
        let entry = result.unwrap();
        if entry.file_name() != OsStr::new("build.rs") {
            continue;
        }
        let path = entry.path();
        let manifest_path = path.with_file_name("Cargo.toml");
        eprintln!("{}", manifest_path.display());
        fetch(&manifest_path);
        if let Err(error) = check_then_build(&manifest_path) {
            failures.push((manifest_path, error));
        }
    }
    assert!(failures.is_empty(), "{failures:#?}");
}

// smoelius: `fetch` is for the `git_dependency` fixture. Note that we must use `cargo nested fetch`
// and not just `cargo fetch` because the command is run in the containing package's directory.
fn fetch(manifest_path: &Path) {
    let manifest_dir = manifest_path.parent_wc().unwrap();
    let mut command = Command::new(&*CARGO_NESTED);
    command.args(["nested", "fetch"]);
    command.current_dir(manifest_dir);
    let status = command.status_wc().unwrap();
    assert!(status.success());
}

fn check_then_build(manifest_path: &Path) -> Result<()> {
    let mut timestamps_before = None;
    for build in [false, true] {
        let mut command = Command::new("cargo");
        command.args([
            if build { "build" } else { "check" },
            "-vv",
            "--offline",
            "--manifest-path",
        ]);
        command.arg(manifest_path);
        let output = command.output_wc()?;
        if build {
            ensure!(output.status.success());
        } else {
            if !output.status.success() {
                eprintln!("command failed: {command:?}: {}", OutputError::new(output));
                return Ok(());
            }
            let stdout = String::from_utf8(output.stdout)?;
            let timestamps = Timestamps::new(&stdout)?;
            eprintln!("`timestamps` before build: {:#?}", timestamps.get());
            ensure!(timestamps.get().is_empty() == false);
            timestamps_before = Some(timestamps);
        }
    }
    let timestamps_before = timestamps_before.unwrap();
    let timestamps_after = timestamps_before.rescan()?;
    eprintln!("`timestamps` after build: {:#?}", timestamps_after.get());
    timestamps_before.compare(&timestamps_after)?;
    Ok(())
}