]> Untitled Git - lemmy.git/blob - crates/apub/src/protocol/objects/group.rs
7820624d91303d97588fb8d35f3c6813182d1ead
[lemmy.git] / crates / apub / src / protocol / objects / group.rs
1 use crate::{
2   check_is_apub_id_valid,
3   collections::{
4     community_moderators::ApubCommunityModerators,
5     community_outbox::ApubCommunityOutbox,
6   },
7   objects::{
8     community::ApubCommunity,
9     get_summary_from_string_or_source,
10     verify_image_domain_matches,
11   },
12   protocol::{objects::Endpoints, ImageObject, Source},
13 };
14 use activitystreams_kinds::actor::GroupType;
15 use chrono::{DateTime, FixedOffset};
16 use lemmy_apub_lib::{object_id::ObjectId, signatures::PublicKey, verify::verify_domains_match};
17 use lemmy_db_schema::{naive_now, source::community::CommunityForm};
18 use lemmy_utils::{
19   utils::{check_slurs, check_slurs_opt},
20   LemmyError,
21 };
22 use lemmy_websocket::LemmyContext;
23 use serde::{Deserialize, Serialize};
24 use serde_with::skip_serializing_none;
25 use url::Url;
26
27 #[skip_serializing_none]
28 #[derive(Clone, Debug, Deserialize, Serialize)]
29 #[serde(rename_all = "camelCase")]
30 pub struct Group {
31   #[serde(rename = "type")]
32   pub(crate) kind: GroupType,
33   pub(crate) id: ObjectId<ApubCommunity>,
34   /// username, set at account creation and usually fixed after that
35   pub(crate) preferred_username: String,
36   /// displayname
37   pub(crate) name: String,
38   pub(crate) inbox: Url,
39   pub(crate) followers: Url,
40   pub(crate) public_key: PublicKey,
41
42   pub(crate) summary: Option<String>,
43   pub(crate) source: Option<Source>,
44   pub(crate) icon: Option<ImageObject>,
45   /// banner
46   pub(crate) image: Option<ImageObject>,
47   // lemmy extension
48   pub(crate) sensitive: Option<bool>,
49   // lemmy extension
50   pub(crate) moderators: Option<ObjectId<ApubCommunityModerators>>,
51   pub(crate) outbox: ObjectId<ApubCommunityOutbox>,
52   pub(crate) endpoints: Option<Endpoints>,
53   pub(crate) published: Option<DateTime<FixedOffset>>,
54   pub(crate) updated: Option<DateTime<FixedOffset>>,
55 }
56
57 impl Group {
58   pub(crate) async fn verify(
59     &self,
60     expected_domain: &Url,
61     context: &LemmyContext,
62   ) -> Result<(), LemmyError> {
63     check_is_apub_id_valid(self.id.inner(), true, &context.settings())?;
64     verify_domains_match(expected_domain, self.id.inner())?;
65     verify_image_domain_matches(expected_domain, &self.icon)?;
66     verify_image_domain_matches(expected_domain, &self.image)?;
67
68     let slur_regex = &context.settings().slur_regex();
69     check_slurs(&self.preferred_username, slur_regex)?;
70     check_slurs(&self.name, slur_regex)?;
71     let description = get_summary_from_string_or_source(&self.summary, &self.source);
72     check_slurs_opt(&description, slur_regex)?;
73     Ok(())
74   }
75
76   pub(crate) fn into_form(self) -> CommunityForm {
77     CommunityForm {
78       name: self.preferred_username,
79       title: self.name,
80       description: get_summary_from_string_or_source(&self.summary, &self.source),
81       removed: None,
82       published: self.published.map(|u| u.naive_local()),
83       updated: self.updated.map(|u| u.naive_local()),
84       deleted: None,
85       nsfw: Some(self.sensitive.unwrap_or(false)),
86       actor_id: Some(self.id.into()),
87       local: Some(false),
88       private_key: None,
89       hidden: Some(false),
90       public_key: self.public_key.public_key_pem,
91       last_refreshed_at: Some(naive_now()),
92       icon: Some(self.icon.map(|i| i.url.into())),
93       banner: Some(self.image.map(|i| i.url.into())),
94       followers_url: Some(self.followers.into()),
95       inbox_url: Some(self.inbox.into()),
96       shared_inbox_url: Some(self.endpoints.map(|e| e.shared_inbox.into())),
97     }
98   }
99 }