n9router 0.4.21 → 0.4.24

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 (499) hide show
  1. package/app/.next/BUILD_ID +1 -1
  2. package/app/.next/app-path-routes-manifest.json +10 -10
  3. package/app/.next/build-manifest.json +2 -2
  4. package/app/.next/prerender-manifest.json +3 -3
  5. package/app/.next/server/app/(dashboard)/dashboard/basic-chat/page.js.nft.json +1 -1
  6. package/app/.next/server/app/(dashboard)/dashboard/basic-chat/page_client-reference-manifest.js +1 -1
  7. package/app/.next/server/app/(dashboard)/dashboard/cli-tools/page.js +1 -1
  8. package/app/.next/server/app/(dashboard)/dashboard/cli-tools/page.js.nft.json +1 -1
  9. package/app/.next/server/app/(dashboard)/dashboard/cli-tools/page_client-reference-manifest.js +1 -1
  10. package/app/.next/server/app/(dashboard)/dashboard/combos/page.js +1 -1
  11. package/app/.next/server/app/(dashboard)/dashboard/combos/page.js.nft.json +1 -1
  12. package/app/.next/server/app/(dashboard)/dashboard/combos/page_client-reference-manifest.js +1 -1
  13. package/app/.next/server/app/(dashboard)/dashboard/console-log/page.js.nft.json +1 -1
  14. package/app/.next/server/app/(dashboard)/dashboard/console-log/page_client-reference-manifest.js +1 -1
  15. package/app/.next/server/app/(dashboard)/dashboard/endpoint/page.js.nft.json +1 -1
  16. package/app/.next/server/app/(dashboard)/dashboard/endpoint/page_client-reference-manifest.js +1 -1
  17. package/app/.next/server/app/(dashboard)/dashboard/media-providers/[kind]/[id]/page.js +1 -1
  18. package/app/.next/server/app/(dashboard)/dashboard/media-providers/[kind]/[id]/page.js.nft.json +1 -1
  19. package/app/.next/server/app/(dashboard)/dashboard/media-providers/[kind]/[id]/page_client-reference-manifest.js +1 -1
  20. package/app/.next/server/app/(dashboard)/dashboard/media-providers/[kind]/page.js.nft.json +1 -1
  21. package/app/.next/server/app/(dashboard)/dashboard/media-providers/[kind]/page_client-reference-manifest.js +1 -1
  22. package/app/.next/server/app/(dashboard)/dashboard/media-providers/web/combo/[id]/page.js.nft.json +1 -1
  23. package/app/.next/server/app/(dashboard)/dashboard/media-providers/web/combo/[id]/page_client-reference-manifest.js +1 -1
  24. package/app/.next/server/app/(dashboard)/dashboard/media-providers/web/page.js.nft.json +1 -1
  25. package/app/.next/server/app/(dashboard)/dashboard/media-providers/web/page_client-reference-manifest.js +1 -1
  26. package/app/.next/server/app/(dashboard)/dashboard/mitm/page.js +1 -1
  27. package/app/.next/server/app/(dashboard)/dashboard/mitm/page.js.nft.json +1 -1
  28. package/app/.next/server/app/(dashboard)/dashboard/mitm/page_client-reference-manifest.js +1 -1
  29. package/app/.next/server/app/(dashboard)/dashboard/page.js.nft.json +1 -1
  30. package/app/.next/server/app/(dashboard)/dashboard/page_client-reference-manifest.js +1 -1
  31. package/app/.next/server/app/(dashboard)/dashboard/profile/page.js +1 -1
  32. package/app/.next/server/app/(dashboard)/dashboard/profile/page.js.nft.json +1 -1
  33. package/app/.next/server/app/(dashboard)/dashboard/profile/page_client-reference-manifest.js +1 -1
  34. package/app/.next/server/app/(dashboard)/dashboard/providers/[id]/page.js +1 -1
  35. package/app/.next/server/app/(dashboard)/dashboard/providers/[id]/page.js.nft.json +1 -1
  36. package/app/.next/server/app/(dashboard)/dashboard/providers/[id]/page_client-reference-manifest.js +1 -1
  37. package/app/.next/server/app/(dashboard)/dashboard/providers/new/page.js.nft.json +1 -1
  38. package/app/.next/server/app/(dashboard)/dashboard/providers/new/page_client-reference-manifest.js +1 -1
  39. package/app/.next/server/app/(dashboard)/dashboard/providers/page.js +1 -1
  40. package/app/.next/server/app/(dashboard)/dashboard/providers/page.js.nft.json +1 -1
  41. package/app/.next/server/app/(dashboard)/dashboard/providers/page_client-reference-manifest.js +1 -1
  42. package/app/.next/server/app/(dashboard)/dashboard/proxy-pools/page.js +1 -1
  43. package/app/.next/server/app/(dashboard)/dashboard/proxy-pools/page.js.nft.json +1 -1
  44. package/app/.next/server/app/(dashboard)/dashboard/proxy-pools/page_client-reference-manifest.js +1 -1
  45. package/app/.next/server/app/(dashboard)/dashboard/quota/page.js +1 -1
  46. package/app/.next/server/app/(dashboard)/dashboard/quota/page.js.nft.json +1 -1
  47. package/app/.next/server/app/(dashboard)/dashboard/quota/page_client-reference-manifest.js +1 -1
  48. package/app/.next/server/app/(dashboard)/dashboard/translator/page.js.nft.json +1 -1
  49. package/app/.next/server/app/(dashboard)/dashboard/translator/page_client-reference-manifest.js +1 -1
  50. package/app/.next/server/app/(dashboard)/dashboard/usage/page.js +1 -1
  51. package/app/.next/server/app/(dashboard)/dashboard/usage/page.js.nft.json +1 -1
  52. package/app/.next/server/app/(dashboard)/dashboard/usage/page_client-reference-manifest.js +1 -1
  53. package/app/.next/server/app/_global-error/page.js.nft.json +1 -1
  54. package/app/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
  55. package/app/.next/server/app/_global-error.html +1 -1
  56. package/app/.next/server/app/_global-error.rsc +1 -1
  57. package/app/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  58. package/app/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  59. package/app/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  60. package/app/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  61. package/app/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  62. package/app/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  63. package/app/.next/server/app/_not-found/page.js.nft.json +1 -1
  64. package/app/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  65. package/app/.next/server/app/_not-found.html +1 -1
  66. package/app/.next/server/app/_not-found.rsc +5 -5
  67. package/app/.next/server/app/_not-found.segments/_full.segment.rsc +5 -5
  68. package/app/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  69. package/app/.next/server/app/_not-found.segments/_index.segment.rsc +4 -4
  70. package/app/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  71. package/app/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  72. package/app/.next/server/app/_not-found.segments/_tree.segment.rsc +2 -2
  73. package/app/.next/server/app/api/antigravity-ide/route.js.nft.json +1 -1
  74. package/app/.next/server/app/api/antigravity-tools/import/route.js.nft.json +1 -1
  75. package/app/.next/server/app/api/antigravity-tools/import-refresh-tokens/route.js.nft.json +1 -1
  76. package/app/.next/server/app/api/auth/login/route.js.nft.json +1 -1
  77. package/app/.next/server/app/api/auth/logout/route.js.nft.json +1 -1
  78. package/app/.next/server/app/api/cli-tools/antigravity-mitm/alias/route.js +2 -2
  79. package/app/.next/server/app/api/cli-tools/antigravity-mitm/alias/route.js.nft.json +1 -1
  80. package/app/.next/server/app/api/cli-tools/antigravity-mitm/route.js.nft.json +1 -1
  81. package/app/.next/server/app/api/cli-tools/claude-settings/route.js +2 -2
  82. package/app/.next/server/app/api/cli-tools/claude-settings/route.js.nft.json +1 -1
  83. package/app/.next/server/app/api/cli-tools/codex-settings/route.js +2 -2
  84. package/app/.next/server/app/api/cli-tools/codex-settings/route.js.nft.json +1 -1
  85. package/app/.next/server/app/api/cli-tools/copilot-settings/route.js +2 -2
  86. package/app/.next/server/app/api/cli-tools/copilot-settings/route.js.nft.json +1 -1
  87. package/app/.next/server/app/api/cli-tools/droid-settings/route.js +1 -1
  88. package/app/.next/server/app/api/cli-tools/droid-settings/route.js.nft.json +1 -1
  89. package/app/.next/server/app/api/cli-tools/hermes-settings/route.js +3 -3
  90. package/app/.next/server/app/api/cli-tools/hermes-settings/route.js.nft.json +1 -1
  91. package/app/.next/server/app/api/cli-tools/openclaw-settings/route.js +2 -2
  92. package/app/.next/server/app/api/cli-tools/openclaw-settings/route.js.nft.json +1 -1
  93. package/app/.next/server/app/api/cli-tools/opencode-settings/route.js +2 -2
  94. package/app/.next/server/app/api/cli-tools/opencode-settings/route.js.nft.json +1 -1
  95. package/app/.next/server/app/api/cloud/auth/route.js.nft.json +1 -1
  96. package/app/.next/server/app/api/cloud/credentials/update/route.js.nft.json +1 -1
  97. package/app/.next/server/app/api/cloud/model/resolve/route.js.nft.json +1 -1
  98. package/app/.next/server/app/api/cloud/models/alias/route.js.nft.json +1 -1
  99. package/app/.next/server/app/api/combos/[id]/route.js +1 -1
  100. package/app/.next/server/app/api/combos/[id]/route.js.nft.json +1 -1
  101. package/app/.next/server/app/api/combos/route.js.nft.json +1 -1
  102. package/app/.next/server/app/api/health/route.js.nft.json +1 -1
  103. package/app/.next/server/app/api/init/route.js.nft.json +1 -1
  104. package/app/.next/server/app/api/internal/account-health/route.js.nft.json +1 -1
  105. package/app/.next/server/app/api/internal/request-detail/route.js.nft.json +1 -1
  106. package/app/.next/server/app/api/internal/usage/route.js.nft.json +1 -1
  107. package/app/.next/server/app/api/keys/[id]/route.js.nft.json +1 -1
  108. package/app/.next/server/app/api/keys/route.js.nft.json +1 -1
  109. package/app/.next/server/app/api/locale/route.js.nft.json +1 -1
  110. package/app/.next/server/app/api/media-providers/tts/deepgram/voices/route.js.nft.json +1 -1
  111. package/app/.next/server/app/api/media-providers/tts/elevenlabs/voices/route.js +1 -1
  112. package/app/.next/server/app/api/media-providers/tts/elevenlabs/voices/route.js.nft.json +1 -1
  113. package/app/.next/server/app/api/media-providers/tts/inworld/voices/route.js.nft.json +1 -1
  114. package/app/.next/server/app/api/media-providers/tts/voices/route.js +1 -1
  115. package/app/.next/server/app/api/media-providers/tts/voices/route.js.nft.json +1 -1
  116. package/app/.next/server/app/api/models/alias/route.js.nft.json +1 -1
  117. package/app/.next/server/app/api/models/availability/route.js.nft.json +1 -1
  118. package/app/.next/server/app/api/models/custom/route.js.nft.json +1 -1
  119. package/app/.next/server/app/api/models/route.js +1 -1
  120. package/app/.next/server/app/api/models/route.js.nft.json +1 -1
  121. package/app/.next/server/app/api/models/test/route.js.nft.json +1 -1
  122. package/app/.next/server/app/api/oauth/[provider]/[action]/route.js +1 -1
  123. package/app/.next/server/app/api/oauth/[provider]/[action]/route.js.nft.json +1 -1
  124. package/app/.next/server/app/api/oauth/cursor/auto-import/route.js.nft.json +1 -1
  125. package/app/.next/server/app/api/oauth/cursor/import/route.js.nft.json +1 -1
  126. package/app/.next/server/app/api/oauth/gitlab/pat/route.js.nft.json +1 -1
  127. package/app/.next/server/app/api/oauth/iflow/cookie/route.js.nft.json +1 -1
  128. package/app/.next/server/app/api/oauth/kiro/auto-import/route.js.nft.json +1 -1
  129. package/app/.next/server/app/api/oauth/kiro/import/route.js.nft.json +1 -1
  130. package/app/.next/server/app/api/oauth/kiro/social-authorize/route.js.nft.json +1 -1
  131. package/app/.next/server/app/api/oauth/kiro/social-exchange/route.js.nft.json +1 -1
  132. package/app/.next/server/app/api/pricing/route.js.nft.json +1 -1
  133. package/app/.next/server/app/api/provider-nodes/[id]/route.js.nft.json +1 -1
  134. package/app/.next/server/app/api/provider-nodes/route.js.nft.json +1 -1
  135. package/app/.next/server/app/api/provider-nodes/validate/route.js.nft.json +1 -1
  136. package/app/.next/server/app/api/providers/[id]/models/route.js +1 -1
  137. package/app/.next/server/app/api/providers/[id]/models/route.js.nft.json +1 -1
  138. package/app/.next/server/app/api/providers/[id]/route.js.nft.json +1 -1
  139. package/app/.next/server/app/api/providers/[id]/test/route.js.nft.json +1 -1
  140. package/app/.next/server/app/api/providers/[id]/test-models/route.js.nft.json +1 -1
  141. package/app/.next/server/app/api/providers/client/route.js +1 -1
  142. package/app/.next/server/app/api/providers/client/route.js.nft.json +1 -1
  143. package/app/.next/server/app/api/providers/kilo/free-models/route.js.nft.json +1 -1
  144. package/app/.next/server/app/api/providers/route.js +1 -1
  145. package/app/.next/server/app/api/providers/route.js.nft.json +1 -1
  146. package/app/.next/server/app/api/providers/suggested-models/route.js.nft.json +1 -1
  147. package/app/.next/server/app/api/providers/test-batch/route.js.nft.json +1 -1
  148. package/app/.next/server/app/api/providers/validate/route.js +1 -1
  149. package/app/.next/server/app/api/providers/validate/route.js.nft.json +1 -1
  150. package/app/.next/server/app/api/proxy-pools/[id]/route.js.nft.json +1 -1
  151. package/app/.next/server/app/api/proxy-pools/[id]/test/route.js.nft.json +1 -1
  152. package/app/.next/server/app/api/proxy-pools/route.js.nft.json +1 -1
  153. package/app/.next/server/app/api/proxy-pools/vercel-deploy/route.js.nft.json +1 -1
  154. package/app/.next/server/app/api/settings/database/route.js.nft.json +1 -1
  155. package/app/.next/server/app/api/settings/proxy-test/route.js.nft.json +1 -1
  156. package/app/.next/server/app/api/settings/require-login/route.js.nft.json +1 -1
  157. package/app/.next/server/app/api/settings/route.js +1 -1
  158. package/app/.next/server/app/api/settings/route.js.nft.json +1 -1
  159. package/app/.next/server/app/api/shutdown/route.js.nft.json +1 -1
  160. package/app/.next/server/app/api/tags/route.js.nft.json +1 -1
  161. package/app/.next/server/app/api/translator/console-logs/route.js +1 -1
  162. package/app/.next/server/app/api/translator/console-logs/route.js.nft.json +1 -1
  163. package/app/.next/server/app/api/translator/console-logs/stream/route.js +1 -1
  164. package/app/.next/server/app/api/translator/console-logs/stream/route.js.nft.json +1 -1
  165. package/app/.next/server/app/api/translator/load/route.js.nft.json +1 -1
  166. package/app/.next/server/app/api/translator/save/route.js.nft.json +1 -1
  167. package/app/.next/server/app/api/translator/send/route.js +1 -1
  168. package/app/.next/server/app/api/translator/send/route.js.nft.json +1 -1
  169. package/app/.next/server/app/api/translator/translate/route.js +1 -1
  170. package/app/.next/server/app/api/translator/translate/route.js.nft.json +1 -1
  171. package/app/.next/server/app/api/tunnel/disable/route.js.nft.json +1 -1
  172. package/app/.next/server/app/api/tunnel/enable/route.js.nft.json +1 -1
  173. package/app/.next/server/app/api/tunnel/status/route.js.nft.json +1 -1
  174. package/app/.next/server/app/api/tunnel/tailscale-check/route.js.nft.json +1 -1
  175. package/app/.next/server/app/api/tunnel/tailscale-disable/route.js.nft.json +1 -1
  176. package/app/.next/server/app/api/tunnel/tailscale-enable/route.js.nft.json +1 -1
  177. package/app/.next/server/app/api/tunnel/tailscale-install/route.js +2 -2
  178. package/app/.next/server/app/api/tunnel/tailscale-install/route.js.nft.json +1 -1
  179. package/app/.next/server/app/api/tunnel/tailscale-login/route.js.nft.json +1 -1
  180. package/app/.next/server/app/api/tunnel/tailscale-start-daemon/route.js +1 -1
  181. package/app/.next/server/app/api/tunnel/tailscale-start-daemon/route.js.nft.json +1 -1
  182. package/app/.next/server/app/api/usage/[connectionId]/route.js +1 -1
  183. package/app/.next/server/app/api/usage/[connectionId]/route.js.nft.json +1 -1
  184. package/app/.next/server/app/api/usage/chart/route.js.nft.json +1 -1
  185. package/app/.next/server/app/api/usage/history/route.js.nft.json +1 -1
  186. package/app/.next/server/app/api/usage/logs/route.js.nft.json +1 -1
  187. package/app/.next/server/app/api/usage/providers/route.js +1 -1
  188. package/app/.next/server/app/api/usage/providers/route.js.nft.json +1 -1
  189. package/app/.next/server/app/api/usage/request-details/route.js.nft.json +1 -1
  190. package/app/.next/server/app/api/usage/request-logs/route.js.nft.json +1 -1
  191. package/app/.next/server/app/api/usage/stats/route.js.nft.json +1 -1
  192. package/app/.next/server/app/api/usage/stream/route.js.nft.json +1 -1
  193. package/app/.next/server/app/api/v1/api/chat/route.js +1 -1
  194. package/app/.next/server/app/api/v1/api/chat/route.js.nft.json +1 -1
  195. package/app/.next/server/app/api/v1/audio/speech/route.js +1 -1
  196. package/app/.next/server/app/api/v1/audio/speech/route.js.nft.json +1 -1
  197. package/app/.next/server/app/api/v1/chat/completions/route.js +1 -1
  198. package/app/.next/server/app/api/v1/chat/completions/route.js.nft.json +1 -1
  199. package/app/.next/server/app/api/v1/embeddings/route.js +1 -1
  200. package/app/.next/server/app/api/v1/embeddings/route.js.nft.json +1 -1
  201. package/app/.next/server/app/api/v1/images/generations/route.js +2 -2
  202. package/app/.next/server/app/api/v1/images/generations/route.js.nft.json +1 -1
  203. package/app/.next/server/app/api/v1/messages/count_tokens/route.js.nft.json +1 -1
  204. package/app/.next/server/app/api/v1/messages/route.js +1 -1
  205. package/app/.next/server/app/api/v1/messages/route.js.nft.json +1 -1
  206. package/app/.next/server/app/api/v1/models/route.js.nft.json +1 -1
  207. package/app/.next/server/app/api/v1/responses/compact/route.js +1 -1
  208. package/app/.next/server/app/api/v1/responses/compact/route.js.nft.json +1 -1
  209. package/app/.next/server/app/api/v1/responses/route.js +1 -1
  210. package/app/.next/server/app/api/v1/responses/route.js.nft.json +1 -1
  211. package/app/.next/server/app/api/v1/route.js.nft.json +1 -1
  212. package/app/.next/server/app/api/v1/search/route.js +1 -1
  213. package/app/.next/server/app/api/v1/search/route.js.nft.json +1 -1
  214. package/app/.next/server/app/api/v1/web/fetch/route.js +1 -1
  215. package/app/.next/server/app/api/v1/web/fetch/route.js.nft.json +1 -1
  216. package/app/.next/server/app/api/v1beta/models/[...path]/route.js +1 -1
  217. package/app/.next/server/app/api/v1beta/models/[...path]/route.js.nft.json +1 -1
  218. package/app/.next/server/app/api/v1beta/models/route.js.nft.json +1 -1
  219. package/app/.next/server/app/api/version/route.js +1 -1
  220. package/app/.next/server/app/api/version/route.js.nft.json +1 -1
  221. package/app/.next/server/app/api/version/update/route.js +1 -1
  222. package/app/.next/server/app/api/version/update/route.js.nft.json +1 -1
  223. package/app/.next/server/app/callback/page.js.nft.json +1 -1
  224. package/app/.next/server/app/callback/page_client-reference-manifest.js +1 -1
  225. package/app/.next/server/app/callback.html +1 -1
  226. package/app/.next/server/app/callback.rsc +5 -5
  227. package/app/.next/server/app/callback.segments/_full.segment.rsc +5 -5
  228. package/app/.next/server/app/callback.segments/_head.segment.rsc +1 -1
  229. package/app/.next/server/app/callback.segments/_index.segment.rsc +4 -4
  230. package/app/.next/server/app/callback.segments/_tree.segment.rsc +2 -2
  231. package/app/.next/server/app/callback.segments/callback/__PAGE__.segment.rsc +1 -1
  232. package/app/.next/server/app/callback.segments/callback.segment.rsc +1 -1
  233. package/app/.next/server/app/dashboard/basic-chat.html +1 -1
  234. package/app/.next/server/app/dashboard/basic-chat.rsc +7 -7
  235. package/app/.next/server/app/dashboard/basic-chat.segments/!KGRhc2hib2FyZCk/dashboard/basic-chat/__PAGE__.segment.rsc +2 -2
  236. package/app/.next/server/app/dashboard/basic-chat.segments/!KGRhc2hib2FyZCk/dashboard/basic-chat.segment.rsc +1 -1
  237. package/app/.next/server/app/dashboard/basic-chat.segments/!KGRhc2hib2FyZCk/dashboard.segment.rsc +1 -1
  238. package/app/.next/server/app/dashboard/basic-chat.segments/!KGRhc2hib2FyZCk.segment.rsc +2 -2
  239. package/app/.next/server/app/dashboard/basic-chat.segments/_full.segment.rsc +7 -7
  240. package/app/.next/server/app/dashboard/basic-chat.segments/_head.segment.rsc +1 -1
  241. package/app/.next/server/app/dashboard/basic-chat.segments/_index.segment.rsc +4 -4
  242. package/app/.next/server/app/dashboard/basic-chat.segments/_tree.segment.rsc +2 -2
  243. package/app/.next/server/app/dashboard/cli-tools.html +1 -1
  244. package/app/.next/server/app/dashboard/cli-tools.rsc +7 -7
  245. package/app/.next/server/app/dashboard/cli-tools.segments/!KGRhc2hib2FyZCk/dashboard/cli-tools/__PAGE__.segment.rsc +2 -2
  246. package/app/.next/server/app/dashboard/cli-tools.segments/!KGRhc2hib2FyZCk/dashboard/cli-tools.segment.rsc +1 -1
  247. package/app/.next/server/app/dashboard/cli-tools.segments/!KGRhc2hib2FyZCk/dashboard.segment.rsc +1 -1
  248. package/app/.next/server/app/dashboard/cli-tools.segments/!KGRhc2hib2FyZCk.segment.rsc +2 -2
  249. package/app/.next/server/app/dashboard/cli-tools.segments/_full.segment.rsc +7 -7
  250. package/app/.next/server/app/dashboard/cli-tools.segments/_head.segment.rsc +1 -1
  251. package/app/.next/server/app/dashboard/cli-tools.segments/_index.segment.rsc +4 -4
  252. package/app/.next/server/app/dashboard/cli-tools.segments/_tree.segment.rsc +2 -2
  253. package/app/.next/server/app/dashboard/combos.html +1 -1
  254. package/app/.next/server/app/dashboard/combos.rsc +7 -7
  255. package/app/.next/server/app/dashboard/combos.segments/!KGRhc2hib2FyZCk/dashboard/combos/__PAGE__.segment.rsc +2 -2
  256. package/app/.next/server/app/dashboard/combos.segments/!KGRhc2hib2FyZCk/dashboard/combos.segment.rsc +1 -1
  257. package/app/.next/server/app/dashboard/combos.segments/!KGRhc2hib2FyZCk/dashboard.segment.rsc +1 -1
  258. package/app/.next/server/app/dashboard/combos.segments/!KGRhc2hib2FyZCk.segment.rsc +2 -2
  259. package/app/.next/server/app/dashboard/combos.segments/_full.segment.rsc +7 -7
  260. package/app/.next/server/app/dashboard/combos.segments/_head.segment.rsc +1 -1
  261. package/app/.next/server/app/dashboard/combos.segments/_index.segment.rsc +4 -4
  262. package/app/.next/server/app/dashboard/combos.segments/_tree.segment.rsc +2 -2
  263. package/app/.next/server/app/dashboard/endpoint.html +1 -1
  264. package/app/.next/server/app/dashboard/endpoint.rsc +7 -7
  265. package/app/.next/server/app/dashboard/endpoint.segments/!KGRhc2hib2FyZCk/dashboard/endpoint/__PAGE__.segment.rsc +2 -2
  266. package/app/.next/server/app/dashboard/endpoint.segments/!KGRhc2hib2FyZCk/dashboard/endpoint.segment.rsc +1 -1
  267. package/app/.next/server/app/dashboard/endpoint.segments/!KGRhc2hib2FyZCk/dashboard.segment.rsc +1 -1
  268. package/app/.next/server/app/dashboard/endpoint.segments/!KGRhc2hib2FyZCk.segment.rsc +2 -2
  269. package/app/.next/server/app/dashboard/endpoint.segments/_full.segment.rsc +7 -7
  270. package/app/.next/server/app/dashboard/endpoint.segments/_head.segment.rsc +1 -1
  271. package/app/.next/server/app/dashboard/endpoint.segments/_index.segment.rsc +4 -4
  272. package/app/.next/server/app/dashboard/endpoint.segments/_tree.segment.rsc +2 -2
  273. package/app/.next/server/app/dashboard/media-providers/web.html +1 -1
  274. package/app/.next/server/app/dashboard/media-providers/web.rsc +7 -7
  275. package/app/.next/server/app/dashboard/media-providers/web.segments/!KGRhc2hib2FyZCk/dashboard/media-providers/web/__PAGE__.segment.rsc +2 -2
  276. package/app/.next/server/app/dashboard/media-providers/web.segments/!KGRhc2hib2FyZCk/dashboard/media-providers/web.segment.rsc +1 -1
  277. package/app/.next/server/app/dashboard/media-providers/web.segments/!KGRhc2hib2FyZCk/dashboard/media-providers.segment.rsc +1 -1
  278. package/app/.next/server/app/dashboard/media-providers/web.segments/!KGRhc2hib2FyZCk/dashboard.segment.rsc +1 -1
  279. package/app/.next/server/app/dashboard/media-providers/web.segments/!KGRhc2hib2FyZCk.segment.rsc +2 -2
  280. package/app/.next/server/app/dashboard/media-providers/web.segments/_full.segment.rsc +7 -7
  281. package/app/.next/server/app/dashboard/media-providers/web.segments/_head.segment.rsc +1 -1
  282. package/app/.next/server/app/dashboard/media-providers/web.segments/_index.segment.rsc +4 -4
  283. package/app/.next/server/app/dashboard/media-providers/web.segments/_tree.segment.rsc +2 -2
  284. package/app/.next/server/app/dashboard/mitm.html +1 -1
  285. package/app/.next/server/app/dashboard/mitm.rsc +7 -7
  286. package/app/.next/server/app/dashboard/mitm.segments/!KGRhc2hib2FyZCk/dashboard/mitm/__PAGE__.segment.rsc +2 -2
  287. package/app/.next/server/app/dashboard/mitm.segments/!KGRhc2hib2FyZCk/dashboard/mitm.segment.rsc +1 -1
  288. package/app/.next/server/app/dashboard/mitm.segments/!KGRhc2hib2FyZCk/dashboard.segment.rsc +1 -1
  289. package/app/.next/server/app/dashboard/mitm.segments/!KGRhc2hib2FyZCk.segment.rsc +2 -2
  290. package/app/.next/server/app/dashboard/mitm.segments/_full.segment.rsc +7 -7
  291. package/app/.next/server/app/dashboard/mitm.segments/_head.segment.rsc +1 -1
  292. package/app/.next/server/app/dashboard/mitm.segments/_index.segment.rsc +4 -4
  293. package/app/.next/server/app/dashboard/mitm.segments/_tree.segment.rsc +2 -2
  294. package/app/.next/server/app/dashboard/profile.html +1 -1
  295. package/app/.next/server/app/dashboard/profile.rsc +7 -7
  296. package/app/.next/server/app/dashboard/profile.segments/!KGRhc2hib2FyZCk/dashboard/profile/__PAGE__.segment.rsc +2 -2
  297. package/app/.next/server/app/dashboard/profile.segments/!KGRhc2hib2FyZCk/dashboard/profile.segment.rsc +1 -1
  298. package/app/.next/server/app/dashboard/profile.segments/!KGRhc2hib2FyZCk/dashboard.segment.rsc +1 -1
  299. package/app/.next/server/app/dashboard/profile.segments/!KGRhc2hib2FyZCk.segment.rsc +2 -2
  300. package/app/.next/server/app/dashboard/profile.segments/_full.segment.rsc +7 -7
  301. package/app/.next/server/app/dashboard/profile.segments/_head.segment.rsc +1 -1
  302. package/app/.next/server/app/dashboard/profile.segments/_index.segment.rsc +4 -4
  303. package/app/.next/server/app/dashboard/profile.segments/_tree.segment.rsc +2 -2
  304. package/app/.next/server/app/dashboard/providers/new.html +1 -1
  305. package/app/.next/server/app/dashboard/providers/new.rsc +7 -7
  306. package/app/.next/server/app/dashboard/providers/new.segments/!KGRhc2hib2FyZCk/dashboard/providers/new/__PAGE__.segment.rsc +2 -2
  307. package/app/.next/server/app/dashboard/providers/new.segments/!KGRhc2hib2FyZCk/dashboard/providers/new.segment.rsc +1 -1
  308. package/app/.next/server/app/dashboard/providers/new.segments/!KGRhc2hib2FyZCk/dashboard/providers.segment.rsc +1 -1
  309. package/app/.next/server/app/dashboard/providers/new.segments/!KGRhc2hib2FyZCk/dashboard.segment.rsc +1 -1
  310. package/app/.next/server/app/dashboard/providers/new.segments/!KGRhc2hib2FyZCk.segment.rsc +2 -2
  311. package/app/.next/server/app/dashboard/providers/new.segments/_full.segment.rsc +7 -7
  312. package/app/.next/server/app/dashboard/providers/new.segments/_head.segment.rsc +1 -1
  313. package/app/.next/server/app/dashboard/providers/new.segments/_index.segment.rsc +4 -4
  314. package/app/.next/server/app/dashboard/providers/new.segments/_tree.segment.rsc +2 -2
  315. package/app/.next/server/app/dashboard/providers.html +1 -1
  316. package/app/.next/server/app/dashboard/providers.rsc +7 -7
  317. package/app/.next/server/app/dashboard/providers.segments/!KGRhc2hib2FyZCk/dashboard/providers/__PAGE__.segment.rsc +2 -2
  318. package/app/.next/server/app/dashboard/providers.segments/!KGRhc2hib2FyZCk/dashboard/providers.segment.rsc +1 -1
  319. package/app/.next/server/app/dashboard/providers.segments/!KGRhc2hib2FyZCk/dashboard.segment.rsc +1 -1
  320. package/app/.next/server/app/dashboard/providers.segments/!KGRhc2hib2FyZCk.segment.rsc +2 -2
  321. package/app/.next/server/app/dashboard/providers.segments/_full.segment.rsc +7 -7
  322. package/app/.next/server/app/dashboard/providers.segments/_head.segment.rsc +1 -1
  323. package/app/.next/server/app/dashboard/providers.segments/_index.segment.rsc +4 -4
  324. package/app/.next/server/app/dashboard/providers.segments/_tree.segment.rsc +2 -2
  325. package/app/.next/server/app/dashboard/proxy-pools.html +1 -1
  326. package/app/.next/server/app/dashboard/proxy-pools.rsc +7 -7
  327. package/app/.next/server/app/dashboard/proxy-pools.segments/!KGRhc2hib2FyZCk/dashboard/proxy-pools/__PAGE__.segment.rsc +2 -2
  328. package/app/.next/server/app/dashboard/proxy-pools.segments/!KGRhc2hib2FyZCk/dashboard/proxy-pools.segment.rsc +1 -1
  329. package/app/.next/server/app/dashboard/proxy-pools.segments/!KGRhc2hib2FyZCk/dashboard.segment.rsc +1 -1
  330. package/app/.next/server/app/dashboard/proxy-pools.segments/!KGRhc2hib2FyZCk.segment.rsc +2 -2
  331. package/app/.next/server/app/dashboard/proxy-pools.segments/_full.segment.rsc +7 -7
  332. package/app/.next/server/app/dashboard/proxy-pools.segments/_head.segment.rsc +1 -1
  333. package/app/.next/server/app/dashboard/proxy-pools.segments/_index.segment.rsc +4 -4
  334. package/app/.next/server/app/dashboard/proxy-pools.segments/_tree.segment.rsc +2 -2
  335. package/app/.next/server/app/dashboard/quota.html +2 -2
  336. package/app/.next/server/app/dashboard/quota.rsc +8 -8
  337. package/app/.next/server/app/dashboard/quota.segments/!KGRhc2hib2FyZCk/dashboard/quota/__PAGE__.segment.rsc +3 -3
  338. package/app/.next/server/app/dashboard/quota.segments/!KGRhc2hib2FyZCk/dashboard/quota.segment.rsc +1 -1
  339. package/app/.next/server/app/dashboard/quota.segments/!KGRhc2hib2FyZCk/dashboard.segment.rsc +1 -1
  340. package/app/.next/server/app/dashboard/quota.segments/!KGRhc2hib2FyZCk.segment.rsc +2 -2
  341. package/app/.next/server/app/dashboard/quota.segments/_full.segment.rsc +8 -8
  342. package/app/.next/server/app/dashboard/quota.segments/_head.segment.rsc +1 -1
  343. package/app/.next/server/app/dashboard/quota.segments/_index.segment.rsc +4 -4
  344. package/app/.next/server/app/dashboard/quota.segments/_tree.segment.rsc +2 -2
  345. package/app/.next/server/app/dashboard/settings/pricing/page.js.nft.json +1 -1
  346. package/app/.next/server/app/dashboard/settings/pricing/page_client-reference-manifest.js +1 -1
  347. package/app/.next/server/app/dashboard/settings/pricing.html +1 -1
  348. package/app/.next/server/app/dashboard/settings/pricing.rsc +5 -5
  349. package/app/.next/server/app/dashboard/settings/pricing.segments/_full.segment.rsc +5 -5
  350. package/app/.next/server/app/dashboard/settings/pricing.segments/_head.segment.rsc +1 -1
  351. package/app/.next/server/app/dashboard/settings/pricing.segments/_index.segment.rsc +4 -4
  352. package/app/.next/server/app/dashboard/settings/pricing.segments/_tree.segment.rsc +2 -2
  353. package/app/.next/server/app/dashboard/settings/pricing.segments/dashboard/settings/pricing/__PAGE__.segment.rsc +1 -1
  354. package/app/.next/server/app/dashboard/settings/pricing.segments/dashboard/settings/pricing.segment.rsc +1 -1
  355. package/app/.next/server/app/dashboard/settings/pricing.segments/dashboard/settings.segment.rsc +1 -1
  356. package/app/.next/server/app/dashboard/settings/pricing.segments/dashboard.segment.rsc +1 -1
  357. package/app/.next/server/app/dashboard/translator.html +1 -1
  358. package/app/.next/server/app/dashboard/translator.rsc +7 -7
  359. package/app/.next/server/app/dashboard/translator.segments/!KGRhc2hib2FyZCk/dashboard/translator/__PAGE__.segment.rsc +2 -2
  360. package/app/.next/server/app/dashboard/translator.segments/!KGRhc2hib2FyZCk/dashboard/translator.segment.rsc +1 -1
  361. package/app/.next/server/app/dashboard/translator.segments/!KGRhc2hib2FyZCk/dashboard.segment.rsc +1 -1
  362. package/app/.next/server/app/dashboard/translator.segments/!KGRhc2hib2FyZCk.segment.rsc +2 -2
  363. package/app/.next/server/app/dashboard/translator.segments/_full.segment.rsc +7 -7
  364. package/app/.next/server/app/dashboard/translator.segments/_head.segment.rsc +1 -1
  365. package/app/.next/server/app/dashboard/translator.segments/_index.segment.rsc +4 -4
  366. package/app/.next/server/app/dashboard/translator.segments/_tree.segment.rsc +2 -2
  367. package/app/.next/server/app/dashboard/usage.html +1 -1
  368. package/app/.next/server/app/dashboard/usage.rsc +7 -7
  369. package/app/.next/server/app/dashboard/usage.segments/!KGRhc2hib2FyZCk/dashboard/usage/__PAGE__.segment.rsc +2 -2
  370. package/app/.next/server/app/dashboard/usage.segments/!KGRhc2hib2FyZCk/dashboard/usage.segment.rsc +1 -1
  371. package/app/.next/server/app/dashboard/usage.segments/!KGRhc2hib2FyZCk/dashboard.segment.rsc +1 -1
  372. package/app/.next/server/app/dashboard/usage.segments/!KGRhc2hib2FyZCk.segment.rsc +2 -2
  373. package/app/.next/server/app/dashboard/usage.segments/_full.segment.rsc +7 -7
  374. package/app/.next/server/app/dashboard/usage.segments/_head.segment.rsc +1 -1
  375. package/app/.next/server/app/dashboard/usage.segments/_index.segment.rsc +4 -4
  376. package/app/.next/server/app/dashboard/usage.segments/_tree.segment.rsc +2 -2
  377. package/app/.next/server/app/dashboard.html +1 -1
  378. package/app/.next/server/app/dashboard.rsc +7 -7
  379. package/app/.next/server/app/dashboard.segments/!KGRhc2hib2FyZCk/dashboard/__PAGE__.segment.rsc +2 -2
  380. package/app/.next/server/app/dashboard.segments/!KGRhc2hib2FyZCk/dashboard.segment.rsc +1 -1
  381. package/app/.next/server/app/dashboard.segments/!KGRhc2hib2FyZCk.segment.rsc +2 -2
  382. package/app/.next/server/app/dashboard.segments/_full.segment.rsc +7 -7
  383. package/app/.next/server/app/dashboard.segments/_head.segment.rsc +1 -1
  384. package/app/.next/server/app/dashboard.segments/_index.segment.rsc +4 -4
  385. package/app/.next/server/app/dashboard.segments/_tree.segment.rsc +2 -2
  386. package/app/.next/server/app/favicon.ico/route.js.nft.json +1 -1
  387. package/app/.next/server/app/index.html +1 -1
  388. package/app/.next/server/app/index.rsc +5 -5
  389. package/app/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  390. package/app/.next/server/app/index.segments/_full.segment.rsc +5 -5
  391. package/app/.next/server/app/index.segments/_head.segment.rsc +1 -1
  392. package/app/.next/server/app/index.segments/_index.segment.rsc +4 -4
  393. package/app/.next/server/app/index.segments/_tree.segment.rsc +2 -2
  394. package/app/.next/server/app/landing/page.js.nft.json +1 -1
  395. package/app/.next/server/app/landing/page_client-reference-manifest.js +1 -1
  396. package/app/.next/server/app/landing.html +1 -1
  397. package/app/.next/server/app/landing.rsc +5 -5
  398. package/app/.next/server/app/landing.segments/_full.segment.rsc +5 -5
  399. package/app/.next/server/app/landing.segments/_head.segment.rsc +1 -1
  400. package/app/.next/server/app/landing.segments/_index.segment.rsc +4 -4
  401. package/app/.next/server/app/landing.segments/_tree.segment.rsc +2 -2
  402. package/app/.next/server/app/landing.segments/landing/__PAGE__.segment.rsc +1 -1
  403. package/app/.next/server/app/landing.segments/landing.segment.rsc +1 -1
  404. package/app/.next/server/app/login/page.js.nft.json +1 -1
  405. package/app/.next/server/app/login/page_client-reference-manifest.js +1 -1
  406. package/app/.next/server/app/login.html +1 -1
  407. package/app/.next/server/app/login.rsc +6 -6
  408. package/app/.next/server/app/login.segments/_full.segment.rsc +6 -6
  409. package/app/.next/server/app/login.segments/_head.segment.rsc +1 -1
  410. package/app/.next/server/app/login.segments/_index.segment.rsc +4 -4
  411. package/app/.next/server/app/login.segments/_tree.segment.rsc +2 -2
  412. package/app/.next/server/app/login.segments/login/__PAGE__.segment.rsc +2 -2
  413. package/app/.next/server/app/login.segments/login.segment.rsc +1 -1
  414. package/app/.next/server/app/manifest.webmanifest/route.js +1 -1
  415. package/app/.next/server/app/manifest.webmanifest/route.js.nft.json +1 -1
  416. package/app/.next/server/app/manifest.webmanifest.body +1 -1
  417. package/app/.next/server/app/page.js.nft.json +1 -1
  418. package/app/.next/server/app/page_client-reference-manifest.js +1 -1
  419. package/app/.next/server/app-paths-manifest.json +10 -10
  420. package/app/.next/server/chunks/1051.js +1 -0
  421. package/app/.next/server/chunks/318.js +3 -3
  422. package/app/.next/server/chunks/3646.js +1 -1
  423. package/app/.next/server/chunks/412.js +2 -2
  424. package/app/.next/server/chunks/4177.js +1 -0
  425. package/app/.next/server/chunks/4989.js +2 -2
  426. package/app/.next/server/chunks/5681.js +1 -1
  427. package/app/.next/server/chunks/5787.js +3 -3
  428. package/app/.next/server/chunks/{4084.js → 622.js} +2 -2
  429. package/app/.next/server/chunks/6327.js +1 -0
  430. package/app/.next/server/chunks/6555.js +1 -1
  431. package/app/.next/server/chunks/7595.js +1 -1
  432. package/app/.next/server/chunks/7994.js +2 -2
  433. package/app/.next/server/chunks/8590.js +1 -1
  434. package/app/.next/server/chunks/8760.js +1 -1
  435. package/app/.next/server/chunks/8895.js +8 -2
  436. package/app/.next/server/chunks/9489.js +1 -1
  437. package/app/.next/server/chunks/9609.js +1 -1
  438. package/app/.next/server/chunks/9718.js +1 -1
  439. package/app/.next/server/middleware-build-manifest.js +1 -1
  440. package/app/.next/server/middleware.js +1 -1
  441. package/app/.next/server/pages/404.html +1 -1
  442. package/app/.next/server/pages/500.html +1 -1
  443. package/app/.next/server/server-reference-manifest.js +1 -1
  444. package/app/.next/server/server-reference-manifest.json +1 -1
  445. package/app/.next/static/chunks/1321-b2081c642c72002a.js +1 -0
  446. package/app/.next/static/chunks/2589-5e8a0fa7cb3f227d.js +3 -0
  447. package/app/.next/static/chunks/5497-28242e3ae0b31a36.js +7 -0
  448. package/app/.next/static/chunks/6309-ff36cf5344142814.js +23 -0
  449. package/app/.next/static/chunks/app/(dashboard)/dashboard/cli-tools/page-28383c480ec5b127.js +1 -0
  450. package/app/.next/static/chunks/app/(dashboard)/dashboard/combos/page-85548757b84b3172.js +1 -0
  451. package/app/.next/static/chunks/app/(dashboard)/dashboard/media-providers/[kind]/[id]/{page-88171c9a70b02d07.js → page-b2837760558bde3d.js} +8 -8
  452. package/app/.next/static/chunks/app/(dashboard)/dashboard/mitm/page-f55f0e573440f5e8.js +1 -0
  453. package/app/.next/static/chunks/app/(dashboard)/dashboard/profile/page-d22744b1e72de612.js +1 -0
  454. package/app/.next/static/chunks/app/(dashboard)/dashboard/providers/[id]/page-6925a4857ab01924.js +4 -0
  455. package/app/.next/static/chunks/app/(dashboard)/dashboard/providers/page-8dad17b86423b20a.js +1 -0
  456. package/app/.next/static/chunks/app/(dashboard)/dashboard/proxy-pools/page-eb76b6bd9948a29e.js +2 -0
  457. package/app/.next/static/chunks/app/(dashboard)/dashboard/quota/page-efaa6dd8c7eb8a65.js +1 -0
  458. package/app/.next/static/chunks/app/(dashboard)/dashboard/usage/page-0baabdbcfbf34c07.js +1 -0
  459. package/app/.next/static/css/bcf0d6683b3c65df.css +1 -0
  460. package/app/open-sse/config/providerModels.js +7 -1
  461. package/app/open-sse/config/providers.js +4 -0
  462. package/app/open-sse/executors/antigravity.js +11 -18
  463. package/app/open-sse/handlers/chatCore.js +19 -2
  464. package/app/open-sse/rtk/caveman.js +100 -0
  465. package/app/open-sse/rtk/cavemanPrompts.js +35 -0
  466. package/app/open-sse/services/combo.js +42 -16
  467. package/app/open-sse/services/model.js +2 -0
  468. package/app/open-sse/translator/helpers/claudeHelper.js +8 -0
  469. package/app/open-sse/translator/helpers/openaiHelper.js +9 -2
  470. package/app/open-sse/translator/index.js +1 -9
  471. package/app/open-sse/translator/request/claude-to-openai.js +11 -9
  472. package/app/open-sse/utils/proxyFetch.js +2 -4
  473. package/app/open-sse/utils/streamHelpers.js +2 -2
  474. package/app/package.json +1 -1
  475. package/app/public/providers/xiaomi-mimo.png +0 -0
  476. package/app/src/mitm/handlers/antigravity.js +2 -2
  477. package/app/src/mitm/server.js +11 -11
  478. package/app/src/mitm/usageTracker.js +14 -1
  479. package/app/src/shared/constants/providers.js +35 -34
  480. package/package.json +1 -1
  481. package/app/.next/server/chunks/2549.js +0 -1
  482. package/app/.next/server/chunks/7828.js +0 -1
  483. package/app/.next/server/chunks/8491.js +0 -1
  484. package/app/.next/static/chunks/1321-758994bc249e4966.js +0 -1
  485. package/app/.next/static/chunks/2589-a1b46e66317dffd7.js +0 -3
  486. package/app/.next/static/chunks/5497-dc001333990a942c.js +0 -7
  487. package/app/.next/static/chunks/6309-a7e57611cfcbe5fa.js +0 -23
  488. package/app/.next/static/chunks/app/(dashboard)/dashboard/cli-tools/page-f033f4a266ecfa10.js +0 -1
  489. package/app/.next/static/chunks/app/(dashboard)/dashboard/combos/page-ba5ac6c8e331e7d3.js +0 -1
  490. package/app/.next/static/chunks/app/(dashboard)/dashboard/mitm/page-e8c6b296929523ee.js +0 -1
  491. package/app/.next/static/chunks/app/(dashboard)/dashboard/profile/page-cf73f6db045a7e2d.js +0 -1
  492. package/app/.next/static/chunks/app/(dashboard)/dashboard/providers/[id]/page-1c46f0fe4f131e65.js +0 -4
  493. package/app/.next/static/chunks/app/(dashboard)/dashboard/providers/page-92d99df7ce85fd58.js +0 -1
  494. package/app/.next/static/chunks/app/(dashboard)/dashboard/proxy-pools/page-86caff27d92376bc.js +0 -2
  495. package/app/.next/static/chunks/app/(dashboard)/dashboard/quota/page-751c8e04880cb0d7.js +0 -1
  496. package/app/.next/static/chunks/app/(dashboard)/dashboard/usage/page-a819f6afc52ee43e.js +0 -1
  497. package/app/.next/static/css/34a45bfcb212973d.css +0 -1
  498. /package/app/.next/static/{wozf5GL2edkjzl9Gac_lV → PSWjQzItQS96mOJ2LNJHi}/_buildManifest.js +0 -0
  499. /package/app/.next/static/{wozf5GL2edkjzl9Gac_lV → PSWjQzItQS96mOJ2LNJHi}/_ssgManifest.js +0 -0
