772ec46c98
- Fix Rust syntax/clippy fallout in client.rs, cli/src/lib.rs, web_search.rs - Fix 0.8.53 release metadata: changelog links, TUI changelog, npm wrapper - Update visible help copy for multi-provider support - Add telegram-bridge integration with deploy configs - Add US remote VM quickstart doc - Update Tencent Cloud deploy scripts and docs - Bump npm wrapper to 0.8.53
305 lines
8.6 KiB
JavaScript
305 lines
8.6 KiB
JavaScript
import test from "node:test";
|
|
import assert from "node:assert/strict";
|
|
|
|
import {
|
|
activeTurnBlock,
|
|
activeTurnKeyboard,
|
|
approvalKeyboard,
|
|
callbackAction,
|
|
commandAction,
|
|
controlKeyboard,
|
|
envFirst,
|
|
helpText,
|
|
isAllowed,
|
|
pairingRefusalText,
|
|
parseApprovalDecisionArgs,
|
|
parseBool,
|
|
parseCommand,
|
|
parseEnvText,
|
|
parseList,
|
|
preservedChatStateFields,
|
|
splitMessage,
|
|
stripGroupPrefix,
|
|
threadListKeyboard,
|
|
telegramIdentity,
|
|
telegramRetryDelayMs,
|
|
looksLikePollingConflict,
|
|
validateBridgeConfig
|
|
} from "../src/lib.mjs";
|
|
|
|
test("envFirst returns first non-empty value", () => {
|
|
assert.equal(envFirst({ A: "", B: " value " }, "A", "B"), "value");
|
|
assert.equal(envFirst({ A: "x" }, "B"), "");
|
|
});
|
|
|
|
test("parseList trims empty values", () => {
|
|
assert.deepEqual(parseList(" 123, @user ,, "), ["123", "@user"]);
|
|
});
|
|
|
|
test("parseBool accepts common truthy values", () => {
|
|
assert.equal(parseBool("yes"), true);
|
|
assert.equal(parseBool("0", true), false);
|
|
assert.equal(parseBool(undefined, true), true);
|
|
});
|
|
|
|
test("parseEnvText handles comments, export, and quoted values", () => {
|
|
assert.deepEqual(
|
|
parseEnvText(`
|
|
# ignored
|
|
export TELEGRAM_GROUP_PREFIX="/cw"
|
|
CODEWHALE_WORKSPACE='/opt/whalebro'
|
|
`),
|
|
{
|
|
TELEGRAM_GROUP_PREFIX: "/cw",
|
|
CODEWHALE_WORKSPACE: "/opt/whalebro"
|
|
}
|
|
);
|
|
});
|
|
|
|
test("telegramIdentity extracts chat and sender identifiers", () => {
|
|
const identity = telegramIdentity({
|
|
update_id: 10,
|
|
message: {
|
|
message_id: 20,
|
|
text: "hello",
|
|
chat: { id: -1001, type: "supergroup" },
|
|
from: { id: 42, username: "hunter", first_name: "Hunter" }
|
|
}
|
|
});
|
|
assert.deepEqual(identity, {
|
|
updateId: 10,
|
|
chatId: "-1001",
|
|
messageId: "20",
|
|
chatType: "supergroup",
|
|
userId: "42",
|
|
username: "@hunter",
|
|
firstName: "Hunter",
|
|
text: "hello",
|
|
isBot: false
|
|
});
|
|
});
|
|
|
|
test("stripGroupPrefix requires prefix in Telegram groups", () => {
|
|
assert.deepEqual(
|
|
stripGroupPrefix("/cw inspect this", {
|
|
chatType: "group",
|
|
requirePrefix: true,
|
|
prefix: "/cw"
|
|
}),
|
|
{ accepted: true, text: "inspect this" }
|
|
);
|
|
assert.equal(
|
|
stripGroupPrefix("inspect this", {
|
|
chatType: "group",
|
|
requirePrefix: true,
|
|
prefix: "/cw"
|
|
}).accepted,
|
|
false
|
|
);
|
|
});
|
|
|
|
test("stripGroupPrefix accepts private chat text without group prefix", () => {
|
|
assert.deepEqual(
|
|
stripGroupPrefix("inspect this", {
|
|
chatType: "private",
|
|
requirePrefix: true,
|
|
prefix: "/cw"
|
|
}),
|
|
{ accepted: true, text: "inspect this" }
|
|
);
|
|
});
|
|
|
|
test("parseCommand handles Telegram bot mentions", () => {
|
|
assert.deepEqual(parseCommand("hello"), { name: "prompt", args: "hello" });
|
|
assert.deepEqual(parseCommand("/allow@CodeWhaleBot abc remember"), {
|
|
name: "allow",
|
|
args: "abc remember"
|
|
});
|
|
});
|
|
|
|
test("commandAction maps bridge commands and falls back to prompts", () => {
|
|
assert.deepEqual(commandAction(parseCommand("/menu")), { kind: "menu" });
|
|
assert.deepEqual(commandAction(parseCommand("/status")), { kind: "status" });
|
|
assert.deepEqual(commandAction(parseCommand("/resume thread-1")), {
|
|
kind: "resume",
|
|
threadId: "thread-1"
|
|
});
|
|
assert.deepEqual(commandAction(parseCommand("/model arcee-trinity")), {
|
|
kind: "set_model",
|
|
modelName: "arcee-trinity"
|
|
});
|
|
assert.deepEqual(commandAction(parseCommand("/unknown value")), {
|
|
kind: "prompt",
|
|
prompt: "/unknown value"
|
|
});
|
|
});
|
|
|
|
test("helpText documents per-chat model switching", () => {
|
|
assert.match(helpText(), /\/model <name\|default>/);
|
|
assert.match(helpText(), /\/menu/);
|
|
});
|
|
|
|
test("control keyboards expose modal actions", () => {
|
|
assert.deepEqual(controlKeyboard().inline_keyboard[0][0], {
|
|
text: "Status",
|
|
callback_data: "cw:status"
|
|
});
|
|
assert.deepEqual(activeTurnKeyboard().inline_keyboard[0][1], {
|
|
text: "Interrupt",
|
|
callback_data: "cw:interrupt"
|
|
});
|
|
assert.deepEqual(approvalKeyboard("tok1").inline_keyboard[1][0], {
|
|
text: "Deny",
|
|
callback_data: "cw:act:tok1:deny"
|
|
});
|
|
assert.deepEqual(threadListKeyboard([{ token: "t1", label: "Resume 1" }]).inline_keyboard[0][0], {
|
|
text: "Resume 1",
|
|
callback_data: "cw:act:t1"
|
|
});
|
|
});
|
|
|
|
test("callbackAction parses modal callback payloads", () => {
|
|
assert.deepEqual(callbackAction("cw:status"), { kind: "status" });
|
|
assert.deepEqual(callbackAction("cw:model:default"), {
|
|
kind: "set_model",
|
|
modelName: "default"
|
|
});
|
|
assert.deepEqual(callbackAction("cw:act:tok1:remember"), {
|
|
kind: "stored_action",
|
|
token: "tok1",
|
|
suffix: "remember"
|
|
});
|
|
assert.equal(callbackAction("unknown"), null);
|
|
});
|
|
|
|
test("preservedChatStateFields carries model across state replacement", () => {
|
|
assert.deepEqual(
|
|
preservedChatStateFields({
|
|
threadId: "old-thread",
|
|
model: "mimo-v2.5-pro",
|
|
activeTurnId: "turn-1"
|
|
}),
|
|
{
|
|
model: "mimo-v2.5-pro"
|
|
}
|
|
);
|
|
assert.deepEqual(preservedChatStateFields({ model: null }), { model: null });
|
|
});
|
|
|
|
test("parseApprovalDecisionArgs extracts remember flag", () => {
|
|
assert.deepEqual(parseApprovalDecisionArgs("ap_123 remember"), {
|
|
approvalId: "ap_123",
|
|
remember: true
|
|
});
|
|
assert.deepEqual(parseApprovalDecisionArgs(""), { approvalId: "", remember: false });
|
|
});
|
|
|
|
test("isAllowed checks Telegram chat/user/username identifiers", () => {
|
|
assert.equal(
|
|
isAllowed({ chatId: "-1001", userId: "42", username: "@hunter" }, ["42"], false),
|
|
true
|
|
);
|
|
assert.equal(isAllowed({ chatId: "-1001" }, [], false), false);
|
|
assert.equal(isAllowed({ chatId: "-1001" }, [], true), true);
|
|
});
|
|
|
|
test("pairingRefusalText includes allowlist identifiers", () => {
|
|
const body = pairingRefusalText({
|
|
chatId: "-1001",
|
|
userId: "42",
|
|
username: "@hunter"
|
|
});
|
|
assert.match(body, /chat_id=-1001/);
|
|
assert.match(body, /user_id=42/);
|
|
assert.match(body, /username=@hunter/);
|
|
});
|
|
|
|
test("activeTurnBlock reports active queued or in-progress turn", () => {
|
|
assert.equal(activeTurnBlock({ turns: [{ id: "done", status: "completed" }] }), null);
|
|
assert.deepEqual(
|
|
activeTurnBlock({
|
|
turns: [
|
|
{ id: "old", status: "completed" },
|
|
{ id: "turn-2", status: "queued" }
|
|
]
|
|
}),
|
|
{
|
|
turnId: "turn-2",
|
|
message: "Thread already has active turn turn-2. Wait for it to finish or send /interrupt."
|
|
}
|
|
);
|
|
});
|
|
|
|
test("splitMessage chunks long text without splitting surrogate pairs", () => {
|
|
assert.deepEqual(splitMessage("a🧪b", 2), ["a🧪", "b"]);
|
|
});
|
|
|
|
test("telegramRetryDelayMs honors retry_after", () => {
|
|
assert.equal(telegramRetryDelayMs({ parameters: { retry_after: 2 } }), 2000);
|
|
});
|
|
|
|
test("looksLikePollingConflict detects Telegram 409 conflicts", () => {
|
|
assert.equal(looksLikePollingConflict({ errorCode: 409 }), true);
|
|
assert.equal(
|
|
looksLikePollingConflict({
|
|
message: "Conflict: terminated by other getUpdates request"
|
|
}),
|
|
true
|
|
);
|
|
});
|
|
|
|
test("validateBridgeConfig accepts locked-down whalebro DM config", () => {
|
|
const result = validateBridgeConfig(
|
|
{
|
|
TELEGRAM_BOT_TOKEN: "123456:token",
|
|
CODEWHALE_RUNTIME_URL: "http://127.0.0.1:7878",
|
|
CODEWHALE_RUNTIME_TOKEN: "token-a",
|
|
CODEWHALE_WORKSPACE: "/opt/whalebro",
|
|
TELEGRAM_CHAT_ALLOWLIST: "42",
|
|
TELEGRAM_ALLOW_UNLISTED: "false",
|
|
TELEGRAM_THREAD_MAP_PATH: "/var/lib/codewhale-telegram-bridge/thread-map.json",
|
|
TELEGRAM_ALLOW_GROUPS: "false",
|
|
TELEGRAM_REQUIRE_PREFIX_IN_GROUP: "true"
|
|
},
|
|
{
|
|
workspaceRoot: "/opt/whalebro",
|
|
runtimeEnv: {
|
|
CODEWHALE_RUNTIME_TOKEN: "token-a",
|
|
CODEWHALE_PROVIDER: "arcee",
|
|
CODEWHALE_RUNTIME_PORT: "7878"
|
|
}
|
|
}
|
|
);
|
|
assert.equal(result.ok, true);
|
|
assert.equal(result.errors.length, 0);
|
|
});
|
|
|
|
test("validateBridgeConfig rejects unsafe group pairing and token mismatch", () => {
|
|
const result = validateBridgeConfig(
|
|
{
|
|
TELEGRAM_BOT_TOKEN: "123456:token",
|
|
CODEWHALE_RUNTIME_URL: "http://127.0.0.1:7878",
|
|
CODEWHALE_RUNTIME_TOKEN: "bridge-token",
|
|
CODEWHALE_WORKSPACE: "/opt/whalebro",
|
|
TELEGRAM_ALLOW_UNLISTED: "true",
|
|
TELEGRAM_THREAD_MAP_PATH: "/var/lib/codewhale-telegram-bridge/thread-map.json",
|
|
TELEGRAM_ALLOW_GROUPS: "true",
|
|
TELEGRAM_REQUIRE_PREFIX_IN_GROUP: "false"
|
|
},
|
|
{
|
|
workspaceRoot: "/opt/whalebro",
|
|
runtimeEnv: {
|
|
CODEWHALE_RUNTIME_TOKEN: "runtime-token",
|
|
CODEWHALE_PROVIDER: "arcee"
|
|
}
|
|
}
|
|
);
|
|
assert.equal(result.ok, false);
|
|
assert.match(
|
|
result.errors.map((item) => item.code).join(","),
|
|
/open_group_control/
|
|
);
|
|
assert.match(result.errors.map((item) => item.code).join(","), /token_mismatch/);
|
|
assert.match(result.warnings.map((item) => item.code).join(","), /group_without_prefix/);
|
|
});
|