use super::*; use std::str::FromStr; use bcrypt::{verify}; #[derive(Serialize, Deserialize, Debug)] pub struct Login { username_or_email: String, password: String } #[derive(Serialize, Deserialize)] pub struct Register { username: String, email: Option, password: String, password_verify: String, admin: bool, spam_timeri: i64, } #[derive(Serialize, Deserialize)] pub struct LoginResponse { op: String, jwt: String } #[derive(Serialize, Deserialize)] pub struct GetUserDetails { user_id: Option, username: Option, sort: String, page: Option, limit: Option, community_id: Option, saved_only: bool, auth: Option, } #[derive(Serialize, Deserialize)] pub struct GetUserDetailsResponse { op: String, user: UserView, follows: Vec, moderates: Vec, comments: Vec, posts: Vec, } #[derive(Serialize, Deserialize)] pub struct GetRepliesResponse { op: String, replies: Vec, } #[derive(Serialize, Deserialize)] pub struct MarkAllAsRead { auth: String } #[derive(Serialize, Deserialize)] pub struct AddAdmin { user_id: i32, added: bool, auth: String } #[derive(Serialize, Deserialize)] pub struct AddAdminResponse { op: String, admins: Vec, } #[derive(Serialize, Deserialize)] pub struct BanUser { user_id: i32, ban: bool, reason: Option, expires: Option, auth: String } #[derive(Serialize, Deserialize)] pub struct BanUserResponse { op: String, user: UserView, banned: bool, } #[derive(Serialize, Deserialize)] pub struct GetReplies { sort: String, page: Option, limit: Option, unread_only: bool, auth: String } impl Perform for Oper { fn perform(&self) -> Result { let data: Login = self.data; let conn = establish_connection(); // Fetch that username / email let user: User_ = match User_::find_by_email_or_username(&conn, &data.username_or_email) { Ok(user) => user, Err(_e) => return Err(APIError::err(self.op, "Couldn't find that username or email"))? }; // Verify the password let valid: bool = verify(&data.password, &user.password_encrypted).unwrap_or(false); if !valid { return Err(APIError::err(self.op, "Password incorrect"))? } // Return the jwt Ok( LoginResponse { op: self.op.to_string(), jwt: user.jwt() } ) } } impl Perform for Oper { fn perform(&self) -> Result { let data: Register = self.data; let conn = establish_connection(); // Make sure passwords match if &data.password != &data.password_verify { return Err(APIError::err(self.op, "Passwords do not match."))? } if data.spam_timeri < 1142 { return Err(APIError::err(self.op, "Too fast"))? } if has_slurs(&data.username) { return Err(APIError::err(self.op, "No slurs"))? } // Make sure there are no admins if data.admin && UserView::admins(&conn)?.len() > 0 { return Err(APIError::err(self.op, "Sorry, there's already an admin."))? } // Register the new user let user_form = UserForm { name: data.username.to_owned(), fedi_name: Settings::get().hostname.into(), email: data.email.to_owned(), password_encrypted: data.password.to_owned(), preferred_username: None, updated: None, admin: data.admin, banned: false, }; // Create the user let inserted_user = match User_::register(&conn, &user_form) { Ok(user) => user, Err(_e) => { return Err(APIError::err(self.op, "User already exists."))? } }; // Sign them up for main community no matter what let community_follower_form = CommunityFollowerForm { community_id: 1, user_id: inserted_user.id, }; let _inserted_community_follower = match CommunityFollower::follow(&conn, &community_follower_form) { Ok(user) => user, Err(_e) => { return Err(APIError::err(self.op, "Community follower already exists."))? } }; // If its an admin, add them as a mod and follower to main if data.admin { let community_moderator_form = CommunityModeratorForm { community_id: 1, user_id: inserted_user.id, }; let _inserted_community_moderator = match CommunityModerator::join(&conn, &community_moderator_form) { Ok(user) => user, Err(_e) => { return Err(APIError::err(self.op, "Community moderator already exists."))? } }; } // Return the jwt Ok( LoginResponse { op: self.op.to_string(), jwt: inserted_user.jwt() } ) } } impl Perform for Oper { fn perform(&self) -> Result { let data: GetUserDetails = self.data; let conn = establish_connection(); let user_id: Option = match &data.auth { Some(auth) => { match Claims::decode(&auth) { Ok(claims) => { let user_id = claims.claims.id; Some(user_id) } Err(_e) => None } } None => None }; //TODO add save let sort = SortType::from_str(&data.sort)?; let user_details_id = match data.user_id { Some(id) => id, None => User_::read_from_name(&conn, data.username.to_owned().unwrap_or("admin".to_string()))?.id }; let user_view = UserView::read(&conn, user_details_id)?; // If its saved only, you don't care what creator it was let posts = if data.saved_only { PostView::list(&conn, PostListingType::All, &sort, data.community_id, None, None, Some(user_details_id), data.saved_only, false, data.page, data.limit)? } else { PostView::list(&conn, PostListingType::All, &sort, data.community_id, Some(user_details_id), None, user_id, data.saved_only, false, data.page, data.limit)? }; let comments = if data.saved_only { CommentView::list(&conn, &sort, None, None, None, Some(user_details_id), data.saved_only, data.page, data.limit)? } else { CommentView::list(&conn, &sort, None, Some(user_details_id), None, user_id, data.saved_only, data.page, data.limit)? }; let follows = CommunityFollowerView::for_user(&conn, user_details_id)?; let moderates = CommunityModeratorView::for_user(&conn, user_details_id)?; // Return the jwt Ok( GetUserDetailsResponse { op: self.op.to_string(), user: user_view, follows: follows, moderates: moderates, comments: comments, posts: posts, } ) } } impl Perform for Oper { fn perform(&self) -> Result { let data: AddAdmin = self.data; let conn = establish_connection(); let claims = match Claims::decode(&data.auth) { Ok(claims) => claims.claims, Err(_e) => { return Err(APIError::err(self.op, "Not logged in."))? } }; let user_id = claims.id; // Make sure user is an admin if UserView::read(&conn, user_id)?.admin == false { return Err(APIError::err(self.op, "Not an admin."))? } let read_user = User_::read(&conn, data.user_id)?; let user_form = UserForm { name: read_user.name, fedi_name: read_user.fedi_name, email: read_user.email, password_encrypted: read_user.password_encrypted, preferred_username: read_user.preferred_username, updated: Some(naive_now()), admin: data.added, banned: read_user.banned, }; match User_::update(&conn, data.user_id, &user_form) { Ok(user) => user, Err(_e) => { return Err(APIError::err(self.op, "Couldn't update user"))? } }; // Mod tables let form = ModAddForm { mod_user_id: user_id, other_user_id: data.user_id, removed: Some(!data.added), }; ModAdd::create(&conn, &form)?; let admins = UserView::admins(&conn)?; Ok( AddAdminResponse { op: self.op.to_string(), admins: admins, } ) } } impl Perform for Oper { fn perform(&self) -> Result { let data: BanUser = self.data; let conn = establish_connection(); let claims = match Claims::decode(&data.auth) { Ok(claims) => claims.claims, Err(_e) => { return Err(APIError::err(self.op, "Not logged in."))? } }; let user_id = claims.id; // Make sure user is an admin if UserView::read(&conn, user_id)?.admin == false { return Err(APIError::err(self.op, "Not an admin."))? } let read_user = User_::read(&conn, data.user_id)?; let user_form = UserForm { name: read_user.name, fedi_name: read_user.fedi_name, email: read_user.email, password_encrypted: read_user.password_encrypted, preferred_username: read_user.preferred_username, updated: Some(naive_now()), admin: read_user.admin, banned: data.ban, }; match User_::update(&conn, data.user_id, &user_form) { Ok(user) => user, Err(_e) => { return Err(APIError::err(self.op, "Couldn't update user"))? } }; // Mod tables let expires = match data.expires { Some(time) => Some(naive_from_unix(time)), None => None }; let form = ModBanForm { mod_user_id: user_id, other_user_id: data.user_id, reason: data.reason.to_owned(), banned: Some(data.ban), expires: expires, }; ModBan::create(&conn, &form)?; let user_view = UserView::read(&conn, data.user_id)?; Ok( BanUserResponse { op: self.op.to_string(), user: user_view, banned: data.ban } ) } } impl Perform for Oper { fn perform(&self) -> Result { let data: GetReplies = self.data; let conn = establish_connection(); let claims = match Claims::decode(&data.auth) { Ok(claims) => claims.claims, Err(_e) => { return Err(APIError::err(self.op, "Not logged in."))? } }; let user_id = claims.id; let sort = SortType::from_str(&data.sort)?; let replies = ReplyView::get_replies(&conn, user_id, &sort, data.unread_only, data.page, data.limit)?; // Return the jwt Ok( GetRepliesResponse { op: self.op.to_string(), replies: replies, } ) } } impl Perform for Oper { fn perform(&self) -> Result { let data: MarkAllAsRead = self.data; let conn = establish_connection(); let claims = match Claims::decode(&data.auth) { Ok(claims) => claims.claims, Err(_e) => { return Err(APIError::err(self.op, "Not logged in."))? } }; let user_id = claims.id; let replies = ReplyView::get_replies(&conn, user_id, &SortType::New, true, Some(1), Some(999))?; for reply in &replies { let comment_form = CommentForm { content: reply.to_owned().content, parent_id: reply.to_owned().parent_id, post_id: reply.to_owned().post_id, creator_id: reply.to_owned().creator_id, removed: None, deleted: None, read: Some(true), updated: reply.to_owned().updated }; let _updated_comment = match Comment::update(&conn, reply.id, &comment_form) { Ok(comment) => comment, Err(_e) => { return Err(APIError::err(self.op, "Couldn't update Comment"))? } }; } let replies = ReplyView::get_replies(&conn, user_id, &SortType::New, true, Some(1), Some(999))?; Ok( GetRepliesResponse { op: self.op.to_string(), replies: replies, } ) } }