- Replace PosterStorage with generic ImageStorage port (IMAGE_STORAGE_BACKEND/PATH env vars)
- Rename poster-storage crate to image-storage; serve at /images/{*key}
- Add bio and avatar_path to User model (migration 0009_user_profile)
- update_profile use case with avatar upload, mime validation, old avatar cleanup
- GET/PUT /api/v1/profile and GET/POST /settings/profile HTML page
- Enrich AP Person actor with summary, icon, url, discoverable fields
- Store remote actor avatar_url (migration 0010_ap_remote_actor_avatar)
- Shared inbox delivery via collect_inboxes deduplication
- Broadcast Update(Person) to followers on UserUpdated event
- Paginated outbox: OrderedCollection + OrderedCollectionPage with cursor
- Announce/boost tracking in ap_announces table (migration 0011_ap_announces)
219 lines
8.0 KiB
Rust
219 lines
8.0 KiB
Rust
use chrono::NaiveDateTime;
|
|
use domain::{
|
|
errors::DomainError,
|
|
events::DomainEvent,
|
|
value_objects::{ExternalMetadataId, MovieId, PosterPath, Rating, ReviewId, UserId},
|
|
};
|
|
use serde::{Deserialize, Serialize};
|
|
use uuid::Uuid;
|
|
|
|
#[derive(Debug, Serialize, Deserialize, PartialEq)]
|
|
#[serde(tag = "type", content = "data")]
|
|
pub enum EventPayload {
|
|
ReviewLogged {
|
|
review_id: String,
|
|
movie_id: String,
|
|
user_id: String,
|
|
rating: u8,
|
|
watched_at: i64,
|
|
},
|
|
ReviewUpdated {
|
|
review_id: String,
|
|
movie_id: String,
|
|
user_id: String,
|
|
rating: u8,
|
|
watched_at: i64,
|
|
},
|
|
MovieDiscovered {
|
|
movie_id: String,
|
|
external_metadata_id: String,
|
|
},
|
|
MovieDeleted {
|
|
movie_id: String,
|
|
poster_path: Option<String>,
|
|
},
|
|
UserUpdated {
|
|
user_id: String,
|
|
},
|
|
}
|
|
|
|
impl EventPayload {
|
|
pub fn event_type(&self) -> &'static str {
|
|
match self {
|
|
EventPayload::ReviewLogged { .. } => "ReviewLogged",
|
|
EventPayload::ReviewUpdated { .. } => "ReviewUpdated",
|
|
EventPayload::MovieDiscovered { .. } => "MovieDiscovered",
|
|
EventPayload::MovieDeleted { .. } => "MovieDeleted",
|
|
EventPayload::UserUpdated { .. } => "UserUpdated",
|
|
}
|
|
}
|
|
}
|
|
|
|
fn parse_uuid(s: &str, field: &str) -> Result<Uuid, DomainError> {
|
|
Uuid::parse_str(s)
|
|
.map_err(|e| DomainError::InfrastructureError(format!("{field}: {e}")))
|
|
}
|
|
|
|
fn parse_ts(ts: i64) -> Result<NaiveDateTime, DomainError> {
|
|
chrono::DateTime::from_timestamp(ts, 0)
|
|
.map(|dt| dt.naive_utc())
|
|
.ok_or_else(|| DomainError::InfrastructureError(format!("invalid timestamp: {ts}")))
|
|
}
|
|
|
|
impl From<&DomainEvent> for EventPayload {
|
|
fn from(event: &DomainEvent) -> Self {
|
|
match event {
|
|
DomainEvent::ReviewLogged { review_id, movie_id, user_id, rating, watched_at } => {
|
|
EventPayload::ReviewLogged {
|
|
review_id: review_id.value().to_string(),
|
|
movie_id: movie_id.value().to_string(),
|
|
user_id: user_id.value().to_string(),
|
|
rating: rating.value(),
|
|
watched_at: watched_at.and_utc().timestamp(),
|
|
}
|
|
}
|
|
DomainEvent::ReviewUpdated { review_id, movie_id, user_id, rating, watched_at } => {
|
|
EventPayload::ReviewUpdated {
|
|
review_id: review_id.value().to_string(),
|
|
movie_id: movie_id.value().to_string(),
|
|
user_id: user_id.value().to_string(),
|
|
rating: rating.value(),
|
|
watched_at: watched_at.and_utc().timestamp(),
|
|
}
|
|
}
|
|
DomainEvent::MovieDiscovered { movie_id, external_metadata_id } => {
|
|
EventPayload::MovieDiscovered {
|
|
movie_id: movie_id.value().to_string(),
|
|
external_metadata_id: external_metadata_id.value().to_owned(),
|
|
}
|
|
}
|
|
DomainEvent::MovieDeleted { movie_id, poster_path } => EventPayload::MovieDeleted {
|
|
movie_id: movie_id.value().to_string(),
|
|
poster_path: poster_path.as_ref().map(|p| p.value().to_string()),
|
|
},
|
|
DomainEvent::UserUpdated { user_id } => EventPayload::UserUpdated {
|
|
user_id: user_id.value().to_string(),
|
|
},
|
|
}
|
|
}
|
|
}
|
|
|
|
impl TryFrom<EventPayload> for DomainEvent {
|
|
type Error = DomainError;
|
|
fn try_from(payload: EventPayload) -> Result<Self, DomainError> {
|
|
match payload {
|
|
EventPayload::ReviewLogged { review_id, movie_id, user_id, rating, watched_at } => {
|
|
Ok(DomainEvent::ReviewLogged {
|
|
review_id: ReviewId::from_uuid(parse_uuid(&review_id, "review_id")?),
|
|
movie_id: MovieId::from_uuid(parse_uuid(&movie_id, "movie_id")?),
|
|
user_id: UserId::from_uuid(parse_uuid(&user_id, "user_id")?),
|
|
rating: Rating::new(rating)?,
|
|
watched_at: parse_ts(watched_at)?,
|
|
})
|
|
}
|
|
EventPayload::ReviewUpdated { review_id, movie_id, user_id, rating, watched_at } => {
|
|
Ok(DomainEvent::ReviewUpdated {
|
|
review_id: ReviewId::from_uuid(parse_uuid(&review_id, "review_id")?),
|
|
movie_id: MovieId::from_uuid(parse_uuid(&movie_id, "movie_id")?),
|
|
user_id: UserId::from_uuid(parse_uuid(&user_id, "user_id")?),
|
|
rating: Rating::new(rating)?,
|
|
watched_at: parse_ts(watched_at)?,
|
|
})
|
|
}
|
|
EventPayload::MovieDiscovered { movie_id, external_metadata_id } => {
|
|
Ok(DomainEvent::MovieDiscovered {
|
|
movie_id: MovieId::from_uuid(parse_uuid(&movie_id, "movie_id")?),
|
|
external_metadata_id: ExternalMetadataId::new(external_metadata_id)?,
|
|
})
|
|
}
|
|
EventPayload::MovieDeleted { movie_id, poster_path } => {
|
|
let movie_id = MovieId::from_uuid(parse_uuid(&movie_id, "movie_id")?);
|
|
let poster_path = poster_path
|
|
.map(|p| PosterPath::new(p))
|
|
.transpose()
|
|
.map_err(|e| DomainError::InfrastructureError(e.to_string()))?;
|
|
Ok(DomainEvent::MovieDeleted { movie_id, poster_path })
|
|
}
|
|
EventPayload::UserUpdated { user_id } => {
|
|
Ok(DomainEvent::UserUpdated {
|
|
user_id: UserId::from_uuid(parse_uuid(&user_id, "user_id")?),
|
|
})
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::*;
|
|
|
|
fn fixed_dt() -> NaiveDateTime {
|
|
chrono::DateTime::from_timestamp(1_700_000_000, 0).unwrap().naive_utc()
|
|
}
|
|
|
|
fn review_logged() -> DomainEvent {
|
|
DomainEvent::ReviewLogged {
|
|
review_id: ReviewId::from_uuid(Uuid::new_v4()),
|
|
movie_id: MovieId::from_uuid(Uuid::new_v4()),
|
|
user_id: UserId::from_uuid(Uuid::new_v4()),
|
|
rating: Rating::new(4).unwrap(),
|
|
watched_at: fixed_dt(),
|
|
}
|
|
}
|
|
|
|
fn review_updated() -> DomainEvent {
|
|
DomainEvent::ReviewUpdated {
|
|
review_id: ReviewId::from_uuid(Uuid::new_v4()),
|
|
movie_id: MovieId::from_uuid(Uuid::new_v4()),
|
|
user_id: UserId::from_uuid(Uuid::new_v4()),
|
|
rating: Rating::new(3).unwrap(),
|
|
watched_at: fixed_dt(),
|
|
}
|
|
}
|
|
|
|
fn movie_discovered() -> DomainEvent {
|
|
DomainEvent::MovieDiscovered {
|
|
movie_id: MovieId::from_uuid(Uuid::new_v4()),
|
|
external_metadata_id: ExternalMetadataId::new("tt1234567".into()).unwrap(),
|
|
}
|
|
}
|
|
|
|
fn round_trip(event: DomainEvent) {
|
|
let payload = EventPayload::from(&event);
|
|
let json = serde_json::to_string(&payload).expect("serialize");
|
|
let back: EventPayload = serde_json::from_str(&json).expect("deserialize");
|
|
let recovered = DomainEvent::try_from(back).expect("try_from");
|
|
assert_eq!(EventPayload::from(&event), EventPayload::from(&recovered));
|
|
}
|
|
|
|
#[test]
|
|
fn round_trip_review_logged() {
|
|
round_trip(review_logged());
|
|
}
|
|
|
|
#[test]
|
|
fn round_trip_review_updated() {
|
|
round_trip(review_updated());
|
|
}
|
|
|
|
#[test]
|
|
fn round_trip_movie_discovered() {
|
|
round_trip(movie_discovered());
|
|
}
|
|
|
|
#[test]
|
|
fn serialized_format_is_tagged() {
|
|
let payload = EventPayload::from(&movie_discovered());
|
|
let json = serde_json::to_string(&payload).unwrap();
|
|
assert!(json.contains(r#""type":"MovieDiscovered""#));
|
|
assert!(json.contains(r#""data":"#));
|
|
}
|
|
|
|
#[test]
|
|
fn event_type_strings() {
|
|
assert_eq!(EventPayload::from(&review_logged()).event_type(), "ReviewLogged");
|
|
assert_eq!(EventPayload::from(&review_updated()).event_type(), "ReviewUpdated");
|
|
assert_eq!(EventPayload::from(&movie_discovered()).event_type(), "MovieDiscovered");
|
|
}
|
|
}
|