feat: with rust build and test
Signed-off-by: kjuulh <contact@kjuulh.io>
This commit is contained in:
193
crates/dagger-rust/src/source.rs
Normal file
193
crates/dagger-rust/src/source.rs
Normal file
@@ -0,0 +1,193 @@
|
||||
use std::{
|
||||
path::{Path, PathBuf},
|
||||
sync::Arc,
|
||||
};
|
||||
|
||||
use eyre::Context;
|
||||
|
||||
pub struct RustSource {
|
||||
client: Arc<dagger_sdk::Query>,
|
||||
|
||||
exclude: Vec<String>,
|
||||
}
|
||||
|
||||
impl RustSource {
|
||||
pub fn new(client: Arc<dagger_sdk::Query>) -> Self {
|
||||
Self {
|
||||
client,
|
||||
exclude: vec!["node_modules/", ".git/", "target/", ".cuddle/"]
|
||||
.into_iter()
|
||||
.map(|s| s.to_string())
|
||||
.collect(),
|
||||
}
|
||||
}
|
||||
|
||||
pub fn with_exclude(
|
||||
&mut self,
|
||||
exclude: impl IntoIterator<Item = impl Into<String>>,
|
||||
) -> &mut Self {
|
||||
self.exclude = exclude.into_iter().map(|s| s.into()).collect();
|
||||
|
||||
self
|
||||
}
|
||||
|
||||
pub fn append_exclude(
|
||||
&mut self,
|
||||
exclude: impl IntoIterator<Item = impl Into<String>>,
|
||||
) -> &mut Self {
|
||||
self.exclude
|
||||
.append(&mut exclude.into_iter().map(|s| s.into()).collect::<Vec<_>>());
|
||||
|
||||
self
|
||||
}
|
||||
|
||||
pub async fn get_rust_src<T, I>(
|
||||
&self,
|
||||
source: Option<T>,
|
||||
crate_paths: I,
|
||||
) -> eyre::Result<(dagger_sdk::Directory, dagger_sdk::Directory)>
|
||||
where
|
||||
T: Into<PathBuf>,
|
||||
T: Clone,
|
||||
I: IntoIterator,
|
||||
I::Item: Into<String>,
|
||||
{
|
||||
let source_path = match source.clone() {
|
||||
Some(s) => s.into(),
|
||||
None => PathBuf::from("."),
|
||||
};
|
||||
|
||||
let (skeleton_files, _crates) = self
|
||||
.get_rust_skeleton_files(&source_path, crate_paths)
|
||||
.await?;
|
||||
|
||||
let src = self.get_src(source.clone()).await?;
|
||||
let rust_src = self.get_rust_dep_src(source).await?;
|
||||
let rust_src = rust_src.with_directory(".", skeleton_files.id().await?);
|
||||
|
||||
Ok((src, rust_src))
|
||||
}
|
||||
|
||||
pub async fn get_src(
|
||||
&self,
|
||||
source: Option<impl Into<PathBuf>>,
|
||||
) -> eyre::Result<dagger_sdk::Directory> {
|
||||
let source = source.map(|s| s.into()).unwrap_or(PathBuf::from("."));
|
||||
|
||||
let directory = self.client.host().directory_opts(
|
||||
source.display().to_string(),
|
||||
dagger_sdk::HostDirectoryOptsBuilder::default()
|
||||
.exclude(self.exclude.iter().map(|s| s.as_str()).collect::<Vec<_>>())
|
||||
.build()?,
|
||||
);
|
||||
|
||||
Ok(directory)
|
||||
}
|
||||
|
||||
pub async fn get_rust_dep_src(
|
||||
&self,
|
||||
source: Option<impl Into<PathBuf>>,
|
||||
) -> eyre::Result<dagger_sdk::Directory> {
|
||||
let source = source.map(|s| s.into()).unwrap_or(PathBuf::from("."));
|
||||
|
||||
let directory = self.client.host().directory_opts(
|
||||
source.display().to_string(),
|
||||
dagger_sdk::HostDirectoryOptsBuilder::default()
|
||||
.include(vec!["**/Cargo.toml", "**/Cargo.lock"])
|
||||
.build()?,
|
||||
);
|
||||
|
||||
Ok(directory)
|
||||
}
|
||||
|
||||
pub async fn get_rust_target_src(
|
||||
&self,
|
||||
source_path: &Path,
|
||||
container: dagger_sdk::Container,
|
||||
crate_paths: impl IntoIterator<Item = impl Into<String>>,
|
||||
) -> eyre::Result<dagger_sdk::Directory> {
|
||||
let (_skeleton_files, crates) = self
|
||||
.get_rust_skeleton_files(source_path, crate_paths)
|
||||
.await?;
|
||||
|
||||
let exclude = crates
|
||||
.iter()
|
||||
.map(|c| format!("**/*{}*", c.replace('-', "_")))
|
||||
.collect::<Vec<_>>();
|
||||
|
||||
let exclude = exclude.iter().map(|c| c.as_str()).collect();
|
||||
|
||||
let incremental_dir = self.client.directory().with_directory_opts(
|
||||
".",
|
||||
container.directory("target").id().await?,
|
||||
dagger_sdk::DirectoryWithDirectoryOpts {
|
||||
exclude: Some(exclude),
|
||||
include: None,
|
||||
},
|
||||
);
|
||||
|
||||
return Ok(incremental_dir);
|
||||
}
|
||||
|
||||
pub async fn get_rust_skeleton_files(
|
||||
&self,
|
||||
source_path: &Path,
|
||||
crate_paths: impl IntoIterator<Item = impl Into<String>>,
|
||||
) -> eyre::Result<(dagger_sdk::Directory, Vec<String>)> {
|
||||
let paths = crate_paths
|
||||
.into_iter()
|
||||
.map(|s| s.into())
|
||||
.collect::<Vec<String>>();
|
||||
|
||||
let mut crates = Vec::new();
|
||||
for path in paths {
|
||||
if path.ends_with("/*") {
|
||||
let mut dirs = tokio::fs::read_dir(source_path.join(path.trim_end_matches("/*")))
|
||||
.await
|
||||
.context(format!("failed to find path: {}", path.clone()))?;
|
||||
while let Some(entry) = dirs.next_entry().await? {
|
||||
if entry.metadata().await?.is_dir() {
|
||||
crates.push(entry.path());
|
||||
}
|
||||
}
|
||||
} else {
|
||||
crates.push(PathBuf::from(path));
|
||||
}
|
||||
}
|
||||
|
||||
fn create_skeleton_files(
|
||||
directory: dagger_sdk::Directory,
|
||||
path: &Path,
|
||||
) -> eyre::Result<dagger_sdk::Directory> {
|
||||
let main_content = r#"
|
||||
#[allow(dead_code)]
|
||||
fn main() { panic!("should never be executed"); }"#;
|
||||
let lib_content = r#"
|
||||
#[allow(dead_code)]
|
||||
fn some() { panic!("should never be executed"); }"#;
|
||||
|
||||
let directory = directory.with_new_file(
|
||||
path.join("src").join("main.rs").display().to_string(),
|
||||
main_content,
|
||||
);
|
||||
let directory = directory.with_new_file(
|
||||
path.join("src").join("lib.rs").display().to_string(),
|
||||
lib_content,
|
||||
);
|
||||
|
||||
Ok(directory)
|
||||
}
|
||||
|
||||
let mut directory = self.client.directory();
|
||||
let mut crate_names = Vec::new();
|
||||
|
||||
for rust_crate in crates.iter() {
|
||||
if let Some(file_name) = rust_crate.file_name() {
|
||||
crate_names.push(file_name.to_str().unwrap().to_string());
|
||||
}
|
||||
directory = create_skeleton_files(directory, rust_crate.strip_prefix(source_path)?)?;
|
||||
}
|
||||
|
||||
Ok((directory, crate_names))
|
||||
}
|
||||
}
|
Reference in New Issue
Block a user