mirror of
https://github.com/claude-code-best/claude-code.git
synced 2026-06-15 12:55: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>
135 lines
3.4 KiB
TypeScript
135 lines
3.4 KiB
TypeScript
import { toString as qrToString } from 'qrcode'
|
|
|
|
export interface QRCodeResult {
|
|
qrcodeUrl?: string
|
|
qrcodeId: string
|
|
message: string
|
|
}
|
|
|
|
export interface LoginResult {
|
|
connected: boolean
|
|
token?: string
|
|
accountId?: string
|
|
baseUrl?: string
|
|
userId?: string
|
|
message: string
|
|
}
|
|
|
|
async function renderQrCodeToTerminal(qrcodeUrl: string): Promise<void> {
|
|
const output = await qrToString(qrcodeUrl, {
|
|
type: 'terminal',
|
|
errorCorrectionLevel: 'L',
|
|
small: true,
|
|
})
|
|
process.stderr.write(`${output}\n`)
|
|
}
|
|
|
|
export async function startLogin(apiBaseUrl: string): Promise<QRCodeResult> {
|
|
const response = await fetch(`${apiBaseUrl}/ilink/bot/get_bot_qrcode?bot_type=3`)
|
|
if (!response.ok) {
|
|
throw new Error(`Failed to get QR code: HTTP ${response.status}`)
|
|
}
|
|
|
|
const data = (await response.json()) as {
|
|
qrcode?: string
|
|
qrcode_img_content?: string
|
|
}
|
|
|
|
if (!data.qrcode) {
|
|
throw new Error('No qrcode in response')
|
|
}
|
|
|
|
const qrcodeUrl = data.qrcode_img_content || ''
|
|
if (qrcodeUrl) {
|
|
await renderQrCodeToTerminal(qrcodeUrl)
|
|
}
|
|
|
|
return {
|
|
qrcodeUrl,
|
|
qrcodeId: data.qrcode,
|
|
message: 'Scan the QR code with WeChat to connect.',
|
|
}
|
|
}
|
|
|
|
export async function waitForLogin(params: {
|
|
qrcodeId: string
|
|
apiBaseUrl: string
|
|
timeoutMs?: number
|
|
maxRetries?: number
|
|
}): Promise<LoginResult> {
|
|
const { qrcodeId, apiBaseUrl, timeoutMs = 480_000, maxRetries = 3 } = params
|
|
const deadline = Date.now() + timeoutMs
|
|
let currentQrcodeId = qrcodeId
|
|
let retryCount = 0
|
|
|
|
while (Date.now() < deadline) {
|
|
try {
|
|
const controller = new AbortController()
|
|
const timeout = setTimeout(() => controller.abort(), 60_000)
|
|
|
|
const response = await fetch(
|
|
`${apiBaseUrl}/ilink/bot/get_qrcode_status?qrcode=${encodeURIComponent(currentQrcodeId)}`,
|
|
{
|
|
headers: { 'iLink-App-ClientVersion': '1' },
|
|
signal: controller.signal,
|
|
},
|
|
)
|
|
clearTimeout(timeout)
|
|
|
|
if (!response.ok) {
|
|
throw new Error(`HTTP ${response.status}`)
|
|
}
|
|
|
|
const data = (await response.json()) as {
|
|
status?: string
|
|
bot_token?: string
|
|
ilink_bot_id?: string
|
|
baseurl?: string
|
|
ilink_user_id?: string
|
|
}
|
|
|
|
switch (data.status) {
|
|
case 'confirmed':
|
|
return {
|
|
connected: true,
|
|
token: data.bot_token,
|
|
accountId: data.ilink_bot_id,
|
|
baseUrl: data.baseurl,
|
|
userId: data.ilink_user_id,
|
|
message: 'Connected to WeChat successfully!',
|
|
}
|
|
case 'scaned':
|
|
process.stderr.write(
|
|
'QR code scanned, waiting for confirmation...\n',
|
|
)
|
|
break
|
|
case 'expired': {
|
|
retryCount += 1
|
|
if (retryCount >= maxRetries) {
|
|
return {
|
|
connected: false,
|
|
message: 'QR code expired after maximum retries.',
|
|
}
|
|
}
|
|
process.stderr.write('QR code expired, refreshing...\n')
|
|
const refreshed = await startLogin(apiBaseUrl)
|
|
currentQrcodeId = refreshed.qrcodeId
|
|
break
|
|
}
|
|
case 'wait':
|
|
default:
|
|
break
|
|
}
|
|
} catch (error) {
|
|
if (error instanceof Error && error.name === 'AbortError') {
|
|
continue
|
|
}
|
|
throw error
|
|
}
|
|
|
|
await new Promise(resolve => setTimeout(resolve, 1000))
|
|
}
|
|
|
|
return { connected: false, message: 'Login timed out.' }
|
|
}
|