9router 0.3.90 → 0.3.96

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 (679) hide show
  1. package/app/.next/BUILD_ID +1 -1
  2. package/app/.next/app-path-routes-manifest.json +7 -6
  3. package/app/.next/build-manifest.json +2 -2
  4. package/app/.next/routes-manifest.json +6 -0
  5. package/app/.next/server/app/(dashboard)/dashboard/basic-chat/page.js +2 -2
  6. package/app/.next/server/app/(dashboard)/dashboard/basic-chat/page.js.nft.json +1 -1
  7. package/app/.next/server/app/(dashboard)/dashboard/basic-chat/page_client-reference-manifest.js +1 -1
  8. package/app/.next/server/app/(dashboard)/dashboard/cli-tools/page.js +2 -2
  9. package/app/.next/server/app/(dashboard)/dashboard/cli-tools/page.js.nft.json +1 -1
  10. package/app/.next/server/app/(dashboard)/dashboard/cli-tools/page_client-reference-manifest.js +1 -1
  11. package/app/.next/server/app/(dashboard)/dashboard/combos/page.js +2 -2
  12. package/app/.next/server/app/(dashboard)/dashboard/combos/page.js.nft.json +1 -1
  13. package/app/.next/server/app/(dashboard)/dashboard/combos/page_client-reference-manifest.js +1 -1
  14. package/app/.next/server/app/(dashboard)/dashboard/console-log/page.js +2 -2
  15. package/app/.next/server/app/(dashboard)/dashboard/console-log/page.js.nft.json +1 -1
  16. package/app/.next/server/app/(dashboard)/dashboard/console-log/page_client-reference-manifest.js +1 -1
  17. package/app/.next/server/app/(dashboard)/dashboard/endpoint/page.js +2 -2
  18. package/app/.next/server/app/(dashboard)/dashboard/endpoint/page.js.nft.json +1 -1
  19. package/app/.next/server/app/(dashboard)/dashboard/endpoint/page_client-reference-manifest.js +1 -1
  20. package/app/.next/server/app/(dashboard)/dashboard/media-providers/[kind]/[id]/page.js +34 -5
  21. package/app/.next/server/app/(dashboard)/dashboard/media-providers/[kind]/[id]/page.js.nft.json +1 -1
  22. package/app/.next/server/app/(dashboard)/dashboard/media-providers/[kind]/[id]/page_client-reference-manifest.js +1 -1
  23. package/app/.next/server/app/(dashboard)/dashboard/media-providers/[kind]/page.js +2 -2
  24. package/app/.next/server/app/(dashboard)/dashboard/media-providers/[kind]/page.js.nft.json +1 -1
  25. package/app/.next/server/app/(dashboard)/dashboard/media-providers/[kind]/page_client-reference-manifest.js +1 -1
  26. package/app/.next/server/app/(dashboard)/dashboard/mitm/page.js +2 -2
  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 +2 -2
  30. package/app/.next/server/app/(dashboard)/dashboard/page.js.nft.json +1 -1
  31. package/app/.next/server/app/(dashboard)/dashboard/page_client-reference-manifest.js +1 -1
  32. package/app/.next/server/app/(dashboard)/dashboard/profile/page.js +2 -2
  33. package/app/.next/server/app/(dashboard)/dashboard/profile/page.js.nft.json +1 -1
  34. package/app/.next/server/app/(dashboard)/dashboard/profile/page_client-reference-manifest.js +1 -1
  35. package/app/.next/server/app/(dashboard)/dashboard/providers/[id]/page.js +2 -2
  36. package/app/.next/server/app/(dashboard)/dashboard/providers/[id]/page.js.nft.json +1 -1
  37. package/app/.next/server/app/(dashboard)/dashboard/providers/[id]/page_client-reference-manifest.js +1 -1
  38. package/app/.next/server/app/(dashboard)/dashboard/providers/new/page.js +2 -2
  39. package/app/.next/server/app/(dashboard)/dashboard/providers/new/page.js.nft.json +1 -1
  40. package/app/.next/server/app/(dashboard)/dashboard/providers/new/page_client-reference-manifest.js +1 -1
  41. package/app/.next/server/app/(dashboard)/dashboard/providers/page.js +2 -2
  42. package/app/.next/server/app/(dashboard)/dashboard/providers/page.js.nft.json +1 -1
  43. package/app/.next/server/app/(dashboard)/dashboard/providers/page_client-reference-manifest.js +1 -1
  44. package/app/.next/server/app/(dashboard)/dashboard/proxy-pools/page.js +3 -3
  45. package/app/.next/server/app/(dashboard)/dashboard/proxy-pools/page.js.nft.json +1 -1
  46. package/app/.next/server/app/(dashboard)/dashboard/proxy-pools/page_client-reference-manifest.js +1 -1
  47. package/app/.next/server/app/(dashboard)/dashboard/quota/page.js +2 -2
  48. package/app/.next/server/app/(dashboard)/dashboard/quota/page.js.nft.json +1 -1
  49. package/app/.next/server/app/(dashboard)/dashboard/quota/page_client-reference-manifest.js +1 -1
  50. package/app/.next/server/app/(dashboard)/dashboard/translator/page.js +2 -2
  51. package/app/.next/server/app/(dashboard)/dashboard/translator/page.js.nft.json +1 -1
  52. package/app/.next/server/app/(dashboard)/dashboard/translator/page_client-reference-manifest.js +1 -1
  53. package/app/.next/server/app/(dashboard)/dashboard/usage/page.js +2 -2
  54. package/app/.next/server/app/(dashboard)/dashboard/usage/page.js.nft.json +1 -1
  55. package/app/.next/server/app/(dashboard)/dashboard/usage/page_client-reference-manifest.js +1 -1
  56. package/app/.next/server/app/_global-error/page.js +2 -2
  57. package/app/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
  58. package/app/.next/server/app/_global-error.html +1 -1
  59. package/app/.next/server/app/_global-error.rsc +1 -1
  60. package/app/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  61. package/app/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  62. package/app/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  63. package/app/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  64. package/app/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  65. package/app/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  66. package/app/.next/server/app/_not-found/page.js +2 -2
  67. package/app/.next/server/app/_not-found/page.js.nft.json +1 -1
  68. package/app/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  69. package/app/.next/server/app/_not-found.html +1 -1
  70. package/app/.next/server/app/_not-found.rsc +4 -4
  71. package/app/.next/server/app/_not-found.segments/_full.segment.rsc +4 -4
  72. package/app/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  73. package/app/.next/server/app/_not-found.segments/_index.segment.rsc +4 -4
  74. package/app/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  75. package/app/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  76. package/app/.next/server/app/_not-found.segments/_tree.segment.rsc +2 -2
  77. package/app/.next/server/app/api/auth/login/route.js +1 -1
  78. package/app/.next/server/app/api/auth/logout/route.js +1 -1
  79. package/app/.next/server/app/api/cli-tools/antigravity-mitm/alias/route.js +2 -2
  80. package/app/.next/server/app/api/cli-tools/antigravity-mitm/alias/route.js.nft.json +1 -1
  81. package/app/.next/server/app/api/cli-tools/antigravity-mitm/route.js +1 -1
  82. package/app/.next/server/app/api/cli-tools/antigravity-mitm/route.js.nft.json +1 -1
  83. package/app/.next/server/app/api/cli-tools/claude-settings/route.js +2 -2
  84. package/app/.next/server/app/api/cli-tools/codex-settings/route.js +2 -2
  85. package/app/.next/server/app/api/cli-tools/copilot-settings/route.js +1 -1
  86. package/app/.next/server/app/api/cli-tools/droid-settings/route.js +2 -2
  87. package/app/.next/server/app/api/cli-tools/openclaw-settings/route.js +1 -1
  88. package/app/.next/server/app/api/cli-tools/opencode-settings/route.js +2 -2
  89. package/app/.next/server/app/api/cloud/auth/route.js +1 -1
  90. package/app/.next/server/app/api/cloud/credentials/update/route.js +1 -1
  91. package/app/.next/server/app/api/cloud/model/resolve/route.js +1 -1
  92. package/app/.next/server/app/api/cloud/models/alias/route.js +1 -1
  93. package/app/.next/server/app/api/combos/[id]/route.js +1 -1
  94. package/app/.next/server/app/api/combos/route.js +1 -1
  95. package/app/.next/server/app/api/health/route.js +1 -1
  96. package/app/.next/server/app/api/init/route.js +1 -1
  97. package/app/.next/server/app/api/init/route.js.nft.json +1 -1
  98. package/app/.next/server/app/api/keys/[id]/route.js +1 -1
  99. package/app/.next/server/app/api/keys/route.js +1 -1
  100. package/app/.next/server/app/api/locale/route.js +1 -1
  101. package/app/.next/server/app/api/media-providers/tts/elevenlabs/voices/route.js +1 -1
  102. package/app/.next/server/app/api/media-providers/tts/elevenlabs/voices/route.js.nft.json +1 -1
  103. package/app/.next/server/app/api/media-providers/tts/voices/route.js +1 -1
  104. package/app/.next/server/app/api/media-providers/tts/voices/route.js.nft.json +1 -1
  105. package/app/.next/server/app/api/models/alias/route.js +1 -1
  106. package/app/.next/server/app/api/models/availability/route.js +1 -1
  107. package/app/.next/server/app/api/models/route.js +1 -1
  108. package/app/.next/server/app/api/models/route.js.nft.json +1 -1
  109. package/app/.next/server/app/api/models/test/route.js +1 -1
  110. package/app/.next/server/app/api/oauth/[provider]/[action]/route.js +1 -1
  111. package/app/.next/server/app/api/oauth/[provider]/[action]/route.js.nft.json +1 -1
  112. package/app/.next/server/app/api/oauth/cursor/auto-import/route.js +2 -2
  113. package/app/.next/server/app/api/oauth/cursor/import/route.js +1 -1
  114. package/app/.next/server/app/api/oauth/gitlab/pat/route.js +1 -1
  115. package/app/.next/server/app/api/oauth/iflow/cookie/route.js +1 -1
  116. package/app/.next/server/app/api/oauth/kiro/auto-import/route.js +1 -1
  117. package/app/.next/server/app/api/oauth/kiro/import/route.js +1 -1
  118. package/app/.next/server/app/api/oauth/kiro/import/route.js.nft.json +1 -1
  119. package/app/.next/server/app/api/oauth/kiro/social-authorize/route.js +1 -1
  120. package/app/.next/server/app/api/oauth/kiro/social-authorize/route.js.nft.json +1 -1
  121. package/app/.next/server/app/api/oauth/kiro/social-exchange/route.js +1 -1
  122. package/app/.next/server/app/api/oauth/kiro/social-exchange/route.js.nft.json +1 -1
  123. package/app/.next/server/app/api/pricing/route.js +1 -1
  124. package/app/.next/server/app/api/provider-nodes/[id]/route.js +1 -1
  125. package/app/.next/server/app/api/provider-nodes/route.js +1 -1
  126. package/app/.next/server/app/api/provider-nodes/route.js.nft.json +1 -1
  127. package/app/.next/server/app/api/provider-nodes/validate/route.js +1 -1
  128. package/app/.next/server/app/api/providers/[id]/models/route.js +1 -1
  129. package/app/.next/server/app/api/providers/[id]/models/route.js.nft.json +1 -1
  130. package/app/.next/server/app/api/providers/[id]/route.js +1 -1
  131. package/app/.next/server/app/api/providers/[id]/test/route.js +1 -1
  132. package/app/.next/server/app/api/providers/[id]/test/route.js.nft.json +1 -1
  133. package/app/.next/server/app/api/providers/[id]/test-models/route.js +1 -1
  134. package/app/.next/server/app/api/providers/[id]/test-models/route.js.nft.json +1 -1
  135. package/app/.next/server/app/api/providers/client/route.js +1 -1
  136. package/app/.next/server/app/api/providers/kilo/free-models/route.js +1 -1
  137. package/app/.next/server/app/api/providers/route.js +1 -1
  138. package/app/.next/server/app/api/providers/route.js.nft.json +1 -1
  139. package/app/.next/server/app/api/providers/suggested-models/route.js +1 -0
  140. package/app/.next/server/app/api/providers/suggested-models/route.js.nft.json +1 -0
  141. package/app/.next/server/app/api/providers/suggested-models/route_client-reference-manifest.js +1 -0
  142. package/app/.next/server/app/api/providers/test-batch/route.js +1 -1
  143. package/app/.next/server/app/api/providers/test-batch/route.js.nft.json +1 -1
  144. package/app/.next/server/app/api/providers/validate/route.js +1 -1
  145. package/app/.next/server/app/api/providers/validate/route.js.nft.json +1 -1
  146. package/app/.next/server/app/api/proxy-pools/[id]/route.js +1 -1
  147. package/app/.next/server/app/api/proxy-pools/[id]/test/route.js +1 -1
  148. package/app/.next/server/app/api/proxy-pools/route.js +1 -1
  149. package/app/.next/server/app/api/proxy-pools/vercel-deploy/route.js +2 -2
  150. package/app/.next/server/app/api/settings/database/route.js +1 -1
  151. package/app/.next/server/app/api/settings/proxy-test/route.js +1 -1
  152. package/app/.next/server/app/api/settings/require-login/route.js +1 -1
  153. package/app/.next/server/app/api/settings/route.js +1 -1
  154. package/app/.next/server/app/api/shutdown/route.js +1 -1
  155. package/app/.next/server/app/api/tags/route.js +1 -1
  156. package/app/.next/server/app/api/translator/console-logs/route.js +1 -1
  157. package/app/.next/server/app/api/translator/console-logs/route.js.nft.json +1 -1
  158. package/app/.next/server/app/api/translator/console-logs/stream/route.js +2 -2
  159. package/app/.next/server/app/api/translator/console-logs/stream/route.js.nft.json +1 -1
  160. package/app/.next/server/app/api/translator/load/route.js +1 -1
  161. package/app/.next/server/app/api/translator/save/route.js +1 -1
  162. package/app/.next/server/app/api/translator/send/route.js +1 -1
  163. package/app/.next/server/app/api/translator/send/route.js.nft.json +1 -1
  164. package/app/.next/server/app/api/translator/translate/route.js +1 -1
  165. package/app/.next/server/app/api/tunnel/disable/route.js +1 -1
  166. package/app/.next/server/app/api/tunnel/disable/route.js.nft.json +1 -1
  167. package/app/.next/server/app/api/tunnel/enable/route.js +1 -1
  168. package/app/.next/server/app/api/tunnel/enable/route.js.nft.json +1 -1
  169. package/app/.next/server/app/api/tunnel/status/route.js +1 -1
  170. package/app/.next/server/app/api/tunnel/status/route.js.nft.json +1 -1
  171. package/app/.next/server/app/api/tunnel/tailscale-check/route.js +2 -2
  172. package/app/.next/server/app/api/tunnel/tailscale-disable/route.js +1 -1
  173. package/app/.next/server/app/api/tunnel/tailscale-disable/route.js.nft.json +1 -1
  174. package/app/.next/server/app/api/tunnel/tailscale-enable/route.js +1 -1
  175. package/app/.next/server/app/api/tunnel/tailscale-enable/route.js.nft.json +1 -1
  176. package/app/.next/server/app/api/tunnel/tailscale-install/route.js +3 -3
  177. package/app/.next/server/app/api/tunnel/tailscale-install/route.js.nft.json +1 -1
  178. package/app/.next/server/app/api/tunnel/tailscale-login/route.js +2 -2
  179. package/app/.next/server/app/api/tunnel/tailscale-start-daemon/route.js +1 -1
  180. package/app/.next/server/app/api/tunnel/tailscale-start-daemon/route.js.nft.json +1 -1
  181. package/app/.next/server/app/api/usage/[connectionId]/route.js +1 -1
  182. package/app/.next/server/app/api/usage/[connectionId]/route.js.nft.json +1 -1
  183. package/app/.next/server/app/api/usage/chart/route.js +1 -1
  184. package/app/.next/server/app/api/usage/history/route.js +1 -1
  185. package/app/.next/server/app/api/usage/providers/route.js +1 -1
  186. package/app/.next/server/app/api/usage/providers/route.js.nft.json +1 -1
  187. package/app/.next/server/app/api/usage/request-details/route.js +1 -1
  188. package/app/.next/server/app/api/usage/request-logs/route.js +1 -1
  189. package/app/.next/server/app/api/usage/stats/route.js +1 -1
  190. package/app/.next/server/app/api/usage/stream/route.js +2 -2
  191. package/app/.next/server/app/api/v1/api/chat/route.js +1 -1
  192. package/app/.next/server/app/api/v1/api/chat/route.js.nft.json +1 -1
  193. package/app/.next/server/app/api/v1/audio/speech/route.js +1 -1
  194. package/app/.next/server/app/api/v1/audio/speech/route.js.nft.json +1 -1
  195. package/app/.next/server/app/api/v1/chat/completions/route.js +1 -1
  196. package/app/.next/server/app/api/v1/chat/completions/route.js.nft.json +1 -1
  197. package/app/.next/server/app/api/v1/embeddings/route.js +1 -1
  198. package/app/.next/server/app/api/v1/embeddings/route.js.nft.json +1 -1
  199. package/app/.next/server/app/api/v1/messages/count_tokens/route.js +1 -1
  200. package/app/.next/server/app/api/v1/messages/route.js +1 -1
  201. package/app/.next/server/app/api/v1/messages/route.js.nft.json +1 -1
  202. package/app/.next/server/app/api/v1/models/route.js +1 -1
  203. package/app/.next/server/app/api/v1/models/route.js.nft.json +1 -1
  204. package/app/.next/server/app/api/v1/responses/compact/route.js +1 -1
  205. package/app/.next/server/app/api/v1/responses/compact/route.js.nft.json +1 -1
  206. package/app/.next/server/app/api/v1/responses/route.js +1 -1
  207. package/app/.next/server/app/api/v1/responses/route.js.nft.json +1 -1
  208. package/app/.next/server/app/api/v1/route.js +1 -1
  209. package/app/.next/server/app/api/v1beta/models/[...path]/route.js +1 -1
  210. package/app/.next/server/app/api/v1beta/models/[...path]/route.js.nft.json +1 -1
  211. package/app/.next/server/app/api/v1beta/models/route.js +1 -1
  212. package/app/.next/server/app/api/v1beta/models/route.js.nft.json +1 -1
  213. package/app/.next/server/app/api/version/route.js +1 -1
  214. package/app/.next/server/app/callback/page.js +2 -2
  215. package/app/.next/server/app/callback/page.js.nft.json +1 -1
  216. package/app/.next/server/app/callback/page_client-reference-manifest.js +1 -1
  217. package/app/.next/server/app/callback.html +1 -1
  218. package/app/.next/server/app/callback.rsc +4 -4
  219. package/app/.next/server/app/callback.segments/_full.segment.rsc +4 -4
  220. package/app/.next/server/app/callback.segments/_head.segment.rsc +1 -1
  221. package/app/.next/server/app/callback.segments/_index.segment.rsc +4 -4
  222. package/app/.next/server/app/callback.segments/_tree.segment.rsc +2 -2
  223. package/app/.next/server/app/callback.segments/callback/__PAGE__.segment.rsc +1 -1
  224. package/app/.next/server/app/callback.segments/callback.segment.rsc +1 -1
  225. package/app/.next/server/app/dashboard/basic-chat.html +1 -1
  226. package/app/.next/server/app/dashboard/basic-chat.rsc +6 -6
  227. package/app/.next/server/app/dashboard/basic-chat.segments/!KGRhc2hib2FyZCk/dashboard/basic-chat/__PAGE__.segment.rsc +2 -2
  228. package/app/.next/server/app/dashboard/basic-chat.segments/!KGRhc2hib2FyZCk/dashboard/basic-chat.segment.rsc +1 -1
  229. package/app/.next/server/app/dashboard/basic-chat.segments/!KGRhc2hib2FyZCk/dashboard.segment.rsc +1 -1
  230. package/app/.next/server/app/dashboard/basic-chat.segments/!KGRhc2hib2FyZCk.segment.rsc +2 -2
  231. package/app/.next/server/app/dashboard/basic-chat.segments/_full.segment.rsc +6 -6
  232. package/app/.next/server/app/dashboard/basic-chat.segments/_head.segment.rsc +1 -1
  233. package/app/.next/server/app/dashboard/basic-chat.segments/_index.segment.rsc +4 -4
  234. package/app/.next/server/app/dashboard/basic-chat.segments/_tree.segment.rsc +2 -2
  235. package/app/.next/server/app/dashboard/cli-tools.html +1 -1
  236. package/app/.next/server/app/dashboard/cli-tools.rsc +6 -6
  237. package/app/.next/server/app/dashboard/cli-tools.segments/!KGRhc2hib2FyZCk/dashboard/cli-tools/__PAGE__.segment.rsc +2 -2
  238. package/app/.next/server/app/dashboard/cli-tools.segments/!KGRhc2hib2FyZCk/dashboard/cli-tools.segment.rsc +1 -1
  239. package/app/.next/server/app/dashboard/cli-tools.segments/!KGRhc2hib2FyZCk/dashboard.segment.rsc +1 -1
  240. package/app/.next/server/app/dashboard/cli-tools.segments/!KGRhc2hib2FyZCk.segment.rsc +2 -2
  241. package/app/.next/server/app/dashboard/cli-tools.segments/_full.segment.rsc +6 -6
  242. package/app/.next/server/app/dashboard/cli-tools.segments/_head.segment.rsc +1 -1
  243. package/app/.next/server/app/dashboard/cli-tools.segments/_index.segment.rsc +4 -4
  244. package/app/.next/server/app/dashboard/cli-tools.segments/_tree.segment.rsc +2 -2
  245. package/app/.next/server/app/dashboard/combos.html +1 -1
  246. package/app/.next/server/app/dashboard/combos.rsc +6 -6
  247. package/app/.next/server/app/dashboard/combos.segments/!KGRhc2hib2FyZCk/dashboard/combos/__PAGE__.segment.rsc +2 -2
  248. package/app/.next/server/app/dashboard/combos.segments/!KGRhc2hib2FyZCk/dashboard/combos.segment.rsc +1 -1
  249. package/app/.next/server/app/dashboard/combos.segments/!KGRhc2hib2FyZCk/dashboard.segment.rsc +1 -1
  250. package/app/.next/server/app/dashboard/combos.segments/!KGRhc2hib2FyZCk.segment.rsc +2 -2
  251. package/app/.next/server/app/dashboard/combos.segments/_full.segment.rsc +6 -6
  252. package/app/.next/server/app/dashboard/combos.segments/_head.segment.rsc +1 -1
  253. package/app/.next/server/app/dashboard/combos.segments/_index.segment.rsc +4 -4
  254. package/app/.next/server/app/dashboard/combos.segments/_tree.segment.rsc +2 -2
  255. package/app/.next/server/app/dashboard/endpoint.html +1 -1
  256. package/app/.next/server/app/dashboard/endpoint.rsc +6 -6
  257. package/app/.next/server/app/dashboard/endpoint.segments/!KGRhc2hib2FyZCk/dashboard/endpoint/__PAGE__.segment.rsc +2 -2
  258. package/app/.next/server/app/dashboard/endpoint.segments/!KGRhc2hib2FyZCk/dashboard/endpoint.segment.rsc +1 -1
  259. package/app/.next/server/app/dashboard/endpoint.segments/!KGRhc2hib2FyZCk/dashboard.segment.rsc +1 -1
  260. package/app/.next/server/app/dashboard/endpoint.segments/!KGRhc2hib2FyZCk.segment.rsc +2 -2
  261. package/app/.next/server/app/dashboard/endpoint.segments/_full.segment.rsc +6 -6
  262. package/app/.next/server/app/dashboard/endpoint.segments/_head.segment.rsc +1 -1
  263. package/app/.next/server/app/dashboard/endpoint.segments/_index.segment.rsc +4 -4
  264. package/app/.next/server/app/dashboard/endpoint.segments/_tree.segment.rsc +2 -2
  265. package/app/.next/server/app/dashboard/mitm.html +1 -1
  266. package/app/.next/server/app/dashboard/mitm.rsc +6 -6
  267. package/app/.next/server/app/dashboard/mitm.segments/!KGRhc2hib2FyZCk/dashboard/mitm/__PAGE__.segment.rsc +2 -2
  268. package/app/.next/server/app/dashboard/mitm.segments/!KGRhc2hib2FyZCk/dashboard/mitm.segment.rsc +1 -1
  269. package/app/.next/server/app/dashboard/mitm.segments/!KGRhc2hib2FyZCk/dashboard.segment.rsc +1 -1
  270. package/app/.next/server/app/dashboard/mitm.segments/!KGRhc2hib2FyZCk.segment.rsc +2 -2
  271. package/app/.next/server/app/dashboard/mitm.segments/_full.segment.rsc +6 -6
  272. package/app/.next/server/app/dashboard/mitm.segments/_head.segment.rsc +1 -1
  273. package/app/.next/server/app/dashboard/mitm.segments/_index.segment.rsc +4 -4
  274. package/app/.next/server/app/dashboard/mitm.segments/_tree.segment.rsc +2 -2
  275. package/app/.next/server/app/dashboard/profile.html +1 -1
  276. package/app/.next/server/app/dashboard/profile.rsc +6 -6
  277. package/app/.next/server/app/dashboard/profile.segments/!KGRhc2hib2FyZCk/dashboard/profile/__PAGE__.segment.rsc +2 -2
  278. package/app/.next/server/app/dashboard/profile.segments/!KGRhc2hib2FyZCk/dashboard/profile.segment.rsc +1 -1
  279. package/app/.next/server/app/dashboard/profile.segments/!KGRhc2hib2FyZCk/dashboard.segment.rsc +1 -1
  280. package/app/.next/server/app/dashboard/profile.segments/!KGRhc2hib2FyZCk.segment.rsc +2 -2
  281. package/app/.next/server/app/dashboard/profile.segments/_full.segment.rsc +6 -6
  282. package/app/.next/server/app/dashboard/profile.segments/_head.segment.rsc +1 -1
  283. package/app/.next/server/app/dashboard/profile.segments/_index.segment.rsc +4 -4
  284. package/app/.next/server/app/dashboard/profile.segments/_tree.segment.rsc +2 -2
  285. package/app/.next/server/app/dashboard/providers/new.html +1 -1
  286. package/app/.next/server/app/dashboard/providers/new.rsc +6 -6
  287. package/app/.next/server/app/dashboard/providers/new.segments/!KGRhc2hib2FyZCk/dashboard/providers/new/__PAGE__.segment.rsc +2 -2
  288. package/app/.next/server/app/dashboard/providers/new.segments/!KGRhc2hib2FyZCk/dashboard/providers/new.segment.rsc +1 -1
  289. package/app/.next/server/app/dashboard/providers/new.segments/!KGRhc2hib2FyZCk/dashboard/providers.segment.rsc +1 -1
  290. package/app/.next/server/app/dashboard/providers/new.segments/!KGRhc2hib2FyZCk/dashboard.segment.rsc +1 -1
  291. package/app/.next/server/app/dashboard/providers/new.segments/!KGRhc2hib2FyZCk.segment.rsc +2 -2
  292. package/app/.next/server/app/dashboard/providers/new.segments/_full.segment.rsc +6 -6
  293. package/app/.next/server/app/dashboard/providers/new.segments/_head.segment.rsc +1 -1
  294. package/app/.next/server/app/dashboard/providers/new.segments/_index.segment.rsc +4 -4
  295. package/app/.next/server/app/dashboard/providers/new.segments/_tree.segment.rsc +2 -2
  296. package/app/.next/server/app/dashboard/providers.html +1 -1
  297. package/app/.next/server/app/dashboard/providers.rsc +6 -6
  298. package/app/.next/server/app/dashboard/providers.segments/!KGRhc2hib2FyZCk/dashboard/providers/__PAGE__.segment.rsc +2 -2
  299. package/app/.next/server/app/dashboard/providers.segments/!KGRhc2hib2FyZCk/dashboard/providers.segment.rsc +1 -1
  300. package/app/.next/server/app/dashboard/providers.segments/!KGRhc2hib2FyZCk/dashboard.segment.rsc +1 -1
  301. package/app/.next/server/app/dashboard/providers.segments/!KGRhc2hib2FyZCk.segment.rsc +2 -2
  302. package/app/.next/server/app/dashboard/providers.segments/_full.segment.rsc +6 -6
  303. package/app/.next/server/app/dashboard/providers.segments/_head.segment.rsc +1 -1
  304. package/app/.next/server/app/dashboard/providers.segments/_index.segment.rsc +4 -4
  305. package/app/.next/server/app/dashboard/providers.segments/_tree.segment.rsc +2 -2
  306. package/app/.next/server/app/dashboard/proxy-pools.html +1 -1
  307. package/app/.next/server/app/dashboard/proxy-pools.rsc +6 -6
  308. package/app/.next/server/app/dashboard/proxy-pools.segments/!KGRhc2hib2FyZCk/dashboard/proxy-pools/__PAGE__.segment.rsc +2 -2
  309. package/app/.next/server/app/dashboard/proxy-pools.segments/!KGRhc2hib2FyZCk/dashboard/proxy-pools.segment.rsc +1 -1
  310. package/app/.next/server/app/dashboard/proxy-pools.segments/!KGRhc2hib2FyZCk/dashboard.segment.rsc +1 -1
  311. package/app/.next/server/app/dashboard/proxy-pools.segments/!KGRhc2hib2FyZCk.segment.rsc +2 -2
  312. package/app/.next/server/app/dashboard/proxy-pools.segments/_full.segment.rsc +6 -6
  313. package/app/.next/server/app/dashboard/proxy-pools.segments/_head.segment.rsc +1 -1
  314. package/app/.next/server/app/dashboard/proxy-pools.segments/_index.segment.rsc +4 -4
  315. package/app/.next/server/app/dashboard/proxy-pools.segments/_tree.segment.rsc +2 -2
  316. package/app/.next/server/app/dashboard/quota.html +2 -2
  317. package/app/.next/server/app/dashboard/quota.rsc +7 -7
  318. package/app/.next/server/app/dashboard/quota.segments/!KGRhc2hib2FyZCk/dashboard/quota/__PAGE__.segment.rsc +3 -3
  319. package/app/.next/server/app/dashboard/quota.segments/!KGRhc2hib2FyZCk/dashboard/quota.segment.rsc +1 -1
  320. package/app/.next/server/app/dashboard/quota.segments/!KGRhc2hib2FyZCk/dashboard.segment.rsc +1 -1
  321. package/app/.next/server/app/dashboard/quota.segments/!KGRhc2hib2FyZCk.segment.rsc +2 -2
  322. package/app/.next/server/app/dashboard/quota.segments/_full.segment.rsc +7 -7
  323. package/app/.next/server/app/dashboard/quota.segments/_head.segment.rsc +1 -1
  324. package/app/.next/server/app/dashboard/quota.segments/_index.segment.rsc +4 -4
  325. package/app/.next/server/app/dashboard/quota.segments/_tree.segment.rsc +2 -2
  326. package/app/.next/server/app/dashboard/settings/pricing/page.js +2 -2
  327. package/app/.next/server/app/dashboard/settings/pricing/page.js.nft.json +1 -1
  328. package/app/.next/server/app/dashboard/settings/pricing/page_client-reference-manifest.js +1 -1
  329. package/app/.next/server/app/dashboard/settings/pricing.html +1 -1
  330. package/app/.next/server/app/dashboard/settings/pricing.rsc +4 -4
  331. package/app/.next/server/app/dashboard/settings/pricing.segments/_full.segment.rsc +4 -4
  332. package/app/.next/server/app/dashboard/settings/pricing.segments/_head.segment.rsc +1 -1
  333. package/app/.next/server/app/dashboard/settings/pricing.segments/_index.segment.rsc +4 -4
  334. package/app/.next/server/app/dashboard/settings/pricing.segments/_tree.segment.rsc +2 -2
  335. package/app/.next/server/app/dashboard/settings/pricing.segments/dashboard/settings/pricing/__PAGE__.segment.rsc +1 -1
  336. package/app/.next/server/app/dashboard/settings/pricing.segments/dashboard/settings/pricing.segment.rsc +1 -1
  337. package/app/.next/server/app/dashboard/settings/pricing.segments/dashboard/settings.segment.rsc +1 -1
  338. package/app/.next/server/app/dashboard/settings/pricing.segments/dashboard.segment.rsc +1 -1
  339. package/app/.next/server/app/dashboard/translator.html +1 -1
  340. package/app/.next/server/app/dashboard/translator.rsc +6 -6
  341. package/app/.next/server/app/dashboard/translator.segments/!KGRhc2hib2FyZCk/dashboard/translator/__PAGE__.segment.rsc +2 -2
  342. package/app/.next/server/app/dashboard/translator.segments/!KGRhc2hib2FyZCk/dashboard/translator.segment.rsc +1 -1
  343. package/app/.next/server/app/dashboard/translator.segments/!KGRhc2hib2FyZCk/dashboard.segment.rsc +1 -1
  344. package/app/.next/server/app/dashboard/translator.segments/!KGRhc2hib2FyZCk.segment.rsc +2 -2
  345. package/app/.next/server/app/dashboard/translator.segments/_full.segment.rsc +6 -6
  346. package/app/.next/server/app/dashboard/translator.segments/_head.segment.rsc +1 -1
  347. package/app/.next/server/app/dashboard/translator.segments/_index.segment.rsc +4 -4
  348. package/app/.next/server/app/dashboard/translator.segments/_tree.segment.rsc +2 -2
  349. package/app/.next/server/app/dashboard/usage.html +1 -1
  350. package/app/.next/server/app/dashboard/usage.rsc +6 -6
  351. package/app/.next/server/app/dashboard/usage.segments/!KGRhc2hib2FyZCk/dashboard/usage/__PAGE__.segment.rsc +2 -2
  352. package/app/.next/server/app/dashboard/usage.segments/!KGRhc2hib2FyZCk/dashboard/usage.segment.rsc +1 -1
  353. package/app/.next/server/app/dashboard/usage.segments/!KGRhc2hib2FyZCk/dashboard.segment.rsc +1 -1
  354. package/app/.next/server/app/dashboard/usage.segments/!KGRhc2hib2FyZCk.segment.rsc +2 -2
  355. package/app/.next/server/app/dashboard/usage.segments/_full.segment.rsc +6 -6
  356. package/app/.next/server/app/dashboard/usage.segments/_head.segment.rsc +1 -1
  357. package/app/.next/server/app/dashboard/usage.segments/_index.segment.rsc +4 -4
  358. package/app/.next/server/app/dashboard/usage.segments/_tree.segment.rsc +2 -2
  359. package/app/.next/server/app/dashboard.html +1 -1
  360. package/app/.next/server/app/dashboard.rsc +6 -6
  361. package/app/.next/server/app/dashboard.segments/!KGRhc2hib2FyZCk/dashboard/__PAGE__.segment.rsc +2 -2
  362. package/app/.next/server/app/dashboard.segments/!KGRhc2hib2FyZCk/dashboard.segment.rsc +1 -1
  363. package/app/.next/server/app/dashboard.segments/!KGRhc2hib2FyZCk.segment.rsc +2 -2
  364. package/app/.next/server/app/dashboard.segments/_full.segment.rsc +6 -6
  365. package/app/.next/server/app/dashboard.segments/_head.segment.rsc +1 -1
  366. package/app/.next/server/app/dashboard.segments/_index.segment.rsc +4 -4
  367. package/app/.next/server/app/dashboard.segments/_tree.segment.rsc +2 -2
  368. package/app/.next/server/app/favicon.ico/route.js +1 -1
  369. package/app/.next/server/app/index.html +1 -1
  370. package/app/.next/server/app/index.rsc +4 -4
  371. package/app/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  372. package/app/.next/server/app/index.segments/_full.segment.rsc +4 -4
  373. package/app/.next/server/app/index.segments/_head.segment.rsc +1 -1
  374. package/app/.next/server/app/index.segments/_index.segment.rsc +4 -4
  375. package/app/.next/server/app/index.segments/_tree.segment.rsc +2 -2
  376. package/app/.next/server/app/landing/page.js +2 -2
  377. package/app/.next/server/app/landing/page.js.nft.json +1 -1
  378. package/app/.next/server/app/landing/page_client-reference-manifest.js +1 -1
  379. package/app/.next/server/app/landing.html +1 -1
  380. package/app/.next/server/app/landing.rsc +4 -4
  381. package/app/.next/server/app/landing.segments/_full.segment.rsc +4 -4
  382. package/app/.next/server/app/landing.segments/_head.segment.rsc +1 -1
  383. package/app/.next/server/app/landing.segments/_index.segment.rsc +4 -4
  384. package/app/.next/server/app/landing.segments/_tree.segment.rsc +2 -2
  385. package/app/.next/server/app/landing.segments/landing/__PAGE__.segment.rsc +1 -1
  386. package/app/.next/server/app/landing.segments/landing.segment.rsc +1 -1
  387. package/app/.next/server/app/login/page.js +2 -2
  388. package/app/.next/server/app/login/page.js.nft.json +1 -1
  389. package/app/.next/server/app/login/page_client-reference-manifest.js +1 -1
  390. package/app/.next/server/app/login.html +1 -1
  391. package/app/.next/server/app/login.rsc +5 -5
  392. package/app/.next/server/app/login.segments/_full.segment.rsc +5 -5
  393. package/app/.next/server/app/login.segments/_head.segment.rsc +1 -1
  394. package/app/.next/server/app/login.segments/_index.segment.rsc +4 -4
  395. package/app/.next/server/app/login.segments/_tree.segment.rsc +2 -2
  396. package/app/.next/server/app/login.segments/login/__PAGE__.segment.rsc +2 -2
  397. package/app/.next/server/app/login.segments/login.segment.rsc +1 -1
  398. package/app/.next/server/app/manifest.webmanifest/route.js +2 -2
  399. package/app/.next/server/app/page.js +2 -2
  400. package/app/.next/server/app/page.js.nft.json +1 -1
  401. package/app/.next/server/app/page_client-reference-manifest.js +1 -1
  402. package/app/.next/server/app-paths-manifest.json +7 -6
  403. package/app/.next/server/chunks/126.js +2 -2
  404. package/app/.next/server/chunks/1389.js +1 -0
  405. package/app/.next/server/chunks/1574.js +1 -1
  406. package/app/.next/server/chunks/2049.js +1 -1
  407. package/app/.next/server/chunks/{242.js → 253.js} +1 -1
  408. package/app/.next/server/chunks/2947.js +1 -0
  409. package/app/.next/server/chunks/3030.js +1 -0
  410. package/app/.next/server/chunks/3159.js +1 -1
  411. package/app/.next/server/chunks/3774.js +3 -3
  412. package/app/.next/server/chunks/450.js +2 -2
  413. package/app/.next/server/chunks/{6560.js → 4627.js} +1 -1
  414. package/app/.next/server/chunks/5053.js +64 -0
  415. package/app/.next/server/chunks/5445.js +1 -0
  416. package/app/.next/server/chunks/6379.js +2 -2
  417. package/app/.next/server/chunks/7770.js +1 -1
  418. package/app/.next/server/chunks/7973.js +1 -1
  419. package/app/.next/server/chunks/8035.js +1 -0
  420. package/app/.next/server/chunks/8202.js +6 -6
  421. package/app/.next/server/chunks/8480.js +2 -2
  422. package/app/.next/server/chunks/8513.js +5 -0
  423. package/app/.next/server/chunks/9737.js +1 -0
  424. package/app/.next/server/middleware-build-manifest.js +1 -1
  425. package/app/.next/server/middleware.js +3 -3
  426. package/app/.next/server/pages/404.html +1 -1
  427. package/app/.next/server/pages/500.html +1 -1
  428. package/app/.next/static/VB9M_Cg8qFTiJ9Qceuwx7/_buildManifest.js +1 -0
  429. package/app/.next/static/chunks/1237-a2c937558839656b.js +18 -0
  430. package/app/.next/static/chunks/4156-ef6abcd3360c5d7d.js +7 -0
  431. package/app/.next/static/chunks/505-aa415671593bcfc6.js +1 -0
  432. package/app/.next/static/chunks/6795-3ddcc1d55c65c001.js +64 -0
  433. package/app/.next/static/chunks/8035-122ffa9582ac6dd4.js +3 -0
  434. package/app/.next/static/chunks/app/(dashboard)/dashboard/basic-chat/{page-2c8c481b0fc0bb1b.js → page-cd02fe7b10810144.js} +1 -1
  435. package/app/.next/static/chunks/app/(dashboard)/dashboard/cli-tools/{page-f2206753c04da01d.js → page-1266463402c761f6.js} +1 -1
  436. package/app/.next/static/chunks/app/(dashboard)/dashboard/combos/{page-8556b646c8002338.js → page-fcb74a9294486c9a.js} +1 -1
  437. package/app/.next/static/chunks/app/(dashboard)/dashboard/console-log/{page-2ffb4689f161ca33.js → page-af58c67a4e409aad.js} +1 -1
  438. package/app/.next/static/chunks/app/(dashboard)/dashboard/endpoint/page-79a243ddf837f243.js +1 -0
  439. package/app/.next/static/chunks/app/(dashboard)/dashboard/media-providers/[kind]/[id]/{page-187f6bdd95bc5640.js → page-49cd02768f07a61a.js} +36 -7
  440. package/app/.next/static/chunks/app/(dashboard)/dashboard/media-providers/[kind]/{page-ebe4ae25d01f731a.js → page-7b7477044f123993.js} +1 -1
  441. package/app/.next/static/chunks/app/(dashboard)/dashboard/mitm/{page-21fffa0f231796e9.js → page-7ab6931e5032e32c.js} +1 -1
  442. package/app/.next/static/chunks/app/(dashboard)/dashboard/page-87aa0d715eec8054.js +1 -0
  443. package/app/.next/static/chunks/app/(dashboard)/dashboard/profile/{page-354de24a25e94694.js → page-8d500331b6e10813.js} +1 -1
  444. package/app/.next/static/chunks/app/(dashboard)/dashboard/providers/[id]/page-a35c8ba68e2a770f.js +1 -0
  445. package/app/.next/static/chunks/app/(dashboard)/dashboard/providers/new/{page-426e797917337fd8.js → page-510b667ad2defe52.js} +1 -1
  446. package/app/.next/static/chunks/app/(dashboard)/dashboard/providers/{page-09e7d6bd4d08202c.js → page-93130d3988f4ba96.js} +1 -1
  447. package/app/.next/static/chunks/app/(dashboard)/dashboard/proxy-pools/{page-34d04900ddd6932a.js → page-b675d50ce82d5577.js} +1 -1
  448. package/app/.next/static/chunks/app/(dashboard)/dashboard/quota/page-2485ee7c89798a7b.js +1 -0
  449. package/app/.next/static/chunks/app/(dashboard)/dashboard/translator/{page-82c184348fd5c917.js → page-4af98d8129a93412.js} +1 -1
  450. package/app/.next/static/chunks/app/(dashboard)/dashboard/usage/{page-1308b9d445962823.js → page-538a897db589c145.js} +1 -1
  451. package/app/.next/static/chunks/app/(dashboard)/layout-380c1f4327a2a3bc.js +1 -0
  452. package/app/.next/static/chunks/app/_global-error/{page-4ed867ac6edb2a89.js → page-786efea7f9ddd575.js} +1 -1
  453. package/app/.next/static/chunks/app/api/auth/login/{route-4ed867ac6edb2a89.js → route-786efea7f9ddd575.js} +1 -1
  454. package/app/.next/static/chunks/app/api/auth/logout/{route-4ed867ac6edb2a89.js → route-786efea7f9ddd575.js} +1 -1
  455. package/app/.next/static/chunks/app/api/cli-tools/antigravity-mitm/alias/{route-4ed867ac6edb2a89.js → route-786efea7f9ddd575.js} +1 -1
  456. package/app/.next/static/chunks/app/api/cli-tools/antigravity-mitm/route-786efea7f9ddd575.js +1 -0
  457. package/app/.next/static/chunks/app/api/cli-tools/claude-settings/route-786efea7f9ddd575.js +1 -0
  458. package/app/.next/static/chunks/app/api/cli-tools/codex-settings/route-786efea7f9ddd575.js +1 -0
  459. package/app/.next/static/chunks/app/api/cli-tools/copilot-settings/route-786efea7f9ddd575.js +1 -0
  460. package/app/.next/static/chunks/app/api/cli-tools/droid-settings/route-786efea7f9ddd575.js +1 -0
  461. package/app/.next/static/chunks/app/api/cli-tools/openclaw-settings/route-786efea7f9ddd575.js +1 -0
  462. package/app/.next/static/chunks/app/api/cli-tools/opencode-settings/route-786efea7f9ddd575.js +1 -0
  463. package/app/.next/static/chunks/app/api/cloud/auth/route-786efea7f9ddd575.js +1 -0
  464. package/app/.next/static/chunks/app/api/cloud/credentials/update/route-786efea7f9ddd575.js +1 -0
  465. package/app/.next/static/chunks/app/api/cloud/model/resolve/route-786efea7f9ddd575.js +1 -0
  466. package/app/.next/static/chunks/app/api/cloud/models/alias/route-786efea7f9ddd575.js +1 -0
  467. package/app/.next/static/chunks/app/api/combos/[id]/route-786efea7f9ddd575.js +1 -0
  468. package/app/.next/static/chunks/app/api/combos/route-786efea7f9ddd575.js +1 -0
  469. package/app/.next/static/chunks/app/api/health/route-786efea7f9ddd575.js +1 -0
  470. package/app/.next/static/chunks/app/api/init/route-786efea7f9ddd575.js +1 -0
  471. package/app/.next/static/chunks/app/api/keys/[id]/route-786efea7f9ddd575.js +1 -0
  472. package/app/.next/static/chunks/app/api/keys/route-786efea7f9ddd575.js +1 -0
  473. package/app/.next/static/chunks/app/api/locale/route-786efea7f9ddd575.js +1 -0
  474. package/app/.next/static/chunks/app/api/media-providers/tts/elevenlabs/voices/route-786efea7f9ddd575.js +1 -0
  475. package/app/.next/static/chunks/app/api/media-providers/tts/voices/route-786efea7f9ddd575.js +1 -0
  476. package/app/.next/static/chunks/app/api/models/alias/route-786efea7f9ddd575.js +1 -0
  477. package/app/.next/static/chunks/app/api/models/availability/route-786efea7f9ddd575.js +1 -0
  478. package/app/.next/static/chunks/app/api/models/route-786efea7f9ddd575.js +1 -0
  479. package/app/.next/static/chunks/app/api/models/test/route-786efea7f9ddd575.js +1 -0
  480. package/app/.next/static/chunks/app/api/oauth/[provider]/[action]/route-786efea7f9ddd575.js +1 -0
  481. package/app/.next/static/chunks/app/api/oauth/cursor/auto-import/route-786efea7f9ddd575.js +1 -0
  482. package/app/.next/static/chunks/app/api/oauth/cursor/import/route-786efea7f9ddd575.js +1 -0
  483. package/app/.next/static/chunks/app/api/oauth/gitlab/pat/route-786efea7f9ddd575.js +1 -0
  484. package/app/.next/static/chunks/app/api/oauth/iflow/cookie/route-786efea7f9ddd575.js +1 -0
  485. package/app/.next/static/chunks/app/api/oauth/kiro/auto-import/route-786efea7f9ddd575.js +1 -0
  486. package/app/.next/static/chunks/app/api/oauth/kiro/import/route-786efea7f9ddd575.js +1 -0
  487. package/app/.next/static/chunks/app/api/oauth/kiro/social-authorize/route-786efea7f9ddd575.js +1 -0
  488. package/app/.next/static/chunks/app/api/oauth/kiro/social-exchange/route-786efea7f9ddd575.js +1 -0
  489. package/app/.next/static/chunks/app/api/pricing/route-786efea7f9ddd575.js +1 -0
  490. package/app/.next/static/chunks/app/api/provider-nodes/[id]/route-786efea7f9ddd575.js +1 -0
  491. package/app/.next/static/chunks/app/api/provider-nodes/route-786efea7f9ddd575.js +1 -0
  492. package/app/.next/static/chunks/app/api/provider-nodes/validate/route-786efea7f9ddd575.js +1 -0
  493. package/app/.next/static/chunks/app/api/providers/[id]/models/route-786efea7f9ddd575.js +1 -0
  494. package/app/.next/static/chunks/app/api/providers/[id]/route-786efea7f9ddd575.js +1 -0
  495. package/app/.next/static/chunks/app/api/providers/[id]/test/route-786efea7f9ddd575.js +1 -0
  496. package/app/.next/static/chunks/app/api/providers/[id]/test-models/route-786efea7f9ddd575.js +1 -0
  497. package/app/.next/static/chunks/app/api/providers/client/route-786efea7f9ddd575.js +1 -0
  498. package/app/.next/static/chunks/app/api/providers/kilo/free-models/route-786efea7f9ddd575.js +1 -0
  499. package/app/.next/static/chunks/app/api/providers/route-786efea7f9ddd575.js +1 -0
  500. package/app/.next/static/chunks/app/api/providers/suggested-models/route-786efea7f9ddd575.js +1 -0
  501. package/app/.next/static/chunks/app/api/providers/test-batch/route-786efea7f9ddd575.js +1 -0
  502. package/app/.next/static/chunks/app/api/providers/validate/route-786efea7f9ddd575.js +1 -0
  503. package/app/.next/static/chunks/app/api/proxy-pools/[id]/route-786efea7f9ddd575.js +1 -0
  504. package/app/.next/static/chunks/app/api/proxy-pools/[id]/test/route-786efea7f9ddd575.js +1 -0
  505. package/app/.next/static/chunks/app/api/proxy-pools/route-786efea7f9ddd575.js +1 -0
  506. package/app/.next/static/chunks/app/api/proxy-pools/vercel-deploy/route-786efea7f9ddd575.js +1 -0
  507. package/app/.next/static/chunks/app/api/settings/database/route-786efea7f9ddd575.js +1 -0
  508. package/app/.next/static/chunks/app/api/settings/proxy-test/route-786efea7f9ddd575.js +1 -0
  509. package/app/.next/static/chunks/app/api/settings/require-login/route-786efea7f9ddd575.js +1 -0
  510. package/app/.next/static/chunks/app/api/settings/route-786efea7f9ddd575.js +1 -0
  511. package/app/.next/static/chunks/app/api/shutdown/route-786efea7f9ddd575.js +1 -0
  512. package/app/.next/static/chunks/app/api/tags/route-786efea7f9ddd575.js +1 -0
  513. package/app/.next/static/chunks/app/api/translator/console-logs/route-786efea7f9ddd575.js +1 -0
  514. package/app/.next/static/chunks/app/api/translator/console-logs/stream/route-786efea7f9ddd575.js +1 -0
  515. package/app/.next/static/chunks/app/api/translator/load/route-786efea7f9ddd575.js +1 -0
  516. package/app/.next/static/chunks/app/api/translator/save/route-786efea7f9ddd575.js +1 -0
  517. package/app/.next/static/chunks/app/api/translator/send/route-786efea7f9ddd575.js +1 -0
  518. package/app/.next/static/chunks/app/api/translator/translate/route-786efea7f9ddd575.js +1 -0
  519. package/app/.next/static/chunks/app/api/tunnel/disable/route-786efea7f9ddd575.js +1 -0
  520. package/app/.next/static/chunks/app/api/tunnel/enable/route-786efea7f9ddd575.js +1 -0
  521. package/app/.next/static/chunks/app/api/tunnel/status/route-786efea7f9ddd575.js +1 -0
  522. package/app/.next/static/chunks/app/api/tunnel/tailscale-check/route-786efea7f9ddd575.js +1 -0
  523. package/app/.next/static/chunks/app/api/tunnel/tailscale-disable/route-786efea7f9ddd575.js +1 -0
  524. package/app/.next/static/chunks/app/api/tunnel/tailscale-enable/route-786efea7f9ddd575.js +1 -0
  525. package/app/.next/static/chunks/app/api/tunnel/tailscale-install/route-786efea7f9ddd575.js +1 -0
  526. package/app/.next/static/chunks/app/api/tunnel/tailscale-login/route-786efea7f9ddd575.js +1 -0
  527. package/app/.next/static/chunks/app/api/tunnel/tailscale-start-daemon/route-786efea7f9ddd575.js +1 -0
  528. package/app/.next/static/chunks/app/api/usage/[connectionId]/route-786efea7f9ddd575.js +1 -0
  529. package/app/.next/static/chunks/app/api/usage/chart/route-786efea7f9ddd575.js +1 -0
  530. package/app/.next/static/chunks/app/api/usage/history/route-786efea7f9ddd575.js +1 -0
  531. package/app/.next/static/chunks/app/api/usage/providers/route-786efea7f9ddd575.js +1 -0
  532. package/app/.next/static/chunks/app/api/usage/request-details/route-786efea7f9ddd575.js +1 -0
  533. package/app/.next/static/chunks/app/api/usage/request-logs/route-786efea7f9ddd575.js +1 -0
  534. package/app/.next/static/chunks/app/api/usage/stats/route-786efea7f9ddd575.js +1 -0
  535. package/app/.next/static/chunks/app/api/usage/stream/route-786efea7f9ddd575.js +1 -0
  536. package/app/.next/static/chunks/app/api/v1/api/chat/route-786efea7f9ddd575.js +1 -0
  537. package/app/.next/static/chunks/app/api/v1/audio/speech/route-786efea7f9ddd575.js +1 -0
  538. package/app/.next/static/chunks/app/api/v1/chat/completions/route-786efea7f9ddd575.js +1 -0
  539. package/app/.next/static/chunks/app/api/v1/embeddings/route-786efea7f9ddd575.js +1 -0
  540. package/app/.next/static/chunks/app/api/v1/messages/count_tokens/route-786efea7f9ddd575.js +1 -0
  541. package/app/.next/static/chunks/app/api/v1/messages/route-786efea7f9ddd575.js +1 -0
  542. package/app/.next/static/chunks/app/api/v1/models/route-786efea7f9ddd575.js +1 -0
  543. package/app/.next/static/chunks/app/api/v1/responses/compact/route-786efea7f9ddd575.js +1 -0
  544. package/app/.next/static/chunks/app/api/v1/responses/route-786efea7f9ddd575.js +1 -0
  545. package/app/.next/static/chunks/app/api/v1/route-786efea7f9ddd575.js +1 -0
  546. package/app/.next/static/chunks/app/api/v1beta/models/[...path]/route-786efea7f9ddd575.js +1 -0
  547. package/app/.next/static/chunks/app/api/v1beta/models/route-786efea7f9ddd575.js +1 -0
  548. package/app/.next/static/chunks/app/api/version/route-786efea7f9ddd575.js +1 -0
  549. package/app/.next/static/chunks/app/login/{page-c0d5d5c52ff7317f.js → page-2424ce8673ecf5cb.js} +1 -1
  550. package/app/.next/static/chunks/app/manifest.webmanifest/route-786efea7f9ddd575.js +1 -0
  551. package/app/.next/static/chunks/app/page-786efea7f9ddd575.js +1 -0
  552. package/app/.next/static/chunks/next/dist/client/components/builtin/app-error-786efea7f9ddd575.js +1 -0
  553. package/app/.next/static/chunks/next/dist/client/components/builtin/forbidden-786efea7f9ddd575.js +1 -0
  554. package/app/.next/static/chunks/next/dist/client/components/builtin/not-found-786efea7f9ddd575.js +1 -0
  555. package/app/.next/static/chunks/next/dist/client/components/builtin/unauthorized-786efea7f9ddd575.js +1 -0
  556. package/app/.next/static/css/f5cf7c3d62f1ba87.css +1 -0
  557. package/app/package.json +2 -1
  558. package/cli.js +45 -36
  559. package/package.json +1 -1
  560. package/app/.next/server/chunks/149.js +0 -1
  561. package/app/.next/server/chunks/2141.js +0 -1
  562. package/app/.next/server/chunks/514.js +0 -5
  563. package/app/.next/server/chunks/6502.js +0 -1
  564. package/app/.next/server/chunks/8354.js +0 -1
  565. package/app/.next/server/chunks/8443.js +0 -1
  566. package/app/.next/server/chunks/8838.js +0 -1
  567. package/app/.next/server/chunks/8872.js +0 -1
  568. package/app/.next/server/chunks/9489.js +0 -1
  569. package/app/.next/static/Seqmp6275x677iq3fTzPt/_buildManifest.js +0 -1
  570. package/app/.next/static/chunks/1237-06ab8ceb0847165d.js +0 -18
  571. package/app/.next/static/chunks/2599-ef3aca3d4fdaa96d.js +0 -1
  572. package/app/.next/static/chunks/4156-d2436f4645532dc5.js +0 -7
  573. package/app/.next/static/chunks/505-e8dec61412d86c58.js +0 -1
  574. package/app/.next/static/chunks/8035-0bbf141cee7024b0.js +0 -3
  575. package/app/.next/static/chunks/app/(dashboard)/dashboard/endpoint/page-0b58256fd5ccd8b7.js +0 -1
  576. package/app/.next/static/chunks/app/(dashboard)/dashboard/page-9533adb8cd2aa0ac.js +0 -1
  577. package/app/.next/static/chunks/app/(dashboard)/dashboard/providers/[id]/page-f532b1565bd8f227.js +0 -1
  578. package/app/.next/static/chunks/app/(dashboard)/dashboard/quota/page-3849d2b36c38ef3d.js +0 -1
  579. package/app/.next/static/chunks/app/(dashboard)/layout-757a35dbb4539aad.js +0 -1
  580. package/app/.next/static/chunks/app/api/cli-tools/antigravity-mitm/route-4ed867ac6edb2a89.js +0 -1
  581. package/app/.next/static/chunks/app/api/cli-tools/claude-settings/route-4ed867ac6edb2a89.js +0 -1
  582. package/app/.next/static/chunks/app/api/cli-tools/codex-settings/route-4ed867ac6edb2a89.js +0 -1
  583. package/app/.next/static/chunks/app/api/cli-tools/copilot-settings/route-4ed867ac6edb2a89.js +0 -1
  584. package/app/.next/static/chunks/app/api/cli-tools/droid-settings/route-4ed867ac6edb2a89.js +0 -1
  585. package/app/.next/static/chunks/app/api/cli-tools/openclaw-settings/route-4ed867ac6edb2a89.js +0 -1
  586. package/app/.next/static/chunks/app/api/cli-tools/opencode-settings/route-4ed867ac6edb2a89.js +0 -1
  587. package/app/.next/static/chunks/app/api/cloud/auth/route-4ed867ac6edb2a89.js +0 -1
  588. package/app/.next/static/chunks/app/api/cloud/credentials/update/route-4ed867ac6edb2a89.js +0 -1
  589. package/app/.next/static/chunks/app/api/cloud/model/resolve/route-4ed867ac6edb2a89.js +0 -1
  590. package/app/.next/static/chunks/app/api/cloud/models/alias/route-4ed867ac6edb2a89.js +0 -1
  591. package/app/.next/static/chunks/app/api/combos/[id]/route-4ed867ac6edb2a89.js +0 -1
  592. package/app/.next/static/chunks/app/api/combos/route-4ed867ac6edb2a89.js +0 -1
  593. package/app/.next/static/chunks/app/api/health/route-4ed867ac6edb2a89.js +0 -1
  594. package/app/.next/static/chunks/app/api/init/route-4ed867ac6edb2a89.js +0 -1
  595. package/app/.next/static/chunks/app/api/keys/[id]/route-4ed867ac6edb2a89.js +0 -1
  596. package/app/.next/static/chunks/app/api/keys/route-4ed867ac6edb2a89.js +0 -1
  597. package/app/.next/static/chunks/app/api/locale/route-4ed867ac6edb2a89.js +0 -1
  598. package/app/.next/static/chunks/app/api/media-providers/tts/elevenlabs/voices/route-4ed867ac6edb2a89.js +0 -1
  599. package/app/.next/static/chunks/app/api/media-providers/tts/voices/route-4ed867ac6edb2a89.js +0 -1
  600. package/app/.next/static/chunks/app/api/models/alias/route-4ed867ac6edb2a89.js +0 -1
  601. package/app/.next/static/chunks/app/api/models/availability/route-4ed867ac6edb2a89.js +0 -1
  602. package/app/.next/static/chunks/app/api/models/route-4ed867ac6edb2a89.js +0 -1
  603. package/app/.next/static/chunks/app/api/models/test/route-4ed867ac6edb2a89.js +0 -1
  604. package/app/.next/static/chunks/app/api/oauth/[provider]/[action]/route-4ed867ac6edb2a89.js +0 -1
  605. package/app/.next/static/chunks/app/api/oauth/cursor/auto-import/route-4ed867ac6edb2a89.js +0 -1
  606. package/app/.next/static/chunks/app/api/oauth/cursor/import/route-4ed867ac6edb2a89.js +0 -1
  607. package/app/.next/static/chunks/app/api/oauth/gitlab/pat/route-4ed867ac6edb2a89.js +0 -1
  608. package/app/.next/static/chunks/app/api/oauth/iflow/cookie/route-4ed867ac6edb2a89.js +0 -1
  609. package/app/.next/static/chunks/app/api/oauth/kiro/auto-import/route-4ed867ac6edb2a89.js +0 -1
  610. package/app/.next/static/chunks/app/api/oauth/kiro/import/route-4ed867ac6edb2a89.js +0 -1
  611. package/app/.next/static/chunks/app/api/oauth/kiro/social-authorize/route-4ed867ac6edb2a89.js +0 -1
  612. package/app/.next/static/chunks/app/api/oauth/kiro/social-exchange/route-4ed867ac6edb2a89.js +0 -1
  613. package/app/.next/static/chunks/app/api/pricing/route-4ed867ac6edb2a89.js +0 -1
  614. package/app/.next/static/chunks/app/api/provider-nodes/[id]/route-4ed867ac6edb2a89.js +0 -1
  615. package/app/.next/static/chunks/app/api/provider-nodes/route-4ed867ac6edb2a89.js +0 -1
  616. package/app/.next/static/chunks/app/api/provider-nodes/validate/route-4ed867ac6edb2a89.js +0 -1
  617. package/app/.next/static/chunks/app/api/providers/[id]/models/route-4ed867ac6edb2a89.js +0 -1
  618. package/app/.next/static/chunks/app/api/providers/[id]/route-4ed867ac6edb2a89.js +0 -1
  619. package/app/.next/static/chunks/app/api/providers/[id]/test/route-4ed867ac6edb2a89.js +0 -1
  620. package/app/.next/static/chunks/app/api/providers/[id]/test-models/route-4ed867ac6edb2a89.js +0 -1
  621. package/app/.next/static/chunks/app/api/providers/client/route-4ed867ac6edb2a89.js +0 -1
  622. package/app/.next/static/chunks/app/api/providers/kilo/free-models/route-4ed867ac6edb2a89.js +0 -1
  623. package/app/.next/static/chunks/app/api/providers/route-4ed867ac6edb2a89.js +0 -1
  624. package/app/.next/static/chunks/app/api/providers/test-batch/route-4ed867ac6edb2a89.js +0 -1
  625. package/app/.next/static/chunks/app/api/providers/validate/route-4ed867ac6edb2a89.js +0 -1
  626. package/app/.next/static/chunks/app/api/proxy-pools/[id]/route-4ed867ac6edb2a89.js +0 -1
  627. package/app/.next/static/chunks/app/api/proxy-pools/[id]/test/route-4ed867ac6edb2a89.js +0 -1
  628. package/app/.next/static/chunks/app/api/proxy-pools/route-4ed867ac6edb2a89.js +0 -1
  629. package/app/.next/static/chunks/app/api/proxy-pools/vercel-deploy/route-4ed867ac6edb2a89.js +0 -1
  630. package/app/.next/static/chunks/app/api/settings/database/route-4ed867ac6edb2a89.js +0 -1
  631. package/app/.next/static/chunks/app/api/settings/proxy-test/route-4ed867ac6edb2a89.js +0 -1
  632. package/app/.next/static/chunks/app/api/settings/require-login/route-4ed867ac6edb2a89.js +0 -1
  633. package/app/.next/static/chunks/app/api/settings/route-4ed867ac6edb2a89.js +0 -1
  634. package/app/.next/static/chunks/app/api/shutdown/route-4ed867ac6edb2a89.js +0 -1
  635. package/app/.next/static/chunks/app/api/tags/route-4ed867ac6edb2a89.js +0 -1
  636. package/app/.next/static/chunks/app/api/translator/console-logs/route-4ed867ac6edb2a89.js +0 -1
  637. package/app/.next/static/chunks/app/api/translator/console-logs/stream/route-4ed867ac6edb2a89.js +0 -1
  638. package/app/.next/static/chunks/app/api/translator/load/route-4ed867ac6edb2a89.js +0 -1
  639. package/app/.next/static/chunks/app/api/translator/save/route-4ed867ac6edb2a89.js +0 -1
  640. package/app/.next/static/chunks/app/api/translator/send/route-4ed867ac6edb2a89.js +0 -1
  641. package/app/.next/static/chunks/app/api/translator/translate/route-4ed867ac6edb2a89.js +0 -1
  642. package/app/.next/static/chunks/app/api/tunnel/disable/route-4ed867ac6edb2a89.js +0 -1
  643. package/app/.next/static/chunks/app/api/tunnel/enable/route-4ed867ac6edb2a89.js +0 -1
  644. package/app/.next/static/chunks/app/api/tunnel/status/route-4ed867ac6edb2a89.js +0 -1
  645. package/app/.next/static/chunks/app/api/tunnel/tailscale-check/route-4ed867ac6edb2a89.js +0 -1
  646. package/app/.next/static/chunks/app/api/tunnel/tailscale-disable/route-4ed867ac6edb2a89.js +0 -1
  647. package/app/.next/static/chunks/app/api/tunnel/tailscale-enable/route-4ed867ac6edb2a89.js +0 -1
  648. package/app/.next/static/chunks/app/api/tunnel/tailscale-install/route-4ed867ac6edb2a89.js +0 -1
  649. package/app/.next/static/chunks/app/api/tunnel/tailscale-login/route-4ed867ac6edb2a89.js +0 -1
  650. package/app/.next/static/chunks/app/api/tunnel/tailscale-start-daemon/route-4ed867ac6edb2a89.js +0 -1
  651. package/app/.next/static/chunks/app/api/usage/[connectionId]/route-4ed867ac6edb2a89.js +0 -1
  652. package/app/.next/static/chunks/app/api/usage/chart/route-4ed867ac6edb2a89.js +0 -1
  653. package/app/.next/static/chunks/app/api/usage/history/route-4ed867ac6edb2a89.js +0 -1
  654. package/app/.next/static/chunks/app/api/usage/providers/route-4ed867ac6edb2a89.js +0 -1
  655. package/app/.next/static/chunks/app/api/usage/request-details/route-4ed867ac6edb2a89.js +0 -1
  656. package/app/.next/static/chunks/app/api/usage/request-logs/route-4ed867ac6edb2a89.js +0 -1
  657. package/app/.next/static/chunks/app/api/usage/stats/route-4ed867ac6edb2a89.js +0 -1
  658. package/app/.next/static/chunks/app/api/usage/stream/route-4ed867ac6edb2a89.js +0 -1
  659. package/app/.next/static/chunks/app/api/v1/api/chat/route-4ed867ac6edb2a89.js +0 -1
  660. package/app/.next/static/chunks/app/api/v1/audio/speech/route-4ed867ac6edb2a89.js +0 -1
  661. package/app/.next/static/chunks/app/api/v1/chat/completions/route-4ed867ac6edb2a89.js +0 -1
  662. package/app/.next/static/chunks/app/api/v1/embeddings/route-4ed867ac6edb2a89.js +0 -1
  663. package/app/.next/static/chunks/app/api/v1/messages/count_tokens/route-4ed867ac6edb2a89.js +0 -1
  664. package/app/.next/static/chunks/app/api/v1/messages/route-4ed867ac6edb2a89.js +0 -1
  665. package/app/.next/static/chunks/app/api/v1/models/route-4ed867ac6edb2a89.js +0 -1
  666. package/app/.next/static/chunks/app/api/v1/responses/compact/route-4ed867ac6edb2a89.js +0 -1
  667. package/app/.next/static/chunks/app/api/v1/responses/route-4ed867ac6edb2a89.js +0 -1
  668. package/app/.next/static/chunks/app/api/v1/route-4ed867ac6edb2a89.js +0 -1
  669. package/app/.next/static/chunks/app/api/v1beta/models/[...path]/route-4ed867ac6edb2a89.js +0 -1
  670. package/app/.next/static/chunks/app/api/v1beta/models/route-4ed867ac6edb2a89.js +0 -1
  671. package/app/.next/static/chunks/app/api/version/route-4ed867ac6edb2a89.js +0 -1
  672. package/app/.next/static/chunks/app/manifest.webmanifest/route-4ed867ac6edb2a89.js +0 -1
  673. package/app/.next/static/chunks/app/page-4ed867ac6edb2a89.js +0 -1
  674. package/app/.next/static/chunks/next/dist/client/components/builtin/app-error-4ed867ac6edb2a89.js +0 -1
  675. package/app/.next/static/chunks/next/dist/client/components/builtin/forbidden-4ed867ac6edb2a89.js +0 -1
  676. package/app/.next/static/chunks/next/dist/client/components/builtin/not-found-4ed867ac6edb2a89.js +0 -1
  677. package/app/.next/static/chunks/next/dist/client/components/builtin/unauthorized-4ed867ac6edb2a89.js +0 -1
  678. package/app/.next/static/css/357a3570ee0fa8c1.css +0 -1
  679. /package/app/.next/static/{Seqmp6275x677iq3fTzPt → VB9M_Cg8qFTiJ9Qceuwx7}/_ssgManifest.js +0 -0
