2021-10-12 20:10:19 +02:00
|
|
|
use std::error::Error;
|
2021-10-12 21:53:41 +02:00
|
|
|
use teloxide::{
|
|
|
|
payloads::SendMessageSetters,
|
2022-03-24 12:25:42 +01:00
|
|
|
prelude::*,
|
2021-10-16 10:07:55 +02:00
|
|
|
types::{
|
|
|
|
InlineKeyboardButton, InlineKeyboardMarkup, InlineQueryResultArticle, InputMessageContent,
|
|
|
|
InputMessageContentText,
|
|
|
|
},
|
2022-03-13 16:37:11 +01:00
|
|
|
utils::command::BotCommands,
|
2021-10-12 21:53:41 +02:00
|
|
|
};
|
2021-10-12 20:10:19 +02:00
|
|
|
|
2022-03-13 16:37:11 +01:00
|
|
|
#[derive(BotCommands)]
|
2021-10-12 21:53:41 +02:00
|
|
|
#[command(rename = "lowercase", description = "These commands are supported:")]
|
2021-10-12 21:55:36 +02:00
|
|
|
enum Command {
|
2021-10-12 21:53:41 +02:00
|
|
|
#[command(description = "Display this text")]
|
|
|
|
Help,
|
|
|
|
#[command(description = "Start")]
|
|
|
|
Start,
|
|
|
|
}
|
|
|
|
|
2022-03-24 12:31:01 +01:00
|
|
|
#[tokio::main]
|
|
|
|
async fn main() -> Result<(), Box<dyn Error>> {
|
|
|
|
pretty_env_logger::init();
|
2022-04-26 21:00:08 +02:00
|
|
|
log::info!("Starting buttons bot...");
|
2022-03-24 12:31:01 +01:00
|
|
|
|
|
|
|
let bot = Bot::from_env().auto_send();
|
|
|
|
|
|
|
|
let handler = dptree::entry()
|
|
|
|
.branch(Update::filter_message().endpoint(message_handler))
|
|
|
|
.branch(Update::filter_callback_query().endpoint(callback_handler))
|
|
|
|
.branch(Update::filter_inline_query().endpoint(inline_query_handler));
|
|
|
|
|
2022-07-21 10:36:57 +02:00
|
|
|
Dispatcher::builder(bot, handler).enable_ctrlc_handler().build().dispatch().await;
|
2022-03-24 12:31:01 +01:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2021-10-12 21:53:41 +02:00
|
|
|
/// Creates a keyboard made by buttons in a big column.
|
2021-10-16 09:12:21 +02:00
|
|
|
fn make_keyboard() -> InlineKeyboardMarkup {
|
2021-10-16 09:24:06 +02:00
|
|
|
let mut keyboard: Vec<Vec<InlineKeyboardButton>> = vec![];
|
2021-10-16 09:33:39 +02:00
|
|
|
|
2021-10-16 09:24:06 +02:00
|
|
|
let debian_versions = [
|
2021-10-12 21:53:41 +02:00
|
|
|
"Buzz", "Rex", "Bo", "Hamm", "Slink", "Potato", "Woody", "Sarge", "Etch", "Lenny",
|
|
|
|
"Squeeze", "Wheezy", "Jessie", "Stretch", "Buster", "Bullseye",
|
|
|
|
];
|
|
|
|
|
2021-10-16 09:24:06 +02:00
|
|
|
for versions in debian_versions.chunks(3) {
|
|
|
|
let row = versions
|
|
|
|
.iter()
|
|
|
|
.map(|&version| InlineKeyboardButton::callback(version.to_owned(), version.to_owned()))
|
|
|
|
.collect();
|
|
|
|
|
|
|
|
keyboard.push(row);
|
2021-10-12 21:53:41 +02:00
|
|
|
}
|
|
|
|
|
2021-10-16 09:24:06 +02:00
|
|
|
InlineKeyboardMarkup::new(keyboard)
|
2021-10-12 21:53:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Parse the text wrote on Telegram and check if that text is a valid command
|
|
|
|
/// or not, then match the command. If the command is `/start` it writes a
|
|
|
|
/// markup with the `InlineKeyboardMarkup`.
|
2021-10-12 21:56:49 +02:00
|
|
|
async fn message_handler(
|
2022-01-12 11:36:52 +01:00
|
|
|
m: Message,
|
|
|
|
bot: AutoSend<Bot>,
|
2021-10-12 21:53:41 +02:00
|
|
|
) -> Result<(), Box<dyn Error + Send + Sync>> {
|
2022-02-03 15:48:40 +01:00
|
|
|
if let Some(text) = m.text() {
|
2022-03-13 16:37:11 +01:00
|
|
|
match BotCommands::parse(text, "buttons") {
|
2022-02-03 15:48:40 +01:00
|
|
|
Ok(Command::Help) => {
|
|
|
|
// Just send the description of all commands.
|
2022-03-14 15:11:57 +01:00
|
|
|
bot.send_message(m.chat.id, Command::descriptions().to_string()).await?;
|
2022-02-03 15:48:40 +01:00
|
|
|
}
|
|
|
|
Ok(Command::Start) => {
|
|
|
|
// Create a list of buttons and send them.
|
|
|
|
let keyboard = make_keyboard();
|
|
|
|
bot.send_message(m.chat.id, "Debian versions:").reply_markup(keyboard).await?;
|
|
|
|
}
|
2021-10-16 09:33:39 +02:00
|
|
|
|
2022-02-03 15:48:40 +01:00
|
|
|
Err(_) => {
|
|
|
|
bot.send_message(m.chat.id, "Command not found!").await?;
|
2021-10-12 21:53:41 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2021-10-16 10:07:55 +02:00
|
|
|
async fn inline_query_handler(
|
2022-01-12 11:36:52 +01:00
|
|
|
q: InlineQuery,
|
|
|
|
bot: AutoSend<Bot>,
|
2021-10-16 10:07:55 +02:00
|
|
|
) -> Result<(), Box<dyn Error + Send + Sync>> {
|
|
|
|
let choose_debian_version = InlineQueryResultArticle::new(
|
|
|
|
"0",
|
|
|
|
"Chose debian version",
|
|
|
|
InputMessageContent::Text(InputMessageContentText::new("Debian versions:")),
|
|
|
|
)
|
|
|
|
.reply_markup(make_keyboard());
|
|
|
|
|
2022-01-12 11:36:52 +01:00
|
|
|
bot.answer_inline_query(q.id, vec![choose_debian_version.into()]).await?;
|
2021-10-16 10:07:55 +02:00
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2021-10-12 21:50:37 +02:00
|
|
|
/// When it receives a callback from a button it edits the message with all
|
|
|
|
/// those buttons writing a text with the selected Debian version.
|
2022-01-12 11:36:52 +01:00
|
|
|
///
|
|
|
|
/// **IMPORTANT**: do not send privacy-sensitive data this way!!!
|
2022-02-04 11:46:26 +01:00
|
|
|
/// Anyone can read data stored in the callback button.
|
2021-10-13 09:19:27 +02:00
|
|
|
async fn callback_handler(
|
2022-01-12 11:36:52 +01:00
|
|
|
q: CallbackQuery,
|
|
|
|
bot: AutoSend<Bot>,
|
2021-10-12 21:50:37 +02:00
|
|
|
) -> Result<(), Box<dyn Error + Send + Sync>> {
|
2022-01-12 11:36:52 +01:00
|
|
|
if let Some(version) = q.data {
|
2022-04-03 12:06:44 +02:00
|
|
|
let text = format!("You chose: {version}");
|
2021-10-16 09:33:39 +02:00
|
|
|
|
2022-01-12 11:36:52 +01:00
|
|
|
match q.message {
|
2021-10-16 10:07:55 +02:00
|
|
|
Some(Message { id, chat, .. }) => {
|
|
|
|
bot.edit_message_text(chat.id, id, text).await?;
|
|
|
|
}
|
|
|
|
None => {
|
2022-01-12 11:36:52 +01:00
|
|
|
if let Some(id) = q.inline_message_id {
|
|
|
|
bot.edit_message_text_inline(id, text).await?;
|
2021-10-16 10:07:55 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-10-16 09:33:39 +02:00
|
|
|
|
2021-10-12 21:50:37 +02:00
|
|
|
log::info!("You chose: {}", version);
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|