agent-relay 2.0.21 → 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.
- package/bin/relay-pty-linux-arm64 +0 -0
- package/dist/src/cli/index.d.ts +3 -3
- package/dist/src/cli/index.js +31 -100
- package/package.json +22 -29
- package/packages/api-types/package.json +1 -1
- package/packages/bridge/package.json +8 -8
- package/packages/cli-tester/package.json +1 -1
- package/packages/cloud/dist/server.js +25 -4
- package/packages/cloud/package.json +6 -6
- package/packages/config/package.json +2 -2
- package/packages/continuity/package.json +1 -1
- package/packages/daemon/dist/orchestrator.js +21 -1
- package/packages/daemon/dist/router.d.ts +5 -0
- package/packages/daemon/dist/router.js +31 -0
- package/packages/daemon/dist/server.d.ts +5 -0
- package/packages/daemon/dist/server.js +131 -1
- package/packages/daemon/package.json +12 -12
- package/packages/hooks/package.json +4 -4
- package/packages/mcp/dist/client.d.ts +15 -0
- package/packages/mcp/dist/client.js +9 -0
- package/packages/mcp/dist/server.js +13 -1
- package/packages/mcp/dist/tools/index.d.ts +2 -0
- package/packages/mcp/dist/tools/index.js +2 -0
- package/packages/mcp/dist/tools/relay-connected.d.ts +17 -0
- package/packages/mcp/dist/tools/relay-connected.js +40 -0
- package/packages/mcp/dist/tools/relay-remove-agent.d.ts +20 -0
- package/packages/mcp/dist/tools/relay-remove-agent.js +50 -0
- package/packages/mcp/package.json +2 -2
- package/packages/memory/package.json +2 -2
- package/packages/policy/package.json +2 -2
- package/packages/protocol/dist/types.d.ts +46 -1
- package/packages/protocol/package.json +1 -1
- package/packages/resiliency/package.json +1 -1
- package/packages/sdk/dist/client.d.ts +22 -1
- package/packages/sdk/dist/client.js +31 -0
- package/packages/sdk/dist/protocol/index.d.ts +1 -1
- package/packages/sdk/dist/protocol/types.d.ts +35 -1
- package/packages/sdk/package.json +2 -2
- package/packages/spawner/package.json +1 -1
- package/packages/state/package.json +1 -1
- package/packages/storage/dist/adapter.d.ts +4 -0
- package/packages/storage/dist/sqlite-adapter.d.ts +10 -0
- package/packages/storage/dist/sqlite-adapter.js +26 -0
- package/packages/storage/package.json +2 -2
- package/packages/telemetry/package.json +1 -1
- package/packages/trajectory/package.json +2 -2
- package/packages/user-directory/package.json +2 -2
- package/packages/utils/dist/update-checker.js +4 -0
- package/packages/utils/package.json +1 -1
- package/packages/wrapper/package.json +6 -6
- package/deploy/workspace/codex.config.toml +0 -20
- package/deploy/workspace/entrypoint-browser.sh +0 -118
- package/deploy/workspace/entrypoint.sh +0 -612
- package/deploy/workspace/gh-credential-relay +0 -90
- package/deploy/workspace/gh-relay +0 -156
- package/deploy/workspace/git-credential-relay +0 -330
- package/deploy/workspace/git-credential-relay.test.sh +0 -230
- package/dist/dashboard/out/404.html +0 -1
- package/dist/dashboard/out/_next/static/7MZPqYkVGw3EGzVBkVmY9/_buildManifest.js +0 -1
- package/dist/dashboard/out/_next/static/7MZPqYkVGw3EGzVBkVmY9/_ssgManifest.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/116-a883fca163f3a5bc.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/117-c8afed19e821a35d.js +0 -2
- package/dist/dashboard/out/_next/static/chunks/282-980c2eb8fff20123.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/320-a6304232cd0ee2ce.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/532-bace199897eeab37.js +0 -9
- package/dist/dashboard/out/_next/static/chunks/631-16b905e5920f9b59.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/648-acb2ff9f77cbfbd3.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/766-2aea80818f7eb0d8.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/83-26d2bde54616ee90.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/847-f1f467060f32afff.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/891-5cb1513eeb97a891.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/app/_not-found/page-60501fddbafba9dc.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/app/app/onboarding/page-9914652442f7e4fb.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/app/app/page-366fb7c078d4e9e0.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/app/cloud/link/page-fa1d5842aa90e8a6.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/app/complete-profile/page-dd64bbdf66b639cd.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/app/connect-repos/page-113060009ef35bc2.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/app/history/page-9965d2483011b846.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/app/layout-6b91e33784c20610.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/app/login/page-435eceb0073be027.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/app/metrics/page-1e37ef8e73940b40.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/app/page-8119d4246743574e.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/app/pricing/page-9db3ebdfa567a7c9.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/app/providers/page-ecb16ffd3b36262b.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/app/providers/setup/[provider]/page-4dbe33f0f7691b7c.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/app/signup/page-c7a0a28341365ae0.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/e868780c-48e5f147c90a3a41.js +0 -18
- package/dist/dashboard/out/_next/static/chunks/fd9d1056-609918ca7b6280bb.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/framework-f66176bb897dc684.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/main-311c3db74dcfadb7.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/main-app-fdbeb09028f57c9f.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/pages/_app-72b849fbd24ac258.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/pages/_error-7ba65e1336b92748.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/polyfills-42372ed130431b0a.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/webpack-1cdd8ed57114d5e1.js +0 -1
- package/dist/dashboard/out/_next/static/css/4034f236dd1a3178.css +0 -1
- package/dist/dashboard/out/_next/static/css/6892f8422896ef7a.css +0 -1
- package/dist/dashboard/out/alt-logos/agent-relay-logo-128.png +0 -0
- package/dist/dashboard/out/alt-logos/agent-relay-logo-256.png +0 -0
- package/dist/dashboard/out/alt-logos/agent-relay-logo-32.png +0 -0
- package/dist/dashboard/out/alt-logos/agent-relay-logo-512.png +0 -0
- package/dist/dashboard/out/alt-logos/agent-relay-logo-64.png +0 -0
- package/dist/dashboard/out/alt-logos/agent-relay-logo.svg +0 -45
- package/dist/dashboard/out/alt-logos/logo.svg +0 -38
- package/dist/dashboard/out/alt-logos/monogram-logo-128.png +0 -0
- package/dist/dashboard/out/alt-logos/monogram-logo-256.png +0 -0
- package/dist/dashboard/out/alt-logos/monogram-logo-32.png +0 -0
- package/dist/dashboard/out/alt-logos/monogram-logo-512.png +0 -0
- package/dist/dashboard/out/alt-logos/monogram-logo-64.png +0 -0
- package/dist/dashboard/out/alt-logos/monogram-logo.svg +0 -38
- package/dist/dashboard/out/app/onboarding.html +0 -1
- package/dist/dashboard/out/app/onboarding.txt +0 -7
- package/dist/dashboard/out/app.html +0 -1
- package/dist/dashboard/out/app.txt +0 -7
- package/dist/dashboard/out/apple-icon.png +0 -0
- package/dist/dashboard/out/cloud/link.html +0 -1
- package/dist/dashboard/out/cloud/link.txt +0 -7
- package/dist/dashboard/out/complete-profile.html +0 -5
- package/dist/dashboard/out/complete-profile.txt +0 -7
- package/dist/dashboard/out/connect-repos.html +0 -1
- package/dist/dashboard/out/connect-repos.txt +0 -7
- package/dist/dashboard/out/history.html +0 -1
- package/dist/dashboard/out/history.txt +0 -7
- package/dist/dashboard/out/index.html +0 -1
- package/dist/dashboard/out/index.txt +0 -7
- package/dist/dashboard/out/login.html +0 -5
- package/dist/dashboard/out/login.txt +0 -7
- package/dist/dashboard/out/metrics.html +0 -1
- package/dist/dashboard/out/metrics.txt +0 -7
- package/dist/dashboard/out/pricing.html +0 -13
- package/dist/dashboard/out/pricing.txt +0 -7
- package/dist/dashboard/out/providers/setup/claude.html +0 -1
- package/dist/dashboard/out/providers/setup/claude.txt +0 -8
- package/dist/dashboard/out/providers/setup/codex.html +0 -1
- package/dist/dashboard/out/providers/setup/codex.txt +0 -8
- package/dist/dashboard/out/providers/setup/cursor.html +0 -1
- package/dist/dashboard/out/providers/setup/cursor.txt +0 -8
- package/dist/dashboard/out/providers.html +0 -1
- package/dist/dashboard/out/providers.txt +0 -7
- package/dist/dashboard/out/signup.html +0 -6
- package/dist/dashboard/out/signup.txt +0 -7
- package/dist/src/dashboard-server/index.d.ts +0 -8
- package/dist/src/dashboard-server/index.js +0 -8
- package/packages/dashboard/README.md +0 -48
- package/packages/dashboard/dist/health-worker-manager.d.ts +0 -62
- package/packages/dashboard/dist/health-worker-manager.js +0 -144
- package/packages/dashboard/dist/health-worker.d.ts +0 -9
- package/packages/dashboard/dist/health-worker.js +0 -79
- package/packages/dashboard/dist/index.d.ts +0 -20
- package/packages/dashboard/dist/index.js +0 -19
- package/packages/dashboard/dist/metrics.d.ts +0 -105
- package/packages/dashboard/dist/metrics.js +0 -193
- package/packages/dashboard/dist/needs-attention.d.ts +0 -24
- package/packages/dashboard/dist/needs-attention.js +0 -78
- package/packages/dashboard/dist/server.d.ts +0 -25
- package/packages/dashboard/dist/server.js +0 -5270
- package/packages/dashboard/dist/start.d.ts +0 -6
- package/packages/dashboard/dist/start.js +0 -13
- package/packages/dashboard/dist/types/threading.d.ts +0 -8
- package/packages/dashboard/dist/types/threading.js +0 -2
- package/packages/dashboard/dist/user-bridge.d.ts +0 -154
- package/packages/dashboard/dist/user-bridge.js +0 -372
- package/packages/dashboard/package.json +0 -65
- package/packages/dashboard/ui/app/app/onboarding/page.tsx +0 -394
- package/packages/dashboard/ui/app/app/page.tsx +0 -667
- package/packages/dashboard/ui/app/apple-icon.png +0 -0
- package/packages/dashboard/ui/app/cloud/link/page.tsx +0 -464
- package/packages/dashboard/ui/app/complete-profile/page.tsx +0 -204
- package/packages/dashboard/ui/app/connect-repos/page.tsx +0 -410
- package/packages/dashboard/ui/app/favicon.png +0 -0
- package/packages/dashboard/ui/app/globals.css +0 -59
- package/packages/dashboard/ui/app/history/page.tsx +0 -658
- package/packages/dashboard/ui/app/layout.tsx +0 -25
- package/packages/dashboard/ui/app/login/page.tsx +0 -424
- package/packages/dashboard/ui/app/metrics/page.tsx +0 -751
- package/packages/dashboard/ui/app/page.tsx +0 -59
- package/packages/dashboard/ui/app/pricing/page.tsx +0 -7
- package/packages/dashboard/ui/app/providers/page.tsx +0 -193
- package/packages/dashboard/ui/app/providers/setup/[provider]/ProviderSetupClient.tsx +0 -148
- package/packages/dashboard/ui/app/providers/setup/[provider]/constants.ts +0 -35
- package/packages/dashboard/ui/app/providers/setup/[provider]/page.tsx +0 -42
- package/packages/dashboard/ui/app/signup/page.tsx +0 -533
- package/packages/dashboard/ui/index.ts +0 -49
- package/packages/dashboard/ui/landing/LandingPage.tsx +0 -713
- package/packages/dashboard/ui/landing/PricingPage.tsx +0 -559
- package/packages/dashboard/ui/landing/index.ts +0 -6
- package/packages/dashboard/ui/landing/styles.css +0 -2850
- package/packages/dashboard/ui/lib/agent-merge.ts +0 -35
- package/packages/dashboard/ui/lib/api.ts +0 -1155
- package/packages/dashboard/ui/lib/cloudApi.ts +0 -877
- package/packages/dashboard/ui/lib/colors.ts +0 -218
- package/packages/dashboard/ui/lib/hierarchy.ts +0 -242
- package/packages/dashboard/ui/lib/stuckDetection.ts +0 -142
- package/packages/dashboard/ui/next-env.d.ts +0 -5
- package/packages/dashboard/ui/next.config.js +0 -41
- package/packages/dashboard/ui/package-lock.json +0 -2882
- package/packages/dashboard/ui/package.json +0 -33
- package/packages/dashboard/ui/postcss.config.js +0 -5
- package/packages/dashboard/ui/react-components/ActivityFeed.tsx +0 -216
- package/packages/dashboard/ui/react-components/AddWorkspaceModal.tsx +0 -170
- package/packages/dashboard/ui/react-components/AgentCard.tsx +0 -587
- package/packages/dashboard/ui/react-components/AgentList.tsx +0 -411
- package/packages/dashboard/ui/react-components/AgentProfilePanel.tsx +0 -564
- package/packages/dashboard/ui/react-components/App.tsx +0 -3033
- package/packages/dashboard/ui/react-components/BillingPanel.tsx +0 -922
- package/packages/dashboard/ui/react-components/BillingResult.tsx +0 -447
- package/packages/dashboard/ui/react-components/BroadcastComposer.tsx +0 -690
- package/packages/dashboard/ui/react-components/ChannelAdminPanel.tsx +0 -773
- package/packages/dashboard/ui/react-components/ChannelBrowser.tsx +0 -385
- package/packages/dashboard/ui/react-components/ChannelChat.tsx +0 -261
- package/packages/dashboard/ui/react-components/ChannelSidebar.tsx +0 -399
- package/packages/dashboard/ui/react-components/CloudSessionProvider.tsx +0 -130
- package/packages/dashboard/ui/react-components/CommandPalette.tsx +0 -815
- package/packages/dashboard/ui/react-components/ConfirmationDialog.tsx +0 -133
- package/packages/dashboard/ui/react-components/ConversationHistory.tsx +0 -518
- package/packages/dashboard/ui/react-components/CoordinatorPanel.tsx +0 -944
- package/packages/dashboard/ui/react-components/DecisionQueue.tsx +0 -717
- package/packages/dashboard/ui/react-components/DirectMessageView.tsx +0 -164
- package/packages/dashboard/ui/react-components/FileAutocomplete.tsx +0 -368
- package/packages/dashboard/ui/react-components/FleetOverview.tsx +0 -278
- package/packages/dashboard/ui/react-components/LogViewer.tsx +0 -310
- package/packages/dashboard/ui/react-components/LogViewerPanel.tsx +0 -482
- package/packages/dashboard/ui/react-components/Logo.tsx +0 -284
- package/packages/dashboard/ui/react-components/MentionAutocomplete.tsx +0 -384
- package/packages/dashboard/ui/react-components/MessageComposer.tsx +0 -457
- package/packages/dashboard/ui/react-components/MessageList.tsx +0 -649
- package/packages/dashboard/ui/react-components/MessageSenderName.tsx +0 -91
- package/packages/dashboard/ui/react-components/MessageStatusIndicator.tsx +0 -142
- package/packages/dashboard/ui/react-components/NewConversationModal.tsx +0 -400
- package/packages/dashboard/ui/react-components/NotificationToast.tsx +0 -488
- package/packages/dashboard/ui/react-components/OnlineUsersIndicator.tsx +0 -164
- package/packages/dashboard/ui/react-components/Pagination.tsx +0 -124
- package/packages/dashboard/ui/react-components/PricingPlans.tsx +0 -386
- package/packages/dashboard/ui/react-components/ProjectList.tsx +0 -625
- package/packages/dashboard/ui/react-components/ProviderAuthFlow.tsx +0 -853
- package/packages/dashboard/ui/react-components/ProviderConnectionList.tsx +0 -378
- package/packages/dashboard/ui/react-components/ProvisioningProgress.tsx +0 -730
- package/packages/dashboard/ui/react-components/RepoAccessPanel.tsx +0 -549
- package/packages/dashboard/ui/react-components/ServerCard.tsx +0 -202
- package/packages/dashboard/ui/react-components/SessionExpiredModal.tsx +0 -128
- package/packages/dashboard/ui/react-components/SpawnModal.tsx +0 -804
- package/packages/dashboard/ui/react-components/TaskAssignmentUI.tsx +0 -375
- package/packages/dashboard/ui/react-components/TerminalProviderSetup.tsx +0 -608
- package/packages/dashboard/ui/react-components/ThemeProvider.tsx +0 -325
- package/packages/dashboard/ui/react-components/ThinkingIndicator.tsx +0 -231
- package/packages/dashboard/ui/react-components/ThreadList.tsx +0 -198
- package/packages/dashboard/ui/react-components/ThreadPanel.tsx +0 -346
- package/packages/dashboard/ui/react-components/TrajectoryViewer.tsx +0 -698
- package/packages/dashboard/ui/react-components/TypingIndicator.tsx +0 -69
- package/packages/dashboard/ui/react-components/UsageBanner.tsx +0 -231
- package/packages/dashboard/ui/react-components/UserProfilePanel.tsx +0 -233
- package/packages/dashboard/ui/react-components/WorkspaceContext.tsx +0 -107
- package/packages/dashboard/ui/react-components/WorkspaceSelector.tsx +0 -234
- package/packages/dashboard/ui/react-components/WorkspaceStatusIndicator.tsx +0 -370
- package/packages/dashboard/ui/react-components/XTermInteractive.tsx +0 -510
- package/packages/dashboard/ui/react-components/XTermLogViewer.tsx +0 -719
- package/packages/dashboard/ui/react-components/channels/ChannelDialogs.tsx +0 -1411
- package/packages/dashboard/ui/react-components/channels/ChannelHeader.tsx +0 -317
- package/packages/dashboard/ui/react-components/channels/ChannelMessageList.tsx +0 -463
- package/packages/dashboard/ui/react-components/channels/ChannelViewV1.tsx +0 -146
- package/packages/dashboard/ui/react-components/channels/MessageInput.tsx +0 -288
- package/packages/dashboard/ui/react-components/channels/SearchInput.tsx +0 -172
- package/packages/dashboard/ui/react-components/channels/SearchResults.tsx +0 -336
- package/packages/dashboard/ui/react-components/channels/api.ts +0 -697
- package/packages/dashboard/ui/react-components/channels/index.ts +0 -76
- package/packages/dashboard/ui/react-components/channels/mockApi.ts +0 -344
- package/packages/dashboard/ui/react-components/channels/types.ts +0 -566
- package/packages/dashboard/ui/react-components/hooks/index.ts +0 -57
- package/packages/dashboard/ui/react-components/hooks/useAgentLogs.ts +0 -394
- package/packages/dashboard/ui/react-components/hooks/useAgents.ts +0 -127
- package/packages/dashboard/ui/react-components/hooks/useBroadcastDedup.ts +0 -86
- package/packages/dashboard/ui/react-components/hooks/useChannelAdmin.ts +0 -329
- package/packages/dashboard/ui/react-components/hooks/useChannelBrowser.ts +0 -239
- package/packages/dashboard/ui/react-components/hooks/useChannelCommands.ts +0 -138
- package/packages/dashboard/ui/react-components/hooks/useChannels.ts +0 -328
- package/packages/dashboard/ui/react-components/hooks/useDebounce.ts +0 -29
- package/packages/dashboard/ui/react-components/hooks/useDirectMessage.ts +0 -141
- package/packages/dashboard/ui/react-components/hooks/useMessages.ts +0 -309
- package/packages/dashboard/ui/react-components/hooks/useOrchestrator.ts +0 -364
- package/packages/dashboard/ui/react-components/hooks/usePinnedAgents.ts +0 -140
- package/packages/dashboard/ui/react-components/hooks/usePresence.ts +0 -340
- package/packages/dashboard/ui/react-components/hooks/useRecentRepos.ts +0 -130
- package/packages/dashboard/ui/react-components/hooks/useSession.ts +0 -209
- package/packages/dashboard/ui/react-components/hooks/useTrajectory.ts +0 -265
- package/packages/dashboard/ui/react-components/hooks/useWebSocket.ts +0 -169
- package/packages/dashboard/ui/react-components/hooks/useWorkspaceMembers.ts +0 -120
- package/packages/dashboard/ui/react-components/hooks/useWorkspaceRepos.ts +0 -73
- package/packages/dashboard/ui/react-components/hooks/useWorkspaceStatus.ts +0 -237
- package/packages/dashboard/ui/react-components/index.ts +0 -81
- package/packages/dashboard/ui/react-components/layout/Header.tsx +0 -355
- package/packages/dashboard/ui/react-components/layout/RepoContextHeader.tsx +0 -361
- package/packages/dashboard/ui/react-components/layout/Sidebar.archive.test.tsx +0 -126
- package/packages/dashboard/ui/react-components/layout/Sidebar.test.tsx +0 -691
- package/packages/dashboard/ui/react-components/layout/Sidebar.tsx +0 -930
- package/packages/dashboard/ui/react-components/layout/index.ts +0 -7
- package/packages/dashboard/ui/react-components/settings/BillingSettingsPanel.tsx +0 -564
- package/packages/dashboard/ui/react-components/settings/SettingsPage.tsx +0 -544
- package/packages/dashboard/ui/react-components/settings/TeamSettingsPanel.tsx +0 -560
- package/packages/dashboard/ui/react-components/settings/WorkspaceSettingsPanel.tsx +0 -1386
- package/packages/dashboard/ui/react-components/settings/index.ts +0 -11
- package/packages/dashboard/ui/react-components/settings/types.ts +0 -53
- package/packages/dashboard/ui/react-components/utils/messageFormatting.tsx +0 -370
- package/packages/dashboard/ui/tailwind.config.js +0 -148
- package/packages/dashboard/ui/types/index.ts +0 -304
- package/packages/dashboard/ui/types/threading.ts +0 -7
- package/packages/dashboard/ui-dist/404.html +0 -1
- package/packages/dashboard/ui-dist/_next/static/7MZPqYkVGw3EGzVBkVmY9/_buildManifest.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/7MZPqYkVGw3EGzVBkVmY9/_ssgManifest.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/116-a883fca163f3a5bc.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/117-c8afed19e821a35d.js +0 -2
- package/packages/dashboard/ui-dist/_next/static/chunks/282-980c2eb8fff20123.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/320-a6304232cd0ee2ce.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/532-bace199897eeab37.js +0 -9
- package/packages/dashboard/ui-dist/_next/static/chunks/631-16b905e5920f9b59.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/648-acb2ff9f77cbfbd3.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/766-2aea80818f7eb0d8.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/83-26d2bde54616ee90.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/847-f1f467060f32afff.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/891-5cb1513eeb97a891.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/app/_not-found/page-60501fddbafba9dc.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/app/app/onboarding/page-9914652442f7e4fb.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/app/app/page-366fb7c078d4e9e0.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/app/cloud/link/page-fa1d5842aa90e8a6.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/app/complete-profile/page-dd64bbdf66b639cd.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/app/connect-repos/page-113060009ef35bc2.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/app/history/page-9965d2483011b846.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/app/layout-6b91e33784c20610.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/app/login/page-435eceb0073be027.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/app/metrics/page-1e37ef8e73940b40.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/app/page-8119d4246743574e.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/app/pricing/page-9db3ebdfa567a7c9.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/app/providers/page-ecb16ffd3b36262b.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/app/providers/setup/[provider]/page-4dbe33f0f7691b7c.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/app/signup/page-c7a0a28341365ae0.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/e868780c-48e5f147c90a3a41.js +0 -18
- package/packages/dashboard/ui-dist/_next/static/chunks/fd9d1056-609918ca7b6280bb.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/framework-f66176bb897dc684.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/main-311c3db74dcfadb7.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/main-app-fdbeb09028f57c9f.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/pages/_app-72b849fbd24ac258.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/pages/_error-7ba65e1336b92748.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/polyfills-42372ed130431b0a.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/chunks/webpack-1cdd8ed57114d5e1.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/css/4034f236dd1a3178.css +0 -1
- package/packages/dashboard/ui-dist/_next/static/css/6892f8422896ef7a.css +0 -1
- package/packages/dashboard/ui-dist/_next/static/iJ3Uiz3IrqUJL7IxKZHiV/_buildManifest.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/iJ3Uiz3IrqUJL7IxKZHiV/_ssgManifest.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/l-jd878zUJ_IlraqEWMZc/_buildManifest.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/l-jd878zUJ_IlraqEWMZc/_ssgManifest.js +0 -1
- package/packages/dashboard/ui-dist/alt-logos/agent-relay-logo-128.png +0 -0
- package/packages/dashboard/ui-dist/alt-logos/agent-relay-logo-256.png +0 -0
- package/packages/dashboard/ui-dist/alt-logos/agent-relay-logo-32.png +0 -0
- package/packages/dashboard/ui-dist/alt-logos/agent-relay-logo-512.png +0 -0
- package/packages/dashboard/ui-dist/alt-logos/agent-relay-logo-64.png +0 -0
- package/packages/dashboard/ui-dist/alt-logos/agent-relay-logo.svg +0 -45
- package/packages/dashboard/ui-dist/alt-logos/logo.svg +0 -38
- package/packages/dashboard/ui-dist/alt-logos/monogram-logo-128.png +0 -0
- package/packages/dashboard/ui-dist/alt-logos/monogram-logo-256.png +0 -0
- package/packages/dashboard/ui-dist/alt-logos/monogram-logo-32.png +0 -0
- package/packages/dashboard/ui-dist/alt-logos/monogram-logo-512.png +0 -0
- package/packages/dashboard/ui-dist/alt-logos/monogram-logo-64.png +0 -0
- package/packages/dashboard/ui-dist/alt-logos/monogram-logo.svg +0 -38
- package/packages/dashboard/ui-dist/app/onboarding.html +0 -1
- package/packages/dashboard/ui-dist/app/onboarding.txt +0 -7
- package/packages/dashboard/ui-dist/app.html +0 -1
- package/packages/dashboard/ui-dist/app.txt +0 -7
- package/packages/dashboard/ui-dist/apple-icon.png +0 -0
- package/packages/dashboard/ui-dist/cloud/link.html +0 -1
- package/packages/dashboard/ui-dist/cloud/link.txt +0 -7
- package/packages/dashboard/ui-dist/complete-profile.html +0 -5
- package/packages/dashboard/ui-dist/complete-profile.txt +0 -7
- package/packages/dashboard/ui-dist/connect-repos.html +0 -1
- package/packages/dashboard/ui-dist/connect-repos.txt +0 -7
- package/packages/dashboard/ui-dist/history.html +0 -1
- package/packages/dashboard/ui-dist/history.txt +0 -7
- package/packages/dashboard/ui-dist/index.html +0 -1
- package/packages/dashboard/ui-dist/index.txt +0 -7
- package/packages/dashboard/ui-dist/login.html +0 -5
- package/packages/dashboard/ui-dist/login.txt +0 -7
- package/packages/dashboard/ui-dist/metrics.html +0 -1
- package/packages/dashboard/ui-dist/metrics.txt +0 -7
- package/packages/dashboard/ui-dist/pricing.html +0 -13
- package/packages/dashboard/ui-dist/pricing.txt +0 -7
- package/packages/dashboard/ui-dist/providers/setup/claude.html +0 -1
- package/packages/dashboard/ui-dist/providers/setup/claude.txt +0 -8
- package/packages/dashboard/ui-dist/providers/setup/codex.html +0 -1
- package/packages/dashboard/ui-dist/providers/setup/codex.txt +0 -8
- package/packages/dashboard/ui-dist/providers/setup/cursor.html +0 -1
- package/packages/dashboard/ui-dist/providers/setup/cursor.txt +0 -8
- package/packages/dashboard/ui-dist/providers.html +0 -1
- package/packages/dashboard/ui-dist/providers.txt +0 -7
- package/packages/dashboard/ui-dist/signup.html +0 -6
- package/packages/dashboard/ui-dist/signup.txt +0 -7
- package/packages/dashboard-server/dist/health-worker-manager.d.ts +0 -62
- package/packages/dashboard-server/dist/health-worker-manager.js +0 -144
- package/packages/dashboard-server/dist/health-worker.d.ts +0 -9
- package/packages/dashboard-server/dist/health-worker.js +0 -79
- package/packages/dashboard-server/dist/index.d.ts +0 -18
- package/packages/dashboard-server/dist/index.js +0 -17
- package/packages/dashboard-server/dist/metrics.d.ts +0 -105
- package/packages/dashboard-server/dist/metrics.js +0 -193
- package/packages/dashboard-server/dist/needs-attention.d.ts +0 -24
- package/packages/dashboard-server/dist/needs-attention.js +0 -78
- package/packages/dashboard-server/dist/server.d.ts +0 -25
- package/packages/dashboard-server/dist/server.js +0 -5158
- package/packages/dashboard-server/dist/start.d.ts +0 -6
- package/packages/dashboard-server/dist/start.js +0 -13
- package/packages/dashboard-server/dist/types/threading.d.ts +0 -8
- package/packages/dashboard-server/dist/types/threading.js +0 -2
- package/packages/dashboard-server/dist/user-bridge.d.ts +0 -158
- package/packages/dashboard-server/dist/user-bridge.js +0 -390
- 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;
|