diff --git a/scarb/Cargo.toml b/scarb/Cargo.toml index d1d093ab7..dc41d39a4 100644 --- a/scarb/Cargo.toml +++ b/scarb/Cargo.toml @@ -89,6 +89,7 @@ windows-sys.workspace = true zstd.workspace = true cargo_metadata.workspace = true flate2.workspace = true +fs_extra.workspace = true [target.'cfg(not(target_os = "linux"))'.dependencies] reqwest = { workspace = true, default-features = true } diff --git a/scarb/src/core/manifest/mod.rs b/scarb/src/core/manifest/mod.rs index bdeb0c175..ef41db954 100644 --- a/scarb/src/core/manifest/mod.rs +++ b/scarb/src/core/manifest/mod.rs @@ -75,6 +75,7 @@ pub struct ManifestMetadata { pub license_file: Option, pub readme: Option, pub repository: Option, + pub include: Option>, #[serde(rename = "tool")] pub tool_metadata: Option>, pub cairo_version: Option, diff --git a/scarb/src/core/manifest/toml_manifest.rs b/scarb/src/core/manifest/toml_manifest.rs index 64485e847..8f95ca26f 100644 --- a/scarb/src/core/manifest/toml_manifest.rs +++ b/scarb/src/core/manifest/toml_manifest.rs @@ -119,6 +119,7 @@ pub struct PackageInheritableFields { pub license_file: Option, pub readme: Option, pub repository: Option, + pub include: Option>, pub cairo_version: Option, } @@ -147,6 +148,7 @@ impl PackageInheritableFields { get_field!(license, String); get_field!(license_file, Utf8PathBuf); get_field!(repository, String); + get_field!(include, VecOfStrings); get_field!(edition, Edition); pub fn readme(&self, workspace_root: &Utf8Path, package_root: &Utf8Path) -> Result { @@ -197,6 +199,7 @@ pub struct TomlPackage { pub license_file: Option>, pub readme: Option>, pub repository: Option>, + pub include: Option>>, /// **UNSTABLE** This package does not depend on Cairo's `core`. pub no_core: Option, pub cairo_version: Option>, @@ -571,6 +574,11 @@ impl TomlManifest { .clone() .map(|mw| mw.resolve("repository", || inheritable_package.repository())) .transpose()?, + include: package + .include + .clone() + .map(|mw| mw.resolve("include", || inheritable_package.include())) + .transpose()?, cairo_version: package .cairo_version .clone() diff --git a/scarb/src/core/publishing/manifest_normalization.rs b/scarb/src/core/publishing/manifest_normalization.rs index 14e41e0ac..45e04e555 100644 --- a/scarb/src/core/publishing/manifest_normalization.rs +++ b/scarb/src/core/publishing/manifest_normalization.rs @@ -73,6 +73,7 @@ fn generate_package(pkg: &Package) -> Box { .clone() .map(|_| MaybeWorkspace::Defined((Utf8PathBuf::from(DEFAULT_README_FILE_NAME)).into())), repository: metadata.repository.clone().map(MaybeWorkspace::Defined), + include: metadata.include.clone().map(MaybeWorkspace::Defined), no_core: summary.no_core.then_some(true), cairo_version: metadata.cairo_version.clone().map(MaybeWorkspace::Defined), experimental_features: pkg.manifest.experimental_features.clone(), diff --git a/scarb/src/lib.rs b/scarb/src/lib.rs index 673632428..4285498db 100644 --- a/scarb/src/lib.rs +++ b/scarb/src/lib.rs @@ -42,3 +42,5 @@ pub const TEST_ASSERTS_PLUGIN_NAME: &str = "assert_macros"; pub const CAIRO_RUN_PLUGIN_NAME: &str = "cairo_run"; pub const CARGO_MANIFEST_FILE_NAME: &str = "Cargo.toml"; pub const CARGO_LOCK_FILE_NAME: &str = "Cargo.lock"; +pub static PREBUILT_LIBRARY_TARGET_DIRECTORY: LazyLock = + LazyLock::new(|| ["target", "scarb", "cairo-plugin"].iter().collect()); diff --git a/scarb/src/ops/package.rs b/scarb/src/ops/package.rs index c01ceda23..2cf102b30 100644 --- a/scarb/src/ops/package.rs +++ b/scarb/src/ops/package.rs @@ -1,10 +1,14 @@ +use anyhow::{anyhow, bail, ensure, Context, Result}; +use camino::{Utf8Path, Utf8PathBuf}; +use core::str; +use fs::copy; +use fs_extra::dir::{copy as fs_extra_copy, CopyOptions}; +use fs_extra::error::{Error as FsExtraError, ErrorKind as FsExtraErrorKind}; +use indoc::{formatdoc, indoc, writedoc}; use std::collections::BTreeMap; -use std::fs::File; +use std::fs::{self, File}; use std::io::{Seek, SeekFrom, Write}; - -use anyhow::{bail, ensure, Context, Result}; -use camino::Utf8PathBuf; -use indoc::{formatdoc, indoc, writedoc}; +use std::path::Path; use crate::core::registry::package_source_store::PackageSourceStore; use crate::sources::client::PackageRepository; @@ -23,9 +27,11 @@ use crate::flock::{FileLockGuard, Filesystem}; use crate::internal::restricted_names; use crate::{ ops, CARGO_MANIFEST_FILE_NAME, DEFAULT_LICENSE_FILE_NAME, DEFAULT_README_FILE_NAME, - MANIFEST_FILE_NAME, VCS_INFO_FILE_NAME, + MANIFEST_FILE_NAME, PREBUILT_LIBRARY_TARGET_DIRECTORY, VCS_INFO_FILE_NAME, }; +use super::execute_script; + const VERSION: u8 = 1; const VERSION_FILE_NAME: &str = "VERSION"; const ORIGINAL_MANIFEST_FILE_NAME: &str = "Scarb.orig.toml"; @@ -157,6 +163,7 @@ fn package_one_impl( ws: &Workspace<'_>, ) -> Result { let pkg = ws.fetch_package(&pkg_id)?; + let target_dir = ws.target_dir().child("package"); ws.config() .ui() @@ -166,14 +173,44 @@ fn package_one_impl( check_metadata(pkg, ws.config())?; } - let recipe = prepare_archive_recipe(pkg, opts, ws)?; + let mut recipe: Vec = Vec::default(); + + if let Some(script_definition) = pkg.manifest.scripts.get("package") { + let target_dir_path = target_dir.path_existent()?; + ws.config().ui().print(Status::new( + "Running package script with package", + &pkg_id.to_string(), + )); + + if let Some(includes) = pkg.manifest.metadata.include.clone() { + copy_items_to_target_dir(includes, target_dir_path)?; + } + + execute_script(script_definition, &[], ws, target_dir_path, None)?; + + let built_macros_target_dir = ws.target_dir().child("scarb").child("cairo-plugin"); + + if built_macros_target_dir.exists() { + let dynamic_library_files = + find_dynamic_library_files(built_macros_target_dir.path_unchecked()); + for dynamic_library_file in dynamic_library_files.into_iter() { + recipe.push(ArchiveFile { + path: PREBUILT_LIBRARY_TARGET_DIRECTORY + .as_path() + .join(dynamic_library_file.file_name().unwrap()), + contents: ArchiveFileContents::OnDisk(dynamic_library_file), + }) + } + } + } + + recipe.extend(prepare_archive_recipe(pkg, opts, ws)?); let num_files = recipe.len(); // Package up and test a temporary tarball and only move it to the final location if it actually // passes all verification checks. Any previously existing tarball can be assumed as corrupt // or invalid, so we can overwrite it if it exists. let filename = pkg_id.tarball_name(); - let target_dir = ws.target_dir().child("package"); let mut dst = target_dir.create_rw(format!(".{filename}"), "package scratch space", ws.config())?; @@ -606,3 +643,47 @@ fn check_metadata(pkg: &Package, config: &Config) -> Result<()> { Ok(()) } + +fn copy_items_to_target_dir(paths: Vec, target_dir: &Utf8Path) -> Result<()> { + let target_path = Path::new(target_dir); + let options = CopyOptions::new().copy_inside(true).overwrite(true); + + for path_str in paths { + let source_path = Path::new(&path_str); + if source_path.exists() { + if source_path.is_dir() { + fs_extra_copy(source_path, target_path, &options)?; + } else { + let file_name = source_path.file_name().ok_or(FsExtraError::new( + fs_extra::error::ErrorKind::Other, + "failed to extract file name", + ))?; + let target_file_path = target_path.join(file_name); + copy(source_path, target_file_path).map_err(|err| { + FsExtraError::new(FsExtraErrorKind::Io(err), "failed to copy file") + })?; + } + } else { + return Err(anyhow!("path does not exist: {}", source_path.display())); + } + } + + Ok(()) +} + +fn find_dynamic_library_files(directory: &Utf8Path) -> Vec { + let mut files = Vec::new(); + if let Ok(entries) = fs::read_dir(directory) { + for entry in entries.filter_map(Result::ok) { + let path = entry.path(); + if let Ok(utf8_path) = Utf8PathBuf::from_path_buf(path) { + if let Some(extension) = utf8_path.extension() { + if extension == "dll" || extension == "so" || extension == "dylib" { + files.push(utf8_path); + } + } + } + } + } + files +} diff --git a/scarb/tests/package.rs b/scarb/tests/package.rs index 00d3cb279..e30315425 100644 --- a/scarb/tests/package.rs +++ b/scarb/tests/package.rs @@ -48,10 +48,11 @@ impl PackageChecker { .path() .expect("failed to get archive entry path") .into_owned(); - let mut contents = String::new(); - entry - .read_to_string(&mut contents) - .expect("failed to read archive entry contents"); + let mut contents = String::default(); + // Some files are not valid utf-8 contents, so we don't want to read them at all. + // We just want to track that they even exist. + let _ = entry.read_to_string(&mut contents); + (name, contents) }) .collect(); @@ -1525,3 +1526,105 @@ fn package_with_publish_disabled() { [..]Packaged [..] files, [..] ([..] compressed) "#}); } + +#[test] +fn package_with_package_script() { + let t = TempDir::new().unwrap(); + + #[cfg(not(windows))] + let script_code = indoc! { r#"cargo build --release && mkdir -p ../scarb/cairo-plugin && cp target/release/libfoo.so ../scarb/cairo-plugin"#}; + + #[cfg(windows)] + let script_code = indoc! { r#"cargo build --release && mkdir -p ../scarb/cairo-plugin && cp target/release/libfoo.dll ../scarb/cairo-plugin"#}; + + CairoPluginProjectBuilder::start() + .name("foo") + .scarb_project(|b| { + b.name("foo") + .version("1.0.0") + .manifest_package_extra(formatdoc! {r#" + include = ["Cargo.lock", "Cargo.toml", "src"] + "#}) + .manifest_extra(formatdoc! {r#" + [cairo-plugin] + + [scripts] + package = "{script_code}" + "#}) + }) + .lib_rs(indoc! {r#" + use cairo_lang_macro::{ProcMacroResult, TokenStream, attribute_macro}; + + #[attribute_macro] + pub fn some(_attr: TokenStream, token_stream: TokenStream) -> ProcMacroResult { + ProcMacroResult::new(token_stream) + } + "#}) + .build(&t); + + Scarb::quick_snapbox() + .current_dir(&t) + .arg("package") + .assert() + .success(); + + #[cfg(not(windows))] + let expected_shared_lib_file = r#"target/scarb/cairo-plugin/libfoo.so"#; + + #[cfg(windows)] + let expected_shared_lib_file = r#"target/scarb/cairo-plugin/libfoo.dll"#; + + PackageChecker::assert(&t.child("target/package/foo-1.0.0.tar.zst")) + .name_and_version("foo", "1.0.0") + .contents(&[ + "VERSION", + "Scarb.orig.toml", + "Scarb.toml", + expected_shared_lib_file, + "Cargo.toml", + "Cargo.orig.toml", + "src/lib.rs", + ]); +} + +#[test] +fn package_with_package_script_not_existing_script() { + let t = TempDir::new().unwrap(); + + #[cfg(not(windows))] + let script_name = "script.sh"; + + #[cfg(windows)] + let script_name = "script.bat"; + + CairoPluginProjectBuilder::start() + .name("foo") + .scarb_project(|b| { + b.name("foo") + .version("1.0.0") + .manifest_package_extra(indoc! {r#" + include = ["Cargo.lock", "Cargo.toml", "src", "script.sh"] + "#}) + .manifest_extra(formatdoc! {r#" + [cairo-plugin] + + [scripts] + package = "{script_name}" + "#}) + }) + .lib_rs(indoc! {r#" + use cairo_lang_macro::{ProcMacroResult, TokenStream, attribute_macro}; + + #[attribute_macro] + pub fn some(_attr: TokenStream, token_stream: TokenStream) -> ProcMacroResult { + ProcMacroResult::new(token_stream) + } + "#}) + .build(&t); + + Scarb::quick_snapbox() + .current_dir(&t) + .arg("package") + .assert() + .failure(); +}