mirror of
https://github.com/teloxide/teloxide.git
synced 2025-01-03 17:52:12 +01:00
commit
4a32963901
47 changed files with 1313 additions and 981 deletions
8
.github/workflows/ci.yml
vendored
8
.github/workflows/ci.yml
vendored
|
@ -15,12 +15,12 @@ env:
|
||||||
CARGO_NET_RETRY: 10
|
CARGO_NET_RETRY: 10
|
||||||
RUSTUP_MAX_RETRIES: 10
|
RUSTUP_MAX_RETRIES: 10
|
||||||
|
|
||||||
rust_nightly: nightly-2022-07-01
|
rust_nightly: nightly-2022-09-01
|
||||||
# When updating this, also update:
|
# When updating this, also update:
|
||||||
# - README.md
|
# - README.md
|
||||||
# - src/lib.rs
|
# - src/lib.rs
|
||||||
# - down below in a matrix
|
# - down below in a matrix
|
||||||
rust_msrv: 1.58.0
|
rust_msrv: 1.64.0
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
# Depends on all action that are required for a "successful" CI run.
|
# Depends on all action that are required for a "successful" CI run.
|
||||||
|
@ -82,10 +82,10 @@ jobs:
|
||||||
toolchain: beta
|
toolchain: beta
|
||||||
features: "--features full"
|
features: "--features full"
|
||||||
- rust: nightly
|
- rust: nightly
|
||||||
toolchain: nightly-2022-07-01
|
toolchain: nightly-2022-09-01
|
||||||
features: "--all-features"
|
features: "--all-features"
|
||||||
- rust: msrv
|
- rust: msrv
|
||||||
toolchain: 1.58.0
|
toolchain: 1.64.0
|
||||||
features: "--features full"
|
features: "--features full"
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
|
|
47
CHANGELOG.md
47
CHANGELOG.md
|
@ -6,6 +6,34 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
||||||
|
|
||||||
## unreleased
|
## unreleased
|
||||||
|
|
||||||
|
## 0.11.0 - 2022-10-07
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
|
||||||
|
- Updated `teloxide-macros` to v0.7.0; see its [changelog](https://github.com/teloxide/teloxide-macros/blob/master/CHANGELOG.md#070---2022-10-06) for more
|
||||||
|
- Updated `teloxide-core` to v0.8.0; see its [changelog](https://github.com/teloxide/teloxide-core/blob/master/CHANGELOG.md#080---2022-10-03) for more
|
||||||
|
- `UpdateListener` now has an associated type `Err` instead of a generic
|
||||||
|
- `AsUpdateStream` now has an associated type `StreamErr` instead of a generic
|
||||||
|
- Rename `dispatching::stop_token::{AsyncStopToken, AsyncStopFlag}` => `stop::{StopToken, StopFlag}`
|
||||||
|
- Replace the generic error type `E` with `RequestError` for REPLs (`repl(_with_listener)`, `commands_repl(_with_listener)`)
|
||||||
|
- The following functions are now `#[must_use]`:
|
||||||
|
- `BotCommands::ty`.
|
||||||
|
- `CommandDescriptions::{new, global_description, username, username_from_me}`.
|
||||||
|
- `teloxide::filter_command`.
|
||||||
|
- `teloxide::dispatching::dialogue::enter`.
|
||||||
|
- `BotCommands::parse` now accept `bot_username` as `&str`
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- `requests::ResponseResult` to `prelude`
|
||||||
|
|
||||||
|
### Removed
|
||||||
|
|
||||||
|
- `dispatching::stop_token::StopToken` trait (all uses are replaced with `stop::StopToken` structure)
|
||||||
|
- Some previously deprecated items
|
||||||
|
- `enable_logging!`, `enable_logging_with_filter!`
|
||||||
|
- `HandlerFactory`, `HandlerExt::dispatch_by`
|
||||||
|
|
||||||
## 0.10.1 - 2022-07-22
|
## 0.10.1 - 2022-07-22
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
@ -34,7 +62,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
||||||
|
|
||||||
- Add the `Key: Clone` requirement for `impl Dispatcher` [**BC**].
|
- Add the `Key: Clone` requirement for `impl Dispatcher` [**BC**].
|
||||||
- `dispatching::update_listeners::{polling_default, polling}` now return a named, `Polling<_>` type.
|
- `dispatching::update_listeners::{polling_default, polling}` now return a named, `Polling<_>` type.
|
||||||
- Update teloxide-core to v0.7.0 with Bot API 6.1 support, see [its changelog][core07c] for more information [**BC**].
|
- Update `teloxide-core` to v0.7.0 with Bot API 6.1 support, see [its changelog][core07c] for more information [**BC**].
|
||||||
|
|
||||||
[core07c]: https://github.com/teloxide/teloxide-core/blob/master/CHANGELOG.md#070---2022-07-19
|
[core07c]: https://github.com/teloxide/teloxide-core/blob/master/CHANGELOG.md#070---2022-07-19
|
||||||
|
|
||||||
|
@ -64,7 +92,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
||||||
|
|
||||||
### Changed
|
### Changed
|
||||||
|
|
||||||
- Update teloxide-core to v0.6.0 with [Bot API 6.0] support [**BC**].
|
- Update `teloxide-core` to v0.6.0 with [Bot API 6.0] support [**BC**].
|
||||||
|
|
||||||
[Bot API 6.0]: https://core.telegram.org/bots/api#april-16-2022
|
[Bot API 6.0]: https://core.telegram.org/bots/api#april-16-2022
|
||||||
|
|
||||||
|
@ -244,6 +272,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
||||||
## 0.4.0 - 2021-03-22
|
## 0.4.0 - 2021-03-22
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
- Integrate [teloxide-core].
|
- Integrate [teloxide-core].
|
||||||
- Allow arbitrary error types to be returned from (sub)transitions ([issue 242](https://github.com/teloxide/teloxide/issues/242)).
|
- Allow arbitrary error types to be returned from (sub)transitions ([issue 242](https://github.com/teloxide/teloxide/issues/242)).
|
||||||
- The `respond` function, a shortcut for `ResponseResult::Ok(())`.
|
- The `respond` function, a shortcut for `ResponseResult::Ok(())`.
|
||||||
|
@ -261,6 +290,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
||||||
- Hide `SubtransitionOutputType` from the docs.
|
- Hide `SubtransitionOutputType` from the docs.
|
||||||
|
|
||||||
### Changed
|
### Changed
|
||||||
|
|
||||||
- Export `teloxide_macros::teloxide` in `prelude`.
|
- Export `teloxide_macros::teloxide` in `prelude`.
|
||||||
- `dispatching::dialogue::serializer::{JSON -> Json, CBOR -> Cbor}`
|
- `dispatching::dialogue::serializer::{JSON -> Json, CBOR -> Cbor}`
|
||||||
- Allow `bot_name` be `N`, where `N: Into<String> + ...` in `commands_repl` & `commands_repl_with_listener`.
|
- Allow `bot_name` be `N`, where `N: Into<String> + ...` in `commands_repl` & `commands_repl_with_listener`.
|
||||||
|
@ -287,20 +317,25 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
||||||
## 0.3.3 - 2020-10-30
|
## 0.3.3 - 2020-10-30
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
|
||||||
- The `dice` field from `MessageDice` is public now ([issue 306](https://github.com/teloxide/teloxide/issues/306))
|
- The `dice` field from `MessageDice` is public now ([issue 306](https://github.com/teloxide/teloxide/issues/306))
|
||||||
|
|
||||||
## 0.3.2 - 2020-10-23
|
## 0.3.2 - 2020-10-23
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
- `LoginUrl::new` ([issue 298](https://github.com/teloxide/teloxide/issues/298))
|
- `LoginUrl::new` ([issue 298](https://github.com/teloxide/teloxide/issues/298))
|
||||||
|
|
||||||
## 0.3.1 - 2020-08-25
|
## 0.3.1 - 2020-08-25
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
- `Bot::builder` method ([PR 269](https://github.com/teloxide/teloxide/pull/269)).
|
- `Bot::builder` method ([PR 269](https://github.com/teloxide/teloxide/pull/269)).
|
||||||
|
|
||||||
## 0.3.0 - 2020-07-31
|
## 0.3.0 - 2020-07-31
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
- Support for typed bot commands ([issue 152](https://github.com/teloxide/teloxide/issues/152)).
|
- Support for typed bot commands ([issue 152](https://github.com/teloxide/teloxide/issues/152)).
|
||||||
- `BotBuilder`, which allows setting a default `ParseMode`.
|
- `BotBuilder`, which allows setting a default `ParseMode`.
|
||||||
- The `Transition`, `Subtransition`, `SubtransitionOutputType` traits.
|
- The `Transition`, `Subtransition`, `SubtransitionOutputType` traits.
|
||||||
|
@ -322,9 +357,11 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
||||||
- Respect the `TELOXIDE_PROXY` environment variable in `Bot::from_env`.
|
- Respect the `TELOXIDE_PROXY` environment variable in `Bot::from_env`.
|
||||||
|
|
||||||
### Deprecated
|
### Deprecated
|
||||||
|
|
||||||
- `Bot::{from_env_with_client, new, with_client}`
|
- `Bot::{from_env_with_client, new, with_client}`
|
||||||
|
|
||||||
### Changed
|
### Changed
|
||||||
|
|
||||||
- `DialogueDispatcherHandlerCx` -> `DialogueWithCx`.
|
- `DialogueDispatcherHandlerCx` -> `DialogueWithCx`.
|
||||||
- `DispatcherHandlerCx` -> `UpdateWithCx`.
|
- `DispatcherHandlerCx` -> `UpdateWithCx`.
|
||||||
- Now provided description of unknown telegram error, by splitting ApiErrorKind at `ApiErrorKind` and `ApiErrorKindKnown` enums ([issue 199](https://github.com/teloxide/teloxide/issues/199)).
|
- Now provided description of unknown telegram error, by splitting ApiErrorKind at `ApiErrorKind` and `ApiErrorKindKnown` enums ([issue 199](https://github.com/teloxide/teloxide/issues/199)).
|
||||||
|
@ -333,24 +370,30 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
||||||
- Replace all `mime_type: String` with `MimeWrapper`.
|
- Replace all `mime_type: String` with `MimeWrapper`.
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
|
||||||
- Now methods which can send file to Telegram returns `tokio::io::Result<T>`. Early its could panic ([issue 216](https://github.com/teloxide/teloxide/issues/216)).
|
- Now methods which can send file to Telegram returns `tokio::io::Result<T>`. Early its could panic ([issue 216](https://github.com/teloxide/teloxide/issues/216)).
|
||||||
- If a bot wasn't triggered for several days, it stops responding ([issue 223](https://github.com/teloxide/teloxide/issues/223)).
|
- If a bot wasn't triggered for several days, it stops responding ([issue 223](https://github.com/teloxide/teloxide/issues/223)).
|
||||||
|
|
||||||
## 0.2.0 - 2020-02-25
|
## 0.2.0 - 2020-02-25
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
- The functionality to parse commands only with a correct bot's name (breaks backwards compatibility) ([Issue 168](https://github.com/teloxide/teloxide/issues/168)).
|
- The functionality to parse commands only with a correct bot's name (breaks backwards compatibility) ([Issue 168](https://github.com/teloxide/teloxide/issues/168)).
|
||||||
- This `CHANGELOG.md`.
|
- This `CHANGELOG.md`.
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
|
||||||
- Fix parsing a pinned message ([Issue 167](https://github.com/teloxide/teloxide/issues/167)).
|
- Fix parsing a pinned message ([Issue 167](https://github.com/teloxide/teloxide/issues/167)).
|
||||||
- Replace `LanguageCode` with `String`, because [the official Telegram documentation](https://core.telegram.org/bots/api#getchat) doesn't specify a concrete version of IETF language tag.
|
- Replace `LanguageCode` with `String`, because [the official Telegram documentation](https://core.telegram.org/bots/api#getchat) doesn't specify a concrete version of IETF language tag.
|
||||||
- Problems with the `poll_type` field ([Issue 178](https://github.com/teloxide/teloxide/issues/178)).
|
- Problems with the `poll_type` field ([Issue 178](https://github.com/teloxide/teloxide/issues/178)).
|
||||||
- Make `polling_default` actually a long polling update listener ([PR 182](https://github.com/teloxide/teloxide/pull/182)).
|
- Make `polling_default` actually a long polling update listener ([PR 182](https://github.com/teloxide/teloxide/pull/182)).
|
||||||
|
|
||||||
### Removed
|
### Removed
|
||||||
|
|
||||||
- [either](https://crates.io/crates/either) from the dependencies in `Cargo.toml`.
|
- [either](https://crates.io/crates/either) from the dependencies in `Cargo.toml`.
|
||||||
- `teloxide-macros` migrated into [the separate repository](https://github.com/teloxide/teloxide-macros) to easier releases and testing.
|
- `teloxide-macros` migrated into [the separate repository](https://github.com/teloxide/teloxide-macros) to easier releases and testing.
|
||||||
|
|
||||||
## 0.1.0 - 2020-02-19
|
## 0.1.0 - 2020-02-19
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
- This project.
|
- This project.
|
||||||
|
|
405
CODE_STYLE.md
405
CODE_STYLE.md
|
@ -1,49 +1,75 @@
|
||||||
# Code style
|
# Code style
|
||||||
This is a description of a coding style that every contributor must follow. Please, read the whole document before you start pushing code.
|
|
||||||
|
This is a description of a coding style that every contributor must follow.
|
||||||
|
Please, read the whole document before you start pushing code.
|
||||||
|
|
||||||
## Generics
|
## Generics
|
||||||
Generics are always written with `where`.
|
|
||||||
|
|
||||||
Bad:
|
All trait bounds should be written in `where`:
|
||||||
|
|
||||||
```rust
|
```rust
|
||||||
|
// GOOD
|
||||||
|
pub fn new<N, T, P, E>(user_id: i32, name: N, title: T, png_sticker: P, emojis: E) -> Self
|
||||||
|
where
|
||||||
|
N: Into<String>,
|
||||||
|
T: Into<String>,
|
||||||
|
P: Into<InputFile>,
|
||||||
|
E: Into<String>,
|
||||||
|
{ ... }
|
||||||
|
|
||||||
|
// BAD
|
||||||
pub fn new<N: Into<String>,
|
pub fn new<N: Into<String>,
|
||||||
T: Into<String>,
|
T: Into<String>,
|
||||||
P: Into<InputFile>,
|
P: Into<InputFile>,
|
||||||
E: Into<String>>
|
E: Into<String>>
|
||||||
(user_id: i32, name: N, title: T, png_sticker: P, emojis: E) -> Self { ... }
|
(user_id: i32, name: N, title: T, png_sticker: P, emojis: E) -> Self { ... }
|
||||||
```
|
```
|
||||||
|
|
||||||
Good:
|
|
||||||
|
|
||||||
```rust
|
```rust
|
||||||
pub fn new<N, T, P, E>(user_id: i32, name: N, title: T, png_sticker: P, emojis: E) -> Self
|
// GOOD
|
||||||
|
impl<T> Trait for Wrap<T>
|
||||||
where
|
where
|
||||||
N: Into<String>,
|
T: Trait
|
||||||
T: Into<String>,
|
{ ... }
|
||||||
P: Into<InputFile>,
|
|
||||||
E: Into<String> { ... }
|
// BAD
|
||||||
|
impl<T: Trait> Trait for Wrap<T> { ... }
|
||||||
```
|
```
|
||||||
|
|
||||||
## Comments
|
**Rationale:**
|
||||||
1. Comments must describe what your code does and mustn't describe how your code does it and bla-bla-bla. Be sure that your comments follow the grammar, including punctuation, the first capital letter and so on.
|
- `where` clauses are easier to read when there are a lot of bounds
|
||||||
|
- uniformity
|
||||||
|
|
||||||
Bad:
|
## Documentation comments
|
||||||
|
|
||||||
|
1. Documentation must describe _what_ your code does and mustn't describe _how_ your code does it and bla-bla-bla.
|
||||||
|
2. Be sure that your comments follow the grammar, including punctuation, the first capital letter and so on:
|
||||||
```rust
|
```rust
|
||||||
|
// GOOD
|
||||||
|
/// This function makes a request to Telegram.
|
||||||
|
pub fn make_request(url: &str) -> String { ... }
|
||||||
|
|
||||||
|
// BAD
|
||||||
/// this function make request to telegram
|
/// this function make request to telegram
|
||||||
pub fn make_request(url: &str) -> String { ... }
|
pub fn make_request(url: &str) -> String { ... }
|
||||||
```
|
```
|
||||||
|
3. Do not use ending punctuation in short list items (usually containing just one phrase or sentence):
|
||||||
|
```md
|
||||||
|
<!-- GOOD -->
|
||||||
|
- Handle different kinds of Update
|
||||||
|
- Pass dependencies to handlers
|
||||||
|
- Disable a default Ctrl-C handling
|
||||||
|
|
||||||
Good:
|
<!-- BAD -->
|
||||||
|
- Handle different kinds of Update.
|
||||||
|
- Pass dependencies to handlers.
|
||||||
|
- Disable a default Ctrl-C handling.
|
||||||
|
|
||||||
```rust
|
<!-- BAD -->
|
||||||
/// This function makes a request to Telegram.
|
- Handle different kinds of Update;
|
||||||
pub fn make_request(url: &str) -> String { ... }
|
- Pass dependencies to handlers;
|
||||||
|
- Disable a default Ctrl-C handling;
|
||||||
```
|
```
|
||||||
|
3. Link resources in your comments when possible:
|
||||||
2. Also, link resources in your comments when possible:
|
|
||||||
|
|
||||||
```rust
|
```rust
|
||||||
/// Download a file from Telegram.
|
/// Download a file from Telegram.
|
||||||
///
|
///
|
||||||
|
@ -56,21 +82,23 @@ pub fn make_request(url: &str) -> String { ... }
|
||||||
/// [`AsyncWrite`]: tokio::io::AsyncWrite
|
/// [`AsyncWrite`]: tokio::io::AsyncWrite
|
||||||
/// [`tokio::fs::File`]: tokio::fs::File
|
/// [`tokio::fs::File`]: tokio::fs::File
|
||||||
/// [`Bot::download_file`]: crate::Bot::download_file
|
/// [`Bot::download_file`]: crate::Bot::download_file
|
||||||
#[cfg(feature = "unstable-stream")]
|
|
||||||
pub async fn download_file_stream(
|
|
||||||
&self,
|
|
||||||
path: &str,
|
|
||||||
) -> Result<impl Stream<Item = Result<Bytes, reqwest::Error>>, reqwest::Error>
|
|
||||||
{
|
|
||||||
download_file_stream(&self.client, &self.token, path).await
|
|
||||||
}
|
|
||||||
```
|
```
|
||||||
|
4. Write `teloxide`, `teloxide-macros`, and `teloxide-core`, not "teloxide", "Teloxide", "teloxide-macros" or any other variant.
|
||||||
|
|
||||||
## Use Self where possible
|
## Use `Self` where possible
|
||||||
Bad:
|
|
||||||
|
When referring to the type for which block is implemented, prefer using `Self`, rather than the name of the type:
|
||||||
|
|
||||||
```rust
|
```rust
|
||||||
impl ErrorKind {
|
impl ErrorKind {
|
||||||
|
// GOOD
|
||||||
|
fn print(&self) {
|
||||||
|
Self::Io => println!("Io"),
|
||||||
|
Self::Network => println!("Network"),
|
||||||
|
Self::Json => println!("Json"),
|
||||||
|
}
|
||||||
|
|
||||||
|
// BAD
|
||||||
fn print(&self) {
|
fn print(&self) {
|
||||||
ErrorKind::Io => println!("Io"),
|
ErrorKind::Io => println!("Io"),
|
||||||
ErrorKind::Network => println!("Network"),
|
ErrorKind::Network => println!("Network"),
|
||||||
|
@ -78,50 +106,303 @@ impl ErrorKind {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
Good:
|
|
||||||
```rust
|
```rust
|
||||||
impl ErrorKind {
|
impl<'a> AnswerCallbackQuery<'a> {
|
||||||
fn print(&self) {
|
// GOOD
|
||||||
Self::Io => println!("Io"),
|
fn new<C>(bot: &'a Bot, callback_query_id: C) -> Self
|
||||||
Self::Network => println!("Network"),
|
where
|
||||||
Self::Json => println!("Json"),
|
C: Into<String>,
|
||||||
|
{ ... }
|
||||||
|
|
||||||
|
// BAD
|
||||||
|
fn new<C>(bot: &'a Bot, callback_query_id: C) -> AnswerCallbackQuery<'a>
|
||||||
|
where
|
||||||
|
C: Into<String>,
|
||||||
|
{ ... }
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Rationale:** `Self` is generally shorter and it's easier to copy-paste code or rename the type.
|
||||||
|
|
||||||
|
## Avoid duplication in fields names
|
||||||
|
|
||||||
|
```rust
|
||||||
|
struct Message {
|
||||||
|
// GOOD
|
||||||
|
#[serde(rename = "message_id")]
|
||||||
|
id: MessageId,
|
||||||
|
|
||||||
|
// BAD
|
||||||
|
message_id: MessageId,
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Rationale:** duplication blurs the focus of code, making it unnecessarily longer.
|
||||||
|
|
||||||
|
## Conventional generic names
|
||||||
|
|
||||||
|
Use a generic parameter name `S` for streams, `Fut` for futures, `F` for functions (where possible).
|
||||||
|
|
||||||
|
**Rationale:** uniformity.
|
||||||
|
|
||||||
|
## Deriving traits
|
||||||
|
|
||||||
|
Derive `Copy`, `Clone`, `Eq`, `PartialEq`, `Hash` and `Debug` for public types when possible.
|
||||||
|
|
||||||
|
**Rationale:** these traits can be useful for users and can be implemented for most types.
|
||||||
|
|
||||||
|
Derive `Default` when there is a reasonable default value for the type.
|
||||||
|
|
||||||
|
**Rationale:** `Default` plays nicely with generic code (for example, `mem::take`).
|
||||||
|
|
||||||
|
## `Into`-polymorphism
|
||||||
|
|
||||||
|
Use `T: Into<Ty>` when this can simplify user code.
|
||||||
|
I.e. when there are types that implement `Into<Ty>` that are likely to be passed to this function.
|
||||||
|
|
||||||
|
**Rationale:** conversions unnecessarily complicate caller code and can be confusing for beginners.
|
||||||
|
|
||||||
|
## `must_use`
|
||||||
|
|
||||||
|
Always mark functions as `#[must_use]` if they don't have side effects and the only reason to call them is to get the result:
|
||||||
|
|
||||||
|
```rust
|
||||||
|
impl User {
|
||||||
|
// GOOD
|
||||||
|
#[must_use]
|
||||||
|
fn full_name(&self) -> String {
|
||||||
|
format!("{} {}", user.first_name, user.last_name)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
<details>
|
**Rationale:** users will get warnings if they forgot to do something with the result, potentially preventing bugs.
|
||||||
<summary>More examples</summary>
|
|
||||||
|
|
||||||
Bad:
|
## Creating boxed futures
|
||||||
|
|
||||||
|
Prefer `Box::pin(async { ... })` instead of `async { ... }.boxed()`.
|
||||||
|
|
||||||
|
**Rationale:** the former is generally formatted better by rustfmt.
|
||||||
|
|
||||||
|
## Full paths for logging
|
||||||
|
|
||||||
|
Always write `log::<op>!(...)` instead of importing `use log::<op>;` and invoking `<op>!(...)`.
|
||||||
|
|
||||||
```rust
|
```rust
|
||||||
impl<'a> AnswerCallbackQuery<'a> {
|
// GOOD
|
||||||
pub(crate) fn new<C>(bot: &'a Bot, callback_query_id: C) -> AnswerCallbackQuery<'a>
|
log::warn!("Everything is on fire");
|
||||||
where
|
|
||||||
C: Into<String>, { ... }
|
// BAD
|
||||||
|
use log::warn;
|
||||||
|
|
||||||
|
warn!("Everything is on fire");
|
||||||
```
|
```
|
||||||
|
|
||||||
Good:
|
**Rationale:**
|
||||||
|
- Less polluted import blocks
|
||||||
|
- Uniformity
|
||||||
|
|
||||||
|
## `&str` -> `String` conversion
|
||||||
|
|
||||||
|
Prefer using `.to_owned()`, rather than `.to_string()`, `.into()`, `String::from`, etc.
|
||||||
|
|
||||||
|
**Rationale:** uniformity, intent clarity.
|
||||||
|
|
||||||
|
## Order of imports
|
||||||
|
|
||||||
|
Separate import groups with blank lines. Use one use per crate.
|
||||||
|
|
||||||
|
Module declarations come before the imports.
|
||||||
|
Order them in "suggested reading order" for a person new to the code base.
|
||||||
|
|
||||||
```rust
|
```rust
|
||||||
impl<'a> AnswerCallbackQuery<'a> {
|
mod x;
|
||||||
pub(crate) fn new<C>(bot: &'a Bot, callback_query_id: C) -> Self
|
mod y;
|
||||||
where
|
|
||||||
C: Into<String>, { ... }
|
// First std.
|
||||||
|
use std::{ ... }
|
||||||
|
|
||||||
|
// Second, external crates (both crates.io crates and other rust-analyzer crates).
|
||||||
|
use crate_foo::{ ... }
|
||||||
|
use crate_bar::{ ... }
|
||||||
|
|
||||||
|
// Then current crate.
|
||||||
|
use crate::{}
|
||||||
|
|
||||||
|
// Finally, parent and child modules, but prefer `use crate::`.
|
||||||
|
use super::{}
|
||||||
|
|
||||||
|
// Re-exports are treated as item definitions rather than imports, so they go
|
||||||
|
// after imports and modules. Use them sparingly.
|
||||||
|
pub use crate::x::Z;
|
||||||
```
|
```
|
||||||
</details>
|
|
||||||
|
|
||||||
## Naming
|
**Rationale:**
|
||||||
1. Avoid unnecessary duplication (`Message::message_id` -> `Message::id` using `#[serde(rename = "message_id")]`).
|
- Reading order is important for new contributors
|
||||||
2. Use a generic parameter name `S` for streams, `Fut` for futures, `F` for functions (where possible).
|
- Grouping by crate allows spotting unwanted dependencies easier
|
||||||
|
- Consistency
|
||||||
|
|
||||||
## Deriving
|
## Import Style
|
||||||
1. Derive `Copy`, `Eq`, `Hash`, `PartialEq`, `Clone`, `Debug` for public types when possible (note: if the default `Debug` implementation is weird, you should manually implement it by yourself).
|
|
||||||
2. Derive `Default` when there is an algorithm to get a default value for your type.
|
|
||||||
|
|
||||||
## Misc
|
When implementing traits from `std::fmt` import the module:
|
||||||
1. Use `Into<...>` only where there exists at least one conversion **and** it will be logically to use.
|
|
||||||
2. Always mark a function as `#[must_use]` if its return value **must** be used.
|
```rust
|
||||||
3. `Box::pin(async [move] { ... })` instead of `async [move] { ... }.boxed()`.
|
// GOOD
|
||||||
4. Always write `log::<op>!(...)` instead of importing `use log::<op>;` and invoking `<op>!(...)`. For example, write `log::info!("blah")`.
|
use std::fmt;
|
||||||
|
|
||||||
|
impl fmt::Display for RenameError {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { .. }
|
||||||
|
}
|
||||||
|
|
||||||
|
// BAD
|
||||||
|
impl std::fmt::Display for RenameError {
|
||||||
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { .. }
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Rationale:**
|
||||||
|
- Makes it clear that a trait is implemented, rather than used
|
||||||
|
- Less typing
|
||||||
|
|
||||||
|
Prefer `use crate::foo::bar` to `use super::bar` or `use self::bar::baz`. **Rationale:**
|
||||||
|
- Works in all cases
|
||||||
|
- Consistency
|
||||||
|
|
||||||
|
## Order of Items
|
||||||
|
|
||||||
|
Optimize for the reader who sees the file for the first time, and wants to get a general idea about what's going on. People read things from top to bottom, so place most important things first.
|
||||||
|
|
||||||
|
Specifically, if all items except one are private, always put the non-private item on top:
|
||||||
|
```rust
|
||||||
|
// GOOD
|
||||||
|
pub(crate) fn frobnicate() {
|
||||||
|
Helper::act()
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Default)]
|
||||||
|
struct Helper { stuff: i32 }
|
||||||
|
|
||||||
|
impl Helper {
|
||||||
|
fn act(&self) {
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// BAD
|
||||||
|
#[derive(Default)]
|
||||||
|
struct Helper { stuff: i32 }
|
||||||
|
|
||||||
|
pub(crate) fn frobnicate() {
|
||||||
|
Helper::act()
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Helper {
|
||||||
|
fn act(&self) {
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
If there's a mixture of private and public items, put public items first.
|
||||||
|
|
||||||
|
Put structs and enums first, functions and impls last. Order type declarations in a top-down manner:
|
||||||
|
|
||||||
|
```rust
|
||||||
|
// GOOD
|
||||||
|
struct Parent {
|
||||||
|
children: Vec<Child>
|
||||||
|
}
|
||||||
|
|
||||||
|
struct Child;
|
||||||
|
|
||||||
|
impl Parent {
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Child {
|
||||||
|
}
|
||||||
|
|
||||||
|
// BAD
|
||||||
|
struct Child;
|
||||||
|
|
||||||
|
impl Child {
|
||||||
|
}
|
||||||
|
|
||||||
|
struct Parent {
|
||||||
|
children: Vec<Child>
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Parent {
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Rationale:**
|
||||||
|
- Easier to get a sense of the API by visually scanning the file
|
||||||
|
- If function bodies are folded in the editor, the source code should be read as documentation for the public API
|
||||||
|
|
||||||
|
## Early Returns
|
||||||
|
|
||||||
|
Do use early returns:
|
||||||
|
|
||||||
|
```rust
|
||||||
|
// GOOD
|
||||||
|
fn foo() -> Option<Bar> {
|
||||||
|
if !condition() {
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
|
||||||
|
Some(...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// BAD
|
||||||
|
fn foo() -> Option<Bar> {
|
||||||
|
if condition() {
|
||||||
|
Some(...)
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Rationale:** reduce cognitive stack usage.
|
||||||
|
|
||||||
|
## If-let
|
||||||
|
|
||||||
|
Avoid the `if let ... { } else { }` construct, use `match` instead:
|
||||||
|
|
||||||
|
```rust
|
||||||
|
// GOOD
|
||||||
|
match ctx.expected_type.as_ref() {
|
||||||
|
Some(expected_type) => completion_ty == expected_type && !expected_type.is_unit(),
|
||||||
|
None => false,
|
||||||
|
}
|
||||||
|
|
||||||
|
// BAD
|
||||||
|
if let Some(expected_type) = ctx.expected_type.as_ref() {
|
||||||
|
completion_ty == expected_type && !expected_type.is_unit()
|
||||||
|
} else {
|
||||||
|
false
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Rationale:**
|
||||||
|
- `match` is almost always more compact
|
||||||
|
- The `else` branch can get a more precise pattern: `None` or `Err(_)` instead of `_`
|
||||||
|
|
||||||
|
## Empty Match Arms
|
||||||
|
|
||||||
|
Use `=> (),` when a match arm is intentionally empty:
|
||||||
|
```rust
|
||||||
|
// GOOD
|
||||||
|
match result {
|
||||||
|
Ok(_) => (),
|
||||||
|
Err(err) => error!("{}", err),
|
||||||
|
}
|
||||||
|
|
||||||
|
// BAD
|
||||||
|
match result {
|
||||||
|
Ok(_) => {}
|
||||||
|
Err(err) => error!("{}", err),
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Rationale:** consistency.
|
||||||
|
|
10
Cargo.toml
10
Cargo.toml
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "teloxide"
|
name = "teloxide"
|
||||||
version = "0.10.1"
|
version = "0.11.0"
|
||||||
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"
|
||||||
|
@ -57,8 +57,8 @@ full = [
|
||||||
]
|
]
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
teloxide-core = { version = "0.7.0", default-features = false }
|
teloxide-core = { version = "0.8.0", default-features = false }
|
||||||
teloxide-macros = { version = "0.6.3", optional = true }
|
teloxide-macros = { version = "0.7.0", optional = true }
|
||||||
|
|
||||||
serde_json = "1.0"
|
serde_json = "1.0"
|
||||||
serde = { version = "1.0", features = ["derive"] }
|
serde = { version = "1.0", features = ["derive"] }
|
||||||
|
@ -66,8 +66,8 @@ serde = { version = "1.0", features = ["derive"] }
|
||||||
dptree = "0.3.0"
|
dptree = "0.3.0"
|
||||||
|
|
||||||
# These lines are used only for development.
|
# These lines are used only for development.
|
||||||
# teloxide-core = { git = "https://github.com/teloxide/teloxide-core", rev = "b13393d", default-features = false }
|
# teloxide-core = { git = "https://github.com/teloxide/teloxide-core", rev = "00165e6", default-features = false }
|
||||||
# teloxide-macros = { git = "https://github.com/teloxide/teloxide-macros", rev = "44d91c5", optional = true }
|
# teloxide-macros = { git = "https://github.com/teloxide/teloxide-macros", rev = "e715105", optional = true }
|
||||||
# dptree = { git = "https://github.com/teloxide/dptree", rev = "df578e4" }
|
# dptree = { git = "https://github.com/teloxide/dptree", rev = "df578e4" }
|
||||||
|
|
||||||
tokio = { version = "1.8", features = ["fs"] }
|
tokio = { version = "1.8", features = ["fs"] }
|
||||||
|
|
|
@ -1,6 +1,108 @@
|
||||||
This document describes breaking changes of `teloxide` crate, as well as the ways to update code.
|
This document describes breaking changes of `teloxide` crate, as well as the ways to update code.
|
||||||
Note that the list of required changes is not fully exhaustive and it may lack something in rare cases.
|
Note that the list of required changes is not fully exhaustive and it may lack something in rare cases.
|
||||||
|
|
||||||
|
## 0.10 -> 0.11
|
||||||
|
|
||||||
|
### core
|
||||||
|
|
||||||
|
Requests can now be `.await`ed directly, without need of `.send()` or `AutoSend`.
|
||||||
|
If you previously used `AutoSend` adaptor, you can safely remove it:
|
||||||
|
|
||||||
|
```diff,rust
|
||||||
|
-let bot = Bot::from_env().auto_send();
|
||||||
|
+let bot = Bot::from_env();
|
||||||
|
```
|
||||||
|
|
||||||
|
```diff,rust
|
||||||
|
-async fn start(bot: AutoSend<Bot>, dialogue: MyDialogue, msg: Message) -> HandlerResult {
|
||||||
|
+async fn start(bot: Bot, dialogue: MyDialogue, msg: Message) -> HandlerResult {
|
||||||
|
```
|
||||||
|
|
||||||
|
`File`'s and `FileMeta`'s fields now don't have `file_` prefix.
|
||||||
|
If you previously accessed the fields, you'll need to change remove the prefix:
|
||||||
|
|
||||||
|
```diff
|
||||||
|
-_ = file.file_size;
|
||||||
|
+_ = file.size;
|
||||||
|
```
|
||||||
|
|
||||||
|
`Animation`, `Audio`, `Document`, `PassportFile`, `PhotoSize`, `Video`, `VideoNote` and `Voice` now contain `FileMeta` instead of its fields.
|
||||||
|
Together with rename of `FileMeta`'s fields, you'll need to change `_` to `.`:
|
||||||
|
|
||||||
|
```diff
|
||||||
|
-_ = animation.file_size;
|
||||||
|
+_ = animation.file.size;
|
||||||
|
```
|
||||||
|
|
||||||
|
Message id fields and parameters now use `MessageId` type, instead of `i32`.
|
||||||
|
You may need to change code accordingly:
|
||||||
|
|
||||||
|
```diff
|
||||||
|
-let id: i32 = message.id;
|
||||||
|
+let id: MessageId = message.id;
|
||||||
|
```
|
||||||
|
```diff,rust
|
||||||
|
let (cid, mid): (ChatId, i32) = get_message_to_delete_from_db();
|
||||||
|
-bot.delete_message(cid, mid).await?;
|
||||||
|
+bot.delete_message(cid, MessageId(mid)).await?;
|
||||||
|
```
|
||||||
|
|
||||||
|
Note that at the same time `MessageId` is now a tuple struct.
|
||||||
|
If you've accessed its only field you'll need to change it too:
|
||||||
|
|
||||||
|
```diff,rust
|
||||||
|
-let MessageId { message_id } = bot.copy_message(dst_chat, src_chat, mid).await?;
|
||||||
|
+let MessageId(message_id) = bot.copy_message(dst_chat, src_chat, mid).await?;
|
||||||
|
save_to_db(message_id);
|
||||||
|
```
|
||||||
|
|
||||||
|
Because of API updates `Sticker` type was refactored again.
|
||||||
|
You may need to change code accordingly.
|
||||||
|
See `Sticker` documentation for more information about the new structure.
|
||||||
|
|
||||||
|
### teloxide
|
||||||
|
|
||||||
|
You can now write `Ok(())` instead of `respond(())` at the end of closures provided to RELPs:
|
||||||
|
|
||||||
|
```diff,rust
|
||||||
|
teloxide::repl(bot, |bot: Bot, msg: Message| async move {
|
||||||
|
bot.send_dice(msg.chat.id).await?;
|
||||||
|
- respond(())
|
||||||
|
+ Ok(())
|
||||||
|
})
|
||||||
|
.await;
|
||||||
|
```
|
||||||
|
|
||||||
|
This is because REPLs now require the closure to return `RequestError` instead of a generic error type, so type inference works perfectly for a return value. If you use something other than `RequestError`, you can transfer your code to `teloxide::dispatching`, which still permits a generic error type.
|
||||||
|
|
||||||
|
### macros
|
||||||
|
|
||||||
|
`parse_with` now accepts a Rust _path_ to a custom parser function instead of a string:
|
||||||
|
|
||||||
|
```diff,rust
|
||||||
|
fn custom_parser(input: String) -> Result<(u8,), ParseError> {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(BotCommands)]
|
||||||
|
enum Command {
|
||||||
|
- #[command(parse_with = "custom_parser")]
|
||||||
|
+ #[command(parse_with = custom_parser)]
|
||||||
|
Num(u8),
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
`rename` now only renames a command literally; use `rename_rule` to change the case of a command:
|
||||||
|
|
||||||
|
```diff,rust
|
||||||
|
#[derive(BotCommands)]
|
||||||
|
- #[command(rename = "lowercase", description = "These commands are supported:")]
|
||||||
|
+ #[command(rename_rule = "lowercase", description = "These commands are supported:")]
|
||||||
|
enum Command {
|
||||||
|
// ...
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
## 0.9 -> 0.10
|
## 0.9 -> 0.10
|
||||||
|
|
||||||
### core
|
### core
|
||||||
|
|
99
README.md
99
README.md
|
@ -1,8 +1,8 @@
|
||||||
> [v0.9 -> v0.10 migration guide >>](MIGRATION_GUIDE.md#09---010)
|
> [v0.10 -> v0.11 migration guide >>](MIGRATION_GUIDE.md#010---011)
|
||||||
|
|
||||||
<div align="center">
|
<div align="center">
|
||||||
<img src="./ICON.png" width="250"/>
|
<img src="./ICON.png" width="250"/>
|
||||||
<h1>teloxide</h1>
|
<h1><code>teloxide</code></h1>
|
||||||
<a href="https://docs.rs/teloxide/">
|
<a href="https://docs.rs/teloxide/">
|
||||||
<img src="https://docs.rs/teloxide/badge.svg">
|
<img src="https://docs.rs/teloxide/badge.svg">
|
||||||
</a>
|
</a>
|
||||||
|
@ -13,7 +13,7 @@
|
||||||
<img src="https://img.shields.io/crates/v/teloxide.svg">
|
<img src="https://img.shields.io/crates/v/teloxide.svg">
|
||||||
</a>
|
</a>
|
||||||
<a href="https://core.telegram.org/bots/api">
|
<a href="https://core.telegram.org/bots/api">
|
||||||
<img src="https://img.shields.io/badge/API%20coverage-Up%20to%206.1%20(inclusively)-green.svg">
|
<img src="https://img.shields.io/badge/API%20coverage-Up%20to%206.2%20(inclusively)-green.svg">
|
||||||
</a>
|
</a>
|
||||||
<a href="https://t.me/teloxide">
|
<a href="https://t.me/teloxide">
|
||||||
<img src="https://img.shields.io/badge/support-t.me%2Fteloxide-blueviolet">
|
<img src="https://img.shields.io/badge/support-t.me%2Fteloxide-blueviolet">
|
||||||
|
@ -24,18 +24,20 @@
|
||||||
|
|
||||||
## Highlights
|
## Highlights
|
||||||
|
|
||||||
- **Declarative design.** teloxide is based upon [`dptree`], a functional [chain of responsibility] pattern that allows you to express pipelines of message processing in a highly declarative and extensible style.
|
- **Declarative design.** `teloxide` is based upon [`dptree`], a functional [chain of responsibility] pattern that allows you to express pipelines of message processing in a highly declarative and extensible style.
|
||||||
|
|
||||||
[`dptree`]: https://github.com/teloxide/dptree
|
[`dptree`]: https://github.com/teloxide/dptree
|
||||||
[chain of responsibility]: https://en.wikipedia.org/wiki/Chain-of-responsibility_pattern
|
[chain of responsibility]: https://en.wikipedia.org/wiki/Chain-of-responsibility_pattern
|
||||||
|
|
||||||
- **Dialogues management subsystem.** Our dialogues management subsystem is simple and easy-to-use, and, furthermore, is agnostic of how/where dialogues are stored. For example, you can just replace a one line to achieve [persistence]. Out-of-the-box storages include [Redis] and [Sqlite].
|
- **Feature-rich.** You can use both long polling and webhooks, configure an underlying HTTPS client, set a custom URL of a Telegram API server, and much more.
|
||||||
|
|
||||||
|
- **Simple dialogues.** Our dialogues subsystem is simple and easy-to-use, and, furthermore, is agnostic of how/where dialogues are stored. For example, you can just replace a one line to achieve [persistence]. Out-of-the-box storages include [Redis] and [Sqlite].
|
||||||
|
|
||||||
[persistence]: https://en.wikipedia.org/wiki/Persistence_(computer_science)
|
[persistence]: https://en.wikipedia.org/wiki/Persistence_(computer_science)
|
||||||
[Redis]: https://redis.io/
|
[Redis]: https://redis.io/
|
||||||
[Sqlite]: https://www.sqlite.org
|
[Sqlite]: https://www.sqlite.org
|
||||||
|
|
||||||
- **Strongly typed commands.** You can describe bot commands as enumerations, and then they'll be automatically constructed from strings — just like JSON structures in [`serde-json`] and command-line arguments in [`structopt`].
|
- **Strongly typed commands.** Define bot commands as an `enum` and teloxide will parse them automatically — just like JSON structures in [`serde-json`] and command-line arguments in [`structopt`].
|
||||||
|
|
||||||
[`structopt`]: https://github.com/TeXitoi/structopt
|
[`structopt`]: https://github.com/TeXitoi/structopt
|
||||||
[`serde-json`]: https://github.com/serde-rs/json
|
[`serde-json`]: https://github.com/serde-rs/json
|
||||||
|
@ -54,9 +56,9 @@ $ set TELOXIDE_TOKEN=<Your token here>
|
||||||
|
|
||||||
# Windows PowerShell
|
# Windows PowerShell
|
||||||
$ $env:TELOXIDE_TOKEN=<Your token here>
|
$ $env:TELOXIDE_TOKEN=<Your token here>
|
||||||
|
|
||||||
```
|
```
|
||||||
4. Make sure that your Rust compiler is up to date (teloxide currently requires rustc at least version 1.58):
|
|
||||||
|
4. Make sure that your Rust compiler is up to date (`teloxide` currently requires rustc at least version 1.64):
|
||||||
```bash
|
```bash
|
||||||
# If you're using stable
|
# If you're using stable
|
||||||
$ rustup update stable
|
$ rustup update stable
|
||||||
|
@ -70,7 +72,7 @@ $ rustup override set nightly
|
||||||
5. Run `cargo new my_bot`, enter the directory and put these lines into your `Cargo.toml`:
|
5. Run `cargo new my_bot`, enter the directory and put these lines into your `Cargo.toml`:
|
||||||
```toml
|
```toml
|
||||||
[dependencies]
|
[dependencies]
|
||||||
teloxide = { version = "0.10", features = ["macros", "auto-send"] }
|
teloxide = { version = "0.11", features = ["macros", "auto-send"] }
|
||||||
log = "0.4"
|
log = "0.4"
|
||||||
pretty_env_logger = "0.4"
|
pretty_env_logger = "0.4"
|
||||||
tokio = { version = "1.8", features = ["rt-multi-thread", "macros"] }
|
tokio = { version = "1.8", features = ["rt-multi-thread", "macros"] }
|
||||||
|
@ -92,11 +94,11 @@ async fn main() {
|
||||||
pretty_env_logger::init();
|
pretty_env_logger::init();
|
||||||
log::info!("Starting throw dice bot...");
|
log::info!("Starting throw dice bot...");
|
||||||
|
|
||||||
let bot = Bot::from_env().auto_send();
|
let bot = Bot::from_env();
|
||||||
|
|
||||||
teloxide::repl(bot, |message: Message, bot: AutoSend<Bot>| async move {
|
teloxide::repl(bot, |bot: Bot, msg: Message| async move {
|
||||||
bot.send_dice(message.chat.id).await?;
|
bot.send_dice(msg.chat.id).await?;
|
||||||
respond(())
|
Ok(())
|
||||||
})
|
})
|
||||||
.await;
|
.await;
|
||||||
}
|
}
|
||||||
|
@ -122,20 +124,18 @@ Commands are strongly typed and defined declaratively, similar to how we define
|
||||||
```rust,no_run
|
```rust,no_run
|
||||||
use teloxide::{prelude::*, utils::command::BotCommands};
|
use teloxide::{prelude::*, utils::command::BotCommands};
|
||||||
|
|
||||||
use std::error::Error;
|
|
||||||
|
|
||||||
#[tokio::main]
|
#[tokio::main]
|
||||||
async fn main() {
|
async fn main() {
|
||||||
pretty_env_logger::init();
|
pretty_env_logger::init();
|
||||||
log::info!("Starting command bot...");
|
log::info!("Starting command bot...");
|
||||||
|
|
||||||
let bot = Bot::from_env().auto_send();
|
let bot = Bot::from_env();
|
||||||
|
|
||||||
teloxide::commands_repl(bot, answer, Command::ty()).await;
|
teloxide::commands_repl(bot, answer, Command::ty()).await;
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(BotCommands, Clone)]
|
#[derive(BotCommands, Clone)]
|
||||||
#[command(rename = "lowercase", description = "These commands are supported:")]
|
#[command(rename_rule = "lowercase", description = "These commands are supported:")]
|
||||||
enum Command {
|
enum Command {
|
||||||
#[command(description = "display this text.")]
|
#[command(description = "display this text.")]
|
||||||
Help,
|
Help,
|
||||||
|
@ -145,23 +145,14 @@ enum Command {
|
||||||
UsernameAndAge { username: String, age: u8 },
|
UsernameAndAge { username: String, age: u8 },
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn answer(
|
async fn answer(bot: Bot, msg: Message, cmd: Command) -> ResponseResult<()> {
|
||||||
bot: AutoSend<Bot>,
|
match cmd {
|
||||||
message: Message,
|
Command::Help => bot.send_message(msg.chat.id, Command::descriptions().to_string()).await?,
|
||||||
command: Command,
|
|
||||||
) -> Result<(), Box<dyn Error + Send + Sync>> {
|
|
||||||
match command {
|
|
||||||
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(msg.chat.id, format!("Your username is @{username}.")).await?
|
||||||
}
|
}
|
||||||
Command::UsernameAndAge { username, age } => {
|
Command::UsernameAndAge { username, age } => {
|
||||||
bot.send_message(
|
bot.send_message(msg.chat.id, format!("Your username is @{username} and age is {age}."))
|
||||||
message.chat.id,
|
|
||||||
format!("Your username is @{username} and age is {age}."),
|
|
||||||
)
|
|
||||||
.await?
|
.await?
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@ -190,18 +181,18 @@ use teloxide::{dispatching::dialogue::InMemStorage, prelude::*};
|
||||||
type MyDialogue = Dialogue<State, InMemStorage<State>>;
|
type MyDialogue = Dialogue<State, InMemStorage<State>>;
|
||||||
type HandlerResult = Result<(), Box<dyn std::error::Error + Send + Sync>>;
|
type HandlerResult = Result<(), Box<dyn std::error::Error + Send + Sync>>;
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone, Default)]
|
||||||
pub enum State {
|
pub enum State {
|
||||||
|
#[default]
|
||||||
Start,
|
Start,
|
||||||
ReceiveFullName,
|
ReceiveFullName,
|
||||||
ReceiveAge { full_name: String },
|
ReceiveAge {
|
||||||
ReceiveLocation { full_name: String, age: u8 },
|
full_name: String,
|
||||||
}
|
},
|
||||||
|
ReceiveLocation {
|
||||||
impl Default for State {
|
full_name: String,
|
||||||
fn default() -> Self {
|
age: u8,
|
||||||
Self::Start
|
},
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tokio::main]
|
#[tokio::main]
|
||||||
|
@ -209,7 +200,7 @@ async fn main() {
|
||||||
pretty_env_logger::init();
|
pretty_env_logger::init();
|
||||||
log::info!("Starting dialogue bot...");
|
log::info!("Starting dialogue bot...");
|
||||||
|
|
||||||
let bot = Bot::from_env().auto_send();
|
let bot = Bot::from_env();
|
||||||
|
|
||||||
Dispatcher::builder(
|
Dispatcher::builder(
|
||||||
bot,
|
bot,
|
||||||
|
@ -229,17 +220,13 @@ async fn main() {
|
||||||
.await;
|
.await;
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn start(bot: AutoSend<Bot>, msg: Message, dialogue: MyDialogue) -> HandlerResult {
|
async fn start(bot: Bot, dialogue: MyDialogue, msg: Message) -> HandlerResult {
|
||||||
bot.send_message(msg.chat.id, "Let's start! What's your full name?").await?;
|
bot.send_message(msg.chat.id, "Let's start! What's your full name?").await?;
|
||||||
dialogue.update(State::ReceiveFullName).await?;
|
dialogue.update(State::ReceiveFullName).await?;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn receive_full_name(
|
async fn receive_full_name(bot: Bot, dialogue: MyDialogue, msg: Message) -> HandlerResult {
|
||||||
bot: AutoSend<Bot>,
|
|
||||||
msg: Message,
|
|
||||||
dialogue: MyDialogue,
|
|
||||||
) -> HandlerResult {
|
|
||||||
match msg.text() {
|
match msg.text() {
|
||||||
Some(text) => {
|
Some(text) => {
|
||||||
bot.send_message(msg.chat.id, "How old are you?").await?;
|
bot.send_message(msg.chat.id, "How old are you?").await?;
|
||||||
|
@ -254,10 +241,10 @@ async fn receive_full_name(
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn receive_age(
|
async fn receive_age(
|
||||||
bot: AutoSend<Bot>,
|
bot: Bot,
|
||||||
msg: Message,
|
|
||||||
dialogue: MyDialogue,
|
dialogue: MyDialogue,
|
||||||
full_name: String, // Available from `State::ReceiveAge`.
|
full_name: String, // Available from `State::ReceiveAge`.
|
||||||
|
msg: Message,
|
||||||
) -> HandlerResult {
|
) -> HandlerResult {
|
||||||
match msg.text().map(|text| text.parse::<u8>()) {
|
match msg.text().map(|text| text.parse::<u8>()) {
|
||||||
Some(Ok(age)) => {
|
Some(Ok(age)) => {
|
||||||
|
@ -273,15 +260,15 @@ async fn receive_age(
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn receive_location(
|
async fn receive_location(
|
||||||
bot: AutoSend<Bot>,
|
bot: Bot,
|
||||||
msg: Message,
|
|
||||||
dialogue: MyDialogue,
|
dialogue: MyDialogue,
|
||||||
(full_name, age): (String, u8), // Available from `State::ReceiveLocation`.
|
(full_name, age): (String, u8), // Available from `State::ReceiveLocation`.
|
||||||
|
msg: Message,
|
||||||
) -> HandlerResult {
|
) -> HandlerResult {
|
||||||
match msg.text() {
|
match msg.text() {
|
||||||
Some(location) => {
|
Some(location) => {
|
||||||
let message = format!("Full name: {full_name}\nAge: {age}\nLocation: {location}");
|
let report = format!("Full name: {full_name}\nAge: {age}\nLocation: {location}");
|
||||||
bot.send_message(msg.chat.id, message).await?;
|
bot.send_message(msg.chat.id, report).await?;
|
||||||
dialogue.exit().await?;
|
dialogue.exit().await?;
|
||||||
}
|
}
|
||||||
None => {
|
None => {
|
||||||
|
@ -319,7 +306,7 @@ A: No, only the bots API.
|
||||||
|
|
||||||
**Q: Can I use webhooks?**
|
**Q: Can I use webhooks?**
|
||||||
|
|
||||||
A: You can! Teloxide has a built-in support for webhooks in `dispatching::update_listeners::webhooks` module. See how it's used in [`examples/ngrok_ping_pong_bot`](examples/ngrok_ping_pong.rs) and [`examples/heroku_ping_pong_bot`](examples/heroku_ping_pong.rs).
|
A: You can! `teloxide` has a built-in support for webhooks in `dispatching::update_listeners::webhooks` module. See how it's used in [`examples/ngrok_ping_pong_bot`](examples/ngrok_ping_pong.rs) and [`examples/heroku_ping_pong_bot`](examples/heroku_ping_pong.rs).
|
||||||
|
|
||||||
**Q: Can I handle both callback queries and messages within a single dialogue?**
|
**Q: Can I handle both callback queries and messages within a single dialogue?**
|
||||||
|
|
||||||
|
@ -343,7 +330,7 @@ Feel free to propose your own bot to our collection!
|
||||||
- [`zamazan4ik/npaperbot-telegram`](https://github.com/zamazan4ik/npaperbot-telegram) — Telegram bot for searching via C++ proposals.
|
- [`zamazan4ik/npaperbot-telegram`](https://github.com/zamazan4ik/npaperbot-telegram) — Telegram bot for searching via C++ proposals.
|
||||||
|
|
||||||
<details>
|
<details>
|
||||||
<summary>Show bots using teloxide older than v0.6.0</summary>
|
<summary>Show bots using `teloxide` older than v0.6.0</summary>
|
||||||
|
|
||||||
- [`mxseev/logram`](https://github.com/mxseev/logram) — Utility that takes logs from anywhere and sends them to Telegram.
|
- [`mxseev/logram`](https://github.com/mxseev/logram) — Utility that takes logs from anywhere and sends them to Telegram.
|
||||||
- [`alexkonovalov/PedigreeBot`](https://github.com/alexkonovalov/PedigreeBot) — A Telegram bot for building family trees.
|
- [`alexkonovalov/PedigreeBot`](https://github.com/alexkonovalov/PedigreeBot) — A Telegram bot for building family trees.
|
||||||
|
@ -355,7 +342,7 @@ Feel free to propose your own bot to our collection!
|
||||||
|
|
||||||
</details>
|
</details>
|
||||||
|
|
||||||
See [600+ other public repositories using teloxide >>](https://github.com/teloxide/teloxide/network/dependents)
|
See [700+ other public repositories using `teloxide` >>](https://github.com/teloxide/teloxide/network/dependents)
|
||||||
|
|
||||||
## Contributing
|
## Contributing
|
||||||
|
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
use std::{error::Error, str::FromStr};
|
use std::str::FromStr;
|
||||||
|
|
||||||
use chrono::Duration;
|
use chrono::Duration;
|
||||||
use teloxide::{prelude::*, types::ChatPermissions, utils::command::BotCommands};
|
use teloxide::{prelude::*, types::ChatPermissions, utils::command::BotCommands};
|
||||||
|
@ -14,7 +14,7 @@ use teloxide::{prelude::*, types::ChatPermissions, utils::command::BotCommands};
|
||||||
// %PREFIX%%COMMAND% - %DESCRIPTION%
|
// %PREFIX%%COMMAND% - %DESCRIPTION%
|
||||||
#[derive(BotCommands, Clone)]
|
#[derive(BotCommands, Clone)]
|
||||||
#[command(
|
#[command(
|
||||||
rename = "lowercase",
|
rename_rule = "lowercase",
|
||||||
description = "Use commands in format /%command% %num% %unit%",
|
description = "Use commands in format /%command% %num% %unit%",
|
||||||
parse_with = "split"
|
parse_with = "split"
|
||||||
)]
|
)]
|
||||||
|
@ -58,19 +58,13 @@ async fn main() {
|
||||||
pretty_env_logger::init();
|
pretty_env_logger::init();
|
||||||
log::info!("Starting admin bot...");
|
log::info!("Starting admin bot...");
|
||||||
|
|
||||||
let bot = teloxide::Bot::from_env().auto_send();
|
let bot = teloxide::Bot::from_env();
|
||||||
|
|
||||||
teloxide::commands_repl(bot, action, Command::ty()).await;
|
teloxide::commands_repl(bot, action, Command::ty()).await;
|
||||||
}
|
}
|
||||||
|
|
||||||
type Bot = AutoSend<teloxide::Bot>;
|
async fn action(bot: Bot, msg: Message, cmd: Command) -> ResponseResult<()> {
|
||||||
|
match cmd {
|
||||||
async fn action(
|
|
||||||
bot: Bot,
|
|
||||||
msg: Message,
|
|
||||||
command: Command,
|
|
||||||
) -> Result<(), Box<dyn Error + Send + Sync>> {
|
|
||||||
match command {
|
|
||||||
Command::Help => {
|
Command::Help => {
|
||||||
bot.send_message(msg.chat.id, Command::descriptions().to_string()).await?;
|
bot.send_message(msg.chat.id, Command::descriptions().to_string()).await?;
|
||||||
}
|
}
|
||||||
|
@ -83,7 +77,7 @@ async fn action(
|
||||||
}
|
}
|
||||||
|
|
||||||
// Kick a user with a replied message.
|
// Kick a user with a replied message.
|
||||||
async fn kick_user(bot: Bot, msg: Message) -> Result<(), Box<dyn Error + Send + Sync>> {
|
async fn kick_user(bot: Bot, msg: Message) -> ResponseResult<()> {
|
||||||
match msg.reply_to_message() {
|
match msg.reply_to_message() {
|
||||||
Some(replied) => {
|
Some(replied) => {
|
||||||
// bot.unban_chat_member can also kicks a user from a group chat.
|
// bot.unban_chat_member can also kicks a user from a group chat.
|
||||||
|
@ -97,11 +91,7 @@ async fn kick_user(bot: Bot, msg: Message) -> Result<(), Box<dyn Error + Send +
|
||||||
}
|
}
|
||||||
|
|
||||||
// Ban a user with replied message.
|
// Ban a user with replied message.
|
||||||
async fn ban_user(
|
async fn ban_user(bot: Bot, msg: Message, time: Duration) -> ResponseResult<()> {
|
||||||
bot: Bot,
|
|
||||||
msg: Message,
|
|
||||||
time: Duration,
|
|
||||||
) -> Result<(), Box<dyn Error + Send + Sync>> {
|
|
||||||
match msg.reply_to_message() {
|
match msg.reply_to_message() {
|
||||||
Some(replied) => {
|
Some(replied) => {
|
||||||
bot.kick_chat_member(
|
bot.kick_chat_member(
|
||||||
|
@ -120,11 +110,7 @@ async fn ban_user(
|
||||||
}
|
}
|
||||||
|
|
||||||
// Mute a user with a replied message.
|
// Mute a user with a replied message.
|
||||||
async fn mute_user(
|
async fn mute_user(bot: Bot, msg: Message, time: Duration) -> ResponseResult<()> {
|
||||||
bot: Bot,
|
|
||||||
msg: Message,
|
|
||||||
time: Duration,
|
|
||||||
) -> Result<(), Box<dyn Error + Send + Sync>> {
|
|
||||||
match msg.reply_to_message() {
|
match msg.reply_to_message() {
|
||||||
Some(replied) => {
|
Some(replied) => {
|
||||||
bot.restrict_chat_member(
|
bot.restrict_chat_member(
|
||||||
|
|
|
@ -4,13 +4,13 @@ use teloxide::{
|
||||||
prelude::*,
|
prelude::*,
|
||||||
types::{
|
types::{
|
||||||
InlineKeyboardButton, InlineKeyboardMarkup, InlineQueryResultArticle, InputMessageContent,
|
InlineKeyboardButton, InlineKeyboardMarkup, InlineQueryResultArticle, InputMessageContent,
|
||||||
InputMessageContentText,
|
InputMessageContentText, Me,
|
||||||
},
|
},
|
||||||
utils::command::BotCommands,
|
utils::command::BotCommands,
|
||||||
};
|
};
|
||||||
|
|
||||||
#[derive(BotCommands)]
|
#[derive(BotCommands)]
|
||||||
#[command(rename = "lowercase", description = "These commands are supported:")]
|
#[command(rename_rule = "lowercase", description = "These commands are supported:")]
|
||||||
enum Command {
|
enum Command {
|
||||||
#[command(description = "Display this text")]
|
#[command(description = "Display this text")]
|
||||||
Help,
|
Help,
|
||||||
|
@ -23,7 +23,7 @@ async fn main() -> Result<(), Box<dyn Error>> {
|
||||||
pretty_env_logger::init();
|
pretty_env_logger::init();
|
||||||
log::info!("Starting buttons bot...");
|
log::info!("Starting buttons bot...");
|
||||||
|
|
||||||
let bot = Bot::from_env().auto_send();
|
let bot = Bot::from_env();
|
||||||
|
|
||||||
let handler = dptree::entry()
|
let handler = dptree::entry()
|
||||||
.branch(Update::filter_message().endpoint(message_handler))
|
.branch(Update::filter_message().endpoint(message_handler))
|
||||||
|
@ -59,23 +59,24 @@ fn make_keyboard() -> InlineKeyboardMarkup {
|
||||||
/// or not, then match the command. If the command is `/start` it writes a
|
/// or not, then match the command. If the command is `/start` it writes a
|
||||||
/// markup with the `InlineKeyboardMarkup`.
|
/// markup with the `InlineKeyboardMarkup`.
|
||||||
async fn message_handler(
|
async fn message_handler(
|
||||||
m: Message,
|
bot: Bot,
|
||||||
bot: AutoSend<Bot>,
|
msg: Message,
|
||||||
|
me: Me,
|
||||||
) -> Result<(), Box<dyn Error + Send + Sync>> {
|
) -> Result<(), Box<dyn Error + Send + Sync>> {
|
||||||
if let Some(text) = m.text() {
|
if let Some(text) = msg.text() {
|
||||||
match BotCommands::parse(text, "buttons") {
|
match BotCommands::parse(text, me.username()) {
|
||||||
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().to_string()).await?;
|
bot.send_message(msg.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.
|
||||||
let keyboard = make_keyboard();
|
let keyboard = make_keyboard();
|
||||||
bot.send_message(m.chat.id, "Debian versions:").reply_markup(keyboard).await?;
|
bot.send_message(msg.chat.id, "Debian versions:").reply_markup(keyboard).await?;
|
||||||
}
|
}
|
||||||
|
|
||||||
Err(_) => {
|
Err(_) => {
|
||||||
bot.send_message(m.chat.id, "Command not found!").await?;
|
bot.send_message(msg.chat.id, "Command not found!").await?;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -84,8 +85,8 @@ async fn message_handler(
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn inline_query_handler(
|
async fn inline_query_handler(
|
||||||
|
bot: Bot,
|
||||||
q: InlineQuery,
|
q: InlineQuery,
|
||||||
bot: AutoSend<Bot>,
|
|
||||||
) -> Result<(), Box<dyn Error + Send + Sync>> {
|
) -> Result<(), Box<dyn Error + Send + Sync>> {
|
||||||
let choose_debian_version = InlineQueryResultArticle::new(
|
let choose_debian_version = InlineQueryResultArticle::new(
|
||||||
"0",
|
"0",
|
||||||
|
@ -104,23 +105,22 @@ async fn inline_query_handler(
|
||||||
///
|
///
|
||||||
/// **IMPORTANT**: do not send privacy-sensitive data this way!!!
|
/// **IMPORTANT**: do not send privacy-sensitive data this way!!!
|
||||||
/// Anyone can read data stored in the callback button.
|
/// Anyone can read data stored in the callback button.
|
||||||
async fn callback_handler(
|
async fn callback_handler(bot: Bot, q: CallbackQuery) -> Result<(), Box<dyn Error + Send + Sync>> {
|
||||||
q: CallbackQuery,
|
|
||||||
bot: AutoSend<Bot>,
|
|
||||||
) -> Result<(), Box<dyn Error + Send + Sync>> {
|
|
||||||
if let Some(version) = q.data {
|
if let Some(version) = q.data {
|
||||||
let text = format!("You chose: {version}");
|
let text = format!("You chose: {version}");
|
||||||
|
|
||||||
match q.message {
|
// Tell telegram that we've seen this query, to remove 🕑 icons from the
|
||||||
Some(Message { id, chat, .. }) => {
|
//
|
||||||
|
// clients. You could also use `answer_callback_query`'s optional
|
||||||
|
// parameters to tweak what happens on the client side.
|
||||||
|
bot.answer_callback_query(q.id).await?;
|
||||||
|
|
||||||
|
// Edit text of the message to which the buttons were attached
|
||||||
|
if let Some(Message { id, chat, .. }) = q.message {
|
||||||
bot.edit_message_text(chat.id, id, text).await?;
|
bot.edit_message_text(chat.id, id, text).await?;
|
||||||
}
|
} else if let Some(id) = q.inline_message_id {
|
||||||
None => {
|
|
||||||
if let Some(id) = q.inline_message_id {
|
|
||||||
bot.edit_message_text_inline(id, text).await?;
|
bot.edit_message_text_inline(id, text).await?;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
log::info!("You chose: {}", version);
|
log::info!("You chose: {}", version);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,19 +1,17 @@
|
||||||
use teloxide::{prelude::*, utils::command::BotCommands};
|
use teloxide::{prelude::*, utils::command::BotCommands};
|
||||||
|
|
||||||
use std::error::Error;
|
|
||||||
|
|
||||||
#[tokio::main]
|
#[tokio::main]
|
||||||
async fn main() {
|
async fn main() {
|
||||||
pretty_env_logger::init();
|
pretty_env_logger::init();
|
||||||
log::info!("Starting command bot...");
|
log::info!("Starting command bot...");
|
||||||
|
|
||||||
let bot = Bot::from_env().auto_send();
|
let bot = Bot::from_env();
|
||||||
|
|
||||||
teloxide::commands_repl(bot, answer, Command::ty()).await;
|
teloxide::commands_repl(bot, answer, Command::ty()).await;
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(BotCommands, Clone)]
|
#[derive(BotCommands, Clone)]
|
||||||
#[command(rename = "lowercase", description = "These commands are supported:")]
|
#[command(rename_rule = "lowercase", description = "These commands are supported:")]
|
||||||
enum Command {
|
enum Command {
|
||||||
#[command(description = "display this text.")]
|
#[command(description = "display this text.")]
|
||||||
Help,
|
Help,
|
||||||
|
@ -23,23 +21,14 @@ enum Command {
|
||||||
UsernameAndAge { username: String, age: u8 },
|
UsernameAndAge { username: String, age: u8 },
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn answer(
|
async fn answer(bot: Bot, msg: Message, cmd: Command) -> ResponseResult<()> {
|
||||||
bot: AutoSend<Bot>,
|
match cmd {
|
||||||
message: Message,
|
Command::Help => bot.send_message(msg.chat.id, Command::descriptions().to_string()).await?,
|
||||||
command: Command,
|
|
||||||
) -> Result<(), Box<dyn Error + Send + Sync>> {
|
|
||||||
match command {
|
|
||||||
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(msg.chat.id, format!("Your username is @{username}.")).await?
|
||||||
}
|
}
|
||||||
Command::UsernameAndAge { username, age } => {
|
Command::UsernameAndAge { username, age } => {
|
||||||
bot.send_message(
|
bot.send_message(msg.chat.id, format!("Your username is @{username} and age is {age}."))
|
||||||
message.chat.id,
|
|
||||||
format!("Your username is @{username} and age is {age}."),
|
|
||||||
)
|
|
||||||
.await?
|
.await?
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
|
@ -22,7 +22,7 @@ pub enum State {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone, BotCommands)]
|
#[derive(Clone, BotCommands)]
|
||||||
#[command(rename = "lowercase", description = "These commands are supported:")]
|
#[command(rename_rule = "lowercase", description = "These commands are supported:")]
|
||||||
pub enum Command {
|
pub enum Command {
|
||||||
#[command(description = "get your number.")]
|
#[command(description = "get your number.")]
|
||||||
Get,
|
Get,
|
||||||
|
@ -35,7 +35,7 @@ async fn main() {
|
||||||
pretty_env_logger::init();
|
pretty_env_logger::init();
|
||||||
log::info!("Starting DB remember bot...");
|
log::info!("Starting DB remember bot...");
|
||||||
|
|
||||||
let bot = Bot::from_env().auto_send();
|
let bot = Bot::from_env();
|
||||||
|
|
||||||
let storage: MyStorage = if std::env::var("DB_REMEMBER_REDIS").is_ok() {
|
let storage: MyStorage = if std::env::var("DB_REMEMBER_REDIS").is_ok() {
|
||||||
RedisStorage::open("redis://127.0.0.1:6379", Bincode).await.unwrap().erase()
|
RedisStorage::open("redis://127.0.0.1:6379", Bincode).await.unwrap().erase()
|
||||||
|
@ -60,7 +60,7 @@ async fn main() {
|
||||||
.await;
|
.await;
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn start(bot: AutoSend<Bot>, msg: Message, dialogue: MyDialogue) -> HandlerResult {
|
async fn start(bot: Bot, dialogue: MyDialogue, msg: Message) -> HandlerResult {
|
||||||
match msg.text().map(|text| text.parse::<i32>()) {
|
match msg.text().map(|text| text.parse::<i32>()) {
|
||||||
Some(Ok(n)) => {
|
Some(Ok(n)) => {
|
||||||
dialogue.update(State::GotNumber(n)).await?;
|
dialogue.update(State::GotNumber(n)).await?;
|
||||||
|
@ -79,10 +79,10 @@ async fn start(bot: AutoSend<Bot>, msg: Message, dialogue: MyDialogue) -> Handle
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn got_number(
|
async fn got_number(
|
||||||
bot: AutoSend<Bot>,
|
bot: Bot,
|
||||||
msg: Message,
|
|
||||||
dialogue: MyDialogue,
|
dialogue: MyDialogue,
|
||||||
num: i32,
|
num: i32, // Available from `State::GotNumber`.
|
||||||
|
msg: Message,
|
||||||
cmd: Command,
|
cmd: Command,
|
||||||
) -> HandlerResult {
|
) -> HandlerResult {
|
||||||
match cmd {
|
match cmd {
|
||||||
|
@ -97,7 +97,7 @@ async fn got_number(
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn invalid_command(bot: AutoSend<Bot>, msg: Message) -> HandlerResult {
|
async fn invalid_command(bot: Bot, msg: Message) -> HandlerResult {
|
||||||
bot.send_message(msg.chat.id, "Please, send /get or /reset.").await?;
|
bot.send_message(msg.chat.id, "Please, send /get or /reset.").await?;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
|
@ -37,7 +37,7 @@ async fn main() {
|
||||||
pretty_env_logger::init();
|
pretty_env_logger::init();
|
||||||
log::info!("Starting dialogue bot...");
|
log::info!("Starting dialogue bot...");
|
||||||
|
|
||||||
let bot = Bot::from_env().auto_send();
|
let bot = Bot::from_env();
|
||||||
|
|
||||||
Dispatcher::builder(
|
Dispatcher::builder(
|
||||||
bot,
|
bot,
|
||||||
|
@ -57,17 +57,13 @@ async fn main() {
|
||||||
.await;
|
.await;
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn start(bot: AutoSend<Bot>, msg: Message, dialogue: MyDialogue) -> HandlerResult {
|
async fn start(bot: Bot, dialogue: MyDialogue, msg: Message) -> HandlerResult {
|
||||||
bot.send_message(msg.chat.id, "Let's start! What's your full name?").await?;
|
bot.send_message(msg.chat.id, "Let's start! What's your full name?").await?;
|
||||||
dialogue.update(State::ReceiveFullName).await?;
|
dialogue.update(State::ReceiveFullName).await?;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn receive_full_name(
|
async fn receive_full_name(bot: Bot, dialogue: MyDialogue, msg: Message) -> HandlerResult {
|
||||||
bot: AutoSend<Bot>,
|
|
||||||
msg: Message,
|
|
||||||
dialogue: MyDialogue,
|
|
||||||
) -> HandlerResult {
|
|
||||||
match msg.text() {
|
match msg.text() {
|
||||||
Some(text) => {
|
Some(text) => {
|
||||||
bot.send_message(msg.chat.id, "How old are you?").await?;
|
bot.send_message(msg.chat.id, "How old are you?").await?;
|
||||||
|
@ -82,10 +78,10 @@ async fn receive_full_name(
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn receive_age(
|
async fn receive_age(
|
||||||
bot: AutoSend<Bot>,
|
bot: Bot,
|
||||||
msg: Message,
|
|
||||||
dialogue: MyDialogue,
|
dialogue: MyDialogue,
|
||||||
full_name: String, // Available from `State::ReceiveAge`.
|
full_name: String, // Available from `State::ReceiveAge`.
|
||||||
|
msg: Message,
|
||||||
) -> HandlerResult {
|
) -> HandlerResult {
|
||||||
match msg.text().map(|text| text.parse::<u8>()) {
|
match msg.text().map(|text| text.parse::<u8>()) {
|
||||||
Some(Ok(age)) => {
|
Some(Ok(age)) => {
|
||||||
|
@ -101,15 +97,15 @@ async fn receive_age(
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn receive_location(
|
async fn receive_location(
|
||||||
bot: AutoSend<Bot>,
|
bot: Bot,
|
||||||
msg: Message,
|
|
||||||
dialogue: MyDialogue,
|
dialogue: MyDialogue,
|
||||||
(full_name, age): (String, u8), // Available from `State::ReceiveLocation`.
|
(full_name, age): (String, u8), // Available from `State::ReceiveLocation`.
|
||||||
|
msg: Message,
|
||||||
) -> HandlerResult {
|
) -> HandlerResult {
|
||||||
match msg.text() {
|
match msg.text() {
|
||||||
Some(location) => {
|
Some(location) => {
|
||||||
let message = format!("Full name: {full_name}\nAge: {age}\nLocation: {location}");
|
let report = format!("Full name: {full_name}\nAge: {age}\nLocation: {location}");
|
||||||
bot.send_message(msg.chat.id, message).await?;
|
bot.send_message(msg.chat.id, report).await?;
|
||||||
dialogue.exit().await?;
|
dialogue.exit().await?;
|
||||||
}
|
}
|
||||||
None => {
|
None => {
|
||||||
|
|
|
@ -14,7 +14,7 @@ async fn main() {
|
||||||
pretty_env_logger::init();
|
pretty_env_logger::init();
|
||||||
log::info!("Starting dispatching features bot...");
|
log::info!("Starting dispatching features bot...");
|
||||||
|
|
||||||
let bot = Bot::from_env().auto_send();
|
let bot = Bot::from_env();
|
||||||
|
|
||||||
let parameters = ConfigParameters {
|
let parameters = ConfigParameters {
|
||||||
bot_maintainer: UserId(0), // Paste your ID to run this bot.
|
bot_maintainer: UserId(0), // Paste your ID to run this bot.
|
||||||
|
@ -33,12 +33,11 @@ async fn main() {
|
||||||
)
|
)
|
||||||
.branch(
|
.branch(
|
||||||
// Filter a maintainer by a used ID.
|
// Filter a maintainer by a used ID.
|
||||||
dptree::filter(|msg: Message, cfg: ConfigParameters| {
|
dptree::filter(|cfg: ConfigParameters, msg: Message| {
|
||||||
msg.from().map(|user| user.id == cfg.bot_maintainer).unwrap_or_default()
|
msg.from().map(|user| user.id == cfg.bot_maintainer).unwrap_or_default()
|
||||||
})
|
})
|
||||||
.filter_command::<MaintainerCommands>()
|
.filter_command::<MaintainerCommands>()
|
||||||
.endpoint(
|
.endpoint(|msg: Message, bot: Bot, cmd: MaintainerCommands| async move {
|
||||||
|msg: Message, bot: AutoSend<Bot>, cmd: MaintainerCommands| async move {
|
|
||||||
match cmd {
|
match cmd {
|
||||||
MaintainerCommands::Rand { from, to } => {
|
MaintainerCommands::Rand { from, to } => {
|
||||||
let mut rng = rand::rngs::OsRng::default();
|
let mut rng = rand::rngs::OsRng::default();
|
||||||
|
@ -48,14 +47,13 @@ async fn main() {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
}),
|
||||||
),
|
|
||||||
)
|
)
|
||||||
.branch(
|
.branch(
|
||||||
// Filtering allow you to filter updates by some condition.
|
// Filtering allow you to filter updates by some condition.
|
||||||
dptree::filter(|msg: Message| msg.chat.is_group() || msg.chat.is_supergroup())
|
dptree::filter(|msg: Message| msg.chat.is_group() || msg.chat.is_supergroup())
|
||||||
// An endpoint is the last update handler.
|
// An endpoint is the last update handler.
|
||||||
.endpoint(|msg: Message, bot: AutoSend<Bot>| async move {
|
.endpoint(|msg: Message, bot: Bot| async move {
|
||||||
log::info!("Received a message from a group chat.");
|
log::info!("Received a message from a group chat.");
|
||||||
bot.send_message(msg.chat.id, "This is a group chat.").await?;
|
bot.send_message(msg.chat.id, "This is a group chat.").await?;
|
||||||
respond(())
|
respond(())
|
||||||
|
@ -64,14 +62,12 @@ async fn main() {
|
||||||
.branch(
|
.branch(
|
||||||
// There are some extension filtering functions on `Message`. The following filter will
|
// There are some extension filtering functions on `Message`. The following filter will
|
||||||
// filter only messages with dices.
|
// filter only messages with dices.
|
||||||
Message::filter_dice().endpoint(
|
Message::filter_dice().endpoint(|bot: Bot, msg: Message, dice: Dice| async move {
|
||||||
|msg: Message, dice: Dice, bot: AutoSend<Bot>| async move {
|
|
||||||
bot.send_message(msg.chat.id, format!("Dice value: {}", dice.value))
|
bot.send_message(msg.chat.id, format!("Dice value: {}", dice.value))
|
||||||
.reply_to_message_id(msg.id)
|
.reply_to_message_id(msg.id)
|
||||||
.await?;
|
.await?;
|
||||||
Ok(())
|
Ok(())
|
||||||
},
|
}),
|
||||||
),
|
|
||||||
);
|
);
|
||||||
|
|
||||||
Dispatcher::builder(bot, handler)
|
Dispatcher::builder(bot, handler)
|
||||||
|
@ -100,7 +96,7 @@ struct ConfigParameters {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(BotCommands, Clone)]
|
#[derive(BotCommands, Clone)]
|
||||||
#[command(rename = "lowercase", description = "Simple commands")]
|
#[command(rename_rule = "lowercase", description = "Simple commands")]
|
||||||
enum SimpleCommand {
|
enum SimpleCommand {
|
||||||
#[command(description = "shows this message.")]
|
#[command(description = "shows this message.")]
|
||||||
Help,
|
Help,
|
||||||
|
@ -111,18 +107,18 @@ enum SimpleCommand {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(BotCommands, Clone)]
|
#[derive(BotCommands, Clone)]
|
||||||
#[command(rename = "lowercase", description = "Maintainer commands")]
|
#[command(rename_rule = "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")]
|
||||||
Rand { from: u64, to: u64 },
|
Rand { from: u64, to: u64 },
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn simple_commands_handler(
|
async fn simple_commands_handler(
|
||||||
msg: Message,
|
|
||||||
bot: AutoSend<Bot>,
|
|
||||||
cmd: SimpleCommand,
|
|
||||||
cfg: ConfigParameters,
|
cfg: ConfigParameters,
|
||||||
|
bot: Bot,
|
||||||
me: teloxide::types::Me,
|
me: teloxide::types::Me,
|
||||||
|
msg: Message,
|
||||||
|
cmd: SimpleCommand,
|
||||||
) -> Result<(), teloxide::RequestError> {
|
) -> Result<(), teloxide::RequestError> {
|
||||||
let text = match cmd {
|
let text = match cmd {
|
||||||
SimpleCommand::Help => {
|
SimpleCommand::Help => {
|
||||||
|
|
|
@ -21,15 +21,13 @@
|
||||||
use std::env;
|
use std::env;
|
||||||
|
|
||||||
use teloxide::{dispatching::update_listeners::webhooks, prelude::*};
|
use teloxide::{dispatching::update_listeners::webhooks, prelude::*};
|
||||||
use url::Url;
|
|
||||||
|
|
||||||
#[tokio::main]
|
#[tokio::main]
|
||||||
async fn main() {
|
async fn main() {
|
||||||
pretty_env_logger::init();
|
pretty_env_logger::init();
|
||||||
log::info!("Starting Heroku ping-pong bot...");
|
log::info!("Starting Heroku ping-pong bot...");
|
||||||
|
|
||||||
let bot = Bot::from_env().auto_send();
|
let bot = Bot::from_env();
|
||||||
let token = bot.inner().token();
|
|
||||||
|
|
||||||
// Heroku auto defines a port value
|
// Heroku auto defines a port value
|
||||||
let port: u16 = env::var("PORT")
|
let port: u16 = env::var("PORT")
|
||||||
|
@ -41,7 +39,7 @@ async fn main() {
|
||||||
|
|
||||||
// Heroku host example: "heroku-ping-pong-bot.herokuapp.com"
|
// Heroku host example: "heroku-ping-pong-bot.herokuapp.com"
|
||||||
let host = env::var("HOST").expect("HOST env variable is not set");
|
let host = env::var("HOST").expect("HOST env variable is not set");
|
||||||
let url = Url::parse(&format!("https://{host}/webhooks/{token}")).unwrap();
|
let url = format!("https://{host}/webhook").parse().unwrap();
|
||||||
|
|
||||||
let listener = webhooks::axum(bot.clone(), webhooks::Options::new(addr, url))
|
let listener = webhooks::axum(bot.clone(), webhooks::Options::new(addr, url))
|
||||||
.await
|
.await
|
||||||
|
@ -49,9 +47,9 @@ async fn main() {
|
||||||
|
|
||||||
teloxide::repl_with_listener(
|
teloxide::repl_with_listener(
|
||||||
bot,
|
bot,
|
||||||
|msg: Message, bot: AutoSend<Bot>| async move {
|
|bot: Bot, msg: Message| async move {
|
||||||
bot.send_message(msg.chat.id, "pong").await?;
|
bot.send_message(msg.chat.id, "pong").await?;
|
||||||
respond(())
|
Ok(())
|
||||||
},
|
},
|
||||||
listener,
|
listener,
|
||||||
)
|
)
|
||||||
|
|
|
@ -11,10 +11,10 @@ async fn main() {
|
||||||
pretty_env_logger::init();
|
pretty_env_logger::init();
|
||||||
log::info!("Starting inline bot...");
|
log::info!("Starting inline bot...");
|
||||||
|
|
||||||
let bot = Bot::from_env().auto_send();
|
let bot = Bot::from_env();
|
||||||
|
|
||||||
let handler = Update::filter_inline_query().branch(dptree::endpoint(
|
let handler = Update::filter_inline_query().branch(dptree::endpoint(
|
||||||
|query: InlineQuery, bot: AutoSend<Bot>| async move {
|
|bot: Bot, q: InlineQuery| async move {
|
||||||
// First, create your actual response
|
// First, create your actual response
|
||||||
let google_search = InlineQueryResultArticle::new(
|
let google_search = InlineQueryResultArticle::new(
|
||||||
// Each item needs a unique ID, as well as the response container for the
|
// Each item needs a unique ID, as well as the response container for the
|
||||||
|
@ -26,7 +26,7 @@ async fn main() {
|
||||||
// What message will be sent when clicked/tapped
|
// What message will be sent when clicked/tapped
|
||||||
InputMessageContent::Text(InputMessageContentText::new(format!(
|
InputMessageContent::Text(InputMessageContentText::new(format!(
|
||||||
"https://www.google.com/search?q={}",
|
"https://www.google.com/search?q={}",
|
||||||
query.query,
|
q.query,
|
||||||
))),
|
))),
|
||||||
);
|
);
|
||||||
// While constructing them from the struct itself is possible, it is preferred
|
// While constructing them from the struct itself is possible, it is preferred
|
||||||
|
@ -38,7 +38,7 @@ async fn main() {
|
||||||
"DuckDuckGo Search".to_string(),
|
"DuckDuckGo Search".to_string(),
|
||||||
InputMessageContent::Text(InputMessageContentText::new(format!(
|
InputMessageContent::Text(InputMessageContentText::new(format!(
|
||||||
"https://duckduckgo.com/?q={}",
|
"https://duckduckgo.com/?q={}",
|
||||||
query.query
|
q.query
|
||||||
))),
|
))),
|
||||||
)
|
)
|
||||||
.description("DuckDuckGo Search")
|
.description("DuckDuckGo Search")
|
||||||
|
@ -52,7 +52,7 @@ async fn main() {
|
||||||
|
|
||||||
// Send it off! One thing to note -- the ID we use here must be of the query
|
// Send it off! One thing to note -- the ID we use here must be of the query
|
||||||
// we're responding to.
|
// we're responding to.
|
||||||
let response = bot.answer_inline_query(&query.id, results).send().await;
|
let response = bot.answer_inline_query(&q.id, results).send().await;
|
||||||
if let Err(err) = response {
|
if let Err(err) = response {
|
||||||
log::error!("Error in handler: {:?}", err);
|
log::error!("Error in handler: {:?}", err);
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,7 +8,7 @@ async fn main() {
|
||||||
pretty_env_logger::init();
|
pretty_env_logger::init();
|
||||||
log::info!("Starting ngrok ping-pong bot...");
|
log::info!("Starting ngrok ping-pong bot...");
|
||||||
|
|
||||||
let bot = Bot::from_env().auto_send();
|
let bot = Bot::from_env();
|
||||||
|
|
||||||
let addr = ([127, 0, 0, 1], 8443).into();
|
let addr = ([127, 0, 0, 1], 8443).into();
|
||||||
let url = "Your HTTPS ngrok URL here. Get it by `ngrok http 8443`".parse().unwrap();
|
let url = "Your HTTPS ngrok URL here. Get it by `ngrok http 8443`".parse().unwrap();
|
||||||
|
@ -18,9 +18,9 @@ async fn main() {
|
||||||
|
|
||||||
teloxide::repl_with_listener(
|
teloxide::repl_with_listener(
|
||||||
bot,
|
bot,
|
||||||
|msg: Message, bot: AutoSend<Bot>| async move {
|
|bot: Bot, msg: Message| async move {
|
||||||
bot.send_message(msg.chat.id, "pong").await?;
|
bot.send_message(msg.chat.id, "pong").await?;
|
||||||
respond(())
|
Ok(())
|
||||||
},
|
},
|
||||||
listener,
|
listener,
|
||||||
)
|
)
|
||||||
|
|
|
@ -33,7 +33,7 @@ pub enum State {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(BotCommands, Clone)]
|
#[derive(BotCommands, Clone)]
|
||||||
#[command(rename = "lowercase", description = "These commands are supported:")]
|
#[command(rename_rule = "lowercase", description = "These commands are supported:")]
|
||||||
enum Command {
|
enum Command {
|
||||||
#[command(description = "display this text.")]
|
#[command(description = "display this text.")]
|
||||||
Help,
|
Help,
|
||||||
|
@ -48,7 +48,7 @@ async fn main() {
|
||||||
pretty_env_logger::init();
|
pretty_env_logger::init();
|
||||||
log::info!("Starting purchase bot...");
|
log::info!("Starting purchase bot...");
|
||||||
|
|
||||||
let bot = Bot::from_env().auto_send();
|
let bot = Bot::from_env();
|
||||||
|
|
||||||
Dispatcher::builder(bot, schema())
|
Dispatcher::builder(bot, schema())
|
||||||
.dependencies(dptree::deps![InMemStorage::<State>::new()])
|
.dependencies(dptree::deps![InMemStorage::<State>::new()])
|
||||||
|
@ -83,34 +83,30 @@ fn schema() -> UpdateHandler<Box<dyn std::error::Error + Send + Sync + 'static>>
|
||||||
.branch(callback_query_handler)
|
.branch(callback_query_handler)
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn start(bot: AutoSend<Bot>, msg: Message, dialogue: MyDialogue) -> HandlerResult {
|
async fn start(bot: Bot, dialogue: MyDialogue, msg: Message) -> HandlerResult {
|
||||||
bot.send_message(msg.chat.id, "Let's start! What's your full name?").await?;
|
bot.send_message(msg.chat.id, "Let's start! What's your full name?").await?;
|
||||||
dialogue.update(State::ReceiveFullName).await?;
|
dialogue.update(State::ReceiveFullName).await?;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn help(bot: AutoSend<Bot>, msg: Message) -> HandlerResult {
|
async fn help(bot: Bot, msg: Message) -> HandlerResult {
|
||||||
bot.send_message(msg.chat.id, Command::descriptions().to_string()).await?;
|
bot.send_message(msg.chat.id, Command::descriptions().to_string()).await?;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn cancel(bot: AutoSend<Bot>, msg: Message, dialogue: MyDialogue) -> HandlerResult {
|
async fn cancel(bot: Bot, dialogue: MyDialogue, msg: Message) -> HandlerResult {
|
||||||
bot.send_message(msg.chat.id, "Cancelling the dialogue.").await?;
|
bot.send_message(msg.chat.id, "Cancelling the dialogue.").await?;
|
||||||
dialogue.exit().await?;
|
dialogue.exit().await?;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn invalid_state(bot: AutoSend<Bot>, msg: Message) -> HandlerResult {
|
async fn invalid_state(bot: Bot, msg: Message) -> HandlerResult {
|
||||||
bot.send_message(msg.chat.id, "Unable to handle the message. Type /help to see the usage.")
|
bot.send_message(msg.chat.id, "Unable to handle the message. Type /help to see the usage.")
|
||||||
.await?;
|
.await?;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn receive_full_name(
|
async fn receive_full_name(bot: Bot, dialogue: MyDialogue, msg: Message) -> HandlerResult {
|
||||||
bot: AutoSend<Bot>,
|
|
||||||
msg: Message,
|
|
||||||
dialogue: MyDialogue,
|
|
||||||
) -> HandlerResult {
|
|
||||||
match msg.text().map(ToOwned::to_owned) {
|
match msg.text().map(ToOwned::to_owned) {
|
||||||
Some(full_name) => {
|
Some(full_name) => {
|
||||||
let products = ["Apple", "Banana", "Orange", "Potato"]
|
let products = ["Apple", "Banana", "Orange", "Potato"]
|
||||||
|
@ -130,10 +126,10 @@ async fn receive_full_name(
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn receive_product_selection(
|
async fn receive_product_selection(
|
||||||
bot: AutoSend<Bot>,
|
bot: Bot,
|
||||||
q: CallbackQuery,
|
|
||||||
dialogue: MyDialogue,
|
dialogue: MyDialogue,
|
||||||
full_name: String,
|
full_name: String, // Available from `State::ReceiveProductChoice`.
|
||||||
|
q: CallbackQuery,
|
||||||
) -> HandlerResult {
|
) -> HandlerResult {
|
||||||
if let Some(product) = &q.data {
|
if let Some(product) = &q.data {
|
||||||
bot.send_message(
|
bot.send_message(
|
||||||
|
|
|
@ -12,11 +12,11 @@ async fn main() {
|
||||||
pretty_env_logger::init();
|
pretty_env_logger::init();
|
||||||
log::info!("Starting shared state bot...");
|
log::info!("Starting shared state bot...");
|
||||||
|
|
||||||
let bot = Bot::from_env().auto_send();
|
let bot = Bot::from_env();
|
||||||
let messages_total = Arc::new(AtomicU64::new(0));
|
let messages_total = Arc::new(AtomicU64::new(0));
|
||||||
|
|
||||||
let handler = Update::filter_message().endpoint(
|
let handler = Update::filter_message().endpoint(
|
||||||
|msg: Message, bot: AutoSend<Bot>, messages_total: Arc<AtomicU64>| async move {
|
|bot: Bot, messages_total: Arc<AtomicU64>, msg: Message| async move {
|
||||||
let previous = messages_total.fetch_add(1, Ordering::Relaxed);
|
let previous = messages_total.fetch_add(1, Ordering::Relaxed);
|
||||||
bot.send_message(msg.chat.id, format!("I received {previous} messages in total."))
|
bot.send_message(msg.chat.id, format!("I received {previous} messages in total."))
|
||||||
.await?;
|
.await?;
|
||||||
|
|
|
@ -7,11 +7,11 @@ async fn main() {
|
||||||
pretty_env_logger::init();
|
pretty_env_logger::init();
|
||||||
log::info!("Starting throw dice bot...");
|
log::info!("Starting throw dice bot...");
|
||||||
|
|
||||||
let bot = Bot::from_env().auto_send();
|
let bot = Bot::from_env();
|
||||||
|
|
||||||
teloxide::repl(bot, |message: Message, bot: AutoSend<Bot>| async move {
|
teloxide::repl(bot, |bot: Bot, msg: Message| async move {
|
||||||
bot.send_dice(message.chat.id).await?;
|
bot.send_dice(msg.chat.id).await?;
|
||||||
respond(())
|
Ok(())
|
||||||
})
|
})
|
||||||
.await;
|
.await;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
[toolchain]
|
[toolchain]
|
||||||
channel = "nightly-2022-07-01"
|
channel = "nightly-2022-09-01"
|
||||||
components = ["rustfmt", "clippy"]
|
components = ["rustfmt", "clippy"]
|
||||||
profile = "minimal"
|
profile = "minimal"
|
||||||
|
|
|
@ -25,7 +25,7 @@
|
||||||
//! ```no_run
|
//! ```no_run
|
||||||
//! # use teloxide::utils::command::BotCommands;
|
//! # use teloxide::utils::command::BotCommands;
|
||||||
//! #[derive(BotCommands, Clone)]
|
//! #[derive(BotCommands, Clone)]
|
||||||
//! #[command(rename = "lowercase", description = "These commands are supported:")]
|
//! #[command(rename_rule = "lowercase", description = "These commands are supported:")]
|
||||||
//! enum Command {
|
//! enum Command {
|
||||||
//! #[command(description = "display this text.")]
|
//! #[command(description = "display this text.")]
|
||||||
//! Help,
|
//! Help,
|
||||||
|
@ -102,10 +102,10 @@
|
||||||
//! -- no problem, reuse [`dptree::Handler::filter`], [`dptree::case!`], and
|
//! -- no problem, reuse [`dptree::Handler::filter`], [`dptree::case!`], and
|
||||||
//! other combinators in the same way!
|
//! other combinators in the same way!
|
||||||
//!
|
//!
|
||||||
//! Finally, we define our endpoints like this:
|
//! Finally, we define our endpoints:
|
||||||
//!
|
//!
|
||||||
//! ```no_run
|
//! ```no_run
|
||||||
//! # use teloxide::{Bot, adaptors::AutoSend};
|
//! # use teloxide::Bot;
|
||||||
//! # use teloxide::types::{Message, CallbackQuery};
|
//! # use teloxide::types::{Message, CallbackQuery};
|
||||||
//! # use teloxide::dispatching::dialogue::{InMemStorage, Dialogue};
|
//! # use teloxide::dispatching::dialogue::{InMemStorage, Dialogue};
|
||||||
//! # enum State{}
|
//! # enum State{}
|
||||||
|
@ -113,47 +113,38 @@
|
||||||
//! type MyDialogue = Dialogue<State, InMemStorage<State>>;
|
//! type MyDialogue = Dialogue<State, InMemStorage<State>>;
|
||||||
//! type HandlerResult = Result<(), Box<dyn std::error::Error + Send + Sync>>;
|
//! type HandlerResult = Result<(), Box<dyn std::error::Error + Send + Sync>>;
|
||||||
//!
|
//!
|
||||||
//! async fn start(bot: AutoSend<Bot>, msg: Message, dialogue: MyDialogue) -> HandlerResult {
|
//! async fn start(bot: Bot, dialogue: MyDialogue, msg: Message) -> HandlerResult {
|
||||||
//! todo!()
|
//! todo!()
|
||||||
//! }
|
//! }
|
||||||
//!
|
//! async fn help(bot: Bot, msg: Message) -> HandlerResult {
|
||||||
//! async fn help(bot: AutoSend<Bot>, msg: Message) -> HandlerResult {
|
|
||||||
//! todo!()
|
//! todo!()
|
||||||
//! }
|
//! }
|
||||||
//!
|
//! async fn cancel(bot: Bot, dialogue: MyDialogue, msg: Message) -> HandlerResult {
|
||||||
//! async fn cancel(bot: AutoSend<Bot>, msg: Message, dialogue: MyDialogue) -> HandlerResult {
|
|
||||||
//! todo!()
|
//! todo!()
|
||||||
//! }
|
//! }
|
||||||
//!
|
//! async fn invalid_state(bot: Bot, msg: Message) -> HandlerResult {
|
||||||
//! async fn invalid_state(bot: AutoSend<Bot>, msg: Message) -> HandlerResult {
|
|
||||||
//! todo!()
|
//! todo!()
|
||||||
//! }
|
//! }
|
||||||
//!
|
//! async fn receive_full_name(bot: Bot, dialogue: MyDialogue, msg: Message) -> HandlerResult {
|
||||||
//! async fn receive_full_name(
|
|
||||||
//! bot: AutoSend<Bot>,
|
|
||||||
//! msg: Message,
|
|
||||||
//! dialogue: MyDialogue,
|
|
||||||
//! ) -> HandlerResult {
|
|
||||||
//! todo!()
|
//! todo!()
|
||||||
//! }
|
//! }
|
||||||
//!
|
|
||||||
//! async fn receive_product_selection(
|
//! async fn receive_product_selection(
|
||||||
//! bot: AutoSend<Bot>,
|
//! bot: Bot,
|
||||||
//! q: CallbackQuery,
|
|
||||||
//! dialogue: MyDialogue,
|
//! dialogue: MyDialogue,
|
||||||
//! full_name: String,
|
//! full_name: String, // Available from `State::ReceiveProductChoice`.
|
||||||
|
//! q: CallbackQuery,
|
||||||
//! ) -> HandlerResult {
|
//! ) -> HandlerResult {
|
||||||
//! todo!()
|
//! todo!()
|
||||||
//! }
|
//! }
|
||||||
//! ```
|
//! ```
|
||||||
//!
|
//!
|
||||||
//! Each parameter is supplied as a dependency by teloxide. In particular:
|
//! Each parameter is supplied as a dependency by `teloxide`. In particular:
|
||||||
//! - `bot: AutoSend<Bot>` comes from the dispatcher (see below);
|
//! - `bot: Bot` comes from the dispatcher (see below)
|
||||||
//! - `msg: Message` comes from [`Update::filter_message`];
|
//! - `msg: Message` comes from [`Update::filter_message`]
|
||||||
//! - `q: CallbackQuery` comes from [`Update::filter_callback_query`];
|
//! - `q: CallbackQuery` comes from [`Update::filter_callback_query`]
|
||||||
//! - `dialogue: MyDialogue` comes from [`dialogue::enter`];
|
//! - `dialogue: MyDialogue` comes from [`dialogue::enter`]
|
||||||
//! - `full_name: String` comes from `dptree::case![State::ReceiveProductChoice
|
//! - `full_name: String` comes from `dptree::case![State::ReceiveProductChoice
|
||||||
//! { full_name }]`.
|
//! { full_name }]`
|
||||||
//!
|
//!
|
||||||
//! Inside `main`, we plug the schema into [`Dispatcher`] like this:
|
//! Inside `main`, we plug the schema into [`Dispatcher`] like this:
|
||||||
//!
|
//!
|
||||||
|
@ -165,7 +156,7 @@
|
||||||
//! # fn schema() -> teloxide::dispatching::UpdateHandler<Box<dyn std::error::Error + Send + Sync + 'static>> { teloxide::dptree::entry() }
|
//! # fn schema() -> teloxide::dispatching::UpdateHandler<Box<dyn std::error::Error + Send + Sync + 'static>> { teloxide::dptree::entry() }
|
||||||
//! #[tokio::main]
|
//! #[tokio::main]
|
||||||
//! async fn main() {
|
//! async fn main() {
|
||||||
//! let bot = Bot::from_env().auto_send();
|
//! let bot = Bot::from_env();
|
||||||
//!
|
//!
|
||||||
//! Dispatcher::builder(bot, schema())
|
//! Dispatcher::builder(bot, schema())
|
||||||
//! .dependencies(dptree::deps![InMemStorage::<State>::new()])
|
//! .dependencies(dptree::deps![InMemStorage::<State>::new()])
|
||||||
|
@ -187,12 +178,36 @@
|
||||||
//! useful features. See [`examples/dispatching_features.rs`] as a more involved
|
//! useful features. See [`examples/dispatching_features.rs`] as a more involved
|
||||||
//! example.
|
//! example.
|
||||||
//!
|
//!
|
||||||
|
//! ## Dispatching or REPLs?
|
||||||
|
//!
|
||||||
|
//! The difference between dispatching and the REPLs ([`crate::repl`] & co) is
|
||||||
|
//! that dispatching gives you a greater degree of flexibility at the cost of a
|
||||||
|
//! bit more complicated setup.
|
||||||
|
//!
|
||||||
|
//! Here are things that dispatching can do, but REPLs can't:
|
||||||
|
//! - Handle different kinds of [`Update`]
|
||||||
|
//! - [Pass dependencies] to handlers
|
||||||
|
//! - Disable a [default Ctrl-C handling]
|
||||||
|
//! - Control your [default] and [error] handlers
|
||||||
|
//! - Use [dialogues]
|
||||||
|
//! - Use [`dptree`]-related functionality
|
||||||
|
//! - Probably more
|
||||||
|
//!
|
||||||
|
//! Thus, REPLs are good for simple bots and rapid prototyping, but for more
|
||||||
|
//! involved scenarios, we recommend using dispatching over REPLs.
|
||||||
|
//!
|
||||||
|
//! [Pass dependencies]: DispatcherBuilder#method.dependencies
|
||||||
|
//! [default Ctrl-C handling]: DispatcherBuilder#method.enable_ctrlc_handler
|
||||||
|
//! [default]: DispatcherBuilder#method.default_handler
|
||||||
|
//! [error]: DispatcherBuilder#method.error_handler
|
||||||
|
//! [dialogues]: dialogue
|
||||||
//! [`examples/purchase.rs`]: https://github.com/teloxide/teloxide/blob/master/examples/purchase.rs
|
//! [`examples/purchase.rs`]: https://github.com/teloxide/teloxide/blob/master/examples/purchase.rs
|
||||||
//! [`Update::filter_message`]: crate::types::Update::filter_message
|
//! [`Update::filter_message`]: crate::types::Update::filter_message
|
||||||
//! [`Update::filter_callback_query`]: crate::types::Update::filter_callback_query
|
//! [`Update::filter_callback_query`]: crate::types::Update::filter_callback_query
|
||||||
//! [chain of responsibility]: https://en.wikipedia.org/wiki/Chain-of-responsibility_pattern
|
//! [chain of responsibility]: https://en.wikipedia.org/wiki/Chain-of-responsibility_pattern
|
||||||
//! [dependency injection (DI)]: https://en.wikipedia.org/wiki/Dependency_injection
|
//! [dependency injection (DI)]: https://en.wikipedia.org/wiki/Dependency_injection
|
||||||
//! [`examples/dispatching_features.rs`]: https://github.com/teloxide/teloxide/blob/master/examples/dispatching_features.rs
|
//! [`examples/dispatching_features.rs`]: https://github.com/teloxide/teloxide/blob/master/examples/dispatching_features.rs
|
||||||
|
//! [`Update`]: crate::types::Update
|
||||||
|
|
||||||
#[cfg(all(feature = "ctrlc_handler"))]
|
#[cfg(all(feature = "ctrlc_handler"))]
|
||||||
pub mod repls;
|
pub mod repls;
|
||||||
|
@ -203,8 +218,6 @@ mod distribution;
|
||||||
mod filter_ext;
|
mod filter_ext;
|
||||||
mod handler_description;
|
mod handler_description;
|
||||||
mod handler_ext;
|
mod handler_ext;
|
||||||
mod handler_factory;
|
|
||||||
pub mod stop_token;
|
|
||||||
pub mod update_listeners;
|
pub mod update_listeners;
|
||||||
|
|
||||||
pub use crate::utils::shutdown_token::{IdleShutdownError, ShutdownToken};
|
pub use crate::utils::shutdown_token::{IdleShutdownError, ShutdownToken};
|
||||||
|
@ -213,5 +226,3 @@ pub use distribution::DefaultKey;
|
||||||
pub use filter_ext::{MessageFilterExt, UpdateFilterExt};
|
pub use filter_ext::{MessageFilterExt, UpdateFilterExt};
|
||||||
pub use handler_description::DpHandlerDescription;
|
pub use handler_description::DpHandlerDescription;
|
||||||
pub use handler_ext::{filter_command, HandlerExt};
|
pub use handler_ext::{filter_command, HandlerExt};
|
||||||
#[allow(deprecated)]
|
|
||||||
pub use handler_factory::HandlerFactory;
|
|
||||||
|
|
|
@ -13,26 +13,35 @@
|
||||||
//! [`examples/dialogue.rs`] clearly demonstrates the typical usage of
|
//! [`examples/dialogue.rs`] clearly demonstrates the typical usage of
|
||||||
//! dialogues. Your dialogue state can be represented as an enumeration:
|
//! dialogues. Your dialogue state can be represented as an enumeration:
|
||||||
//!
|
//!
|
||||||
//! ```ignore
|
//! ```no_run
|
||||||
//! #[derive(Clone, Default)]
|
//! #[derive(Clone, Default)]
|
||||||
//! pub enum State {
|
//! pub enum State {
|
||||||
//! #[default]
|
//! #[default]
|
||||||
//! Start,
|
//! Start,
|
||||||
//! ReceiveFullName,
|
//! ReceiveFullName,
|
||||||
//! ReceiveAge { full_name: String },
|
//! ReceiveAge {
|
||||||
//! ReceiveLocation { full_name: String, age: u8 },
|
//! full_name: String,
|
||||||
|
//! },
|
||||||
|
//! 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`, `receive_age` is invoked:
|
//! state is `ReceiveAge`, `receive_age` is invoked:
|
||||||
//!
|
//!
|
||||||
//! ```ignore
|
//! ```no_run
|
||||||
|
//! # use teloxide::{dispatching::dialogue::InMemStorage, prelude::*};
|
||||||
|
//! # type MyDialogue = Dialogue<State, InMemStorage<State>>;
|
||||||
|
//! # type HandlerResult = Result<(), Box<dyn std::error::Error + Send + Sync>>;
|
||||||
|
//! # #[derive(Clone, Debug)] enum State { ReceiveLocation { full_name: String, age: u8 } }
|
||||||
//! async fn receive_age(
|
//! async fn receive_age(
|
||||||
//! bot: AutoSend<Bot>,
|
//! bot: Bot,
|
||||||
//! msg: Message,
|
|
||||||
//! dialogue: MyDialogue,
|
//! dialogue: MyDialogue,
|
||||||
//! full_name: String, // Available from `State::ReceiveAge`.
|
//! full_name: String, // Available from `State::ReceiveAge`.
|
||||||
|
//! msg: Message,
|
||||||
//! ) -> HandlerResult {
|
//! ) -> HandlerResult {
|
||||||
//! match msg.text().map(|text| text.parse::<u8>()) {
|
//! match msg.text().map(|text| text.parse::<u8>()) {
|
||||||
//! Some(Ok(age)) => {
|
//! Some(Ok(age)) => {
|
||||||
|
@ -55,13 +64,17 @@
|
||||||
//! the dialogue, just call [`Dialogue::exit`] and it will be removed from the
|
//! the dialogue, just call [`Dialogue::exit`] and it will be removed from the
|
||||||
//! underlying storage:
|
//! underlying storage:
|
||||||
//!
|
//!
|
||||||
//! ```ignore
|
//! ```no_run
|
||||||
|
//! # use teloxide::{dispatching::dialogue::InMemStorage, prelude::*};
|
||||||
|
//! # type MyDialogue = Dialogue<State, InMemStorage<State>>;
|
||||||
|
//! # type HandlerResult = Result<(), Box<dyn std::error::Error + Send + Sync>>;
|
||||||
|
//! # #[derive(Clone, Debug)] enum State {}
|
||||||
//! async fn receive_location(
|
//! async fn receive_location(
|
||||||
//! bot: AutoSend<Bot>,
|
//! bot: Bot,
|
||||||
//! msg: Message,
|
|
||||||
//! dialogue: MyDialogue,
|
//! dialogue: MyDialogue,
|
||||||
//! (full_name, age): (String, u8), // Available from `State::ReceiveLocation`.
|
//! (full_name, age): (String, u8), // Available from `State::ReceiveLocation`.
|
||||||
//! ) -> anyhow::Result<()> {
|
//! msg: Message,
|
||||||
|
//! ) -> HandlerResult {
|
||||||
//! match msg.text() {
|
//! match msg.text() {
|
||||||
//! Some(location) => {
|
//! Some(location) => {
|
||||||
//! let message =
|
//! let message =
|
||||||
|
@ -198,6 +211,7 @@ where
|
||||||
/// - `Upd`
|
/// - `Upd`
|
||||||
///
|
///
|
||||||
/// [`HandlerExt::enter_dialogue`]: super::HandlerExt::enter_dialogue
|
/// [`HandlerExt::enter_dialogue`]: super::HandlerExt::enter_dialogue
|
||||||
|
#[must_use]
|
||||||
pub fn enter<Upd, S, D, Output>() -> Handler<'static, DependencyMap, Output, DpHandlerDescription>
|
pub fn enter<Upd, S, D, Output>() -> Handler<'static, DependencyMap, Output, DpHandlerDescription>
|
||||||
where
|
where
|
||||||
S: Storage<D> + ?Sized + Send + Sync + 'static,
|
S: Storage<D> + ?Sized + Send + Sync + 'static,
|
||||||
|
@ -206,12 +220,11 @@ where
|
||||||
Upd: GetChatId + Clone + Send + Sync + 'static,
|
Upd: GetChatId + Clone + Send + Sync + 'static,
|
||||||
Output: Send + Sync + 'static,
|
Output: Send + Sync + 'static,
|
||||||
{
|
{
|
||||||
dptree::entry()
|
dptree::filter_map(|storage: Arc<S>, upd: Upd| {
|
||||||
.chain(dptree::filter_map(|storage: Arc<S>, upd: Upd| {
|
|
||||||
let chat_id = upd.chat_id()?;
|
let chat_id = upd.chat_id()?;
|
||||||
Some(Dialogue::new(storage, chat_id))
|
Some(Dialogue::new(storage, chat_id))
|
||||||
}))
|
})
|
||||||
.chain(dptree::filter_map_async(|dialogue: Dialogue<D, S>| async move {
|
.filter_map_async(|dialogue: Dialogue<D, S>| async move {
|
||||||
match dialogue.get_or_default().await {
|
match dialogue.get_or_default().await {
|
||||||
Ok(dialogue) => Some(dialogue),
|
Ok(dialogue) => Some(dialogue),
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
|
@ -219,5 +232,5 @@ where
|
||||||
None
|
None
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}))
|
})
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
use crate::{
|
use crate::{
|
||||||
dispatching::{
|
dispatching::{
|
||||||
distribution::default_distribution_function, stop_token::StopToken, update_listeners,
|
distribution::default_distribution_function, update_listeners,
|
||||||
update_listeners::UpdateListener, DefaultKey, DpHandlerDescription, ShutdownToken,
|
update_listeners::UpdateListener, DefaultKey, DpHandlerDescription, ShutdownToken,
|
||||||
},
|
},
|
||||||
error_handlers::{ErrorHandler, LoggingErrorHandler},
|
error_handlers::{ErrorHandler, LoggingErrorHandler},
|
||||||
|
@ -27,6 +27,9 @@ use std::{
|
||||||
};
|
};
|
||||||
|
|
||||||
/// The builder for [`Dispatcher`].
|
/// The builder for [`Dispatcher`].
|
||||||
|
///
|
||||||
|
/// See also: ["Dispatching or
|
||||||
|
/// REPLs?"](../dispatching/index.html#dispatching-or-repls)
|
||||||
pub struct DispatcherBuilder<R, Err, Key> {
|
pub struct DispatcherBuilder<R, Err, Key> {
|
||||||
bot: R,
|
bot: R,
|
||||||
dependencies: DependencyMap,
|
dependencies: DependencyMap,
|
||||||
|
@ -171,11 +174,14 @@ where
|
||||||
|
|
||||||
/// The base for update dispatching.
|
/// The base for update dispatching.
|
||||||
///
|
///
|
||||||
/// Updates from different chats are handles concurrently, whereas updates from
|
/// Updates from different chats are handled concurrently, whereas updates from
|
||||||
/// the same chats are handled sequentially. If the dispatcher is unable to
|
/// the same chats are handled sequentially. If the dispatcher is unable to
|
||||||
/// determine a chat ID of an incoming update, it will be handled concurrently.
|
/// determine a chat ID of an incoming update, it will be handled concurrently.
|
||||||
/// Note that this behaviour can be altered with [`distribution_function`].
|
/// Note that this behaviour can be altered with [`distribution_function`].
|
||||||
///
|
///
|
||||||
|
/// See also: ["Dispatching or
|
||||||
|
/// REPLs?"](../dispatching/index.html#dispatching-or-repls)
|
||||||
|
///
|
||||||
/// [`distribution_function`]: DispatcherBuilder::distribution_function
|
/// [`distribution_function`]: DispatcherBuilder::distribution_function
|
||||||
pub struct Dispatcher<R, Err, Key> {
|
pub struct Dispatcher<R, Err, Key> {
|
||||||
bot: R,
|
bot: R,
|
||||||
|
@ -281,14 +287,14 @@ where
|
||||||
/// This method adds the same dependencies as [`Dispatcher::dispatch`].
|
/// This method adds the same dependencies as [`Dispatcher::dispatch`].
|
||||||
///
|
///
|
||||||
/// [`shutdown`]: ShutdownToken::shutdown
|
/// [`shutdown`]: ShutdownToken::shutdown
|
||||||
pub async fn dispatch_with_listener<'a, UListener, ListenerE, Eh>(
|
pub async fn dispatch_with_listener<'a, UListener, Eh>(
|
||||||
&'a mut self,
|
&'a mut self,
|
||||||
mut update_listener: UListener,
|
mut update_listener: UListener,
|
||||||
update_listener_error_handler: Arc<Eh>,
|
update_listener_error_handler: Arc<Eh>,
|
||||||
) where
|
) where
|
||||||
UListener: UpdateListener<ListenerE> + 'a,
|
UListener: UpdateListener + 'a,
|
||||||
Eh: ErrorHandler<ListenerE> + 'a,
|
Eh: ErrorHandler<UListener::Err> + 'a,
|
||||||
ListenerE: Debug,
|
UListener::Err: Debug,
|
||||||
{
|
{
|
||||||
// FIXME: there should be a way to check if dependency is already inserted
|
// FIXME: there should be a way to check if dependency is already inserted
|
||||||
let me = self.bot.get_me().send().await.expect("Failed to retrieve 'me'");
|
let me = self.bot.get_me().send().await.expect("Failed to retrieve 'me'");
|
||||||
|
|
|
@ -56,11 +56,11 @@ mod private {
|
||||||
macro_rules! define_message_ext {
|
macro_rules! define_message_ext {
|
||||||
($( ($func:ident, $fn_name:path) ,)*) => {
|
($( ($func:ident, $fn_name:path) ,)*) => {
|
||||||
define_ext! {
|
define_ext! {
|
||||||
MessageFilterExt, crate::types::Message =>
|
MessageFilterExt, Message =>
|
||||||
$((
|
$((
|
||||||
$func,
|
$func,
|
||||||
(|x| $fn_name(&x).map(ToOwned::to_owned)),
|
(|x| $fn_name(&x).map(ToOwned::to_owned)),
|
||||||
concat!("Applies the [`crate::types::", stringify!($fn_name), "`] filter.")
|
concat!("Applies the [`", stringify!($fn_name), "`] filter.")
|
||||||
),)*
|
),)*
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -89,14 +89,14 @@ define_message_ext! {
|
||||||
macro_rules! define_update_ext {
|
macro_rules! define_update_ext {
|
||||||
($( ($func:ident, $kind:path, $Allowed:ident) ,)*) => {
|
($( ($func:ident, $kind:path, $Allowed:ident) ,)*) => {
|
||||||
define_ext! {
|
define_ext! {
|
||||||
UpdateFilterExt, crate::types::Update =>
|
UpdateFilterExt, Update =>
|
||||||
$((
|
$((
|
||||||
$func,
|
$func,
|
||||||
|update: Update| match update.kind {
|
|update: Update| match update.kind {
|
||||||
$kind(x) => Some(x),
|
$kind(x) => Some(x),
|
||||||
_ => None,
|
_ => None,
|
||||||
},
|
},
|
||||||
concat!("Filters out [`crate::types::", stringify!($kind), "`] objects."),
|
concat!("Filters out [`", stringify!($kind), "`] objects."),
|
||||||
$Allowed
|
$Allowed
|
||||||
),)*
|
),)*
|
||||||
}
|
}
|
||||||
|
|
|
@ -86,7 +86,7 @@ mod tests {
|
||||||
use crate as teloxide; // fixup for the `BotCommands` macro
|
use crate as teloxide; // fixup for the `BotCommands` macro
|
||||||
|
|
||||||
#[derive(BotCommands, Clone)]
|
#[derive(BotCommands, Clone)]
|
||||||
#[command(rename = "lowercase")]
|
#[command(rename_rule = "lowercase")]
|
||||||
enum Cmd {
|
enum Cmd {
|
||||||
B,
|
B,
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,9 +8,6 @@ use crate::{
|
||||||
};
|
};
|
||||||
use dptree::{di::DependencyMap, Handler};
|
use dptree::{di::DependencyMap, Handler};
|
||||||
|
|
||||||
#[allow(deprecated)]
|
|
||||||
use crate::dispatching::HandlerFactory;
|
|
||||||
|
|
||||||
use std::fmt::Debug;
|
use std::fmt::Debug;
|
||||||
|
|
||||||
/// Extension methods for working with `dptree` handlers.
|
/// Extension methods for working with `dptree` handlers.
|
||||||
|
@ -51,13 +48,6 @@ pub trait HandlerExt<Output> {
|
||||||
<S as Storage<D>>::Error: Debug + Send,
|
<S as Storage<D>>::Error: Debug + Send,
|
||||||
D: Default + Send + Sync + 'static,
|
D: Default + Send + Sync + 'static,
|
||||||
Upd: GetChatId + Clone + Send + Sync + 'static;
|
Upd: GetChatId + Clone + Send + Sync + 'static;
|
||||||
|
|
||||||
#[must_use]
|
|
||||||
#[deprecated(note = "Use the teloxide::handler! API")]
|
|
||||||
#[allow(deprecated)]
|
|
||||||
fn dispatch_by<F>(self) -> Self
|
|
||||||
where
|
|
||||||
F: HandlerFactory<Out = Output>;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<Output> HandlerExt<Output> for Handler<'static, DependencyMap, Output, DpHandlerDescription>
|
impl<Output> HandlerExt<Output> for Handler<'static, DependencyMap, Output, DpHandlerDescription>
|
||||||
|
@ -80,14 +70,6 @@ where
|
||||||
{
|
{
|
||||||
self.chain(super::dialogue::enter::<Upd, S, D, Output>())
|
self.chain(super::dialogue::enter::<Upd, S, D, Output>())
|
||||||
}
|
}
|
||||||
|
|
||||||
#[allow(deprecated)]
|
|
||||||
fn dispatch_by<F>(self) -> Self
|
|
||||||
where
|
|
||||||
F: HandlerFactory<Out = Output>,
|
|
||||||
{
|
|
||||||
self.chain(F::handler())
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returns a handler that accepts a parsed command `C`.
|
/// Returns a handler that accepts a parsed command `C`.
|
||||||
|
@ -100,13 +82,14 @@ where
|
||||||
///
|
///
|
||||||
/// - [`crate::types::Message`]
|
/// - [`crate::types::Message`]
|
||||||
/// - [`crate::types::Me`]
|
/// - [`crate::types::Me`]
|
||||||
|
#[must_use]
|
||||||
pub fn filter_command<C, Output>() -> Handler<'static, DependencyMap, Output, DpHandlerDescription>
|
pub fn filter_command<C, Output>() -> Handler<'static, DependencyMap, Output, DpHandlerDescription>
|
||||||
where
|
where
|
||||||
C: BotCommands + Send + Sync + 'static,
|
C: BotCommands + Send + Sync + 'static,
|
||||||
Output: Send + Sync + 'static,
|
Output: Send + Sync + 'static,
|
||||||
{
|
{
|
||||||
dptree::entry().chain(dptree::filter_map(move |message: Message, me: Me| {
|
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");
|
||||||
message.text().and_then(|text| C::parse(text, bot_name).ok())
|
message.text().and_then(|text| C::parse(text, &bot_name).ok())
|
||||||
}))
|
})
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,11 +0,0 @@
|
||||||
use dptree::{di::DependencyMap, Handler};
|
|
||||||
|
|
||||||
use crate::dispatching::DpHandlerDescription;
|
|
||||||
|
|
||||||
/// Something that can construct a handler.
|
|
||||||
#[deprecated(note = "Use the teloxide::handler! API")]
|
|
||||||
pub trait HandlerFactory {
|
|
||||||
type Out;
|
|
||||||
|
|
||||||
fn handler() -> Handler<'static, DependencyMap, Self::Out, DpHandlerDescription>;
|
|
||||||
}
|
|
|
@ -1,4 +1,12 @@
|
||||||
//! REPLs for dispatching updates.
|
//! [REPL]s for dispatching updates.
|
||||||
|
//!
|
||||||
|
//! This module provides utilities for easy update handling. They accept a
|
||||||
|
//! single "handler" function that processes all updates of a certain kind. Note
|
||||||
|
//! that REPLs are meant to be used for simple scenarios, such as prototyping,
|
||||||
|
//! inasmuch they lack configuration and some [advanced features].
|
||||||
|
//!
|
||||||
|
//! [REPL]: https://en.wikipedia.org/wiki/Read-eval-print_loop
|
||||||
|
//! [advanced features]: crate::dispatching#dispatching-or-repls
|
||||||
|
|
||||||
mod commands_repl;
|
mod commands_repl;
|
||||||
mod repl;
|
mod repl;
|
||||||
|
|
2
src/dispatching/repls/caution.md
Normal file
2
src/dispatching/repls/caution.md
Normal file
|
@ -0,0 +1,2 @@
|
||||||
|
**DO NOT** use this function together with [`Dispatcher`] and other REPLs,
|
||||||
|
because Telegram disallow multiple requests at the same time from the same bot.
|
|
@ -3,41 +3,68 @@ use crate::{
|
||||||
update_listeners, update_listeners::UpdateListener, HandlerExt, UpdateFilterExt,
|
update_listeners, update_listeners::UpdateListener, HandlerExt, UpdateFilterExt,
|
||||||
},
|
},
|
||||||
error_handlers::LoggingErrorHandler,
|
error_handlers::LoggingErrorHandler,
|
||||||
|
requests::{Requester, ResponseResult},
|
||||||
types::Update,
|
types::Update,
|
||||||
utils::command::BotCommands,
|
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};
|
||||||
use teloxide_core::requests::Requester;
|
|
||||||
|
|
||||||
/// A [REPL] for commands.
|
/// A [REPL] for commands.
|
||||||
|
//
|
||||||
///
|
///
|
||||||
/// All errors from an update listener and handler will be logged.
|
//
|
||||||
///
|
#[doc = include_str!("preamble.md")]
|
||||||
/// REPLs are meant only for simple bots and rapid prototyping. If you need to
|
|
||||||
/// supply dependencies or describe more complex dispatch logic, please use
|
|
||||||
/// [`Dispatcher`].
|
|
||||||
///
|
|
||||||
/// ## 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.
|
|
||||||
///
|
|
||||||
/// ## Dependency requirements
|
|
||||||
///
|
|
||||||
/// - Those of [`HandlerExt::filter_command`].
|
|
||||||
///
|
///
|
||||||
/// [REPL]: https://en.wikipedia.org/wiki/Read-eval-print_loop
|
/// [REPL]: https://en.wikipedia.org/wiki/Read-eval-print_loop
|
||||||
/// [`Dispatcher`]: crate::dispatching::Dispatcher
|
///
|
||||||
|
/// ## Signature
|
||||||
|
///
|
||||||
|
/// Don't be scared by many trait bounds in the signature, in essence they
|
||||||
|
/// require:
|
||||||
|
///
|
||||||
|
/// 1. `bot` is a bot, client for the Telegram bot API. It is represented via
|
||||||
|
/// the [`Requester`] trait.
|
||||||
|
/// 2. `handler` is an `async` function that takes arguments from
|
||||||
|
/// [`DependencyMap`] (see below) and returns [`ResponseResult`].
|
||||||
|
/// 3. `cmd` is a type hint for your command enumeration
|
||||||
|
/// `MyCommand`: just write `MyCommand::ty()`. Note that `MyCommand` must
|
||||||
|
/// implement the [`BotCommands`] trait, typically via
|
||||||
|
/// `#[derive(BotCommands)]`.
|
||||||
|
///
|
||||||
|
/// All the other requirements are about thread safety and data validity and can
|
||||||
|
/// be ignored for most of the time.
|
||||||
|
///
|
||||||
|
/// ## Handler arguments
|
||||||
|
///
|
||||||
|
/// `teloxide` provides the following types to the `handler`:
|
||||||
|
/// - [`Message`]
|
||||||
|
/// - `R` (type of the `bot`)
|
||||||
|
/// - `Cmd` (type of the parsed command)
|
||||||
|
/// - [`Me`]
|
||||||
|
///
|
||||||
|
/// Each of these types can be accepted as a handler parameter. Note that they
|
||||||
|
/// aren't all required at the same time: e.g., you can take only the bot and
|
||||||
|
/// the command without [`Me`] and [`Message`].
|
||||||
|
///
|
||||||
|
/// [`Me`]: crate::types::Me
|
||||||
|
/// [`Message`]: crate::types::Message
|
||||||
|
///
|
||||||
|
/// ## Stopping
|
||||||
|
//
|
||||||
|
#[doc = include_str!("stopping.md")]
|
||||||
|
///
|
||||||
|
/// ## Caution
|
||||||
|
//
|
||||||
|
#[doc = include_str!("caution.md")]
|
||||||
|
///
|
||||||
#[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, Args>(bot: R, handler: H, cmd: PhantomData<Cmd>)
|
||||||
where
|
where
|
||||||
Cmd: BotCommands + 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,
|
||||||
E: Debug + Send + Sync + 'static,
|
H: Injectable<DependencyMap, ResponseResult<()>, Args> + Send + Sync + 'static,
|
||||||
|
Cmd: BotCommands + Send + Sync + 'static,
|
||||||
{
|
{
|
||||||
let cloned_bot = bot.clone();
|
let cloned_bot = bot.clone();
|
||||||
|
|
||||||
|
@ -50,51 +77,77 @@ where
|
||||||
.await;
|
.await;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Like [`commands_repl`], but with a custom [`UpdateListener`].
|
/// A [REPL] for commands, with a custom [`UpdateListener`].
|
||||||
|
//
|
||||||
///
|
///
|
||||||
/// All errors from an update listener and handler will be logged.
|
//
|
||||||
|
#[doc = include_str!("preamble.md")]
|
||||||
///
|
///
|
||||||
/// REPLs are meant only for simple bots and rapid prototyping. If you need to
|
/// [REPL]: https://en.wikipedia.org/wiki/Read-eval-print_loop
|
||||||
/// supply dependencies or describe more complex dispatch logic, please use
|
///
|
||||||
/// [`Dispatcher`].
|
/// ## Signature
|
||||||
|
///
|
||||||
|
/// Don't be scared by many trait bounds in the signature, in essence they
|
||||||
|
/// require:
|
||||||
|
///
|
||||||
|
/// 1. `bot` is a bot, client for the Telegram bot API. It is represented via
|
||||||
|
/// the [`Requester`] trait.
|
||||||
|
/// 2. `handler` is an `async` function that takes arguments from
|
||||||
|
/// [`DependencyMap`] (see below) and returns [`ResponseResult`].
|
||||||
|
/// 3. `listener` is something that takes updates from a Telegram server and
|
||||||
|
/// implements [`UpdateListener`].
|
||||||
|
/// 4. `cmd` is a type hint for your command enumeration `MyCommand`: just
|
||||||
|
/// write `MyCommand::ty()`. Note that `MyCommand` must implement the
|
||||||
|
/// [`BotCommands`] trait, typically via `#[derive(BotCommands)]`.
|
||||||
|
///
|
||||||
|
/// All the other requirements are about thread safety and data validity and can
|
||||||
|
/// be ignored for most of the time.
|
||||||
|
///
|
||||||
|
/// ## Handler arguments
|
||||||
|
///
|
||||||
|
/// `teloxide` provides the following types to the `handler`:
|
||||||
|
/// - [`Message`]
|
||||||
|
/// - `R` (type of the `bot`)
|
||||||
|
/// - `Cmd` (type of the parsed command)
|
||||||
|
/// - [`Me`]
|
||||||
|
///
|
||||||
|
/// Each of these types can be accepted as a handler parameter. Note that they
|
||||||
|
/// aren't all required at the same time: e.g., you can take only the bot and
|
||||||
|
/// the command without [`Me`] and [`Message`].
|
||||||
|
///
|
||||||
|
/// [`Me`]: crate::types::Me
|
||||||
|
/// [`Message`]: crate::types::Message
|
||||||
|
///
|
||||||
|
/// ## Stopping
|
||||||
|
//
|
||||||
|
#[doc = include_str!("stopping.md")]
|
||||||
///
|
///
|
||||||
/// ## Caution
|
/// ## Caution
|
||||||
|
//
|
||||||
|
#[doc = include_str!("caution.md")]
|
||||||
///
|
///
|
||||||
/// **DO NOT** use this function together with [`Dispatcher`] and other REPLs,
|
|
||||||
/// because Telegram disallow multiple requests at the same time from the same
|
|
||||||
/// bot.
|
|
||||||
///
|
|
||||||
/// ## Dependency requirements
|
|
||||||
///
|
|
||||||
/// - Those of [`HandlerExt::filter_command`].
|
|
||||||
///
|
|
||||||
/// [`Dispatcher`]: crate::dispatching::Dispatcher
|
|
||||||
/// [`commands_repl`]: crate::dispatching::repls::commands_repl()
|
|
||||||
/// [`UpdateListener`]: crate::dispatching::update_listeners::UpdateListener
|
|
||||||
#[cfg(feature = "ctrlc_handler")]
|
#[cfg(feature = "ctrlc_handler")]
|
||||||
pub async fn commands_repl_with_listener<'a, R, Cmd, H, L, ListenerE, E, Args>(
|
pub async fn commands_repl_with_listener<'a, R, Cmd, H, L, Args>(
|
||||||
bot: R,
|
bot: R,
|
||||||
handler: H,
|
handler: H,
|
||||||
listener: L,
|
listener: L,
|
||||||
_cmd: PhantomData<Cmd>,
|
cmd: PhantomData<Cmd>,
|
||||||
) where
|
) where
|
||||||
Cmd: BotCommands + Send + Sync + 'static,
|
Cmd: BotCommands + Send + Sync + 'static,
|
||||||
H: Injectable<DependencyMap, Result<(), E>, Args> + Send + Sync + 'static,
|
H: Injectable<DependencyMap, ResponseResult<()>, Args> + Send + Sync + 'static,
|
||||||
L: UpdateListener<ListenerE> + Send + 'a,
|
L: UpdateListener + Send + 'a,
|
||||||
ListenerE: Debug + Send + 'a,
|
L::Err: Debug + Send + 'a,
|
||||||
R: Requester + Clone + Send + Sync + 'static,
|
R: Requester + Clone + Send + Sync + 'static,
|
||||||
E: Debug + Send + Sync + 'static,
|
|
||||||
{
|
{
|
||||||
use crate::dispatching::Dispatcher;
|
use crate::dispatching::Dispatcher;
|
||||||
|
|
||||||
|
let _ = cmd;
|
||||||
|
|
||||||
// Other update types are of no interest to use since this REPL is only for
|
// Other update types are of no interest to use since this REPL is only for
|
||||||
// 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 {});
|
||||||
|
|
||||||
Dispatcher::builder(
|
Dispatcher::builder(bot, Update::filter_message().filter_command::<Cmd>().endpoint(handler))
|
||||||
bot,
|
|
||||||
Update::filter_message().filter_command::<Cmd>().chain(dptree::endpoint(handler)),
|
|
||||||
)
|
|
||||||
.default_handler(ignore_update)
|
.default_handler(ignore_update)
|
||||||
.enable_ctrlc_handler()
|
.enable_ctrlc_handler()
|
||||||
.build()
|
.build()
|
||||||
|
|
7
src/dispatching/repls/preamble.md
Normal file
7
src/dispatching/repls/preamble.md
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
REPLs are meant only for simple bots and rapid prototyping.
|
||||||
|
If you need to supply dependencies or describe more complex dispatch logic, please use [`Dispatcher`].
|
||||||
|
See also: ["Dispatching or REPLs?"](dispatching/index.html#dispatching-or-repls).
|
||||||
|
|
||||||
|
[`Dispatcher`]: crate::dispatching::Dispatcher
|
||||||
|
|
||||||
|
All errors from the handler and update listener will be logged.
|
|
@ -1,67 +1,113 @@
|
||||||
use crate::{
|
use crate::{
|
||||||
dispatching::{update_listeners, update_listeners::UpdateListener, UpdateFilterExt},
|
dispatching::{update_listeners, update_listeners::UpdateListener, UpdateFilterExt},
|
||||||
error_handlers::{LoggingErrorHandler, OnError},
|
error_handlers::LoggingErrorHandler,
|
||||||
|
requests::{Requester, ResponseResult},
|
||||||
types::Update,
|
types::Update,
|
||||||
};
|
};
|
||||||
use dptree::di::{DependencyMap, Injectable};
|
use dptree::di::{DependencyMap, Injectable};
|
||||||
use std::fmt::Debug;
|
use std::fmt::Debug;
|
||||||
use teloxide_core::requests::Requester;
|
|
||||||
|
|
||||||
/// A [REPL] for messages.
|
/// A [REPL] for messages.
|
||||||
|
//
|
||||||
///
|
///
|
||||||
/// All errors from an update listener and a handler will be logged.
|
//
|
||||||
///
|
#[doc = include_str!("preamble.md")]
|
||||||
/// REPLs are meant only for simple bots and rapid prototyping. If you need to
|
|
||||||
/// supply dependencies or describe more complex dispatch logic, please use
|
|
||||||
/// [`Dispatcher`].
|
|
||||||
///
|
|
||||||
/// ## 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
|
/// [REPL]: https://en.wikipedia.org/wiki/Read-eval-print_loop
|
||||||
/// [`Dispatcher`]: crate::dispatching::Dispatcher
|
///
|
||||||
|
/// ## Signature
|
||||||
|
///
|
||||||
|
/// Don't be scared by many trait bounds in the signature, in essence they
|
||||||
|
/// require:
|
||||||
|
///
|
||||||
|
/// 1. `bot` is a bot, client for the Telegram bot API. It is represented via
|
||||||
|
/// the [`Requester`] trait.
|
||||||
|
/// 2. `handler` is an `async` function that takes arguments from
|
||||||
|
/// [`DependencyMap`] (see below) and returns [`ResponseResult`].
|
||||||
|
///
|
||||||
|
/// ## Handler arguments
|
||||||
|
///
|
||||||
|
/// `teloxide` provides the following types to the `handler`:
|
||||||
|
/// - [`Message`]
|
||||||
|
/// - `R` (type of the `bot`)
|
||||||
|
/// - [`Me`]
|
||||||
|
///
|
||||||
|
/// Each of these types can be accepted as a handler parameter. Note that they
|
||||||
|
/// aren't all required at the same time: e.g., you can take only the bot and
|
||||||
|
/// the message without [`Me`].
|
||||||
|
///
|
||||||
|
/// [`Me`]: crate::types::Me
|
||||||
|
/// [`Message`]: crate::types::Message
|
||||||
|
///
|
||||||
|
/// ## Stopping
|
||||||
|
//
|
||||||
|
#[doc = include_str!("stopping.md")]
|
||||||
|
///
|
||||||
|
/// ## Caution
|
||||||
|
//
|
||||||
|
#[doc = include_str!("caution.md")]
|
||||||
|
///
|
||||||
#[cfg(feature = "ctrlc_handler")]
|
#[cfg(feature = "ctrlc_handler")]
|
||||||
pub async fn repl<R, H, E, Args>(bot: R, handler: H)
|
pub async fn repl<R, H, Args>(bot: R, handler: H)
|
||||||
where
|
where
|
||||||
H: Injectable<DependencyMap, Result<(), E>, Args> + Send + Sync + 'static,
|
|
||||||
Result<(), E>: OnError<E>,
|
|
||||||
E: Debug + Send + Sync + 'static,
|
|
||||||
R: Requester + Send + Sync + Clone + 'static,
|
R: Requester + Send + Sync + Clone + 'static,
|
||||||
<R as Requester>::GetUpdates: Send,
|
<R as Requester>::GetUpdates: Send,
|
||||||
|
H: Injectable<DependencyMap, ResponseResult<()>, Args> + Send + Sync + 'static,
|
||||||
{
|
{
|
||||||
let cloned_bot = bot.clone();
|
let cloned_bot = bot.clone();
|
||||||
repl_with_listener(bot, handler, update_listeners::polling_default(cloned_bot).await).await;
|
repl_with_listener(bot, handler, update_listeners::polling_default(cloned_bot).await).await;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Like [`repl`], but with a custom [`UpdateListener`].
|
/// A [REPL] for messages, with a custom [`UpdateListener`].
|
||||||
|
//
|
||||||
///
|
///
|
||||||
/// All errors from an update listener and handler will be logged.
|
//
|
||||||
|
#[doc = include_str!("preamble.md")]
|
||||||
///
|
///
|
||||||
/// REPLs are meant only for simple bots and rapid prototyping. If you need to
|
/// [REPL]: https://en.wikipedia.org/wiki/Read-eval-print_loop
|
||||||
/// supply dependencies or describe more complex dispatch logic, please use
|
|
||||||
/// [`Dispatcher`].
|
|
||||||
///
|
|
||||||
/// # 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
|
|
||||||
/// [`repl`]: crate::dispatching::repls::repl()
|
|
||||||
/// [`UpdateListener`]: crate::dispatching::update_listeners::UpdateListener
|
/// [`UpdateListener`]: crate::dispatching::update_listeners::UpdateListener
|
||||||
|
///
|
||||||
|
/// ## Signature
|
||||||
|
///
|
||||||
|
/// Don't be scared by many trait bounds in the signature, in essence they
|
||||||
|
/// require:
|
||||||
|
///
|
||||||
|
/// 1. `bot` is a bot, client for the Telegram bot API. It is represented via
|
||||||
|
/// the [`Requester`] trait.
|
||||||
|
/// 2. `handler` is an `async` function that takes arguments from
|
||||||
|
/// [`DependencyMap`] (see below) and returns [`ResponseResult`].
|
||||||
|
/// 3. `listener` is something that takes updates from a Telegram server and
|
||||||
|
/// implements [`UpdateListener`].
|
||||||
|
///
|
||||||
|
/// ## Handler arguments
|
||||||
|
///
|
||||||
|
/// `teloxide` provides the following types to the `handler`:
|
||||||
|
/// - [`Message`]
|
||||||
|
/// - `R` (type of the `bot`)
|
||||||
|
/// - [`Me`]
|
||||||
|
///
|
||||||
|
/// Each of these types can be accepted as a handler parameter. Note that they
|
||||||
|
/// aren't all required at the same time: e.g., you can take only the bot and
|
||||||
|
/// the message without [`Me`].
|
||||||
|
///
|
||||||
|
/// [`Me`]: crate::types::Me
|
||||||
|
/// [`Message`]: crate::types::Message
|
||||||
|
///
|
||||||
|
/// ## Stopping
|
||||||
|
//
|
||||||
|
#[doc = include_str!("stopping.md")]
|
||||||
|
///
|
||||||
|
/// ## Caution
|
||||||
|
//
|
||||||
|
#[doc = include_str!("caution.md")]
|
||||||
|
///
|
||||||
#[cfg(feature = "ctrlc_handler")]
|
#[cfg(feature = "ctrlc_handler")]
|
||||||
pub async fn repl_with_listener<'a, R, H, E, L, ListenerE, Args>(bot: R, handler: H, listener: L)
|
pub async fn repl_with_listener<R, H, L, Args>(bot: R, handler: H, listener: L)
|
||||||
where
|
where
|
||||||
H: Injectable<DependencyMap, Result<(), E>, Args> + Send + Sync + 'static,
|
|
||||||
L: UpdateListener<ListenerE> + Send + 'a,
|
|
||||||
ListenerE: Debug,
|
|
||||||
Result<(), E>: OnError<E>,
|
|
||||||
E: Debug + Send + Sync + 'static,
|
|
||||||
R: Requester + Clone + Send + Sync + 'static,
|
R: Requester + Clone + Send + Sync + 'static,
|
||||||
|
H: Injectable<DependencyMap, ResponseResult<()>, Args> + Send + Sync + 'static,
|
||||||
|
L: UpdateListener + Send,
|
||||||
|
L::Err: Debug,
|
||||||
{
|
{
|
||||||
use crate::dispatching::Dispatcher;
|
use crate::dispatching::Dispatcher;
|
||||||
|
|
||||||
|
@ -69,7 +115,7 @@ 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 {});
|
||||||
|
|
||||||
Dispatcher::builder(bot, Update::filter_message().chain(dptree::endpoint(handler)))
|
Dispatcher::builder(bot, Update::filter_message().endpoint(handler))
|
||||||
.default_handler(ignore_update)
|
.default_handler(ignore_update)
|
||||||
.enable_ctrlc_handler()
|
.enable_ctrlc_handler()
|
||||||
.build()
|
.build()
|
||||||
|
@ -83,7 +129,7 @@ where
|
||||||
#[test]
|
#[test]
|
||||||
fn repl_is_send() {
|
fn repl_is_send() {
|
||||||
let bot = crate::Bot::new("");
|
let bot = crate::Bot::new("");
|
||||||
let repl = crate::repl(bot, || async { crate::respond(()) });
|
let repl = crate::repl(bot, || async { Ok(()) });
|
||||||
assert_send(&repl);
|
assert_send(&repl);
|
||||||
|
|
||||||
fn assert_send(_: &impl Send) {}
|
fn assert_send(_: &impl Send) {}
|
||||||
|
|
4
src/dispatching/repls/stopping.md
Normal file
4
src/dispatching/repls/stopping.md
Normal file
|
@ -0,0 +1,4 @@
|
||||||
|
To stop a REPL, simply press `Ctrl`+`C` in the terminal where you run the program.
|
||||||
|
Note that graceful shutdown may take some time (we plan to improve this, see [#711]).
|
||||||
|
|
||||||
|
[#711]: https://github.com/teloxide/teloxide/issues/711
|
|
@ -1,79 +0,0 @@
|
||||||
//! A stop token used to stop a listener.
|
|
||||||
|
|
||||||
use std::{future::Future, pin::Pin, task};
|
|
||||||
|
|
||||||
use futures::future::{pending, AbortHandle, Abortable, Pending};
|
|
||||||
|
|
||||||
/// A stop token allows you to stop a listener.
|
|
||||||
///
|
|
||||||
/// See also: [`UpdateListener::stop_token`].
|
|
||||||
///
|
|
||||||
/// [`UpdateListener::stop_token`]:
|
|
||||||
/// crate::dispatching::update_listeners::UpdateListener::stop_token
|
|
||||||
pub trait StopToken {
|
|
||||||
/// Stop the listener linked to this token.
|
|
||||||
fn stop(self);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// A stop token which does nothing. May be used in prototyping or in cases
|
|
||||||
/// where you do not care about graceful shutdowning.
|
|
||||||
pub struct Noop;
|
|
||||||
|
|
||||||
impl StopToken for Noop {
|
|
||||||
fn stop(self) {}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// A stop token which corresponds to [`AsyncStopFlag`].
|
|
||||||
#[derive(Clone)]
|
|
||||||
pub struct AsyncStopToken(AbortHandle);
|
|
||||||
|
|
||||||
/// A flag which corresponds to [`AsyncStopToken`].
|
|
||||||
///
|
|
||||||
/// To know if the stop token was used you can either repeatedly call
|
|
||||||
/// [`is_stopped`] or use this type as a `Future`.
|
|
||||||
///
|
|
||||||
/// [`is_stopped`]: AsyncStopFlag::is_stopped
|
|
||||||
#[pin_project::pin_project]
|
|
||||||
#[derive(Clone)]
|
|
||||||
pub struct AsyncStopFlag(#[pin] Abortable<Pending<()>>);
|
|
||||||
|
|
||||||
impl AsyncStopToken {
|
|
||||||
/// Create a new token/flag pair.
|
|
||||||
#[must_use = "This function is pure, that is does nothing unless its output is used"]
|
|
||||||
pub fn new_pair() -> (Self, AsyncStopFlag) {
|
|
||||||
let (handle, reg) = AbortHandle::new_pair();
|
|
||||||
let token = Self(handle);
|
|
||||||
let flag = AsyncStopFlag(Abortable::new(pending(), reg));
|
|
||||||
|
|
||||||
(token, flag)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl StopToken for AsyncStopToken {
|
|
||||||
fn stop(self) {
|
|
||||||
self.0.abort()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl AsyncStopFlag {
|
|
||||||
/// Returns true if the stop token linked to `self` was used.
|
|
||||||
#[must_use = "This function is pure, that is does nothing unless its output is used"]
|
|
||||||
pub fn is_stopped(&self) -> bool {
|
|
||||||
self.0.is_aborted()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// This future resolves when a stop token was used.
|
|
||||||
impl Future for AsyncStopFlag {
|
|
||||||
type Output = ();
|
|
||||||
|
|
||||||
fn poll(self: Pin<&mut Self>, cx: &mut task::Context<'_>) -> task::Poll<Self::Output> {
|
|
||||||
self.project().0.poll(cx).map(|res| {
|
|
||||||
debug_assert!(
|
|
||||||
res.is_err(),
|
|
||||||
"Pending Future can't ever be resolved, so Abortable is only resolved when \
|
|
||||||
canceled"
|
|
||||||
);
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -35,7 +35,7 @@ use futures::Stream;
|
||||||
use std::time::Duration;
|
use std::time::Duration;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
dispatching::stop_token::StopToken,
|
stop::StopToken,
|
||||||
types::{AllowedUpdate, Update},
|
types::{AllowedUpdate, Update},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -59,9 +59,11 @@ pub use self::{
|
||||||
/// - [`AsUpdateStream::as_stream`]
|
/// - [`AsUpdateStream::as_stream`]
|
||||||
///
|
///
|
||||||
/// [module-level documentation]: mod@self
|
/// [module-level documentation]: mod@self
|
||||||
pub trait UpdateListener<E>: for<'a> AsUpdateStream<'a, E> {
|
pub trait UpdateListener:
|
||||||
/// The type of token which allows to stop this listener.
|
for<'a> AsUpdateStream<'a, StreamErr = <Self as UpdateListener>::Err>
|
||||||
type StopToken: StopToken + Send;
|
{
|
||||||
|
/// The type of errors that can be returned from this listener.
|
||||||
|
type Err;
|
||||||
|
|
||||||
/// Returns a token which stops this listener.
|
/// Returns a token which stops this listener.
|
||||||
///
|
///
|
||||||
|
@ -76,7 +78,7 @@ pub trait UpdateListener<E>: for<'a> AsUpdateStream<'a, E> {
|
||||||
/// soon as all cached updates are returned.
|
/// soon as all cached updates are returned.
|
||||||
#[must_use = "This function doesn't stop listening, to stop listening you need to call `stop` \
|
#[must_use = "This function doesn't stop listening, to stop listening you need to call `stop` \
|
||||||
on the returned token"]
|
on the returned token"]
|
||||||
fn stop_token(&mut self) -> Self::StopToken;
|
fn stop_token(&mut self) -> StopToken;
|
||||||
|
|
||||||
/// Hint which updates should the listener listen for.
|
/// Hint which updates should the listener listen for.
|
||||||
///
|
///
|
||||||
|
@ -110,16 +112,19 @@ pub trait UpdateListener<E>: for<'a> AsUpdateStream<'a, E> {
|
||||||
/// [`UpdateListener`]'s supertrait/extension.
|
/// [`UpdateListener`]'s supertrait/extension.
|
||||||
///
|
///
|
||||||
/// This trait is a workaround to not require GAT.
|
/// This trait is a workaround to not require GAT.
|
||||||
pub trait AsUpdateStream<'a, E> {
|
pub trait AsUpdateStream<'a> {
|
||||||
|
/// Error that can be returned from the [`Stream`]
|
||||||
|
///
|
||||||
|
/// [`Stream`]: AsUpdateStream::Stream
|
||||||
|
// NB: This should be named differently to `UpdateListener::Err`, so that it's
|
||||||
|
// unambiguous
|
||||||
|
type StreamErr;
|
||||||
|
|
||||||
/// The stream of updates from Telegram.
|
/// The stream of updates from Telegram.
|
||||||
// HACK: There is currently no way to write something like
|
// NB: `Send` is not strictly required here, but it makes it easier to return
|
||||||
// `-> impl for<'a> AsUpdateStream<'a, E, Stream: Send>`. Since we return
|
// `impl AsUpdateStream` and also you want `Send` streams almost (?) always
|
||||||
// `impl UpdateListener<E>` from `polling`, we need to have `Send` bound here,
|
// anyway.
|
||||||
// to make the stream `Send`.
|
type Stream: Stream<Item = Result<Update, Self::StreamErr>> + Send + 'a;
|
||||||
//
|
|
||||||
// Without this it's, for example, impossible to spawn a tokio task with
|
|
||||||
// teloxide polling.
|
|
||||||
type Stream: Stream<Item = Result<Update, E>> + Send + 'a;
|
|
||||||
|
|
||||||
/// Creates the update [`Stream`].
|
/// Creates the update [`Stream`].
|
||||||
///
|
///
|
||||||
|
@ -128,9 +133,9 @@ pub trait AsUpdateStream<'a, E> {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline(always)]
|
#[inline(always)]
|
||||||
pub(crate) fn assert_update_listener<L, E>(listener: L) -> L
|
pub(crate) fn assert_update_listener<L>(listener: L) -> L
|
||||||
where
|
where
|
||||||
L: UpdateListener<E>,
|
L: UpdateListener,
|
||||||
{
|
{
|
||||||
listener
|
listener
|
||||||
}
|
}
|
||||||
|
|
|
@ -13,11 +13,9 @@ use std::{
|
||||||
use futures::{ready, stream::Stream};
|
use futures::{ready, stream::Stream};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
dispatching::{
|
dispatching::update_listeners::{assert_update_listener, AsUpdateStream, UpdateListener},
|
||||||
stop_token::{AsyncStopFlag, AsyncStopToken},
|
|
||||||
update_listeners::{assert_update_listener, AsUpdateStream, UpdateListener},
|
|
||||||
},
|
|
||||||
requests::{HasPayload, Request, Requester},
|
requests::{HasPayload, Request, Requester},
|
||||||
|
stop::{mk_stop_token, StopFlag, StopToken},
|
||||||
types::{AllowedUpdate, Update},
|
types::{AllowedUpdate, Update},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -69,7 +67,7 @@ where
|
||||||
///
|
///
|
||||||
/// ## Note
|
/// ## Note
|
||||||
///
|
///
|
||||||
/// Teloxide normally (when using [`Dispatcher`] or [`repl`]s) sets this
|
/// `teloxide` normally (when using [`Dispatcher`] or [`repl`]s) sets this
|
||||||
/// automatically via [`hint_allowed_updates`], so you rarely need to use
|
/// automatically via [`hint_allowed_updates`], so you rarely need to use
|
||||||
/// `allowed_updates` explicitly.
|
/// `allowed_updates` explicitly.
|
||||||
///
|
///
|
||||||
|
@ -98,7 +96,7 @@ where
|
||||||
/// See also: [`polling_default`], [`Polling`].
|
/// See also: [`polling_default`], [`Polling`].
|
||||||
pub fn build(self) -> Polling<R> {
|
pub fn build(self) -> Polling<R> {
|
||||||
let Self { bot, timeout, limit, allowed_updates, drop_pending_updates } = self;
|
let Self { bot, timeout, limit, allowed_updates, drop_pending_updates } = self;
|
||||||
let (token, flag) = AsyncStopToken::new_pair();
|
let (token, flag) = mk_stop_token();
|
||||||
let polling =
|
let polling =
|
||||||
Polling { bot, timeout, limit, allowed_updates, drop_pending_updates, flag, token };
|
Polling { bot, timeout, limit, allowed_updates, drop_pending_updates, flag, token };
|
||||||
|
|
||||||
|
@ -242,8 +240,8 @@ pub struct Polling<B: Requester> {
|
||||||
limit: Option<u8>,
|
limit: Option<u8>,
|
||||||
allowed_updates: Option<Vec<AllowedUpdate>>,
|
allowed_updates: Option<Vec<AllowedUpdate>>,
|
||||||
drop_pending_updates: bool,
|
drop_pending_updates: bool,
|
||||||
flag: AsyncStopFlag,
|
flag: StopFlag,
|
||||||
token: AsyncStopToken,
|
token: StopToken,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<R> Polling<R>
|
impl<R> Polling<R>
|
||||||
|
@ -291,10 +289,10 @@ pub struct PollingStream<'a, B: Requester> {
|
||||||
in_flight: Option<<B::GetUpdates as Request>::Send>,
|
in_flight: Option<<B::GetUpdates as Request>::Send>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<B: Requester + Send + 'static> UpdateListener<B::Err> for Polling<B> {
|
impl<B: Requester + Send + 'static> UpdateListener for Polling<B> {
|
||||||
type StopToken = AsyncStopToken;
|
type Err = B::Err;
|
||||||
|
|
||||||
fn stop_token(&mut self) -> Self::StopToken {
|
fn stop_token(&mut self) -> StopToken {
|
||||||
self.token.clone()
|
self.token.clone()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -309,7 +307,8 @@ impl<B: Requester + Send + 'static> UpdateListener<B::Err> for Polling<B> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'a, B: Requester + Send + 'a> AsUpdateStream<'a, B::Err> for Polling<B> {
|
impl<'a, B: Requester + Send + 'a> AsUpdateStream<'a> for Polling<B> {
|
||||||
|
type StreamErr = B::Err;
|
||||||
type Stream = PollingStream<'a, B>;
|
type Stream = PollingStream<'a, B>;
|
||||||
|
|
||||||
fn as_stream(&'a mut self) -> Self::Stream {
|
fn as_stream(&'a mut self) -> Self::Stream {
|
||||||
|
|
|
@ -3,10 +3,8 @@ use std::time::Duration;
|
||||||
use futures::Stream;
|
use futures::Stream;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
dispatching::{
|
dispatching::update_listeners::{AsUpdateStream, UpdateListener},
|
||||||
stop_token::{self, StopToken},
|
stop::StopToken,
|
||||||
update_listeners::{AsUpdateStream, UpdateListener},
|
|
||||||
},
|
|
||||||
types::{AllowedUpdate, Update},
|
types::{AllowedUpdate, Update},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -30,7 +28,7 @@ pub struct StatefulListener<St, Assf, Sf, Hauf, Thf> {
|
||||||
|
|
||||||
/// The function used as [`UpdateListener::stop_token`].
|
/// The function used as [`UpdateListener::stop_token`].
|
||||||
///
|
///
|
||||||
/// Must implement `FnMut(&mut St) -> impl StopToken`.
|
/// Must implement `FnMut(&mut St) -> StopToken`.
|
||||||
pub stop_token: Sf,
|
pub stop_token: Sf,
|
||||||
|
|
||||||
/// The function used as [`UpdateListener::hint_allowed_updates`].
|
/// The function used as [`UpdateListener::hint_allowed_updates`].
|
||||||
|
@ -68,42 +66,7 @@ impl<St, Assf, Sf, Hauf, Thf> StatefulListener<St, Assf, Sf, Hauf, Thf> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<S, E>
|
impl<'a, St, Assf, Sf, Hauf, Thf, Strm, E> AsUpdateStream<'a>
|
||||||
StatefulListener<
|
|
||||||
S,
|
|
||||||
for<'a> fn(&'a mut S) -> &'a mut S,
|
|
||||||
for<'a> fn(&'a mut S) -> stop_token::Noop,
|
|
||||||
Haufn<S>,
|
|
||||||
Thfn<S>,
|
|
||||||
>
|
|
||||||
where
|
|
||||||
S: Stream<Item = Result<Update, E>> + Unpin + Send + 'static,
|
|
||||||
{
|
|
||||||
/// Creates a new update listener from a stream of updates which ignores
|
|
||||||
/// stop signals.
|
|
||||||
///
|
|
||||||
/// It won't be possible to ever stop this listener with a stop token.
|
|
||||||
pub fn from_stream_without_graceful_shutdown(stream: S) -> Self {
|
|
||||||
let this = Self::new_with_hints(
|
|
||||||
stream,
|
|
||||||
|s| s,
|
|
||||||
|_| stop_token::Noop,
|
|
||||||
None,
|
|
||||||
Some(|_| {
|
|
||||||
// FIXME: replace this by just Duration::MAX once 1.53 releases
|
|
||||||
// be released
|
|
||||||
const NANOS_PER_SEC: u32 = 1_000_000_000;
|
|
||||||
let dmax = Duration::new(u64::MAX, NANOS_PER_SEC - 1);
|
|
||||||
|
|
||||||
Some(dmax)
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_update_listener(this)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'a, St, Assf, Sf, Hauf, Thf, Strm, E> AsUpdateStream<'a, E>
|
|
||||||
for StatefulListener<St, Assf, Hauf, Sf, Thf>
|
for StatefulListener<St, Assf, Hauf, Sf, Thf>
|
||||||
where
|
where
|
||||||
(St, Strm): 'a,
|
(St, Strm): 'a,
|
||||||
|
@ -111,6 +74,7 @@ where
|
||||||
Assf: FnMut(&'a mut St) -> Strm,
|
Assf: FnMut(&'a mut St) -> Strm,
|
||||||
Strm: Stream<Item = Result<Update, E>>,
|
Strm: Stream<Item = Result<Update, E>>,
|
||||||
{
|
{
|
||||||
|
type StreamErr = E;
|
||||||
type Stream = Strm;
|
type Stream = Strm;
|
||||||
|
|
||||||
fn as_stream(&'a mut self) -> Self::Stream {
|
fn as_stream(&'a mut self) -> Self::Stream {
|
||||||
|
@ -118,18 +82,16 @@ where
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<St, Assf, Sf, Hauf, Stt, Thf, E> UpdateListener<E>
|
impl<St, Assf, Sf, Hauf, Thf, E> UpdateListener for StatefulListener<St, Assf, Sf, Hauf, Thf>
|
||||||
for StatefulListener<St, Assf, Sf, Hauf, Thf>
|
|
||||||
where
|
where
|
||||||
Self: for<'a> AsUpdateStream<'a, E>,
|
Self: for<'a> AsUpdateStream<'a, StreamErr = E>,
|
||||||
Sf: FnMut(&mut St) -> Stt,
|
Sf: FnMut(&mut St) -> 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>,
|
||||||
{
|
{
|
||||||
type StopToken = Stt;
|
type Err = E;
|
||||||
|
|
||||||
fn stop_token(&mut self) -> Stt {
|
fn stop_token(&mut self) -> StopToken {
|
||||||
(self.stop_token)(&mut self.state)
|
(self.stop_token)(&mut self.state)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -143,10 +105,3 @@ where
|
||||||
self.timeout_hint.as_ref().and_then(|f| f(&self.state))
|
self.timeout_hint.as_ref().and_then(|f| f(&self.state))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn assert_update_listener<L, E>(l: L) -> L
|
|
||||||
where
|
|
||||||
L: UpdateListener<E>,
|
|
||||||
{
|
|
||||||
l
|
|
||||||
}
|
|
||||||
|
|
|
@ -49,7 +49,7 @@ pub struct Options {
|
||||||
/// `a-z`, `0-9`, `_` and `-` are allowed. The header is useful to ensure
|
/// `a-z`, `0-9`, `_` and `-` are allowed. The header is useful to ensure
|
||||||
/// that the request comes from a webhook set by you.
|
/// that the request comes from a webhook set by you.
|
||||||
///
|
///
|
||||||
/// Default - teloxide will generate a random token.
|
/// Default - `teloxide` will generate a random token.
|
||||||
pub secret_token: Option<String>,
|
pub secret_token: Option<String>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -6,11 +6,9 @@ use axum::{
|
||||||
};
|
};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
dispatching::{
|
dispatching::update_listeners::{webhooks::Options, UpdateListener},
|
||||||
stop_token::{AsyncStopFlag, StopToken},
|
|
||||||
update_listeners::{webhooks::Options, UpdateListener},
|
|
||||||
},
|
|
||||||
requests::Requester,
|
requests::Requester,
|
||||||
|
stop::StopFlag,
|
||||||
};
|
};
|
||||||
|
|
||||||
/// Webhook implementation based on the [mod@axum] framework.
|
/// Webhook implementation based on the [mod@axum] framework.
|
||||||
|
@ -22,7 +20,7 @@ use crate::{
|
||||||
///
|
///
|
||||||
/// [`set_webhook`]: crate::payloads::SetWebhook
|
/// [`set_webhook`]: crate::payloads::SetWebhook
|
||||||
/// [`delete_webhook`]: crate::payloads::DeleteWebhook
|
/// [`delete_webhook`]: crate::payloads::DeleteWebhook
|
||||||
/// [`stop`]: StopToken::stop
|
/// [`stop`]: crate::stop::StopToken::stop
|
||||||
///
|
///
|
||||||
/// ## Panics
|
/// ## Panics
|
||||||
///
|
///
|
||||||
|
@ -38,7 +36,10 @@ use crate::{
|
||||||
///
|
///
|
||||||
/// [`axum_to_router`] and [`axum_no_setup`] for lower-level versions of this
|
/// [`axum_to_router`] and [`axum_no_setup`] for lower-level versions of this
|
||||||
/// function.
|
/// function.
|
||||||
pub async fn axum<R>(bot: R, options: Options) -> Result<impl UpdateListener<Infallible>, R::Err>
|
pub async fn axum<R>(
|
||||||
|
bot: R,
|
||||||
|
options: Options,
|
||||||
|
) -> Result<impl UpdateListener<Err = Infallible>, R::Err>
|
||||||
where
|
where
|
||||||
R: Requester + Send + 'static,
|
R: Requester + Send + 'static,
|
||||||
<R as Requester>::DeleteWebhook: Send,
|
<R as Requester>::DeleteWebhook: Send,
|
||||||
|
@ -85,7 +86,7 @@ where
|
||||||
///
|
///
|
||||||
/// [`set_webhook`]: crate::payloads::SetWebhook
|
/// [`set_webhook`]: crate::payloads::SetWebhook
|
||||||
/// [`delete_webhook`]: crate::payloads::DeleteWebhook
|
/// [`delete_webhook`]: crate::payloads::DeleteWebhook
|
||||||
/// [`stop`]: StopToken::stop
|
/// [`stop`]: crate::stop::StopToken::stop
|
||||||
/// [`options.address`]: Options::address
|
/// [`options.address`]: Options::address
|
||||||
/// [`with_graceful_shutdown`]: axum::Server::with_graceful_shutdown
|
/// [`with_graceful_shutdown`]: axum::Server::with_graceful_shutdown
|
||||||
///
|
///
|
||||||
|
@ -107,7 +108,10 @@ where
|
||||||
pub async fn axum_to_router<R>(
|
pub async fn axum_to_router<R>(
|
||||||
bot: R,
|
bot: R,
|
||||||
mut options: Options,
|
mut options: Options,
|
||||||
) -> Result<(impl UpdateListener<Infallible>, impl Future<Output = ()> + Send, axum::Router), R::Err>
|
) -> Result<
|
||||||
|
(impl UpdateListener<Err = Infallible>, impl Future<Output = ()> + Send, axum::Router),
|
||||||
|
R::Err,
|
||||||
|
>
|
||||||
where
|
where
|
||||||
R: Requester + Send,
|
R: Requester + Send,
|
||||||
<R as Requester>::DeleteWebhook: Send,
|
<R as Requester>::DeleteWebhook: Send,
|
||||||
|
@ -148,12 +152,10 @@ where
|
||||||
/// function.
|
/// function.
|
||||||
pub fn axum_no_setup(
|
pub fn axum_no_setup(
|
||||||
options: Options,
|
options: Options,
|
||||||
) -> (impl UpdateListener<Infallible>, impl Future<Output = ()>, axum::Router) {
|
) -> (impl UpdateListener<Err = Infallible>, impl Future<Output = ()>, axum::Router) {
|
||||||
use crate::{
|
use crate::{
|
||||||
dispatching::{
|
dispatching::update_listeners::{self, webhooks::tuple_first_mut},
|
||||||
stop_token::AsyncStopToken,
|
stop::{mk_stop_token, StopToken},
|
||||||
update_listeners::{self, webhooks::tuple_first_mut},
|
|
||||||
},
|
|
||||||
types::Update,
|
types::Update,
|
||||||
};
|
};
|
||||||
use axum::{extract::Extension, response::IntoResponse, routing::post};
|
use axum::{extract::Extension, response::IntoResponse, routing::post};
|
||||||
|
@ -172,7 +174,7 @@ pub fn axum_no_setup(
|
||||||
secret_header: XTelegramBotApiSecretToken,
|
secret_header: XTelegramBotApiSecretToken,
|
||||||
secret: Extension<Option<String>>,
|
secret: Extension<Option<String>>,
|
||||||
tx: Extension<CSender>,
|
tx: Extension<CSender>,
|
||||||
flag: Extension<AsyncStopFlag>,
|
flag: Extension<StopFlag>,
|
||||||
) -> impl IntoResponse {
|
) -> impl IntoResponse {
|
||||||
// FIXME: use constant time comparison here
|
// FIXME: use constant time comparison here
|
||||||
if secret_header.0.as_deref() != secret.as_deref().map(str::as_bytes) {
|
if secret_header.0.as_deref() != secret.as_deref().map(str::as_bytes) {
|
||||||
|
@ -208,7 +210,7 @@ pub fn axum_no_setup(
|
||||||
StatusCode::OK
|
StatusCode::OK
|
||||||
}
|
}
|
||||||
|
|
||||||
let (stop_token, stop_flag) = AsyncStopToken::new_pair();
|
let (stop_token, stop_flag) = mk_stop_token();
|
||||||
|
|
||||||
let app = axum::Router::new().route(options.url.path(), post(telegram_request)).layer(
|
let app = axum::Router::new().route(options.url.path(), post(telegram_request)).layer(
|
||||||
ServiceBuilder::new()
|
ServiceBuilder::new()
|
||||||
|
@ -225,7 +227,7 @@ pub fn axum_no_setup(
|
||||||
let listener = update_listeners::StatefulListener::new(
|
let listener = update_listeners::StatefulListener::new(
|
||||||
(stream, stop_token),
|
(stream, stop_token),
|
||||||
tuple_first_mut,
|
tuple_first_mut,
|
||||||
|state: &mut (_, AsyncStopToken)| state.1.clone(),
|
|state: &mut (_, StopToken)| state.1.clone(),
|
||||||
);
|
);
|
||||||
|
|
||||||
(listener, stop_flag, app)
|
(listener, stop_flag, app)
|
||||||
|
|
|
@ -6,13 +6,13 @@
|
||||||
| `webhooks-axum` | Enables webhook implementation based on axum framework |
|
| `webhooks-axum` | Enables webhook implementation based on axum framework |
|
||||||
| `macros` | Re-exports macros from [`teloxide-macros`]. |
|
| `macros` | Re-exports macros from [`teloxide-macros`]. |
|
||||||
| `ctrlc_handler` | Enables the [`DispatcherBuilder::enable_ctrlc_handler`] function (**enabled by default**). |
|
| `ctrlc_handler` | Enables the [`DispatcherBuilder::enable_ctrlc_handler`] function (**enabled by default**). |
|
||||||
| `auto-send` | Enables the [`AutoSend`](adaptors::AutoSend) bot adaptor (**enabled by default**). |
|
| `auto-send` | Enables the [`AutoSend`](adaptors::AutoSend) bot adaptor (**enabled by default; DEPRECATED**). |
|
||||||
| `throttle` | Enables the [`Throttle`](adaptors::Throttle) bot adaptor. |
|
| `throttle` | Enables the [`Throttle`](adaptors::Throttle) bot adaptor. |
|
||||||
| `cache-me` | Enables the [`CacheMe`](adaptors::CacheMe) bot adaptor. |
|
| `cache-me` | Enables the [`CacheMe`](adaptors::CacheMe) bot adaptor. |
|
||||||
| `trace-adaptor` | Enables the [`Trace`](adaptors::Trace) bot adaptor. |
|
| `trace-adaptor` | Enables the [`Trace`](adaptors::Trace) bot adaptor. |
|
||||||
| `erased` | Enables the [`ErasedRequester`](adaptors::ErasedRequester) bot adaptor. |
|
| `erased` | Enables the [`ErasedRequester`](adaptors::ErasedRequester) bot adaptor. |
|
||||||
| `full` | Enables all the features except `nightly`. |
|
| `full` | Enables all the features except `nightly`. |
|
||||||
| `nightly` | Enables nightly-only features (see the [teloxide-core features]). |
|
| `nightly` | Enables nightly-only features (see the [`teloxide-core` features]). |
|
||||||
| `native-tls` | Enables the [`native-tls`] TLS implementation (**enabled by default**). |
|
| `native-tls` | Enables the [`native-tls`] TLS implementation (**enabled by default**). |
|
||||||
| `rustls` | Enables the [`rustls`] TLS implementation. |
|
| `rustls` | Enables the [`rustls`] TLS implementation. |
|
||||||
| `redis-storage` | Enables the [Redis] storage support for dialogues. |
|
| `redis-storage` | Enables the [Redis] storage support for dialogues. |
|
||||||
|
@ -29,6 +29,6 @@
|
||||||
[`native-tls`]: https://docs.rs/native-tls
|
[`native-tls`]: https://docs.rs/native-tls
|
||||||
[`rustls`]: https://docs.rs/rustls
|
[`rustls`]: https://docs.rs/rustls
|
||||||
[`teloxide::utils::UpState`]: utils::UpState
|
[`teloxide::utils::UpState`]: utils::UpState
|
||||||
[teloxide-core features]: https://docs.rs/teloxide-core/latest/teloxide_core/#cargo-features
|
[`teloxide-core` features]: https://docs.rs/teloxide-core/latest/teloxide_core/#cargo-features
|
||||||
|
|
||||||
[`DispatcherBuilder::enable_ctrlc_handler`]: dispatching::DispatcherBuilder::enable_ctrlc_handler
|
[`DispatcherBuilder::enable_ctrlc_handler`]: dispatching::DispatcherBuilder::enable_ctrlc_handler
|
11
src/lib.rs
11
src/lib.rs
|
@ -13,11 +13,11 @@
|
||||||
//! pretty_env_logger::init();
|
//! pretty_env_logger::init();
|
||||||
//! log::info!("Starting throw dice bot...");
|
//! log::info!("Starting throw dice bot...");
|
||||||
//!
|
//!
|
||||||
//! let bot = Bot::from_env().auto_send();
|
//! let bot = Bot::from_env();
|
||||||
//!
|
//!
|
||||||
//! teloxide::repl(bot, |message: Message, bot: AutoSend<Bot>| async move {
|
//! teloxide::repl(bot, |bot: Bot, msg: Message| async move {
|
||||||
//! bot.send_dice(message.chat.id).await?;
|
//! bot.send_dice(msg.chat.id).await?;
|
||||||
//! respond(())
|
//! Ok(())
|
||||||
//! })
|
//! })
|
||||||
//! .await;
|
//! .await;
|
||||||
//! # }
|
//! # }
|
||||||
|
@ -62,11 +62,10 @@ pub use dispatching::repls::{
|
||||||
commands_repl, commands_repl_with_listener, repl, repl_with_listener,
|
commands_repl, commands_repl_with_listener, repl, repl_with_listener,
|
||||||
};
|
};
|
||||||
|
|
||||||
mod logging;
|
|
||||||
|
|
||||||
pub mod dispatching;
|
pub mod dispatching;
|
||||||
pub mod error_handlers;
|
pub mod error_handlers;
|
||||||
pub mod prelude;
|
pub mod prelude;
|
||||||
|
pub mod stop;
|
||||||
pub mod utils;
|
pub mod utils;
|
||||||
|
|
||||||
#[doc(inline)]
|
#[doc(inline)]
|
||||||
|
|
|
@ -1,53 +0,0 @@
|
||||||
/// Enables logging through [pretty-env-logger].
|
|
||||||
///
|
|
||||||
/// A logger will **only** print errors, warnings, and general information from
|
|
||||||
/// teloxide and **all** logs from your program.
|
|
||||||
///
|
|
||||||
/// # Note
|
|
||||||
///
|
|
||||||
/// Calling this macro **is not mandatory**; you can setup if your own logger if
|
|
||||||
/// you want.
|
|
||||||
///
|
|
||||||
/// [pretty-env-logger]: https://crates.io/crates/pretty_env_logger
|
|
||||||
#[macro_export]
|
|
||||||
#[deprecated = "Choose logging implementation yourself"]
|
|
||||||
macro_rules! enable_logging {
|
|
||||||
() => {
|
|
||||||
#[allow(deprecated)]
|
|
||||||
teloxide::enable_logging_with_filter!(log::LevelFilter::Trace);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Enables logging through [pretty-env-logger] with a custom filter for your
|
|
||||||
/// program.
|
|
||||||
///
|
|
||||||
/// A logger will **only** print errors, warnings, and general information from
|
|
||||||
/// teloxide and restrict logs from your program by the specified filter.
|
|
||||||
///
|
|
||||||
/// # Example
|
|
||||||
///
|
|
||||||
/// Allow printing all logs from your program up to [`LevelFilter::Debug`] (i.e.
|
|
||||||
/// do not print traces):
|
|
||||||
///
|
|
||||||
/// ```no_compile
|
|
||||||
/// teloxide::enable_logging_with_filter!(log::LevelFilter::Debug);
|
|
||||||
/// ```
|
|
||||||
///
|
|
||||||
/// # Note
|
|
||||||
///
|
|
||||||
/// Calling this macro **is not mandatory**; you can setup if your own logger if
|
|
||||||
/// you want.
|
|
||||||
///
|
|
||||||
/// [pretty-env-logger]: https://crates.io/crates/pretty_env_logger
|
|
||||||
/// [`LevelFilter::Debug`]: https://docs.rs/log/0.4.10/log/enum.LevelFilter.html
|
|
||||||
#[macro_export]
|
|
||||||
#[deprecated = "Choose logging implementation yourself"]
|
|
||||||
macro_rules! enable_logging_with_filter {
|
|
||||||
($filter:expr) => {
|
|
||||||
pretty_env_logger::formatted_builder()
|
|
||||||
.write_style(pretty_env_logger::env_logger::WriteStyle::Auto)
|
|
||||||
.filter(Some(&env!("CARGO_CRATE_NAME").replace("-", "_")), $filter)
|
|
||||||
.filter(Some("teloxide"), log::LevelFilter::Info)
|
|
||||||
.init();
|
|
||||||
};
|
|
||||||
}
|
|
|
@ -1,20 +1,24 @@
|
||||||
//! Commonly used items.
|
//! Commonly used items.
|
||||||
|
|
||||||
pub use crate::{
|
pub use crate::error_handlers::{LoggingErrorHandler, OnError};
|
||||||
error_handlers::{LoggingErrorHandler, OnError},
|
|
||||||
respond,
|
#[allow(deprecated)]
|
||||||
};
|
pub use crate::respond;
|
||||||
|
|
||||||
pub use crate::dispatching::{
|
pub use crate::dispatching::{
|
||||||
dialogue::Dialogue, Dispatcher, HandlerExt as _, MessageFilterExt as _, UpdateFilterExt as _,
|
dialogue::Dialogue, Dispatcher, HandlerExt as _, MessageFilterExt as _, UpdateFilterExt as _,
|
||||||
};
|
};
|
||||||
|
|
||||||
pub use teloxide_core::types::{
|
pub use teloxide_core::{
|
||||||
CallbackQuery, ChatMemberUpdated, ChosenInlineResult, InlineQuery, Message, Poll, PollAnswer,
|
requests::ResponseResult,
|
||||||
PreCheckoutQuery, ShippingQuery, Update,
|
types::{
|
||||||
|
CallbackQuery, ChatMemberUpdated, ChosenInlineResult, InlineQuery, Message, Poll,
|
||||||
|
PollAnswer, PreCheckoutQuery, ShippingQuery, Update,
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
#[cfg(feature = "auto-send")]
|
#[cfg(feature = "auto-send")]
|
||||||
|
#[allow(deprecated)]
|
||||||
pub use crate::adaptors::AutoSend;
|
pub use crate::adaptors::AutoSend;
|
||||||
|
|
||||||
#[doc(no_inline)]
|
#[doc(no_inline)]
|
||||||
|
|
64
src/stop.rs
Normal file
64
src/stop.rs
Normal file
|
@ -0,0 +1,64 @@
|
||||||
|
//! This module contains stop [token] and stop [flag] that are used to stop
|
||||||
|
//! async tasks, for example [listeners].
|
||||||
|
//!
|
||||||
|
//! [token]: StopToken
|
||||||
|
//! [flag]: StopFlag
|
||||||
|
//! [listeners]: crate::dispatching::update_listeners
|
||||||
|
|
||||||
|
use std::{convert::Infallible, future::Future, pin::Pin, task};
|
||||||
|
|
||||||
|
use futures::future::{pending, AbortHandle, Abortable, Pending};
|
||||||
|
|
||||||
|
/// Create a new token/flag pair.
|
||||||
|
#[must_use]
|
||||||
|
pub fn mk_stop_token() -> (StopToken, StopFlag) {
|
||||||
|
let (handle, reg) = AbortHandle::new_pair();
|
||||||
|
let token = StopToken(handle);
|
||||||
|
let flag = StopFlag(Abortable::new(pending(), reg));
|
||||||
|
|
||||||
|
(token, flag)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// A stop token which corresponds to a [`StopFlag`].
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct StopToken(AbortHandle);
|
||||||
|
|
||||||
|
/// A flag which corresponds to [`StopToken`].
|
||||||
|
///
|
||||||
|
/// To know if the stop token was used you can either repeatedly call
|
||||||
|
/// [`is_stopped`] or use this type as a `Future`.
|
||||||
|
///
|
||||||
|
/// [`is_stopped`]: StopFlag::is_stopped
|
||||||
|
#[pin_project::pin_project]
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct StopFlag(#[pin] Abortable<Pending<Infallible>>);
|
||||||
|
|
||||||
|
impl StopToken {
|
||||||
|
/// "Stops" the flag associated with this token.
|
||||||
|
///
|
||||||
|
/// Note that calling this function multiple times does nothing, only the
|
||||||
|
/// first call changes the state.
|
||||||
|
pub fn stop(&self) {
|
||||||
|
self.0.abort()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl StopFlag {
|
||||||
|
/// Returns true if the stop token linked to `self` was used.
|
||||||
|
#[must_use]
|
||||||
|
pub fn is_stopped(&self) -> bool {
|
||||||
|
self.0.is_aborted()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// This future resolves when a stop token was used.
|
||||||
|
impl Future for StopFlag {
|
||||||
|
type Output = ();
|
||||||
|
|
||||||
|
fn poll(self: Pin<&mut Self>, cx: &mut task::Context<'_>) -> task::Poll<Self::Output> {
|
||||||
|
self.project().0.poll(cx).map(|res| match res {
|
||||||
|
Err(_aborted) => (),
|
||||||
|
Ok(unreachable) => match unreachable {},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
|
@ -13,7 +13,7 @@
|
||||||
//! type UnitOfTime = u8;
|
//! type UnitOfTime = u8;
|
||||||
//!
|
//!
|
||||||
//! #[derive(BotCommands, PartialEq, Debug)]
|
//! #[derive(BotCommands, PartialEq, Debug)]
|
||||||
//! #[command(rename = "lowercase", parse_with = "split")]
|
//! #[command(rename_rule = "lowercase", parse_with = "split")]
|
||||||
//! enum AdminCommand {
|
//! enum AdminCommand {
|
||||||
//! Mute(UnitOfTime, char),
|
//! Mute(UnitOfTime, char),
|
||||||
//! Ban(UnitOfTime, char),
|
//! Ban(UnitOfTime, char),
|
||||||
|
@ -70,7 +70,7 @@ pub use teloxide_macros::BotCommands;
|
||||||
/// type UnitOfTime = u8;
|
/// type UnitOfTime = u8;
|
||||||
///
|
///
|
||||||
/// #[derive(BotCommands, PartialEq, Debug)]
|
/// #[derive(BotCommands, PartialEq, Debug)]
|
||||||
/// #[command(rename = "lowercase", parse_with = "split")]
|
/// #[command(rename_rule = "lowercase", parse_with = "split")]
|
||||||
/// enum AdminCommand {
|
/// enum AdminCommand {
|
||||||
/// Mute(UnitOfTime, char),
|
/// Mute(UnitOfTime, char),
|
||||||
/// Ban(UnitOfTime, char),
|
/// Ban(UnitOfTime, char),
|
||||||
|
@ -82,11 +82,10 @@ pub use teloxide_macros::BotCommands;
|
||||||
/// ```
|
/// ```
|
||||||
///
|
///
|
||||||
/// # Enum attributes
|
/// # Enum attributes
|
||||||
/// 1. `#[command(rename = "rule")]`
|
/// 1. `#[command(rename_rule = "rule")]`
|
||||||
/// Rename all commands by `rule`. If you will not use this attribute, commands
|
/// Rename all commands by `rule`. Allowed rules are `lowercase`, `UPPERCASE`,
|
||||||
/// will be parsed by their original names. Allowed rules are `lowercase`,
|
/// `PascalCase`, `camelCase`, `snake_case`, `SCREAMING_SNAKE_CASE`,
|
||||||
/// `UPPERCASE`, `PascalCase`, `camelCase`, `snake_case`,
|
/// `kebab-case`, and `SCREAMING-KEBAB-CASE`.
|
||||||
/// `SCREAMING_SNAKE_CASE`, `kebab-case`, and `SCREAMING-KEBAB-CASE`.
|
|
||||||
///
|
///
|
||||||
/// 2. `#[command(prefix = "prefix")]`
|
/// 2. `#[command(prefix = "prefix")]`
|
||||||
/// Change a prefix for all commands (the default is `/`).
|
/// Change a prefix for all commands (the default is `/`).
|
||||||
|
@ -106,7 +105,7 @@ pub use teloxide_macros::BotCommands;
|
||||||
/// use teloxide::utils::command::BotCommands;
|
/// use teloxide::utils::command::BotCommands;
|
||||||
///
|
///
|
||||||
/// #[derive(BotCommands, PartialEq, Debug)]
|
/// #[derive(BotCommands, PartialEq, Debug)]
|
||||||
/// #[command(rename = "lowercase")]
|
/// #[command(rename_rule = "lowercase")]
|
||||||
/// enum Command {
|
/// enum Command {
|
||||||
/// Text(String),
|
/// Text(String),
|
||||||
/// }
|
/// }
|
||||||
|
@ -126,7 +125,7 @@ pub use teloxide_macros::BotCommands;
|
||||||
/// use teloxide::utils::command::BotCommands;
|
/// use teloxide::utils::command::BotCommands;
|
||||||
///
|
///
|
||||||
/// #[derive(BotCommands, PartialEq, Debug)]
|
/// #[derive(BotCommands, PartialEq, Debug)]
|
||||||
/// #[command(rename = "lowercase", parse_with = "split")]
|
/// #[command(rename_rule = "lowercase", parse_with = "split")]
|
||||||
/// enum Command {
|
/// enum Command {
|
||||||
/// Nums(u8, u16, i32),
|
/// Nums(u8, u16, i32),
|
||||||
/// }
|
/// }
|
||||||
|
@ -146,7 +145,7 @@ pub use teloxide_macros::BotCommands;
|
||||||
/// use teloxide::utils::command::BotCommands;
|
/// use teloxide::utils::command::BotCommands;
|
||||||
///
|
///
|
||||||
/// #[derive(BotCommands, PartialEq, Debug)]
|
/// #[derive(BotCommands, PartialEq, Debug)]
|
||||||
/// #[command(rename = "lowercase", parse_with = "split", separator = "|")]
|
/// #[command(rename_rule = "lowercase", parse_with = "split", separator = "|")]
|
||||||
/// enum Command {
|
/// enum Command {
|
||||||
/// Nums(u8, u16, i32),
|
/// Nums(u8, u16, i32),
|
||||||
/// }
|
/// }
|
||||||
|
@ -159,21 +158,23 @@ pub use teloxide_macros::BotCommands;
|
||||||
/// # Variant attributes
|
/// # Variant attributes
|
||||||
/// All variant attributes override the corresponding `enum` attributes.
|
/// All variant attributes override the corresponding `enum` attributes.
|
||||||
///
|
///
|
||||||
/// 1. `#[command(rename = "rule")]`
|
/// 1. `#[command(rename_rule = "rule")]`
|
||||||
/// Rename one command by a rule. Allowed rules are `lowercase`, `UPPERCASE`,
|
/// Rename one command by a rule. Allowed rules are `lowercase`, `UPPERCASE`,
|
||||||
/// `PascalCase`, `camelCase`, `snake_case`, `SCREAMING_SNAKE_CASE`,
|
/// `PascalCase`, `camelCase`, `snake_case`, `SCREAMING_SNAKE_CASE`,
|
||||||
/// `kebab-case`, `SCREAMING-KEBAB-CASE`, and `%some_name%`, where `%some_name%`
|
/// `kebab-case`, `SCREAMING-KEBAB-CASE`.
|
||||||
/// is any string, a new name.
|
|
||||||
///
|
///
|
||||||
/// 2. `#[command(description = "description")]`
|
/// 2. `#[command(rename = "name")]`
|
||||||
|
/// Rename one command to `name` (literal renaming; do not confuse with
|
||||||
|
/// `rename_rule`).
|
||||||
|
///
|
||||||
|
/// 3. `#[command(description = "description")]`
|
||||||
/// Give your command a description. Write `"off"` for `"description"` to hide a
|
/// Give your command a description. Write `"off"` for `"description"` to hide a
|
||||||
/// command.
|
/// command.
|
||||||
///
|
///
|
||||||
/// 3. `#[command(parse_with = "parser")]`
|
/// 4. `#[command(parse_with = "parser")]`
|
||||||
/// One more option is available for variants.
|
/// Parse arguments of one command with a given parser. `parser` must be a
|
||||||
/// - `custom_parser` - your own parser of the signature `fn(String) ->
|
/// function of the signature `fn(String) -> Result<Tuple, ParseError>`, where
|
||||||
/// Result<Tuple, ParseError>`, where `Tuple` corresponds to the variant's
|
/// `Tuple` corresponds to the variant's arguments.
|
||||||
/// arguments.
|
|
||||||
///
|
///
|
||||||
/// ## Example
|
/// ## Example
|
||||||
/// ```
|
/// ```
|
||||||
|
@ -191,9 +192,9 @@ pub use teloxide_macros::BotCommands;
|
||||||
/// }
|
/// }
|
||||||
///
|
///
|
||||||
/// #[derive(BotCommands, PartialEq, Debug)]
|
/// #[derive(BotCommands, PartialEq, Debug)]
|
||||||
/// #[command(rename = "lowercase")]
|
/// #[command(rename_rule = "lowercase")]
|
||||||
/// enum Command {
|
/// enum Command {
|
||||||
/// #[command(parse_with = "accept_two_digits")]
|
/// #[command(parse_with = accept_two_digits)]
|
||||||
/// Num(u8),
|
/// Num(u8),
|
||||||
/// }
|
/// }
|
||||||
///
|
///
|
||||||
|
@ -204,8 +205,8 @@ pub use teloxide_macros::BotCommands;
|
||||||
/// # }
|
/// # }
|
||||||
/// ```
|
/// ```
|
||||||
///
|
///
|
||||||
/// 4. `#[command(prefix = "prefix")]`
|
/// 5. `#[command(prefix = "prefix")]`
|
||||||
/// 5. `#[command(separator = "sep")]`
|
/// 6. `#[command(separator = "sep")]`
|
||||||
///
|
///
|
||||||
/// These attributes just override the corresponding `enum` attributes for a
|
/// These attributes just override the corresponding `enum` attributes for a
|
||||||
/// specific variant.
|
/// specific variant.
|
||||||
|
@ -217,9 +218,7 @@ pub trait BotCommands: Sized {
|
||||||
///
|
///
|
||||||
/// `bot_username` is required to parse commands like
|
/// `bot_username` is required to parse commands like
|
||||||
/// `/cmd@username_of_the_bot`.
|
/// `/cmd@username_of_the_bot`.
|
||||||
fn parse<N>(s: &str, bot_username: N) -> Result<Self, ParseError>
|
fn parse(s: &str, bot_username: &str) -> Result<Self, ParseError>;
|
||||||
where
|
|
||||||
N: Into<String>;
|
|
||||||
|
|
||||||
/// Returns descriptions of the commands suitable to be shown to the user
|
/// Returns descriptions of the commands suitable to be shown to the user
|
||||||
/// (for example when `/help` command is used).
|
/// (for example when `/help` command is used).
|
||||||
|
@ -235,6 +234,7 @@ pub trait BotCommands: Sized {
|
||||||
/// Returns `PhantomData<Self>` that is used as a param of [`commands_repl`]
|
/// Returns `PhantomData<Self>` that is used as a param of [`commands_repl`]
|
||||||
///
|
///
|
||||||
/// [`commands_repl`]: (crate::repls2::commands_repl)
|
/// [`commands_repl`]: (crate::repls2::commands_repl)
|
||||||
|
#[must_use]
|
||||||
fn ty() -> PhantomData<Self> {
|
fn ty() -> PhantomData<Self> {
|
||||||
PhantomData
|
PhantomData
|
||||||
}
|
}
|
||||||
|
@ -296,11 +296,13 @@ pub struct CommandDescription<'a> {
|
||||||
|
|
||||||
impl<'a> CommandDescriptions<'a> {
|
impl<'a> CommandDescriptions<'a> {
|
||||||
/// Creates new [`CommandDescriptions`] from a list of command descriptions.
|
/// Creates new [`CommandDescriptions`] from a list of command descriptions.
|
||||||
|
#[must_use]
|
||||||
pub fn new(descriptions: &'a [CommandDescription<'a>]) -> Self {
|
pub fn new(descriptions: &'a [CommandDescription<'a>]) -> Self {
|
||||||
Self { global_description: None, descriptions, bot_username: None }
|
Self { global_description: None, descriptions, bot_username: None }
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Sets the global description of these commands.
|
/// Sets the global description of these commands.
|
||||||
|
#[must_use]
|
||||||
pub fn global_description(self, global_description: &'a str) -> Self {
|
pub fn global_description(self, global_description: &'a str) -> Self {
|
||||||
Self { global_description: Some(global_description), ..self }
|
Self { global_description: Some(global_description), ..self }
|
||||||
}
|
}
|
||||||
|
@ -328,6 +330,7 @@ impl<'a> CommandDescriptions<'a> {
|
||||||
/// message"
|
/// message"
|
||||||
/// );
|
/// );
|
||||||
/// ```
|
/// ```
|
||||||
|
#[must_use]
|
||||||
pub fn username(self, bot_username: &'a str) -> Self {
|
pub fn username(self, bot_username: &'a str) -> Self {
|
||||||
Self { bot_username: Some(bot_username), ..self }
|
Self { bot_username: Some(bot_username), ..self }
|
||||||
}
|
}
|
||||||
|
@ -338,6 +341,7 @@ impl<'a> CommandDescriptions<'a> {
|
||||||
/// method to get the username.
|
/// method to get the username.
|
||||||
///
|
///
|
||||||
/// [`username`]: self::CommandDescriptions::username
|
/// [`username`]: self::CommandDescriptions::username
|
||||||
|
#[must_use]
|
||||||
pub fn username_from_me(self, me: &'a Me) -> CommandDescriptions<'a> {
|
pub fn username_from_me(self, me: &'a Me) -> CommandDescriptions<'a> {
|
||||||
self.username(me.user.username.as_deref().expect("Bots must have usernames"))
|
self.username(me.user.username.as_deref().expect("Bots must have usernames"))
|
||||||
}
|
}
|
||||||
|
|
|
@ -93,7 +93,7 @@ impl fmt::Display for IdleShutdownError {
|
||||||
|
|
||||||
impl std::error::Error for IdleShutdownError {}
|
impl std::error::Error for IdleShutdownError {}
|
||||||
|
|
||||||
pub(crate) fn shutdown_check_timeout_for<E>(update_listener: &impl UpdateListener<E>) -> Duration {
|
pub(crate) fn shutdown_check_timeout_for(update_listener: &impl UpdateListener) -> Duration {
|
||||||
const MIN_SHUTDOWN_CHECK_TIMEOUT: Duration = Duration::from_secs(1);
|
const MIN_SHUTDOWN_CHECK_TIMEOUT: Duration = Duration::from_secs(1);
|
||||||
const DZERO: Duration = Duration::ZERO;
|
const DZERO: Duration = Duration::ZERO;
|
||||||
|
|
||||||
|
|
|
@ -11,7 +11,7 @@ use teloxide::utils::command::BotCommands;
|
||||||
#[cfg(feature = "macros")]
|
#[cfg(feature = "macros")]
|
||||||
fn parse_command_with_args() {
|
fn parse_command_with_args() {
|
||||||
#[derive(BotCommands, Debug, PartialEq)]
|
#[derive(BotCommands, Debug, PartialEq)]
|
||||||
#[command(rename = "lowercase")]
|
#[command(rename_rule = "lowercase")]
|
||||||
enum DefaultCommands {
|
enum DefaultCommands {
|
||||||
Start(String),
|
Start(String),
|
||||||
Help,
|
Help,
|
||||||
|
@ -27,7 +27,7 @@ fn parse_command_with_args() {
|
||||||
#[cfg(feature = "macros")]
|
#[cfg(feature = "macros")]
|
||||||
fn parse_command_with_non_string_arg() {
|
fn parse_command_with_non_string_arg() {
|
||||||
#[derive(BotCommands, Debug, PartialEq)]
|
#[derive(BotCommands, Debug, PartialEq)]
|
||||||
#[command(rename = "lowercase")]
|
#[command(rename_rule = "lowercase")]
|
||||||
enum DefaultCommands {
|
enum DefaultCommands {
|
||||||
Start(i32),
|
Start(i32),
|
||||||
Help,
|
Help,
|
||||||
|
@ -43,7 +43,7 @@ fn parse_command_with_non_string_arg() {
|
||||||
#[cfg(feature = "macros")]
|
#[cfg(feature = "macros")]
|
||||||
fn attribute_prefix() {
|
fn attribute_prefix() {
|
||||||
#[derive(BotCommands, Debug, PartialEq)]
|
#[derive(BotCommands, Debug, PartialEq)]
|
||||||
#[command(rename = "lowercase")]
|
#[command(rename_rule = "lowercase")]
|
||||||
enum DefaultCommands {
|
enum DefaultCommands {
|
||||||
#[command(prefix = "!")]
|
#[command(prefix = "!")]
|
||||||
Start(String),
|
Start(String),
|
||||||
|
@ -60,7 +60,7 @@ fn attribute_prefix() {
|
||||||
#[cfg(feature = "macros")]
|
#[cfg(feature = "macros")]
|
||||||
fn many_attributes() {
|
fn many_attributes() {
|
||||||
#[derive(BotCommands, Debug, PartialEq)]
|
#[derive(BotCommands, Debug, PartialEq)]
|
||||||
#[command(rename = "lowercase")]
|
#[command(rename_rule = "lowercase")]
|
||||||
enum DefaultCommands {
|
enum DefaultCommands {
|
||||||
#[command(prefix = "!", description = "desc")]
|
#[command(prefix = "!", description = "desc")]
|
||||||
Start,
|
Start,
|
||||||
|
@ -75,7 +75,7 @@ fn many_attributes() {
|
||||||
#[cfg(feature = "macros")]
|
#[cfg(feature = "macros")]
|
||||||
fn global_attributes() {
|
fn global_attributes() {
|
||||||
#[derive(BotCommands, Debug, PartialEq)]
|
#[derive(BotCommands, Debug, PartialEq)]
|
||||||
#[command(prefix = "!", rename = "lowercase", description = "Bot commands")]
|
#[command(prefix = "!", rename_rule = "lowercase", description = "Bot commands")]
|
||||||
enum DefaultCommands {
|
enum DefaultCommands {
|
||||||
#[command(prefix = "/")]
|
#[command(prefix = "/")]
|
||||||
Start,
|
Start,
|
||||||
|
@ -91,7 +91,7 @@ fn global_attributes() {
|
||||||
#[cfg(feature = "macros")]
|
#[cfg(feature = "macros")]
|
||||||
fn parse_command_with_bot_name() {
|
fn parse_command_with_bot_name() {
|
||||||
#[derive(BotCommands, Debug, PartialEq)]
|
#[derive(BotCommands, Debug, PartialEq)]
|
||||||
#[command(rename = "lowercase")]
|
#[command(rename_rule = "lowercase")]
|
||||||
enum DefaultCommands {
|
enum DefaultCommands {
|
||||||
#[command(prefix = "/")]
|
#[command(prefix = "/")]
|
||||||
Start,
|
Start,
|
||||||
|
@ -108,7 +108,7 @@ fn parse_command_with_bot_name() {
|
||||||
#[cfg(feature = "macros")]
|
#[cfg(feature = "macros")]
|
||||||
fn parse_with_split() {
|
fn parse_with_split() {
|
||||||
#[derive(BotCommands, Debug, PartialEq)]
|
#[derive(BotCommands, Debug, PartialEq)]
|
||||||
#[command(rename = "lowercase")]
|
#[command(rename_rule = "lowercase")]
|
||||||
#[command(parse_with = "split")]
|
#[command(parse_with = "split")]
|
||||||
enum DefaultCommands {
|
enum DefaultCommands {
|
||||||
Start(u8, String),
|
Start(u8, String),
|
||||||
|
@ -125,7 +125,7 @@ fn parse_with_split() {
|
||||||
#[cfg(feature = "macros")]
|
#[cfg(feature = "macros")]
|
||||||
fn parse_with_split2() {
|
fn parse_with_split2() {
|
||||||
#[derive(BotCommands, Debug, PartialEq)]
|
#[derive(BotCommands, Debug, PartialEq)]
|
||||||
#[command(rename = "lowercase")]
|
#[command(rename_rule = "lowercase")]
|
||||||
#[command(parse_with = "split", separator = "|")]
|
#[command(parse_with = "split", separator = "|")]
|
||||||
enum DefaultCommands {
|
enum DefaultCommands {
|
||||||
Start(u8, String),
|
Start(u8, String),
|
||||||
|
@ -159,13 +159,13 @@ fn parse_custom_parser() {
|
||||||
use parser::custom_parse_function;
|
use parser::custom_parse_function;
|
||||||
|
|
||||||
#[derive(BotCommands, Debug, PartialEq)]
|
#[derive(BotCommands, Debug, PartialEq)]
|
||||||
#[command(rename = "lowercase")]
|
#[command(rename_rule = "lowercase")]
|
||||||
enum DefaultCommands {
|
enum DefaultCommands {
|
||||||
#[command(parse_with = "custom_parse_function")]
|
#[command(parse_with = custom_parse_function)]
|
||||||
Start(u8, String),
|
Start(u8, String),
|
||||||
|
|
||||||
// Test <https://github.com/teloxide/teloxide/issues/668>.
|
// Test <https://github.com/teloxide/teloxide/issues/668>.
|
||||||
#[command(parse_with = "parser::custom_parse_function")]
|
#[command(parse_with = parser::custom_parse_function)]
|
||||||
TestPath(u8, String),
|
TestPath(u8, String),
|
||||||
|
|
||||||
Help,
|
Help,
|
||||||
|
@ -185,7 +185,7 @@ fn parse_custom_parser() {
|
||||||
#[cfg(feature = "macros")]
|
#[cfg(feature = "macros")]
|
||||||
fn parse_named_fields() {
|
fn parse_named_fields() {
|
||||||
#[derive(BotCommands, Debug, PartialEq)]
|
#[derive(BotCommands, Debug, PartialEq)]
|
||||||
#[command(rename = "lowercase")]
|
#[command(rename_rule = "lowercase")]
|
||||||
#[command(parse_with = "split")]
|
#[command(parse_with = "split")]
|
||||||
enum DefaultCommands {
|
enum DefaultCommands {
|
||||||
Start { num: u8, data: String },
|
Start { num: u8, data: String },
|
||||||
|
@ -202,7 +202,7 @@ fn parse_named_fields() {
|
||||||
#[cfg(feature = "macros")]
|
#[cfg(feature = "macros")]
|
||||||
fn descriptions_off() {
|
fn descriptions_off() {
|
||||||
#[derive(BotCommands, Debug, PartialEq)]
|
#[derive(BotCommands, Debug, PartialEq)]
|
||||||
#[command(rename = "lowercase")]
|
#[command(rename_rule = "lowercase")]
|
||||||
enum DefaultCommands {
|
enum DefaultCommands {
|
||||||
#[command(description = "off")]
|
#[command(description = "off")]
|
||||||
Start,
|
Start,
|
||||||
|
@ -217,21 +217,21 @@ fn descriptions_off() {
|
||||||
fn rename_rules() {
|
fn rename_rules() {
|
||||||
#[derive(BotCommands, Debug, PartialEq)]
|
#[derive(BotCommands, Debug, PartialEq)]
|
||||||
enum DefaultCommands {
|
enum DefaultCommands {
|
||||||
#[command(rename = "lowercase")]
|
#[command(rename_rule = "lowercase")]
|
||||||
AaaAaa,
|
AaaAaa,
|
||||||
#[command(rename = "UPPERCASE")]
|
#[command(rename_rule = "UPPERCASE")]
|
||||||
BbbBbb,
|
BbbBbb,
|
||||||
#[command(rename = "PascalCase")]
|
#[command(rename_rule = "PascalCase")]
|
||||||
CccCcc,
|
CccCcc,
|
||||||
#[command(rename = "camelCase")]
|
#[command(rename_rule = "camelCase")]
|
||||||
DddDdd,
|
DddDdd,
|
||||||
#[command(rename = "snake_case")]
|
#[command(rename_rule = "snake_case")]
|
||||||
EeeEee,
|
EeeEee,
|
||||||
#[command(rename = "SCREAMING_SNAKE_CASE")]
|
#[command(rename_rule = "SCREAMING_SNAKE_CASE")]
|
||||||
FffFff,
|
FffFff,
|
||||||
#[command(rename = "kebab-case")]
|
#[command(rename_rule = "kebab-case")]
|
||||||
GggGgg,
|
GggGgg,
|
||||||
#[command(rename = "SCREAMING-KEBAB-CASE")]
|
#[command(rename_rule = "SCREAMING-KEBAB-CASE")]
|
||||||
HhhHhh,
|
HhhHhh,
|
||||||
#[command(rename = "Bar")]
|
#[command(rename = "Bar")]
|
||||||
Foo,
|
Foo,
|
||||||
|
|
|
@ -1,64 +0,0 @@
|
||||||
#![allow(deprecated)]
|
|
||||||
|
|
||||||
#[cfg(feature = "macros")]
|
|
||||||
use teloxide::macros::DialogueState;
|
|
||||||
// We put tests here because macro expand in unit tests in the crate was a
|
|
||||||
// failure
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
#[cfg(feature = "macros")]
|
|
||||||
fn compile_test() {
|
|
||||||
#[allow(dead_code)]
|
|
||||||
#[derive(DialogueState, Clone)]
|
|
||||||
#[handler_out(Result<(), teloxide::RequestError>)]
|
|
||||||
enum State {
|
|
||||||
#[handler(handle_start)]
|
|
||||||
Start,
|
|
||||||
|
|
||||||
#[handler(handle_have_data)]
|
|
||||||
HaveData(String),
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Default for State {
|
|
||||||
fn default() -> Self {
|
|
||||||
Self::Start
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn handle_start() -> Result<(), teloxide::RequestError> {
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn handle_have_data() -> Result<(), teloxide::RequestError> {
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
#[cfg(feature = "macros")]
|
|
||||||
fn compile_test_generics() {
|
|
||||||
#[allow(dead_code)]
|
|
||||||
#[derive(DialogueState, Clone)]
|
|
||||||
#[handler_out(Result<(), teloxide::RequestError>)]
|
|
||||||
enum State<X: Clone + Send + Sync + 'static> {
|
|
||||||
#[handler(handle_start)]
|
|
||||||
Start,
|
|
||||||
|
|
||||||
#[handler(handle_have_data)]
|
|
||||||
HaveData(X),
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<X: Clone + Send + Sync + 'static> Default for State<X> {
|
|
||||||
fn default() -> Self {
|
|
||||||
Self::Start
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn handle_start() -> Result<(), teloxide::RequestError> {
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn handle_have_data() -> Result<(), teloxide::RequestError> {
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
}
|
|
Loading…
Reference in a new issue