]> Untitled Git - lemmy.git/blob - crates/api_crud/src/site/update.rs
Add cargo feature for building lemmy_api_common with mininum deps (#2243)
[lemmy.git] / crates / api_crud / src / site / update.rs
1 use crate::PerformCrud;
2 use actix_web::web::Data;
3 use lemmy_api_common::{
4   site::{EditSite, SiteResponse},
5   utils::{
6     blocking,
7     check_image_has_local_domain,
8     get_local_user_view_from_jwt,
9     is_admin,
10     site_description_length_check,
11   },
12 };
13 use lemmy_db_schema::{
14   source::{
15     local_user::LocalUser,
16     site::{Site, SiteForm},
17   },
18   traits::Crud,
19   utils::{diesel_option_overwrite, diesel_option_overwrite_to_url, naive_now, ListingType},
20 };
21 use lemmy_db_views::structs::SiteView;
22 use lemmy_utils::{utils::check_slurs_opt, ConnectionId, LemmyError};
23 use lemmy_websocket::{messages::SendAllMessage, LemmyContext, UserOperationCrud};
24 use std::{default::Default, str::FromStr};
25
26 #[async_trait::async_trait(?Send)]
27 impl PerformCrud for EditSite {
28   type Response = SiteResponse;
29
30   #[tracing::instrument(skip(context, websocket_id))]
31   async fn perform(
32     &self,
33     context: &Data<LemmyContext>,
34     websocket_id: Option<ConnectionId>,
35   ) -> Result<SiteResponse, LemmyError> {
36     let data: &EditSite = self;
37     let local_user_view =
38       get_local_user_view_from_jwt(&data.auth, context.pool(), context.secret()).await?;
39
40     // Make sure user is an admin
41     is_admin(&local_user_view)?;
42
43     let local_site = blocking(context.pool(), Site::read_local_site).await??;
44
45     let sidebar = diesel_option_overwrite(&data.sidebar);
46     let description = diesel_option_overwrite(&data.description);
47     let application_question = diesel_option_overwrite(&data.application_question);
48     let icon = diesel_option_overwrite_to_url(&data.icon)?;
49     let banner = diesel_option_overwrite_to_url(&data.banner)?;
50
51     check_slurs_opt(&data.name, &context.settings().slur_regex())?;
52     check_slurs_opt(&data.description, &context.settings().slur_regex())?;
53     check_image_has_local_domain(icon.as_ref().unwrap_or(&None))?;
54     check_image_has_local_domain(banner.as_ref().unwrap_or(&None))?;
55
56     if let Some(Some(desc)) = &description {
57       site_description_length_check(desc)?;
58     }
59
60     // Make sure if applications are required, that there is an application questionnaire
61     if data.require_application.unwrap_or(false)
62       && application_question.as_ref().unwrap_or(&None).is_none()
63     {
64       return Err(LemmyError::from_message("application_question_required"));
65     }
66
67     if let Some(default_post_listing_type) = &data.default_post_listing_type {
68       // only allow all or local as default listing types
69       let val = ListingType::from_str(default_post_listing_type);
70       if val != Ok(ListingType::All) && val != Ok(ListingType::Local) {
71         return Err(LemmyError::from_message(
72           "invalid_default_post_listing_type",
73         ));
74       }
75     }
76
77     let site_form = SiteForm {
78       name: data.name.to_owned().unwrap_or(local_site.name),
79       sidebar,
80       description,
81       icon,
82       banner,
83       updated: Some(naive_now()),
84       enable_downvotes: data.enable_downvotes,
85       open_registration: data.open_registration,
86       enable_nsfw: data.enable_nsfw,
87       community_creation_admin_only: data.community_creation_admin_only,
88       require_email_verification: data.require_email_verification,
89       require_application: data.require_application,
90       application_question,
91       private_instance: data.private_instance,
92       default_theme: data.default_theme.clone(),
93       default_post_listing_type: data.default_post_listing_type.clone(),
94       ..SiteForm::default()
95     };
96
97     let update_site = blocking(context.pool(), move |conn| {
98       Site::update(conn, local_site.id, &site_form)
99     })
100     .await?
101     .map_err(|e| LemmyError::from_error_message(e, "couldnt_update_site"))?;
102
103     // TODO can't think of a better way to do this.
104     // If the server suddenly requires email verification, or required applications, no old users
105     // will be able to log in. It really only wants this to be a requirement for NEW signups.
106     // So if it was set from false, to true, you need to update all current users columns to be verified.
107
108     if !local_site.require_application && update_site.require_application {
109       blocking(context.pool(), move |conn| {
110         LocalUser::set_all_users_registration_applications_accepted(conn)
111       })
112       .await?
113       .map_err(|e| LemmyError::from_error_message(e, "couldnt_set_all_registrations_accepted"))?;
114     }
115
116     if !local_site.require_email_verification && update_site.require_email_verification {
117       blocking(context.pool(), move |conn| {
118         LocalUser::set_all_users_email_verified(conn)
119       })
120       .await?
121       .map_err(|e| LemmyError::from_error_message(e, "couldnt_set_all_email_verified"))?;
122     }
123
124     let site_view = blocking(context.pool(), SiteView::read_local).await??;
125
126     let res = SiteResponse { site_view };
127
128     context.chat_server().do_send(SendAllMessage {
129       op: UserOperationCrud::EditSite,
130       response: res.clone(),
131       websocket_id,
132     });
133
134     Ok(res)
135   }
136 }