wyxrouter 0.4.71
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/CHANGELOG.md +222 -0
- package/LICENSE +21 -0
- package/README.md +96 -0
- package/README.zh-CN.md +1311 -0
- package/assets/pixel-router-2d.png +0 -0
- package/cli/LICENSE +42 -0
- package/cli/README.md +125 -0
- package/cli/app/package.json +58 -0
- package/cli/cli.js +806 -0
- package/cli/package.json +48 -0
- package/i18n/README.ja-JP.md +1209 -0
- package/i18n/README.ru.md +1311 -0
- package/i18n/README.vi.md +1310 -0
- package/i18n/README.zh-CN.md +1306 -0
- package/images/9router.png +0 -0
- package/jsconfig.json +12 -0
- package/next.config.mjs +71 -0
- package/open-sse/config/appConstants.js +203 -0
- package/open-sse/config/codexInstructions.js +119 -0
- package/open-sse/config/constants.js +4 -0
- package/open-sse/config/defaultThinkingSignature.js +12 -0
- package/open-sse/config/errorConfig.js +85 -0
- package/open-sse/config/googleTtsLanguages.js +62 -0
- package/open-sse/config/kiroConstants.js +322 -0
- package/open-sse/config/models.js +13 -0
- package/open-sse/config/ollamaModels.js +19 -0
- package/open-sse/config/providerModels.js +944 -0
- package/open-sse/config/providers.js +458 -0
- package/open-sse/config/runtimeConfig.js +72 -0
- package/open-sse/config/ttsModels.js +129 -0
- package/open-sse/executors/antigravity.js +504 -0
- package/open-sse/executors/azure.js +57 -0
- package/open-sse/executors/base.js +185 -0
- package/open-sse/executors/codex.js +469 -0
- package/open-sse/executors/commandcode.js +88 -0
- package/open-sse/executors/cursor.js +795 -0
- package/open-sse/executors/default.js +497 -0
- package/open-sse/executors/gemini-cli.js +89 -0
- package/open-sse/executors/github.js +379 -0
- package/open-sse/executors/grok-web.js +345 -0
- package/open-sse/executors/iflow.js +108 -0
- package/open-sse/executors/index.js +75 -0
- package/open-sse/executors/kiro.js +508 -0
- package/open-sse/executors/ollama-local.js +14 -0
- package/open-sse/executors/opencode-go.js +41 -0
- package/open-sse/executors/opencode.js +32 -0
- package/open-sse/executors/perplexity-web.js +507 -0
- package/open-sse/executors/qoder.js +450 -0
- package/open-sse/executors/qwen.js +129 -0
- package/open-sse/executors/vertex.js +131 -0
- package/open-sse/executors/xiaomi-tokenplan.js +19 -0
- package/open-sse/handlers/chatCore/nonStreamingHandler.js +230 -0
- package/open-sse/handlers/chatCore/requestDetail.js +102 -0
- package/open-sse/handlers/chatCore/sseToJsonHandler.js +231 -0
- package/open-sse/handlers/chatCore/streamingHandler.js +103 -0
- package/open-sse/handlers/chatCore.js +287 -0
- package/open-sse/handlers/embeddingProviders/_base.js +4 -0
- package/open-sse/handlers/embeddingProviders/gemini.js +54 -0
- package/open-sse/handlers/embeddingProviders/index.js +23 -0
- package/open-sse/handlers/embeddingProviders/openai.js +39 -0
- package/open-sse/handlers/embeddingProviders/openaiCompatNode.js +13 -0
- package/open-sse/handlers/embeddingsCore.js +126 -0
- package/open-sse/handlers/fetch/index.js +237 -0
- package/open-sse/handlers/imageGenerationCore.js +189 -0
- package/open-sse/handlers/imageProviders/_base.js +31 -0
- package/open-sse/handlers/imageProviders/blackForestLabs.js +43 -0
- package/open-sse/handlers/imageProviders/cloudflareAi.js +178 -0
- package/open-sse/handlers/imageProviders/codex.js +198 -0
- package/open-sse/handlers/imageProviders/comfyui.js +8 -0
- package/open-sse/handlers/imageProviders/falAi.js +41 -0
- package/open-sse/handlers/imageProviders/gemini.js +25 -0
- package/open-sse/handlers/imageProviders/huggingface.js +22 -0
- package/open-sse/handlers/imageProviders/index.js +40 -0
- package/open-sse/handlers/imageProviders/nanobanana.js +58 -0
- package/open-sse/handlers/imageProviders/openai.js +40 -0
- package/open-sse/handlers/imageProviders/runwayml.js +47 -0
- package/open-sse/handlers/imageProviders/sdwebui.js +17 -0
- package/open-sse/handlers/imageProviders/stabilityAi.js +34 -0
- package/open-sse/handlers/responsesHandler.js +103 -0
- package/open-sse/handlers/search/callers.js +371 -0
- package/open-sse/handlers/search/chatSearch.js +409 -0
- package/open-sse/handlers/search/index.js +201 -0
- package/open-sse/handlers/search/normalizers.js +223 -0
- package/open-sse/handlers/sttCore.js +194 -0
- package/open-sse/handlers/ttsCore.js +74 -0
- package/open-sse/handlers/ttsProviders/_base.js +39 -0
- package/open-sse/handlers/ttsProviders/edgeTts.js +89 -0
- package/open-sse/handlers/ttsProviders/elevenlabs.js +48 -0
- package/open-sse/handlers/ttsProviders/gemini.js +117 -0
- package/open-sse/handlers/ttsProviders/genericFormats.js +169 -0
- package/open-sse/handlers/ttsProviders/googleTts.js +54 -0
- package/open-sse/handlers/ttsProviders/index.js +50 -0
- package/open-sse/handlers/ttsProviders/localDevice.js +87 -0
- package/open-sse/handlers/ttsProviders/minimax.js +59 -0
- package/open-sse/handlers/ttsProviders/openai.js +30 -0
- package/open-sse/handlers/ttsProviders/openrouter.js +70 -0
- package/open-sse/index.js +82 -0
- package/open-sse/rtk/applyFilter.js +15 -0
- package/open-sse/rtk/autodetect.js +111 -0
- package/open-sse/rtk/caveman.js +100 -0
- package/open-sse/rtk/cavemanPrompts.js +78 -0
- package/open-sse/rtk/constants.js +55 -0
- package/open-sse/rtk/filters/buildOutput.js +127 -0
- package/open-sse/rtk/filters/dedupLog.js +44 -0
- package/open-sse/rtk/filters/find.js +49 -0
- package/open-sse/rtk/filters/gitDiff.js +92 -0
- package/open-sse/rtk/filters/gitStatus.js +117 -0
- package/open-sse/rtk/filters/grep.js +48 -0
- package/open-sse/rtk/filters/ls.js +79 -0
- package/open-sse/rtk/filters/readNumbered.js +27 -0
- package/open-sse/rtk/filters/searchList.js +52 -0
- package/open-sse/rtk/filters/smartTruncate.js +15 -0
- package/open-sse/rtk/filters/tree.js +32 -0
- package/open-sse/rtk/index.js +155 -0
- package/open-sse/rtk/registry.js +38 -0
- package/open-sse/services/accountFallback.js +238 -0
- package/open-sse/services/combo.js +198 -0
- package/open-sse/services/compact.js +71 -0
- package/open-sse/services/kiroModels.js +332 -0
- package/open-sse/services/model.js +261 -0
- package/open-sse/services/oauthCredentialManager.js +151 -0
- package/open-sse/services/projectId.js +306 -0
- package/open-sse/services/provider.js +356 -0
- package/open-sse/services/qoderModels.js +214 -0
- package/open-sse/services/tokenRefresh.js +939 -0
- package/open-sse/services/usage.js +1496 -0
- package/open-sse/transformer/responsesTransformer.js +439 -0
- package/open-sse/transformer/streamToJsonConverter.js +103 -0
- package/open-sse/translator/formats.js +36 -0
- package/open-sse/translator/helpers/claudeHelper.js +216 -0
- package/open-sse/translator/helpers/geminiHelper.js +372 -0
- package/open-sse/translator/helpers/imageHelper.js +34 -0
- package/open-sse/translator/helpers/maxTokensHelper.js +27 -0
- package/open-sse/translator/helpers/openaiHelper.js +130 -0
- package/open-sse/translator/helpers/responsesApiHelper.js +139 -0
- package/open-sse/translator/helpers/toolCallHelper.js +148 -0
- package/open-sse/translator/index.js +251 -0
- package/open-sse/translator/request/antigravity-to-openai.js +229 -0
- package/open-sse/translator/request/claude-to-openai.js +232 -0
- package/open-sse/translator/request/gemini-to-openai.js +147 -0
- package/open-sse/translator/request/openai-responses.js +318 -0
- package/open-sse/translator/request/openai-to-claude.js +401 -0
- package/open-sse/translator/request/openai-to-commandcode.js +170 -0
- package/open-sse/translator/request/openai-to-cursor.js +183 -0
- package/open-sse/translator/request/openai-to-gemini.js +470 -0
- package/open-sse/translator/request/openai-to-kiro.js +629 -0
- package/open-sse/translator/request/openai-to-kiro.old.js +278 -0
- package/open-sse/translator/request/openai-to-ollama.js +192 -0
- package/open-sse/translator/request/openai-to-vertex.js +42 -0
- package/open-sse/translator/response/claude-to-openai.js +206 -0
- package/open-sse/translator/response/commandcode-to-openai.js +197 -0
- package/open-sse/translator/response/cursor-to-openai.js +30 -0
- package/open-sse/translator/response/gemini-to-openai.js +245 -0
- package/open-sse/translator/response/kiro-to-openai.js +195 -0
- package/open-sse/translator/response/ollama-to-openai.js +152 -0
- package/open-sse/translator/response/openai-responses.js +590 -0
- package/open-sse/translator/response/openai-to-antigravity.js +122 -0
- package/open-sse/translator/response/openai-to-claude.js +266 -0
- package/open-sse/utils/bypassHandler.js +298 -0
- package/open-sse/utils/claudeCloaking.js +155 -0
- package/open-sse/utils/claudeHeaderCache.js +70 -0
- package/open-sse/utils/clientDetector.js +63 -0
- package/open-sse/utils/cursorChecksum.js +149 -0
- package/open-sse/utils/cursorProtobuf.js +904 -0
- package/open-sse/utils/debugLog.js +14 -0
- package/open-sse/utils/error.js +147 -0
- package/open-sse/utils/ollamaTransform.js +85 -0
- package/open-sse/utils/proxyFetch.js +368 -0
- package/open-sse/utils/reasoningContentInjector.js +79 -0
- package/open-sse/utils/requestLogger.js +260 -0
- package/open-sse/utils/responsesStreamHelpers.js +49 -0
- package/open-sse/utils/sessionManager.js +82 -0
- package/open-sse/utils/stream.js +462 -0
- package/open-sse/utils/streamHandler.js +250 -0
- package/open-sse/utils/streamHelpers.js +122 -0
- package/open-sse/utils/toolDeduper.js +49 -0
- package/open-sse/utils/usageTracking.js +347 -0
- package/package.json +100 -0
- package/postcss.config.mjs +12 -0
- package/public/favicon.svg +11 -0
- package/public/file.svg +1 -0
- package/public/globe.svg +1 -0
- package/public/i18n/literals/ar.json +195 -0
- package/public/i18n/literals/bn.json +195 -0
- package/public/i18n/literals/cs.json +195 -0
- package/public/i18n/literals/da.json +195 -0
- package/public/i18n/literals/de.json +195 -0
- package/public/i18n/literals/el.json +195 -0
- package/public/i18n/literals/es.json +195 -0
- package/public/i18n/literals/fi.json +195 -0
- package/public/i18n/literals/fr.json +195 -0
- package/public/i18n/literals/he.json +195 -0
- package/public/i18n/literals/hi.json +195 -0
- package/public/i18n/literals/hu.json +195 -0
- package/public/i18n/literals/id.json +195 -0
- package/public/i18n/literals/it.json +195 -0
- package/public/i18n/literals/ja.json +195 -0
- package/public/i18n/literals/ko.json +195 -0
- package/public/i18n/literals/nl.json +195 -0
- package/public/i18n/literals/no.json +195 -0
- package/public/i18n/literals/pl.json +195 -0
- package/public/i18n/literals/pt-BR.json +195 -0
- package/public/i18n/literals/pt-PT.json +195 -0
- package/public/i18n/literals/ro.json +195 -0
- package/public/i18n/literals/ru.json +195 -0
- package/public/i18n/literals/sv.json +195 -0
- package/public/i18n/literals/th.json +195 -0
- package/public/i18n/literals/tl.json +195 -0
- package/public/i18n/literals/tr.json +195 -0
- package/public/i18n/literals/uk.json +195 -0
- package/public/i18n/literals/ur.json +195 -0
- package/public/i18n/literals/vi.json +195 -0
- package/public/i18n/literals/zh-CN.json +772 -0
- package/public/i18n/literals/zh-TW.json +195 -0
- package/public/icons/discord.svg +4 -0
- package/public/icons/icon-192.svg +4 -0
- package/public/icons/icon-512.svg +4 -0
- package/public/next.svg +1 -0
- package/public/providers/alicode-intl.png +0 -0
- package/public/providers/alicode.png +0 -0
- package/public/providers/amp.png +0 -0
- package/public/providers/anthropic-m.png +0 -0
- package/public/providers/anthropic.png +0 -0
- package/public/providers/antigravity.png +0 -0
- package/public/providers/assemblyai.png +0 -0
- package/public/providers/aws-polly.png +0 -0
- package/public/providers/azure.png +0 -0
- package/public/providers/black-forest-labs.png +0 -0
- package/public/providers/blackbox.png +0 -0
- package/public/providers/brave-search.png +0 -0
- package/public/providers/byteplus.png +0 -0
- package/public/providers/cartesia.png +0 -0
- package/public/providers/cerebras.png +0 -0
- package/public/providers/chutes.png +0 -0
- package/public/providers/claude.png +0 -0
- package/public/providers/cline.png +0 -0
- package/public/providers/cloudflare-ai.png +0 -0
- package/public/providers/codebuddy.svg +37 -0
- package/public/providers/codex.png +0 -0
- package/public/providers/cohere.png +0 -0
- package/public/providers/comfyui.png +0 -0
- package/public/providers/commandcode.png +0 -0
- package/public/providers/continue.png +0 -0
- package/public/providers/copilot.png +0 -0
- package/public/providers/coqui.png +0 -0
- package/public/providers/cursor.png +0 -0
- package/public/providers/deepgram.png +0 -0
- package/public/providers/deepseek-tui.png +0 -0
- package/public/providers/deepseek.png +0 -0
- package/public/providers/droid.png +0 -0
- package/public/providers/edge-tts.png +0 -0
- package/public/providers/elevenlabs.png +0 -0
- package/public/providers/exa.png +0 -0
- package/public/providers/fal-ai.png +0 -0
- package/public/providers/firecrawl.png +0 -0
- package/public/providers/fireworks.png +0 -0
- package/public/providers/gemini-cli.png +0 -0
- package/public/providers/gemini.png +0 -0
- package/public/providers/github.png +0 -0
- package/public/providers/glm-cn.png +0 -0
- package/public/providers/glm.png +0 -0
- package/public/providers/google-pse.png +0 -0
- package/public/providers/google-tts.png +0 -0
- package/public/providers/grok-web.png +0 -0
- package/public/providers/groq.png +0 -0
- package/public/providers/hermes.png +0 -0
- package/public/providers/huggingface.png +0 -0
- package/public/providers/hyperbolic.png +0 -0
- package/public/providers/iflow.png +0 -0
- package/public/providers/inworld.png +0 -0
- package/public/providers/jcode.png +0 -0
- package/public/providers/jina-ai.png +0 -0
- package/public/providers/jina-reader.png +0 -0
- package/public/providers/kilocode.png +0 -0
- package/public/providers/kimi-coding.png +0 -0
- package/public/providers/kimi.png +0 -0
- package/public/providers/kiro.png +0 -0
- package/public/providers/linkup.png +0 -0
- package/public/providers/local-device.png +0 -0
- package/public/providers/minimax-cn.png +0 -0
- package/public/providers/minimax.png +0 -0
- package/public/providers/mistral.png +0 -0
- package/public/providers/nanobanana.png +0 -0
- package/public/providers/nebius.png +0 -0
- package/public/providers/nvidia.png +0 -0
- package/public/providers/oai-cc.png +0 -0
- package/public/providers/oai-r.png +0 -0
- package/public/providers/ollama-local.png +0 -0
- package/public/providers/ollama.png +0 -0
- package/public/providers/openai.png +0 -0
- package/public/providers/openclaw.png +0 -0
- package/public/providers/opencode-go.png +0 -0
- package/public/providers/opencode.png +0 -0
- package/public/providers/openrouter.png +0 -0
- package/public/providers/perplexity-web.png +0 -0
- package/public/providers/perplexity.png +0 -0
- package/public/providers/playht.png +0 -0
- package/public/providers/qoder.png +0 -0
- package/public/providers/qwen.png +0 -0
- package/public/providers/recraft.png +0 -0
- package/public/providers/roo.png +0 -0
- package/public/providers/runwayml.png +0 -0
- package/public/providers/sdwebui.png +0 -0
- package/public/providers/searchapi.png +0 -0
- package/public/providers/searxng.png +0 -0
- package/public/providers/serper.png +0 -0
- package/public/providers/siliconflow.png +0 -0
- package/public/providers/stability-ai.png +0 -0
- package/public/providers/tavily.png +0 -0
- package/public/providers/together.png +0 -0
- package/public/providers/topaz.png +0 -0
- package/public/providers/tortoise.png +0 -0
- package/public/providers/vertex-partner.png +0 -0
- package/public/providers/vertex.png +0 -0
- package/public/providers/volcengine-ark.png +0 -0
- package/public/providers/voyage-ai.png +0 -0
- package/public/providers/xai.png +0 -0
- package/public/providers/xiaomi-mimo.png +0 -0
- package/public/providers/xiaomi-tokenplan.png +0 -0
- package/public/providers/youcom.png +0 -0
- package/public/sw.js +22 -0
- package/public/vercel.svg +1 -0
- package/public/window.svg +1 -0
- package/scripts/compact-request-details.mjs +71 -0
- package/scripts/import-codex-gptjson.mjs +342 -0
- package/scripts/start-standalone.mjs +25 -0
- package/scripts/translate-readme.js +201 -0
- package/src/app/(dashboard)/dashboard/automation/page.js +294 -0
- package/src/app/(dashboard)/dashboard/basic-chat/BasicChatPageClient.js +967 -0
- package/src/app/(dashboard)/dashboard/basic-chat/page.js +5 -0
- package/src/app/(dashboard)/dashboard/cli-tools/CLIToolsPageClient.js +66 -0
- package/src/app/(dashboard)/dashboard/cli-tools/[toolId]/ToolDetailClient.js +173 -0
- package/src/app/(dashboard)/dashboard/cli-tools/[toolId]/page.js +11 -0
- package/src/app/(dashboard)/dashboard/cli-tools/components/AntigravityToolCard.js +481 -0
- package/src/app/(dashboard)/dashboard/cli-tools/components/ApiKeySelect.js +66 -0
- package/src/app/(dashboard)/dashboard/cli-tools/components/BaseUrlSelect.js +174 -0
- package/src/app/(dashboard)/dashboard/cli-tools/components/ClaudeToolCard.js +390 -0
- package/src/app/(dashboard)/dashboard/cli-tools/components/ClineToolCard.js +301 -0
- package/src/app/(dashboard)/dashboard/cli-tools/components/CodexToolCard.js +458 -0
- package/src/app/(dashboard)/dashboard/cli-tools/components/CopilotToolCard.js +323 -0
- package/src/app/(dashboard)/dashboard/cli-tools/components/CoworkToolCard.js +640 -0
- package/src/app/(dashboard)/dashboard/cli-tools/components/DeepSeekTuiToolCard.js +338 -0
- package/src/app/(dashboard)/dashboard/cli-tools/components/DefaultToolCard.js +271 -0
- package/src/app/(dashboard)/dashboard/cli-tools/components/DroidToolCard.js +410 -0
- package/src/app/(dashboard)/dashboard/cli-tools/components/EndpointPresetControl.js +128 -0
- package/src/app/(dashboard)/dashboard/cli-tools/components/HermesToolCard.js +317 -0
- package/src/app/(dashboard)/dashboard/cli-tools/components/JcodeToolCard.js +380 -0
- package/src/app/(dashboard)/dashboard/cli-tools/components/KiloToolCard.js +275 -0
- package/src/app/(dashboard)/dashboard/cli-tools/components/MitmLinkCard.js +40 -0
- package/src/app/(dashboard)/dashboard/cli-tools/components/MitmServerCard.js +329 -0
- package/src/app/(dashboard)/dashboard/cli-tools/components/MitmToolCard.js +318 -0
- package/src/app/(dashboard)/dashboard/cli-tools/components/OpenClawToolCard.js +388 -0
- package/src/app/(dashboard)/dashboard/cli-tools/components/OpenCodeToolCard.js +500 -0
- package/src/app/(dashboard)/dashboard/cli-tools/components/ToolSummaryCard.js +39 -0
- package/src/app/(dashboard)/dashboard/cli-tools/components/cliEndpointMatch.js +13 -0
- package/src/app/(dashboard)/dashboard/cli-tools/components/index.js +19 -0
- package/src/app/(dashboard)/dashboard/cli-tools/page.js +7 -0
- package/src/app/(dashboard)/dashboard/combos/page.js +612 -0
- package/src/app/(dashboard)/dashboard/console-log/ConsoleLogClient.js +91 -0
- package/src/app/(dashboard)/dashboard/console-log/page.js +8 -0
- package/src/app/(dashboard)/dashboard/endpoint/EndpointPageClient.js +1555 -0
- package/src/app/(dashboard)/dashboard/endpoint/page.js +7 -0
- package/src/app/(dashboard)/dashboard/media-providers/[kind]/[id]/page.js +1903 -0
- package/src/app/(dashboard)/dashboard/media-providers/[kind]/page.js +289 -0
- package/src/app/(dashboard)/dashboard/media-providers/combo/[id]/page.js +410 -0
- package/src/app/(dashboard)/dashboard/media-providers/web/page.js +209 -0
- package/src/app/(dashboard)/dashboard/mitm/MitmPageClient.js +117 -0
- package/src/app/(dashboard)/dashboard/mitm/page.js +5 -0
- package/src/app/(dashboard)/dashboard/page.js +7 -0
- package/src/app/(dashboard)/dashboard/profile/page.js +1140 -0
- package/src/app/(dashboard)/dashboard/providers/[id]/AddApiKeyModal.js +389 -0
- package/src/app/(dashboard)/dashboard/providers/[id]/AddCustomModelModal.js +125 -0
- package/src/app/(dashboard)/dashboard/providers/[id]/CompatibleModelsSection.js +250 -0
- package/src/app/(dashboard)/dashboard/providers/[id]/ConnectionRow.js +299 -0
- package/src/app/(dashboard)/dashboard/providers/[id]/CooldownTimer.js +42 -0
- package/src/app/(dashboard)/dashboard/providers/[id]/EditCompatibleNodeModal.js +161 -0
- package/src/app/(dashboard)/dashboard/providers/[id]/ModelRow.js +95 -0
- package/src/app/(dashboard)/dashboard/providers/[id]/PassthroughModelsSection.js +183 -0
- package/src/app/(dashboard)/dashboard/providers/[id]/page.js +2053 -0
- package/src/app/(dashboard)/dashboard/providers/[id]/page.new.js +1724 -0
- package/src/app/(dashboard)/dashboard/providers/components/ConnectionsCard.js +497 -0
- package/src/app/(dashboard)/dashboard/providers/components/ModelAvailabilityBadge.js +185 -0
- package/src/app/(dashboard)/dashboard/providers/components/ModelsCard.js +294 -0
- package/src/app/(dashboard)/dashboard/providers/new/page.js +220 -0
- package/src/app/(dashboard)/dashboard/providers/page.js +1365 -0
- package/src/app/(dashboard)/dashboard/proxy-pools/page.js +1092 -0
- package/src/app/(dashboard)/dashboard/quota/page.js +11 -0
- package/src/app/(dashboard)/dashboard/skills/page.js +112 -0
- package/src/app/(dashboard)/dashboard/translator/page.js +303 -0
- package/src/app/(dashboard)/dashboard/usage/components/OverviewCards.js +35 -0
- package/src/app/(dashboard)/dashboard/usage/components/ProviderLimits/ProviderLimitCard.js +185 -0
- package/src/app/(dashboard)/dashboard/usage/components/ProviderLimits/QuotaProgressBar.js +127 -0
- package/src/app/(dashboard)/dashboard/usage/components/ProviderLimits/QuotaTable.js +259 -0
- package/src/app/(dashboard)/dashboard/usage/components/ProviderLimits/index.js +1394 -0
- package/src/app/(dashboard)/dashboard/usage/components/ProviderLimits/utils.js +244 -0
- package/src/app/(dashboard)/dashboard/usage/components/ProviderTopology.js +327 -0
- package/src/app/(dashboard)/dashboard/usage/components/RequestDetailsTab.js +433 -0
- package/src/app/(dashboard)/dashboard/usage/components/UsageChart.js +141 -0
- package/src/app/(dashboard)/dashboard/usage/components/UsageTable.js +247 -0
- package/src/app/(dashboard)/dashboard/usage/page.js +75 -0
- package/src/app/(dashboard)/layout.js +6 -0
- package/src/app/api/auth/login/route.js +76 -0
- package/src/app/api/auth/logout/route.js +12 -0
- package/src/app/api/auth/oidc/callback/route.js +87 -0
- package/src/app/api/auth/oidc/start/route.js +52 -0
- package/src/app/api/auth/oidc/test/route.js +84 -0
- package/src/app/api/auth/status/route.js +45 -0
- package/src/app/api/cli-tools/all-statuses/route.js +46 -0
- package/src/app/api/cli-tools/antigravity-mitm/alias/route.js +53 -0
- package/src/app/api/cli-tools/antigravity-mitm/route.js +202 -0
- package/src/app/api/cli-tools/claude-settings/route.js +203 -0
- package/src/app/api/cli-tools/cline-settings/route.js +133 -0
- package/src/app/api/cli-tools/codex-gateway/accounts/route.js +16 -0
- package/src/app/api/cli-tools/codex-settings/route.js +239 -0
- package/src/app/api/cli-tools/copilot-settings/route.js +148 -0
- package/src/app/api/cli-tools/cowork-mcp-registry/route.js +77 -0
- package/src/app/api/cli-tools/cowork-mcp-tools/route.js +95 -0
- package/src/app/api/cli-tools/cowork-settings/route.js +412 -0
- package/src/app/api/cli-tools/deepseek-tui-settings/route.js +164 -0
- package/src/app/api/cli-tools/droid-settings/route.js +213 -0
- package/src/app/api/cli-tools/hermes-settings/route.js +175 -0
- package/src/app/api/cli-tools/jcode-settings/route.js +216 -0
- package/src/app/api/cli-tools/kilo-settings/route.js +131 -0
- package/src/app/api/cli-tools/openclaw-settings/route.js +292 -0
- package/src/app/api/cli-tools/opencode-settings/route.js +259 -0
- package/src/app/api/combos/[id]/route.js +81 -0
- package/src/app/api/combos/route.js +48 -0
- package/src/app/api/health/route.js +15 -0
- package/src/app/api/init/route.js +4 -0
- package/src/app/api/keys/[id]/route.js +58 -0
- package/src/app/api/keys/route.js +42 -0
- package/src/app/api/locale/route.js +30 -0
- package/src/app/api/mcp/[plugin]/message/route.js +21 -0
- package/src/app/api/mcp/[plugin]/sse/route.js +37 -0
- package/src/app/api/media-providers/tts/deepgram/voices/route.js +65 -0
- package/src/app/api/media-providers/tts/elevenlabs/voices/route.js +71 -0
- package/src/app/api/media-providers/tts/inworld/voices/route.js +61 -0
- package/src/app/api/media-providers/tts/minimax/voices/route.js +113 -0
- package/src/app/api/media-providers/tts/voices/route.js +99 -0
- package/src/app/api/models/alias/route.js +53 -0
- package/src/app/api/models/availability/route.js +103 -0
- package/src/app/api/models/custom/route.js +48 -0
- package/src/app/api/models/disabled/route.js +50 -0
- package/src/app/api/models/route.js +64 -0
- package/src/app/api/models/test/ping.js +191 -0
- package/src/app/api/models/test/route.js +14 -0
- package/src/app/api/oauth/[provider]/[action]/route.js +343 -0
- package/src/app/api/oauth/codebuddy/bulk-import/[jobId]/cancel/route.js +19 -0
- package/src/app/api/oauth/codebuddy/bulk-import/[jobId]/manual/[workerId]/route.js +30 -0
- package/src/app/api/oauth/codebuddy/bulk-import/[jobId]/route.js +23 -0
- package/src/app/api/oauth/codebuddy/bulk-import/latest/route.js +25 -0
- package/src/app/api/oauth/codebuddy/bulk-import/route.js +49 -0
- package/src/app/api/oauth/codebuddy/quota-cookie/route.js +133 -0
- package/src/app/api/oauth/codex/import-token/route.js +96 -0
- package/src/app/api/oauth/cursor/auto-import/route.js +258 -0
- package/src/app/api/oauth/cursor/import/route.js +100 -0
- package/src/app/api/oauth/gitlab/pat/route.js +62 -0
- package/src/app/api/oauth/iflow/cookie/route.js +137 -0
- package/src/app/api/oauth/kiro/auto-import/route.js +85 -0
- package/src/app/api/oauth/kiro/bulk-import/[jobId]/cancel/route.js +18 -0
- package/src/app/api/oauth/kiro/bulk-import/[jobId]/manual/[workerId]/route.js +29 -0
- package/src/app/api/oauth/kiro/bulk-import/[jobId]/route.js +22 -0
- package/src/app/api/oauth/kiro/bulk-import/latest/route.js +25 -0
- package/src/app/api/oauth/kiro/bulk-import/route.js +49 -0
- package/src/app/api/oauth/kiro/import/route.js +110 -0
- package/src/app/api/oauth/kiro/social-authorize/route.js +27 -0
- package/src/app/api/oauth/kiro/social-exchange/route.js +41 -0
- package/src/app/api/pricing/route.js +134 -0
- package/src/app/api/provider-nodes/[id]/route.js +101 -0
- package/src/app/api/provider-nodes/route.js +104 -0
- package/src/app/api/provider-nodes/validate/route.js +201 -0
- package/src/app/api/providers/[id]/models/route.js +526 -0
- package/src/app/api/providers/[id]/route.js +189 -0
- package/src/app/api/providers/[id]/test/route.js +23 -0
- package/src/app/api/providers/[id]/test/testUtils.js +714 -0
- package/src/app/api/providers/[id]/test-models/route.js +66 -0
- package/src/app/api/providers/client/route.js +126 -0
- package/src/app/api/providers/kilo/free-models/route.js +55 -0
- package/src/app/api/providers/route.js +206 -0
- package/src/app/api/providers/suggested-models/filters.js +20 -0
- package/src/app/api/providers/suggested-models/route.js +32 -0
- package/src/app/api/providers/test-batch/route.js +131 -0
- package/src/app/api/providers/validate/route.js +637 -0
- package/src/app/api/proxy-pools/[id]/route.js +123 -0
- package/src/app/api/proxy-pools/[id]/test/route.js +70 -0
- package/src/app/api/proxy-pools/cloudflare-deploy/route.js +145 -0
- package/src/app/api/proxy-pools/deno-deploy/route.js +175 -0
- package/src/app/api/proxy-pools/route.js +93 -0
- package/src/app/api/proxy-pools/vercel-deploy/route.js +142 -0
- package/src/app/api/settings/database/route.js +36 -0
- package/src/app/api/settings/proxy-test/route.js +23 -0
- package/src/app/api/settings/require-login/route.js +15 -0
- package/src/app/api/settings/route.js +100 -0
- package/src/app/api/shutdown/route.js +24 -0
- package/src/app/api/tags/route.js +18 -0
- package/src/app/api/translator/console-logs/route.js +24 -0
- package/src/app/api/translator/console-logs/stream/route.js +79 -0
- package/src/app/api/translator/load/route.js +45 -0
- package/src/app/api/translator/save/route.js +44 -0
- package/src/app/api/translator/send/route.js +94 -0
- package/src/app/api/translator/translate/route.js +90 -0
- package/src/app/api/tunnel/disable/route.js +12 -0
- package/src/app/api/tunnel/enable/route.js +16 -0
- package/src/app/api/tunnel/status/route.js +13 -0
- package/src/app/api/tunnel/tailscale-check/route.js +50 -0
- package/src/app/api/tunnel/tailscale-disable/route.js +12 -0
- package/src/app/api/tunnel/tailscale-enable/route.js +12 -0
- package/src/app/api/tunnel/tailscale-install/route.js +72 -0
- package/src/app/api/usage/[connectionId]/route.js +188 -0
- package/src/app/api/usage/chart/route.js +21 -0
- package/src/app/api/usage/history/route.js +12 -0
- package/src/app/api/usage/logs/route.js +12 -0
- package/src/app/api/usage/providers/route.js +42 -0
- package/src/app/api/usage/request-details/route.js +57 -0
- package/src/app/api/usage/request-logs/route.js +13 -0
- package/src/app/api/usage/stats/route.js +23 -0
- package/src/app/api/usage/stream/route.js +79 -0
- package/src/app/api/v1/api/chat/route.js +37 -0
- package/src/app/api/v1/audio/speech/route.js +16 -0
- package/src/app/api/v1/audio/transcriptions/route.js +19 -0
- package/src/app/api/v1/audio/voices/route.js +68 -0
- package/src/app/api/v1/chat/completions/route.js +35 -0
- package/src/app/api/v1/embeddings/route.js +21 -0
- package/src/app/api/v1/images/generations/route.js +16 -0
- package/src/app/api/v1/messages/count_tokens/route.js +52 -0
- package/src/app/api/v1/messages/route.js +36 -0
- package/src/app/api/v1/models/[kind]/route.js +55 -0
- package/src/app/api/v1/models/info/route.js +110 -0
- package/src/app/api/v1/models/route.js +451 -0
- package/src/app/api/v1/responses/compact/route.js +37 -0
- package/src/app/api/v1/responses/route.js +30 -0
- package/src/app/api/v1/route.js +1 -0
- package/src/app/api/v1/search/route.js +21 -0
- package/src/app/api/v1/web/fetch/route.js +21 -0
- package/src/app/api/v1beta/models/[...path]/route.js +328 -0
- package/src/app/api/v1beta/models/route.js +44 -0
- package/src/app/api/version/route.js +45 -0
- package/src/app/api/version/shutdown/route.js +15 -0
- package/src/app/api/version/update/route.js +21 -0
- package/src/app/callback/page.js +148 -0
- package/src/app/dashboard/settings/pricing/page.js +173 -0
- package/src/app/favicon.ico +0 -0
- package/src/app/globals.css +496 -0
- package/src/app/landing/components/AnimatedBackground.js +57 -0
- package/src/app/landing/components/Features.js +133 -0
- package/src/app/landing/components/FlowAnimation.js +175 -0
- package/src/app/landing/components/Footer.js +61 -0
- package/src/app/landing/components/GetStarted.js +97 -0
- package/src/app/landing/components/HeroSection.js +47 -0
- package/src/app/landing/components/HowItWorks.js +66 -0
- package/src/app/landing/components/Navigation.js +72 -0
- package/src/app/landing/page.js +106 -0
- package/src/app/layout.js +49 -0
- package/src/app/login/page.js +197 -0
- package/src/app/manifest.js +30 -0
- package/src/app/page.js +5 -0
- package/src/dashboardGuard.js +242 -0
- package/src/i18n/RuntimeI18nProvider.js +27 -0
- package/src/i18n/config.js +146 -0
- package/src/i18n/runtime.js +162 -0
- package/src/lib/appUpdater.js +200 -0
- package/src/lib/auth/dashboardSession.js +68 -0
- package/src/lib/auth/loginLimiter.js +52 -0
- package/src/lib/auth/oidc.js +234 -0
- package/src/lib/consoleLogBuffer.js +79 -0
- package/src/lib/dataDir.js +29 -0
- package/src/lib/db/adapters/betterSqliteAdapter.js +55 -0
- package/src/lib/db/adapters/bunSqliteAdapter.js +63 -0
- package/src/lib/db/adapters/nodeSqliteAdapter.js +84 -0
- package/src/lib/db/adapters/sqljsAdapter.js +115 -0
- package/src/lib/db/backup.js +35 -0
- package/src/lib/db/driver.js +85 -0
- package/src/lib/db/helpers/jsonCol.js +9 -0
- package/src/lib/db/helpers/kvStore.js +39 -0
- package/src/lib/db/helpers/metaStore.js +22 -0
- package/src/lib/db/index.js +171 -0
- package/src/lib/db/migrate.js +286 -0
- package/src/lib/db/migrations/001-initial.js +14 -0
- package/src/lib/db/migrations/index.js +10 -0
- package/src/lib/db/paths.js +18 -0
- package/src/lib/db/repos/aliasRepo.js +62 -0
- package/src/lib/db/repos/apiKeysRepo.js +75 -0
- package/src/lib/db/repos/combosRepo.js +73 -0
- package/src/lib/db/repos/connectionsRepo.js +226 -0
- package/src/lib/db/repos/disabledModelsRepo.js +56 -0
- package/src/lib/db/repos/nodesRepo.js +95 -0
- package/src/lib/db/repos/pricingRepo.js +108 -0
- package/src/lib/db/repos/proxyPoolsRepo.js +103 -0
- package/src/lib/db/repos/requestDetailsRepo.js +259 -0
- package/src/lib/db/repos/settingsRepo.js +104 -0
- package/src/lib/db/repos/usageRepo.js +731 -0
- package/src/lib/db/schema.js +157 -0
- package/src/lib/db/version.js +21 -0
- package/src/lib/disabledModelsDb.js +4 -0
- package/src/lib/localDb.js +21 -0
- package/src/lib/mcp/stdioSseBridge.js +198 -0
- package/src/lib/mitmAliasCache.js +46 -0
- package/src/lib/network/connectionProxy.js +160 -0
- package/src/lib/network/initOutboundProxy.js +25 -0
- package/src/lib/network/outboundProxy.js +68 -0
- package/src/lib/network/proxyTest.js +91 -0
- package/src/lib/oauth/constants/oauth.js +284 -0
- package/src/lib/oauth/constants/xai.js +61 -0
- package/src/lib/oauth/providers.js +1506 -0
- package/src/lib/oauth/services/antigravity.js +321 -0
- package/src/lib/oauth/services/claude.js +136 -0
- package/src/lib/oauth/services/codebuddyBulkImportManager.js +454 -0
- package/src/lib/oauth/services/codex.js +144 -0
- package/src/lib/oauth/services/cursor.js +179 -0
- package/src/lib/oauth/services/gemini.js +240 -0
- package/src/lib/oauth/services/github.js +225 -0
- package/src/lib/oauth/services/iflow.js +202 -0
- package/src/lib/oauth/services/index.js +17 -0
- package/src/lib/oauth/services/kiro.js +334 -0
- package/src/lib/oauth/services/kiroBulkImportManager.js +778 -0
- package/src/lib/oauth/services/kiroConnections.js +93 -0
- package/src/lib/oauth/services/kiroGoogleAutomation.js +1136 -0
- package/src/lib/oauth/services/oauth.js +157 -0
- package/src/lib/oauth/services/openai.js +123 -0
- package/src/lib/oauth/services/qoder.js +216 -0
- package/src/lib/oauth/services/qwen.js +170 -0
- package/src/lib/oauth/services/xai.js +238 -0
- package/src/lib/oauth/utils/banner.js +63 -0
- package/src/lib/oauth/utils/pkce.js +39 -0
- package/src/lib/oauth/utils/server.js +415 -0
- package/src/lib/oauth/utils/ui.js +48 -0
- package/src/lib/providerNormalization.js +45 -0
- package/src/lib/qoder/constants.js +64 -0
- package/src/lib/qoder/cosy.js +175 -0
- package/src/lib/qoder/encoding.js +55 -0
- package/src/lib/requestDetailsDb.js +4 -0
- package/src/lib/tunnel/cloudflare/cloudflared.js +449 -0
- package/src/lib/tunnel/cloudflare/config.js +9 -0
- package/src/lib/tunnel/cloudflare/healthCheck.js +29 -0
- package/src/lib/tunnel/cloudflare/manager.js +151 -0
- package/src/lib/tunnel/cloudflare/pid.js +23 -0
- package/src/lib/tunnel/index.js +48 -0
- package/src/lib/tunnel/shared/dnsResolver.js +17 -0
- package/src/lib/tunnel/shared/internetCheck.js +26 -0
- package/src/lib/tunnel/shared/state.js +41 -0
- package/src/lib/tunnel/shared/watchdogConfig.js +8 -0
- package/src/lib/tunnel/tailscale/config.js +7 -0
- package/src/lib/tunnel/tailscale/healthCheck.js +29 -0
- package/src/lib/tunnel/tailscale/manager.js +129 -0
- package/src/lib/tunnel/tailscale/tailscale.js +790 -0
- package/src/lib/updater/updater.js +235 -0
- package/src/lib/usage/fetcher.js +208 -0
- package/src/lib/usageDb.js +7 -0
- package/src/mitm/antigravityIdeVersion.js +50 -0
- package/src/mitm/cert/generate.js +32 -0
- package/src/mitm/cert/install.js +269 -0
- package/src/mitm/cert/rootCA.js +173 -0
- package/src/mitm/config.js +87 -0
- package/src/mitm/dbReader.js +22 -0
- package/src/mitm/dns/dnsConfig.js +266 -0
- package/src/mitm/handlers/antigravity.js +33 -0
- package/src/mitm/handlers/base.js +226 -0
- package/src/mitm/handlers/copilot.js +35 -0
- package/src/mitm/handlers/cursor.js +15 -0
- package/src/mitm/handlers/kiro.js +526 -0
- package/src/mitm/logger.js +106 -0
- package/src/mitm/manager.js +851 -0
- package/src/mitm/paths.js +32 -0
- package/src/mitm/server.js +435 -0
- package/src/mitm/winElevated.js +81 -0
- package/src/models/index.js +38 -0
- package/src/proxy.js +5 -0
- package/src/shared/components/AddCustomEmbeddingModal.js +183 -0
- package/src/shared/components/Avatar.js +88 -0
- package/src/shared/components/Badge.js +54 -0
- package/src/shared/components/BulkAccountAutomationModal.js +508 -0
- package/src/shared/components/Button.js +56 -0
- package/src/shared/components/Card.js +116 -0
- package/src/shared/components/ChangelogModal.js +97 -0
- package/src/shared/components/CodeBuddyQuotaCookieModal.js +109 -0
- package/src/shared/components/ComboFormModal.js +176 -0
- package/src/shared/components/CursorAuthModal.js +212 -0
- package/src/shared/components/DonateModal.js +136 -0
- package/src/shared/components/Drawer.js +82 -0
- package/src/shared/components/EditConnectionModal.js +286 -0
- package/src/shared/components/Footer.js +132 -0
- package/src/shared/components/GitLabAuthModal.js +194 -0
- package/src/shared/components/Header.js +380 -0
- package/src/shared/components/HeaderLanguage.js +46 -0
- package/src/shared/components/HeaderMenu.js +126 -0
- package/src/shared/components/IFlowCookieModal.js +132 -0
- package/src/shared/components/Input.js +65 -0
- package/src/shared/components/KiroAuthModal.js +1171 -0
- package/src/shared/components/KiroOAuthWrapper.js +149 -0
- package/src/shared/components/KiroSocialOAuthModal.js +205 -0
- package/src/shared/components/LanguageSwitcher.js +190 -0
- package/src/shared/components/Loading.js +75 -0
- package/src/shared/components/ManualConfigModal.js +44 -0
- package/src/shared/components/McpMarketplaceModal.js +255 -0
- package/src/shared/components/Modal.js +146 -0
- package/src/shared/components/ModelSelectModal.js +537 -0
- package/src/shared/components/NineRemoteButton.js +23 -0
- package/src/shared/components/NineRemotePromoModal.js +99 -0
- package/src/shared/components/NoAuthProxyCard.js +86 -0
- package/src/shared/components/OAuthModal.js +682 -0
- package/src/shared/components/Pagination.js +150 -0
- package/src/shared/components/PricingModal.js +208 -0
- package/src/shared/components/ProviderIcon.js +63 -0
- package/src/shared/components/ProviderInfoCard.js +82 -0
- package/src/shared/components/RequestLogger.js +121 -0
- package/src/shared/components/SegmentedControl.js +48 -0
- package/src/shared/components/Select.js +67 -0
- package/src/shared/components/Sidebar.js +441 -0
- package/src/shared/components/ThemeProvider.js +15 -0
- package/src/shared/components/ThemeToggle.js +42 -0
- package/src/shared/components/Toggle.js +69 -0
- package/src/shared/components/Tooltip.js +25 -0
- package/src/shared/components/UsageStats.js +505 -0
- package/src/shared/components/index.js +46 -0
- package/src/shared/components/layouts/AuthLayout.js +29 -0
- package/src/shared/components/layouts/DashboardLayout.js +104 -0
- package/src/shared/components/layouts/index.js +4 -0
- package/src/shared/constants/cliTools.js +397 -0
- package/src/shared/constants/colors.js +77 -0
- package/src/shared/constants/config.js +99 -0
- package/src/shared/constants/coworkPlugins.js +75 -0
- package/src/shared/constants/index.js +4 -0
- package/src/shared/constants/locales.js +36 -0
- package/src/shared/constants/mitmToolHosts.js +12 -0
- package/src/shared/constants/models.js +38 -0
- package/src/shared/constants/pricing.js +303 -0
- package/src/shared/constants/providers.js +289 -0
- package/src/shared/constants/skills.js +78 -0
- package/src/shared/constants/ttsProviders.js +138 -0
- package/src/shared/hooks/index.js +2 -0
- package/src/shared/hooks/useCopyToClipboard.js +43 -0
- package/src/shared/hooks/useTheme.js +60 -0
- package/src/shared/services/bootstrap.js +12 -0
- package/src/shared/services/initializeApp.js +268 -0
- package/src/shared/utils/api.js +93 -0
- package/src/shared/utils/apiKey.js +98 -0
- package/src/shared/utils/clineAuth.js +37 -0
- package/src/shared/utils/cn.js +11 -0
- package/src/shared/utils/connectionStatus.js +162 -0
- package/src/shared/utils/index.js +40 -0
- package/src/shared/utils/machine.js +6 -0
- package/src/shared/utils/machineId.js +66 -0
- package/src/shared/utils/providerModelsFetcher.js +30 -0
- package/src/sse/handlers/chat.js +261 -0
- package/src/sse/handlers/embeddings.js +141 -0
- package/src/sse/handlers/fetch.js +213 -0
- package/src/sse/handlers/imageGeneration.js +142 -0
- package/src/sse/handlers/search.js +206 -0
- package/src/sse/handlers/stt.js +88 -0
- package/src/sse/handlers/tts.js +114 -0
- package/src/sse/services/auth.js +346 -0
- package/src/sse/services/codexGateway.js +215 -0
- package/src/sse/services/model.js +99 -0
- package/src/sse/services/tokenRefresh.js +319 -0
- package/src/sse/utils/logger.js +75 -0
- package/src/store/headerSearchStore.js +19 -0
- package/src/store/index.js +6 -0
- package/src/store/notificationStore.js +45 -0
- package/src/store/providerStore.js +55 -0
- package/src/store/settingsStore.js +51 -0
- package/src/store/themeStore.js +54 -0
- package/src/store/userStore.js +20 -0
- package/start.sh +4 -0
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
// Debug logging utility — only active in dev mode (NODE_ENV !== "production")
|
|
2
|
+
// Outputs are tagged with [DBG:tag] for easy grep/filter
|
|
3
|
+
const isDev = process.env.NODE_ENV !== "production";
|
|
4
|
+
|
|
5
|
+
function ts() {
|
|
6
|
+
return new Date().toLocaleTimeString("en-US", { hour12: false, hour: "2-digit", minute: "2-digit", second: "2-digit" });
|
|
7
|
+
}
|
|
8
|
+
|
|
9
|
+
export function dbg(tag, msg) {
|
|
10
|
+
if (!isDev) return;
|
|
11
|
+
console.log(`[${ts()}] 🐛 [DBG:${tag}] ${msg}`);
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
export const isDebugEnabled = isDev;
|
|
@@ -0,0 +1,147 @@
|
|
|
1
|
+
import { ERROR_TYPES, DEFAULT_ERROR_MESSAGES } from "../config/errorConfig.js";
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Build OpenAI-compatible error response body
|
|
5
|
+
* @param {number} statusCode - HTTP status code
|
|
6
|
+
* @param {string} message - Error message
|
|
7
|
+
* @returns {object} Error response object
|
|
8
|
+
*/
|
|
9
|
+
export function buildErrorBody(statusCode, message) {
|
|
10
|
+
const errorInfo = ERROR_TYPES[statusCode] ||
|
|
11
|
+
(statusCode >= 500
|
|
12
|
+
? { type: "server_error", code: "internal_server_error" }
|
|
13
|
+
: { type: "invalid_request_error", code: "" });
|
|
14
|
+
|
|
15
|
+
return {
|
|
16
|
+
error: {
|
|
17
|
+
message: message || DEFAULT_ERROR_MESSAGES[statusCode] || "An error occurred",
|
|
18
|
+
type: errorInfo.type,
|
|
19
|
+
code: errorInfo.code
|
|
20
|
+
}
|
|
21
|
+
};
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
/**
|
|
25
|
+
* Create error Response object (for non-streaming)
|
|
26
|
+
* @param {number} statusCode - HTTP status code
|
|
27
|
+
* @param {string} message - Error message
|
|
28
|
+
* @returns {Response} HTTP Response object
|
|
29
|
+
*/
|
|
30
|
+
export function errorResponse(statusCode, message) {
|
|
31
|
+
return new Response(JSON.stringify(buildErrorBody(statusCode, message)), {
|
|
32
|
+
status: statusCode,
|
|
33
|
+
headers: {
|
|
34
|
+
"Content-Type": "application/json",
|
|
35
|
+
"Access-Control-Allow-Origin": "*"
|
|
36
|
+
}
|
|
37
|
+
});
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
/**
|
|
41
|
+
* Write error to SSE stream (for streaming)
|
|
42
|
+
* @param {WritableStreamDefaultWriter} writer - Stream writer
|
|
43
|
+
* @param {number} statusCode - HTTP status code
|
|
44
|
+
* @param {string} message - Error message
|
|
45
|
+
*/
|
|
46
|
+
export async function writeStreamError(writer, statusCode, message) {
|
|
47
|
+
const errorBody = buildErrorBody(statusCode, message);
|
|
48
|
+
const encoder = new TextEncoder();
|
|
49
|
+
await writer.write(encoder.encode(`data: ${JSON.stringify(errorBody)}\n\n`));
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
/**
|
|
53
|
+
* Parse upstream provider error response
|
|
54
|
+
* @param {Response} response - Fetch response from provider
|
|
55
|
+
* @param {object} [executor] - Optional executor with parseError() override for provider-specific parsing
|
|
56
|
+
* @returns {Promise<{statusCode: number, message: string, resetsAtMs?: number}>}
|
|
57
|
+
*/
|
|
58
|
+
export async function parseUpstreamError(response, executor = null) {
|
|
59
|
+
let bodyText = "";
|
|
60
|
+
try {
|
|
61
|
+
bodyText = await response.text();
|
|
62
|
+
} catch {
|
|
63
|
+
bodyText = "";
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
// Let executor-specific parser extract provider-specific fields (e.g. codex resetsAtMs)
|
|
67
|
+
if (executor && typeof executor.parseError === "function") {
|
|
68
|
+
try {
|
|
69
|
+
const parsed = executor.parseError(response, bodyText);
|
|
70
|
+
if (parsed && typeof parsed === "object") {
|
|
71
|
+
const msg = parsed.message || DEFAULT_ERROR_MESSAGES[response.status] || `Upstream error: ${response.status}`;
|
|
72
|
+
return { statusCode: parsed.status || response.status, message: msg, resetsAtMs: parsed.resetsAtMs };
|
|
73
|
+
}
|
|
74
|
+
} catch { /* fall through to default parsing */ }
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
let message = "";
|
|
78
|
+
try {
|
|
79
|
+
const json = JSON.parse(bodyText);
|
|
80
|
+
message = json.error?.message || json.message || json.error || bodyText;
|
|
81
|
+
} catch {
|
|
82
|
+
message = bodyText;
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
const messageStr = typeof message === "string" ? message : JSON.stringify(message);
|
|
86
|
+
const finalMessage = messageStr || DEFAULT_ERROR_MESSAGES[response.status] || `Upstream error: ${response.status}`;
|
|
87
|
+
|
|
88
|
+
return { statusCode: response.status, message: finalMessage };
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
/**
|
|
92
|
+
* Create error result for chatCore handler
|
|
93
|
+
* @param {number} statusCode - HTTP status code
|
|
94
|
+
* @param {string} message - Error message
|
|
95
|
+
* @param {number} [resetsAtMs] - Optional precise cooldown expiry (ms epoch) for provider-specific quota errors
|
|
96
|
+
* @returns {{ success: false, status: number, error: string, response: Response, resetsAtMs?: number }}
|
|
97
|
+
*/
|
|
98
|
+
export function createErrorResult(statusCode, message, resetsAtMs) {
|
|
99
|
+
return {
|
|
100
|
+
success: false,
|
|
101
|
+
status: statusCode,
|
|
102
|
+
error: message,
|
|
103
|
+
resetsAtMs,
|
|
104
|
+
response: errorResponse(statusCode, message)
|
|
105
|
+
};
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
/**
|
|
109
|
+
* Create unavailable response when all accounts are rate limited
|
|
110
|
+
* @param {number} statusCode - Original error status code
|
|
111
|
+
* @param {string} message - Error message (without retry info)
|
|
112
|
+
* @param {string} retryAfter - ISO timestamp when earliest account becomes available
|
|
113
|
+
* @param {string} retryAfterHuman - Human-readable retry info e.g. "reset after 30s"
|
|
114
|
+
* @returns {Response}
|
|
115
|
+
*/
|
|
116
|
+
export function unavailableResponse(statusCode, message, retryAfter, retryAfterHuman) {
|
|
117
|
+
const retryAfterSec = Math.max(Math.ceil((new Date(retryAfter).getTime() - Date.now()) / 1000), 1);
|
|
118
|
+
const msg = `${message} (${retryAfterHuman})`;
|
|
119
|
+
return new Response(
|
|
120
|
+
JSON.stringify({ error: { message: msg } }),
|
|
121
|
+
{
|
|
122
|
+
status: statusCode,
|
|
123
|
+
headers: {
|
|
124
|
+
"Content-Type": "application/json",
|
|
125
|
+
"Retry-After": String(retryAfterSec)
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
);
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
/**
|
|
132
|
+
* Format provider error with context
|
|
133
|
+
* @param {Error} error - Original error
|
|
134
|
+
* @param {string} provider - Provider name
|
|
135
|
+
* @param {string} model - Model name
|
|
136
|
+
* @param {number|string} statusCode - HTTP status code or error code
|
|
137
|
+
* @returns {string} Formatted error message
|
|
138
|
+
*/
|
|
139
|
+
export function formatProviderError(error, provider, model, statusCode) {
|
|
140
|
+
const code = statusCode || error.code || "FETCH_FAILED";
|
|
141
|
+
const message = error.message || "Unknown error";
|
|
142
|
+
// Expose low-level cause (e.g. UND_ERR_SOCKET, ECONNRESET, ETIMEDOUT) for diagnosing fetch failures
|
|
143
|
+
const causeCode = error.cause?.code;
|
|
144
|
+
const causeMsg = error.cause?.message;
|
|
145
|
+
const causeStr = causeCode || causeMsg ? ` (cause: ${[causeCode, causeMsg].filter(Boolean).join(": ")})` : "";
|
|
146
|
+
return `[${code}]: ${message}${causeStr}`;
|
|
147
|
+
}
|
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
// Transform OpenAI SSE stream to Ollama JSON lines format
|
|
2
|
+
export function transformToOllama(response, model) {
|
|
3
|
+
let buffer = "";
|
|
4
|
+
let pendingToolCalls = {};
|
|
5
|
+
|
|
6
|
+
const transform = new TransformStream({
|
|
7
|
+
transform(chunk, controller) {
|
|
8
|
+
const text = new TextDecoder().decode(chunk);
|
|
9
|
+
buffer += text;
|
|
10
|
+
const lines = buffer.split("\n");
|
|
11
|
+
buffer = lines.pop() || "";
|
|
12
|
+
|
|
13
|
+
for (const line of lines) {
|
|
14
|
+
if (!line.startsWith("data:")) continue;
|
|
15
|
+
const data = line.slice(5).trim();
|
|
16
|
+
|
|
17
|
+
if (data === "[DONE]") {
|
|
18
|
+
const ollamaEnd = JSON.stringify({ model, message: { role: "assistant", content: "" }, done: true }) + "\n";
|
|
19
|
+
controller.enqueue(new TextEncoder().encode(ollamaEnd));
|
|
20
|
+
return;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
try {
|
|
24
|
+
const parsed = JSON.parse(data);
|
|
25
|
+
const delta = parsed.choices?.[0]?.delta || {};
|
|
26
|
+
const content = delta.content || "";
|
|
27
|
+
const toolCalls = delta.tool_calls;
|
|
28
|
+
|
|
29
|
+
if (toolCalls) {
|
|
30
|
+
for (const tc of toolCalls) {
|
|
31
|
+
const idx = tc.index;
|
|
32
|
+
if (!pendingToolCalls[idx]) {
|
|
33
|
+
pendingToolCalls[idx] = { id: tc.id, function: { name: "", arguments: "" } };
|
|
34
|
+
}
|
|
35
|
+
if (tc.function?.name) pendingToolCalls[idx].function.name += tc.function.name;
|
|
36
|
+
if (tc.function?.arguments) pendingToolCalls[idx].function.arguments += tc.function.arguments;
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
if (content) {
|
|
41
|
+
const ollama = JSON.stringify({ model, message: { role: "assistant", content }, done: false }) + "\n";
|
|
42
|
+
controller.enqueue(new TextEncoder().encode(ollama));
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
const finishReason = parsed.choices?.[0]?.finish_reason;
|
|
46
|
+
if (finishReason === "tool_calls" || finishReason === "stop") {
|
|
47
|
+
const toolCallsArr = Object.values(pendingToolCalls);
|
|
48
|
+
if (toolCallsArr.length > 0) {
|
|
49
|
+
const formattedCalls = toolCallsArr.map(tc => ({
|
|
50
|
+
function: {
|
|
51
|
+
name: tc.function.name,
|
|
52
|
+
arguments: (() => { try { return JSON.parse(tc.function.arguments || "{}"); } catch { return {}; } })()
|
|
53
|
+
}
|
|
54
|
+
}));
|
|
55
|
+
const ollama = JSON.stringify({
|
|
56
|
+
model,
|
|
57
|
+
message: { role: "assistant", content: "", tool_calls: formattedCalls },
|
|
58
|
+
done: true
|
|
59
|
+
}) + "\n";
|
|
60
|
+
controller.enqueue(new TextEncoder().encode(ollama));
|
|
61
|
+
pendingToolCalls = {};
|
|
62
|
+
} else if (finishReason === "stop") {
|
|
63
|
+
const ollamaEnd = JSON.stringify({ model, message: { role: "assistant", content: "" }, done: true }) + "\n";
|
|
64
|
+
controller.enqueue(new TextEncoder().encode(ollamaEnd));
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
} catch (e) {
|
|
68
|
+
// Silently ignore parse errors
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
},
|
|
72
|
+
flush(controller) {
|
|
73
|
+
const ollamaEnd = JSON.stringify({ model, message: { role: "assistant", content: "" }, done: true }) + "\n";
|
|
74
|
+
controller.enqueue(new TextEncoder().encode(ollamaEnd));
|
|
75
|
+
}
|
|
76
|
+
});
|
|
77
|
+
|
|
78
|
+
if (!response.body) {
|
|
79
|
+
return new Response("", { status: response.status, headers: { "Content-Type": "application/x-ndjson" } });
|
|
80
|
+
}
|
|
81
|
+
return new Response(response.body.pipeThrough(transform), {
|
|
82
|
+
headers: { "Content-Type": "application/x-ndjson", "Access-Control-Allow-Origin": "*" }
|
|
83
|
+
});
|
|
84
|
+
}
|
|
85
|
+
|
|
@@ -0,0 +1,368 @@
|
|
|
1
|
+
import { Readable } from "stream";
|
|
2
|
+
import { MEMORY_CONFIG } from "../config/runtimeConfig.js";
|
|
3
|
+
import { dbg } from "./debugLog.js";
|
|
4
|
+
|
|
5
|
+
const originalFetch = globalThis.fetch;
|
|
6
|
+
const proxyDispatchers = new Map();
|
|
7
|
+
|
|
8
|
+
// ─── TLS fingerprinting via got-scraping (browser-like JA3) ───────────────
|
|
9
|
+
// Disabled: not in use. Kept commented for future re-enable.
|
|
10
|
+
// Restore the original block to re-enable per-host JA3 spoofing.
|
|
11
|
+
/*
|
|
12
|
+
let _gotScraping = null;
|
|
13
|
+
let _gotScrapingChecked = false;
|
|
14
|
+
const _gotScrapingLoggedHosts = new Set();
|
|
15
|
+
|
|
16
|
+
async function getGotScraping() {
|
|
17
|
+
if (_gotScrapingChecked) return _gotScraping;
|
|
18
|
+
_gotScrapingChecked = true;
|
|
19
|
+
try {
|
|
20
|
+
const mod = await import("got-scraping");
|
|
21
|
+
_gotScraping = typeof mod.gotScraping === "function" ? mod.gotScraping : null;
|
|
22
|
+
if (_gotScraping) dbg("TLS", "got-scraping loaded (browser-like JA3 enabled)");
|
|
23
|
+
} catch (e) {
|
|
24
|
+
console.warn(`[ProxyFetch] got-scraping unavailable, falling back to native fetch: ${e.message}`);
|
|
25
|
+
_gotScraping = null;
|
|
26
|
+
}
|
|
27
|
+
return _gotScraping;
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
async function gotScrapingFetch(url, options) {
|
|
31
|
+
const gs = await getGotScraping();
|
|
32
|
+
if (!gs) return null;
|
|
33
|
+
|
|
34
|
+
const method = (options.method || "GET").toUpperCase();
|
|
35
|
+
const headersInit = options.headers || {};
|
|
36
|
+
const headers = headersInit instanceof Headers
|
|
37
|
+
? Object.fromEntries(headersInit.entries())
|
|
38
|
+
: { ...headersInit };
|
|
39
|
+
|
|
40
|
+
return new Promise((resolve, reject) => {
|
|
41
|
+
let settled = false;
|
|
42
|
+
const stream = gs.stream({
|
|
43
|
+
url,
|
|
44
|
+
method,
|
|
45
|
+
headers,
|
|
46
|
+
body: method === "GET" || method === "HEAD" ? undefined : options.body,
|
|
47
|
+
throwHttpErrors: false,
|
|
48
|
+
retry: { limit: 0 },
|
|
49
|
+
timeout: { request: undefined },
|
|
50
|
+
followRedirect: false,
|
|
51
|
+
decompress: true,
|
|
52
|
+
});
|
|
53
|
+
|
|
54
|
+
if (options.signal) {
|
|
55
|
+
const onAbort = () => { try { stream.destroy(new Error("aborted")); } catch { } };
|
|
56
|
+
if (options.signal.aborted) onAbort();
|
|
57
|
+
else options.signal.addEventListener("abort", onAbort, { once: true });
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
stream.once("response", (res) => {
|
|
61
|
+
if (settled) return;
|
|
62
|
+
settled = true;
|
|
63
|
+
const resHeaders = new Headers();
|
|
64
|
+
for (const [k, v] of Object.entries(res.headers || {})) {
|
|
65
|
+
if (Array.isArray(v)) v.forEach((x) => resHeaders.append(k, String(x)));
|
|
66
|
+
else if (v != null) resHeaders.set(k, String(v));
|
|
67
|
+
}
|
|
68
|
+
const body = Readable.toWeb(stream);
|
|
69
|
+
resolve(new Response(body, { status: res.statusCode, statusText: res.statusMessage || "", headers: resHeaders }));
|
|
70
|
+
});
|
|
71
|
+
|
|
72
|
+
stream.once("error", (err) => {
|
|
73
|
+
if (settled) return;
|
|
74
|
+
settled = true;
|
|
75
|
+
reject(err);
|
|
76
|
+
});
|
|
77
|
+
});
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
async function tryGotScrapingFetch(url, options) {
|
|
81
|
+
try {
|
|
82
|
+
const res = await gotScrapingFetch(url, options);
|
|
83
|
+
if (res) {
|
|
84
|
+
try {
|
|
85
|
+
const host = new URL(typeof url === "string" ? url : url.toString()).hostname;
|
|
86
|
+
if (!_gotScrapingLoggedHosts.has(host)) {
|
|
87
|
+
_gotScrapingLoggedHosts.add(host);
|
|
88
|
+
dbg("TLS", `using got-scraping for ${host}`);
|
|
89
|
+
}
|
|
90
|
+
} catch { }
|
|
91
|
+
}
|
|
92
|
+
return res;
|
|
93
|
+
} catch (e) {
|
|
94
|
+
console.warn(`[ProxyFetch] got-scraping request failed, fallback to native fetch: ${e.message}`);
|
|
95
|
+
return null;
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
*/
|
|
99
|
+
|
|
100
|
+
// DNS cache — use Map to avoid prototype pollution via malformed hostnames
|
|
101
|
+
const DNS_CACHE = new Map();
|
|
102
|
+
const MITM_BYPASS_HOSTS = [
|
|
103
|
+
"cloudcode-pa.googleapis.com",
|
|
104
|
+
"daily-cloudcode-pa.googleapis.com",
|
|
105
|
+
"api.individual.githubcopilot.com",
|
|
106
|
+
"q.us-east-1.amazonaws.com",
|
|
107
|
+
"codewhisperer.us-east-1.amazonaws.com",
|
|
108
|
+
"api2.cursor.sh",
|
|
109
|
+
];
|
|
110
|
+
const GOOGLE_DNS_SERVERS = ["8.8.8.8", "8.8.4.4"];
|
|
111
|
+
const HTTPS_PORT = 443;
|
|
112
|
+
const HTTP_SUCCESS_MIN = 200;
|
|
113
|
+
const HTTP_SUCCESS_MAX = 300;
|
|
114
|
+
|
|
115
|
+
function normalizeString(value) {
|
|
116
|
+
if (value === undefined || value === null) return "";
|
|
117
|
+
return String(value).trim();
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
/**
|
|
121
|
+
* Resolve real IP using Google DNS (bypass system DNS)
|
|
122
|
+
*/
|
|
123
|
+
async function resolveRealIP(hostname) {
|
|
124
|
+
const cached = DNS_CACHE.get(hostname);
|
|
125
|
+
if (cached && Date.now() < cached.expiry) return cached.ip;
|
|
126
|
+
|
|
127
|
+
try {
|
|
128
|
+
const dns = await import("dns");
|
|
129
|
+
const { promisify } = await import("util");
|
|
130
|
+
const resolver = new dns.Resolver();
|
|
131
|
+
resolver.setServers(GOOGLE_DNS_SERVERS);
|
|
132
|
+
const resolve4 = promisify(resolver.resolve4.bind(resolver));
|
|
133
|
+
const addresses = await resolve4(hostname);
|
|
134
|
+
DNS_CACHE.set(hostname, { ip: addresses[0], expiry: Date.now() + MEMORY_CONFIG.dnsCacheTtlMs });
|
|
135
|
+
return addresses[0];
|
|
136
|
+
} catch (error) {
|
|
137
|
+
console.warn(`[ProxyFetch] DNS resolve failed for ${hostname}:`, error.message);
|
|
138
|
+
return null;
|
|
139
|
+
}
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
/**
|
|
143
|
+
* Check if request should bypass MITM DNS redirect
|
|
144
|
+
*/
|
|
145
|
+
function shouldBypassMitmDns(url) {
|
|
146
|
+
try {
|
|
147
|
+
const hostname = new URL(url).hostname;
|
|
148
|
+
return MITM_BYPASS_HOSTS.some(host => hostname.includes(host));
|
|
149
|
+
} catch { return false; }
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
function shouldBypassByNoProxy(targetUrl, noProxyValue) {
|
|
153
|
+
const noProxy = normalizeString(noProxyValue);
|
|
154
|
+
if (!noProxy) return false;
|
|
155
|
+
|
|
156
|
+
let hostname;
|
|
157
|
+
try { hostname = new URL(targetUrl).hostname.toLowerCase(); } catch { return false; }
|
|
158
|
+
const patterns = noProxy.split(",").map((p) => p.trim().toLowerCase()).filter(Boolean);
|
|
159
|
+
|
|
160
|
+
return patterns.some((pattern) => {
|
|
161
|
+
if (pattern === "*") return true;
|
|
162
|
+
if (pattern.startsWith(".")) return hostname.endsWith(pattern) || hostname === pattern.slice(1);
|
|
163
|
+
return hostname === pattern || hostname.endsWith(`.${pattern}`);
|
|
164
|
+
});
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
/**
|
|
168
|
+
* Get proxy URL from environment
|
|
169
|
+
*/
|
|
170
|
+
function getEnvProxyUrl(targetUrl) {
|
|
171
|
+
const noProxy = process.env.NO_PROXY || process.env.no_proxy;
|
|
172
|
+
if (shouldBypassByNoProxy(targetUrl, noProxy)) return null;
|
|
173
|
+
|
|
174
|
+
let protocol;
|
|
175
|
+
try { protocol = new URL(targetUrl).protocol; } catch { return null; }
|
|
176
|
+
|
|
177
|
+
if (protocol === "https:") {
|
|
178
|
+
return process.env.HTTPS_PROXY || process.env.https_proxy ||
|
|
179
|
+
process.env.ALL_PROXY || process.env.all_proxy;
|
|
180
|
+
}
|
|
181
|
+
|
|
182
|
+
return process.env.HTTP_PROXY || process.env.http_proxy ||
|
|
183
|
+
process.env.ALL_PROXY || process.env.all_proxy;
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
/**
|
|
187
|
+
* Normalize proxy URL (allow host:port)
|
|
188
|
+
*/
|
|
189
|
+
function normalizeProxyUrl(proxyUrl) {
|
|
190
|
+
const normalizedInput = normalizeString(proxyUrl);
|
|
191
|
+
if (!normalizedInput) return null;
|
|
192
|
+
|
|
193
|
+
try {
|
|
194
|
+
|
|
195
|
+
new URL(normalizedInput);
|
|
196
|
+
return normalizedInput;
|
|
197
|
+
} catch {
|
|
198
|
+
// Allow "127.0.0.1:7890" style values
|
|
199
|
+
return `http://${normalizedInput}`;
|
|
200
|
+
}
|
|
201
|
+
}
|
|
202
|
+
|
|
203
|
+
function resolveConnectionProxyUrl(targetUrl, proxyOptions) {
|
|
204
|
+
const enabled = proxyOptions?.enabled === true || proxyOptions?.connectionProxyEnabled === true;
|
|
205
|
+
if (!enabled) return null;
|
|
206
|
+
|
|
207
|
+
const proxyUrlRaw = normalizeString(proxyOptions?.url ?? proxyOptions?.connectionProxyUrl);
|
|
208
|
+
if (!proxyUrlRaw) return null;
|
|
209
|
+
|
|
210
|
+
const noProxy = normalizeString(proxyOptions?.noProxy ?? proxyOptions?.connectionNoProxy);
|
|
211
|
+
if (noProxy && shouldBypassByNoProxy(targetUrl, noProxy)) return null;
|
|
212
|
+
|
|
213
|
+
return normalizeProxyUrl(proxyUrlRaw);
|
|
214
|
+
}
|
|
215
|
+
|
|
216
|
+
/**
|
|
217
|
+
* Create proxy dispatcher lazily (undici-compatible)
|
|
218
|
+
*/
|
|
219
|
+
async function getDispatcher(proxyUrl) {
|
|
220
|
+
const normalized = normalizeProxyUrl(proxyUrl);
|
|
221
|
+
if (!normalized) return null;
|
|
222
|
+
|
|
223
|
+
if (!proxyDispatchers.has(normalized)) {
|
|
224
|
+
// Evict oldest entry if max size reached
|
|
225
|
+
if (proxyDispatchers.size >= MEMORY_CONFIG.proxyDispatchersMaxSize) {
|
|
226
|
+
proxyDispatchers.delete(proxyDispatchers.keys().next().value);
|
|
227
|
+
}
|
|
228
|
+
const { ProxyAgent } = await import("undici");
|
|
229
|
+
proxyDispatchers.set(normalized, new ProxyAgent({ uri: normalized }));
|
|
230
|
+
}
|
|
231
|
+
|
|
232
|
+
return proxyDispatchers.get(normalized);
|
|
233
|
+
}
|
|
234
|
+
|
|
235
|
+
/**
|
|
236
|
+
* Create HTTPS request with manual socket connection (bypass DNS)
|
|
237
|
+
*/
|
|
238
|
+
async function createBypassRequest(parsedUrl, realIP, options) {
|
|
239
|
+
const httpsModule = await import("https");
|
|
240
|
+
const netModule = await import("net");
|
|
241
|
+
// CJS modules expose exports via .default in ESM dynamic import context
|
|
242
|
+
const https = httpsModule.default ?? httpsModule;
|
|
243
|
+
const net = netModule.default ?? netModule;
|
|
244
|
+
|
|
245
|
+
return new Promise((resolve, reject) => {
|
|
246
|
+
const socket = new net.Socket();
|
|
247
|
+
|
|
248
|
+
socket.connect(HTTPS_PORT, realIP, () => {
|
|
249
|
+
const reqOptions = {
|
|
250
|
+
socket,
|
|
251
|
+
// SNI + cert hostname are validated against the hostname the caller
|
|
252
|
+
// asked for, not the IP we connected to. This keeps the DNS-bypass
|
|
253
|
+
// (avoiding /etc/hosts MITM) while still rejecting on-path attackers
|
|
254
|
+
// that present a different cert. The MITM_BYPASS_HOSTS targets are
|
|
255
|
+
// all public-CA-issued (Google / GitHub / AWS / Cursor) so default
|
|
256
|
+
// verification works without any extra trust store.
|
|
257
|
+
servername: parsedUrl.hostname,
|
|
258
|
+
path: parsedUrl.pathname + parsedUrl.search,
|
|
259
|
+
method: options.method || "POST",
|
|
260
|
+
headers: {
|
|
261
|
+
...options.headers,
|
|
262
|
+
Host: parsedUrl.hostname,
|
|
263
|
+
},
|
|
264
|
+
};
|
|
265
|
+
|
|
266
|
+
const req = https.request(reqOptions, (res) => {
|
|
267
|
+
const response = {
|
|
268
|
+
ok: res.statusCode >= HTTP_SUCCESS_MIN && res.statusCode < HTTP_SUCCESS_MAX,
|
|
269
|
+
status: res.statusCode,
|
|
270
|
+
statusText: res.statusMessage,
|
|
271
|
+
headers: new Map(Object.entries(res.headers)),
|
|
272
|
+
body: Readable.toWeb(res),
|
|
273
|
+
text: async () => {
|
|
274
|
+
const chunks = [];
|
|
275
|
+
for await (const chunk of res) chunks.push(chunk);
|
|
276
|
+
return Buffer.concat(chunks).toString();
|
|
277
|
+
},
|
|
278
|
+
json: async () => JSON.parse(await response.text()),
|
|
279
|
+
};
|
|
280
|
+
resolve(response);
|
|
281
|
+
});
|
|
282
|
+
|
|
283
|
+
req.on("error", reject);
|
|
284
|
+
if (options.body) {
|
|
285
|
+
req.write(typeof options.body === "string" ? options.body : JSON.stringify(options.body));
|
|
286
|
+
}
|
|
287
|
+
req.end();
|
|
288
|
+
});
|
|
289
|
+
|
|
290
|
+
socket.on("error", reject);
|
|
291
|
+
});
|
|
292
|
+
}
|
|
293
|
+
|
|
294
|
+
export async function proxyAwareFetch(url, options = {}, proxyOptions = null) {
|
|
295
|
+
const targetUrl = typeof url === "string" ? url : url.toString();
|
|
296
|
+
|
|
297
|
+
// Vercel relay: forward request via relay headers
|
|
298
|
+
const vercelRelayUrl = normalizeString(proxyOptions?.vercelRelayUrl);
|
|
299
|
+
if (vercelRelayUrl) {
|
|
300
|
+
const parsed = new URL(targetUrl);
|
|
301
|
+
const relayHeaders = {
|
|
302
|
+
...options.headers,
|
|
303
|
+
"x-relay-target": `${parsed.protocol}//${parsed.host}`,
|
|
304
|
+
"x-relay-path": `${parsed.pathname}${parsed.search}`,
|
|
305
|
+
};
|
|
306
|
+
return originalFetch(vercelRelayUrl, { ...options, headers: relayHeaders });
|
|
307
|
+
}
|
|
308
|
+
|
|
309
|
+
const connectionProxyUrl = resolveConnectionProxyUrl(targetUrl, proxyOptions);
|
|
310
|
+
const envProxyUrl = connectionProxyUrl ? null : normalizeProxyUrl(getEnvProxyUrl(targetUrl));
|
|
311
|
+
const proxyUrl = connectionProxyUrl || envProxyUrl;
|
|
312
|
+
|
|
313
|
+
// MITM DNS bypass: for known MITM-intercepted hosts, resolve real IP to avoid DNS spoof
|
|
314
|
+
if (shouldBypassMitmDns(targetUrl)) {
|
|
315
|
+
if (proxyUrl) {
|
|
316
|
+
// Proxy resolves DNS externally (not affected by /etc/hosts) — use proxy directly
|
|
317
|
+
try {
|
|
318
|
+
const dispatcher = await getDispatcher(proxyUrl);
|
|
319
|
+
return await originalFetch(url, { ...options, dispatcher });
|
|
320
|
+
} catch (proxyError) {
|
|
321
|
+
if (proxyOptions?.strictProxy === true) {
|
|
322
|
+
throw new Error(`[ProxyFetch] Proxy required but failed (strictProxy=true): ${proxyError.message}`);
|
|
323
|
+
}
|
|
324
|
+
console.warn(`[ProxyFetch] Proxy failed, falling back to direct bypass: ${proxyError.message}`);
|
|
325
|
+
}
|
|
326
|
+
}
|
|
327
|
+
// No proxy — manually resolve real IP to bypass DNS spoof
|
|
328
|
+
try {
|
|
329
|
+
const parsedUrl = new URL(targetUrl);
|
|
330
|
+
const realIP = await resolveRealIP(parsedUrl.hostname);
|
|
331
|
+
if (realIP) return await createBypassRequest(parsedUrl, realIP, options);
|
|
332
|
+
} catch (error) {
|
|
333
|
+
console.warn(`[ProxyFetch] MITM bypass failed: ${error.message}`);
|
|
334
|
+
}
|
|
335
|
+
}
|
|
336
|
+
|
|
337
|
+
if (proxyUrl) {
|
|
338
|
+
try {
|
|
339
|
+
const dispatcher = await getDispatcher(proxyUrl);
|
|
340
|
+
return await originalFetch(url, { ...options, dispatcher });
|
|
341
|
+
} catch (proxyError) {
|
|
342
|
+
// If strictProxy is enabled, fail hard instead of falling back to direct
|
|
343
|
+
if (proxyOptions?.strictProxy === true) {
|
|
344
|
+
throw new Error(`[ProxyFetch] Proxy required but failed (strictProxy=true): ${proxyError.message}`);
|
|
345
|
+
}
|
|
346
|
+
console.warn(`[ProxyFetch] Proxy failed, falling back to direct: ${proxyError.message}`);
|
|
347
|
+
return originalFetch(url, options);
|
|
348
|
+
}
|
|
349
|
+
}
|
|
350
|
+
|
|
351
|
+
// got-scraping disabled — use native fetch directly
|
|
352
|
+
// (Re-enable per-host by wrapping with tryGotScrapingFetch when needed)
|
|
353
|
+
return originalFetch(url, options);
|
|
354
|
+
}
|
|
355
|
+
|
|
356
|
+
/**
|
|
357
|
+
* Patched global fetch with env-proxy support and MITM DNS bypass
|
|
358
|
+
*/
|
|
359
|
+
async function patchedFetch(url, options = {}) {
|
|
360
|
+
return proxyAwareFetch(url, options, null);
|
|
361
|
+
}
|
|
362
|
+
|
|
363
|
+
// Idempotency guard — only patch once to avoid wrapping multiple times
|
|
364
|
+
if (globalThis.fetch !== patchedFetch) {
|
|
365
|
+
globalThis.fetch = patchedFetch;
|
|
366
|
+
}
|
|
367
|
+
|
|
368
|
+
export default patchedFetch;
|
|
@@ -0,0 +1,79 @@
|
|
|
1
|
+
// Some thinking-mode providers (DeepSeek, Kimi, MiniMax, ...) require reasoning_content
|
|
2
|
+
// to be echoed back on assistant messages. Clients in OpenAI format don't send it,
|
|
3
|
+
// so we inject a non-empty placeholder to satisfy upstream validation.
|
|
4
|
+
|
|
5
|
+
const PLACEHOLDER = " ";
|
|
6
|
+
|
|
7
|
+
// Provider-level rules: keyed by executor.provider
|
|
8
|
+
const PROVIDER_RULES = {
|
|
9
|
+
deepseek: { scope: "all" },
|
|
10
|
+
minimax: { scope: "all" },
|
|
11
|
+
"minimax-cn": { scope: "all" }
|
|
12
|
+
};
|
|
13
|
+
|
|
14
|
+
// Model-level rules: matched by predicate against model id
|
|
15
|
+
const MODEL_RULES = [
|
|
16
|
+
{ match: m => /^kimi-/i.test(m || ""), scope: "toolCalls" },
|
|
17
|
+
{ match: m => /deepseek/i.test(m || ""), scope: "all" }
|
|
18
|
+
];
|
|
19
|
+
|
|
20
|
+
const DEEPSEEK_V4_PRO = "deepseek-v4-pro";
|
|
21
|
+
const DEEPSEEK_V4_PRO_ALIASES = {
|
|
22
|
+
[`${DEEPSEEK_V4_PRO}-max`]: {
|
|
23
|
+
thinkingType: "enabled",
|
|
24
|
+
reasoningEffort: "max"
|
|
25
|
+
},
|
|
26
|
+
[`${DEEPSEEK_V4_PRO}-none`]: {
|
|
27
|
+
thinkingType: "disabled",
|
|
28
|
+
reasoningEffort: null
|
|
29
|
+
}
|
|
30
|
+
};
|
|
31
|
+
|
|
32
|
+
function shouldInject(message, scope) {
|
|
33
|
+
if (message?.role !== "assistant") return false;
|
|
34
|
+
const rc = message.reasoning_content;
|
|
35
|
+
if (typeof rc === "string" && rc.length > 0) return false;
|
|
36
|
+
if (scope === "toolCalls") return Array.isArray(message.tool_calls) && message.tool_calls.length > 0;
|
|
37
|
+
return true;
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
function applyRule(body, rule) {
|
|
41
|
+
if (!rule || !body?.messages) return body;
|
|
42
|
+
const messages = body.messages.map(m =>
|
|
43
|
+
shouldInject(m, rule.scope) ? { ...m, reasoning_content: PLACEHOLDER } : m
|
|
44
|
+
);
|
|
45
|
+
return { ...body, messages };
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
function applyDeepSeekV4ProAlias({ provider, model, body }) {
|
|
49
|
+
const alias = DEEPSEEK_V4_PRO_ALIASES[model];
|
|
50
|
+
if (provider !== "deepseek" || !alias || !body) return body;
|
|
51
|
+
|
|
52
|
+
const nextBody = {
|
|
53
|
+
...body,
|
|
54
|
+
model: DEEPSEEK_V4_PRO,
|
|
55
|
+
extra_body: {
|
|
56
|
+
...(body.extra_body || {}),
|
|
57
|
+
thinking: {
|
|
58
|
+
...(body.extra_body?.thinking || {}),
|
|
59
|
+
type: alias.thinkingType
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
};
|
|
63
|
+
|
|
64
|
+
if (alias.reasoningEffort) {
|
|
65
|
+
nextBody.reasoning_effort = alias.reasoningEffort;
|
|
66
|
+
} else {
|
|
67
|
+
delete nextBody.reasoning_effort;
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
return nextBody;
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
export function injectReasoningContent({ provider, model, body }) {
|
|
74
|
+
const providerRule = PROVIDER_RULES[provider];
|
|
75
|
+
const modelRule = MODEL_RULES.find(r => r.match(model));
|
|
76
|
+
const rule = providerRule || modelRule;
|
|
77
|
+
const nextBody = applyDeepSeekV4ProAlias({ provider, model, body });
|
|
78
|
+
return applyRule(nextBody, rule);
|
|
79
|
+
}
|