diff --git a/twoliter/src/cargo_make.rs b/twoliter/src/cargo_make.rs new file mode 100644 index 000000000..138599e49 --- /dev/null +++ b/twoliter/src/cargo_make.rs @@ -0,0 +1,213 @@ +use std::path::PathBuf; + +use anyhow::{bail, Result}; +use log::trace; +use tokio::process::Command; + +use crate::common::exec; +use crate::docker::ImageArchUri; +use crate::project::Project; + +fn require_sdk(project: &Project, arch: &str) -> Result<(ImageArchUri, ImageArchUri)> { + match (project.sdk(arch), project.toolchain(arch)) { + (Some(s), Some(t)) => Ok((s, t)), + _ => bail!( + "When using twoliter make, it is required that the SDK and toolchain be specified in \ + Twoliter.toml" + ), + } +} + +#[derive(Debug, Clone, Default)] +pub(crate) struct CargoMake { + makefile_path: Option, + project_dir: Option, + args: Vec, +} + +impl CargoMake { + pub(crate) fn new(project: &Project, arch: S) -> Result + where + S: Into, + { + let (sdk, toolchain) = require_sdk(project, &arch.into())?; + Ok(Self::default() + .env("TLPRIVATE_SDK_IMAGE", sdk) + .env("TLPRIVATE_TOOLCHAIN", toolchain)) + } + + pub(crate) fn makefile

(mut self, makefile_path: P) -> Self + where + P: Into, + { + self.makefile_path = Some(makefile_path.into()); + self + } + + pub(crate) fn project_dir

