Merge pull request #600 from teloxide/dev

Merge v0.8.1
This commit is contained in:
Hirrolot 2022-04-24 02:05:17 +06:00 committed by GitHub
commit f8b2f010c9
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
8 changed files with 211 additions and 23 deletions

View file

@ -55,6 +55,7 @@ jobs:
- stable - stable
- beta - beta
- nightly - nightly
- msrv
include: include:
- rust: stable - rust: stable
@ -66,6 +67,9 @@ jobs:
- rust: nightly - rust: nightly
toolchain: nightly-2022-01-17 toolchain: nightly-2022-01-17
features: "--all-features" features: "--all-features"
- rust: msrv
toolchain: "1.58.0"
features: "--features full"
steps: steps:
- uses: actions/checkout@v2 - uses: actions/checkout@v2

View file

@ -6,6 +6,13 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
## unreleased ## unreleased
## 0.8.1 - 2022-04-24
### 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

View file

@ -1,6 +1,6 @@
[package] [package]
name = "teloxide" name = "teloxide"
version = "0.8.0" version = "0.8.1"
edition = "2021" edition = "2021"
description = "An elegant Telegram bots framework for Rust" description = "An elegant Telegram bots framework for Rust"
repository = "https://github.com/teloxide/teloxide" repository = "https://github.com/teloxide/teloxide"
@ -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"]

View file

@ -58,7 +58,7 @@ $ set TELOXIDE_TOKEN=<Your token here>
$ $env:TELOXIDE_TOKEN=<Your token here> $ $env:TELOXIDE_TOKEN=<Your token here>
``` ```
4. Make sure that your Rust compiler is up to date: 4. Make sure that your Rust compiler is up to date (teloxide currently requires rustc at least version 1.58):
```bash ```bash
# If you're using stable # If you're using stable
$ rustup update stable $ rustup update stable
@ -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
View 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(())
}

View file

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

View file

@ -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
@ -203,8 +237,8 @@ where
/// - For `State::MyVariant(param,)` and `State::MyVariant { param, }`, the /// - For `State::MyVariant(param,)` and `State::MyVariant { param, }`, the
/// payload is `(param,)`. /// payload is `(param,)`.
/// - For `State::MyVariant(param1, ..., paramN)` and `State::MyVariant { /// - For `State::MyVariant(param1, ..., paramN)` and `State::MyVariant {
/// param1, ..., paramN }`, the payload is `(param1, ..., paramN)` (where `N` /// param1, ..., paramN }`, the payload is `(param1, ..., paramN)` (where
/// > 1). /// `N`>1).
/// ///
/// ## Dependency requirements /// ## Dependency requirements
/// ///

View file

@ -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)]