Compare commits
3 Commits
82ccdefd93
...
7a1ad63b57
Author | SHA1 | Date | |
---|---|---|---|
7a1ad63b57 | |||
80782e70f9 | |||
3939940c01 |
1
Cargo.lock
generated
1
Cargo.lock
generated
@ -326,6 +326,7 @@ version = "0.2.0"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"async-trait",
|
"async-trait",
|
||||||
"clap",
|
"clap",
|
||||||
|
"dagger-rust",
|
||||||
"dagger-sdk",
|
"dagger-sdk",
|
||||||
"eyre",
|
"eyre",
|
||||||
"futures",
|
"futures",
|
||||||
|
@ -1,5 +1,4 @@
|
|||||||
use std::path::PathBuf;
|
use std::path::PathBuf;
|
||||||
use std::sync::Arc;
|
|
||||||
|
|
||||||
use clap::Args;
|
use clap::Args;
|
||||||
use clap::Parser;
|
use clap::Parser;
|
||||||
@ -62,10 +61,10 @@ async fn main() -> eyre::Result<()> {
|
|||||||
let _ = dotenv::dotenv();
|
let _ = dotenv::dotenv();
|
||||||
let _ = color_eyre::install();
|
let _ = color_eyre::install();
|
||||||
|
|
||||||
let client = dagger_sdk::connect().await?;
|
|
||||||
|
|
||||||
let cli = Command::parse();
|
let cli = Command::parse();
|
||||||
|
|
||||||
|
let client = dagger_sdk::connect().await?;
|
||||||
|
|
||||||
match &cli.commands {
|
match &cli.commands {
|
||||||
Commands::Local { command } => match command {
|
Commands::Local { command } => match command {
|
||||||
LocalCommands::Test => {
|
LocalCommands::Test => {
|
||||||
@ -107,7 +106,6 @@ async fn main() -> eyre::Result<()> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
mod please_release {
|
mod please_release {
|
||||||
use std::sync::Arc;
|
|
||||||
|
|
||||||
use dagger_cuddle_please::{models::CuddlePleaseSrcArgs, DaggerCuddlePleaseAction};
|
use dagger_cuddle_please::{models::CuddlePleaseSrcArgs, DaggerCuddlePleaseAction};
|
||||||
|
|
||||||
@ -136,7 +134,7 @@ mod please_release {
|
|||||||
}
|
}
|
||||||
|
|
||||||
mod test {
|
mod test {
|
||||||
use std::{path::PathBuf, sync::Arc};
|
use std::path::PathBuf;
|
||||||
|
|
||||||
use dagger_rust::build::RustVersion;
|
use dagger_rust::build::RustVersion;
|
||||||
|
|
||||||
|
@ -10,6 +10,8 @@ repository.workspace = true
|
|||||||
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
|
dagger-rust.workspace = true
|
||||||
|
|
||||||
dagger-sdk.workspace = true
|
dagger-sdk.workspace = true
|
||||||
eyre.workspace = true
|
eyre.workspace = true
|
||||||
clap.workspace = true
|
clap.workspace = true
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
use std::{env::Args, sync::Arc};
|
use std::{sync::Arc};
|
||||||
|
|
||||||
use async_trait::async_trait;
|
use async_trait::async_trait;
|
||||||
|
|
||||||
@ -50,17 +50,17 @@ impl CuddleCI {
|
|||||||
|
|
||||||
match matches.subcommand() {
|
match matches.subcommand() {
|
||||||
Some((name, args)) => match (name, args) {
|
Some((name, args)) => match (name, args) {
|
||||||
("pr", args) => {
|
("pr", _args) => {
|
||||||
eprintln!("starting pr validate");
|
eprintln!("starting pr validate");
|
||||||
self.pr_action.execute_pull_request().await?;
|
self.pr_action.execute_pull_request().await?;
|
||||||
eprintln!("finished pr validate");
|
eprintln!("finished pr validate");
|
||||||
}
|
}
|
||||||
("main", args) => {
|
("main", _args) => {
|
||||||
eprintln!("starting main validate");
|
eprintln!("starting main validate");
|
||||||
self.main_action.execute_main().await?;
|
self.main_action.execute_main().await?;
|
||||||
eprintln!("finished main validate");
|
eprintln!("finished main validate");
|
||||||
}
|
}
|
||||||
("release", args) => {
|
("release", _args) => {
|
||||||
eprintln!("starting release validate");
|
eprintln!("starting release validate");
|
||||||
self.release_action.execute_release().await?;
|
self.release_action.execute_release().await?;
|
||||||
eprintln!("finished release validate");
|
eprintln!("finished release validate");
|
||||||
|
@ -5,7 +5,7 @@ use std::{future::Future, pin::Pin};
|
|||||||
#[async_trait]
|
#[async_trait]
|
||||||
pub trait DaggerMiddleware {
|
pub trait DaggerMiddleware {
|
||||||
async fn handle(
|
async fn handle(
|
||||||
&mut self,
|
&self,
|
||||||
container: dagger_sdk::Container,
|
container: dagger_sdk::Container,
|
||||||
) -> eyre::Result<dagger_sdk::Container> {
|
) -> eyre::Result<dagger_sdk::Container> {
|
||||||
Ok(container)
|
Ok(container)
|
||||||
@ -14,14 +14,14 @@ pub trait DaggerMiddleware {
|
|||||||
|
|
||||||
pub struct DaggerMiddlewareFn<F>
|
pub struct DaggerMiddlewareFn<F>
|
||||||
where
|
where
|
||||||
F: FnMut(Container) -> Pin<Box<dyn Future<Output = eyre::Result<Container>> + Send>>,
|
F: Fn(Container) -> Pin<Box<dyn Future<Output = eyre::Result<Container>> + Send>>,
|
||||||
{
|
{
|
||||||
pub func: F,
|
pub func: F,
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn middleware<F>(func: F) -> Box<DaggerMiddlewareFn<F>>
|
pub fn middleware<F>(func: F) -> Box<DaggerMiddlewareFn<F>>
|
||||||
where
|
where
|
||||||
F: FnMut(Container) -> Pin<Box<dyn Future<Output = eyre::Result<Container>> + Send>>,
|
F: Fn(Container) -> Pin<Box<dyn Future<Output = eyre::Result<Container>> + Send>>,
|
||||||
{
|
{
|
||||||
Box::new(DaggerMiddlewareFn { func })
|
Box::new(DaggerMiddlewareFn { func })
|
||||||
}
|
}
|
||||||
@ -29,11 +29,9 @@ where
|
|||||||
#[async_trait]
|
#[async_trait]
|
||||||
impl<F> DaggerMiddleware for DaggerMiddlewareFn<F>
|
impl<F> DaggerMiddleware for DaggerMiddlewareFn<F>
|
||||||
where
|
where
|
||||||
F: FnMut(Container) -> Pin<Box<dyn Future<Output = eyre::Result<Container>> + Send>>
|
F: Fn(Container) -> Pin<Box<dyn Future<Output = eyre::Result<Container>> + Send>> + Send + Sync,
|
||||||
+ Send
|
|
||||||
+ Sync,
|
|
||||||
{
|
{
|
||||||
async fn handle(&mut self, container: Container) -> eyre::Result<Container> {
|
async fn handle(&self, container: Container) -> eyre::Result<Container> {
|
||||||
// Call the closure stored in the struct
|
// Call the closure stored in the struct
|
||||||
(self.func)(container).await
|
(self.func)(container).await
|
||||||
}
|
}
|
||||||
|
@ -2,110 +2,4 @@ pub mod cli;
|
|||||||
pub use cli::*;
|
pub use cli::*;
|
||||||
|
|
||||||
pub mod dagger_middleware;
|
pub mod dagger_middleware;
|
||||||
|
pub mod rust_service;
|
||||||
pub mod rust_service {
|
|
||||||
use std::{future::Future, pin::Pin, sync::Arc};
|
|
||||||
|
|
||||||
use async_trait::async_trait;
|
|
||||||
use dagger_sdk::Container;
|
|
||||||
use futures::future::BoxFuture;
|
|
||||||
|
|
||||||
use crate::{dagger_middleware::DaggerMiddleware, MainAction, PullRequestAction};
|
|
||||||
|
|
||||||
pub type DynMiddleware = Box<dyn DaggerMiddleware + Send + Sync>;
|
|
||||||
|
|
||||||
pub enum RustServiceStage {
|
|
||||||
BeforeBase(DynMiddleware),
|
|
||||||
AfterBase(DynMiddleware),
|
|
||||||
BeforeRelease(DynMiddleware),
|
|
||||||
AfterRelease(DynMiddleware),
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct RustService {
|
|
||||||
client: dagger_sdk::Query,
|
|
||||||
|
|
||||||
base_image: Option<dagger_sdk::Container>,
|
|
||||||
|
|
||||||
stages: Vec<RustServiceStage>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl From<dagger_sdk::Query> for RustService {
|
|
||||||
fn from(value: dagger_sdk::Query) -> Self {
|
|
||||||
Self {
|
|
||||||
client: value,
|
|
||||||
base_image: None,
|
|
||||||
stages: Vec::new(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl RustService {
|
|
||||||
pub async fn new() -> eyre::Result<Self> {
|
|
||||||
Ok(Self {
|
|
||||||
client: dagger_sdk::connect().await?,
|
|
||||||
base_image: None,
|
|
||||||
stages: Vec::new(),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn with_base_image(&mut self, base: dagger_sdk::Container) -> &mut Self {
|
|
||||||
self.base_image = Some(base);
|
|
||||||
|
|
||||||
self
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn add_stage(&mut self, stage: RustServiceStage) -> &mut Self {
|
|
||||||
self
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn with_sqlx(&mut self) -> &mut Self {
|
|
||||||
self
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn build_release(&self) -> eyre::Result<Vec<Container>> {
|
|
||||||
Ok(Vec::new())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[async_trait]
|
|
||||||
impl PullRequestAction for RustService {
|
|
||||||
async fn execute_pull_request(&self) -> eyre::Result<()> {
|
|
||||||
self.build_release().await?;
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[async_trait]
|
|
||||||
impl MainAction for RustService {
|
|
||||||
async fn execute_main(&self) -> eyre::Result<()> {
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(test)]
|
|
||||||
mod test {
|
|
||||||
use futures::FutureExt;
|
|
||||||
|
|
||||||
use crate::{
|
|
||||||
dagger_middleware::middleware,
|
|
||||||
rust_service::{RustService, RustServiceStage},
|
|
||||||
};
|
|
||||||
|
|
||||||
use super::*;
|
|
||||||
|
|
||||||
#[tokio::test]
|
|
||||||
async fn 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()
|
|
||||||
.add_stage(RustServiceStage::BeforeBase(middleware(|c| async move { Ok(c) }.boxed())))
|
|
||||||
.build_release()
|
|
||||||
.await?;
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
344
crates/cuddle-ci/src/rust_service.rs
Normal file
344
crates/cuddle-ci/src/rust_service.rs
Normal file
@ -0,0 +1,344 @@
|
|||||||
|
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};
|
||||||
|
|
||||||
|
pub type DynMiddleware = Box<dyn DaggerMiddleware + Send + Sync>;
|
||||||
|
|
||||||
|
pub enum RustServiceStage {
|
||||||
|
BeforeDeps(DynMiddleware),
|
||||||
|
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 {
|
||||||
|
fn from(value: dagger_sdk::Query) -> Self {
|
||||||
|
Self {
|
||||||
|
client: value,
|
||||||
|
base_image: None,
|
||||||
|
final_image: None,
|
||||||
|
stages: Vec::new(),
|
||||||
|
source: None,
|
||||||
|
crates: Vec::new(),
|
||||||
|
bin_name: String::new(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl RustService {
|
||||||
|
pub async fn new() -> eyre::Result<Self> {
|
||||||
|
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(),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn with_base_image(&mut self, base: dagger_sdk::Container) -> &mut Self {
|
||||||
|
self.base_image = Some(base);
|
||||||
|
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn with_stage(&mut self, stage: RustServiceStage) -> &mut Self {
|
||||||
|
self.stages.push(stage);
|
||||||
|
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn with_source(&mut self, path: impl Into<PathBuf>) -> &mut Self {
|
||||||
|
self.source = Some(path.into());
|
||||||
|
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
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<()> {
|
||||||
|
let _container = self.build_release().await?;
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[async_trait]
|
||||||
|
impl MainAction for RustService {
|
||||||
|
async fn execute_main(&self) -> eyre::Result<()> {
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
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::{
|
||||||
|
apt::AptExt,
|
||||||
|
architecture::{Architecture, Os},
|
||||||
|
cargo_binstall::CargoBInstallExt,
|
||||||
|
clap_sanity_test::ClapSanityTestExt,
|
||||||
|
mold::MoldActionExt,
|
||||||
|
RustService, RustServiceStage,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
#[tokio::test]
|
||||||
|
async fn test_can_build_rust() -> eyre::Result<()> {
|
||||||
|
let client = dagger_sdk::connect().await?;
|
||||||
|
|
||||||
|
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(())
|
||||||
|
}
|
||||||
|
}
|
52
crates/cuddle-ci/src/rust_service/apt.rs
Normal file
52
crates/cuddle-ci/src/rust_service/apt.rs
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
use async_trait::async_trait;
|
||||||
|
use dagger_sdk::Container;
|
||||||
|
|
||||||
|
use crate::dagger_middleware::DaggerMiddleware;
|
||||||
|
|
||||||
|
use super::RustService;
|
||||||
|
|
||||||
|
pub struct Apt {
|
||||||
|
deps: Vec<String>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Apt {
|
||||||
|
pub fn new() -> Self {
|
||||||
|
Self { deps: Vec::new() }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn add(mut self, dep_name: impl Into<String>) -> Self {
|
||||||
|
self.deps.push(dep_name.into());
|
||||||
|
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn extend(mut self, deps: &[&str]) -> Self {
|
||||||
|
self.deps.extend(deps.iter().map(|s| s.to_string()));
|
||||||
|
|
||||||
|
self
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[async_trait]
|
||||||
|
impl DaggerMiddleware for Apt {
|
||||||
|
async fn handle(&self, container: Container) -> eyre::Result<Container> {
|
||||||
|
let mut deps = vec!["apt", "install", "-y"];
|
||||||
|
deps.extend(self.deps.iter().map(|s| s.as_str()));
|
||||||
|
|
||||||
|
let c = container.with_exec(vec!["apt", "update"]).with_exec(deps);
|
||||||
|
|
||||||
|
Ok(c)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub trait AptExt {
|
||||||
|
fn with_apt(&mut self, deps: &[&str]) -> &mut Self {
|
||||||
|
self
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl AptExt for RustService {
|
||||||
|
fn with_apt(&mut self, deps: &[&str]) -> &mut Self {
|
||||||
|
self.with_stage(super::RustServiceStage::BeforeDeps(Box::new(Apt::new().extend(deps))))
|
||||||
|
}
|
||||||
|
}
|
109
crates/cuddle-ci/src/rust_service/cargo_binstall.rs
Normal file
109
crates/cuddle-ci/src/rust_service/cargo_binstall.rs
Normal file
@ -0,0 +1,109 @@
|
|||||||
|
use async_trait::async_trait;
|
||||||
|
use dagger_sdk::Container;
|
||||||
|
|
||||||
|
use crate::dagger_middleware::DaggerMiddleware;
|
||||||
|
|
||||||
|
use super::{
|
||||||
|
architecture::{Architecture, Os},
|
||||||
|
RustService,
|
||||||
|
};
|
||||||
|
|
||||||
|
pub struct CargoBInstall {
|
||||||
|
arch: Architecture,
|
||||||
|
os: Os,
|
||||||
|
version: String,
|
||||||
|
crates: Vec<String>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl CargoBInstall {
|
||||||
|
pub fn new(
|
||||||
|
arch: Architecture,
|
||||||
|
os: Os,
|
||||||
|
version: impl Into<String>,
|
||||||
|
crates: impl Into<Vec<String>>,
|
||||||
|
) -> Self {
|
||||||
|
Self {
|
||||||
|
arch,
|
||||||
|
os,
|
||||||
|
version: version.into(),
|
||||||
|
crates: crates.into(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get_arch(&self) -> String {
|
||||||
|
match self.arch {
|
||||||
|
Architecture::Amd64 => "x86_64",
|
||||||
|
Architecture::Arm64 => "armv7",
|
||||||
|
}
|
||||||
|
.into()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get_os(&self) -> String {
|
||||||
|
match self.os {
|
||||||
|
Os::Linux => "linux",
|
||||||
|
Os::MacOS => "darwin",
|
||||||
|
}
|
||||||
|
.into()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_download_url(&self) -> String {
|
||||||
|
format!("https://github.com/cargo-bins/cargo-binstall/releases/{}/download/cargo-binstall-{}-unknown-{}-musl.tgz", self.version, self.get_arch(), self.get_os())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_archive(&self) -> String {
|
||||||
|
format!(
|
||||||
|
"cargo-binstall-{}-unknown-{}-musl.tgz",
|
||||||
|
self.get_arch(),
|
||||||
|
self.get_os()
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[async_trait]
|
||||||
|
impl DaggerMiddleware for CargoBInstall {
|
||||||
|
async fn handle(&self, container: Container) -> eyre::Result<Container> {
|
||||||
|
let c =
|
||||||
|
container
|
||||||
|
.with_exec(vec!["wget", &self.get_download_url()])
|
||||||
|
.with_exec(vec!["tar", "-xvf", &self.get_archive()])
|
||||||
|
.with_exec(
|
||||||
|
"mv cargo-binstall /usr/local/cargo/bin"
|
||||||
|
.split_whitespace()
|
||||||
|
.collect(),
|
||||||
|
);
|
||||||
|
|
||||||
|
let c = self.crates.iter().cloned().fold(c, |acc, item| {
|
||||||
|
acc.with_exec(vec!["cargo", "binstall", &item, "-y"])
|
||||||
|
});
|
||||||
|
|
||||||
|
Ok(c)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub trait CargoBInstallExt {
|
||||||
|
fn with_cargo_binstall(
|
||||||
|
&mut self,
|
||||||
|
arch: Architecture,
|
||||||
|
os: Os,
|
||||||
|
version: impl Into<String>,
|
||||||
|
crates: impl IntoIterator<Item = impl Into<String>>,
|
||||||
|
) -> &mut Self {
|
||||||
|
self
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl CargoBInstallExt for RustService {
|
||||||
|
fn with_cargo_binstall(
|
||||||
|
&mut self,
|
||||||
|
arch: Architecture,
|
||||||
|
os: Os,
|
||||||
|
version: impl Into<String>,
|
||||||
|
crates: impl IntoIterator<Item = impl Into<String>>,
|
||||||
|
) -> &mut Self {
|
||||||
|
let crates: Vec<String> = crates.into_iter().map(|s| s.into()).collect();
|
||||||
|
|
||||||
|
self.with_stage(super::RustServiceStage::BeforeDeps(
|
||||||
|
Box::new(CargoBInstall::new(arch, os, version, crates))
|
||||||
|
))
|
||||||
|
}
|
||||||
|
}
|
41
crates/cuddle-ci/src/rust_service/clap_sanity_test.rs
Normal file
41
crates/cuddle-ci/src/rust_service/clap_sanity_test.rs
Normal file
@ -0,0 +1,41 @@
|
|||||||
|
use async_trait::async_trait;
|
||||||
|
use dagger_sdk::Container;
|
||||||
|
|
||||||
|
use crate::dagger_middleware::DaggerMiddleware;
|
||||||
|
|
||||||
|
use super::RustService;
|
||||||
|
|
||||||
|
pub struct ClapSanityTest {
|
||||||
|
bin_name: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ClapSanityTest {
|
||||||
|
pub fn new(bin_name: impl Into<String>) -> Self {
|
||||||
|
Self {
|
||||||
|
bin_name: bin_name.into(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[async_trait]
|
||||||
|
impl DaggerMiddleware for ClapSanityTest {
|
||||||
|
async fn handle(&self, container: Container) -> eyre::Result<Container> {
|
||||||
|
Ok(container.with_exec(vec![&self.bin_name, "--help"]))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub trait ClapSanityTestExt {
|
||||||
|
fn with_clap_sanity_test(&mut self) -> &mut Self {
|
||||||
|
self
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ClapSanityTestExt for RustService {
|
||||||
|
fn with_clap_sanity_test(&mut self) -> &mut Self {
|
||||||
|
self.with_stage(
|
||||||
|
super::RustServiceStage::AfterPackage(Box::new(ClapSanityTest::new(&self.bin_name)))
|
||||||
|
);
|
||||||
|
|
||||||
|
self
|
||||||
|
}
|
||||||
|
}
|
112
crates/cuddle-ci/src/rust_service/mold.rs
Normal file
112
crates/cuddle-ci/src/rust_service/mold.rs
Normal file
@ -0,0 +1,112 @@
|
|||||||
|
use async_trait::async_trait;
|
||||||
|
|
||||||
|
use crate::dagger_middleware::DaggerMiddleware;
|
||||||
|
|
||||||
|
use super::{
|
||||||
|
architecture::{Architecture, Os},
|
||||||
|
RustService,
|
||||||
|
};
|
||||||
|
|
||||||
|
pub struct MoldInstall {
|
||||||
|
arch: Architecture,
|
||||||
|
os: Os,
|
||||||
|
version: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl MoldInstall {
|
||||||
|
pub fn new(arch: Architecture, os: Os, version: impl Into<String>) -> Self {
|
||||||
|
Self {
|
||||||
|
arch,
|
||||||
|
os,
|
||||||
|
version: version.into(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get_arch(&self) -> String {
|
||||||
|
match self.arch {
|
||||||
|
Architecture::Amd64 => "x86_64",
|
||||||
|
Architecture::Arm64 => "arm",
|
||||||
|
}
|
||||||
|
.into()
|
||||||
|
}
|
||||||
|
fn get_os(&self) -> String {
|
||||||
|
match &self.os {
|
||||||
|
Os::Linux => "linux",
|
||||||
|
o => todo!("os not implemented for mold: {:?}", o),
|
||||||
|
}
|
||||||
|
.into()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_download_url(&self) -> String {
|
||||||
|
format!(
|
||||||
|
"https://github.com/rui314/mold/releases/download/v{}/mold-{}-{}-{}.tar.gz",
|
||||||
|
self.version,
|
||||||
|
self.version,
|
||||||
|
self.get_arch(),
|
||||||
|
self.get_os()
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_folder(&self) -> String {
|
||||||
|
format!(
|
||||||
|
"mold-{}-{}-{}",
|
||||||
|
self.version,
|
||||||
|
self.get_arch(),
|
||||||
|
self.get_os()
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_archive_name(&self) -> String {
|
||||||
|
format!(
|
||||||
|
"mold-{}-{}-{}.tar.gz",
|
||||||
|
self.version,
|
||||||
|
self.get_arch(),
|
||||||
|
self.get_os()
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[async_trait]
|
||||||
|
impl DaggerMiddleware for MoldInstall {
|
||||||
|
async fn handle(
|
||||||
|
&self,
|
||||||
|
container: dagger_sdk::Container,
|
||||||
|
) -> eyre::Result<dagger_sdk::Container> {
|
||||||
|
println!("installing mold");
|
||||||
|
|
||||||
|
let c = container
|
||||||
|
.with_exec(vec!["wget", &self.get_download_url()])
|
||||||
|
.with_exec(vec!["tar", "-xvf", &self.get_archive_name()])
|
||||||
|
.with_exec(vec![
|
||||||
|
"mv",
|
||||||
|
&format!("{}/bin/mold", self.get_folder()),
|
||||||
|
"/usr/bin/mold",
|
||||||
|
]);
|
||||||
|
|
||||||
|
Ok(c)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub trait MoldActionExt {
|
||||||
|
fn with_mold(
|
||||||
|
&mut self,
|
||||||
|
architecture: Architecture,
|
||||||
|
os: Os,
|
||||||
|
version: impl Into<String>,
|
||||||
|
) -> &mut Self {
|
||||||
|
self
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl MoldActionExt for RustService {
|
||||||
|
fn with_mold(
|
||||||
|
&mut self,
|
||||||
|
architecture: Architecture,
|
||||||
|
os: Os,
|
||||||
|
version: impl Into<String>,
|
||||||
|
) -> &mut Self {
|
||||||
|
self.with_stage(super::RustServiceStage::AfterDeps(
|
||||||
|
Box::new(MoldInstall::new(architecture, os, version))
|
||||||
|
))
|
||||||
|
}
|
||||||
|
}
|
12
crates/cuddle-ci/src/rust_service/sqlx.rs
Normal file
12
crates/cuddle-ci/src/rust_service/sqlx.rs
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
use async_trait::async_trait;
|
||||||
|
use dagger_sdk::Container;
|
||||||
|
|
||||||
|
use crate::dagger_middleware::DaggerMiddleware;
|
||||||
|
|
||||||
|
pub struct Sqlx {}
|
||||||
|
#[async_trait]
|
||||||
|
impl DaggerMiddleware for Sqlx {
|
||||||
|
async fn handle(&self, container: Container) -> eyre::Result<Container> {
|
||||||
|
Ok(container.with_env_variable("SQLX_OFFLINE", "true"))
|
||||||
|
}
|
||||||
|
}
|
@ -1,4 +1,4 @@
|
|||||||
use std::sync::{Arc, Mutex};
|
use std::sync::{Arc};
|
||||||
|
|
||||||
use models::{CuddlePleaseArgs, CuddlePleaseSrcArgs};
|
use models::{CuddlePleaseArgs, CuddlePleaseSrcArgs};
|
||||||
use traits::CuddlePlease;
|
use traits::CuddlePlease;
|
||||||
|
@ -97,7 +97,8 @@ impl RustBuild {
|
|||||||
|
|
||||||
let mut containers = Vec::new();
|
let mut containers = Vec::new();
|
||||||
for container_image in images {
|
for container_image in images {
|
||||||
let container = match &container_image {
|
let container =
|
||||||
|
match &container_image {
|
||||||
SlimImage::Debian { image, deps, .. } => {
|
SlimImage::Debian { image, deps, .. } => {
|
||||||
let target = BuildTarget::from_target(&container_image);
|
let target = BuildTarget::from_target(&container_image);
|
||||||
|
|
||||||
@ -124,11 +125,7 @@ impl RustBuild {
|
|||||||
"-p",
|
"-p",
|
||||||
bin_name,
|
bin_name,
|
||||||
])
|
])
|
||||||
.file(format!(
|
.file(format!("target/{}/release/{}", target.to_string(), bin_name));
|
||||||
"target/{}/release/{}",
|
|
||||||
target.to_string(),
|
|
||||||
bin_name
|
|
||||||
));
|
|
||||||
|
|
||||||
self.build_debian_image(
|
self.build_debian_image(
|
||||||
bin,
|
bin,
|
||||||
@ -166,11 +163,7 @@ impl RustBuild {
|
|||||||
"-p",
|
"-p",
|
||||||
bin_name,
|
bin_name,
|
||||||
])
|
])
|
||||||
.file(format!(
|
.file(format!("target/{}/release/{}", target.to_string(), bin_name));
|
||||||
"target/{}/release/{}",
|
|
||||||
target.to_string(),
|
|
||||||
bin_name
|
|
||||||
));
|
|
||||||
|
|
||||||
self.build_alpine_image(
|
self.build_alpine_image(
|
||||||
bin,
|
bin,
|
||||||
|
@ -110,7 +110,7 @@ impl HtmxBuild {
|
|||||||
let container =
|
let container =
|
||||||
match &container_image {
|
match &container_image {
|
||||||
SlimImage::Debian { image, deps, .. } => {
|
SlimImage::Debian { image, deps, .. } => {
|
||||||
let target = BuildTarget::from_target(&container_image);
|
let _target = BuildTarget::from_target(&container_image);
|
||||||
|
|
||||||
let build_container = self
|
let build_container = self
|
||||||
.build(
|
.build(
|
||||||
|
@ -55,7 +55,7 @@ impl LeptosBuild {
|
|||||||
.client
|
.client
|
||||||
.cache_volume(format!("rust_leptos_{}", profile.to_string()));
|
.cache_volume(format!("rust_leptos_{}", profile.to_string()));
|
||||||
|
|
||||||
let mut build_options = vec!["cargo", "leptos", "build", "--release", "-vv"];
|
let build_options = vec!["cargo", "leptos", "build", "--release", "-vv"];
|
||||||
|
|
||||||
let rust_prebuild = rust_build_image
|
let rust_prebuild = rust_build_image
|
||||||
.with_workdir("/mnt/src")
|
.with_workdir("/mnt/src")
|
||||||
@ -95,7 +95,7 @@ impl LeptosBuild {
|
|||||||
for container_image in images {
|
for container_image in images {
|
||||||
let container = match &container_image {
|
let container = match &container_image {
|
||||||
SlimImage::Debian { image, deps, .. } => {
|
SlimImage::Debian { image, deps, .. } => {
|
||||||
let target = BuildTarget::from_target(&container_image);
|
let _target = BuildTarget::from_target(&container_image);
|
||||||
|
|
||||||
let build_container = self
|
let build_container = self
|
||||||
.build(
|
.build(
|
||||||
|
@ -1,6 +1,5 @@
|
|||||||
use std::{
|
use std::{
|
||||||
path::{Path, PathBuf},
|
path::{Path, PathBuf},
|
||||||
sync::Arc,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
use eyre::Context;
|
use eyre::Context;
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
use std::{path::PathBuf, sync::Arc};
|
use std::{path::PathBuf};
|
||||||
|
|
||||||
use crate::{build::RustVersion, source::RustSource};
|
use crate::{build::RustVersion, source::RustSource};
|
||||||
|
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
use dagger_rust::build::{BuildProfile, RustVersion, SlimImage};
|
use dagger_rust::build::{RustVersion, SlimImage};
|
||||||
|
|
||||||
#[tokio::main]
|
#[tokio::main]
|
||||||
pub async fn main() -> eyre::Result<()> {
|
pub async fn main() -> eyre::Result<()> {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user