mirror of
https://github.com/claude-code-best/claude-code.git
synced 2026-06-17 13:55:50 +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>
149 lines
3.1 KiB
TypeScript
149 lines
3.1 KiB
TypeScript
import { randomBytes } from 'node:crypto'
|
|
import type {
|
|
BaseInfo,
|
|
GetConfigResp,
|
|
GetUpdatesReq,
|
|
GetUpdatesResp,
|
|
GetUploadUrlReq,
|
|
GetUploadUrlResp,
|
|
SendMessageReq,
|
|
SendTypingReq,
|
|
SendTypingResp,
|
|
} from './types.js'
|
|
|
|
const CHANNEL_VERSION = '0.1.0'
|
|
|
|
function baseInfo(): BaseInfo {
|
|
return { channel_version: CHANNEL_VERSION }
|
|
}
|
|
|
|
function randomUin(): string {
|
|
return randomBytes(4).toString('base64')
|
|
}
|
|
|
|
function buildHeaders(token?: string): Record<string, string> {
|
|
const headers: Record<string, string> = {
|
|
'Content-Type': 'application/json',
|
|
'X-WECHAT-UIN': randomUin(),
|
|
}
|
|
if (token) {
|
|
headers.AuthorizationType = 'ilink_bot_token'
|
|
headers.Authorization = `Bearer ${token}`
|
|
}
|
|
return headers
|
|
}
|
|
|
|
async function post<T>(
|
|
baseUrl: string,
|
|
path: string,
|
|
body: unknown,
|
|
token?: string,
|
|
timeoutMs = 40_000,
|
|
signal?: AbortSignal,
|
|
): Promise<T> {
|
|
const controller = new AbortController()
|
|
const timeout = setTimeout(() => controller.abort(), timeoutMs)
|
|
|
|
if (signal) {
|
|
signal.addEventListener('abort', () => controller.abort(), { once: true })
|
|
}
|
|
|
|
try {
|
|
const response = await fetch(`${baseUrl}${path}`, {
|
|
method: 'POST',
|
|
headers: buildHeaders(token),
|
|
body: JSON.stringify(body),
|
|
signal: controller.signal,
|
|
})
|
|
|
|
if (!response.ok) {
|
|
throw new Error(`HTTP ${response.status}: ${response.statusText}`)
|
|
}
|
|
|
|
return (await response.json()) as T
|
|
} finally {
|
|
clearTimeout(timeout)
|
|
}
|
|
}
|
|
|
|
export async function getUpdates(
|
|
baseUrl: string,
|
|
token: string,
|
|
getUpdatesBuf: string,
|
|
signal?: AbortSignal,
|
|
): Promise<GetUpdatesResp> {
|
|
const body: GetUpdatesReq = {
|
|
get_updates_buf: getUpdatesBuf,
|
|
base_info: baseInfo(),
|
|
}
|
|
|
|
try {
|
|
return await post<GetUpdatesResp>(
|
|
baseUrl,
|
|
'/ilink/bot/getupdates',
|
|
body,
|
|
token,
|
|
40_000,
|
|
signal,
|
|
)
|
|
} catch (error) {
|
|
if (error instanceof Error && error.name === 'AbortError') {
|
|
return { ret: 0, msgs: [], get_updates_buf: getUpdatesBuf }
|
|
}
|
|
throw error
|
|
}
|
|
}
|
|
|
|
export async function sendMessage(
|
|
baseUrl: string,
|
|
token: string,
|
|
msg: SendMessageReq['msg'],
|
|
): Promise<void> {
|
|
const body: SendMessageReq = { msg, base_info: baseInfo() }
|
|
await post(baseUrl, '/ilink/bot/sendmessage', body, token)
|
|
}
|
|
|
|
export async function getUploadUrl(
|
|
baseUrl: string,
|
|
token: string,
|
|
params: Omit<GetUploadUrlReq, 'base_info'>,
|
|
): Promise<GetUploadUrlResp> {
|
|
return post<GetUploadUrlResp>(
|
|
baseUrl,
|
|
'/ilink/bot/getuploadurl',
|
|
{ ...params, base_info: baseInfo() },
|
|
token,
|
|
)
|
|
}
|
|
|
|
export async function getConfig(
|
|
baseUrl: string,
|
|
token: string,
|
|
userId: string,
|
|
contextToken?: string,
|
|
): Promise<GetConfigResp> {
|
|
return post<GetConfigResp>(
|
|
baseUrl,
|
|
'/ilink/bot/getconfig',
|
|
{
|
|
ilink_user_id: userId,
|
|
context_token: contextToken,
|
|
base_info: baseInfo(),
|
|
},
|
|
token,
|
|
)
|
|
}
|
|
|
|
export async function sendTyping(
|
|
baseUrl: string,
|
|
token: string,
|
|
req: Omit<SendTypingReq, 'base_info'>,
|
|
): Promise<SendTypingResp> {
|
|
return post<SendTypingResp>(
|
|
baseUrl,
|
|
'/ilink/bot/sendtyping',
|
|
{ ...req, base_info: baseInfo() },
|
|
token,
|
|
)
|
|
}
|