mirror of
https://github.com/Wei-Shaw/claude-relay-service.git
synced 2026-01-22 16:43:35 +00:00
主要变更: 1. **僵尸流看门狗 (Zombie Stream Watchdog)**: - 新增 resetActivityTimeout 机制,45秒无数据强制断开连接,防止服务假死。 2. **智能重试机制**: - 针对 Antigravity 429 (Resource Exhausted) 错误,自动清理会话并切换账号重试。 - 涵盖流式 (Stream) 和非流式 (Non-stream) 请求。 3. **Thought Signature 增强**: - 新增签名缓存与恢复机制 (signatureCache)。 - 增加 skip_thought_signature_validator 兜底签名策略。 - 强制补充 thought: true 标记以满足上游校验。 4. **系统稳定性与调试**: - 使用 util.inspect 替代 JSON.stringify 打印错误日志,彻底修复循环引用导致的服务崩溃。 - 新增针对 Antigravity 参数错误 (400) 的详细请求结构分析日志。 - 优化日志写入为轮转模式 (safeRotatingAppend)。 5. **其他优化**: - antigravityClient 数据处理安全增强 (safeDataToString)。
122 lines
3.4 KiB
JavaScript
122 lines
3.4 KiB
JavaScript
const path = require('path')
|
|
const logger = require('./logger')
|
|
const { getProjectRoot } = require('./projectPaths')
|
|
const { safeRotatingAppend } = require('./safeRotatingAppend')
|
|
|
|
const UPSTREAM_REQUEST_DUMP_ENV = 'ANTIGRAVITY_DEBUG_UPSTREAM_REQUEST_DUMP'
|
|
const UPSTREAM_REQUEST_DUMP_MAX_BYTES_ENV = 'ANTIGRAVITY_DEBUG_UPSTREAM_REQUEST_DUMP_MAX_BYTES'
|
|
const UPSTREAM_REQUEST_DUMP_FILENAME = 'antigravity-upstream-requests-dump.jsonl'
|
|
|
|
function isEnabled() {
|
|
const raw = process.env[UPSTREAM_REQUEST_DUMP_ENV]
|
|
if (!raw) {
|
|
return false
|
|
}
|
|
const normalized = String(raw).trim().toLowerCase()
|
|
return normalized === '1' || normalized === 'true'
|
|
}
|
|
|
|
function getMaxBytes() {
|
|
const raw = process.env[UPSTREAM_REQUEST_DUMP_MAX_BYTES_ENV]
|
|
if (!raw) {
|
|
return 2 * 1024 * 1024
|
|
}
|
|
const parsed = Number.parseInt(raw, 10)
|
|
if (!Number.isFinite(parsed) || parsed <= 0) {
|
|
return 2 * 1024 * 1024
|
|
}
|
|
return parsed
|
|
}
|
|
|
|
function redact(value) {
|
|
if (!value) {
|
|
return value
|
|
}
|
|
const s = String(value)
|
|
if (s.length <= 10) {
|
|
return '***'
|
|
}
|
|
return `${s.slice(0, 3)}...${s.slice(-4)}`
|
|
}
|
|
|
|
function safeJsonStringify(payload, maxBytes) {
|
|
let json = ''
|
|
try {
|
|
json = JSON.stringify(payload)
|
|
} catch (e) {
|
|
return JSON.stringify({
|
|
type: 'antigravity_upstream_dump_error',
|
|
error: 'JSON.stringify_failed',
|
|
message: e?.message || String(e)
|
|
})
|
|
}
|
|
|
|
if (Buffer.byteLength(json, 'utf8') <= maxBytes) {
|
|
return json
|
|
}
|
|
|
|
const truncated = Buffer.from(json, 'utf8').subarray(0, maxBytes).toString('utf8')
|
|
return JSON.stringify({
|
|
type: 'antigravity_upstream_dump_truncated',
|
|
maxBytes,
|
|
originalBytes: Buffer.byteLength(json, 'utf8'),
|
|
partialJson: truncated
|
|
})
|
|
}
|
|
|
|
async function dumpAntigravityUpstreamRequest(requestInfo) {
|
|
if (!isEnabled()) {
|
|
return
|
|
}
|
|
|
|
const maxBytes = getMaxBytes()
|
|
const filename = path.join(getProjectRoot(), UPSTREAM_REQUEST_DUMP_FILENAME)
|
|
|
|
const record = {
|
|
ts: new Date().toISOString(),
|
|
type: 'antigravity_upstream_request',
|
|
requestId: requestInfo?.requestId || null,
|
|
model: requestInfo?.model || null,
|
|
stream: Boolean(requestInfo?.stream),
|
|
url: requestInfo?.url || null,
|
|
baseUrl: requestInfo?.baseUrl || null,
|
|
params: requestInfo?.params || null,
|
|
headers: requestInfo?.headers
|
|
? {
|
|
Host: requestInfo.headers.Host || requestInfo.headers.host || null,
|
|
'User-Agent':
|
|
requestInfo.headers['User-Agent'] || requestInfo.headers['user-agent'] || null,
|
|
Authorization: (() => {
|
|
const raw = requestInfo.headers.Authorization || requestInfo.headers.authorization
|
|
if (!raw) {
|
|
return null
|
|
}
|
|
const value = String(raw)
|
|
const m = value.match(/^Bearer\\s+(.+)$/i)
|
|
const token = m ? m[1] : value
|
|
return `Bearer ${redact(token)}`
|
|
})()
|
|
}
|
|
: null,
|
|
envelope: requestInfo?.envelope || null
|
|
}
|
|
|
|
const line = `${safeJsonStringify(record, maxBytes)}\n`
|
|
try {
|
|
await safeRotatingAppend(filename, line)
|
|
} catch (e) {
|
|
logger.warn('Failed to dump Antigravity upstream request', {
|
|
filename,
|
|
requestId: requestInfo?.requestId || null,
|
|
error: e?.message || String(e)
|
|
})
|
|
}
|
|
}
|
|
|
|
module.exports = {
|
|
dumpAntigravityUpstreamRequest,
|
|
UPSTREAM_REQUEST_DUMP_ENV,
|
|
UPSTREAM_REQUEST_DUMP_MAX_BYTES_ENV,
|
|
UPSTREAM_REQUEST_DUMP_FILENAME
|
|
}
|