域名 AXUM.RS 将于2025年10月到期。我们无意再对其进行续费,我们希望你能够接续这个域名,让更多 AXUM 开发者继续受益。
  • 方案1️⃣AXUM.RS 域名 = 3000
  • 方案2️⃣方案1️⃣ + 本站所有专题原始 Markdown 文档 = 5000
  • 方案3️⃣方案2️⃣ + 本站原始数据库 = 5500
如果你有意接续这份 AXUM 情怀,请与我们取得联系。
说明:
  1. 如果有人购买 AXUM.RS 域名(方案1️⃣),或者该域名到期,本站将启用新的免费域名继续提供服务。
  2. 如果有人购买了 AXUM.RS 域名,且同时购买了内容和/或数据库(方案2️⃣/方案3️⃣),本站将关闭。届时我们或许会以另一种方式与你再相遇。

发送图片

本章继续完善我们的机器人。收到用户的/logo指令,我们需要把我们的 LOGO 图片发送给用户。让我们来看看如何让 Telegram 机器人发送图片信息。

本章代码在04/发送图片分支。

由于发送图片消息和前一章说的发送文本消息的 API 不一样,所以我们需要对不同类型的消息进行识别,而 Rust 强大的枚举类型能让我们很容易的实现。

消息类型枚举

// src/types/mod.rs

pub enum MsgType {
    Text(String),
    Photo(String),
}

有了消息类型枚举,我们在 hook 中判断出不同指令后,就可以使用这个枚举进行标识、处理了。

// src/handler/mod.rs

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()),
        _ => MsgType::Text(echo(msg_text.clone())),
    };

    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
        }
    }
    .map_err(log_error(msg_text));

    // ..
}

对于不同的指令,我们将其映射到不同的消息类型。然后通过消息类型来调用不同的 API,实现消息的发送。

调用发送图片消息的 API

// src/bot.rs

pub async fn send_photo_message(token: &str, chat_id: u64, photo: String) -> Result<Response> {
    let data = request::PhotoMessage { chat_id, photo };
    invoke_api(&data, "sendPhoto", token).await
}

这里使用了 request::PhotoMessage,我们还没定义它。

定义 PhotoMessage

// src/types/request.rs

#[derive(Serialize, Debug)]
pub struct PhotoMessage {
    pub chat_id: u64,
    pub photo: String,
}

定义 command::logo()

最后,别忘了定义返回 LOGO 图片的函数:

Telegram 支持多种方式发送图片。我们使用的是直接返回一个有效的图片的 URL,其它方式还有:使用 Telegram 服务器上已存在的图片以及从本地上传。详细信息请参见官方文档。

要查看完整内容,请先登录