| import { afterEach, describe, expect, it } from "vitest"; |
| import { clearRuntimeConfigSnapshot, setRuntimeConfigSnapshot } from "../config/config.js"; |
| import { buildTelegramMessageContextForTest } from "./bot-message-context.test-harness.js"; |
|
|
| describe("buildTelegramMessageContext dm thread sessions", () => { |
| const buildContext = async (message: Record<string, unknown>) => |
| await buildTelegramMessageContextForTest({ |
| message, |
| }); |
|
|
| it("uses thread session key for dm topics", async () => { |
| const ctx = await buildContext({ |
| message_id: 1, |
| chat: { id: 1234, type: "private" }, |
| date: 1700000000, |
| text: "hello", |
| message_thread_id: 42, |
| from: { id: 42, first_name: "Alice" }, |
| }); |
|
|
| expect(ctx).not.toBeNull(); |
| expect(ctx?.ctxPayload?.MessageThreadId).toBe(42); |
| expect(ctx?.ctxPayload?.SessionKey).toBe("agent:main:main:thread:1234:42"); |
| }); |
|
|
| it("keeps legacy dm session key when no thread id", async () => { |
| const ctx = await buildContext({ |
| message_id: 2, |
| chat: { id: 1234, type: "private" }, |
| date: 1700000001, |
| text: "hello", |
| from: { id: 42, first_name: "Alice" }, |
| }); |
|
|
| expect(ctx).not.toBeNull(); |
| expect(ctx?.ctxPayload?.MessageThreadId).toBeUndefined(); |
| expect(ctx?.ctxPayload?.SessionKey).toBe("agent:main:main"); |
| }); |
| }); |
|
|
| describe("buildTelegramMessageContext group sessions without forum", () => { |
| const buildContext = async (message: Record<string, unknown>) => |
| await buildTelegramMessageContextForTest({ |
| message, |
| options: { forceWasMentioned: true }, |
| resolveGroupActivation: () => true, |
| }); |
|
|
| it("ignores message_thread_id for regular groups (not forums)", async () => { |
| |
| |
| const ctx = await buildContext({ |
| message_id: 1, |
| chat: { id: -1001234567890, type: "supergroup", title: "Test Group" }, |
| date: 1700000000, |
| text: "@bot hello", |
| message_thread_id: 42, |
| from: { id: 42, first_name: "Alice" }, |
| }); |
|
|
| expect(ctx).not.toBeNull(); |
| |
| expect(ctx?.ctxPayload?.SessionKey).toBe("agent:main:telegram:group:-1001234567890"); |
| |
| expect(ctx?.ctxPayload?.MessageThreadId).toBeUndefined(); |
| }); |
|
|
| it("keeps same session for regular group with and without message_thread_id", async () => { |
| const ctxWithThread = await buildContext({ |
| message_id: 1, |
| chat: { id: -1001234567890, type: "supergroup", title: "Test Group" }, |
| date: 1700000000, |
| text: "@bot hello", |
| message_thread_id: 42, |
| from: { id: 42, first_name: "Alice" }, |
| }); |
|
|
| const ctxWithoutThread = await buildContext({ |
| message_id: 2, |
| chat: { id: -1001234567890, type: "supergroup", title: "Test Group" }, |
| date: 1700000001, |
| text: "@bot world", |
| from: { id: 42, first_name: "Alice" }, |
| }); |
|
|
| expect(ctxWithThread).not.toBeNull(); |
| expect(ctxWithoutThread).not.toBeNull(); |
| |
| expect(ctxWithThread?.ctxPayload?.SessionKey).toBe(ctxWithoutThread?.ctxPayload?.SessionKey); |
| }); |
|
|
| it("uses topic session for forum groups with message_thread_id", async () => { |
| const ctx = await buildContext({ |
| message_id: 1, |
| chat: { id: -1001234567890, type: "supergroup", title: "Test Forum", is_forum: true }, |
| date: 1700000000, |
| text: "@bot hello", |
| message_thread_id: 99, |
| from: { id: 42, first_name: "Alice" }, |
| }); |
|
|
| expect(ctx).not.toBeNull(); |
| |
| expect(ctx?.ctxPayload?.SessionKey).toBe("agent:main:telegram:group:-1001234567890:topic:99"); |
| expect(ctx?.ctxPayload?.MessageThreadId).toBe(99); |
| }); |
|
|
| it("recovers forum sessions via getChat when Telegram omits is_forum", async () => { |
| const ctx = await buildTelegramMessageContextForTest({ |
| message: { |
| message_id: 1, |
| chat: { id: -1001234567890, type: "supergroup", title: "Test Forum" }, |
| date: 1700000000, |
| text: "@bot hello", |
| message_thread_id: 99, |
| from: { id: 42, first_name: "Alice" }, |
| }, |
| getChat: async () => ({ id: -1001234567890, type: "supergroup", is_forum: true }), |
| options: { forceWasMentioned: true }, |
| resolveGroupActivation: () => true, |
| }); |
|
|
| expect(ctx).not.toBeNull(); |
| expect(ctx?.ctxPayload?.SessionKey).toBe("agent:main:telegram:group:-1001234567890:topic:99"); |
| expect(ctx?.ctxPayload?.MessageThreadId).toBe(99); |
| }); |
| }); |
|
|
| describe("buildTelegramMessageContext direct peer routing", () => { |
| afterEach(() => { |
| clearRuntimeConfigSnapshot(); |
| }); |
|
|
| it("isolates dm sessions by sender id when chat id differs", async () => { |
| const runtimeCfg = { |
| agents: { defaults: { model: "anthropic/claude-opus-4-5", workspace: "/tmp/openclaw" } }, |
| channels: { telegram: {} }, |
| messages: { groupChat: { mentionPatterns: [] } }, |
| session: { dmScope: "per-channel-peer" as const }, |
| }; |
| setRuntimeConfigSnapshot(runtimeCfg); |
|
|
| const baseMessage = { |
| chat: { id: 777777777, type: "private" as const }, |
| date: 1700000000, |
| text: "hello", |
| }; |
|
|
| const first = await buildTelegramMessageContextForTest({ |
| cfg: runtimeCfg, |
| message: { |
| ...baseMessage, |
| message_id: 1, |
| from: { id: 123456789, first_name: "Alice" }, |
| }, |
| }); |
| const second = await buildTelegramMessageContextForTest({ |
| cfg: runtimeCfg, |
| message: { |
| ...baseMessage, |
| message_id: 2, |
| from: { id: 987654321, first_name: "Bob" }, |
| }, |
| }); |
|
|
| expect(first?.ctxPayload?.SessionKey).toBe("agent:main:telegram:direct:123456789"); |
| expect(second?.ctxPayload?.SessionKey).toBe("agent:main:telegram:direct:987654321"); |
| }); |
| }); |
|
|