Skip to content

Commit

Permalink
work with pre-built bins
Browse files Browse the repository at this point in the history
  • Loading branch information
zeeshanlakhani committed Jul 13, 2024
1 parent 4a1cb65 commit a4f1097
Show file tree
Hide file tree
Showing 7 changed files with 192 additions and 60 deletions.
14 changes: 6 additions & 8 deletions .github/buildomat/jobs/check-features.sh
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,9 @@
#: variety = "basic"
#: target = "helios-2.0"
#: rust_toolchain = true
#: output_rules = []
#: output_rules = [
#: "/out/*",
#: ]

# Run the check-features `xtask` on illumos, testing compilation of feature combinations.

Expand All @@ -15,22 +17,18 @@ set -o xtrace
cargo --version
rustc --version

# NOTE: This version should be in sync with the recommended version in
# ./dev-tools/xtask/src/check-features.rs.
CARGO_HACK_VERSION='0.6.28'

#
# Set up our PATH for use with this workspace.
#
source ./env.sh
export PATH="$PATH:$PWD/out/cargo-hack"

banner prerequisites
ptime -m bash ./tools/install_builder_prerequisites.sh -y

#
# Check the feature set with the `cargo xtask check-features` command.
# Check feature combinations with the `cargo xtask check-features` command.
#
banner hack-check
export CARGO_INCREMENTAL=0
ptime -m timeout 2h cargo xtask check-features --ci --install-version "$CARGO_HACK_VERSION"
RUSTDOCFLAGS="--document-private-items -D warnings" ptime -m cargo doc --workspace --no-deps --no-default-features
ptime -m timeout 2h cargo xtask check-features --ci
12 changes: 5 additions & 7 deletions .github/workflows/rust.yml
Original file line number Diff line number Diff line change
Expand Up @@ -86,7 +86,6 @@ jobs:
runs-on: ubuntu-22.04
env:
CARGO_INCREMENTAL: 0
CARGO_HACK_VERSION: 0.6.28
steps:
# This repo is unstable and unnecessary: https://github.com/microsoft/linux-package-repositories/issues/34
- name: Disable packages.microsoft.com repo
Expand All @@ -99,18 +98,17 @@ jobs:
- name: Report cargo version
run: cargo --version
- name: Update PATH
run: source "./env.sh"; echo "PATH=$PATH" >> "$GITHUB_ENV"
run: |
set -x
export PATH="./out/cargo-hack:$PATH"
source "./env.sh"; echo "PATH=$PATH" >> "$GITHUB_ENV"
- name: Print PATH
run: echo $PATH
- name: Print GITHUB_ENV
run: cat "$GITHUB_ENV"
- name: Install Pre-Requisites
run: ./tools/install_builder_prerequisites.sh -y
# Uses manifest for install
- uses: taiki-e/install-action@v2
with:
tool: cargo-hack@${{ env.CARGO_HACK_VERSION }}
- name: Run Check on Features (Feature-Powerset, No-Dev-Deps)
- name: Run Check on Feature Combinations (Feature-Powerset, No-Dev-Deps)
timeout-minutes: 120 # 2 hours
run: cargo xtask check-features --ci

Expand Down
8 changes: 6 additions & 2 deletions README.adoc
Original file line number Diff line number Diff line change
Expand Up @@ -114,9 +114,13 @@ We check that certain system library dependencies are not leaked outside of thei

=== Checking feature flag combinations

To ensure that varying combinations of features compile, run `cargo xtask check-features`, which executes the https://github.com/taiki-e/cargo-hack[`cargo hack`] subcommand under the hood. This `xtask` is run in CI using the `--ci` parameter , which automatically exludes certain `image-*` features that purposefully cause compiler errors if set.
To ensure that varying combinations of features compile, run `cargo xtask check-features`, which executes the https://github.com/taiki-e/cargo-hack[`cargo hack`] subcommand under the hood.

If you don't have `cargo hack` installed locally, run the the `xtask` with the `install-version <VERSION>` option, which will install it into your user's `.cargo` directory:
This `xtask` is run in CI using the `--ci` parameter , which automatically exludes certain `image-*` features that purposefully cause compiler errors if set and uses a pre-built binary.

If `cargo hack` is not already installed in omicron's `out/` directory, a pre-built binary will be installed automatically depending on your operating system and architecture.

You can also run the the `xtask` with the `install-version <VERSION>` option, which will install the cargo subcommand into your user's `.cargo` directory:

[source,text]
----
Expand Down
137 changes: 94 additions & 43 deletions dev-tools/xtask/src/check_features.rs
Original file line number Diff line number Diff line change
Expand Up @@ -4,15 +4,12 @@

//! Subcommand: cargo xtask check-features
use anyhow::{bail, Context, Result};
use anyhow::{bail, Result};
use camino::{Utf8Path, Utf8PathBuf};
use clap::Parser;
use std::{collections::HashSet, process::Command};

/// The default version of `cargo-hack` to install.
/// We use a patch-floating version to avoid breaking the build when a new
/// version is released (locally).
const FLOAT_VERSION: &str = "~0.6.28";

const SUPPORTED_ARCHITECTURES: [&str; 1] = ["x86_64"];
const CI_EXCLUDED_FEATURES: [&str; 2] = ["image-trampoline", "image-standard"];

