关注点二:高负载
你的 bot 能够处理高负载的消息取决于你有没有让它处于 长轮询状态运行或者通过使用 webhooks。 不论使用哪种方式,你都应该阅读 下面的一些坑点。
长轮询
大多数 bot 都不需要在每分钟处理超过一些信息("高峰值"期间)。 换句话来说,伸缩性不是它们要关心的问题。 为了便于预测,grammY 会按顺序处理更新。 这是操作的顺序:
- 通过
get
(Telegram Bot API 参考) 获取超过100个 update。Updates - 对于每一次更新,
等待
中间件去堆栈。
不过,如果你的 bot 在高负载期间每秒钟处理一条信息(或者在类似这样的场景下),这可能会对响应造成负面影响。 例如, Bob 的消息必须得等待 Alice 的消息处理完。
这能够通过使用并发去解决,从而不需要等待 Alice 的消息被处理完再执行下一条消息,而同时去处理两条消息。 为了实现最大限度的响应,我们还希望在 Bob 和 Alice 的消息还在处理时引入新的消息。 理想情况下,我们还希望将并发限制在某个固定数量,以约束最大服务器负载。
并发处理的模块并没有被打包在 grammy 的核心库中。 相应替代的是, grammY runner 这个包可以在你的 bot 中使用。 它支持上述所有开箱即用的功能,而且使用起来非常简单。
// 之前的写法
bot.start();
// 使用 grammY runner 中导出的 `run`
run(bot);
默认的并发量限制在了500。 如果你想对这个库有更深入的了解,看这里
并发是困难的,所以当你使用 grammY runner
的时候你应该记住一些东西,详情看 下面的小节
Webhooks
如果你让你的 bot 运行在 webhooks模式下,只要接收到更新它就会并发去处理。 当然,为了让它能够在高负载下运行良好,你应该熟悉 怎样去使用 webhooks。 这就意味着你不得不去意识到一些并发带来的后果,详见 下一小节。
Telegram 将按顺序传送来自同一聊天的更新,但也同时并发传送来自不同聊天的更新 (source)。
并发是困难的
如果你的 bot 使用并发来处理所有更新,这将会引起一系列需要特别注意的问题。 比如,如果两条来自同一聊天的信息最终被同一个 get
处理,它们将会被并发处理。 同一聊天中消息的顺序不再得到保证。
发生冲突的主要原因是,当你使用 sessions,有可能会发生读后写的风险。 想一下这些事件的顺序:
- Alice 发送消息 A
- Bot 开始处理消息 A
- Bot 从数据库中为 Alice 读 session 的数据
- Alice 发送消息 B
- Bot 开始处理消息 B
- Bot 从数据库中为 Alice 读 session 的数据
- Bot 完成处理消息 A ,并且写入新的 session 到数据库中。
- Bot 完成处理消息B, 并且写入新的 session到数据库中,并且覆盖掉在处理消息 A 期间执行的操作。 数据因为读后写而丢失掉了!
Note: 你可以尝试使用数据库事务对 session 进行处理 ,但你只能检测到风险而不能阻止它。 尝试使用锁将有效地排除所有并发带来的问题。 首先避免风险要容易得多。
大多数其他web框架的 session 系统只是简单地接受竞态的风险,因为它们在web上并不经常发生 但是,我们不希望这样,因为Telegram bots 更有可能遇到对同一会话密钥的并行请求冲突。 因此,为了避免这些危险的竞态风险,我们必须确保访问相同的 session 数据的更新是按顺序的。
grammY runner 中封装了 sequentialize()
中间件来确保发生冲突的更新可以被按顺序处理。 你可以将其配置为与确定 session 密钥相同的功能 。 它将通过减慢那些(也仅仅是那些)可能引起冲突的更新来避免上述所说的竞态。
import { Bot, Context, session } from "grammy";
import { run, sequentialize } from "@grammyjs/runner";
// 创建一个 bot。
const bot = new Bot("<token>");
// 为 `Context` 对象构建唯一标识符。
function getSessionKey(ctx: Context) {
return ctx.chat?.id.toString();
}
// 在访问 session 数据之前顺序化
bot.use(sequentialize(getSessionKey));
bot.use(session({ getSessionKey }));
// 添加常用中间件,现在提供了安全的 session 支持。
bot.on("message", (ctx) => ctx.reply("Got your message."));
// 仍然让它并发运行!
run(bot);
const { Bot, Context, session } = require("grammy");
const { run, sequentialize } = require("@grammyjs/runner");
// 创建一个 bot。
const bot = new Bot("<token>");
// 为 `Context` 对象构建唯一标识符。
function getSessionKey(ctx) {
return ctx.chat?.id.toString();
}
// 在访问 session 数据之前顺序化
bot.use(sequentialize(getSessionKey));
bot.use(session({ getSessionKey }));
// 添加常用中间件,现在提供了安全的 session 支持。
bot.on("message", (ctx) => ctx.reply("Got your message."));
// 仍然让它并发运行!
run(bot);
import { Bot, Context, session } from "https://deno.land/x/grammy@v1.11.2/mod.ts";
import { run, sequentialize } from "https://deno.land/x/grammy_runner@v1.0.4/mod.ts";
// 创建一个 bot。
const bot = new Bot("<token>");
// 为 `Context` 对象构建唯一标识符。
function getSessionKey(ctx: Context) {
return ctx.chat?.id.toString();
}
// 在访问 session 数据之前顺序化
bot.use(sequentialize(getSessionKey));
bot.use(session({ getSessionKey }));
// 添加常用中间件,现在提供了安全的 session 支持。
bot.on("message", (ctx) => ctx.reply("Got your message."));
// 仍然让它并发运行!
run(bot);
加入 Telegram chat 自由的讨论怎样在你的 bot 中使用 grammY。 我们总是很高兴收到维护大型 bot 项目的朋友的来信,我们可以根据他们的经验来不断改进 grammY。