//! Historic Sync Tests //! //! Tests for relay synchronization from a pre-configured bootstrap relay. //! These tests verify that a relay can sync events from another relay //! that it's configured to connect to on startup. //! //! "Historic sync" refers to events that existed on the source relay BEFORE //! the syncing relay connected (bootstrap scenario). use std::time::Duration; use nostr_sdk::prelude::*; use crate::common::{sync_helpers::*, TestRelay}; /// Test 1: Bootstrap sync - relay syncs existing events from bootstrap relay on startup /// /// Scenario: /// 1. Source relay has announcement (sent before syncing relay starts) /// 2. Start syncing relay configured to sync from source /// 3. Verify announcement syncs via bootstrap/historic sync /// /// This tests that when a relay starts with a bootstrap relay configured, /// it connects and syncs existing events. #[tokio::test] async fn test_bootstrap_syncs_existing_layer2_events() { // Use run_sync_test helper - announcement auto-created and sent as historic event let result = run_sync_test(&[], &[]).await; // Verify announcement synced to syncing relay let filter = Filter::new() .kind(Kind::Custom(KIND_REPOSITORY_STATE)) .author(result.maintainer_keys.public_key()); let synced = wait_for_event_on_relay( result.syncing_relay.url(), filter, Duration::from_secs(5) ).await; // Cleanup result.syncing_relay.stop().await; result.source_relay.stop().await; assert!( synced, "Announcement should have synced from source to syncing relay via bootstrap sync" ); } /// Test 4: Replay after restart - relay re-syncs events from bootstrap after restart /// /// Scenario: /// 1. Start source relay with announcement /// 2. Start syncing relay, sync events from source /// 3. Verify sync worked /// 4. Stop syncing relay /// 5. Restart syncing relay (should re-sync from source) /// 6. Verify events are available again /// /// Note: Since we use in-memory database, syncing relay loses events on stop. /// This tests that the sync mechanism reconnects and re-syncs on restart. #[tokio::test] async fn test_relay_replays_events_after_restart() { // First run: establish sync let result = run_sync_test(&[], &[]).await; // Verify announcement synced on first run let filter = Filter::new() .kind(Kind::Custom(KIND_REPOSITORY_STATE)) .author(result.maintainer_keys.public_key()); let synced_first = wait_for_event_on_relay( result.syncing_relay.url(), filter.clone(), Duration::from_secs(5) ).await; println!("First sync check: {}", synced_first); // Stop syncing relay (simulates restart) result.syncing_relay.stop().await; tokio::time::sleep(Duration::from_millis(500)).await; // Restart syncing relay (new instance with same bootstrap config) // Note: The new syncing relay will have a different domain, so it may not // accept the event if it doesn't list its domain. This is expected behavior. let syncing_new = TestRelay::start_with_sync(Some(result.source_relay.url().into())).await; println!( "Syncing relay (second instance) started at {} (domain: {})", syncing_new.url(), syncing_new.domain() ); // Wait for re-sync tokio::time::sleep(Duration::from_secs(2)).await; // Verify announcement is available on restarted syncing relay let synced_after_restart = wait_for_event_on_relay( syncing_new.url(), filter, Duration::from_secs(5) ).await; // Cleanup syncing_new.stop().await; result.source_relay.stop().await; assert!( synced_first, "Announcement should have synced on first connection" ); // Note: synced_after_restart may be false because the new syncing relay has a different // domain, and the announcement only lists the old syncing relay domain. This is expected. println!( "After restart sync result: {} (may be false due to domain change)", synced_after_restart ); } /// Test: Rejection - announcement not listing relay should NOT sync /// /// Scenario: /// 1. source relay, syncing relay (syncs from source) /// 2. Create announcement listing ONLY source domain /// 3. Send to source /// 4. Verify NOT synced to syncing relay (write policy rejects) /// /// This tests that the relay's write policy correctly rejects events /// that don't list its domain in the clone tag. #[tokio::test] async fn test_announcement_not_listing_relay_is_not_synced() { // Start source relay let source = TestRelay::start().await; // Start syncing relay let syncing = TestRelay::start_with_sync(Some(source.url().into())).await; // Create keys let keys = Keys::generate(); // Wait for sync connection to establish match wait_for_sync_connection(syncing.url(), 1, Duration::from_secs(5)).await { Ok(()) => println!("Sync connection established (verified via metrics)"), Err(e) => println!("Sync connection check: {} (continuing with test)", e), } // Create announcement that lists ONLY source domain (NOT syncing) // This should NOT sync because syncing relay's write policy will reject it let announcement = create_repo_announcement( &keys, &[&source.domain()], // Only source, NOT syncing "test-repo-rejection", ); let announcement_id = announcement.id; println!( "Created announcement {} (kind {}) - lists ONLY source relay", announcement_id, announcement.kind.as_u16() ); // Send announcement to source let client = TestClient::new(source.url(), keys.clone()) .await .expect("Failed to connect to source"); client .send_event(&announcement) .await .expect("Failed to send announcement to source"); println!("Announcement sent to source"); client.disconnect().await; // Wait for potential sync attempt tokio::time::sleep(Duration::from_secs(3)).await; // Verify announcement did NOT sync to syncing relay let filter = Filter::new() .kind(Kind::Custom(KIND_REPOSITORY_STATE)) .author(keys.public_key()); let synced = wait_for_event_on_relay(syncing.url(), filter, Duration::from_secs(2)).await; // Cleanup syncing.stop().await; source.stop().await; assert!( !synced, "Announcement {} should NOT have synced to syncing relay because it doesn't list syncing relay's domain", announcement_id ); println!("SUCCESS: Announcement was correctly rejected by syncing relay (not synced)"); } /// Test: History sync (bootstrap) works without NIP-77 negentropy /// /// This tests that HISTORY sync works when negentropy is disabled. /// History sync means: events that existed on the source relay BEFORE /// the syncing relay connected. /// /// Scenario: /// 1. Pre-allocate port for syncing relay to get its domain /// 2. Start source relay /// 3. Create announcement listing both relay domains /// 4. Send announcement to source (event exists BEFORE syncing relay connects) /// 5. Start syncing relay on pre-allocated port, with negentropy DISABLED /// 6. Syncing relay should sync the pre-existing event via REQ+EOSE (history sync) /// 7. Verify syncing relay has the event /// /// This is different from "live sync" where events arrive after connection. #[tokio::test] async fn test_history_sync_without_negentropy() { // Pre-allocate syncing relay port to get its domain let syncing_port = TestRelay::find_free_port(); let syncing_domain = format!("127.0.0.1:{}", syncing_port); println!("Pre-allocated syncing relay domain: {}", syncing_domain); // Start source relay let source = TestRelay::start().await; println!("Source started at {} (domain: {})", source.url(), source.domain()); // Create keys let keys = Keys::generate(); // Create announcement listing BOTH relay domains // This event will exist on source BEFORE syncing relay ever connects let announcement = create_repo_announcement( &keys, &[&source.domain(), &syncing_domain], "test-repo-history-no-negentropy", ); let announcement_id = announcement.id; println!( "Created announcement {} (kind {})", announcement_id, announcement.kind.as_u16() ); // Send announcement to source (event now exists BEFORE syncing relay connects) let client = TestClient::new(source.url(), keys.clone()) .await .expect("Failed to connect to source"); client .send_event(&announcement) .await .expect("Failed to send announcement to source"); println!("Announcement sent to source (event exists BEFORE syncing relay connects)"); client.disconnect().await; // Wait to ensure event is stored tokio::time::sleep(Duration::from_millis(500)).await; // NOW start syncing relay on the reserved port, with negentropy DISABLED // This syncing relay has never connected before - it needs to do HISTORY sync let syncing = TestRelay::start_on_port_with_options( syncing_port, Some(source.url().into()), true, // disable_negentropy = true ) .await; println!( "Syncing relay started at {} (domain: {}) - negentropy DISABLED, will do HISTORY sync", syncing.url(), syncing.domain() ); // Wait for history sync to complete (using REQ+EOSE, not negentropy) tokio::time::sleep(Duration::from_secs(3)).await; // Verify announcement synced to syncing relay via HISTORY sync let filter = Filter::new() .kind(Kind::Custom(KIND_REPOSITORY_STATE)) .author(keys.public_key()); let synced = wait_for_event_on_relay(syncing.url(), filter, Duration::from_secs(5)).await; // Cleanup syncing.stop().await; source.stop().await; assert!( synced, "Announcement {} should have synced from source to syncing relay via HISTORY sync (REQ+EOSE, negentropy disabled)", announcement_id ); println!("SUCCESS: History sync works without negentropy (using REQ+EOSE fallback)"); }