Compare commits

...

1 Commits

Author SHA1 Message Date
Azur bdfa7edf52 feature shadow ban 2026-05-14 15:18:02 +02:00
6 changed files with 218 additions and 7 deletions
Generated
+1
View File
@@ -1029,6 +1029,7 @@ dependencies = [
"poise", "poise",
"serenity", "serenity",
"sqlx", "sqlx",
"thiserror 2.0.18",
"tokio", "tokio",
"tracing-subscriber", "tracing-subscriber",
] ]
+1
View File
@@ -8,5 +8,6 @@ dotenvy = "0.15.7"
poise = "0.6.2" poise = "0.6.2"
serenity = { version = "0.12.5", features = ["cache"] } serenity = { version = "0.12.5", features = ["cache"] }
sqlx = { version = "0.8.6", features = ["any", "mysql", "runtime-tokio", "sqlite"] } sqlx = { version = "0.8.6", features = ["any", "mysql", "runtime-tokio", "sqlite"] }
thiserror = "2.0.18"
tokio = { version = "1.52.3", features = ["macros", "rt-multi-thread"] } tokio = { version = "1.52.3", features = ["macros", "rt-multi-thread"] }
tracing-subscriber = "0.3.23" tracing-subscriber = "0.3.23"
+17 -2
View File
@@ -1,6 +1,7 @@
-- Add migration script here -- Add migration script here
create TABLE `Guild` ( create TABLE `Guild` (
`id` INT UNSIGNED NOT NULL PRIMARY KEY `id` INT UNSIGNED NOT NULL PRIMARY KEY,
`shadow_ban_role` INT UNSIGNED
); );
create TABLE `User` ( create TABLE `User` (
@@ -30,4 +31,18 @@ create TABLE `Auto_Channel` (
`channel_id` INT UNSIGNED NOT NULL PRIMARY KEY, `channel_id` INT UNSIGNED NOT NULL PRIMARY KEY,
`category_id` INT UNSIGNED, `category_id` INT UNSIGNED,
FOREIGN KEY(channel_id) REFERENCES User(id) FOREIGN KEY(channel_id) REFERENCES User(id)
) );
create TABLE `Shadow_Ban` (
`id` INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
`guild_id` INT UNSIGNED NOT NULL,
`user_id` INT UNSIGNED NOT NULL,
FOREIGN KEY(guild_id) REFERENCES Guild(id),
FOREIGN KEY(user_id) REFERENCES User(id)
);
create TABLE `Shadow_Ban_Role` (
`shadow_ban_id` INTEGER NOT NULL,
`role_id` INT UNSIGNED NOT NULL,
FOREIGN KEY(shadow_ban_id) REFERENCES Shadow_Ban(id)
);
+166 -1
View File
@@ -1,4 +1,31 @@
use crate::{Context, Error}; use std::convert::Infallible;
use poise::CreateReply;
use serenity::{
builder::{CreateEmbed, EditRole},
cache::CacheRef,
futures::future::join_all,
http::Http,
model::{
self, Permissions,
channel::{Embed, PermissionOverwrite, PermissionOverwriteType},
guild::Guild,
id::{GuildId, RoleId},
user::User,
},
};
use thiserror::Error;
use tokio::join;
use crate::{Context, Error, database::DataBase};
#[derive(Error, Debug)]
pub enum BotError {
#[error("Error related to the database")]
DataBaseError(#[from] sqlx::Error),
#[error("Error related to serenity")]
SerenityError(#[from] serenity::Error),
}
/// ️ Display all commands and their uses /// ️ Display all commands and their uses
/// ///
@@ -20,3 +47,141 @@ pub async fn help(
.await?; .await?;
Ok(()) Ok(())
} }
#[poise::command(
slash_command,
subcommands("shadowban_ban", "shadowban_unban", "shadowban_setup")
)]
pub async fn shadowban(_ctx: Context<'_>) -> Result<(), Error> {
Ok(())
}
#[poise::command(
slash_command,
required_permissions = "BAN_MEMBERS",
rename = "ban",
guild_only
)]
/// Remove the access of all channels to an user
pub async fn shadowban_ban(
ctx: Context<'_>,
#[description = "The user to shadow ban"] user: User,
) -> Result<(), Error> {
if let Some(guild) = ctx.guild_id() {
let role_id = ctx.data().database.get_shadow_ban_role(guild).await;
let role = if let Some(role_id) = role_id {
role_id
} else {
shadowban_setup_internal(&ctx.data().database, ctx.http(), guild, None).await?
};
guild
.member(ctx.http(), user.id)
.await?
.add_role(ctx.http(), role)
.await?;
} else {
let embed = CreateEmbed::new()
.title("Error")
.description("This command must execute in a guild.");
ctx.send(CreateReply {
embeds: vec![embed],
..Default::default()
})
.await?;
}
Ok(())
}
/// Remove a previous shadow ban
#[poise::command(
slash_command,
required_permissions = "BAN_MEMBERS",
rename = "unban",
guild_only
)]
pub async fn shadowban_unban(
ctx: Context<'_>,
#[description = "The user to remove the shadow ban role"] user: User,
) -> Result<(), Error> {
if let Some(guild) = ctx.guild_id() {
let role_id = ctx.data().database.get_shadow_ban_role(guild).await;
let role = if let Some(role_id) = role_id {
role_id
} else {
shadowban_setup_internal(&ctx.data().database, ctx.http(), guild, None).await?
};
guild
.member(ctx.http(), user.id)
.await?
.remove_role(ctx.http(), role)
.await?;
} else {
let embed = CreateEmbed::new()
.title("Error")
.description("This command must execute in a guild.");
ctx.send(CreateReply {
embeds: vec![embed],
..Default::default()
})
.await?;
}
Ok(())
}
#[poise::command(
slash_command,
required_permissions = "BAN_MEMBERS",
rename = "setup",
guild_only
)]
pub async fn shadowban_setup(ctx: Context<'_>, role: Option<RoleId>) -> Result<(), Error> {
ctx.defer_ephemeral().await?;
if let Some(guild) = ctx.guild_id() {
shadowban_setup_internal(&ctx.data().database, ctx.http(), guild, role).await?;
} else {
let embed = CreateEmbed::new()
.title("Error")
.description("This command must execute in a guild.");
ctx.send(CreateReply {
embeds: vec![embed],
..Default::default()
})
.await?;
}
Ok(())
}
async fn shadowban_setup_internal(
database: &DataBase,
http: &Http,
guild: GuildId,
role: Option<RoleId>,
) -> Result<RoleId, BotError> {
let role_id = if let Some(role_id) = role {
role_id
} else {
let role = EditRole::new()
.colour(0x000000)
.name("Shadow Ban")
.hoist(true)
.mentionable(false)
.permissions(Permissions::empty());
let role = guild.create_role(http, role).await?;
role.id
};
database.set_shadow_ban_role(guild, role_id).await?;
let channels = guild.channels(http).await?;
let perms = PermissionOverwrite {
allow: Permissions::empty(),
deny: Permissions::VIEW_CHANNEL.union(Permissions::SEND_MESSAGES),
kind: PermissionOverwriteType::Role(role_id),
};
for (_, channel) in channels {
channel.create_permission(http, perms.clone()).await.ok();
}
Ok(role_id)
}
+29 -1
View File
@@ -1,9 +1,10 @@
use std::env::var; use std::env::var;
use serenity::model::id::{GuildId, RoleId};
use sqlx::any::install_default_drivers; use sqlx::any::install_default_drivers;
use sqlx::migrate::Migrator; use sqlx::migrate::Migrator;
use sqlx::sqlite::SqlitePoolOptions; use sqlx::sqlite::SqlitePoolOptions;
use sqlx::{Error, SqlitePool}; use sqlx::{Error, SqlitePool, query};
static MIGRATOR: Migrator = sqlx::migrate!(); static MIGRATOR: Migrator = sqlx::migrate!();
@@ -24,4 +25,31 @@ impl DataBase {
Ok(Self { pool: pool }) Ok(Self { pool: pool })
} }
pub async fn set_shadow_ban_role(&self, guild: GuildId, role: RoleId) -> Result<(), Error> {
let guild = guild.get() as i64;
let role = role.get() as i64;
query(
"INSERT INTO Guild (id, shadow_ban_role)
VALUES (?1,?2)
ON CONFLICT(id) DO UPDATE SET
shadow_ban_role = excluded.shadow_ban_role;",
)
.bind(guild)
.bind(role)
.execute(&self.pool)
.await?;
Ok(())
}
pub async fn get_shadow_ban_role(&self, guild: GuildId) -> Option<RoleId> {
let guild = guild.get() as i64;
let role = query!("SELECT (shadow_ban_role) FROM Guild where id = ?;", guild)
.fetch_one(&self.pool)
.await
.map(|x| x.shadow_ban_role)
.unwrap_or(None);
role.map(|x| RoleId::new(x as u64))
}
} }
+4 -3
View File
@@ -91,11 +91,12 @@ async fn main() {
let intents = let intents =
serenity::GatewayIntents::non_privileged() | serenity::GatewayIntents::MESSAGE_CONTENT; serenity::GatewayIntents::non_privileged() | serenity::GatewayIntents::MESSAGE_CONTENT;
let client = serenity::ClientBuilder::new(token, intents) let mut client = serenity::ClientBuilder::new(token, intents)
.framework(framework) .framework(framework)
.status(serenity::OnlineStatus::Idle) .status(serenity::OnlineStatus::Idle)
.activity(ActivityData::playing("next station ⭐ /help")) .activity(ActivityData::playing("next station ⭐ /help"))
.await; .await
.unwrap();
client.unwrap().start().await.unwrap() client.start().await.unwrap()
} }