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,140 +0,0 @@
1
- /**
2
- * usePinnedAgents Hook
3
- *
4
- * Manages pinned agents with localStorage persistence.
5
- * Pinned agents appear at the top of the agents panel.
6
- */
7
-
8
- import { useState, useCallback, useEffect, useMemo } from 'react';
9
-
10
- export const STORAGE_KEY = 'agent-relay-pinned-agents';
11
- export const MAX_PINNED = 5;
12
-
13
- /**
14
- * Load pinned agents from localStorage
15
- * Exported for testing
16
- */
17
- export function loadPinnedAgents(): string[] {
18
- try {
19
- if (typeof localStorage === 'undefined') return [];
20
- const stored = localStorage.getItem(STORAGE_KEY);
21
- if (stored) {
22
- const parsed = JSON.parse(stored);
23
- if (Array.isArray(parsed)) {
24
- return parsed.slice(0, MAX_PINNED);
25
- }
26
- }
27
- } catch {
28
- // localStorage not available or invalid data
29
- }
30
- return [];
31
- }
32
-
33
- /**
34
- * Save pinned agents to localStorage
35
- * Exported for testing
36
- */
37
- export function savePinnedAgents(agents: string[]): void {
38
- try {
39
- if (typeof localStorage === 'undefined') return;
40
- localStorage.setItem(STORAGE_KEY, JSON.stringify(agents));
41
- } catch {
42
- // localStorage not available
43
- }
44
- }
45
-
46
- /**
47
- * Pin an agent to the list
48
- * Returns the new list and whether the pin was successful
49
- */
50
- export function pinAgent(
51
- currentPinned: string[],
52
- agentName: string
53
- ): { newPinned: string[]; success: boolean } {
54
- if (currentPinned.includes(agentName)) {
55
- return { newPinned: currentPinned, success: true }; // Already pinned
56
- }
57
- if (currentPinned.length >= MAX_PINNED) {
58
- return { newPinned: currentPinned, success: false }; // At max capacity
59
- }
60
- return { newPinned: [...currentPinned, agentName], success: true };
61
- }
62
-
63
- /**
64
- * Unpin an agent from the list
65
- */
66
- export function unpinAgent(currentPinned: string[], agentName: string): string[] {
67
- return currentPinned.filter((name) => name !== agentName);
68
- }
69
-
70
- export interface UsePinnedAgentsReturn {
71
- /** Array of pinned agent names */
72
- pinnedAgents: string[];
73
- /** Check if an agent is pinned */
74
- isPinned: (agentName: string) => boolean;
75
- /** Toggle pin status for an agent */
76
- togglePin: (agentName: string) => void;
77
- /** Pin an agent (no-op if already pinned or at max) */
78
- pin: (agentName: string) => boolean;
79
- /** Unpin an agent */
80
- unpin: (agentName: string) => void;
81
- /** Whether max pins reached */
82
- isMaxPinned: boolean;
83
- /** Maximum number of pinned agents allowed */
84
- maxPinned: number;
85
- }
86
-
87
- export function usePinnedAgents(): UsePinnedAgentsReturn {
88
- const [pinnedAgents, setPinnedAgents] = useState<string[]>(() => loadPinnedAgents());
89
-
90
- // Persist to localStorage when pinnedAgents changes
91
- useEffect(() => {
92
- savePinnedAgents(pinnedAgents);
93
- }, [pinnedAgents]);
94
-
95
- const isPinned = useCallback(
96
- (agentName: string) => pinnedAgents.includes(agentName),
97
- [pinnedAgents]
98
- );
99
-
100
- const pin = useCallback(
101
- (agentName: string): boolean => {
102
- const { newPinned, success } = pinAgent(pinnedAgents, agentName);
103
- if (newPinned !== pinnedAgents) {
104
- setPinnedAgents(newPinned);
105
- }
106
- return success;
107
- },
108
- [pinnedAgents]
109
- );
110
-
111
- const unpin = useCallback((agentName: string) => {
112
- setPinnedAgents((prev) => unpinAgent(prev, agentName));
113
- }, []);
114
-
115
- const togglePin = useCallback(
116
- (agentName: string) => {
117
- if (isPinned(agentName)) {
118
- unpin(agentName);
119
- } else {
120
- pin(agentName);
121
- }
122
- },
123
- [isPinned, pin, unpin]
124
- );
125
-
126
- const isMaxPinned = useMemo(
127
- () => pinnedAgents.length >= MAX_PINNED,
128
- [pinnedAgents]
129
- );
130
-
131
- return {
132
- pinnedAgents,
133
- isPinned,
134
- togglePin,
135
- pin,
136
- unpin,
137
- isMaxPinned,
138
- maxPinned: MAX_PINNED,
139
- };
140
- }
@@ -1,340 +0,0 @@
1
- /**
2
- * usePresence Hook
3
- *
4
- * Manages user presence and typing indicators via WebSocket.
5
- * - Tracks which users are currently online
6
- * - Sends/receives typing indicator events
7
- * - Handles user presence announcements
8
- */
9
-
10
- import { useState, useEffect, useCallback, useRef } from 'react';
11
-
12
- /** User presence information */
13
- export interface UserPresence {
14
- /** Username (GitHub username in cloud mode) */
15
- username: string;
16
- /** Optional avatar URL */
17
- avatarUrl?: string;
18
- /** When the user came online */
19
- connectedAt: string;
20
- /** Last activity timestamp */
21
- lastSeen: string;
22
- /** Whether user is currently typing */
23
- isTyping?: boolean;
24
- }
25
-
26
- /** Typing indicator information */
27
- export interface TypingIndicator {
28
- /** Username of the person typing */
29
- username: string;
30
- /** Avatar URL if available */
31
- avatarUrl?: string;
32
- /** Timestamp when typing started */
33
- startedAt: number;
34
- }
35
-
36
- export interface UsePresenceOptions {
37
- /** Current user info (if logged in) */
38
- currentUser?: {
39
- username: string;
40
- avatarUrl?: string;
41
- };
42
- /** WebSocket URL (defaults to same as main WebSocket) */
43
- wsUrl?: string;
44
- /** Whether to auto-connect */
45
- autoConnect?: boolean;
46
- /** Optional handler for additional messages (e.g., channel_message) */
47
- onEvent?: (event: any) => void;
48
- /** Workspace ID for channel subscription (cloud mode) */
49
- workspaceId?: string;
50
- }
51
-
52
- export interface UsePresenceReturn {
53
- /** List of online users */
54
- onlineUsers: UserPresence[];
55
- /** Currently typing users (excluding self) */
56
- typingUsers: TypingIndicator[];
57
- /** Send typing indicator */
58
- sendTyping: (isTyping: boolean) => void;
59
- /** Whether connected to presence system */
60
- isConnected: boolean;
61
- }
62
-
63
- /**
64
- * Get the presence WebSocket URL
65
- */
66
- function getPresenceUrl(): string {
67
- const isDev = process.env.NODE_ENV === 'development';
68
-
69
- if (typeof window === 'undefined') {
70
- return 'ws://localhost:3889/ws/presence';
71
- }
72
-
73
- // Dev mode only: Next.js on 3888, dashboard server on 3889
74
- // In production (static export), use same host regardless of port
75
- if (isDev && window.location.port === '3888') {
76
- const host = window.location.hostname || 'localhost';
77
- return `ws://${host}:3889/ws/presence`;
78
- }
79
-
80
- const protocol = window.location.protocol === 'https:' ? 'wss:' : 'ws:';
81
- return `${protocol}//${window.location.host}/ws/presence`;
82
- }
83
-
84
- export function usePresence(options: UsePresenceOptions = {}): UsePresenceReturn {
85
- const { currentUser, wsUrl, autoConnect = true, onEvent, workspaceId } = options;
86
-
87
- const [onlineUsers, setOnlineUsers] = useState<UserPresence[]>([]);
88
- const [typingUsers, setTypingUsers] = useState<TypingIndicator[]>([]);
89
- const [isConnected, setIsConnected] = useState(false);
90
-
91
- const wsRef = useRef<WebSocket | null>(null);
92
- const reconnectTimeoutRef = useRef<NodeJS.Timeout | null>(null);
93
- const typingTimeoutRef = useRef<NodeJS.Timeout | null>(null);
94
- const isConnectingRef = useRef(false); // Prevent race conditions
95
- const currentUserRef = useRef(currentUser);
96
- currentUserRef.current = currentUser; // Keep ref in sync with prop
97
- const workspaceIdRef = useRef(workspaceId);
98
- workspaceIdRef.current = workspaceId; // Keep ref in sync with prop
99
- const onEventRef = useRef(onEvent);
100
- onEventRef.current = onEvent; // Keep ref in sync with callback prop
101
-
102
- // Clear stale typing indicators (after 3 seconds of no update)
103
- useEffect(() => {
104
- const interval = setInterval(() => {
105
- const now = Date.now();
106
- setTypingUsers((prev) =>
107
- prev.filter((t) => now - t.startedAt < 3000)
108
- );
109
- }, 1000);
110
-
111
- return () => clearInterval(interval);
112
- }, []);
113
-
114
- const connect = useCallback(() => {
115
- const user = currentUserRef.current;
116
- if (!user) return; // Don't connect without user info
117
- if (wsRef.current?.readyState === WebSocket.OPEN) return;
118
- if (isConnectingRef.current) return; // Prevent concurrent connect attempts
119
-
120
- isConnectingRef.current = true;
121
- const url = wsUrl || getPresenceUrl();
122
-
123
- try {
124
- const ws = new WebSocket(url);
125
-
126
- ws.onopen = () => {
127
- isConnectingRef.current = false;
128
- setIsConnected(true);
129
-
130
- // Announce presence (use ref to get latest user info)
131
- const currentUserInfo = currentUserRef.current;
132
- if (currentUserInfo) {
133
- ws.send(JSON.stringify({
134
- type: 'presence',
135
- action: 'join',
136
- user: {
137
- username: currentUserInfo.username,
138
- avatarUrl: currentUserInfo.avatarUrl,
139
- },
140
- }));
141
-
142
- // Subscribe to channel messages for this workspace (cloud mode)
143
- // This enables receiving real-time channel messages from other users
144
- const wsId = workspaceIdRef.current;
145
- if (wsId) {
146
- ws.send(JSON.stringify({
147
- type: 'subscribe_channels',
148
- workspaceId: wsId,
149
- }));
150
- }
151
- }
152
- };
153
-
154
- ws.onclose = () => {
155
- isConnectingRef.current = false;
156
- setIsConnected(false);
157
- wsRef.current = null;
158
-
159
- // Reconnect after 2 seconds (only if not intentionally disconnected)
160
- if (currentUserRef.current) {
161
- reconnectTimeoutRef.current = setTimeout(() => {
162
- connect();
163
- }, 2000);
164
- }
165
- };
166
-
167
- ws.onerror = (event) => {
168
- console.error('[usePresence] Error:', event);
169
- };
170
-
171
- ws.onmessage = (event) => {
172
- try {
173
- const msg = JSON.parse(event.data);
174
-
175
- switch (msg.type) {
176
- case 'presence_list':
177
- // Full list of online users
178
- setOnlineUsers(msg.users || []);
179
- break;
180
-
181
- case 'presence_join':
182
- // User came online
183
- setOnlineUsers((prev) => {
184
- const filtered = prev.filter((u) => u.username !== msg.user.username);
185
- return [...filtered, msg.user];
186
- });
187
- // Also forward to onEvent for activity feed
188
- onEventRef.current?.(msg);
189
- break;
190
-
191
- case 'presence_leave':
192
- // User went offline
193
- setOnlineUsers((prev) =>
194
- prev.filter((u) => u.username !== msg.username)
195
- );
196
- setTypingUsers((prev) =>
197
- prev.filter((t) => t.username !== msg.username)
198
- );
199
- // Also forward to onEvent for activity feed
200
- onEventRef.current?.(msg);
201
- break;
202
-
203
- case 'typing':
204
- // Typing indicator update
205
- if (msg.username === currentUserRef.current?.username) break; // Ignore self
206
-
207
- if (msg.isTyping) {
208
- setTypingUsers((prev) => {
209
- const filtered = prev.filter((t) => t.username !== msg.username);
210
- return [
211
- ...filtered,
212
- {
213
- username: msg.username,
214
- avatarUrl: msg.avatarUrl,
215
- startedAt: Date.now(),
216
- },
217
- ];
218
- });
219
- } else {
220
- setTypingUsers((prev) =>
221
- prev.filter((t) => t.username !== msg.username)
222
- );
223
- }
224
- break;
225
-
226
- default:
227
- onEventRef.current?.(msg);
228
- }
229
- } catch (e) {
230
- console.error('[usePresence] Failed to parse message:', e);
231
- }
232
- };
233
-
234
- wsRef.current = ws;
235
- } catch (e) {
236
- console.error('[usePresence] Failed to create WebSocket:', e);
237
- }
238
- }, [wsUrl]); // Use ref for currentUser to avoid dependency
239
-
240
- const disconnect = useCallback(() => {
241
- // Clear reconnect timeout first
242
- if (reconnectTimeoutRef.current) {
243
- clearTimeout(reconnectTimeoutRef.current);
244
- reconnectTimeoutRef.current = null;
245
- }
246
-
247
- // Reset connecting flag
248
- isConnectingRef.current = false;
249
-
250
- if (wsRef.current) {
251
- // Prevent auto-reconnect by removing onclose handler before closing
252
- const ws = wsRef.current;
253
- ws.onclose = null;
254
- ws.onerror = null;
255
-
256
- // Send leave message before closing
257
- const user = currentUserRef.current;
258
- if (ws.readyState === WebSocket.OPEN && user) {
259
- ws.send(JSON.stringify({
260
- type: 'presence',
261
- action: 'leave',
262
- username: user.username,
263
- }));
264
- }
265
- ws.close();
266
- wsRef.current = null;
267
- }
268
-
269
- setIsConnected(false);
270
- }, []); // Use ref for currentUser to avoid dependency
271
-
272
- // Send typing indicator
273
- const sendTyping = useCallback((isTyping: boolean) => {
274
- if (!wsRef.current || wsRef.current.readyState !== WebSocket.OPEN) return;
275
- const user = currentUserRef.current;
276
- if (!user) return;
277
-
278
- // Clear any existing timeout first
279
- if (typingTimeoutRef.current) {
280
- clearTimeout(typingTimeoutRef.current);
281
- typingTimeoutRef.current = null;
282
- }
283
-
284
- wsRef.current.send(JSON.stringify({
285
- type: 'typing',
286
- isTyping,
287
- username: user.username,
288
- avatarUrl: user.avatarUrl,
289
- }));
290
-
291
- // Only set auto-clear timeout when starting to type
292
- if (isTyping) {
293
- typingTimeoutRef.current = setTimeout(() => {
294
- typingTimeoutRef.current = null;
295
- sendTyping(false);
296
- }, 3000);
297
- }
298
- }, []); // Use ref for currentUser to avoid dependency
299
-
300
- // Connect when user is available
301
- useEffect(() => {
302
- if (!autoConnect || !currentUserRef.current) return;
303
-
304
- // Prevent connecting if already connected or connecting
305
- if (wsRef.current && wsRef.current.readyState !== WebSocket.CLOSED) {
306
- return;
307
- }
308
-
309
- connect();
310
-
311
- return () => {
312
- disconnect();
313
- };
314
- // Callbacks are now stable (use refs internally), so only need to depend on user identity
315
- }, [autoConnect, currentUser?.username, connect, disconnect]);
316
-
317
- // Send leave on page unload
318
- useEffect(() => {
319
- const handleUnload = () => {
320
- const user = currentUserRef.current;
321
- if (wsRef.current?.readyState === WebSocket.OPEN && user) {
322
- wsRef.current.send(JSON.stringify({
323
- type: 'presence',
324
- action: 'leave',
325
- username: user.username,
326
- }));
327
- }
328
- };
329
-
330
- window.addEventListener('beforeunload', handleUnload);
331
- return () => window.removeEventListener('beforeunload', handleUnload);
332
- }, []); // Use ref for currentUser to avoid dependency
333
-
334
- return {
335
- onlineUsers,
336
- typingUsers,
337
- sendTyping,
338
- isConnected,
339
- };
340
- }
@@ -1,130 +0,0 @@
1
- /**
2
- * useRecentRepos Hook
3
- *
4
- * Tracks and persists recently accessed repositories/projects.
5
- * Stores in localStorage for persistence across sessions.
6
- */
7
-
8
- import { useState, useEffect, useCallback } from 'react';
9
- import type { Project } from '../../types';
10
-
11
- const STORAGE_KEY = 'relay:recentRepos';
12
- const MAX_RECENT = 5;
13
-
14
- export interface RecentRepo {
15
- id: string;
16
- path: string;
17
- name?: string;
18
- lastAccessed: number;
19
- }
20
-
21
- export interface UseRecentReposOptions {
22
- /** Maximum number of recent repos to track (default: 5) */
23
- maxRecent?: number;
24
- }
25
-
26
- export interface UseRecentReposReturn {
27
- /** List of recent repos, most recent first */
28
- recentRepos: RecentRepo[];
29
- /** Add or update a repo in recent list */
30
- addRecentRepo: (project: Project) => void;
31
- /** Remove a repo from recent list */
32
- removeRecentRepo: (id: string) => void;
33
- /** Clear all recent repos */
34
- clearRecentRepos: () => void;
35
- /** Get recent repos as Project-like objects for display */
36
- getRecentProjects: (allProjects: Project[]) => Project[];
37
- }
38
-
39
- /**
40
- * Load recent repos from localStorage
41
- */
42
- function loadRecentRepos(): RecentRepo[] {
43
- try {
44
- const stored = localStorage.getItem(STORAGE_KEY);
45
- if (!stored) return [];
46
- const parsed = JSON.parse(stored);
47
- if (!Array.isArray(parsed)) return [];
48
- return parsed;
49
- } catch {
50
- return [];
51
- }
52
- }
53
-
54
- /**
55
- * Save recent repos to localStorage
56
- */
57
- function saveRecentRepos(repos: RecentRepo[]): void {
58
- try {
59
- localStorage.setItem(STORAGE_KEY, JSON.stringify(repos));
60
- } catch {
61
- // Silently fail if localStorage is not available
62
- }
63
- }
64
-
65
- export function useRecentRepos(options: UseRecentReposOptions = {}): UseRecentReposReturn {
66
- const maxRecent = options.maxRecent ?? MAX_RECENT;
67
- const [recentRepos, setRecentRepos] = useState<RecentRepo[]>([]);
68
-
69
- // Load from localStorage on mount
70
- useEffect(() => {
71
- setRecentRepos(loadRecentRepos());
72
- }, []);
73
-
74
- // Add or update a repo in recent list
75
- const addRecentRepo = useCallback((project: Project) => {
76
- setRecentRepos((prev) => {
77
- // Remove if already exists
78
- const filtered = prev.filter((r) => r.id !== project.id);
79
-
80
- // Add to front with current timestamp
81
- const newRepo: RecentRepo = {
82
- id: project.id,
83
- path: project.path,
84
- name: project.name,
85
- lastAccessed: Date.now(),
86
- };
87
-
88
- // Keep only maxRecent items
89
- const updated = [newRepo, ...filtered].slice(0, maxRecent);
90
-
91
- // Persist to localStorage
92
- saveRecentRepos(updated);
93
-
94
- return updated;
95
- });
96
- }, [maxRecent]);
97
-
98
- // Remove a repo from recent list
99
- const removeRecentRepo = useCallback((id: string) => {
100
- setRecentRepos((prev) => {
101
- const updated = prev.filter((r) => r.id !== id);
102
- saveRecentRepos(updated);
103
- return updated;
104
- });
105
- }, []);
106
-
107
- // Clear all recent repos
108
- const clearRecentRepos = useCallback(() => {
109
- setRecentRepos([]);
110
- saveRecentRepos([]);
111
- }, []);
112
-
113
- // Get recent repos as Project objects (matched against current projects)
114
- const getRecentProjects = useCallback((allProjects: Project[]): Project[] => {
115
- const projectMap = new Map(allProjects.map((p) => [p.id, p]));
116
- return recentRepos
117
- .map((r) => projectMap.get(r.id))
118
- .filter((p): p is Project => p !== undefined);
119
- }, [recentRepos]);
120
-
121
- return {
122
- recentRepos,
123
- addRecentRepo,
124
- removeRecentRepo,
125
- clearRecentRepos,
126
- getRecentProjects,
127
- };
128
- }
129
-
130
- export default useRecentRepos;