teloxide/examples/admin.rs

154 lines
4.2 KiB
Rust
Raw Normal View History

2021-07-05 22:16:29 +03:00
use std::{error::Error, str::FromStr};
2020-02-14 00:26:06 +06:00
2022-01-12 14:03:10 +03:00
use chrono::Duration;
use teloxide::{prelude2::*, types::ChatPermissions, utils::command::BotCommand};
2020-02-11 21:35:22 +02:00
2020-02-14 00:11:56 +06:00
// Derive BotCommand to parse text with a command into this enumeration.
//
// 1. rename = "lowercase" turns all the commands into lowercase letters.
// 2. `description = "..."` specifies a text before all the commands.
//
// That is, you can just call Command::descriptions() to get a description of
// your commands in this format:
// %GENERAL-DESCRIPTION%
// %PREFIX%%COMMAND% - %DESCRIPTION%
2021-12-26 19:59:32 +02:00
#[derive(BotCommand, Clone)]
2020-02-13 23:23:22 +06:00
#[command(
rename = "lowercase",
2020-06-24 20:43:21 +03:00
description = "Use commands in format /%command% %num% %unit%",
parse_with = "split"
2020-02-13 23:23:22 +06:00
)]
2020-02-11 21:35:22 +02:00
enum Command {
#[command(description = "kick user from chat.")]
Kick,
#[command(description = "ban user in chat.")]
2020-06-24 20:43:21 +03:00
Ban {
2021-07-05 22:16:29 +03:00
time: u64,
2020-06-24 20:43:21 +03:00
unit: UnitOfTime,
},
2020-02-11 21:35:22 +02:00
#[command(description = "mute user in chat.")]
2020-06-24 20:43:21 +03:00
Mute {
2021-07-05 22:16:29 +03:00
time: u64,
2020-06-24 20:43:21 +03:00
unit: UnitOfTime,
},
2020-02-11 21:35:22 +02:00
Help,
}
2021-12-26 19:59:32 +02:00
#[derive(Clone)]
2020-06-24 20:43:21 +03:00
enum UnitOfTime {
Seconds,
Minutes,
Hours,
2020-02-11 21:35:22 +02:00
}
2020-06-24 20:43:21 +03:00
impl FromStr for UnitOfTime {
type Err = &'static str;
fn from_str(s: &str) -> Result<Self, <Self as FromStr>::Err> {
match s {
"h" | "hours" => Ok(UnitOfTime::Hours),
"m" | "minutes" => Ok(UnitOfTime::Minutes),
"s" | "seconds" => Ok(UnitOfTime::Seconds),
_ => Err("Allowed units: h, m, s"),
}
2020-02-11 21:35:22 +02:00
}
}
2020-06-24 20:43:21 +03:00
// Calculates time of user restriction.
2021-07-05 22:16:29 +03:00
fn calc_restrict_time(time: u64, unit: UnitOfTime) -> Duration {
2020-06-24 20:43:21 +03:00
match unit {
2021-07-05 22:16:29 +03:00
UnitOfTime::Hours => Duration::hours(time as i64),
UnitOfTime::Minutes => Duration::minutes(time as i64),
UnitOfTime::Seconds => Duration::seconds(time as i64),
2020-06-24 20:43:21 +03:00
}
2020-02-11 21:35:22 +02:00
}
2022-02-04 19:09:32 +06:00
type Bot = AutoSend<teloxide::Bot>;
2020-02-14 00:11:56 +06:00
2021-12-26 19:59:32 +02:00
// Kick a user with a replied message.
2022-02-04 19:09:32 +06:00
async fn kick_user(bot: Bot, msg: Message) -> Result<(), Box<dyn Error + Send + Sync>> {
match msg.reply_to_message() {
2021-12-26 19:59:32 +02:00
Some(replied) => {
// bot.unban_chat_member can also kicks a user from a group chat.
2022-02-04 19:45:35 +06:00
bot.unban_chat_member(msg.chat.id, replied.from().unwrap().id).await?;
2020-06-24 20:43:21 +03:00
}
2020-02-13 17:42:24 +02:00
None => {
2022-02-04 19:45:35 +06:00
bot.send_message(msg.chat.id, "Use this command in reply to another message").await?;
2020-02-13 23:23:22 +06:00
}
2020-02-13 17:42:24 +02:00
}
Ok(())
}
2021-12-26 19:59:32 +02:00
// Mute a user with a replied message.
async fn mute_user(
2022-02-04 19:09:32 +06:00
bot: Bot,
msg: Message,
2021-12-26 19:59:32 +02:00
time: Duration,
) -> Result<(), Box<dyn Error + Send + Sync>> {
match msg.reply_to_message() {
2021-12-26 19:59:32 +02:00
Some(replied) => {
bot.restrict_chat_member(
2022-02-04 19:45:35 +06:00
msg.chat.id,
2021-12-26 19:59:32 +02:00
replied.from().expect("Must be MessageKind::Common").id,
2022-02-03 20:39:32 +06:00
ChatPermissions::empty(),
2021-12-26 19:59:32 +02:00
)
2022-02-03 20:39:32 +06:00
.until_date(msg.date + time)
2021-12-26 19:59:32 +02:00
.await?;
2020-02-13 23:23:22 +06:00
}
2020-02-13 17:42:24 +02:00
None => {
2022-02-04 19:45:35 +06:00
bot.send_message(msg.chat.id, "Use this command in a reply to another message!")
2021-12-26 19:59:32 +02:00
.await?;
2020-02-13 17:42:24 +02:00
}
}
Ok(())
}
2020-02-14 00:11:56 +06:00
// Ban a user with replied message.
2021-12-26 19:59:32 +02:00
async fn ban_user(
2022-02-04 19:09:32 +06:00
bot: Bot,
msg: Message,
2021-12-26 19:59:32 +02:00
time: Duration,
) -> Result<(), Box<dyn Error + Send + Sync>> {
match msg.reply_to_message() {
2021-12-26 19:59:32 +02:00
Some(replied) => {
bot.kick_chat_member(
2022-02-04 19:45:35 +06:00
msg.chat.id,
2021-12-26 19:59:32 +02:00
replied.from().expect("Must be MessageKind::Common").id,
)
2022-02-03 20:39:32 +06:00
.until_date(msg.date + time)
2021-12-26 19:59:32 +02:00
.await?;
2020-06-24 20:43:21 +03:00
}
2020-02-13 17:42:24 +02:00
None => {
2022-02-04 19:45:35 +06:00
bot.send_message(msg.chat.id, "Use this command in a reply to another message!")
2021-12-26 19:59:32 +02:00
.await?;
2020-02-13 23:23:22 +06:00
}
2020-02-13 17:42:24 +02:00
}
Ok(())
}
2021-12-26 19:59:32 +02:00
async fn action(
2022-02-04 19:09:32 +06:00
bot: Bot,
msg: Message,
2021-12-26 19:59:32 +02:00
command: Command,
) -> Result<(), Box<dyn Error + Send + Sync>> {
2020-02-19 04:54:41 +06:00
match command {
2021-12-26 19:59:32 +02:00
Command::Help => {
2022-02-04 19:45:35 +06:00
bot.send_message(msg.chat.id, Command::descriptions()).await?;
2021-12-26 19:59:32 +02:00
}
Command::Kick => kick_user(bot, msg).await?,
Command::Ban { time, unit } => ban_user(bot, msg, calc_restrict_time(time, unit)).await?,
Command::Mute { time, unit } => mute_user(bot, msg, calc_restrict_time(time, unit)).await?,
2020-02-19 04:54:41 +06:00
};
2020-02-11 21:35:22 +02:00
Ok(())
}
#[tokio::main]
async fn main() {
2020-02-13 23:23:22 +06:00
teloxide::enable_logging!();
log::info!("Starting admin_bot...");
2020-02-13 23:23:22 +06:00
2022-02-04 19:09:32 +06:00
let bot = teloxide::Bot::from_env().auto_send();
2020-02-13 23:23:22 +06:00
teloxide::repls2::commands_repl(bot, action, Command::ty()).await;
2020-02-11 21:35:22 +02:00
}