1 use crate::PerformCrud;
2 use actix_web::web::Data;
3 use lemmy_api_common::{
4 site::{EditSite, SiteResponse},
7 check_image_has_local_domain,
8 get_local_user_view_from_jwt,
10 site_description_length_check,
13 use lemmy_db_schema::{
15 local_user::LocalUser,
16 site::{Site, SiteForm},
19 utils::{diesel_option_overwrite, diesel_option_overwrite_to_url, naive_now},
22 use lemmy_db_views::structs::SiteView;
23 use lemmy_utils::{utils::check_slurs_opt, ConnectionId, LemmyError};
24 use lemmy_websocket::{messages::SendAllMessage, LemmyContext, UserOperationCrud};
25 use std::{default::Default, str::FromStr};
27 #[async_trait::async_trait(?Send)]
28 impl PerformCrud for EditSite {
29 type Response = SiteResponse;
31 #[tracing::instrument(skip(context, websocket_id))]
34 context: &Data<LemmyContext>,
35 websocket_id: Option<ConnectionId>,
36 ) -> Result<SiteResponse, LemmyError> {
37 let data: &EditSite = self;
39 get_local_user_view_from_jwt(&data.auth, context.pool(), context.secret()).await?;
41 // Make sure user is an admin
42 is_admin(&local_user_view)?;
44 let local_site = blocking(context.pool(), Site::read_local_site).await??;
46 let sidebar = diesel_option_overwrite(&data.sidebar);
47 let description = diesel_option_overwrite(&data.description);
48 let application_question = diesel_option_overwrite(&data.application_question);
49 let icon = diesel_option_overwrite_to_url(&data.icon)?;
50 let banner = diesel_option_overwrite_to_url(&data.banner)?;
52 check_slurs_opt(&data.name, &context.settings().slur_regex())?;
53 check_slurs_opt(&data.description, &context.settings().slur_regex())?;
54 check_image_has_local_domain(icon.as_ref().unwrap_or(&None))?;
55 check_image_has_local_domain(banner.as_ref().unwrap_or(&None))?;
57 if let Some(Some(desc)) = &description {
58 site_description_length_check(desc)?;
61 // Make sure if applications are required, that there is an application questionnaire
62 if data.require_application.unwrap_or(false)
63 && application_question.as_ref().unwrap_or(&None).is_none()
65 return Err(LemmyError::from_message("application_question_required"));
68 if let Some(default_post_listing_type) = &data.default_post_listing_type {
69 // only allow all or local as default listing types
70 let val = ListingType::from_str(default_post_listing_type);
71 if val != Ok(ListingType::All) && val != Ok(ListingType::Local) {
72 return Err(LemmyError::from_message(
73 "invalid_default_post_listing_type",
78 let site_form = SiteForm {
79 name: data.name.to_owned().unwrap_or(local_site.name),
84 updated: Some(naive_now()),
85 enable_downvotes: data.enable_downvotes,
86 open_registration: data.open_registration,
87 enable_nsfw: data.enable_nsfw,
88 community_creation_admin_only: data.community_creation_admin_only,
89 require_email_verification: data.require_email_verification,
90 require_application: data.require_application,
92 private_instance: data.private_instance,
93 default_theme: data.default_theme.clone(),
94 default_post_listing_type: data.default_post_listing_type.clone(),
98 let update_site = blocking(context.pool(), move |conn| {
99 Site::update(conn, local_site.id, &site_form)
102 .map_err(|e| LemmyError::from_error_message(e, "couldnt_update_site"))?;
104 // TODO can't think of a better way to do this.
105 // If the server suddenly requires email verification, or required applications, no old users
106 // will be able to log in. It really only wants this to be a requirement for NEW signups.
107 // So if it was set from false, to true, you need to update all current users columns to be verified.
109 if !local_site.require_application && update_site.require_application {
110 blocking(context.pool(), move |conn| {
111 LocalUser::set_all_users_registration_applications_accepted(conn)
114 .map_err(|e| LemmyError::from_error_message(e, "couldnt_set_all_registrations_accepted"))?;
117 if !local_site.require_email_verification && update_site.require_email_verification {
118 blocking(context.pool(), move |conn| {
119 LocalUser::set_all_users_email_verified(conn)
122 .map_err(|e| LemmyError::from_error_message(e, "couldnt_set_all_email_verified"))?;
125 let site_view = blocking(context.pool(), SiteView::read_local).await??;
127 let res = SiteResponse { site_view };
129 context.chat_server().do_send(SendAllMessage {
130 op: UserOperationCrud::EditSite,
131 response: res.clone(),