@renxqoo/renx-code 0.0.4 → 0.0.5

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 (209) hide show
  1. package/bin/renx.cjs +16 -0
  2. package/package.json +2 -45
  3. package/src/agent/runtime/runtime.context-usage.test.ts +4 -5
  4. package/src/agent/runtime/runtime.error-handling.test.ts +4 -5
  5. package/src/agent/runtime/runtime.test.ts +7 -4
  6. package/src/agent/runtime/runtime.ts +3 -9
  7. package/src/agent/runtime/runtime.usage-forwarding.test.ts +4 -5
  8. package/src/agent/runtime/source-modules.test.ts +16 -35
  9. package/src/agent/runtime/source-modules.ts +17 -0
  10. package/vendor/agent-root/src/agent/ENTERPRISE_ACCEPTANCE_CHECKLIST.md +95 -0
  11. package/vendor/agent-root/src/agent/ENTERPRISE_REALTIME.html +1345 -0
  12. package/vendor/agent-root/src/agent/ENTERPRISE_REALTIME.md +1353 -0
  13. package/vendor/agent-root/src/agent/ERROR_CONTRACT.md +60 -0
  14. package/vendor/agent-root/src/agent/TEST_COVERAGE_ANALYSIS.md +278 -0
  15. package/vendor/agent-root/src/agent/__test__/error-contract.test.ts +72 -0
  16. package/vendor/agent-root/src/agent/__test__/types.test.ts +137 -0
  17. package/vendor/agent-root/src/agent/agent/__test__/abort-runtime.test.ts +83 -0
  18. package/vendor/agent-root/src/agent/agent/__test__/callback-safety.test.ts +34 -0
  19. package/vendor/agent-root/src/agent/agent/__test__/compaction.test.ts +323 -0
  20. package/vendor/agent-root/src/agent/agent/__test__/concurrency.test.ts +290 -0
  21. package/vendor/agent-root/src/agent/agent/__test__/error-normalizer.test.ts +377 -0
  22. package/vendor/agent-root/src/agent/agent/__test__/error.test.ts +212 -0
  23. package/vendor/agent-root/src/agent/agent/__test__/fault-injection.test.ts +295 -0
  24. package/vendor/agent-root/src/agent/agent/__test__/index.test.ts +3607 -0
  25. package/vendor/agent-root/src/agent/agent/__test__/logger.test.ts +35 -0
  26. package/vendor/agent-root/src/agent/agent/__test__/message-utils.test.ts +517 -0
  27. package/vendor/agent-root/src/agent/agent/__test__/telemetry.test.ts +97 -0
  28. package/vendor/agent-root/src/agent/agent/__test__/timeout-budget.test.ts +479 -0
  29. package/vendor/agent-root/src/agent/agent/__test__/tool-call-merge.test.ts +80 -0
  30. package/vendor/agent-root/src/agent/agent/__test__/tool-execution-ledger.test.ts +76 -0
  31. package/vendor/agent-root/src/agent/agent/__test__/write-buffer.test.ts +173 -0
  32. package/vendor/agent-root/src/agent/agent/__test__/write-file-session.test.ts +109 -0
  33. package/vendor/agent-root/src/agent/agent/abort-runtime.ts +71 -0
  34. package/vendor/agent-root/src/agent/agent/callback-safety.ts +33 -0
  35. package/vendor/agent-root/src/agent/agent/compaction.ts +291 -0
  36. package/vendor/agent-root/src/agent/agent/concurrency.ts +103 -0
  37. package/vendor/agent-root/src/agent/agent/error-normalizer.ts +190 -0
  38. package/vendor/agent-root/src/agent/agent/error.ts +198 -0
  39. package/vendor/agent-root/src/agent/agent/index.ts +1772 -0
  40. package/vendor/agent-root/src/agent/agent/logger.ts +65 -0
  41. package/vendor/agent-root/src/agent/agent/message-utils.ts +101 -0
  42. package/vendor/agent-root/src/agent/agent/stream-events.ts +61 -0
  43. package/vendor/agent-root/src/agent/agent/telemetry.ts +123 -0
  44. package/vendor/agent-root/src/agent/agent/timeout-budget.ts +227 -0
  45. package/vendor/agent-root/src/agent/agent/tool-call-merge.ts +111 -0
  46. package/vendor/agent-root/src/agent/agent/tool-execution-ledger.ts +164 -0
  47. package/vendor/agent-root/src/agent/agent/write-buffer.ts +188 -0
  48. package/vendor/agent-root/src/agent/agent/write-file-session.ts +238 -0
  49. package/vendor/agent-root/src/agent/app/__test__/agent-app-service.test.ts +1053 -0
  50. package/vendor/agent-root/src/agent/app/__test__/minimal-agent-application.test.ts +158 -0
  51. package/vendor/agent-root/src/agent/app/__test__/sqlite-agent-app-store.test.ts +437 -0
  52. package/vendor/agent-root/src/agent/app/agent-app-service.ts +748 -0
  53. package/vendor/agent-root/src/agent/app/contracts.ts +109 -0
  54. package/vendor/agent-root/src/agent/app/index.ts +5 -0
  55. package/vendor/agent-root/src/agent/app/minimal-agent-application.ts +151 -0
  56. package/vendor/agent-root/src/agent/app/ports.ts +72 -0
  57. package/vendor/agent-root/src/agent/app/sqlite-agent-app-store.ts +1182 -0
  58. package/vendor/agent-root/src/agent/app/sqlite-client.ts +177 -0
  59. package/vendor/agent-root/src/agent/docs/cli-app-layer/00-README.md +36 -0
  60. package/vendor/agent-root/src/agent/docs/cli-app-layer/01-scope-and-goals.md +33 -0
  61. package/vendor/agent-root/src/agent/docs/cli-app-layer/02-architecture-overview.md +40 -0
  62. package/vendor/agent-root/src/agent/docs/cli-app-layer/03-domain-model-and-contracts.md +91 -0
  63. package/vendor/agent-root/src/agent/docs/cli-app-layer/04-ports-and-interfaces.md +116 -0
  64. package/vendor/agent-root/src/agent/docs/cli-app-layer/05-run-orchestration-and-state-machine.md +52 -0
  65. package/vendor/agent-root/src/agent/docs/cli-app-layer/06-cli-commands-and-ux.md +53 -0
  66. package/vendor/agent-root/src/agent/docs/cli-app-layer/07-storage-design-local.md +52 -0
  67. package/vendor/agent-root/src/agent/docs/cli-app-layer/08-error-and-observability.md +40 -0
  68. package/vendor/agent-root/src/agent/docs/cli-app-layer/09-security-and-policy-boundary.md +19 -0
  69. package/vendor/agent-root/src/agent/docs/cli-app-layer/10-test-plan-and-acceptance.md +28 -0
  70. package/vendor/agent-root/src/agent/docs/cli-app-layer/11-implementation-phases.md +26 -0
  71. package/vendor/agent-root/src/agent/docs/cli-app-layer/12-open-questions-and-risks.md +30 -0
  72. package/vendor/agent-root/src/agent/docs/cli-app-layer/13-sqlite-schema-fields-and-rationale.md +567 -0
  73. package/vendor/agent-root/src/agent/docs/cli-app-layer/14-project-flow-mermaid.md +583 -0
  74. package/vendor/agent-root/src/agent/docs/cli-app-layer/15-openclaw-style-project-blueprint.md +972 -0
  75. package/vendor/agent-root/src/agent/error-contract.ts +154 -0
  76. package/vendor/agent-root/src/agent/prompts/system.ts +246 -0
  77. package/vendor/agent-root/src/agent/prompts/system1.ts +208 -0
  78. package/vendor/agent-root/src/agent/storage/__test__/file-history-store.test.ts +98 -0
  79. package/vendor/agent-root/src/agent/storage/file-history-store.ts +313 -0
  80. package/vendor/agent-root/src/agent/storage/file-storage-config.ts +94 -0
  81. package/vendor/agent-root/src/agent/storage/file-system.ts +31 -0
  82. package/vendor/agent-root/src/agent/storage/file-write-service.ts +21 -0
  83. package/vendor/agent-root/src/agent/tool/__test__/base-tool.test.ts +413 -0
  84. package/vendor/agent-root/src/agent/tool/__test__/bash-policy.test.ts +356 -0
  85. package/vendor/agent-root/src/agent/tool/__test__/bash.mocked-coverage.test.ts +375 -0
  86. package/vendor/agent-root/src/agent/tool/__test__/bash.test.ts +372 -0
  87. package/vendor/agent-root/src/agent/tool/__test__/error.test.ts +108 -0
  88. package/vendor/agent-root/src/agent/tool/__test__/file-edit-tool.test.ts +258 -0
  89. package/vendor/agent-root/src/agent/tool/__test__/file-history-tools.test.ts +121 -0
  90. package/vendor/agent-root/src/agent/tool/__test__/file-read-tool.test.ts +210 -0
  91. package/vendor/agent-root/src/agent/tool/__test__/glob.test.ts +139 -0
  92. package/vendor/agent-root/src/agent/tool/__test__/grep.mocked-coverage.test.ts +456 -0
  93. package/vendor/agent-root/src/agent/tool/__test__/grep.test.ts +192 -0
  94. package/vendor/agent-root/src/agent/tool/__test__/lsp.test.ts +300 -0
  95. package/vendor/agent-root/src/agent/tool/__test__/outside-workspace-confirmation.test.ts +214 -0
  96. package/vendor/agent-root/src/agent/tool/__test__/path-security.test.ts +336 -0
  97. package/vendor/agent-root/src/agent/tool/__test__/skill-loader.test.ts +494 -0
  98. package/vendor/agent-root/src/agent/tool/__test__/skill-parser.test.ts +543 -0
  99. package/vendor/agent-root/src/agent/tool/__test__/skill-tool.test.ts +172 -0
  100. package/vendor/agent-root/src/agent/tool/__test__/task-concurrency-and-version.test.ts +116 -0
  101. package/vendor/agent-root/src/agent/tool/__test__/task-create-get-list-update.test.ts +267 -0
  102. package/vendor/agent-root/src/agent/tool/__test__/task-create.test.ts +519 -0
  103. package/vendor/agent-root/src/agent/tool/__test__/task-errors.test.ts +225 -0
  104. package/vendor/agent-root/src/agent/tool/__test__/task-output-blocking.test.ts +223 -0
  105. package/vendor/agent-root/src/agent/tool/__test__/task-output.test.ts +184 -0
  106. package/vendor/agent-root/src/agent/tool/__test__/task-parent-abort.test.ts +287 -0
  107. package/vendor/agent-root/src/agent/tool/__test__/task-real-runner-adapter.test.ts +190 -0
  108. package/vendor/agent-root/src/agent/tool/__test__/task-run-lifecycle.test.ts +352 -0
  109. package/vendor/agent-root/src/agent/tool/__test__/task-store-runner-branches.test.ts +395 -0
  110. package/vendor/agent-root/src/agent/tool/__test__/task-store.test.ts +391 -0
  111. package/vendor/agent-root/src/agent/tool/__test__/task-subagent-config-integration.test.ts +176 -0
  112. package/vendor/agent-root/src/agent/tool/__test__/task-subagent-config.test.ts +68 -0
  113. package/vendor/agent-root/src/agent/tool/__test__/task-tools-core-edges.test.ts +630 -0
  114. package/vendor/agent-root/src/agent/tool/__test__/task-tools-runtime-edges.test.ts +732 -0
  115. package/vendor/agent-root/src/agent/tool/__test__/task-types.test.ts +494 -0
  116. package/vendor/agent-root/src/agent/tool/__test__/task-utils-branches.test.ts +175 -0
  117. package/vendor/agent-root/src/agent/tool/__test__/tool-manager.test.ts +505 -0
  118. package/vendor/agent-root/src/agent/tool/__test__/types.test.ts +55 -0
  119. package/vendor/agent-root/src/agent/tool/__test__/web-fetch.test.ts +244 -0
  120. package/vendor/agent-root/src/agent/tool/__test__/web-search.test.ts +290 -0
  121. package/vendor/agent-root/src/agent/tool/__test__/write-file.test.ts +368 -0
  122. package/vendor/agent-root/src/agent/tool/base-tool.ts +345 -0
  123. package/vendor/agent-root/src/agent/tool/bash-policy.ts +636 -0
  124. package/vendor/agent-root/src/agent/tool/bash.ts +688 -0
  125. package/vendor/agent-root/src/agent/tool/error.ts +131 -0
  126. package/vendor/agent-root/src/agent/tool/file-edit-tool.ts +264 -0
  127. package/vendor/agent-root/src/agent/tool/file-history-list.ts +103 -0
  128. package/vendor/agent-root/src/agent/tool/file-history-restore.ts +149 -0
  129. package/vendor/agent-root/src/agent/tool/file-read-tool.ts +211 -0
  130. package/vendor/agent-root/src/agent/tool/glob.ts +171 -0
  131. package/vendor/agent-root/src/agent/tool/grep.ts +496 -0
  132. package/vendor/agent-root/src/agent/tool/lsp.ts +481 -0
  133. package/vendor/agent-root/src/agent/tool/path-security.ts +117 -0
  134. package/vendor/agent-root/src/agent/tool/search/common.ts +153 -0
  135. package/vendor/agent-root/src/agent/tool/skill/index.ts +13 -0
  136. package/vendor/agent-root/src/agent/tool/skill/loader.ts +229 -0
  137. package/vendor/agent-root/src/agent/tool/skill/parser.ts +124 -0
  138. package/vendor/agent-root/src/agent/tool/skill/types.ts +27 -0
  139. package/vendor/agent-root/src/agent/tool/skill-tool.ts +143 -0
  140. package/vendor/agent-root/src/agent/tool/task-create.ts +186 -0
  141. package/vendor/agent-root/src/agent/tool/task-errors.ts +42 -0
  142. package/vendor/agent-root/src/agent/tool/task-get.ts +116 -0
  143. package/vendor/agent-root/src/agent/tool/task-graph.ts +78 -0
  144. package/vendor/agent-root/src/agent/tool/task-list.ts +141 -0
  145. package/vendor/agent-root/src/agent/tool/task-mock-runner-adapter.ts +232 -0
  146. package/vendor/agent-root/src/agent/tool/task-output.ts +223 -0
  147. package/vendor/agent-root/src/agent/tool/task-parent-abort.ts +115 -0
  148. package/vendor/agent-root/src/agent/tool/task-real-runner-adapter.ts +336 -0
  149. package/vendor/agent-root/src/agent/tool/task-runner-adapter.ts +55 -0
  150. package/vendor/agent-root/src/agent/tool/task-stop.ts +187 -0
  151. package/vendor/agent-root/src/agent/tool/task-store.ts +217 -0
  152. package/vendor/agent-root/src/agent/tool/task-subagent-config.ts +149 -0
  153. package/vendor/agent-root/src/agent/tool/task-types.ts +264 -0
  154. package/vendor/agent-root/src/agent/tool/task-update.ts +315 -0
  155. package/vendor/agent-root/src/agent/tool/task.ts +209 -0
  156. package/vendor/agent-root/src/agent/tool/tool-manager.ts +362 -0
  157. package/vendor/agent-root/src/agent/tool/tool-prompts.ts +242 -0
  158. package/vendor/agent-root/src/agent/tool/types.ts +116 -0
  159. package/vendor/agent-root/src/agent/tool/web-fetch.ts +227 -0
  160. package/vendor/agent-root/src/agent/tool/web-search.ts +208 -0
  161. package/vendor/agent-root/src/agent/tool/write-file.ts +497 -0
  162. package/vendor/agent-root/src/agent/types.ts +232 -0
  163. package/vendor/agent-root/src/agent/utils/__tests__/index.test.ts +18 -0
  164. package/vendor/agent-root/src/agent/utils/__tests__/message-utils.test.ts +610 -0
  165. package/vendor/agent-root/src/agent/utils/__tests__/message.test.ts +223 -0
  166. package/vendor/agent-root/src/agent/utils/__tests__/token.test.ts +42 -0
  167. package/vendor/agent-root/src/agent/utils/index.ts +16 -0
  168. package/vendor/agent-root/src/agent/utils/message.ts +171 -0
  169. package/vendor/agent-root/src/agent/utils/token.ts +28 -0
  170. package/vendor/agent-root/src/config/__tests__/load-config-to-env.test.ts +129 -0
  171. package/vendor/agent-root/src/config/__tests__/loader.test.ts +247 -0
  172. package/vendor/agent-root/src/config/__tests__/runtime.test.ts +88 -0
  173. package/vendor/agent-root/src/config/index.ts +54 -0
  174. package/vendor/agent-root/src/config/loader.ts +431 -0
  175. package/vendor/agent-root/src/config/paths.ts +30 -0
  176. package/vendor/agent-root/src/config/runtime.ts +163 -0
  177. package/vendor/agent-root/src/config/types.ts +70 -0
  178. package/vendor/agent-root/src/logger/index.ts +57 -0
  179. package/vendor/agent-root/src/logger/logger.ts +819 -0
  180. package/vendor/agent-root/src/logger/types.ts +150 -0
  181. package/vendor/agent-root/src/providers/__tests__/errors.test.ts +441 -0
  182. package/vendor/agent-root/src/providers/__tests__/index.test.ts +16 -0
  183. package/vendor/agent-root/src/providers/__tests__/openai-compatible.options.test.ts +318 -0
  184. package/vendor/agent-root/src/providers/__tests__/openai-compatible.test.ts +600 -0
  185. package/vendor/agent-root/src/providers/__tests__/registry.test.ts +449 -0
  186. package/vendor/agent-root/src/providers/__tests__/responses-adapter.test.ts +298 -0
  187. package/vendor/agent-root/src/providers/adapters/__tests__/anthropic.test.ts +354 -0
  188. package/vendor/agent-root/src/providers/adapters/__tests__/kimi.test.ts +58 -0
  189. package/vendor/agent-root/src/providers/adapters/__tests__/standard.test.ts +261 -0
  190. package/vendor/agent-root/src/providers/adapters/anthropic.ts +572 -0
  191. package/vendor/agent-root/src/providers/adapters/base.ts +131 -0
  192. package/vendor/agent-root/src/providers/adapters/kimi.ts +48 -0
  193. package/vendor/agent-root/src/providers/adapters/responses.ts +732 -0
  194. package/vendor/agent-root/src/providers/adapters/standard.ts +120 -0
  195. package/vendor/agent-root/src/providers/http/__tests__/client.timeout.test.ts +313 -0
  196. package/vendor/agent-root/src/providers/http/client.ts +289 -0
  197. package/vendor/agent-root/src/providers/http/stream-parser.ts +109 -0
  198. package/vendor/agent-root/src/providers/index.ts +76 -0
  199. package/vendor/agent-root/src/providers/kimi-headers.ts +177 -0
  200. package/vendor/agent-root/src/providers/openai-compatible.ts +387 -0
  201. package/vendor/agent-root/src/providers/registry/model-config.ts +230 -0
  202. package/vendor/agent-root/src/providers/registry/provider-factory.ts +123 -0
  203. package/vendor/agent-root/src/providers/registry.ts +135 -0
  204. package/vendor/agent-root/src/providers/types/api.ts +284 -0
  205. package/vendor/agent-root/src/providers/types/config.ts +58 -0
  206. package/vendor/agent-root/src/providers/types/errors.ts +323 -0
  207. package/vendor/agent-root/src/providers/types/index.ts +72 -0
  208. package/vendor/agent-root/src/providers/types/provider.ts +45 -0
  209. package/vendor/agent-root/src/providers/types/registry.ts +88 -0
