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>
181 lines
4.7 KiB
TypeScript
181 lines
4.7 KiB
TypeScript
import { randomUUID } from 'node:crypto'
|
|
import type { CDNMedia, MessageItem } from './types.js'
|
|
import { sendMessage } from './api.js'
|
|
import { guessMediaType, uploadFile } from './media.js'
|
|
import { MessageItemType, MessageState, MessageType } from './types.js'
|
|
|
|
function stripCodeBlocks(text: string): string {
|
|
// Non-regex approach to avoid ReDoS on inputs with many ``` sequences.
|
|
let result = ''
|
|
let i = 0
|
|
while (i < text.length) {
|
|
if (text.startsWith('```', i)) {
|
|
// Skip the opening fence (including optional language tag on same line)
|
|
let j = i + 3
|
|
// skip to end of first line (the fence line itself)
|
|
while (j < text.length && text[j] !== '\n') j++
|
|
if (j < text.length) j++ // skip the \n
|
|
// Collect content until closing ```
|
|
const contentStart = j
|
|
while (j < text.length) {
|
|
if (text.startsWith('```', j)) {
|
|
result += text.slice(contentStart, j)
|
|
// skip closing fence and its trailing newline
|
|
j += 3
|
|
while (j < text.length && text[j] !== '\n') j++
|
|
if (j < text.length) j++ // skip \n
|
|
break
|
|
}
|
|
j++
|
|
}
|
|
// If no closing fence found, include rest as-is
|
|
if (j >= text.length && !text.startsWith('```', j - 3)) {
|
|
result += text.slice(i)
|
|
}
|
|
i = j
|
|
} else {
|
|
result += text[i]
|
|
i++
|
|
}
|
|
}
|
|
return result
|
|
}
|
|
|
|
export function markdownToPlainText(text: string): string {
|
|
return stripCodeBlocks(text)
|
|
.replace(/`([^`]+)`/g, '$1')
|
|
.replace(/\*\*\*(.+?)\*\*\*/g, '$1')
|
|
.replace(/\*\*(.+?)\*\*/g, '$1')
|
|
.replace(/\*(.+?)\*/g, '$1')
|
|
.replace(/___(.+?)___/g, '$1')
|
|
.replace(/__(.+?)__/g, '$1')
|
|
.replace(/_(.+?)_/g, '$1')
|
|
.replace(/~~(.+?)~~/g, '$1')
|
|
.replace(/^#{1,6}\s+/gm, '')
|
|
.replace(/\[([^\]]+)\]\(([^)]+)\)/g, '$1 ($2)')
|
|
.replace(/!\[([^\]]*)\]\([^)]+\)/g, '[$1]')
|
|
.replace(/^>\s+/gm, '')
|
|
.replace(/^[-*_]{3,}$/gm, '---')
|
|
.replace(/^[\s]*[-*+]\s+/gm, '- ')
|
|
.replace(/^[\s]*(\d+)\.\s+/gm, '$1. ')
|
|
.replace(/\n{3,}/g, '\n\n')
|
|
.trim()
|
|
}
|
|
|
|
export async function sendText(params: {
|
|
to: string
|
|
text: string
|
|
baseUrl: string
|
|
token: string
|
|
contextToken: string
|
|
}): Promise<{ messageId: string }> {
|
|
const clientId = randomUUID()
|
|
await sendMessage(params.baseUrl, params.token, {
|
|
to_user_id: params.to,
|
|
from_user_id: '',
|
|
client_id: clientId,
|
|
message_type: MessageType.BOT,
|
|
message_state: MessageState.FINISH,
|
|
context_token: params.contextToken,
|
|
item_list: [
|
|
{
|
|
type: MessageItemType.TEXT,
|
|
text_item: { text: markdownToPlainText(params.text) },
|
|
},
|
|
],
|
|
})
|
|
|
|
return { messageId: clientId }
|
|
}
|
|
|
|
async function sendItems(params: {
|
|
items: MessageItem[]
|
|
to: string
|
|
baseUrl: string
|
|
token: string
|
|
contextToken: string
|
|
}): Promise<string> {
|
|
let lastClientId = ''
|
|
for (const item of params.items) {
|
|
lastClientId = randomUUID()
|
|
await sendMessage(params.baseUrl, params.token, {
|
|
to_user_id: params.to,
|
|
from_user_id: '',
|
|
client_id: lastClientId,
|
|
message_type: MessageType.BOT,
|
|
message_state: MessageState.FINISH,
|
|
context_token: params.contextToken,
|
|
item_list: [item],
|
|
})
|
|
}
|
|
return lastClientId
|
|
}
|
|
|
|
export async function sendMediaFile(params: {
|
|
filePath: string
|
|
to: string
|
|
text: string
|
|
baseUrl: string
|
|
token: string
|
|
contextToken: string
|
|
cdnBaseUrl: string
|
|
}): Promise<{ messageId: string }> {
|
|
const mediaType = guessMediaType(params.filePath)
|
|
const uploaded = await uploadFile({
|
|
filePath: params.filePath,
|
|
toUserId: params.to,
|
|
mediaType,
|
|
apiBaseUrl: params.baseUrl,
|
|
token: params.token,
|
|
cdnBaseUrl: params.cdnBaseUrl,
|
|
})
|
|
|
|
const cdnMedia: CDNMedia = {
|
|
encrypt_query_param: uploaded.encryptQueryParam,
|
|
aes_key: uploaded.aesKey,
|
|
encrypt_type: 1,
|
|
}
|
|
|
|
const items: MessageItem[] = []
|
|
if (params.text) {
|
|
items.push({
|
|
type: MessageItemType.TEXT,
|
|
text_item: { text: markdownToPlainText(params.text) },
|
|
})
|
|
}
|
|
|
|
switch (mediaType) {
|
|
case 1:
|
|
items.push({
|
|
type: MessageItemType.IMAGE,
|
|
image_item: { media: cdnMedia, mid_size: uploaded.fileSize },
|
|
})
|
|
break
|
|
case 2:
|
|
items.push({
|
|
type: MessageItemType.VIDEO,
|
|
video_item: { media: cdnMedia, video_size: uploaded.fileSize },
|
|
})
|
|
break
|
|
default:
|
|
items.push({
|
|
type: MessageItemType.FILE,
|
|
file_item: {
|
|
media: cdnMedia,
|
|
file_name: uploaded.fileName,
|
|
len: String(uploaded.rawSize),
|
|
},
|
|
})
|
|
break
|
|
}
|
|
|
|
const messageId = await sendItems({
|
|
items,
|
|
to: params.to,
|
|
baseUrl: params.baseUrl,
|
|
token: params.token,
|
|
contextToken: params.contextToken,
|
|
})
|
|
return { messageId }
|
|
}
|