(mut self, project_dir: P) -> Self + where + P: Into, + { + self.project_dir = Some(project_dir.into()); + self + } + + pub(crate) fn env(mut self, key: S1, value: S2) -> Self + where + S1: Into, + S2: Into, + { + self.args + .push(format!("-e={}={}", key.into(), value.into())); + self + } + + pub(crate) fn _envs(mut self, env_vars: V) -> Self + where + S1: Into, + S2: Into, + V: Into>, + { + for (key, value) in env_vars.into() { + self.args + .push(format!("-e={}={}", key.into(), value.into())); + } + self + } + + pub(crate) fn _arg(mut self, arg: S) -> Self + where + S: Into, + { + self.args.push(arg.into()); + self + } + + pub(crate) fn _args(mut self, args: V) -> Self + where + S: Into, + V: Into>, + { + self.args.extend(args.into().into_iter().map(Into::into)); + self + } + + pub(crate) async fn _exec(&self, task: S) -> Result<()> + where + S: Into, + { + self.exec_with_args(task, Vec::::new()).await + } + + pub(crate) async fn exec_with_args(&self, task: S1, args: V) -> Result<()> + where + S1: Into, + S2: Into, + V: Into>, + { + exec( + Command::new("cargo") + .arg("make") + .arg("--disable-check-for-updates") + .args( + self.makefile_path.iter().flat_map(|path| { + vec!["--makefile".to_string(), path.display().to_string()] + }), + ) + .args( + self.makefile_path + .iter() + .flat_map(|path| vec!["--cwd".to_string(), path.display().to_string()]), + ) + .args(build_system_env_vars()?) + .args(&self.args) + .arg(task.into()) + .args(args.into().into_iter().map(Into::into)), + ) + .await + } +} + +fn build_system_env_vars() -> Result> { + let mut args = Vec::new(); + for (key, val) in std::env::vars() { + if is_build_system_env(key.as_str()) { + trace!("Passing env var {} to cargo make", key); + args.push("-e".to_string()); + args.push(format!("{}={}", key, val)); + } + + // To avoid confusion, environment variables whose values have been moved to + // Twoliter.toml are expressly disallowed here. + check_for_disallowed_var(&key)?; + } + Ok(args) +} + +/// A list of environment variables that don't conform to naming conventions but need to be passed +/// through to the `cargo make` invocation. +const ENV_VARS: [&str; 12] = [ + "ALLOW_MISSING_KEY", + "AMI_DATA_FILE_SUFFIX", + "CARGO_MAKE_CARGO_ARGS", + "CARGO_MAKE_CARGO_LIMIT_JOBS", + "CARGO_MAKE_DEFAULT_TESTSYS_KUBECONFIG_PATH", + "CARGO_MAKE_TESTSYS_ARGS", + "CARGO_MAKE_TESTSYS_KUBECONFIG_ARG", + "MARK_OVA_AS_TEMPLATE", + "RELEASE_START_TIME", + "SSM_DATA_FILE_SUFFIX", + "VMWARE_IMPORT_SPEC_PATH", + "VMWARE_VM_NAME_DEFAULT", +]; + +const DISALLOWED_SDK_VARS: [&str; 4] = [ + "BUILDSYS_SDK_NAME", + "BUILDSYS_SDK_VERSION", + "BUILDSYS_REGISTRY", + "BUILDSYS_TOOLCHAIN", +]; + +/// Returns `true` if `key` is an environment variable that needs to be passed to `cargo make`. +fn is_build_system_env(key: impl AsRef) -> bool { + let key = key.as_ref(); + key.starts_with("BUILDSYS_") + || key.starts_with("PUBLISH_") + || key.starts_with("REPO_") + || key.starts_with("TESTSYS_") + || key.starts_with("BOOT_CONFIG") + || key.starts_with("AWS_") + || ENV_VARS.contains(&key) +} + +fn check_for_disallowed_var(key: &str) -> Result<()> { + if DISALLOWED_SDK_VARS.contains(&key) { + bail!( + "The environment variable '{}' can no longer be used. Specify the SDK in Twoliter.toml", + key + ) + } + Ok(()) +} + +#[test] +fn test_is_build_system_env() { + assert!(is_build_system_env( + "CARGO_MAKE_DEFAULT_TESTSYS_KUBECONFIG_PATH" + )); + assert!(is_build_system_env("BUILDSYS_PRETTY_NAME")); + assert!(is_build_system_env("PUBLISH_FOO_BAR")); + assert!(is_build_system_env("TESTSYS_!")); + assert!(is_build_system_env("BOOT_CONFIG!")); + assert!(is_build_system_env("BOOT_CONFIG_INPUT")); + assert!(is_build_system_env("AWS_REGION")); + assert!(!is_build_system_env("PATH")); + assert!(!is_build_system_env("HOME")); + assert!(!is_build_system_env("COLORTERM")); +} + +#[test] +fn test_check_for_disallowed_var() { + assert!(check_for_disallowed_var("BUILDSYS_REGISTRY").is_err()); + assert!(check_for_disallowed_var("BUILDSYS_PRETTY_NAME").is_ok()); +} diff --git a/twoliter/src/cmd/make.rs b/twoliter/src/cmd/make.rs index 04731ea0d..f8201d261 100644 --- a/twoliter/src/cmd/make.rs +++ b/twoliter/src/cmd/make.rs @@ -1,13 +1,9 @@ -use crate::common::exec; -use crate::docker::ImageArchUri; +use crate::cargo_make::CargoMake; use crate::project; -use crate::project::Project; use crate::tools::{install_tools, tools_tempdir}; -use anyhow::{bail, ensure, Result}; +use anyhow::Result; use clap::Parser; -use log::trace; use std::path::PathBuf; -use tokio::process::Command; /// Run a cargo make command in Twoliter's build environment. Known Makefile.toml environment /// variables will be passed-through to the cargo make invocation. @@ -29,6 +25,9 @@ pub(crate) struct Make { /// Uninspected arguments to be passed to cargo make after the target name. For example, --foo /// in the following command : cargo make test --foo. additional_args: Vec, + + #[clap(env = "BUILDSYS_ARCH")] + arch: String, } impl Make { @@ -37,134 +36,12 @@ impl Make { let tempdir = tools_tempdir()?; install_tools(&tempdir).await?; let makefile_path = tempdir.path().join("Makefile.toml"); - - let mut args = vec![ - "make".to_string(), - "--disable-check-for-updates".to_string(), - "--makefile".to_string(), - makefile_path.display().to_string(), - "--cwd".to_string(), - project.project_dir().display().to_string(), - ]; - - let mut arch = String::new(); - - for (key, val) in std::env::vars() { - if is_build_system_env(key.as_str()) { - trace!("Passing env var {} to cargo make", key); - args.push("-e".to_string()); - args.push(format!("{}={}", key, val)); - } - - // To avoid confusion, environment variables whose values have been moved to - // Twoliter.toml are expressly disallowed here. - check_for_disallowed_var(&key)?; - - if key == "BUILDSYS_ARCH" { - arch = val.clone(); - } - } - - ensure!( - !arch.is_empty(), - "It is required to pass a non-zero string as the value of environment variable \ - 'BUILDSYS_ARCH' when running twoliter make" - ); - - let (sdk, toolchain) = require_sdk(&project, &arch)?; - - args.push(format!("-e=TLPRIVATE_SDK_IMAGE={}", sdk)); - args.push(format!("-e=TLPRIVATE_TOOLCHAIN={}", toolchain)); - args.push(format!("-e=CARGO_HOME={}", self.cargo_home.display())); - args.push(format!( - "-e=TWOLITER_TOOLS_DIR={}", - tempdir.path().display() - )); - - args.push(self.makefile_task.clone()); - - for cargo_make_arg in &self.additional_args { - args.push(cargo_make_arg.clone()); - } - - exec(Command::new("cargo").args(args)).await - } -} - -/// A list of environment variables that don't conform to naming conventions but need to be passed -/// through to the `cargo make` invocation. -const ENV_VARS: [&str; 12] = [ - "ALLOW_MISSING_KEY", - "AMI_DATA_FILE_SUFFIX", - "CARGO_MAKE_CARGO_ARGS", - "CARGO_MAKE_CARGO_LIMIT_JOBS", - "CARGO_MAKE_DEFAULT_TESTSYS_KUBECONFIG_PATH", - "CARGO_MAKE_TESTSYS_ARGS", - "CARGO_MAKE_TESTSYS_KUBECONFIG_ARG", - "MARK_OVA_AS_TEMPLATE", - "RELEASE_START_TIME", - "SSM_DATA_FILE_SUFFIX", - "VMWARE_IMPORT_SPEC_PATH", - "VMWARE_VM_NAME_DEFAULT", -]; - -const DISALLOWED_SDK_VARS: [&str; 4] = [ - "BUILDSYS_SDK_NAME", - "BUILDSYS_SDK_VERSION", - "BUILDSYS_REGISTRY", - "BUILDSYS_TOOLCHAIN", -]; - -/// Returns `true` if `key` is an environment variable that needs to be passed to `cargo make`. -fn is_build_system_env(key: impl AsRef) -> bool { - let key = key.as_ref(); - key.starts_with("BUILDSYS_") - || key.starts_with("PUBLISH_") - || key.starts_with("REPO_") - || key.starts_with("TESTSYS_") - || key.starts_with("BOOT_CONFIG") - || key.starts_with("AWS_") - || ENV_VARS.contains(&key) -} - -fn check_for_disallowed_var(key: &str) -> Result<()> { - if DISALLOWED_SDK_VARS.contains(&key) { - bail!( - "The environment variable '{}' can no longer be used. Specify the SDK in Twoliter.toml", - key - ) + CargoMake::new(&project, &self.arch)? + .env("CARGO_HOME", self.cargo_home.display().to_string()) + .env("TWOLITER_TOOLS_DIR", tempdir.path().display().to_string()) + .makefile(makefile_path) + .project_dir(project.project_dir()) + .exec_with_args(&self.makefile_task, self.additional_args.clone()) + .await } - Ok(()) -} - -fn require_sdk(project: &Project, arch: &str) -> Result<(ImageArchUri, ImageArchUri)> { - match (project.sdk(arch), project.toolchain(arch)) { - (Some(s), Some(t)) => Ok((s, t)), - _ => bail!( - "When using twoliter make, it is required that the SDK and toolchain be specified in \ - Twoliter.toml" - ), - } -} - -#[test] -fn test_is_build_system_env() { - assert!(is_build_system_env( - "CARGO_MAKE_DEFAULT_TESTSYS_KUBECONFIG_PATH" - )); - assert!(is_build_system_env("BUILDSYS_PRETTY_NAME")); - assert!(is_build_system_env("PUBLISH_FOO_BAR")); - assert!(is_build_system_env("TESTSYS_!")); - assert!(is_build_system_env("BOOT_CONFIG!")); - assert!(is_build_system_env("BOOT_CONFIG_INPUT")); - assert!(is_build_system_env("AWS_REGION")); - assert!(!is_build_system_env("PATH")); - assert!(!is_build_system_env("HOME")); - assert!(!is_build_system_env("COLORTERM")); -} - -#[test] -fn test_check_for_disallowed_var() { - assert!(check_for_disallowed_var("BUILDSYS_REGISTRY").is_err()); - assert!(check_for_disallowed_var("BUILDSYS_PRETTY_NAME").is_ok()); } diff --git a/twoliter/src/docker/image.rs b/twoliter/src/docker/image.rs index 98b48fd09..eb23582ac 100644 --- a/twoliter/src/docker/image.rs +++ b/twoliter/src/docker/image.rs @@ -101,6 +101,12 @@ impl Display for ImageArchUri { } } +impl From for String { + fn from(value: ImageArchUri) -> Self { + value.to_string() + } +} + #[test] fn image_arch_uri_no_registry() { let uri = ImageArchUri::new(None, "my-sdk", "i386", "v0.33.1"); diff --git a/twoliter/src/main.rs b/twoliter/src/main.rs index cb4d8c28c..9a3ea4be6 100644 --- a/twoliter/src/main.rs +++ b/twoliter/src/main.rs @@ -2,6 +2,7 @@ use crate::cmd::{init_logger, Args}; use anyhow::Result; use clap::Parser; +mod cargo_make; mod cmd; mod common; mod docker;