#[derive(Parser)]
Expand All @@ -29,27 +26,31 @@ pub struct Args {
/// Error format passed to `cargo hack check`.
#[clap(long, value_name = "FMT")]
message_format: Option<String>,
/// Version of `cargo-hack` to install.
/// Version of `cargo-hack` to install. By default, we download a pre-built
/// version.
#[clap(long, value_name = "VERSION")]
install_version: Option<String>,
}

/// Run `cargo hack check`.
pub fn run_cmd(args: Args) -> Result<()> {
// Install `cargo-hack` if the `install-version` was specified.
if let Some(version) = args.install_version {
install_cargo_hack(Some(version))?;
// We cannot specify both `--ci` and `--install-version`, as the former
// implies we are using a pre-built version.
if args.ci && args.install_version.is_some() {
bail!("cannot specify --ci and --install-version together");
}

let cargo =
std::env::var("CARGO").unwrap_or_else(|_| String::from("cargo"));

let mut command = Command::new(&cargo);

// Add the `hack check` subcommand.
command.args(&["hack", "check"]);

// Add the `--exclude-features` flag if we are running in CI mode.
if args.ci {
install_prebuilt_cargo_hack(&cargo)?;

let ex = if let Some(mut features) = args.exclude_features {
// Extend the list of features to exclude with the CI defaults.
features.extend(
Expand All @@ -64,8 +65,10 @@ pub fn run_cmd(args: Args) -> Result<()> {
CI_EXCLUDED_FEATURES.join(",")
};

// Add the `--exclude-features` flag if we are running in CI mode.
command.args(["--exclude-features", &ex]);
} else {
install_cargo_hack(&cargo, args.install_version)?;
// Add "only" the `--exclude-features` flag if it was provided.
if let Some(features) = args.exclude_features {
command.args(["--exclude-features", &features.join(",")]);
Expand All @@ -91,48 +94,96 @@ pub fn run_cmd(args: Args) -> Result<()> {
// We will not check the dev-dependencies, which should covered by tests.
.arg("--no-dev-deps");

eprintln!(
"running: {:?} {}",
&cargo,
command
.get_args()
.map(|arg| format!("{:?}", arg.to_str().unwrap()))
.collect::<Vec<_>>()
.join(" ")
);

let exit_status = command
.spawn()
.context("failed to spawn child process")?
.wait()
.context("failed to wait for child process")?;

if !exit_status.success() {
bail!("check-features failed: {}", exit_status);
}
exec(command)
}

Ok(())
/// The supported operating systems.
enum Os {
Illumos,
Linux,
Mac,
}

/// Install `cargo-hack` at the specified version or the default version.
fn install_cargo_hack(version: Option<String>) -> Result<()> {
let cargo =
std::env::var("CARGO").unwrap_or_else(|_| String::from("cargo"));
/// Get the current OS.
fn os_name() -> Result<Os> {
let os = match std::env::consts::OS {
"linux" => Os::Linux,
"macos" => Os::Mac,
"solaris" | "illumos" => Os::Illumos,
other => bail!("OS not supported: {other}"),
};
Ok(os)
}

let mut command = Command::new(&cargo);
/// Get the path to the `out` directory.
fn out_dir() -> Utf8PathBuf {
if let Ok(omicron_dir) = std::env::var("OMICRON") {
Utf8Path::new(format!("{}/out/cargo-hack", omicron_dir).as_str())
.to_path_buf()
} else {
Utf8Path::new("out/cargo-hack").to_path_buf()
}
}

/// Install `cargo-hack` if the `install-version` was specified; otherwise,
/// download a pre-built version if it's not already in our `out` directory.
fn install_cargo_hack(cargo: &str, version: Option<String>) -> Result<()> {
if let Some(version) = version {
let mut command = Command::new(cargo);

eprintln!(
"installing cargo-hack at version {} to {}",
version,
env!("CARGO_HOME")
);
command.args(&["install", "cargo-hack", "--version", &version]);
exec(command)
} else if !out_dir().exists() {
install_prebuilt_cargo_hack(cargo)
} else {
command.args(&[
"install",
"cargo-hack",
"--locked",
"--version",
FLOAT_VERSION,
]);
let out_dir = out_dir();
eprintln!("cargo-hack found in {}", out_dir);
Ok(())
}
}

/// Download a pre-built version of `cargo-hack` to the `out` directory via the
/// download `xtask`.
fn install_prebuilt_cargo_hack(cargo: &str) -> Result<()> {
let mut command = Command::new(cargo);

let out_dir = out_dir();
eprintln!(
"cargo-hack not found in {}, downloading a pre-built version",
out_dir
);

let os = os_name()?;
match os {
Os::Illumos | Os::Linux | Os::Mac
if SUPPORTED_ARCHITECTURES.contains(&std::env::consts::ARCH) =>
{
// Download the pre-built version of `cargo-hack` via our
// download `xtask`.
command.args(&["xtask", "download", "cargo-hack"]);
}
_ => {
bail!(
"cargo-hack is not pre-built for this os {} / arch {}",
std::env::consts::OS,
std::env::consts::ARCH
);
}
}

exec(command)
}

/// Execute the command and check the exit status.
fn exec(mut command: Command) -> Result<()> {
let cargo =
std::env::var("CARGO").unwrap_or_else(|_| String::from("cargo"));

eprintln!(
"running: {:?} {}",
&cargo,
Expand Down
Loading

0 comments on commit a4f1097

Please sign in to comment.