use super::*; use diesel::PgConnection; use std::str::FromStr; #[derive(Serialize, Deserialize)] pub struct ListCategories; #[derive(Serialize, Deserialize)] pub struct ListCategoriesResponse { categories: Vec, } #[derive(Serialize, Deserialize)] pub struct Search { q: String, type_: String, community_id: Option, sort: String, page: Option, limit: Option, } #[derive(Serialize, Deserialize)] pub struct SearchResponse { type_: String, comments: Vec, posts: Vec, communities: Vec, users: Vec, } #[derive(Serialize, Deserialize)] pub struct GetModlog { mod_user_id: Option, community_id: Option, page: Option, limit: Option, } #[derive(Serialize, Deserialize)] pub struct GetModlogResponse { removed_posts: Vec, locked_posts: Vec, stickied_posts: Vec, removed_comments: Vec, removed_communities: Vec, banned_from_community: Vec, banned: Vec, added_to_community: Vec, added: Vec, } #[derive(Serialize, Deserialize)] pub struct CreateSite { name: String, description: Option, enable_downvotes: bool, open_registration: bool, enable_nsfw: bool, auth: String, } #[derive(Serialize, Deserialize)] pub struct EditSite { name: String, description: Option, enable_downvotes: bool, open_registration: bool, enable_nsfw: bool, auth: String, } #[derive(Serialize, Deserialize)] pub struct GetSite; #[derive(Serialize, Deserialize)] pub struct SiteResponse { site: SiteView, } #[derive(Serialize, Deserialize)] pub struct GetSiteResponse { site: Option, admins: Vec, banned: Vec, pub online: usize, } #[derive(Serialize, Deserialize)] pub struct TransferSite { user_id: i32, auth: String, } impl Perform for Oper { fn perform(&self, conn: &PgConnection) -> Result { let _data: &ListCategories = &self.data; let categories: Vec = Category::list_all(&conn)?; // Return the jwt Ok(ListCategoriesResponse { categories }) } } impl Perform for Oper { fn perform(&self, conn: &PgConnection) -> Result { let data: &GetModlog = &self.data; let removed_posts = ModRemovePostView::list( &conn, data.community_id, data.mod_user_id, data.page, data.limit, )?; let locked_posts = ModLockPostView::list( &conn, data.community_id, data.mod_user_id, data.page, data.limit, )?; let stickied_posts = ModStickyPostView::list( &conn, data.community_id, data.mod_user_id, data.page, data.limit, )?; let removed_comments = ModRemoveCommentView::list( &conn, data.community_id, data.mod_user_id, data.page, data.limit, )?; let banned_from_community = ModBanFromCommunityView::list( &conn, data.community_id, data.mod_user_id, data.page, data.limit, )?; let added_to_community = ModAddCommunityView::list( &conn, data.community_id, data.mod_user_id, data.page, data.limit, )?; // These arrays are only for the full modlog, when a community isn't given let (removed_communities, banned, added) = if data.community_id.is_none() { ( ModRemoveCommunityView::list(&conn, data.mod_user_id, data.page, data.limit)?, ModBanView::list(&conn, data.mod_user_id, data.page, data.limit)?, ModAddView::list(&conn, data.mod_user_id, data.page, data.limit)?, ) } else { (Vec::new(), Vec::new(), Vec::new()) }; // Return the jwt Ok(GetModlogResponse { removed_posts, locked_posts, stickied_posts, removed_comments, removed_communities, banned_from_community, banned, added_to_community, added, }) } } impl Perform for Oper { fn perform(&self, conn: &PgConnection) -> Result { let data: &CreateSite = &self.data; let claims = match Claims::decode(&data.auth) { Ok(claims) => claims.claims, Err(_e) => return Err(APIError::err("not_logged_in").into()), }; if has_slurs(&data.name) || (data.description.is_some() && has_slurs(&data.description.to_owned().unwrap())) { return Err(APIError::err("no_slurs").into()); } let user_id = claims.id; // Make sure user is an admin if !UserView::read(&conn, user_id)?.admin { return Err(APIError::err("not_an_admin").into()); } let site_form = SiteForm { name: data.name.to_owned(), description: data.description.to_owned(), creator_id: user_id, enable_downvotes: data.enable_downvotes, open_registration: data.open_registration, enable_nsfw: data.enable_nsfw, updated: None, }; match Site::create(&conn, &site_form) { Ok(site) => site, Err(_e) => return Err(APIError::err("site_already_exists").into()), }; let site_view = SiteView::read(&conn)?; Ok(SiteResponse { site: site_view }) } } impl Perform for Oper { fn perform(&self, conn: &PgConnection) -> Result { let data: &EditSite = &self.data; let claims = match Claims::decode(&data.auth) { Ok(claims) => claims.claims, Err(_e) => return Err(APIError::err("not_logged_in").into()), }; if has_slurs(&data.name) || (data.description.is_some() && has_slurs(&data.description.to_owned().unwrap())) { return Err(APIError::err("no_slurs").into()); } let user_id = claims.id; // Make sure user is an admin if !UserView::read(&conn, user_id)?.admin { return Err(APIError::err("not_an_admin").into()); } let found_site = Site::read(&conn, 1)?; let site_form = SiteForm { name: data.name.to_owned(), description: data.description.to_owned(), creator_id: found_site.creator_id, updated: Some(naive_now()), enable_downvotes: data.enable_downvotes, open_registration: data.open_registration, enable_nsfw: data.enable_nsfw, }; match Site::update(&conn, 1, &site_form) { Ok(site) => site, Err(_e) => return Err(APIError::err("couldnt_update_site").into()), }; let site_view = SiteView::read(&conn)?; Ok(SiteResponse { site: site_view }) } } impl Perform for Oper { fn perform(&self, conn: &PgConnection) -> Result { let _data: &GetSite = &self.data; // It can return a null site in order to redirect let site_view = match Site::read(&conn, 1) { Ok(_site) => Some(SiteView::read(&conn)?), Err(_e) => None, }; let mut admins = UserView::admins(&conn)?; if site_view.is_some() { let site_creator_id = site_view.to_owned().unwrap().creator_id; let creator_index = admins.iter().position(|r| r.id == site_creator_id).unwrap(); let creator_user = admins.remove(creator_index); admins.insert(0, creator_user); } let banned = UserView::banned(&conn)?; Ok(GetSiteResponse { site: site_view, admins, banned, online: 0, }) } } impl Perform for Oper { fn perform(&self, conn: &PgConnection) -> Result { let data: &Search = &self.data; let sort = SortType::from_str(&data.sort)?; let type_ = SearchType::from_str(&data.type_)?; let mut posts = Vec::new(); let mut comments = Vec::new(); let mut communities = Vec::new(); let mut users = Vec::new(); // TODO no clean / non-nsfw searching rn match type_ { SearchType::Posts => { posts = PostQueryBuilder::create(&conn) .sort(&sort) .show_nsfw(true) .for_community_id(data.community_id) .search_term(data.q.to_owned()) .page(data.page) .limit(data.limit) .list()?; } SearchType::Comments => { comments = CommentQueryBuilder::create(&conn) .sort(&sort) .search_term(data.q.to_owned()) .page(data.page) .limit(data.limit) .list()?; } SearchType::Communities => { communities = CommunityQueryBuilder::create(&conn) .sort(&sort) .search_term(data.q.to_owned()) .page(data.page) .limit(data.limit) .list()?; } SearchType::Users => { users = UserQueryBuilder::create(&conn) .sort(&sort) .search_term(data.q.to_owned()) .page(data.page) .limit(data.limit) .list()?; } SearchType::All => { posts = PostQueryBuilder::create(&conn) .sort(&sort) .show_nsfw(true) .for_community_id(data.community_id) .search_term(data.q.to_owned()) .page(data.page) .limit(data.limit) .list()?; comments = CommentQueryBuilder::create(&conn) .sort(&sort) .search_term(data.q.to_owned()) .page(data.page) .limit(data.limit) .list()?; communities = CommunityQueryBuilder::create(&conn) .sort(&sort) .search_term(data.q.to_owned()) .page(data.page) .limit(data.limit) .list()?; users = UserQueryBuilder::create(&conn) .sort(&sort) .search_term(data.q.to_owned()) .page(data.page) .limit(data.limit) .list()?; } SearchType::Url => { posts = PostQueryBuilder::create(&conn) .sort(&sort) .show_nsfw(true) .for_community_id(data.community_id) .url_search(data.q.to_owned()) .page(data.page) .limit(data.limit) .list()?; } }; // Return the jwt Ok(SearchResponse { type_: data.type_.to_owned(), comments, posts, communities, users, }) } } impl Perform for Oper { fn perform(&self, conn: &PgConnection) -> Result { let data: &TransferSite = &self.data; let claims = match Claims::decode(&data.auth) { Ok(claims) => claims.claims, Err(_e) => return Err(APIError::err("not_logged_in").into()), }; let user_id = claims.id; let read_site = Site::read(&conn, 1)?; // Make sure user is the creator if read_site.creator_id != user_id { return Err(APIError::err("not_an_admin").into()); } let site_form = SiteForm { name: read_site.name, description: read_site.description, creator_id: data.user_id, updated: Some(naive_now()), enable_downvotes: read_site.enable_downvotes, open_registration: read_site.open_registration, enable_nsfw: read_site.enable_nsfw, }; match Site::update(&conn, 1, &site_form) { Ok(site) => site, Err(_e) => return Err(APIError::err("couldnt_update_site").into()), }; // Mod tables let form = ModAddForm { mod_user_id: user_id, other_user_id: data.user_id, removed: Some(false), }; ModAdd::create(&conn, &form)?; let site_view = SiteView::read(&conn)?; let mut admins = UserView::admins(&conn)?; let creator_index = admins .iter() .position(|r| r.id == site_view.creator_id) .unwrap(); let creator_user = admins.remove(creator_index); admins.insert(0, creator_user); let banned = UserView::banned(&conn)?; Ok(GetSiteResponse { site: Some(site_view), admins, banned, online: 0, }) } }