driller 0.10.2

A clean HTTP load-test drill. Ansible-style YAML plans, Rust runtime, RPS and percentiles per run -- no fancy bits.
//! Build script for driller.
//!
//! Captures git commit hash, build timestamp, and target triple at compile
//! time and exposes them as `GIT_HASH`, `BUILD_TIME`, and `BUILD_TARGET`
//! environment variables. `src/main.rs` reads them via `env!()` to build
//! the `--version` string so users (and the b-bug investigation harness)
//! can tell which exact build is running, not just which version was
//! published.
//!
//! Pattern lifted from accentcms's `build.rs`.

use std::process::Command;

fn main() {
  // Short git commit hash. Falls back to "unknown" if the build is not
  // happening inside a git working tree (e.g. crates.io publish unpacks
  // a tarball without .git).
  let git_hash = Command::new("git").args(["rev-parse", "--short", "HEAD"]).output().ok().and_then(|o| String::from_utf8(o.stdout).ok()).map_or_else(|| "unknown".to_string(), |s| s.trim().to_string());

  println!("cargo:rustc-env=GIT_HASH={git_hash}");

  // Build timestamp in UTC, in a stable `YYYY-MM-DD HH:MM:SS UTC` format.
  // Pure Rust (no chrono/time dependency) so the build script stays
  // dependency-free and cross-platform.
  let build_time = {
    use std::time::SystemTime;
    let now = SystemTime::now().duration_since(SystemTime::UNIX_EPOCH).unwrap_or_default().as_secs();
    let secs_per_day = 86400u64;
    let days = now / secs_per_day;
    let day_secs = now % secs_per_day;
    let hours = day_secs / 3600;
    let minutes = (day_secs % 3600) / 60;
    let seconds = day_secs % 60;
    // Days since epoch comfortably fits in i64 for any realistic build time.
    #[allow(clippy::cast_possible_wrap)]
    let (y, m, d) = civil_from_days(days as i64);
    format!("{y:04}-{m:02}-{d:02} {hours:02}:{minutes:02}:{seconds:02} UTC")
  };

  println!("cargo:rustc-env=BUILD_TIME={build_time}");

  // Target triple (cargo sets TARGET in the build script's environment).
  let target = std::env::var("TARGET").unwrap_or_default();
  println!("cargo:rustc-env=BUILD_TARGET={target}");

  // Trigger a rebuild when HEAD moves so the embedded git hash stays in
  // sync with the working tree without needing `cargo clean`.
  println!("cargo:rerun-if-changed=.git/HEAD");
  println!("cargo:rerun-if-changed=.git/refs/heads/");
}

/// Converts days since the Unix epoch (1970-01-01) into a calendar
/// `(year, month, day)` triple, using Howard Hinnant's `civil_from_days`
/// algorithm.
///
/// Reference: <https://howardhinnant.github.io/date_algorithms.html>
#[allow(clippy::cast_possible_truncation, clippy::cast_sign_loss)]
fn civil_from_days(days: i64) -> (i64, u32, u32) {
  let z = days + 719_468;
  let era = (if z >= 0 {
    z
  } else {
    z - 146_096
  }) / 146_097;
  let doe = (z - era * 146_097) as u32;
  let yoe = (doe - doe / 1460 + doe / 36524 - doe / 146_096) / 365;
  let y = i64::from(yoe) + era * 400;
  let doy = doe - (365 * yoe + yoe / 4 - yoe / 100);
  let mp = (5 * doy + 2) / 153;
  let d = doy - (153 * mp + 2) / 5 + 1;
  let m = if mp < 10 {
    mp + 3
  } else {
    mp - 9
  };
  let y = if m <= 2 {
    y + 1
  } else {
    y
  };
  (y, m, d)
}