mirror of
https://github.com/claude-code-best/claude-code.git
synced 2026-06-22 00:05:51 +00:00
* feat: 接入 weixin 服务层与命令入口 Ultraworked with [Sisyphus](https://github.com/code-yeongyu/oh-my-openagent) Co-authored-by: Sisyphus <clio-agent@sisyphuslabs.ai> * feat: 注册内建 weixin channel 插件 Ultraworked with [Sisyphus](https://github.com/code-yeongyu/oh-my-openagent) Co-authored-by: Sisyphus <clio-agent@sisyphuslabs.ai> * fix: 修正 channel permission relay 路由与能力判定 Ultraworked with [Sisyphus](https://github.com/code-yeongyu/oh-my-openagent) Co-authored-by: Sisyphus <clio-agent@sisyphuslabs.ai> * fix: 修复 builtin channel 的 ChannelsNotice 误报 Ultraworked with [Sisyphus](https://github.com/code-yeongyu/oh-my-openagent) Co-authored-by: Sisyphus <clio-agent@sisyphuslabs.ai> * docs: 补充内建 weixin channel 使用说明 Ultraworked with [Sisyphus](https://github.com/code-yeongyu/oh-my-openagent) Co-authored-by: Sisyphus <clio-agent@sisyphuslabs.ai> * docs: 更新微信 channel 接入计划状态 Ultraworked with [Sisyphus](https://github.com/code-yeongyu/oh-my-openagent) Co-authored-by: Sisyphus <clio-agent@sisyphuslabs.ai> * fix: 延迟加载 weixin 登录二维码依赖 Ultraworked with [Sisyphus](https://github.com/code-yeongyu/oh-my-openagent) Co-authored-by: Sisyphus <clio-agent@sisyphuslabs.ai> * fix: 改用 qrcode 生成 weixin 登录二维码 Ultraworked with [Sisyphus](https://github.com/code-yeongyu/oh-my-openagent) Co-authored-by: Sisyphus <clio-agent@sisyphuslabs.ai> * fix: 修正 vite 构建的 Windows 路径解析 Ultraworked with [Sisyphus](https://github.com/code-yeongyu/oh-my-openagent) Co-authored-by: Sisyphus <clio-agent@sisyphuslabs.ai> * chore: 删除临时规划文档 wx_channel.md 并还原 package.json 排序 wx_channel.md 内容已整合到 docs/features/channels.md,不再需要。 package.json 中 @ant/model-provider 位置从原始位置被无意移动,还原。 Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com> * refactor: 将 weixin 模块从 src/ 迁移至 packages/weixin 工作区包 将 src/services/weixin/ 中的纯业务逻辑迁入 @claude-code-best/weixin workspace 包,降低 src/ 耦合度。仅保留 server.ts 作为薄适配层。 - 迁移 7 个无修改的纯模块 (types/api/accounts/login/pairing/media/send) - monitor.ts 内联 PERMISSION_REPLY_RE 正则,解除对 src/ 的依赖 - permissions.ts 本地定义 ChannelPermissionRequestParams 接口 - cli.ts 拆分:serve 子命令通过回调注入,login/access 保留在包内 - server.ts 重写为从 @claude-code-best/weixin 导入 - 新增 cli-serve.ts 作为 serve 入口薄壳 Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com> * fix: 修正 weixin barrel export 中 interface 的导出方式 ChannelPermissionRequestParams 是纯类型,必须用 export type 导出, 否则 Bun 运行时会报 "export not found" 错误。 Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com> * refactor: 将 server.ts 迁入 packages/weixin,彻底移除 src/services/weixin/ 通过依赖注入(WeixinServerDeps)解耦 src/ 依赖(analytics、config、 MCP channel schema),server.ts 完全移入包内。cli.tsx 入口处一次性 注入所有依赖。 src/services/weixin/ 目录已完全删除。 Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com> * fix: 修复 markdownToPlainText 中代码块正则的 ReDoS 风险 用非正则的线性扫描替代 \`\`\`[\s\S]*?\n([\s\S]*?)\`\`\` 匹配, 避免在含有大量重复 \`\`\` 序列的输入上触发多项式回溯。 Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com> --------- Co-authored-by: 1111 <11111@asd.c> Co-authored-by: Sisyphus <clio-agent@sisyphuslabs.ai> Co-authored-by: Claude Opus 4.6 <noreply@anthropic.com>
195 lines
5.6 KiB
TypeScript
195 lines
5.6 KiB
TypeScript
import { mock, describe, expect, test } from "bun:test";
|
|
|
|
mock.module("src/services/analytics/growthbook.js", () => ({
|
|
getFeatureValue_CACHED_MAY_BE_STALE: () => false,
|
|
}));
|
|
|
|
const {
|
|
filterPermissionRelayClients,
|
|
shortRequestId,
|
|
truncateForPreview,
|
|
PERMISSION_REPLY_RE,
|
|
createChannelPermissionCallbacks,
|
|
} = await import("../channelPermissions");
|
|
|
|
describe("shortRequestId", () => {
|
|
test("returns 5-char string from tool use ID", () => {
|
|
const result = shortRequestId("toolu_abc123");
|
|
expect(result).toHaveLength(5);
|
|
});
|
|
|
|
test("is deterministic (same input = same output)", () => {
|
|
const a = shortRequestId("toolu_abc123");
|
|
const b = shortRequestId("toolu_abc123");
|
|
expect(a).toBe(b);
|
|
});
|
|
|
|
test("different inputs produce different outputs", () => {
|
|
const a = shortRequestId("toolu_aaa");
|
|
const b = shortRequestId("toolu_bbb");
|
|
expect(a).not.toBe(b);
|
|
});
|
|
|
|
test("result contains only valid letters (no 'l')", () => {
|
|
const validChars = new Set("abcdefghijkmnopqrstuvwxyz");
|
|
for (let i = 0; i < 50; i++) {
|
|
const result = shortRequestId(`toolu_${i}`);
|
|
for (const ch of result) {
|
|
expect(validChars.has(ch)).toBe(true);
|
|
}
|
|
}
|
|
});
|
|
|
|
test("handles empty string", () => {
|
|
const result = shortRequestId("");
|
|
expect(result).toHaveLength(5);
|
|
});
|
|
});
|
|
|
|
describe("truncateForPreview", () => {
|
|
test("returns JSON string for object input", () => {
|
|
const result = truncateForPreview({ key: "value" });
|
|
expect(result).toBe('{"key":"value"}');
|
|
});
|
|
|
|
test("truncates to <=200 chars with ellipsis when input is long", () => {
|
|
const longObj = { data: "x".repeat(300) };
|
|
const result = truncateForPreview(longObj);
|
|
expect(result.length).toBeLessThanOrEqual(203); // 200 + '…'
|
|
expect(result.endsWith("…")).toBe(true);
|
|
});
|
|
|
|
test("returns short input unchanged", () => {
|
|
const result = truncateForPreview({ a: 1 });
|
|
expect(result).toBe('{"a":1}');
|
|
expect(result.endsWith("…")).toBe(false);
|
|
});
|
|
|
|
test("handles string input", () => {
|
|
const result = truncateForPreview("hello");
|
|
expect(result).toBe('"hello"');
|
|
});
|
|
|
|
test("handles null input", () => {
|
|
const result = truncateForPreview(null);
|
|
expect(result).toBe("null");
|
|
});
|
|
|
|
test("handles undefined input", () => {
|
|
const result = truncateForPreview(undefined);
|
|
// JSON.stringify(undefined) returns undefined, then .length throws → catch returns '(unserializable)'
|
|
expect(result).toBe("(unserializable)");
|
|
});
|
|
});
|
|
|
|
describe("PERMISSION_REPLY_RE", () => {
|
|
test("matches 'y abcde'", () => {
|
|
expect(PERMISSION_REPLY_RE.test("y abcde")).toBe(true);
|
|
});
|
|
|
|
test("matches 'yes abcde'", () => {
|
|
expect(PERMISSION_REPLY_RE.test("yes abcde")).toBe(true);
|
|
});
|
|
|
|
test("matches 'n abcde'", () => {
|
|
expect(PERMISSION_REPLY_RE.test("n abcde")).toBe(true);
|
|
});
|
|
|
|
test("matches 'no abcde'", () => {
|
|
expect(PERMISSION_REPLY_RE.test("no abcde")).toBe(true);
|
|
});
|
|
|
|
test("is case-insensitive", () => {
|
|
expect(PERMISSION_REPLY_RE.test("Y abcde")).toBe(true);
|
|
expect(PERMISSION_REPLY_RE.test("YES abcde")).toBe(true);
|
|
});
|
|
|
|
test("does not match without ID", () => {
|
|
expect(PERMISSION_REPLY_RE.test("yes")).toBe(false);
|
|
});
|
|
|
|
test("captures the ID from reply", () => {
|
|
const match = "y abcde".match(PERMISSION_REPLY_RE);
|
|
expect(match?.[2]).toBe("abcde");
|
|
});
|
|
});
|
|
|
|
describe("createChannelPermissionCallbacks", () => {
|
|
test("resolve returns false for unknown request ID", () => {
|
|
const cb = createChannelPermissionCallbacks();
|
|
expect(cb.resolve("unknown-id", "allow", "server")).toBe(false);
|
|
});
|
|
|
|
test("onResponse + resolve triggers handler", () => {
|
|
const cb = createChannelPermissionCallbacks();
|
|
let received: any = null;
|
|
cb.onResponse("test-id", (response) => {
|
|
received = response;
|
|
});
|
|
expect(cb.resolve("test-id", "allow", "test-server")).toBe(true);
|
|
expect(received).toEqual({
|
|
behavior: "allow",
|
|
fromServer: "test-server",
|
|
});
|
|
});
|
|
|
|
test("onResponse unsubscribe prevents resolve", () => {
|
|
const cb = createChannelPermissionCallbacks();
|
|
let called = false;
|
|
const unsub = cb.onResponse("test-id", () => {
|
|
called = true;
|
|
});
|
|
unsub();
|
|
expect(cb.resolve("test-id", "allow", "server")).toBe(false);
|
|
expect(called).toBe(false);
|
|
});
|
|
|
|
test("duplicate resolve returns false (already consumed)", () => {
|
|
const cb = createChannelPermissionCallbacks();
|
|
cb.onResponse("test-id", () => {});
|
|
expect(cb.resolve("test-id", "allow", "server")).toBe(true);
|
|
expect(cb.resolve("test-id", "allow", "server")).toBe(false);
|
|
});
|
|
|
|
test("is case-insensitive for request IDs", () => {
|
|
const cb = createChannelPermissionCallbacks();
|
|
let received: any = null;
|
|
cb.onResponse("ABC", (response) => {
|
|
received = response;
|
|
});
|
|
expect(cb.resolve("abc", "deny", "server")).toBe(true);
|
|
expect(received?.behavior).toBe("deny");
|
|
});
|
|
});
|
|
|
|
describe("filterPermissionRelayClients", () => {
|
|
test("requires truthy permission capability", () => {
|
|
const clients = [
|
|
{
|
|
type: "connected",
|
|
name: "plugin:weixin:weixin",
|
|
capabilities: {
|
|
experimental: {
|
|
"claude/channel": {},
|
|
"claude/channel/permission": false,
|
|
},
|
|
},
|
|
},
|
|
{
|
|
type: "connected",
|
|
name: "plugin:telegram:telegram",
|
|
capabilities: {
|
|
experimental: {
|
|
"claude/channel": {},
|
|
"claude/channel/permission": {},
|
|
},
|
|
},
|
|
},
|
|
];
|
|
|
|
expect(
|
|
filterPermissionRelayClients(clients, () => true).map(client => client.name),
|
|
).toEqual(["plugin:telegram:telegram"]);
|
|
});
|
|
});
|