use serde::Serialize; use uuid::Uuid; use mitra_config::Instance; use crate::activitypub::{ actors::types::Actor, constants::AP_PUBLIC, deliverer::OutgoingActivity, identifiers::{local_actor_id, local_object_id, post_object_id}, types::{build_default_context, Context}, vocabulary::LIKE, }; use crate::database::{DatabaseClient, DatabaseError}; use crate::models::{ posts::types::{Post, Visibility}, profiles::types::DbActorProfile, users::types::User, }; #[derive(Serialize)] struct Like { #[serde(rename = "@context")] context: Context, #[serde(rename = "type")] activity_type: String, id: String, actor: String, object: String, to: Vec, cc: Vec, } pub fn get_like_audience( note_author_id: &str, note_visibility: &Visibility, ) -> (Vec, Vec) { let mut primary_audience = vec![note_author_id.to_string()]; if matches!(note_visibility, Visibility::Public) { primary_audience.push(AP_PUBLIC.to_string()); }; let secondary_audience = vec![]; (primary_audience, secondary_audience) } fn build_like( instance_url: &str, actor_profile: &DbActorProfile, object_id: &str, reaction_id: &Uuid, post_author_id: &str, post_visibility: &Visibility, ) -> Like { let activity_id = local_object_id(instance_url, reaction_id); let actor_id = local_actor_id(instance_url, &actor_profile.username); let (primary_audience, secondary_audience) = get_like_audience(post_author_id, post_visibility); Like { context: build_default_context(), activity_type: LIKE.to_string(), id: activity_id, actor: actor_id, object: object_id.to_string(), to: primary_audience, cc: secondary_audience, } } pub async fn get_like_recipients( _db_client: &impl DatabaseClient, _instance_url: &str, post: &Post, ) -> Result, DatabaseError> { let mut recipients: Vec = Vec::new(); if let Some(remote_actor) = post.author.actor_json.as_ref() { recipients.push(remote_actor.clone()); }; Ok(recipients) } pub async fn prepare_like( db_client: &impl DatabaseClient, instance: &Instance, sender: &User, post: &Post, reaction_id: &Uuid, ) -> Result { let recipients = get_like_recipients( db_client, &instance.url(), post, ).await?; let object_id = post_object_id(&instance.url(), post); let post_author_id = post.author.actor_id(&instance.url()); let activity = build_like( &instance.url(), &sender.profile, &object_id, reaction_id, &post_author_id, &post.visibility, ); Ok(OutgoingActivity::new( instance, sender, activity, recipients, )) } #[cfg(test)] mod tests { use mitra_utils::id::generate_ulid; use super::*; const INSTANCE_URL: &str = "https://example.com"; #[test] fn test_build_like() { let author = DbActorProfile::default(); let post_id = "https://example.com/objects/123"; let post_author_id = "https://example.com/users/test"; let reaction_id = generate_ulid(); let activity = build_like( INSTANCE_URL, &author, post_id, &reaction_id, post_author_id, &Visibility::Public, ); assert_eq!( activity.id, format!("{}/objects/{}", INSTANCE_URL, reaction_id), ); assert_eq!(activity.object, post_id); assert_eq!(activity.to, vec![post_author_id, AP_PUBLIC]); assert_eq!(activity.cc.is_empty(), true); } }