mirror of
https://github.com/flibusta-apps/book_bot.git
synced 2025-12-06 15:35:35 +01:00
343 lines
11 KiB
Rust
343 lines
11 KiB
Rust
use std::str::FromStr;
|
|
|
|
use regex::Regex;
|
|
use teloxide::{dispatching::UpdateFilterExt, dptree, prelude::*};
|
|
|
|
use crate::bots::approved_bot::{
|
|
services::{
|
|
book_library::{
|
|
formaters::Format, get_author_books, get_sequence_books, get_translator_books,
|
|
types::Page,
|
|
},
|
|
user_settings::get_user_or_default_lang_codes,
|
|
},
|
|
tools::filter_callback_query,
|
|
};
|
|
|
|
use super::utils::{
|
|
filter_command, generic_get_pagination_keyboard, CommandParse, GetPaginationCallbackData,
|
|
};
|
|
|
|
#[derive(Clone)]
|
|
pub enum BookCommand {
|
|
Author { id: u32 },
|
|
Translator { id: u32 },
|
|
Sequence { id: u32 },
|
|
}
|
|
|
|
impl CommandParse<Self> for BookCommand {
|
|
fn parse(s: &str, bot_name: &str) -> Result<Self, strum::ParseError> {
|
|
let re = Regex::new(r"^/(?P<an_type>a|t|s)_(?P<id>\d+)$").unwrap();
|
|
|
|
let full_bot_name = format!("@{bot_name}");
|
|
let after_replace = s.replace(&full_bot_name, "");
|
|
|
|
let caps = re.captures(&after_replace);
|
|
let caps = match caps {
|
|
Some(v) => v,
|
|
None => return Err(strum::ParseError::VariantNotFound),
|
|
};
|
|
|
|
let annotation_type = &caps["an_type"];
|
|
let id: u32 = caps["id"].parse().unwrap();
|
|
|
|
match annotation_type {
|
|
"a" => Ok(BookCommand::Author { id }),
|
|
"t" => Ok(BookCommand::Translator { id }),
|
|
"s" => Ok(BookCommand::Sequence { id }),
|
|
_ => Err(strum::ParseError::VariantNotFound),
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(Clone)]
|
|
pub enum BookCallbackData {
|
|
Author { id: u32, page: u32 },
|
|
Translator { id: u32, page: u32 },
|
|
Sequence { id: u32, page: u32 },
|
|
}
|
|
|
|
impl FromStr for BookCallbackData {
|
|
type Err = strum::ParseError;
|
|
|
|
fn from_str(s: &str) -> Result<Self, Self::Err> {
|
|
let re = Regex::new(r"^b(?P<an_type>a|t|s)_(?P<id>\d+)_(?P<page>\d+)$").unwrap();
|
|
|
|
let caps = re.captures(s);
|
|
let caps = match caps {
|
|
Some(v) => v,
|
|
None => return Err(strum::ParseError::VariantNotFound),
|
|
};
|
|
|
|
let annotation_type = &caps["an_type"];
|
|
let id = caps["id"].parse::<u32>().unwrap();
|
|
let page = caps["page"].parse::<u32>().unwrap();
|
|
|
|
match annotation_type {
|
|
"a" => Ok(BookCallbackData::Author { id, page }),
|
|
"t" => Ok(BookCallbackData::Translator { id, page }),
|
|
"s" => Ok(BookCallbackData::Sequence { id, page }),
|
|
_ => Err(strum::ParseError::VariantNotFound),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl ToString for BookCallbackData {
|
|
fn to_string(&self) -> String {
|
|
match self {
|
|
BookCallbackData::Author { id, page } => format!("ba_{id}_{page}"),
|
|
BookCallbackData::Translator { id, page } => format!("bt_{id}_{page}"),
|
|
BookCallbackData::Sequence { id, page } => format!("bs_{id}_{page}"),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl GetPaginationCallbackData for BookCallbackData {
|
|
fn get_pagination_callback_data(&self, target_page: u32) -> String {
|
|
match self {
|
|
BookCallbackData::Author { id, .. } => BookCallbackData::Author {
|
|
id: id.clone(),
|
|
page: target_page,
|
|
},
|
|
BookCallbackData::Translator { id, .. } => BookCallbackData::Translator {
|
|
id: id.clone(),
|
|
page: target_page,
|
|
},
|
|
BookCallbackData::Sequence { id, .. } => BookCallbackData::Sequence {
|
|
id: id.clone(),
|
|
page: target_page,
|
|
},
|
|
}
|
|
.to_string()
|
|
}
|
|
}
|
|
|
|
async fn send_book_handler<T, Fut>(
|
|
message: Message,
|
|
bot: AutoSend<Bot>,
|
|
command: BookCommand,
|
|
books_getter: fn(id: u32, page: u32, allowed_langs: Vec<String>) -> Fut,
|
|
) -> crate::bots::BotHandlerInternal
|
|
where
|
|
T: Format + Clone,
|
|
Fut: std::future::Future<Output = Result<Page<T>, Box<dyn std::error::Error + Send + Sync>>>,
|
|
{
|
|
let id = match command {
|
|
BookCommand::Author { id } => id,
|
|
BookCommand::Translator { id } => id,
|
|
BookCommand::Sequence { id } => id,
|
|
};
|
|
|
|
let chat_id = message.chat.id;
|
|
let user_id = message.reply_to_message().map(|message| message.from().map(|from| from.id)).unwrap_or(None);
|
|
|
|
let user_id = match user_id {
|
|
Some(v) => v,
|
|
None => {
|
|
return match bot
|
|
.send_message(chat_id, "Повторите запрос сначала")
|
|
.send()
|
|
.await
|
|
{
|
|
Ok(_) => Ok(()),
|
|
Err(err) => Err(Box::new(err)),
|
|
}
|
|
}
|
|
};
|
|
|
|
let allowed_langs = get_user_or_default_lang_codes(user_id).await;
|
|
|
|
let items_page = match books_getter(id, 1, allowed_langs.clone()).await {
|
|
Ok(v) => v,
|
|
Err(err) => {
|
|
match bot
|
|
.send_message(chat_id, "Ошибка! Попробуйте позже :(")
|
|
.send()
|
|
.await
|
|
{
|
|
Ok(_) => (),
|
|
Err(err) => log::error!("{:?}", err),
|
|
}
|
|
return Err(err);
|
|
}
|
|
};
|
|
|
|
if items_page.total_pages == 0 {
|
|
match bot.send_message(chat_id, "Книги не найдены!").send().await {
|
|
Ok(_) => (),
|
|
Err(err) => return Err(Box::new(err)),
|
|
};
|
|
};
|
|
|
|
let formated_items = items_page.format_items();
|
|
let total_pages = items_page.total_pages;
|
|
|
|
let footer = format!("\n\nСтраница 1/{total_pages}");
|
|
let message_text = format!("{formated_items}{footer}");
|
|
|
|
let callback_data = match command {
|
|
BookCommand::Author { id } => BookCallbackData::Author { id, page: 1 },
|
|
BookCommand::Translator { id } => BookCallbackData::Translator { id, page: 1 },
|
|
BookCommand::Sequence { id } => BookCallbackData::Sequence { id, page: 1 },
|
|
};
|
|
|
|
let keyboard = generic_get_pagination_keyboard(1, total_pages, callback_data, true);
|
|
|
|
match bot
|
|
.send_message(chat_id, message_text)
|
|
.reply_markup(keyboard)
|
|
.send()
|
|
.await
|
|
{
|
|
Ok(_) => Ok(()),
|
|
Err(err) => Err(Box::new(err)),
|
|
}
|
|
}
|
|
|
|
async fn send_pagination_book_handler<T, Fut>(
|
|
cq: CallbackQuery,
|
|
bot: AutoSend<Bot>,
|
|
callback_data: BookCallbackData,
|
|
books_getter: fn(id: u32, page: u32, allowed_langs: Vec<String>) -> Fut,
|
|
) -> crate::bots::BotHandlerInternal
|
|
where
|
|
T: Format + Clone,
|
|
Fut: std::future::Future<Output = Result<Page<T>, Box<dyn std::error::Error + Send + Sync>>>,
|
|
{
|
|
let (id, page) = match callback_data {
|
|
BookCallbackData::Author { id, page } => (id, page),
|
|
BookCallbackData::Translator { id, page } => (id, page),
|
|
BookCallbackData::Sequence { id, page } => (id, page),
|
|
};
|
|
|
|
let chat_id = cq.message.as_ref().map(|message| message.chat.id);
|
|
let user_id = cq.from.id;
|
|
let message_id = cq.message.as_ref().map(|message| message.id);
|
|
|
|
let (chat_id, message_id) = match (chat_id, message_id) {
|
|
(Some(chat_id), Some(message_id)) => (chat_id, message_id),
|
|
_ => {
|
|
return match chat_id {
|
|
Some(v) => match bot.send_message(v, "Повторите поиск сначала").send().await
|
|
{
|
|
Ok(_) => Ok(()),
|
|
Err(err) => Err(Box::new(err)),
|
|
},
|
|
None => return Ok(()),
|
|
}
|
|
}
|
|
};
|
|
|
|
let allowed_langs = get_user_or_default_lang_codes(user_id).await;
|
|
|
|
let mut items_page = match books_getter(id, page, allowed_langs.clone()).await {
|
|
Ok(v) => v,
|
|
Err(err) => {
|
|
match bot
|
|
.send_message(chat_id, "Ошибка! Попробуйте позже :(")
|
|
.send()
|
|
.await
|
|
{
|
|
Ok(_) => (),
|
|
Err(err) => log::error!("{:?}", err),
|
|
}
|
|
return Err(err);
|
|
}
|
|
};
|
|
|
|
if items_page.total_pages == 0 {
|
|
match bot.send_message(chat_id, "Книги не найдены!").send().await {
|
|
Ok(_) => (),
|
|
Err(err) => return Err(Box::new(err)),
|
|
};
|
|
};
|
|
|
|
if page > items_page.total_pages {
|
|
items_page = match books_getter(id, items_page.total_pages, allowed_langs.clone()).await {
|
|
Ok(v) => v,
|
|
Err(err) => {
|
|
match bot
|
|
.send_message(chat_id, "Ошибка! Попробуйте позже :(")
|
|
.send()
|
|
.await
|
|
{
|
|
Ok(_) => (),
|
|
Err(err) => log::error!("{:?}", err),
|
|
}
|
|
return Err(err);
|
|
}
|
|
};
|
|
}
|
|
|
|
let formated_items = items_page.format_items();
|
|
|
|
let total_pages = items_page.total_pages;
|
|
|
|
let footer = format!("\n\nСтраница {page}/{total_pages}");
|
|
let message_text = format!("{formated_items}{footer}");
|
|
|
|
let keyboard = generic_get_pagination_keyboard(page, total_pages, callback_data, true);
|
|
|
|
match bot
|
|
.edit_message_text(chat_id, message_id, message_text)
|
|
.reply_markup(keyboard)
|
|
.send()
|
|
.await
|
|
{
|
|
Ok(_) => Ok(()),
|
|
Err(err) => Err(Box::new(err)),
|
|
}
|
|
}
|
|
|
|
pub fn get_book_handler() -> crate::bots::BotHandler {
|
|
dptree::entry()
|
|
.branch(
|
|
Update::filter_message()
|
|
.chain(filter_command::<BookCommand>())
|
|
.endpoint(
|
|
|message: Message, bot: AutoSend<Bot>, command: BookCommand| async move {
|
|
match command {
|
|
BookCommand::Author { .. } => {
|
|
send_book_handler(
|
|
message,
|
|
bot,
|
|
command,
|
|
get_author_books,
|
|
)
|
|
.await
|
|
}
|
|
BookCommand::Translator { .. } => {
|
|
send_book_handler(
|
|
message,
|
|
bot,
|
|
command,
|
|
get_translator_books,
|
|
)
|
|
.await
|
|
}
|
|
BookCommand::Sequence { .. } => {
|
|
send_book_handler(
|
|
message,
|
|
bot,
|
|
command,
|
|
get_sequence_books,
|
|
)
|
|
.await
|
|
}
|
|
}
|
|
},
|
|
),
|
|
)
|
|
.branch(
|
|
Update::filter_callback_query()
|
|
.chain(filter_callback_query::<BookCallbackData>())
|
|
.endpoint(|cq: CallbackQuery, bot: AutoSend<Bot>, callback_data: BookCallbackData| async move {
|
|
match callback_data {
|
|
BookCallbackData::Author { .. } => send_pagination_book_handler(cq, bot, callback_data, get_author_books).await,
|
|
BookCallbackData::Translator { .. } => send_pagination_book_handler(cq, bot, callback_data, get_translator_books).await,
|
|
BookCallbackData::Sequence { .. } => send_pagination_book_handler(cq, bot, callback_data, get_sequence_books).await,
|
|
}
|
|
}),
|
|
)
|
|
}
|