From b503ebc4d82e5b7f9fa6ecf7787e4f05e8ac4fc3 Mon Sep 17 00:00:00 2001 From: Tarek Date: Thu, 18 Jul 2024 19:46:50 +0300 Subject: [PATCH] test(fs-index): add macro for parameterized hash function tests - Implement `hash_tests!` macro to generate tests for function and hash type pairs. - Implement test functions parameterized by hash type to validate `ResourceIndex` operations. - Add various tests for both `Blake3` and `Crc32` hash algorithms. Signed-off-by: Tarek --- fs-index/src/tests.rs | 997 ++++++++++++++++++++---------------------- 1 file changed, 474 insertions(+), 523 deletions(-) diff --git a/fs-index/src/tests.rs b/fs-index/src/tests.rs index bf0fd7b4..8b001649 100644 --- a/fs-index/src/tests.rs +++ b/fs-index/src/tests.rs @@ -1,539 +1,490 @@ -/*! -# Test Structure for `fs-index` Crate - -This test suite is designed to validate the functionality of `fs-index` crate using different hash functions. -The crate is tested with both cryptographic and non-cryptographic hash functions -to ensure that the resource index works correctly regardless of the underlying hash function used. - -## Modules - -- `blake3`: This module contains tests using the `Blake3` hash function. -- `crc32`: This module contains tests using the `CRC32` hash function. -- `test_utils`: This module contains common helper macro to generate tests for different hash functions. - -*/ - -mod blake3 { - use dev_hash::Blake3; - - use crate::generate_tests_for_hash; - - generate_tests_for_hash!(Blake3); +//! This module provides tests for the `ResourceIndex` functionality using +//! different hash algorithms. +//! +//! The tests are parameterized by various hash types, such as `Blake3` and +//! `Crc32`, to ensure the implementation works consistently across different +//! hashing algorithms. +//! +//! # Structure +//! +//! - **Macros**: +//! - `hash_tests!`: Generates test functions for pairs of test functions and +//! hash types. +//! +//! - **Test Functions**: +//! - Defined to test various aspects of `ResourceIndex`, parameterized by +//! hash type. +//! +//! - **Helper Functions**: +//! - `get_indexed_resource_from_file`: Helper to create `IndexedResource` +//! from a file path. +//! +//! # Usage +//! +//! To add a new test for a specific hash type, add a new entry in the +//! `hash_tests!` macro invocation with the appropriate function and hash type. + +use dev_hash::{Blake3, Crc32}; +use std::{fs, path::Path}; + +use anyhow::{anyhow, Result}; +use tempfile::TempDir; + +use data_resource::ResourceId; + +use crate::{ + index::IndexedResource, utils::load_or_build_index, ResourceIndex, +}; + +/// A macro to generate tests for function and hash type pairs. +#[macro_export] +macro_rules! hash_tests { + ($($name:ident: ($func:ident, $hash_type:ty),)*) => { + $( + #[test] + fn $name() { + $func::<$hash_type>(); + } + )* + }; } -mod crc32 { - use dev_hash::Crc32; - - use crate::generate_tests_for_hash; - - generate_tests_for_hash!(Crc32); +// Use the macro to generate tests for the specified function and hash type +// pairs +hash_tests! { + // CRC32 + test_store_and_load_index_crc32: (test_store_and_load_index, Crc32), + test_store_and_load_index_with_collisions_crc32: (test_store_and_load_index_with_collisions, Crc32), + test_build_index_with_file_crc32: (test_build_index_with_file, Crc32), + test_build_index_with_empty_file_crc32: (test_build_index_with_empty_file, Crc32), + test_build_index_with_directory_crc32: (test_build_index_with_directory, Crc32), + test_build_index_with_multiple_files_crc32: (test_build_index_with_multiple_files, Crc32), + test_build_index_with_multiple_directories_crc32: (test_build_index_with_multiple_directories, Crc32), + test_resource_index_update_crc32: (test_resource_index_update, Crc32), + test_add_colliding_files_crc32: (test_add_colliding_files, Crc32), + test_num_collisions_crc32: (test_num_collisions, Crc32), + test_hidden_files_crc32: (test_hidden_files, Crc32), + + // Blake3 + test_store_and_load_index_blake3: (test_store_and_load_index, Blake3), + test_store_and_load_index_with_collisions_blake3: (test_store_and_load_index_with_collisions, Blake3), + test_build_index_with_file_blake3: (test_build_index_with_file, Blake3), + test_build_index_with_empty_file_blake3: (test_build_index_with_empty_file, Blake3), + test_build_index_with_directory_blake3: (test_build_index_with_directory, Blake3), + test_build_index_with_multiple_files_blake3: (test_build_index_with_multiple_files, Blake3), + test_build_index_with_multiple_directories_blake3: (test_build_index_with_multiple_directories, Blake3), + test_resource_index_update_blake3: (test_resource_index_update, Blake3), + test_add_colliding_files_blake3: (test_add_colliding_files, Blake3), + test_num_collisions_blake3: (test_num_collisions, Blake3), + test_hidden_files_blake3: (test_hidden_files, Blake3), } -mod test_utils { - /// A macro to generate tests for the resource index. - /// - /// This macro generates tests for a given hash type. The hash type must - /// implement the `ResourceId` trait. - #[macro_export] - macro_rules! generate_tests_for_hash { - ($hash_type:ty) => { - use std::{fs, path::Path}; - - use anyhow::{anyhow, Result}; - use tempfile::TempDir; - - use data_resource::ResourceId; - - use crate::{ - index::IndexedResource, utils::load_or_build_index, - ResourceIndex, - }; - - /// A helper function to get [`IndexedResource`] from a file path - fn get_indexed_resource_from_file>( - path: P, - parent_dir: P, - ) -> Result> { - let id = <$hash_type>::from_path(&path)?; - - let relative_path = path - .as_ref() - .strip_prefix(parent_dir) - .map_err(|_| anyhow!("Failed to get relative path"))?; - - Ok(IndexedResource::new( - id, - relative_path.to_path_buf(), - fs::metadata(path)?.modified()?, - )) - } +/// A helper function to get [`IndexedResource`] from a file path +fn get_indexed_resource_from_file>( + path: P, + parent_dir: P, +) -> Result> { + let id = H::from_path(&path)?; + + let relative_path = path + .as_ref() + .strip_prefix(parent_dir) + .map_err(|_| anyhow!("Failed to get relative path"))?; + + Ok(IndexedResource::new( + id, + relative_path.to_path_buf(), + fs::metadata(path)?.modified()?, + )) +} - /// Test storing and loading the resource index. - /// - /// ## Test scenario: - /// - Build a resource index in the temporary directory. - /// - Store the index. - /// - Load the stored index. - /// - Assert that the loaded index matches the original index. - #[test] - fn test_store_and_load_index() { - let temp_dir = - TempDir::with_prefix("ark_test_store_and_load_index") - .expect("Failed to create temp dir"); - let root_path = temp_dir.path(); - - let file_path = root_path.join("file.txt"); - fs::write(&file_path, "file content") - .expect("Failed to write to file"); - - let index: ResourceIndex<$hash_type> = - ResourceIndex::build(root_path) - .expect("Failed to build index"); - assert_eq!(index.len(), 1); - index.store().expect("Failed to store index"); - - let loaded_index = load_or_build_index(root_path, false) - .expect("Failed to load index"); - - assert_eq!(index, loaded_index); - } +/// Test storing and loading the resource index. +/// +/// ## Test scenario: +/// - Build a resource index in the temporary directory. +/// - Store the index. +/// - Load the stored index. +/// - Assert that the loaded index matches the original index. +fn test_store_and_load_index() { + let temp_dir = TempDir::with_prefix("ark_test_store_and_load_index") + .expect("Failed to create temp dir"); + let root_path = temp_dir.path(); + + let file_path = root_path.join("file.txt"); + fs::write(&file_path, "file content").expect("Failed to write to file"); + + let index: ResourceIndex = + ResourceIndex::build(root_path).expect("Failed to build index"); + assert_eq!(index.len(), 1); + index.store().expect("Failed to store index"); + + let loaded_index = + load_or_build_index(root_path, false).expect("Failed to load index"); + + assert_eq!(index, loaded_index); +} - /// Test storing and loading the resource index with collisions. - /// - /// ## Test scenario: - /// - Build a resource index in the temporary directory. - /// - Write duplicate files with the same content. - /// - Store the index. - /// - Load the stored index. - /// - Assert that the loaded index matches the original index. - #[test] - fn test_store_and_load_index_with_collisions() { - let temp_dir = TempDir::with_prefix( - "ark_test_store_and_load_index_with_collisions", - ) - .expect("Failed to create temp dir"); - let root_path = temp_dir.path(); - - let file_path = root_path.join("file.txt"); - fs::write(&file_path, "file content") - .expect("Failed to write to file"); - - let file_path2 = root_path.join("file2.txt"); - fs::write(&file_path2, "file content") - .expect("Failed to write to file"); - - let file_path3 = root_path.join("file3.txt"); - fs::write(&file_path3, "file content") - .expect("Failed to write to file"); - - let file_path4 = root_path.join("file4.txt"); - fs::write(&file_path4, "file content") - .expect("Failed to write to file"); - - // Now we have 4 files with the same content (same checksum) - - let index: ResourceIndex<$hash_type> = - ResourceIndex::build(root_path) - .expect("Failed to build index"); - let checksum = <$hash_type>::from_path(&file_path) - .expect("Failed to get checksum"); - assert_eq!(index.len(), 4); - assert_eq!(index.collisions().len(), 1); - assert_eq!(index.collisions()[&checksum].len(), 4); - index.store().expect("Failed to store index"); - - let loaded_index = load_or_build_index(root_path, false) - .expect("Failed to load index"); - - assert_eq!(index, loaded_index); - } +/// Test storing and loading the resource index with collisions. +/// +/// ## Test scenario: +/// - Build a resource index in the temporary directory. +/// - Write duplicate files with the same content. +/// - Store the index. +/// - Load the stored index. +/// - Assert that the loaded index matches the original index. +fn test_store_and_load_index_with_collisions() { + let temp_dir = + TempDir::with_prefix("ark_test_store_and_load_index_with_collisions") + .expect("Failed to create temp dir"); + let root_path = temp_dir.path(); + + let file_path = root_path.join("file.txt"); + fs::write(&file_path, "file content").expect("Failed to write to file"); + + let file_path2 = root_path.join("file2.txt"); + fs::write(&file_path2, "file content").expect("Failed to write to file"); + + let file_path3 = root_path.join("file3.txt"); + fs::write(&file_path3, "file content").expect("Failed to write to file"); + + let file_path4 = root_path.join("file4.txt"); + fs::write(&file_path4, "file content").expect("Failed to write to file"); + + // Now we have 4 files with the same content (same checksum) + + let index: ResourceIndex = + ResourceIndex::build(root_path).expect("Failed to build index"); + let checksum = H::from_path(&file_path).expect("Failed to get checksum"); + assert_eq!(index.len(), 4); + assert_eq!(index.collisions().len(), 1); + assert_eq!(index.collisions()[&checksum].len(), 4); + index.store().expect("Failed to store index"); + + let loaded_index = + load_or_build_index(root_path, false).expect("Failed to load index"); + + assert_eq!(index, loaded_index); +} - /// Test building an index with a file. - /// - /// ## Test scenario: - /// - Create a file within the temporary directory. - /// - Build a resource index in the temporary directory. - /// - Assert that the index contains one entry. - /// - Assert that the resource retrieved by path matches the - /// expected resource. - /// - Assert that the resource retrieved by ID matches the expected - /// resource. - #[test] - fn test_build_index_with_file() { - let temp_dir = - TempDir::with_prefix("ark_test_build_index_with_file") - .expect("Failed to create temp dir"); - let root_path = temp_dir.path(); - - let file_path = root_path.join("file.txt"); - fs::write(&file_path, "file content") - .expect("Failed to write to file"); - let expected_resource = get_indexed_resource_from_file( - &file_path, - &root_path.to_path_buf(), - ) - .expect("Failed to get indexed resource"); - - let index = ResourceIndex::build(root_path) - .expect("Failed to build index"); - assert_eq!(index.len(), 1); - - let resource = index - .get_resource_by_path("file.txt") - .expect("Failed to get resource"); - assert_eq!(resource, &expected_resource); - } +/// Test building an index with a file. +/// +/// ## Test scenario: +/// - Create a file within the temporary directory. +/// - Build a resource index in the temporary directory. +/// - Assert that the index contains one entry. +/// - Assert that the resource retrieved by path matches the expected resource. +/// - Assert that the resource retrieved by ID matches the expected resource. +fn test_build_index_with_file() { + let temp_dir = TempDir::with_prefix("ark_test_build_index_with_file") + .expect("Failed to create temp dir"); + let root_path = temp_dir.path(); + + let file_path = root_path.join("file.txt"); + fs::write(&file_path, "file content").expect("Failed to write to file"); + let expected_resource: IndexedResource = + get_indexed_resource_from_file(&file_path, &root_path.to_path_buf()) + .expect("Failed to get indexed resource"); + + let index = ResourceIndex::build(root_path).expect("Failed to build index"); + assert_eq!(index.len(), 1); + + let resource = index + .get_resource_by_path("file.txt") + .expect("Failed to get resource"); + assert_eq!(resource, &expected_resource); +} - /// Test building an index with an empty file. - /// - /// ## Test scenario: - /// - Create an empty file within the temporary directory. - /// - Create a file with content within the temporary directory. - /// - Build a resource index in the temporary directory. - /// - Assert that the index contains one entries. - #[test] - fn test_build_index_with_empty_file() { - let temp_dir = TempDir::with_prefix( - "ark_test_build_index_with_empty_file", - ) - .expect("Failed to create temp dir"); - let root_path = temp_dir.path(); - - let empty_file_path = root_path.join("empty_file.txt"); - fs::write(&empty_file_path, "") - .expect("Failed to write to file"); - - let file_path = root_path.join("file.txt"); - fs::write(&file_path, "file content") - .expect("Failed to write to file"); - - let index: ResourceIndex<$hash_type> = - ResourceIndex::build(root_path) - .expect("Failed to build index"); - assert_eq!(index.len(), 1); - } +/// Test building an index with an empty file. +/// +/// ## Test scenario: +/// - Create an empty file within the temporary directory. +/// - Create a file with content within the temporary directory. +/// - Build a resource index in the temporary directory. +/// - Assert that the index contains one entries. +fn test_build_index_with_empty_file() { + let temp_dir = TempDir::with_prefix("ark_test_build_index_with_empty_file") + .expect("Failed to create temp dir"); + let root_path = temp_dir.path(); + + let empty_file_path = root_path.join("empty_file.txt"); + fs::write(&empty_file_path, "").expect("Failed to write to file"); + + let file_path = root_path.join("file.txt"); + fs::write(&file_path, "file content").expect("Failed to write to file"); + + let index: ResourceIndex = + ResourceIndex::build(root_path).expect("Failed to build index"); + assert_eq!(index.len(), 1); +} - /// Test building an index with a directory. - /// - /// ## Test scenario: - /// - Create a subdirectory within the temporary directory. - /// - Create a file within the subdirectory. - /// - Build a resource index in the temporary directory. - /// - Assert that the index contains one entry. - /// - Assert that the resource retrieved by path matches the - /// expected resource. - /// - Assert that the resource retrieved by ID matches the expected - /// resource. - #[test] - fn test_build_index_with_directory() { - let temp_dir = - TempDir::with_prefix("ark_test_build_index_with_directory") - .expect("Failed to create temp dir"); - let root_path = temp_dir.path(); - - let dir_path = root_path.join("dir"); - fs::create_dir(&dir_path).expect("Failed to create dir"); - let file_path = dir_path.join("file.txt"); - fs::write(&file_path, "file content") - .expect("Failed to write to file"); - let expected_resource = get_indexed_resource_from_file( - &file_path, - &root_path.to_path_buf(), - ) - .expect("Failed to get indexed resource"); - - let index = ResourceIndex::build(root_path) - .expect("Failed to build index"); - assert_eq!(index.len(), 1); - - let resource = index - .get_resource_by_path("dir/file.txt") - .expect("Failed to get resource"); - assert_eq!(resource, &expected_resource); - } +/// Test building an index with a directory. +/// +/// ## Test scenario: +/// - Create a subdirectory within the temporary directory. +/// - Create a file within the subdirectory. +/// - Build a resource index in the temporary directory. +/// - Assert that the index contains one entry. +/// - Assert that the resource retrieved by path matches the expected resource. +/// - Assert that the resource retrieved by ID matches the expected resource. +fn test_build_index_with_directory() { + let temp_dir = TempDir::with_prefix("ark_test_build_index_with_directory") + .expect("Failed to create temp dir"); + let root_path = temp_dir.path(); + + let dir_path = root_path.join("dir"); + fs::create_dir(&dir_path).expect("Failed to create dir"); + let file_path = dir_path.join("file.txt"); + fs::write(&file_path, "file content").expect("Failed to write to file"); + let expected_resource: IndexedResource = + get_indexed_resource_from_file(&file_path, &root_path.to_path_buf()) + .expect("Failed to get indexed resource"); + + let index = ResourceIndex::build(root_path).expect("Failed to build index"); + assert_eq!(index.len(), 1); + + let resource = index + .get_resource_by_path("dir/file.txt") + .expect("Failed to get resource"); + assert_eq!(resource, &expected_resource); +} - /// Test building an index with multiple files. - /// - /// ## Test scenario: - /// - Create multiple files within the temporary directory. - /// - Build a resource index in the temporary directory. - /// - Assert that the index contains two entries. - /// - Assert that the resource retrieved by path for each file - /// matches the expected resource. - #[test] - fn test_build_index_with_multiple_files() { - let temp_dir = TempDir::with_prefix( - "ark_test_build_index_with_multiple_files", - ) - .expect("Failed to create temp dir"); - let root_path = temp_dir.path(); - - let file1_path = root_path.join("file1.txt"); - fs::write(&file1_path, "file1 content") - .expect("Failed to write to file"); - let file2_path = root_path.join("file2.txt"); - fs::write(&file2_path, "file2 content") - .expect("Failed to write to file"); - - let expected_resource1 = get_indexed_resource_from_file( - &file1_path, - &root_path.to_path_buf(), - ) - .expect("Failed to get indexed resource"); - let expected_resource2 = get_indexed_resource_from_file( - &file2_path, - &root_path.to_path_buf(), - ) - .expect("Failed to get indexed resource"); - - let index = ResourceIndex::build(root_path) - .expect("Failed to build index"); - assert_eq!(index.len(), 2); - - let resource = index - .get_resource_by_path("file1.txt") - .expect("Failed to get resource"); - assert_eq!(resource, &expected_resource1); - - let resource = index - .get_resource_by_path("file2.txt") - .expect("Failed to get resource"); - assert_eq!(resource, &expected_resource2); - } +/// Test building an index with multiple files. +/// +/// ## Test scenario: +/// - Create multiple files within the temporary directory. +/// - Build a resource index in the temporary directory. +/// - Assert that the index contains two entries. +/// - Assert that the resource retrieved by path for each file matches the +/// expected resource. +fn test_build_index_with_multiple_files() { + let temp_dir = + TempDir::with_prefix("ark_test_build_index_with_multiple_files") + .expect("Failed to create temp dir"); + let root_path = temp_dir.path(); + + let file1_path = root_path.join("file1.txt"); + fs::write(&file1_path, "file1 content").expect("Failed to write to file"); + let file2_path = root_path.join("file2.txt"); + fs::write(&file2_path, "file2 content").expect("Failed to write to file"); + + let expected_resource1: IndexedResource = + get_indexed_resource_from_file(&file1_path, &root_path.to_path_buf()) + .expect("Failed to get indexed resource"); + let expected_resource2 = + get_indexed_resource_from_file(&file2_path, &root_path.to_path_buf()) + .expect("Failed to get indexed resource"); + + let index = ResourceIndex::build(root_path).expect("Failed to build index"); + assert_eq!(index.len(), 2); + + let resource = index + .get_resource_by_path("file1.txt") + .expect("Failed to get resource"); + assert_eq!(resource, &expected_resource1); + + let resource = index + .get_resource_by_path("file2.txt") + .expect("Failed to get resource"); + assert_eq!(resource, &expected_resource2); +} - /// Test building an index with multiple directories. - /// - /// ## Test scenario: - /// - Create multiple directories within the temporary directory, - /// each containing a file. - /// - Build a resource index in the temporary directory. - /// - Assert that the index contains two entries. - /// - Assert that the resources retrieved by path for each file - /// match the expected resources. - #[test] - fn test_build_index_with_multiple_directories() { - let temp_dir = TempDir::with_prefix( - "ark_test_build_index_with_multiple_directories", - ) - .expect("Failed to create temp dir"); - let root_path = temp_dir.path(); - - let dir1_path = root_path.join("dir1"); - fs::create_dir(&dir1_path).expect("Failed to create dir"); - let file1_path = dir1_path.join("file1.txt"); - fs::write(&file1_path, "file1 content") - .expect("Failed to write to file"); - - let dir2_path = root_path.join("dir2"); - fs::create_dir(&dir2_path).expect("Failed to create dir"); - let file2_path = dir2_path.join("file2.txt"); - fs::write(&file2_path, "file2 content") - .expect("Failed to write to file"); - - let expected_resource1 = get_indexed_resource_from_file( - &file1_path, - &root_path.to_path_buf(), - ) - .expect("Failed to get indexed resource"); - let expected_resource2 = get_indexed_resource_from_file( - &file2_path, - &root_path.to_path_buf(), - ) - .expect("Failed to get indexed resource"); - - let index = ResourceIndex::build(root_path) - .expect("Failed to build index"); - assert_eq!(index.len(), 2); - - let resource = index - .get_resource_by_path("dir1/file1.txt") - .expect("Resource not found"); - assert_eq!(resource, &expected_resource1); - - let resource = index - .get_resource_by_path("dir2/file2.txt") - .expect("Resource not found"); - assert_eq!(resource, &expected_resource2); - } +/// Test building an index with multiple directories. +/// +/// ## Test scenario: +/// - Create multiple directories within the temporary directory, each +/// containing a file. +/// - Build a resource index in the temporary directory. +/// - Assert that the index contains two entries. +/// - Assert that the resources retrieved by path for each file match the +/// expected resources. +fn test_build_index_with_multiple_directories() { + let temp_dir = + TempDir::with_prefix("ark_test_build_index_with_multiple_directories") + .expect("Failed to create temp dir"); + let root_path = temp_dir.path(); + + let dir1_path = root_path.join("dir1"); + fs::create_dir(&dir1_path).expect("Failed to create dir"); + let file1_path = dir1_path.join("file1.txt"); + fs::write(&file1_path, "file1 content").expect("Failed to write to file"); + + let dir2_path = root_path.join("dir2"); + fs::create_dir(&dir2_path).expect("Failed to create dir"); + let file2_path = dir2_path.join("file2.txt"); + fs::write(&file2_path, "file2 content").expect("Failed to write to file"); + + let expected_resource1: IndexedResource = + get_indexed_resource_from_file(&file1_path, &root_path.to_path_buf()) + .expect("Failed to get indexed resource"); + let expected_resource2 = + get_indexed_resource_from_file(&file2_path, &root_path.to_path_buf()) + .expect("Failed to get indexed resource"); + + let index = ResourceIndex::build(root_path).expect("Failed to build index"); + assert_eq!(index.len(), 2); + + let resource = index + .get_resource_by_path("dir1/file1.txt") + .expect("Resource not found"); + assert_eq!(resource, &expected_resource1); + + let resource = index + .get_resource_by_path("dir2/file2.txt") + .expect("Resource not found"); + assert_eq!(resource, &expected_resource2); +} - /// Test updating the resource index. - /// - /// ## Test scenario: - /// - Create files within the temporary directory. - /// - Build a resource index in the temporary directory. - /// - Assert that the index initially contains the expected number - /// of entries. - /// - Create a new file, modify an existing file, and remove another - /// file. - /// - Update the resource index. - /// - Assert that the index contains the expected number of entries - /// after the update. - /// - Assert that the entries in the index match the expected state - /// after the update. - #[test] - fn test_resource_index_update() { - let temp_dir = - TempDir::with_prefix("ark_test_resource_index_update") - .expect("Failed to create temp dir"); - let root_path = temp_dir.path(); - - let file_path = root_path.join("file.txt"); - fs::write(&file_path, "file content") - .expect("Failed to write to file"); - - let image_path = root_path.join("image.png"); - fs::write(&image_path, "image content") - .expect("Failed to write to file"); - - let mut index = ResourceIndex::build(root_path) - .expect("Failed to build index"); - index.store().expect("Failed to store index"); - assert_eq!(index.len(), 2); - - // create new file - let new_file_path = root_path.join("new_file.txt"); - fs::write(&new_file_path, "new file content") - .expect("Failed to write to file"); - - // modify file - fs::write(&file_path, "updated file content") - .expect("Failed to write to file"); - - // remove file - fs::remove_file(&image_path).expect("Failed to remove file"); - - index - .update_all() - .expect("Failed to update index"); - // Index now contains 2 resources (file.txt and new_file.txt) - assert_eq!(index.len(), 2); - - let resource = index - .get_resource_by_path("file.txt") - .expect("Resource not found"); - let expected_resource = get_indexed_resource_from_file( - &file_path, - &root_path.to_path_buf(), - ) - .expect("Failed to get indexed resource"); - assert_eq!(resource, &expected_resource); - - let _resource = index - .get_resource_by_path("new_file.txt") - .expect("Resource not found"); - - assert!(index.get_resource_by_path("image.png").is_none()); - } +/// Test updating the resource index. +/// +/// ## Test scenario: +/// - Create files within the temporary directory. +/// - Build a resource index in the temporary directory. +/// - Assert that the index initially contains the expected number of entries. +/// - Create a new file, modify an existing file, and remove another file. +/// - Update the resource index. +/// - Assert that the index contains the expected number of entries after the +/// update. +/// - Assert that the entries in the index match the expected state after the +/// update. +fn test_resource_index_update() { + let temp_dir = TempDir::with_prefix("ark_test_resource_index_update") + .expect("Failed to create temp dir"); + let root_path = temp_dir.path(); + + let file_path = root_path.join("file.txt"); + fs::write(&file_path, "file content").expect("Failed to write to file"); + + let image_path = root_path.join("image.png"); + fs::write(&image_path, "image content").expect("Failed to write to file"); + + let mut index: ResourceIndex = + ResourceIndex::build(root_path).expect("Failed to build index"); + index.store().expect("Failed to store index"); + assert_eq!(index.len(), 2); + + // create new file + let new_file_path = root_path.join("new_file.txt"); + fs::write(&new_file_path, "new file content") + .expect("Failed to write to file"); + + // modify file + fs::write(&file_path, "updated file content") + .expect("Failed to write to file"); + + // remove file + fs::remove_file(&image_path).expect("Failed to remove file"); + + index + .update_all() + .expect("Failed to update index"); + // Index now contains 2 resources (file.txt and new_file.txt) + assert_eq!(index.len(), 2); + + let resource = index + .get_resource_by_path("file.txt") + .expect("Resource not found"); + let expected_resource = + get_indexed_resource_from_file(&file_path, &root_path.to_path_buf()) + .expect("Failed to get indexed resource"); + assert_eq!(resource, &expected_resource); + + let _resource = index + .get_resource_by_path("new_file.txt") + .expect("Resource not found"); + + assert!(index.get_resource_by_path("image.png").is_none()); +} - /// Test adding colliding files to the index. - /// - /// ## Test scenario: - /// - Create a file within the temporary directory. - /// - Build a resource index in the temporary directory. - /// - Assert that the index initially contains the expected number - /// of entries. - /// - Create a new file with the same checksum as the existing file. - /// - Track the addition of the new file in the index. - /// - Assert that the index contains the expected number of entries - /// after the addition. - /// - Assert index.collisions contains the expected number of - /// entries. - #[test] - fn test_add_colliding_files() { - let temp_dir = - TempDir::with_prefix("ark_test_add_colliding_files") - .expect("Failed to create temp dir"); - let root_path = temp_dir.path(); - - let file_path = root_path.join("file.txt"); - fs::write(&file_path, "file content") - .expect("Failed to write to file"); - - let mut index: ResourceIndex<$hash_type> = - ResourceIndex::build(root_path) - .expect("Failed to build index"); - index.store().expect("Failed to store index"); - assert_eq!(index.len(), 1); - - let new_file_path = root_path.join("new_file.txt"); - fs::write(&new_file_path, "file content") - .expect("Failed to write to file"); - - index - .update_all() - .expect("Failed to update index"); - - assert_eq!(index.len(), 2); - assert_eq!(index.collisions().len(), 1); - } +/// Test adding colliding files to the index. +/// +/// ## Test scenario: +/// - Create a file within the temporary directory. +/// - Build a resource index in the temporary directory. +/// - Assert that the index initially contains the expected number of entries. +/// - Create a new file with the same checksum as the existing file. +/// - Track the addition of the new file in the index. +/// - Assert that the index contains the expected number of entries after the +/// addition. +/// - Assert index.collisions contains the expected number of entries. +fn test_add_colliding_files() { + let temp_dir = TempDir::with_prefix("ark_test_add_colliding_files") + .expect("Failed to create temp dir"); + let root_path = temp_dir.path(); + + let file_path = root_path.join("file.txt"); + fs::write(&file_path, "file content").expect("Failed to write to file"); + + let mut index: ResourceIndex = + ResourceIndex::build(root_path).expect("Failed to build index"); + index.store().expect("Failed to store index"); + assert_eq!(index.len(), 1); + + let new_file_path = root_path.join("new_file.txt"); + fs::write(&new_file_path, "file content").expect("Failed to write to file"); + + index + .update_all() + .expect("Failed to update index"); + + assert_eq!(index.len(), 2); + assert_eq!(index.collisions().len(), 1); +} - /// Test `ResourceIndex::num_collisions()` method. - /// - /// ## Test scenario: - /// - Create a file within the temporary directory. - /// - Build a resource index in the temporary directory. - /// - Assert that the index initially contains the expected number - /// of entries. - /// - Create 2 new files with the same checksum as the existing - /// file. - /// - Update the index. - /// - Assert that the index contains the expected number of entries - /// after the update. - #[test] - fn test_num_collisions() { - let temp_dir = TempDir::with_prefix("ark_test_num_collisions") - .expect("Failed to create temp dir"); - let root_path = temp_dir.path(); - - let file_path = root_path.join("file.txt"); - fs::write(&file_path, "file content") - .expect("Failed to write to file"); - - let mut index: ResourceIndex<$hash_type> = - ResourceIndex::build(root_path) - .expect("Failed to build index"); - index.store().expect("Failed to store index"); - assert_eq!(index.len(), 1); - - let new_file_path = root_path.join("new_file.txt"); - fs::write(&new_file_path, "file content") - .expect("Failed to write to file"); - - let new_file_path2 = root_path.join("new_file2.txt"); - fs::write(&new_file_path2, "file content") - .expect("Failed to write to file"); - - index - .update_all() - .expect("Failed to update index"); - - assert_eq!(index.len(), 3); - assert_eq!(index.num_collisions(), 3); - } +/// Test `ResourceIndex::num_collisions()` method. +/// +/// ## Test scenario: +/// - Create a file within the temporary directory. +/// - Build a resource index in the temporary directory. +/// - Assert that the index initially contains the expected number of entries. +/// - Create 2 new files with the same checksum as the existing file. +/// - Update the index. +/// - Assert that the index contains the expected number of entries after the +/// update. +fn test_num_collisions() { + let temp_dir = TempDir::with_prefix("ark_test_num_collisions") + .expect("Failed to create temp dir"); + let root_path = temp_dir.path(); + + let file_path = root_path.join("file.txt"); + fs::write(&file_path, "file content").expect("Failed to write to file"); + + let mut index: ResourceIndex = + ResourceIndex::build(root_path).expect("Failed to build index"); + index.store().expect("Failed to store index"); + assert_eq!(index.len(), 1); + + let new_file_path = root_path.join("new_file.txt"); + fs::write(&new_file_path, "file content").expect("Failed to write to file"); + + let new_file_path2 = root_path.join("new_file2.txt"); + fs::write(&new_file_path2, "file content") + .expect("Failed to write to file"); + + index + .update_all() + .expect("Failed to update index"); + + assert_eq!(index.len(), 3); + assert_eq!(index.num_collisions(), 3); +} - /// Test that we don't index hidden files. - /// - /// ## Test scenario: - /// - Create a hidden file within the temporary directory. - /// - Build a resource index in the temporary directory. - /// - Assert that the index initially contains the expected number - /// of entries. (0) - #[test] - fn test_hidden_files() { - let temp_dir = TempDir::with_prefix("ark_test_hidden_files") - .expect("Failed to create temp dir"); - let root_path = temp_dir.path(); - - let file_path = root_path.join(".hidden_file.txt"); - fs::write(&file_path, "file content") - .expect("Failed to write to file"); - - let index: ResourceIndex<$hash_type> = - ResourceIndex::build(root_path) - .expect("Failed to build index"); - index.store().expect("Failed to store index"); - assert_eq!(index.len(), 0); - } - }; - } +/// Test that we don't index hidden files. +/// +/// ## Test scenario: +/// - Create a hidden file within the temporary directory. +/// - Build a resource index in the temporary directory. +/// - Assert that the index initially contains the expected number of entries. +/// (0) +fn test_hidden_files() { + let temp_dir = TempDir::with_prefix("ark_test_hidden_files") + .expect("Failed to create temp dir"); + let root_path = temp_dir.path(); + + let file_path = root_path.join(".hidden_file.txt"); + fs::write(&file_path, "file content").expect("Failed to write to file"); + + let index: ResourceIndex = + ResourceIndex::build(root_path).expect("Failed to build index"); + index.store().expect("Failed to store index"); + assert_eq!(index.len(), 0); }