@renxqoo/renx-code 0.0.4 → 0.0.6
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.
- package/README.md +82 -51
- package/bin/renx.cjs +16 -0
- package/package.json +2 -45
- package/src/agent/runtime/runtime.context-usage.test.ts +4 -5
- package/src/agent/runtime/runtime.error-handling.test.ts +4 -5
- package/src/agent/runtime/runtime.test.ts +7 -4
- package/src/agent/runtime/runtime.ts +3 -9
- package/src/agent/runtime/runtime.usage-forwarding.test.ts +4 -5
- package/src/agent/runtime/source-modules.test.ts +16 -35
- package/src/agent/runtime/source-modules.ts +17 -0
- package/vendor/agent-root/src/agent/ENTERPRISE_ACCEPTANCE_CHECKLIST.md +95 -0
- package/vendor/agent-root/src/agent/ENTERPRISE_REALTIME.html +1345 -0
- package/vendor/agent-root/src/agent/ENTERPRISE_REALTIME.md +1353 -0
- package/vendor/agent-root/src/agent/ERROR_CONTRACT.md +60 -0
- package/vendor/agent-root/src/agent/TEST_COVERAGE_ANALYSIS.md +278 -0
- package/vendor/agent-root/src/agent/__test__/error-contract.test.ts +72 -0
- package/vendor/agent-root/src/agent/__test__/types.test.ts +137 -0
- package/vendor/agent-root/src/agent/agent/__test__/abort-runtime.test.ts +83 -0
- package/vendor/agent-root/src/agent/agent/__test__/callback-safety.test.ts +34 -0
- package/vendor/agent-root/src/agent/agent/__test__/compaction.test.ts +323 -0
- package/vendor/agent-root/src/agent/agent/__test__/concurrency.test.ts +290 -0
- package/vendor/agent-root/src/agent/agent/__test__/error-normalizer.test.ts +377 -0
- package/vendor/agent-root/src/agent/agent/__test__/error.test.ts +212 -0
- package/vendor/agent-root/src/agent/agent/__test__/fault-injection.test.ts +295 -0
- package/vendor/agent-root/src/agent/agent/__test__/index.test.ts +3607 -0
- package/vendor/agent-root/src/agent/agent/__test__/logger.test.ts +35 -0
- package/vendor/agent-root/src/agent/agent/__test__/message-utils.test.ts +517 -0
- package/vendor/agent-root/src/agent/agent/__test__/telemetry.test.ts +97 -0
- package/vendor/agent-root/src/agent/agent/__test__/timeout-budget.test.ts +479 -0
- package/vendor/agent-root/src/agent/agent/__test__/tool-call-merge.test.ts +80 -0
- package/vendor/agent-root/src/agent/agent/__test__/tool-execution-ledger.test.ts +76 -0
- package/vendor/agent-root/src/agent/agent/__test__/write-buffer.test.ts +173 -0
- package/vendor/agent-root/src/agent/agent/__test__/write-file-session.test.ts +109 -0
- package/vendor/agent-root/src/agent/agent/abort-runtime.ts +71 -0
- package/vendor/agent-root/src/agent/agent/callback-safety.ts +33 -0
- package/vendor/agent-root/src/agent/agent/compaction.ts +291 -0
- package/vendor/agent-root/src/agent/agent/concurrency.ts +103 -0
- package/vendor/agent-root/src/agent/agent/error-normalizer.ts +190 -0
- package/vendor/agent-root/src/agent/agent/error.ts +198 -0
- package/vendor/agent-root/src/agent/agent/index.ts +1772 -0
- package/vendor/agent-root/src/agent/agent/logger.ts +65 -0
- package/vendor/agent-root/src/agent/agent/message-utils.ts +101 -0
- package/vendor/agent-root/src/agent/agent/stream-events.ts +61 -0
- package/vendor/agent-root/src/agent/agent/telemetry.ts +123 -0
- package/vendor/agent-root/src/agent/agent/timeout-budget.ts +227 -0
- package/vendor/agent-root/src/agent/agent/tool-call-merge.ts +111 -0
- package/vendor/agent-root/src/agent/agent/tool-execution-ledger.ts +164 -0
- package/vendor/agent-root/src/agent/agent/write-buffer.ts +188 -0
- package/vendor/agent-root/src/agent/agent/write-file-session.ts +238 -0
- package/vendor/agent-root/src/agent/app/__test__/agent-app-service.test.ts +1053 -0
- package/vendor/agent-root/src/agent/app/__test__/minimal-agent-application.test.ts +158 -0
- package/vendor/agent-root/src/agent/app/__test__/sqlite-agent-app-store.test.ts +437 -0
- package/vendor/agent-root/src/agent/app/agent-app-service.ts +748 -0
- package/vendor/agent-root/src/agent/app/contracts.ts +109 -0
- package/vendor/agent-root/src/agent/app/index.ts +5 -0
- package/vendor/agent-root/src/agent/app/minimal-agent-application.ts +151 -0
- package/vendor/agent-root/src/agent/app/ports.ts +72 -0
- package/vendor/agent-root/src/agent/app/sqlite-agent-app-store.ts +1182 -0
- package/vendor/agent-root/src/agent/app/sqlite-client.ts +177 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/00-README.md +36 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/01-scope-and-goals.md +33 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/02-architecture-overview.md +40 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/03-domain-model-and-contracts.md +91 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/04-ports-and-interfaces.md +116 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/05-run-orchestration-and-state-machine.md +52 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/06-cli-commands-and-ux.md +53 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/07-storage-design-local.md +52 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/08-error-and-observability.md +40 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/09-security-and-policy-boundary.md +19 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/10-test-plan-and-acceptance.md +28 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/11-implementation-phases.md +26 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/12-open-questions-and-risks.md +30 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/13-sqlite-schema-fields-and-rationale.md +567 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/14-project-flow-mermaid.md +583 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/15-openclaw-style-project-blueprint.md +972 -0
- package/vendor/agent-root/src/agent/error-contract.ts +154 -0
- package/vendor/agent-root/src/agent/prompts/system.ts +246 -0
- package/vendor/agent-root/src/agent/prompts/system1.ts +208 -0
- package/vendor/agent-root/src/agent/storage/__test__/file-history-store.test.ts +98 -0
- package/vendor/agent-root/src/agent/storage/file-history-store.ts +313 -0
- package/vendor/agent-root/src/agent/storage/file-storage-config.ts +94 -0
- package/vendor/agent-root/src/agent/storage/file-system.ts +31 -0
- package/vendor/agent-root/src/agent/storage/file-write-service.ts +21 -0
- package/vendor/agent-root/src/agent/tool/__test__/base-tool.test.ts +413 -0
- package/vendor/agent-root/src/agent/tool/__test__/bash-policy.test.ts +356 -0
- package/vendor/agent-root/src/agent/tool/__test__/bash.mocked-coverage.test.ts +375 -0
- package/vendor/agent-root/src/agent/tool/__test__/bash.test.ts +372 -0
- package/vendor/agent-root/src/agent/tool/__test__/error.test.ts +108 -0
- package/vendor/agent-root/src/agent/tool/__test__/file-edit-tool.test.ts +258 -0
- package/vendor/agent-root/src/agent/tool/__test__/file-history-tools.test.ts +121 -0
- package/vendor/agent-root/src/agent/tool/__test__/file-read-tool.test.ts +210 -0
- package/vendor/agent-root/src/agent/tool/__test__/glob.test.ts +139 -0
- package/vendor/agent-root/src/agent/tool/__test__/grep.mocked-coverage.test.ts +456 -0
- package/vendor/agent-root/src/agent/tool/__test__/grep.test.ts +192 -0
- package/vendor/agent-root/src/agent/tool/__test__/lsp.test.ts +300 -0
- package/vendor/agent-root/src/agent/tool/__test__/outside-workspace-confirmation.test.ts +214 -0
- package/vendor/agent-root/src/agent/tool/__test__/path-security.test.ts +336 -0
- package/vendor/agent-root/src/agent/tool/__test__/skill-loader.test.ts +494 -0
- package/vendor/agent-root/src/agent/tool/__test__/skill-parser.test.ts +543 -0
- package/vendor/agent-root/src/agent/tool/__test__/skill-tool.test.ts +172 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-concurrency-and-version.test.ts +116 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-create-get-list-update.test.ts +267 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-create.test.ts +519 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-errors.test.ts +225 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-output-blocking.test.ts +223 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-output.test.ts +184 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-parent-abort.test.ts +287 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-real-runner-adapter.test.ts +190 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-run-lifecycle.test.ts +352 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-store-runner-branches.test.ts +395 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-store.test.ts +391 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-subagent-config-integration.test.ts +176 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-subagent-config.test.ts +68 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-tools-core-edges.test.ts +630 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-tools-runtime-edges.test.ts +732 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-types.test.ts +494 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-utils-branches.test.ts +175 -0
- package/vendor/agent-root/src/agent/tool/__test__/tool-manager.test.ts +505 -0
- package/vendor/agent-root/src/agent/tool/__test__/types.test.ts +55 -0
- package/vendor/agent-root/src/agent/tool/__test__/web-fetch.test.ts +244 -0
- package/vendor/agent-root/src/agent/tool/__test__/web-search.test.ts +290 -0
- package/vendor/agent-root/src/agent/tool/__test__/write-file.test.ts +368 -0
- package/vendor/agent-root/src/agent/tool/base-tool.ts +345 -0
- package/vendor/agent-root/src/agent/tool/bash-policy.ts +636 -0
- package/vendor/agent-root/src/agent/tool/bash.ts +688 -0
- package/vendor/agent-root/src/agent/tool/error.ts +131 -0
- package/vendor/agent-root/src/agent/tool/file-edit-tool.ts +264 -0
- package/vendor/agent-root/src/agent/tool/file-history-list.ts +103 -0
- package/vendor/agent-root/src/agent/tool/file-history-restore.ts +149 -0
- package/vendor/agent-root/src/agent/tool/file-read-tool.ts +211 -0
- package/vendor/agent-root/src/agent/tool/glob.ts +171 -0
- package/vendor/agent-root/src/agent/tool/grep.ts +496 -0
- package/vendor/agent-root/src/agent/tool/lsp.ts +481 -0
- package/vendor/agent-root/src/agent/tool/path-security.ts +117 -0
- package/vendor/agent-root/src/agent/tool/search/common.ts +153 -0
- package/vendor/agent-root/src/agent/tool/skill/index.ts +13 -0
- package/vendor/agent-root/src/agent/tool/skill/loader.ts +229 -0
- package/vendor/agent-root/src/agent/tool/skill/parser.ts +124 -0
- package/vendor/agent-root/src/agent/tool/skill/types.ts +27 -0
- package/vendor/agent-root/src/agent/tool/skill-tool.ts +143 -0
- package/vendor/agent-root/src/agent/tool/task-create.ts +186 -0
- package/vendor/agent-root/src/agent/tool/task-errors.ts +42 -0
- package/vendor/agent-root/src/agent/tool/task-get.ts +116 -0
- package/vendor/agent-root/src/agent/tool/task-graph.ts +78 -0
- package/vendor/agent-root/src/agent/tool/task-list.ts +141 -0
- package/vendor/agent-root/src/agent/tool/task-mock-runner-adapter.ts +232 -0
- package/vendor/agent-root/src/agent/tool/task-output.ts +223 -0
- package/vendor/agent-root/src/agent/tool/task-parent-abort.ts +115 -0
- package/vendor/agent-root/src/agent/tool/task-real-runner-adapter.ts +336 -0
- package/vendor/agent-root/src/agent/tool/task-runner-adapter.ts +55 -0
- package/vendor/agent-root/src/agent/tool/task-stop.ts +187 -0
- package/vendor/agent-root/src/agent/tool/task-store.ts +217 -0
- package/vendor/agent-root/src/agent/tool/task-subagent-config.ts +149 -0
- package/vendor/agent-root/src/agent/tool/task-types.ts +264 -0
- package/vendor/agent-root/src/agent/tool/task-update.ts +315 -0
- package/vendor/agent-root/src/agent/tool/task.ts +209 -0
- package/vendor/agent-root/src/agent/tool/tool-manager.ts +362 -0
- package/vendor/agent-root/src/agent/tool/tool-prompts.ts +242 -0
- package/vendor/agent-root/src/agent/tool/types.ts +116 -0
- package/vendor/agent-root/src/agent/tool/web-fetch.ts +227 -0
- package/vendor/agent-root/src/agent/tool/web-search.ts +208 -0
- package/vendor/agent-root/src/agent/tool/write-file.ts +497 -0
- package/vendor/agent-root/src/agent/types.ts +232 -0
- package/vendor/agent-root/src/agent/utils/__tests__/index.test.ts +18 -0
- package/vendor/agent-root/src/agent/utils/__tests__/message-utils.test.ts +610 -0
- package/vendor/agent-root/src/agent/utils/__tests__/message.test.ts +223 -0
- package/vendor/agent-root/src/agent/utils/__tests__/token.test.ts +42 -0
- package/vendor/agent-root/src/agent/utils/index.ts +16 -0
- package/vendor/agent-root/src/agent/utils/message.ts +171 -0
- package/vendor/agent-root/src/agent/utils/token.ts +28 -0
- package/vendor/agent-root/src/config/__tests__/load-config-to-env.test.ts +129 -0
- package/vendor/agent-root/src/config/__tests__/loader.test.ts +247 -0
- package/vendor/agent-root/src/config/__tests__/runtime.test.ts +88 -0
- package/vendor/agent-root/src/config/index.ts +54 -0
- package/vendor/agent-root/src/config/loader.ts +431 -0
- package/vendor/agent-root/src/config/paths.ts +30 -0
- package/vendor/agent-root/src/config/runtime.ts +163 -0
- package/vendor/agent-root/src/config/types.ts +70 -0
- package/vendor/agent-root/src/logger/index.ts +57 -0
- package/vendor/agent-root/src/logger/logger.ts +819 -0
- package/vendor/agent-root/src/logger/types.ts +150 -0
- package/vendor/agent-root/src/providers/__tests__/errors.test.ts +441 -0
- package/vendor/agent-root/src/providers/__tests__/index.test.ts +16 -0
- package/vendor/agent-root/src/providers/__tests__/openai-compatible.options.test.ts +318 -0
- package/vendor/agent-root/src/providers/__tests__/openai-compatible.test.ts +600 -0
- package/vendor/agent-root/src/providers/__tests__/registry.test.ts +449 -0
- package/vendor/agent-root/src/providers/__tests__/responses-adapter.test.ts +298 -0
- package/vendor/agent-root/src/providers/adapters/__tests__/anthropic.test.ts +354 -0
- package/vendor/agent-root/src/providers/adapters/__tests__/kimi.test.ts +58 -0
- package/vendor/agent-root/src/providers/adapters/__tests__/standard.test.ts +261 -0
- package/vendor/agent-root/src/providers/adapters/anthropic.ts +572 -0
- package/vendor/agent-root/src/providers/adapters/base.ts +131 -0
- package/vendor/agent-root/src/providers/adapters/kimi.ts +48 -0
- package/vendor/agent-root/src/providers/adapters/responses.ts +732 -0
- package/vendor/agent-root/src/providers/adapters/standard.ts +120 -0
- package/vendor/agent-root/src/providers/http/__tests__/client.timeout.test.ts +313 -0
- package/vendor/agent-root/src/providers/http/client.ts +289 -0
- package/vendor/agent-root/src/providers/http/stream-parser.ts +109 -0
- package/vendor/agent-root/src/providers/index.ts +76 -0
- package/vendor/agent-root/src/providers/kimi-headers.ts +177 -0
- package/vendor/agent-root/src/providers/openai-compatible.ts +387 -0
- package/vendor/agent-root/src/providers/registry/model-config.ts +230 -0
- package/vendor/agent-root/src/providers/registry/provider-factory.ts +123 -0
- package/vendor/agent-root/src/providers/registry.ts +135 -0
- package/vendor/agent-root/src/providers/types/api.ts +284 -0
- package/vendor/agent-root/src/providers/types/config.ts +58 -0
- package/vendor/agent-root/src/providers/types/errors.ts +323 -0
- package/vendor/agent-root/src/providers/types/index.ts +72 -0
- package/vendor/agent-root/src/providers/types/provider.ts +45 -0
- package/vendor/agent-root/src/providers/types/registry.ts +88 -0
|
@@ -0,0 +1,289 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* HTTP 客户端工具
|
|
3
|
+
*
|
|
4
|
+
* 提供统一的 HTTP 客户端,具有以下功能:
|
|
5
|
+
* - 单次请求执行(不包含重试)
|
|
6
|
+
* - Abort 信号支持(优先使用上层传入 signal)
|
|
7
|
+
* - 可选默认超时兜底(调用方未传 signal 时生效)
|
|
8
|
+
* - 与 LLM 错误类型集成的错误处理
|
|
9
|
+
*
|
|
10
|
+
* 超时控制设计:
|
|
11
|
+
* - Agent 层通过 LLMCaller 创建 AbortSignal.timeout() 控制主链路超时
|
|
12
|
+
* - 当调用方未传 signal 且配置了 defaultTimeoutMs,本层会创建兜底超时信号
|
|
13
|
+
* - 这样兼顾统一控制与 standalone 调用安全性
|
|
14
|
+
*/
|
|
15
|
+
|
|
16
|
+
import { LLMError, LLMAbortedError, LLMRetryableError, createErrorFromStatus } from '../types';
|
|
17
|
+
import { classifyAbortReason } from '../types';
|
|
18
|
+
import type { ProviderLogger } from '../types';
|
|
19
|
+
|
|
20
|
+
export interface HttpClientOptions {
|
|
21
|
+
/** 日志器(推荐注入 Agent logger 的 child) */
|
|
22
|
+
logger?: ProviderLogger;
|
|
23
|
+
/** 默认超时(毫秒,仅在调用方未传 signal 时生效) */
|
|
24
|
+
defaultTimeoutMs?: number;
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
export type RequestInitWithOptions = RequestInit;
|
|
28
|
+
|
|
29
|
+
/**
|
|
30
|
+
* HTTP 客户端
|
|
31
|
+
*
|
|
32
|
+
* 超时优先由调用方通过 options.signal 传入;
|
|
33
|
+
* 如未传入且配置了 defaultTimeoutMs,会自动应用默认超时信号。
|
|
34
|
+
*/
|
|
35
|
+
export class HTTPClient {
|
|
36
|
+
readonly logger?: ProviderLogger;
|
|
37
|
+
readonly defaultTimeoutMs?: number;
|
|
38
|
+
|
|
39
|
+
constructor(options: HttpClientOptions = {}) {
|
|
40
|
+
this.logger = options.logger?.child('HTTPClient');
|
|
41
|
+
this.defaultTimeoutMs = this.normalizeTimeoutMs(options.defaultTimeoutMs);
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
/**
|
|
45
|
+
* 单次 Fetch(重试由上层 Agent 负责)
|
|
46
|
+
*
|
|
47
|
+
* @param url - 请求 URL
|
|
48
|
+
* @param options - 请求选项,signal 应已包含超时逻辑
|
|
49
|
+
*/
|
|
50
|
+
async fetch(url: string, options: RequestInitWithOptions = {}): Promise<Response> {
|
|
51
|
+
const requestOptions = this.applyDefaultSignal(options);
|
|
52
|
+
try {
|
|
53
|
+
const response = await this.executeFetch(url, requestOptions);
|
|
54
|
+
|
|
55
|
+
// 检查 HTTP 错误
|
|
56
|
+
if (!response.ok) {
|
|
57
|
+
const errorText = await response.text();
|
|
58
|
+
// 提取 Retry-After 响应头(用于 429 等错误)
|
|
59
|
+
const retryAfterMs = this.extractRetryAfterMs(response);
|
|
60
|
+
throw createErrorFromStatus(response.status, response.statusText, errorText, retryAfterMs);
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
return response;
|
|
64
|
+
} catch (rawError) {
|
|
65
|
+
throw this.normalizeError(rawError, requestOptions.signal ?? undefined);
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
/**
|
|
70
|
+
* 执行 Fetch 请求
|
|
71
|
+
*
|
|
72
|
+
* 直接使用传入的 signal,不创建额外的超时信号
|
|
73
|
+
*/
|
|
74
|
+
private async executeFetch(url: string, options: RequestInit): Promise<Response> {
|
|
75
|
+
const upstreamSignal = options.signal;
|
|
76
|
+
const method = options.method || 'GET';
|
|
77
|
+
|
|
78
|
+
try {
|
|
79
|
+
this.logger?.debug('Sending request', {
|
|
80
|
+
method,
|
|
81
|
+
url,
|
|
82
|
+
hasSignal: Boolean(upstreamSignal),
|
|
83
|
+
header: JSON.stringify(options),
|
|
84
|
+
});
|
|
85
|
+
|
|
86
|
+
const response = await fetch(url, {
|
|
87
|
+
...options,
|
|
88
|
+
signal: upstreamSignal,
|
|
89
|
+
});
|
|
90
|
+
|
|
91
|
+
this.logger?.debug('Response received', {
|
|
92
|
+
method,
|
|
93
|
+
url,
|
|
94
|
+
status: response.status,
|
|
95
|
+
});
|
|
96
|
+
|
|
97
|
+
return response;
|
|
98
|
+
} catch (error) {
|
|
99
|
+
this.logger?.warn(
|
|
100
|
+
'Request failed',
|
|
101
|
+
{
|
|
102
|
+
method,
|
|
103
|
+
url,
|
|
104
|
+
aborted: Boolean(upstreamSignal?.aborted),
|
|
105
|
+
},
|
|
106
|
+
error
|
|
107
|
+
);
|
|
108
|
+
|
|
109
|
+
// 检查是否为超时或中止错误
|
|
110
|
+
if (upstreamSignal?.aborted) {
|
|
111
|
+
const reason = this.getAbortReason(upstreamSignal);
|
|
112
|
+
if (reason === 'timeout') {
|
|
113
|
+
throw new LLMRetryableError('Request timeout', undefined, 'TIMEOUT');
|
|
114
|
+
}
|
|
115
|
+
throw new LLMAbortedError('Request was cancelled by upstream signal');
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
throw error;
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
/**
|
|
123
|
+
* 获取 AbortSignal 的中止原因
|
|
124
|
+
*/
|
|
125
|
+
private getAbortReason(signal: AbortSignal): 'timeout' | 'abort' | 'unknown' {
|
|
126
|
+
try {
|
|
127
|
+
const reason = classifyAbortReason(signal.reason);
|
|
128
|
+
if (reason === 'idle_timeout' || reason === 'timeout') {
|
|
129
|
+
return 'timeout';
|
|
130
|
+
}
|
|
131
|
+
if (reason === 'abort') {
|
|
132
|
+
return 'abort';
|
|
133
|
+
}
|
|
134
|
+
} catch {
|
|
135
|
+
// 忽略访问 reason 的错误
|
|
136
|
+
}
|
|
137
|
+
return 'unknown';
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
/**
|
|
141
|
+
* 从响应头中提取 Retry-After 值(毫秒)
|
|
142
|
+
*
|
|
143
|
+
* Retry-After 可以是秒数或日期字符串
|
|
144
|
+
* @see https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Retry-After
|
|
145
|
+
*/
|
|
146
|
+
private extractRetryAfterMs(response: Response): number | undefined {
|
|
147
|
+
const retryAfterMsHeader = response.headers.get('retry-after-ms');
|
|
148
|
+
if (retryAfterMsHeader) {
|
|
149
|
+
const ms = Number(retryAfterMsHeader);
|
|
150
|
+
if (Number.isFinite(ms) && ms > 0) {
|
|
151
|
+
return ms;
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
|
|
155
|
+
const retryAfter = response.headers.get('Retry-After');
|
|
156
|
+
if (!retryAfter) {
|
|
157
|
+
return undefined;
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
// 尝试解析为秒数(支持小数,避免精度损失)
|
|
161
|
+
const seconds = Number(retryAfter);
|
|
162
|
+
if (Number.isFinite(seconds) && seconds > 0) {
|
|
163
|
+
return Math.ceil(seconds * 1000);
|
|
164
|
+
}
|
|
165
|
+
|
|
166
|
+
// 尝试解析为日期字符串
|
|
167
|
+
const dateMs = Date.parse(retryAfter);
|
|
168
|
+
if (Number.isFinite(dateMs)) {
|
|
169
|
+
const diffMs = dateMs - Date.now();
|
|
170
|
+
if (diffMs > 0) {
|
|
171
|
+
return Math.ceil(diffMs);
|
|
172
|
+
}
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
return undefined;
|
|
176
|
+
}
|
|
177
|
+
|
|
178
|
+
/**
|
|
179
|
+
* 归一化错误
|
|
180
|
+
*/
|
|
181
|
+
private normalizeError(error: unknown, signal?: AbortSignal): Error {
|
|
182
|
+
// 已经是 LLM 错误,直接返回
|
|
183
|
+
if (error instanceof LLMError) {
|
|
184
|
+
return error;
|
|
185
|
+
}
|
|
186
|
+
|
|
187
|
+
// 检查中止信号
|
|
188
|
+
if (signal?.aborted) {
|
|
189
|
+
const reason = this.getAbortReason(signal);
|
|
190
|
+
if (reason === 'timeout') {
|
|
191
|
+
return new LLMRetryableError('Request timeout', undefined, 'TIMEOUT');
|
|
192
|
+
}
|
|
193
|
+
return new LLMAbortedError('Request was cancelled');
|
|
194
|
+
}
|
|
195
|
+
|
|
196
|
+
// Body 超时类错误
|
|
197
|
+
if (this.isBodyTimeoutLikeError(error)) {
|
|
198
|
+
return new LLMRetryableError('Response body timeout', undefined, 'BODY_TIMEOUT');
|
|
199
|
+
}
|
|
200
|
+
|
|
201
|
+
// 网络类错误
|
|
202
|
+
if (this.isNetworkLikeError(error)) {
|
|
203
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
204
|
+
return new LLMRetryableError(
|
|
205
|
+
`Network request failed: ${message}`,
|
|
206
|
+
undefined,
|
|
207
|
+
'NETWORK_ERROR'
|
|
208
|
+
);
|
|
209
|
+
}
|
|
210
|
+
|
|
211
|
+
if (error instanceof Error) {
|
|
212
|
+
return error;
|
|
213
|
+
}
|
|
214
|
+
|
|
215
|
+
return new LLMError(String(error));
|
|
216
|
+
}
|
|
217
|
+
|
|
218
|
+
private isBodyTimeoutLikeError(error: unknown): boolean {
|
|
219
|
+
if (!(error instanceof Error)) return false;
|
|
220
|
+
|
|
221
|
+
const code = this.getErrorCode(error);
|
|
222
|
+
const message = `${error.name} ${error.message}`.toLowerCase();
|
|
223
|
+
|
|
224
|
+
return (
|
|
225
|
+
code === 'UND_ERR_BODY_TIMEOUT' ||
|
|
226
|
+
message.includes('body timeout') ||
|
|
227
|
+
message.includes('terminated')
|
|
228
|
+
);
|
|
229
|
+
}
|
|
230
|
+
|
|
231
|
+
private isNetworkLikeError(error: unknown): boolean {
|
|
232
|
+
if (!(error instanceof Error)) return false;
|
|
233
|
+
|
|
234
|
+
const code = this.getErrorCode(error);
|
|
235
|
+
const message = `${error.name} ${error.message}`.toLowerCase();
|
|
236
|
+
if (!code) {
|
|
237
|
+
// Bun/Node fetch 的网络失败既可能是 TypeError,也可能是普通 Error 文案
|
|
238
|
+
return (
|
|
239
|
+
error instanceof TypeError ||
|
|
240
|
+
message.includes('unable to connect') ||
|
|
241
|
+
message.includes('access the url') ||
|
|
242
|
+
message.includes('failed to fetch') ||
|
|
243
|
+
message.includes('fetch failed') ||
|
|
244
|
+
message.includes('network request failed')
|
|
245
|
+
);
|
|
246
|
+
}
|
|
247
|
+
|
|
248
|
+
return [
|
|
249
|
+
'ECONNRESET',
|
|
250
|
+
'ECONNREFUSED',
|
|
251
|
+
'ENOTFOUND',
|
|
252
|
+
'EAI_AGAIN',
|
|
253
|
+
'ETIMEDOUT',
|
|
254
|
+
'UND_ERR_SOCKET',
|
|
255
|
+
'UND_ERR_CONNECT_TIMEOUT',
|
|
256
|
+
'UND_ERR_HEADERS_TIMEOUT',
|
|
257
|
+
'UND_ERR_ABORTED',
|
|
258
|
+
].includes(code);
|
|
259
|
+
}
|
|
260
|
+
|
|
261
|
+
private getErrorCode(error: Error): string | undefined {
|
|
262
|
+
const withCode = error as Error & { code?: unknown; cause?: unknown };
|
|
263
|
+
if (typeof withCode.code === 'string') {
|
|
264
|
+
return withCode.code;
|
|
265
|
+
}
|
|
266
|
+
const cause = withCode.cause as { code?: unknown } | undefined;
|
|
267
|
+
if (cause && typeof cause.code === 'string') {
|
|
268
|
+
return cause.code;
|
|
269
|
+
}
|
|
270
|
+
return undefined;
|
|
271
|
+
}
|
|
272
|
+
|
|
273
|
+
private applyDefaultSignal(options: RequestInitWithOptions): RequestInitWithOptions {
|
|
274
|
+
if (options.signal || !this.defaultTimeoutMs) {
|
|
275
|
+
return options;
|
|
276
|
+
}
|
|
277
|
+
return {
|
|
278
|
+
...options,
|
|
279
|
+
signal: AbortSignal.timeout(this.defaultTimeoutMs),
|
|
280
|
+
};
|
|
281
|
+
}
|
|
282
|
+
|
|
283
|
+
private normalizeTimeoutMs(value: number | undefined): number | undefined {
|
|
284
|
+
if (typeof value !== 'number' || !Number.isFinite(value) || value <= 0) {
|
|
285
|
+
return undefined;
|
|
286
|
+
}
|
|
287
|
+
return value;
|
|
288
|
+
}
|
|
289
|
+
}
|
|
@@ -0,0 +1,109 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* 流解析器工具
|
|
3
|
+
*
|
|
4
|
+
* 处理 SSE(服务器发送事件)流解析,用于 LLM 流式响应。
|
|
5
|
+
*/
|
|
6
|
+
|
|
7
|
+
import type { Chunk } from '../types';
|
|
8
|
+
|
|
9
|
+
/**
|
|
10
|
+
* SSE(服务器发送事件)的流解析器
|
|
11
|
+
*/
|
|
12
|
+
export class StreamParser {
|
|
13
|
+
/**
|
|
14
|
+
* 解析 SSE 行并提取数据
|
|
15
|
+
* 对于空行、注释或非数据行返回 null
|
|
16
|
+
*/
|
|
17
|
+
static parseSseLine(line: string): string | null {
|
|
18
|
+
const trimmed = line.trim();
|
|
19
|
+
if (!trimmed || trimmed.startsWith(':')) {
|
|
20
|
+
return null;
|
|
21
|
+
}
|
|
22
|
+
if (trimmed.startsWith('data:')) {
|
|
23
|
+
return trimmed.slice(5).trim();
|
|
24
|
+
}
|
|
25
|
+
if (trimmed.startsWith('{')) {
|
|
26
|
+
return trimmed;
|
|
27
|
+
}
|
|
28
|
+
return null;
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
/**
|
|
32
|
+
* 检查数据是否指示流结束
|
|
33
|
+
*/
|
|
34
|
+
static isStreamEnd(data: string): boolean {
|
|
35
|
+
return data === '[DONE]';
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
/**
|
|
39
|
+
* 安全地解析 JSON,失败时返回 null
|
|
40
|
+
*/
|
|
41
|
+
static safeJsonParse<T>(data: string): T | null {
|
|
42
|
+
try {
|
|
43
|
+
return JSON.parse(data) as T;
|
|
44
|
+
} catch {
|
|
45
|
+
return null;
|
|
46
|
+
}
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
/**
|
|
50
|
+
* 处理可读流并返回 AsyncGenerator
|
|
51
|
+
*
|
|
52
|
+
* 注意:使用 try/finally 确保在生成器被提前终止时(如 break 或 return)
|
|
53
|
+
* reader 能够被正确释放,避免流锁定问题
|
|
54
|
+
*
|
|
55
|
+
* @param reader - 可读流读取器
|
|
56
|
+
* @returns 异步生成器,每次 yield 一个 chunk
|
|
57
|
+
*/
|
|
58
|
+
static async *parseAsync(reader: ReadableStreamDefaultReader<Uint8Array>): AsyncGenerator<Chunk> {
|
|
59
|
+
const decoder = new TextDecoder();
|
|
60
|
+
let buffer = '';
|
|
61
|
+
let shouldStop = false;
|
|
62
|
+
|
|
63
|
+
try {
|
|
64
|
+
while (!shouldStop) {
|
|
65
|
+
const { done, value } = await reader.read();
|
|
66
|
+
|
|
67
|
+
if (done) {
|
|
68
|
+
// 处理流结束时剩余的缓冲区
|
|
69
|
+
if (buffer.trim()) {
|
|
70
|
+
const data = this.parseSseLine(buffer);
|
|
71
|
+
if (data && !this.isStreamEnd(data)) {
|
|
72
|
+
const chunk = this.safeJsonParse<Chunk>(data);
|
|
73
|
+
if (chunk) yield chunk;
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
break;
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
buffer += decoder.decode(value, { stream: true });
|
|
80
|
+
const lines = buffer.split(/[\r\n]+/);
|
|
81
|
+
buffer = lines.pop() ?? '';
|
|
82
|
+
|
|
83
|
+
for (const line of lines) {
|
|
84
|
+
const data = this.parseSseLine(line);
|
|
85
|
+
|
|
86
|
+
if (!data) continue;
|
|
87
|
+
|
|
88
|
+
if (this.isStreamEnd(data)) {
|
|
89
|
+
shouldStop = true;
|
|
90
|
+
break;
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
const chunk = this.safeJsonParse<Chunk>(data);
|
|
94
|
+
if (!chunk) continue;
|
|
95
|
+
|
|
96
|
+
yield chunk;
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
} finally {
|
|
100
|
+
// 确保在生成器结束时释放 reader 锁
|
|
101
|
+
// 这样即使消费者提前 break,也不会导致流被锁定
|
|
102
|
+
try {
|
|
103
|
+
reader.releaseLock();
|
|
104
|
+
} catch {
|
|
105
|
+
// 忽略释放锁时的错误(如流已关闭)
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
}
|
|
@@ -0,0 +1,76 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Providers 统一导出
|
|
3
|
+
*/
|
|
4
|
+
|
|
5
|
+
// Registry 相关
|
|
6
|
+
export { Models, MODEL_CONFIGS, ProviderRegistry } from './registry';
|
|
7
|
+
export type { ProviderType, ModelId } from './registry';
|
|
8
|
+
|
|
9
|
+
// Provider 相关
|
|
10
|
+
export { LLMProvider } from './types';
|
|
11
|
+
export type { BaseProviderConfig, ProviderLogger } from './types';
|
|
12
|
+
|
|
13
|
+
// OpenAI Compatible Provider
|
|
14
|
+
export { OpenAICompatibleProvider } from './openai-compatible';
|
|
15
|
+
export type { OpenAICompatibleConfig } from './openai-compatible';
|
|
16
|
+
|
|
17
|
+
// 适配器
|
|
18
|
+
export { BaseAPIAdapter } from './adapters/base';
|
|
19
|
+
export { StandardAdapter, type StandardTransformOptions } from './adapters/standard';
|
|
20
|
+
|
|
21
|
+
// HTTP 客户端
|
|
22
|
+
export { HTTPClient, type HttpClientOptions, type RequestInitWithOptions } from './http/client';
|
|
23
|
+
export { StreamParser } from './http/stream-parser';
|
|
24
|
+
|
|
25
|
+
// 错误类型
|
|
26
|
+
export {
|
|
27
|
+
PERMANENT_STREAM_ERROR_CODE_MARKERS,
|
|
28
|
+
PERMANENT_STREAM_ERROR_MESSAGE_PATTERNS,
|
|
29
|
+
isPermanentStreamChunkError,
|
|
30
|
+
abortReasonToText,
|
|
31
|
+
isIdleTimeoutReasonText,
|
|
32
|
+
isTimeoutReasonText,
|
|
33
|
+
classifyAbortReason,
|
|
34
|
+
LLMError,
|
|
35
|
+
LLMRetryableError,
|
|
36
|
+
LLMRateLimitError,
|
|
37
|
+
LLMPermanentError,
|
|
38
|
+
LLMAuthError,
|
|
39
|
+
LLMNotFoundError,
|
|
40
|
+
LLMBadRequestError,
|
|
41
|
+
LLMAbortedError,
|
|
42
|
+
createErrorFromStatus,
|
|
43
|
+
isRetryableError,
|
|
44
|
+
isPermanentError,
|
|
45
|
+
isAbortedError,
|
|
46
|
+
calculateBackoff,
|
|
47
|
+
DEFAULT_BACKOFF_CONFIG,
|
|
48
|
+
} from './types';
|
|
49
|
+
|
|
50
|
+
export type { BackoffConfig } from './types';
|
|
51
|
+
|
|
52
|
+
// 类型定义
|
|
53
|
+
export type {
|
|
54
|
+
ToolCall,
|
|
55
|
+
Role,
|
|
56
|
+
TextContentPart,
|
|
57
|
+
ImageUrlContentPart,
|
|
58
|
+
InputAudioContentPart,
|
|
59
|
+
InputVideoContentPart,
|
|
60
|
+
FileContentPart,
|
|
61
|
+
InputContentPart,
|
|
62
|
+
MessageContent,
|
|
63
|
+
Usage,
|
|
64
|
+
StreamOptions,
|
|
65
|
+
BaseLLMMessage,
|
|
66
|
+
LLMRequestMessage,
|
|
67
|
+
FinishReason,
|
|
68
|
+
LLMResponse,
|
|
69
|
+
StreamChunkError,
|
|
70
|
+
Chunk,
|
|
71
|
+
StreamCallback,
|
|
72
|
+
Tool,
|
|
73
|
+
LLMGenerateOptions,
|
|
74
|
+
LLMRequest,
|
|
75
|
+
AnthropicStreamEvent,
|
|
76
|
+
} from './types';
|
|
@@ -0,0 +1,177 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* 平台信息工具
|
|
3
|
+
*
|
|
4
|
+
* 用于获取客户端平台信息,生成请求头标识
|
|
5
|
+
*/
|
|
6
|
+
|
|
7
|
+
import * as os from 'os';
|
|
8
|
+
import * as fs from 'fs';
|
|
9
|
+
import * as path from 'path';
|
|
10
|
+
import { createHash } from 'crypto';
|
|
11
|
+
|
|
12
|
+
// 从 package.json 读取版本号
|
|
13
|
+
import { readFileSync } from 'fs';
|
|
14
|
+
import { fileURLToPath } from 'url';
|
|
15
|
+
|
|
16
|
+
const PACKAGE_VERSION = (() => {
|
|
17
|
+
try {
|
|
18
|
+
// ESM 环境获取 package.json 路径
|
|
19
|
+
const currentDir = path.dirname(fileURLToPath(import.meta.url));
|
|
20
|
+
const packageJsonPath = path.resolve(currentDir, '../../package.json');
|
|
21
|
+
const content = readFileSync(packageJsonPath, 'utf-8');
|
|
22
|
+
const pkg = JSON.parse(content);
|
|
23
|
+
return pkg.version || '1.0.0';
|
|
24
|
+
} catch {
|
|
25
|
+
return '1.0.0';
|
|
26
|
+
}
|
|
27
|
+
})();
|
|
28
|
+
|
|
29
|
+
/**
|
|
30
|
+
* 获取设备唯一 ID
|
|
31
|
+
*
|
|
32
|
+
* 优先从缓存文件读取,不存在则生成新的并保存
|
|
33
|
+
*/
|
|
34
|
+
function getDeviceId(): string {
|
|
35
|
+
const cacheDir = path.join(os.homedir(), '.renx');
|
|
36
|
+
const deviceIdPath = path.join(cacheDir, 'device_id');
|
|
37
|
+
|
|
38
|
+
try {
|
|
39
|
+
// 尝试读取已存在的设备 ID
|
|
40
|
+
if (fs.existsSync(deviceIdPath)) {
|
|
41
|
+
const id = fs.readFileSync(deviceIdPath, 'utf-8').trim();
|
|
42
|
+
if (id && /^[a-f0-9]{32}$/i.test(id)) {
|
|
43
|
+
return id;
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
// 生成新的设备 ID
|
|
48
|
+
const id = createHash('md5')
|
|
49
|
+
.update(`${os.hostname()}-${os.userInfo().username}-${Date.now()}-${Math.random()}`)
|
|
50
|
+
.digest('hex');
|
|
51
|
+
|
|
52
|
+
// 确保目录存在
|
|
53
|
+
if (!fs.existsSync(cacheDir)) {
|
|
54
|
+
fs.mkdirSync(cacheDir, { recursive: true });
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
// 保存设备 ID(权限 600)
|
|
58
|
+
fs.writeFileSync(deviceIdPath, id, { mode: 0o600 });
|
|
59
|
+
return id;
|
|
60
|
+
} catch {
|
|
61
|
+
// 生成临时 ID(不持久化)
|
|
62
|
+
return createHash('md5').update(`${os.hostname()}-${os.userInfo().username}`).digest('hex');
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
/**
|
|
67
|
+
* 获取设备型号
|
|
68
|
+
*/
|
|
69
|
+
function getDeviceModel(): string {
|
|
70
|
+
const platform = os.platform();
|
|
71
|
+
const arch = os.arch();
|
|
72
|
+
const release = os.release();
|
|
73
|
+
|
|
74
|
+
switch (platform) {
|
|
75
|
+
case 'darwin': {
|
|
76
|
+
const version = release.split('.')[0];
|
|
77
|
+
const macVersion = getMacOSVersion(version);
|
|
78
|
+
return `macOS ${macVersion} ${arch}`;
|
|
79
|
+
}
|
|
80
|
+
case 'win32': {
|
|
81
|
+
const releaseNum = os.release();
|
|
82
|
+
let windowsVersion = releaseNum;
|
|
83
|
+
// Windows 10 build >= 22000 是 Windows 11
|
|
84
|
+
try {
|
|
85
|
+
const build = parseInt(releaseNum.split('.')[2] || '0', 10);
|
|
86
|
+
if (build >= 22000) {
|
|
87
|
+
windowsVersion = '11';
|
|
88
|
+
} else {
|
|
89
|
+
windowsVersion = '10';
|
|
90
|
+
}
|
|
91
|
+
} catch {
|
|
92
|
+
// 忽略解析错误
|
|
93
|
+
}
|
|
94
|
+
return `Windows ${windowsVersion} ${arch}`;
|
|
95
|
+
}
|
|
96
|
+
case 'linux': {
|
|
97
|
+
return `Linux ${release} ${arch}`;
|
|
98
|
+
}
|
|
99
|
+
default:
|
|
100
|
+
return `${platform} ${release} ${arch}`;
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
/**
|
|
105
|
+
* 获取 macOS 版本号
|
|
106
|
+
*/
|
|
107
|
+
function getMacOSVersion(darwinVersion: string): string {
|
|
108
|
+
const major = parseInt(darwinVersion, 10);
|
|
109
|
+
// Darwin 版本到 macOS 版本的映射
|
|
110
|
+
const versionMap: Record<number, string> = {
|
|
111
|
+
23: '14.0', // Sonoma
|
|
112
|
+
22: '13.0', // Ventura
|
|
113
|
+
21: '12.0', // Monterey
|
|
114
|
+
20: '11.0', // Big Sur
|
|
115
|
+
};
|
|
116
|
+
return versionMap[major] || `${major}.0`;
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
/**
|
|
120
|
+
* ASCII 化头部值(移除非 ASCII 字符)
|
|
121
|
+
*/
|
|
122
|
+
function asciiHeaderValue(value: string, fallback = 'unknown'): string {
|
|
123
|
+
try {
|
|
124
|
+
// 检查是否为纯 ASCII
|
|
125
|
+
if (/^[\x20-\x7E]*$/.test(value)) {
|
|
126
|
+
return value;
|
|
127
|
+
}
|
|
128
|
+
// 移除非 ASCII 字符
|
|
129
|
+
const sanitized = value.replace(/[^\x20-\x7E]/g, '').trim();
|
|
130
|
+
return sanitized || fallback;
|
|
131
|
+
} catch {
|
|
132
|
+
return fallback;
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
// 缓存设备 ID(避免重复 I/O)
|
|
137
|
+
let cachedDeviceId: string | null = null;
|
|
138
|
+
|
|
139
|
+
/**
|
|
140
|
+
* 获取公共请求头
|
|
141
|
+
*
|
|
142
|
+
* 用于 Kimi API 请求的客户端标识
|
|
143
|
+
*/
|
|
144
|
+
export function getCommonHeaders(): Record<string, string> {
|
|
145
|
+
const deviceId = cachedDeviceId || (cachedDeviceId = getDeviceId());
|
|
146
|
+
const deviceName = os.hostname() || 'unknown';
|
|
147
|
+
const deviceModel = getDeviceModel();
|
|
148
|
+
const osVersion = os.release();
|
|
149
|
+
const version = PACKAGE_VERSION;
|
|
150
|
+
|
|
151
|
+
return {
|
|
152
|
+
'X-Msh-Platform': 'kimi_cli',
|
|
153
|
+
'X-Msh-Version': asciiHeaderValue(version),
|
|
154
|
+
'X-Msh-Device-Name': asciiHeaderValue(deviceName),
|
|
155
|
+
'X-Msh-Device-Model': asciiHeaderValue(deviceModel),
|
|
156
|
+
'X-Msh-Os-Version': asciiHeaderValue(osVersion),
|
|
157
|
+
'X-Msh-Device-Id': deviceId,
|
|
158
|
+
};
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
/**
|
|
162
|
+
* 获取 User-Agent
|
|
163
|
+
*/
|
|
164
|
+
export function getUserAgent(): string {
|
|
165
|
+
// return 'claude-cli/2.1.19 (external, cli)';
|
|
166
|
+
return `KimiCLI/${PACKAGE_VERSION}`;
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
/**
|
|
170
|
+
* 获取完整的 Kimi 请求头
|
|
171
|
+
*/
|
|
172
|
+
export function getKimiHeaders(): Record<string, string> {
|
|
173
|
+
return {
|
|
174
|
+
...getCommonHeaders(),
|
|
175
|
+
'User-Agent': getUserAgent(),
|
|
176
|
+
};
|
|
177
|
+
}
|