mirror of
https://github.com/teloxide/teloxide.git
synced 2025-03-22 06:45:37 +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
|
## unreleased
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- Implement `GetChatId` for `Update`.
|
||||||
|
- The `dialogue::enter()` function as a shortcut for `dptree::entry().enter_dialogue()`.
|
||||||
|
|
||||||
## 0.8.0 - 2022-04-18
|
## 0.8.0 - 2022-04-18
|
||||||
|
|
||||||
### Removed
|
### Removed
|
||||||
|
|
|
@ -152,3 +152,7 @@ required-features = ["macros"]
|
||||||
[[example]]
|
[[example]]
|
||||||
name = "ngrok_ping_pong"
|
name = "ngrok_ping_pong"
|
||||||
required-features = ["webhooks-axum"]
|
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)
|
- [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)
|
- [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
|
## Community bots
|
||||||
|
|
||||||
Feel free to propose your own bot to our collection!
|
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 crate::types::{CallbackQuery, ChatId, Message, Update};
|
||||||
use teloxide_core::types::{ChatId, Message};
|
|
||||||
|
|
||||||
/// Something that may has a chat ID.
|
/// Something that may has a chat ID.
|
||||||
pub trait GetChatId {
|
pub trait GetChatId {
|
||||||
|
@ -18,3 +17,9 @@ impl GetChatId for CallbackQuery {
|
||||||
self.message.as_ref().map(|mes| mes.chat.id)
|
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")]
|
#[cfg(feature = "sqlite-storage")]
|
||||||
pub use crate::dispatching::dialogue::{SqliteStorage, SqliteStorageError};
|
pub use crate::dispatching::dialogue::{SqliteStorage, SqliteStorageError};
|
||||||
|
|
||||||
|
use dptree::{prelude::DependencyMap, Handler};
|
||||||
pub use get_chat_id::GetChatId;
|
pub use get_chat_id::GetChatId;
|
||||||
pub use storage::*;
|
pub use storage::*;
|
||||||
use teloxide_core::types::ChatId;
|
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 get_chat_id;
|
||||||
mod storage;
|
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.
|
/// Perform a dialogue FSM transition.
|
||||||
///
|
///
|
||||||
/// This macro expands to a [`dptree::Handler`] that filters your dialogue
|
/// This macro expands to a [`dptree::Handler`] that filters your dialogue
|
||||||
|
|
|
@ -1,8 +1,6 @@
|
||||||
use std::sync::Arc;
|
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
dispatching::{
|
dispatching::{
|
||||||
dialogue::{Dialogue, GetChatId, Storage},
|
dialogue::{GetChatId, Storage},
|
||||||
DpHandlerDescription,
|
DpHandlerDescription,
|
||||||
},
|
},
|
||||||
types::{Me, Message},
|
types::{Me, Message},
|
||||||
|
@ -82,19 +80,7 @@ where
|
||||||
D: Default + Send + Sync + 'static,
|
D: Default + Send + Sync + 'static,
|
||||||
Upd: GetChatId + Clone + Send + Sync + 'static,
|
Upd: GetChatId + Clone + Send + Sync + 'static,
|
||||||
{
|
{
|
||||||
self.chain(dptree::filter_map(|storage: Arc<S>, upd: Upd| {
|
self.chain(super::dialogue::enter::<Upd, S, D, Output>())
|
||||||
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
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[allow(deprecated)]
|
#[allow(deprecated)]
|
||||||
|
|
Loading…
Add table
Reference in a new issue