agent-relay 2.0.22 → 2.0.23

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 (391) hide show
  1. package/bin/relay-pty-linux-arm64 +0 -0
  2. package/dist/src/cli/index.d.ts +3 -3
  3. package/dist/src/cli/index.js +31 -100
  4. package/package.json +22 -29
  5. package/packages/api-types/package.json +1 -1
  6. package/packages/bridge/package.json +8 -8
  7. package/packages/cli-tester/package.json +1 -1
  8. package/packages/cloud/dist/server.js +25 -4
  9. package/packages/cloud/package.json +6 -6
  10. package/packages/config/package.json +2 -2
  11. package/packages/continuity/package.json +1 -1
  12. package/packages/daemon/dist/orchestrator.js +19 -1
  13. package/packages/daemon/package.json +12 -12
  14. package/packages/hooks/package.json +4 -4
  15. package/packages/mcp/package.json +2 -2
  16. package/packages/memory/package.json +2 -2
  17. package/packages/policy/package.json +2 -2
  18. package/packages/protocol/package.json +1 -1
  19. package/packages/resiliency/package.json +1 -1
  20. package/packages/sdk/package.json +2 -2
  21. package/packages/spawner/package.json +1 -1
  22. package/packages/state/package.json +1 -1
  23. package/packages/storage/package.json +2 -2
  24. package/packages/telemetry/package.json +1 -1
  25. package/packages/trajectory/package.json +2 -2
  26. package/packages/user-directory/package.json +2 -2
  27. package/packages/utils/dist/update-checker.js +4 -0
  28. package/packages/utils/package.json +1 -1
  29. package/packages/wrapper/package.json +6 -6
  30. package/deploy/workspace/codex.config.toml +0 -20
  31. package/deploy/workspace/entrypoint-browser.sh +0 -118
  32. package/deploy/workspace/entrypoint.sh +0 -612
  33. package/deploy/workspace/gh-credential-relay +0 -90
  34. package/deploy/workspace/gh-relay +0 -156
  35. package/deploy/workspace/git-credential-relay +0 -330
  36. package/deploy/workspace/git-credential-relay.test.sh +0 -230
  37. package/dist/dashboard/out/404.html +0 -1
  38. package/dist/dashboard/out/_next/static/91mkGYq3qbG8WHE6VytQ8/_buildManifest.js +0 -1
  39. package/dist/dashboard/out/_next/static/91mkGYq3qbG8WHE6VytQ8/_ssgManifest.js +0 -1
  40. package/dist/dashboard/out/_next/static/chunks/116-a883fca163f3a5bc.js +0 -1
  41. package/dist/dashboard/out/_next/static/chunks/117-c8afed19e821a35d.js +0 -2
  42. package/dist/dashboard/out/_next/static/chunks/282-980c2eb8fff20123.js +0 -1
  43. package/dist/dashboard/out/_next/static/chunks/320-a6304232cd0ee2ce.js +0 -1
  44. package/dist/dashboard/out/_next/static/chunks/532-bace199897eeab37.js +0 -9
  45. package/dist/dashboard/out/_next/static/chunks/631-16b905e5920f9b59.js +0 -1
  46. package/dist/dashboard/out/_next/static/chunks/648-acb2ff9f77cbfbd3.js +0 -1
  47. package/dist/dashboard/out/_next/static/chunks/766-2aea80818f7eb0d8.js +0 -1
  48. package/dist/dashboard/out/_next/static/chunks/83-26d2bde54616ee90.js +0 -1
  49. package/dist/dashboard/out/_next/static/chunks/847-f1f467060f32afff.js +0 -1
  50. package/dist/dashboard/out/_next/static/chunks/891-5cb1513eeb97a891.js +0 -1
  51. package/dist/dashboard/out/_next/static/chunks/app/_not-found/page-60501fddbafba9dc.js +0 -1
  52. package/dist/dashboard/out/_next/static/chunks/app/app/onboarding/page-9914652442f7e4fb.js +0 -1
  53. package/dist/dashboard/out/_next/static/chunks/app/app/page-366fb7c078d4e9e0.js +0 -1
  54. package/dist/dashboard/out/_next/static/chunks/app/cloud/link/page-fa1d5842aa90e8a6.js +0 -1
  55. package/dist/dashboard/out/_next/static/chunks/app/complete-profile/page-dd64bbdf66b639cd.js +0 -1
  56. package/dist/dashboard/out/_next/static/chunks/app/connect-repos/page-113060009ef35bc2.js +0 -1
  57. package/dist/dashboard/out/_next/static/chunks/app/history/page-9965d2483011b846.js +0 -1
  58. package/dist/dashboard/out/_next/static/chunks/app/layout-6b91e33784c20610.js +0 -1
  59. package/dist/dashboard/out/_next/static/chunks/app/login/page-435eceb0073be027.js +0 -1
  60. package/dist/dashboard/out/_next/static/chunks/app/metrics/page-1e37ef8e73940b40.js +0 -1
  61. package/dist/dashboard/out/_next/static/chunks/app/page-8119d4246743574e.js +0 -1
  62. package/dist/dashboard/out/_next/static/chunks/app/pricing/page-9db3ebdfa567a7c9.js +0 -1
  63. package/dist/dashboard/out/_next/static/chunks/app/providers/page-ecb16ffd3b36262b.js +0 -1
  64. package/dist/dashboard/out/_next/static/chunks/app/providers/setup/[provider]/page-4dbe33f0f7691b7c.js +0 -1
  65. package/dist/dashboard/out/_next/static/chunks/app/signup/page-c7a0a28341365ae0.js +0 -1
  66. package/dist/dashboard/out/_next/static/chunks/e868780c-48e5f147c90a3a41.js +0 -18
  67. package/dist/dashboard/out/_next/static/chunks/fd9d1056-609918ca7b6280bb.js +0 -1
  68. package/dist/dashboard/out/_next/static/chunks/framework-f66176bb897dc684.js +0 -1
  69. package/dist/dashboard/out/_next/static/chunks/main-311c3db74dcfadb7.js +0 -1
  70. package/dist/dashboard/out/_next/static/chunks/main-app-fdbeb09028f57c9f.js +0 -1
  71. package/dist/dashboard/out/_next/static/chunks/pages/_app-72b849fbd24ac258.js +0 -1
  72. package/dist/dashboard/out/_next/static/chunks/pages/_error-7ba65e1336b92748.js +0 -1
  73. package/dist/dashboard/out/_next/static/chunks/polyfills-42372ed130431b0a.js +0 -1
  74. package/dist/dashboard/out/_next/static/chunks/webpack-1cdd8ed57114d5e1.js +0 -1
  75. package/dist/dashboard/out/_next/static/css/4034f236dd1a3178.css +0 -1
  76. package/dist/dashboard/out/_next/static/css/6892f8422896ef7a.css +0 -1
  77. package/dist/dashboard/out/alt-logos/agent-relay-logo-128.png +0 -0
  78. package/dist/dashboard/out/alt-logos/agent-relay-logo-256.png +0 -0
  79. package/dist/dashboard/out/alt-logos/agent-relay-logo-32.png +0 -0
  80. package/dist/dashboard/out/alt-logos/agent-relay-logo-512.png +0 -0
  81. package/dist/dashboard/out/alt-logos/agent-relay-logo-64.png +0 -0
  82. package/dist/dashboard/out/alt-logos/agent-relay-logo.svg +0 -45
  83. package/dist/dashboard/out/alt-logos/logo.svg +0 -38
  84. package/dist/dashboard/out/alt-logos/monogram-logo-128.png +0 -0
  85. package/dist/dashboard/out/alt-logos/monogram-logo-256.png +0 -0
  86. package/dist/dashboard/out/alt-logos/monogram-logo-32.png +0 -0
  87. package/dist/dashboard/out/alt-logos/monogram-logo-512.png +0 -0
  88. package/dist/dashboard/out/alt-logos/monogram-logo-64.png +0 -0
  89. package/dist/dashboard/out/alt-logos/monogram-logo.svg +0 -38
  90. package/dist/dashboard/out/app/onboarding.html +0 -1
  91. package/dist/dashboard/out/app/onboarding.txt +0 -7
  92. package/dist/dashboard/out/app.html +0 -1
  93. package/dist/dashboard/out/app.txt +0 -7
  94. package/dist/dashboard/out/apple-icon.png +0 -0
  95. package/dist/dashboard/out/cloud/link.html +0 -1
  96. package/dist/dashboard/out/cloud/link.txt +0 -7
  97. package/dist/dashboard/out/complete-profile.html +0 -5
  98. package/dist/dashboard/out/complete-profile.txt +0 -7
  99. package/dist/dashboard/out/connect-repos.html +0 -1
  100. package/dist/dashboard/out/connect-repos.txt +0 -7
  101. package/dist/dashboard/out/history.html +0 -1
  102. package/dist/dashboard/out/history.txt +0 -7
  103. package/dist/dashboard/out/index.html +0 -1
  104. package/dist/dashboard/out/index.txt +0 -7
  105. package/dist/dashboard/out/login.html +0 -5
  106. package/dist/dashboard/out/login.txt +0 -7
  107. package/dist/dashboard/out/metrics.html +0 -1
  108. package/dist/dashboard/out/metrics.txt +0 -7
  109. package/dist/dashboard/out/pricing.html +0 -13
  110. package/dist/dashboard/out/pricing.txt +0 -7
  111. package/dist/dashboard/out/providers/setup/claude.html +0 -1
  112. package/dist/dashboard/out/providers/setup/claude.txt +0 -8
  113. package/dist/dashboard/out/providers/setup/codex.html +0 -1
  114. package/dist/dashboard/out/providers/setup/codex.txt +0 -8
  115. package/dist/dashboard/out/providers/setup/cursor.html +0 -1
  116. package/dist/dashboard/out/providers/setup/cursor.txt +0 -8
  117. package/dist/dashboard/out/providers.html +0 -1
  118. package/dist/dashboard/out/providers.txt +0 -7
  119. package/dist/dashboard/out/signup.html +0 -6
  120. package/dist/dashboard/out/signup.txt +0 -7
  121. package/dist/src/dashboard-server/index.d.ts +0 -8
  122. package/dist/src/dashboard-server/index.js +0 -8
  123. package/packages/dashboard/README.md +0 -48
  124. package/packages/dashboard/dist/health-worker-manager.d.ts +0 -62
  125. package/packages/dashboard/dist/health-worker-manager.js +0 -144
  126. package/packages/dashboard/dist/health-worker.d.ts +0 -9
  127. package/packages/dashboard/dist/health-worker.js +0 -79
  128. package/packages/dashboard/dist/index.d.ts +0 -20
  129. package/packages/dashboard/dist/index.js +0 -19
  130. package/packages/dashboard/dist/metrics.d.ts +0 -105
  131. package/packages/dashboard/dist/metrics.js +0 -193
  132. package/packages/dashboard/dist/needs-attention.d.ts +0 -24
  133. package/packages/dashboard/dist/needs-attention.js +0 -78
  134. package/packages/dashboard/dist/server.d.ts +0 -25
  135. package/packages/dashboard/dist/server.js +0 -5270
  136. package/packages/dashboard/dist/start.d.ts +0 -6
  137. package/packages/dashboard/dist/start.js +0 -13
  138. package/packages/dashboard/dist/types/threading.d.ts +0 -8
  139. package/packages/dashboard/dist/types/threading.js +0 -2
  140. package/packages/dashboard/dist/user-bridge.d.ts +0 -154
  141. package/packages/dashboard/dist/user-bridge.js +0 -372
  142. package/packages/dashboard/package.json +0 -65
  143. package/packages/dashboard/ui/app/app/onboarding/page.tsx +0 -394
  144. package/packages/dashboard/ui/app/app/page.tsx +0 -667
  145. package/packages/dashboard/ui/app/apple-icon.png +0 -0
  146. package/packages/dashboard/ui/app/cloud/link/page.tsx +0 -464
  147. package/packages/dashboard/ui/app/complete-profile/page.tsx +0 -204
  148. package/packages/dashboard/ui/app/connect-repos/page.tsx +0 -410
  149. package/packages/dashboard/ui/app/favicon.png +0 -0
  150. package/packages/dashboard/ui/app/globals.css +0 -59
  151. package/packages/dashboard/ui/app/history/page.tsx +0 -658
  152. package/packages/dashboard/ui/app/layout.tsx +0 -25
  153. package/packages/dashboard/ui/app/login/page.tsx +0 -424
  154. package/packages/dashboard/ui/app/metrics/page.tsx +0 -751
  155. package/packages/dashboard/ui/app/page.tsx +0 -59
  156. package/packages/dashboard/ui/app/pricing/page.tsx +0 -7
  157. package/packages/dashboard/ui/app/providers/page.tsx +0 -193
  158. package/packages/dashboard/ui/app/providers/setup/[provider]/ProviderSetupClient.tsx +0 -148
  159. package/packages/dashboard/ui/app/providers/setup/[provider]/constants.ts +0 -35
  160. package/packages/dashboard/ui/app/providers/setup/[provider]/page.tsx +0 -42
  161. package/packages/dashboard/ui/app/signup/page.tsx +0 -533
  162. package/packages/dashboard/ui/index.ts +0 -49
  163. package/packages/dashboard/ui/landing/LandingPage.tsx +0 -713
  164. package/packages/dashboard/ui/landing/PricingPage.tsx +0 -559
  165. package/packages/dashboard/ui/landing/index.ts +0 -6
  166. package/packages/dashboard/ui/landing/styles.css +0 -2850
  167. package/packages/dashboard/ui/lib/agent-merge.ts +0 -35
  168. package/packages/dashboard/ui/lib/api.ts +0 -1155
  169. package/packages/dashboard/ui/lib/cloudApi.ts +0 -877
  170. package/packages/dashboard/ui/lib/colors.ts +0 -218
  171. package/packages/dashboard/ui/lib/hierarchy.ts +0 -242
  172. package/packages/dashboard/ui/lib/stuckDetection.ts +0 -142
  173. package/packages/dashboard/ui/next-env.d.ts +0 -5
  174. package/packages/dashboard/ui/next.config.js +0 -41
  175. package/packages/dashboard/ui/package-lock.json +0 -2882
  176. package/packages/dashboard/ui/package.json +0 -33
  177. package/packages/dashboard/ui/postcss.config.js +0 -5
  178. package/packages/dashboard/ui/react-components/ActivityFeed.tsx +0 -216
  179. package/packages/dashboard/ui/react-components/AddWorkspaceModal.tsx +0 -170
  180. package/packages/dashboard/ui/react-components/AgentCard.tsx +0 -587
  181. package/packages/dashboard/ui/react-components/AgentList.tsx +0 -411
  182. package/packages/dashboard/ui/react-components/AgentProfilePanel.tsx +0 -564
  183. package/packages/dashboard/ui/react-components/App.tsx +0 -3033
  184. package/packages/dashboard/ui/react-components/BillingPanel.tsx +0 -922
  185. package/packages/dashboard/ui/react-components/BillingResult.tsx +0 -447
  186. package/packages/dashboard/ui/react-components/BroadcastComposer.tsx +0 -690
  187. package/packages/dashboard/ui/react-components/ChannelAdminPanel.tsx +0 -773
  188. package/packages/dashboard/ui/react-components/ChannelBrowser.tsx +0 -385
  189. package/packages/dashboard/ui/react-components/ChannelChat.tsx +0 -261
  190. package/packages/dashboard/ui/react-components/ChannelSidebar.tsx +0 -399
  191. package/packages/dashboard/ui/react-components/CloudSessionProvider.tsx +0 -130
  192. package/packages/dashboard/ui/react-components/CommandPalette.tsx +0 -815
  193. package/packages/dashboard/ui/react-components/ConfirmationDialog.tsx +0 -133
  194. package/packages/dashboard/ui/react-components/ConversationHistory.tsx +0 -518
  195. package/packages/dashboard/ui/react-components/CoordinatorPanel.tsx +0 -944
  196. package/packages/dashboard/ui/react-components/DecisionQueue.tsx +0 -717
  197. package/packages/dashboard/ui/react-components/DirectMessageView.tsx +0 -164
  198. package/packages/dashboard/ui/react-components/FileAutocomplete.tsx +0 -368
  199. package/packages/dashboard/ui/react-components/FleetOverview.tsx +0 -278
  200. package/packages/dashboard/ui/react-components/LogViewer.tsx +0 -310
  201. package/packages/dashboard/ui/react-components/LogViewerPanel.tsx +0 -482
  202. package/packages/dashboard/ui/react-components/Logo.tsx +0 -284
  203. package/packages/dashboard/ui/react-components/MentionAutocomplete.tsx +0 -384
  204. package/packages/dashboard/ui/react-components/MessageComposer.tsx +0 -457
  205. package/packages/dashboard/ui/react-components/MessageList.tsx +0 -649
  206. package/packages/dashboard/ui/react-components/MessageSenderName.tsx +0 -91
  207. package/packages/dashboard/ui/react-components/MessageStatusIndicator.tsx +0 -142
  208. package/packages/dashboard/ui/react-components/NewConversationModal.tsx +0 -400
  209. package/packages/dashboard/ui/react-components/NotificationToast.tsx +0 -488
  210. package/packages/dashboard/ui/react-components/OnlineUsersIndicator.tsx +0 -164
  211. package/packages/dashboard/ui/react-components/Pagination.tsx +0 -124
  212. package/packages/dashboard/ui/react-components/PricingPlans.tsx +0 -386
  213. package/packages/dashboard/ui/react-components/ProjectList.tsx +0 -625
  214. package/packages/dashboard/ui/react-components/ProviderAuthFlow.tsx +0 -853
  215. package/packages/dashboard/ui/react-components/ProviderConnectionList.tsx +0 -378
  216. package/packages/dashboard/ui/react-components/ProvisioningProgress.tsx +0 -730
  217. package/packages/dashboard/ui/react-components/RepoAccessPanel.tsx +0 -549
  218. package/packages/dashboard/ui/react-components/ServerCard.tsx +0 -202
  219. package/packages/dashboard/ui/react-components/SessionExpiredModal.tsx +0 -128
  220. package/packages/dashboard/ui/react-components/SpawnModal.tsx +0 -804
  221. package/packages/dashboard/ui/react-components/TaskAssignmentUI.tsx +0 -375
  222. package/packages/dashboard/ui/react-components/TerminalProviderSetup.tsx +0 -608
  223. package/packages/dashboard/ui/react-components/ThemeProvider.tsx +0 -325
  224. package/packages/dashboard/ui/react-components/ThinkingIndicator.tsx +0 -231
  225. package/packages/dashboard/ui/react-components/ThreadList.tsx +0 -198
  226. package/packages/dashboard/ui/react-components/ThreadPanel.tsx +0 -346
  227. package/packages/dashboard/ui/react-components/TrajectoryViewer.tsx +0 -698
  228. package/packages/dashboard/ui/react-components/TypingIndicator.tsx +0 -69
  229. package/packages/dashboard/ui/react-components/UsageBanner.tsx +0 -231
  230. package/packages/dashboard/ui/react-components/UserProfilePanel.tsx +0 -233
  231. package/packages/dashboard/ui/react-components/WorkspaceContext.tsx +0 -107
  232. package/packages/dashboard/ui/react-components/WorkspaceSelector.tsx +0 -234
  233. package/packages/dashboard/ui/react-components/WorkspaceStatusIndicator.tsx +0 -370
  234. package/packages/dashboard/ui/react-components/XTermInteractive.tsx +0 -510
  235. package/packages/dashboard/ui/react-components/XTermLogViewer.tsx +0 -719
  236. package/packages/dashboard/ui/react-components/channels/ChannelDialogs.tsx +0 -1411
  237. package/packages/dashboard/ui/react-components/channels/ChannelHeader.tsx +0 -317
  238. package/packages/dashboard/ui/react-components/channels/ChannelMessageList.tsx +0 -463
  239. package/packages/dashboard/ui/react-components/channels/ChannelViewV1.tsx +0 -146
  240. package/packages/dashboard/ui/react-components/channels/MessageInput.tsx +0 -288
  241. package/packages/dashboard/ui/react-components/channels/SearchInput.tsx +0 -172
  242. package/packages/dashboard/ui/react-components/channels/SearchResults.tsx +0 -336
  243. package/packages/dashboard/ui/react-components/channels/api.ts +0 -697
  244. package/packages/dashboard/ui/react-components/channels/index.ts +0 -76
  245. package/packages/dashboard/ui/react-components/channels/mockApi.ts +0 -344
  246. package/packages/dashboard/ui/react-components/channels/types.ts +0 -566
  247. package/packages/dashboard/ui/react-components/hooks/index.ts +0 -57
  248. package/packages/dashboard/ui/react-components/hooks/useAgentLogs.ts +0 -394
  249. package/packages/dashboard/ui/react-components/hooks/useAgents.ts +0 -127
  250. package/packages/dashboard/ui/react-components/hooks/useBroadcastDedup.ts +0 -86
  251. package/packages/dashboard/ui/react-components/hooks/useChannelAdmin.ts +0 -329
  252. package/packages/dashboard/ui/react-components/hooks/useChannelBrowser.ts +0 -239
  253. package/packages/dashboard/ui/react-components/hooks/useChannelCommands.ts +0 -138
  254. package/packages/dashboard/ui/react-components/hooks/useChannels.ts +0 -328
  255. package/packages/dashboard/ui/react-components/hooks/useDebounce.ts +0 -29
  256. package/packages/dashboard/ui/react-components/hooks/useDirectMessage.ts +0 -141
  257. package/packages/dashboard/ui/react-components/hooks/useMessages.ts +0 -309
  258. package/packages/dashboard/ui/react-components/hooks/useOrchestrator.ts +0 -364
  259. package/packages/dashboard/ui/react-components/hooks/usePinnedAgents.ts +0 -140
  260. package/packages/dashboard/ui/react-components/hooks/usePresence.ts +0 -340
  261. package/packages/dashboard/ui/react-components/hooks/useRecentRepos.ts +0 -130
  262. package/packages/dashboard/ui/react-components/hooks/useSession.ts +0 -209
  263. package/packages/dashboard/ui/react-components/hooks/useTrajectory.ts +0 -265
  264. package/packages/dashboard/ui/react-components/hooks/useWebSocket.ts +0 -169
  265. package/packages/dashboard/ui/react-components/hooks/useWorkspaceMembers.ts +0 -120
  266. package/packages/dashboard/ui/react-components/hooks/useWorkspaceRepos.ts +0 -73
  267. package/packages/dashboard/ui/react-components/hooks/useWorkspaceStatus.ts +0 -237
  268. package/packages/dashboard/ui/react-components/index.ts +0 -81
  269. package/packages/dashboard/ui/react-components/layout/Header.tsx +0 -355
  270. package/packages/dashboard/ui/react-components/layout/RepoContextHeader.tsx +0 -361
  271. package/packages/dashboard/ui/react-components/layout/Sidebar.archive.test.tsx +0 -126
  272. package/packages/dashboard/ui/react-components/layout/Sidebar.test.tsx +0 -691
  273. package/packages/dashboard/ui/react-components/layout/Sidebar.tsx +0 -930
  274. package/packages/dashboard/ui/react-components/layout/index.ts +0 -7
  275. package/packages/dashboard/ui/react-components/settings/BillingSettingsPanel.tsx +0 -564
  276. package/packages/dashboard/ui/react-components/settings/SettingsPage.tsx +0 -544
  277. package/packages/dashboard/ui/react-components/settings/TeamSettingsPanel.tsx +0 -560
  278. package/packages/dashboard/ui/react-components/settings/WorkspaceSettingsPanel.tsx +0 -1386
  279. package/packages/dashboard/ui/react-components/settings/index.ts +0 -11
  280. package/packages/dashboard/ui/react-components/settings/types.ts +0 -53
  281. package/packages/dashboard/ui/react-components/utils/messageFormatting.tsx +0 -370
  282. package/packages/dashboard/ui/tailwind.config.js +0 -148
  283. package/packages/dashboard/ui/types/index.ts +0 -304
  284. package/packages/dashboard/ui/types/threading.ts +0 -7
  285. package/packages/dashboard/ui-dist/404.html +0 -1
  286. package/packages/dashboard/ui-dist/_next/static/91mkGYq3qbG8WHE6VytQ8/_buildManifest.js +0 -1
  287. package/packages/dashboard/ui-dist/_next/static/91mkGYq3qbG8WHE6VytQ8/_ssgManifest.js +0 -1
  288. package/packages/dashboard/ui-dist/_next/static/T2rV14eEU5OweDeV29SvG/_buildManifest.js +0 -1
  289. package/packages/dashboard/ui-dist/_next/static/T2rV14eEU5OweDeV29SvG/_ssgManifest.js +0 -1
  290. package/packages/dashboard/ui-dist/_next/static/chunks/116-a883fca163f3a5bc.js +0 -1
  291. package/packages/dashboard/ui-dist/_next/static/chunks/117-c8afed19e821a35d.js +0 -2
  292. package/packages/dashboard/ui-dist/_next/static/chunks/282-980c2eb8fff20123.js +0 -1
  293. package/packages/dashboard/ui-dist/_next/static/chunks/320-a6304232cd0ee2ce.js +0 -1
  294. package/packages/dashboard/ui-dist/_next/static/chunks/532-bace199897eeab37.js +0 -9
  295. package/packages/dashboard/ui-dist/_next/static/chunks/631-16b905e5920f9b59.js +0 -1
  296. package/packages/dashboard/ui-dist/_next/static/chunks/648-acb2ff9f77cbfbd3.js +0 -1
  297. package/packages/dashboard/ui-dist/_next/static/chunks/766-2aea80818f7eb0d8.js +0 -1
  298. package/packages/dashboard/ui-dist/_next/static/chunks/83-26d2bde54616ee90.js +0 -1
  299. package/packages/dashboard/ui-dist/_next/static/chunks/847-f1f467060f32afff.js +0 -1
  300. package/packages/dashboard/ui-dist/_next/static/chunks/891-5cb1513eeb97a891.js +0 -1
  301. package/packages/dashboard/ui-dist/_next/static/chunks/app/_not-found/page-60501fddbafba9dc.js +0 -1
  302. package/packages/dashboard/ui-dist/_next/static/chunks/app/app/onboarding/page-9914652442f7e4fb.js +0 -1
  303. package/packages/dashboard/ui-dist/_next/static/chunks/app/app/page-366fb7c078d4e9e0.js +0 -1
  304. package/packages/dashboard/ui-dist/_next/static/chunks/app/cloud/link/page-fa1d5842aa90e8a6.js +0 -1
  305. package/packages/dashboard/ui-dist/_next/static/chunks/app/complete-profile/page-dd64bbdf66b639cd.js +0 -1
  306. package/packages/dashboard/ui-dist/_next/static/chunks/app/connect-repos/page-113060009ef35bc2.js +0 -1
  307. package/packages/dashboard/ui-dist/_next/static/chunks/app/history/page-9965d2483011b846.js +0 -1
  308. package/packages/dashboard/ui-dist/_next/static/chunks/app/layout-6b91e33784c20610.js +0 -1
  309. package/packages/dashboard/ui-dist/_next/static/chunks/app/login/page-435eceb0073be027.js +0 -1
  310. package/packages/dashboard/ui-dist/_next/static/chunks/app/metrics/page-1e37ef8e73940b40.js +0 -1
  311. package/packages/dashboard/ui-dist/_next/static/chunks/app/page-8119d4246743574e.js +0 -1
  312. package/packages/dashboard/ui-dist/_next/static/chunks/app/pricing/page-9db3ebdfa567a7c9.js +0 -1
  313. package/packages/dashboard/ui-dist/_next/static/chunks/app/providers/page-ecb16ffd3b36262b.js +0 -1
  314. package/packages/dashboard/ui-dist/_next/static/chunks/app/providers/setup/[provider]/page-4dbe33f0f7691b7c.js +0 -1
  315. package/packages/dashboard/ui-dist/_next/static/chunks/app/signup/page-c7a0a28341365ae0.js +0 -1
  316. package/packages/dashboard/ui-dist/_next/static/chunks/e868780c-48e5f147c90a3a41.js +0 -18
  317. package/packages/dashboard/ui-dist/_next/static/chunks/fd9d1056-609918ca7b6280bb.js +0 -1
  318. package/packages/dashboard/ui-dist/_next/static/chunks/framework-f66176bb897dc684.js +0 -1
  319. package/packages/dashboard/ui-dist/_next/static/chunks/main-311c3db74dcfadb7.js +0 -1
  320. package/packages/dashboard/ui-dist/_next/static/chunks/main-app-fdbeb09028f57c9f.js +0 -1
  321. package/packages/dashboard/ui-dist/_next/static/chunks/pages/_app-72b849fbd24ac258.js +0 -1
  322. package/packages/dashboard/ui-dist/_next/static/chunks/pages/_error-7ba65e1336b92748.js +0 -1
  323. package/packages/dashboard/ui-dist/_next/static/chunks/polyfills-42372ed130431b0a.js +0 -1
  324. package/packages/dashboard/ui-dist/_next/static/chunks/webpack-1cdd8ed57114d5e1.js +0 -1
  325. package/packages/dashboard/ui-dist/_next/static/css/4034f236dd1a3178.css +0 -1
  326. package/packages/dashboard/ui-dist/_next/static/css/6892f8422896ef7a.css +0 -1
  327. package/packages/dashboard/ui-dist/_next/static/l8L2OscDSR2vsMIlWcC48/_buildManifest.js +0 -1
  328. package/packages/dashboard/ui-dist/_next/static/l8L2OscDSR2vsMIlWcC48/_ssgManifest.js +0 -1
  329. package/packages/dashboard/ui-dist/alt-logos/agent-relay-logo-128.png +0 -0
  330. package/packages/dashboard/ui-dist/alt-logos/agent-relay-logo-256.png +0 -0
  331. package/packages/dashboard/ui-dist/alt-logos/agent-relay-logo-32.png +0 -0
  332. package/packages/dashboard/ui-dist/alt-logos/agent-relay-logo-512.png +0 -0
  333. package/packages/dashboard/ui-dist/alt-logos/agent-relay-logo-64.png +0 -0
  334. package/packages/dashboard/ui-dist/alt-logos/agent-relay-logo.svg +0 -45
  335. package/packages/dashboard/ui-dist/alt-logos/logo.svg +0 -38
  336. package/packages/dashboard/ui-dist/alt-logos/monogram-logo-128.png +0 -0
  337. package/packages/dashboard/ui-dist/alt-logos/monogram-logo-256.png +0 -0
  338. package/packages/dashboard/ui-dist/alt-logos/monogram-logo-32.png +0 -0
  339. package/packages/dashboard/ui-dist/alt-logos/monogram-logo-512.png +0 -0
  340. package/packages/dashboard/ui-dist/alt-logos/monogram-logo-64.png +0 -0
  341. package/packages/dashboard/ui-dist/alt-logos/monogram-logo.svg +0 -38
  342. package/packages/dashboard/ui-dist/app/onboarding.html +0 -1
  343. package/packages/dashboard/ui-dist/app/onboarding.txt +0 -7
  344. package/packages/dashboard/ui-dist/app.html +0 -1
  345. package/packages/dashboard/ui-dist/app.txt +0 -7
  346. package/packages/dashboard/ui-dist/apple-icon.png +0 -0
  347. package/packages/dashboard/ui-dist/cloud/link.html +0 -1
  348. package/packages/dashboard/ui-dist/cloud/link.txt +0 -7
  349. package/packages/dashboard/ui-dist/complete-profile.html +0 -5
  350. package/packages/dashboard/ui-dist/complete-profile.txt +0 -7
  351. package/packages/dashboard/ui-dist/connect-repos.html +0 -1
  352. package/packages/dashboard/ui-dist/connect-repos.txt +0 -7
  353. package/packages/dashboard/ui-dist/history.html +0 -1
  354. package/packages/dashboard/ui-dist/history.txt +0 -7
  355. package/packages/dashboard/ui-dist/index.html +0 -1
  356. package/packages/dashboard/ui-dist/index.txt +0 -7
  357. package/packages/dashboard/ui-dist/login.html +0 -5
  358. package/packages/dashboard/ui-dist/login.txt +0 -7
  359. package/packages/dashboard/ui-dist/metrics.html +0 -1
  360. package/packages/dashboard/ui-dist/metrics.txt +0 -7
  361. package/packages/dashboard/ui-dist/pricing.html +0 -13
  362. package/packages/dashboard/ui-dist/pricing.txt +0 -7
  363. package/packages/dashboard/ui-dist/providers/setup/claude.html +0 -1
  364. package/packages/dashboard/ui-dist/providers/setup/claude.txt +0 -8
  365. package/packages/dashboard/ui-dist/providers/setup/codex.html +0 -1
  366. package/packages/dashboard/ui-dist/providers/setup/codex.txt +0 -8
  367. package/packages/dashboard/ui-dist/providers/setup/cursor.html +0 -1
  368. package/packages/dashboard/ui-dist/providers/setup/cursor.txt +0 -8
  369. package/packages/dashboard/ui-dist/providers.html +0 -1
  370. package/packages/dashboard/ui-dist/providers.txt +0 -7
  371. package/packages/dashboard/ui-dist/signup.html +0 -6
  372. package/packages/dashboard/ui-dist/signup.txt +0 -7
  373. package/packages/dashboard-server/dist/health-worker-manager.d.ts +0 -62
  374. package/packages/dashboard-server/dist/health-worker-manager.js +0 -144
  375. package/packages/dashboard-server/dist/health-worker.d.ts +0 -9
  376. package/packages/dashboard-server/dist/health-worker.js +0 -79
  377. package/packages/dashboard-server/dist/index.d.ts +0 -18
  378. package/packages/dashboard-server/dist/index.js +0 -17
  379. package/packages/dashboard-server/dist/metrics.d.ts +0 -105
  380. package/packages/dashboard-server/dist/metrics.js +0 -193
  381. package/packages/dashboard-server/dist/needs-attention.d.ts +0 -24
  382. package/packages/dashboard-server/dist/needs-attention.js +0 -78
  383. package/packages/dashboard-server/dist/server.d.ts +0 -25
  384. package/packages/dashboard-server/dist/server.js +0 -5158
  385. package/packages/dashboard-server/dist/start.d.ts +0 -6
  386. package/packages/dashboard-server/dist/start.js +0 -13
  387. package/packages/dashboard-server/dist/types/threading.d.ts +0 -8
  388. package/packages/dashboard-server/dist/types/threading.js +0 -2
  389. package/packages/dashboard-server/dist/user-bridge.d.ts +0 -158
  390. package/packages/dashboard-server/dist/user-bridge.js +0 -390
  391. package/packages/dashboard-server/package.json +0 -55
