]> Untitled Git - lemmy.git/blob - crates/apub/src/protocol/objects/group.rs
Fix Issue #3075 by using None for local fields instead of Some(false) (#3088)
[lemmy.git] / crates / apub / src / protocol / objects / group.rs
1 use crate::{
2   check_apub_id_valid_with_strictness,
3   collections::{
4     community_featured::ApubCommunityFeatured,
5     community_moderators::ApubCommunityModerators,
6     community_outbox::ApubCommunityOutbox,
7   },
8   fetch_local_site_data,
9   objects::{community::ApubCommunity, read_from_string_or_source_opt},
10   protocol::{
11     objects::{Endpoints, LanguageTag},
12     ImageObject,
13     Source,
14   },
15 };
16 use activitypub_federation::{
17   fetch::{collection_id::CollectionId, object_id::ObjectId},
18   kinds::actor::GroupType,
19   protocol::{
20     helpers::deserialize_skip_error,
21     public_key::PublicKey,
22     verification::verify_domains_match,
23   },
24 };
25 use chrono::{DateTime, FixedOffset};
26 use lemmy_api_common::{context::LemmyContext, utils::local_site_opt_to_slur_regex};
27 use lemmy_db_schema::{
28   newtypes::InstanceId,
29   source::community::{CommunityInsertForm, CommunityUpdateForm},
30   utils::naive_now,
31 };
32 use lemmy_utils::{
33   error::LemmyError,
34   utils::slurs::{check_slurs, check_slurs_opt},
35 };
36 use serde::{Deserialize, Serialize};
37 use serde_with::skip_serializing_none;
38 use std::fmt::Debug;
39 use url::Url;
40
41 #[skip_serializing_none]
42 #[derive(Clone, Debug, Deserialize, Serialize)]
43 #[serde(rename_all = "camelCase")]
44 pub struct Group {
45   #[serde(rename = "type")]
46   pub(crate) kind: GroupType,
47   pub(crate) id: ObjectId<ApubCommunity>,
48   /// username, set at account creation and usually fixed after that
49   pub(crate) preferred_username: String,
50   pub(crate) inbox: Url,
51   pub(crate) followers: Url,
52   pub(crate) public_key: PublicKey,
53
54   /// title
55   pub(crate) name: Option<String>,
56   pub(crate) summary: Option<String>,
57   #[serde(deserialize_with = "deserialize_skip_error", default)]
58   pub(crate) source: Option<Source>,
59   pub(crate) icon: Option<ImageObject>,
60   /// banner
61   pub(crate) image: Option<ImageObject>,
62   // lemmy extension
63   pub(crate) sensitive: Option<bool>,
64   #[serde(deserialize_with = "deserialize_skip_error", default)]
65   pub(crate) attributed_to: Option<CollectionId<ApubCommunityModerators>>,
66   // lemmy extension
67   pub(crate) posting_restricted_to_mods: Option<bool>,
68   pub(crate) outbox: CollectionId<ApubCommunityOutbox>,
69   pub(crate) endpoints: Option<Endpoints>,
70   pub(crate) featured: Option<CollectionId<ApubCommunityFeatured>>,
71   #[serde(default)]
72   pub(crate) language: Vec<LanguageTag>,
73   pub(crate) published: Option<DateTime<FixedOffset>>,
74   pub(crate) updated: Option<DateTime<FixedOffset>>,
75 }
76
77 impl Group {
78   pub(crate) async fn verify(
79     &self,
80     expected_domain: &Url,
81     context: &LemmyContext,
82   ) -> Result<(), LemmyError> {
83     let local_site_data = fetch_local_site_data(context.pool()).await?;
84
85     check_apub_id_valid_with_strictness(
86       self.id.inner(),
87       true,
88       &local_site_data,
89       context.settings(),
90     )?;
91     verify_domains_match(expected_domain, self.id.inner())?;
92
93     let slur_regex = &local_site_opt_to_slur_regex(&local_site_data.local_site);
94
95     check_slurs(&self.preferred_username, slur_regex)?;
96     check_slurs_opt(&self.name, slur_regex)?;
97     let description = read_from_string_or_source_opt(&self.summary, &None, &self.source);
98     check_slurs_opt(&description, slur_regex)?;
99     Ok(())
100   }
101
102   pub(crate) fn into_insert_form(self, instance_id: InstanceId) -> CommunityInsertForm {
103     CommunityInsertForm {
104       name: self.preferred_username.clone(),
105       title: self.name.unwrap_or(self.preferred_username),
106       description: read_from_string_or_source_opt(&self.summary, &None, &self.source),
107       removed: None,
108       published: self.published.map(|u| u.naive_local()),
109       updated: self.updated.map(|u| u.naive_local()),
110       deleted: Some(false),
111       nsfw: Some(self.sensitive.unwrap_or(false)),
112       actor_id: Some(self.id.into()),
113       local: Some(false),
114       private_key: None,
115       hidden: Some(false),
116       public_key: self.public_key.public_key_pem,
117       last_refreshed_at: Some(naive_now()),
118       icon: self.icon.map(|i| i.url.into()),
119       banner: self.image.map(|i| i.url.into()),
120       followers_url: Some(self.followers.into()),
121       inbox_url: Some(self.inbox.into()),
122       shared_inbox_url: self.endpoints.map(|e| e.shared_inbox.into()),
123       moderators_url: self.attributed_to.map(Into::into),
124       posting_restricted_to_mods: self.posting_restricted_to_mods,
125       instance_id,
126       featured_url: self.featured.map(Into::into),
127     }
128   }
129
130   pub(crate) fn into_update_form(self) -> CommunityUpdateForm {
131     CommunityUpdateForm {
132       title: Some(self.name.unwrap_or(self.preferred_username)),
133       description: Some(read_from_string_or_source_opt(
134         &self.summary,
135         &None,
136         &self.source,
137       )),
138       removed: None,
139       published: self.published.map(|u| u.naive_local()),
140       updated: Some(self.updated.map(|u| u.naive_local())),
141       deleted: None,
142       nsfw: Some(self.sensitive.unwrap_or(false)),
143       actor_id: Some(self.id.into()),
144       local: None,
145       private_key: None,
146       hidden: Some(false),
147       public_key: Some(self.public_key.public_key_pem),
148       last_refreshed_at: Some(naive_now()),
149       icon: Some(self.icon.map(|i| i.url.into())),
150       banner: Some(self.image.map(|i| i.url.into())),
151       followers_url: Some(self.followers.into()),
152       inbox_url: Some(self.inbox.into()),
153       shared_inbox_url: Some(self.endpoints.map(|e| e.shared_inbox.into())),
154       moderators_url: self.attributed_to.map(Into::into),
155       posting_restricted_to_mods: self.posting_restricted_to_mods,
156       featured_url: self.featured.map(Into::into),
157     }
158   }
159 }