coding-proxy 0.2.4a3__tar.gz → 0.2.4a4__tar.gz

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (149) hide show
  1. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/CHANGELOG.md +2 -0
  2. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/PKG-INFO +1 -1
  3. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/pyproject.toml +1 -1
  4. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/routing/executor.py +42 -10
  5. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/server/request_normalizer.py +144 -0
  6. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_request_normalizer.py +511 -0
  7. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/uv.lock +1 -1
  8. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/.github/workflows/ci.yml +0 -0
  9. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/.github/workflows/coverage.yml +0 -0
  10. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/.github/workflows/release.yml +0 -0
  11. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/.gitignore +0 -0
  12. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/.pre-commit-config.yaml +0 -0
  13. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/AGENTS.md +0 -0
  14. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/CLAUDE.md +0 -0
  15. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/LICENSE +0 -0
  16. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/README.md +0 -0
  17. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/assets/dashboard-v0.2.3.png +0 -0
  18. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/docs/ci-cd.md +0 -0
  19. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/docs/framework.md +0 -0
  20. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/docs/user-guide.md +0 -0
  21. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/docs/zh-CN/README.md +0 -0
  22. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/__init__.py +0 -0
  23. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/__init__.py +0 -0
  24. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/__main__.py +0 -0
  25. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/auth/__init__.py +0 -0
  26. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/auth/providers/__init__.py +0 -0
  27. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/auth/providers/base.py +0 -0
  28. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/auth/providers/github.py +0 -0
  29. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/auth/providers/google.py +0 -0
  30. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/auth/runtime.py +0 -0
  31. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/auth/store.py +0 -0
  32. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/cli/__init__.py +0 -0
  33. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/cli/auth_commands.py +0 -0
  34. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/cli/banner.py +0 -0
  35. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/compat/__init__.py +0 -0
  36. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/compat/canonical.py +0 -0
  37. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/compat/session_store.py +0 -0
  38. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/config/__init__.py +0 -0
  39. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/config/auth_schema.py +0 -0
  40. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/config/config.default.yaml +0 -0
  41. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/config/loader.py +0 -0
  42. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/config/resiliency.py +0 -0
  43. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/config/routing.py +0 -0
  44. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/config/schema.py +0 -0
  45. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/config/server.py +0 -0
  46. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/config/vendors.py +0 -0
  47. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/convert/__init__.py +0 -0
  48. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/convert/anthropic_to_gemini.py +0 -0
  49. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/convert/anthropic_to_openai.py +0 -0
  50. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/convert/gemini_sse_adapter.py +0 -0
  51. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/convert/gemini_to_anthropic.py +0 -0
  52. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/convert/openai_to_anthropic.py +0 -0
  53. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/logging/__init__.py +0 -0
  54. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/logging/db.py +0 -0
  55. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/logging/formatters.py +0 -0
  56. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/logging/stats.py +0 -0
  57. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/model/__init__.py +0 -0
  58. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/model/auth.py +0 -0
  59. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/model/compat.py +0 -0
  60. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/model/constants.py +0 -0
  61. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/model/pricing.py +0 -0
  62. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/model/token.py +0 -0
  63. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/model/vendor.py +0 -0
  64. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/pricing.py +0 -0
  65. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/routing/__init__.py +0 -0
  66. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/routing/circuit_breaker.py +0 -0
  67. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/routing/error_classifier.py +0 -0
  68. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/routing/model_mapper.py +0 -0
  69. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/routing/quota_guard.py +0 -0
  70. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/routing/rate_limit.py +0 -0
  71. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/routing/retry.py +0 -0
  72. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/routing/router.py +0 -0
  73. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/routing/session_manager.py +0 -0
  74. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/routing/tier.py +0 -0
  75. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/routing/usage_parser.py +0 -0
  76. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/routing/usage_recorder.py +0 -0
  77. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/server/__init__.py +0 -0
  78. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/server/app.py +0 -0
  79. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/server/dashboard.py +0 -0
  80. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/server/factory.py +0 -0
  81. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/server/responses.py +0 -0
  82. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/server/routes.py +0 -0
  83. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/streaming/__init__.py +0 -0
  84. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/streaming/anthropic_compat.py +0 -0
  85. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/vendors/__init__.py +0 -0
  86. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/vendors/alibaba.py +0 -0
  87. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/vendors/anthropic.py +0 -0
  88. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/vendors/antigravity.py +0 -0
  89. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/vendors/base.py +0 -0
  90. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/vendors/copilot.py +0 -0
  91. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/vendors/copilot_models.py +0 -0
  92. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/vendors/copilot_token_manager.py +0 -0
  93. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/vendors/copilot_urls.py +0 -0
  94. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/vendors/doubao.py +0 -0
  95. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/vendors/kimi.py +0 -0
  96. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/vendors/minimax.py +0 -0
  97. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/vendors/mixins.py +0 -0
  98. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/vendors/native_anthropic.py +0 -0
  99. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/vendors/token_manager.py +0 -0
  100. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/vendors/xiaomi.py +0 -0
  101. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/src/coding/proxy/vendors/zhipu.py +0 -0
  102. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/__init__.py +0 -0
  103. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_antigravity.py +0 -0
  104. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_app_routes.py +0 -0
  105. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_auto_login.py +0 -0
  106. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_banner.py +0 -0
  107. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_circuit_breaker.py +0 -0
  108. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_cli_usage.py +0 -0
  109. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_compat.py +0 -0
  110. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_config_init.py +0 -0
  111. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_config_loader.py +0 -0
  112. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_convert_request.py +0 -0
  113. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_convert_response.py +0 -0
  114. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_convert_sse.py +0 -0
  115. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_copilot.py +0 -0
  116. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_copilot_convert_request.py +0 -0
  117. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_copilot_convert_response.py +0 -0
  118. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_copilot_models.py +0 -0
  119. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_copilot_urls.py +0 -0
  120. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_currency.py +0 -0
  121. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_error_classifier.py +0 -0
  122. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_logging_dual_write.py +0 -0
  123. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_mixins.py +0 -0
  124. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_model_auth.py +0 -0
  125. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_model_compat.py +0 -0
  126. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_model_constants.py +0 -0
  127. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_model_mapper.py +0 -0
  128. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_model_pricing.py +0 -0
  129. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_model_token.py +0 -0
  130. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_model_vendor.py +0 -0
  131. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_native_vendors.py +0 -0
  132. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_parse_usage.py +0 -0
  133. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_pricing.py +0 -0
  134. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_quota_guard.py +0 -0
  135. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_rate_limit.py +0 -0
  136. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_router_chain.py +0 -0
  137. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_router_executor.py +0 -0
  138. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_runtime_reauth.py +0 -0
  139. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_schema.py +0 -0
  140. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_streaming_anthropic_compat.py +0 -0
  141. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_tier.py +0 -0
  142. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_tiers_config.py +0 -0
  143. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_time_range.py +0 -0
  144. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_token_logger.py +0 -0
  145. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_token_manager.py +0 -0
  146. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_types.py +0 -0
  147. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_vendor_streaming.py +0 -0
  148. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_vendors.py +0 -0
  149. {coding_proxy-0.2.4a3 → coding_proxy-0.2.4a4}/tests/test_zhipu.py +0 -0
