summaryrefslogtreecommitdiffstats
path: root/server/src/api/community.rs
diff options
context:
space:
mode:
Diffstat (limited to 'server/src/api/community.rs')
-rw-r--r--server/src/api/community.rs559
1 files changed, 559 insertions, 0 deletions
diff --git a/server/src/api/community.rs b/server/src/api/community.rs
new file mode 100644
index 00000000..be4bb41a
--- /dev/null
+++ b/server/src/api/community.rs
@@ -0,0 +1,559 @@
+use super::*;
+use std::str::FromStr;
+
+#[derive(Serialize, Deserialize)]
+pub struct GetCommunity {
+ id: Option<i32>,
+ name: Option<String>,
+ auth: Option<String>
+}
+
+#[derive(Serialize, Deserialize)]
+pub struct GetCommunityResponse {
+ op: String,
+ community: CommunityView,
+ moderators: Vec<CommunityModeratorView>,
+ admins: Vec<UserView>,
+}
+
+
+#[derive(Serialize, Deserialize)]
+pub struct CreateCommunity {
+ name: String,
+ title: String,
+ description: Option<String>,
+ category_id: i32 ,
+ auth: String
+}
+
+#[derive(Serialize, Deserialize, Clone)]
+pub struct CommunityResponse {
+ op: String,
+ pub community: CommunityView
+}
+
+#[derive(Serialize, Deserialize)]
+pub struct ListCommunities {
+ sort: String,
+ page: Option<i64>,
+ limit: Option<i64>,
+ auth: Option<String>
+}
+
+#[derive(Serialize, Deserialize)]
+pub struct ListCommunitiesResponse {
+ op: String,
+ communities: Vec<CommunityView>
+}
+
+#[derive(Serialize, Deserialize, Clone)]
+pub struct BanFromCommunity {
+ pub community_id: i32,
+ user_id: i32,
+ ban: bool,
+ reason: Option<String>,
+ expires: Option<i64>,
+ auth: String
+}
+
+#[derive(Serialize, Deserialize)]
+pub struct BanFromCommunityResponse {
+ op: String,
+ user: UserView,
+ banned: bool,
+}
+
+#[derive(Serialize, Deserialize)]
+pub struct AddModToCommunity {
+ pub community_id: i32,
+ user_id: i32,
+ added: bool,
+ auth: String
+}
+
+#[derive(Serialize, Deserialize)]
+pub struct AddModToCommunityResponse {
+ op: String,
+ moderators: Vec<CommunityModeratorView>,
+}
+
+#[derive(Serialize, Deserialize)]
+pub struct EditCommunity {
+ pub edit_id: i32,
+ name: String,
+ title: String,
+ description: Option<String>,
+ category_id: i32,
+ removed: Option<bool>,
+ deleted: Option<bool>,
+ reason: Option<String>,
+ expires: Option<i64>,
+ auth: String
+}
+
+#[derive(Serialize, Deserialize)]
+pub struct FollowCommunity {
+ community_id: i32,
+ follow: bool,
+ auth: String
+}
+
+#[derive(Serialize, Deserialize)]
+pub struct GetFollowedCommunities {
+ auth: String
+}
+
+#[derive(Serialize, Deserialize)]
+pub struct GetFollowedCommunitiesResponse {
+ op: String,
+ communities: Vec<CommunityFollowerView>
+}
+
+impl Perform<GetCommunityResponse> for Oper<GetCommunity> {
+ fn perform(&self) -> Result<GetCommunityResponse, Error> {
+ let data: &GetCommunity = &self.data;
+ let conn = establish_connection();
+
+ let user_id: Option<i32> = 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
+ };
+
+ let community_id = match data.id {
+ Some(id) => id,
+ None => Community::read_from_name(&conn, data.name.to_owned().unwrap_or("main".to_string()))?.id
+ };
+
+ let community_view = match CommunityView::read(&conn, community_id, user_id) {
+ Ok(community) => community,
+ Err(_e) => {
+ return Err(APIError::err(&self.op, "Couldn't find Community"))?
+ }
+ };
+
+ let moderators = match CommunityModeratorView::for_community(&conn, community_id) {
+ Ok(moderators) => moderators,
+ Err(_e) => {
+ return Err(APIError::err(&self.op, "Couldn't find Community"))?
+ }
+ };
+
+ let admins = UserView::admins(&conn)?;
+
+ // Return the jwt
+ Ok(
+ GetCommunityResponse {
+ op: self.op.to_string(),
+ community: community_view,
+ moderators: moderators,
+ admins: admins,
+ }
+ )
+ }
+}
+
+impl Perform<CommunityResponse> for Oper<CreateCommunity> {
+ fn perform(&self) -> Result<CommunityResponse, Error> {
+ let data: &CreateCommunity = &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."))?
+ }
+ };
+
+ if has_slurs(&data.name) ||
+ has_slurs(&data.title) ||
+ (data.description.is_some() && has_slurs(&data.description.to_owned().unwrap())) {
+ return Err(APIError::err(&self.op, "No slurs"))?
+ }
+
+ let user_id = claims.id;
+
+ // Check for a site ban
+ if UserView::read(&conn, user_id)?.banned {
+ return Err(APIError::err(&self.op, "You have been banned from the site"))?
+ }
+
+ // When you create a community, make sure the user becomes a moderator and a follower
+ let community_form = CommunityForm {
+ name: data.name.to_owned(),
+ title: data.title.to_owned(),
+ description: data.description.to_owned(),
+ category_id: data.category_id,
+ creator_id: user_id,
+ removed: None,
+ deleted: None,
+ updated: None,
+ };
+
+ let inserted_community = match Community::create(&conn, &community_form) {
+ Ok(community) => community,
+ Err(_e) => {
+ return Err(APIError::err(&self.op, "Community already exists."))?
+ }
+ };
+
+ let community_moderator_form = CommunityModeratorForm {
+ community_id: inserted_community.id,
+ user_id: 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."))?
+ }
+ };
+
+ let community_follower_form = CommunityFollowerForm {
+ community_id: inserted_community.id,
+ user_id: 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."))?
+ }
+ };
+
+ let community_view = CommunityView::read(&conn, inserted_community.id, Some(user_id))?;
+
+ Ok(
+ CommunityResponse {
+ op: self.op.to_string(),
+ community: community_view
+ }
+ )
+ }
+}
+
+impl Perform<CommunityResponse> for Oper<EditCommunity> {
+ fn perform(&self) -> Result<CommunityResponse, Error> {
+ let data: &EditCommunity = &self.data;
+
+ if has_slurs(&data.name) || has_slurs(&data.title) {
+ return Err(APIError::err(&self.op, "No slurs"))?
+ }
+
+ 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;
+
+ // Check for a site ban
+ if UserView::read(&conn, user_id)?.banned {
+ return Err(APIError::err(&self.op, "You have been banned from the site"))?
+ }
+
+ // Verify its a mod
+ let mut editors: Vec<i32> = Vec::new();
+ editors.append(
+ &mut CommunityModeratorView::for_community(&conn, data.edit_id)
+ ?
+ .into_iter()
+ .map(|m| m.user_id)
+ .collect()
+ );
+ editors.append(
+ &mut UserView::admins(&conn)
+ ?
+ .into_iter()
+ .map(|a| a.id)
+ .collect()
+ );
+ if !editors.contains(&user_id) {
+ return Err(APIError::err(&self.op, "Not allowed to edit community"))?
+ }
+
+ let community_form = CommunityForm {
+ name: data.name.to_owned(),
+ title: data.title.to_owned(),
+ description: data.description.to_owned(),
+ category_id: data.category_id.to_owned(),
+ creator_id: user_id,
+ removed: data.removed.to_owned(),
+ deleted: data.deleted.to_owned(),
+ updated: Some(naive_now())
+ };
+
+ let _updated_community = match Community::update(&conn, data.edit_id, &community_form) {
+ Ok(community) => community,
+ Err(_e) => {
+ return Err(APIError::err(&self.op, "Couldn't update Community"))?
+ }
+ };
+
+ // Mod tables
+ if let Some(removed) = data.removed.to_owned() {
+ let expires = match data.expires {
+ Some(time) => Some(naive_from_unix(time)),
+ None => None
+ };
+ let form = ModRemoveCommunityForm {
+ mod_user_id: user_id,
+ community_id: data.edit_id,
+ removed: Some(removed),
+ reason: data.reason.to_owned(),
+ expires: expires
+ };
+ ModRemoveCommunity::create(&conn, &form)?;
+ }
+
+ let community_view = CommunityView::read(&conn, data.edit_id, Some(user_id))?;
+
+ Ok(
+ CommunityResponse {
+ op: self.op.to_string(),
+ community: community_view
+ }
+ )
+ }
+}
+
+impl Perform<ListCommunitiesResponse> for Oper<ListCommunities> {
+ fn perform(&self) -> Result<ListCommunitiesResponse, Error> {
+ let data: &ListCommunities = &self.data;
+ let conn = establish_connection();
+
+ let user_id: Option<i32> = 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
+ };
+
+ let sort = SortType::from_str(&data.sort)?;
+
+ let communities: Vec<CommunityView> = CommunityView::list(&conn, user_id, sort, data.page, data.limit)?;
+
+ // Return the jwt
+ Ok(
+ ListCommunitiesResponse {
+ op: self.op.to_string(),
+ communities: communities
+ }
+ )
+ }
+}
+
+
+impl Perform<CommunityResponse> for Oper<FollowCommunity> {
+ fn perform(&self) -> Result<CommunityResponse, Error> {
+ let data: &FollowCommunity = &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 community_follower_form = CommunityFollowerForm {
+ community_id: data.community_id,
+ user_id: user_id
+ };
+
+ if data.follow {
+ match CommunityFollower::follow(&conn, &community_follower_form) {
+ Ok(user) => user,
+ Err(_e) => {
+ return Err(APIError::err(&self.op, "Community follower already exists."))?
+ }
+ };
+ } else {
+ match CommunityFollower::ignore(&conn, &community_follower_form) {
+ Ok(user) => user,
+ Err(_e) => {
+ return Err(APIError::err(&self.op, "Community follower already exists."))?
+ }
+ };
+ }
+
+ let community_view = CommunityView::read(&conn, data.community_id, Some(user_id))?;
+
+ Ok(
+ CommunityResponse {
+ op: self.op.to_string(),
+ community: community_view
+ }
+ )
+ }
+}
+
+
+impl Perform<GetFollowedCommunitiesResponse> for Oper<GetFollowedCommunities> {
+ fn perform(&self) -> Result<GetFollowedCommunitiesResponse, Error> {
+ let data: &GetFollowedCommunities = &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 communities: Vec<CommunityFollowerView> = match CommunityFollowerView::for_user(&conn, user_id) {
+ Ok(communities) => communities,
+ Err(_e) => {
+ return Err(APIError::err(&self.op, "System error, try logging out and back in."))?
+ }
+ };
+
+ // Return the jwt
+ Ok(
+ GetFollowedCommunitiesResponse {
+ op: self.op.to_string(),
+ communities: communities
+ }
+ )
+ }
+}
+
+
+impl Perform<BanFromCommunityResponse> for Oper<BanFromCommunity> {
+ fn perform(&self) -> Result<BanFromCommunityResponse, Error> {
+ let data: &BanFromCommunity = &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 community_user_ban_form = CommunityUserBanForm {
+ community_id: data.community_id,
+ user_id: data.user_id,
+ };
+
+ if data.ban {
+ match CommunityUserBan::ban(&conn, &community_user_ban_form) {
+ Ok(user) => user,
+ Err(_e) => {
+ return Err(APIError::err(&self.op, "Community user ban already exists"))?
+ }
+ };
+ } else {
+ match CommunityUserBan::unban(&conn, &community_user_ban_form) {
+ Ok(user) => user,
+ Err(_e) => {
+ return Err(APIError::err(&self.op, "Community user ban already exists"))?
+ }
+ };
+ }
+
+ // Mod tables
+ let expires = match data.expires {
+ Some(time) => Some(naive_from_unix(time)),
+ None => None
+ };
+
+ let form = ModBanFromCommunityForm {
+ mod_user_id: user_id,
+ other_user_id: data.user_id,
+ community_id: data.community_id,
+ reason: data.reason.to_owned(),
+ banned: Some(data.ban),
+ expires: expires,
+ };
+ ModBanFromCommunity::create(&conn, &form)?;
+
+ let user_view = UserView::read(&conn, data.user_id)?;
+
+ Ok(
+ BanFromCommunityResponse {
+ op: self.op.to_string(),
+ user: user_view,
+ banned: data.ban
+ }
+ )
+ }
+}
+
+impl Perform<AddModToCommunityResponse> for Oper<AddModToCommunity> {
+ fn perform(&self) -> Result<AddModToCommunityResponse, Error> {
+ let data: &AddModToCommunity = &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 community_moderator_form = CommunityModeratorForm {
+ community_id: data.community_id,
+ user_id: data.user_id
+ };
+
+ if data.added {
+ match CommunityModerator::join(&conn, &community_moderator_form) {
+ Ok(user) => user,
+ Err(_e) => {
+ return Err(APIError::err(&self.op, "Community moderator already exists."))?
+ }
+ };
+ } else {
+ match CommunityModerator::leave(&conn, &community_moderator_form) {
+ Ok(user) => user,
+ Err(_e) => {
+ return Err(APIError::err(&self.op, "Community moderator already exists."))?
+ }
+ };
+ }
+
+ // Mod tables
+ let form = ModAddCommunityForm {
+ mod_user_id: user_id,
+ other_user_id: data.user_id,
+ community_id: data.community_id,
+ removed: Some(!data.added),
+ };
+ ModAddCommunity::create(&conn, &form)?;
+
+ let moderators = CommunityModeratorView::for_community(&conn, data.community_id)?;
+
+ Ok(
+ AddModToCommunityResponse {
+ op: self.op.to_string(),
+ moderators: moderators,
+ }
+ )
+ }
+}