@@ -76,6 +76,8 @@ export function fixToolUseOrdering(messages) {
76
76
  return merged;
77
77
  }
78
78
 
79
+ const CLAUDE_FORMAT_PROVIDERS_WITHOUT_OUTPUT_CONFIG = new Set(["minimax", "minimax-cn"]);
80
+
79
81
  // Prepare request for Claude format endpoints
80
82
  // - Cleanup cache_control
81
83
  // - Filter empty messages
@@ -83,6 +85,12 @@ export function fixToolUseOrdering(messages) {
83
85
  // - Fix tool_use/tool_result ordering
84
86
  // - Apply cloaking (billing header + fake user ID) for OAuth tokens
85
87
  export function prepareClaudeRequest(body, provider = null, apiKey = null, connectionId = null) {
88
+ // MiniMax exposes a Claude-compatible endpoint but rejects Anthropic's extended
89
+ // structured output parameter with a generic 400 "invalid params" response.
90
+ if (CLAUDE_FORMAT_PROVIDERS_WITHOUT_OUTPUT_CONFIG.has(provider)) {
91
+ delete body.output_config;
92
+ }
93
+
86
94
  // 1. System: remove all cache_control, add only to last block with ttl 1h
87
95
  if (body.system && Array.isArray(body.system)) {
88
96
  body.system = body.system.map((block, i) => {
@@ -4,6 +4,14 @@
4
4
  export const VALID_OPENAI_CONTENT_TYPES = ["text", "image_url", "image"];
5
5
  export const VALID_OPENAI_MESSAGE_TYPES = ["text", "image_url", "image", "tool_calls", "tool_result"];
6
6
 
7
+ function normalizeOpenAIContent(content) {
8
+ if (content.length === 0) return "";
9
+ if (content.every((block) => block.type === "text")) {
10
+ return content.map((block) => block.text || "").join("\n");
11
+ }
12
+ return content;
13
+ }
14
+
7
15
  // Filter messages to OpenAI standard format
8
16
  // Remove: thinking, redacted_thinking, signature, and other non-OpenAI blocks
9
17
  export function filterToOpenAIFormat(body) {
@@ -47,7 +55,7 @@ export function filterToOpenAIFormat(body) {
47
55
  filteredContent.push({ type: "text", text: "" });
48
56
  }
49
57
 
50
- return { ...msg, content: filteredContent };
58
+ return { ...msg, content: normalizeOpenAIContent(filteredContent) };
51
59
  }
52
60
 
53
61
  return msg;
@@ -124,4 +132,3 @@ export function filterToOpenAIFormat(body) {
124
132
 
125
133
  return body;
126
134
  }
127
-
@@ -5,7 +5,6 @@ import { cloakClaudeTools } from "../utils/claudeCloaking.js";
5
5
  import { filterToOpenAIFormat } from "./helpers/openaiHelper.js";
6
6
  import { normalizeThinkingConfig } from "../services/provider.js";
7
7
  import { AntigravityExecutor } from "../executors/antigravity.js";
8
- import { compressMessages, formatRtkLog } from "../rtk/index.js";
9
8
 
10
9
  // Registry for translators
11
10
  const requestRegistry = new Map();
@@ -71,17 +70,10 @@ function stripContentTypes(body, stripList = []) {
71
70
  }
72
71
 
73
72
  // Translate request: source -> openai -> target
74
- export function translateRequest(sourceFormat, targetFormat, model, body, stream = true, credentials = null, provider = null, reqLogger = null, stripList = [], connectionId = null, rtkEnabled = false, clientTool = null) {
73
+ export function translateRequest(sourceFormat, targetFormat, model, body, stream = true, credentials = null, provider = null, reqLogger = null, stripList = [], connectionId = null, clientTool = null) {
75
74
  ensureInitialized();
76
75
  let result = body;
77
76
 
78
- // RTK: compress tool_result content before any translation (shape-agnostic)
79
- const rtkStats = compressMessages(result, rtkEnabled);
80
- if (rtkStats) {
81
- const line = formatRtkLog(rtkStats);
82
- if (line) console.log(line);
83
- }
84
-
85
77
  // Strip explicit content types (opt-in via strip[] in PROVIDER_MODELS entry)
86
78
  stripContentTypes(result, stripList);
87
79
 
@@ -109,6 +109,14 @@ function fixMissingToolResponses(messages) {
109
109
  }
110
110
  }
111
111
 
112
+ function openAIContentFromParts(parts) {
113
+ if (parts.length === 0) return "";
114
+ if (parts.every((part) => part.type === "text")) {
115
+ return parts.map((part) => part.text || "").join("\n");
116
+ }
117
+ return parts;
118
+ }
119
+
112
120
  // Convert single Claude message - returns single message or array of messages
113
121
  function convertClaudeMessage(msg) {
114
122
  const role = msg.role === "user" || msg.role === "tool" ? "user" : "assistant";
@@ -177,10 +185,7 @@ function convertClaudeMessage(msg) {
177
185
  // If has tool results, return array of tool messages
178
186
  if (toolResults.length > 0) {
179
187
  if (parts.length > 0) {
180
- const textContent = parts.length === 1 && parts[0].type === "text"
181
- ? parts[0].text
182
- : parts;
183
- return [...toolResults, { role: "user", content: textContent }];
188
+ return [...toolResults, { role: "user", content: openAIContentFromParts(parts) }];
184
189
  }
185
190
  return toolResults;
186
191
  }
@@ -189,9 +194,7 @@ function convertClaudeMessage(msg) {
189
194
  if (toolCalls.length > 0) {
190
195
  const result = { role: "assistant" };
191
196
  if (parts.length > 0) {
192
- result.content = parts.length === 1 && parts[0].type === "text"
193
- ? parts[0].text
194
- : parts;
197
+ result.content = openAIContentFromParts(parts);
195
198
  }
196
199
  result.tool_calls = toolCalls;
197
200
  return result;
@@ -201,7 +204,7 @@ function convertClaudeMessage(msg) {
201
204
  if (parts.length > 0) {
202
205
  return {
203
206
  role,
204
- content: parts.length === 1 && parts[0].type === "text" ? parts[0].text : parts
207
+ content: openAIContentFromParts(parts)
205
208
  };
206
209
  }
207
210
 
@@ -229,4 +232,3 @@ function convertToolChoice(choice) {
229
232
 
230
233
  // Register
231
234
  register(FORMATS.CLAUDE, FORMATS.OPENAI, claudeToOpenAIRequest, null);
232
-
@@ -1,8 +1,6 @@
1
1
  import { Readable } from "stream";
2
2
  import { MEMORY_CONFIG } from "../config/runtimeConfig.js";
3
3
 
4
- const isCloud = typeof caches !== "undefined" && typeof caches === "object";
5
-
6
4
  const originalFetch = globalThis.fetch;
7
5
  const proxyDispatchers = new Map();
8
6
 
@@ -263,8 +261,8 @@ async function patchedFetch(url, options = {}) {
263
261
  }
264
262
 
265
263
  // Idempotency guard — only patch once to avoid wrapping multiple times
266
- if (!isCloud && globalThis.fetch !== patchedFetch) {
264
+ if (globalThis.fetch !== patchedFetch) {
267
265
  globalThis.fetch = patchedFetch;
268
266
  }
269
267
 
270
- export default isCloud ? originalFetch : patchedFetch;
268
+ export default patchedFetch;
@@ -5,8 +5,8 @@ export function parseSSELine(line, format = null) {
5
5
  if (!line) return null;
6
6
 
7
7
  // NDJSON format (Ollama): raw JSON lines without "data:" prefix
8
- if (format === FORMATS.OLLAMA) {
9
- const trimmed = line.trim();
8
+ const trimmed = line.trim();
9
+ if (format === FORMATS.OLLAMA || trimmed.startsWith("{")) {
10
10
  if (trimmed.startsWith("{")) {
11
11
  try {
12
12
  return JSON.parse(trimmed);
package/app/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "n9router",
3
- "version": "0.4.21",
3
+ "version": "0.4.24",
4
4
  "description": "Self-hosted AI routing gateway — local proxy for Claude, Gemini, OpenAI and 40+ providers",
5
5
  "keywords": [
6
6
  "ai",
@@ -11,10 +11,10 @@ async function intercept(req, res, bodyBuffer, mappedModel) {
11
11
  const isStream = req.url.includes(":streamGenerateContent");
12
12
  try {
13
13
  const body = JSON.parse(bodyBuffer.toString());
14
- if (body.model) body.model = mappedModel;
14
+ body.model = mappedModel;
15
15
 
16
16
  const routerRes = await fetchRouter(body, "/v1/chat/completions", req.headers);
17
- await pipeSSE(routerRes, res, dumper);
17
+ await pipeSSE(routerRes, res);
18
18
  } catch (error) {
19
19
  err(`[antigravity] ${error.message}`);
20
20
  if (dumper) { dumper.writeChunk(`\n[ERROR] ${error.message}\n`); dumper.end(); }
@@ -82,7 +82,6 @@ function sniCallback(servername, cb) {
82
82
  cert: `${certData.cert}\n${rootCAPem}`
83
83
  });
84
84
  certCache.set(servername, ctx);
85
- log(`🔐 Cert generated: ${servername}`);
86
85
  cb(null, ctx);
87
86
  } catch (e) {
88
87
  err(`SNI error for ${servername}: ${e.message}`);
@@ -204,7 +203,7 @@ async function passthrough(req, res, bodyBuffer, onResponse, debugContext = null
204
203
 
205
204
  // ── Token swap forward ────────────────────────────────────────
206
205
  // Unlike passthrough(), this checks upstream statusCode BEFORE
207
- // piping to client — enabling auto-retry on 429/503.
206
+ // piping to client — enabling auto-retry on 403/429/503.
208
207
 
209
208
  async function tokenSwapForward(req, res, bodyBuffer, connections, model, strategy, provider, requestStartTime, debugContext = null) {
210
209
  let targetHost = (req.headers.host || TARGET_HOSTS[0]).split(":")[0];
@@ -267,12 +266,16 @@ async function tokenSwapForward(req, res, bodyBuffer, connections, model, strate
267
266
  servername: targetHost,
268
267
  rejectUnauthorized: false
269
268
  }, (forwardRes) => {
270
- if (forwardRes.statusCode === 429 || forwardRes.statusCode === 503 || forwardRes.statusCode === 401) {
269
+ if (forwardRes.statusCode === 403 || forwardRes.statusCode === 429 || forwardRes.statusCode === 503 || forwardRes.statusCode === 401) {
271
270
  const chunks = [];
272
271
  forwardRes.on("data", c => chunks.push(c));
273
272
  forwardRes.on("end", () => {
274
273
  const body = Buffer.concat(chunks).toString();
275
- const retryType = forwardRes.statusCode === 401 ? "auth" : "quota";
274
+ const retryType = forwardRes.statusCode === 401
275
+ ? "auth"
276
+ : forwardRes.statusCode === 403
277
+ ? "permission"
278
+ : "quota";
276
279
  resolve({ retry: true, retryType, body, headers: forwardRes.headers, statusCode: forwardRes.statusCode });
277
280
  });
278
281
  } else {
@@ -284,8 +287,8 @@ async function tokenSwapForward(req, res, bodyBuffer, connections, model, strate
284
287
  forwardReq.end();
285
288
  });
286
289
 
287
- if (result.retry && result.retryType === "quota") {
288
- // ── Unified quota retry: 429 and 503 treated identically ──
290
+ if (result.retry && (result.retryType === "quota" || result.retryType === "permission")) {
291
+ // ── Unified retry: 403 IAM denial, 429 and 503 treated identically ──
289
292
  // Both are false-positive-prone from Antigravity; retry same account
290
293
  // with exponential backoff before moving on.
291
294
  const maxRetries = getAntigravity503RetryCount(conn.antigravity503RetryCount);
@@ -310,7 +313,7 @@ async function tokenSwapForward(req, res, bodyBuffer, connections, model, strate
310
313
 
311
314
  // ── 2-consecutive-fail rule: only apply cooldown/strike if this
312
315
  // account's last health event was already a fail. A single transient
313
- // 429/503 burst doesn't warrant a cooldown. ──
316
+ // 403/429/503 burst doesn't warrant a cooldown. ──
314
317
  const accountKey = conn.email || conn.id;
315
318
  const lastStatus = getLastEventStatus(accountKey);
316
319
  const isConsecutiveFail = lastStatus === "fl";
@@ -432,7 +435,7 @@ async function tokenSwapForward(req, res, bodyBuffer, connections, model, strate
432
435
  clearStrikes(conn.id);
433
436
  if (model) clearModelStrikes(conn.id, model);
434
437
  markAccountUsed(conn.id);
435
- // Record health: success on first try vs retry success (429+503 quota retries both count)
438
+ // Record health: success on first try vs retry success (403/429/503 retries count)
436
439
  const _healthAttempts = (conn._quotaRetryCount || 0) + 1;
437
440
  reportHealth(conn.email || conn.id, _healthAttempts > 1 ? "retry_success" : (i > 0 ? "retry_success" : "success"), _healthAttempts, model);
438
441
  res.writeHead(statusCode, result.response.headers);
@@ -665,12 +668,9 @@ const server = https.createServer(sslOptions, async (req, res) => {
665
668
  }
666
669
  }
667
670
 
668
- log(`🔍 [${tool}] url=${req.url} | bodyLen=${bodyBuffer.length}`);
669
-
670
671
  // Cursor uses binary proto — model extraction not possible at this layer.
671
672
  // Delegate directly to handler which decodes proto internally.
672
673
  if (tool === "cursor") {
673
- log(`⚡ intercept | cursor | proto`);
674
674
  return handlers[tool].intercept(req, res, bodyBuffer, null, passthrough);
675
675
  }
676
676
 
@@ -105,6 +105,11 @@ function extractUsage(chunk) {
105
105
  return null;
106
106
  }
107
107
 
108
+ function extractModelFromResponseChunk(chunk) {
109
+ if (!chunk || typeof chunk !== "object") return null;
110
+ return chunk.model || chunk.modelVersion || chunk.response?.model || chunk.response?.modelVersion || null;
111
+ }
112
+
108
113
  function getPartsTextLength(parts) {
109
114
  if (!Array.isArray(parts)) return 0;
110
115
  return parts.reduce((total, part) => total + (typeof part?.text === "string" ? part.text.length : 0), 0);
@@ -468,6 +473,7 @@ function createTokenSwapUsageObserver({ provider, model, connectionId, accountLa
468
473
  let buffer = "";
469
474
  let contentLength = 0;
470
475
  let usage = null;
476
+ let responseModel = null;
471
477
  let finished = false;
472
478
  let responseContent = "";
473
479
  let thinkingContent = "";
@@ -477,6 +483,7 @@ function createTokenSwapUsageObserver({ provider, model, connectionId, accountLa
477
483
  const processParsedChunk = (parsed) => {
478
484
  const extracted = extractUsage(parsed);
479
485
  if (extracted) usage = extracted;
486
+ responseModel = responseModel || extractModelFromResponseChunk(parsed);
480
487
  contentLength += extractContentLength(parsed);
481
488
  if (!contentCapped) {
482
489
  const text = extractContentText(parsed);
@@ -552,6 +559,11 @@ function createTokenSwapUsageObserver({ provider, model, connectionId, accountLa
552
559
 
553
560
  if (!hasValidUsage(usage)) return;
554
561
 
562
+ const trackedModel = model
563
+ || responseModel
564
+ || detailRecord?.model
565
+ || "unknown";
566
+
555
567
  const inTokens = usage.prompt_tokens || usage.input_tokens || 0;
556
568
  const outTokens = usage.completion_tokens || usage.output_tokens || 0;
557
569
  log(`📊 [token-swap] usage "${accountLabel}" in=${inTokens} out=${outTokens}${usage.estimated ? " estimated" : ""}`);
@@ -559,7 +571,7 @@ function createTokenSwapUsageObserver({ provider, model, connectionId, accountLa
559
571
  try {
560
572
  await persistUsage({
561
573
  provider,
562
- model,
574
+ model: trackedModel,
563
575
  connectionId,
564
576
  tokens: usage,
565
577
  status: `${statusCode} OK`
@@ -573,6 +585,7 @@ function createTokenSwapUsageObserver({ provider, model, connectionId, accountLa
573
585
  try {
574
586
  await persistRequestDetail({
575
587
  ...detailRecord,
588
+ model: trackedModel,
576
589
  status: "completed",
577
590
  tokens: usage,
578
591
  latency: {
@@ -18,7 +18,7 @@ export const FREE_TIER_PROVIDERS = {
18
18
  nvidia: { id: "nvidia", alias: "nvidia", name: "NVIDIA NIM", icon: "developer_board", color: "#76B900", textIcon: "NV", website: "https://developer.nvidia.com/nim", notice: { text: "Free access for NVIDIA Developer Program members (prototyping & testing).", apiKeyUrl: "https://build.nvidia.com/settings/api-keys" }, serviceKinds: ["llm", "tts", "embedding", "stt"], ttsConfig: { baseUrl: "https://integrate.api.nvidia.com/v1/audio/speech", authType: "apikey", authHeader: "bearer", format: "nvidia-tts", models: [{ id: "fastpitch", name: "FastPitch" }, { id: "tacotron2", name: "Tacotron2" }] }, embeddingConfig: { baseUrl: "https://integrate.api.nvidia.com/v1/embeddings", authType: "apikey", authHeader: "bearer", models: [{ id: "nvidia/nv-embedqa-e5-v5", name: "NV EmbedQA E5 v5", dimensions: 1024 }] } },
19
19
  ollama: { id: "ollama", alias: "ollama", name: "Ollama Cloud", icon: "cloud", color: "#ffffffff", textIcon: "OL", website: "https://ollama.com", notice: { text: "Free tier: light usage, 1 cloud model at a time (limits reset every 5h & 7d). Pro $20/mo · Max $100/mo.", apiKeyUrl: "https://ollama.com/settings/keys" } },
20
20
  vertex: { id: "vertex", alias: "vx", name: "Vertex AI", icon: "cloud", color: "#4285F4", textIcon: "VX", website: "https://cloud.google.com/vertex-ai", notice: { text: "New Google Cloud accounts get $300 free credits. Requires GCP project + Service Account with Vertex AI API enabled.", apiKeyUrl: "https://console.cloud.google.com/iam-admin/serviceaccounts" } },
21
- gemini: { id: "gemini", alias: "gemini", name: "Gemini", icon: "diamond", color: "#4285F4", textIcon: "GE", website: "https://ai.google.dev", serviceKinds: ["llm", "embedding", "image", "imageToText", "webSearch"], searchViaChat: { defaultModel: "gemini-2.5-flash", pricingUrl: "https://ai.google.dev/pricing", freeTier: "Free tier: 15 RPM, 1M tokens/day on gemini-2.5-flash via AI Studio." }, embeddingConfig: { baseUrl: "https://generativelanguage.googleapis.com/v1beta/models", authType: "apikey", authHeader: "key", models: [{ id: "text-embedding-004", name: "Text Embedding 004", dimensions: 768 }, { id: "embedding-001", name: "Embedding 001", dimensions: 768 }] } },
21
+ gemini: { id: "gemini", alias: "gemini", name: "Gemini", icon: "diamond", color: "#4285F4", textIcon: "GE", website: "https://ai.google.dev", notice: { apiKeyUrl: "https://aistudio.google.com/app/apikey" }, serviceKinds: ["llm", "embedding", "image", "imageToText", "webSearch"], searchViaChat: { defaultModel: "gemini-2.5-flash", pricingUrl: "https://ai.google.dev/pricing", freeTier: "Free tier: 15 RPM, 1M tokens/day on gemini-2.5-flash via AI Studio." }, embeddingConfig: { baseUrl: "https://generativelanguage.googleapis.com/v1beta/models", authType: "apikey", authHeader: "key", models: [{ id: "text-embedding-004", name: "Text Embedding 004", dimensions: 768 }, { id: "embedding-001", name: "Embedding 001", dimensions: 768 }] } },
22
22
  byteplus: { id: "byteplus", alias: "bpm", name: "BytePlus ModelArk", icon: "cloud", color: "#2563EB", textIcon: "BP", website: "https://console.byteplus.com/ark", notice: { text: "Free credits for new accounts. Access to Seed 2.0, Kimi K2 Thinking, GLM 4.7, GPT-OSS-120B models.", apiKeyUrl: "https://console.byteplus.com/ark/region:ark+ap-southeast-1/apiKey" }, serviceKinds: ["llm"] },
23
23
  };
24
24
 
@@ -53,37 +53,38 @@ export const OAUTH_PROVIDERS = {
53
53
  };
54
54
 
55
55
  export const APIKEY_PROVIDERS = {
56
- glm: { id: "glm", alias: "glm", name: "GLM Coding", icon: "code", color: "#2563EB", textIcon: "GL", website: "https://open.bigmodel.cn" },
57
- "glm-cn": { id: "glm-cn", alias: "glm-cn", name: "GLM (China)", icon: "code", color: "#DC2626", textIcon: "GC", website: "https://open.bigmodel.cn" },
58
- kimi: { id: "kimi", alias: "kimi", name: "Kimi", icon: "psychology", color: "#1E3A8A", textIcon: "KM", website: "https://kimi.moonshot.cn", serviceKinds: ["llm", "webSearch"], searchViaChat: { defaultModel: "kimi-k2.5", pricingUrl: "https://platform.moonshot.ai/docs/pricing/chat" } },
59
- minimax: { id: "minimax", alias: "minimax", name: "Minimax Coding", icon: "memory", color: "#7C3AED", textIcon: "MM", website: "https://www.minimaxi.com", serviceKinds: ["llm", "image", "imageToText", "webSearch"], searchViaChat: { defaultModel: "MiniMax-M2.7", pricingUrl: "https://www.minimaxi.com/document/price" } },
60
- "minimax-cn": { id: "minimax-cn", alias: "minimax-cn", name: "Minimax (China)", icon: "memory", color: "#DC2626", textIcon: "MC", website: "https://www.minimaxi.com" },
61
- alicode: { id: "alicode", alias: "alicode", name: "Alibaba", icon: "cloud", color: "#FF6A00", textIcon: "ALi" },
62
- "alicode-intl": { id: "alicode-intl", alias: "alicode-intl", name: "Alibaba Intl", icon: "cloud", color: "#FF6A00", textIcon: "ALi" },
63
- "volcengine-ark": { id: "volcengine-ark", alias: "ark", name: "Volcengine Ark", icon: "cloud", color: "#1677FF", textIcon: "ARK", website: "https://ark.cn-beijing.volces.com" },
64
- openai: { id: "openai", alias: "openai", name: "OpenAI", icon: "auto_awesome", color: "#10A37F", textIcon: "OA", website: "https://platform.openai.com", serviceKinds: ["llm", "embedding", "tts", "image", "imageToText", "webSearch"], thinkingConfig: THINKING_CONFIG.effort, searchViaChat: { defaultModel: "gpt-4o-mini", pricingUrl: "https://openai.com/api/pricing" }, ttsConfig: { baseUrl: "https://api.openai.com/v1/audio/speech", authType: "apikey", authHeader: "bearer", format: "openai", models: [{ id: "tts-1", name: "TTS-1" }, { id: "tts-1-hd", name: "TTS-1 HD" }, { id: "gpt-4o-mini-tts", name: "GPT-4o Mini TTS" }] }, embeddingConfig: { baseUrl: "https://api.openai.com/v1/embeddings", authType: "apikey", authHeader: "bearer", models: [{ id: "text-embedding-3-small", name: "Text Embedding 3 Small", dimensions: 1536 }, { id: "text-embedding-3-large", name: "Text Embedding 3 Large", dimensions: 3072 }, { id: "text-embedding-ada-002", name: "Text Embedding Ada 002", dimensions: 1536 }] } },
65
- anthropic: { id: "anthropic", alias: "anthropic", name: "Anthropic", icon: "smart_toy", color: "#D97757", textIcon: "AN", website: "https://console.anthropic.com", serviceKinds: ["llm", "imageToText"] },
56
+ glm: { id: "glm", alias: "glm", name: "GLM Coding", icon: "code", color: "#2563EB", textIcon: "GL", website: "https://open.bigmodel.cn", notice: { apiKeyUrl: "https://open.bigmodel.cn/usercenter/apikeys" } },
57
+ "glm-cn": { id: "glm-cn", alias: "glm-cn", name: "GLM (China)", icon: "code", color: "#DC2626", textIcon: "GC", website: "https://open.bigmodel.cn", notice: { apiKeyUrl: "https://open.bigmodel.cn/usercenter/apikeys" } },
58
+ kimi: { id: "kimi", alias: "kimi", name: "Kimi", icon: "psychology", color: "#1E3A8A", textIcon: "KM", website: "https://kimi.moonshot.cn", notice: { apiKeyUrl: "https://platform.moonshot.ai/console/api-keys" }, serviceKinds: ["llm", "webSearch"], searchViaChat: { defaultModel: "kimi-k2.5", pricingUrl: "https://platform.moonshot.ai/docs/pricing/chat" } },
59
+ minimax: { id: "minimax", alias: "minimax", name: "Minimax Coding", icon: "memory", color: "#7C3AED", textIcon: "MM", website: "https://www.minimaxi.com", notice: { apiKeyUrl: "https://platform.minimaxi.com/user-center/basic-information/interface-key" }, serviceKinds: ["llm", "image", "imageToText", "webSearch"], searchViaChat: { defaultModel: "MiniMax-M2.7", pricingUrl: "https://www.minimaxi.com/document/price" } },
60
+ "minimax-cn": { id: "minimax-cn", alias: "minimax-cn", name: "Minimax (China)", icon: "memory", color: "#DC2626", textIcon: "MC", website: "https://www.minimaxi.com", notice: { apiKeyUrl: "https://platform.minimaxi.com/user-center/basic-information/interface-key" } },
61
+ alicode: { id: "alicode", alias: "alicode", name: "Alibaba", icon: "cloud", color: "#FF6A00", textIcon: "ALi", website: "https://bailian.console.aliyun.com", notice: { apiKeyUrl: "https://bailian.console.aliyun.com/?apiKey=1" } },
62
+ "alicode-intl": { id: "alicode-intl", alias: "alicode-intl", name: "Alibaba Intl", icon: "cloud", color: "#FF6A00", textIcon: "ALi", website: "https://modelstudio.console.alibabacloud.com", notice: { apiKeyUrl: "https://modelstudio.console.alibabacloud.com/?apiKey=1" } },
63
+ "xiaomi-mimo": { id: "xiaomi-mimo", alias: "mimo", name: "Xiaomi MiMo", icon: "smart_toy", color: "#FF6900", textIcon: "XM", website: "https://xiaomimimo.com", notice: { apiKeyUrl: "https://xiaomimimo.com" } },
64
+ "volcengine-ark": { id: "volcengine-ark", alias: "ark", name: "Volcengine Ark", icon: "cloud", color: "#1677FF", textIcon: "ARK", website: "https://ark.cn-beijing.volces.com", notice: { apiKeyUrl: "https://console.volcengine.com/ark/region:ark+cn-beijing/apiKey" } },
65
+ openai: { id: "openai", alias: "openai", name: "OpenAI", icon: "auto_awesome", color: "#10A37F", textIcon: "OA", website: "https://platform.openai.com", notice: { apiKeyUrl: "https://platform.openai.com/api-keys" }, serviceKinds: ["llm", "embedding", "tts", "image", "imageToText", "webSearch"], thinkingConfig: THINKING_CONFIG.effort, searchViaChat: { defaultModel: "gpt-4o-mini", pricingUrl: "https://openai.com/api/pricing" }, ttsConfig: { baseUrl: "https://api.openai.com/v1/audio/speech", authType: "apikey", authHeader: "bearer", format: "openai", models: [{ id: "tts-1", name: "TTS-1" }, { id: "tts-1-hd", name: "TTS-1 HD" }, { id: "gpt-4o-mini-tts", name: "GPT-4o Mini TTS" }] }, embeddingConfig: { baseUrl: "https://api.openai.com/v1/embeddings", authType: "apikey", authHeader: "bearer", models: [{ id: "text-embedding-3-small", name: "Text Embedding 3 Small", dimensions: 1536 }, { id: "text-embedding-3-large", name: "Text Embedding 3 Large", dimensions: 3072 }, { id: "text-embedding-ada-002", name: "Text Embedding Ada 002", dimensions: 1536 }] } },
66
+ anthropic: { id: "anthropic", alias: "anthropic", name: "Anthropic", icon: "smart_toy", color: "#D97757", textIcon: "AN", website: "https://console.anthropic.com", notice: { apiKeyUrl: "https://console.anthropic.com/settings/keys" }, serviceKinds: ["llm", "imageToText"] },
66
67
  "opencode-go": { id: "opencode-go", alias: "ocg", name: "OpenCode Go", icon: "terminal", color: "#E87040", textIcon: "OC", website: "https://opencode.ai/auth", notice: { text: "OpenCode Go subscription: $5/mo (then $10/mo). Access to Kimi, GLM, Qwen, MiMo, MiniMax models.", apiKeyUrl: "https://opencode.ai/auth" } },
67
- azure: { id: "azure", alias: "azure", name: "Azure OpenAI", icon: "cloud", color: "#0078D4", textIcon: "AZ", website: "https://azure.microsoft.com/en-us/products/ai-services/openai-service", hasProviderSpecificData: true },
68
-
69
- deepseek: { id: "deepseek", alias: "ds", name: "DeepSeek", icon: "bolt", color: "#4D6BFE", textIcon: "DS", website: "https://deepseek.com" },
70
- groq: { id: "groq", alias: "groq", name: "Groq", icon: "speed", color: "#F55036", textIcon: "GQ", website: "https://groq.com", serviceKinds: ["llm", "imageToText"] },
71
- xai: { id: "xai", alias: "xai", name: "xAI (Grok)", icon: "auto_awesome", color: "#1DA1F2", textIcon: "XA", website: "https://x.ai", serviceKinds: ["llm", "imageToText", "webSearch"], searchViaChat: { defaultModel: "grok-4.20-reasoning", pricingUrl: "https://x.ai/api#pricing" } },
72
- mistral: { id: "mistral", alias: "mistral", name: "Mistral", icon: "air", color: "#FF7000", textIcon: "MI", website: "https://mistral.ai", serviceKinds: ["llm", "imageToText", "embedding"], embeddingConfig: { baseUrl: "https://api.mistral.ai/v1/embeddings", authType: "apikey", authHeader: "bearer", models: [{ id: "mistral-embed", name: "Mistral Embed", dimensions: 1024 }] } },
73
- perplexity: { id: "perplexity", alias: "pplx", name: "Perplexity", icon: "search", color: "#20808D", textIcon: "PP", website: "https://www.perplexity.ai", serviceKinds: ["llm", "webSearch"], searchConfig: { baseUrl: "https://api.perplexity.ai/search", method: "POST", authType: "apikey", authHeader: "bearer", costPerQuery: 0.005, freeMonthlyQuota: 0, searchTypes: ["web"], defaultMaxResults: 5, maxMaxResults: 20, timeoutMs: 10000, cacheTTLMs: 300000 } },
74
- together: { id: "together", alias: "together", name: "Together AI", icon: "group_work", color: "#0F6FFF", textIcon: "TG", website: "https://www.together.ai", serviceKinds: ["llm", "embedding"], embeddingConfig: { baseUrl: "https://api.together.xyz/v1/embeddings", authType: "apikey", authHeader: "bearer", models: [{ id: "BAAI/bge-large-en-v1.5", name: "BGE Large EN v1.5", dimensions: 1024 }, { id: "togethercomputer/m2-bert-80M-8k-retrieval", name: "M2 BERT 80M 8K", dimensions: 768 }] } },
75
- fireworks: { id: "fireworks", alias: "fireworks", name: "Fireworks AI", icon: "local_fire_department", color: "#7B2EF2", textIcon: "FW", website: "https://fireworks.ai", serviceKinds: ["llm", "embedding"], embeddingConfig: { baseUrl: "https://api.fireworks.ai/inference/v1/embeddings", authType: "apikey", authHeader: "bearer", models: [{ id: "nomic-ai/nomic-embed-text-v1.5", name: "Nomic Embed Text v1.5", dimensions: 768 }] } },
76
- cerebras: { id: "cerebras", alias: "cerebras", name: "Cerebras", icon: "memory", color: "#FF4F00", textIcon: "CB", website: "https://www.cerebras.ai" },
77
- cohere: { id: "cohere", alias: "cohere", name: "Cohere", icon: "hub", color: "#39594D", textIcon: "CO", website: "https://cohere.com" },
78
- nebius: { id: "nebius", alias: "nebius", name: "Nebius AI", icon: "cloud", color: "#6C5CE7", textIcon: "NB", website: "https://nebius.com", serviceKinds: ["llm", "embedding"], embeddingConfig: { baseUrl: "https://api.tokenfactory.nebius.com/v1/embeddings", authType: "apikey", authHeader: "bearer", models: [{ id: "Qwen/Qwen3-Embedding-8B", name: "Qwen3 Embedding 8B", dimensions: 4096 }] } },
79
- siliconflow: { id: "siliconflow", alias: "siliconflow", name: "SiliconFlow", icon: "cloud_queue", color: "#5B6EF5", textIcon: "SF", website: "https://cloud.siliconflow.com" },
80
- hyperbolic: { id: "hyperbolic", alias: "hyp", name: "Hyperbolic", icon: "bolt", color: "#00D4FF", textIcon: "HY", website: "https://hyperbolic.xyz", serviceKinds: ["llm", "tts"], ttsConfig: { baseUrl: "https://api.hyperbolic.xyz/v1/audio/generation", authType: "apikey", authHeader: "bearer", format: "hyperbolic", models: [{ id: "melo-tts", name: "Melo TTS" }] } },
68
+ azure: { id: "azure", alias: "azure", name: "Azure OpenAI", icon: "cloud", color: "#0078D4", textIcon: "AZ", website: "https://azure.microsoft.com/en-us/products/ai-services/openai-service", notice: { apiKeyUrl: "https://portal.azure.com/#view/Microsoft_Azure_ProjectOxford/CognitiveServicesHub/~/OpenAI" }, hasProviderSpecificData: true },
69
+
70
+ deepseek: { id: "deepseek", alias: "ds", name: "DeepSeek", icon: "bolt", color: "#4D6BFE", textIcon: "DS", website: "https://deepseek.com", notice: { apiKeyUrl: "https://platform.deepseek.com/api_keys" } },
71
+ groq: { id: "groq", alias: "groq", name: "Groq", icon: "speed", color: "#F55036", textIcon: "GQ", website: "https://groq.com", notice: { apiKeyUrl: "https://console.groq.com/keys" }, serviceKinds: ["llm", "imageToText"] },
72
+ xai: { id: "xai", alias: "xai", name: "xAI (Grok)", icon: "auto_awesome", color: "#1DA1F2", textIcon: "XA", website: "https://x.ai", notice: { apiKeyUrl: "https://console.x.ai" }, serviceKinds: ["llm", "imageToText", "webSearch"], searchViaChat: { defaultModel: "grok-4.20-reasoning", pricingUrl: "https://x.ai/api#pricing" } },
73
+ mistral: { id: "mistral", alias: "mistral", name: "Mistral", icon: "air", color: "#FF7000", textIcon: "MI", website: "https://mistral.ai", notice: { apiKeyUrl: "https://console.mistral.ai/api-keys" }, serviceKinds: ["llm", "imageToText", "embedding"], embeddingConfig: { baseUrl: "https://api.mistral.ai/v1/embeddings", authType: "apikey", authHeader: "bearer", models: [{ id: "mistral-embed", name: "Mistral Embed", dimensions: 1024 }] } },
74
+ perplexity: { id: "perplexity", alias: "pplx", name: "Perplexity", icon: "search", color: "#20808D", textIcon: "PP", website: "https://www.perplexity.ai", notice: { apiKeyUrl: "https://www.perplexity.ai/settings/api" }, serviceKinds: ["llm", "webSearch"], searchConfig: { baseUrl: "https://api.perplexity.ai/search", method: "POST", authType: "apikey", authHeader: "bearer", costPerQuery: 0.005, freeMonthlyQuota: 0, searchTypes: ["web"], defaultMaxResults: 5, maxMaxResults: 20, timeoutMs: 10000, cacheTTLMs: 300000 } },
75
+ together: { id: "together", alias: "together", name: "Together AI", icon: "group_work", color: "#0F6FFF", textIcon: "TG", website: "https://www.together.ai", notice: { apiKeyUrl: "https://api.together.xyz/settings/api-keys" }, serviceKinds: ["llm", "embedding"], embeddingConfig: { baseUrl: "https://api.together.xyz/v1/embeddings", authType: "apikey", authHeader: "bearer", models: [{ id: "BAAI/bge-large-en-v1.5", name: "BGE Large EN v1.5", dimensions: 1024 }, { id: "togethercomputer/m2-bert-80M-8k-retrieval", name: "M2 BERT 80M 8K", dimensions: 768 }] } },
76
+ fireworks: { id: "fireworks", alias: "fireworks", name: "Fireworks AI", icon: "local_fire_department", color: "#7B2EF2", textIcon: "FW", website: "https://fireworks.ai", notice: { apiKeyUrl: "https://fireworks.ai/account/api-keys" }, serviceKinds: ["llm", "embedding"], embeddingConfig: { baseUrl: "https://api.fireworks.ai/inference/v1/embeddings", authType: "apikey", authHeader: "bearer", models: [{ id: "nomic-ai/nomic-embed-text-v1.5", name: "Nomic Embed Text v1.5", dimensions: 768 }] } },
77
+ cerebras: { id: "cerebras", alias: "cerebras", name: "Cerebras", icon: "memory", color: "#FF4F00", textIcon: "CB", website: "https://www.cerebras.ai", notice: { apiKeyUrl: "https://cloud.cerebras.ai/platform" } },
78
+ cohere: { id: "cohere", alias: "cohere", name: "Cohere", icon: "hub", color: "#39594D", textIcon: "CO", website: "https://cohere.com", notice: { apiKeyUrl: "https://dashboard.cohere.com/api-keys" } },
79
+ nebius: { id: "nebius", alias: "nebius", name: "Nebius AI", icon: "cloud", color: "#6C5CE7", textIcon: "NB", website: "https://nebius.com", notice: { apiKeyUrl: "https://studio.nebius.com/settings/api-keys" }, serviceKinds: ["llm", "embedding"], embeddingConfig: { baseUrl: "https://api.tokenfactory.nebius.com/v1/embeddings", authType: "apikey", authHeader: "bearer", models: [{ id: "Qwen/Qwen3-Embedding-8B", name: "Qwen3 Embedding 8B", dimensions: 4096 }] } },
80
+ siliconflow: { id: "siliconflow", alias: "siliconflow", name: "SiliconFlow", icon: "cloud_queue", color: "#5B6EF5", textIcon: "SF", website: "https://cloud.siliconflow.com", notice: { apiKeyUrl: "https://cloud.siliconflow.com/account/ak" } },
81
+ hyperbolic: { id: "hyperbolic", alias: "hyp", name: "Hyperbolic", icon: "bolt", color: "#00D4FF", textIcon: "HY", website: "https://hyperbolic.xyz", notice: { apiKeyUrl: "https://app.hyperbolic.xyz/settings" }, serviceKinds: ["llm", "tts"], ttsConfig: { baseUrl: "https://api.hyperbolic.xyz/v1/audio/generation", authType: "apikey", authHeader: "bearer", format: "hyperbolic", models: [{ id: "melo-tts", name: "Melo TTS" }] } },
81
82
  deepgram: { id: "deepgram", alias: "dg", name: "Deepgram", icon: "mic", color: "#13EF93", textIcon: "DG", website: "https://deepgram.com", notice: { text: "$200 free credit on signup (no card required). Aura-1: $0.015/1k chars, Aura-2: $0.030/1k chars (Pay-As-You-Go).", apiKeyUrl: "https://console.deepgram.com/api-keys" }, serviceKinds: ["stt", "imageToText", "tts"], ttsConfig: { baseUrl: "https://api.deepgram.com/v1/speak", authType: "apikey", authHeader: "token", format: "deepgram", models: [] } },
82
- assemblyai: { id: "assemblyai", alias: "aai", name: "AssemblyAI", icon: "record_voice_over", color: "#0062FF", textIcon: "AA", website: "https://assemblyai.com", serviceKinds: ["stt"] },
83
- nanobanana: { id: "nanobanana", alias: "nb", name: "NanoBanana", icon: "image", color: "#FFD700", textIcon: "NB", website: "https://nanobananaapi.ai", serviceKinds: ["image"] },
84
- elevenlabs: { id: "elevenlabs", alias: "el", name: "ElevenLabs", icon: "record_voice_over", color: "#6C47FF", textIcon: "EL", website: "https://elevenlabs.io", serviceKinds: ["tts"], ttsConfig: { baseUrl: "https://api.elevenlabs.io/v1/text-to-speech", authType: "apikey", authHeader: "xi-api-key", format: "elevenlabs", models: [{ id: "eleven_multilingual_v2", name: "Eleven Multilingual v2" }, { id: "eleven_turbo_v2_5", name: "Eleven Turbo v2.5" }] } },
85
- cartesia: { id: "cartesia", alias: "cartesia", name: "Cartesia", icon: "spatial_audio", color: "#FF4F8B", textIcon: "CA", website: "https://cartesia.ai", serviceKinds: ["tts"], hidden: true, ttsConfig: { baseUrl: "https://api.cartesia.ai/tts/bytes", authType: "apikey", authHeader: "x-api-key", format: "cartesia", models: [{ id: "sonic-2", name: "Sonic 2" }, { id: "sonic-3", name: "Sonic 3" }] } },
86
- playht: { id: "playht", alias: "playht", name: "PlayHT", icon: "play_circle", color: "#00B4D8", textIcon: "PH", website: "https://play.ht", serviceKinds: ["tts"], hidden: true, ttsConfig: { baseUrl: "https://api.play.ht/api/v2/tts/stream", authType: "apikey", authHeader: "playht", format: "playht", models: [{ id: "PlayDialog", name: "PlayDialog" }, { id: "Play3.0-mini", name: "Play 3.0 Mini" }] } },
83
+ assemblyai: { id: "assemblyai", alias: "aai", name: "AssemblyAI", icon: "record_voice_over", color: "#0062FF", textIcon: "AA", website: "https://assemblyai.com", notice: { apiKeyUrl: "https://www.assemblyai.com/app/api-keys" }, serviceKinds: ["stt"] },
84
+ nanobanana: { id: "nanobanana", alias: "nb", name: "NanoBanana", icon: "image", color: "#FFD700", textIcon: "NB", website: "https://nanobananaapi.ai", notice: { apiKeyUrl: "https://nanobananaapi.ai/dashboard" }, serviceKinds: ["image"] },
85
+ elevenlabs: { id: "elevenlabs", alias: "el", name: "ElevenLabs", icon: "record_voice_over", color: "#6C47FF", textIcon: "EL", website: "https://elevenlabs.io", notice: { apiKeyUrl: "https://elevenlabs.io/app/settings/api-keys" }, serviceKinds: ["tts"], ttsConfig: { baseUrl: "https://api.elevenlabs.io/v1/text-to-speech", authType: "apikey", authHeader: "xi-api-key", format: "elevenlabs", models: [{ id: "eleven_multilingual_v2", name: "Eleven Multilingual v2" }, { id: "eleven_turbo_v2_5", name: "Eleven Turbo v2.5" }] } },
86
+ cartesia: { id: "cartesia", alias: "cartesia", name: "Cartesia", icon: "spatial_audio", color: "#FF4F8B", textIcon: "CA", website: "https://cartesia.ai", notice: { apiKeyUrl: "https://play.cartesia.ai/keys" }, serviceKinds: ["tts"], hidden: true, ttsConfig: { baseUrl: "https://api.cartesia.ai/tts/bytes", authType: "apikey", authHeader: "x-api-key", format: "cartesia", models: [{ id: "sonic-2", name: "Sonic 2" }, { id: "sonic-3", name: "Sonic 3" }] } },
87
+ playht: { id: "playht", alias: "playht", name: "PlayHT", icon: "play_circle", color: "#00B4D8", textIcon: "PH", website: "https://play.ht", notice: { apiKeyUrl: "https://play.ht/studio/api-access" }, serviceKinds: ["tts"], hidden: true, ttsConfig: { baseUrl: "https://api.play.ht/api/v2/tts/stream", authType: "apikey", authHeader: "playht", format: "playht", models: [{ id: "PlayDialog", name: "PlayDialog" }, { id: "Play3.0-mini", name: "Play 3.0 Mini" }] } },
87
88
  "local-device": { id: "local-device", alias: "local-device", name: "Local Device", icon: "speaker", color: "#64748B", textIcon: "LD", serviceKinds: ["tts"], noAuth: true, ttsConfig: { baseUrl: "local-device", authType: "none", authHeader: "none", format: "local-device", models: [] } },
88
89
  "google-tts": { id: "google-tts", alias: "google-tts", name: "Google TTS", icon: "record_voice_over", color: "#4285F4", textIcon: "GT", serviceKinds: ["tts"], noAuth: true, ttsConfig: { baseUrl: "google-tts", authType: "none", authHeader: "none", format: "google-tts", models: [] } },
89
90
  "edge-tts": { id: "edge-tts", alias: "edge-tts", name: "Edge TTS", icon: "record_voice_over", color: "#0078D4", textIcon: "ET", serviceKinds: ["tts"], noAuth: true, ttsConfig: { baseUrl: "edge-tts", authType: "none", authHeader: "none", format: "edge-tts", models: [] } },
@@ -93,11 +94,11 @@ export const APIKEY_PROVIDERS = {
93
94
  "voyage-ai": { id: "voyage-ai", alias: "voyage", name: "Voyage AI", icon: "data_array", color: "#0EA5E9", textIcon: "VG", website: "https://www.voyageai.com", notice: { apiKeyUrl: "https://dash.voyageai.com/api-keys" }, serviceKinds: ["embedding"], embeddingConfig: { baseUrl: "https://api.voyageai.com/v1/embeddings", authType: "apikey", authHeader: "bearer", models: [{ id: "voyage-3-large", name: "Voyage 3 Large", dimensions: 1024 }, { id: "voyage-3.5", name: "Voyage 3.5", dimensions: 1024 }, { id: "voyage-3.5-lite", name: "Voyage 3.5 Lite", dimensions: 1024 }, { id: "voyage-code-3", name: "Voyage Code 3", dimensions: 1024 }, { id: "voyage-finance-2", name: "Voyage Finance 2", dimensions: 1024 }, { id: "voyage-law-2", name: "Voyage Law 2", dimensions: 1024 }, { id: "voyage-multilingual-2", name: "Voyage Multilingual 2", dimensions: 1024 }] } },
94
95
  sdwebui: { id: "sdwebui", alias: "sdwebui", name: "SD WebUI", icon: "brush", color: "#FF7043", textIcon: "SD", website: "https://github.com/AUTOMATIC1111/stable-diffusion-webui", serviceKinds: ["image"] },
95
96
  comfyui: { id: "comfyui", alias: "comfyui", name: "ComfyUI", icon: "account_tree", color: "#4CAF50", textIcon: "CF", website: "https://github.com/comfyanonymous/ComfyUI", serviceKinds: ["image"] },
96
- huggingface: { id: "huggingface", alias: "hf", name: "HuggingFace", icon: "face", color: "#FFD21E", textIcon: "HF", website: "https://huggingface.co", serviceKinds: ["image", "imageToText", "tts"], hiddenKinds: ["tts"], ttsConfig: { baseUrl: "https://api-inference.huggingface.co/models", authType: "apikey", authHeader: "bearer", format: "huggingface-tts", models: [{ id: "facebook/mms-tts-eng", name: "MMS TTS English" }, { id: "microsoft/speecht5_tts", name: "SpeechT5 TTS" }] } },
97
- blackbox: { id: "blackbox", alias: "bb", name: "Blackbox AI", icon: "smart_toy", color: "#5B5FEF", textIcon: "BB", website: "https://blackbox.ai", serviceKinds: ["llm"] },
98
- chutes: { id: "chutes", alias: "ch", name: "Chutes AI", icon: "water_drop", color: "#ffffffff", textIcon: "CH", website: "https://chutes.ai" },
97
+ huggingface: { id: "huggingface", alias: "hf", name: "HuggingFace", icon: "face", color: "#FFD21E", textIcon: "HF", website: "https://huggingface.co", notice: { apiKeyUrl: "https://huggingface.co/settings/tokens" }, serviceKinds: ["image", "imageToText", "tts"], hiddenKinds: ["tts"], ttsConfig: { baseUrl: "https://api-inference.huggingface.co/models", authType: "apikey", authHeader: "bearer", format: "huggingface-tts", models: [{ id: "facebook/mms-tts-eng", name: "MMS TTS English" }, { id: "microsoft/speecht5_tts", name: "SpeechT5 TTS" }] } },
98
+ blackbox: { id: "blackbox", alias: "bb", name: "Blackbox AI", icon: "smart_toy", color: "#5B5FEF", textIcon: "BB", website: "https://blackbox.ai", notice: { apiKeyUrl: "https://www.blackbox.ai/api-management" }, serviceKinds: ["llm"] },
99
+ chutes: { id: "chutes", alias: "ch", name: "Chutes AI", icon: "water_drop", color: "#ffffffff", textIcon: "CH", website: "https://chutes.ai", notice: { apiKeyUrl: "https://chutes.ai/app/api" } },
99
100
  "ollama-local": { id: "ollama-local", alias: "ollama-local", name: "Ollama Local", icon: "cloud", color: "#ffffffff", textIcon: "OL", website: "https://ollama.com" },
100
- "vertex-partner": { id: "vertex-partner", alias: "vxp", name: "Vertex Partner", icon: "cloud", color: "#34A853", textIcon: "VP", website: "https://cloud.google.com/vertex-ai/generative-ai/docs/partner-models/use-partner-models" },
101
+ "vertex-partner": { id: "vertex-partner", alias: "vxp", name: "Vertex Partner", icon: "cloud", color: "#34A853", textIcon: "VP", website: "https://cloud.google.com/vertex-ai/generative-ai/docs/partner-models/use-partner-models", notice: { apiKeyUrl: "https://console.cloud.google.com/iam-admin/serviceaccounts" } },
101
102
  tavily: { id: "tavily", alias: "tavily", name: "Tavily", icon: "search", color: "#5B21B6", textIcon: "TV", website: "https://tavily.com", notice: { apiKeyUrl: "https://app.tavily.com/home" }, serviceKinds: ["webSearch", "webFetch"], searchConfig: { baseUrl: "https://api.tavily.com/search", method: "POST", authType: "apikey", authHeader: "bearer", costPerQuery: 0.008, freeMonthlyQuota: 1000, searchTypes: ["web", "news"], defaultMaxResults: 5, maxMaxResults: 20, timeoutMs: 10000, cacheTTLMs: 300000 }, fetchConfig: { baseUrl: "https://api.tavily.com/extract", method: "POST", authType: "apikey", authHeader: "bearer", costPerQuery: 0.008, freeMonthlyQuota: 1000, formats: ["markdown", "text"], maxCharacters: 100000, timeoutMs: 15000 } },
102
103
  "brave-search": { id: "brave-search", alias: "brave", name: "Brave Search", icon: "travel_explore", color: "#FB542B", textIcon: "BR", website: "https://brave.com/search/api", notice: { apiKeyUrl: "https://api-dashboard.search.brave.com/app/keys" }, serviceKinds: ["webSearch"], searchConfig: { baseUrl: "https://api.search.brave.com/res/v1", method: "GET", authType: "apikey", authHeader: "x-subscription-token", costPerQuery: 0.005, freeMonthlyQuota: 1000, searchTypes: ["web", "news"], defaultMaxResults: 5, maxMaxResults: 20, timeoutMs: 10000, cacheTTLMs: 300000 } },
103
104
  serper: { id: "serper", alias: "serper", name: "Serper", icon: "search", color: "#4F46E5", textIcon: "SP", website: "https://serper.dev", notice: { apiKeyUrl: "https://serper.dev/api-key" }, serviceKinds: ["webSearch"], searchConfig: { baseUrl: "https://google.serper.dev", method: "POST", authType: "apikey", authHeader: "x-api-key", costPerQuery: 0.001, freeMonthlyQuota: 2500, searchTypes: ["web", "news"], defaultMaxResults: 5, maxMaxResults: 100, timeoutMs: 10000, cacheTTLMs: 300000 } },
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "n9router",
3
- "version": "0.4.21",
3
+ "version": "0.4.24",
4
4
  "description": "Self-hosted AI routing gateway — local proxy for Claude, Gemini, OpenAI and 40+ providers",
5
5
  "keywords": [
6
6
  "ai",
@@ -1 +0,0 @@
1
- "use strict";exports.id=2549,exports.ids=[2549],exports.modules={12557:(a,b,c)=>{c.d(b,{Bl:()=>j,Qo:()=>f,S5:()=>l,hk:()=>e,kJ:()=>k});var d=c(3662);function e(a,b,c=0){let f=b?("string"==typeof b?b:JSON.stringify(b)).toLowerCase():"";for(let b of d.t2)if(b.text&&f&&f.includes(b.text)||b.status&&b.status===a){if(b.backoff){let a=Math.min(c+1,d.EQ.maxLevel);return{shouldFallback:!0,cooldownMs:function(a=0){let b=Math.max(0,a-1);return Math.min(d.EQ.base*Math.pow(2,b),d.EQ.max)}(a),newBackoffLevel:a}}return{shouldFallback:!0,cooldownMs:b.cooldownMs}}return{shouldFallback:!0,cooldownMs:d.wf}}function f(a){if(!a)return"";let b=new Date(a).getTime()-Date.now();if(b<=0)return"reset after 0s";let c=Math.ceil(b/1e3),d=Math.floor(c/3600),e=Math.floor(c%3600/60),f=c%60,g=[];return d>0&&g.push(`${d}h`),e>0&&g.push(`${e}m`),(f>0||0===g.length)&&g.push(`${f}s`),`reset after ${g.join(" ")}`}let g="modelLock_",h=`${g}__all`;function i(a){return a?`${g}${a}`:h}function j(a,b){let c=a[i(b)]||a[h];return!!c&&new Date(c).getTime()>Date.now()}function k(a){if(!a)return null;let b=null,c=Date.now();for(let[d,e]of Object.entries(a)){if(!d.startsWith(g)||!e)continue;let a=new Date(e).getTime();a<=c||(!b||a<b)&&(b=a)}return b?new Date(b).toISOString():null}function l(a,b){return{[i(a)]:new Date(Date.now()+b).toISOString()}}},13808:(a,b,c)=>{c.d(b,{lz:()=>f});var d=c(55511),e=c.n(d);function f(){let a=e().randomBytes(32).toString("base64url"),b=e().createHash("sha256").update(a).digest("base64url");return{codeVerifier:a,codeChallenge:b,state:e().randomBytes(32).toString("base64url")}}},43659:(a,b,c)=>{c.d(b,{A1:()=>g,lR:()=>i,wO:()=>h,yj:()=>e,zL:()=>f});var d=c(3662);function e(a,b){let c;return new Response(JSON.stringify((c=d.LY[a]||(a>=500?{type:"server_error",code:"internal_server_error"}:{type:"invalid_request_error",code:""}),{error:{message:b||d.O[a]||"An error occurred",type:c.type,code:c.code}})),{status:a,headers:{"Content-Type":"application/json","Access-Control-Allow-Origin":"*"}})}async function f(a,b=null){let c="";try{c=await a.text()}catch{c=""}if(b&&"function"==typeof b.parseError)try{let e=b.parseError(a,c);if(e&&"object"==typeof e){let b=e.message||d.O[a.status]||`Upstream error: ${a.status}`;return{statusCode:e.status||a.status,message:b,resetsAtMs:e.resetsAtMs}}}catch{}let e="";try{let a=JSON.parse(c);e=a.error?.message||a.message||a.error||c}catch{e=c}let g=("string"==typeof e?e:JSON.stringify(e))||d.O[a.status]||`Upstream error: ${a.status}`;return{statusCode:a.status,message:g}}function g(a,b,c){return{success:!1,status:a,error:b,resetsAtMs:c,response:e(a,b)}}function h(a,b,c,d){let e=Math.max(Math.ceil((new Date(c).getTime()-Date.now())/1e3),1);return new Response(JSON.stringify({error:{message:`${b} (${d})`}}),{status:a,headers:{"Content-Type":"application/json","Retry-After":String(e)}})}function i(a,b,c,d){let e=d||a.code||"FETCH_FAILED",f=a.message||"Unknown error",g=a.cause?.code,h=a.cause?.message,i=g||h?` (cause: ${[g,h].filter(Boolean).join(": ")})`:"";return`[${e}]: ${f}${i}`}},74177:(a,b,c)=>{c.a(a,async(a,d)=>{try{c.d(b,{Gj:()=>m,PY:()=>p,Re:()=>n,Su:()=>o,kP:()=>l,sO:()=>k});var e=c(48895),f=c(13808),g=c(92990),h=a([e]);function i(a){try{if(!a||"string"!=typeof a)return null;let b=a.split(".");if(3!==b.length)return null;let c=b[1].replace(/-/g,"+").replace(/_/g,"/"),d=(4-c.length%4)%4,e=c+"=".repeat(d);return JSON.parse(Buffer.from(e,"base64").toString("utf8"))}catch{return null}}function j(a){let b=i(a);if(!b)return{};let c=b["https://api.openai.com/auth"]||{};return{email:b.email,chatgptAccountId:c.chatgpt_account_id,chatgptPlanType:c.chatgpt_plan_type}}e=(h.then?(await h)():h)[0];let q={claude:{config:g.OV,flowType:"authorization_code_pkce",buildAuthUrl:(a,b,c,d)=>{let e=new URLSearchParams({code:"true",client_id:a.clientId,response_type:"code",redirect_uri:b,scope:a.scopes.join(" "),code_challenge:d,code_challenge_method:a.codeChallengeMethod,state:c});return`${a.authorizeUrl}?${e.toString()}`},exchangeToken:async(a,b,c,d,e)=>{let f=b,g="";if(f.includes("#")){let a=f.split("#");f=a[0],g=a[1]||""}let h=await fetch(a.tokenUrl,{method:"POST",headers:{"Content-Type":"application/json",Accept:"application/json"},body:JSON.stringify({code:f,state:g||e,grant_type:"authorization_code",client_id:a.clientId,redirect_uri:c,code_verifier:d})});if(!h.ok){let a=await h.text();throw Error(`Token exchange failed: ${a}`)}return await h.json()},mapTokens:a=>({accessToken:a.access_token,refreshToken:a.refresh_token,expiresIn:a.expires_in,scope:a.scope})},codex:{config:g.DI,flowType:"authorization_code_pkce",fixedPort:1455,callbackPath:"/auth/callback",buildAuthUrl:(a,b,c,d)=>{let e={response_type:"code",client_id:a.clientId,redirect_uri:b,scope:a.scope,code_challenge:d,code_challenge_method:a.codeChallengeMethod,...a.extraParams,state:c},f=Object.entries(e).map(([a,b])=>`${a}=${encodeURIComponent(b)}`).join("&");return`${a.authorizeUrl}?${f}`},exchangeToken:async(a,b,c,d)=>{let e=await fetch(a.tokenUrl,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded",Accept:"application/json"},body:new URLSearchParams({grant_type:"authorization_code",client_id:a.clientId,code:b,redirect_uri:c,code_verifier:d})});if(!e.ok){let a=await e.text();throw Error(`Token exchange failed: ${a}`)}return await e.json()},mapTokens:a=>{let b=j(a.id_token),c={accessToken:a.access_token,refreshToken:a.refresh_token,idToken:a.id_token,expiresIn:a.expires_in};return b.email&&(c.email=b.email),(b.chatgptAccountId||b.chatgptPlanType)&&(c.providerSpecificData={chatgptAccountId:b.chatgptAccountId,chatgptPlanType:b.chatgptPlanType}),c}},"gemini-cli":{config:g.LT,flowType:"authorization_code",buildAuthUrl:(a,b,c)=>{let d=new URLSearchParams({client_id:a.clientId,response_type:"code",redirect_uri:b,scope:a.scopes.join(" "),state:c,access_type:"offline",prompt:"consent"});return`${a.authorizeUrl}?${d.toString()}`},exchangeToken:async(a,b,c)=>{let d=await fetch(a.tokenUrl,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded",Accept:"application/json"},body:new URLSearchParams({grant_type:"authorization_code",client_id:a.clientId,client_secret:a.clientSecret,code:b,redirect_uri:c})});if(!d.ok){let a=await d.text();throw Error(`Token exchange failed: ${a}`)}return await d.json()},postExchange:async a=>{let b=await fetch(`${g.LT.userInfoUrl}?alt=json`,{headers:{Authorization:`Bearer ${a.access_token}`}}),c=b.ok?await b.json():{},d="";try{let b=await fetch("https://cloudcode-pa.googleapis.com/v1internal:loadCodeAssist",{method:"POST",headers:{Authorization:`Bearer ${a.access_token}`,"Content-Type":"application/json"},body:JSON.stringify({metadata:getOAuthClientMetadata(),mode:1})});if(b.ok){let a=await b.json();d=a.cloudaicompanionProject?.id||a.cloudaicompanionProject||""}}catch(a){console.log("Failed to fetch project ID:",a)}return{userInfo:c,projectId:d}},mapTokens:(a,b)=>({accessToken:a.access_token,refreshToken:a.refresh_token,expiresIn:a.expires_in,scope:a.scope,email:b?.userInfo?.email,projectId:b?.projectId})},antigravity:{config:g.YT,flowType:"authorization_code",buildAuthUrl:(a,b,c)=>{let d=new URLSearchParams({client_id:a.clientId,response_type:"code",redirect_uri:b,scope:a.scopes.join(" "),state:c,access_type:"offline",prompt:"consent"});return`${a.authorizeUrl}?${d.toString()}`},exchangeToken:async(a,b,c)=>{let d=await fetch(a.tokenUrl,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded",Accept:"application/json"},body:new URLSearchParams({grant_type:"authorization_code",client_id:a.clientId,client_secret:a.clientSecret,code:b,redirect_uri:c})});if(!d.ok){let a=await d.text();throw Error(`Token exchange failed: ${a}`)}return await d.json()},postExchange:async a=>{let b={Authorization:`Bearer ${a.access_token}`,"Content-Type":"application/json","User-Agent":g.YT.loadCodeAssistUserAgent,"X-Goog-Api-Client":g.YT.loadCodeAssistApiClient,"Client-Metadata":g.YT.loadCodeAssistClientMetadata,"x-request-source":"local"},c={ideType:"IDE_UNSPECIFIED",platform:"PLATFORM_UNSPECIFIED",pluginType:"GEMINI"},d=await fetch(`${g.YT.userInfoUrl}?alt=json`,{headers:{Authorization:`Bearer ${a.access_token}`,"x-request-source":"local"}}),e=d.ok?await d.json():{},f="",h="legacy-tier";try{let a=await fetch(g.YT.loadCodeAssistEndpoint,{method:"POST",headers:b,body:JSON.stringify({metadata:c})});if(a.ok){let b=await a.json();if(f=b.cloudaicompanionProject?.id||b.cloudaicompanionProject||"",Array.isArray(b.allowedTiers)){for(let a of b.allowedTiers)if(a.isDefault&&a.id){h=a.id.trim();break}}}}catch(a){console.log("Failed to load code assist:",a)}return f&&(async()=>{for(let a=0;a<10;a++){try{let a=await fetch(g.YT.onboardUserEndpoint,{method:"POST",headers:b,body:JSON.stringify({tierId:h,metadata:c})});if(a.ok){let b=await a.json();if(!0===b.done)break}}catch(a){break}await new Promise(a=>setTimeout(a,5e3))}})().catch(()=>{}),{userInfo:e,projectId:f}},mapTokens:(a,b)=>({accessToken:a.access_token,refreshToken:a.refresh_token,expiresIn:a.expires_in,scope:a.scope,email:b?.userInfo?.email,projectId:b?.projectId})},iflow:{config:g.ZL,flowType:"authorization_code",buildAuthUrl:(a,b,c)=>{let d=new URLSearchParams({loginMethod:a.extraParams.loginMethod,type:a.extraParams.type,redirect:b,state:c,client_id:a.clientId});return`${a.authorizeUrl}?${d.toString()}`},exchangeToken:async(a,b,c)=>{let d=Buffer.from(`${a.clientId}:${a.clientSecret}`).toString("base64"),e=await fetch(a.tokenUrl,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded",Accept:"application/json",Authorization:`Basic ${d}`},body:new URLSearchParams({grant_type:"authorization_code",code:b,redirect_uri:c,client_id:a.clientId,client_secret:a.clientSecret})});if(!e.ok){let a=await e.text();throw Error(`Token exchange failed: ${a}`)}return await e.json()},postExchange:async a=>{let b=await fetch(`${g.ZL.userInfoUrl}?accessToken=${encodeURIComponent(a.access_token)}`,{headers:{Accept:"application/json"}});if(!b.ok){let a=await b.text();throw Error(`Failed to fetch user info: ${a}`)}let c=await b.json();if(!c.success)throw Error(`User info request failed: ${c.message||"Unknown error"}`);let d=c.data||{};if(!d.apiKey||""===d.apiKey.trim())throw Error("Empty API key returned from iFlow");if(!(d.email?.trim()||d.phone?.trim()))throw Error("Missing account email/phone in user info");return{userInfo:d}},mapTokens:(a,b)=>({accessToken:a.access_token,refreshToken:a.refresh_token,expiresIn:a.expires_in,apiKey:b?.userInfo?.apiKey,email:b?.userInfo?.email||b?.userInfo?.phone,displayName:b?.userInfo?.nickname||b?.userInfo?.name})},qoder:{config:g.hF,flowType:"authorization_code",buildAuthUrl:(a,b,c)=>{let d=new URLSearchParams({client_id:a.clientId,response_type:"code",redirect_uri:b,state:c});return`${a.authorizeUrl}?${d.toString()}`},exchangeToken:async(a,b,c)=>{let d=Buffer.from(`${a.clientId}:${a.clientSecret}`).toString("base64"),e=await fetch(a.tokenUrl,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded",Accept:"application/json",Authorization:`Basic ${d}`},body:new URLSearchParams({grant_type:"authorization_code",code:b,redirect_uri:c,client_id:a.clientId,client_secret:a.clientSecret})});if(!e.ok){let a=await e.text();throw Error(`Token exchange failed: ${a}`)}return await e.json()},postExchange:async a=>{let b=await fetch(`${g.hF.userInfoUrl}?accessToken=${encodeURIComponent(a.access_token)}`,{headers:{Accept:"application/json"}});if(!b.ok){let a=await b.text();throw Error(`Failed to fetch user info: ${a}`)}let c=await b.json();if(!c.success)throw Error(`User info request failed: ${c.message||"Unknown error"}`);let d=c.data||{};if(!d.apiKey||""===d.apiKey.trim())throw Error("Empty API key returned from Qoder");if(!(d.email?.trim()||d.phone?.trim()))throw Error("Missing account email/phone in user info");return{userInfo:d}},mapTokens:(a,b)=>({accessToken:a.access_token,refreshToken:a.refresh_token,expiresIn:a.expires_in,apiKey:b?.userInfo?.apiKey,email:b?.userInfo?.email||b?.userInfo?.phone,displayName:b?.userInfo?.nickname||b?.userInfo?.name})},qwen:{config:g.Hp,flowType:"device_code",requestDeviceCode:async(a,b)=>{let c=await fetch(a.deviceCodeUrl,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded",Accept:"application/json"},body:new URLSearchParams({client_id:a.clientId,scope:a.scope,code_challenge:b,code_challenge_method:a.codeChallengeMethod})});if(!c.ok){let a=await c.text();throw Error(`Device code request failed: ${a}`)}return await c.json()},pollToken:async(a,b,c)=>{let d=await fetch(a.tokenUrl,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded",Accept:"application/json"},body:new URLSearchParams({grant_type:"urn:ietf:params:oauth:grant-type:device_code",client_id:a.clientId,device_code:b,code_verifier:c})});return{ok:d.ok,data:await d.json()}},mapTokens:a=>({accessToken:a.access_token,refreshToken:a.refresh_token,expiresIn:a.expires_in,providerSpecificData:{resourceUrl:a.resource_url}})},github:{config:g.Nu,flowType:"device_code",requestDeviceCode:async a=>{let b=await fetch(a.deviceCodeUrl,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded",Accept:"application/json"},body:new URLSearchParams({client_id:a.clientId,scope:a.scopes})});if(!b.ok){let a=await b.text();throw Error(`Device code request failed: ${a}`)}return await b.json()},pollToken:async(a,b)=>{let c,d=await fetch(a.tokenUrl,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded",Accept:"application/json"},body:new URLSearchParams({client_id:a.clientId,device_code:b,grant_type:"urn:ietf:params:oauth:grant-type:device_code"})});try{c=await d.json()}catch(b){let a=await d.text();c={error:"invalid_response",error_description:a}}return{ok:d.ok,data:c}},postExchange:async a=>{let b=await fetch(g.Nu.copilotTokenUrl,{headers:{Authorization:`Bearer ${a.access_token}`,Accept:"application/json","X-GitHub-Api-Version":g.Nu.apiVersion,"User-Agent":g.Nu.userAgent}}),c=b.ok?await b.json():{},d=await fetch(g.Nu.userInfoUrl,{headers:{Authorization:`Bearer ${a.access_token}`,Accept:"application/json","X-GitHub-Api-Version":g.Nu.apiVersion,"User-Agent":g.Nu.userAgent}}),e=d.ok?await d.json():{};return{copilotToken:c,userInfo:e}},mapTokens:(a,b)=>({accessToken:a.access_token,refreshToken:a.refresh_token,expiresIn:a.expires_in,providerSpecificData:{copilotToken:b?.copilotToken?.token,copilotTokenExpiresAt:b?.copilotToken?.expires_at,githubUserId:b?.userInfo?.id,githubLogin:b?.userInfo?.login,githubName:b?.userInfo?.name,githubEmail:b?.userInfo?.email}})},kiro:{config:g.Tx,flowType:"device_code",requestDeviceCode:async(a,b,c={})=>{let d=("string"==typeof c.region?c.region.trim():"")||"us-east-1",e=("string"==typeof c.startUrl?c.startUrl.trim():"")||a.startUrl,f="idc"===c.authMethod?"idc":"builder-id",g=`https://oidc.${d}.amazonaws.com/client/register`,h=`https://oidc.${d}.amazonaws.com/device_authorization`,i=await fetch(g,{method:"POST",headers:{"Content-Type":"application/json",Accept:"application/json"},body:JSON.stringify({clientName:a.clientName,clientType:a.clientType,scopes:a.scopes,grantTypes:a.grantTypes,issuerUrl:a.issuerUrl})});if(!i.ok){let a=await i.text();throw Error(`Client registration failed: ${a}`)}let j=await i.json(),k=await fetch(h,{method:"POST",headers:{"Content-Type":"application/json",Accept:"application/json"},body:JSON.stringify({clientId:j.clientId,clientSecret:j.clientSecret,startUrl:e})});if(!k.ok){let a=await k.text();throw Error(`Device authorization failed: ${a}`)}let l=await k.json();return{device_code:l.deviceCode,user_code:l.userCode,verification_uri:l.verificationUri,verification_uri_complete:l.verificationUriComplete,expires_in:l.expiresIn,interval:l.interval||5,_clientId:j.clientId,_clientSecret:j.clientSecret,_region:d,_authMethod:f,_startUrl:e}},pollToken:async(a,b,c,d)=>{let e,f=d?._region||"us-east-1",g=`https://oidc.${f}.amazonaws.com/token`,h=await fetch(g,{method:"POST",headers:{"Content-Type":"application/json",Accept:"application/json"},body:JSON.stringify({clientId:d?._clientId,clientSecret:d?._clientSecret,deviceCode:b,grantType:"urn:ietf:params:oauth:grant-type:device_code"})});try{e=await h.json()}catch(b){let a=await h.text();e={error:"invalid_response",error_description:a}}return e.accessToken?{ok:!0,data:{access_token:e.accessToken,refresh_token:e.refreshToken,expires_in:e.expiresIn,profile_arn:e?.profileArn||null,_clientId:d?._clientId,_clientSecret:d?._clientSecret,_region:d?._region,_authMethod:d?._authMethod,_startUrl:d?._startUrl}}:{ok:!1,data:{error:e.error||"authorization_pending",error_description:e.error_description||e.message}}},mapTokens:a=>{let b=function(a){let b=i(a);if(b)return b.email||b.preferred_username||b.sub||void 0}(a.access_token);return{accessToken:a.access_token,refreshToken:a.refresh_token,expiresIn:a.expires_in,email:b,providerSpecificData:{profileArn:a?.profile_arn||null,clientId:a._clientId,clientSecret:a._clientSecret,region:a._region||"us-east-1",authMethod:a._authMethod||"builder-id",startUrl:a._startUrl||g.Tx.startUrl}}}},cursor:{config:g.WN,flowType:"import_token",mapTokens:a=>({accessToken:a.accessToken,refreshToken:null,expiresIn:a.expiresIn||86400,providerSpecificData:{machineId:a.machineId,authMethod:"imported"}})},"kimi-coding":{config:g.t,flowType:"device_code",requestDeviceCode:async a=>{let b=await fetch(a.deviceCodeUrl,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded",Accept:"application/json"},body:new URLSearchParams({client_id:a.clientId})});if(!b.ok){let a=await b.text();throw Error(`Device code request failed: ${a}`)}let c=await b.json();return{device_code:c.device_code,user_code:c.user_code,verification_uri:c.verification_uri||"https://www.kimi.com/code/authorize_device",verification_uri_complete:c.verification_uri_complete||`https://www.kimi.com/code/authorize_device?user_code=${c.user_code}`,expires_in:c.expires_in,interval:c.interval||5}},pollToken:async(a,b)=>{let c,d=await fetch(a.tokenUrl,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded",Accept:"application/json"},body:new URLSearchParams({grant_type:"urn:ietf:params:oauth:grant-type:device_code",client_id:a.clientId,device_code:b})});try{c=await d.json()}catch(b){let a=await d.text();c={error:"invalid_response",error_description:a}}return{ok:d.ok,data:c}},mapTokens:a=>({accessToken:a.access_token,refreshToken:a.refresh_token,expiresIn:a.expires_in})},kilocode:{config:g.MZ,flowType:"device_code",requestDeviceCode:async a=>{let b=await fetch(a.initiateUrl,{method:"POST",headers:{"Content-Type":"application/json"}});if(!b.ok){if(429===b.status)throw Error("Too many pending authorization requests. Please try again later.");let a=await b.text();throw Error(`Device auth initiation failed: ${a}`)}let c=await b.json();return{device_code:c.code,user_code:c.code,verification_uri:c.verificationUrl,verification_uri_complete:c.verificationUrl,expires_in:c.expiresIn||300,interval:3}},pollToken:async(a,b)=>{let c=await fetch(`${a.pollUrlBase}/${b}`);if(202===c.status)return{ok:!1,data:{error:"authorization_pending"}};if(403===c.status)return{ok:!1,data:{error:"access_denied",error_description:"Authorization denied by user"}};if(410===c.status)return{ok:!1,data:{error:"expired_token",error_description:"Authorization code expired"}};if(!c.ok)return{ok:!1,data:{error:"poll_failed",error_description:`Poll failed: ${c.status}`}};let d=await c.json();if("approved"===d.status&&d.token){let b=null;try{let c=await fetch(`${a.apiBaseUrl}/api/profile`,{headers:{Authorization:`Bearer ${d.token}`}});if(c.ok){let a=await c.json();b=a.organizations?.[0]?.id||null}}catch{}return{ok:!0,data:{access_token:d.token,_userEmail:d.userEmail,_orgId:b}}}return{ok:!1,data:{error:"authorization_pending"}}},mapTokens:a=>({accessToken:a.access_token,refreshToken:null,expiresIn:null,email:a._userEmail,...a._orgId?{providerSpecificData:{orgId:a._orgId}}:{}})},cline:{config:g.lB,flowType:"authorization_code",buildAuthUrl:(a,b)=>{let c=new URLSearchParams({client_type:"extension",callback_url:b,redirect_uri:b});return`${a.authorizeUrl}?${c.toString()}`},exchangeToken:async(a,b,c)=>{try{let a=b,c=4-a.length%4;4!==c&&(a+="=".repeat(c));let d=Buffer.from(a,"base64").toString("utf-8"),e=d.lastIndexOf("}");if(-1===e)throw Error("No JSON found in decoded code");let f=JSON.parse(d.substring(0,e+1));return{access_token:f.accessToken,refresh_token:f.refreshToken,email:f.email,firstName:f.firstName,lastName:f.lastName,expires_at:f.expiresAt}}catch(f){let d=await fetch(a.tokenExchangeUrl,{method:"POST",headers:{"Content-Type":"application/json",Accept:"application/json"},body:JSON.stringify({grant_type:"authorization_code",code:b,client_type:"extension",redirect_uri:c})});if(!d.ok){let a=await d.text();throw Error(`Cline token exchange failed: ${a}`)}let e=await d.json();return{access_token:e.data?.accessToken||e.accessToken,refresh_token:e.data?.refreshToken||e.refreshToken,email:e.data?.userInfo?.email||"",expires_at:e.data?.expiresAt||e.expiresAt}}},mapTokens:a=>({accessToken:a.access_token,refreshToken:a.refresh_token,expiresIn:a.expires_at?Math.floor((new Date(a.expires_at).getTime()-Date.now())/1e3):3600,email:a.email,providerSpecificData:{firstName:a.firstName,lastName:a.lastName}})},gitlab:{config:g.f7,flowType:"authorization_code_pkce",buildAuthUrl:(a,b,c,d,e={})=>{let f=e.baseUrl||a.defaultBaseUrl,g=e.clientId||"",h=new URLSearchParams({client_id:g,redirect_uri:b,response_type:"code",state:c,scope:a.scope,code_challenge:d,code_challenge_method:a.codeChallengeMethod});return`${f}${a.authorizeUrlPath}?${h.toString()}`},exchangeToken:async(a,b,c,d,e,f={})=>{let g=f.baseUrl||a.defaultBaseUrl,h=f.clientId||"",i=f.clientSecret||"",j=new URLSearchParams({client_id:h,grant_type:"authorization_code",code:b,redirect_uri:c,code_verifier:d});i&&j.set("client_secret",i);let k=await fetch(`${g}${a.tokenUrlPath}`,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded",Accept:"application/json"},body:j.toString()});if(!k.ok)throw Error(`GitLab token exchange failed: ${await k.text()}`);let l=await k.json(),m=await fetch(`${g}${a.userInfoUrlPath}`,{headers:{Authorization:`Bearer ${l.access_token}`}}),n=m.ok?await m.json():{};return{...l,_user:n,_baseUrl:g,_clientId:h}},mapTokens:a=>({accessToken:a.access_token,refreshToken:a.refresh_token,expiresIn:a.expires_in,scope:a.scope,providerSpecificData:{username:a._user?.username||"",email:a._user?.email||a._user?.public_email||"",name:a._user?.name||"",baseUrl:a._baseUrl,clientId:a._clientId,authKind:"oauth"}})},codebuddy:{config:g.Nl,flowType:"device_code",requestDeviceCode:async a=>{let b=await fetch(`${a.stateUrl}?platform=${a.platform}`,{method:"POST",headers:{"Content-Type":"application/json",Accept:"application/json","User-Agent":a.userAgent,"X-Requested-With":"XMLHttpRequest","X-Domain":"copilot.tencent.com","X-No-Authorization":"true","X-No-User-Id":"true","X-Product":"SaaS"},body:"{}"});if(!b.ok)throw Error(`CodeBuddy state request failed: ${await b.text()}`);let c=await b.json();if(0!==c.code||!c.data?.state||!c.data?.authUrl)throw Error(`CodeBuddy state error: ${c.msg||"missing state/authUrl"}`);return{device_code:c.data.state,verification_uri:c.data.authUrl,user_code:"",interval:a.pollInterval/1e3,_isCodeBuddy:!0}},pollToken:async(a,b)=>{let c=await fetch(a.tokenUrl,{method:"POST",headers:{"Content-Type":"application/json",Accept:"application/json","User-Agent":a.userAgent,"X-Requested-With":"XMLHttpRequest","X-Domain":"copilot.tencent.com","X-No-Authorization":"true","X-No-User-Id":"true","X-Product":"SaaS"},body:JSON.stringify({state:b})});if(!c.ok)return{ok:!1,data:{error:"request_failed"}};let d=await c.json();return 0===d.code&&d.data?.accessToken?{ok:!0,data:{access_token:d.data.accessToken,refresh_token:d.data.refreshToken||"",token_type:d.data.tokenType||"Bearer"}}:11217===d.code?{ok:!0,data:{error:"authorization_pending"}}:{ok:!1,data:{error:d.msg||"unknown_error"}}},mapTokens:a=>({accessToken:a.access_token,refreshToken:a.refresh_token,expiresIn:86400,providerSpecificData:{}})}};function k(a){let b=q[a];if(!b)throw Error(`Unknown provider: ${a}`);return b}function l(a,b,c){let d=k(a),{codeVerifier:e,codeChallenge:g,state:h}=(0,f.lz)();return{authUrl:"device_code"===d.flowType?null:"authorization_code_pkce"===d.flowType?d.buildAuthUrl(d.config,b,h,g,c||{}):d.buildAuthUrl(d.config,b,h,void 0,c||{}),state:h,codeVerifier:e,codeChallenge:g,redirectUri:b,flowType:d.flowType,fixedPort:d.fixedPort,callbackPath:d.callbackPath||"/callback"}}async function m(a,b,c,d,e,f){let g=k(a),h=await g.exchangeToken(g.config,b,c,d,e,f||{}),i=null;return g.postExchange&&(i=await g.postExchange(h)),g.mapTokens(h,i)}async function n(a,b,c){let d=k(a);if("device_code"!==d.flowType)throw Error(`Provider ${a} does not support device code flow`);return await d.requestDeviceCode(d.config,b,c||{})}async function o(a,b,c,d){let e=k(a);if("device_code"!==e.flowType)throw Error(`Provider ${a} does not support device code flow`);let f=await e.pollToken(e.config,b,c,d);if(f.ok)if(f.data.access_token){let a=null;return e.postExchange&&(a=await e.postExchange(f.data)),{success:!0,tokens:e.mapTokens(f.data,a)}}else if("authorization_pending"===f.data.error||"slow_down"===f.data.error)return{success:!1,error:f.data.error,errorDescription:f.data.error_description||f.data.message,pending:"authorization_pending"===f.data.error};else return{success:!1,error:f.data.error||"no_access_token",errorDescription:f.data.error_description||f.data.message||"No access token received"};return{success:!1,error:f.data.error,errorDescription:f.data.error_description}}let r=!1;async function p(){if(!r){r=!0;try{let{getProviderConnections:a,updateProviderConnection:b}=await Promise.resolve().then(c.bind(c,89718));for(let c of(await a()).filter(a=>{if("codex"!==a.provider||"oauth"!==a.authType||!a.idToken)return!1;let b=!!a.email,c=!!a.providerSpecificData?.chatgptAccountId;return!b||!c})){let a=j(c.idToken);if(!a.email&&!a.chatgptAccountId)continue;let d={};!c.email&&a.email&&(d.email=a.email),(a.chatgptAccountId||a.chatgptPlanType)&&(d.providerSpecificData={...c.providerSpecificData||{},chatgptAccountId:a.chatgptAccountId,chatgptPlanType:a.chatgptPlanType}),Object.keys(d).length&&await b(c.id,d)}}catch(a){r=!1,console.log("backfillCodexEmails failed:",a?.message||a)}}}d()}catch(a){d(a)}})},92990:(a,b,c)=>{c.d(b,{DI:()=>e,Hp:()=>g,LT:()=>f,MZ:()=>o,Nl:()=>r,Nu:()=>k,OV:()=>d,Tx:()=>l,WN:()=>m,YT:()=>j,ZL:()=>i,f7:()=>q,hF:()=>h,lB:()=>p,t:()=>n}),c(21820);let d={clientId:"9d1c250a-e61b-44d9-88ed-5944d1962f5e",authorizeUrl:"https://claude.ai/oauth/authorize",tokenUrl:"https://api.anthropic.com/v1/oauth/token",scopes:["org:create_api_key","user:profile","user:inference"],codeChallengeMethod:"S256"},e={clientId:"app_EMoamEEZ73f0CkXaXp7hrann",authorizeUrl:"https://auth.openai.com/oauth/authorize",tokenUrl:"https://auth.openai.com/oauth/token",scope:"openid profile email offline_access",codeChallengeMethod:"S256",extraParams:{id_token_add_organizations:"true",codex_cli_simplified_flow:"true",originator:"codex_cli_rs"}},f={clientId:"681255809395-oo8ft2oprdrnp9e3aqf6av3hmdib135j.apps.googleusercontent.com",clientSecret:"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",authorizeUrl:"https://accounts.google.com/o/oauth2/v2/auth",tokenUrl:"https://oauth2.googleapis.com/token",userInfoUrl:"https://www.googleapis.com/oauth2/v1/userinfo",scopes:["https://www.googleapis.com/auth/cloud-platform","https://www.googleapis.com/auth/userinfo.email","https://www.googleapis.com/auth/userinfo.profile"]},g={clientId:"f0304373b74a44d2b584a3fb70ca9e56",deviceCodeUrl:"https://qwen.ai/api/v1/oauth2/device/code",tokenUrl:"https://qwen.ai/api/v1/oauth2/token",scope:"openid profile email model.completion",codeChallengeMethod:"S256"},h={apiBaseUrl:"https://api2.qoder.sh",deviceTokenUrl:"https://api2.qoder.sh/api/v1/deviceToken/poll",deviceRefreshUrl:"https://api2.qoder.sh/api/v1/deviceToken/refresh",refreshUrl:"https://api2.qoder.sh/api/v3/user/refresh_token",userInfoUrl:"https://api2.qoder.sh/api/v1/userinfo",statusUrl:"https://api2.qoder.sh/api/v3/user/status",loginUrl:"https://qoder.com/login"},i={clientId:"10009311001",clientSecret:"4Z3YjXycVsQvyGF1etiNlIBB4RsqSDtW",authorizeUrl:"https://iflow.cn/oauth",tokenUrl:"https://iflow.cn/oauth/token",userInfoUrl:"https://iflow.cn/api/oauth/getUserInfo",extraParams:{loginMethod:"phone",type:"phone"}},j={clientId:"1071006060591-tmhssin2h21lcre235vtolojh4g403ep.apps.googleusercontent.com",clientSecret:"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",authorizeUrl:"https://accounts.google.com/o/oauth2/v2/auth",tokenUrl:"https://oauth2.googleapis.com/token",userInfoUrl:"https://www.googleapis.com/oauth2/v1/userinfo",scopes:["https://www.googleapis.com/auth/cloud-platform","https://www.googleapis.com/auth/userinfo.email","https://www.googleapis.com/auth/userinfo.profile","https://www.googleapis.com/auth/cclog","https://www.googleapis.com/auth/experimentsandconfigs"],apiEndpoint:"https://cloudcode-pa.googleapis.com",apiVersion:"v1internal",loadCodeAssistEndpoint:"https://cloudcode-pa.googleapis.com/v1internal:loadCodeAssist",onboardUserEndpoint:"https://cloudcode-pa.googleapis.com/v1internal:onboardUser",loadCodeAssistUserAgent:"google-api-nodejs-client/9.15.1",loadCodeAssistApiClient:"google-cloud-sdk vscode_cloudshelleditor/0.1",loadCodeAssistClientMetadata:JSON.stringify({ideType:"IDE_UNSPECIFIED",platform:"PLATFORM_UNSPECIFIED",pluginType:"GEMINI"})},k={clientId:"Iv1.b507a08c87ecfe98",deviceCodeUrl:"https://github.com/login/device/code",tokenUrl:"https://github.com/login/oauth/access_token",userInfoUrl:"https://api.github.com/user",scopes:"read:user",apiVersion:"2022-11-28",copilotTokenUrl:"https://api.github.com/copilot_internal/v2/token",userAgent:"GitHubCopilotChat/0.26.7",editorVersion:"vscode/1.85.0",editorPluginVersion:"copilot-chat/0.26.7"},l={ssoOidcEndpoint:"https://oidc.us-east-1.amazonaws.com",registerClientUrl:"https://oidc.us-east-1.amazonaws.com/client/register",deviceAuthUrl:"https://oidc.us-east-1.amazonaws.com/device_authorization",tokenUrl:"https://oidc.us-east-1.amazonaws.com/token",startUrl:"https://view.awsapps.com/start",clientName:"kiro-oauth-client",clientType:"public",scopes:["codewhisperer:completions","codewhisperer:analysis","codewhisperer:conversations"],grantTypes:["urn:ietf:params:oauth:grant-type:device_code","refresh_token"],issuerUrl:"https://identitycenter.amazonaws.com/ssoins-722374e8c3c8e6c6",socialAuthEndpoint:"https://prod.us-east-1.auth.desktop.kiro.dev",socialLoginUrl:"https://prod.us-east-1.auth.desktop.kiro.dev/login",socialTokenUrl:"https://prod.us-east-1.auth.desktop.kiro.dev/oauth/token",socialRefreshUrl:"https://prod.us-east-1.auth.desktop.kiro.dev/refreshToken",authMethods:["builder-id","idc","google","github","import"]},m={apiEndpoint:"https://api2.cursor.sh",chatEndpoint:"/aiserver.v1.ChatService/StreamUnifiedChatWithTools",modelsEndpoint:"/aiserver.v1.AiService/GetDefaultModelNudgeData",api3Endpoint:"https://api3.cursor.sh",agentEndpoint:"https://agent.api5.cursor.sh",agentNonPrivacyEndpoint:"https://agentn.api5.cursor.sh",clientVersion:"3.1.0",clientType:"ide",tokenStoragePaths:{linux:"~/.config/Cursor/User/globalStorage/state.vscdb",macos:"/Users/<user>/Library/Application Support/Cursor/User/globalStorage/state.vscdb",windows:"%APPDATA%\\Cursor\\User\\globalStorage\\state.vscdb"},dbKeys:{accessToken:"cursorAuth/accessToken",machineId:"storage.serviceMachineId"}},n={clientId:process.env.KIMI_CODING_OAUTH_CLIENT_ID||"17e5f671-d194-4dfb-9706-5516cb48c098",deviceCodeUrl:"https://auth.kimi.com/api/oauth/device_authorization",tokenUrl:"https://auth.kimi.com/api/oauth/token"},o={apiBaseUrl:"https://api.kilo.ai",initiateUrl:"https://api.kilo.ai/api/device-auth/codes",pollUrlBase:"https://api.kilo.ai/api/device-auth/codes"},p={appBaseUrl:"https://app.cline.bot",apiBaseUrl:"https://api.cline.bot",authorizeUrl:"https://api.cline.bot/api/v1/auth/authorize",tokenExchangeUrl:"https://api.cline.bot/api/v1/auth/token",refreshUrl:"https://api.cline.bot/api/v1/auth/refresh"},q={defaultBaseUrl:"https://gitlab.com",authorizeUrlPath:"/oauth/authorize",tokenUrlPath:"/oauth/token",userInfoUrlPath:"/api/v4/user",scope:"api read_user",codeChallengeMethod:"S256"},r={baseUrl:"https://copilot.tencent.com",stateUrl:"https://copilot.tencent.com/v2/plugin/auth/state",tokenUrl:"https://copilot.tencent.com/v2/plugin/auth/token",refreshUrl:"https://copilot.tencent.com/v2/plugin/auth/token/refresh",userAgent:"CLI/2.63.2 CodeBuddy/2.63.2",platform:"CLI",pollInterval:5e3}}};
@@ -1 +0,0 @@
1
- "use strict";exports.id=7828,exports.ids=[7828],exports.modules={38160:(a,b,c)=>{c.d(b,{Ou:()=>i,eG:()=>h});var d=c(75924);let e=new Map,f=new Map,g=null;async function h(a,b){if(!a||!b)return null;let c=e.get(a);if(c&&Date.now()-c.fetchedAt<36e5)return c.projectId;if(f.has(a))return f.get(a).promise;let d=new AbortController,g=(async()=>{try{let c=await j(b,d.signal);if(c)return e.set(a,{projectId:c,fetchedAt:Date.now()}),c;return console.warn("[ProjectId] could not fetch projectId for connection",a.slice(0,8)),null}catch(a){return console.warn(`[ProjectId] Error fetching project ID: ${a.message}`),null}finally{f.delete(a)}})();return f.set(a,{promise:g,controller:d,startedAt:Date.now()}),g}function i(a){e.delete(a)}async function j(a,b){let c=await fetch(d.nZ.loadCodeAssist,{method:"POST",headers:{...d.Ic,Authorization:`Bearer ${a}`},body:JSON.stringify({metadata:d.zv}),signal:b});if(!c.ok){let a=await c.text().catch(()=>"");throw Error(`loadCodeAssist failed: HTTP ${c.status} ${a.slice(0,200)}`)}let e=await c.json(),f=function(a){if(!a)return null;if("string"==typeof a.cloudaicompanionProject){let b=a.cloudaicompanionProject.trim();if(b)return b}if(a.cloudaicompanionProject&&"object"==typeof a.cloudaicompanionProject){let b=a.cloudaicompanionProject.id;if("string"==typeof b&&b.trim())return b.trim()}return null}(e);if(f)return f;let g="legacy-tier";if(Array.isArray(e.allowedTiers)){for(let a of e.allowedTiers)if(a&&"object"==typeof a&&!0===a.isDefault&&a.id&&"string"==typeof a.id&&a.id.trim()){g=a.id.trim();break}}return k(a,g,b)}async function k(a,b,c){console.log(`[ProjectId] Onboarding user with tier: ${b}`);let e={tierId:b,metadata:d.zv};for(let b=1;b<=5&&!c?.aborted;b++){let f=new AbortController,g=setTimeout(()=>f.abort(),3e4),h=()=>f.abort();c?.addEventListener("abort",h);try{let c=await fetch(d.nZ.onboardUser,{method:"POST",headers:{...d.Ic,Authorization:`Bearer ${a}`},body:JSON.stringify(e),signal:f.signal});if(clearTimeout(g),!c.ok){let a=await c.text().catch(()=>"");throw Error(`onboardUser HTTP ${c.status}: ${a.slice(0,200)}`)}let h=await c.json();if(!0===h.done){let a=function(a){if(!a?.response)return null;let b=a.response.cloudaicompanionProject;if("string"==typeof b){let a=b.trim();if(a)return a}if(b&&"object"==typeof b){let a=b.id;if("string"==typeof a&&a.trim())return a.trim()}return null}(h);if(a)return console.log(`[ProjectId] Successfully onboarded, project ID: ${a}`),a;throw Error("onboardUser done but no project_id in response")}console.log(`[ProjectId] Onboard attempt ${b}/5: not done yet, waiting...`),await new Promise(a=>setTimeout(a,2e3))}catch(a){if(clearTimeout(g),"AbortError"===a.name){if(console.warn(`[ProjectId] onboardUser attempt ${b} aborted (timeout or connection removed)`),c?.aborted)return null;continue}if(5===b)return console.warn(`[ProjectId] onboardUser failed after 5 attempts: ${a.message}`),null;console.warn(`[ProjectId] onboardUser attempt ${b} failed: ${a.message}, retrying...`),await new Promise(a=>setTimeout(a,2e3))}finally{clearTimeout(g),c?.removeEventListener("abort",h)}}return null}g=setInterval(()=>{try{let a=Date.now();for(let[b,c]of e)(!c||a-c.fetchedAt>=36e5)&&e.delete(b);for(let[b,c]of f){if(!c||"number"!=typeof c.startedAt){f.delete(b);continue}if(a-c.startedAt>12e4){try{c.controller.abort()}catch(a){}f.delete(b)}}}catch(a){console.warn("[ProjectId] cleanup sweep error:",a?.message??a)}},6e5),g?.unref?.()},67828:(a,b,c)=>{c.d(b,{I9:()=>i,Ql:()=>j,eU:()=>m,vN:()=>l});var d=c(7803),e=c(89718),f=c(38160),g=c(8590);let h=g.oD,i=(a,b,c)=>(0,g.I9)(a,b,c,d),j=(a,b)=>(0,g.Ql)(a,b,d);function k(a){return new Date(Date.now()+1e3*a).toISOString()}async function l(a,b){try{let c={};b.accessToken&&(c.accessToken=b.accessToken),b.refreshToken&&(c.refreshToken=b.refreshToken),b.expiresIn&&(c.expiresAt=k(b.expiresIn),c.expiresIn=b.expiresIn),b.providerSpecificData&&(c.providerSpecificData={...b.existingProviderSpecificData||{},...b.providerSpecificData}),b.projectId&&(c.projectId=b.projectId);let f=await (0,e.updateProviderConnection)(a,c);return d.info("TOKEN_REFRESH","Credentials updated in localDb",{connectionId:a,success:!!f}),!!f}catch(b){return d.error("TOKEN_REFRESH","Error updating credentials in localDb",{connectionId:a,error:b.message}),!1}}async function m(a,b){let c={...b};if(c.expiresAt){let b=new Date(c.expiresAt).getTime()-Date.now(),e=(0,g.Og)(a);if(b<e){let h;d.info("TOKEN_REFRESH","Token expiring soon, refreshing proactively",{provider:a,expiresIn:Math.round(b/1e3),refreshLeadMs:e});let i=await (h=c,(0,g.iD)(a,h,d));if(i?.accessToken){let b={...i,existingProviderSpecificData:c.providerSpecificData};await l(c.connectionId,b),function(a,b,c){("antigravity"===a||"gemini-cli"===a)&&b&&c&&((0,f.Ou)(b),(0,f.eG)(b,c).then(a=>{a&&l(b,{projectId:a}).catch(a=>{d.debug("TOKEN_REFRESH","Failed to persist refreshed projectId",{connectionId:b,error:a?.message??a})})}).catch(a=>{d.debug("TOKEN_REFRESH","Failed to fetch projectId after token refresh",{connectionId:b,error:a?.message??a})}))}(a,(c={...c,accessToken:i.accessToken,refreshToken:i.refreshToken??c.refreshToken,providerSpecificData:i.providerSpecificData?{...c.providerSpecificData,...i.providerSpecificData}:c.providerSpecificData,expiresAt:i.expiresIn?k(i.expiresIn):c.expiresAt}).connectionId,c.accessToken)}}}if("github"===a&&c.providerSpecificData?.copilotTokenExpiresAt){let b=1e3*c.providerSpecificData.copilotTokenExpiresAt-Date.now();if(b<h){let e;d.info("TOKEN_REFRESH","Copilot token expiring soon, refreshing proactively",{provider:a,expiresIn:Math.round(b/1e3)});let f=await (e=c.accessToken,(0,g.jR)(e,d));if(f){let a={...c.providerSpecificData,copilotToken:f.token,copilotTokenExpiresAt:f.expiresAt};await l(c.connectionId,{providerSpecificData:a}),c.providerSpecificData=a,c.copilotToken=f.token}}}return c}}};