@@ -4,6 +4,8 @@
4
4
 
5
5
  ## [Unreleased]
6
6
 
7
+ - fix(request-normalizer): 重设计 zhipu→anthropic 跨供应商 tool_use/tool_result 配对修复——以单遍自包含 `enforce_anthropic_tool_pairing` 替代原有多步串联管线(剥离→重定位→孤儿修复),消除步骤间隐式依赖导致的孤儿 tool_use 漏修问题,彻底根治 `tool_use ids were found without tool_result blocks` 400 异常;
8
+
7
9
  ## [v0.2.3](https://github.com/ThreeFish-AI/coding-proxy/releases/tag/v0.2.3) — 2026-04-16
8
10
 
9
11
  - feat(dashboard): 新增实时 Web Dashboard 页面,聚合展示流量与用量统计;
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: coding-proxy
3
- Version: 0.2.4a3
3
+ Version: 0.2.4a4
4
4
  Summary: A High-Availability, Transparent, and Smart Multi-Vendor Proxy for Claude Code. Support Claude Plans, GitHub Copilot, Google Antigravity, ZAI/GLM, MiniMax, Qwen, Xiaomi, Kimi, Doubao...
5
5
  Project-URL: Source Code, https://github.com/ThreeFish-AI/coding-proxy
6
6
  Project-URL: User Guide, https://github.com/ThreeFish-AI/coding-proxy/blob/master/docs/user-guide.md
