mirror of
https://github.com/teloxide/teloxide.git
synced 2025-01-05 10:24:32 +01:00
Merge branch 'dev' into teloxide-handler
This commit is contained in:
commit
9bb2a85aba
24 changed files with 335 additions and 276 deletions
8
.github/workflows/ci.yml
vendored
8
.github/workflows/ci.yml
vendored
|
@ -6,6 +6,9 @@ on:
|
||||||
|
|
||||||
name: Continuous integration
|
name: Continuous integration
|
||||||
|
|
||||||
|
env:
|
||||||
|
RUSTFLAGS: "--cfg CI_REDIS"
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
style:
|
style:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
@ -55,10 +58,13 @@ jobs:
|
||||||
|
|
||||||
include:
|
include:
|
||||||
- rust: stable
|
- rust: stable
|
||||||
|
toolchain: stable
|
||||||
features: "--features full"
|
features: "--features full"
|
||||||
- rust: beta
|
- rust: beta
|
||||||
|
toolchain: beta
|
||||||
features: "--features full"
|
features: "--features full"
|
||||||
- rust: nightly
|
- rust: nightly
|
||||||
|
toolchain: nightly-2022-01-17
|
||||||
features: "--all-features"
|
features: "--all-features"
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
|
@ -67,7 +73,7 @@ jobs:
|
||||||
- uses: actions-rs/toolchain@v1
|
- uses: actions-rs/toolchain@v1
|
||||||
with:
|
with:
|
||||||
profile: minimal
|
profile: minimal
|
||||||
toolchain: ${{ matrix.rust }}
|
toolchain: ${{ matrix.toolchain }}
|
||||||
override: true
|
override: true
|
||||||
|
|
||||||
- name: build
|
- name: build
|
||||||
|
|
20
CHANGELOG.md
20
CHANGELOG.md
|
@ -6,20 +6,23 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
||||||
|
|
||||||
## unreleased
|
## unreleased
|
||||||
|
|
||||||
|
### Removed
|
||||||
|
|
||||||
|
- The old dispatching system and related stuff: `dispatching`, `utils::UpState`, `prelude`, `repls2`, `crate::{dialogues_repl, dialogues_repl_with_listener}`, and `#[teloxide(subtransition)]` [**BC**].
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
- The new API for dialogue handlers: `teloxide::handler!` ([issue 567](https://github.com/teloxide/teloxide/issues/567)).
|
- The new API for dialogue handlers: `teloxide::handler!` ([issue 567](https://github.com/teloxide/teloxide/issues/567)).
|
||||||
|
|
||||||
### Removed
|
|
||||||
|
|
||||||
- The old dispatching system and related stuff: `dispatching`, `utils::UpState`, `prelude`, `repls2`, `crate::{dialogues_repl, dialogues_repl_with_listener}`, and `#[teloxide(...)]`.
|
|
||||||
|
|
||||||
### Changed
|
### Changed
|
||||||
|
|
||||||
- Rename `dispatching2` => `dispatching`.
|
- Rename `dispatching2` => `dispatching` [**BC**].
|
||||||
- Rename `prelude2` => `prelude`.
|
- Rename `prelude2` => `prelude` [**BC**].
|
||||||
- Move `update_listeners`, `stop_token`, `IdleShutdownError`, and `ShutdownToken` from the old `dispatching` to the new `dispatching`.
|
- Move `update_listeners`, `stop_token`, `IdleShutdownError`, and `ShutdownToken` from the old `dispatching` to the new `dispatching` (previously `dispatching2`).
|
||||||
- Replace `crate::{commands_repl, commands_repl_with_listener, repl, repl_with_listener}` with those of the new `dispatching`.
|
- Replace `crate::{commands_repl, commands_repl_with_listener, repl, repl_with_listener}` with those of the new `dispatching` [**BC**].
|
||||||
|
- `UpdateListener::StopToken` is now always `Send` [**BC**].
|
||||||
|
- Rename `BotCommand` trait to `BotCommands` [**BC**].
|
||||||
|
- `BotCommands::descriptions` now returns `CommandDescriptions` instead of `String` [**BC**].
|
||||||
|
|
||||||
## 0.7.2 - 2022-03-23
|
## 0.7.2 - 2022-03-23
|
||||||
|
|
||||||
|
@ -37,6 +40,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
||||||
|
|
||||||
- Log `UpdateKind::Error` in `teloxide::dispatching2::Dispatcher`.
|
- Log `UpdateKind::Error` in `teloxide::dispatching2::Dispatcher`.
|
||||||
- Don't warn about unhandled updates in `repls2` ([issue 557](https://github.com/teloxide/teloxide/issues/557)).
|
- Don't warn about unhandled updates in `repls2` ([issue 557](https://github.com/teloxide/teloxide/issues/557)).
|
||||||
|
- `parse_command` and `parse_command_with_prefix` now ignores case of the bot username
|
||||||
|
|
||||||
## 0.7.1 - 2022-03-09
|
## 0.7.1 - 2022-03-09
|
||||||
|
|
||||||
|
|
|
@ -55,7 +55,7 @@ full = [
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
teloxide-core = { version = "0.4", default-features = false }
|
teloxide-core = { version = "0.4", default-features = false }
|
||||||
teloxide-macros = { git = "https://github.com/teloxide/teloxide-macros.git", rev = "144eb73aaf39145bf8f6b57eec5c76730961c2f1", optional = true }
|
teloxide-macros = { git = "https://github.com/teloxide/teloxide-macros.git", rev = "dfba097c7146ba6244a36001d703e04b771baa05", optional = true }
|
||||||
|
|
||||||
serde_json = "1.0"
|
serde_json = "1.0"
|
||||||
serde = { version = "1.0", features = ["derive"] }
|
serde = { version = "1.0", features = ["derive"] }
|
||||||
|
|
|
@ -125,7 +125,7 @@ Commands are strongly typed and defined declaratively, similar to how we define
|
||||||
([Full](examples/simple_commands.rs))
|
([Full](examples/simple_commands.rs))
|
||||||
|
|
||||||
```rust,no_run
|
```rust,no_run
|
||||||
use teloxide::{prelude::*, utils::command::BotCommand};
|
use teloxide::{prelude::*, utils::command::BotCommands};
|
||||||
|
|
||||||
use std::error::Error;
|
use std::error::Error;
|
||||||
|
|
||||||
|
@ -139,7 +139,7 @@ async fn main() {
|
||||||
teloxide::commands_repl(bot, answer, Command::ty()).await;
|
teloxide::commands_repl(bot, answer, Command::ty()).await;
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(BotCommand, Clone)]
|
#[derive(BotCommands, Clone)]
|
||||||
#[command(rename = "lowercase", description = "These commands are supported:")]
|
#[command(rename = "lowercase", description = "These commands are supported:")]
|
||||||
enum Command {
|
enum Command {
|
||||||
#[command(description = "display this text.")]
|
#[command(description = "display this text.")]
|
||||||
|
@ -156,7 +156,9 @@ async fn answer(
|
||||||
command: Command,
|
command: Command,
|
||||||
) -> Result<(), Box<dyn Error + Send + Sync>> {
|
) -> Result<(), Box<dyn Error + Send + Sync>> {
|
||||||
match command {
|
match command {
|
||||||
Command::Help => bot.send_message(message.chat.id, Command::descriptions()).await?,
|
Command::Help => {
|
||||||
|
bot.send_message(message.chat.id, Command::descriptions().to_string()).await?
|
||||||
|
}
|
||||||
Command::Username(username) => {
|
Command::Username(username) => {
|
||||||
bot.send_message(message.chat.id, format!("Your username is @{}.", username)).await?
|
bot.send_message(message.chat.id, format!("Your username is @{}.", username)).await?
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,9 +1,9 @@
|
||||||
use std::{error::Error, str::FromStr};
|
use std::{error::Error, str::FromStr};
|
||||||
|
|
||||||
use chrono::Duration;
|
use chrono::Duration;
|
||||||
use teloxide::{prelude::*, types::ChatPermissions, utils::command::BotCommand};
|
use teloxide::{prelude::*, types::ChatPermissions, utils::command::BotCommands};
|
||||||
|
|
||||||
// Derive BotCommand to parse text with a command into this enumeration.
|
// Derive BotCommands to parse text with a command into this enumeration.
|
||||||
//
|
//
|
||||||
// 1. rename = "lowercase" turns all the commands into lowercase letters.
|
// 1. rename = "lowercase" turns all the commands into lowercase letters.
|
||||||
// 2. `description = "..."` specifies a text before all the commands.
|
// 2. `description = "..."` specifies a text before all the commands.
|
||||||
|
@ -12,7 +12,7 @@ use teloxide::{prelude::*, types::ChatPermissions, utils::command::BotCommand};
|
||||||
// your commands in this format:
|
// your commands in this format:
|
||||||
// %GENERAL-DESCRIPTION%
|
// %GENERAL-DESCRIPTION%
|
||||||
// %PREFIX%%COMMAND% - %DESCRIPTION%
|
// %PREFIX%%COMMAND% - %DESCRIPTION%
|
||||||
#[derive(BotCommand, Clone)]
|
#[derive(BotCommands, Clone)]
|
||||||
#[command(
|
#[command(
|
||||||
rename = "lowercase",
|
rename = "lowercase",
|
||||||
description = "Use commands in format /%command% %num% %unit%",
|
description = "Use commands in format /%command% %num% %unit%",
|
||||||
|
@ -72,7 +72,7 @@ async fn action(
|
||||||
) -> Result<(), Box<dyn Error + Send + Sync>> {
|
) -> Result<(), Box<dyn Error + Send + Sync>> {
|
||||||
match command {
|
match command {
|
||||||
Command::Help => {
|
Command::Help => {
|
||||||
bot.send_message(msg.chat.id, Command::descriptions()).await?;
|
bot.send_message(msg.chat.id, Command::descriptions().to_string()).await?;
|
||||||
}
|
}
|
||||||
Command::Kick => kick_user(bot, msg).await?,
|
Command::Kick => kick_user(bot, msg).await?,
|
||||||
Command::Ban { time, unit } => ban_user(bot, msg, calc_restrict_time(time, unit)).await?,
|
Command::Ban { time, unit } => ban_user(bot, msg, calc_restrict_time(time, unit)).await?,
|
||||||
|
|
|
@ -6,10 +6,10 @@ use teloxide::{
|
||||||
InlineKeyboardButton, InlineKeyboardMarkup, InlineQueryResultArticle, InputMessageContent,
|
InlineKeyboardButton, InlineKeyboardMarkup, InlineQueryResultArticle, InputMessageContent,
|
||||||
InputMessageContentText,
|
InputMessageContentText,
|
||||||
},
|
},
|
||||||
utils::command::BotCommand,
|
utils::command::BotCommands,
|
||||||
};
|
};
|
||||||
|
|
||||||
#[derive(BotCommand)]
|
#[derive(BotCommands)]
|
||||||
#[command(rename = "lowercase", description = "These commands are supported:")]
|
#[command(rename = "lowercase", description = "These commands are supported:")]
|
||||||
enum Command {
|
enum Command {
|
||||||
#[command(description = "Display this text")]
|
#[command(description = "Display this text")]
|
||||||
|
@ -31,9 +31,6 @@ async fn main() -> Result<(), Box<dyn Error>> {
|
||||||
.branch(Update::filter_inline_query().endpoint(inline_query_handler));
|
.branch(Update::filter_inline_query().endpoint(inline_query_handler));
|
||||||
|
|
||||||
Dispatcher::builder(bot, handler).build().setup_ctrlc_handler().dispatch().await;
|
Dispatcher::builder(bot, handler).build().setup_ctrlc_handler().dispatch().await;
|
||||||
|
|
||||||
log::info!("Closing bot... Goodbye!");
|
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -66,10 +63,10 @@ async fn message_handler(
|
||||||
bot: AutoSend<Bot>,
|
bot: AutoSend<Bot>,
|
||||||
) -> Result<(), Box<dyn Error + Send + Sync>> {
|
) -> Result<(), Box<dyn Error + Send + Sync>> {
|
||||||
if let Some(text) = m.text() {
|
if let Some(text) = m.text() {
|
||||||
match BotCommand::parse(text, "buttons") {
|
match BotCommands::parse(text, "buttons") {
|
||||||
Ok(Command::Help) => {
|
Ok(Command::Help) => {
|
||||||
// Just send the description of all commands.
|
// Just send the description of all commands.
|
||||||
bot.send_message(m.chat.id, Command::descriptions()).await?;
|
bot.send_message(m.chat.id, Command::descriptions().to_string()).await?;
|
||||||
}
|
}
|
||||||
Ok(Command::Start) => {
|
Ok(Command::Start) => {
|
||||||
// Create a list of buttons and send them.
|
// Create a list of buttons and send them.
|
||||||
|
|
|
@ -7,7 +7,8 @@ use teloxide::{
|
||||||
ErasedStorage, RedisStorage, SqliteStorage, Storage,
|
ErasedStorage, RedisStorage, SqliteStorage, Storage,
|
||||||
},
|
},
|
||||||
prelude::*,
|
prelude::*,
|
||||||
utils::command::BotCommand,
|
types::Me,
|
||||||
|
utils::command::BotCommands,
|
||||||
};
|
};
|
||||||
|
|
||||||
type MyDialogue = Dialogue<State, ErasedStorage<State>>;
|
type MyDialogue = Dialogue<State, ErasedStorage<State>>;
|
||||||
|
@ -26,7 +27,7 @@ impl Default for State {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(BotCommand, Clone)]
|
#[derive(BotCommands)]
|
||||||
#[command(rename = "lowercase", description = "These commands are supported:")]
|
#[command(rename = "lowercase", description = "These commands are supported:")]
|
||||||
pub enum Command {
|
pub enum Command {
|
||||||
#[command(description = "get your number.")]
|
#[command(description = "get your number.")]
|
||||||
|
|
|
@ -6,7 +6,7 @@ use rand::Rng;
|
||||||
use teloxide::{
|
use teloxide::{
|
||||||
prelude::*,
|
prelude::*,
|
||||||
types::{Dice, Update},
|
types::{Dice, Update},
|
||||||
utils::command::BotCommand,
|
utils::command::BotCommands,
|
||||||
};
|
};
|
||||||
|
|
||||||
#[tokio::main]
|
#[tokio::main]
|
||||||
|
@ -24,28 +24,6 @@ async fn main() {
|
||||||
let handler = Update::filter_message()
|
let handler = Update::filter_message()
|
||||||
// You can use branching to define multiple ways in which an update will be handled. If the
|
// You can use branching to define multiple ways in which an update will be handled. If the
|
||||||
// first branch fails, an update will be passed to the second branch, and so on.
|
// first branch fails, an update will be passed to the second branch, and so on.
|
||||||
.branch(
|
|
||||||
// Filtering allow you to filter updates by some condition.
|
|
||||||
dptree::filter(|msg: Message| msg.chat.is_group() || msg.chat.is_supergroup())
|
|
||||||
// An endpoint is the last update handler.
|
|
||||||
.endpoint(|msg: Message, bot: AutoSend<Bot>| async move {
|
|
||||||
log::info!("Received a message from a group chat.");
|
|
||||||
bot.send_message(msg.chat.id, "This is a group chat.").await?;
|
|
||||||
respond(())
|
|
||||||
}),
|
|
||||||
)
|
|
||||||
.branch(
|
|
||||||
// There are some extension filtering functions on `Message`. The following filter will
|
|
||||||
// filter only messages with dices.
|
|
||||||
Message::filter_dice().endpoint(
|
|
||||||
|msg: Message, dice: Dice, bot: AutoSend<Bot>| async move {
|
|
||||||
bot.send_message(msg.chat.id, format!("Dice value: {}", dice.value))
|
|
||||||
.reply_to_message_id(msg.id)
|
|
||||||
.await?;
|
|
||||||
Ok(())
|
|
||||||
},
|
|
||||||
),
|
|
||||||
)
|
|
||||||
.branch(
|
.branch(
|
||||||
dptree::entry()
|
dptree::entry()
|
||||||
// Filter commands: the next handlers will receive a parsed `SimpleCommand`.
|
// Filter commands: the next handlers will receive a parsed `SimpleCommand`.
|
||||||
|
@ -72,6 +50,28 @@ async fn main() {
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
|
)
|
||||||
|
.branch(
|
||||||
|
// Filtering allow you to filter updates by some condition.
|
||||||
|
dptree::filter(|msg: Message| msg.chat.is_group() || msg.chat.is_supergroup())
|
||||||
|
// An endpoint is the last update handler.
|
||||||
|
.endpoint(|msg: Message, bot: AutoSend<Bot>| async move {
|
||||||
|
log::info!("Received a message from a group chat.");
|
||||||
|
bot.send_message(msg.chat.id, "This is a group chat.").await?;
|
||||||
|
respond(())
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
.branch(
|
||||||
|
// There are some extension filtering functions on `Message`. The following filter will
|
||||||
|
// filter only messages with dices.
|
||||||
|
Message::filter_dice().endpoint(
|
||||||
|
|msg: Message, dice: Dice, bot: AutoSend<Bot>| async move {
|
||||||
|
bot.send_message(msg.chat.id, format!("Dice value: {}", dice.value))
|
||||||
|
.reply_to_message_id(msg.id)
|
||||||
|
.await?;
|
||||||
|
Ok(())
|
||||||
|
},
|
||||||
|
),
|
||||||
);
|
);
|
||||||
|
|
||||||
Dispatcher::builder(bot, handler)
|
Dispatcher::builder(bot, handler)
|
||||||
|
@ -99,7 +99,7 @@ struct ConfigParameters {
|
||||||
maintainer_username: Option<String>,
|
maintainer_username: Option<String>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(BotCommand, Clone)]
|
#[derive(BotCommands, Clone)]
|
||||||
#[command(rename = "lowercase", description = "Simple commands")]
|
#[command(rename = "lowercase", description = "Simple commands")]
|
||||||
enum SimpleCommand {
|
enum SimpleCommand {
|
||||||
#[command(description = "shows this message.")]
|
#[command(description = "shows this message.")]
|
||||||
|
@ -110,7 +110,7 @@ enum SimpleCommand {
|
||||||
MyId,
|
MyId,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(BotCommand, Clone)]
|
#[derive(BotCommands, Clone)]
|
||||||
#[command(rename = "lowercase", description = "Maintainer commands")]
|
#[command(rename = "lowercase", description = "Maintainer commands")]
|
||||||
enum MaintainerCommands {
|
enum MaintainerCommands {
|
||||||
#[command(parse_with = "split", description = "generate a number within range")]
|
#[command(parse_with = "split", description = "generate a number within range")]
|
||||||
|
@ -122,13 +122,20 @@ async fn simple_commands_handler(
|
||||||
bot: AutoSend<Bot>,
|
bot: AutoSend<Bot>,
|
||||||
cmd: SimpleCommand,
|
cmd: SimpleCommand,
|
||||||
cfg: ConfigParameters,
|
cfg: ConfigParameters,
|
||||||
|
me: teloxide::types::Me,
|
||||||
) -> Result<(), teloxide::RequestError> {
|
) -> Result<(), teloxide::RequestError> {
|
||||||
let text = match cmd {
|
let text = match cmd {
|
||||||
SimpleCommand::Help => {
|
SimpleCommand::Help => {
|
||||||
if msg.from().unwrap().id == cfg.bot_maintainer {
|
if msg.from().unwrap().id == cfg.bot_maintainer {
|
||||||
format!("{}\n{}", SimpleCommand::descriptions(), MaintainerCommands::descriptions())
|
format!(
|
||||||
|
"{}\n\n{}",
|
||||||
|
SimpleCommand::descriptions(),
|
||||||
|
MaintainerCommands::descriptions()
|
||||||
|
)
|
||||||
|
} else if msg.chat.is_group() || msg.chat.is_supergroup() {
|
||||||
|
SimpleCommand::descriptions().username_from_me(&me).to_string()
|
||||||
} else {
|
} else {
|
||||||
SimpleCommand::descriptions()
|
SimpleCommand::descriptions().to_string()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
SimpleCommand::Maintainer => {
|
SimpleCommand::Maintainer => {
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
use teloxide::{prelude::*, utils::command::BotCommand};
|
use teloxide::{prelude::*, utils::command::BotCommands};
|
||||||
|
|
||||||
use std::error::Error;
|
use std::error::Error;
|
||||||
|
|
||||||
|
@ -12,7 +12,7 @@ async fn main() {
|
||||||
teloxide::commands_repl(bot, answer, Command::ty()).await;
|
teloxide::commands_repl(bot, answer, Command::ty()).await;
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(BotCommand, Clone)]
|
#[derive(BotCommands, Clone)]
|
||||||
#[command(rename = "lowercase", description = "These commands are supported:")]
|
#[command(rename = "lowercase", description = "These commands are supported:")]
|
||||||
enum Command {
|
enum Command {
|
||||||
#[command(description = "display this text.")]
|
#[command(description = "display this text.")]
|
||||||
|
@ -29,7 +29,9 @@ async fn answer(
|
||||||
command: Command,
|
command: Command,
|
||||||
) -> Result<(), Box<dyn Error + Send + Sync>> {
|
) -> Result<(), Box<dyn Error + Send + Sync>> {
|
||||||
match command {
|
match command {
|
||||||
Command::Help => bot.send_message(message.chat.id, Command::descriptions()).await?,
|
Command::Help => {
|
||||||
|
bot.send_message(message.chat.id, Command::descriptions().to_string()).await?
|
||||||
|
}
|
||||||
Command::Username(username) => {
|
Command::Username(username) => {
|
||||||
bot.send_message(message.chat.id, format!("Your username is @{}.", username)).await?
|
bot.send_message(message.chat.id, format!("Your username is @{}.", username)).await?
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
[build]
|
[build]
|
||||||
command = "rustup install nightly --profile minimal && cargo +nightly doc --all-features --no-deps -Zunstable-options -Zrustdoc-scrape-examples=examples && cp -r target/doc _netlify_out"
|
command = "rustup install nightly-2022-02-02 --profile minimal && cargo +nightly doc --all-features --no-deps -Zunstable-options -Zrustdoc-scrape-examples=examples && cp -r target/doc _netlify_out"
|
||||||
publish = "_netlify_out"
|
publish = "_netlify_out"
|
||||||
|
|
||||||
[build.environment]
|
[build.environment]
|
||||||
|
|
|
@ -13,28 +13,20 @@
|
||||||
//! dialogues. Your dialogue state can be represented as an enumeration:
|
//! dialogues. Your dialogue state can be represented as an enumeration:
|
||||||
//!
|
//!
|
||||||
//! ```ignore
|
//! ```ignore
|
||||||
//! #[derive(DialogueState, Clone)]
|
//! #[derive(Clone)]
|
||||||
//! #[handler_out(anyhow::Result<()>)]
|
|
||||||
//! pub enum State {
|
//! pub enum State {
|
||||||
//! #[handler(handle_start)]
|
|
||||||
//! Start,
|
//! Start,
|
||||||
//!
|
|
||||||
//! #[handler(handle_receive_full_name)]
|
|
||||||
//! ReceiveFullName,
|
//! ReceiveFullName,
|
||||||
//!
|
|
||||||
//! #[handler(handle_receive_age)]
|
|
||||||
//! ReceiveAge { full_name: String },
|
//! ReceiveAge { full_name: String },
|
||||||
//!
|
|
||||||
//! #[handler(handle_receive_location)]
|
|
||||||
//! ReceiveLocation { full_name: String, age: u8 },
|
//! ReceiveLocation { full_name: String, age: u8 },
|
||||||
//! }
|
//! }
|
||||||
//! ```
|
//! ```
|
||||||
//!
|
//!
|
||||||
//! Each state is associated with its respective handler: e.g., when a dialogue
|
//! Each state is associated with its respective handler: e.g., when a dialogue
|
||||||
//! state is `ReceiveAge`, `handle_receive_age` is invoked:
|
//! state is `ReceiveAge`, `receive_age` is invoked:
|
||||||
//!
|
//!
|
||||||
//! ```ignore
|
//! ```ignore
|
||||||
//! async fn handle_receive_age(
|
//! async fn receive_age(
|
||||||
//! bot: AutoSend<Bot>,
|
//! bot: AutoSend<Bot>,
|
||||||
//! msg: Message,
|
//! msg: Message,
|
||||||
//! dialogue: MyDialogue,
|
//! dialogue: MyDialogue,
|
||||||
|
@ -61,7 +53,7 @@
|
||||||
//! the inner storage:
|
//! the inner storage:
|
||||||
//!
|
//!
|
||||||
//! ```ignore
|
//! ```ignore
|
||||||
//! async fn handle_receive_location(
|
//! async fn receive_location(
|
||||||
//! bot: AutoSend<Bot>,
|
//! bot: AutoSend<Bot>,
|
||||||
//! msg: Message,
|
//! msg: Message,
|
||||||
//! dialogue: MyDialogue,
|
//! dialogue: MyDialogue,
|
||||||
|
|
|
@ -6,7 +6,7 @@ use crate::{
|
||||||
HandlerFactory,
|
HandlerFactory,
|
||||||
},
|
},
|
||||||
types::{Me, Message},
|
types::{Me, Message},
|
||||||
utils::command::BotCommand,
|
utils::command::BotCommands,
|
||||||
};
|
};
|
||||||
use dptree::{di::DependencyMap, Handler};
|
use dptree::{di::DependencyMap, Handler};
|
||||||
|
|
||||||
|
@ -23,7 +23,7 @@ pub trait HandlerExt<Output> {
|
||||||
#[must_use]
|
#[must_use]
|
||||||
fn filter_command<C>(self) -> Self
|
fn filter_command<C>(self) -> Self
|
||||||
where
|
where
|
||||||
C: BotCommand + Send + Sync + 'static;
|
C: BotCommands + Send + Sync + 'static;
|
||||||
|
|
||||||
/// Passes [`Dialogue<D, S>`] and `D` as handler dependencies.
|
/// Passes [`Dialogue<D, S>`] and `D` as handler dependencies.
|
||||||
///
|
///
|
||||||
|
@ -62,7 +62,7 @@ where
|
||||||
{
|
{
|
||||||
fn filter_command<C>(self) -> Self
|
fn filter_command<C>(self) -> Self
|
||||||
where
|
where
|
||||||
C: BotCommand + Send + Sync + 'static,
|
C: BotCommands + Send + Sync + 'static,
|
||||||
{
|
{
|
||||||
self.chain(dptree::filter_map(move |message: Message, me: Me| {
|
self.chain(dptree::filter_map(move |message: Message, me: Me| {
|
||||||
let bot_name = me.user.username.expect("Bots must have a username");
|
let bot_name = me.user.username.expect("Bots must have a username");
|
||||||
|
|
|
@ -4,7 +4,7 @@ use crate::{
|
||||||
},
|
},
|
||||||
error_handlers::LoggingErrorHandler,
|
error_handlers::LoggingErrorHandler,
|
||||||
types::Update,
|
types::Update,
|
||||||
utils::command::BotCommand,
|
utils::command::BotCommands,
|
||||||
};
|
};
|
||||||
use dptree::di::{DependencyMap, Injectable};
|
use dptree::di::{DependencyMap, Injectable};
|
||||||
use std::{fmt::Debug, marker::PhantomData};
|
use std::{fmt::Debug, marker::PhantomData};
|
||||||
|
@ -28,7 +28,7 @@ use teloxide_core::requests::Requester;
|
||||||
#[cfg(feature = "ctrlc_handler")]
|
#[cfg(feature = "ctrlc_handler")]
|
||||||
pub async fn commands_repl<'a, R, Cmd, H, E, Args>(bot: R, handler: H, cmd: PhantomData<Cmd>)
|
pub async fn commands_repl<'a, R, Cmd, H, E, Args>(bot: R, handler: H, cmd: PhantomData<Cmd>)
|
||||||
where
|
where
|
||||||
Cmd: BotCommand + Send + Sync + 'static,
|
Cmd: BotCommands + Send + Sync + 'static,
|
||||||
H: Injectable<DependencyMap, Result<(), E>, Args> + Send + Sync + 'static,
|
H: Injectable<DependencyMap, Result<(), E>, Args> + Send + Sync + 'static,
|
||||||
R: Requester + Clone + Send + Sync + 'static,
|
R: Requester + Clone + Send + Sync + 'static,
|
||||||
<R as Requester>::GetUpdates: Send,
|
<R as Requester>::GetUpdates: Send,
|
||||||
|
@ -68,7 +68,7 @@ pub async fn commands_repl_with_listener<'a, R, Cmd, H, L, ListenerE, E, Args>(
|
||||||
listener: L,
|
listener: L,
|
||||||
_cmd: PhantomData<Cmd>,
|
_cmd: PhantomData<Cmd>,
|
||||||
) where
|
) where
|
||||||
Cmd: BotCommand + Send + Sync + 'static,
|
Cmd: BotCommands + Send + Sync + 'static,
|
||||||
H: Injectable<DependencyMap, Result<(), E>, Args> + Send + Sync + 'static,
|
H: Injectable<DependencyMap, Result<(), E>, Args> + Send + Sync + 'static,
|
||||||
L: UpdateListener<ListenerE> + Send + 'a,
|
L: UpdateListener<ListenerE> + Send + 'a,
|
||||||
ListenerE: Debug + Send + 'a,
|
ListenerE: Debug + Send + 'a,
|
||||||
|
@ -81,20 +81,13 @@ pub async fn commands_repl_with_listener<'a, R, Cmd, H, L, ListenerE, E, Args>(
|
||||||
// commands. See <https://github.com/teloxide/teloxide/issues/557>.
|
// commands. See <https://github.com/teloxide/teloxide/issues/557>.
|
||||||
let ignore_update = |_upd| Box::pin(async {});
|
let ignore_update = |_upd| Box::pin(async {});
|
||||||
|
|
||||||
let mut dispatcher = Dispatcher::builder(
|
Dispatcher::builder(
|
||||||
bot,
|
bot,
|
||||||
Update::filter_message().filter_command::<Cmd>().branch(dptree::endpoint(handler)),
|
Update::filter_message().filter_command::<Cmd>().branch(dptree::endpoint(handler)),
|
||||||
)
|
)
|
||||||
.default_handler(ignore_update)
|
.default_handler(ignore_update)
|
||||||
.build();
|
.build()
|
||||||
|
.setup_ctrlc_handler()
|
||||||
#[cfg(feature = "ctrlc_handler")]
|
|
||||||
dispatcher.setup_ctrlc_handler();
|
|
||||||
|
|
||||||
// To make mutable var from immutable.
|
|
||||||
let mut dispatcher = dispatcher;
|
|
||||||
|
|
||||||
dispatcher
|
|
||||||
.dispatch_with_listener(
|
.dispatch_with_listener(
|
||||||
listener,
|
listener,
|
||||||
LoggingErrorHandler::with_custom_text("An error from the update listener"),
|
LoggingErrorHandler::with_custom_text("An error from the update listener"),
|
||||||
|
|
|
@ -1,96 +0,0 @@
|
||||||
use crate::{
|
|
||||||
dispatching::{
|
|
||||||
dialogue::{DialogueDispatcher, DialogueStage, DialogueWithCx, InMemStorageError},
|
|
||||||
update_listeners,
|
|
||||||
update_listeners::UpdateListener,
|
|
||||||
Dispatcher, UpdateWithCx,
|
|
||||||
},
|
|
||||||
error_handlers::LoggingErrorHandler,
|
|
||||||
};
|
|
||||||
use std::{fmt::Debug, future::Future, sync::Arc};
|
|
||||||
use teloxide_core::{requests::Requester, types::Message};
|
|
||||||
|
|
||||||
/// A [REPL] for dialogues.
|
|
||||||
///
|
|
||||||
/// All errors from an update listener and handler will be logged. This function
|
|
||||||
/// uses [`InMemStorage`].
|
|
||||||
///
|
|
||||||
/// # Caution
|
|
||||||
/// **DO NOT** use this function together with [`Dispatcher`] and other REPLs,
|
|
||||||
/// because Telegram disallow multiple requests at the same time from the same
|
|
||||||
/// bot.
|
|
||||||
///
|
|
||||||
/// [REPL]: https://en.wikipedia.org/wiki/Read-eval-print_loop
|
|
||||||
/// [`Dispatcher`]: crate::dispatching::Dispatcher
|
|
||||||
/// [`InMemStorage`]: crate::dispatching::dialogue::InMemStorage
|
|
||||||
#[cfg(feature = "ctrlc_handler")]
|
|
||||||
pub async fn dialogues_repl<'a, R, H, D, Fut>(requester: R, handler: H)
|
|
||||||
where
|
|
||||||
H: Fn(UpdateWithCx<R, Message>, D) -> Fut + Send + Sync + 'static,
|
|
||||||
D: Clone + Default + Send + 'static,
|
|
||||||
Fut: Future<Output = DialogueStage<D>> + Send + 'static,
|
|
||||||
R: Requester + Send + Clone + 'static,
|
|
||||||
<R as Requester>::GetUpdatesFaultTolerant: Send,
|
|
||||||
{
|
|
||||||
let cloned_requester = requester.clone();
|
|
||||||
|
|
||||||
dialogues_repl_with_listener(
|
|
||||||
requester,
|
|
||||||
handler,
|
|
||||||
update_listeners::polling_default(cloned_requester).await,
|
|
||||||
)
|
|
||||||
.await;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Like [`dialogues_repl`], but with a custom [`UpdateListener`].
|
|
||||||
///
|
|
||||||
/// All errors from an update listener and handler will be logged. This function
|
|
||||||
/// uses [`InMemStorage`].
|
|
||||||
///
|
|
||||||
/// # Caution
|
|
||||||
/// **DO NOT** use this function together with [`Dispatcher`] and other REPLs,
|
|
||||||
/// because Telegram disallow multiple requests at the same time from the same
|
|
||||||
/// bot.
|
|
||||||
///
|
|
||||||
/// [`Dispatcher`]: crate::dispatching::Dispatcher
|
|
||||||
/// [`dialogues_repl`]: crate::dispatching::repls::dialogues_repl()
|
|
||||||
/// [`UpdateListener`]: crate::dispatching::update_listeners::UpdateListener
|
|
||||||
/// [`InMemStorage`]: crate::dispatching::dialogue::InMemStorage
|
|
||||||
#[cfg(feature = "ctrlc_handler")]
|
|
||||||
pub async fn dialogues_repl_with_listener<'a, R, H, D, Fut, L, ListenerE>(
|
|
||||||
requester: R,
|
|
||||||
handler: H,
|
|
||||||
listener: L,
|
|
||||||
) where
|
|
||||||
H: Fn(UpdateWithCx<R, Message>, D) -> Fut + Send + Sync + 'static,
|
|
||||||
D: Clone + Default + Send + 'static,
|
|
||||||
Fut: Future<Output = DialogueStage<D>> + Send + 'static,
|
|
||||||
L: UpdateListener<ListenerE> + Send + 'a,
|
|
||||||
ListenerE: Debug + Send + 'a,
|
|
||||||
R: Requester + Send + Clone + 'static,
|
|
||||||
{
|
|
||||||
let handler = Arc::new(handler);
|
|
||||||
|
|
||||||
Dispatcher::new(requester)
|
|
||||||
.messages_handler(DialogueDispatcher::new(
|
|
||||||
move |DialogueWithCx { cx, dialogue }: DialogueWithCx<
|
|
||||||
R,
|
|
||||||
Message,
|
|
||||||
D,
|
|
||||||
InMemStorageError,
|
|
||||||
>| {
|
|
||||||
let handler = Arc::clone(&handler);
|
|
||||||
|
|
||||||
async move {
|
|
||||||
let dialogue = dialogue.expect("std::convert::Infallible");
|
|
||||||
handler(cx, dialogue).await
|
|
||||||
}
|
|
||||||
},
|
|
||||||
))
|
|
||||||
.setup_ctrlc_handler()
|
|
||||||
.dispatch_with_listener(
|
|
||||||
listener,
|
|
||||||
LoggingErrorHandler::with_custom_text("An error from the update listener"),
|
|
||||||
)
|
|
||||||
.await;
|
|
||||||
}
|
|
|
@ -4,5 +4,4 @@ mod commands_repl;
|
||||||
mod repl;
|
mod repl;
|
||||||
|
|
||||||
pub use commands_repl::{commands_repl, commands_repl_with_listener};
|
pub use commands_repl::{commands_repl, commands_repl_with_listener};
|
||||||
//pub use dialogues_repl::{dialogues_repl, dialogues_repl_with_listener};
|
|
||||||
pub use repl::{repl, repl_with_listener};
|
pub use repl::{repl, repl_with_listener};
|
||||||
|
|
|
@ -59,19 +59,22 @@ where
|
||||||
// messages. See <https://github.com/teloxide/teloxide/issues/557>.
|
// messages. See <https://github.com/teloxide/teloxide/issues/557>.
|
||||||
let ignore_update = |_upd| Box::pin(async {});
|
let ignore_update = |_upd| Box::pin(async {});
|
||||||
|
|
||||||
#[allow(unused_mut)]
|
|
||||||
let mut dispatcher =
|
|
||||||
Dispatcher::builder(bot, Update::filter_message().branch(dptree::endpoint(handler)))
|
Dispatcher::builder(bot, Update::filter_message().branch(dptree::endpoint(handler)))
|
||||||
.default_handler(ignore_update)
|
.default_handler(ignore_update)
|
||||||
.build();
|
.build()
|
||||||
|
.setup_ctrlc_handler()
|
||||||
#[cfg(feature = "ctrlc_handler")]
|
|
||||||
dispatcher.setup_ctrlc_handler();
|
|
||||||
|
|
||||||
dispatcher
|
|
||||||
.dispatch_with_listener(
|
.dispatch_with_listener(
|
||||||
listener,
|
listener,
|
||||||
LoggingErrorHandler::with_custom_text("An error from the update listener"),
|
LoggingErrorHandler::with_custom_text("An error from the update listener"),
|
||||||
)
|
)
|
||||||
.await;
|
.await;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn repl_is_send() {
|
||||||
|
let bot = crate::Bot::new("");
|
||||||
|
let repl = crate::repl(bot, || async { crate::respond(()) });
|
||||||
|
assert_send(&repl);
|
||||||
|
|
||||||
|
fn assert_send(_: &impl Send) {}
|
||||||
|
}
|
||||||
|
|
|
@ -53,7 +53,7 @@ pub use self::{
|
||||||
/// [module-level documentation]: mod@self
|
/// [module-level documentation]: mod@self
|
||||||
pub trait UpdateListener<E>: for<'a> AsUpdateStream<'a, E> {
|
pub trait UpdateListener<E>: for<'a> AsUpdateStream<'a, E> {
|
||||||
/// The type of token which allows to stop this listener.
|
/// The type of token which allows to stop this listener.
|
||||||
type StopToken: StopToken;
|
type StopToken: StopToken + Send;
|
||||||
|
|
||||||
/// Returns a token which stops this listener.
|
/// Returns a token which stops this listener.
|
||||||
///
|
///
|
||||||
|
|
|
@ -7,7 +7,7 @@ use futures::{
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
dispatching::{
|
dispatching::{
|
||||||
stop_token::{AsyncStopFlag, AsyncStopToken, StopToken},
|
stop_token::{AsyncStopFlag, AsyncStopToken},
|
||||||
update_listeners::{stateful_listener::StatefulListener, UpdateListener},
|
update_listeners::{stateful_listener::StatefulListener, UpdateListener},
|
||||||
},
|
},
|
||||||
payloads::{GetUpdates, GetUpdatesSetters as _},
|
payloads::{GetUpdates, GetUpdatesSetters as _},
|
||||||
|
@ -22,9 +22,7 @@ use crate::{
|
||||||
/// ## Notes
|
/// ## Notes
|
||||||
///
|
///
|
||||||
/// This function will automatically delete a webhook if it was set up.
|
/// This function will automatically delete a webhook if it was set up.
|
||||||
pub async fn polling_default<R>(
|
pub async fn polling_default<R>(requester: R) -> impl UpdateListener<R::Err>
|
||||||
requester: R,
|
|
||||||
) -> impl UpdateListener<R::Err, StopToken = impl Send + StopToken>
|
|
||||||
where
|
where
|
||||||
R: Requester + Send + 'static,
|
R: Requester + Send + 'static,
|
||||||
<R as Requester>::GetUpdates: Send,
|
<R as Requester>::GetUpdates: Send,
|
||||||
|
@ -130,7 +128,7 @@ pub fn polling<R>(
|
||||||
timeout: Option<Duration>,
|
timeout: Option<Duration>,
|
||||||
limit: Option<u8>,
|
limit: Option<u8>,
|
||||||
allowed_updates: Option<Vec<AllowedUpdate>>,
|
allowed_updates: Option<Vec<AllowedUpdate>>,
|
||||||
) -> impl UpdateListener<R::Err, StopToken = impl Send + StopToken>
|
) -> impl UpdateListener<R::Err>
|
||||||
where
|
where
|
||||||
R: Requester + Send + 'static,
|
R: Requester + Send + 'static,
|
||||||
<R as Requester>::GetUpdates: Send,
|
<R as Requester>::GetUpdates: Send,
|
||||||
|
|
|
@ -123,7 +123,7 @@ impl<St, Assf, Sf, Hauf, Stt, Thf, E> UpdateListener<E>
|
||||||
where
|
where
|
||||||
Self: for<'a> AsUpdateStream<'a, E>,
|
Self: for<'a> AsUpdateStream<'a, E>,
|
||||||
Sf: FnMut(&mut St) -> Stt,
|
Sf: FnMut(&mut St) -> Stt,
|
||||||
Stt: StopToken,
|
Stt: StopToken + Send,
|
||||||
Hauf: FnMut(&mut St, &mut dyn Iterator<Item = AllowedUpdate>),
|
Hauf: FnMut(&mut St, &mut dyn Iterator<Item = AllowedUpdate>),
|
||||||
Thf: Fn(&St) -> Option<Duration>,
|
Thf: Fn(&St) -> Option<Duration>,
|
||||||
{
|
{
|
||||||
|
|
|
@ -1,17 +1,18 @@
|
||||||
//! Command parsers.
|
//! Command parsers.
|
||||||
//!
|
//!
|
||||||
//! You can either create an `enum` with derived [`BotCommand`], containing
|
//! You can either create an `enum` with derived [`BotCommands`], containing
|
||||||
//! commands of your bot, or use functions, which split input text into a string
|
//! commands of your bot, or use functions, which split input text into a string
|
||||||
//! command with its arguments.
|
//! command with its arguments.
|
||||||
//!
|
//!
|
||||||
//! # Using BotCommand
|
//! # Using BotCommands
|
||||||
|
//!
|
||||||
//! ```
|
//! ```
|
||||||
//! # #[cfg(feature = "macros")] {
|
//! # #[cfg(feature = "macros")] {
|
||||||
//! use teloxide::utils::command::BotCommand;
|
//! use teloxide::utils::command::BotCommands;
|
||||||
//!
|
//!
|
||||||
//! type UnitOfTime = u8;
|
//! type UnitOfTime = u8;
|
||||||
//!
|
//!
|
||||||
//! #[derive(BotCommand, PartialEq, Debug)]
|
//! #[derive(BotCommands, PartialEq, Debug)]
|
||||||
//! #[command(rename = "lowercase", parse_with = "split")]
|
//! #[command(rename = "lowercase", parse_with = "split")]
|
||||||
//! enum AdminCommand {
|
//! enum AdminCommand {
|
||||||
//! Mute(UnitOfTime, char),
|
//! Mute(UnitOfTime, char),
|
||||||
|
@ -24,6 +25,7 @@
|
||||||
//! ```
|
//! ```
|
||||||
//!
|
//!
|
||||||
//! # Using parse_command
|
//! # Using parse_command
|
||||||
|
//!
|
||||||
//! ```
|
//! ```
|
||||||
//! use teloxide::utils::command::parse_command;
|
//! use teloxide::utils::command::parse_command;
|
||||||
//!
|
//!
|
||||||
|
@ -33,6 +35,7 @@
|
||||||
//! ```
|
//! ```
|
||||||
//!
|
//!
|
||||||
//! # Using parse_command_with_prefix
|
//! # Using parse_command_with_prefix
|
||||||
|
//!
|
||||||
//! ```
|
//! ```
|
||||||
//! use teloxide::utils::command::parse_command_with_prefix;
|
//! use teloxide::utils::command::parse_command_with_prefix;
|
||||||
//!
|
//!
|
||||||
|
@ -46,25 +49,27 @@
|
||||||
//!
|
//!
|
||||||
//! [examples/admin_bot]: https://github.com/teloxide/teloxide/blob/master/examples/admin_bot/
|
//! [examples/admin_bot]: https://github.com/teloxide/teloxide/blob/master/examples/admin_bot/
|
||||||
|
|
||||||
|
use core::fmt;
|
||||||
use std::{
|
use std::{
|
||||||
error::Error,
|
error::Error,
|
||||||
fmt::{Display, Formatter},
|
fmt::{Display, Formatter, Write},
|
||||||
};
|
};
|
||||||
|
|
||||||
use std::marker::PhantomData;
|
use std::marker::PhantomData;
|
||||||
|
use teloxide_core::types::{BotCommand, Me};
|
||||||
#[cfg(feature = "macros")]
|
#[cfg(feature = "macros")]
|
||||||
pub use teloxide_macros::BotCommand;
|
pub use teloxide_macros::BotCommands;
|
||||||
|
|
||||||
/// An enumeration of bot's commands.
|
/// An enumeration of bot's commands.
|
||||||
///
|
///
|
||||||
/// # Example
|
/// # Example
|
||||||
/// ```
|
/// ```
|
||||||
/// # #[cfg(feature = "macros")] {
|
/// # #[cfg(feature = "macros")] {
|
||||||
/// use teloxide::utils::command::BotCommand;
|
/// use teloxide::utils::command::BotCommands;
|
||||||
///
|
///
|
||||||
/// type UnitOfTime = u8;
|
/// type UnitOfTime = u8;
|
||||||
///
|
///
|
||||||
/// #[derive(BotCommand, PartialEq, Debug)]
|
/// #[derive(BotCommands, PartialEq, Debug)]
|
||||||
/// #[command(rename = "lowercase", parse_with = "split")]
|
/// #[command(rename = "lowercase", parse_with = "split")]
|
||||||
/// enum AdminCommand {
|
/// enum AdminCommand {
|
||||||
/// Mute(UnitOfTime, char),
|
/// Mute(UnitOfTime, char),
|
||||||
|
@ -98,9 +103,9 @@ pub use teloxide_macros::BotCommand;
|
||||||
/// ## Example
|
/// ## Example
|
||||||
/// ```
|
/// ```
|
||||||
/// # #[cfg(feature = "macros")] {
|
/// # #[cfg(feature = "macros")] {
|
||||||
/// use teloxide::utils::command::BotCommand;
|
/// use teloxide::utils::command::BotCommands;
|
||||||
///
|
///
|
||||||
/// #[derive(BotCommand, PartialEq, Debug)]
|
/// #[derive(BotCommands, PartialEq, Debug)]
|
||||||
/// #[command(rename = "lowercase")]
|
/// #[command(rename = "lowercase")]
|
||||||
/// enum Command {
|
/// enum Command {
|
||||||
/// Text(String),
|
/// Text(String),
|
||||||
|
@ -118,9 +123,9 @@ pub use teloxide_macros::BotCommand;
|
||||||
/// ## Example
|
/// ## Example
|
||||||
/// ```
|
/// ```
|
||||||
/// # #[cfg(feature = "macros")] {
|
/// # #[cfg(feature = "macros")] {
|
||||||
/// use teloxide::utils::command::BotCommand;
|
/// use teloxide::utils::command::BotCommands;
|
||||||
///
|
///
|
||||||
/// #[derive(BotCommand, PartialEq, Debug)]
|
/// #[derive(BotCommands, PartialEq, Debug)]
|
||||||
/// #[command(rename = "lowercase", parse_with = "split")]
|
/// #[command(rename = "lowercase", parse_with = "split")]
|
||||||
/// enum Command {
|
/// enum Command {
|
||||||
/// Nums(u8, u16, i32),
|
/// Nums(u8, u16, i32),
|
||||||
|
@ -138,9 +143,9 @@ pub use teloxide_macros::BotCommand;
|
||||||
/// ## Example
|
/// ## Example
|
||||||
/// ```
|
/// ```
|
||||||
/// # #[cfg(feature = "macros")] {
|
/// # #[cfg(feature = "macros")] {
|
||||||
/// use teloxide::utils::command::BotCommand;
|
/// use teloxide::utils::command::BotCommands;
|
||||||
///
|
///
|
||||||
/// #[derive(BotCommand, PartialEq, Debug)]
|
/// #[derive(BotCommands, PartialEq, Debug)]
|
||||||
/// #[command(rename = "lowercase", parse_with = "split", separator = "|")]
|
/// #[command(rename = "lowercase", parse_with = "split", separator = "|")]
|
||||||
/// enum Command {
|
/// enum Command {
|
||||||
/// Nums(u8, u16, i32),
|
/// Nums(u8, u16, i32),
|
||||||
|
@ -171,7 +176,7 @@ pub use teloxide_macros::BotCommand;
|
||||||
/// ## Example
|
/// ## Example
|
||||||
/// ```
|
/// ```
|
||||||
/// # #[cfg(feature = "macros")] {
|
/// # #[cfg(feature = "macros")] {
|
||||||
/// use teloxide::utils::command::{BotCommand, ParseError};
|
/// use teloxide::utils::command::{BotCommands, ParseError};
|
||||||
///
|
///
|
||||||
/// fn accept_two_digits(input: String) -> Result<(u8,), ParseError> {
|
/// fn accept_two_digits(input: String) -> Result<(u8,), ParseError> {
|
||||||
/// match input.len() {
|
/// match input.len() {
|
||||||
|
@ -183,7 +188,7 @@ pub use teloxide_macros::BotCommand;
|
||||||
/// }
|
/// }
|
||||||
/// }
|
/// }
|
||||||
///
|
///
|
||||||
/// #[derive(BotCommand, PartialEq, Debug)]
|
/// #[derive(BotCommands, PartialEq, Debug)]
|
||||||
/// #[command(rename = "lowercase")]
|
/// #[command(rename = "lowercase")]
|
||||||
/// enum Command {
|
/// enum Command {
|
||||||
/// #[command(parse_with = "accept_two_digits")]
|
/// #[command(parse_with = "accept_two_digits")]
|
||||||
|
@ -204,24 +209,41 @@ pub use teloxide_macros::BotCommand;
|
||||||
/// specific variant.
|
/// specific variant.
|
||||||
///
|
///
|
||||||
/// [`FromStr`]: https://doc.rust-lang.org/std/str/trait.FromStr.html
|
/// [`FromStr`]: https://doc.rust-lang.org/std/str/trait.FromStr.html
|
||||||
/// [`BotCommand`]: crate::utils::command::BotCommand
|
/// [`BotCommands`]: crate::utils::command::BotCommands
|
||||||
pub trait BotCommand: Sized {
|
pub trait BotCommands: Sized {
|
||||||
fn descriptions() -> String;
|
/// Parses a command.
|
||||||
fn parse<N>(s: &str, bot_name: N) -> Result<Self, ParseError>
|
///
|
||||||
|
/// `bot_username` is required to parse commands like
|
||||||
|
/// `/cmd@username_of_the_bot`.
|
||||||
|
fn parse<N>(s: &str, bot_username: N) -> Result<Self, ParseError>
|
||||||
where
|
where
|
||||||
N: Into<String>;
|
N: Into<String>;
|
||||||
|
|
||||||
|
/// Returns descriptions of the commands suitable to be shown to the user
|
||||||
|
/// (for example when `/help` command is used).
|
||||||
|
fn descriptions() -> CommandDescriptions<'static>;
|
||||||
|
|
||||||
|
/// Returns a vector of [`BotCommand`] that can be used with
|
||||||
|
/// [`set_my_commands`].
|
||||||
|
///
|
||||||
|
/// [`BotCommand`]: crate::types::BotCommand
|
||||||
|
/// [`set_my_commands`]: crate::requests::Requester::set_my_commands
|
||||||
|
fn bot_commands() -> Vec<BotCommand>;
|
||||||
|
|
||||||
|
/// Returns `PhantomData<Self>` that is used as a param of [`commands_repl`]
|
||||||
|
///
|
||||||
|
/// [`commands_repl`]: (crate::repls2::commands_repl)
|
||||||
fn ty() -> PhantomData<Self> {
|
fn ty() -> PhantomData<Self> {
|
||||||
PhantomData
|
PhantomData
|
||||||
}
|
}
|
||||||
fn bot_commands() -> Vec<crate::types::BotCommand>;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub type PrefixedBotCommand = String;
|
pub type PrefixedBotCommand = String;
|
||||||
pub type BotName = String;
|
pub type BotName = String;
|
||||||
|
|
||||||
/// Errors returned from [`BotCommand::parse`].
|
/// Errors returned from [`BotCommands::parse`].
|
||||||
///
|
///
|
||||||
/// [`BotCommand::parse`]: crate::utils::command::BotCommand::parse
|
/// [`BotCommands::parse`]: BotCommands::parse
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub enum ParseError {
|
pub enum ParseError {
|
||||||
TooFewArguments {
|
TooFewArguments {
|
||||||
|
@ -247,28 +269,77 @@ pub enum ParseError {
|
||||||
Custom(Box<dyn Error + Send + Sync + 'static>),
|
Custom(Box<dyn Error + Send + Sync + 'static>),
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Display for ParseError {
|
/// Command descriptions that can be shown to the user (e.g. as a part of
|
||||||
fn fmt(&self, f: &mut Formatter<'_>) -> Result<(), std::fmt::Error> {
|
/// `/help` message)
|
||||||
match self {
|
///
|
||||||
ParseError::TooFewArguments { expected, found, message } => write!(
|
/// Most of the time you don't need to create this struct yourself as it's
|
||||||
f,
|
/// returned from [`BotCommands::descriptions`].
|
||||||
"Too few arguments (expected {}, found {}, message = '{}')",
|
#[derive(Debug, Clone)]
|
||||||
expected, found, message
|
pub struct CommandDescriptions<'a> {
|
||||||
),
|
global_description: Option<&'a str>,
|
||||||
ParseError::TooManyArguments { expected, found, message } => write!(
|
descriptions: &'a [CommandDescription<'a>],
|
||||||
f,
|
bot_username: Option<&'a str>,
|
||||||
"Too many arguments (expected {}, found {}, message = '{}')",
|
|
||||||
expected, found, message
|
|
||||||
),
|
|
||||||
ParseError::IncorrectFormat(e) => write!(f, "Incorrect format of command args: {}", e),
|
|
||||||
ParseError::UnknownCommand(e) => write!(f, "Unknown command: {}", e),
|
|
||||||
ParseError::WrongBotName(n) => write!(f, "Wrong bot name: {}", n),
|
|
||||||
ParseError::Custom(e) => write!(f, "{}", e),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl std::error::Error for ParseError {}
|
/// Description of a particular command, used in [`CommandDescriptions`].
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
pub struct CommandDescription<'a> {
|
||||||
|
/// Prefix of the command, usually `/`.
|
||||||
|
pub prefix: &'a str,
|
||||||
|
/// The command itself, e.g. `start`.
|
||||||
|
pub command: &'a str,
|
||||||
|
/// Human-readable description of the command.
|
||||||
|
pub description: &'a str,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> CommandDescriptions<'a> {
|
||||||
|
/// Creates new [`CommandDescriptions`] from a list of command descriptions.
|
||||||
|
pub fn new(descriptions: &'a [CommandDescription<'a>]) -> Self {
|
||||||
|
Self { global_description: None, descriptions, bot_username: None }
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Sets the global description of these commands.
|
||||||
|
pub fn global_description(self, global_description: &'a str) -> Self {
|
||||||
|
Self { global_description: Some(global_description), ..self }
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Sets the username of the bot.
|
||||||
|
///
|
||||||
|
/// After this method is called, returned instance of
|
||||||
|
/// [`CommandDescriptions`] will append `@bot_username` to all commands.
|
||||||
|
/// This is useful in groups, to disambiguate commands for different bots.
|
||||||
|
///
|
||||||
|
/// ## Examples
|
||||||
|
///
|
||||||
|
/// ```
|
||||||
|
/// use teloxide::utils::command::{CommandDescription, CommandDescriptions};
|
||||||
|
///
|
||||||
|
/// let descriptions = CommandDescriptions::new(&[
|
||||||
|
/// CommandDescription { prefix: "/", command: "start", description: "start this bot" },
|
||||||
|
/// CommandDescription { prefix: "/", command: "help", description: "show this message" },
|
||||||
|
/// ]);
|
||||||
|
///
|
||||||
|
/// assert_eq!(descriptions.to_string(), "/start — start this bot\n/help — show this message");
|
||||||
|
/// assert_eq!(
|
||||||
|
/// descriptions.username("username_of_the_bot").to_string(),
|
||||||
|
/// "/start@username_of_the_bot — start this bot\n/help@username_of_the_bot — show this \
|
||||||
|
/// message"
|
||||||
|
/// );
|
||||||
|
/// ```
|
||||||
|
pub fn username(self, bot_username: &'a str) -> Self {
|
||||||
|
Self { bot_username: Some(bot_username), ..self }
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Sets the username of the bot.
|
||||||
|
///
|
||||||
|
/// This is the same as [`username`], but uses value returned from `get_me`
|
||||||
|
/// method to get the username.
|
||||||
|
///
|
||||||
|
/// [`username`]: self::CommandDescriptions::username
|
||||||
|
pub fn username_from_me(self, me: &'a Me) -> CommandDescriptions<'a> {
|
||||||
|
self.username(me.user.username.as_deref().expect("Bots must have usernames"))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Parses a string into a command with args.
|
/// Parses a string into a command with args.
|
||||||
///
|
///
|
||||||
|
@ -346,6 +417,68 @@ where
|
||||||
Some((command, words.collect()))
|
Some((command, words.collect()))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl Display for ParseError {
|
||||||
|
fn fmt(&self, f: &mut Formatter<'_>) -> Result<(), std::fmt::Error> {
|
||||||
|
match self {
|
||||||
|
ParseError::TooFewArguments { expected, found, message } => write!(
|
||||||
|
f,
|
||||||
|
"Too few arguments (expected {}, found {}, message = '{}')",
|
||||||
|
expected, found, message
|
||||||
|
),
|
||||||
|
ParseError::TooManyArguments { expected, found, message } => write!(
|
||||||
|
f,
|
||||||
|
"Too many arguments (expected {}, found {}, message = '{}')",
|
||||||
|
expected, found, message
|
||||||
|
),
|
||||||
|
ParseError::IncorrectFormat(e) => write!(f, "Incorrect format of command args: {}", e),
|
||||||
|
ParseError::UnknownCommand(e) => write!(f, "Unknown command: {}", e),
|
||||||
|
ParseError::WrongBotName(n) => write!(f, "Wrong bot name: {}", n),
|
||||||
|
ParseError::Custom(e) => write!(f, "{}", e),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl std::error::Error for ParseError {}
|
||||||
|
|
||||||
|
impl Display for CommandDescriptions<'_> {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||||
|
if let Some(global_description) = self.global_description {
|
||||||
|
f.write_str(global_description)?;
|
||||||
|
f.write_str("\n\n")?;
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut write = |&CommandDescription { prefix, command, description }, nls| {
|
||||||
|
if nls {
|
||||||
|
f.write_char('\n')?;
|
||||||
|
}
|
||||||
|
|
||||||
|
f.write_str(prefix)?;
|
||||||
|
f.write_str(command)?;
|
||||||
|
|
||||||
|
if let Some(username) = self.bot_username {
|
||||||
|
f.write_char('@')?;
|
||||||
|
f.write_str(username)?;
|
||||||
|
}
|
||||||
|
|
||||||
|
if !description.is_empty() {
|
||||||
|
f.write_str(" — ")?;
|
||||||
|
f.write_str(description)?;
|
||||||
|
}
|
||||||
|
|
||||||
|
fmt::Result::Ok(())
|
||||||
|
};
|
||||||
|
|
||||||
|
if let Some(descr) = self.descriptions.first() {
|
||||||
|
write(descr, false)?;
|
||||||
|
for descr in &self.descriptions[1..] {
|
||||||
|
write(descr, true)?;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// The rest of tests are integrational due to problems with macro expansion in
|
// The rest of tests are integrational due to problems with macro expansion in
|
||||||
// unit tests.
|
// unit tests.
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
|
|
|
@ -12,7 +12,9 @@ use tokio::sync::Notify;
|
||||||
|
|
||||||
use crate::dispatching::update_listeners::UpdateListener;
|
use crate::dispatching::update_listeners::UpdateListener;
|
||||||
|
|
||||||
/// A token which used to shutdown [`crate::dispatching::Dispatcher`].
|
/// A token which used to shutdown [`Dispatcher`].
|
||||||
|
///
|
||||||
|
/// [`Dispatcher`]: crate::dispatching::Dispatcher
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
pub struct ShutdownToken {
|
pub struct ShutdownToken {
|
||||||
dispatcher_state: Arc<DispatcherState>,
|
dispatcher_state: Arc<DispatcherState>,
|
||||||
|
@ -20,7 +22,9 @@ pub struct ShutdownToken {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// This error is returned from [`ShutdownToken::shutdown`] when trying to
|
/// This error is returned from [`ShutdownToken::shutdown`] when trying to
|
||||||
/// shutdown an idle [`crate::dispatching::Dispatcher`].
|
/// shutdown an idle [`Dispatcher`].
|
||||||
|
///
|
||||||
|
/// [`Dispatcher`]: crate::dispatching::Dispatcher
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct IdleShutdownError;
|
pub struct IdleShutdownError;
|
||||||
|
|
||||||
|
|
|
@ -2,7 +2,7 @@
|
||||||
#![allow(clippy::nonstandard_macro_braces)]
|
#![allow(clippy::nonstandard_macro_braces)]
|
||||||
|
|
||||||
#[cfg(feature = "macros")]
|
#[cfg(feature = "macros")]
|
||||||
use teloxide::utils::command::{BotCommand, ParseError};
|
use teloxide::utils::command::{BotCommands, ParseError};
|
||||||
|
|
||||||
// We put tests here because macro expand in unit tests in module
|
// We put tests here because macro expand in unit tests in module
|
||||||
// teloxide::utils::command was a failure
|
// teloxide::utils::command was a failure
|
||||||
|
@ -10,7 +10,7 @@ use teloxide::utils::command::{BotCommand, ParseError};
|
||||||
#[test]
|
#[test]
|
||||||
#[cfg(feature = "macros")]
|
#[cfg(feature = "macros")]
|
||||||
fn parse_command_with_args() {
|
fn parse_command_with_args() {
|
||||||
#[derive(BotCommand, Debug, PartialEq)]
|
#[derive(BotCommands, Debug, PartialEq)]
|
||||||
#[command(rename = "lowercase")]
|
#[command(rename = "lowercase")]
|
||||||
enum DefaultCommands {
|
enum DefaultCommands {
|
||||||
Start(String),
|
Start(String),
|
||||||
|
@ -26,7 +26,7 @@ fn parse_command_with_args() {
|
||||||
#[test]
|
#[test]
|
||||||
#[cfg(feature = "macros")]
|
#[cfg(feature = "macros")]
|
||||||
fn parse_command_with_non_string_arg() {
|
fn parse_command_with_non_string_arg() {
|
||||||
#[derive(BotCommand, Debug, PartialEq)]
|
#[derive(BotCommands, Debug, PartialEq)]
|
||||||
#[command(rename = "lowercase")]
|
#[command(rename = "lowercase")]
|
||||||
enum DefaultCommands {
|
enum DefaultCommands {
|
||||||
Start(i32),
|
Start(i32),
|
||||||
|
@ -42,7 +42,7 @@ fn parse_command_with_non_string_arg() {
|
||||||
#[test]
|
#[test]
|
||||||
#[cfg(feature = "macros")]
|
#[cfg(feature = "macros")]
|
||||||
fn attribute_prefix() {
|
fn attribute_prefix() {
|
||||||
#[derive(BotCommand, Debug, PartialEq)]
|
#[derive(BotCommands, Debug, PartialEq)]
|
||||||
#[command(rename = "lowercase")]
|
#[command(rename = "lowercase")]
|
||||||
enum DefaultCommands {
|
enum DefaultCommands {
|
||||||
#[command(prefix = "!")]
|
#[command(prefix = "!")]
|
||||||
|
@ -59,7 +59,7 @@ fn attribute_prefix() {
|
||||||
#[test]
|
#[test]
|
||||||
#[cfg(feature = "macros")]
|
#[cfg(feature = "macros")]
|
||||||
fn many_attributes() {
|
fn many_attributes() {
|
||||||
#[derive(BotCommand, Debug, PartialEq)]
|
#[derive(BotCommands, Debug, PartialEq)]
|
||||||
#[command(rename = "lowercase")]
|
#[command(rename = "lowercase")]
|
||||||
enum DefaultCommands {
|
enum DefaultCommands {
|
||||||
#[command(prefix = "!", description = "desc")]
|
#[command(prefix = "!", description = "desc")]
|
||||||
|
@ -68,13 +68,13 @@ fn many_attributes() {
|
||||||
}
|
}
|
||||||
|
|
||||||
assert_eq!(DefaultCommands::Start, DefaultCommands::parse("!start", "").unwrap());
|
assert_eq!(DefaultCommands::Start, DefaultCommands::parse("!start", "").unwrap());
|
||||||
assert_eq!(DefaultCommands::descriptions(), "!start - desc\n/help\n");
|
assert_eq!(DefaultCommands::descriptions().to_string(), "!start — desc\n/help");
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
#[cfg(feature = "macros")]
|
#[cfg(feature = "macros")]
|
||||||
fn global_attributes() {
|
fn global_attributes() {
|
||||||
#[derive(BotCommand, Debug, PartialEq)]
|
#[derive(BotCommands, Debug, PartialEq)]
|
||||||
#[command(prefix = "!", rename = "lowercase", description = "Bot commands")]
|
#[command(prefix = "!", rename = "lowercase", description = "Bot commands")]
|
||||||
enum DefaultCommands {
|
enum DefaultCommands {
|
||||||
#[command(prefix = "/")]
|
#[command(prefix = "/")]
|
||||||
|
@ -84,13 +84,13 @@ fn global_attributes() {
|
||||||
|
|
||||||
assert_eq!(DefaultCommands::Start, DefaultCommands::parse("/start", "MyNameBot").unwrap());
|
assert_eq!(DefaultCommands::Start, DefaultCommands::parse("/start", "MyNameBot").unwrap());
|
||||||
assert_eq!(DefaultCommands::Help, DefaultCommands::parse("!help", "MyNameBot").unwrap());
|
assert_eq!(DefaultCommands::Help, DefaultCommands::parse("!help", "MyNameBot").unwrap());
|
||||||
assert_eq!(DefaultCommands::descriptions(), "Bot commands\n/start\n!help\n");
|
assert_eq!(DefaultCommands::descriptions().to_string(), "Bot commands\n\n/start\n!help");
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
#[cfg(feature = "macros")]
|
#[cfg(feature = "macros")]
|
||||||
fn parse_command_with_bot_name() {
|
fn parse_command_with_bot_name() {
|
||||||
#[derive(BotCommand, Debug, PartialEq)]
|
#[derive(BotCommands, Debug, PartialEq)]
|
||||||
#[command(rename = "lowercase")]
|
#[command(rename = "lowercase")]
|
||||||
enum DefaultCommands {
|
enum DefaultCommands {
|
||||||
#[command(prefix = "/")]
|
#[command(prefix = "/")]
|
||||||
|
@ -107,7 +107,7 @@ fn parse_command_with_bot_name() {
|
||||||
#[test]
|
#[test]
|
||||||
#[cfg(feature = "macros")]
|
#[cfg(feature = "macros")]
|
||||||
fn parse_with_split() {
|
fn parse_with_split() {
|
||||||
#[derive(BotCommand, Debug, PartialEq)]
|
#[derive(BotCommands, Debug, PartialEq)]
|
||||||
#[command(rename = "lowercase")]
|
#[command(rename = "lowercase")]
|
||||||
#[command(parse_with = "split")]
|
#[command(parse_with = "split")]
|
||||||
enum DefaultCommands {
|
enum DefaultCommands {
|
||||||
|
@ -124,7 +124,7 @@ fn parse_with_split() {
|
||||||
#[test]
|
#[test]
|
||||||
#[cfg(feature = "macros")]
|
#[cfg(feature = "macros")]
|
||||||
fn parse_with_split2() {
|
fn parse_with_split2() {
|
||||||
#[derive(BotCommand, Debug, PartialEq)]
|
#[derive(BotCommands, Debug, PartialEq)]
|
||||||
#[command(rename = "lowercase")]
|
#[command(rename = "lowercase")]
|
||||||
#[command(parse_with = "split", separator = "|")]
|
#[command(parse_with = "split", separator = "|")]
|
||||||
enum DefaultCommands {
|
enum DefaultCommands {
|
||||||
|
@ -152,7 +152,7 @@ fn parse_custom_parser() {
|
||||||
.map_err(|_| ParseError::Custom("First argument must be a integer!".to_owned().into()))
|
.map_err(|_| ParseError::Custom("First argument must be a integer!".to_owned().into()))
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(BotCommand, Debug, PartialEq)]
|
#[derive(BotCommands, Debug, PartialEq)]
|
||||||
#[command(rename = "lowercase")]
|
#[command(rename = "lowercase")]
|
||||||
enum DefaultCommands {
|
enum DefaultCommands {
|
||||||
#[command(parse_with = "custom_parse_function")]
|
#[command(parse_with = "custom_parse_function")]
|
||||||
|
@ -169,7 +169,7 @@ fn parse_custom_parser() {
|
||||||
#[test]
|
#[test]
|
||||||
#[cfg(feature = "macros")]
|
#[cfg(feature = "macros")]
|
||||||
fn parse_named_fields() {
|
fn parse_named_fields() {
|
||||||
#[derive(BotCommand, Debug, PartialEq)]
|
#[derive(BotCommands, Debug, PartialEq)]
|
||||||
#[command(rename = "lowercase")]
|
#[command(rename = "lowercase")]
|
||||||
#[command(parse_with = "split")]
|
#[command(parse_with = "split")]
|
||||||
enum DefaultCommands {
|
enum DefaultCommands {
|
||||||
|
@ -186,7 +186,7 @@ fn parse_named_fields() {
|
||||||
#[test]
|
#[test]
|
||||||
#[cfg(feature = "macros")]
|
#[cfg(feature = "macros")]
|
||||||
fn descriptions_off() {
|
fn descriptions_off() {
|
||||||
#[derive(BotCommand, Debug, PartialEq)]
|
#[derive(BotCommands, Debug, PartialEq)]
|
||||||
#[command(rename = "lowercase")]
|
#[command(rename = "lowercase")]
|
||||||
enum DefaultCommands {
|
enum DefaultCommands {
|
||||||
#[command(description = "off")]
|
#[command(description = "off")]
|
||||||
|
@ -194,13 +194,13 @@ fn descriptions_off() {
|
||||||
Help,
|
Help,
|
||||||
}
|
}
|
||||||
|
|
||||||
assert_eq!(DefaultCommands::descriptions(), "/help\n".to_owned());
|
assert_eq!(DefaultCommands::descriptions().to_string(), "/help".to_owned());
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
#[cfg(feature = "macros")]
|
#[cfg(feature = "macros")]
|
||||||
fn rename_rules() {
|
fn rename_rules() {
|
||||||
#[derive(BotCommand, Debug, PartialEq)]
|
#[derive(BotCommands, Debug, PartialEq)]
|
||||||
enum DefaultCommands {
|
enum DefaultCommands {
|
||||||
#[command(rename = "lowercase")]
|
#[command(rename = "lowercase")]
|
||||||
AaaAaa,
|
AaaAaa,
|
||||||
|
@ -230,7 +230,7 @@ fn rename_rules() {
|
||||||
assert_eq!(DefaultCommands::HhhHhh, DefaultCommands::parse("/HHH-HHH", "").unwrap());
|
assert_eq!(DefaultCommands::HhhHhh, DefaultCommands::parse("/HHH-HHH", "").unwrap());
|
||||||
|
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
"/aaaaaa\n/BBBBBB\n/CccCcc\n/dddDdd\n/eee_eee\n/FFF_FFF\n/ggg-ggg\n/HHH-HHH\n",
|
"/aaaaaa\n/BBBBBB\n/CccCcc\n/dddDdd\n/eee_eee\n/FFF_FFF\n/ggg-ggg\n/HHH-HHH",
|
||||||
DefaultCommands::descriptions()
|
DefaultCommands::descriptions().to_string()
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,6 +5,7 @@ use std::{
|
||||||
use teloxide::dispatching::dialogue::{RedisStorage, RedisStorageError, Serializer, Storage};
|
use teloxide::dispatching::dialogue::{RedisStorage, RedisStorageError, Serializer, Storage};
|
||||||
|
|
||||||
#[tokio::test]
|
#[tokio::test]
|
||||||
|
#[cfg_attr(not(CI_REDIS), ignore)]
|
||||||
async fn test_redis_json() {
|
async fn test_redis_json() {
|
||||||
let storage = RedisStorage::open(
|
let storage = RedisStorage::open(
|
||||||
"redis://127.0.0.1:7777",
|
"redis://127.0.0.1:7777",
|
||||||
|
@ -16,6 +17,7 @@ async fn test_redis_json() {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tokio::test]
|
#[tokio::test]
|
||||||
|
#[cfg_attr(not(CI_REDIS), ignore)]
|
||||||
async fn test_redis_bincode() {
|
async fn test_redis_bincode() {
|
||||||
let storage = RedisStorage::open(
|
let storage = RedisStorage::open(
|
||||||
"redis://127.0.0.1:7778",
|
"redis://127.0.0.1:7778",
|
||||||
|
@ -27,6 +29,7 @@ async fn test_redis_bincode() {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tokio::test]
|
#[tokio::test]
|
||||||
|
#[cfg_attr(not(CI_REDIS), ignore)]
|
||||||
async fn test_redis_cbor() {
|
async fn test_redis_cbor() {
|
||||||
let storage = RedisStorage::open(
|
let storage = RedisStorage::open(
|
||||||
"redis://127.0.0.1:7779",
|
"redis://127.0.0.1:7779",
|
||||||
|
|
|
@ -1,36 +1,47 @@
|
||||||
use std::{
|
use std::{
|
||||||
fmt::{Debug, Display},
|
fmt::{Debug, Display},
|
||||||
|
fs,
|
||||||
sync::Arc,
|
sync::Arc,
|
||||||
};
|
};
|
||||||
use teloxide::dispatching::dialogue::{Serializer, SqliteStorage, SqliteStorageError, Storage};
|
use teloxide::dispatching::dialogue::{Serializer, SqliteStorage, SqliteStorageError, Storage};
|
||||||
|
|
||||||
#[tokio::test(flavor = "multi_thread")]
|
#[tokio::test(flavor = "multi_thread")]
|
||||||
async fn test_sqlite_json() {
|
async fn test_sqlite_json() {
|
||||||
let storage =
|
fs::create_dir("./test_db1").unwrap();
|
||||||
SqliteStorage::open("./test_db1.sqlite", teloxide::dispatching::dialogue::serializer::Json)
|
let storage = SqliteStorage::open(
|
||||||
|
"./test_db1/test_db1.sqlite",
|
||||||
|
teloxide::dispatching::dialogue::serializer::Json,
|
||||||
|
)
|
||||||
.await
|
.await
|
||||||
.unwrap();
|
.unwrap();
|
||||||
test_sqlite(storage).await;
|
test_sqlite(storage).await;
|
||||||
|
fs::remove_dir_all("./test_db1").unwrap();
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tokio::test(flavor = "multi_thread")]
|
#[tokio::test(flavor = "multi_thread")]
|
||||||
async fn test_sqlite_bincode() {
|
async fn test_sqlite_bincode() {
|
||||||
|
fs::create_dir("./test_db2").unwrap();
|
||||||
let storage = SqliteStorage::open(
|
let storage = SqliteStorage::open(
|
||||||
"./test_db2.sqlite",
|
"./test_db2/test_db2.sqlite",
|
||||||
teloxide::dispatching::dialogue::serializer::Bincode,
|
teloxide::dispatching::dialogue::serializer::Bincode,
|
||||||
)
|
)
|
||||||
.await
|
.await
|
||||||
.unwrap();
|
.unwrap();
|
||||||
test_sqlite(storage).await;
|
test_sqlite(storage).await;
|
||||||
|
fs::remove_dir_all("./test_db2").unwrap();
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tokio::test(flavor = "multi_thread")]
|
#[tokio::test(flavor = "multi_thread")]
|
||||||
async fn test_sqlite_cbor() {
|
async fn test_sqlite_cbor() {
|
||||||
let storage =
|
fs::create_dir("./test_db3").unwrap();
|
||||||
SqliteStorage::open("./test_db3.sqlite", teloxide::dispatching::dialogue::serializer::Cbor)
|
let storage = SqliteStorage::open(
|
||||||
|
"./test_db3/test_db3.sqlite",
|
||||||
|
teloxide::dispatching::dialogue::serializer::Cbor,
|
||||||
|
)
|
||||||
.await
|
.await
|
||||||
.unwrap();
|
.unwrap();
|
||||||
test_sqlite(storage).await;
|
test_sqlite(storage).await;
|
||||||
|
fs::remove_dir_all("./test_db3").unwrap();
|
||||||
}
|
}
|
||||||
|
|
||||||
type Dialogue = String;
|
type Dialogue = String;
|
||||||
|
|
Loading…
Reference in a new issue