@@ -1,1386 +0,0 @@
1
- /**
2
- * Workspace Settings Panel
3
- *
4
- * Manage workspace configuration including repositories,
5
- * AI providers, custom domains, and agent policies.
6
- *
7
- * Design: Mission Control theme with deep space aesthetic
8
- */
9
-
10
- import React, { useState, useEffect, useCallback } from 'react';
11
- import { cloudApi } from '../../lib/cloudApi';
12
- import { ProviderAuthFlow } from '../ProviderAuthFlow';
13
- import { TerminalProviderSetup } from '../TerminalProviderSetup';
14
- import { RepoAccessPanel } from '../RepoAccessPanel';
15
-
16
- export interface WorkspaceSettingsPanelProps {
17
- workspaceId: string;
18
- csrfToken?: string;
19
- onClose?: () => void;
20
- }
21
-
22
- interface WorkspaceDetails {
23
- id: string;
24
- name: string;
25
- status: string;
26
- publicUrl?: string;
27
- computeProvider: string;
28
- config: {
29
- providers: string[];
30
- repositories: string[];
31
- supervisorEnabled?: boolean;
32
- maxAgents?: number;
33
- };
34
- customDomain?: string;
35
- customDomainStatus?: string;
36
- errorMessage?: string;
37
- repositories: Array<{
38
- id: string;
39
- fullName: string;
40
- syncStatus: string;
41
- lastSyncedAt?: string;
42
- }>;
43
- createdAt: string;
44
- updatedAt: string;
45
- }
46
-
47
- interface AvailableRepo {
48
- id: string;
49
- fullName: string;
50
- isPrivate: boolean;
51
- defaultBranch: string;
52
- syncStatus: string;
53
- hasNangoConnection: boolean;
54
- lastSyncedAt?: string;
55
- }
56
-
57
- interface AIProvider {
58
- id: string;
59
- name: string;
60
- displayName: string;
61
- description: string;
62
- color: string;
63
- cliCommand: string;
64
- apiKeyUrl?: string;
65
- apiKeyName?: string;
66
- supportsOAuth?: boolean;
67
- supportsDeviceFlow?: boolean; // Provider supports device flow (easier for headless environments)
68
- preferApiKey?: boolean; // Show API key input by default (simpler for mobile/containers)
69
- isConnected?: boolean;
70
- comingSoon?: boolean; // Provider is not yet fully tested/available
71
- }
72
-
73
- const AI_PROVIDERS: AIProvider[] = [
74
- {
75
- id: 'anthropic',
76
- name: 'anthropic', // Must be lowercase to match backend validation
77
- displayName: 'Claude',
78
- description: 'Claude Code - recommended for code tasks',
79
- color: '#D97757',
80
- cliCommand: 'claude',
81
- apiKeyUrl: 'https://console.anthropic.com/settings/keys',
82
- apiKeyName: 'API key',
83
- supportsOAuth: true,
84
- },
85
- {
86
- id: 'codex',
87
- name: 'openai', // Must be lowercase to match backend validation
88
- displayName: 'Codex',
89
- description: 'Codex - OpenAI coding assistant',
90
- color: '#10A37F',
91
- cliCommand: 'codex login',
92
- apiKeyUrl: 'https://platform.openai.com/api-keys',
93
- apiKeyName: 'API key',
94
- supportsOAuth: true,
95
- supportsDeviceFlow: true, // Codex supports --device-auth for headless environments
96
- },
97
- {
98
- id: 'google',
99
- name: 'google', // Must be lowercase to match backend validation
100
- displayName: 'Gemini',
101
- description: 'Gemini - Google AI coding assistant',
102
- color: '#4285F4',
103
- cliCommand: 'gemini',
104
- // No apiKeyUrl - Gemini uses interactive terminal where user can choose OAuth or API key
105
- supportsOAuth: true,
106
- },
107
- {
108
- id: 'opencode',
109
- name: 'opencode', // Must be lowercase to match backend validation
110
- displayName: 'OpenCode',
111
- description: 'OpenCode - AI coding assistant',
112
- color: '#00D4AA',
113
- cliCommand: 'opencode',
114
- supportsOAuth: true,
115
- comingSoon: true, // Not yet fully tested
116
- },
117
- {
118
- id: 'droid',
119
- name: 'factory', // Must be lowercase to match backend validation
120
- displayName: 'Droid',
121
- description: 'Droid - Factory AI coding agent',
122
- color: '#6366F1',
123
- cliCommand: 'droid',
124
- supportsOAuth: true,
125
- comingSoon: true, // Not yet fully tested
126
- },
127
- {
128
- id: 'cursor',
129
- name: 'cursor', // Must be lowercase to match backend validation
130
- displayName: 'Cursor',
131
- description: 'Cursor - AI-first code editor agent',
132
- color: '#7C3AED',
133
- cliCommand: 'agent',
134
- supportsOAuth: true,
135
- },
136
- ];
137
-
138
- export function WorkspaceSettingsPanel({
139
- workspaceId,
140
- csrfToken,
141
- onClose,
142
- }: WorkspaceSettingsPanelProps) {
143
- const [workspace, setWorkspace] = useState<WorkspaceDetails | null>(null);
144
- const [availableRepos, setAvailableRepos] = useState<AvailableRepo[]>([]);
145
- const [isLoading, setIsLoading] = useState(true);
146
- const [error, setError] = useState<string | null>(null);
147
- const [activeSection, setActiveSection] = useState<'general' | 'providers' | 'repos' | 'github-access' | 'domain' | 'danger'>('general');
148
-
149
- // Provider connection state
150
- const [providerStatus, setProviderStatus] = useState<Record<string, boolean>>({});
151
- const [connectingProvider, setConnectingProvider] = useState<string | null>(null);
152
- const [apiKeyInput, setApiKeyInput] = useState('');
153
- const [providerError, setProviderError] = useState<string | null>(null);
154
- const [showApiKeyFallback, setShowApiKeyFallback] = useState<Record<string, boolean>>({});
155
- // Device flow preference for providers that support it
156
- const [useDeviceFlow, setUseDeviceFlow] = useState<Record<string, boolean>>({});
157
- // Use terminal-based setup (default for Claude, Cursor, and Gemini - Codex uses CLI helper flow)
158
- const [useTerminalSetup, setUseTerminalSetup] = useState<Record<string, boolean>>({
159
- anthropic: true, // Default to terminal for Claude
160
- cursor: true, // Default to terminal for Cursor
161
- google: true, // Default to terminal for Gemini - allows choosing OAuth or API key
162
- });
163
-
164
- // Provider disconnection state
165
- const [disconnectingProvider, setDisconnectingProvider] = useState<string | null>(null);
166
-
167
- // Repo sync state
168
- const [syncingRepoId, setSyncingRepoId] = useState<string | null>(null);
169
-
170
- // Custom domain form
171
- const [customDomain, setCustomDomain] = useState('');
172
- const [domainLoading, setDomainLoading] = useState(false);
173
- const [domainError, setDomainError] = useState<string | null>(null);
174
- const [domainInstructions, setDomainInstructions] = useState<{
175
- type: string;
176
- name: string;
177
- value: string;
178
- ttl: number;
179
- } | null>(null);
180
-
181
- // Load workspace details
182
- useEffect(() => {
183
- async function loadWorkspace() {
184
- setIsLoading(true);
185
- setError(null);
186
-
187
- const [wsResult, reposResult, providersResult] = await Promise.all([
188
- cloudApi.getWorkspaceDetails(workspaceId),
189
- cloudApi.getRepos(),
190
- cloudApi.getProviders(workspaceId),
191
- ]);
192
-
193
- if (wsResult.success) {
194
- setWorkspace(wsResult.data);
195
- if (wsResult.data.customDomain) {
196
- setCustomDomain(wsResult.data.customDomain);
197
- }
198
- } else {
199
- setError(wsResult.error);
200
- }
201
-
202
- if (reposResult.success) {
203
- setAvailableRepos(reposResult.data.repositories);
204
- }
205
-
206
- // Mark connected providers for this workspace
207
- if (providersResult.success) {
208
- // Map backend IDs to frontend IDs for consistency
209
- const BACKEND_TO_FRONTEND_MAP: Record<string, string> = {
210
- openai: 'codex', // Backend stores 'openai', frontend uses 'codex'
211
- };
212
- const connected: Record<string, boolean> = {};
213
- providersResult.data.providers.forEach((p) => {
214
- if (p.isConnected) {
215
- connected[p.id] = true;
216
- // Also mark the frontend ID as connected if there's a mapping
217
- const frontendId = BACKEND_TO_FRONTEND_MAP[p.id];
218
- if (frontendId) {
219
- connected[frontendId] = true;
220
- }
221
- }
222
- });
223
- setProviderStatus(connected);
224
- }
225
-
226
- setIsLoading(false);
227
- }
228
-
229
- loadWorkspace();
230
- }, [workspaceId]);
231
-
232
- // Start CLI-based OAuth flow for a provider
233
- // This just sets state to show the ProviderAuthFlow component, which handles the actual auth
234
- const startOAuthFlow = (provider: AIProvider) => {
235
- setProviderError(null);
236
- setConnectingProvider(provider.id);
237
- // ProviderAuthFlow will handle the rest when it mounts
238
- };
239
-
240
- // Disconnect a provider
241
- const handleDisconnectProvider = useCallback(async (provider: AIProvider) => {
242
- const confirmed = window.confirm(
243
- `Are you sure you want to disconnect ${provider.displayName}? This will remove the authentication and delete credential files from the workspace.`
244
- );
245
- if (!confirmed) return;
246
-
247
- setDisconnectingProvider(provider.id);
248
- setProviderError(null);
249
-
250
- try {
251
- const result = await cloudApi.disconnectProvider(provider.id, workspaceId);
252
- if (result.success) {
253
- setProviderStatus(prev => {
254
- const updated = { ...prev };
255
- delete updated[provider.id];
256
- return updated;
257
- });
258
- } else {
259
- setProviderError(result.error);
260
- }
261
- } catch (err) {
262
- setProviderError(err instanceof Error ? err.message : 'Failed to disconnect provider');
263
- } finally {
264
- setDisconnectingProvider(null);
265
- }
266
- }, [workspaceId]);
267
-
268
- const submitApiKey = async (provider: AIProvider) => {
269
- if (!apiKeyInput.trim()) {
270
- setProviderError('Please enter an API key');
271
- return;
272
- }
273
-
274
- setProviderError(null);
275
- setConnectingProvider(provider.id);
276
-
277
- try {
278
- const headers: Record<string, string> = { 'Content-Type': 'application/json' };
279
- if (csrfToken) headers['X-CSRF-Token'] = csrfToken;
280
-
281
- const res = await fetch(`/api/onboarding/token/${provider.id}`, {
282
- method: 'POST',
283
- credentials: 'include',
284
- headers,
285
- body: JSON.stringify({ token: apiKeyInput.trim(), workspaceId }),
286
- });
287
-
288
- if (!res.ok) {
289
- const data = await res.json();
290
- throw new Error(data.error || 'Failed to connect');
291
- }
292
-
293
- setProviderStatus(prev => ({ ...prev, [provider.id]: true }));
294
- setApiKeyInput('');
295
- setConnectingProvider(null);
296
- setShowApiKeyFallback(prev => ({ ...prev, [provider.id]: false }));
297
- } catch (err) {
298
- setProviderError(err instanceof Error ? err.message : 'Failed to connect');
299
- setConnectingProvider(null);
300
- }
301
- };
302
-
303
- // Restart workspace
304
- const handleRestart = useCallback(async () => {
305
- if (!workspace) return;
306
-
307
- const confirmed = window.confirm('Are you sure you want to restart this workspace?');
308
- if (!confirmed) return;
309
-
310
- const result = await cloudApi.restartWorkspace(workspace.id);
311
- if (result.success) {
312
- const wsResult = await cloudApi.getWorkspaceDetails(workspaceId);
313
- if (wsResult.success) {
314
- setWorkspace(wsResult.data);
315
- }
316
- } else {
317
- setError(result.error);
318
- }
319
- }, [workspace, workspaceId]);
320
-
321
- // Stop workspace
322
- const handleStop = useCallback(async () => {
323
- if (!workspace) return;
324
-
325
- const confirmed = window.confirm('Are you sure you want to stop this workspace?');
326
- if (!confirmed) return;
327
-
328
- const result = await cloudApi.stopWorkspace(workspace.id);
329
- if (result.success) {
330
- const wsResult = await cloudApi.getWorkspaceDetails(workspaceId);
331
- if (wsResult.success) {
332
- setWorkspace(wsResult.data);
333
- }
334
- } else {
335
- setError(result.error);
336
- }
337
- }, [workspace, workspaceId]);
338
-
339
- // Add repository to workspace
340
- const handleAddRepo = useCallback(async (repoId: string) => {
341
- if (!workspace) return;
342
-
343
- const result = await cloudApi.addReposToWorkspace(workspace.id, [repoId]);
344
- if (result.success) {
345
- const wsResult = await cloudApi.getWorkspaceDetails(workspaceId);
346
- if (wsResult.success) {
347
- setWorkspace(wsResult.data);
348
- }
349
- } else {
350
- setError(result.error);
351
- }
352
- }, [workspace, workspaceId]);
353
-
354
- // Sync repository to workspace (clone/pull)
355
- const handleSyncRepo = useCallback(async (repoId: string) => {
356
- if (!workspace) return;
357
-
358
- setSyncingRepoId(repoId);
359
- setError(null);
360
-
361
- const result = await cloudApi.syncRepo(repoId);
362
- if (result.success) {
363
- // Refresh workspace to get updated sync status
364
- const wsResult = await cloudApi.getWorkspaceDetails(workspaceId);
365
- if (wsResult.success) {
366
- setWorkspace(wsResult.data);
367
- }
368
- } else {
369
- setError(result.error);
370
- }
371
-
372
- setSyncingRepoId(null);
373
- }, [workspace, workspaceId]);
374
-
375
- // Set custom domain
376
- const handleSetDomain = useCallback(async () => {
377
- if (!workspace || !customDomain.trim()) return;
378
-
379
- setDomainLoading(true);
380
- setDomainError(null);
381
- setDomainInstructions(null);
382
-
383
- const result = await cloudApi.setCustomDomain(workspace.id, customDomain.trim());
384
- if (result.success) {
385
- setDomainInstructions(result.data.instructions);
386
- const wsResult = await cloudApi.getWorkspaceDetails(workspaceId);
387
- if (wsResult.success) {
388
- setWorkspace(wsResult.data);
389
- }
390
- } else {
391
- setDomainError(result.error);
392
- }
393
-
394
- setDomainLoading(false);
395
- }, [workspace, customDomain, workspaceId]);
396
-
397
- // Verify custom domain
398
- const handleVerifyDomain = useCallback(async () => {
399
- if (!workspace) return;
400
-
401
- setDomainLoading(true);
402
- setDomainError(null);
403
-
404
- const result = await cloudApi.verifyCustomDomain(workspace.id);
405
- if (result.success) {
406
- const wsResult = await cloudApi.getWorkspaceDetails(workspaceId);
407
- if (wsResult.success) {
408
- setWorkspace(wsResult.data);
409
- }
410
- if (result.data.status === 'active') {
411
- setDomainInstructions(null);
412
- }
413
- } else {
414
- setDomainError(result.error);
415
- }
416
-
417
- setDomainLoading(false);
418
- }, [workspace, workspaceId]);
419
-
420
- // Remove custom domain
421
- const handleRemoveDomain = useCallback(async () => {
422
- if (!workspace) return;
423
-
424
- const confirmed = window.confirm('Are you sure you want to remove the custom domain?');
425
- if (!confirmed) return;
426
-
427
- setDomainLoading(true);
428
- const result = await cloudApi.removeCustomDomain(workspace.id);
429
- if (result.success) {
430
- setCustomDomain('');
431
- setDomainInstructions(null);
432
- const wsResult = await cloudApi.getWorkspaceDetails(workspaceId);
433
- if (wsResult.success) {
434
- setWorkspace(wsResult.data);
435
- }
436
- } else {
437
- setDomainError(result.error);
438
- }
439
- setDomainLoading(false);
440
- }, [workspace, workspaceId]);
441
-
442
- // Delete workspace
443
- const handleDelete = useCallback(async () => {
444
- if (!workspace) return;
445
-
446
- const confirmed = window.confirm(
447
- `Are you sure you want to delete "${workspace.name}"? This action cannot be undone.`
448
- );
449
- if (!confirmed) return;
450
-
451
- const doubleConfirm = window.confirm(
452
- 'This will permanently delete all workspace data. Are you absolutely sure?'
453
- );
454
- if (!doubleConfirm) return;
455
-
456
- const result = await cloudApi.deleteWorkspace(workspace.id);
457
- if (result.success) {
458
- // Redirect to onboarding page with deleted reason
459
- window.location.href = '/app/onboarding?reason=deleted';
460
- } else {
461
- setError(result.error);
462
- }
463
- }, [workspace]);
464
-
465
- if (isLoading) {
466
- return (
467
- <div className="flex items-center justify-center h-64">
468
- <div className="relative">
469
- <div className="w-12 h-12 rounded-full border-2 border-accent-cyan/20 border-t-accent-cyan animate-spin" />
470
- <div className="absolute inset-0 flex items-center justify-center">
471
- <div className="w-4 h-4 rounded-full bg-accent-cyan/40 animate-pulse" />
472
- </div>
473
- </div>
474
- <span className="ml-4 text-text-muted font-mono text-sm tracking-wide">
475
- LOADING WORKSPACE CONFIG...
476
- </span>
477
- </div>
478
- );
479
- }
480
-
481
- if (error && !workspace) {
482
- return (
483
- <div className="p-6">
484
- <div className="p-4 bg-error/10 border border-error/30 rounded-lg text-error flex items-center gap-3">
485
- <AlertIcon />
486
- <span>{error}</span>
487
- </div>
488
- </div>
489
- );
490
- }
491
-
492
- if (!workspace) {
493
- return null;
494
- }
495
-
496
- const unassignedRepos = availableRepos.filter(
497
- (r) => !workspace.repositories.some((wr) => wr.id === r.id)
498
- );
499
-
500
- const sections = [
501
- { id: 'general', label: 'General', icon: <SettingsGearIcon /> },
502
- { id: 'providers', label: 'AI Providers', icon: <ProviderIcon /> },
503
- { id: 'repos', label: 'Repositories', icon: <RepoIcon /> },
504
- { id: 'github-access', label: 'GitHub Access', icon: <GitHubIcon /> },
505
- { id: 'domain', label: 'Domain', icon: <GlobeIcon /> },
506
- { id: 'danger', label: 'Danger', icon: <AlertIcon /> },
507
- ];
508
-
509
- return (
510
- <div className="flex flex-col h-full bg-bg-primary">
511
- {/* Section Navigation - horizontally scrollable on mobile */}
512
- <div
513
- className="flex gap-1 p-2 sm:p-3 border-b border-border-subtle bg-gradient-to-b from-bg-tertiary to-bg-primary overflow-x-auto scrollbar-hide scroll-smooth snap-x snap-mandatory touch-pan-x"
514
- style={{ WebkitOverflowScrolling: 'touch' }}
515
- >
516
- {sections.map((section) => (
517
- <button
518
- key={section.id}
519
- onClick={() => setActiveSection(section.id as typeof activeSection)}
520
- className={`flex items-center gap-1.5 sm:gap-2 px-3 sm:px-4 py-2 sm:py-2.5 rounded-lg text-xs sm:text-sm font-medium transition-all duration-200 whitespace-nowrap shrink-0 snap-start ${
521
- activeSection === section.id
522
- ? 'bg-accent-cyan/15 text-accent-cyan border border-accent-cyan/30 shadow-[0_0_12px_rgba(0,217,255,0.15)]'
523
- : 'text-text-secondary hover:bg-bg-hover hover:text-text-primary border border-transparent'
524
- }`}
525
- >
526
- <span className={activeSection === section.id ? 'text-accent-cyan' : 'text-text-muted'}>
527
- {section.icon}
528
- </span>
529
- {section.label}
530
- </button>
531
- ))}
532
- </div>
533
-
534
- {/* Content */}
535
- <div className="flex-1 overflow-y-auto p-4 sm:p-6">
536
- {error && (
537
- <div className="mb-6 p-4 bg-error/10 border border-error/30 rounded-lg text-error text-sm flex items-center gap-3">
538
- <AlertIcon />
539
- <span className="flex-1">{error}</span>
540
- <button onClick={() => setError(null)} className="text-error/60 hover:text-error">
541
- <CloseIcon />
542
- </button>
543
- </div>
544
- )}
545
-
546
- {/* General Section */}
547
- {activeSection === 'general' && (
548
- <div className="space-y-8">
549
- <SectionHeader
550
- title="Workspace Overview"
551
- subtitle="Core configuration and status"
552
- />
553
-
554
- <div className="grid grid-cols-2 gap-4">
555
- <InfoCard label="Name" value={workspace.name} />
556
- <InfoCard
557
- label="Status"
558
- value={workspace.status.charAt(0).toUpperCase() + workspace.status.slice(1)}
559
- valueColor={
560
- workspace.status === 'running' ? 'text-success' :
561
- workspace.status === 'stopped' ? 'text-amber-400' :
562
- workspace.status === 'error' ? 'text-error' : 'text-text-muted'
563
- }
564
- indicator={workspace.status === 'running'}
565
- />
566
- <InfoCard
567
- label="Public URL"
568
- value={workspace.publicUrl || 'Not available'}
569
- mono
570
- />
571
- <InfoCard
572
- label="Compute Provider"
573
- value={workspace.computeProvider.charAt(0).toUpperCase() + workspace.computeProvider.slice(1)}
574
- />
575
- </div>
576
-
577
- <div>
578
- <SectionHeader title="Actions" subtitle="Manage workspace state" />
579
- <div className="flex gap-3 mt-4">
580
- {workspace.status === 'running' && (
581
- <ActionButton
582
- onClick={handleStop}
583
- variant="warning"
584
- icon={<StopIcon />}
585
- >
586
- Stop Workspace
587
- </ActionButton>
588
- )}
589
- <ActionButton
590
- onClick={handleRestart}
591
- variant="primary"
592
- icon={<RestartIcon />}
593
- >
594
- Restart Workspace
595
- </ActionButton>
596
- </div>
597
- </div>
598
- </div>
599
- )}
600
-
601
- {/* AI Providers Section */}
602
- {activeSection === 'providers' && (
603
- <div className="space-y-8">
604
- <SectionHeader
605
- title="AI Providers"
606
- subtitle="Connect AI providers to spawn agents in this workspace"
607
- />
608
-
609
- {providerError && (
610
- <div className="p-4 bg-error/10 border border-error/30 rounded-lg text-error text-sm flex items-center gap-3">
611
- <AlertIcon />
612
- <span>{providerError}</span>
613
- </div>
614
- )}
615
-
616
- <div className="space-y-4">
617
- {AI_PROVIDERS.map((provider) => (
618
- <div
619
- key={provider.id}
620
- className={`p-5 bg-bg-tertiary rounded-xl border border-border-subtle transition-all duration-200 ${
621
- provider.comingSoon ? 'opacity-60' : 'hover:border-border-medium'
622
- }`}
623
- >
624
- <div className="flex items-center justify-between">
625
- <div className="flex items-center gap-4">
626
- <div
627
- className={`w-12 h-12 rounded-xl flex items-center justify-center text-white font-bold text-lg shadow-lg ${
628
- provider.comingSoon ? 'grayscale' : ''
629
- }`}
630
- style={{
631
- backgroundColor: provider.color,
632
- boxShadow: provider.comingSoon ? 'none' : `0 4px 20px ${provider.color}40`,
633
- }}
634
- >
635
- {provider.displayName[0]}
636
- </div>
637
- <div>
638
- <h4 className="text-base font-semibold text-text-primary flex items-center gap-2">
639
- {provider.displayName}
640
- {provider.comingSoon && (
641
- <span className="px-2 py-0.5 bg-amber-400/20 text-amber-400 text-xs font-medium rounded-full">
642
- Coming Soon
643
- </span>
644
- )}
645
- </h4>
646
- <p className="text-sm text-text-muted">{provider.description}</p>
647
- </div>
648
- </div>
649
-
650
- {provider.comingSoon ? (
651
- <div className="px-4 py-2 bg-bg-card rounded-full border border-border-subtle">
652
- <span className="text-sm text-text-muted">Not available yet</span>
653
- </div>
654
- ) : providerStatus[provider.id] ? (
655
- <div className="flex items-center gap-3">
656
- <div className="flex items-center gap-2 px-4 py-2 bg-success/15 rounded-full border border-success/30">
657
- <div className="w-2 h-2 rounded-full bg-success animate-pulse" />
658
- <span className="text-sm font-medium text-success">Connected</span>
659
- </div>
660
- <button
661
- onClick={() => handleDisconnectProvider(provider)}
662
- disabled={disconnectingProvider === provider.id}
663
- className="px-3 py-2 text-xs font-medium text-error/80 hover:text-error hover:bg-error/10 rounded-lg border border-transparent hover:border-error/30 transition-all disabled:opacity-50 disabled:cursor-not-allowed"
664
- title={`Disconnect ${provider.displayName}`}
665
- >
666
- {disconnectingProvider === provider.id ? 'Disconnecting...' : 'Disconnect'}
667
- </button>
668
- </div>
669
- ) : null}
670
- </div>
671
-
672
- {!providerStatus[provider.id] && !provider.comingSoon && (
673
- <div className="mt-5 pt-5 border-t border-border-subtle">
674
- {connectingProvider === provider.id && !showApiKeyFallback[provider.id] ? (
675
- useTerminalSetup[provider.id] ? (
676
- <TerminalProviderSetup
677
- provider={{
678
- id: provider.id,
679
- name: provider.name,
680
- displayName: provider.displayName,
681
- color: provider.color,
682
- }}
683
- workspaceId={workspaceId}
684
- csrfToken={csrfToken}
685
- maxHeight="350px"
686
- onSuccess={() => {
687
- setProviderStatus(prev => ({ ...prev, [provider.id]: true }));
688
- setConnectingProvider(null);
689
- }}
690
- onCancel={() => {
691
- setConnectingProvider(null);
692
- }}
693
- onError={(err) => {
694
- setProviderError(err);
695
- setConnectingProvider(null);
696
- }}
697
- onConnectAnother={() => {
698
- // Mark current provider as connected and clear selection
699
- // User can then click another provider to connect
700
- setProviderStatus(prev => ({ ...prev, [provider.id]: true }));
701
- setConnectingProvider(null);
702
- }}
703
- />
704
- ) : (
705
- <ProviderAuthFlow
706
- provider={{
707
- id: provider.id,
708
- name: provider.name,
709
- displayName: provider.displayName,
710
- color: provider.color,
711
- requiresUrlCopy: provider.id === 'codex',
712
- supportsDeviceFlow: provider.supportsDeviceFlow,
713
- }}
714
- workspaceId={workspaceId}
715
- csrfToken={csrfToken}
716
- useDeviceFlow={useDeviceFlow[provider.id] || false}
717
- onSuccess={() => {
718
- setProviderStatus(prev => ({ ...prev, [provider.id]: true }));
719
- setConnectingProvider(null);
720
- }}
721
- onCancel={() => {
722
- setConnectingProvider(null);
723
- }}
724
- onError={(err) => {
725
- setProviderError(err);
726
- setConnectingProvider(null);
727
- }}
728
- />
729
- )
730
- ) : showApiKeyFallback[provider.id] ? (
731
- <div className="space-y-4">
732
- <div className="flex gap-3">
733
- <input
734
- type="password"
735
- placeholder={`Enter ${provider.displayName} ${provider.apiKeyName || 'API key'}`}
736
- value={connectingProvider === provider.id ? apiKeyInput : ''}
737
- onChange={(e) => {
738
- setConnectingProvider(provider.id);
739
- setApiKeyInput(e.target.value);
740
- }}
741
- onFocus={() => setConnectingProvider(provider.id)}
742
- className="flex-1 px-4 py-3 bg-bg-card border border-border-subtle rounded-lg text-sm text-text-primary placeholder:text-text-muted focus:outline-none focus:border-accent-cyan focus:ring-1 focus:ring-accent-cyan/30 transition-all"
743
- />
744
- <button
745
- onClick={() => submitApiKey(provider)}
746
- disabled={connectingProvider !== provider.id || !apiKeyInput.trim()}
747
- className="px-5 py-3 bg-accent-cyan text-bg-deep font-semibold rounded-lg text-sm hover:bg-accent-cyan/90 disabled:opacity-50 disabled:cursor-not-allowed transition-all"
748
- >
749
- Connect
750
- </button>
751
- </div>
752
- {provider.apiKeyUrl && (
753
- <p className="text-xs text-text-muted">
754
- Get your API key from{' '}
755
- <a
756
- href={provider.apiKeyUrl}
757
- target="_blank"
758
- rel="noopener noreferrer"
759
- className="text-accent-cyan hover:underline"
760
- >
761
- {new URL(provider.apiKeyUrl).hostname}
762
- </a>
763
- </p>
764
- )}
765
- {provider.supportsOAuth && (
766
- <button
767
- onClick={() => setShowApiKeyFallback(prev => ({ ...prev, [provider.id]: false }))}
768
- className="text-xs text-text-muted hover:text-text-secondary transition-colors"
769
- >
770
- ← Back to OAuth login
771
- </button>
772
- )}
773
- </div>
774
- ) : provider.supportsOAuth ? (
775
- <div className="space-y-3">
776
- {/* CLI info for Codex */}
777
- {provider.id === 'codex' && (
778
- <div className="p-3 bg-accent-cyan/10 border border-accent-cyan/30 rounded-lg">
779
- <p className="text-sm text-accent-cyan font-medium mb-1">CLI-assisted authentication</p>
780
- <p className="text-xs text-accent-cyan/80">
781
- Codex auth uses a CLI command to capture the OAuth callback locally.
782
- Click the button below and we&apos;ll show you a command with a unique session token
783
- to run in your terminal before signing in with OpenAI.
784
- </p>
785
- </div>
786
- )}
787
- {/* Device flow toggle for providers that support it */}
788
- {provider.supportsDeviceFlow && (
789
- <label className="flex items-center gap-2 text-xs text-text-secondary cursor-pointer">
790
- <input
791
- type="checkbox"
792
- checked={useDeviceFlow[provider.id] || false}
793
- onChange={(e) => setUseDeviceFlow(prev => ({
794
- ...prev,
795
- [provider.id]: e.target.checked,
796
- }))}
797
- className="w-4 h-4 rounded border-border-subtle bg-bg-card text-accent-cyan focus:ring-accent-cyan/30 cursor-pointer"
798
- />
799
- Use device flow (easier for containers/headless)
800
- </label>
801
- )}
802
- <button
803
- onClick={() => startOAuthFlow(provider)}
804
- disabled={connectingProvider !== null}
805
- className="w-full py-3 px-4 bg-gradient-to-r from-accent-cyan to-[#00b8d9] text-bg-deep font-semibold rounded-lg text-sm hover:shadow-glow-cyan hover:-translate-y-0.5 disabled:opacity-50 disabled:cursor-not-allowed disabled:hover:translate-y-0 disabled:hover:shadow-none transition-all duration-200 flex items-center justify-center gap-2"
806
- >
807
- <LockIcon />
808
- Connect with {provider.displayName}
809
- </button>
810
- {provider.apiKeyUrl && (
811
- <button
812
- onClick={() => setShowApiKeyFallback(prev => ({ ...prev, [provider.id]: true }))}
813
- className="w-full text-xs text-text-muted hover:text-text-secondary transition-colors"
814
- >
815
- Or enter API key manually
816
- </button>
817
- )}
818
- </div>
819
- ) : (
820
- /* Provider doesn't support OAuth - show API key input directly */
821
- <div className="space-y-4">
822
- <div className="flex gap-3">
823
- <input
824
- type="password"
825
- placeholder={`Enter ${provider.displayName} ${provider.apiKeyName || 'API key'}`}
826
- value={connectingProvider === provider.id ? apiKeyInput : ''}
827
- onChange={(e) => {
828
- setConnectingProvider(provider.id);
829
- setApiKeyInput(e.target.value);
830
- }}
831
- onFocus={() => setConnectingProvider(provider.id)}
832
- className="flex-1 px-4 py-3 bg-bg-card border border-border-subtle rounded-lg text-sm text-text-primary placeholder:text-text-muted focus:outline-none focus:border-accent-cyan focus:ring-1 focus:ring-accent-cyan/30 transition-all"
833
- />
834
- <button
835
- onClick={() => submitApiKey(provider)}
836
- disabled={connectingProvider !== provider.id || !apiKeyInput.trim()}
837
- className="px-5 py-3 bg-accent-cyan text-bg-deep font-semibold rounded-lg text-sm hover:bg-accent-cyan/90 disabled:opacity-50 disabled:cursor-not-allowed transition-all"
838
- >
839
- Connect
840
- </button>
841
- </div>
842
- {provider.apiKeyUrl && (
843
- <p className="text-xs text-text-muted">
844
- Get your API key from{' '}
845
- <a
846
- href={provider.apiKeyUrl}
847
- target="_blank"
848
- rel="noopener noreferrer"
849
- className="text-accent-cyan hover:underline"
850
- >
851
- {new URL(provider.apiKeyUrl).hostname}
852
- </a>
853
- </p>
854
- )}
855
- <p className="text-xs text-amber-400/80">
856
- OAuth not available for {provider.displayName} in container environments
857
- </p>
858
- </div>
859
- )}
860
- </div>
861
- )}
862
-
863
- <div className="mt-4 pt-4 border-t border-border-subtle">
864
- <p className="text-xs text-text-muted">
865
- CLI: <code className="px-2 py-1 bg-bg-card rounded font-mono">{provider.cliCommand}</code>
866
- </p>
867
- </div>
868
- </div>
869
- ))}
870
- </div>
871
- </div>
872
- )}
873
-
874
- {/* Repositories Section */}
875
- {activeSection === 'repos' && (
876
- <div className="space-y-8">
877
- <SectionHeader
878
- title="Connected Repositories"
879
- subtitle="Repositories linked to this workspace"
880
- />
881
-
882
- <div className="space-y-3">
883
- {workspace.repositories.length > 0 ? (
884
- workspace.repositories.map((repo) => (
885
- <div
886
- key={repo.id}
887
- className="flex items-center justify-between p-4 bg-bg-tertiary rounded-lg border border-border-subtle"
888
- >
889
- <div className="flex items-center gap-3">
890
- <div className="w-10 h-10 rounded-lg bg-bg-card flex items-center justify-center">
891
- <RepoIcon />
892
- </div>
893
- <div>
894
- <p className="text-sm font-medium text-text-primary">{repo.fullName}</p>
895
- <p className="text-xs text-text-muted">
896
- {repo.lastSyncedAt
897
- ? `Synced ${new Date(repo.lastSyncedAt).toLocaleDateString()}`
898
- : 'Not synced'}
899
- </p>
900
- </div>
901
- </div>
902
- <div className="flex items-center gap-3">
903
- <button
904
- onClick={() => handleSyncRepo(repo.id)}
905
- disabled={syncingRepoId === repo.id || workspace.status !== 'running'}
906
- className="px-3 py-1.5 bg-accent-cyan/10 border border-accent-cyan/30 text-accent-cyan rounded-lg text-xs font-semibold hover:bg-accent-cyan/20 disabled:opacity-50 disabled:cursor-not-allowed transition-colors flex items-center gap-1.5"
907
- title={workspace.status !== 'running' ? 'Workspace must be running to sync' : 'Sync repository'}
908
- >
909
- {syncingRepoId === repo.id ? (
910
- <>
911
- <SyncIcon spinning />
912
- Syncing...
913
- </>
914
- ) : (
915
- <>
916
- <SyncIcon />
917
- Sync
918
- </>
919
- )}
920
- </button>
921
- <StatusBadge status={repo.syncStatus} />
922
- </div>
923
- </div>
924
- ))
925
- ) : (
926
- <div className="p-6 bg-bg-tertiary rounded-lg border border-border-subtle border-dashed text-center">
927
- <RepoIcon className="w-8 h-8 mx-auto mb-3 text-text-muted" />
928
- <p className="text-sm text-text-muted">No repositories connected</p>
929
- </div>
930
- )}
931
- </div>
932
-
933
- {unassignedRepos.length > 0 && (
934
- <>
935
- <SectionHeader
936
- title="Available Repositories"
937
- subtitle="Add more repositories to this workspace"
938
- />
939
- <div className="space-y-3">
940
- {unassignedRepos.map((repo) => (
941
- <div
942
- key={repo.id}
943
- className="flex items-center justify-between p-4 bg-bg-tertiary rounded-lg border border-border-subtle hover:border-accent-cyan/30 transition-colors"
944
- >
945
- <div className="flex items-center gap-3">
946
- <div className="w-10 h-10 rounded-lg bg-bg-card flex items-center justify-center">
947
- <RepoIcon />
948
- </div>
949
- <div>
950
- <p className="text-sm font-medium text-text-primary">{repo.fullName}</p>
951
- <p className="text-xs text-text-muted">
952
- {repo.isPrivate ? 'Private' : 'Public'}
953
- </p>
954
- </div>
955
- </div>
956
- <button
957
- onClick={() => handleAddRepo(repo.id)}
958
- className="px-4 py-2 bg-accent-cyan/10 border border-accent-cyan/30 text-accent-cyan rounded-lg text-xs font-semibold hover:bg-accent-cyan/20 transition-colors"
959
- >
960
- Add to Workspace
961
- </button>
962
- </div>
963
- ))}
964
- </div>
965
- </>
966
- )}
967
- </div>
968
- )}
969
-
970
- {/* GitHub Access Section */}
971
- {activeSection === 'github-access' && (
972
- <div className="space-y-6">
973
- <SectionHeader
974
- title="GitHub Repository Access"
975
- subtitle="Repositories you have access to via your GitHub account"
976
- />
977
- <RepoAccessPanel
978
- workspaces={
979
- workspace && workspace.repositories?.length > 0
980
- ? [{
981
- id: workspace.id,
982
- name: workspace.name,
983
- repositoryFullName: workspace.repositories[0].fullName,
984
- status: workspace.status as 'provisioning' | 'running' | 'stopped' | 'error',
985
- }]
986
- : []
987
- }
988
- onWorkspaceCreated={(workspaceId, repoFullName) => {
989
- // Refresh workspace data after creating
990
- cloudApi.getWorkspaceDetails(workspaceId).then(result => {
991
- if (result.success) {
992
- setWorkspace(result.data);
993
- }
994
- });
995
- }}
996
- onOpenWorkspace={(workspaceId) => {
997
- // Navigate to workspace or close settings
998
- if (onClose) {
999
- onClose();
1000
- }
1001
- }}
1002
- csrfToken={csrfToken}
1003
- className="bg-bg-tertiary rounded-xl border border-border-subtle overflow-hidden"
1004
- />
1005
- </div>
1006
- )}
1007
-
1008
- {/* Custom Domain Section */}
1009
- {activeSection === 'domain' && (
1010
- <div className="space-y-8">
1011
- <SectionHeader
1012
- title="Custom Domain"
1013
- subtitle="Connect your own domain to this workspace"
1014
- />
1015
-
1016
- <div className="p-5 bg-gradient-to-r from-accent-purple/10 to-accent-cyan/10 border border-accent-purple/20 rounded-xl">
1017
- <div className="flex items-center gap-3 mb-3">
1018
- <div className="w-10 h-10 rounded-lg bg-accent-purple/20 flex items-center justify-center">
1019
- <GlobeIcon className="text-accent-purple" />
1020
- </div>
1021
- <div>
1022
- <h4 className="text-sm font-semibold text-text-primary">Premium Feature</h4>
1023
- <p className="text-xs text-text-secondary">Requires Team or Enterprise plan</p>
1024
- </div>
1025
- </div>
1026
- </div>
1027
-
1028
- {workspace.customDomain ? (
1029
- <div className="space-y-4">
1030
- <div className="p-5 bg-bg-tertiary rounded-xl border border-border-subtle">
1031
- <div className="flex items-center justify-between mb-3">
1032
- <span className="text-xs text-text-muted uppercase tracking-wide font-semibold">
1033
- Current Domain
1034
- </span>
1035
- <StatusBadge status={workspace.customDomainStatus || 'pending'} />
1036
- </div>
1037
- <p className="text-lg font-mono text-text-primary">{workspace.customDomain}</p>
1038
- </div>
1039
-
1040
- {workspace.customDomainStatus === 'pending' && (
1041
- <ActionButton
1042
- onClick={handleVerifyDomain}
1043
- disabled={domainLoading}
1044
- variant="primary"
1045
- icon={<CheckIcon />}
1046
- fullWidth
1047
- >
1048
- {domainLoading ? 'Verifying...' : 'Verify DNS Configuration'}
1049
- </ActionButton>
1050
- )}
1051
-
1052
- <ActionButton
1053
- onClick={handleRemoveDomain}
1054
- disabled={domainLoading}
1055
- variant="danger"
1056
- icon={<TrashIcon />}
1057
- fullWidth
1058
- >
1059
- Remove Custom Domain
1060
- </ActionButton>
1061
- </div>
1062
- ) : (
1063
- <div className="space-y-4">
1064
- <div>
1065
- <label className="text-xs font-semibold text-text-muted uppercase tracking-wide mb-2 block">
1066
- Domain Name
1067
- </label>
1068
- <input
1069
- type="text"
1070
- value={customDomain}
1071
- onChange={(e) => setCustomDomain(e.target.value)}
1072
- placeholder="workspace.yourdomain.com"
1073
- className="w-full px-4 py-3 bg-bg-tertiary border border-border-subtle rounded-lg text-sm text-text-primary font-mono placeholder:text-text-muted focus:outline-none focus:border-accent-cyan focus:ring-1 focus:ring-accent-cyan/30 transition-all"
1074
- />
1075
- </div>
1076
-
1077
- <ActionButton
1078
- onClick={handleSetDomain}
1079
- disabled={domainLoading || !customDomain.trim()}
1080
- variant="primary"
1081
- icon={<GlobeIcon />}
1082
- fullWidth
1083
- >
1084
- {domainLoading ? 'Setting up...' : 'Set Custom Domain'}
1085
- </ActionButton>
1086
- </div>
1087
- )}
1088
-
1089
- {domainError && (
1090
- <div className="p-4 bg-error/10 border border-error/30 rounded-lg text-error text-sm">
1091
- {domainError}
1092
- </div>
1093
- )}
1094
-
1095
- {domainInstructions && (
1096
- <div className="p-5 bg-bg-tertiary rounded-xl border border-border-subtle space-y-4">
1097
- <h4 className="text-sm font-semibold text-text-primary flex items-center gap-2">
1098
- <InfoIcon />
1099
- DNS Configuration Required
1100
- </h4>
1101
- <p className="text-xs text-text-secondary">
1102
- Add the following DNS record to your domain provider:
1103
- </p>
1104
- <div className="grid grid-cols-3 gap-3">
1105
- <DNSField label="Type" value={domainInstructions.type} />
1106
- <DNSField label="Name" value={domainInstructions.name} />
1107
- <DNSField label="Value" value={domainInstructions.value} />
1108
- </div>
1109
- </div>
1110
- )}
1111
- </div>
1112
- )}
1113
-
1114
- {/* Danger Zone Section */}
1115
- {activeSection === 'danger' && (
1116
- <div className="space-y-8">
1117
- <div className="p-6 bg-error/5 border-2 border-error/20 rounded-xl">
1118
- <div className="flex items-center gap-3 mb-4">
1119
- <div className="w-10 h-10 rounded-lg bg-error/20 flex items-center justify-center">
1120
- <AlertIcon className="text-error" />
1121
- </div>
1122
- <div>
1123
- <h3 className="text-base font-semibold text-error">Danger Zone</h3>
1124
- <p className="text-xs text-text-secondary">
1125
- These actions are destructive and cannot be undone
1126
- </p>
1127
- </div>
1128
- </div>
1129
-
1130
- <div className="p-5 border border-error/30 rounded-lg bg-bg-primary">
1131
- <div className="flex items-center justify-between">
1132
- <div>
1133
- <h4 className="text-sm font-semibold text-text-primary">Delete Workspace</h4>
1134
- <p className="text-xs text-text-muted mt-1">
1135
- Permanently delete this workspace and all its data
1136
- </p>
1137
- </div>
1138
- <button
1139
- onClick={handleDelete}
1140
- className="px-5 py-2.5 bg-error text-white rounded-lg text-sm font-semibold hover:bg-error/90 transition-colors"
1141
- >
1142
- Delete Workspace
1143
- </button>
1144
- </div>
1145
- </div>
1146
- </div>
1147
- </div>
1148
- )}
1149
- </div>
1150
- </div>
1151
- );
1152
- }
1153
-
1154
- // Utility Components
1155
- function SectionHeader({ title, subtitle }: { title: string; subtitle: string }) {
1156
- return (
1157
- <div className="mb-4">
1158
- <h3 className="text-sm font-semibold text-text-muted uppercase tracking-wide">{title}</h3>
1159
- <p className="text-xs text-text-muted mt-1">{subtitle}</p>
1160
- </div>
1161
- );
1162
- }
1163
-
1164
- function InfoCard({
1165
- label,
1166
- value,
1167
- valueColor = 'text-text-primary',
1168
- mono = false,
1169
- indicator = false,
1170
- }: {
1171
- label: string;
1172
- value: string;
1173
- valueColor?: string;
1174
- mono?: boolean;
1175
- indicator?: boolean;
1176
- }) {
1177
- return (
1178
- <div className="p-4 bg-bg-tertiary rounded-lg border border-border-subtle">
1179
- <label className="text-xs text-text-muted uppercase tracking-wide font-medium">{label}</label>
1180
- <div className="flex items-center gap-2 mt-1">
1181
- {indicator && <div className="w-2 h-2 rounded-full bg-success animate-pulse" />}
1182
- <p className={`text-sm font-medium ${valueColor} ${mono ? 'font-mono' : ''} break-all`}>
1183
- {value}
1184
- </p>
1185
- </div>
1186
- </div>
1187
- );
1188
- }
1189
-
1190
- function ActionButton({
1191
- children,
1192
- onClick,
1193
- disabled,
1194
- variant,
1195
- icon,
1196
- fullWidth,
1197
- }: {
1198
- children: React.ReactNode;
1199
- onClick: () => void;
1200
- disabled?: boolean;
1201
- variant: 'primary' | 'warning' | 'danger';
1202
- icon?: React.ReactNode;
1203
- fullWidth?: boolean;
1204
- }) {
1205
- const variants = {
1206
- primary: 'bg-accent-cyan/10 border-accent-cyan/30 text-accent-cyan hover:bg-accent-cyan/20',
1207
- warning: 'bg-amber-400/10 border-amber-400/30 text-amber-400 hover:bg-amber-400/20',
1208
- danger: 'bg-error/10 border-error/30 text-error hover:bg-error/20',
1209
- };
1210
-
1211
- return (
1212
- <button
1213
- onClick={onClick}
1214
- disabled={disabled}
1215
- className={`${fullWidth ? 'w-full' : ''} px-5 py-2.5 border rounded-lg text-sm font-semibold transition-all duration-200 disabled:opacity-50 disabled:cursor-not-allowed flex items-center justify-center gap-2 ${variants[variant]}`}
1216
- >
1217
- {icon}
1218
- {children}
1219
- </button>
1220
- );
1221
- }
1222
-
1223
- function StatusBadge({ status }: { status: string }) {
1224
- const styles: Record<string, string> = {
1225
- synced: 'bg-success/15 text-success border-success/30',
1226
- active: 'bg-success/15 text-success border-success/30',
1227
- syncing: 'bg-accent-cyan/15 text-accent-cyan border-accent-cyan/30',
1228
- verifying: 'bg-accent-cyan/15 text-accent-cyan border-accent-cyan/30',
1229
- pending: 'bg-amber-400/15 text-amber-400 border-amber-400/30',
1230
- error: 'bg-error/15 text-error border-error/30',
1231
- };
1232
-
1233
- return (
1234
- <span className={`text-xs px-3 py-1 rounded-full border ${styles[status] || 'bg-bg-hover text-text-muted border-border-subtle'}`}>
1235
- {status}
1236
- </span>
1237
- );
1238
- }
1239
-
1240
- function DNSField({ label, value }: { label: string; value: string }) {
1241
- return (
1242
- <div className="p-3 bg-bg-card rounded-lg">
1243
- <label className="text-xs text-text-muted block mb-1">{label}</label>
1244
- <p className="font-mono text-sm text-text-primary break-all">{value}</p>
1245
- </div>
1246
- );
1247
- }
1248
-
1249
- // Icons
1250
- function SettingsGearIcon() {
1251
- return (
1252
- <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
1253
- <circle cx="12" cy="12" r="3" />
1254
- <path d="M19.4 15a1.65 1.65 0 0 0 .33 1.82l.06.06a2 2 0 0 1 0 2.83 2 2 0 0 1-2.83 0l-.06-.06a1.65 1.65 0 0 0-1.82-.33 1.65 1.65 0 0 0-1 1.51V21a2 2 0 0 1-2 2 2 2 0 0 1-2-2v-.09A1.65 1.65 0 0 0 9 19.4a1.65 1.65 0 0 0-1.82.33l-.06.06a2 2 0 0 1-2.83 0 2 2 0 0 1 0-2.83l.06-.06a1.65 1.65 0 0 0 .33-1.82 1.65 1.65 0 0 0-1.51-1H3a2 2 0 0 1-2-2 2 2 0 0 1 2-2h.09A1.65 1.65 0 0 0 4.6 9a1.65 1.65 0 0 0-.33-1.82l-.06-.06a2 2 0 0 1 0-2.83 2 2 0 0 1 2.83 0l.06.06a1.65 1.65 0 0 0 1.82.33H9a1.65 1.65 0 0 0 1-1.51V3a2 2 0 0 1 2-2 2 2 0 0 1 2 2v.09a1.65 1.65 0 0 0 1 1.51 1.65 1.65 0 0 0 1.82-.33l.06-.06a2 2 0 0 1 2.83 0 2 2 0 0 1 0 2.83l-.06.06a1.65 1.65 0 0 0-.33 1.82V9a1.65 1.65 0 0 0 1.51 1H21a2 2 0 0 1 2 2 2 2 0 0 1-2 2h-.09a1.65 1.65 0 0 0-1.51 1z" />
1255
- </svg>
1256
- );
1257
- }
1258
-
1259
- function ProviderIcon() {
1260
- return (
1261
- <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
1262
- <path d="M12 2L2 7l10 5 10-5-10-5z" />
1263
- <path d="M2 17l10 5 10-5" />
1264
- <path d="M2 12l10 5 10-5" />
1265
- </svg>
1266
- );
1267
- }
1268
-
1269
- function RepoIcon({ className = '' }: { className?: string }) {
1270
- return (
1271
- <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round" className={`text-text-muted ${className}`}>
1272
- <path d="M22 19a2 2 0 0 1-2 2H4a2 2 0 0 1-2-2V5a2 2 0 0 1 2-2h5l2 3h9a2 2 0 0 1 2 2z" />
1273
- </svg>
1274
- );
1275
- }
1276
-
1277
- function GlobeIcon({ className = '' }: { className?: string }) {
1278
- return (
1279
- <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round" className={className}>
1280
- <circle cx="12" cy="12" r="10" />
1281
- <line x1="2" y1="12" x2="22" y2="12" />
1282
- <path d="M12 2a15.3 15.3 0 0 1 4 10 15.3 15.3 0 0 1-4 10 15.3 15.3 0 0 1-4-10 15.3 15.3 0 0 1 4-10z" />
1283
- </svg>
1284
- );
1285
- }
1286
-
1287
- function GitHubIcon() {
1288
- return (
1289
- <svg width="14" height="14" viewBox="0 0 24 24" fill="currentColor">
1290
- <path d="M12 0C5.37 0 0 5.37 0 12c0 5.31 3.435 9.795 8.205 11.385.6.105.825-.255.825-.57 0-.285-.015-1.23-.015-2.235-3.015.555-3.795-.735-4.035-1.41-.135-.345-.72-1.41-1.23-1.695-.42-.225-1.02-.78-.015-.795.945-.015 1.62.87 1.845 1.23 1.08 1.815 2.805 1.305 3.495.99.105-.78.42-1.305.765-1.605-2.67-.3-5.46-1.335-5.46-5.925 0-1.305.465-2.385 1.23-3.225-.12-.3-.54-1.53.12-3.18 0 0 1.005-.315 3.3 1.23.96-.27 1.98-.405 3-.405s2.04.135 3 .405c2.295-1.56 3.3-1.23 3.3-1.23.66 1.65.24 2.88.12 3.18.765.84 1.23 1.905 1.23 3.225 0 4.605-2.805 5.625-5.475 5.925.435.375.81 1.095.81 2.22 0 1.605-.015 2.895-.015 3.3 0 .315.225.69.825.57A12.02 12.02 0 0024 12c0-6.63-5.37-12-12-12z" />
1291
- </svg>
1292
- );
1293
- }
1294
-
1295
- function AlertIcon({ className = '' }: { className?: string }) {
1296
- return (
1297
- <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round" className={className}>
1298
- <path d="M10.29 3.86L1.82 18a2 2 0 0 0 1.71 3h16.94a2 2 0 0 0 1.71-3L13.71 3.86a2 2 0 0 0-3.42 0z" />
1299
- <line x1="12" y1="9" x2="12" y2="13" />
1300
- <line x1="12" y1="17" x2="12.01" y2="17" />
1301
- </svg>
1302
- );
1303
- }
1304
-
1305
- function LockIcon() {
1306
- return (
1307
- <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
1308
- <rect x="3" y="11" width="18" height="11" rx="2" ry="2" />
1309
- <path d="M7 11V7a5 5 0 0 1 10 0v4" />
1310
- </svg>
1311
- );
1312
- }
1313
-
1314
- function StopIcon() {
1315
- return (
1316
- <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
1317
- <rect x="6" y="6" width="12" height="12" />
1318
- </svg>
1319
- );
1320
- }
1321
-
1322
- function RestartIcon() {
1323
- return (
1324
- <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
1325
- <path d="M23 4v6h-6" />
1326
- <path d="M20.49 15a9 9 0 1 1-2.12-9.36L23 10" />
1327
- </svg>
1328
- );
1329
- }
1330
-
1331
- function CheckIcon() {
1332
- return (
1333
- <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
1334
- <polyline points="20 6 9 17 4 12" />
1335
- </svg>
1336
- );
1337
- }
1338
-
1339
- function TrashIcon() {
1340
- return (
1341
- <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
1342
- <polyline points="3 6 5 6 21 6" />
1343
- <path d="M19 6v14a2 2 0 0 1-2 2H7a2 2 0 0 1-2-2V6m3 0V4a2 2 0 0 1 2-2h4a2 2 0 0 1 2 2v2" />
1344
- </svg>
1345
- );
1346
- }
1347
-
1348
- function CloseIcon() {
1349
- return (
1350
- <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
1351
- <line x1="18" y1="6" x2="6" y2="18" />
1352
- <line x1="6" y1="6" x2="18" y2="18" />
1353
- </svg>
1354
- );
1355
- }
1356
-
1357
- function InfoIcon() {
1358
- return (
1359
- <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
1360
- <circle cx="12" cy="12" r="10" />
1361
- <line x1="12" y1="16" x2="12" y2="12" />
1362
- <line x1="12" y1="8" x2="12.01" y2="8" />
1363
- </svg>
1364
- );
1365
- }
1366
-
1367
- function SyncIcon({ spinning = false }: { spinning?: boolean } = {}) {
1368
- return (
1369
- <svg
1370
- width="12"
1371
- height="12"
1372
- viewBox="0 0 24 24"
1373
- fill="none"
1374
- stroke="currentColor"
1375
- strokeWidth="2"
1376
- strokeLinecap="round"
1377
- strokeLinejoin="round"
1378
- className={spinning ? 'animate-spin' : ''}
1379
- >
1380
- <path d="M23 4v6h-6" />
1381
- <path d="M1 20v-6h6" />
1382
- <path d="M3.51 9a9 9 0 0 1 14.85-3.36L23 10" />
1383
- <path d="M20.49 15a9 9 0 0 1-14.85 3.36L1 14" />
1384
- </svg>
1385
- );
1386
- }