@@ -1,6 +1,6 @@
1
1
  [project]
2
2
  name = "coding-proxy"
3
- version = "0.2.4a3"
3
+ version = "0.2.4a4"
4
4
  description = "A High-Availability, Transparent, and Smart Multi-Vendor Proxy for Claude Code. Support Claude Plans, GitHub Copilot, Google Antigravity, ZAI/GLM, MiniMax, Qwen, Xiaomi, Kimi, Doubao..."
5
5
  readme = "README.md"
6
6
  requires-python = ">=3.12"
@@ -233,7 +233,7 @@ class _RouteExecutor:
233
233
  """为指定 tier 准备请求体,必要时应用 Anthropic 专属修复(Phase 2).
234
234
 
235
235
  仅当 tier 为 Anthropic 时才执行以下处理:
236
- 1. tool_result 重定位 + 孤儿修复(需 normalization.has_anthropic_fixes)
236
+ 1. 跨供应商 tool_use/tool_result 配对强制修复(单遍自包含扫描)
237
237
  2. 条件化 thinking block 剥离(仅跨供应商场景)
238
238
 
239
239
  确保 Zhipu 等其他 vendor 不受影响。
@@ -241,30 +241,28 @@ class _RouteExecutor:
241
241
  if tier.name != "anthropic":
242
242
  return body
243
243
 
244
- needs_tool_fixes = (
245
- normalization is not None and normalization.has_anthropic_fixes
244
+ needs_tool_pairing = self._needs_tool_pairing_enforcement(
245
+ normalization, session_record
246
246
  )
247
247
  needs_thinking_strip = self._needs_thinking_strip(normalization, session_record)
248
248
 
249
- if not needs_tool_fixes and not needs_thinking_strip:
249
+ if not needs_tool_pairing and not needs_thinking_strip:
250
250
  return body
251
251
 
252
252
  from ..server.request_normalizer import (
253
- apply_anthropic_specific_fixes,
253
+ enforce_anthropic_tool_pairing,
254
254
  strip_thinking_blocks,
255
255
  )
256
256
 
257
257
  body_for_vendor = copy.deepcopy(body)
258
258
 
259
- if needs_tool_fixes:
260
- fixes = apply_anthropic_specific_fixes(
259
+ if needs_tool_pairing:
260
+ fixes = enforce_anthropic_tool_pairing(
261
261
  body_for_vendor.get("messages", []),
262
- normalization.misplaced_tool_results,
263
- normalization.misplaced_log_info,
264
262
  )
265
263
  if fixes:
266
264
  logger.debug(
267
- "Applied Anthropic-specific fixes for tier %s: %s",
265
+ "Applied tool pairing enforcement for tier %s: %s",
268
266
  tier.name,
269
267
  ", ".join(fixes),
270
268
  )
@@ -279,6 +277,40 @@ class _RouteExecutor:
279
277
 
280
278
  return body_for_vendor
281
279
 
280
+ @staticmethod
281
+ def _needs_tool_pairing_enforcement(
282
+ normalization: Any, session_record: Any
283
+ ) -> bool:
284
+ """判断是否需要强制执行 Anthropic tool_use/tool_result 配对修复.
285
+
286
+ 此方法扩展了原有 ``has_anthropic_fixes`` 的触发条件,覆盖以下场景:
287
+
288
+ 1. 请求体中检测到跨供应商产物(如非标准 ID、misplaced tool_result)
289
+ 2. Phase 1 检测到需要 Anthropic 修复(misplaced 或 ID 重写)
290
+ 3. 会话历史中存在非 Anthropic 供应商记录(如 zhipu)
291
+ 4. 无会话追踪能力时安全回退
292
+
293
+ 条件 3 和 4 确保即使请求体本身无跨供应商产物(如 zhipu 使用标准
294
+ ``toolu_*`` ID 时),只要会话曾经过非 Anthropic 供应商,仍会执行配对修复。
295
+ """
296
+ # Signal 1: 当前请求体有跨供应商产物
297
+ if normalization is not None and normalization.has_cross_vendor_signals:
298
+ return True
299
+ # Signal 2: Phase 1 检测到需要 Anthropic 修复
300
+ if normalization is not None and normalization.has_anthropic_fixes:
301
+ return True
302
+ # Signal 3: 无会话追踪 → 安全回退
303
+ if session_record is None:
304
+ return True
305
+ # Signal 4: 会话历史中有非 Anthropic 供应商
306
+ if session_record.provider_state:
307
+ non_anthropic = {
308
+ v for v in session_record.provider_state if v != "anthropic"
309
+ }
310
+ if non_anthropic:
311
+ return True
312
+ return False
313
+
282
314
  @staticmethod
283
315
  def _needs_thinking_strip(normalization: Any, session_record: Any) -> bool:
284
316
  """判断是否需要剥离 thinking blocks(仅跨供应商场景).
