use crate::{ activities::{ community::{announce::AnnouncableActivities, send_to_community}, generate_activity_id, verify_activity, verify_is_public, verify_person_in_community, voting::{vote_comment, vote_post}, }, context::lemmy_context, fetcher::object_id::ObjectId, objects::{community::ApubCommunity, person::ApubPerson}, PostOrComment, }; use activitystreams::{base::AnyBase, primitives::OneOrMany, public, unparsed::Unparsed}; use anyhow::anyhow; use lemmy_api_common::blocking; use lemmy_apub_lib::{ data::Data, traits::{ActivityFields, ActivityHandler, ActorType}, }; use lemmy_db_schema::{newtypes::CommunityId, source::community::Community, traits::Crud}; use lemmy_utils::LemmyError; use lemmy_websocket::LemmyContext; use serde::{Deserialize, Serialize}; use std::{convert::TryFrom, ops::Deref}; use strum_macros::ToString; use url::Url; #[derive(Clone, Debug, ToString, Deserialize, Serialize)] pub enum VoteType { Like, Dislike, } impl TryFrom for VoteType { type Error = LemmyError; fn try_from(value: i16) -> Result { match value { 1 => Ok(VoteType::Like), -1 => Ok(VoteType::Dislike), _ => Err(anyhow!("invalid vote value").into()), } } } impl From<&VoteType> for i16 { fn from(value: &VoteType) -> i16 { match value { VoteType::Like => 1, VoteType::Dislike => -1, } } } #[derive(Clone, Debug, Deserialize, Serialize, ActivityFields)] #[serde(rename_all = "camelCase")] pub struct Vote { actor: ObjectId, to: Vec, pub(in crate::activities::voting) object: ObjectId, cc: [ObjectId; 1], #[serde(rename = "type")] pub(in crate::activities::voting) kind: VoteType, id: Url, #[serde(rename = "@context")] context: OneOrMany, #[serde(flatten)] unparsed: Unparsed, } impl Vote { pub(in crate::activities::voting) fn new( object: &PostOrComment, actor: &ApubPerson, community: &ApubCommunity, kind: VoteType, context: &LemmyContext, ) -> Result { Ok(Vote { actor: ObjectId::new(actor.actor_id()), to: vec![public()], object: ObjectId::new(object.ap_id()), cc: [ObjectId::new(community.actor_id())], kind: kind.clone(), id: generate_activity_id(kind, &context.settings().get_protocol_and_hostname())?, context: lemmy_context(), unparsed: Default::default(), }) } pub async fn send( object: &PostOrComment, actor: &ApubPerson, community_id: CommunityId, kind: VoteType, context: &LemmyContext, ) -> Result<(), LemmyError> { let community = blocking(context.pool(), move |conn| { Community::read(conn, community_id) }) .await?? .into(); let vote = Vote::new(object, actor, &community, kind, context)?; let vote_id = vote.id.clone(); let activity = AnnouncableActivities::Vote(vote); send_to_community(activity, &vote_id, actor, &community, vec![], context).await } } #[async_trait::async_trait(?Send)] impl ActivityHandler for Vote { type DataType = LemmyContext; async fn verify( &self, context: &Data, request_counter: &mut i32, ) -> Result<(), LemmyError> { verify_is_public(&self.to)?; verify_activity(self, &context.settings())?; verify_person_in_community(&self.actor, &self.cc[0], context, request_counter).await?; Ok(()) } async fn receive( self, context: &Data, request_counter: &mut i32, ) -> Result<(), LemmyError> { let actor = self.actor.dereference(context, request_counter).await?; let object = self.object.dereference(context, request_counter).await?; match object { PostOrComment::Post(p) => vote_post(&self.kind, actor, p.deref(), context).await, PostOrComment::Comment(c) => vote_comment(&self.kind, actor, &c, context).await, } } }