cc-switch连接one-api,token消耗但claude界面无返回内容

Anthropic Adaptor Streaming 兼容性修复记录

前置条件

中转服务,使用one-api中转连接小米MIMO渠道,渠道类型选择Anthropic,设置好令牌后,局域网其他设备使用这个对外令牌,连接方式也用cc-switch连接,API格式选择OpenAI Chat Completions,并开启路由(cc使用apenai格式必须开启路由,one-api中转对外也必须是openai格式)。但使用claude code开启服务后,发送请求有token消耗,但是界面没有任何输出,查询cc-switch日志发现下述问题。

问题描述

使用 cc-switch 通过 one-api 中转调用 Anthropic Claude 模型时,流式响应报错:

复制代码
Unknown finish_reason in streaming

根本原因

relay/adaptor/anthropic/main.go 中存在 3 个兼容性缺陷:

  1. stopReasonClaude2OpenAI 直接透传未知的 stop_reason --- Claude 返回了 OpenAI 协议中不存在的 finish_reason 值,导致客户端解析失败
  2. 未过滤 thinking 类型的 content block --- Claude 的 extended thinking 返回 thinking / thinking_delta / signature_delta 等块,OpenAI 协议不支持这些类型,直接传递会导致客户端异常
  3. 缺少对控制事件的处理 --- content_block_stopmessage_stop 等事件未被识别,进入了默认的响应构建逻辑,产生了无效数据

修复方案

修改文件:relay/adaptor/anthropic/main.go(commit fe82335

1. 未知 stop_reason 映射为 "stop"

go 复制代码
// 修复前
default:
    return *reason

// 修复后
default:
    // unknown stop reason, map to "stop" to avoid client errors
    return "stop"

2. 跳过 thinking 类型的 content block

go 复制代码
// content_block_start 阶段
case "content_block_start":
    if claudeResponse.ContentBlock != nil {
+       if claudeResponse.ContentBlock.Type == "thinking" {
+           return nil, nil
+       }
        // ...
    }

// content_block_delta 阶段
case "content_block_delta":
    if claudeResponse.Delta != nil {
+       if claudeResponse.Delta.Type == "thinking_delta" || claudeResponse.Delta.Type == "signature_delta" {
+           return nil, nil
+       }
        // ...
    }

3. 处理未知控制事件

go 复制代码
// 修复前:没有 default 分支,控制事件进入响应构建逻辑

// 修复后
default:
    // content_block_stop, message_stop and other control events produce no OpenAI chunk
    return nil, nil

4. 仅在有 stop_reason 时设置 finish_reason

go 复制代码
// 修复前
finishReason := stopReasonClaude2OpenAI(&stopReason)
if finishReason != "null" {
    choice.FinishReason = &finishReason
}

// 修复后
if stopReason != "" {
    finishReason := stopReasonClaude2OpenAI(&stopReason)
    choice.FinishReason = &finishReason
}

影响范围

  • 仅修改 relay/adaptor/anthropic/main.go 一个文件(+15 / -3)
  • 仅影响 Anthropic → OpenAI 协议转换的流式响应路径
  • 不影响非流式调用、不影响其他 adaptor

验证方式

通过 cc-switch 向 one-api 中转发起 Claude 流式调用,确认不再出现 Unknown finish_reason in streaming 错误。Claude code界面也正常有返回内容输出,问题解决。

相关代码已经提交到github源仓库,待通过审核~

相关推荐
m0_535817553 小时前
macOS上Claude Code安装配置保姆级教程:国内直连API,从0到1跑通(附避坑指南)
gpt·macos·ai·node.js·claude·claudecode·88api
四眼蒙面侠6 小时前
从 Agent 到代码:Claude Code 编排模型的演进
claudecode·动态工作流·dynamicworkflows
Soari1 天前
【功能演进】Claude Code v2.1.153:交互逻辑重大反转,后台 Agent 体验大修
aiagent·claudecode·model逻辑反转
m0_535817552 天前
从0到1上手Claude Code:Windows安装+88api配置全流程
windows·gpt·node.js·claude·claudecode·88api
LSG_Dawn3 天前
ubuntu22.04 部署 claudecode详细步骤
linux·ai编程·claudecode
m0_535817553 天前
Claude Code在Linux/WSL2环境完整部署指南:API中转+避坑配置一篇搞定
linux·服务器·node.js·claude·wsl2·claudecode·88api
拙野5 天前
【保姆级教程】Claude Code无缝集成DeepSeek V4 Pro
java·人工智能·deepseek·claudecode·ai coding
深念Y6 天前
理解大模型API缓存机制:从Claude Code的缓存失效到DeepSeek的硬盘缓存
缓存·ai·api·提示词·kvcache·vibecoding·claudecode