mirror of
https://github.com/teloxide/teloxide.git
synced 2025-01-03 17:52:12 +01:00
Merge pull request #598 from teloxide/purchase-bot
Add `examples/purchase.rs`
This commit is contained in:
commit
0dd7aa3dab
7 changed files with 201 additions and 19 deletions
|
@ -6,6 +6,11 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
|||
|
||||
## unreleased
|
||||
|
||||
### Added
|
||||
|
||||
- Implement `GetChatId` for `Update`.
|
||||
- The `dialogue::enter()` function as a shortcut for `dptree::entry().enter_dialogue()`.
|
||||
|
||||
## 0.8.0 - 2022-04-18
|
||||
|
||||
### Removed
|
||||
|
|
|
@ -152,3 +152,7 @@ required-features = ["macros"]
|
|||
[[example]]
|
||||
name = "ngrok_ping_pong"
|
||||
required-features = ["webhooks-axum"]
|
||||
|
||||
[[example]]
|
||||
name = "purchase"
|
||||
required-features = ["macros"]
|
||||
|
|
|
@ -334,6 +334,10 @@ Associated links:
|
|||
- [Marvin's Marvellous Guide to All Things Webhook](https://core.telegram.org/bots/webhooks)
|
||||
- [Using self-signed certificates](https://core.telegram.org/bots/self-signed)
|
||||
|
||||
**Q: Can I handle both callback queries and messages within a single dialogue?**
|
||||
|
||||
A: Yes, see [`examples/purchase.rs`](examples/purchase.rs).
|
||||
|
||||
## Community bots
|
||||
|
||||
Feel free to propose your own bot to our collection!
|
||||
|
|
144
examples/purchase.rs
Normal file
144
examples/purchase.rs
Normal file
|
@ -0,0 +1,144 @@
|
|||
// This example demonstrates how to deal with messages and callback queries
|
||||
// within a single dialogue.
|
||||
//
|
||||
// # Example
|
||||
// ```
|
||||
// - /start
|
||||
// - Let's start! What's your full name?
|
||||
// - John Doe
|
||||
// - Select a product:
|
||||
// [Apple, Banana, Orange, Potato]
|
||||
// - <A user selects "Banana">
|
||||
// - John Doe, product 'Banana' has been purchased successfully!
|
||||
// ```
|
||||
|
||||
use teloxide::{
|
||||
dispatching::dialogue::{self, GetChatId, InMemStorage},
|
||||
prelude::*,
|
||||
types::{InlineKeyboardButton, InlineKeyboardMarkup},
|
||||
utils::command::BotCommands,
|
||||
};
|
||||
|
||||
type MyDialogue = Dialogue<State, InMemStorage<State>>;
|
||||
type HandlerResult = Result<(), Box<dyn std::error::Error + Send + Sync>>;
|
||||
|
||||
#[derive(Clone)]
|
||||
pub enum State {
|
||||
Start,
|
||||
ReceiveFullName,
|
||||
ReceiveProductChoice { full_name: String },
|
||||
}
|
||||
|
||||
impl Default for State {
|
||||
fn default() -> Self {
|
||||
Self::Start
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(BotCommands, Clone)]
|
||||
#[command(rename = "lowercase", description = "These commands are supported:")]
|
||||
enum Command {
|
||||
#[command(description = "display this text.")]
|
||||
Help,
|
||||
#[command(description = "start the purchase procedure.")]
|
||||
Start,
|
||||
}
|
||||
|
||||
#[tokio::main]
|
||||
async fn main() {
|
||||
pretty_env_logger::init();
|
||||
log::info!("Starting dialogue_bot...");
|
||||
|
||||
let bot = Bot::from_env().auto_send();
|
||||
|
||||
Dispatcher::builder(
|
||||
bot,
|
||||
dialogue::enter::<Update, InMemStorage<State>, State, _>()
|
||||
.branch(
|
||||
Update::filter_message()
|
||||
.branch(teloxide::handler![State::ReceiveFullName].endpoint(receive_full_name))
|
||||
.branch(dptree::entry().filter_command::<Command>().endpoint(handle_command))
|
||||
.branch(dptree::endpoint(invalid_state)),
|
||||
)
|
||||
.branch(
|
||||
Update::filter_callback_query().chain(
|
||||
teloxide::handler![State::ReceiveProductChoice { full_name }]
|
||||
.endpoint(receive_product_selection),
|
||||
),
|
||||
),
|
||||
)
|
||||
.dependencies(dptree::deps![InMemStorage::<State>::new()])
|
||||
.build()
|
||||
.setup_ctrlc_handler()
|
||||
.dispatch()
|
||||
.await;
|
||||
}
|
||||
|
||||
async fn handle_command(
|
||||
bot: AutoSend<Bot>,
|
||||
msg: Message,
|
||||
cmd: Command,
|
||||
dialogue: MyDialogue,
|
||||
) -> HandlerResult {
|
||||
match cmd {
|
||||
Command::Help => {
|
||||
bot.send_message(msg.chat.id, Command::descriptions().to_string()).await?;
|
||||
}
|
||||
Command::Start => {
|
||||
bot.send_message(msg.chat.id, "Let's start! What's your full name?").await?;
|
||||
dialogue.update(State::ReceiveFullName).await?;
|
||||
}
|
||||
}
|
||||
|
||||
Ok(())
|
||||
}
|
||||
|
||||
async fn receive_full_name(
|
||||
bot: AutoSend<Bot>,
|
||||
msg: Message,
|
||||
dialogue: MyDialogue,
|
||||
) -> HandlerResult {
|
||||
match msg.text().map(ToOwned::to_owned) {
|
||||
Some(full_name) => {
|
||||
let products = InlineKeyboardMarkup::default().append_row(
|
||||
vec!["Apple", "Banana", "Orange", "Potato"].into_iter().map(|product| {
|
||||
InlineKeyboardButton::callback(product.to_owned(), product.to_owned())
|
||||
}),
|
||||
);
|
||||
|
||||
bot.send_message(msg.chat.id, "Select a product:").reply_markup(products).await?;
|
||||
dialogue.update(State::ReceiveProductChoice { full_name }).await?;
|
||||
}
|
||||
None => {
|
||||
bot.send_message(msg.chat.id, "Please, send me your full name.").await?;
|
||||
}
|
||||
}
|
||||
|
||||
Ok(())
|
||||
}
|
||||
|
||||
async fn receive_product_selection(
|
||||
bot: AutoSend<Bot>,
|
||||
q: CallbackQuery,
|
||||
dialogue: MyDialogue,
|
||||
full_name: String,
|
||||
) -> HandlerResult {
|
||||
if let Some(product) = &q.data {
|
||||
if let Some(chat_id) = q.chat_id() {
|
||||
bot.send_message(
|
||||
chat_id,
|
||||
format!("{full_name}, product '{product}' has been purchased successfully!"),
|
||||
)
|
||||
.await?;
|
||||
dialogue.exit().await?;
|
||||
}
|
||||
}
|
||||
|
||||
Ok(())
|
||||
}
|
||||
|
||||
async fn invalid_state(bot: AutoSend<Bot>, msg: Message) -> HandlerResult {
|
||||
bot.send_message(msg.chat.id, "Unable to handle the message. Type /help to see the usage.")
|
||||
.await?;
|
||||
Ok(())
|
||||
}
|
|
@ -1,5 +1,4 @@
|
|||
use crate::types::CallbackQuery;
|
||||
use teloxide_core::types::{ChatId, Message};
|
||||
use crate::types::{CallbackQuery, ChatId, Message, Update};
|
||||
|
||||
/// Something that may has a chat ID.
|
||||
pub trait GetChatId {
|
||||
|
@ -18,3 +17,9 @@ impl GetChatId for CallbackQuery {
|
|||
self.message.as_ref().map(|mes| mes.chat.id)
|
||||
}
|
||||
}
|
||||
|
||||
impl GetChatId for Update {
|
||||
fn chat_id(&self) -> Option<ChatId> {
|
||||
self.chat().map(|chat| chat.id)
|
||||
}
|
||||
}
|
||||
|
|
|
@ -83,11 +83,14 @@ pub use crate::dispatching::dialogue::{RedisStorage, RedisStorageError};
|
|||
#[cfg(feature = "sqlite-storage")]
|
||||
pub use crate::dispatching::dialogue::{SqliteStorage, SqliteStorageError};
|
||||
|
||||
use dptree::{prelude::DependencyMap, Handler};
|
||||
pub use get_chat_id::GetChatId;
|
||||
pub use storage::*;
|
||||
use teloxide_core::types::ChatId;
|
||||
|
||||
use std::{marker::PhantomData, sync::Arc};
|
||||
use std::{fmt::Debug, marker::PhantomData, sync::Arc};
|
||||
|
||||
use super::DpHandlerDescription;
|
||||
|
||||
mod get_chat_id;
|
||||
mod storage;
|
||||
|
@ -180,6 +183,37 @@ where
|
|||
}
|
||||
}
|
||||
|
||||
/// Enters a dialogue context.
|
||||
///
|
||||
/// A call to this function is the same as `dptree::entry().enter_dialogue()`.
|
||||
///
|
||||
/// See [`HandlerExt::enter_dialogue`].
|
||||
///
|
||||
/// [`HandlerExt::enter_dialogue`]: super::HandlerExt::enter_dialogue
|
||||
pub fn enter<Upd, S, D, Output>() -> Handler<'static, DependencyMap, Output, DpHandlerDescription>
|
||||
where
|
||||
S: Storage<D> + ?Sized + Send + Sync + 'static,
|
||||
<S as Storage<D>>::Error: Debug + Send,
|
||||
D: Default + Send + Sync + 'static,
|
||||
Upd: GetChatId + Clone + Send + Sync + 'static,
|
||||
Output: Send + Sync + 'static,
|
||||
{
|
||||
dptree::entry()
|
||||
.chain(dptree::filter_map(|storage: Arc<S>, upd: Upd| {
|
||||
let chat_id = upd.chat_id()?;
|
||||
Some(Dialogue::new(storage, chat_id))
|
||||
}))
|
||||
.chain(dptree::filter_map_async(|dialogue: Dialogue<D, S>| async move {
|
||||
match dialogue.get_or_default().await {
|
||||
Ok(dialogue) => Some(dialogue),
|
||||
Err(err) => {
|
||||
log::error!("dialogue.get_or_default() failed: {:?}", err);
|
||||
None
|
||||
}
|
||||
}
|
||||
}))
|
||||
}
|
||||
|
||||
/// Perform a dialogue FSM transition.
|
||||
///
|
||||
/// This macro expands to a [`dptree::Handler`] that filters your dialogue
|
||||
|
|
|
@ -1,8 +1,6 @@
|
|||
use std::sync::Arc;
|
||||
|
||||
use crate::{
|
||||
dispatching::{
|
||||
dialogue::{Dialogue, GetChatId, Storage},
|
||||
dialogue::{GetChatId, Storage},
|
||||
DpHandlerDescription,
|
||||
},
|
||||
types::{Me, Message},
|
||||
|
@ -82,19 +80,7 @@ where
|
|||
D: Default + Send + Sync + 'static,
|
||||
Upd: GetChatId + Clone + Send + Sync + 'static,
|
||||
{
|
||||
self.chain(dptree::filter_map(|storage: Arc<S>, upd: Upd| {
|
||||
let chat_id = upd.chat_id()?;
|
||||
Some(Dialogue::new(storage, chat_id))
|
||||
}))
|
||||
.chain(dptree::filter_map_async(|dialogue: Dialogue<D, S>| async move {
|
||||
match dialogue.get_or_default().await {
|
||||
Ok(dialogue) => Some(dialogue),
|
||||
Err(err) => {
|
||||
log::error!("dialogue.get_or_default() failed: {:?}", err);
|
||||
None
|
||||
}
|
||||
}
|
||||
}))
|
||||
self.chain(super::dialogue::enter::<Upd, S, D, Output>())
|
||||
}
|
||||
|
||||
#[allow(deprecated)]
|
||||
|
|
Loading…
Reference in a new issue