mirror of
https://github.com/openclaw/openclaw.git
synced 2026-03-27 09:52:25 +00:00
* refactor: move Telegram channel implementation to extensions/telegram/src/ Move all Telegram channel code (123 files + 10 bot/ files + 8 channel plugin files) from src/telegram/ and src/channels/plugins/*/telegram.ts to extensions/telegram/src/. Leave thin re-export shims at original locations so cross-cutting src/ imports continue to resolve. - Fix all relative import paths in moved files (../X/ -> ../../../src/X/) - Fix vi.mock paths in 60 test files - Fix inline typeof import() expressions - Update tsconfig.plugin-sdk.dts.json rootDir to "." for cross-directory DTS - Update write-plugin-sdk-entry-dts.ts for new rootDir structure - Move channel plugin files with correct path remapping * fix: support keyed telegram send deps * fix: sync telegram extension copies with latest main * fix: correct import paths and remove misplaced files in telegram extension * fix: sync outbound-adapter with main (add sendTelegramPayloadMessages) and fix delivery.test import path
156 lines
5.0 KiB
TypeScript
156 lines
5.0 KiB
TypeScript
import { afterEach, describe, expect, it, vi } from "vitest";
|
|
import {
|
|
clearRuntimeConfigSnapshot,
|
|
setRuntimeConfigSnapshot,
|
|
} from "../../../src/config/config.js";
|
|
import { buildTelegramMessageContextForTest } from "./bot-message-context.test-harness.js";
|
|
|
|
const recordInboundSessionMock = vi.fn().mockResolvedValue(undefined);
|
|
vi.mock("../../../src/channels/session.js", () => ({
|
|
recordInboundSession: (...args: unknown[]) => recordInboundSessionMock(...args),
|
|
}));
|
|
|
|
describe("buildTelegramMessageContext named-account DM fallback", () => {
|
|
const baseCfg = {
|
|
agents: { defaults: { model: "anthropic/claude-opus-4-5", workspace: "/tmp/openclaw" } },
|
|
channels: { telegram: {} },
|
|
messages: { groupChat: { mentionPatterns: [] } },
|
|
};
|
|
|
|
afterEach(() => {
|
|
clearRuntimeConfigSnapshot();
|
|
recordInboundSessionMock.mockClear();
|
|
});
|
|
|
|
function getLastUpdateLastRoute(): { sessionKey?: string } | undefined {
|
|
const callArgs = recordInboundSessionMock.mock.calls.at(-1)?.[0] as {
|
|
updateLastRoute?: { sessionKey?: string };
|
|
};
|
|
return callArgs?.updateLastRoute;
|
|
}
|
|
|
|
function buildNamedAccountDmMessage(messageId = 1) {
|
|
return {
|
|
message_id: messageId,
|
|
chat: { id: 814912386, type: "private" as const },
|
|
date: 1700000000 + messageId - 1,
|
|
text: "hello",
|
|
from: { id: 814912386, first_name: "Alice" },
|
|
};
|
|
}
|
|
|
|
async function buildNamedAccountDmContext(accountId = "atlas", messageId = 1) {
|
|
setRuntimeConfigSnapshot(baseCfg);
|
|
return await buildTelegramMessageContextForTest({
|
|
cfg: baseCfg,
|
|
accountId,
|
|
message: buildNamedAccountDmMessage(messageId),
|
|
});
|
|
}
|
|
|
|
it("allows DM through for a named account with no explicit binding", async () => {
|
|
setRuntimeConfigSnapshot(baseCfg);
|
|
|
|
const ctx = await buildTelegramMessageContextForTest({
|
|
cfg: baseCfg,
|
|
accountId: "atlas",
|
|
message: {
|
|
message_id: 1,
|
|
chat: { id: 814912386, type: "private" },
|
|
date: 1700000000,
|
|
text: "hello",
|
|
from: { id: 814912386, first_name: "Alice" },
|
|
},
|
|
});
|
|
|
|
expect(ctx).not.toBeNull();
|
|
expect(ctx?.route.matchedBy).toBe("default");
|
|
expect(ctx?.route.accountId).toBe("atlas");
|
|
});
|
|
|
|
it("uses a per-account session key for named-account DMs", async () => {
|
|
const ctx = await buildNamedAccountDmContext();
|
|
|
|
expect(ctx?.ctxPayload?.SessionKey).toBe("agent:main:telegram:atlas:direct:814912386");
|
|
});
|
|
|
|
it("keeps named-account fallback lastRoute on the isolated DM session", async () => {
|
|
const ctx = await buildNamedAccountDmContext();
|
|
|
|
expect(ctx?.ctxPayload?.SessionKey).toBe("agent:main:telegram:atlas:direct:814912386");
|
|
expect(getLastUpdateLastRoute()?.sessionKey).toBe("agent:main:telegram:atlas:direct:814912386");
|
|
});
|
|
|
|
it("isolates sessions between named accounts that share the default agent", async () => {
|
|
const atlas = await buildNamedAccountDmContext("atlas", 1);
|
|
const skynet = await buildNamedAccountDmContext("skynet", 2);
|
|
|
|
expect(atlas?.ctxPayload?.SessionKey).toBe("agent:main:telegram:atlas:direct:814912386");
|
|
expect(skynet?.ctxPayload?.SessionKey).toBe("agent:main:telegram:skynet:direct:814912386");
|
|
expect(atlas?.ctxPayload?.SessionKey).not.toBe(skynet?.ctxPayload?.SessionKey);
|
|
});
|
|
|
|
it("keeps identity-linked peer canonicalization in the named-account fallback path", async () => {
|
|
const cfg = {
|
|
...baseCfg,
|
|
session: {
|
|
identityLinks: {
|
|
"alice-shared": ["telegram:814912386"],
|
|
},
|
|
},
|
|
};
|
|
setRuntimeConfigSnapshot(cfg);
|
|
|
|
const ctx = await buildTelegramMessageContextForTest({
|
|
cfg,
|
|
accountId: "atlas",
|
|
message: {
|
|
message_id: 1,
|
|
chat: { id: 999999999, type: "private" },
|
|
date: 1700000000,
|
|
text: "hello",
|
|
from: { id: 814912386, first_name: "Alice" },
|
|
},
|
|
});
|
|
|
|
expect(ctx?.ctxPayload?.SessionKey).toBe("agent:main:telegram:atlas:direct:alice-shared");
|
|
});
|
|
|
|
it("still drops named-account group messages without an explicit binding", async () => {
|
|
setRuntimeConfigSnapshot(baseCfg);
|
|
|
|
const ctx = await buildTelegramMessageContextForTest({
|
|
cfg: baseCfg,
|
|
accountId: "atlas",
|
|
options: { forceWasMentioned: true },
|
|
resolveGroupActivation: () => true,
|
|
message: {
|
|
message_id: 1,
|
|
chat: { id: -1001234567890, type: "supergroup", title: "Test Group" },
|
|
date: 1700000000,
|
|
text: "@bot hello",
|
|
from: { id: 814912386, first_name: "Alice" },
|
|
},
|
|
});
|
|
|
|
expect(ctx).toBeNull();
|
|
});
|
|
|
|
it("does not change the default-account DM session key", async () => {
|
|
setRuntimeConfigSnapshot(baseCfg);
|
|
|
|
const ctx = await buildTelegramMessageContextForTest({
|
|
cfg: baseCfg,
|
|
message: {
|
|
message_id: 1,
|
|
chat: { id: 42, type: "private" },
|
|
date: 1700000000,
|
|
text: "hello",
|
|
from: { id: 42, first_name: "Alice" },
|
|
},
|
|
});
|
|
|
|
expect(ctx?.ctxPayload?.SessionKey).toBe("agent:main:main");
|
|
});
|
|
});
|