Files
book_bot/src/bots/approved_bot/modules/book.rs

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,
}
}),
)
}