mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-11 20:43:42 +00:00
refactor(telegram): extract sequential key module
This commit is contained in:
@@ -1,7 +1,6 @@
|
|||||||
import fs from "node:fs";
|
import fs from "node:fs";
|
||||||
import os from "node:os";
|
import os from "node:os";
|
||||||
import path from "node:path";
|
import path from "node:path";
|
||||||
import type { Chat, Message } from "@grammyjs/types";
|
|
||||||
import { afterAll, beforeAll, describe, expect, it, vi } from "vitest";
|
import { afterAll, beforeAll, describe, expect, it, vi } from "vitest";
|
||||||
import { escapeRegExp, formatEnvelopeTimestamp } from "../../test/helpers/envelope-timestamp.js";
|
import { escapeRegExp, formatEnvelopeTimestamp } from "../../test/helpers/envelope-timestamp.js";
|
||||||
import { withEnvAsync } from "../test-utils/env.js";
|
import { withEnvAsync } from "../test-utils/env.js";
|
||||||
@@ -39,14 +38,6 @@ const readChannelAllowFromStore = getReadChannelAllowFromStoreMock();
|
|||||||
const upsertChannelPairingRequest = getUpsertChannelPairingRequestMock();
|
const upsertChannelPairingRequest = getUpsertChannelPairingRequestMock();
|
||||||
|
|
||||||
const ORIGINAL_TZ = process.env.TZ;
|
const ORIGINAL_TZ = process.env.TZ;
|
||||||
const mockChat = (chat: Pick<Chat, "id"> & Partial<Pick<Chat, "type" | "is_forum">>): Chat =>
|
|
||||||
chat as Chat;
|
|
||||||
const mockMessage = (message: Pick<Message, "chat"> & Partial<Message>): Message =>
|
|
||||||
({
|
|
||||||
message_id: 1,
|
|
||||||
date: 0,
|
|
||||||
...message,
|
|
||||||
}) as Message;
|
|
||||||
const TELEGRAM_TEST_TIMINGS = {
|
const TELEGRAM_TEST_TIMINGS = {
|
||||||
mediaGroupFlushMs: 20,
|
mediaGroupFlushMs: 20,
|
||||||
textFragmentGapMs: 30,
|
textFragmentGapMs: 30,
|
||||||
@@ -124,87 +115,6 @@ describe("createTelegramBot", () => {
|
|||||||
expect(sequentializeSpy).toHaveBeenCalledTimes(1);
|
expect(sequentializeSpy).toHaveBeenCalledTimes(1);
|
||||||
expect(middlewareUseSpy).toHaveBeenCalledWith(sequentializeSpy.mock.results[0]?.value);
|
expect(middlewareUseSpy).toHaveBeenCalledWith(sequentializeSpy.mock.results[0]?.value);
|
||||||
expect(sequentializeKey).toBe(getTelegramSequentialKey);
|
expect(sequentializeKey).toBe(getTelegramSequentialKey);
|
||||||
const cases = [
|
|
||||||
[{ message: mockMessage({ chat: mockChat({ id: 123 }) }) }, "telegram:123"],
|
|
||||||
[
|
|
||||||
{
|
|
||||||
message: mockMessage({
|
|
||||||
chat: mockChat({ id: 123, type: "private" }),
|
|
||||||
message_thread_id: 9,
|
|
||||||
}),
|
|
||||||
},
|
|
||||||
"telegram:123:topic:9",
|
|
||||||
],
|
|
||||||
[
|
|
||||||
{
|
|
||||||
message: mockMessage({
|
|
||||||
chat: mockChat({ id: 123, type: "supergroup" }),
|
|
||||||
message_thread_id: 9,
|
|
||||||
}),
|
|
||||||
},
|
|
||||||
"telegram:123",
|
|
||||||
],
|
|
||||||
[
|
|
||||||
{
|
|
||||||
message: mockMessage({
|
|
||||||
chat: mockChat({ id: 123, type: "supergroup", is_forum: true }),
|
|
||||||
}),
|
|
||||||
},
|
|
||||||
"telegram:123:topic:1",
|
|
||||||
],
|
|
||||||
[{ update: { message: mockMessage({ chat: mockChat({ id: 555 }) }) } }, "telegram:555"],
|
|
||||||
[
|
|
||||||
{
|
|
||||||
channelPost: mockMessage({ chat: mockChat({ id: -100777111222, type: "channel" }) }),
|
|
||||||
},
|
|
||||||
"telegram:-100777111222",
|
|
||||||
],
|
|
||||||
[
|
|
||||||
{
|
|
||||||
update: {
|
|
||||||
channel_post: mockMessage({ chat: mockChat({ id: -100777111223, type: "channel" }) }),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"telegram:-100777111223",
|
|
||||||
],
|
|
||||||
[
|
|
||||||
{ message: mockMessage({ chat: mockChat({ id: 123 }), text: "/stop" }) },
|
|
||||||
"telegram:123:control",
|
|
||||||
],
|
|
||||||
[{ message: mockMessage({ chat: mockChat({ id: 123 }), text: "/status" }) }, "telegram:123"],
|
|
||||||
[
|
|
||||||
{ message: mockMessage({ chat: mockChat({ id: 123 }), text: "stop" }) },
|
|
||||||
"telegram:123:control",
|
|
||||||
],
|
|
||||||
[
|
|
||||||
{ message: mockMessage({ chat: mockChat({ id: 123 }), text: "stop please" }) },
|
|
||||||
"telegram:123:control",
|
|
||||||
],
|
|
||||||
[
|
|
||||||
{ message: mockMessage({ chat: mockChat({ id: 123 }), text: "do not do that" }) },
|
|
||||||
"telegram:123:control",
|
|
||||||
],
|
|
||||||
[
|
|
||||||
{ message: mockMessage({ chat: mockChat({ id: 123 }), text: "остановись" }) },
|
|
||||||
"telegram:123:control",
|
|
||||||
],
|
|
||||||
[
|
|
||||||
{ message: mockMessage({ chat: mockChat({ id: 123 }), text: "halt" }) },
|
|
||||||
"telegram:123:control",
|
|
||||||
],
|
|
||||||
[{ message: mockMessage({ chat: mockChat({ id: 123 }), text: "/abort" }) }, "telegram:123"],
|
|
||||||
[
|
|
||||||
{ message: mockMessage({ chat: mockChat({ id: 123 }), text: "/abort now" }) },
|
|
||||||
"telegram:123",
|
|
||||||
],
|
|
||||||
[
|
|
||||||
{ message: mockMessage({ chat: mockChat({ id: 123 }), text: "please do not do that" }) },
|
|
||||||
"telegram:123",
|
|
||||||
],
|
|
||||||
] as const;
|
|
||||||
for (const [input, expected] of cases) {
|
|
||||||
expect(getTelegramSequentialKey(input)).toBe(expected);
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
it("routes callback_query payloads as messages and answers callbacks", async () => {
|
it("routes callback_query payloads as messages and answers callbacks", async () => {
|
||||||
createTelegramBot({ token: "tok" });
|
createTelegramBot({ token: "tok" });
|
||||||
|
|||||||
@@ -1,11 +1,9 @@
|
|||||||
import { sequentialize } from "@grammyjs/runner";
|
import { sequentialize } from "@grammyjs/runner";
|
||||||
import { apiThrottler } from "@grammyjs/transformer-throttler";
|
import { apiThrottler } from "@grammyjs/transformer-throttler";
|
||||||
import { type Message, type UserFromGetMe } from "@grammyjs/types";
|
|
||||||
import type { ApiClientOptions } from "grammy";
|
import type { ApiClientOptions } from "grammy";
|
||||||
import { Bot, webhookCallback } from "grammy";
|
import { Bot, webhookCallback } from "grammy";
|
||||||
import { resolveDefaultAgentId } from "../agents/agent-scope.js";
|
import { resolveDefaultAgentId } from "../agents/agent-scope.js";
|
||||||
import { resolveTextChunkLimit } from "../auto-reply/chunk.js";
|
import { resolveTextChunkLimit } from "../auto-reply/chunk.js";
|
||||||
import { isAbortRequestText } from "../auto-reply/reply/abort.js";
|
|
||||||
import { DEFAULT_GROUP_HISTORY_LIMIT, type HistoryEntry } from "../auto-reply/reply/history.js";
|
import { DEFAULT_GROUP_HISTORY_LIMIT, type HistoryEntry } from "../auto-reply/reply/history.js";
|
||||||
import {
|
import {
|
||||||
isNativeCommandsExplicitlyDisabled,
|
isNativeCommandsExplicitlyDisabled,
|
||||||
@@ -34,13 +32,10 @@ import {
|
|||||||
resolveTelegramUpdateId,
|
resolveTelegramUpdateId,
|
||||||
type TelegramUpdateKeyContext,
|
type TelegramUpdateKeyContext,
|
||||||
} from "./bot-updates.js";
|
} from "./bot-updates.js";
|
||||||
import {
|
import { buildTelegramGroupPeerId, resolveTelegramStreamMode } from "./bot/helpers.js";
|
||||||
buildTelegramGroupPeerId,
|
|
||||||
resolveTelegramForumThreadId,
|
|
||||||
resolveTelegramStreamMode,
|
|
||||||
} from "./bot/helpers.js";
|
|
||||||
import { resolveTelegramFetch } from "./fetch.js";
|
import { resolveTelegramFetch } from "./fetch.js";
|
||||||
import { createTelegramSendChatActionHandler } from "./sendchataction-401-backoff.js";
|
import { createTelegramSendChatActionHandler } from "./sendchataction-401-backoff.js";
|
||||||
|
import { getTelegramSequentialKey } from "./sequential-key.js";
|
||||||
|
|
||||||
export type TelegramBotOptions = {
|
export type TelegramBotOptions = {
|
||||||
token: string;
|
token: string;
|
||||||
@@ -63,55 +58,7 @@ export type TelegramBotOptions = {
|
|||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
export function getTelegramSequentialKey(ctx: {
|
export { getTelegramSequentialKey };
|
||||||
chat?: { id?: number };
|
|
||||||
me?: UserFromGetMe;
|
|
||||||
message?: Message;
|
|
||||||
channelPost?: Message;
|
|
||||||
editedChannelPost?: Message;
|
|
||||||
update?: {
|
|
||||||
message?: Message;
|
|
||||||
edited_message?: Message;
|
|
||||||
channel_post?: Message;
|
|
||||||
edited_channel_post?: Message;
|
|
||||||
callback_query?: { message?: Message };
|
|
||||||
message_reaction?: { chat?: { id?: number } };
|
|
||||||
};
|
|
||||||
}): string {
|
|
||||||
// Handle reaction updates
|
|
||||||
const reaction = ctx.update?.message_reaction;
|
|
||||||
if (reaction?.chat?.id) {
|
|
||||||
return `telegram:${reaction.chat.id}`;
|
|
||||||
}
|
|
||||||
const msg =
|
|
||||||
ctx.message ??
|
|
||||||
ctx.channelPost ??
|
|
||||||
ctx.editedChannelPost ??
|
|
||||||
ctx.update?.message ??
|
|
||||||
ctx.update?.edited_message ??
|
|
||||||
ctx.update?.channel_post ??
|
|
||||||
ctx.update?.edited_channel_post ??
|
|
||||||
ctx.update?.callback_query?.message;
|
|
||||||
const chatId = msg?.chat?.id ?? ctx.chat?.id;
|
|
||||||
const rawText = msg?.text ?? msg?.caption;
|
|
||||||
const botUsername = ctx.me?.username;
|
|
||||||
if (isAbortRequestText(rawText, botUsername ? { botUsername } : undefined)) {
|
|
||||||
if (typeof chatId === "number") {
|
|
||||||
return `telegram:${chatId}:control`;
|
|
||||||
}
|
|
||||||
return "telegram:control";
|
|
||||||
}
|
|
||||||
const isGroup = msg?.chat?.type === "group" || msg?.chat?.type === "supergroup";
|
|
||||||
const messageThreadId = msg?.message_thread_id;
|
|
||||||
const isForum = msg?.chat?.is_forum;
|
|
||||||
const threadId = isGroup
|
|
||||||
? resolveTelegramForumThreadId({ isForum, messageThreadId })
|
|
||||||
: messageThreadId;
|
|
||||||
if (typeof chatId === "number") {
|
|
||||||
return threadId != null ? `telegram:${chatId}:topic:${threadId}` : `telegram:${chatId}`;
|
|
||||||
}
|
|
||||||
return "telegram:unknown";
|
|
||||||
}
|
|
||||||
|
|
||||||
export function createTelegramBot(opts: TelegramBotOptions) {
|
export function createTelegramBot(opts: TelegramBotOptions) {
|
||||||
const runtime: RuntimeEnv = opts.runtime ?? createNonExitingRuntime();
|
const runtime: RuntimeEnv = opts.runtime ?? createNonExitingRuntime();
|
||||||
|
|||||||
92
src/telegram/sequential-key.test.ts
Normal file
92
src/telegram/sequential-key.test.ts
Normal file
@@ -0,0 +1,92 @@
|
|||||||
|
import type { Chat, Message } from "@grammyjs/types";
|
||||||
|
import { describe, expect, it } from "vitest";
|
||||||
|
import { getTelegramSequentialKey } from "./sequential-key.js";
|
||||||
|
|
||||||
|
const mockChat = (chat: Pick<Chat, "id"> & Partial<Pick<Chat, "type" | "is_forum">>): Chat =>
|
||||||
|
chat as Chat;
|
||||||
|
const mockMessage = (message: Pick<Message, "chat"> & Partial<Message>): Message =>
|
||||||
|
({
|
||||||
|
message_id: 1,
|
||||||
|
date: 0,
|
||||||
|
...message,
|
||||||
|
}) as Message;
|
||||||
|
|
||||||
|
describe("getTelegramSequentialKey", () => {
|
||||||
|
it.each([
|
||||||
|
[{ message: mockMessage({ chat: mockChat({ id: 123 }) }) }, "telegram:123"],
|
||||||
|
[
|
||||||
|
{
|
||||||
|
message: mockMessage({
|
||||||
|
chat: mockChat({ id: 123, type: "private" }),
|
||||||
|
message_thread_id: 9,
|
||||||
|
}),
|
||||||
|
},
|
||||||
|
"telegram:123:topic:9",
|
||||||
|
],
|
||||||
|
[
|
||||||
|
{
|
||||||
|
message: mockMessage({
|
||||||
|
chat: mockChat({ id: 123, type: "supergroup" }),
|
||||||
|
message_thread_id: 9,
|
||||||
|
}),
|
||||||
|
},
|
||||||
|
"telegram:123",
|
||||||
|
],
|
||||||
|
[
|
||||||
|
{
|
||||||
|
message: mockMessage({
|
||||||
|
chat: mockChat({ id: 123, type: "supergroup", is_forum: true }),
|
||||||
|
}),
|
||||||
|
},
|
||||||
|
"telegram:123:topic:1",
|
||||||
|
],
|
||||||
|
[{ update: { message: mockMessage({ chat: mockChat({ id: 555 }) }) } }, "telegram:555"],
|
||||||
|
[
|
||||||
|
{
|
||||||
|
channelPost: mockMessage({ chat: mockChat({ id: -100777111222, type: "channel" }) }),
|
||||||
|
},
|
||||||
|
"telegram:-100777111222",
|
||||||
|
],
|
||||||
|
[
|
||||||
|
{
|
||||||
|
update: {
|
||||||
|
channel_post: mockMessage({ chat: mockChat({ id: -100777111223, type: "channel" }) }),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"telegram:-100777111223",
|
||||||
|
],
|
||||||
|
[
|
||||||
|
{ message: mockMessage({ chat: mockChat({ id: 123 }), text: "/stop" }) },
|
||||||
|
"telegram:123:control",
|
||||||
|
],
|
||||||
|
[{ message: mockMessage({ chat: mockChat({ id: 123 }), text: "/status" }) }, "telegram:123"],
|
||||||
|
[
|
||||||
|
{ message: mockMessage({ chat: mockChat({ id: 123 }), text: "stop" }) },
|
||||||
|
"telegram:123:control",
|
||||||
|
],
|
||||||
|
[
|
||||||
|
{ message: mockMessage({ chat: mockChat({ id: 123 }), text: "stop please" }) },
|
||||||
|
"telegram:123:control",
|
||||||
|
],
|
||||||
|
[
|
||||||
|
{ message: mockMessage({ chat: mockChat({ id: 123 }), text: "do not do that" }) },
|
||||||
|
"telegram:123:control",
|
||||||
|
],
|
||||||
|
[
|
||||||
|
{ message: mockMessage({ chat: mockChat({ id: 123 }), text: "остановись" }) },
|
||||||
|
"telegram:123:control",
|
||||||
|
],
|
||||||
|
[
|
||||||
|
{ message: mockMessage({ chat: mockChat({ id: 123 }), text: "halt" }) },
|
||||||
|
"telegram:123:control",
|
||||||
|
],
|
||||||
|
[{ message: mockMessage({ chat: mockChat({ id: 123 }), text: "/abort" }) }, "telegram:123"],
|
||||||
|
[{ message: mockMessage({ chat: mockChat({ id: 123 }), text: "/abort now" }) }, "telegram:123"],
|
||||||
|
[
|
||||||
|
{ message: mockMessage({ chat: mockChat({ id: 123 }), text: "please do not do that" }) },
|
||||||
|
"telegram:123",
|
||||||
|
],
|
||||||
|
])("resolves key %#", (input, expected) => {
|
||||||
|
expect(getTelegramSequentialKey(input)).toBe(expected);
|
||||||
|
});
|
||||||
|
});
|
||||||
54
src/telegram/sequential-key.ts
Normal file
54
src/telegram/sequential-key.ts
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
import { type Message, type UserFromGetMe } from "@grammyjs/types";
|
||||||
|
import { isAbortRequestText } from "../auto-reply/reply/abort.js";
|
||||||
|
import { resolveTelegramForumThreadId } from "./bot/helpers.js";
|
||||||
|
|
||||||
|
export type TelegramSequentialKeyContext = {
|
||||||
|
chat?: { id?: number };
|
||||||
|
me?: UserFromGetMe;
|
||||||
|
message?: Message;
|
||||||
|
channelPost?: Message;
|
||||||
|
editedChannelPost?: Message;
|
||||||
|
update?: {
|
||||||
|
message?: Message;
|
||||||
|
edited_message?: Message;
|
||||||
|
channel_post?: Message;
|
||||||
|
edited_channel_post?: Message;
|
||||||
|
callback_query?: { message?: Message };
|
||||||
|
message_reaction?: { chat?: { id?: number } };
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
export function getTelegramSequentialKey(ctx: TelegramSequentialKeyContext): string {
|
||||||
|
const reaction = ctx.update?.message_reaction;
|
||||||
|
if (reaction?.chat?.id) {
|
||||||
|
return `telegram:${reaction.chat.id}`;
|
||||||
|
}
|
||||||
|
const msg =
|
||||||
|
ctx.message ??
|
||||||
|
ctx.channelPost ??
|
||||||
|
ctx.editedChannelPost ??
|
||||||
|
ctx.update?.message ??
|
||||||
|
ctx.update?.edited_message ??
|
||||||
|
ctx.update?.channel_post ??
|
||||||
|
ctx.update?.edited_channel_post ??
|
||||||
|
ctx.update?.callback_query?.message;
|
||||||
|
const chatId = msg?.chat?.id ?? ctx.chat?.id;
|
||||||
|
const rawText = msg?.text ?? msg?.caption;
|
||||||
|
const botUsername = ctx.me?.username;
|
||||||
|
if (isAbortRequestText(rawText, botUsername ? { botUsername } : undefined)) {
|
||||||
|
if (typeof chatId === "number") {
|
||||||
|
return `telegram:${chatId}:control`;
|
||||||
|
}
|
||||||
|
return "telegram:control";
|
||||||
|
}
|
||||||
|
const isGroup = msg?.chat?.type === "group" || msg?.chat?.type === "supergroup";
|
||||||
|
const messageThreadId = msg?.message_thread_id;
|
||||||
|
const isForum = msg?.chat?.is_forum;
|
||||||
|
const threadId = isGroup
|
||||||
|
? resolveTelegramForumThreadId({ isForum, messageThreadId })
|
||||||
|
: messageThreadId;
|
||||||
|
if (typeof chatId === "number") {
|
||||||
|
return threadId != null ? `telegram:${chatId}:topic:${threadId}` : `telegram:${chatId}`;
|
||||||
|
}
|
||||||
|
return "telegram:unknown";
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user