2 check_is_apub_id_valid,
4 community_moderators::ApubCommunityModerators,
5 community_outbox::ApubCommunityOutbox,
8 community::ApubCommunity,
9 read_from_string_or_source_opt,
10 verify_image_domain_matches,
12 protocol::{objects::Endpoints, ImageObject, SourceCompat},
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};
19 utils::{check_slurs, check_slurs_opt},
22 use lemmy_websocket::LemmyContext;
23 use serde::{Deserialize, Serialize};
24 use serde_with::skip_serializing_none;
27 #[skip_serializing_none]
28 #[derive(Clone, Debug, Deserialize, Serialize)]
29 #[serde(rename_all = "camelCase")]
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 pub(crate) inbox: Url,
37 pub(crate) followers: Url,
38 pub(crate) public_key: PublicKey,
41 pub(crate) name: Option<String>,
42 pub(crate) summary: Option<String>,
43 pub(crate) source: Option<SourceCompat>,
44 pub(crate) icon: Option<ImageObject>,
46 pub(crate) image: Option<ImageObject>,
48 pub(crate) sensitive: Option<bool>,
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>>,
58 pub(crate) async fn verify(
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)?;
68 let slur_regex = &context.settings().slur_regex();
69 check_slurs(&self.preferred_username, slur_regex)?;
70 check_slurs_opt(&self.name, slur_regex)?;
71 let description = read_from_string_or_source_opt(&self.summary, &self.source);
72 check_slurs_opt(&description, slur_regex)?;
76 pub(crate) fn into_form(self) -> CommunityForm {
78 name: self.preferred_username.clone(),
79 title: self.name.unwrap_or(self.preferred_username),
80 description: read_from_string_or_source_opt(&self.summary, &self.source),
82 published: self.published.map(|u| u.naive_local()),
83 updated: self.updated.map(|u| u.naive_local()),
85 nsfw: Some(self.sensitive.unwrap_or(false)),
86 actor_id: Some(self.id.into()),
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())),