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.
Files changed (763) hide show
  1. package/CHANGELOG.md +222 -0
  2. package/LICENSE +21 -0
  3. package/README.md +96 -0
  4. package/README.zh-CN.md +1311 -0
  5. package/assets/pixel-router-2d.png +0 -0
  6. package/cli/LICENSE +42 -0
  7. package/cli/README.md +125 -0
  8. package/cli/app/package.json +58 -0
  9. package/cli/cli.js +806 -0
  10. package/cli/package.json +48 -0
  11. package/i18n/README.ja-JP.md +1209 -0
  12. package/i18n/README.ru.md +1311 -0
  13. package/i18n/README.vi.md +1310 -0
  14. package/i18n/README.zh-CN.md +1306 -0
  15. package/images/9router.png +0 -0
  16. package/jsconfig.json +12 -0
  17. package/next.config.mjs +71 -0
  18. package/open-sse/config/appConstants.js +203 -0
  19. package/open-sse/config/codexInstructions.js +119 -0
  20. package/open-sse/config/constants.js +4 -0
  21. package/open-sse/config/defaultThinkingSignature.js +12 -0
  22. package/open-sse/config/errorConfig.js +85 -0
  23. package/open-sse/config/googleTtsLanguages.js +62 -0
  24. package/open-sse/config/kiroConstants.js +322 -0
  25. package/open-sse/config/models.js +13 -0
  26. package/open-sse/config/ollamaModels.js +19 -0
  27. package/open-sse/config/providerModels.js +944 -0
  28. package/open-sse/config/providers.js +458 -0
  29. package/open-sse/config/runtimeConfig.js +72 -0
  30. package/open-sse/config/ttsModels.js +129 -0
  31. package/open-sse/executors/antigravity.js +504 -0
  32. package/open-sse/executors/azure.js +57 -0
  33. package/open-sse/executors/base.js +185 -0
  34. package/open-sse/executors/codex.js +469 -0
  35. package/open-sse/executors/commandcode.js +88 -0
  36. package/open-sse/executors/cursor.js +795 -0
  37. package/open-sse/executors/default.js +497 -0
  38. package/open-sse/executors/gemini-cli.js +89 -0
  39. package/open-sse/executors/github.js +379 -0
  40. package/open-sse/executors/grok-web.js +345 -0
  41. package/open-sse/executors/iflow.js +108 -0
  42. package/open-sse/executors/index.js +75 -0
  43. package/open-sse/executors/kiro.js +508 -0
  44. package/open-sse/executors/ollama-local.js +14 -0
  45. package/open-sse/executors/opencode-go.js +41 -0
  46. package/open-sse/executors/opencode.js +32 -0
  47. package/open-sse/executors/perplexity-web.js +507 -0
  48. package/open-sse/executors/qoder.js +450 -0
  49. package/open-sse/executors/qwen.js +129 -0
  50. package/open-sse/executors/vertex.js +131 -0
  51. package/open-sse/executors/xiaomi-tokenplan.js +19 -0
  52. package/open-sse/handlers/chatCore/nonStreamingHandler.js +230 -0
  53. package/open-sse/handlers/chatCore/requestDetail.js +102 -0
  54. package/open-sse/handlers/chatCore/sseToJsonHandler.js +231 -0
  55. package/open-sse/handlers/chatCore/streamingHandler.js +103 -0
  56. package/open-sse/handlers/chatCore.js +287 -0
  57. package/open-sse/handlers/embeddingProviders/_base.js +4 -0
  58. package/open-sse/handlers/embeddingProviders/gemini.js +54 -0
  59. package/open-sse/handlers/embeddingProviders/index.js +23 -0
  60. package/open-sse/handlers/embeddingProviders/openai.js +39 -0
  61. package/open-sse/handlers/embeddingProviders/openaiCompatNode.js +13 -0
  62. package/open-sse/handlers/embeddingsCore.js +126 -0
  63. package/open-sse/handlers/fetch/index.js +237 -0
  64. package/open-sse/handlers/imageGenerationCore.js +189 -0
  65. package/open-sse/handlers/imageProviders/_base.js +31 -0
  66. package/open-sse/handlers/imageProviders/blackForestLabs.js +43 -0
  67. package/open-sse/handlers/imageProviders/cloudflareAi.js +178 -0
  68. package/open-sse/handlers/imageProviders/codex.js +198 -0
  69. package/open-sse/handlers/imageProviders/comfyui.js +8 -0
  70. package/open-sse/handlers/imageProviders/falAi.js +41 -0
  71. package/open-sse/handlers/imageProviders/gemini.js +25 -0
  72. package/open-sse/handlers/imageProviders/huggingface.js +22 -0
  73. package/open-sse/handlers/imageProviders/index.js +40 -0
  74. package/open-sse/handlers/imageProviders/nanobanana.js +58 -0
  75. package/open-sse/handlers/imageProviders/openai.js +40 -0
  76. package/open-sse/handlers/imageProviders/runwayml.js +47 -0
  77. package/open-sse/handlers/imageProviders/sdwebui.js +17 -0
  78. package/open-sse/handlers/imageProviders/stabilityAi.js +34 -0
  79. package/open-sse/handlers/responsesHandler.js +103 -0
  80. package/open-sse/handlers/search/callers.js +371 -0
  81. package/open-sse/handlers/search/chatSearch.js +409 -0
  82. package/open-sse/handlers/search/index.js +201 -0
  83. package/open-sse/handlers/search/normalizers.js +223 -0
  84. package/open-sse/handlers/sttCore.js +194 -0
  85. package/open-sse/handlers/ttsCore.js +74 -0
  86. package/open-sse/handlers/ttsProviders/_base.js +39 -0
  87. package/open-sse/handlers/ttsProviders/edgeTts.js +89 -0
  88. package/open-sse/handlers/ttsProviders/elevenlabs.js +48 -0
  89. package/open-sse/handlers/ttsProviders/gemini.js +117 -0
  90. package/open-sse/handlers/ttsProviders/genericFormats.js +169 -0
  91. package/open-sse/handlers/ttsProviders/googleTts.js +54 -0
  92. package/open-sse/handlers/ttsProviders/index.js +50 -0
  93. package/open-sse/handlers/ttsProviders/localDevice.js +87 -0
  94. package/open-sse/handlers/ttsProviders/minimax.js +59 -0
  95. package/open-sse/handlers/ttsProviders/openai.js +30 -0
  96. package/open-sse/handlers/ttsProviders/openrouter.js +70 -0
  97. package/open-sse/index.js +82 -0
  98. package/open-sse/rtk/applyFilter.js +15 -0
  99. package/open-sse/rtk/autodetect.js +111 -0
  100. package/open-sse/rtk/caveman.js +100 -0
  101. package/open-sse/rtk/cavemanPrompts.js +78 -0
  102. package/open-sse/rtk/constants.js +55 -0
  103. package/open-sse/rtk/filters/buildOutput.js +127 -0
  104. package/open-sse/rtk/filters/dedupLog.js +44 -0
  105. package/open-sse/rtk/filters/find.js +49 -0
  106. package/open-sse/rtk/filters/gitDiff.js +92 -0
  107. package/open-sse/rtk/filters/gitStatus.js +117 -0
  108. package/open-sse/rtk/filters/grep.js +48 -0
  109. package/open-sse/rtk/filters/ls.js +79 -0
  110. package/open-sse/rtk/filters/readNumbered.js +27 -0
  111. package/open-sse/rtk/filters/searchList.js +52 -0
  112. package/open-sse/rtk/filters/smartTruncate.js +15 -0
  113. package/open-sse/rtk/filters/tree.js +32 -0
  114. package/open-sse/rtk/index.js +155 -0
  115. package/open-sse/rtk/registry.js +38 -0
  116. package/open-sse/services/accountFallback.js +238 -0
  117. package/open-sse/services/combo.js +198 -0
  118. package/open-sse/services/compact.js +71 -0
  119. package/open-sse/services/kiroModels.js +332 -0
  120. package/open-sse/services/model.js +261 -0
  121. package/open-sse/services/oauthCredentialManager.js +151 -0
  122. package/open-sse/services/projectId.js +306 -0
  123. package/open-sse/services/provider.js +356 -0
  124. package/open-sse/services/qoderModels.js +214 -0
  125. package/open-sse/services/tokenRefresh.js +939 -0
  126. package/open-sse/services/usage.js +1496 -0
  127. package/open-sse/transformer/responsesTransformer.js +439 -0
  128. package/open-sse/transformer/streamToJsonConverter.js +103 -0
  129. package/open-sse/translator/formats.js +36 -0
  130. package/open-sse/translator/helpers/claudeHelper.js +216 -0
  131. package/open-sse/translator/helpers/geminiHelper.js +372 -0
  132. package/open-sse/translator/helpers/imageHelper.js +34 -0
  133. package/open-sse/translator/helpers/maxTokensHelper.js +27 -0
  134. package/open-sse/translator/helpers/openaiHelper.js +130 -0
  135. package/open-sse/translator/helpers/responsesApiHelper.js +139 -0
  136. package/open-sse/translator/helpers/toolCallHelper.js +148 -0
  137. package/open-sse/translator/index.js +251 -0
  138. package/open-sse/translator/request/antigravity-to-openai.js +229 -0
  139. package/open-sse/translator/request/claude-to-openai.js +232 -0
  140. package/open-sse/translator/request/gemini-to-openai.js +147 -0
  141. package/open-sse/translator/request/openai-responses.js +318 -0
  142. package/open-sse/translator/request/openai-to-claude.js +401 -0
  143. package/open-sse/translator/request/openai-to-commandcode.js +170 -0
  144. package/open-sse/translator/request/openai-to-cursor.js +183 -0
  145. package/open-sse/translator/request/openai-to-gemini.js +470 -0
  146. package/open-sse/translator/request/openai-to-kiro.js +629 -0
  147. package/open-sse/translator/request/openai-to-kiro.old.js +278 -0
  148. package/open-sse/translator/request/openai-to-ollama.js +192 -0
  149. package/open-sse/translator/request/openai-to-vertex.js +42 -0
  150. package/open-sse/translator/response/claude-to-openai.js +206 -0
  151. package/open-sse/translator/response/commandcode-to-openai.js +197 -0
  152. package/open-sse/translator/response/cursor-to-openai.js +30 -0
  153. package/open-sse/translator/response/gemini-to-openai.js +245 -0
  154. package/open-sse/translator/response/kiro-to-openai.js +195 -0
  155. package/open-sse/translator/response/ollama-to-openai.js +152 -0
  156. package/open-sse/translator/response/openai-responses.js +590 -0
  157. package/open-sse/translator/response/openai-to-antigravity.js +122 -0
  158. package/open-sse/translator/response/openai-to-claude.js +266 -0
  159. package/open-sse/utils/bypassHandler.js +298 -0
  160. package/open-sse/utils/claudeCloaking.js +155 -0
  161. package/open-sse/utils/claudeHeaderCache.js +70 -0
  162. package/open-sse/utils/clientDetector.js +63 -0
  163. package/open-sse/utils/cursorChecksum.js +149 -0
  164. package/open-sse/utils/cursorProtobuf.js +904 -0
  165. package/open-sse/utils/debugLog.js +14 -0
  166. package/open-sse/utils/error.js +147 -0
  167. package/open-sse/utils/ollamaTransform.js +85 -0
  168. package/open-sse/utils/proxyFetch.js +368 -0
  169. package/open-sse/utils/reasoningContentInjector.js +79 -0
  170. package/open-sse/utils/requestLogger.js +260 -0
  171. package/open-sse/utils/responsesStreamHelpers.js +49 -0
  172. package/open-sse/utils/sessionManager.js +82 -0
  173. package/open-sse/utils/stream.js +462 -0
  174. package/open-sse/utils/streamHandler.js +250 -0
  175. package/open-sse/utils/streamHelpers.js +122 -0
  176. package/open-sse/utils/toolDeduper.js +49 -0
  177. package/open-sse/utils/usageTracking.js +347 -0
  178. package/package.json +100 -0
  179. package/postcss.config.mjs +12 -0
  180. package/public/favicon.svg +11 -0
  181. package/public/file.svg +1 -0
  182. package/public/globe.svg +1 -0
  183. package/public/i18n/literals/ar.json +195 -0
  184. package/public/i18n/literals/bn.json +195 -0
  185. package/public/i18n/literals/cs.json +195 -0
  186. package/public/i18n/literals/da.json +195 -0
  187. package/public/i18n/literals/de.json +195 -0
  188. package/public/i18n/literals/el.json +195 -0
  189. package/public/i18n/literals/es.json +195 -0
  190. package/public/i18n/literals/fi.json +195 -0
  191. package/public/i18n/literals/fr.json +195 -0
  192. package/public/i18n/literals/he.json +195 -0
  193. package/public/i18n/literals/hi.json +195 -0
  194. package/public/i18n/literals/hu.json +195 -0
  195. package/public/i18n/literals/id.json +195 -0
  196. package/public/i18n/literals/it.json +195 -0
  197. package/public/i18n/literals/ja.json +195 -0
  198. package/public/i18n/literals/ko.json +195 -0
  199. package/public/i18n/literals/nl.json +195 -0
  200. package/public/i18n/literals/no.json +195 -0
  201. package/public/i18n/literals/pl.json +195 -0
  202. package/public/i18n/literals/pt-BR.json +195 -0
  203. package/public/i18n/literals/pt-PT.json +195 -0
  204. package/public/i18n/literals/ro.json +195 -0
  205. package/public/i18n/literals/ru.json +195 -0
  206. package/public/i18n/literals/sv.json +195 -0
  207. package/public/i18n/literals/th.json +195 -0
  208. package/public/i18n/literals/tl.json +195 -0
  209. package/public/i18n/literals/tr.json +195 -0
  210. package/public/i18n/literals/uk.json +195 -0
  211. package/public/i18n/literals/ur.json +195 -0
  212. package/public/i18n/literals/vi.json +195 -0
  213. package/public/i18n/literals/zh-CN.json +772 -0
  214. package/public/i18n/literals/zh-TW.json +195 -0
  215. package/public/icons/discord.svg +4 -0
  216. package/public/icons/icon-192.svg +4 -0
  217. package/public/icons/icon-512.svg +4 -0
  218. package/public/next.svg +1 -0
  219. package/public/providers/alicode-intl.png +0 -0
  220. package/public/providers/alicode.png +0 -0
  221. package/public/providers/amp.png +0 -0
  222. package/public/providers/anthropic-m.png +0 -0
  223. package/public/providers/anthropic.png +0 -0
  224. package/public/providers/antigravity.png +0 -0
  225. package/public/providers/assemblyai.png +0 -0
  226. package/public/providers/aws-polly.png +0 -0
  227. package/public/providers/azure.png +0 -0
  228. package/public/providers/black-forest-labs.png +0 -0
  229. package/public/providers/blackbox.png +0 -0
  230. package/public/providers/brave-search.png +0 -0
  231. package/public/providers/byteplus.png +0 -0
  232. package/public/providers/cartesia.png +0 -0
  233. package/public/providers/cerebras.png +0 -0
  234. package/public/providers/chutes.png +0 -0
  235. package/public/providers/claude.png +0 -0
  236. package/public/providers/cline.png +0 -0
  237. package/public/providers/cloudflare-ai.png +0 -0
  238. package/public/providers/codebuddy.svg +37 -0
  239. package/public/providers/codex.png +0 -0
  240. package/public/providers/cohere.png +0 -0
  241. package/public/providers/comfyui.png +0 -0
  242. package/public/providers/commandcode.png +0 -0
  243. package/public/providers/continue.png +0 -0
  244. package/public/providers/copilot.png +0 -0
  245. package/public/providers/coqui.png +0 -0
  246. package/public/providers/cursor.png +0 -0
  247. package/public/providers/deepgram.png +0 -0
  248. package/public/providers/deepseek-tui.png +0 -0
  249. package/public/providers/deepseek.png +0 -0
  250. package/public/providers/droid.png +0 -0
  251. package/public/providers/edge-tts.png +0 -0
  252. package/public/providers/elevenlabs.png +0 -0
  253. package/public/providers/exa.png +0 -0
  254. package/public/providers/fal-ai.png +0 -0
  255. package/public/providers/firecrawl.png +0 -0
  256. package/public/providers/fireworks.png +0 -0
  257. package/public/providers/gemini-cli.png +0 -0
  258. package/public/providers/gemini.png +0 -0
  259. package/public/providers/github.png +0 -0
  260. package/public/providers/glm-cn.png +0 -0
  261. package/public/providers/glm.png +0 -0
  262. package/public/providers/google-pse.png +0 -0
  263. package/public/providers/google-tts.png +0 -0
  264. package/public/providers/grok-web.png +0 -0
  265. package/public/providers/groq.png +0 -0
  266. package/public/providers/hermes.png +0 -0
  267. package/public/providers/huggingface.png +0 -0
  268. package/public/providers/hyperbolic.png +0 -0
  269. package/public/providers/iflow.png +0 -0
  270. package/public/providers/inworld.png +0 -0
  271. package/public/providers/jcode.png +0 -0
  272. package/public/providers/jina-ai.png +0 -0
  273. package/public/providers/jina-reader.png +0 -0
  274. package/public/providers/kilocode.png +0 -0
  275. package/public/providers/kimi-coding.png +0 -0
  276. package/public/providers/kimi.png +0 -0
  277. package/public/providers/kiro.png +0 -0
  278. package/public/providers/linkup.png +0 -0
  279. package/public/providers/local-device.png +0 -0
  280. package/public/providers/minimax-cn.png +0 -0
  281. package/public/providers/minimax.png +0 -0
  282. package/public/providers/mistral.png +0 -0
  283. package/public/providers/nanobanana.png +0 -0
  284. package/public/providers/nebius.png +0 -0
  285. package/public/providers/nvidia.png +0 -0
  286. package/public/providers/oai-cc.png +0 -0
  287. package/public/providers/oai-r.png +0 -0
  288. package/public/providers/ollama-local.png +0 -0
  289. package/public/providers/ollama.png +0 -0
  290. package/public/providers/openai.png +0 -0
  291. package/public/providers/openclaw.png +0 -0
  292. package/public/providers/opencode-go.png +0 -0
  293. package/public/providers/opencode.png +0 -0
  294. package/public/providers/openrouter.png +0 -0
  295. package/public/providers/perplexity-web.png +0 -0
  296. package/public/providers/perplexity.png +0 -0
  297. package/public/providers/playht.png +0 -0
  298. package/public/providers/qoder.png +0 -0
  299. package/public/providers/qwen.png +0 -0
  300. package/public/providers/recraft.png +0 -0
  301. package/public/providers/roo.png +0 -0
  302. package/public/providers/runwayml.png +0 -0
  303. package/public/providers/sdwebui.png +0 -0
  304. package/public/providers/searchapi.png +0 -0
  305. package/public/providers/searxng.png +0 -0
  306. package/public/providers/serper.png +0 -0
  307. package/public/providers/siliconflow.png +0 -0
  308. package/public/providers/stability-ai.png +0 -0
  309. package/public/providers/tavily.png +0 -0
  310. package/public/providers/together.png +0 -0
  311. package/public/providers/topaz.png +0 -0
  312. package/public/providers/tortoise.png +0 -0
  313. package/public/providers/vertex-partner.png +0 -0
  314. package/public/providers/vertex.png +0 -0
  315. package/public/providers/volcengine-ark.png +0 -0
  316. package/public/providers/voyage-ai.png +0 -0
  317. package/public/providers/xai.png +0 -0
  318. package/public/providers/xiaomi-mimo.png +0 -0
  319. package/public/providers/xiaomi-tokenplan.png +0 -0
  320. package/public/providers/youcom.png +0 -0
  321. package/public/sw.js +22 -0
  322. package/public/vercel.svg +1 -0
  323. package/public/window.svg +1 -0
  324. package/scripts/compact-request-details.mjs +71 -0
  325. package/scripts/import-codex-gptjson.mjs +342 -0
  326. package/scripts/start-standalone.mjs +25 -0
  327. package/scripts/translate-readme.js +201 -0
  328. package/src/app/(dashboard)/dashboard/automation/page.js +294 -0
  329. package/src/app/(dashboard)/dashboard/basic-chat/BasicChatPageClient.js +967 -0
  330. package/src/app/(dashboard)/dashboard/basic-chat/page.js +5 -0
  331. package/src/app/(dashboard)/dashboard/cli-tools/CLIToolsPageClient.js +66 -0
  332. package/src/app/(dashboard)/dashboard/cli-tools/[toolId]/ToolDetailClient.js +173 -0
  333. package/src/app/(dashboard)/dashboard/cli-tools/[toolId]/page.js +11 -0
  334. package/src/app/(dashboard)/dashboard/cli-tools/components/AntigravityToolCard.js +481 -0
  335. package/src/app/(dashboard)/dashboard/cli-tools/components/ApiKeySelect.js +66 -0
  336. package/src/app/(dashboard)/dashboard/cli-tools/components/BaseUrlSelect.js +174 -0
  337. package/src/app/(dashboard)/dashboard/cli-tools/components/ClaudeToolCard.js +390 -0
  338. package/src/app/(dashboard)/dashboard/cli-tools/components/ClineToolCard.js +301 -0
  339. package/src/app/(dashboard)/dashboard/cli-tools/components/CodexToolCard.js +458 -0
  340. package/src/app/(dashboard)/dashboard/cli-tools/components/CopilotToolCard.js +323 -0
  341. package/src/app/(dashboard)/dashboard/cli-tools/components/CoworkToolCard.js +640 -0
  342. package/src/app/(dashboard)/dashboard/cli-tools/components/DeepSeekTuiToolCard.js +338 -0
  343. package/src/app/(dashboard)/dashboard/cli-tools/components/DefaultToolCard.js +271 -0
  344. package/src/app/(dashboard)/dashboard/cli-tools/components/DroidToolCard.js +410 -0
  345. package/src/app/(dashboard)/dashboard/cli-tools/components/EndpointPresetControl.js +128 -0
  346. package/src/app/(dashboard)/dashboard/cli-tools/components/HermesToolCard.js +317 -0
  347. package/src/app/(dashboard)/dashboard/cli-tools/components/JcodeToolCard.js +380 -0
  348. package/src/app/(dashboard)/dashboard/cli-tools/components/KiloToolCard.js +275 -0
  349. package/src/app/(dashboard)/dashboard/cli-tools/components/MitmLinkCard.js +40 -0
  350. package/src/app/(dashboard)/dashboard/cli-tools/components/MitmServerCard.js +329 -0
  351. package/src/app/(dashboard)/dashboard/cli-tools/components/MitmToolCard.js +318 -0
  352. package/src/app/(dashboard)/dashboard/cli-tools/components/OpenClawToolCard.js +388 -0
  353. package/src/app/(dashboard)/dashboard/cli-tools/components/OpenCodeToolCard.js +500 -0
  354. package/src/app/(dashboard)/dashboard/cli-tools/components/ToolSummaryCard.js +39 -0
  355. package/src/app/(dashboard)/dashboard/cli-tools/components/cliEndpointMatch.js +13 -0
  356. package/src/app/(dashboard)/dashboard/cli-tools/components/index.js +19 -0
  357. package/src/app/(dashboard)/dashboard/cli-tools/page.js +7 -0
  358. package/src/app/(dashboard)/dashboard/combos/page.js +612 -0
  359. package/src/app/(dashboard)/dashboard/console-log/ConsoleLogClient.js +91 -0
  360. package/src/app/(dashboard)/dashboard/console-log/page.js +8 -0
  361. package/src/app/(dashboard)/dashboard/endpoint/EndpointPageClient.js +1555 -0
  362. package/src/app/(dashboard)/dashboard/endpoint/page.js +7 -0
  363. package/src/app/(dashboard)/dashboard/media-providers/[kind]/[id]/page.js +1903 -0
  364. package/src/app/(dashboard)/dashboard/media-providers/[kind]/page.js +289 -0
  365. package/src/app/(dashboard)/dashboard/media-providers/combo/[id]/page.js +410 -0
  366. package/src/app/(dashboard)/dashboard/media-providers/web/page.js +209 -0
  367. package/src/app/(dashboard)/dashboard/mitm/MitmPageClient.js +117 -0
  368. package/src/app/(dashboard)/dashboard/mitm/page.js +5 -0
  369. package/src/app/(dashboard)/dashboard/page.js +7 -0
  370. package/src/app/(dashboard)/dashboard/profile/page.js +1140 -0
  371. package/src/app/(dashboard)/dashboard/providers/[id]/AddApiKeyModal.js +389 -0
  372. package/src/app/(dashboard)/dashboard/providers/[id]/AddCustomModelModal.js +125 -0
  373. package/src/app/(dashboard)/dashboard/providers/[id]/CompatibleModelsSection.js +250 -0
  374. package/src/app/(dashboard)/dashboard/providers/[id]/ConnectionRow.js +299 -0
  375. package/src/app/(dashboard)/dashboard/providers/[id]/CooldownTimer.js +42 -0
  376. package/src/app/(dashboard)/dashboard/providers/[id]/EditCompatibleNodeModal.js +161 -0
  377. package/src/app/(dashboard)/dashboard/providers/[id]/ModelRow.js +95 -0
  378. package/src/app/(dashboard)/dashboard/providers/[id]/PassthroughModelsSection.js +183 -0
  379. package/src/app/(dashboard)/dashboard/providers/[id]/page.js +2053 -0
  380. package/src/app/(dashboard)/dashboard/providers/[id]/page.new.js +1724 -0
  381. package/src/app/(dashboard)/dashboard/providers/components/ConnectionsCard.js +497 -0
  382. package/src/app/(dashboard)/dashboard/providers/components/ModelAvailabilityBadge.js +185 -0
  383. package/src/app/(dashboard)/dashboard/providers/components/ModelsCard.js +294 -0
  384. package/src/app/(dashboard)/dashboard/providers/new/page.js +220 -0
  385. package/src/app/(dashboard)/dashboard/providers/page.js +1365 -0
  386. package/src/app/(dashboard)/dashboard/proxy-pools/page.js +1092 -0
  387. package/src/app/(dashboard)/dashboard/quota/page.js +11 -0
  388. package/src/app/(dashboard)/dashboard/skills/page.js +112 -0
  389. package/src/app/(dashboard)/dashboard/translator/page.js +303 -0
  390. package/src/app/(dashboard)/dashboard/usage/components/OverviewCards.js +35 -0
  391. package/src/app/(dashboard)/dashboard/usage/components/ProviderLimits/ProviderLimitCard.js +185 -0
  392. package/src/app/(dashboard)/dashboard/usage/components/ProviderLimits/QuotaProgressBar.js +127 -0
  393. package/src/app/(dashboard)/dashboard/usage/components/ProviderLimits/QuotaTable.js +259 -0
  394. package/src/app/(dashboard)/dashboard/usage/components/ProviderLimits/index.js +1394 -0
  395. package/src/app/(dashboard)/dashboard/usage/components/ProviderLimits/utils.js +244 -0
  396. package/src/app/(dashboard)/dashboard/usage/components/ProviderTopology.js +327 -0
  397. package/src/app/(dashboard)/dashboard/usage/components/RequestDetailsTab.js +433 -0
  398. package/src/app/(dashboard)/dashboard/usage/components/UsageChart.js +141 -0
  399. package/src/app/(dashboard)/dashboard/usage/components/UsageTable.js +247 -0
  400. package/src/app/(dashboard)/dashboard/usage/page.js +75 -0
  401. package/src/app/(dashboard)/layout.js +6 -0
  402. package/src/app/api/auth/login/route.js +76 -0
  403. package/src/app/api/auth/logout/route.js +12 -0
  404. package/src/app/api/auth/oidc/callback/route.js +87 -0
  405. package/src/app/api/auth/oidc/start/route.js +52 -0
  406. package/src/app/api/auth/oidc/test/route.js +84 -0
  407. package/src/app/api/auth/status/route.js +45 -0
  408. package/src/app/api/cli-tools/all-statuses/route.js +46 -0
  409. package/src/app/api/cli-tools/antigravity-mitm/alias/route.js +53 -0
  410. package/src/app/api/cli-tools/antigravity-mitm/route.js +202 -0
  411. package/src/app/api/cli-tools/claude-settings/route.js +203 -0
  412. package/src/app/api/cli-tools/cline-settings/route.js +133 -0
  413. package/src/app/api/cli-tools/codex-gateway/accounts/route.js +16 -0
  414. package/src/app/api/cli-tools/codex-settings/route.js +239 -0
  415. package/src/app/api/cli-tools/copilot-settings/route.js +148 -0
  416. package/src/app/api/cli-tools/cowork-mcp-registry/route.js +77 -0
  417. package/src/app/api/cli-tools/cowork-mcp-tools/route.js +95 -0
  418. package/src/app/api/cli-tools/cowork-settings/route.js +412 -0
  419. package/src/app/api/cli-tools/deepseek-tui-settings/route.js +164 -0
  420. package/src/app/api/cli-tools/droid-settings/route.js +213 -0
  421. package/src/app/api/cli-tools/hermes-settings/route.js +175 -0
  422. package/src/app/api/cli-tools/jcode-settings/route.js +216 -0
  423. package/src/app/api/cli-tools/kilo-settings/route.js +131 -0
  424. package/src/app/api/cli-tools/openclaw-settings/route.js +292 -0
  425. package/src/app/api/cli-tools/opencode-settings/route.js +259 -0
  426. package/src/app/api/combos/[id]/route.js +81 -0
  427. package/src/app/api/combos/route.js +48 -0
  428. package/src/app/api/health/route.js +15 -0
  429. package/src/app/api/init/route.js +4 -0
  430. package/src/app/api/keys/[id]/route.js +58 -0
  431. package/src/app/api/keys/route.js +42 -0
  432. package/src/app/api/locale/route.js +30 -0
  433. package/src/app/api/mcp/[plugin]/message/route.js +21 -0
  434. package/src/app/api/mcp/[plugin]/sse/route.js +37 -0
  435. package/src/app/api/media-providers/tts/deepgram/voices/route.js +65 -0
  436. package/src/app/api/media-providers/tts/elevenlabs/voices/route.js +71 -0
  437. package/src/app/api/media-providers/tts/inworld/voices/route.js +61 -0
  438. package/src/app/api/media-providers/tts/minimax/voices/route.js +113 -0
  439. package/src/app/api/media-providers/tts/voices/route.js +99 -0
  440. package/src/app/api/models/alias/route.js +53 -0
  441. package/src/app/api/models/availability/route.js +103 -0
  442. package/src/app/api/models/custom/route.js +48 -0
  443. package/src/app/api/models/disabled/route.js +50 -0
  444. package/src/app/api/models/route.js +64 -0
  445. package/src/app/api/models/test/ping.js +191 -0
  446. package/src/app/api/models/test/route.js +14 -0
  447. package/src/app/api/oauth/[provider]/[action]/route.js +343 -0
  448. package/src/app/api/oauth/codebuddy/bulk-import/[jobId]/cancel/route.js +19 -0
  449. package/src/app/api/oauth/codebuddy/bulk-import/[jobId]/manual/[workerId]/route.js +30 -0
  450. package/src/app/api/oauth/codebuddy/bulk-import/[jobId]/route.js +23 -0
  451. package/src/app/api/oauth/codebuddy/bulk-import/latest/route.js +25 -0
  452. package/src/app/api/oauth/codebuddy/bulk-import/route.js +49 -0
  453. package/src/app/api/oauth/codebuddy/quota-cookie/route.js +133 -0
  454. package/src/app/api/oauth/codex/import-token/route.js +96 -0
  455. package/src/app/api/oauth/cursor/auto-import/route.js +258 -0
  456. package/src/app/api/oauth/cursor/import/route.js +100 -0
  457. package/src/app/api/oauth/gitlab/pat/route.js +62 -0
  458. package/src/app/api/oauth/iflow/cookie/route.js +137 -0
  459. package/src/app/api/oauth/kiro/auto-import/route.js +85 -0
  460. package/src/app/api/oauth/kiro/bulk-import/[jobId]/cancel/route.js +18 -0
  461. package/src/app/api/oauth/kiro/bulk-import/[jobId]/manual/[workerId]/route.js +29 -0
  462. package/src/app/api/oauth/kiro/bulk-import/[jobId]/route.js +22 -0
  463. package/src/app/api/oauth/kiro/bulk-import/latest/route.js +25 -0
  464. package/src/app/api/oauth/kiro/bulk-import/route.js +49 -0
  465. package/src/app/api/oauth/kiro/import/route.js +110 -0
  466. package/src/app/api/oauth/kiro/social-authorize/route.js +27 -0
  467. package/src/app/api/oauth/kiro/social-exchange/route.js +41 -0
  468. package/src/app/api/pricing/route.js +134 -0
  469. package/src/app/api/provider-nodes/[id]/route.js +101 -0
  470. package/src/app/api/provider-nodes/route.js +104 -0
  471. package/src/app/api/provider-nodes/validate/route.js +201 -0
  472. package/src/app/api/providers/[id]/models/route.js +526 -0
  473. package/src/app/api/providers/[id]/route.js +189 -0
  474. package/src/app/api/providers/[id]/test/route.js +23 -0
  475. package/src/app/api/providers/[id]/test/testUtils.js +714 -0
  476. package/src/app/api/providers/[id]/test-models/route.js +66 -0
  477. package/src/app/api/providers/client/route.js +126 -0
  478. package/src/app/api/providers/kilo/free-models/route.js +55 -0
  479. package/src/app/api/providers/route.js +206 -0
  480. package/src/app/api/providers/suggested-models/filters.js +20 -0
  481. package/src/app/api/providers/suggested-models/route.js +32 -0
  482. package/src/app/api/providers/test-batch/route.js +131 -0
  483. package/src/app/api/providers/validate/route.js +637 -0
  484. package/src/app/api/proxy-pools/[id]/route.js +123 -0
  485. package/src/app/api/proxy-pools/[id]/test/route.js +70 -0
  486. package/src/app/api/proxy-pools/cloudflare-deploy/route.js +145 -0
  487. package/src/app/api/proxy-pools/deno-deploy/route.js +175 -0
  488. package/src/app/api/proxy-pools/route.js +93 -0
  489. package/src/app/api/proxy-pools/vercel-deploy/route.js +142 -0
  490. package/src/app/api/settings/database/route.js +36 -0
  491. package/src/app/api/settings/proxy-test/route.js +23 -0
  492. package/src/app/api/settings/require-login/route.js +15 -0
  493. package/src/app/api/settings/route.js +100 -0
  494. package/src/app/api/shutdown/route.js +24 -0
  495. package/src/app/api/tags/route.js +18 -0
  496. package/src/app/api/translator/console-logs/route.js +24 -0
  497. package/src/app/api/translator/console-logs/stream/route.js +79 -0
  498. package/src/app/api/translator/load/route.js +45 -0
  499. package/src/app/api/translator/save/route.js +44 -0
  500. package/src/app/api/translator/send/route.js +94 -0
  501. package/src/app/api/translator/translate/route.js +90 -0
  502. package/src/app/api/tunnel/disable/route.js +12 -0
  503. package/src/app/api/tunnel/enable/route.js +16 -0
  504. package/src/app/api/tunnel/status/route.js +13 -0
  505. package/src/app/api/tunnel/tailscale-check/route.js +50 -0
  506. package/src/app/api/tunnel/tailscale-disable/route.js +12 -0
  507. package/src/app/api/tunnel/tailscale-enable/route.js +12 -0
  508. package/src/app/api/tunnel/tailscale-install/route.js +72 -0
  509. package/src/app/api/usage/[connectionId]/route.js +188 -0
  510. package/src/app/api/usage/chart/route.js +21 -0
  511. package/src/app/api/usage/history/route.js +12 -0
  512. package/src/app/api/usage/logs/route.js +12 -0
  513. package/src/app/api/usage/providers/route.js +42 -0
  514. package/src/app/api/usage/request-details/route.js +57 -0
  515. package/src/app/api/usage/request-logs/route.js +13 -0
  516. package/src/app/api/usage/stats/route.js +23 -0
  517. package/src/app/api/usage/stream/route.js +79 -0
  518. package/src/app/api/v1/api/chat/route.js +37 -0
  519. package/src/app/api/v1/audio/speech/route.js +16 -0
  520. package/src/app/api/v1/audio/transcriptions/route.js +19 -0
  521. package/src/app/api/v1/audio/voices/route.js +68 -0
  522. package/src/app/api/v1/chat/completions/route.js +35 -0
  523. package/src/app/api/v1/embeddings/route.js +21 -0
  524. package/src/app/api/v1/images/generations/route.js +16 -0
  525. package/src/app/api/v1/messages/count_tokens/route.js +52 -0
  526. package/src/app/api/v1/messages/route.js +36 -0
  527. package/src/app/api/v1/models/[kind]/route.js +55 -0
  528. package/src/app/api/v1/models/info/route.js +110 -0
  529. package/src/app/api/v1/models/route.js +451 -0
  530. package/src/app/api/v1/responses/compact/route.js +37 -0
  531. package/src/app/api/v1/responses/route.js +30 -0
  532. package/src/app/api/v1/route.js +1 -0
  533. package/src/app/api/v1/search/route.js +21 -0
  534. package/src/app/api/v1/web/fetch/route.js +21 -0
  535. package/src/app/api/v1beta/models/[...path]/route.js +328 -0
  536. package/src/app/api/v1beta/models/route.js +44 -0
  537. package/src/app/api/version/route.js +45 -0
  538. package/src/app/api/version/shutdown/route.js +15 -0
  539. package/src/app/api/version/update/route.js +21 -0
  540. package/src/app/callback/page.js +148 -0
  541. package/src/app/dashboard/settings/pricing/page.js +173 -0
  542. package/src/app/favicon.ico +0 -0
  543. package/src/app/globals.css +496 -0
  544. package/src/app/landing/components/AnimatedBackground.js +57 -0
  545. package/src/app/landing/components/Features.js +133 -0
  546. package/src/app/landing/components/FlowAnimation.js +175 -0
  547. package/src/app/landing/components/Footer.js +61 -0
  548. package/src/app/landing/components/GetStarted.js +97 -0
  549. package/src/app/landing/components/HeroSection.js +47 -0
  550. package/src/app/landing/components/HowItWorks.js +66 -0
  551. package/src/app/landing/components/Navigation.js +72 -0
  552. package/src/app/landing/page.js +106 -0
  553. package/src/app/layout.js +49 -0
  554. package/src/app/login/page.js +197 -0
  555. package/src/app/manifest.js +30 -0
  556. package/src/app/page.js +5 -0
  557. package/src/dashboardGuard.js +242 -0
  558. package/src/i18n/RuntimeI18nProvider.js +27 -0
  559. package/src/i18n/config.js +146 -0
  560. package/src/i18n/runtime.js +162 -0
  561. package/src/lib/appUpdater.js +200 -0
  562. package/src/lib/auth/dashboardSession.js +68 -0
  563. package/src/lib/auth/loginLimiter.js +52 -0
  564. package/src/lib/auth/oidc.js +234 -0
  565. package/src/lib/consoleLogBuffer.js +79 -0
  566. package/src/lib/dataDir.js +29 -0
  567. package/src/lib/db/adapters/betterSqliteAdapter.js +55 -0
  568. package/src/lib/db/adapters/bunSqliteAdapter.js +63 -0
  569. package/src/lib/db/adapters/nodeSqliteAdapter.js +84 -0
  570. package/src/lib/db/adapters/sqljsAdapter.js +115 -0
  571. package/src/lib/db/backup.js +35 -0
  572. package/src/lib/db/driver.js +85 -0
  573. package/src/lib/db/helpers/jsonCol.js +9 -0
  574. package/src/lib/db/helpers/kvStore.js +39 -0
  575. package/src/lib/db/helpers/metaStore.js +22 -0
  576. package/src/lib/db/index.js +171 -0
  577. package/src/lib/db/migrate.js +286 -0
  578. package/src/lib/db/migrations/001-initial.js +14 -0
  579. package/src/lib/db/migrations/index.js +10 -0
  580. package/src/lib/db/paths.js +18 -0
  581. package/src/lib/db/repos/aliasRepo.js +62 -0
  582. package/src/lib/db/repos/apiKeysRepo.js +75 -0
  583. package/src/lib/db/repos/combosRepo.js +73 -0
  584. package/src/lib/db/repos/connectionsRepo.js +226 -0
  585. package/src/lib/db/repos/disabledModelsRepo.js +56 -0
  586. package/src/lib/db/repos/nodesRepo.js +95 -0
  587. package/src/lib/db/repos/pricingRepo.js +108 -0
  588. package/src/lib/db/repos/proxyPoolsRepo.js +103 -0
  589. package/src/lib/db/repos/requestDetailsRepo.js +259 -0
  590. package/src/lib/db/repos/settingsRepo.js +104 -0
  591. package/src/lib/db/repos/usageRepo.js +731 -0
  592. package/src/lib/db/schema.js +157 -0
  593. package/src/lib/db/version.js +21 -0
  594. package/src/lib/disabledModelsDb.js +4 -0
  595. package/src/lib/localDb.js +21 -0
  596. package/src/lib/mcp/stdioSseBridge.js +198 -0
  597. package/src/lib/mitmAliasCache.js +46 -0
  598. package/src/lib/network/connectionProxy.js +160 -0
  599. package/src/lib/network/initOutboundProxy.js +25 -0
  600. package/src/lib/network/outboundProxy.js +68 -0
  601. package/src/lib/network/proxyTest.js +91 -0
  602. package/src/lib/oauth/constants/oauth.js +284 -0
  603. package/src/lib/oauth/constants/xai.js +61 -0
  604. package/src/lib/oauth/providers.js +1506 -0
  605. package/src/lib/oauth/services/antigravity.js +321 -0
  606. package/src/lib/oauth/services/claude.js +136 -0
  607. package/src/lib/oauth/services/codebuddyBulkImportManager.js +454 -0
  608. package/src/lib/oauth/services/codex.js +144 -0
  609. package/src/lib/oauth/services/cursor.js +179 -0
  610. package/src/lib/oauth/services/gemini.js +240 -0
  611. package/src/lib/oauth/services/github.js +225 -0
  612. package/src/lib/oauth/services/iflow.js +202 -0
  613. package/src/lib/oauth/services/index.js +17 -0
  614. package/src/lib/oauth/services/kiro.js +334 -0
  615. package/src/lib/oauth/services/kiroBulkImportManager.js +778 -0
  616. package/src/lib/oauth/services/kiroConnections.js +93 -0
  617. package/src/lib/oauth/services/kiroGoogleAutomation.js +1136 -0
  618. package/src/lib/oauth/services/oauth.js +157 -0
  619. package/src/lib/oauth/services/openai.js +123 -0
  620. package/src/lib/oauth/services/qoder.js +216 -0
  621. package/src/lib/oauth/services/qwen.js +170 -0
  622. package/src/lib/oauth/services/xai.js +238 -0
  623. package/src/lib/oauth/utils/banner.js +63 -0
  624. package/src/lib/oauth/utils/pkce.js +39 -0
  625. package/src/lib/oauth/utils/server.js +415 -0
  626. package/src/lib/oauth/utils/ui.js +48 -0
  627. package/src/lib/providerNormalization.js +45 -0
  628. package/src/lib/qoder/constants.js +64 -0
  629. package/src/lib/qoder/cosy.js +175 -0
  630. package/src/lib/qoder/encoding.js +55 -0
  631. package/src/lib/requestDetailsDb.js +4 -0
  632. package/src/lib/tunnel/cloudflare/cloudflared.js +449 -0
  633. package/src/lib/tunnel/cloudflare/config.js +9 -0
  634. package/src/lib/tunnel/cloudflare/healthCheck.js +29 -0
  635. package/src/lib/tunnel/cloudflare/manager.js +151 -0
  636. package/src/lib/tunnel/cloudflare/pid.js +23 -0
  637. package/src/lib/tunnel/index.js +48 -0
  638. package/src/lib/tunnel/shared/dnsResolver.js +17 -0
  639. package/src/lib/tunnel/shared/internetCheck.js +26 -0
  640. package/src/lib/tunnel/shared/state.js +41 -0
  641. package/src/lib/tunnel/shared/watchdogConfig.js +8 -0
  642. package/src/lib/tunnel/tailscale/config.js +7 -0
  643. package/src/lib/tunnel/tailscale/healthCheck.js +29 -0
  644. package/src/lib/tunnel/tailscale/manager.js +129 -0
  645. package/src/lib/tunnel/tailscale/tailscale.js +790 -0
  646. package/src/lib/updater/updater.js +235 -0
  647. package/src/lib/usage/fetcher.js +208 -0
  648. package/src/lib/usageDb.js +7 -0
  649. package/src/mitm/antigravityIdeVersion.js +50 -0
  650. package/src/mitm/cert/generate.js +32 -0
  651. package/src/mitm/cert/install.js +269 -0
  652. package/src/mitm/cert/rootCA.js +173 -0
  653. package/src/mitm/config.js +87 -0
  654. package/src/mitm/dbReader.js +22 -0
  655. package/src/mitm/dns/dnsConfig.js +266 -0
  656. package/src/mitm/handlers/antigravity.js +33 -0
  657. package/src/mitm/handlers/base.js +226 -0
  658. package/src/mitm/handlers/copilot.js +35 -0
  659. package/src/mitm/handlers/cursor.js +15 -0
  660. package/src/mitm/handlers/kiro.js +526 -0
  661. package/src/mitm/logger.js +106 -0
  662. package/src/mitm/manager.js +851 -0
  663. package/src/mitm/paths.js +32 -0
  664. package/src/mitm/server.js +435 -0
  665. package/src/mitm/winElevated.js +81 -0
  666. package/src/models/index.js +38 -0
  667. package/src/proxy.js +5 -0
  668. package/src/shared/components/AddCustomEmbeddingModal.js +183 -0
  669. package/src/shared/components/Avatar.js +88 -0
  670. package/src/shared/components/Badge.js +54 -0
  671. package/src/shared/components/BulkAccountAutomationModal.js +508 -0
  672. package/src/shared/components/Button.js +56 -0
  673. package/src/shared/components/Card.js +116 -0
  674. package/src/shared/components/ChangelogModal.js +97 -0
  675. package/src/shared/components/CodeBuddyQuotaCookieModal.js +109 -0
  676. package/src/shared/components/ComboFormModal.js +176 -0
  677. package/src/shared/components/CursorAuthModal.js +212 -0
  678. package/src/shared/components/DonateModal.js +136 -0
  679. package/src/shared/components/Drawer.js +82 -0
  680. package/src/shared/components/EditConnectionModal.js +286 -0
  681. package/src/shared/components/Footer.js +132 -0
  682. package/src/shared/components/GitLabAuthModal.js +194 -0
  683. package/src/shared/components/Header.js +380 -0
  684. package/src/shared/components/HeaderLanguage.js +46 -0
  685. package/src/shared/components/HeaderMenu.js +126 -0
  686. package/src/shared/components/IFlowCookieModal.js +132 -0
  687. package/src/shared/components/Input.js +65 -0
  688. package/src/shared/components/KiroAuthModal.js +1171 -0
  689. package/src/shared/components/KiroOAuthWrapper.js +149 -0
  690. package/src/shared/components/KiroSocialOAuthModal.js +205 -0
  691. package/src/shared/components/LanguageSwitcher.js +190 -0
  692. package/src/shared/components/Loading.js +75 -0
  693. package/src/shared/components/ManualConfigModal.js +44 -0
  694. package/src/shared/components/McpMarketplaceModal.js +255 -0
  695. package/src/shared/components/Modal.js +146 -0
  696. package/src/shared/components/ModelSelectModal.js +537 -0
  697. package/src/shared/components/NineRemoteButton.js +23 -0
  698. package/src/shared/components/NineRemotePromoModal.js +99 -0
  699. package/src/shared/components/NoAuthProxyCard.js +86 -0
  700. package/src/shared/components/OAuthModal.js +682 -0
  701. package/src/shared/components/Pagination.js +150 -0
  702. package/src/shared/components/PricingModal.js +208 -0
  703. package/src/shared/components/ProviderIcon.js +63 -0
  704. package/src/shared/components/ProviderInfoCard.js +82 -0
  705. package/src/shared/components/RequestLogger.js +121 -0
  706. package/src/shared/components/SegmentedControl.js +48 -0
  707. package/src/shared/components/Select.js +67 -0
  708. package/src/shared/components/Sidebar.js +441 -0
  709. package/src/shared/components/ThemeProvider.js +15 -0
  710. package/src/shared/components/ThemeToggle.js +42 -0
  711. package/src/shared/components/Toggle.js +69 -0
  712. package/src/shared/components/Tooltip.js +25 -0
  713. package/src/shared/components/UsageStats.js +505 -0
  714. package/src/shared/components/index.js +46 -0
  715. package/src/shared/components/layouts/AuthLayout.js +29 -0
  716. package/src/shared/components/layouts/DashboardLayout.js +104 -0
  717. package/src/shared/components/layouts/index.js +4 -0
  718. package/src/shared/constants/cliTools.js +397 -0
  719. package/src/shared/constants/colors.js +77 -0
  720. package/src/shared/constants/config.js +99 -0
  721. package/src/shared/constants/coworkPlugins.js +75 -0
  722. package/src/shared/constants/index.js +4 -0
  723. package/src/shared/constants/locales.js +36 -0
  724. package/src/shared/constants/mitmToolHosts.js +12 -0
  725. package/src/shared/constants/models.js +38 -0
  726. package/src/shared/constants/pricing.js +303 -0
  727. package/src/shared/constants/providers.js +289 -0
  728. package/src/shared/constants/skills.js +78 -0
  729. package/src/shared/constants/ttsProviders.js +138 -0
  730. package/src/shared/hooks/index.js +2 -0
  731. package/src/shared/hooks/useCopyToClipboard.js +43 -0
  732. package/src/shared/hooks/useTheme.js +60 -0
  733. package/src/shared/services/bootstrap.js +12 -0
  734. package/src/shared/services/initializeApp.js +268 -0
  735. package/src/shared/utils/api.js +93 -0
  736. package/src/shared/utils/apiKey.js +98 -0
  737. package/src/shared/utils/clineAuth.js +37 -0
  738. package/src/shared/utils/cn.js +11 -0
  739. package/src/shared/utils/connectionStatus.js +162 -0
  740. package/src/shared/utils/index.js +40 -0
  741. package/src/shared/utils/machine.js +6 -0
  742. package/src/shared/utils/machineId.js +66 -0
  743. package/src/shared/utils/providerModelsFetcher.js +30 -0
  744. package/src/sse/handlers/chat.js +261 -0
  745. package/src/sse/handlers/embeddings.js +141 -0
  746. package/src/sse/handlers/fetch.js +213 -0
  747. package/src/sse/handlers/imageGeneration.js +142 -0
  748. package/src/sse/handlers/search.js +206 -0
  749. package/src/sse/handlers/stt.js +88 -0
  750. package/src/sse/handlers/tts.js +114 -0
  751. package/src/sse/services/auth.js +346 -0
  752. package/src/sse/services/codexGateway.js +215 -0
  753. package/src/sse/services/model.js +99 -0
  754. package/src/sse/services/tokenRefresh.js +319 -0
  755. package/src/sse/utils/logger.js +75 -0
  756. package/src/store/headerSearchStore.js +19 -0
  757. package/src/store/index.js +6 -0
  758. package/src/store/notificationStore.js +45 -0
  759. package/src/store/providerStore.js +55 -0
  760. package/src/store/settingsStore.js +51 -0
  761. package/src/store/themeStore.js +54 -0
  762. package/src/store/userStore.js +20 -0
  763. package/start.sh +4 -0
