forked from nutomic/lemmy
560 lines
14 KiB
Rust
560 lines
14 KiB
Rust
|
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)]
|
||
|
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,
|
||
|
}
|
||
|
)
|
||
|
}
|
||
|
}
|