Skip to content

Commit

Permalink
refactor moderation; change prefix to r
Browse files Browse the repository at this point in the history
  • Loading branch information
TheKodeToad committed Dec 6, 2023
1 parent 174d935 commit 20e2dbb
Show file tree
Hide file tree
Showing 3 changed files with 123 additions and 52 deletions.
4 changes: 2 additions & 2 deletions src/commands/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@ pub fn to_global_commands() -> Vec<Command<Data, Report>> {
general::say(),
general::stars(),
general::tag(),
moderation::ban_user(),
moderation::kick_user(),
moderation::ban(),
moderation::kick(),
]
}
169 changes: 120 additions & 49 deletions src/commands/moderation/actions.rs
Original file line number Diff line number Diff line change
@@ -1,54 +1,123 @@
use crate::{consts::COLORS, Context};

Check warning

Code scanning / clippy

unused import: consts::COLORS Warning

unused import: consts::COLORS

Check warning

Code scanning / clippy

unused import: consts::COLORS Warning

unused import: consts::COLORS

use color_eyre::eyre::{eyre, Result};
use poise::serenity_prelude::{CreateEmbed, User};
use poise::serenity_prelude::{
futures::TryFutureExt, CreateEmbed, CreateMessage, FutureExt, Guild, Timestamp, User, UserId,

Check warning

Code scanning / clippy

unused imports: CreateEmbed, User Warning

unused imports: CreateEmbed, User

Check warning

Code scanning / clippy

unused imports: CreateEmbed, User Warning

unused imports: CreateEmbed, User

Check warning

Code scanning / clippy

unused imports: CreateEmbed, User Warning

unused imports: CreateEmbed, User

Check warning

Code scanning / clippy

unused imports: CreateEmbed, User Warning

unused imports: CreateEmbed, User

Check warning

Code scanning / clippy

unused import: futures::TryFutureExt Warning

unused import: futures::TryFutureExt

Check warning

Code scanning / clippy

unused import: FutureExt Warning

unused import: FutureExt

Check warning

Code scanning / clippy

unused import: futures::TryFutureExt Warning

unused import: futures::TryFutureExt

Check warning

Code scanning / clippy

unused import: FutureExt Warning

unused import: FutureExt
};