@@ -0,0 +1,66 @@
1
+ import { NextResponse } from "next/server";
2
+ import { getProviderConnectionById } from "@/lib/localDb";
3
+ import { getProviderModels, PROVIDER_ID_TO_ALIAS } from "open-sse/config/providerModels.js";
4
+ import { isOpenAICompatibleProvider, isAnthropicCompatibleProvider } from "@/shared/constants/providers";
5
+ import { UPDATER_CONFIG } from "@/shared/constants/config";
6
+ import { pingModelByKind } from "@/app/api/models/test/ping";
7
+
8
+ /**
9
+ * POST /api/providers/[id]/test-models
10
+ * id = connectionId — used only to resolve provider + model list.
11
+ * Actual requests go through the internal endpoint that matches each model kind.
12
+ */
13
+ export async function POST(request, { params }) {
14
+ try {
15
+ const { id } = await params;
16
+ const connection = await getProviderConnectionById(id);
17
+ if (!connection) {
18
+ return NextResponse.json({ error: "Connection not found" }, { status: 404 });
19
+ }
20
+
21
+ const providerId = connection.provider;
22
+ const isCompatible = isOpenAICompatibleProvider(providerId) || isAnthropicCompatibleProvider(providerId);
23
+ const alias = PROVIDER_ID_TO_ALIAS[providerId] || providerId;
24
+
25
+ let models = getProviderModels(alias);
26
+
27
+ const baseUrl = `http://127.0.0.1:${process.env.PORT || UPDATER_CONFIG.appPort}`;
28
+
29
+ // Compatible providers: fetch live model list
30
+ if (isCompatible && models.length === 0) {
31
+ try {
32
+ const modelsRes = await fetch(`${baseUrl}/api/providers/${id}/models`);
33
+ if (modelsRes.ok) {
34
+ const data = await modelsRes.json();
35
+ models = (data.models || []).map((m) => ({ id: m.id || m.name, name: m.name || m.id }));
36
+ }
37
+ } catch { /* fallback to empty */ }
38
+ }
39
+
40
+ if (models.length === 0) {
41
+ return NextResponse.json({ error: "No models configured for this provider" }, { status: 400 });
42
+ }
43
+
44
+ // Warm up with first model to trigger token refresh (if needed) before parallel calls.
45
+ // This prevents race condition where multiple requests concurrently refresh the same token.
46
+ const [first, ...rest] = models;
47
+ const firstKind = first.type || "llm";
48
+ const firstResult = await pingModelByKind(`${alias}/${first.id}`, firstKind, baseUrl);
49
+ const results = [{ modelId: first.id, name: first.name || first.id, ...firstResult }];
50
+
51
+ if (rest.length > 0) {
52
+ const restResults = await Promise.all(
53
+ rest.map(async (model) => {
54
+ const result = await pingModelByKind(`${alias}/${model.id}`, model.type || "llm", baseUrl);
55
+ return { modelId: model.id, name: model.name || model.id, ...result };
56
+ })
57
+ );
58
+ results.push(...restResults);
59
+ }
60
+
61
+ return NextResponse.json({ provider: providerId, connectionId: id, results });
62
+ } catch (error) {
63
+ console.log("Error testing models:", error);
64
+ return NextResponse.json({ error: "Test failed" }, { status: 500 });
65
+ }
66
+ }
@@ -0,0 +1,126 @@
1
+ import { NextResponse } from "next/server";
2
+ import { getProviderConnections } from "@/lib/localDb";
3
+ import { backfillCodexEmails } from "@/lib/oauth/providers";
4
+ import { USAGE_APIKEY_PROVIDERS, USAGE_SUPPORTED_PROVIDERS } from "@/shared/constants/providers";
5
+
6
+ const SAFE_FIELDS = [
7
+ "id", "provider", "authType", "name", "email", "displayName",
8
+ "priority", "globalPriority", "isActive", "defaultModel",
9
+ "testStatus", "lastError", "lastErrorAt", "errorCode",
10
+ "expiresAt", "lastUsedAt", "consecutiveUseCount",
11
+ "createdAt", "updatedAt",
12
+ ];
13
+
14
+ const SAFE_PSD_FIELDS = [
15
+ "baseUrl", "azureEndpoint", "deployment", "apiVersion", "accountId",
16
+ "region", "projectId", "resourceUrl", "proxyPoolId",
17
+ "connectionProxyEnabled", "connectionProxyUrl", "connectionNoProxy",
18
+ "githubLogin", "githubName", "githubEmail", "githubUserId",
19
+ "username", "firstName", "lastName", "authMethod", "authKind",
20
+ ];
21
+
22
+ const DEFAULT_PAGE_SIZE = 20;
23
+ const MAX_PAGE_SIZE = 500;
24
+
25
+ function maskName(name) {
26
+ if (typeof name !== "string" || name.length <= 16) return name;
27
+ if (/[a-zA-Z0-9_-]{32,}/.test(name)) return `${name.slice(0, 8)}***`;
28
+ return name;
29
+ }
30
+
31
+ function sanitize(c) {
32
+ const safe = {};
33
+ for (const f of SAFE_FIELDS) if (c[f] !== undefined) safe[f] = c[f];
34
+ if (safe.name) safe.name = maskName(safe.name);
35
+ if (c.providerSpecificData) {
36
+ const psd = {};
37
+ for (const f of SAFE_PSD_FIELDS) {
38
+ if (c.providerSpecificData[f] !== undefined) psd[f] = c.providerSpecificData[f];
39
+ }
40
+ safe.providerSpecificData = psd;
41
+ }
42
+ return safe;
43
+ }
44
+
45
+ function isUsageEligible(connection) {
46
+ return USAGE_SUPPORTED_PROVIDERS.includes(connection.provider) && (
47
+ connection.authType === "oauth" || USAGE_APIKEY_PROVIDERS.includes(connection.provider)
48
+ );
49
+ }
50
+
51
+ function parsePositiveInt(value, fallback) {
52
+ const parsed = Number.parseInt(value, 10);
53
+ return Number.isFinite(parsed) && parsed > 0 ? parsed : fallback;
54
+ }
55
+
56
+ function sortConnections(connections, sort) {
57
+ const list = [...connections];
58
+
59
+ if (sort === "provider") {
60
+ return list.sort((a, b) => {
61
+ const orderA = USAGE_SUPPORTED_PROVIDERS.indexOf(a.provider);
62
+ const orderB = USAGE_SUPPORTED_PROVIDERS.indexOf(b.provider);
63
+ if (orderA !== orderB) return orderA - orderB;
64
+ return a.provider.localeCompare(b.provider);
65
+ });
66
+ }
67
+
68
+ return list.sort((a, b) => {
69
+ const priorityA = a.priority ?? Number.MAX_SAFE_INTEGER;
70
+ const priorityB = b.priority ?? Number.MAX_SAFE_INTEGER;
71
+ if (priorityA !== priorityB) return priorityA - priorityB;
72
+ return (a.provider || "").localeCompare(b.provider || "");
73
+ });
74
+ }
75
+
76
+ export async function GET(request) {
77
+ try {
78
+ await backfillCodexEmails();
79
+
80
+ const { searchParams } = new URL(request.url);
81
+ const provider = searchParams.get("provider") || "all";
82
+ const accountStatus = searchParams.get("accountStatus") || "all";
83
+ const sort = searchParams.get("sort") || "priority";
84
+ const page = parsePositiveInt(searchParams.get("page"), 1);
85
+ const pageSize = Math.min(parsePositiveInt(searchParams.get("pageSize"), DEFAULT_PAGE_SIZE), MAX_PAGE_SIZE);
86
+
87
+ const allConnections = await getProviderConnections();
88
+ const eligibleConnections = allConnections.filter(isUsageEligible);
89
+ const providerOptions = Array.from(new Set(eligibleConnections.map((conn) => conn.provider))).sort();
90
+
91
+ const providerFilteredConnections = eligibleConnections.filter((conn) => (
92
+ provider === "all" || conn.provider === provider
93
+ ));
94
+
95
+ const accountFilteredConnections = providerFilteredConnections.filter((conn) => {
96
+ if (accountStatus === "active") return conn.isActive ?? true;
97
+ if (accountStatus === "inactive") return !(conn.isActive ?? true);
98
+ return true;
99
+ });
100
+
101
+ const sortedConnections = sortConnections(accountFilteredConnections, sort);
102
+ const total = sortedConnections.length;
103
+ const totalPages = Math.max(1, Math.ceil(total / pageSize));
104
+ const currentPage = Math.min(page, totalPages);
105
+ const offset = (currentPage - 1) * pageSize;
106
+ const pageConnections = sortedConnections.slice(offset, offset + pageSize).map(sanitize);
107
+
108
+ return NextResponse.json({
109
+ connections: pageConnections,
110
+ providerOptions,
111
+ pagination: {
112
+ page: currentPage,
113
+ pageSize,
114
+ total,
115
+ totalPages,
116
+ },
117
+ totals: {
118
+ eligibleConnections: eligibleConnections.length,
119
+ providerFilteredConnections: providerFilteredConnections.length,
120
+ },
121
+ });
122
+ } catch (error) {
123
+ console.log("Error fetching providers for client:", error);
124
+ return NextResponse.json({ error: "Failed to fetch providers" }, { status: 500 });
125
+ }
126
+ }
@@ -0,0 +1,55 @@
1
+ import { NextResponse } from "next/server";
2
+
3
+ const KILO_MODELS_URL = "https://api.kilo.ai/api/gateway/models";
4
+
5
+ // In-memory cache with TTL
6
+ let cachedModels = null;
7
+ let cacheTimestamp = 0;
8
+ const CACHE_TTL_MS = 60 * 60 * 1000; // 1 hour
9
+
10
+ export async function GET() {
11
+ const now = Date.now();
12
+
13
+ // Return cached result if still valid
14
+ if (cachedModels && now - cacheTimestamp < CACHE_TTL_MS) {
15
+ return NextResponse.json({ models: cachedModels, cached: true });
16
+ }
17
+
18
+ try {
19
+ const res = await fetch(KILO_MODELS_URL, {
20
+ headers: { "Accept": "application/json" },
21
+ signal: AbortSignal.timeout(10000),
22
+ });
23
+
24
+ if (!res.ok) {
25
+ throw new Error(`Kilo API returned ${res.status}`);
26
+ }
27
+
28
+ const json = await res.json();
29
+ const allModels = json.data || [];
30
+
31
+ const freeModels = allModels
32
+ .filter((m) => m.isFree === true)
33
+ .map((m) => ({
34
+ id: m.id,
35
+ name: m.name,
36
+ isFree: true,
37
+ context_length: m.context_length || 0,
38
+ }));
39
+
40
+ cachedModels = freeModels;
41
+ cacheTimestamp = now;
42
+
43
+ return NextResponse.json({ models: freeModels, cached: false });
44
+ } catch (error) {
45
+ // Return cached data if available, even if expired
46
+ if (cachedModels) {
47
+ return NextResponse.json({ models: cachedModels, cached: true, warning: error.message });
48
+ }
49
+
50
+ return NextResponse.json(
51
+ { models: [], error: `Failed to fetch Kilo models: ${error.message}` },
52
+ { status: 502 }
53
+ );
54
+ }
55
+ }
@@ -0,0 +1,206 @@
1
+ import { NextResponse } from "next/server";
2
+ import {
3
+ getProviderConnections,
4
+ createProviderConnection,
5
+ getProviderNodeById,
6
+ getProviderNodes,
7
+ getProxyPoolById,
8
+ } from "@/models";
9
+ import { APIKEY_PROVIDERS } from "@/shared/constants/config";
10
+ import { AI_PROVIDERS, FREE_TIER_PROVIDERS, WEB_COOKIE_PROVIDERS, isOpenAICompatibleProvider, isAnthropicCompatibleProvider, isCustomEmbeddingProvider } from "@/shared/constants/providers";
11
+ import { normalizeProviderId, normalizeProviderSpecificData } from "@/lib/providerNormalization";
12
+
13
+ export const dynamic = "force-dynamic";
14
+
15
+ function normalizeProxyConfig(body = {}) {
16
+ const enabled = body?.connectionProxyEnabled === true;
17
+ const url = typeof body?.connectionProxyUrl === "string" ? body.connectionProxyUrl.trim() : "";
18
+ const noProxy = typeof body?.connectionNoProxy === "string" ? body.connectionNoProxy.trim() : "";
19
+
20
+ if (enabled && !url) {
21
+ return { error: "Connection proxy URL is required when connection proxy is enabled" };
22
+ }
23
+
24
+ return {
25
+ connectionProxyEnabled: enabled,
26
+ connectionProxyUrl: url,
27
+ connectionNoProxy: noProxy,
28
+ };
29
+ }
30
+
31
+ async function normalizeProxyPoolId(proxyPoolId) {
32
+ if (proxyPoolId === undefined || proxyPoolId === null || proxyPoolId === "" || proxyPoolId === "__none__") {
33
+ return { proxyPoolId: null };
34
+ }
35
+
36
+ const normalizedId = String(proxyPoolId).trim();
37
+ if (!normalizedId) {
38
+ return { proxyPoolId: null };
39
+ }
40
+
41
+ const proxyPool = await getProxyPoolById(normalizedId);
42
+ if (!proxyPool) {
43
+ return { error: "Proxy pool not found" };
44
+ }
45
+
46
+ return { proxyPoolId: normalizedId };
47
+ }
48
+
49
+ // GET /api/providers - List all connections
50
+ export async function GET() {
51
+ try {
52
+ const connections = await getProviderConnections();
53
+
54
+ // Build nodeNameMap for compatible providers (id → name)
55
+ let nodeNameMap = {};
56
+ try {
57
+ const nodes = await getProviderNodes();
58
+ for (const node of nodes) {
59
+ if (node.id && node.name) nodeNameMap[node.id] = node.name;
60
+ }
61
+ } catch { }
62
+
63
+ // Hide sensitive fields, enrich name for compatible providers
64
+ const safeConnections = connections.map(c => {
65
+ const isCompatible = isOpenAICompatibleProvider(c.provider) || isAnthropicCompatibleProvider(c.provider);
66
+ const name = isCompatible
67
+ ? (c.name || nodeNameMap[c.provider] || c.providerSpecificData?.nodeName || c.provider)
68
+ : c.name;
69
+ return {
70
+ ...c,
71
+ name,
72
+ apiKey: undefined,
73
+ accessToken: undefined,
74
+ refreshToken: undefined,
75
+ idToken: undefined,
76
+ };
77
+ });
78
+
79
+ return NextResponse.json({ connections: safeConnections });
80
+ } catch (error) {
81
+ console.log("Error fetching providers:", error);
82
+ return NextResponse.json({ error: "Failed to fetch providers" }, { status: 500 });
83
+ }
84
+ }
85
+
86
+ // POST /api/providers - Create new connection (API Key only, OAuth via separate flow)
87
+ export async function POST(request) {
88
+ try {
89
+ const body = await request.json();
90
+ const provider = normalizeProviderId(body.provider);
91
+ const { apiKey, name, displayName, priority, globalPriority, defaultModel, testStatus } = body;
92
+ const proxyConfig = normalizeProxyConfig(body);
93
+ if (proxyConfig.error) {
94
+ return NextResponse.json({ error: proxyConfig.error }, { status: 400 });
95
+ }
96
+
97
+ const proxyPoolResult = await normalizeProxyPoolId(body.proxyPoolId);
98
+ if (proxyPoolResult.error) {
99
+ return NextResponse.json({ error: proxyPoolResult.error }, { status: 400 });
100
+ }
101
+ const proxyPoolId = proxyPoolResult.proxyPoolId;
102
+
103
+ // Validation
104
+ const isWebCookieProvider = !!WEB_COOKIE_PROVIDERS[provider];
105
+ const isValidProvider = APIKEY_PROVIDERS[provider] ||
106
+ FREE_TIER_PROVIDERS[provider] ||
107
+ isWebCookieProvider ||
108
+ isOpenAICompatibleProvider(provider) ||
109
+ isAnthropicCompatibleProvider(provider) ||
110
+ isCustomEmbeddingProvider(provider);
111
+
112
+ if (!provider || !isValidProvider) {
113
+ return NextResponse.json({ error: "Invalid provider" }, { status: 400 });
114
+ }
115
+ if (!apiKey && provider !== "ollama-local") {
116
+ return NextResponse.json({ error: `${isWebCookieProvider ? "Cookie value" : "API Key"} is required` }, { status: 400 });
117
+ }
118
+ const connectionName = name || displayName || AI_PROVIDERS[provider]?.name;
119
+ if (!connectionName) {
120
+ return NextResponse.json({ error: "Name is required" }, { status: 400 });
121
+ }
122
+
123
+ let providerSpecificData = normalizeProviderSpecificData(provider, body, body.providerSpecificData);
124
+
125
+ // Compatible/embedding nodes allow exactly one connection each. These guards were
126
+ // dropped accidentally during the bun:sqlite refactor (v0.4.28); restored to honor
127
+ // the contract locked in by tests/unit/compatible-provider-connections.test.js (#925).
128
+ if (isOpenAICompatibleProvider(provider)) {
129
+ const node = await getProviderNodeById(provider);
130
+ if (!node) {
131
+ return NextResponse.json({ error: "OpenAI Compatible node not found" }, { status: 404 });
132
+ }
133
+ const existingConnections = await getProviderConnections({ provider });
134
+ if (existingConnections.length > 0) {
135
+ return NextResponse.json({ error: "Only one connection is allowed for this OpenAI Compatible node" }, { status: 400 });
136
+ }
137
+ providerSpecificData = {
138
+ prefix: node.prefix,
139
+ apiType: node.apiType,
140
+ baseUrl: node.baseUrl,
141
+ nodeName: node.name,
142
+ };
143
+ } else if (isAnthropicCompatibleProvider(provider)) {
144
+ const node = await getProviderNodeById(provider);
145
+ if (!node) {
146
+ return NextResponse.json({ error: "Anthropic Compatible node not found" }, { status: 404 });
147
+ }
148
+ const existingConnections = await getProviderConnections({ provider });
149
+ if (existingConnections.length > 0) {
150
+ return NextResponse.json({ error: "Only one connection is allowed for this Anthropic Compatible node" }, { status: 400 });
151
+ }
152
+ providerSpecificData = {
153
+ prefix: node.prefix,
154
+ baseUrl: node.baseUrl,
155
+ nodeName: node.name,
156
+ };
157
+ } else if (isCustomEmbeddingProvider(provider)) {
158
+ const node = await getProviderNodeById(provider);
159
+ if (!node) {
160
+ return NextResponse.json({ error: "Custom Embedding node not found" }, { status: 404 });
161
+ }
162
+ const existingConnections = await getProviderConnections({ provider });
163
+ if (existingConnections.length > 0) {
164
+ return NextResponse.json({ error: "Only one connection is allowed for this Custom Embedding node" }, { status: 400 });
165
+ }
166
+ providerSpecificData = {
167
+ prefix: node.prefix,
168
+ baseUrl: node.baseUrl,
169
+ nodeName: node.name,
170
+ };
171
+ }
172
+
173
+ const mergedProviderSpecificData = {
174
+ ...(providerSpecificData || {}),
175
+ connectionProxyEnabled: proxyConfig.connectionProxyEnabled,
176
+ connectionProxyUrl: proxyConfig.connectionProxyUrl,
177
+ connectionNoProxy: proxyConfig.connectionNoProxy,
178
+ };
179
+
180
+ if (proxyPoolId !== null) {
181
+ mergedProviderSpecificData.proxyPoolId = proxyPoolId;
182
+ }
183
+
184
+ const newConnection = await createProviderConnection({
185
+ provider,
186
+ authType: isWebCookieProvider ? "cookie" : "apikey",
187
+ name: connectionName,
188
+ apiKey: apiKey || "",
189
+ priority: priority || 1,
190
+ globalPriority: globalPriority || null,
191
+ defaultModel: defaultModel || null,
192
+ providerSpecificData: mergedProviderSpecificData,
193
+ isActive: true,
194
+ testStatus: testStatus || "unknown",
195
+ });
196
+
197
+ // Hide sensitive fields
198
+ const result = { ...newConnection };
199
+ delete result.apiKey;
200
+
201
+ return NextResponse.json({ connection: result }, { status: 201 });
202
+ } catch (error) {
203
+ console.log("Error creating provider:", error);
204
+ return NextResponse.json({ error: "Failed to create provider" }, { status: 500 });
205
+ }
206
+ }
@@ -0,0 +1,20 @@
1
+ // Free OpenCode models that don't use the "-free" id suffix
2
+ const KNOWN_FREE_OPENCODE_MODELS = ["big-pickle"];
3
+
4
+ export const FILTERS = {
5
+ "openrouter-free": (models) =>
6
+ models
7
+ .filter(
8
+ (m) =>
9
+ m.pricing?.prompt === "0" &&
10
+ m.pricing?.completion === "0" &&
11
+ m.context_length >= 200000
12
+ )
13
+ .map((m) => ({ id: m.id, name: m.name, contextLength: m.context_length }))
14
+ .sort((a, b) => b.contextLength - a.contextLength),
15
+
16
+ "opencode-free": (models) =>
17
+ models
18
+ .filter((m) => m.id?.endsWith("-free") || KNOWN_FREE_OPENCODE_MODELS.includes(m.id))
19
+ .map((m) => ({ id: m.id, name: m.id })),
20
+ };
@@ -0,0 +1,32 @@
1
+ import { NextResponse } from "next/server";
2
+ import { FILTERS } from "./filters.js";
3
+
4
+ export const dynamic = "force-dynamic";
5
+
6
+ export async function GET(request) {
7
+ const { searchParams } = new URL(request.url);
8
+ const url = searchParams.get("url");
9
+ const type = searchParams.get("type");
10
+
11
+ if (!url || !type) {
12
+ return NextResponse.json({ error: "Missing url or type" }, { status: 400 });
13
+ }
14
+
15
+ const filter = FILTERS[type];
16
+ if (!filter) {
17
+ return NextResponse.json({ error: "Unknown filter type" }, { status: 400 });
18
+ }
19
+
20
+ try {
21
+ const res = await fetch(url);
22
+ if (!res.ok) {
23
+ return NextResponse.json({ data: [] });
24
+ }
25
+ const json = await res.json();
26
+ const raw = json.data ?? json.models ?? json;
27
+ const data = filter(Array.isArray(raw) ? raw : []);
28
+ return NextResponse.json({ data });
29
+ } catch {
30
+ return NextResponse.json({ data: [] });
31
+ }
32
+ }
@@ -0,0 +1,131 @@
1
+ import { NextResponse } from "next/server";
2
+ import { getProviderConnections } from "@/models";
3
+ import {
4
+ FREE_PROVIDERS,
5
+ OAUTH_PROVIDERS,
6
+ APIKEY_PROVIDERS,
7
+ OPENAI_COMPATIBLE_PREFIX,
8
+ ANTHROPIC_COMPATIBLE_PREFIX,
9
+ } from "@/shared/constants/providers";
10
+ import { testSingleConnection } from "../[id]/test/testUtils.js";
11
+
12
+ function getAuthGroup(providerId, connection = null) {
13
+ // Prioritize authType from connection if available
14
+ if (connection?.authType) {
15
+ if (connection.authType === "oauth") {
16
+ // Check if it's a free provider
17
+ if (FREE_PROVIDERS[providerId]) return "free";
18
+ return "oauth";
19
+ }
20
+ return connection.authType;
21
+ }
22
+
23
+ // Fallback to constants
24
+ if (FREE_PROVIDERS[providerId]) return "free";
25
+ if (OAUTH_PROVIDERS[providerId]) return "oauth";
26
+ if (APIKEY_PROVIDERS[providerId]) return "apikey";
27
+ if (
28
+ typeof providerId === "string" &&
29
+ (providerId.startsWith(OPENAI_COMPATIBLE_PREFIX) || providerId.startsWith(ANTHROPIC_COMPATIBLE_PREFIX))
30
+ )
31
+ return "compatible";
32
+ return "apikey";
33
+ }
34
+
35
+ function isCompatibleProvider(providerId) {
36
+ return (
37
+ typeof providerId === "string" &&
38
+ (providerId.startsWith(OPENAI_COMPATIBLE_PREFIX) || providerId.startsWith(ANTHROPIC_COMPATIBLE_PREFIX))
39
+ );
40
+ }
41
+
42
+ // POST /api/providers/test-batch - Test multiple connections by group
43
+ export async function POST(request) {
44
+ try {
45
+ const body = await request.json();
46
+ const { mode, providerId } = body;
47
+
48
+ if (!mode) {
49
+ return NextResponse.json({ error: "mode is required" }, { status: 400 });
50
+ }
51
+
52
+ const allConnections = await getProviderConnections({ isActive: true });
53
+
54
+ let connectionsToTest = [];
55
+ if (mode === "provider" && providerId) {
56
+ connectionsToTest = allConnections.filter((c) => c.provider === providerId);
57
+ } else if (mode === "oauth") {
58
+ connectionsToTest = allConnections.filter((c) => getAuthGroup(c.provider, c) === "oauth");
59
+ } else if (mode === "free") {
60
+ connectionsToTest = allConnections.filter((c) => getAuthGroup(c.provider, c) === "free");
61
+ } else if (mode === "apikey") {
62
+ connectionsToTest = allConnections.filter((c) => getAuthGroup(c.provider, c) === "apikey");
63
+ } else if (mode === "compatible") {
64
+ connectionsToTest = allConnections.filter((c) => isCompatibleProvider(c.provider));
65
+ } else if (mode === "all") {
66
+ connectionsToTest = allConnections;
67
+ } else {
68
+ return NextResponse.json(
69
+ { error: "Invalid mode. Use: provider, oauth, free, apikey, compatible, all" },
70
+ { status: 400 }
71
+ );
72
+ }
73
+
74
+ if (connectionsToTest.length === 0) {
75
+ return NextResponse.json({
76
+ mode,
77
+ providerId: providerId || null,
78
+ results: [],
79
+ summary: { total: 0, passed: 0, failed: 0 },
80
+ testedAt: new Date().toISOString(),
81
+ });
82
+ }
83
+
84
+ const results = [];
85
+ for (const conn of connectionsToTest) {
86
+ try {
87
+ const data = await testSingleConnection(conn.id);
88
+ results.push({
89
+ provider: conn.provider,
90
+ connectionId: conn.id,
91
+ connectionName: conn.name || conn.email || conn.provider,
92
+ authType: conn.authType || getAuthGroup(conn.provider, conn),
93
+ valid: data.valid,
94
+ latencyMs: data.latencyMs || 0,
95
+ error: data.error || null,
96
+ diagnosis: data.diagnosis || null,
97
+ statusCode: data.statusCode || null,
98
+ testedAt: data.testedAt || new Date().toISOString(),
99
+ });
100
+ } catch (error) {
101
+ results.push({
102
+ provider: conn.provider,
103
+ connectionId: conn.id,
104
+ connectionName: conn.name || conn.email || conn.provider,
105
+ authType: conn.authType || getAuthGroup(conn.provider, conn),
106
+ valid: false,
107
+ latencyMs: 0,
108
+ error: error.message,
109
+ diagnosis: { type: "network_error", source: "local", code: null, message: error.message },
110
+ statusCode: null,
111
+ testedAt: new Date().toISOString(),
112
+ });
113
+ }
114
+ }
115
+
116
+ return NextResponse.json({
117
+ mode,
118
+ providerId: providerId || null,
119
+ results,
120
+ testedAt: new Date().toISOString(),
121
+ summary: {
122
+ total: results.length,
123
+ passed: results.filter((r) => r.valid).length,
124
+ failed: results.filter((r) => !r.valid).length,
125
+ },
126
+ });
127
+ } catch (error) {
128
+ console.log("Error in batch test:", error);
129
+ return NextResponse.json({ error: "Batch test failed" }, { status: 500 });
130
+ }
131
+ }