use std::sync::Arc; use domain::models::{WatchEvent, WatchEventSource}; use domain::ports::{MovieRepository, WatchEventRepository}; use domain::testing::{InMemoryWatchEventRepository, NoopEventPublisher}; use domain::value_objects::UserId; use uuid::Uuid; use crate::integrations::commands::{ConfirmWatchEventsCommand, WatchEventConfirmation}; use crate::integrations::confirm; use crate::test_helpers::NoopReviewLogger; fn noop_logger() -> Arc { Arc::new(NoopReviewLogger) } #[tokio::test] async fn confirms_watch_event_via_review_logger() { let watch_events: Arc = InMemoryWatchEventRepository::new(); let uid = Uuid::new_v4(); let event = WatchEvent::new( UserId::from_uuid(uid), "Test Movie".into(), Some(2024), None, WatchEventSource::Jellyfin, chrono::Utc::now().naive_utc(), None, ); let event_id = event.id().value(); watch_events.save(&event).await.unwrap(); let result = confirm::execute( Arc::clone(&watch_events), noop_logger(), ConfirmWatchEventsCommand { user_id: uid, confirmations: vec![WatchEventConfirmation { watch_event_id: event_id, rating: 4, comment: None, }], }, ) .await .unwrap(); assert_eq!(result, 1); } #[tokio::test] async fn empty_confirmations_returns_zero() { let watch_events: Arc = InMemoryWatchEventRepository::new(); let result = confirm::execute( Arc::clone(&watch_events), noop_logger(), ConfirmWatchEventsCommand { user_id: Uuid::new_v4(), confirmations: vec![], }, ) .await .unwrap(); assert_eq!(result, 0); } #[tokio::test] async fn confirms_event_with_external_metadata_id_and_no_movie_id() { let watch_events: Arc = InMemoryWatchEventRepository::new(); let uid = Uuid::new_v4(); let event = WatchEvent::new( UserId::from_uuid(uid), "External Movie".into(), Some(2023), Some("tt1234567".into()), WatchEventSource::Jellyfin, chrono::Utc::now().naive_utc(), None, ); let event_id = event.id().value(); watch_events.save(&event).await.unwrap(); let result = confirm::execute( Arc::clone(&watch_events), noop_logger(), ConfirmWatchEventsCommand { user_id: uid, confirmations: vec![WatchEventConfirmation { watch_event_id: event_id, rating: 3, comment: Some("Great film".into()), }], }, ) .await .unwrap(); assert_eq!(result, 1); } #[tokio::test] async fn rejects_other_users_event() { let watch_events: Arc = InMemoryWatchEventRepository::new(); let owner = Uuid::new_v4(); let intruder = Uuid::new_v4(); let event = WatchEvent::new( UserId::from_uuid(owner), "Movie".into(), None, None, WatchEventSource::Jellyfin, chrono::Utc::now().naive_utc(), None, ); let event_id = event.id().value(); watch_events.save(&event).await.unwrap(); let result = confirm::execute( Arc::clone(&watch_events), noop_logger(), ConfirmWatchEventsCommand { user_id: intruder, confirmations: vec![WatchEventConfirmation { watch_event_id: event_id, rating: 3, comment: None, }], }, ) .await; assert!(result.is_err()); } #[tokio::test] async fn fails_when_event_not_found() { let watch_events: Arc = InMemoryWatchEventRepository::new(); let result = confirm::execute( Arc::clone(&watch_events), noop_logger(), ConfirmWatchEventsCommand { user_id: Uuid::new_v4(), confirmations: vec![WatchEventConfirmation { watch_event_id: Uuid::new_v4(), rating: 4, comment: None, }], }, ) .await; assert!(result.is_err()); } #[tokio::test] async fn confirms_event_with_movie_id() { let watch_events: Arc = InMemoryWatchEventRepository::new(); let events = NoopEventPublisher::new(); let uid = Uuid::new_v4(); let movie_uuid = Uuid::new_v4(); let event = WatchEvent::new( UserId::from_uuid(uid), "Movie With Id".into(), Some(2024), None, WatchEventSource::Jellyfin, chrono::Utc::now().naive_utc(), Some(domain::value_objects::MovieId::from_uuid(movie_uuid)), ); let event_id = event.id().value(); watch_events.save(&event).await.unwrap(); // Also seed movie repo so review_logger can find it let movies = domain::testing::InMemoryMovieRepository::new(); let movie = domain::models::Movie::from_persistence( domain::value_objects::MovieId::from_uuid(movie_uuid), None, domain::value_objects::MovieTitle::new("Movie With Id".into()).unwrap(), domain::value_objects::ReleaseYear::new(2024).unwrap(), None, None, ); movies.upsert_movie(&movie).await.unwrap(); // Build a real review logger let reviews = domain::testing::InMemoryReviewRepository::new(); let watchlist = domain::testing::InMemoryWatchlistRepository::new(); let review_logger: Arc = Arc::new(crate::diary::review_logger::DefaultReviewLogger::new( Arc::clone(&movies) as _, Arc::clone(&reviews) as _, Arc::clone(&watchlist) as _, Arc::new(domain::testing::FakeMetadataClient) as _, Arc::clone(&events) as _, )); let result = confirm::execute( Arc::clone(&watch_events), review_logger, ConfirmWatchEventsCommand { user_id: uid, confirmations: vec![WatchEventConfirmation { watch_event_id: event_id, rating: 4, comment: None, }], }, ) .await .unwrap(); assert_eq!(result, 1); } #[tokio::test] async fn confirms_event_without_movie_id_and_without_external_metadata_id() { let watch_events: Arc = InMemoryWatchEventRepository::new(); let uid = Uuid::new_v4(); let event = WatchEvent::new( UserId::from_uuid(uid), "Title Only Movie".into(), Some(2022), None, WatchEventSource::Jellyfin, chrono::Utc::now().naive_utc(), None, ); let event_id = event.id().value(); watch_events.save(&event).await.unwrap(); let result = confirm::execute( Arc::clone(&watch_events), noop_logger(), ConfirmWatchEventsCommand { user_id: uid, confirmations: vec![WatchEventConfirmation { watch_event_id: event_id, rating: 5, comment: Some("Amazing".into()), }], }, ) .await .unwrap(); assert_eq!(result, 1); } #[tokio::test] async fn confirms_multiple_events() { let watch_events: Arc = InMemoryWatchEventRepository::new(); let uid = Uuid::new_v4(); let event1 = WatchEvent::new( UserId::from_uuid(uid), "Movie One".into(), Some(2020), None, WatchEventSource::Jellyfin, chrono::Utc::now().naive_utc(), None, ); let id1 = event1.id().value(); let event2 = WatchEvent::new( UserId::from_uuid(uid), "Movie Two".into(), Some(2021), None, WatchEventSource::Jellyfin, chrono::Utc::now().naive_utc(), None, ); let id2 = event2.id().value(); watch_events.save(&event1).await.unwrap(); watch_events.save(&event2).await.unwrap(); let result = confirm::execute( Arc::clone(&watch_events), noop_logger(), ConfirmWatchEventsCommand { user_id: uid, confirmations: vec![ WatchEventConfirmation { watch_event_id: id1, rating: 3, comment: None, }, WatchEventConfirmation { watch_event_id: id2, rating: 4, comment: None, }, ], }, ) .await .unwrap(); assert_eq!(result, 2); } #[tokio::test] async fn confirms_event_without_year() { let watch_events: Arc = InMemoryWatchEventRepository::new(); let uid = Uuid::new_v4(); let event = WatchEvent::new( UserId::from_uuid(uid), "No Year Movie".into(), None, None, WatchEventSource::Jellyfin, chrono::Utc::now().naive_utc(), None, ); let event_id = event.id().value(); watch_events.save(&event).await.unwrap(); let result = confirm::execute( Arc::clone(&watch_events), noop_logger(), ConfirmWatchEventsCommand { user_id: uid, confirmations: vec![WatchEventConfirmation { watch_event_id: event_id, rating: 3, comment: None, }], }, ) .await .unwrap(); assert_eq!(result, 1); }