fn create_moderation_embed(
title: String,
user: &User,
delete_messages_days: Option<u8>,
reason: Option<String>,
) -> impl FnOnce(&mut CreateEmbed) -> &mut CreateEmbed {
let fields = [
("User", format!("{} ({})", user.name, user.id), false),
("Reason", reason.unwrap_or("None".to_string()), false),
(
"Deleted messages",
format!("Last {} days", delete_messages_days.unwrap_or(0)),
false,
),
];
struct Action {
reason: String,
data: ActionData,
}

|e: &mut CreateEmbed| e.title(title).fields(fields).color(COLORS["red"])
enum ActionData {

Check warning

Code scanning / clippy

variant Timeout is never constructed Warning

variant Timeout is never constructed

Check warning

Code scanning / clippy

variant Timeout is never constructed Warning

variant Timeout is never constructed
Kick,
Ban { purge: u8 },
Timeout { until: Timestamp },

Check warning

Code scanning / clippy

variant Timeout is never constructed Warning

variant Timeout is never constructed

Check warning

Code scanning / clippy

variant Timeout is never constructed Warning

variant Timeout is never constructed
}

fn build_dm<'a, 'b>(
message: &'b mut CreateMessage<'a>,
guild: &Guild,
action: &Action,
) -> &'b mut CreateMessage<'a> {
let description = match &action.data {
ActionData::Kick => "kicked from".to_string(),
ActionData::Ban { purge: _ } => "banned from".to_string(),
ActionData::Timeout { until } => {
format!("timed out until <t:{}> in", until.unix_timestamp())
}
};
let guild_name = &guild.name;
let reason = &action.reason;
message.content(format!(
"You have been {description} {guild_name}.\nReason: {reason}"
))
}

async fn moderate(
ctx: &Context<'_>,
users: &Vec<UserId>,
action: &Action,
quiet: bool,
) -> Result<()> {
let guild = ctx
.guild()
.ok_or_else(|| eyre!("Couldn't get guild from message!"))?;
let reason = &action.reason;

let mut count = 0;

for user in users {
if quiet {
if let Ok(channel) = user.create_dm_channel(ctx.http()).await {
let _ = channel
.send_message(ctx.http(), |message| build_dm(message, &guild, action))
.await;
}
}

let success = match action.data {
ActionData::Kick => guild
.kick_with_reason(ctx.http(), user, reason)
.await
.is_ok(),

ActionData::Ban { purge } => guild
.ban_with_reason(ctx.http(), user, purge, reason)
.await
.is_ok(),

ActionData::Timeout { until } => guild
.edit_member(ctx.http(), user, |member| {
member.disable_communication_until_datetime(until)
})
.await
.is_ok(),
};
if success {
count += 1;
}
}

let total = users.len();
if count == total {
ctx.reply("✅ Done!").await?;
} else {
ctx.reply(format!("⚠️ {count}/{total} succeeded!"))
.await?;
}

Ok(())
}

/// Ban a user
#[poise::command(
slash_command,
prefix_command,
default_member_permissions = "BAN_MEMBERS",
required_permissions = "BAN_MEMBERS"
required_permissions = "BAN_MEMBERS",
aliases("ban")
)]
pub async fn ban_user(
pub async fn ban(
ctx: Context<'_>,
user: User,
delete_messages_days: Option<u8>,
users: Vec<UserId>,
purge: Option<u8>,
reason: Option<String>,
quiet: Option<bool>,
) -> Result<()> {
let days = delete_messages_days.unwrap_or(1);
let guild = ctx
.guild()
.ok_or_else(|| eyre!("Couldn't get guild from message; Unable to ban!"))?;

guild
.ban_with_reason(ctx, &user, days, reason.clone().unwrap_or_default())
.await?;

let embed = create_moderation_embed("User banned!".to_string(), &user, Some(days), reason);

ctx.send(|m| m.embed(embed)).await?;

Ok(())
moderate(
&ctx,
&users,
&Action {
reason: reason.unwrap_or_default(),
data: ActionData::Ban {
purge: purge.unwrap_or(0),
},
},
quiet.unwrap_or(false),
)
.await
}

/// Kick a user
Expand All @@ -58,18 +127,20 @@ pub async fn ban_user(
default_member_permissions = "KICK_MEMBERS",
required_permissions = "KICK_MEMBERS"
)]
pub async fn kick_user(ctx: Context<'_>, user: User, reason: Option<String>) -> Result<()> {
let guild = ctx
.guild()
.ok_or_else(|| eyre!("Couldn't get guild from message; Unable to ban!"))?;

guild
.kick_with_reason(ctx, &user, &reason.clone().unwrap_or_default())
.await?;

let embed = create_moderation_embed("User kicked!".to_string(), &user, None, reason);

ctx.send(|m| m.embed(embed)).await?;

Ok(())
pub async fn kick(
ctx: Context<'_>,
users: Vec<UserId>,
reason: Option<String>,
quiet: Option<bool>,
) -> Result<()> {
moderate(
&ctx,
&users,
&Action {
reason: reason.unwrap_or_default(),
data: ActionData::Kick {},
},
quiet.unwrap_or(false),
)
.await
}
2 changes: 1 addition & 1 deletion src/main.rs
Original file line number Diff line number Diff line change
Expand Up @@ -62,7 +62,7 @@ async fn main() -> Result<()> {
Box::pin(handlers::handle_event(ctx, event, framework, data))
},
prefix_options: PrefixFrameworkOptions {
prefix: Some("!".into()),
prefix: Some("r".into()),
edit_tracker: Some(EditTracker::for_timespan(Duration::from_secs(3600))),
..Default::default()
},
Expand Down

0 comments on commit 20e2dbb

Please sign in to comment.