teloxide/README.md

300 lines
11 KiB
Markdown
Raw Normal View History

2019-09-02 09:28:56 +02:00
<div align="center">
2020-01-06 11:25:37 +01:00
<img src="ICON.png" width="250"/>
2019-12-07 20:30:15 +01:00
<h1>teloxide</h1>
2019-12-07 20:30:15 +01:00
<a href="https://docs.rs/teloxide/">
2020-01-06 09:27:05 +01:00
<img src="https://img.shields.io/badge/docs.rs-v0.1.0-blue.svg">
2019-10-14 19:07:58 +02:00
</a>
2019-12-08 07:57:53 +01:00
<a href="https://github.com/teloxide/teloxide/actions">
<img src="https://github.com/teloxide/teloxide/workflows/Continuous%20integration/badge.svg">
2019-09-02 09:28:56 +02:00
</a>
2019-12-07 20:30:15 +01:00
<a href="https://crates.io/crates/teloxide">
2019-10-14 19:04:55 +02:00
<img src="https://img.shields.io/badge/crates.io-v0.1.0-orange.svg">
</a>
2019-10-14 19:10:41 +02:00
A full-featured framework that empowers you to easily build [Telegram bots](https://telegram.org/blog/bot-revolution) using the [`async`/`.await`](https://rust-lang.github.io/async-book/01_getting_started/01_chapter.html) syntax in [Rust](https://www.rust-lang.org/). It handles all the difficult stuff so you can focus only on your business logic.
2019-09-02 09:28:56 +02:00
</div>
2020-01-26 22:36:36 +01:00
2020-02-13 21:19:08 +01:00
## Features
2020-02-18 23:54:41 +01:00
- **Type-safe.** teloxide leverages the Rust's type system with two serious implications: resistance to human mistakes and tight integration with IDEs. Write fast, avoid debugging as much as possible.
- **Flexible API.** teloxide gives you the power of [streams](https://docs.rs/futures/0.3.4/futures/stream/index.html): you can combine [all 30+ patterns](https://docs.rs/futures/0.3.4/futures/stream/trait.StreamExt.html) when working with updates from Telegram.
2020-02-13 23:47:53 +01:00
2020-02-13 23:47:38 +01:00
- **Persistency.** By default, teloxide stores all user dialogues in RAM, but you can store them somewhere else (for example, in DB) just by implementing 2 functions.
2020-02-13 23:58:36 +01:00
- **Convenient dialogues system.** Define a type-safe [finite automaton](https://en.wikipedia.org/wiki/Finite-state_machine)
2020-02-18 23:54:41 +01:00
and transition functions to drive a user dialogue with ease (see [the guess-a-number example](#guess-a-number) below).
2020-02-13 23:52:44 +01:00
2020-02-12 11:17:20 +01:00
## Getting started
2020-02-12 11:20:15 +01:00
1. Create a new bot using [@Botfather](https://t.me/botfather) to get a token in the format `123456789:blablabla`.
2020-02-12 11:47:51 +01:00
2. Initialise the `TELOXIDE_TOKEN` environmental variable to your token:
```bash
2020-02-14 11:28:35 +01:00
# Unix
2020-02-12 11:49:59 +01:00
$ export TELOXIDE_TOKEN=MyAwesomeToken
2020-02-14 11:28:35 +01:00
# Windows
$ set TELOXITE_TOKEN=MyAwesomeToken
2020-02-12 11:47:51 +01:00
```
2020-02-12 16:37:57 +01:00
3. Be sure that you are up to date:
2020-02-12 11:09:53 +01:00
```bash
$ rustup update stable
2020-01-26 22:36:36 +01:00
```
2020-02-12 11:09:53 +01:00
2020-02-12 16:37:57 +01:00
4. Execute `cargo new my_bot`, enter the directory and put these lines into your `Cargo.toml`:
2020-01-26 22:36:36 +01:00
```toml
2020-02-12 11:09:53 +01:00
[dependencies]
2020-01-26 22:36:36 +01:00
teloxide = "0.1.0"
2020-02-12 17:00:54 +01:00
log = "0.4.8"
2020-02-18 23:54:41 +01:00
futures = "0.3.4"
2020-02-12 18:21:29 +01:00
tokio = "0.2.11"
2020-02-13 18:08:50 +01:00
pretty_env_logger = "0.4.0"
2020-01-26 22:36:36 +01:00
```
2020-02-12 11:35:51 +01:00
## The ping-pong bot
2020-02-13 23:00:37 +01:00
This bot has a single message handler, which answers "pong" to each incoming message:
2020-02-13 18:07:28 +01:00
2020-02-14 13:28:52 +01:00
([Full](https://github.com/teloxide/teloxide/blob/master/examples/ping_pong_bot/src/main.rs))
2020-01-26 22:36:36 +01:00
```rust
2020-02-12 11:17:20 +01:00
use teloxide::prelude::*;
2020-01-26 22:36:36 +01:00
#[tokio::main]
async fn main() {
2020-02-13 18:07:28 +01:00
teloxide::enable_logging!();
2020-02-18 23:54:41 +01:00
log::info!("Starting ping_pong_bot!");
2020-02-12 11:17:20 +01:00
2020-02-13 18:07:28 +01:00
let bot = Bot::from_env();
2020-02-18 23:54:41 +01:00
Dispatcher::new(bot)
.messages_handler(|rx: DispatcherHandlerRx<Message>| {
rx.for_each(|message| async move {
message.answer("pong").send().await.log_on_error().await;
})
2020-02-12 11:17:20 +01:00
})
.dispatch()
.await;
2020-01-26 22:36:36 +01:00
}
2020-02-18 23:54:41 +01:00
2020-01-26 22:36:36 +01:00
```
2020-02-13 21:19:08 +01:00
2020-02-15 14:34:06 +01:00
<details>
2020-02-18 23:54:41 +01:00
<summary>Click here to run it!</summary>
2020-02-15 14:34:06 +01:00
```bash
git clone https://github.com/teloxide/teloxide.git
cd teloxide/examples/ping_pong_bot
TELOXIDE_TOKEN=MyAwesomeToken cargo run
```
</details>
2020-02-14 13:17:57 +01:00
<div align="center">
2020-02-14 13:54:52 +01:00
<img src=https://github.com/teloxide/teloxide/raw/master/media/PING_PONG_BOT.png width="400" />
2020-02-14 13:17:57 +01:00
</div>
2020-02-13 22:49:49 +01:00
## Commands
2020-02-13 23:40:17 +01:00
Commands are defined similar to how we define CLI using [structopt](https://docs.rs/structopt/0.3.9/structopt/). This bot says "I am a cat! Meow!" on `/meow`, generates a random number within [0; 1) on `/generate`, and shows the usage guide on `/help`:
2020-02-13 22:49:49 +01:00
2020-02-14 13:28:52 +01:00
([Full](https://github.com/teloxide/teloxide/blob/master/examples/simple_commands_bot/src/main.rs))
2020-02-13 22:49:49 +01:00
```rust
// Imports are omitted...
#[derive(BotCommand)]
#[command(rename = "lowercase", description = "These commands are supported:")]
enum Command {
#[command(description = "display this text.")]
Help,
#[command(description = "be a cat.")]
Meow,
2020-02-13 23:36:57 +01:00
#[command(description = "generate a random number within [0; 1).")]
Generate,
2020-02-13 22:49:49 +01:00
}
2020-02-18 23:54:41 +01:00
fn generate() -> String {
thread_rng().gen_range(0.0, 1.0).to_string()
}
2020-02-13 22:49:49 +01:00
2020-02-18 23:54:41 +01:00
async fn answer(
cx: DispatcherHandlerCx<Message>,
command: Command,
) -> ResponseResult<()> {
2020-02-13 22:49:49 +01:00
match command {
2020-02-18 23:54:41 +01:00
Command::Help => cx.answer(Command::descriptions()).send().await?,
Command::Generate => cx.answer(generate()).send().await?,
Command::Meow => cx.answer("I am a cat! Meow!").send().await?,
2020-02-13 22:49:49 +01:00
};
Ok(())
}
2020-02-18 23:54:41 +01:00
async fn handle_command(rx: DispatcherHandlerRx<Message>) {
rx.filter_map(|cx| {
future::ready(cx.update.text_owned().map(|text| (cx, text)))
})
.filter_map(|(cx, text)| {
future::ready(Command::parse(&text).map(|(command, _)| (cx, command)))
})
.for_each_concurrent(None, |(cx, command)| async move {
answer(cx, command).await.log_on_error().await;
})
.await;
}
2020-02-13 22:49:49 +01:00
#[tokio::main]
async fn main() {
// Setup is omitted...
}
```
2020-02-15 14:34:06 +01:00
<details>
2020-02-18 23:54:41 +01:00
<summary>Click here to run it!</summary>
2020-02-15 14:34:06 +01:00
```bash
git clone https://github.com/teloxide/teloxide.git
cd teloxide/examples/simple_commands_bot
TELOXIDE_TOKEN=MyAwesomeToken cargo run
```
</details>
2020-02-14 13:17:57 +01:00
<div align="center">
2020-02-14 13:19:39 +01:00
<img src=https://github.com/teloxide/teloxide/raw/master/media/SIMPLE_COMMANDS_BOT.png width="400" />
2020-02-14 13:17:57 +01:00
</div>
2020-02-18 23:54:41 +01:00
See? The dispatcher gives us a stream of messages, so we can handle it as we want! Here we use [`.filter_map()`](https://docs.rs/futures/0.3.4/futures/stream/trait.StreamExt.html#method.filter_map) and [`.for_each_concurrent()`](https://docs.rs/futures/0.3.4/futures/stream/trait.StreamExt.html#method.for_each_concurrent), but others are also available:
- [`.flatten()`](https://docs.rs/futures/0.3.4/futures/stream/trait.StreamExt.html#method.flatten)
- [`.left_stream()`](https://docs.rs/futures/0.3.4/futures/stream/trait.StreamExt.html#method.left_stream)
- [`.scan()`](https://docs.rs/futures/0.3.4/futures/stream/trait.StreamExt.html#method.scan)
- [`.skip_while()`](https://docs.rs/futures/0.3.4/futures/stream/trait.StreamExt.html#method.skip_while)
- [`.zip()`](https://docs.rs/futures/0.3.4/futures/stream/trait.StreamExt.html#method.zip)
- [`.select_next_some()`](https://docs.rs/futures/0.3.4/futures/stream/trait.StreamExt.html#method.select_next_some)
- [`.fold()`](https://docs.rs/futures/0.3.4/futures/stream/trait.StreamExt.html#method.fold)
- [`.inspect()`](https://docs.rs/futures/0.3.4/futures/stream/trait.StreamExt.html#method.inspect)
- ... And lots of [others](https://docs.rs/futures/0.3.4/futures/stream/trait.StreamExt.html)!
2020-02-13 21:19:08 +01:00
## Guess a number
Wanna see more? This is a bot, which starts a game on each incoming message. You must guess a number from 1 to 10 (inclusively):
2020-02-14 13:28:52 +01:00
([Full](https://github.com/teloxide/teloxide/blob/master/examples/guess_a_number_bot/src/main.rs))
2020-02-13 21:19:08 +01:00
```rust
2020-02-18 23:54:41 +01:00
// Setup is omitted...
2020-02-13 21:19:08 +01:00
#[derive(SmartDefault)]
enum Dialogue {
#[default]
Start,
ReceiveAttempt(u8),
}
2020-02-14 10:57:14 +01:00
2020-02-13 21:19:08 +01:00
async fn handle_message(
2020-02-18 23:54:41 +01:00
cx: DialogueDispatcherHandlerCx<Message, Dialogue>,
) -> ResponseResult<DialogueStage<Dialogue>> {
match cx.dialogue {
2020-02-13 21:19:08 +01:00
Dialogue::Start => {
2020-02-18 23:54:41 +01:00
cx.answer(
2020-02-13 21:19:08 +01:00
"Let's play a game! Guess a number from 1 to 10 (inclusively).",
)
.send()
.await?;
next(Dialogue::ReceiveAttempt(thread_rng().gen_range(1, 11)))
}
2020-02-18 23:54:41 +01:00
Dialogue::ReceiveAttempt(secret) => match cx.update.text() {
2020-02-13 21:19:08 +01:00
None => {
2020-02-18 23:54:41 +01:00
cx.answer("Oh, please, send me a text message!").send().await?;
next(cx.dialogue)
2020-02-13 21:19:08 +01:00
}
Some(text) => match text.parse::<u8>() {
Ok(attempt) => match attempt {
x if !(1..=10).contains(&x) => {
2020-02-18 23:54:41 +01:00
cx.answer(
2020-02-13 21:19:08 +01:00
"Oh, please, send me a number in the range [1; \
10]!",
)
.send()
.await?;
2020-02-18 23:54:41 +01:00
next(cx.dialogue)
2020-02-13 21:19:08 +01:00
}
x if x == secret => {
2020-02-18 23:54:41 +01:00
cx.answer("Congratulations! You won!").send().await?;
2020-02-13 21:19:08 +01:00
exit()
}
_ => {
2020-02-18 23:54:41 +01:00
cx.answer("No.").send().await?;
next(cx.dialogue)
2020-02-13 21:19:08 +01:00
}
},
Err(_) => {
2020-02-18 23:54:41 +01:00
cx.answer(
2020-02-13 23:11:29 +01:00
"Oh, please, send me a number in the range [1; 10]!",
)
.send()
.await?;
2020-02-18 23:54:41 +01:00
next(cx.dialogue)
2020-02-13 21:19:08 +01:00
}
},
},
}
}
#[tokio::main]
async fn main() {
2020-02-14 08:31:42 +01:00
// Setup is omitted...
2020-02-13 21:19:08 +01:00
}
```
2020-02-15 14:34:06 +01:00
<details>
2020-02-18 23:54:41 +01:00
<summary>Click here to run it!</summary>
2020-02-15 14:34:06 +01:00
```bash
git clone https://github.com/teloxide/teloxide.git
cd teloxide/examples/guess_a_number_bot
TELOXIDE_TOKEN=MyAwesomeToken cargo run
```
</details>
2020-02-14 13:17:57 +01:00
<div align="center">
2020-02-14 13:19:39 +01:00
<img src=https://github.com/teloxide/teloxide/raw/master/media/GUESS_A_NUMBER_BOT.png width="400" />
2020-02-14 13:17:57 +01:00
</div>
2020-02-19 00:08:45 +01:00
Our [finite automaton](https://en.wikipedia.org/wiki/Finite-state_machine), designating a user dialogue, cannot be in an invalid state, and this is why it is called "type-safe". We could use `enum` + `Option`s instead, but it will lead is to lots of unpleasure `.unwrap()`s.
2020-02-13 21:38:58 +01:00
2020-02-19 00:08:45 +01:00
Remember that a classical [finite automaton](https://en.wikipedia.org/wiki/Finite-state_machine) is defined by its initial state, a list of its possible states and a transition function? We can think that `Dialogue` is a finite automaton with a context type at each state (`Dialogue::Start` has `()`, `Dialogue::ReceiveAttempt` has `u8`).
2020-02-18 23:59:44 +01:00
2020-02-19 00:10:10 +01:00
See [examples/dialogue_bot](https://github.com/teloxide/teloxide/blob/master/examples/dialogue_bot/src/main.rs) to see a bit more complicated bot with dialogues.
## More examples!
- [ping_pong_bot](ping_pong_bot) - Answers "pong" to each incoming message.
- [simple_commands_bot](simple_commands_bot) - Shows how to deal with bot's commands.
- [guess_a_number_bot](guess_a_number_bot) - The "guess a number" game.
- [dialogue_bot](dialogue_bot) - Drive a dialogue with a user using a type-safe finite automaton.
- [admin_bot](admin_bot) - A bot, which can ban, kick, and mute on a command.
2020-02-14 00:01:40 +01:00
2020-02-13 21:38:58 +01:00
## Recommendations
- Use this pattern:
```rust
#[tokio::main]
async fn main() {
run().await;
}
async fn run() {
// Your logic here...
}
```
Instead of this:
```rust
#[tokio::main]
async fn main() {
// Your logic here...
}
```
2020-02-14 10:53:14 +01:00
The second one produces very strange compiler messages because of the `#[tokio::main]` macro. However, the examples in this README use the second variant for brevity.
2020-02-14 09:45:12 +01:00
## Contributing
2020-02-14 13:28:52 +01:00
See [CONRIBUTING.md](https://github.com/teloxide/teloxide/blob/master/CONTRIBUTING.md).