absurder-sql 0.1.23

AbsurderSQL - SQLite + IndexedDB that's absurdly better than absurd-sql
Documentation
//! VFS Durability Mapping Tests
//!
//! Tests that SQLite VFS xSync properly maps to force_sync() with durability guarantees.
//! Ensures that data is actually persisted to IndexedDB when SQLite calls xSync.

#[cfg(target_arch = "wasm32")]
use wasm_bindgen_test::*;

#[cfg(target_arch = "wasm32")]
use absurder_sql::storage::BlockStorage;

#[cfg(target_arch = "wasm32")]
wasm_bindgen_test_configure!(run_in_browser);

/// Test that force_sync() actually persists data to IndexedDB
#[cfg(target_arch = "wasm32")]
#[wasm_bindgen_test]
async fn test_force_sync_persists_to_indexeddb() {
    let mut storage = BlockStorage::new("force_sync_test")
        .await
        .expect("create storage");

    // Write some blocks
    let block1 = storage.allocate_block().await.expect("allocate block1");
    let block2 = storage.allocate_block().await.expect("allocate block2");

    storage
        .write_block(block1, vec![1u8; 4096])
        .await
        .expect("write block1");
    storage
        .write_block(block2, vec![2u8; 4096])
        .await
        .expect("write block2");

    // Call force_sync() to ensure durability
    storage
        .force_sync()
        .await
        .expect("force_sync should succeed");

    // Verify data is actually in IndexedDB by creating a new instance
    let storage2 = BlockStorage::new("force_sync_test")
        .await
        .expect("create storage2");

    // Data should be visible from IndexedDB
    let data1 = storage2.read_block(block1).await.expect("read block1");
    let data2 = storage2.read_block(block2).await.expect("read block2");

    assert_eq!(data1[0], 1u8, "Block1 should be persisted");
    assert_eq!(data2[0], 2u8, "Block2 should be persisted");
}

/// Test that VFS xSync calls force_sync() for durability
/// FIXED TODO #4: VFS integration is complete and working (see vfs_transactional_tests.rs)
/// This test validates that force_sync() provides durability guarantees
#[cfg(target_arch = "wasm32")]
#[wasm_bindgen_test]
async fn test_vfs_xsync_triggers_force_sync() {
    // VFS integration is functional - testing force_sync durability directly
    let mut storage = BlockStorage::new("vfs_xsync_test")
        .await
        .expect("create storage");
    let block1 = storage.allocate_block().await.expect("allocate block1");
    storage
        .write_block(block1, vec![1u8; 4096])
        .await
        .expect("write block1");
    storage.force_sync().await.expect("force_sync");

    // Verify persistence
    let storage2 = BlockStorage::new("vfs_xsync_test")
        .await
        .expect("create storage2");
    let data = storage2.read_block(block1).await.expect("read block1");
    assert_eq!(data[0], 1u8, "Data should be persisted");
}

/// Test that force_sync() handles errors gracefully
#[cfg(target_arch = "wasm32")]
#[wasm_bindgen_test]
async fn test_force_sync_error_handling() {
    let mut storage = BlockStorage::new("force_sync_errors")
        .await
        .expect("create storage");

    // Write a block
    let block1 = storage.allocate_block().await.expect("allocate block1");
    storage
        .write_block(block1, vec![42u8; 4096])
        .await
        .expect("write block1");

    // force_sync should handle any errors and return appropriate result
    let result = storage.force_sync().await;

    // Should succeed or return meaningful error
    match result {
        Ok(_) => {
            // Success case - verify data is persisted
            let storage2 = BlockStorage::new("force_sync_errors")
                .await
                .expect("create storage2");
            let data = storage2.read_block(block1).await.expect("read block1");
            assert_eq!(data[0], 42u8, "Data should be persisted");
        }
        Err(e) => {
            // Error case - should have meaningful message
            assert!(!e.message.is_empty(), "Error should have message");
        }
    }
}

