内容介绍
本专题将带你使用axum实现一个 Webhook 形式的 Telegram 机器人。webhook
Telegram 机器人支持两种方式:轮询和 Webhook。为了节约资源我们将使用 Webhook 的方式开发 Telegram 机器人。处理文本消息
本章将开始与 Telegram 机器人进行交互。首先从最简单的文本消息开始。处理指令
本章我们将学习如何处理 Telegram 的“指令”(command)。开始之前,我们对之前的代码进行必要的封装。发送图片
本章继续完善我们的机器人。收到用户的`/logo`指令,我们需要把我们的 LOGO 图片发送给用户。让我们来看看如何让 Telegram 机器人发送图片信息。发送Markdown
Telegram 还支持 Markdown 和 HTML 类型的文本消息。本章我们将实现`/help`指令,它会将帮助信息以 Markdown 格式发送给用户。总结
本专题带你实现了一个简单的 Telegram 机器人。我们实现的功能是很简单的,其实 Telegram 支持多种消息
发送Markdown
Telegram 还支持 Markdown 和 HTML 类型的文本消息。本章我们将实现/help
指令,它会将帮助信息以 Markdown 格式发送给用户。
本章代码在05/发送 Markdown分支。
本章代码在05/发送 Markdown分支。
Telegram 只是有限支持 Markdown 和 HTML,即便如此,我们也可以利用这有限的支持发送格式多样的文本消息。
我们需要给MsgType
增加一个枚举值:
// src/types/mod.rs
pub enum MsgType {
Text(String),
Photo(String),
Markdown(String),
}
hook 处理
pub async fn hook(
Json(payload): Json<Update>,
Extension(state): Extension<AppState>,
) -> Result<String> {
// ..
let msg_text = payload.message.text.unwrap_or("".to_string());
let msg_type = match msg_text.as_str() {
"/website" => MsgType::Text(command::website()),
"/logo" => MsgType::Photo(command::logo()),
"/help" => MsgType::Markdown(command::help(None)),
_ => MsgType::Markdown(command::help(Some(&msg_text))),
};
let res = match msg_type {
MsgType::Text(reply_msg) => {
bot::send_text_message(&state.bot.token, payload.message.chat.id, reply_msg).await
}
MsgType::Photo(reply_msg) => {
bot::send_photo_message(&state.bot.token, payload.message.chat.id, reply_msg).await
}
MsgType::Markdown(reply_msg) => {
bot::send_markdown_message(&state.bot.token, payload.message.chat.id, reply_msg).await
}
}
.map_err(log_error(msg_text));
// ..
}
pub async fn send_markdown_message(token: &str, chat_id: u64, text: String) -> Result<Response> {
let data = request::MarkdownMessage {
chat_id,
text,
parse_mode: "MarkdownV2".to_string(),
};
invoke_api(&data, "sendMessage", token).await
}
Telegram 是通过 parse_mode
字段来识别采用何种格式解析文本消息的。所以我们的 MarkdownMessage
和之前的 TextMessage
相比,只需要加上一个parse_mode
字段。
// src/types/request.rs
#[derive(Serialize, Debug)]
pub struct MarkdownMessage {
pub chat_id: u64,
pub text: String,
pub parse_mode: String,
}
你可以只定义一个结构体来发送不同格式的文本消息,而不是像本专题这样分别定义不同的结构体。
你可以只定义一个结构体来发送不同格式的文本消息,而不是像本专题这样分别定义不同的结构体。
返回 Markdown 消息内容
// src/handler/command.rs
pub fn help(msg: Option<&str>) -> String {
let header = match msg {
Some(txt) => format!("你输入的 `{}` 有误", txt),
None => "".to_string(),
};
let body = r#"
__使用帮助__
`/website`:访问官方网站
`/logo`:获取官方LOGO
`/help`:显示帮助信息
"#;
format!(r#"{}{}"#, header, body)
}