use crate::{ activities::{block::SiteOrCommunity, verify_community_matches}, objects::{community::ApubCommunity, person::ApubPerson}, protocol::InCommunity, }; use activitypub_federation::{ config::Data, fetch::object_id::ObjectId, kinds::activity::BlockType, protocol::helpers::deserialize_one_or_many, }; use anyhow::anyhow; use chrono::{DateTime, Utc}; use lemmy_api_common::context::LemmyContext; use lemmy_utils::error::LemmyError; use serde::{Deserialize, Serialize}; use serde_with::skip_serializing_none; use url::Url; #[skip_serializing_none] #[derive(Clone, Debug, Deserialize, Serialize)] #[serde(rename_all = "camelCase")] pub struct BlockUser { pub(crate) actor: ObjectId, #[serde(deserialize_with = "deserialize_one_or_many")] pub(crate) to: Vec, pub(crate) object: ObjectId, #[serde(deserialize_with = "deserialize_one_or_many")] pub(crate) cc: Vec, pub(crate) target: ObjectId, #[serde(rename = "type")] pub(crate) kind: BlockType, pub(crate) id: Url, pub(crate) audience: Option>, /// Quick and dirty solution. /// TODO: send a separate Delete activity instead pub(crate) remove_data: Option, /// block reason, written to mod log pub(crate) summary: Option, pub(crate) expires: Option>, } #[async_trait::async_trait] impl InCommunity for BlockUser { async fn community(&self, context: &Data) -> Result { let target = self.target.dereference(context).await?; let community = match target { SiteOrCommunity::Community(c) => c, SiteOrCommunity::Site(_) => return Err(anyhow!("activity is not in community").into()), }; if let Some(audience) = &self.audience { verify_community_matches(audience, community.actor_id.clone())?; } Ok(community) } }