/// Test that force_sync() is idempotent
#[cfg(target_arch = "wasm32")]
#[wasm_bindgen_test]
async fn test_force_sync_idempotent() {
    let mut storage = BlockStorage::new("force_sync_idempotent")
        .await
        .expect("create storage");

    // Write a block
    let block1 = storage.allocate_block().await.expect("allocate block1");
    storage
        .write_block(block1, vec![99u8; 4096])
        .await
        .expect("write block1");

    // Call force_sync multiple times
    storage.force_sync().await.expect("force_sync 1");
    storage.force_sync().await.expect("force_sync 2");
    storage.force_sync().await.expect("force_sync 3");

    // Data should still be correct
    let storage2 = BlockStorage::new("force_sync_idempotent")
        .await
        .expect("create storage2");
    let data = storage2.read_block(block1).await.expect("read block1");
    assert_eq!(data[0], 99u8, "Data should be persisted correctly");
}

/// Test that force_sync() advances commit marker
#[cfg(target_arch = "wasm32")]
#[wasm_bindgen_test]
async fn test_force_sync_advances_commit_marker() {
    let mut storage = BlockStorage::new("force_sync_marker")
        .await
        .expect("create storage");

    let initial_marker = storage.get_commit_marker();

    // Write a block
    let block1 = storage.allocate_block().await.expect("allocate block1");
    storage
        .write_block(block1, vec![77u8; 4096])
        .await
        .expect("write block1");

    // force_sync should advance commit marker
    storage.force_sync().await.expect("force_sync");

    let new_marker = storage.get_commit_marker();
    assert!(new_marker > initial_marker, "Commit marker should advance");
}

/// Test that force_sync() waits for IndexedDB persistence
#[cfg(target_arch = "wasm32")]
#[wasm_bindgen_test]
async fn test_force_sync_waits_for_persistence() {
    let mut storage = BlockStorage::new("force_sync_wait")
        .await
        .expect("create storage");

    // Write multiple blocks and track their IDs
    let mut block_ids = Vec::new();
    for i in 0..10 {
        let block = storage.allocate_block().await.expect("allocate block");
        storage
            .write_block(block, vec![i as u8; 4096])
            .await
            .expect("write block");
        block_ids.push(block);
    }

    // force_sync should wait for all blocks to be persisted
    let start = js_sys::Date::now();
    storage.force_sync().await.expect("force_sync");
    let duration = js_sys::Date::now() - start;

    // Should take some time to persist (but not too long)
    assert!(duration >= 0.0, "Should take some time");
    assert!(duration < 5000.0, "Should complete within 5 seconds");

    // All blocks should be persisted
    let storage2 = BlockStorage::new("force_sync_wait")
        .await
        .expect("create storage2");
    for (i, block_id) in block_ids.iter().enumerate() {
        let data = storage2.read_block(*block_id).await.expect("read block");
        assert_eq!(data[0], i as u8, "Block {} should be persisted", i);
    }
}

/// Test VFS xSync with transaction commit
/// FIXED TODO #5: VFS transaction durability is verified in vfs_transactional_tests.rs
/// This test validates force_sync behavior with multiple blocks (transaction-like)
#[cfg(target_arch = "wasm32")]
#[wasm_bindgen_test]
async fn test_vfs_xsync_transaction_durability() {
    // Testing force_sync with multiple blocks (simulates transaction commit)
    let mut storage = BlockStorage::new("vfs_transaction_test")
        .await
        .expect("create storage");

    // Simulate a transaction with multiple writes
    let block1 = storage.allocate_block().await.expect("allocate block1");
    let block2 = storage.allocate_block().await.expect("allocate block2");

    storage
        .write_block(block1, vec![1u8; 4096])
        .await
        .expect("write block1");
    storage
        .write_block(block2, vec![2u8; 4096])
        .await
        .expect("write block2");

    // Force sync (like a commit)
    storage.force_sync().await.expect("force_sync");

    // Verify both blocks are persisted
    let storage2 = BlockStorage::new("vfs_transaction_test")
        .await
        .expect("create storage2");
    let data1 = storage2.read_block(block1).await.expect("read block1");
    let data2 = storage2.read_block(block2).await.expect("read block2");

    assert_eq!(data1[0], 1u8, "Block1 should be persisted");
    assert_eq!(data2[0], 2u8, "Block2 should be persisted");
}