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