feat: with full support for rust services
Some checks failed
continuous-integration/drone/push Build is failing
Some checks failed
continuous-integration/drone/push Build is failing
Signed-off-by: kjuulh <contact@kjuulh.io>
This commit is contained in:
@@ -1,5 +1,9 @@
|
||||
use std::path::PathBuf;
|
||||
|
||||
use async_trait::async_trait;
|
||||
use dagger_rust::source::RustSource;
|
||||
use dagger_sdk::Container;
|
||||
use futures::{stream, StreamExt};
|
||||
|
||||
use crate::{dagger_middleware::DaggerMiddleware, MainAction, PullRequestAction};
|
||||
|
||||
@@ -10,16 +14,22 @@ pub enum RustServiceStage {
|
||||
AfterDeps(DynMiddleware),
|
||||
BeforeBase(DynMiddleware),
|
||||
AfterBase(DynMiddleware),
|
||||
BeforeBuild(DynMiddleware),
|
||||
AfterBuild(DynMiddleware),
|
||||
BeforePackage(DynMiddleware),
|
||||
AfterPackage(DynMiddleware),
|
||||
BeforeRelease(DynMiddleware),
|
||||
AfterRelease(DynMiddleware),
|
||||
}
|
||||
|
||||
pub struct RustService {
|
||||
client: dagger_sdk::Query,
|
||||
|
||||
base_image: Option<dagger_sdk::Container>,
|
||||
|
||||
final_image: Option<dagger_sdk::Container>,
|
||||
stages: Vec<RustServiceStage>,
|
||||
source: Option<PathBuf>,
|
||||
crates: Vec<String>,
|
||||
bin_name: String,
|
||||
}
|
||||
|
||||
impl From<dagger_sdk::Query> for RustService {
|
||||
@@ -27,7 +37,11 @@ impl From<dagger_sdk::Query> for RustService {
|
||||
Self {
|
||||
client: value,
|
||||
base_image: None,
|
||||
final_image: None,
|
||||
stages: Vec::new(),
|
||||
source: None,
|
||||
crates: Vec::new(),
|
||||
bin_name: String::new(),
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -37,7 +51,11 @@ impl RustService {
|
||||
Ok(Self {
|
||||
client: dagger_sdk::connect().await?,
|
||||
base_image: None,
|
||||
final_image: None,
|
||||
stages: Vec::new(),
|
||||
source: None,
|
||||
crates: Vec::new(),
|
||||
bin_name: String::new(),
|
||||
})
|
||||
}
|
||||
|
||||
@@ -47,23 +65,206 @@ impl RustService {
|
||||
self
|
||||
}
|
||||
|
||||
pub fn with_stage(&mut self, _stage: RustServiceStage) -> &mut Self {
|
||||
pub fn with_stage(&mut self, stage: RustServiceStage) -> &mut Self {
|
||||
self.stages.push(stage);
|
||||
|
||||
self
|
||||
}
|
||||
|
||||
pub fn with_sqlx(&mut self) -> &mut Self {
|
||||
pub fn with_source(&mut self, path: impl Into<PathBuf>) -> &mut Self {
|
||||
self.source = Some(path.into());
|
||||
|
||||
self
|
||||
}
|
||||
|
||||
pub async fn build_release(&self) -> eyre::Result<Vec<Container>> {
|
||||
Ok(Vec::new())
|
||||
pub fn with_bin_name(&mut self, bin_name: impl Into<String>) -> &mut Self {
|
||||
self.bin_name = bin_name.into();
|
||||
|
||||
self
|
||||
}
|
||||
|
||||
pub fn with_crates(
|
||||
&mut self,
|
||||
crates: impl IntoIterator<Item = impl Into<String>>,
|
||||
) -> &mut Self {
|
||||
self.crates = crates.into_iter().map(|c| c.into()).collect();
|
||||
|
||||
self
|
||||
}
|
||||
|
||||
fn get_src(&self) -> PathBuf {
|
||||
self.source
|
||||
.clone()
|
||||
.unwrap_or(std::env::current_dir().unwrap())
|
||||
}
|
||||
|
||||
async fn run_stage(
|
||||
&self,
|
||||
stages: impl IntoIterator<Item = &Box<dyn DaggerMiddleware + Send + Sync>>,
|
||||
container: Container,
|
||||
) -> eyre::Result<Container> {
|
||||
let before_deps_stream = stream::iter(stages.into_iter().map(Ok));
|
||||
let res = StreamExt::fold(before_deps_stream, Ok(container), |base, m| async move {
|
||||
match (base, m) {
|
||||
(Ok(base), Ok(m)) => m.handle(base).await,
|
||||
(_, Err(e)) | (Err(e), _) => eyre::bail!("failed with {e}"),
|
||||
}
|
||||
})
|
||||
.await?;
|
||||
|
||||
Ok(res)
|
||||
}
|
||||
|
||||
pub async fn build_base(&self) -> eyre::Result<Container> {
|
||||
let rust_src = RustSource::new(self.client.clone());
|
||||
|
||||
let (src, dep_src) = rust_src
|
||||
.get_rust_src(Some(&self.get_src()), self.crates.clone())
|
||||
.await?;
|
||||
|
||||
let base_image = self
|
||||
.base_image
|
||||
.clone()
|
||||
.unwrap_or(self.client.container().from("rustlang/rust:nightly"));
|
||||
|
||||
let before_deps = self
|
||||
.stages
|
||||
.iter()
|
||||
.filter_map(|s| match s {
|
||||
RustServiceStage::BeforeDeps(middleware) => Some(middleware),
|
||||
_ => None,
|
||||
})
|
||||
.collect::<Vec<_>>();
|
||||
let image = self.run_stage(before_deps, base_image).await?;
|
||||
|
||||
let after_deps = self
|
||||
.stages
|
||||
.iter()
|
||||
.filter_map(|s| match s {
|
||||
RustServiceStage::AfterDeps(m) => Some(m),
|
||||
_ => None,
|
||||
})
|
||||
.collect::<Vec<_>>();
|
||||
let image = self.run_stage(after_deps, image).await?;
|
||||
|
||||
let before_base = self
|
||||
.stages
|
||||
.iter()
|
||||
.filter_map(|s| match s {
|
||||
RustServiceStage::BeforeBase(m) => Some(m),
|
||||
_ => None,
|
||||
})
|
||||
.collect::<Vec<_>>();
|
||||
let image = self.run_stage(before_base, image).await?;
|
||||
|
||||
let cache = self.client.cache_volume("rust_target_cache");
|
||||
|
||||
let rust_prebuild = image
|
||||
.with_workdir("/mnt/src")
|
||||
.with_directory("/mnt/src", dep_src)
|
||||
.with_exec(vec!["cargo", "build", "--release", "--bin", &self.bin_name])
|
||||
.with_mounted_cache("/mnt/src/target/", cache);
|
||||
|
||||
let incremental_dir = rust_src
|
||||
.get_rust_target_src(&self.get_src(), rust_prebuild.clone(), self.crates.clone())
|
||||
.await?;
|
||||
|
||||
let rust_with_src = image
|
||||
.with_workdir("/mnt/src")
|
||||
.with_directory(
|
||||
"/usr/local/cargo",
|
||||
rust_prebuild.directory("/usr/local/cargo"),
|
||||
)
|
||||
.with_directory("/mnt/src/target", incremental_dir)
|
||||
.with_directory("/mnt/src/", src);
|
||||
|
||||
let after_base = self
|
||||
.stages
|
||||
.iter()
|
||||
.filter_map(|s| match s {
|
||||
RustServiceStage::AfterBase(m) => Some(m),
|
||||
_ => None,
|
||||
})
|
||||
.collect::<Vec<_>>();
|
||||
let image = self.run_stage(after_base, rust_with_src).await?;
|
||||
|
||||
Ok(image)
|
||||
}
|
||||
|
||||
pub async fn build_release(&self) -> eyre::Result<Container> {
|
||||
let base = self.build_base().await?;
|
||||
|
||||
let before_build = self
|
||||
.stages
|
||||
.iter()
|
||||
.filter_map(|s| match s {
|
||||
RustServiceStage::BeforeBuild(m) => Some(m),
|
||||
_ => None,
|
||||
})
|
||||
.collect::<Vec<_>>();
|
||||
let base = self.run_stage(before_build, base).await?;
|
||||
|
||||
let binary_build =
|
||||
base.with_exec(vec!["cargo", "build", "--release", "--bin", &self.bin_name]);
|
||||
|
||||
let after_build = self
|
||||
.stages
|
||||
.iter()
|
||||
.filter_map(|s| match s {
|
||||
RustServiceStage::AfterBuild(m) => Some(m),
|
||||
_ => None,
|
||||
})
|
||||
.collect::<Vec<_>>();
|
||||
let binary_build = self.run_stage(after_build, binary_build).await?;
|
||||
|
||||
let dest = self
|
||||
.final_image
|
||||
.clone()
|
||||
.unwrap_or(self.client.container().from("debian:bullseye"));
|
||||
|
||||
let before_package = self
|
||||
.stages
|
||||
.iter()
|
||||
.filter_map(|s| match s {
|
||||
RustServiceStage::BeforePackage(m) => Some(m),
|
||||
_ => None,
|
||||
})
|
||||
.collect::<Vec<_>>();
|
||||
let dest = self.run_stage(before_package, dest).await?;
|
||||
|
||||
let final_image = dest.with_workdir("/mnt/app").with_file(
|
||||
format!("/usr/local/bin/{}", self.bin_name),
|
||||
binary_build.file(format!("/mnt/src/target/release/{}", self.bin_name)),
|
||||
);
|
||||
|
||||
let after_package = self
|
||||
.stages
|
||||
.iter()
|
||||
.filter_map(|s| match s {
|
||||
RustServiceStage::AfterPackage(m) => Some(m),
|
||||
_ => None,
|
||||
})
|
||||
.collect::<Vec<_>>();
|
||||
let final_image = self.run_stage(after_package, final_image).await?;
|
||||
|
||||
Ok(final_image)
|
||||
}
|
||||
|
||||
pub async fn build_test(&self) -> eyre::Result<()> {
|
||||
let base = self.build_base().await?;
|
||||
|
||||
base.with_exec(vec!["cargo", "test", "--release"])
|
||||
.sync()
|
||||
.await?;
|
||||
|
||||
Ok(())
|
||||
}
|
||||
}
|
||||
|
||||
#[async_trait]
|
||||
impl PullRequestAction for RustService {
|
||||
async fn execute_pull_request(&self) -> eyre::Result<()> {
|
||||
self.build_release().await?;
|
||||
let _container = self.build_release().await?;
|
||||
|
||||
Ok(())
|
||||
}
|
||||
@@ -76,26 +277,68 @@ impl MainAction for RustService {
|
||||
}
|
||||
}
|
||||
|
||||
pub mod architecture {
|
||||
#[derive(Debug)]
|
||||
pub enum Architecture {
|
||||
Amd64,
|
||||
Arm64,
|
||||
}
|
||||
|
||||
#[derive(Debug)]
|
||||
pub enum Os {
|
||||
Linux,
|
||||
MacOS,
|
||||
}
|
||||
}
|
||||
|
||||
pub mod apt;
|
||||
pub mod cargo_binstall;
|
||||
pub mod clap_sanity_test;
|
||||
pub mod mold;
|
||||
pub mod sqlx;
|
||||
|
||||
#[cfg(test)]
|
||||
mod test {
|
||||
use futures::FutureExt;
|
||||
|
||||
use crate::{
|
||||
dagger_middleware::middleware,
|
||||
rust_service::{RustService, RustServiceStage},
|
||||
rust_service::{
|
||||
apt::AptExt,
|
||||
architecture::{Architecture, Os},
|
||||
cargo_binstall::CargoBInstallExt,
|
||||
clap_sanity_test::ClapSanityTestExt,
|
||||
mold::MoldActionExt,
|
||||
RustService, RustServiceStage,
|
||||
},
|
||||
};
|
||||
|
||||
#[tokio::test]
|
||||
async fn can_build_rust() -> eyre::Result<()> {
|
||||
async fn test_can_build_rust() -> eyre::Result<()> {
|
||||
let client = dagger_sdk::connect().await?;
|
||||
|
||||
RustService::from(client.clone())
|
||||
.with_base_image(client.container().from("rustlang/rust:nightly"))
|
||||
.with_sqlx()
|
||||
.with_stage(RustServiceStage::BeforeDeps(middleware(|c| async move { Ok(c) }.boxed())))
|
||||
let root_dir = std::path::PathBuf::from("../../").canonicalize()?;
|
||||
|
||||
let container = RustService::from(client.clone())
|
||||
.with_apt(&["git"])
|
||||
.with_cargo_binstall(Architecture::Amd64, Os::Linux, "latest", ["sqlx-cli"])
|
||||
.with_source(root_dir)
|
||||
.with_bin_name("ci")
|
||||
.with_crates(["crates/*", "examples/*", "ci"])
|
||||
.with_mold(Architecture::Amd64, Os::Linux, "2.3.3")
|
||||
.with_stage(RustServiceStage::BeforeDeps(middleware(|c| {
|
||||
async move {
|
||||
// Noop
|
||||
Ok(c)
|
||||
}
|
||||
.boxed()
|
||||
})))
|
||||
.with_clap_sanity_test()
|
||||
.build_release()
|
||||
.await?;
|
||||
|
||||
container.sync().await?;
|
||||
|
||||
Ok(())
|
||||
}
|
||||
}
|
||||
|
Reference in New Issue
Block a user