@@ -0,0 +1,572 @@
1
+ /**
2
+ * Anthropic API Adapter
3
+ *
4
+ * Supports Anthropic Claude API specification:
5
+ * - Separate system field (not a message role)
6
+ * - Unique authentication headers (x-api-key + anthropic-version)
7
+ * - Different response and streaming formats
8
+ */
9
+
10
+ import { BaseAPIAdapter } from './base';
11
+ import type {
12
+ LLMRequest,
13
+ LLMResponse,
14
+ LLMRequestMessage,
15
+ LLMResponseMessage,
16
+ Chunk,
17
+ MessageContent,
18
+ ToolCall,
19
+ } from '../types';
20
+ import type { ProviderLogger } from '../types';
21
+
22
+ /**
23
+ * Anthropic content block
24
+ */
25
+ interface AnthropicContentBlock {
26
+ type: 'text' | 'image' | 'tool_use' | 'tool_result';
27
+ text?: string;
28
+ source?: {
29
+ type: 'base64' | 'url';
30
+ media_type?: string;
31
+ data?: string;
32
+ url?: string;
33
+ };
34
+ id?: string;
35
+ name?: string;
36
+ input?: Record<string, unknown>;
37
+ tool_use_id?: string;
38
+ content?: string | AnthropicContentBlock[];
39
+ is_error?: boolean;
40
+ }
41
+
42
+ /**
43
+ * Anthropic request message format
44
+ */
45
+ interface AnthropicMessage {
46
+ role: 'user' | 'assistant';
47
+ content: string | AnthropicContentBlock[];
48
+ }
49
+
50
+ /**
51
+ * Anthropic request body
52
+ */
53
+ interface AnthropicRequestBody {
54
+ model: string;
55
+ max_tokens: number;
56
+ system?: string;
57
+ messages: AnthropicMessage[];
58
+ stream?: boolean;
59
+ tools?: Array<{
60
+ name: string;
61
+ description: string;
62
+ input_schema: Record<string, unknown>;
63
+ }>;
64
+ temperature?: number;
65
+ }
66
+
67
+ /**
68
+ * Anthropic response format
69
+ */
70
+ interface AnthropicResponse {
71
+ id: string;
72
+ type: 'message';
73
+ role: 'assistant';
74
+ content: AnthropicContentBlock[];
75
+ model: string;
76
+ stop_reason: 'end_turn' | 'max_tokens' | 'stop_sequence' | 'tool_use' | null;
77
+ stop_sequence: string | null;
78
+ usage: {
79
+ input_tokens: number;
80
+ output_tokens: number;
81
+ [key: string]: unknown;
82
+ };
83
+ }
84
+
85
+ /**
86
+ * Anthropic stream event
87
+ */
88
+ interface AnthropicStreamEvent {
89
+ type: string;
90
+ message?: AnthropicResponse;
91
+ index?: number;
92
+ content_block?: AnthropicContentBlock;
93
+ delta?: {
94
+ type: string;
95
+ text?: string;
96
+ stop_reason?: string;
97
+ partial_json?: string;
98
+ };
99
+ usage?: {
100
+ input_tokens: number;
101
+ output_tokens: number;
102
+ [key: string]: unknown;
103
+ };
104
+ }
105
+
106
+ interface ToolCallStreamState {
107
+ id: string;
108
+ name: string;
109
+ }
110
+
111
+ /**
112
+ * Anthropic API Adapter
113
+ */
114
+ export class AnthropicAdapter extends BaseAPIAdapter {
115
+ readonly endpointPath: string;
116
+ readonly defaultModel: string;
117
+ readonly apiVersion: string;
118
+ readonly logger?: ProviderLogger;
119
+
120
+ constructor(
121
+ options: {
122
+ endpointPath?: string;
123
+ defaultModel?: string;
124
+ apiVersion?: string;
125
+ logger?: ProviderLogger;
126
+ } = {}
127
+ ) {
128
+ super();
129
+ this.endpointPath = options.endpointPath ?? '/v1/messages';
130
+ this.defaultModel = options.defaultModel ?? 'claude-opus-4-6-20250528';
131
+ this.apiVersion = options.apiVersion ?? '2023-06-01';
132
+ this.logger = options.logger?.child('AnthropicAdapter');
133
+ }
134
+
135
+ /**
136
+ * Transform request to Anthropic format
137
+ */
138
+ transformRequest(options?: LLMRequest): LLMRequest {
139
+ if (!options) {
140
+ return { model: this.defaultModel, messages: [] };
141
+ }
142
+
143
+ const { messages, tools, ...rest } = options;
144
+
145
+ // Extract system messages
146
+ let systemPrompt = '';
147
+ const conversationMessages: LLMRequestMessage[] = [];
148
+
149
+ for (const msg of messages || []) {
150
+ if (msg.role === 'system') {
151
+ const content = this.extractTextContent(msg.content);
152
+ if (content) {
153
+ systemPrompt += (systemPrompt ? '\n\n' : '') + content;
154
+ }
155
+ } else {
156
+ conversationMessages.push(msg);
157
+ }
158
+ }
159
+ // Convert message format
160
+ const anthropicMessages = this.convertMessages(conversationMessages);
161
+
162
+ const body: AnthropicRequestBody = {
163
+ model: rest.model || this.defaultModel,
164
+ max_tokens: rest.max_tokens || 4096,
165
+ messages: anthropicMessages,
166
+ stream: rest.stream ?? false,
167
+ };
168
+ if (systemPrompt) {
169
+ body.system = systemPrompt;
170
+ }
171
+ if (rest.temperature !== undefined) {
172
+ body.temperature = rest.temperature;
173
+ }
174
+ // convert tool definitions
175
+ if (tools && tools.length > 0) {
176
+ body.tools = tools.map((tool) => ({
177
+ name: tool.function.name,
178
+ description: tool.function.description,
179
+ input_schema: tool.function.parameters,
180
+ }));
181
+ }
182
+ return body as unknown as LLMRequest;
183
+ }
184
+ /**
185
+ * Transform response to standard format
186
+ */
187
+ transformResponse(response: unknown): LLMResponse {
188
+ const anthropicResp = response as AnthropicResponse;
189
+ // extract text content and tool calls
190
+ const content: string[] = [];
191
+ const toolCalls: ToolCall[] = [];
192
+ for (const block of anthropicResp.content || []) {
193
+ if (block.type === 'text' && block.text) {
194
+ content.push(block.text);
195
+ } else if (block.type === 'tool_use' && block.id && block.name) {
196
+ toolCalls.push({
197
+ id: block.id,
198
+ type: 'function',
199
+ index: toolCalls.length,
200
+ function: {
201
+ name: block.name,
202
+ arguments: typeof block.input === 'string' ? block.input : JSON.stringify(block.input),
203
+ },
204
+ });
205
+ }
206
+ }
207
+ const message: LLMResponseMessage = {
208
+ role: 'assistant',
209
+ content: content.join(''),
210
+ };
211
+ if (toolCalls.length > 0) {
212
+ message.tool_calls = toolCalls;
213
+ }
214
+ // convert stop_reason
215
+ let finishReason: 'stop' | 'length' | 'tool_calls' | null = null;
216
+ if (anthropicResp.stop_reason === 'end_turn') {
217
+ finishReason = 'stop';
218
+ } else if (anthropicResp.stop_reason === 'max_tokens') {
219
+ finishReason = 'length';
220
+ } else if (anthropicResp.stop_reason === 'tool_use') {
221
+ finishReason = 'tool_calls';
222
+ }
223
+ return {
224
+ id: anthropicResp.id,
225
+ object: 'chat.completion',
226
+ created: Math.floor(Date.now() / 1000),
227
+ model: anthropicResp.model,
228
+ choices: [
229
+ {
230
+ index: 0,
231
+ message,
232
+ finish_reason: finishReason,
233
+ },
234
+ ],
235
+ usage: {
236
+ ...anthropicResp.usage,
237
+ prompt_tokens: anthropicResp.usage?.input_tokens || 0,
238
+ completion_tokens: anthropicResp.usage?.output_tokens || 0,
239
+ total_tokens:
240
+ (anthropicResp.usage?.input_tokens || 0) + (anthropicResp.usage?.output_tokens || 0),
241
+ },
242
+ };
243
+ }
244
+ /**
245
+ * Get request headers
246
+ */
247
+ getHeaders(apiKey: string): Headers {
248
+ return new Headers({
249
+ 'Content-Type': 'application/json',
250
+ 'x-api-key': apiKey,
251
+ 'anthropic-version': this.apiVersion,
252
+ });
253
+ }
254
+ /**
255
+ * Get endpoint path
256
+ */
257
+ getEndpointPath(): string {
258
+ return this.endpointPath;
259
+ }
260
+ /**
261
+ * Parse Anthropic stream response
262
+ */
263
+ async *parseStreamAsync(reader: ReadableStreamDefaultReader<Uint8Array>): AsyncGenerator<Chunk> {
264
+ const decoder = new TextDecoder();
265
+ let buffer = '';
266
+ const toolCallStates = new Map<number, ToolCallStreamState>();
267
+
268
+ const baseChunk: Partial<Chunk> = {};
269
+ try {
270
+ while (true) {
271
+ const { done, value } = await reader.read();
272
+ if (done) {
273
+ break;
274
+ }
275
+ buffer += decoder.decode(value, { stream: true });
276
+ const lines = buffer.split('\n');
277
+ buffer = lines.pop() ?? '';
278
+ for (const line of lines) {
279
+ const trimmed = line.trim();
280
+ if (!trimmed || !trimmed.startsWith('data:')) continue;
281
+ const data = trimmed.slice(5).trim();
282
+ if (!data) continue;
283
+ let event: AnthropicStreamEvent;
284
+ try {
285
+ event = JSON.parse(data);
286
+ } catch {
287
+ continue;
288
+ }
289
+ // update base chunk info
290
+ if (event.type === 'message_start' && event.message) {
291
+ baseChunk.id = event.message.id;
292
+ baseChunk.model = event.message.model;
293
+ }
294
+ // check if stream end
295
+ if (this.isStreamEndEvent(event)) {
296
+ return;
297
+ }
298
+ // parse event
299
+ const chunk = this.parseStreamEvent(event, baseChunk, toolCallStates);
300
+ if (chunk) {
301
+ yield chunk;
302
+ }
303
+ }
304
+ }
305
+ } finally {
306
+ try {
307
+ reader.releaseLock();
308
+ } catch {
309
+ // ignore lock release errors
310
+ }
311
+ }
312
+ }
313
+ /**
314
+ * Parse stream event to standard Chunk
315
+ */
316
+ parseStreamEvent(
317
+ event: AnthropicStreamEvent,
318
+ baseChunk: Partial<Chunk>,
319
+ toolCallStates?: Map<number, ToolCallStreamState>
320
+ ): Chunk | null {
321
+ switch (event.type) {
322
+ case 'message_start':
323
+ return {
324
+ id: event.message?.id || baseChunk.id || '',
325
+ index: 0,
326
+ model: event.message?.model || baseChunk.model || '',
327
+ object: 'chat.completion.chunk',
328
+ created: Math.floor(Date.now() / 1000),
329
+ choices: [
330
+ {
331
+ index: 0,
332
+ delta: { role: 'assistant', content: '' },
333
+ finish_reason: null,
334
+ },
335
+ ],
336
+ };
337
+ case 'content_block_start':
338
+ if (event.content_block?.type === 'tool_use') {
339
+ const toolIndex = event.index ?? 0;
340
+ const toolState: ToolCallStreamState = {
341
+ id: event.content_block.id || `tool_call_${toolIndex}`,
342
+ name: event.content_block.name || '',
343
+ };
344
+ toolCallStates?.set(toolIndex, toolState);
345
+
346
+ return {
347
+ ...baseChunk,
348
+ index: toolIndex,
349
+ choices: [
350
+ {
351
+ index: 0,
352
+ delta: {
353
+ role: 'assistant',
354
+ tool_calls: [
355
+ {
356
+ id: toolState.id,
357
+ type: 'function',
358
+ index: toolIndex,
359
+ function: {
360
+ name: toolState.name,
361
+ arguments: '',
362
+ },
363
+ },
364
+ ],
365
+ },
366
+ finish_reason: null,
367
+ },
368
+ ],
369
+ } as Chunk;
370
+ }
371
+ return null;
372
+ case 'content_block_delta':
373
+ if (event.delta?.type === 'text_delta' && event.delta.text) {
374
+ return {
375
+ ...baseChunk,
376
+ index: event.index ?? 0,
377
+ choices: [
378
+ {
379
+ index: 0,
380
+ delta: { role: 'assistant', content: event.delta.text },
381
+ finish_reason: null,
382
+ },
383
+ ],
384
+ } as Chunk;
385
+ } else if (event.delta?.type === 'input_json_delta' && event.delta.partial_json) {
386
+ const toolIndex = event.index ?? 0;
387
+ const toolState = toolCallStates?.get(toolIndex);
388
+ return {
389
+ ...baseChunk,
390
+ index: toolIndex,
391
+ choices: [
392
+ {
393
+ index: 0,
394
+ delta: {
395
+ role: 'assistant',
396
+ tool_calls: [
397
+ {
398
+ id: toolState?.id || `tool_call_${toolIndex}`,
399
+ type: 'function',
400
+ index: toolIndex,
401
+ function: {
402
+ name: toolState?.name || '',
403
+ arguments: event.delta.partial_json,
404
+ },
405
+ },
406
+ ],
407
+ },
408
+ finish_reason: null,
409
+ },
410
+ ],
411
+ } as Chunk;
412
+ }
413
+ return null;
414
+ case 'message_delta':
415
+ return {
416
+ ...baseChunk,
417
+ index: 0,
418
+ choices: [
419
+ {
420
+ index: 0,
421
+ delta: { role: 'assistant', content: '' },
422
+ finish_reason: this.convertStopReason(event.delta?.stop_reason),
423
+ },
424
+ ],
425
+ usage: event.usage
426
+ ? {
427
+ ...event.usage,
428
+ prompt_tokens: 0,
429
+ completion_tokens: event.usage.output_tokens || 0,
430
+ total_tokens: event.usage.output_tokens || 0,
431
+ }
432
+ : undefined,
433
+ } as Chunk;
434
+ case 'message_stop':
435
+ return null;
436
+ default:
437
+ return null;
438
+ }
439
+ }
440
+ /**
441
+ * Check if stream end event
442
+ */
443
+ isStreamEndEvent(event: AnthropicStreamEvent): boolean {
444
+ return event.type === 'message_stop';
445
+ }
446
+ /**
447
+ * Convert messages to Anthropic format
448
+ */
449
+ private convertMessages(messages: LLMRequestMessage[]): AnthropicMessage[] {
450
+ const result: AnthropicMessage[] = [];
451
+ for (const msg of messages) {
452
+ // handle tool response
453
+ if (msg.role === 'tool' && msg.tool_call_id) {
454
+ result.push({
455
+ role: 'user',
456
+ content: [
457
+ {
458
+ type: 'tool_result',
459
+ tool_use_id: msg.tool_call_id,
460
+ content: this.extractTextContent(msg.content),
461
+ },
462
+ ],
463
+ });
464
+ continue;
465
+ }
466
+ // handle assistant message with tool calls
467
+ const toolCalls = msg.tool_calls as ToolCall[] | undefined;
468
+ if (msg.role === 'assistant' && toolCalls && toolCalls.length > 0) {
469
+ const content: AnthropicContentBlock[] = [];
470
+ const textContent = this.extractTextContent(msg.content);
471
+ if (textContent) {
472
+ content.push({ type: 'text', text: textContent });
473
+ }
474
+ for (const toolCall of toolCalls) {
475
+ content.push({
476
+ type: 'tool_use',
477
+ id: toolCall.id,
478
+ name: toolCall.function.name,
479
+ input: this.parseJsonSafe(toolCall.function.arguments) || {},
480
+ });
481
+ }
482
+ result.push({ role: 'assistant', content });
483
+ continue;
484
+ }
485
+ // handle normal message
486
+ const content = this.convertContent(msg.content);
487
+ result.push({
488
+ role: msg.role as 'user' | 'assistant',
489
+ content,
490
+ });
491
+ }
492
+ return result;
493
+ }
494
+ /**
495
+ * Convert content to Anthropic format
496
+ */
497
+ private convertContent(content: MessageContent): string | AnthropicContentBlock[] {
498
+ if (typeof content === 'string') {
499
+ return content;
500
+ }
501
+ if (!Array.isArray(content)) {
502
+ return String(content);
503
+ }
504
+ const blocks: AnthropicContentBlock[] = [];
505
+ for (const part of content) {
506
+ if (part.type === 'text') {
507
+ blocks.push({ type: 'text', text: part.text });
508
+ } else if (part.type === 'image_url') {
509
+ const imageUrl = part.image_url.url;
510
+ if (imageUrl.startsWith('data:')) {
511
+ const matches = imageUrl.match(/^data:([^;]+);base64,(.+)$/);
512
+ if (matches) {
513
+ blocks.push({
514
+ type: 'image',
515
+ source: {
516
+ type: 'base64',
517
+ media_type: matches[1],
518
+ data: matches[2],
519
+ },
520
+ });
521
+ }
522
+ } else {
523
+ this.logger?.warn('Anthropic does not support image URLs directly; expected base64 data');
524
+ }
525
+ }
526
+ }
527
+ return blocks.length === 1 && blocks[0].type === 'text' ? (blocks[0].text ?? '') : blocks;
528
+ }
529
+ /**
530
+ * Extract text content
531
+ */
532
+ private extractTextContent(content: MessageContent): string {
533
+ if (typeof content === 'string') {
534
+ return content;
535
+ }
536
+ if (Array.isArray(content)) {
537
+ return content
538
+ .filter((part): part is { type: 'text'; text: string } => part.type === 'text')
539
+ .map((part) => part.text)
540
+ .join('');
541
+ }
542
+ return '';
543
+ }
544
+ /**
545
+ * Safe JSON parse
546
+ */
547
+ private parseJsonSafe(str: string): Record<string, unknown> | null {
548
+ try {
549
+ return JSON.parse(str);
550
+ } catch {
551
+ return null;
552
+ }
553
+ }
554
+ /**
555
+ * Convert stop_reason
556
+ */
557
+ private convertStopReason(
558
+ reason: string | undefined
559
+ ): 'stop' | 'length' | 'content_filter' | 'tool_calls' | null {
560
+ if (!reason) return null;
561
+ switch (reason) {
562
+ case 'end_turn':
563
+ return 'stop';
564
+ case 'max_tokens':
565
+ return 'length';
566
+ case 'tool_use':
567
+ return 'tool_calls';
568
+ default:
569
+ return null;
570
+ }
571
+ }
572
+ }
@@ -0,0 +1,131 @@
1
+ /**
2
+ * 基础 API 适配器
3
+ *
4
+ * API 适配器的抽象基类,用于处理特定于提供商的
5
+ * 请求/响应转换。
6
+ */
7
+
8
+ import type {
9
+ InputContentPart,
10
+ LLMRequest,
11
+ LLMResponse,
12
+ LLMRequestMessage,
13
+ MessageContent,
14
+ Role,
15
+ Chunk,
16
+ ToolCall,
17
+ } from '../types';
18
+
19
+ export abstract class BaseAPIAdapter {
20
+ /**
21
+ * 将消息和选项转换为特定于提供商的请求体
22
+ */
23
+ abstract transformRequest(options?: LLMRequest): Record<string, unknown>;
24
+
25
+ /**
26
+ * 将特定于提供商的响应转换为标准格式
27
+ */
28
+ abstract transformResponse(response: unknown): LLMResponse;
29
+
30
+ /**
31
+ * 获取请求的 HTTP 头(包括身份验证)
32
+ */
33
+ abstract getHeaders(apiKey: string, config?: Record<string, unknown>): Headers;
34
+
35
+ /**
36
+ * 获取聊天补全的端点路径
37
+ * 例如:'/v1/chat/completions' 或 '/api/paas/v4/chat/completions'
38
+ */
39
+ abstract getEndpointPath(): string;
40
+
41
+ /**
42
+ * 可选:解析流式响应
43
+ * 子类可以覆盖此方法以支持非标准流式格式
44
+ * 如果返回 null,则使用默认的 OpenAI 流式解析器
45
+ */
46
+ parseStreamAsync?(reader: ReadableStreamDefaultReader<Uint8Array>): AsyncGenerator<Chunk>;
47
+
48
+ /**
49
+ * 工具方法:检查清理后的消息是否应该发送
50
+ */
51
+ protected isMessageUsable(msg: {
52
+ role: string;
53
+ content?: unknown;
54
+ tool_call_id?: string;
55
+ tool_calls?: Array<{
56
+ id: string;
57
+ type: string;
58
+ function: { name: string; arguments: string };
59
+ }>;
60
+ }): boolean {
61
+ if (!msg) return false;
62
+ const hasContent =
63
+ msg.content !== undefined &&
64
+ msg.content !== null &&
65
+ (typeof msg.content !== 'string' || msg.content !== '') &&
66
+ (!Array.isArray(msg.content) || msg.content.length > 0);
67
+ const hasToolCalls = Array.isArray(msg.tool_calls) && msg.tool_calls.length > 0;
68
+ const hasToolCallId = Boolean(msg.tool_call_id);
69
+ return hasContent || hasToolCalls || hasToolCallId;
70
+ }
71
+
72
+ /**
73
+ * 清理消息以用于 API 请求(移除内部字段)
74
+ */
75
+ protected cleanMessage(msg: Array<Record<string, unknown>>): LLMRequestMessage[] {
76
+ const cleaned: LLMRequestMessage[] = [];
77
+
78
+ for (const item of msg) {
79
+ const normalizedContent = this.normalizeMessageContent(item.content);
80
+ const message: LLMRequestMessage = {
81
+ content: normalizedContent,
82
+ role: item.role as Role,
83
+ };
84
+
85
+ if (item.reasoning_content !== undefined && item.reasoning_content !== null) {
86
+ message.reasoning_content = item.reasoning_content as string;
87
+ }
88
+
89
+ if (item.tool_call_id !== undefined && item.tool_call_id !== null) {
90
+ message.tool_call_id = item.tool_call_id as string;
91
+ }
92
+
93
+ if (item.tool_calls !== undefined && item.tool_calls !== null) {
94
+ message.tool_calls = item.tool_calls as ToolCall[];
95
+ }
96
+
97
+ cleaned.push(message);
98
+ }
99
+
100
+ return cleaned;
101
+ }
102
+
103
+ /**
104
+ * 标准化消息 content,保留 OpenAI 多模态数组结构
105
+ */
106
+ private normalizeMessageContent(content: unknown): MessageContent {
107
+ if (typeof content === 'string') return content;
108
+
109
+ if (Array.isArray(content)) {
110
+ return content.filter((part): part is InputContentPart => this.isValidContentPart(part));
111
+ }
112
+
113
+ if (content === undefined || content === null) {
114
+ return '';
115
+ }
116
+
117
+ return String(content);
118
+ }
119
+
120
+ private isValidContentPart(part: unknown): part is InputContentPart {
121
+ if (!part || typeof part !== 'object') return false;
122
+ const type = (part as { type?: unknown }).type;
123
+ return (
124
+ type === 'text' ||
125
+ type === 'image_url' ||
126
+ type === 'input_audio' ||
127
+ type === 'input_video' ||
128
+ type === 'file'
129
+ );
130
+ }
131
+ }