Compare commits

...

5 Commits

Author SHA1 Message Date
pyranota
d39b870b2a add windows flags 2025-04-11 15:55:07 +02:00
pyranota
f107936d44 fix compilation error 2025-04-11 15:34:58 +02:00
pyranota
9221a63edb Merge branch 'main' into rust-inc-comp 2025-04-11 15:12:50 +02:00
pyranota
99fcacb2e6 update dockerfiles 2025-04-11 15:10:53 +02:00
pyranota
cb49ae5458 feat: rust incremental compilation 2025-04-11 13:55:22 +02:00
5 changed files with 95 additions and 10 deletions

View File

@@ -4,7 +4,11 @@ use uuid::Uuid;
use windmill_parser_rust::parse_rust_deps_into_manifest;
use itertools::Itertools;
use tokio::{fs::File, io::AsyncReadExt, process::Command};
use tokio::{
fs::{create_dir_all, File},
io::AsyncReadExt,
process::Command,
};
use windmill_common::{
error::{self, Error},
utils::calculate_hash,
@@ -33,6 +37,10 @@ lazy_static::lazy_static! {
static ref CARGO_HOME: String = std::env::var("CARGO_HOME").unwrap_or_else(|_| { CARGO_HOME_DEFAULT.clone() });
static ref RUSTUP_HOME: String = std::env::var("RUSTUP_HOME").unwrap_or_else(|_| { RUSTUP_HOME_DEFAULT.clone() });
static ref CARGO_PATH: String = std::env::var("CARGO_PATH").unwrap_or_else(|_| format!("{}/bin/cargo", CARGO_HOME.as_str()));
static ref CARGO_SWEEP_PATH: String = std::env::var("CARGO_SWEEP_PATH").unwrap_or_else(|_| format!("{}/bin/cargo-sweep", CARGO_HOME.as_str()));
static ref SWEEP_MAXSIZE: String = std::env::var("CARGO_SWEEP_MAXSIZE").unwrap_or("5GB".to_owned());
static ref NO_SHARED_BUILD_DIR: bool = std::env::var("RUST_NO_SHARED_BUILD_DIR").ok().map(|flag| flag == "true").unwrap_or(false);
}
#[cfg(windows)]
@@ -176,18 +184,85 @@ pub async fn generate_cargo_lockfile(
}
pub async fn build_rust_crate(
job_id: &Uuid,
job: &MiniPulledJob,
mem_peak: &mut i32,
canceled_by: &mut Option<CanceledBy>,
job_dir: &str,
conn: &Connection,
worker_name: &str,
w_id: &str,
base_internal_url: &str,
hash: &str,
occupancy_metrics: &mut OccupancyMetrics,
) -> error::Result<String> {
let bin_path = format!("{}/{hash}", RUST_CACHE_DIR);
let target = if let Some(path) = &job.runnable_path {
let t = format!(
"{RUST_CACHE_DIR}/targets/{}@{}@{}",
&job.workspace_id,
path.replace('/', "."),
&job.created_by
);
if *NO_SHARED_BUILD_DIR {
None
} else if let Err(e) = create_dir_all(&t).await {
tracing::warn!("Could not create shared target dir. Incremental compilation is not possible.\ne: {e}");
None
} else {
let mut sweep_cmd = Command::new(CARGO_SWEEP_PATH.as_str());
sweep_cmd
.current_dir(job_dir)
.env_clear()
.env("PATH", PATH_ENV.as_str())
.env("CARGO_HOME", CARGO_HOME.as_str())
.env("HOME", HOME_ENV.as_str())
.env("CARGO_TARGET_DIR", &(t.clone() + "/target"))
.env("RUSTUP_HOME", RUSTUP_HOME.as_str())
.args(["sweep", "--maxsize", SWEEP_MAXSIZE.as_str()])
.stdout(Stdio::piped())
.stderr(Stdio::piped());
#[cfg(windows)]
{
sweep_cmd.env("SystemRoot", SYSTEM_ROOT.as_str());
sweep_cmd.env(
"TMP",
std::env::var("TMP").unwrap_or_else(|_| "C:\\tmp".to_string()),
);
sweep_cmd.env("USERPROFILE", crate::USERPROFILE_ENV.as_str());
}
if let Err(e) = match start_child_process(sweep_cmd, CARGO_SWEEP_PATH.as_str()).await {
Ok(sweep_process) => {
handle_child(
&job.id,
conn,
mem_peak,
canceled_by,
sweep_process,
false,
worker_name,
&job.workspace_id,
"cargo sweep",
None,
false,
&mut Some(occupancy_metrics),
None,
)
.await
}
Err(e) => Err(e),
} {
tracing::warn!(
workspace_id = %job.workspace_id,
job_id = %job.id,
"Failed to run `cargo sweep`. Rust cache may grow over time, cargo sweep is meant to clean up unused cache.\ne: {e}\n"
);
}
Some(t)
}
} else {
tracing::warn!("Runnable path is None. Incremental compilation is not possible.");
None
};
let mut build_rust_cmd = Command::new(CARGO_PATH.as_str());
build_rust_cmd
@@ -203,6 +278,9 @@ pub async fn build_rust_crate(
.stdout(Stdio::piped())
.stderr(Stdio::piped());
if let Some(ref t) = target {
build_rust_cmd.env("CARGO_TARGET_DIR", &(t.clone() + "/target"));
}
#[cfg(windows)]
{
build_rust_cmd.env("SystemRoot", SYSTEM_ROOT.as_str());
@@ -215,14 +293,14 @@ pub async fn build_rust_crate(
let build_rust_process = start_child_process(build_rust_cmd, CARGO_PATH.as_str()).await?;
handle_child(
job_id,
&job.id,
conn,
mem_peak,
canceled_by,
build_rust_process,
false,
worker_name,
w_id,
&job.workspace_id,
"rust build",
None,
false,
@@ -230,10 +308,13 @@ pub async fn build_rust_crate(
None,
)
.await?;
append_logs(job_id, w_id, "\n\n", conn).await;
append_logs(&job.id, &job.workspace_id, "\n\n", conn).await;
tokio::fs::copy(
&format!("{job_dir}/target/release/main"),
&format!(
"{}/target/release/main",
target.as_ref().unwrap_or(&(job_dir.to_owned()))
),
format! {"{job_dir}/main"},
)
.await
@@ -331,13 +412,12 @@ pub async fn handle_rust_job(
create_args_and_out_file(client, job, job_dir, conn).await?;
build_rust_crate(
&job.id,
&job,
mem_peak,
canceled_by,
job_dir,
conn,
worker_name,
&job.workspace_id,
base_internal_url,
&hash,
occupancy_metrics,

View File

@@ -3,6 +3,7 @@ FROM ghcr.io/windmill-labs/windmill:dev
# Rust
COPY --from=rust:1.81.0 /usr/local/cargo /usr/local/cargo
COPY --from=rust:1.81.0 /usr/local/rustup /usr/local/rustup
RUN /usr/local/cargo/bin/cargo install cargo-sweep --version ^0.7
# Ansible
RUN uv tool install ansible && [ -d "$(uv tool dir)/ansible/bin/" ] && find "$(uv tool dir)/ansible/bin/" -mindepth 1 -maxdepth 1 -type f -executable -regextype posix-extended -regex '^((.+/)?)[^.]+' -print0 | xargs -0 ln -s -t "$UV_TOOL_BIN_DIR/" || true

View File

@@ -22,6 +22,7 @@ FROM ghcr.io/windmill-labs/windmill-ee:dev
# Rust
COPY --from=rust:1.81.0 /usr/local/cargo /usr/local/cargo
COPY --from=rust:1.81.0 /usr/local/rustup /usr/local/rustup
RUN /usr/local/cargo/bin/cargo install cargo-sweep --version ^0.7
# Ansible
RUN uv tool install ansible && [ -d "$(uv tool dir)/ansible/bin/" ] && find "$(uv tool dir)/ansible/bin/" -mindepth 1 -maxdepth 1 -type f -executable -regextype posix-extended -regex '^((.+/)?)[^.]+' -print0 | xargs -0 ln -s -t "$UV_TOOL_BIN_DIR/" || true

View File

@@ -45,6 +45,7 @@ RUN apt-get update && apt-get install -y libprotobuf-dev libnl-route-3-dev
# Rust
COPY --from=rust:1.80.1 /usr/local/cargo /usr/local/cargo
COPY --from=rust:1.80.1 /usr/local/rustup /usr/local/rustup
RUN /usr/local/cargo/bin/cargo install cargo-sweep --version ^0.7
# Ansible
RUN uv tool install ansible && [ -d "$(uv tool dir)/ansible/bin/" ] && find "$(uv tool dir)/ansible/bin/" -mindepth 1 -maxdepth 1 -type f -executable -regextype posix-extended -regex '^((.+/)?)[^.]+' -print0 | xargs -0 ln -s -t "$UV_TOOL_BIN_DIR/" || true

View File

@@ -77,6 +77,7 @@
devShells.default = pkgs.mkShell {
buildInputs = buildInputs ++ (with pkgs; [
rust
cargo-sweep
git
xcaddy
sqlx-cli
@@ -172,12 +173,13 @@
JAVA_PATH = "${pkgs.jdk21}/bin/java";
JAVAC_PATH = "${pkgs.jdk21}/bin/javac";
COURSIER_PATH = "${coursier}/coursier";
# for related places search: ADD_NEW_LANG
FLOCK_PATH = "${pkgs.flock}/bin/flock";
CARGO_PATH = "${rust}/bin/cargo";
CARGO_SWEEP_PATH = "${pkgs.cargo-sweep}/bin/cargo-sweep";
DOTNET_PATH = "${pkgs.dotnet-sdk_9}/bin/dotnet";
DOTNET_ROOT = "${pkgs.dotnet-sdk_9}/share/dotnet";
ORACLE_LIB_DIR = "${pkgs.oracle-instantclient.lib}/lib";
# for related places search: ADD_NEW_LANG
RUST_LOG = "debug";
};
packages.default = self.packages.${system}.windmill;