#[cfg(test)]
mod tests {
use super::super::*;
use std::path::PathBuf;
use tempfile::TempDir;
fn create_test_cargo_toml() -> types::CargoToml {
types::CargoToml {
package: Some(types::CargoPackage {
name: "test-package".to_string(),
version: "1.0.0".to_string(),
description: Some("Test description".to_string()),
authors: Some(vec!["Test Author <test@example.com>".to_string()]),
keywords: Some(vec!["test".to_string(), "packaging".to_string()]),
rust_version: None,
}),
lib: Some(types::CargoLib {
crate_type: Some(vec!["cdylib".to_string()]),
}),
dependencies: None,
}
}
fn create_mock_library_file() -> (TempDir, PathBuf) {
let temp_dir = TempDir::new().expect("Failed to create temp dir");
let lib_path = temp_dir.path().join("libtest.so");
std::fs::write(&lib_path, b"mock library content").expect("Failed to write mock library");
(temp_dir, lib_path)
}
fn create_test_project() -> (TempDir, PathBuf) {
let temp_dir = TempDir::new().expect("Failed to create temp dir");
let project_path = temp_dir.path().to_path_buf();
let src_dir = project_path.join("src");
std::fs::create_dir_all(&src_dir).expect("Failed to create src dir");
let lib_rs_content = r#"
use cloacina::packaged_workflow;
#[packaged_workflow(package = "test-package")]
pub fn simple_task() -> Result<serde_json::Value, Box<dyn std::error::Error>> {
Ok(serde_json::json!({"status": "complete"}))
}
#[packaged_workflow(package = "test-package")]
pub fn complex_task() -> Result<serde_json::Value, Box<dyn std::error::Error>> {
Ok(serde_json::json!({"data": "processed"}))
}
"#;
std::fs::write(src_dir.join("lib.rs"), lib_rs_content).expect("Failed to write lib.rs");
(temp_dir, project_path)
}
#[test]
fn test_generate_manifest_basic() {
let cargo_toml = create_test_cargo_toml();
let (_temp_dir, lib_path) = create_mock_library_file();
let (_project_temp, project_path) = create_test_project();
let result = manifest::generate_manifest(&cargo_toml, &lib_path, &None, &project_path);
match result {
Ok(manifest) => {
assert_eq!(manifest.package.name, "test-package");
assert_eq!(manifest.package.version, "1.0.0");
assert!(manifest.package.description.contains("Packaged workflow"));
assert!(!manifest.library.filename.is_empty());
assert_eq!(manifest.library.symbols, vec!["cloacina_execute_task"]);
}
Err(e) => {
let error_msg = format!("{}", e);
assert!(
error_msg.contains("Failed to load library")
|| error_msg.contains("metadata")
|| error_msg.contains("symbol"),
"Error should be FFI-related: {}",
error_msg
);
}
}
}
#[test]
fn test_generate_manifest_with_target() {
let cargo_toml = create_test_cargo_toml();
let (_temp_dir, lib_path) = create_mock_library_file();
let (_project_temp, project_path) = create_test_project();
let target = Some("x86_64-unknown-linux-gnu".to_string());
let result = manifest::generate_manifest(&cargo_toml, &lib_path, &target, &project_path);
match result {
Ok(manifest) => {
assert_eq!(manifest.library.architecture, "x86_64-unknown-linux-gnu");
}
Err(_) => {
}
}
}
#[test]
fn test_generate_manifest_missing_package() {
let mut cargo_toml = create_test_cargo_toml();
cargo_toml.package = None;
let (_temp_dir, lib_path) = create_mock_library_file();
let (_project_temp, project_path) = create_test_project();
let result = manifest::generate_manifest(&cargo_toml, &lib_path, &None, &project_path);
assert!(result.is_err());
let error_msg = format!("{}", result.unwrap_err());
assert!(error_msg.contains("Missing package section"));
}
#[test]
fn test_extract_package_names_from_source() {
let (_temp_dir, project_path) = create_test_project();
let result = manifest::extract_package_names_from_source(&project_path);
match result {
Ok(package_names) => {
assert!(!package_names.is_empty());
assert!(package_names.contains(&"test-package".to_string()));
}
Err(e) => {
panic!("Should be able to extract package names: {}", e);
}
}
}
#[test]
fn test_extract_package_names_no_packages() {
let temp_dir = TempDir::new().expect("Failed to create temp dir");
let project_path = temp_dir.path().to_path_buf();
let src_dir = project_path.join("src");
std::fs::create_dir_all(&src_dir).expect("Failed to create src dir");
let lib_rs_content = r#"
pub fn regular_function() -> String {
"not a packaged workflow".to_string()
}
"#;
std::fs::write(src_dir.join("lib.rs"), lib_rs_content).expect("Failed to write lib.rs");
let result = manifest::extract_package_names_from_source(&project_path);
match result {
Ok(package_names) => {
assert!(package_names.is_empty());
}
Err(e) => {
panic!("Should handle no packages gracefully: {}", e);
}
}
}
#[test]
fn test_extract_package_names_missing_src() {
let temp_dir = TempDir::new().expect("Failed to create temp dir");
let project_path = temp_dir.path().to_path_buf();
let result = manifest::extract_package_names_from_source(&project_path);
assert!(result.is_err());
let error_msg = format!("{}", result.unwrap_err());
assert!(error_msg.contains("Failed to read src directory"));
}
#[test]
fn test_get_current_architecture() {
let arch = manifest::get_current_architecture();
assert!(!arch.is_empty());
assert!(
arch == "x86_64"
|| arch == "aarch64"
|| arch == "arm"
|| arch.starts_with("x86")
|| arch.starts_with("aarch")
|| arch.starts_with("arm")
);
}
#[test]
fn test_compile_options_builder_pattern() {
let options = CompileOptions {
target: Some("aarch64-apple-darwin".to_string()),
profile: "release".to_string(),
cargo_flags: vec!["--features".to_string(), "postgres".to_string()],
jobs: Some(8),
};
assert_eq!(options.target.as_ref().unwrap(), "aarch64-apple-darwin");
assert_eq!(options.profile, "release");
assert_eq!(options.cargo_flags.len(), 2);
assert_eq!(options.jobs.unwrap(), 8);
}
#[test]
fn test_package_info_creation() {
let package_info = types::PackageInfo {
name: "test-workflow".to_string(),
version: "2.1.0".to_string(),
description: "Test workflow package".to_string(),
author: Some("Test Author".to_string()),
workflow_fingerprint: Some("abc123def456".to_string()),
cloacina_version: "0.2.0".to_string(),
};
assert_eq!(package_info.name, "test-workflow");
assert_eq!(package_info.version, "2.1.0");
assert!(!package_info.description.is_empty());
assert!(!package_info.cloacina_version.is_empty());
}
#[test]
fn test_library_info_creation() {
let library_info = types::LibraryInfo {
filename: "libworkflow.dylib".to_string(),
symbols: vec![
"cloacina_execute_task".to_string(),
"cloacina_get_task_metadata".to_string(),
],
architecture: "aarch64-apple-darwin".to_string(),
};
assert!(library_info.filename.ends_with(".dylib"));
assert_eq!(library_info.symbols.len(), 2);
assert!(library_info
.symbols
.contains(&"cloacina_execute_task".to_string()));
assert!(!library_info.architecture.is_empty());
}
#[test]
fn test_task_info_creation() {
let task_info = types::TaskInfo {
index: 0,
id: "process_data".to_string(),
dependencies: vec!["validate_input".to_string()],
description: "Process the input data".to_string(),
source_location: "src/tasks.rs:42".to_string(),
};
assert_eq!(task_info.index, 0);
assert_eq!(task_info.id, "process_data");
assert_eq!(task_info.dependencies.len(), 1);
assert!(!task_info.description.is_empty());
assert!(task_info.source_location.contains("src/"));
}
#[test]
fn test_package_manifest_serialization_roundtrip() {
let original_manifest = types::PackageManifest {
package: types::PackageInfo {
name: "test-package".to_string(),
version: "1.0.0".to_string(),
description: "Test package".to_string(),
author: None,
workflow_fingerprint: None,
cloacina_version: "0.2.0".to_string(),
},
library: types::LibraryInfo {
filename: "libtest.so".to_string(),
symbols: vec!["cloacina_execute_task".to_string()],
architecture: "x86_64".to_string(),
},
tasks: vec![
types::TaskInfo {
index: 0,
id: "task1".to_string(),
dependencies: vec![],
description: "First task".to_string(),
source_location: "src/lib.rs:10".to_string(),
},
types::TaskInfo {
index: 1,
id: "task2".to_string(),
dependencies: vec!["task1".to_string()],
description: "Second task".to_string(),
source_location: "src/lib.rs:20".to_string(),
},
],
graph: None,
};
let json = serde_json::to_string(&original_manifest).expect("Should serialize");
assert!(!json.is_empty());
let deserialized: types::PackageManifest =
serde_json::from_str(&json).expect("Should deserialize");
assert_eq!(deserialized.package.name, original_manifest.package.name);
assert_eq!(
deserialized.package.version,
original_manifest.package.version
);
assert_eq!(
deserialized.library.filename,
original_manifest.library.filename
);
assert_eq!(deserialized.tasks.len(), original_manifest.tasks.len());
assert_eq!(deserialized.tasks[0].id, original_manifest.tasks[0].id);
assert_eq!(
deserialized.tasks[1].dependencies,
original_manifest.tasks[1].dependencies
);
}
#[test]
fn test_constants() {
assert_eq!(types::MANIFEST_FILENAME, "manifest.json");
assert_eq!(types::EXECUTE_TASK_SYMBOL, "cloacina_execute_task");
assert!(!types::CLOACINA_VERSION.is_empty());
let version_parts: Vec<&str> = types::CLOACINA_VERSION.split('.').collect();
assert!(
version_parts.len() >= 2,
"Version should have at least major.minor"
);
for part in version_parts.iter().take(2) {
assert!(
part.parse::<u32>().is_ok(),
"Version parts should be numeric: {}",
part
);
}
}
#[test]
fn test_safe_cstr_to_string_null_pointer() {
use super::super::manifest::{safe_cstr_to_string, ManifestError};
use std::ptr;
let result = safe_cstr_to_string(ptr::null(), "test_field");
assert!(result.is_err());
match result.unwrap_err() {
ManifestError::NullString { field } => {
assert_eq!(field, "test_field");
}
_ => panic!("Expected NullString error"),
}
}
#[test]
fn test_safe_cstr_to_string_valid() {
use super::super::manifest::safe_cstr_to_string;
use std::ffi::CString;
let c_string = CString::new("hello world").unwrap();
let result = safe_cstr_to_string(c_string.as_ptr(), "test_field");
assert!(result.is_ok());
assert_eq!(result.unwrap(), "hello world");
}
#[test]
fn test_safe_cstr_to_option_string_null_returns_none() {
use super::super::manifest::safe_cstr_to_option_string;
use std::ptr;
let result = safe_cstr_to_option_string(ptr::null(), "test_field");
assert!(result.is_ok());
assert!(result.unwrap().is_none());
}
#[test]
fn test_safe_cstr_to_option_string_valid() {
use super::super::manifest::safe_cstr_to_option_string;
use std::ffi::CString;
let c_string = CString::new("optional value").unwrap();
let result = safe_cstr_to_option_string(c_string.as_ptr(), "test_field");
assert!(result.is_ok());
assert_eq!(result.unwrap(), Some("optional value".to_string()));
}
#[test]
fn test_validate_ptr_null_pointer() {
use super::super::manifest::{validate_ptr, ManifestError};
use std::ptr;
let result: Result<&u32, ManifestError> = unsafe { validate_ptr(ptr::null(), "test_ptr") };
assert!(result.is_err());
match result.unwrap_err() {
ManifestError::NullPointer { field } => {
assert_eq!(field, "test_ptr");
}
_ => panic!("Expected NullPointer error"),
}
}
#[test]
fn test_validate_ptr_valid() {
use super::super::manifest::validate_ptr;
let value: u32 = 42;
let result = unsafe { validate_ptr(&value as *const u32, "test_ptr") };
assert!(result.is_ok());
assert_eq!(*result.unwrap(), 42);
}
#[test]
fn test_validate_slice_null_with_nonzero_count() {
use super::super::manifest::{validate_slice, ManifestError};
use std::ptr;
let result: Result<&[u32], ManifestError> =
unsafe { validate_slice(ptr::null(), 5, "test_slice") };
assert!(result.is_err());
match result.unwrap_err() {
ManifestError::NullTaskSlice { count } => {
assert_eq!(count, 5);
}
_ => panic!("Expected NullTaskSlice error"),
}
}
#[test]
fn test_validate_slice_null_with_zero_count() {
use super::super::manifest::validate_slice;
use std::ptr;
let result: Result<&[u32], _> = unsafe { validate_slice(ptr::null(), 0, "test_slice") };
assert!(result.is_ok());
assert!(result.unwrap().is_empty());
}
#[test]
fn test_validate_slice_exceeds_max_tasks() {
use super::super::manifest::{validate_slice, ManifestError};
let value: u32 = 42;
let result: Result<&[u32], ManifestError> =
unsafe { validate_slice(&value as *const u32, 10_001, "test_slice") };
assert!(result.is_err());
match result.unwrap_err() {
ManifestError::TooManyTasks { count, max } => {
assert_eq!(count, 10_001);
assert_eq!(max, 10_000);
}
_ => panic!("Expected TooManyTasks error"),
}
}
#[test]
fn test_validate_slice_valid() {
use super::super::manifest::validate_slice;
let values: [u32; 3] = [1, 2, 3];
let result = unsafe { validate_slice(values.as_ptr(), 3, "test_slice") };
assert!(result.is_ok());
let slice = result.unwrap();
assert_eq!(slice.len(), 3);
assert_eq!(slice[0], 1);
assert_eq!(slice[1], 2);
assert_eq!(slice[2], 3);
}
#[test]
fn test_manifest_error_display() {
use super::super::manifest::ManifestError;
let err = ManifestError::NullPointer {
field: "test_field",
};
assert!(err.to_string().contains("test_field"));
let err = ManifestError::NullString {
field: "string_field".to_string(),
};
assert!(err.to_string().contains("string_field"));
let err = ManifestError::NullTaskSlice { count: 42 };
assert!(err.to_string().contains("42"));
let err = ManifestError::TooManyTasks {
count: 20000,
max: 10000,
};
assert!(err.to_string().contains("20000"));
assert!(err.to_string().contains("10000"));
}
}