@@ -0,0 +1,3 @@
1
+ (self.webpackChunk_N_E=self.webpackChunk_N_E||[]).push([[8035],{64795:(e,t,s)=>{"use strict";s.d(t,{default:()=>d});var a=s(73365),l=s(1521),r=s(86447),n=s.n(r),i=s(74156),o=s(96459);let c=[{icon:"public",title:"Access Anywhere",desc:"Use your API from any network"},{icon:"group",title:"Share Endpoint",desc:"Share URL with team members"},{icon:"code",title:"Use in Cursor/Cline",desc:"Connect AI tools remotely"},{icon:"lock",title:"Encrypted",desc:"End-to-end TLS via Cloudflare"}];function d({machineId:e}){let[t,s]=(0,l.useState)([]),[r,n]=(0,l.useState)(!0),[p,g]=(0,l.useState)(!1),[b,y]=(0,l.useState)(""),[f,j]=(0,l.useState)(null),[k,w]=(0,l.useState)(!1),[v,N]=(0,l.useState)(!0),[C,S]=(0,l.useState)(!0),[T,_]=(0,l.useState)(!1),[P,$]=(0,l.useState)(!0),[E,A]=(0,l.useState)(!1),[U,D]=(0,l.useState)(""),[F,I]=(0,l.useState)(""),[O,W]=(0,l.useState)(!1),[R,q]=(0,l.useState)(""),[L,K]=(0,l.useState)(null),[z,J]=(0,l.useState)(!1),[H,M]=(0,l.useState)(!1),[Q,Z]=(0,l.useState)(!1),[B,V]=(0,l.useState)(""),[G,X]=(0,l.useState)(!1),[Y,ee]=(0,l.useState)(""),[et,es]=(0,l.useState)(null),[ea,el]=(0,l.useState)(null),[er,en]=(0,l.useState)(!1),[ei,eo]=(0,l.useState)([]),[ec,ed]=(0,l.useState)(""),[ex,em]=(0,l.useState)(!1),[eh,eu]=(0,l.useState)(!1),[ep,eg]=(0,l.useState)(!1),eb=(0,l.useRef)(null),[ey,ef]=(0,l.useState)(new Set),{copied:ej,copy:ek}=(0,o.C)();(0,l.useEffect)(()=>{eb.current&&(eb.current.scrollTop=eb.current.scrollHeight)},[ei]),(0,l.useEffect)(()=>{eC(),ew()},[]);let ew=async()=>{$(!0);try{let[e,t]=await Promise.all([fetch("/api/settings"),fetch("/api/tunnel/status")]);if(e.ok){let t=await e.json();w(t.requireApiKey||!1),N(!1!==t.requireLogin),S(t.hasPassword||!1),_(t.tunnelDashboardAccess||!1)}if(t.ok){let e=await t.json(),s=e.tunnel?.enabled||!1,a=e.tunnel?.tunnelUrl||"",l=e.tunnel?.publicUrl||"";D(a),I(l);let r=e.tailscale?.enabled||!1,n=e.tailscale?.tunnelUrl||"";if(V(n),r&&n){X(!0),ee("Checking Tailscale...");let e=`${n}/api/health`;try{let t=await fetch(e,{mode:"no-cors",cache:"no-store"});if(t.ok||"opaque"===t.type)Z(!0);else{let e=await eE(n);Z(e),e||es({type:"warning",message:"Tailscale not reachable."})}}catch{let e=await eE(n);Z(e),e||es({type:"warning",message:"Tailscale not reachable."})}finally{X(!1),ee("")}}else Z(r);if(s&&(l||a)){let e=`${l||a}/api/health`;try{(await fetch(e,{cache:"no-store"})).ok?A(!0):eS(l||a)}catch{eS(l||a)}}else A(s)}}catch(e){console.log("Error loading settings:",e)}finally{$(!1)}},ev=async e=>{try{(await fetch("/api/settings",{method:"PATCH",headers:{"Content-Type":"application/json"},body:JSON.stringify({tunnelDashboardAccess:e})})).ok&&_(e)}catch(e){console.log("Error updating tunnelDashboardAccess:",e)}},eN=async e=>{try{(await fetch("/api/settings",{method:"PATCH",headers:{"Content-Type":"application/json"},body:JSON.stringify({requireApiKey:e})})).ok&&w(e)}catch(e){console.log("Error updating requireApiKey:",e)}},eC=async()=>{try{let e=await fetch("/api/keys"),t=await e.json();e.ok&&s(t.keys||[])}catch(e){console.log("Error fetching data:",e)}finally{n(!1)}},eS=async e=>{W(!0),q("Waiting for tunnel ready...");let t=`${e}/api/health`,s=Date.now();for(;Date.now()-s<3e5;){await new Promise(e=>setTimeout(e,2e3));try{let e=await fetch(t,{mode:"no-cors",cache:"no-store"});if(e.ok||"opaque"===e.type)return A(!0),W(!1),q(""),!0}catch{}if((Date.now()-s)%1e4<2e3)try{let e=await fetch("/api/tunnel/status");if(e.ok){let t=await e.json();if(!t.tunnel?.enabled)return K({type:"error",message:"Tunnel process stopped unexpectedly."}),W(!1),q(""),!1}}catch{}}return K({type:"error",message:"Tunnel created but not reachable. Please try again."}),W(!1),q(""),!1},eT=async()=>{J(!1),W(!0),K(null),q("Creating tunnel...");let e=!0;(async()=>{for(;e;){try{let t=await fetch("/api/tunnel/status");if(t.ok){let s=await t.json();s.download?.downloading?q(`Downloading cloudflared... ${s.download.progress}%`):e&&q("Creating tunnel...")}}catch{}await new Promise(e=>setTimeout(e,1e3))}})();try{let t=await fetch("/api/tunnel/enable",{method:"POST"});e=!1;let s=await t.json();if(!t.ok)return void K({type:"error",message:s.error||"Failed to enable tunnel"});let a=s.publicUrl||s.tunnelUrl;if(!a)return void K({type:"error",message:"No tunnel URL returned"});D(s.tunnelUrl||""),I(s.publicUrl||""),await eS(a)}catch(e){K({type:"error",message:e.message})}finally{e=!1,W(!1),q("")}},e_=async()=>{W(!0),K(null);try{let e=await fetch("/api/tunnel/disable",{method:"POST"}),t=await e.json();e.ok?(A(!1),D(""),I(""),M(!1),K({type:"success",message:"Tunnel disabled"})):K({type:"error",message:t.error||"Failed to disable tunnel"})}catch(e){K({type:"error",message:e.message})}finally{W(!1)}},eP=async()=>{el(null);try{let e=await fetch("/api/tunnel/tailscale-check");if(e.ok){let t=await e.json();return el(t.installed),t}}catch{}return el(!1),{installed:!1}},e$=async()=>{en(!0),es(null),eo([]);try{let e=await fetch("/api/tunnel/tailscale-install",{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({sudoPassword:ec})});ed("");let t=e.body.getReader(),s=new TextDecoder,a="";for(;;){let{done:e,value:l}=await t.read();if(e)break;let r=(a+=s.decode(l,{stream:!0})).split("\n\n");for(let e of(a=r.pop()||"",r)){let t=e.split("\n"),s="progress",a=null;for(let e of t)if(e.startsWith("event: ")&&(s=e.slice(7).trim()),e.startsWith("data: "))try{a=JSON.parse(e.slice(6))}catch{}if(a)if("progress"===s)eo(e=>[...e.slice(-50),a.message]);else if("done"===s){el(!0),en(!1);return}else"error"===s&&es({type:"error",message:a.error||"Install failed"})}}}catch(e){es({type:"error",message:e.message})}finally{en(!1)}},eE=async e=>{ee("Waiting for Tailscale ready...");let t=`${e}/api/health`,s=Date.now();for(;Date.now()-s<3e5;){await new Promise(e=>setTimeout(e,2e3));try{let e=await fetch(t,{mode:"no-cors",cache:"no-store"});if(e.ok||"opaque"===e.type)return!0}catch{}}return!1},eA=async e=>{let t=e||null;eu(!1),em(!0),X(!0),es(null),ee("Connecting...");try{let e=await fetch("/api/tunnel/tailscale-enable",{method:"POST"}),s=await e.json();if(e.ok&&s.success){t&&t.close(),V(s.tunnelUrl||""),await eE(s.tunnelUrl)?(Z(!0),es(null)):(Z(!0),es({type:"warning",message:"Connected but not reachable yet."}));return}if(s.needsLogin&&s.authUrl){t?t.location.href=s.authUrl:window.open(s.authUrl,"tailscale_auth","width=600,height=700"),ee("Waiting for login...");for(let e=0;e<40;e++){await new Promise(e=>setTimeout(e,3e3));try{let e=await fetch("/api/tunnel/tailscale-check");if(e.ok&&(await e.json()).loggedIn){ee("Starting funnel...");let e=await fetch("/api/tunnel/tailscale-enable",{method:"POST"}),s=await e.json();e.ok&&s.success?(t&&t.close(),V(s.tunnelUrl||""),await eE(s.tunnelUrl)?(Z(!0),es(null)):(Z(!0),es({type:"warning",message:"Connected but not reachable yet."}))):s.funnelNotEnabled&&s.enableUrl?await eU(s.enableUrl,t):es({type:"error",message:s.error||"Failed to start funnel"});return}}catch{}}es({type:"error",message:"Login timed out. Please try again."});return}if(s.funnelNotEnabled&&s.enableUrl)return void await eU(s.enableUrl,t);t&&t.close(),es({type:"error",message:s.error||"Failed to connect"})}catch(e){t&&t.close(),es({type:"error",message:e.message})}finally{X(!1),em(!1),ee("")}},eU=async(e,t)=>{t?t.location.href=e:window.open(e,"tailscale_auth","width=600,height=700"),ee("Enable Funnel in browser, waiting...");for(let e=0;e<40;e++){await new Promise(e=>setTimeout(e,3e3));try{let e=await fetch("/api/tunnel/tailscale-enable",{method:"POST"}),s=await e.json();if(e.ok&&s.success){t&&t.close(),V(s.tunnelUrl||""),await eE(s.tunnelUrl)?(Z(!0),es(null)):(Z(!0),es({type:"warning",message:"Connected but not reachable yet."}));return}if(s.funnelNotEnabled)continue;if(s.error)return void es({type:"error",message:s.error})}catch{}}es({type:"error",message:"Timed out waiting for Funnel to be enabled."})},eD=async()=>{X(!0),es(null);try{let e=await fetch("/api/tunnel/tailscale-disable",{method:"POST"}),t=await e.json();e.ok?(Z(!1),V(""),eg(!1),es({type:"success",message:"Tailscale disabled"})):es({type:"error",message:t.error||"Failed to disable Tailscale"})}catch(e){es({type:"error",message:e.message})}finally{X(!1)}},eF=async()=>{es(null),eo([]),eu(!0),await eP()},eI=async()=>{if(b.trim())try{let e=await fetch("/api/keys",{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({name:b})}),t=await e.json();e.ok&&(j(t.key),await eC(),y(""),g(!1))}catch(e){console.log("Error creating key:",e)}},eO=async e=>{if(confirm("Delete this API key?"))try{(await fetch(`/api/keys/${e}`,{method:"DELETE"})).ok&&(s(t.filter(t=>t.id!==e)),ef(t=>{let s=new Set(t);return s.delete(e),s}))}catch(e){console.log("Error deleting key:",e)}},eW=async(e,t)=>{try{(await fetch(`/api/keys/${e}`,{method:"PUT",headers:{"Content-Type":"application/json"},body:JSON.stringify({isActive:t})})).ok&&s(s=>s.map(s=>s.id===e?{...s,isActive:t}:s))}catch(e){console.log("Error toggling key:",e)}},[eR,eq]=(0,l.useState)("/v1");return((0,l.useEffect)(()=>{eq(`${window.location.origin}/v1`)},[]),r)?(0,a.jsxs)("div",{className:"flex flex-col gap-8",children:[(0,a.jsx)(i.Qv,{}),(0,a.jsx)(i.Qv,{})]}):(0,a.jsxs)("div",{className:"flex flex-col gap-8",children:[(0,a.jsxs)(i.Zp,{children:[(0,a.jsx)("h2",{className:"text-lg font-semibold mb-4",children:"API Endpoint"}),(0,a.jsxs)("div",{className:"flex flex-col gap-2",children:[(0,a.jsx)(x,{label:"Local",url:eR,copyId:"local_url",copied:ej,onCopy:ek}),(0,a.jsxs)("div",{className:"flex items-center gap-2",children:[(0,a.jsx)("span",{className:`text-xs font-mono px-1.5 py-0.5 rounded shrink-0 min-w-[68px] text-center ${E?"bg-orange-100 dark:bg-orange-900/30 text-orange-600 dark:text-orange-400":"bg-sidebar text-text-muted"}`,children:"Tunnel"}),E&&!O?(0,a.jsxs)(a.Fragment,{children:[(0,a.jsx)(i.pd,{value:`${F||U}/v1`,readOnly:!0,className:"flex-1 font-mono text-sm"}),(0,a.jsx)("button",{onClick:()=>ek(`${F||U}/v1`,"tunnel_url"),className:"p-2 hover:bg-black/5 dark:hover:bg-white/5 rounded text-text-muted hover:text-primary transition-colors shrink-0",children:(0,a.jsx)("span",{className:"material-symbols-outlined text-[18px]",children:"tunnel_url"===ej?"check":"content_copy"})}),(0,a.jsx)("button",{onClick:()=>M(!0),className:"p-2 hover:bg-red-500/10 rounded text-red-500 transition-colors shrink-0",title:"Disable Tunnel",children:(0,a.jsx)("span",{className:"material-symbols-outlined text-[18px]",children:"power_settings_new"})})]}):O?(0,a.jsxs)(a.Fragment,{children:[(0,a.jsxs)("div",{className:"flex-1 flex items-center gap-2 px-3 py-1.5 rounded border border-border bg-input text-sm text-text-muted",children:[(0,a.jsx)("span",{className:"material-symbols-outlined animate-spin text-sm",children:"progress_activity"}),R||"Creating tunnel..."]}),(0,a.jsx)("button",{onClick:()=>{W(!1),q("")},className:"p-2 hover:bg-red-500/10 rounded text-red-500 transition-colors shrink-0",title:"Stop",children:(0,a.jsx)("span",{className:"material-symbols-outlined text-[18px]",children:"power_settings_new"})})]}):L?.type==="error"?(0,a.jsxs)(a.Fragment,{children:[(0,a.jsxs)("div",{className:"flex-1 flex items-center gap-2 px-3 py-1.5 rounded border border-red-300 dark:border-red-800 bg-red-500/5 text-sm text-red-600 dark:text-red-400",children:[(0,a.jsx)("span",{className:"material-symbols-outlined text-sm",children:"error"}),L.message]}),(0,a.jsx)(i.$n,{size:"sm",icon:"cloud_upload",onClick:()=>J(!0),children:"Enable"})]}):P?(0,a.jsxs)(a.Fragment,{children:[(0,a.jsxs)("div",{className:"flex-1 flex items-center gap-2 px-3 py-1.5 rounded border border-border bg-input text-sm text-text-muted",children:[(0,a.jsx)("span",{className:"material-symbols-outlined animate-spin text-sm",children:"progress_activity"}),"Checking..."]}),(0,a.jsx)("button",{onClick:()=>$(!1),className:"p-2 hover:bg-red-500/10 rounded text-red-500 transition-colors shrink-0",title:"Stop",children:(0,a.jsx)("span",{className:"material-symbols-outlined text-[18px]",children:"power_settings_new"})})]}):(0,a.jsx)(i.$n,{size:"sm",icon:"cloud_upload",onClick:()=>{k?J(!0):K({type:"error",message:'Security required: Enable "Require API key" before activating the tunnel.'})},className:"bg-linear-to-r from-primary to-blue-500 hover:from-primary-hover hover:to-blue-600 text-white!",children:"Enable"})]}),(0,a.jsxs)("div",{className:"flex items-center gap-2",children:[(0,a.jsx)("span",{className:`text-xs font-mono px-1.5 py-0.5 rounded shrink-0 min-w-[68px] text-center ${Q?"bg-purple-100 dark:bg-purple-900/30 text-purple-600 dark:text-purple-400":"bg-sidebar text-text-muted"}`,children:"Tailscale"}),Q&&!G?(0,a.jsxs)(a.Fragment,{children:[(0,a.jsx)(i.pd,{value:`${B}/v1`,readOnly:!0,className:"flex-1 font-mono text-sm"}),(0,a.jsx)("button",{onClick:()=>ek(`${B}/v1`,"ts_url"),className:"p-2 hover:bg-black/5 dark:hover:bg-white/5 rounded text-text-muted hover:text-primary transition-colors shrink-0",children:(0,a.jsx)("span",{className:"material-symbols-outlined text-[18px]",children:"ts_url"===ej?"check":"content_copy"})}),(0,a.jsx)("button",{onClick:()=>eg(!0),className:"p-2 hover:bg-red-500/10 rounded text-red-500 transition-colors shrink-0",title:"Disable Tailscale",children:(0,a.jsx)("span",{className:"material-symbols-outlined text-[18px]",children:"power_settings_new"})})]}):G||ex?(0,a.jsxs)(a.Fragment,{children:[(0,a.jsxs)("div",{className:"flex-1 flex items-center gap-2 px-3 py-1.5 rounded border border-border bg-input text-sm text-text-muted",children:[(0,a.jsx)("span",{className:"material-symbols-outlined animate-spin text-sm",children:"progress_activity"}),Y||"Connecting..."]}),(0,a.jsx)("button",{onClick:()=>{X(!1),em(!1),ee("")},className:"p-2 hover:bg-red-500/10 rounded text-red-500 transition-colors shrink-0",title:"Stop",children:(0,a.jsx)("span",{className:"material-symbols-outlined text-[18px]",children:"power_settings_new"})})]}):et?.type==="error"?(0,a.jsxs)(a.Fragment,{children:[(0,a.jsxs)("div",{className:"flex-1 flex items-center gap-2 px-3 py-1.5 rounded border border-red-300 dark:border-red-800 bg-red-500/5 text-sm text-red-600 dark:text-red-400",children:[(0,a.jsx)("span",{className:"material-symbols-outlined text-sm",children:"error"}),et.message]}),(0,a.jsx)(i.$n,{size:"sm",icon:"vpn_lock",onClick:eF,children:"Enable"})]}):(0,a.jsx)(i.$n,{size:"sm",icon:"vpn_lock",onClick:eF,className:"bg-linear-to-r from-indigo-500 to-purple-500 hover:from-indigo-600 hover:to-purple-600 text-white!",children:"Enable"})]})]}),(E||Q)&&(0,a.jsxs)("div",{className:"mt-4 flex flex-col gap-2",children:[!k&&(0,a.jsx)(u,{message:"Require API key is disabled — your endpoint is publicly accessible without authentication.",action:{label:"Enable",href:"#require-api-key"}}),(!v||!C)&&(0,a.jsx)(u,{message:v?"Dashboard uses the default password — change it in Profile settings.":"Require login is disabled — anyone can access your dashboard via tunnel.",action:{label:v?"Change password":"Enable",href:"/dashboard/profile"}})]}),(E||Q)&&(0,a.jsxs)("div",{className:"mt-4 pt-4 border-t border-border flex items-center gap-3",children:[(0,a.jsx)(i.lM,{checked:T,onChange:()=>ev(!T)}),(0,a.jsxs)("div",{className:"flex items-center gap-1.5",children:[(0,a.jsx)("p",{className:"font-medium text-sm",children:"Allow dashboard access via tunnel"}),(0,a.jsx)(h,{text:"When enabled, the dashboard can be accessed through your tunnel or Tailscale URL (login still required). When disabled, dashboard access via tunnel/Tailscale is completely blocked."})]})]})]}),(0,a.jsxs)(i.Zp,{id:"require-api-key",children:[(0,a.jsxs)("div",{className:"flex items-center justify-between mb-4",children:[(0,a.jsx)("h2",{className:"text-lg font-semibold",children:"API Keys"}),(0,a.jsx)(i.$n,{icon:"add",onClick:()=>g(!0),children:"Create Key"})]}),(0,a.jsxs)("div",{className:"flex items-center justify-between pb-4 mb-4 border-b border-border",children:[(0,a.jsxs)("div",{children:[(0,a.jsx)("p",{className:"font-medium",children:"Require API key"}),(0,a.jsx)("p",{className:"text-sm text-text-muted",children:"Requests without a valid key will be rejected"})]}),(0,a.jsx)(i.lM,{checked:k,onChange:()=>eN(!k)})]}),0===t.length?(0,a.jsxs)("div",{className:"text-center py-12",children:[(0,a.jsx)("div",{className:"inline-flex items-center justify-center w-16 h-16 rounded-full bg-primary/10 text-primary mb-4",children:(0,a.jsx)("span",{className:"material-symbols-outlined text-[32px]",children:"vpn_key"})}),(0,a.jsx)("p",{className:"text-text-main font-medium mb-1",children:"No API keys yet"}),(0,a.jsx)("p",{className:"text-sm text-text-muted mb-4",children:"Create your first API key to get started"}),(0,a.jsx)(i.$n,{icon:"add",onClick:()=>g(!0),children:"Create Key"})]}):(0,a.jsx)("div",{className:"flex flex-col",children:t.map(e=>{var t;return(0,a.jsxs)("div",{className:`group flex items-center justify-between py-3 border-b border-black/[0.03] dark:border-white/[0.03] last:border-b-0 ${!1===e.isActive?"opacity-60":""}`,children:[(0,a.jsxs)("div",{className:"flex-1 min-w-0",children:[(0,a.jsx)("p",{className:"text-sm font-medium",children:e.name}),(0,a.jsxs)("div",{className:"flex items-center gap-2 mt-1",children:[(0,a.jsx)("code",{className:"text-xs text-text-muted font-mono",children:ey.has(e.id)?e.key:(t=e.key)?t.length>8?t.slice(0,8)+"...":t:""}),(0,a.jsx)("button",{onClick:()=>{var t;return t=e.id,void ef(e=>{let s=new Set(e);return s.has(t)?s.delete(t):s.add(t),s})},className:"p-1 hover:bg-black/5 dark:hover:bg-white/5 rounded text-text-muted hover:text-primary opacity-0 group-hover:opacity-100 transition-all",title:ey.has(e.id)?"Hide key":"Show key",children:(0,a.jsx)("span",{className:"material-symbols-outlined text-[14px]",children:ey.has(e.id)?"visibility_off":"visibility"})}),(0,a.jsx)("button",{onClick:()=>ek(e.key,e.id),className:"p-1 hover:bg-black/5 dark:hover:bg-white/5 rounded text-text-muted hover:text-primary opacity-0 group-hover:opacity-100 transition-all",children:(0,a.jsx)("span",{className:"material-symbols-outlined text-[14px]",children:ej===e.id?"check":"content_copy"})})]}),(0,a.jsxs)("p",{className:"text-xs text-text-muted mt-1",children:["Created ",new Date(e.createdAt).toLocaleDateString()]}),!1===e.isActive&&(0,a.jsx)("p",{className:"text-xs text-orange-500 mt-1",children:"Paused"})]}),(0,a.jsxs)("div",{className:"flex items-center gap-2",children:[(0,a.jsx)(i.lM,{size:"sm",checked:e.isActive??!0,onChange:t=>{e.isActive&&!t?confirm(`Pause API key "${e.name}"?
2
+
3
+ This key will stop working immediately but can be resumed later.`)&&eW(e.id,t):eW(e.id,t)},title:e.isActive?"Pause key":"Resume key"}),(0,a.jsx)("button",{onClick:()=>eO(e.id),className:"p-2 hover:bg-red-500/10 rounded text-red-500 opacity-0 group-hover:opacity-100 transition-all",children:(0,a.jsx)("span",{className:"material-symbols-outlined text-[18px]",children:"delete"})})]})]},e.id)})})]}),(0,a.jsx)(i.aF,{isOpen:p,title:"Create API Key",onClose:()=>{g(!1),y("")},children:(0,a.jsxs)("div",{className:"flex flex-col gap-4",children:[(0,a.jsx)(i.pd,{label:"Key Name",value:b,onChange:e=>y(e.target.value),placeholder:"Production Key"}),(0,a.jsxs)("div",{className:"flex gap-2",children:[(0,a.jsx)(i.$n,{onClick:eI,fullWidth:!0,disabled:!b.trim(),children:"Create"}),(0,a.jsx)(i.$n,{onClick:()=>{g(!1),y("")},variant:"ghost",fullWidth:!0,children:"Cancel"})]})]})}),(0,a.jsx)(i.aF,{isOpen:!!f,title:"API Key Created",onClose:()=>j(null),children:(0,a.jsxs)("div",{className:"flex flex-col gap-4",children:[(0,a.jsxs)("div",{className:"bg-yellow-50 dark:bg-yellow-900/20 border border-yellow-200 dark:border-yellow-800 rounded-lg p-4",children:[(0,a.jsx)("p",{className:"text-sm text-yellow-800 dark:text-yellow-200 mb-2 font-medium",children:"Save this key now!"}),(0,a.jsx)("p",{className:"text-sm text-yellow-700 dark:text-yellow-300",children:"This is the only time you will see this key. Store it securely."})]}),(0,a.jsxs)("div",{className:"flex gap-2",children:[(0,a.jsx)(i.pd,{value:f||"",readOnly:!0,className:"flex-1 font-mono text-sm"}),(0,a.jsx)(i.$n,{variant:"secondary",icon:"created_key"===ej?"check":"content_copy",onClick:()=>ek(f,"created_key"),children:"created_key"===ej?"Copied!":"Copy"})]}),(0,a.jsx)(i.$n,{onClick:()=>j(null),fullWidth:!0,children:"Done"})]})}),(0,a.jsx)(i.aF,{isOpen:z,title:"Enable Tunnel",onClose:()=>J(!1),children:(0,a.jsxs)("div",{className:"flex flex-col gap-4",children:[(0,a.jsx)("div",{className:"bg-blue-50 dark:bg-blue-900/20 border border-blue-200 dark:border-blue-800 rounded-lg p-4",children:(0,a.jsxs)("div",{className:"flex items-start gap-3",children:[(0,a.jsx)("span",{className:"material-symbols-outlined text-blue-600 dark:text-blue-400",children:"cloud_upload"}),(0,a.jsxs)("div",{children:[(0,a.jsx)("p",{className:"text-sm text-blue-800 dark:text-blue-200 font-medium mb-1",children:"Cloudflare Tunnel"}),(0,a.jsx)("p",{className:"text-sm text-blue-700 dark:text-blue-300",children:"Expose your local 9Router to the internet. No port forwarding, no static IP needed. Share endpoint URL with your team or use it in Cursor, Cline, and other AI tools from anywhere."})]})]})}),(0,a.jsx)("div",{className:"grid grid-cols-2 gap-3",children:c.map(e=>(0,a.jsxs)("div",{className:"flex flex-col items-center text-center p-3 rounded-lg bg-sidebar/50",children:[(0,a.jsx)("span",{className:"material-symbols-outlined text-xl text-primary mb-1",children:e.icon}),(0,a.jsx)("p",{className:"text-xs font-semibold",children:e.title}),(0,a.jsx)("p",{className:"text-xs text-text-muted",children:e.desc})]},e.title))}),(0,a.jsx)("p",{className:"text-xs text-text-muted",children:"Requires outbound port 7844 (TCP/UDP). Connection may take 10-30s."}),(0,a.jsxs)("div",{className:"flex gap-2",children:[(0,a.jsx)(i.$n,{onClick:eT,fullWidth:!0,className:"bg-linear-to-r from-primary to-blue-500 hover:from-primary-hover hover:to-blue-600 text-white!",children:"Start Tunnel"}),(0,a.jsx)(i.$n,{onClick:()=>J(!1),variant:"ghost",fullWidth:!0,children:"Cancel"})]})]})}),(0,a.jsx)(i.aF,{isOpen:H,title:"Disable Tunnel",onClose:()=>!O&&M(!1),children:(0,a.jsxs)("div",{className:"flex flex-col gap-4",children:[(0,a.jsx)("p",{className:"text-sm text-text-muted",children:"The Cloudflare tunnel will be disconnected. Remote access via tunnel URL will stop working."}),(0,a.jsxs)("div",{className:"flex gap-2",children:[(0,a.jsx)(i.$n,{onClick:e_,fullWidth:!0,disabled:O,className:"bg-red-500! hover:bg-red-600! text-white!",children:O?"Disabling...":"Disable"}),(0,a.jsx)(i.$n,{onClick:()=>M(!1),variant:"ghost",fullWidth:!0,disabled:O,children:"Cancel"})]})]})}),(0,a.jsx)(i.aF,{isOpen:eh,title:"Tailscale Funnel",onClose:()=>{er||(eu(!1),ed(""),es(null))},children:(0,a.jsxs)("div",{className:"flex flex-col gap-4",children:[null===ea&&(0,a.jsxs)("p",{className:"text-sm text-text-muted flex items-center gap-2",children:[(0,a.jsx)("span",{className:"material-symbols-outlined animate-spin text-sm",children:"progress_activity"}),"Checking..."]}),!1===ea&&!er&&(0,a.jsxs)("div",{className:"flex flex-col gap-3",children:[(0,a.jsx)("p",{className:"text-sm text-text-muted",children:"Tailscale is not installed. Install it to enable Funnel."}),(0,a.jsxs)("div",{className:"flex gap-2",children:[(0,a.jsx)(i.$n,{onClick:e$,fullWidth:!0,className:"bg-linear-to-r from-indigo-500 to-purple-500 hover:from-indigo-600 hover:to-purple-600 text-white!",children:"Install Tailscale"}),(0,a.jsx)(i.$n,{onClick:()=>eu(!1),variant:"ghost",fullWidth:!0,children:"Cancel"})]})]}),er&&(0,a.jsxs)("div",{className:"flex flex-col gap-2",children:[(0,a.jsxs)("div",{className:"flex items-center gap-2 text-sm text-text-muted",children:[(0,a.jsx)("span",{className:"material-symbols-outlined animate-spin text-sm",children:"progress_activity"}),"Installing Tailscale..."]}),ei.length>0&&(0,a.jsx)("div",{ref:eb,className:"bg-black/5 dark:bg-white/5 rounded p-2 max-h-40 overflow-y-auto font-mono text-xs text-text-muted",children:ei.map((e,t)=>(0,a.jsx)("div",{children:e},t))})]}),!0===ea&&!er&&(0,a.jsxs)("div",{className:"flex flex-col gap-3",children:[(0,a.jsxs)("div",{className:"flex items-center gap-2 text-sm text-green-600 dark:text-green-400",children:[(0,a.jsx)("span",{className:"material-symbols-outlined text-[16px]",children:"check_circle"}),"Tailscale installed"]}),(0,a.jsxs)("div",{className:"flex gap-2",children:[(0,a.jsx)(i.$n,{onClick:()=>{let e=window.open("","tailscale_auth","width=600,height=700");e&&e.document.write("<p style='font-family:sans-serif;text-align:center;margin-top:40px'>Connecting to Tailscale...</p>"),eA(e)},fullWidth:!0,className:"bg-linear-to-r from-indigo-500 to-purple-500 hover:from-indigo-600 hover:to-purple-600 text-white!",children:"Connect"}),(0,a.jsx)(i.$n,{onClick:()=>eu(!1),variant:"ghost",fullWidth:!0,children:"Cancel"})]})]}),et&&(0,a.jsx)(m,{status:et})]})}),(0,a.jsx)(i.aF,{isOpen:ep,title:"Disable Tailscale",onClose:()=>!G&&eg(!1),children:(0,a.jsxs)("div",{className:"flex flex-col gap-4",children:[(0,a.jsx)("p",{className:"text-sm text-text-muted",children:"Tailscale Funnel will be stopped. Remote access via Tailscale URL will stop working."}),(0,a.jsxs)("div",{className:"flex gap-2",children:[(0,a.jsx)(i.$n,{onClick:eD,fullWidth:!0,disabled:G,className:"bg-red-500! hover:bg-red-600! text-white!",children:G?"Disabling...":"Disable"}),(0,a.jsx)(i.$n,{onClick:()=>eg(!1),variant:"ghost",fullWidth:!0,disabled:G,children:"Cancel"})]})]})})]})}function x({label:e,url:t,copyId:s,copied:l,onCopy:r,badge:n,actions:o}){return(0,a.jsxs)("div",{className:"flex items-center gap-2",children:[(0,a.jsx)("span",{className:`text-xs font-mono px-1.5 py-0.5 rounded shrink-0 min-w-[68px] text-center ${"CF"===n?"bg-orange-100 dark:bg-orange-900/30 text-orange-600 dark:text-orange-400":"TS"===n?"bg-purple-100 dark:bg-purple-900/30 text-purple-600 dark:text-purple-400":"bg-sidebar text-text-muted"}`,children:e}),(0,a.jsx)(i.pd,{value:t,readOnly:!0,className:"flex-1 font-mono text-sm"}),(0,a.jsx)("button",{onClick:()=>r(t,s),className:"p-2 hover:bg-black/5 dark:hover:bg-white/5 rounded text-text-muted hover:text-primary transition-colors shrink-0",children:(0,a.jsx)("span",{className:"material-symbols-outlined text-[18px]",children:l===s?"check":"content_copy"})}),o]})}function m({status:e,className:t=""}){return(0,a.jsx)("div",{className:`p-2 rounded text-sm ${t} ${"success"===e.type?"bg-green-500/10 text-green-600 dark:text-green-400":"warning"===e.type?"bg-yellow-500/10 text-yellow-600 dark:text-yellow-400":"info"===e.type?"bg-blue-500/10 text-blue-600 dark:text-blue-400":"bg-red-500/10 text-red-600 dark:text-red-400"}`,children:e.message.split(/(https?:\/\/[^\s]+)/g).map((e,t)=>/^https?:\/\//.test(e)?(0,a.jsx)("a",{href:e,target:"_blank",rel:"noreferrer",className:"underline font-medium",children:e},t):e)})}function h({text:e}){return(0,a.jsxs)("span",{className:"relative group inline-flex items-center",children:[(0,a.jsx)("span",{className:"material-symbols-outlined text-[14px] text-text-muted cursor-help",children:"help"}),(0,a.jsx)("span",{className:"pointer-events-none absolute left-5 top-1/2 -translate-y-1/2 z-50 w-64 rounded bg-gray-900 dark:bg-gray-800 text-white text-xs px-2.5 py-1.5 opacity-0 group-hover:opacity-100 transition-opacity shadow-lg",children:e})]})}function u({message:e,action:t}){return(0,a.jsxs)("div",{className:"flex items-center gap-2 px-3 py-2 rounded-lg bg-amber-500/10 border border-amber-500/20 text-amber-700 dark:text-amber-400",children:[(0,a.jsx)("span",{className:"material-symbols-outlined text-[16px] shrink-0 mt-0.5",children:"warning"}),(0,a.jsx)("p",{className:"text-xs flex-1",children:e}),t&&(0,a.jsx)("a",{href:t.href,className:"text-xs font-medium underline shrink-0 hover:opacity-80",onClick:t.href.startsWith("#")?e=>{e.preventDefault(),document.getElementById(t.href.slice(1))?.scrollIntoView({behavior:"smooth"})}:void 0,children:t.label})]})}d.propTypes={machineId:n().string.isRequired}},78035:(e,t,s)=>{Promise.resolve().then(s.bind(s,64795))}}]);
@@ -1 +1 @@
1
- (self.webpackChunk_N_E=self.webpackChunk_N_E||[]).push([[4465],{11710:(e,t,r)=>{Promise.resolve().then(r.bind(r,72600))},72600:(e,t,r)=>{"use strict";r.d(t,{default:()=>g});var a=r(73365),s=r(1521),i=r(74156),l=r(69633),n=r(82495);let o="basic-chat.sessions",d="basic-chat.activeSessionId",c="basic-chat.activeProviderId",m="basic-chat.draft";function u(){return globalThis.crypto?.randomUUID?globalThis.crypto.randomUUID():`chat_${Date.now()}_${Math.random().toString(16).slice(2)}`}function h(e){if("string"==typeof e)return e;if(null==e)return"";if(Array.isArray(e))return e.map(h).filter(Boolean).join(" ");if("object"==typeof e){if("string"==typeof e.message)return e.message;if("string"==typeof e.error)return e.error;try{return JSON.stringify(e)}catch{}}return String(e)}function p(e=""){let t=h(e).replace(/\s+/g," ").trim();return t?t.length>52?`${t.slice(0,52).trimEnd()}…`:t:"New chat"}async function x(e){return await new Promise((t,r)=>{let a=new FileReader;a.onload=()=>t(String(a.result||"")),a.onerror=()=>r(a.error||Error("Failed to read file")),a.readAsDataURL(e)})}function f(e){return e?.name||function(e=""){return String(e).replace(/[-_]/g," ").replace(/\b\w/g,e=>e.toUpperCase()).trim()||"Unknown"}(e?.provider||e?.id||"provider")}function g(){let[e,t]=(0,s.useState)([]),[r,g]=(0,s.useState)(!0),[w,v]=(0,s.useState)(""),[b,N]=(0,s.useState)([]),[y,j]=(0,s.useState)(""),[S,I]=(0,s.useState)(""),[A,k]=(0,s.useState)(""),[C,D]=(0,s.useState)(""),[$,O]=(0,s.useState)([]),[_,M]=(0,s.useState)(!1),[E,T]=(0,s.useState)(""),[U,R]=(0,s.useState)(""),[z,q]=(0,s.useState)(!1),[P,J]=(0,s.useState)(!1),[K,B]=(0,s.useState)(!1),L=(0,s.useRef)(null),F=(0,s.useRef)(null),W=(0,s.useRef)(!1),H=(0,s.useRef)(null),Y=(0,s.useRef)(null);(0,s.useEffect)(()=>{try{let e=function(e,t){try{return JSON.parse(e)}catch{return t}}(globalThis.localStorage.getItem(o),[]);N(Array.isArray(e)?e.map(e=>({...e,messages:Array.isArray(e.messages)?e.messages:[]})):[]),j(globalThis.localStorage.getItem(d)||""),I(globalThis.localStorage.getItem(c)||""),D(globalThis.localStorage.getItem(m)||"")}catch{}finally{q(!0)}},[]),(0,s.useEffect)(()=>{let e=!1;return async function(){g(!0),v("");try{let r=await fetch("/api/providers",{cache:"no-store"}),a=await r.json().catch(()=>({})),s=Array.isArray(a.connections)?a.connections.filter(e=>e?.isActive!==!1):[];if(0===s.length){e||(t([]),v("Chưa c\xf3 provider n\xe0o được connect."));return}let i=new Map;for(let e of s){let t=e.provider||e.id,r=f(e),a=(0,n.mq)(t)?"openai-compatible":(0,n.gb)(t)?"anthropic-compatible":t;i.has(t)||i.set(t,{providerId:t,providerName:r,providerType:a,connections:[],models:[]});let s=i.get(t);s.providerName=s.providerName||r,s.providerType=s.providerType||a,s.connections.push(e);let o=(0,l.KC)(t).map(t=>t?.id?{id:`${e.provider}/${t.id}`,requestModel:`${e.provider}/${t.id}`,name:t.name||t.id,providerId:e.provider,providerName:f(e),source:"static"}:null).filter(Boolean);s.models.push(...o)}for(let e of(await Promise.all(s.map(async e=>{try{let t=await fetch(`/api/providers/${e.id}/models`,{cache:"no-store"}),r=await t.json().catch(()=>({}));if(!t.ok)return{connection:e,models:[]};let a=(Array.isArray(r?.models)?r.models:Array.isArray(r?.data)?r.data:Array.isArray(r?.results)?r.results:Array.isArray(r)?r:[]).map(t=>(function(e,t){let r="string"==typeof e?e:e?.id||e?.name||e?.model||"";if(!r)return null;let a="string"==typeof e?e:e?.name||e?.displayName||r,s=r;return((0,n.mq)(t.provider)||(0,n.gb)(t.provider))&&!r.includes("/")&&(s=`${t.provider}/${r}`),{id:s,requestModel:s,name:a,providerId:t.provider,providerName:f(t),source:"live"}})(t,e)).filter(Boolean);return{connection:e,models:a}}catch{return{connection:e,models:[]}}})))){let t=e.connection.provider||e.connection.id,r=i.get(t);r&&r.models.push(...e.models)}let o=Array.from(i.values()).map(e=>({...e,models:(function(e){let t=new Map;for(let r of e)r?.id&&(t.has(r.id)||t.set(r.id,r));return Array.from(t.values())})(e.models).sort((e,t)=>e.name.localeCompare(t.name))})).filter(e=>e.models.length>0).sort((e,t)=>e.providerName.localeCompare(t.providerName));e||(t(o),0===o.length&&v("Đ\xe3 c\xf3 provider connect nhưng chưa lấy được model n\xe0o."))}catch(r){e||(v(h(r?.message)||"Kh\xf4ng thể tải danh s\xe1ch provider/model."),t([]))}finally{e||g(!1)}}(),()=>{e=!0}},[]),(0,s.useEffect)(()=>{let e=e=>{H.current&&!H.current.contains(e.target)&&J(!1),Y.current&&!Y.current.contains(e.target)&&B(!1)};return document.addEventListener("mousedown",e),()=>document.removeEventListener("mousedown",e)},[]);let G=(0,s.useMemo)(()=>{let t=new Map;for(let r of e)for(let e of r.models)t.set(e.id,{...e,providerId:r.providerId,providerName:r.providerName});return t},[e]),Q=(0,s.useMemo)(()=>e.find(e=>e.providerId===S)||e[0]||null,[e,S]),V=(0,s.useMemo)(()=>{if(A&&G.has(A))return G.get(A);if(y){let e=b.find(e=>e.id===y);if(e?.modelId&&G.has(e.modelId))return G.get(e.modelId)}return Q?.models?.[0]||null},[A,G,Q,b,y]),X=(0,s.useMemo)(()=>b.find(e=>e.id===y)||null,[b,y]),Z=X?.messages||[],ee=(0,s.useMemo)(()=>[...b].sort((e,t)=>new Date(t.updatedAt).getTime()-new Date(e.updatedAt).getTime()),[b]),et=!_&&!!V&&(C.trim().length>0||$.length>0);(0,s.useEffect)(()=>{if(z)try{globalThis.localStorage.setItem(o,JSON.stringify(b)),globalThis.localStorage.setItem(d,y),globalThis.localStorage.setItem(c,S),globalThis.localStorage.setItem(m,C)}catch{}},[z,b,y,S,C]),(0,s.useEffect)(()=>{if(!z||r||W.current||0===e.length)return;let t=e.find(e=>e.providerId===S)||e[0],a=A&&G.has(A)?G.get(A):t.models[0];if(b.length>0){let e=b.find(e=>e.id===y)||b[0],r=e?.modelId&&G.has(e.modelId)?G.get(e.modelId):a;W.current=!0,j(e.id),I(r?.providerId||t.providerId),k(r?.id||a.id);return}let s={id:u(),title:"New chat",providerId:t.providerId,providerName:t.providerName,modelId:a.id,modelName:a.name,createdAt:new Date().toISOString(),updatedAt:new Date().toISOString(),messages:[]};W.current=!0,N([s]),j(s.id),I(t.providerId),k(a.id)},[z,r,e,G,b,y,S,A]);let er=(e,t)=>{N(r=>r.map(r=>r.id===e?t({...r,messages:Array.isArray(r.messages)?r.messages.map(e=>({...e})):[]}):r))},ea=e=>e?{id:u(),title:"New chat",providerId:e.providerId,providerName:e.providerName,modelId:e.id,modelName:e.name,createdAt:new Date().toISOString(),updatedAt:new Date().toISOString(),messages:[]}:null,es=async e=>{let t=Array.from(e.target.files||[]);if(0===t.length)return;let r=t.filter(e=>e.type.startsWith("image/"));if(0===r.length){e.target.value="";return}let a=await Promise.all(r.map(async e=>({id:u(),name:e.name,type:e.type,size:e.size,dataUrl:await x(e)})));O(e=>[...e,...a]),e.target.value=""},ei=async()=>{let e=V||Q?.models?.[0]||null;if(!e)return;let t=C.trim();if(!t&&0===$.length)return;let r=y,a=b.find(e=>e.id===r);if(!a){if(!(a=ea(e)))return;r=a.id,N(e=>[a,...e]),j(r)}let s={id:u(),role:"user",content:t,attachments:$.map(e=>({id:e.id,name:e.name,type:e.type,dataUrl:e.dataUrl})),createdAt:new Date().toISOString()},i=u(),l={id:i,role:"assistant",content:"",createdAt:new Date().toISOString(),status:"streaming"},n=[...a.messages||[],s,l];N(a=>a.map(a=>a.id===r?{...a,providerId:e.providerId,providerName:e.providerName,modelId:e.id,modelName:e.name,messages:n,updatedAt:new Date().toISOString(),title:"New chat"===a.title?p(t):a.title}:a)),D(""),O([]),M(!0),T(i),R(""),F.current?.abort(),F.current=new AbortController;let o=n.filter(e=>"assistant"!==e.role||e.id!==i).map(e=>({role:e.role,content:"user"===e.role?function(e){let t=h(e.content).trim(),r=Array.isArray(e.attachments)?e.attachments:[];if(0===r.length)return t;let a=[];for(let e of(t&&a.push({type:"text",text:t}),r))e?.dataUrl&&a.push({type:"image_url",image_url:{url:e.dataUrl}});return a.length>0?a:t}(e):e.content}));try{var d;let a,s=await fetch("/api/dashboard/chat/completions",{method:"POST",headers:{"Content-Type":"application/json",Accept:"text/event-stream"},body:JSON.stringify({model:e.requestModel||e.id,messages:o,stream:!0}),signal:F.current.signal});if(!s.ok){let e=await s.json().catch(()=>({}));throw Error(h(e.error||e.message||`Request failed (${s.status})`))}let l=s.body?.getReader();if(!l){let e=await s.json().catch(()=>({})),t=h(e?.choices?.[0]?.message?.content||e?.output_text||e?.error||e?.message||"");er(r,e=>({...e,messages:e.messages.map(e=>e.id===i?{...e,content:t,status:"done"}:e),updatedAt:new Date().toISOString()}));return}let n=new TextDecoder,c="",m="";for(;;){let{value:e,done:t}=await l.read();if(t)break;let a=(c+=n.decode(e,{stream:!0})).split(/\r?\n/);for(let e of(c=a.pop()||"",a)){let t=e.trim();if(!t.startsWith("data:"))continue;let a=t.slice(5).trim();if(a&&"[DONE]"!==a)try{let e=JSON.parse(a),t=function(e){if(!e||"object"!=typeof e)return"";let t=e.choices?.[0];return[(t?.delta||{}).content,t?.message?.content,e.output_text,e.text].map(h).filter(Boolean)[0]||""}(e);if(!t)continue;m+=t,R(m),er(r,e=>({...e,messages:e.messages.map(e=>e.id===i?{...e,content:m,status:"streaming"}:e),updatedAt:new Date().toISOString()}))}catch{}}}er(r,e=>({...e,messages:e.messages.map(e=>e.id===i?{...e,content:m||e.content,status:"done"}:e),updatedAt:new Date().toISOString()})),d=r,a=p(t),er(d,e=>({...e,title:"New chat"===e.title?a:e.title,updatedAt:new Date().toISOString()}))}catch(e){if("AbortError"!==e.name){let t=h(e?.message||e);er(r,e=>({...e,messages:e.messages.map(e=>e.id===i?{...e,content:e.content||`Error: ${t}`,status:"error"}:e),updatedAt:new Date().toISOString()})),v(t||"Kh\xf4ng thể gửi tin nhắn.")}}finally{M(!1),T(""),R(""),F.current=null}},el=V?`${V.name}`:"Select model",en=V?V.requestModel:"Choose from connected providers";return(0,a.jsx)("div",{className:"relative flex-1 flex flex-col h-full min-h-0 min-w-0 bg-[#212121] text-white overflow-hidden",children:(0,a.jsxs)("div",{className:"relative mx-auto flex flex-1 h-full min-h-0 w-full max-w-4xl flex-col",children:[(0,a.jsxs)("div",{className:"flex shrink-0 items-center justify-between gap-3 px-4 py-3 lg:px-6",children:[(0,a.jsxs)("div",{ref:H,className:"relative",children:[(0,a.jsx)("button",{type:"button",onClick:()=>J(e=>!e),className:"flex items-center gap-3 rounded-2xl border border-white/10 bg-white/5 px-4 py-3 text-left transition hover:bg-white/8",children:(0,a.jsxs)("div",{className:"min-w-0",children:[(0,a.jsxs)("div",{className:"flex items-center gap-2",children:[(0,a.jsx)("span",{className:"text-sm font-semibold text-white",children:el}),(0,a.jsx)("span",{className:"material-symbols-outlined text-[18px] text-white/70",children:"expand_more"})]}),(0,a.jsx)("p",{className:"truncate text-xs text-white/55",children:en})]})}),P?(0,a.jsxs)("div",{className:"absolute left-0 top-[calc(100%+10px)] z-30 w-[min(520px,calc(100vw-2rem))] overflow-hidden rounded-[20px] border border-white/10 bg-[#262626] shadow-2xl shadow-black/50",children:[(0,a.jsxs)("div",{className:"border-b border-white/10 px-4 py-3",children:[(0,a.jsx)("p",{className:"text-xs uppercase tracking-[0.22em] text-white/45",children:"Models"}),(0,a.jsx)("p",{className:"text-sm text-white/75",children:"Chỉ lấy từ provider đ\xe3 connect"})]}),(0,a.jsx)("div",{className:"max-h-[60vh] overflow-y-auto p-2 custom-scrollbar",children:e.map(e=>(0,a.jsxs)("div",{className:"mb-2 rounded-[16px] border border-white/10 bg-black/20 p-2",children:[(0,a.jsxs)("div",{className:"flex items-center justify-between px-2 py-2",children:[(0,a.jsx)("p",{className:"text-sm font-semibold text-white",children:e.providerName}),(0,a.jsx)(i.Ex,{size:"sm",variant:"default",children:e.models.length})]}),(0,a.jsx)("div",{className:"grid gap-2 sm:grid-cols-2",children:e.models.map(e=>{let t=e.id===A;return(0,a.jsx)("button",{type:"button",onClick:()=>(e=>{let t=G.get(e);if(!t)return;let r=b.find(e=>e.id===y);if(r&&r.messages.length>0){let e=ea(t);if(!e)return;N(t=>[e,...t]),j(e.id)}else if(r)N(e=>e.map(e=>e.id===r.id?{...e,providerId:t.providerId,providerName:t.providerName,modelId:t.id,modelName:t.name}:e)),j(r.id);else{let e=ea(t);if(!e)return;N(t=>[e,...t]),j(e.id)}I(t.providerId),k(t.id),J(!1)})(e.id),className:`rounded-[14px] border px-3 py-3 text-left transition ${t?"border-blue-400/40 bg-blue-500/15":"border-white/10 bg-white/5 hover:bg-white/8"}`,children:(0,a.jsxs)("div",{className:"flex items-start justify-between gap-3",children:[(0,a.jsxs)("div",{className:"min-w-0",children:[(0,a.jsx)("p",{className:"truncate text-sm font-medium text-white",children:e.name}),(0,a.jsx)("p",{className:"truncate text-[11px] text-white/45",children:e.requestModel})]}),t?(0,a.jsx)("span",{className:"material-symbols-outlined text-[18px] text-blue-300",children:"check_circle"}):null]})},e.id)})})]},e.providerId))})]}):null]}),(0,a.jsxs)("div",{className:"flex items-center gap-2",children:[(0,a.jsx)("button",{type:"button",onClick:()=>B(e=>!e),className:"rounded-2xl border border-white/10 bg-white/5 px-4 py-3 text-sm text-white/80 transition hover:bg-white/8",children:"History"}),(0,a.jsx)(i.$n,{variant:"ghost",size:"sm",icon:"delete",onClick:()=>{if(!y)return;let e=b.filter(e=>e.id!==y),t=e[0]||null;N(e),t?(j(t.id),I(t.providerId),k(t.modelId)):(j(""),I(""),k(""))},disabled:!y||0===b.length,children:"Clear"})]})]}),K?(0,a.jsxs)("div",{ref:Y,className:"absolute right-4 top-[72px] z-20 w-[min(360px,calc(100vw-2rem))] rounded-[20px] border border-white/10 bg-[#262626] p-2 shadow-2xl shadow-black/50 lg:right-6",children:[(0,a.jsx)("div",{className:"px-3 py-2",children:(0,a.jsx)("p",{className:"text-xs uppercase tracking-[0.22em] text-white/45",children:"Recent chats"})}),(0,a.jsx)("div",{className:"max-h-[48vh] space-y-2 overflow-y-auto p-1 custom-scrollbar",children:0===ee.length?(0,a.jsx)("div",{className:"rounded-[16px] border border-dashed border-white/10 bg-white/5 p-4 text-sm text-white/55",children:"Chưa c\xf3 cuộc tr\xf2 chuyện n\xe0o."}):ee.map(e=>{let t=e.id===y,r=[...e.messages||[]].reverse().find(e=>"user"===e.role)||e.messages?.[0];return(0,a.jsx)("button",{type:"button",onClick:()=>{var t;let r;return t=e.id,void((r=b.find(e=>e.id===t))&&(j(t),I(r.providerId||S),k(r.modelId||A),B(!1)))},className:`w-full rounded-[16px] border px-3 py-3 text-left transition ${t?"border-blue-400/40 bg-blue-500/15":"border-white/10 bg-white/5 hover:bg-white/8"}`,children:(0,a.jsxs)("div",{className:"flex items-start justify-between gap-3",children:[(0,a.jsxs)("div",{className:"min-w-0 flex-1",children:[(0,a.jsx)("p",{className:"truncate text-sm font-medium text-white",children:e.title}),(0,a.jsx)("p",{className:"mt-1 truncate text-xs text-white/50",children:h(r?.content)||"Empty chat"})]}),(0,a.jsx)("span",{className:"text-[10px] text-white/40 shrink-0",children:function(e){if(!e)return"Now";let t=new Date(e).getTime();if(Number.isNaN(t))return"Now";let r=Math.max(1,Math.round((Date.now()-t)/6e4));if(r<60)return`${r}m`;let a=Math.round(r/60);return a<24?`${a}h`:`${Math.round(a/24)}d`}(e.updatedAt)})]})},e.id)})})]}):null,w?(0,a.jsx)("div",{className:"mt-4 rounded-[18px] border border-rose-500/20 bg-rose-500/10 px-4 py-3 text-rose-100",children:(0,a.jsxs)("div",{className:"flex items-start gap-3",children:[(0,a.jsx)("span",{className:"material-symbols-outlined text-[20px]",children:"error"}),(0,a.jsx)("p",{className:"text-sm leading-6",children:w})]})}):null,(0,a.jsxs)("div",{className:"flex flex-1 flex-col min-h-0",children:[(0,a.jsxs)("div",{className:"flex-1 overflow-y-auto py-4 custom-scrollbar",children:[0===Z.length?(0,a.jsx)("div",{className:"flex min-h-[50vh] items-center justify-center px-4 text-center",children:(0,a.jsxs)("div",{className:"max-w-xl space-y-4",children:[(0,a.jsx)("div",{className:"mx-auto flex size-16 items-center justify-center rounded-[20px] border border-white/10 bg-white/5 text-white/80",children:(0,a.jsx)("span",{className:"material-symbols-outlined text-[30px]",children:"chat"})}),(0,a.jsxs)("div",{className:"space-y-2",children:[(0,a.jsx)("h2",{className:"text-2xl font-semibold text-white",children:"Start a conversation"}),(0,a.jsx)("p",{className:"text-sm leading-6 text-white/60",children:"Simple chat interface to interact with any AI model from connected providers. Select a model and start chatting!"})]})]})}):null,(0,a.jsx)("div",{className:"mx-auto flex w-full max-w-3xl flex-col gap-4 px-4",children:Z.map(e=>{let t="user"===e.role,r="assistant"===e.role,s=r&&e.id===E&&"streaming"===e.status,i=h(e.content)||(r?U:"");return(0,a.jsx)("div",{className:`flex w-full ${t?"justify-end":"justify-start"} mb-6`,children:(0,a.jsxs)("div",{className:`max-w-[min(88%,42rem)] ${t?"rounded-3xl bg-[#2f2f2f] px-5 py-3.5 text-white":"text-white/90"}`,children:[(0,a.jsx)("div",{className:"mb-1 flex items-center justify-between gap-3",children:(0,a.jsx)("span",{className:"text-xs font-semibold",children:t?"You":V?.name||"Assistant"})}),e.attachments?.length?(0,a.jsx)("div",{className:"mb-3 grid grid-cols-2 gap-2 sm:grid-cols-3 mt-2",children:e.attachments.map(e=>(0,a.jsx)("a",{href:e.dataUrl,target:"_blank",rel:"noreferrer",className:"overflow-hidden rounded-[18px] border border-white/10 bg-black/20",children:(0,a.jsx)("img",{src:e.dataUrl,alt:e.name,className:"h-28 w-full object-cover"})},e.id))}):null,(0,a.jsxs)("div",{className:"whitespace-pre-wrap break-words text-[15px] leading-7",children:[i,r&&s&&!U?(0,a.jsx)("span",{className:"inline-block animate-pulse",children:"▋"}):null]})]})},e.id)})})]}),(0,a.jsxs)("div",{className:"shrink-0 pt-2",children:[$.length>0?(0,a.jsx)("div",{className:"mx-auto mb-3 flex w-full max-w-3xl flex-wrap gap-2 px-4",children:$.map(e=>(0,a.jsxs)("div",{className:"flex items-center gap-2 rounded-full border border-white/10 bg-white/5 px-3 py-2",children:[(0,a.jsx)("span",{className:"text-xs text-white/80 max-w-[12rem] truncate",children:e.name}),(0,a.jsx)("button",{type:"button",onClick:()=>{var t;return t=e.id,void O(e=>e.filter(e=>e.id!==t))},className:"text-white/55 hover:text-white","aria-label":"Remove attachment",children:(0,a.jsx)("span",{className:"material-symbols-outlined text-[18px]",children:"close"})})]},e.id))}):null,(0,a.jsx)("div",{className:"mx-auto w-full max-w-3xl px-4 pb-2",children:(0,a.jsxs)("div",{className:"rounded-[26px] bg-[#2f2f2f] px-3 pt-3 pb-2 shadow-[0_0_15px_rgba(0,0,0,0.10)] ring-1 ring-white/5",children:[(0,a.jsx)("textarea",{value:C,onChange:e=>D(e.target.value),onKeyDown:e=>{"Enter"===e.key&&!e.shiftKey&&(e.preventDefault(),et&&ei())},placeholder:"Message AI",rows:1,className:"w-full resize-none bg-transparent px-2 text-[15px] leading-6 text-white outline-none placeholder:text-white/40 custom-scrollbar max-h-[25vh] overflow-y-auto"}),(0,a.jsxs)("div",{className:"mt-2 flex items-center justify-between gap-3",children:[(0,a.jsxs)("div",{className:"flex items-center gap-2",children:[(0,a.jsx)("button",{type:"button",onClick:()=>L.current?.click(),disabled:!V||r,className:"p-2 text-white/50 hover:text-white transition rounded-full hover:bg-white/5",children:(0,a.jsx)("span",{className:"material-symbols-outlined text-[20px]",children:"attach_file"})}),(0,a.jsx)("input",{ref:L,type:"file",accept:"image/*",multiple:!0,className:"hidden",onChange:es}),(0,a.jsx)("span",{className:"text-xs font-medium text-white/30 truncate max-w-[120px]",children:V?V.name:"No model"})]}),(0,a.jsxs)("div",{className:"flex items-center gap-2",children:[_?(0,a.jsx)("button",{type:"button",onClick:()=>{F.current?.abort()},className:"p-2 text-white bg-white/10 hover:bg-white/20 transition rounded-full h-8 w-8 flex items-center justify-center",children:(0,a.jsx)("span",{className:"material-symbols-outlined text-[16px]",children:"stop"})}):null,(0,a.jsx)("button",{onClick:ei,disabled:!et,className:`h-8 w-8 rounded-full flex items-center justify-center transition ${et?"bg-white text-black hover:opacity-90":"bg-white/10 text-white/30 cursor-not-allowed"}`,children:(0,a.jsx)("span",{className:"material-symbols-outlined text-[16px]",children:"arrow_upward"})})]})]})]})})]}),(0,a.jsx)("p",{className:"mx-auto mt-2 max-w-3xl px-4 pb-4 text-center text-[11px] text-white/30",children:"Model list is filtered from connected providers."})]})]})})}}},e=>{e.O(0,[7848,4335,2599,505,4156,2347,5158,7358],()=>e(e.s=11710)),_N_E=e.O()}]);
1
+ (self.webpackChunk_N_E=self.webpackChunk_N_E||[]).push([[4465],{11710:(e,t,r)=>{Promise.resolve().then(r.bind(r,72600))},72600:(e,t,r)=>{"use strict";r.d(t,{default:()=>g});var a=r(73365),s=r(1521),i=r(74156),l=r(69633),n=r(82495);let o="basic-chat.sessions",d="basic-chat.activeSessionId",c="basic-chat.activeProviderId",m="basic-chat.draft";function u(){return globalThis.crypto?.randomUUID?globalThis.crypto.randomUUID():`chat_${Date.now()}_${Math.random().toString(16).slice(2)}`}function h(e){if("string"==typeof e)return e;if(null==e)return"";if(Array.isArray(e))return e.map(h).filter(Boolean).join(" ");if("object"==typeof e){if("string"==typeof e.message)return e.message;if("string"==typeof e.error)return e.error;try{return JSON.stringify(e)}catch{}}return String(e)}function p(e=""){let t=h(e).replace(/\s+/g," ").trim();return t?t.length>52?`${t.slice(0,52).trimEnd()}…`:t:"New chat"}async function x(e){return await new Promise((t,r)=>{let a=new FileReader;a.onload=()=>t(String(a.result||"")),a.onerror=()=>r(a.error||Error("Failed to read file")),a.readAsDataURL(e)})}function f(e){return e?.name||function(e=""){return String(e).replace(/[-_]/g," ").replace(/\b\w/g,e=>e.toUpperCase()).trim()||"Unknown"}(e?.provider||e?.id||"provider")}function g(){let[e,t]=(0,s.useState)([]),[r,g]=(0,s.useState)(!0),[w,v]=(0,s.useState)(""),[b,N]=(0,s.useState)([]),[y,j]=(0,s.useState)(""),[S,I]=(0,s.useState)(""),[A,k]=(0,s.useState)(""),[C,D]=(0,s.useState)(""),[$,O]=(0,s.useState)([]),[_,M]=(0,s.useState)(!1),[E,T]=(0,s.useState)(""),[U,R]=(0,s.useState)(""),[z,q]=(0,s.useState)(!1),[P,J]=(0,s.useState)(!1),[K,B]=(0,s.useState)(!1),L=(0,s.useRef)(null),F=(0,s.useRef)(null),W=(0,s.useRef)(!1),H=(0,s.useRef)(null),Y=(0,s.useRef)(null);(0,s.useEffect)(()=>{try{let e=function(e,t){try{return JSON.parse(e)}catch{return t}}(globalThis.localStorage.getItem(o),[]);N(Array.isArray(e)?e.map(e=>({...e,messages:Array.isArray(e.messages)?e.messages:[]})):[]),j(globalThis.localStorage.getItem(d)||""),I(globalThis.localStorage.getItem(c)||""),D(globalThis.localStorage.getItem(m)||"")}catch{}finally{q(!0)}},[]),(0,s.useEffect)(()=>{let e=!1;return async function(){g(!0),v("");try{let r=await fetch("/api/providers",{cache:"no-store"}),a=await r.json().catch(()=>({})),s=Array.isArray(a.connections)?a.connections.filter(e=>e?.isActive!==!1):[];if(0===s.length){e||(t([]),v("Chưa c\xf3 provider n\xe0o được connect."));return}let i=new Map;for(let e of s){let t=e.provider||e.id,r=f(e),a=(0,n.mq)(t)?"openai-compatible":(0,n.gb)(t)?"anthropic-compatible":t;i.has(t)||i.set(t,{providerId:t,providerName:r,providerType:a,connections:[],models:[]});let s=i.get(t);s.providerName=s.providerName||r,s.providerType=s.providerType||a,s.connections.push(e);let o=(0,l.KC)(t).map(t=>t?.id?{id:`${e.provider}/${t.id}`,requestModel:`${e.provider}/${t.id}`,name:t.name||t.id,providerId:e.provider,providerName:f(e),source:"static"}:null).filter(Boolean);s.models.push(...o)}for(let e of(await Promise.all(s.map(async e=>{try{let t=await fetch(`/api/providers/${e.id}/models`,{cache:"no-store"}),r=await t.json().catch(()=>({}));if(!t.ok)return{connection:e,models:[]};let a=(Array.isArray(r?.models)?r.models:Array.isArray(r?.data)?r.data:Array.isArray(r?.results)?r.results:Array.isArray(r)?r:[]).map(t=>(function(e,t){let r="string"==typeof e?e:e?.id||e?.name||e?.model||"";if(!r)return null;let a="string"==typeof e?e:e?.name||e?.displayName||r,s=r;return((0,n.mq)(t.provider)||(0,n.gb)(t.provider))&&!r.includes("/")&&(s=`${t.provider}/${r}`),{id:s,requestModel:s,name:a,providerId:t.provider,providerName:f(t),source:"live"}})(t,e)).filter(Boolean);return{connection:e,models:a}}catch{return{connection:e,models:[]}}})))){let t=e.connection.provider||e.connection.id,r=i.get(t);r&&r.models.push(...e.models)}let o=Array.from(i.values()).map(e=>({...e,models:(function(e){let t=new Map;for(let r of e)r?.id&&(t.has(r.id)||t.set(r.id,r));return Array.from(t.values())})(e.models).sort((e,t)=>e.name.localeCompare(t.name))})).filter(e=>e.models.length>0).sort((e,t)=>e.providerName.localeCompare(t.providerName));e||(t(o),0===o.length&&v("Đ\xe3 c\xf3 provider connect nhưng chưa lấy được model n\xe0o."))}catch(r){e||(v(h(r?.message)||"Kh\xf4ng thể tải danh s\xe1ch provider/model."),t([]))}finally{e||g(!1)}}(),()=>{e=!0}},[]),(0,s.useEffect)(()=>{let e=e=>{H.current&&!H.current.contains(e.target)&&J(!1),Y.current&&!Y.current.contains(e.target)&&B(!1)};return document.addEventListener("mousedown",e),()=>document.removeEventListener("mousedown",e)},[]);let G=(0,s.useMemo)(()=>{let t=new Map;for(let r of e)for(let e of r.models)t.set(e.id,{...e,providerId:r.providerId,providerName:r.providerName});return t},[e]),Q=(0,s.useMemo)(()=>e.find(e=>e.providerId===S)||e[0]||null,[e,S]),V=(0,s.useMemo)(()=>{if(A&&G.has(A))return G.get(A);if(y){let e=b.find(e=>e.id===y);if(e?.modelId&&G.has(e.modelId))return G.get(e.modelId)}return Q?.models?.[0]||null},[A,G,Q,b,y]),X=(0,s.useMemo)(()=>b.find(e=>e.id===y)||null,[b,y]),Z=X?.messages||[],ee=(0,s.useMemo)(()=>[...b].sort((e,t)=>new Date(t.updatedAt).getTime()-new Date(e.updatedAt).getTime()),[b]),et=!_&&!!V&&(C.trim().length>0||$.length>0);(0,s.useEffect)(()=>{if(z)try{globalThis.localStorage.setItem(o,JSON.stringify(b)),globalThis.localStorage.setItem(d,y),globalThis.localStorage.setItem(c,S),globalThis.localStorage.setItem(m,C)}catch{}},[z,b,y,S,C]),(0,s.useEffect)(()=>{if(!z||r||W.current||0===e.length)return;let t=e.find(e=>e.providerId===S)||e[0],a=A&&G.has(A)?G.get(A):t.models[0];if(b.length>0){let e=b.find(e=>e.id===y)||b[0],r=e?.modelId&&G.has(e.modelId)?G.get(e.modelId):a;W.current=!0,j(e.id),I(r?.providerId||t.providerId),k(r?.id||a.id);return}let s={id:u(),title:"New chat",providerId:t.providerId,providerName:t.providerName,modelId:a.id,modelName:a.name,createdAt:new Date().toISOString(),updatedAt:new Date().toISOString(),messages:[]};W.current=!0,N([s]),j(s.id),I(t.providerId),k(a.id)},[z,r,e,G,b,y,S,A]);let er=(e,t)=>{N(r=>r.map(r=>r.id===e?t({...r,messages:Array.isArray(r.messages)?r.messages.map(e=>({...e})):[]}):r))},ea=e=>e?{id:u(),title:"New chat",providerId:e.providerId,providerName:e.providerName,modelId:e.id,modelName:e.name,createdAt:new Date().toISOString(),updatedAt:new Date().toISOString(),messages:[]}:null,es=async e=>{let t=Array.from(e.target.files||[]);if(0===t.length)return;let r=t.filter(e=>e.type.startsWith("image/"));if(0===r.length){e.target.value="";return}let a=await Promise.all(r.map(async e=>({id:u(),name:e.name,type:e.type,size:e.size,dataUrl:await x(e)})));O(e=>[...e,...a]),e.target.value=""},ei=async()=>{let e=V||Q?.models?.[0]||null;if(!e)return;let t=C.trim();if(!t&&0===$.length)return;let r=y,a=b.find(e=>e.id===r);if(!a){if(!(a=ea(e)))return;r=a.id,N(e=>[a,...e]),j(r)}let s={id:u(),role:"user",content:t,attachments:$.map(e=>({id:e.id,name:e.name,type:e.type,dataUrl:e.dataUrl})),createdAt:new Date().toISOString()},i=u(),l={id:i,role:"assistant",content:"",createdAt:new Date().toISOString(),status:"streaming"},n=[...a.messages||[],s,l];N(a=>a.map(a=>a.id===r?{...a,providerId:e.providerId,providerName:e.providerName,modelId:e.id,modelName:e.name,messages:n,updatedAt:new Date().toISOString(),title:"New chat"===a.title?p(t):a.title}:a)),D(""),O([]),M(!0),T(i),R(""),F.current?.abort(),F.current=new AbortController;let o=n.filter(e=>"assistant"!==e.role||e.id!==i).map(e=>({role:e.role,content:"user"===e.role?function(e){let t=h(e.content).trim(),r=Array.isArray(e.attachments)?e.attachments:[];if(0===r.length)return t;let a=[];for(let e of(t&&a.push({type:"text",text:t}),r))e?.dataUrl&&a.push({type:"image_url",image_url:{url:e.dataUrl}});return a.length>0?a:t}(e):e.content}));try{var d;let a,s=await fetch("/api/dashboard/chat/completions",{method:"POST",headers:{"Content-Type":"application/json",Accept:"text/event-stream"},body:JSON.stringify({model:e.requestModel||e.id,messages:o,stream:!0}),signal:F.current.signal});if(!s.ok){let e=await s.json().catch(()=>({}));throw Error(h(e.error||e.message||`Request failed (${s.status})`))}let l=s.body?.getReader();if(!l){let e=await s.json().catch(()=>({})),t=h(e?.choices?.[0]?.message?.content||e?.output_text||e?.error||e?.message||"");er(r,e=>({...e,messages:e.messages.map(e=>e.id===i?{...e,content:t,status:"done"}:e),updatedAt:new Date().toISOString()}));return}let n=new TextDecoder,c="",m="";for(;;){let{value:e,done:t}=await l.read();if(t)break;let a=(c+=n.decode(e,{stream:!0})).split(/\r?\n/);for(let e of(c=a.pop()||"",a)){let t=e.trim();if(!t.startsWith("data:"))continue;let a=t.slice(5).trim();if(a&&"[DONE]"!==a)try{let e=JSON.parse(a),t=function(e){if(!e||"object"!=typeof e)return"";let t=e.choices?.[0];return[(t?.delta||{}).content,t?.message?.content,e.output_text,e.text].map(h).filter(Boolean)[0]||""}(e);if(!t)continue;m+=t,R(m),er(r,e=>({...e,messages:e.messages.map(e=>e.id===i?{...e,content:m,status:"streaming"}:e),updatedAt:new Date().toISOString()}))}catch{}}}er(r,e=>({...e,messages:e.messages.map(e=>e.id===i?{...e,content:m||e.content,status:"done"}:e),updatedAt:new Date().toISOString()})),d=r,a=p(t),er(d,e=>({...e,title:"New chat"===e.title?a:e.title,updatedAt:new Date().toISOString()}))}catch(e){if("AbortError"!==e.name){let t=h(e?.message||e);er(r,e=>({...e,messages:e.messages.map(e=>e.id===i?{...e,content:e.content||`Error: ${t}`,status:"error"}:e),updatedAt:new Date().toISOString()})),v(t||"Kh\xf4ng thể gửi tin nhắn.")}}finally{M(!1),T(""),R(""),F.current=null}},el=V?`${V.name}`:"Select model",en=V?V.requestModel:"Choose from connected providers";return(0,a.jsx)("div",{className:"relative flex-1 flex flex-col h-full min-h-0 min-w-0 bg-[#212121] text-white overflow-hidden",children:(0,a.jsxs)("div",{className:"relative mx-auto flex flex-1 h-full min-h-0 w-full max-w-4xl flex-col",children:[(0,a.jsxs)("div",{className:"flex shrink-0 items-center justify-between gap-3 px-4 py-3 lg:px-6",children:[(0,a.jsxs)("div",{ref:H,className:"relative",children:[(0,a.jsx)("button",{type:"button",onClick:()=>J(e=>!e),className:"flex items-center gap-3 rounded-2xl border border-white/10 bg-white/5 px-4 py-3 text-left transition hover:bg-white/8",children:(0,a.jsxs)("div",{className:"min-w-0",children:[(0,a.jsxs)("div",{className:"flex items-center gap-2",children:[(0,a.jsx)("span",{className:"text-sm font-semibold text-white",children:el}),(0,a.jsx)("span",{className:"material-symbols-outlined text-[18px] text-white/70",children:"expand_more"})]}),(0,a.jsx)("p",{className:"truncate text-xs text-white/55",children:en})]})}),P?(0,a.jsxs)("div",{className:"absolute left-0 top-[calc(100%+10px)] z-30 w-[min(520px,calc(100vw-2rem))] overflow-hidden rounded-[20px] border border-white/10 bg-[#262626] shadow-2xl shadow-black/50",children:[(0,a.jsxs)("div",{className:"border-b border-white/10 px-4 py-3",children:[(0,a.jsx)("p",{className:"text-xs uppercase tracking-[0.22em] text-white/45",children:"Models"}),(0,a.jsx)("p",{className:"text-sm text-white/75",children:"Chỉ lấy từ provider đ\xe3 connect"})]}),(0,a.jsx)("div",{className:"max-h-[60vh] overflow-y-auto p-2 custom-scrollbar",children:e.map(e=>(0,a.jsxs)("div",{className:"mb-2 rounded-[16px] border border-white/10 bg-black/20 p-2",children:[(0,a.jsxs)("div",{className:"flex items-center justify-between px-2 py-2",children:[(0,a.jsx)("p",{className:"text-sm font-semibold text-white",children:e.providerName}),(0,a.jsx)(i.Ex,{size:"sm",variant:"default",children:e.models.length})]}),(0,a.jsx)("div",{className:"grid gap-2 sm:grid-cols-2",children:e.models.map(e=>{let t=e.id===A;return(0,a.jsx)("button",{type:"button",onClick:()=>(e=>{let t=G.get(e);if(!t)return;let r=b.find(e=>e.id===y);if(r&&r.messages.length>0){let e=ea(t);if(!e)return;N(t=>[e,...t]),j(e.id)}else if(r)N(e=>e.map(e=>e.id===r.id?{...e,providerId:t.providerId,providerName:t.providerName,modelId:t.id,modelName:t.name}:e)),j(r.id);else{let e=ea(t);if(!e)return;N(t=>[e,...t]),j(e.id)}I(t.providerId),k(t.id),J(!1)})(e.id),className:`rounded-[14px] border px-3 py-3 text-left transition ${t?"border-blue-400/40 bg-blue-500/15":"border-white/10 bg-white/5 hover:bg-white/8"}`,children:(0,a.jsxs)("div",{className:"flex items-start justify-between gap-3",children:[(0,a.jsxs)("div",{className:"min-w-0",children:[(0,a.jsx)("p",{className:"truncate text-sm font-medium text-white",children:e.name}),(0,a.jsx)("p",{className:"truncate text-[11px] text-white/45",children:e.requestModel})]}),t?(0,a.jsx)("span",{className:"material-symbols-outlined text-[18px] text-blue-300",children:"check_circle"}):null]})},e.id)})})]},e.providerId))})]}):null]}),(0,a.jsxs)("div",{className:"flex items-center gap-2",children:[(0,a.jsx)("button",{type:"button",onClick:()=>B(e=>!e),className:"rounded-2xl border border-white/10 bg-white/5 px-4 py-3 text-sm text-white/80 transition hover:bg-white/8",children:"History"}),(0,a.jsx)(i.$n,{variant:"ghost",size:"sm",icon:"delete",onClick:()=>{if(!y)return;let e=b.filter(e=>e.id!==y),t=e[0]||null;N(e),t?(j(t.id),I(t.providerId),k(t.modelId)):(j(""),I(""),k(""))},disabled:!y||0===b.length,children:"Clear"})]})]}),K?(0,a.jsxs)("div",{ref:Y,className:"absolute right-4 top-[72px] z-20 w-[min(360px,calc(100vw-2rem))] rounded-[20px] border border-white/10 bg-[#262626] p-2 shadow-2xl shadow-black/50 lg:right-6",children:[(0,a.jsx)("div",{className:"px-3 py-2",children:(0,a.jsx)("p",{className:"text-xs uppercase tracking-[0.22em] text-white/45",children:"Recent chats"})}),(0,a.jsx)("div",{className:"max-h-[48vh] space-y-2 overflow-y-auto p-1 custom-scrollbar",children:0===ee.length?(0,a.jsx)("div",{className:"rounded-[16px] border border-dashed border-white/10 bg-white/5 p-4 text-sm text-white/55",children:"Chưa c\xf3 cuộc tr\xf2 chuyện n\xe0o."}):ee.map(e=>{let t=e.id===y,r=[...e.messages||[]].reverse().find(e=>"user"===e.role)||e.messages?.[0];return(0,a.jsx)("button",{type:"button",onClick:()=>{var t;let r;return t=e.id,void((r=b.find(e=>e.id===t))&&(j(t),I(r.providerId||S),k(r.modelId||A),B(!1)))},className:`w-full rounded-[16px] border px-3 py-3 text-left transition ${t?"border-blue-400/40 bg-blue-500/15":"border-white/10 bg-white/5 hover:bg-white/8"}`,children:(0,a.jsxs)("div",{className:"flex items-start justify-between gap-3",children:[(0,a.jsxs)("div",{className:"min-w-0 flex-1",children:[(0,a.jsx)("p",{className:"truncate text-sm font-medium text-white",children:e.title}),(0,a.jsx)("p",{className:"mt-1 truncate text-xs text-white/50",children:h(r?.content)||"Empty chat"})]}),(0,a.jsx)("span",{className:"text-[10px] text-white/40 shrink-0",children:function(e){if(!e)return"Now";let t=new Date(e).getTime();if(Number.isNaN(t))return"Now";let r=Math.max(1,Math.round((Date.now()-t)/6e4));if(r<60)return`${r}m`;let a=Math.round(r/60);return a<24?`${a}h`:`${Math.round(a/24)}d`}(e.updatedAt)})]})},e.id)})})]}):null,w?(0,a.jsx)("div",{className:"mt-4 rounded-[18px] border border-rose-500/20 bg-rose-500/10 px-4 py-3 text-rose-100",children:(0,a.jsxs)("div",{className:"flex items-start gap-3",children:[(0,a.jsx)("span",{className:"material-symbols-outlined text-[20px]",children:"error"}),(0,a.jsx)("p",{className:"text-sm leading-6",children:w})]})}):null,(0,a.jsxs)("div",{className:"flex flex-1 flex-col min-h-0",children:[(0,a.jsxs)("div",{className:"flex-1 overflow-y-auto py-4 custom-scrollbar",children:[0===Z.length?(0,a.jsx)("div",{className:"flex min-h-[50vh] items-center justify-center px-4 text-center",children:(0,a.jsxs)("div",{className:"max-w-xl space-y-4",children:[(0,a.jsx)("div",{className:"mx-auto flex size-16 items-center justify-center rounded-[20px] border border-white/10 bg-white/5 text-white/80",children:(0,a.jsx)("span",{className:"material-symbols-outlined text-[30px]",children:"chat"})}),(0,a.jsxs)("div",{className:"space-y-2",children:[(0,a.jsx)("h2",{className:"text-2xl font-semibold text-white",children:"Start a conversation"}),(0,a.jsx)("p",{className:"text-sm leading-6 text-white/60",children:"Simple chat interface to interact with any AI model from connected providers. Select a model and start chatting!"})]})]})}):null,(0,a.jsx)("div",{className:"mx-auto flex w-full max-w-3xl flex-col gap-4 px-4",children:Z.map(e=>{let t="user"===e.role,r="assistant"===e.role,s=r&&e.id===E&&"streaming"===e.status,i=h(e.content)||(r?U:"");return(0,a.jsx)("div",{className:`flex w-full ${t?"justify-end":"justify-start"} mb-6`,children:(0,a.jsxs)("div",{className:`max-w-[min(88%,42rem)] ${t?"rounded-3xl bg-[#2f2f2f] px-5 py-3.5 text-white":"text-white/90"}`,children:[(0,a.jsx)("div",{className:"mb-1 flex items-center justify-between gap-3",children:(0,a.jsx)("span",{className:"text-xs font-semibold",children:t?"You":V?.name||"Assistant"})}),e.attachments?.length?(0,a.jsx)("div",{className:"mb-3 grid grid-cols-2 gap-2 sm:grid-cols-3 mt-2",children:e.attachments.map(e=>(0,a.jsx)("a",{href:e.dataUrl,target:"_blank",rel:"noreferrer",className:"overflow-hidden rounded-[18px] border border-white/10 bg-black/20",children:(0,a.jsx)("img",{src:e.dataUrl,alt:e.name,className:"h-28 w-full object-cover"})},e.id))}):null,(0,a.jsxs)("div",{className:"whitespace-pre-wrap break-words text-[15px] leading-7",children:[i,r&&s&&!U?(0,a.jsx)("span",{className:"inline-block animate-pulse",children:"▋"}):null]})]})},e.id)})})]}),(0,a.jsxs)("div",{className:"shrink-0 pt-2",children:[$.length>0?(0,a.jsx)("div",{className:"mx-auto mb-3 flex w-full max-w-3xl flex-wrap gap-2 px-4",children:$.map(e=>(0,a.jsxs)("div",{className:"flex items-center gap-2 rounded-full border border-white/10 bg-white/5 px-3 py-2",children:[(0,a.jsx)("span",{className:"text-xs text-white/80 max-w-[12rem] truncate",children:e.name}),(0,a.jsx)("button",{type:"button",onClick:()=>{var t;return t=e.id,void O(e=>e.filter(e=>e.id!==t))},className:"text-white/55 hover:text-white","aria-label":"Remove attachment",children:(0,a.jsx)("span",{className:"material-symbols-outlined text-[18px]",children:"close"})})]},e.id))}):null,(0,a.jsx)("div",{className:"mx-auto w-full max-w-3xl px-4 pb-2",children:(0,a.jsxs)("div",{className:"rounded-[26px] bg-[#2f2f2f] px-3 pt-3 pb-2 shadow-[0_0_15px_rgba(0,0,0,0.10)] ring-1 ring-white/5",children:[(0,a.jsx)("textarea",{value:C,onChange:e=>D(e.target.value),onKeyDown:e=>{"Enter"===e.key&&!e.shiftKey&&(e.preventDefault(),et&&ei())},placeholder:"Message AI",rows:1,className:"w-full resize-none bg-transparent px-2 text-[15px] leading-6 text-white outline-none placeholder:text-white/40 custom-scrollbar max-h-[25vh] overflow-y-auto"}),(0,a.jsxs)("div",{className:"mt-2 flex items-center justify-between gap-3",children:[(0,a.jsxs)("div",{className:"flex items-center gap-2",children:[(0,a.jsx)("button",{type:"button",onClick:()=>L.current?.click(),disabled:!V||r,className:"p-2 text-white/50 hover:text-white transition rounded-full hover:bg-white/5",children:(0,a.jsx)("span",{className:"material-symbols-outlined text-[20px]",children:"attach_file"})}),(0,a.jsx)("input",{ref:L,type:"file",accept:"image/*",multiple:!0,className:"hidden",onChange:es}),(0,a.jsx)("span",{className:"text-xs font-medium text-white/30 truncate max-w-[120px]",children:V?V.name:"No model"})]}),(0,a.jsxs)("div",{className:"flex items-center gap-2",children:[_?(0,a.jsx)("button",{type:"button",onClick:()=>{F.current?.abort()},className:"p-2 text-white bg-white/10 hover:bg-white/20 transition rounded-full h-8 w-8 flex items-center justify-center",children:(0,a.jsx)("span",{className:"material-symbols-outlined text-[16px]",children:"stop"})}):null,(0,a.jsx)("button",{onClick:ei,disabled:!et,className:`h-8 w-8 rounded-full flex items-center justify-center transition ${et?"bg-white text-black hover:opacity-90":"bg-white/10 text-white/30 cursor-not-allowed"}`,children:(0,a.jsx)("span",{className:"material-symbols-outlined text-[16px]",children:"arrow_upward"})})]})]})]})})]}),(0,a.jsx)("p",{className:"mx-auto mt-2 max-w-3xl px-4 pb-4 text-center text-[11px] text-white/30",children:"Model list is filtered from connected providers."})]})]})})}}},e=>{e.O(0,[7848,4335,6795,505,4156,2347,5158,7358],()=>e(e.s=11710)),_N_E=e.O()}]);
@@ -1 +1 @@
1
- (self.webpackChunk_N_E=self.webpackChunk_N_E||[]).push([[2326],{46772:(e,t,a)=>{"use strict";a.d(t,{default:()=>u});var s=a(73365),l=a(1521),i=a(74156),o=a(97990),r=a(69633),c=a(41658);let n=a(89651).env.NEXT_PUBLIC_CLOUD_URL,d={claude:"/api/cli-tools/claude-settings",codex:"/api/cli-tools/codex-settings",opencode:"/api/cli-tools/opencode-settings",droid:"/api/cli-tools/droid-settings",openclaw:"/api/cli-tools/openclaw-settings"};function u({machineId:e}){let t,a,p,[v,f]=(0,l.useState)([]),[h,g]=(0,l.useState)(!0),[x,j]=(0,l.useState)(null),[w,E]=(0,l.useState)({}),[b,m]=(0,l.useState)(!1),[y,S]=(0,l.useState)(!1),[P,k]=(0,l.useState)(""),[N,_]=(0,l.useState)([]),[O,C]=(0,l.useState)({});(0,l.useEffect)(()=>{I(),M(),U(),A()},[]);let A=async()=>{try{let e=await Promise.all(Object.entries(d).map(async([e,t])=>{try{let a=await fetch(t),s=await a.json();return[e,s]}catch{return[e,null]}}));C(Object.fromEntries(e))}catch(e){console.log("Error fetching tool statuses:",e)}},M=async()=>{try{let[e,t]=await Promise.all([fetch("/api/settings"),fetch("/api/tunnel/status")]);if(e.ok){let t=await e.json();m(t.cloudEnabled||!1)}if(t.ok){let e=await t.json();S(e.enabled||!1),k(e.publicUrl||"")}}catch(e){console.log("Error loading settings:",e)}},U=async()=>{try{let e=await fetch("/api/keys");if(e.ok){let t=await e.json();_(t.keys||[])}}catch(e){console.log("Error fetching API keys:",e)}},I=async()=>{try{let e=await fetch("/api/providers"),t=await e.json();e.ok&&f(t.connections||[])}catch(e){console.log("Error fetching connections:",e)}finally{g(!1)}},$=()=>v.filter(e=>!1!==e.isActive),L=(0,l.useCallback)((e,t,a)=>{E(s=>s[e]?.[t]===a?s:{...s,[e]:{...s[e],[t]:a}})},[]);if(h)return(0,s.jsxs)("div",{className:"flex flex-col gap-4",children:[(0,s.jsx)(i.Qv,{}),(0,s.jsx)(i.Qv,{}),(0,s.jsx)(i.Qv,{})]});let Q=(t=$(),a=[],p=new Set,t.forEach(e=>{let t=r.Xg[e.provider]||e.provider;(0,r.KC)(e.provider).forEach(s=>{let l=`${t}/${s.id}`;p.has(l)||(p.add(l),a.push({value:l,label:`${t}/${s.id}`,provider:e.provider,alias:t,connectionName:e.name,modelId:s.id}))})}),a).length>0,T=Object.entries(o.dM),X=Object.entries(o.wn);return(0,s.jsxs)("div",{className:"flex flex-col gap-6",children:[(0,s.jsx)("div",{className:"flex flex-col gap-4",children:T.map(([e,t])=>((e,t)=>{let a={tool:t,isExpanded:x===e,onToggle:()=>j(x===e?null:e),baseUrl:y&&P?P:b&&n?n:window.location.origin,apiKeys:N};switch(e){case"claude":return(0,s.jsx)(c.Tk,{...a,activeProviders:$(),modelMappings:w[e]||{},onModelMappingChange:(t,a)=>L(e,t,a),hasActiveProviders:Q,cloudEnabled:b,initialStatus:O.claude},e);case"codex":return(0,s.jsx)(c.Ah,{...a,activeProviders:$(),cloudEnabled:b,initialStatus:O.codex},e);case"opencode":return(0,s.jsx)(c.qO,{...a,activeProviders:$(),cloudEnabled:b,initialStatus:O.opencode},e);case"droid":return(0,s.jsx)(c.ZM,{...a,activeProviders:$(),hasActiveProviders:Q,cloudEnabled:b,initialStatus:O.droid},e);case"openclaw":return(0,s.jsx)(c.yZ,{...a,activeProviders:$(),hasActiveProviders:Q,cloudEnabled:b,initialStatus:O.openclaw},e);default:return(0,s.jsx)(c.a7,{toolId:e,...a,activeProviders:$(),cloudEnabled:b,tunnelEnabled:y},e)}})(e,t))}),(0,s.jsx)("div",{className:"flex flex-col gap-4",children:X.map(([e,t])=>(0,s.jsx)(c.gY,{tool:t},e))})]})}},71438:(e,t,a)=>{Promise.resolve().then(a.bind(a,46772))}},e=>{e.O(0,[7848,4335,2599,8872,505,4156,1237,2347,5158,7358],()=>e(e.s=71438)),_N_E=e.O()}]);
1
+ (self.webpackChunk_N_E=self.webpackChunk_N_E||[]).push([[2326],{46772:(e,t,a)=>{"use strict";a.d(t,{default:()=>u});var s=a(73365),l=a(1521),i=a(74156),o=a(97990),r=a(69633),c=a(41658);let n=a(89651).env.NEXT_PUBLIC_CLOUD_URL,d={claude:"/api/cli-tools/claude-settings",codex:"/api/cli-tools/codex-settings",opencode:"/api/cli-tools/opencode-settings",droid:"/api/cli-tools/droid-settings",openclaw:"/api/cli-tools/openclaw-settings"};function u({machineId:e}){let t,a,p,[v,f]=(0,l.useState)([]),[h,g]=(0,l.useState)(!0),[x,j]=(0,l.useState)(null),[w,E]=(0,l.useState)({}),[b,m]=(0,l.useState)(!1),[y,S]=(0,l.useState)(!1),[P,k]=(0,l.useState)(""),[N,_]=(0,l.useState)([]),[O,C]=(0,l.useState)({});(0,l.useEffect)(()=>{I(),M(),U(),A()},[]);let A=async()=>{try{let e=await Promise.all(Object.entries(d).map(async([e,t])=>{try{let a=await fetch(t),s=await a.json();return[e,s]}catch{return[e,null]}}));C(Object.fromEntries(e))}catch(e){console.log("Error fetching tool statuses:",e)}},M=async()=>{try{let[e,t]=await Promise.all([fetch("/api/settings"),fetch("/api/tunnel/status")]);if(e.ok){let t=await e.json();m(t.cloudEnabled||!1)}if(t.ok){let e=await t.json();S(e.enabled||!1),k(e.publicUrl||"")}}catch(e){console.log("Error loading settings:",e)}},U=async()=>{try{let e=await fetch("/api/keys");if(e.ok){let t=await e.json();_(t.keys||[])}}catch(e){console.log("Error fetching API keys:",e)}},I=async()=>{try{let e=await fetch("/api/providers"),t=await e.json();e.ok&&f(t.connections||[])}catch(e){console.log("Error fetching connections:",e)}finally{g(!1)}},$=()=>v.filter(e=>!1!==e.isActive),L=(0,l.useCallback)((e,t,a)=>{E(s=>s[e]?.[t]===a?s:{...s,[e]:{...s[e],[t]:a}})},[]);if(h)return(0,s.jsxs)("div",{className:"flex flex-col gap-4",children:[(0,s.jsx)(i.Qv,{}),(0,s.jsx)(i.Qv,{}),(0,s.jsx)(i.Qv,{})]});let Q=(t=$(),a=[],p=new Set,t.forEach(e=>{let t=r.Xg[e.provider]||e.provider;(0,r.KC)(e.provider).forEach(s=>{let l=`${t}/${s.id}`;p.has(l)||(p.add(l),a.push({value:l,label:`${t}/${s.id}`,provider:e.provider,alias:t,connectionName:e.name,modelId:s.id}))})}),a).length>0,T=Object.entries(o.dM),X=Object.entries(o.wn);return(0,s.jsxs)("div",{className:"flex flex-col gap-6",children:[(0,s.jsx)("div",{className:"flex flex-col gap-4",children:T.map(([e,t])=>((e,t)=>{let a={tool:t,isExpanded:x===e,onToggle:()=>j(x===e?null:e),baseUrl:y&&P?P:b&&n?n:window.location.origin,apiKeys:N};switch(e){case"claude":return(0,s.jsx)(c.Tk,{...a,activeProviders:$(),modelMappings:w[e]||{},onModelMappingChange:(t,a)=>L(e,t,a),hasActiveProviders:Q,cloudEnabled:b,initialStatus:O.claude},e);case"codex":return(0,s.jsx)(c.Ah,{...a,activeProviders:$(),cloudEnabled:b,initialStatus:O.codex},e);case"opencode":return(0,s.jsx)(c.qO,{...a,activeProviders:$(),cloudEnabled:b,initialStatus:O.opencode},e);case"droid":return(0,s.jsx)(c.ZM,{...a,activeProviders:$(),hasActiveProviders:Q,cloudEnabled:b,initialStatus:O.droid},e);case"openclaw":return(0,s.jsx)(c.yZ,{...a,activeProviders:$(),hasActiveProviders:Q,cloudEnabled:b,initialStatus:O.openclaw},e);default:return(0,s.jsx)(c.a7,{toolId:e,...a,activeProviders:$(),cloudEnabled:b,tunnelEnabled:y},e)}})(e,t))}),(0,s.jsx)("div",{className:"flex flex-col gap-4",children:X.map(([e,t])=>(0,s.jsx)(c.gY,{tool:t},e))})]})}},71438:(e,t,a)=>{Promise.resolve().then(a.bind(a,46772))}},e=>{e.O(0,[7848,4335,6795,8872,505,4156,1237,2347,5158,7358],()=>e(e.s=71438)),_N_E=e.O()}]);
@@ -1 +1 @@
1
- (self.webpackChunk_N_E=self.webpackChunk_N_E||[]).push([[3649],{37175:(e,t,s)=>{"use strict";s.r(t),s.d(t,{default:()=>i});var a=s(73365),l=s(1521),o=s(74156),r=s(96459);s(82495);let n=/^[a-zA-Z0-9_.\-]+$/;function i(){let[e,t]=(0,l.useState)([]),[s,n]=(0,l.useState)(!0),[i,c]=(0,l.useState)(!1),[x,h]=(0,l.useState)(null),[p,u]=(0,l.useState)([]),[b,f]=(0,l.useState)({}),{copied:g,copy:j}=(0,r.C)();(0,l.useEffect)(()=>{y()},[]);let y=async()=>{try{let[e,s,a]=await Promise.all([fetch("/api/combos"),fetch("/api/providers"),fetch("/api/settings")]),l=await e.json(),o=await s.json(),r=a.ok?await a.json():{};e.ok&&t(l.combos||[]),s.ok&&u(o.connections||[]),f(r.comboStrategies||{})}catch(e){console.log("Error fetching data:",e)}finally{n(!1)}},v=async e=>{try{let t=await fetch("/api/combos",{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify(e)});if(t.ok)await y(),c(!1);else{let e=await t.json();alert(e.error||"Failed to create combo")}}catch(e){console.log("Error creating combo:",e)}},N=async(e,t)=>{try{let s=await fetch(`/api/combos/${e}`,{method:"PUT",headers:{"Content-Type":"application/json"},body:JSON.stringify(t)});if(s.ok)await y(),h(null);else{let e=await s.json();alert(e.error||"Failed to update combo")}}catch(e){console.log("Error updating combo:",e)}},k=async s=>{if(confirm("Delete this combo?"))try{(await fetch(`/api/combos/${s}`,{method:"DELETE"})).ok&&t(e.filter(e=>e.id!==s))}catch(e){console.log("Error deleting combo:",e)}},w=async(e,t)=>{try{let s={...b};t?s[e]={fallbackStrategy:"round-robin"}:delete s[e],await fetch("/api/settings",{method:"PATCH",headers:{"Content-Type":"application/json"},body:JSON.stringify({comboStrategies:s})}),f(s)}catch(e){console.log("Error updating combo strategy:",e)}};return s?(0,a.jsxs)("div",{className:"flex flex-col gap-6",children:[(0,a.jsx)(o.Qv,{}),(0,a.jsx)(o.Qv,{})]}):(0,a.jsxs)("div",{className:"flex flex-col gap-6",children:[(0,a.jsxs)("div",{className:"flex items-center justify-between",children:[(0,a.jsxs)("div",{children:[(0,a.jsx)("h1",{className:"text-2xl font-semibold",children:"Combos"}),(0,a.jsx)("p",{className:"text-sm text-text-muted mt-1",children:"Create model combos with fallback support"})]}),(0,a.jsx)(o.$n,{icon:"add",onClick:()=>c(!0),children:"Create Combo"})]}),0===e.length?(0,a.jsx)(o.Zp,{children:(0,a.jsxs)("div",{className:"text-center py-12",children:[(0,a.jsx)("div",{className:"inline-flex items-center justify-center w-16 h-16 rounded-full bg-primary/10 text-primary mb-4",children:(0,a.jsx)("span",{className:"material-symbols-outlined text-[32px]",children:"layers"})}),(0,a.jsx)("p",{className:"text-text-main font-medium mb-1",children:"No combos yet"}),(0,a.jsx)("p",{className:"text-sm text-text-muted mb-4",children:"Create model combos with fallback support"}),(0,a.jsx)(o.$n,{icon:"add",onClick:()=>c(!0),children:"Create Combo"})]})}):(0,a.jsx)("div",{className:"flex flex-col gap-4",children:e.map(e=>(0,a.jsx)(d,{combo:e,copied:g,onCopy:j,onEdit:()=>h(e),onDelete:()=>k(e.id),roundRobinEnabled:b[e.name]?.fallbackStrategy==="round-robin",onToggleRoundRobin:t=>w(e.name,t)},e.id))}),(0,a.jsx)(m,{isOpen:i,onClose:()=>c(!1),onSave:v,activeProviders:p},"create"),(0,a.jsx)(m,{isOpen:!!x,combo:x,onClose:()=>h(null),onSave:e=>N(x.id,e),activeProviders:p},x?.id||"new")]})}function d({combo:e,copied:t,onCopy:s,onEdit:l,onDelete:r,roundRobinEnabled:n,onToggleRoundRobin:i}){return(0,a.jsx)(o.Zp,{padding:"sm",className:"group",children:(0,a.jsxs)("div",{className:"flex items-center justify-between",children:[(0,a.jsxs)("div",{className:"flex items-center gap-3 flex-1 min-w-0",children:[(0,a.jsx)("div",{className:"size-8 rounded-lg bg-primary/10 flex items-center justify-center shrink-0",children:(0,a.jsx)("span",{className:"material-symbols-outlined text-primary text-[18px]",children:"layers"})}),(0,a.jsxs)("div",{className:"min-w-0 flex-1",children:[(0,a.jsx)("code",{className:"text-sm font-medium font-mono truncate",children:e.name}),(0,a.jsxs)("div",{className:"flex items-center gap-1 mt-0.5 flex-wrap",children:[0===e.models.length?(0,a.jsx)("span",{className:"text-xs text-text-muted italic",children:"No models"}):e.models.slice(0,3).map((e,t)=>(0,a.jsx)("code",{className:"text-[10px] font-mono bg-black/5 dark:bg-white/5 px-1.5 py-0.5 rounded text-text-muted",children:e},t)),e.models.length>3&&(0,a.jsxs)("span",{className:"text-[10px] text-text-muted",children:["+",e.models.length-3," more"]})]})]})]}),(0,a.jsxs)("div",{className:"flex items-center gap-3 shrink-0",children:[(0,a.jsxs)("div",{className:"flex items-center gap-1.5",children:[(0,a.jsx)("span",{className:"text-xs text-text-muted font-medium",children:"Round Robin"}),(0,a.jsx)(o.lM,{size:"sm",checked:n,onChange:i})]}),(0,a.jsxs)("div",{className:"flex gap-1",children:[(0,a.jsxs)("button",{onClick:t=>{t.stopPropagation(),s(e.name,`combo-${e.id}`)},className:"flex flex-col items-center px-2 py-1 rounded hover:bg-black/5 dark:hover:bg-white/5 text-text-muted hover:text-primary transition-colors",title:"Copy combo name",children:[(0,a.jsx)("span",{className:"material-symbols-outlined text-[18px]",children:t===`combo-${e.id}`?"check":"content_copy"}),(0,a.jsx)("span",{className:"text-[10px] leading-tight",children:"Copy"})]}),(0,a.jsxs)("button",{onClick:l,className:"flex flex-col items-center px-2 py-1 rounded hover:bg-black/5 dark:hover:bg-white/5 text-text-muted hover:text-primary transition-colors",title:"Edit",children:[(0,a.jsx)("span",{className:"material-symbols-outlined text-[18px]",children:"edit"}),(0,a.jsx)("span",{className:"text-[10px] leading-tight",children:"Edit"})]}),(0,a.jsxs)("button",{onClick:r,className:"flex flex-col items-center px-2 py-1 rounded hover:bg-red-500/10 text-red-500 transition-colors",title:"Delete",children:[(0,a.jsx)("span",{className:"material-symbols-outlined text-[18px]",children:"delete"}),(0,a.jsx)("span",{className:"text-[10px] leading-tight",children:"Delete"})]})]})]})]})})}function c({index:e,model:t,isFirst:s,isLast:o,onEdit:r,onMoveUp:n,onMoveDown:i,onRemove:d}){let[m,x]=(0,l.useState)(!1),[h,p]=(0,l.useState)(t),u=()=>{let e=h.trim();e&&e!==t?r(e):p(t),x(!1)};return(0,a.jsxs)("div",{className:"group flex items-center gap-1.5 px-2 py-1 rounded-md bg-black/[0.02] dark:bg-white/[0.02] hover:bg-black/[0.04] dark:hover:bg-white/[0.04] transition-colors",children:[(0,a.jsx)("span",{className:"text-[10px] font-medium text-text-muted w-3 text-center shrink-0",children:e+1}),m?(0,a.jsx)("input",{autoFocus:!0,value:h,onChange:e=>p(e.target.value),onBlur:u,onKeyDown:e=>{"Enter"===e.key&&u(),"Escape"===e.key&&(p(t),x(!1))},className:"flex-1 min-w-0 px-1.5 py-0.5 text-xs font-mono bg-white dark:bg-black/20 border border-primary/40 rounded outline-none text-text-main"}):(0,a.jsx)("div",{className:"flex-1 min-w-0 px-1.5 py-0.5 text-xs font-mono text-text-main truncate cursor-text hover:bg-black/5 dark:hover:bg-white/5 rounded",onClick:()=>x(!0),title:"Click to edit",children:t}),(0,a.jsxs)("div",{className:"flex items-center gap-0.5",children:[(0,a.jsx)("button",{onClick:n,disabled:s,className:`p-0.5 rounded ${s?"text-text-muted/20 cursor-not-allowed":"text-text-muted hover:text-primary hover:bg-black/5 dark:hover:bg-white/5"}`,title:"Move up",children:(0,a.jsx)("span",{className:"material-symbols-outlined text-[12px]",children:"arrow_upward"})}),(0,a.jsx)("button",{onClick:i,disabled:o,className:`p-0.5 rounded ${o?"text-text-muted/20 cursor-not-allowed":"text-text-muted hover:text-primary hover:bg-black/5 dark:hover:bg-white/5"}`,title:"Move down",children:(0,a.jsx)("span",{className:"material-symbols-outlined text-[12px]",children:"arrow_downward"})})]}),(0,a.jsx)("button",{onClick:d,className:"p-0.5 hover:bg-red-500/10 rounded text-text-muted hover:text-red-500 transition-all",title:"Remove",children:(0,a.jsx)("span",{className:"material-symbols-outlined text-[12px]",children:"close"})})]})}function m({isOpen:e,combo:t,onClose:s,onSave:r,activeProviders:i}){let[d,x]=(0,l.useState)(t?.name||""),[h,p]=(0,l.useState)(t?.models||[]),[u,b]=(0,l.useState)(!1),[f,g]=(0,l.useState)(!1),[j,y]=(0,l.useState)(""),[v,N]=(0,l.useState)({}),k=async()=>{try{let e=await fetch("/api/models/alias");if(!e.ok)return;let t=await e.json();N(t.aliases||{})}catch(e){console.error("Error fetching modal data:",e)}};(0,l.useEffect)(()=>{e&&k()},[e]);let w=e=>e.trim()?n.test(e)?(y(""),!0):(y("Only letters, numbers, -, _ and . allowed"),!1):(y("Name is required"),!1),C=async()=>{w(d)&&(g(!0),await r({name:d.trim(),models:h}),g(!1))},S=!!t;return(0,a.jsxs)(a.Fragment,{children:[(0,a.jsx)(o.aF,{isOpen:e,onClose:s,title:S?"Edit Combo":"Create Combo",children:(0,a.jsxs)("div",{className:"flex flex-col gap-3",children:[(0,a.jsxs)("div",{children:[(0,a.jsx)(o.pd,{label:"Combo Name",value:d,onChange:e=>{let t=e.target.value;x(t),t?w(t):y("")},placeholder:"my-combo",error:j}),(0,a.jsx)("p",{className:"text-[10px] text-text-muted mt-0.5",children:"Only letters, numbers, -, _ and . allowed"})]}),(0,a.jsxs)("div",{children:[(0,a.jsx)("label",{className:"text-sm font-medium mb-1.5 block",children:"Models"}),0===h.length?(0,a.jsxs)("div",{className:"text-center py-4 border border-dashed border-black/10 dark:border-white/10 rounded-lg bg-black/[0.01] dark:bg-white/[0.01]",children:[(0,a.jsx)("span",{className:"material-symbols-outlined text-text-muted text-xl mb-1",children:"layers"}),(0,a.jsx)("p",{className:"text-xs text-text-muted",children:"No models added yet"})]}):(0,a.jsx)("div",{className:"flex flex-col gap-1 max-h-[350px] overflow-y-auto",children:h.map((e,t)=>(0,a.jsx)(c,{index:t,model:e,isFirst:0===t,isLast:t===h.length-1,onEdit:e=>{let s=[...h];s[t]=e,p(s)},onMoveUp:()=>(e=>{if(0===e)return;let t=[...h];[t[e-1],t[e]]=[t[e],t[e-1]],p(t)})(t),onMoveDown:()=>(e=>{if(e===h.length-1)return;let t=[...h];[t[e],t[e+1]]=[t[e+1],t[e]],p(t)})(t),onRemove:()=>{p(h.filter((e,s)=>s!==t))}},t))}),(0,a.jsxs)("button",{onClick:()=>b(!0),className:"w-full mt-2 py-2 border border-dashed border-black/10 dark:border-white/10 rounded-lg text-xs text-primary font-medium hover:text-primary hover:border-primary/50 transition-colors flex items-center justify-center gap-1",children:[(0,a.jsx)("span",{className:"material-symbols-outlined text-[16px]",children:"add"}),"Add Model"]})]}),(0,a.jsxs)("div",{className:"flex gap-2 pt-1",children:[(0,a.jsx)(o.$n,{onClick:s,variant:"ghost",fullWidth:!0,size:"sm",children:"Cancel"}),(0,a.jsx)(o.$n,{onClick:C,fullWidth:!0,size:"sm",disabled:!d.trim()||!!j||f,children:f?"Saving...":S?"Save":"Create"})]})]})}),(0,a.jsx)(o.rq,{isOpen:u,onClose:()=>b(!1),onSelect:e=>{h.includes(e.value)||p([...h,e.value])},activeProviders:i,modelAliases:v,title:"Add Model to Combo"})]})}},73540:(e,t,s)=>{Promise.resolve().then(s.bind(s,37175))}},e=>{e.O(0,[7848,4335,2599,505,4156,2347,5158,7358],()=>e(e.s=73540)),_N_E=e.O()}]);
1
+ (self.webpackChunk_N_E=self.webpackChunk_N_E||[]).push([[3649],{37175:(e,t,s)=>{"use strict";s.r(t),s.d(t,{default:()=>i});var a=s(73365),l=s(1521),o=s(74156),r=s(96459);s(82495);let n=/^[a-zA-Z0-9_.\-]+$/;function i(){let[e,t]=(0,l.useState)([]),[s,n]=(0,l.useState)(!0),[i,c]=(0,l.useState)(!1),[x,h]=(0,l.useState)(null),[p,u]=(0,l.useState)([]),[b,f]=(0,l.useState)({}),{copied:g,copy:j}=(0,r.C)();(0,l.useEffect)(()=>{y()},[]);let y=async()=>{try{let[e,s,a]=await Promise.all([fetch("/api/combos"),fetch("/api/providers"),fetch("/api/settings")]),l=await e.json(),o=await s.json(),r=a.ok?await a.json():{};e.ok&&t(l.combos||[]),s.ok&&u(o.connections||[]),f(r.comboStrategies||{})}catch(e){console.log("Error fetching data:",e)}finally{n(!1)}},v=async e=>{try{let t=await fetch("/api/combos",{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify(e)});if(t.ok)await y(),c(!1);else{let e=await t.json();alert(e.error||"Failed to create combo")}}catch(e){console.log("Error creating combo:",e)}},N=async(e,t)=>{try{let s=await fetch(`/api/combos/${e}`,{method:"PUT",headers:{"Content-Type":"application/json"},body:JSON.stringify(t)});if(s.ok)await y(),h(null);else{let e=await s.json();alert(e.error||"Failed to update combo")}}catch(e){console.log("Error updating combo:",e)}},k=async s=>{if(confirm("Delete this combo?"))try{(await fetch(`/api/combos/${s}`,{method:"DELETE"})).ok&&t(e.filter(e=>e.id!==s))}catch(e){console.log("Error deleting combo:",e)}},w=async(e,t)=>{try{let s={...b};t?s[e]={fallbackStrategy:"round-robin"}:delete s[e],await fetch("/api/settings",{method:"PATCH",headers:{"Content-Type":"application/json"},body:JSON.stringify({comboStrategies:s})}),f(s)}catch(e){console.log("Error updating combo strategy:",e)}};return s?(0,a.jsxs)("div",{className:"flex flex-col gap-6",children:[(0,a.jsx)(o.Qv,{}),(0,a.jsx)(o.Qv,{})]}):(0,a.jsxs)("div",{className:"flex flex-col gap-6",children:[(0,a.jsxs)("div",{className:"flex items-center justify-between",children:[(0,a.jsxs)("div",{children:[(0,a.jsx)("h1",{className:"text-2xl font-semibold",children:"Combos"}),(0,a.jsx)("p",{className:"text-sm text-text-muted mt-1",children:"Create model combos with fallback support"})]}),(0,a.jsx)(o.$n,{icon:"add",onClick:()=>c(!0),children:"Create Combo"})]}),0===e.length?(0,a.jsx)(o.Zp,{children:(0,a.jsxs)("div",{className:"text-center py-12",children:[(0,a.jsx)("div",{className:"inline-flex items-center justify-center w-16 h-16 rounded-full bg-primary/10 text-primary mb-4",children:(0,a.jsx)("span",{className:"material-symbols-outlined text-[32px]",children:"layers"})}),(0,a.jsx)("p",{className:"text-text-main font-medium mb-1",children:"No combos yet"}),(0,a.jsx)("p",{className:"text-sm text-text-muted mb-4",children:"Create model combos with fallback support"}),(0,a.jsx)(o.$n,{icon:"add",onClick:()=>c(!0),children:"Create Combo"})]})}):(0,a.jsx)("div",{className:"flex flex-col gap-4",children:e.map(e=>(0,a.jsx)(d,{combo:e,copied:g,onCopy:j,onEdit:()=>h(e),onDelete:()=>k(e.id),roundRobinEnabled:b[e.name]?.fallbackStrategy==="round-robin",onToggleRoundRobin:t=>w(e.name,t)},e.id))}),(0,a.jsx)(m,{isOpen:i,onClose:()=>c(!1),onSave:v,activeProviders:p},"create"),(0,a.jsx)(m,{isOpen:!!x,combo:x,onClose:()=>h(null),onSave:e=>N(x.id,e),activeProviders:p},x?.id||"new")]})}function d({combo:e,copied:t,onCopy:s,onEdit:l,onDelete:r,roundRobinEnabled:n,onToggleRoundRobin:i}){return(0,a.jsx)(o.Zp,{padding:"sm",className:"group",children:(0,a.jsxs)("div",{className:"flex items-center justify-between",children:[(0,a.jsxs)("div",{className:"flex items-center gap-3 flex-1 min-w-0",children:[(0,a.jsx)("div",{className:"size-8 rounded-lg bg-primary/10 flex items-center justify-center shrink-0",children:(0,a.jsx)("span",{className:"material-symbols-outlined text-primary text-[18px]",children:"layers"})}),(0,a.jsxs)("div",{className:"min-w-0 flex-1",children:[(0,a.jsx)("code",{className:"text-sm font-medium font-mono truncate",children:e.name}),(0,a.jsxs)("div",{className:"flex items-center gap-1 mt-0.5 flex-wrap",children:[0===e.models.length?(0,a.jsx)("span",{className:"text-xs text-text-muted italic",children:"No models"}):e.models.slice(0,3).map((e,t)=>(0,a.jsx)("code",{className:"text-[10px] font-mono bg-black/5 dark:bg-white/5 px-1.5 py-0.5 rounded text-text-muted",children:e},t)),e.models.length>3&&(0,a.jsxs)("span",{className:"text-[10px] text-text-muted",children:["+",e.models.length-3," more"]})]})]})]}),(0,a.jsxs)("div",{className:"flex items-center gap-3 shrink-0",children:[(0,a.jsxs)("div",{className:"flex items-center gap-1.5",children:[(0,a.jsx)("span",{className:"text-xs text-text-muted font-medium",children:"Round Robin"}),(0,a.jsx)(o.lM,{size:"sm",checked:n,onChange:i})]}),(0,a.jsxs)("div",{className:"flex gap-1",children:[(0,a.jsxs)("button",{onClick:t=>{t.stopPropagation(),s(e.name,`combo-${e.id}`)},className:"flex flex-col items-center px-2 py-1 rounded hover:bg-black/5 dark:hover:bg-white/5 text-text-muted hover:text-primary transition-colors",title:"Copy combo name",children:[(0,a.jsx)("span",{className:"material-symbols-outlined text-[18px]",children:t===`combo-${e.id}`?"check":"content_copy"}),(0,a.jsx)("span",{className:"text-[10px] leading-tight",children:"Copy"})]}),(0,a.jsxs)("button",{onClick:l,className:"flex flex-col items-center px-2 py-1 rounded hover:bg-black/5 dark:hover:bg-white/5 text-text-muted hover:text-primary transition-colors",title:"Edit",children:[(0,a.jsx)("span",{className:"material-symbols-outlined text-[18px]",children:"edit"}),(0,a.jsx)("span",{className:"text-[10px] leading-tight",children:"Edit"})]}),(0,a.jsxs)("button",{onClick:r,className:"flex flex-col items-center px-2 py-1 rounded hover:bg-red-500/10 text-red-500 transition-colors",title:"Delete",children:[(0,a.jsx)("span",{className:"material-symbols-outlined text-[18px]",children:"delete"}),(0,a.jsx)("span",{className:"text-[10px] leading-tight",children:"Delete"})]})]})]})]})})}function c({index:e,model:t,isFirst:s,isLast:o,onEdit:r,onMoveUp:n,onMoveDown:i,onRemove:d}){let[m,x]=(0,l.useState)(!1),[h,p]=(0,l.useState)(t),u=()=>{let e=h.trim();e&&e!==t?r(e):p(t),x(!1)};return(0,a.jsxs)("div",{className:"group flex items-center gap-1.5 px-2 py-1 rounded-md bg-black/[0.02] dark:bg-white/[0.02] hover:bg-black/[0.04] dark:hover:bg-white/[0.04] transition-colors",children:[(0,a.jsx)("span",{className:"text-[10px] font-medium text-text-muted w-3 text-center shrink-0",children:e+1}),m?(0,a.jsx)("input",{autoFocus:!0,value:h,onChange:e=>p(e.target.value),onBlur:u,onKeyDown:e=>{"Enter"===e.key&&u(),"Escape"===e.key&&(p(t),x(!1))},className:"flex-1 min-w-0 px-1.5 py-0.5 text-xs font-mono bg-white dark:bg-black/20 border border-primary/40 rounded outline-none text-text-main"}):(0,a.jsx)("div",{className:"flex-1 min-w-0 px-1.5 py-0.5 text-xs font-mono text-text-main truncate cursor-text hover:bg-black/5 dark:hover:bg-white/5 rounded",onClick:()=>x(!0),title:"Click to edit",children:t}),(0,a.jsxs)("div",{className:"flex items-center gap-0.5",children:[(0,a.jsx)("button",{onClick:n,disabled:s,className:`p-0.5 rounded ${s?"text-text-muted/20 cursor-not-allowed":"text-text-muted hover:text-primary hover:bg-black/5 dark:hover:bg-white/5"}`,title:"Move up",children:(0,a.jsx)("span",{className:"material-symbols-outlined text-[12px]",children:"arrow_upward"})}),(0,a.jsx)("button",{onClick:i,disabled:o,className:`p-0.5 rounded ${o?"text-text-muted/20 cursor-not-allowed":"text-text-muted hover:text-primary hover:bg-black/5 dark:hover:bg-white/5"}`,title:"Move down",children:(0,a.jsx)("span",{className:"material-symbols-outlined text-[12px]",children:"arrow_downward"})})]}),(0,a.jsx)("button",{onClick:d,className:"p-0.5 hover:bg-red-500/10 rounded text-text-muted hover:text-red-500 transition-all",title:"Remove",children:(0,a.jsx)("span",{className:"material-symbols-outlined text-[12px]",children:"close"})})]})}function m({isOpen:e,combo:t,onClose:s,onSave:r,activeProviders:i}){let[d,x]=(0,l.useState)(t?.name||""),[h,p]=(0,l.useState)(t?.models||[]),[u,b]=(0,l.useState)(!1),[f,g]=(0,l.useState)(!1),[j,y]=(0,l.useState)(""),[v,N]=(0,l.useState)({}),k=async()=>{try{let e=await fetch("/api/models/alias");if(!e.ok)return;let t=await e.json();N(t.aliases||{})}catch(e){console.error("Error fetching modal data:",e)}};(0,l.useEffect)(()=>{e&&k()},[e]);let w=e=>e.trim()?n.test(e)?(y(""),!0):(y("Only letters, numbers, -, _ and . allowed"),!1):(y("Name is required"),!1),C=async()=>{w(d)&&(g(!0),await r({name:d.trim(),models:h}),g(!1))},S=!!t;return(0,a.jsxs)(a.Fragment,{children:[(0,a.jsx)(o.aF,{isOpen:e,onClose:s,title:S?"Edit Combo":"Create Combo",children:(0,a.jsxs)("div",{className:"flex flex-col gap-3",children:[(0,a.jsxs)("div",{children:[(0,a.jsx)(o.pd,{label:"Combo Name",value:d,onChange:e=>{let t=e.target.value;x(t),t?w(t):y("")},placeholder:"my-combo",error:j}),(0,a.jsx)("p",{className:"text-[10px] text-text-muted mt-0.5",children:"Only letters, numbers, -, _ and . allowed"})]}),(0,a.jsxs)("div",{children:[(0,a.jsx)("label",{className:"text-sm font-medium mb-1.5 block",children:"Models"}),0===h.length?(0,a.jsxs)("div",{className:"text-center py-4 border border-dashed border-black/10 dark:border-white/10 rounded-lg bg-black/[0.01] dark:bg-white/[0.01]",children:[(0,a.jsx)("span",{className:"material-symbols-outlined text-text-muted text-xl mb-1",children:"layers"}),(0,a.jsx)("p",{className:"text-xs text-text-muted",children:"No models added yet"})]}):(0,a.jsx)("div",{className:"flex flex-col gap-1 max-h-[350px] overflow-y-auto",children:h.map((e,t)=>(0,a.jsx)(c,{index:t,model:e,isFirst:0===t,isLast:t===h.length-1,onEdit:e=>{let s=[...h];s[t]=e,p(s)},onMoveUp:()=>(e=>{if(0===e)return;let t=[...h];[t[e-1],t[e]]=[t[e],t[e-1]],p(t)})(t),onMoveDown:()=>(e=>{if(e===h.length-1)return;let t=[...h];[t[e],t[e+1]]=[t[e+1],t[e]],p(t)})(t),onRemove:()=>{p(h.filter((e,s)=>s!==t))}},t))}),(0,a.jsxs)("button",{onClick:()=>b(!0),className:"w-full mt-2 py-2 border border-dashed border-black/10 dark:border-white/10 rounded-lg text-xs text-primary font-medium hover:text-primary hover:border-primary/50 transition-colors flex items-center justify-center gap-1",children:[(0,a.jsx)("span",{className:"material-symbols-outlined text-[16px]",children:"add"}),"Add Model"]})]}),(0,a.jsxs)("div",{className:"flex gap-2 pt-1",children:[(0,a.jsx)(o.$n,{onClick:s,variant:"ghost",fullWidth:!0,size:"sm",children:"Cancel"}),(0,a.jsx)(o.$n,{onClick:C,fullWidth:!0,size:"sm",disabled:!d.trim()||!!j||f,children:f?"Saving...":S?"Save":"Create"})]})]})}),(0,a.jsx)(o.rq,{isOpen:u,onClose:()=>b(!1),onSelect:e=>{h.includes(e.value)||p([...h,e.value])},activeProviders:i,modelAliases:v,title:"Add Model to Combo"})]})}},73540:(e,t,s)=>{Promise.resolve().then(s.bind(s,37175))}},e=>{e.O(0,[7848,4335,6795,505,4156,2347,5158,7358],()=>e(e.s=73540)),_N_E=e.O()}]);
@@ -1 +1 @@
1
- (self.webpackChunk_N_E=self.webpackChunk_N_E||[]).push([[712],{25733:(e,t,l)=>{"use strict";l.d(t,{default:()=>o});var s=l(73365),n=l(1521),r=l(74156),a=l(43202);let c={LOG:"text-green-400",INFO:"text-blue-400",WARN:"text-yellow-400",ERROR:"text-red-400",DEBUG:"text-purple-400"};function o(){let[e,t]=(0,n.useState)([]),[l,o]=(0,n.useState)(!1),i=(0,n.useRef)(null),u=async()=>{try{await fetch("/api/translator/console-logs",{method:"DELETE"})}catch(e){console.error("Failed to clear console logs:",e)}};return(0,n.useEffect)(()=>{let e=new EventSource("/api/translator/console-logs/stream");return e.onopen=()=>o(!0),e.onmessage=e=>{let l=JSON.parse(e.data);"init"===l.type?t(l.logs.slice(-a.UY.maxLines)):"line"===l.type?t(e=>{let t=[...e,l.line];return t.length>a.UY.maxLines?t.slice(-a.UY.maxLines):t}):"clear"===l.type&&t([])},e.onerror=()=>o(!1),()=>e.close()},[]),(0,n.useEffect)(()=>{i.current&&(i.current.scrollTop=i.current.scrollHeight)},[e]),(0,s.jsx)("div",{className:"",children:(0,s.jsxs)(r.Zp,{children:[(0,s.jsx)("div",{className:"flex items-center justify-end px-4 pt-3 pb-2",children:(0,s.jsx)(r.$n,{size:"sm",variant:"outline",icon:"delete",onClick:u,children:"Clear"})}),(0,s.jsx)("div",{ref:i,className:"bg-black rounded-b-lg p-4 text-xs font-mono h-[calc(100vh-220px)] overflow-y-auto",children:0===e.length?(0,s.jsx)("span",{className:"text-text-muted",children:"No console logs yet."}):(0,s.jsx)("div",{className:"space-y-0.5",children:e.map((e,t)=>{let l,n;return(0,s.jsx)("div",{children:(n=c[(l=e.match(/\[(\w+)\]/g))?l[1]?.replace(/\[|\]/g,""):null]||"text-green-400",(0,s.jsx)("span",{className:n,children:e}))},t)})})})]})})}},88795:(e,t,l)=>{Promise.resolve().then(l.bind(l,25733))}},e=>{e.O(0,[7848,4335,2599,505,4156,2347,5158,7358],()=>e(e.s=88795)),_N_E=e.O()}]);
1
+ (self.webpackChunk_N_E=self.webpackChunk_N_E||[]).push([[712],{25733:(e,t,l)=>{"use strict";l.d(t,{default:()=>o});var s=l(73365),n=l(1521),r=l(74156),a=l(43202);let c={LOG:"text-green-400",INFO:"text-blue-400",WARN:"text-yellow-400",ERROR:"text-red-400",DEBUG:"text-purple-400"};function o(){let[e,t]=(0,n.useState)([]),[l,o]=(0,n.useState)(!1),i=(0,n.useRef)(null),u=async()=>{try{await fetch("/api/translator/console-logs",{method:"DELETE"})}catch(e){console.error("Failed to clear console logs:",e)}};return(0,n.useEffect)(()=>{let e=new EventSource("/api/translator/console-logs/stream");return e.onopen=()=>o(!0),e.onmessage=e=>{let l=JSON.parse(e.data);"init"===l.type?t(l.logs.slice(-a.UY.maxLines)):"line"===l.type?t(e=>{let t=[...e,l.line];return t.length>a.UY.maxLines?t.slice(-a.UY.maxLines):t}):"clear"===l.type&&t([])},e.onerror=()=>o(!1),()=>e.close()},[]),(0,n.useEffect)(()=>{i.current&&(i.current.scrollTop=i.current.scrollHeight)},[e]),(0,s.jsx)("div",{className:"",children:(0,s.jsxs)(r.Zp,{children:[(0,s.jsx)("div",{className:"flex items-center justify-end px-4 pt-3 pb-2",children:(0,s.jsx)(r.$n,{size:"sm",variant:"outline",icon:"delete",onClick:u,children:"Clear"})}),(0,s.jsx)("div",{ref:i,className:"bg-black rounded-b-lg p-4 text-xs font-mono h-[calc(100vh-220px)] overflow-y-auto",children:0===e.length?(0,s.jsx)("span",{className:"text-text-muted",children:"No console logs yet."}):(0,s.jsx)("div",{className:"space-y-0.5",children:e.map((e,t)=>{let l,n;return(0,s.jsx)("div",{children:(n=c[(l=e.match(/\[(\w+)\]/g))?l[1]?.replace(/\[|\]/g,""):null]||"text-green-400",(0,s.jsx)("span",{className:n,children:e}))},t)})})})]})})}},88795:(e,t,l)=>{Promise.resolve().then(l.bind(l,25733))}},e=>{e.O(0,[7848,4335,6795,505,4156,2347,5158,7358],()=>e(e.s=88795)),_N_E=e.O()}]);
@@ -0,0 +1 @@
1
+ (self.webpackChunk_N_E=self.webpackChunk_N_E||[]).push([[6099],{},_=>{_.O(0,[7848,4335,6795,505,4156,8035,2347,5158,7358],()=>_(_.s=78035)),_N_E=_.O()}]);