use crate::{ activities::{ community::announce::AnnouncableActivities, generate_activity_id, verify_activity, verify_person_in_community, voting::{vote_comment, vote_post}, }, activity_queue::send_to_community_new, extensions::context::lemmy_context, fetcher::{ objects::get_or_fetch_and_insert_post_or_comment, person::get_or_fetch_and_upsert_person, }, ActorType, PostOrComment, }; use anyhow::anyhow; use lemmy_api_common::blocking; use lemmy_apub_lib::{values::PublicUrl, ActivityCommonFields, ActivityHandler}; use lemmy_db_queries::Crud; use lemmy_db_schema::{ source::{community::Community, person::Person}, CommunityId, }; 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)] #[serde(rename_all = "camelCase")] pub struct Vote { pub(in crate::activities::voting) to: PublicUrl, pub(in crate::activities::voting) object: Url, pub(in crate::activities::voting) cc: [Url; 1], #[serde(rename = "type")] pub(in crate::activities::voting) kind: VoteType, #[serde(flatten)] pub(in crate::activities::voting) common: ActivityCommonFields, } impl Vote { pub async fn send( object: &PostOrComment, actor: &Person, community_id: CommunityId, kind: VoteType, context: &LemmyContext, ) -> Result<(), LemmyError> { let community = blocking(context.pool(), move |conn| { Community::read(conn, community_id) }) .await??; let id = generate_activity_id(kind.clone())?; let vote = Vote { to: PublicUrl::Public, object: object.ap_id(), cc: [community.actor_id()], kind, common: ActivityCommonFields { context: lemmy_context(), id: id.clone(), actor: actor.actor_id(), unparsed: Default::default(), }, }; let activity = AnnouncableActivities::Vote(vote); send_to_community_new(activity, &id, actor, &community, vec![], context).await } } #[async_trait::async_trait(?Send)] impl ActivityHandler for Vote { async fn verify( &self, context: &LemmyContext, request_counter: &mut i32, ) -> Result<(), LemmyError> { verify_activity(self.common())?; verify_person_in_community(&self.common.actor, &self.cc[0], context, request_counter).await?; Ok(()) } async fn receive( &self, context: &LemmyContext, request_counter: &mut i32, ) -> Result<(), LemmyError> { let actor = get_or_fetch_and_upsert_person(&self.common.actor, context, request_counter).await?; let object = get_or_fetch_and_insert_post_or_comment(&self.object, 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.deref(), context).await, } } fn common(&self) -> &ActivityCommonFields { &self.common } }