@@ -487,6 +487,150 @@ def _repair_orphaned_tool_use(
487
487
  return repaired
488
488
 
489
489
 
490
+ # ── Phase 2: 跨供应商 tool_use/tool_result 配对强制修复 ─────────
491
+
492
+
493
+ def enforce_anthropic_tool_pairing(
494
+ messages_list: list[dict[str, Any]],
495
+ ) -> list[str]:
496
+ """为跨供应商场景强制保证 Anthropic tool_use/tool_result 配对约束.
497
+
498
+ 单次正向遍历所有消息,对每个 assistant 消息执行:
499
+
500
+ 1. 剥离所有 tool_result 块(跨供应商产物,如 GLM-5 内联的 tool_result)
501
+ 2. 收集所有 tool_use ID
502
+ 3. 确保紧邻的下一条消息是 user 消息且包含所有必需的 tool_result
503
+ 4. 将剥离的 tool_result 重定位到正确的 user 消息
504
+ 5. 为仍缺失的 tool_result 合成 ``is_error=True`` 的占位块
505
+
506
+ 此函数是一个**自包含的单遍处理**,不依赖 Phase 1 收集的 misplaced 信息,
507
+ 通过直接扫描消息列表确保处理的完备性。替代此前多步串联管线
508
+ (剥离 → 重定位 → 孤儿修复)因步骤间隐式依赖导致的漏修问题。
509
+
510
+ 仅在请求实际发送给 Anthropic tier 且检测到跨供应商信号时调用,
511
+ 确保 Zhipu 等其他 vendor 不受影响。
512
+
513
+ Args:
514
+ messages_list: 消息列表(就地修改)。
515
+
516
+ Returns:
517
+ 新增的 adaptation 标签列表。
518
+ """
519
+ adaptations: list[str] = []
520
+ relocated_count = 0
521
+ synthesized_ids: list[str] = []
522
+
523
+ i = 0
524
+ while i < len(messages_list):
525
+ msg = messages_list[i]
526
+ if not isinstance(msg, dict) or msg.get("role") != "assistant":
527
+ i += 1
528
+ continue
529
+
530
+ content = msg.get("content")
531
+ if not isinstance(content, list):
532
+ i += 1
533
+ continue
534
+
535
+ # ── A. 从 assistant 消息中剥离所有 tool_result 块 ─────────
536
+ extracted_tool_results: dict[str, dict[str, Any]] = {} # tool_use_id → block
537
+ retained_content: list[Any] = []
538
+ for block in content:
539
+ if isinstance(block, dict) and block.get("type") == "tool_result":
540
+ tid = block.get("tool_use_id")
541
+ if tid:
542
+ extracted_tool_results[tid] = block
543
+ relocated_count += 1
544
+ # 无 tool_use_id 的 tool_result 直接丢弃(无效块)
545
+ else:
546
+ retained_content.append(block)
547
+
548
+ if extracted_tool_results:
549
+ msg["content"] = retained_content
550
+
551
+ # ── B. 收集所有 tool_use ID ───────────────────────────────
552
+ tool_use_ids: list[str] = [
553
+ b["id"]
554
+ for b in (
555
+ msg.get("content") if isinstance(msg.get("content"), list) else []
556
+ )
557
+ if isinstance(b, dict) and b.get("type") == "tool_use" and b.get("id")
558
+ ]
559
+ if not tool_use_ids:
560
+ # 无 tool_use 块:若剥离后 content 为空,插入占位
561
+ current_content = msg.get("content")
562
+ if isinstance(current_content, list) and not current_content:
563
+ msg["content"] = [{"type": "text", "text": ""}]
564
+ i += 1
565
+ continue
566
+
567
+ # ── C. 确保 messages[i+1] 是 user 消息 ───────────────────
568
+ next_idx = i + 1
569
+ if (
570
+ next_idx < len(messages_list)
571
+ and isinstance(messages_list[next_idx], dict)
572
+ and messages_list[next_idx].get("role") == "user"
573
+ ):
574
+ user_msg = messages_list[next_idx]
575
+ else:
576
+ # 插入合成 user 消息
577
+ user_msg: dict[str, Any] = {"role": "user", "content": []}
578
+ messages_list.insert(next_idx, user_msg)
579
+
580
+ # ── D. 确保 user_msg.content 是 list ─────────────────────
581
+ user_content = user_msg.get("content")
582
+ if isinstance(user_content, str):
583
+ user_msg["content"] = [{"type": "text", "text": user_content}]
584
+ elif not isinstance(user_content, list):
585
+ user_msg["content"] = []
586
+
587
+ # ── E. 收集 user 消息中已有的 tool_result IDs ─────────────
588
+ existing_result_ids: set[str] = {
589
+ b["tool_use_id"]
590
+ for b in user_msg["content"]
591
+ if isinstance(b, dict)
592
+ and b.get("type") == "tool_result"
593
+ and b.get("tool_use_id")
594
+ }
595
+
596
+ # ── F. 为每个 tool_use_id 确保 tool_result 存在 ──────────
597
+ for uid in tool_use_ids:
598
+ if uid in existing_result_ids:
599
+ continue # 已有匹配的 tool_result
600
+
601
+ if uid in extracted_tool_results:
602
+ # 从 assistant 剥离的 tool_result 重定位到 user
603
+ user_msg["content"].append(extracted_tool_results[uid])
604
+ else:
605
+ # 完全缺失:合成 is_error=True 占位块
606
+ user_msg["content"].append(
607
+ {
608
+ "type": "tool_result",
609
+ "tool_use_id": uid,
610
+ "content": "",
611
+ "is_error": True,
612
+ }
613
+ )
614
+ synthesized_ids.append(uid)
615
+
616
+ i += 1
617
+
618
+ # ── 构建 adaptation 标签与日志 ────────────────────────────
619
+ if relocated_count:
620
+ adaptations.append("misplaced_tool_result_relocated")
621
+ if synthesized_ids:
622
+ adaptations.append("orphaned_tool_use_repaired")
623
+ logger.warning(
624
+ "Vendor degradation adaptation: synthesized %d tool_result block(s) "
625
+ "for orphaned tool_use to satisfy Anthropic pairing constraint. "
626
+ "Affected tool_use_ids: %s",
627
+ len(synthesized_ids),
628
+ ", ".join(synthesized_ids),
629
+ )
630
+
631
+ return adaptations
632
+
633
+
490
634
  # ── Phase 2: Thinking block 剥离 ──────────────────────────────
491
635
 
492
636
  # 需要从 assistant messages 中剥离的 thinking block 类型