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.
- 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 +19 -1
- package/packages/daemon/package.json +12 -12
- package/packages/hooks/package.json +4 -4
- package/packages/mcp/package.json +2 -2
- package/packages/memory/package.json +2 -2
- package/packages/policy/package.json +2 -2
- package/packages/protocol/package.json +1 -1
- package/packages/resiliency/package.json +1 -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/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/91mkGYq3qbG8WHE6VytQ8/_buildManifest.js +0 -1
- package/dist/dashboard/out/_next/static/91mkGYq3qbG8WHE6VytQ8/_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/91mkGYq3qbG8WHE6VytQ8/_buildManifest.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/91mkGYq3qbG8WHE6VytQ8/_ssgManifest.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/T2rV14eEU5OweDeV29SvG/_buildManifest.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/T2rV14eEU5OweDeV29SvG/_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/l8L2OscDSR2vsMIlWcC48/_buildManifest.js +0 -1
- package/packages/dashboard/ui-dist/_next/static/l8L2OscDSR2vsMIlWcC48/_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,164 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* DirectMessageView Component
|
|
3
|
-
*
|
|
4
|
-
* Handles direct message conversations with humans and optional agent participants.
|
|
5
|
-
* Manages agent invitations, message deduplication, and group DM functionality.
|
|
6
|
-
*/
|
|
7
|
-
|
|
8
|
-
import React, { useMemo, useCallback } from 'react';
|
|
9
|
-
import type { Agent, Message } from '../types';
|
|
10
|
-
|
|
11
|
-
export interface DirectMessageViewProps {
|
|
12
|
-
/** The human user being DM'd */
|
|
13
|
-
currentHuman: { name: string; isHuman: boolean } | null;
|
|
14
|
-
/** All messages */
|
|
15
|
-
messages: Message[];
|
|
16
|
-
/** All agents */
|
|
17
|
-
agents: Agent[];
|
|
18
|
-
/** Currently selected agents for this DM */
|
|
19
|
-
selectedAgents: string[];
|
|
20
|
-
/** Agents removed from this DM */
|
|
21
|
-
removedAgents: string[];
|
|
22
|
-
/** Callback to toggle agent participation */
|
|
23
|
-
onAgentToggle: (agentName: string) => void;
|
|
24
|
-
/** Children to render (message list, composer, etc.) */
|
|
25
|
-
children: (props: {
|
|
26
|
-
visibleMessages: Message[];
|
|
27
|
-
participantAgents: string[];
|
|
28
|
-
}) => React.ReactNode;
|
|
29
|
-
}
|
|
30
|
-
|
|
31
|
-
export function DirectMessageView({
|
|
32
|
-
currentHuman,
|
|
33
|
-
messages,
|
|
34
|
-
agents,
|
|
35
|
-
selectedAgents,
|
|
36
|
-
removedAgents,
|
|
37
|
-
onAgentToggle,
|
|
38
|
-
children,
|
|
39
|
-
}: DirectMessageViewProps) {
|
|
40
|
-
const agentNameSet = useMemo(() => new Set(agents.map((a) => a.name)), [agents]);
|
|
41
|
-
|
|
42
|
-
// Derive agents participating in this conversation from message history
|
|
43
|
-
const dmParticipantAgents = useMemo(() => {
|
|
44
|
-
if (!currentHuman) return [];
|
|
45
|
-
const humanName = currentHuman.name;
|
|
46
|
-
const derived = new Set<string>();
|
|
47
|
-
|
|
48
|
-
for (const msg of messages) {
|
|
49
|
-
const { from, to } = msg;
|
|
50
|
-
if (!from || !to) continue;
|
|
51
|
-
if (from === humanName && agentNameSet.has(to)) derived.add(to);
|
|
52
|
-
if (to === humanName && agentNameSet.has(from)) derived.add(from);
|
|
53
|
-
if (selectedAgents.includes(from) && agentNameSet.has(to)) derived.add(to);
|
|
54
|
-
if (selectedAgents.includes(to) && agentNameSet.has(from)) derived.add(from);
|
|
55
|
-
}
|
|
56
|
-
|
|
57
|
-
const participants = new Set<string>([...selectedAgents, ...derived]);
|
|
58
|
-
removedAgents.forEach((a) => participants.delete(a));
|
|
59
|
-
return Array.from(participants);
|
|
60
|
-
}, [agentNameSet, currentHuman, messages, removedAgents, selectedAgents]);
|
|
61
|
-
|
|
62
|
-
// Filter messages for this DM conversation
|
|
63
|
-
const visibleMessages = useMemo(() => {
|
|
64
|
-
if (!currentHuman) return messages;
|
|
65
|
-
const participants = new Set<string>([currentHuman.name, ...dmParticipantAgents]);
|
|
66
|
-
return messages.filter(
|
|
67
|
-
(msg) => msg.from && msg.to && participants.has(msg.from) && participants.has(msg.to)
|
|
68
|
-
);
|
|
69
|
-
}, [currentHuman, dmParticipantAgents, messages]);
|
|
70
|
-
|
|
71
|
-
// Deduplicate DM messages (merge duplicates sent to multiple participants)
|
|
72
|
-
const dedupedVisibleMessages = useMemo(() => {
|
|
73
|
-
if (!currentHuman) return visibleMessages;
|
|
74
|
-
|
|
75
|
-
const normalizeBody = (content?: string) => (content ?? '').trim().replace(/\s+/g, ' ');
|
|
76
|
-
const rank = (msg: Message) => (msg.status === 'sending' ? 1 : 0);
|
|
77
|
-
const choose = (current: Message, incoming: Message) => {
|
|
78
|
-
const currentRank = rank(current);
|
|
79
|
-
const incomingRank = rank(incoming);
|
|
80
|
-
const currentTs = new Date(current.timestamp).getTime();
|
|
81
|
-
const incomingTs = new Date(incoming.timestamp).getTime();
|
|
82
|
-
if (incomingRank < currentRank) return incoming;
|
|
83
|
-
if (incomingRank > currentRank) return current;
|
|
84
|
-
return incomingTs >= currentTs ? incoming : current;
|
|
85
|
-
};
|
|
86
|
-
|
|
87
|
-
const sorted = [...visibleMessages].sort(
|
|
88
|
-
(a, b) => new Date(a.timestamp).getTime() - new Date(b.timestamp).getTime()
|
|
89
|
-
);
|
|
90
|
-
|
|
91
|
-
const byId = new Map<string, Message>();
|
|
92
|
-
const byFuzzy = new Map<string, Message>();
|
|
93
|
-
|
|
94
|
-
for (const msg of sorted) {
|
|
95
|
-
if (msg.id) {
|
|
96
|
-
const existing = byId.get(msg.id);
|
|
97
|
-
byId.set(msg.id, existing ? choose(existing, msg) : msg);
|
|
98
|
-
continue;
|
|
99
|
-
}
|
|
100
|
-
|
|
101
|
-
const sender = msg.from?.toLowerCase() ?? '';
|
|
102
|
-
const bucket = Math.floor(new Date(msg.timestamp).getTime() / 5000);
|
|
103
|
-
const key = `${sender}|${bucket}|${normalizeBody(msg.content)}`;
|
|
104
|
-
const existing = byFuzzy.get(key);
|
|
105
|
-
byFuzzy.set(key, existing ? choose(existing, msg) : msg);
|
|
106
|
-
}
|
|
107
|
-
|
|
108
|
-
const merged = [...byId.values(), ...byFuzzy.values()];
|
|
109
|
-
|
|
110
|
-
// Final pass: deduplicate by sender + recipient + content (no time bucket)
|
|
111
|
-
const finalDedup = new Map<string, Message>();
|
|
112
|
-
for (const msg of merged) {
|
|
113
|
-
const sender = msg.from?.toLowerCase() ?? '';
|
|
114
|
-
const recipient = msg.to?.toLowerCase() ?? '';
|
|
115
|
-
const key = `${sender}|${recipient}|${normalizeBody(msg.content)}`;
|
|
116
|
-
const existing = finalDedup.get(key);
|
|
117
|
-
finalDedup.set(key, existing ? choose(existing, msg) : msg);
|
|
118
|
-
}
|
|
119
|
-
|
|
120
|
-
return Array.from(finalDedup.values()).sort(
|
|
121
|
-
(a, b) => new Date(a.timestamp).getTime() - new Date(b.timestamp).getTime()
|
|
122
|
-
);
|
|
123
|
-
}, [currentHuman, visibleMessages]);
|
|
124
|
-
|
|
125
|
-
if (!currentHuman) return null;
|
|
126
|
-
|
|
127
|
-
return (
|
|
128
|
-
<>
|
|
129
|
-
{/* DM Header with Agent Invites */}
|
|
130
|
-
<div className="px-4 py-2 border-b border-border-subtle bg-bg-secondary flex flex-col gap-2 sticky top-0 z-10">
|
|
131
|
-
<div className="text-xs text-text-muted">
|
|
132
|
-
DM with <span className="font-semibold text-text-primary">{currentHuman.name}</span>. Invite agents:
|
|
133
|
-
</div>
|
|
134
|
-
<div className="flex flex-wrap gap-2">
|
|
135
|
-
{agents
|
|
136
|
-
.filter((a) => !a.isHuman)
|
|
137
|
-
.map((agent) => {
|
|
138
|
-
const isSelected = selectedAgents.includes(agent.name);
|
|
139
|
-
return (
|
|
140
|
-
<button
|
|
141
|
-
key={agent.name}
|
|
142
|
-
onClick={() => onAgentToggle(agent.name)}
|
|
143
|
-
className={`px-3 py-1.5 text-xs font-medium rounded-lg transition-colors ${
|
|
144
|
-
isSelected
|
|
145
|
-
? 'bg-accent-cyan text-bg-deep'
|
|
146
|
-
: 'bg-bg-tertiary text-text-secondary hover:bg-bg-tertiary/80'
|
|
147
|
-
}`}
|
|
148
|
-
title={agent.name}
|
|
149
|
-
>
|
|
150
|
-
{isSelected ? '✓ ' : ''}{agent.name}
|
|
151
|
-
</button>
|
|
152
|
-
);
|
|
153
|
-
})}
|
|
154
|
-
</div>
|
|
155
|
-
</div>
|
|
156
|
-
|
|
157
|
-
{/* Render children with deduped messages */}
|
|
158
|
-
{children({
|
|
159
|
-
visibleMessages: dedupedVisibleMessages,
|
|
160
|
-
participantAgents: dmParticipantAgents,
|
|
161
|
-
})}
|
|
162
|
-
</>
|
|
163
|
-
);
|
|
164
|
-
}
|
|
@@ -1,368 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* FileAutocomplete Component
|
|
3
|
-
*
|
|
4
|
-
* Provides @-file autocomplete for the message composer.
|
|
5
|
-
* Shows a dropdown list of files when typing @ followed by a path pattern.
|
|
6
|
-
* Triggered by @path/to/file or @filename patterns containing / or .
|
|
7
|
-
*/
|
|
8
|
-
|
|
9
|
-
import React, { useState, useCallback, useEffect, useRef, useMemo } from 'react';
|
|
10
|
-
import { getApiUrl } from '../lib/api';
|
|
11
|
-
|
|
12
|
-
export interface FileAutocompleteProps {
|
|
13
|
-
/** Current input value */
|
|
14
|
-
inputValue: string;
|
|
15
|
-
/** Cursor position in input */
|
|
16
|
-
cursorPosition: number;
|
|
17
|
-
/** Called when a file is selected */
|
|
18
|
-
onSelect: (filePath: string, newValue: string) => void;
|
|
19
|
-
/** Called when autocomplete should be hidden */
|
|
20
|
-
onClose: () => void;
|
|
21
|
-
/** Whether the autocomplete is visible */
|
|
22
|
-
isVisible: boolean;
|
|
23
|
-
/** API base URL for fetching files */
|
|
24
|
-
apiBase?: string;
|
|
25
|
-
}
|
|
26
|
-
|
|
27
|
-
interface FileOption {
|
|
28
|
-
path: string;
|
|
29
|
-
name: string;
|
|
30
|
-
isDirectory: boolean;
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
/**
|
|
34
|
-
* Check if the input has an @-file path being typed at the cursor position.
|
|
35
|
-
* Returns the query if it looks like a file path.
|
|
36
|
-
* Works for @ at any position in the text, not just the start.
|
|
37
|
-
*
|
|
38
|
-
* Trigger conditions (to avoid conflict with agent mentions):
|
|
39
|
-
* - Contains `/` (path separator) - e.g., @src/components
|
|
40
|
-
* - Contains `.` with more characters after (file extension) - e.g., @package.json
|
|
41
|
-
* - Starts with `./` or `../` (relative path) - e.g., @./src
|
|
42
|
-
*
|
|
43
|
-
* Does NOT trigger for:
|
|
44
|
-
* - Simple names like @Alice (could be an agent name)
|
|
45
|
-
* - Names with trailing dot like @config. (user still typing)
|
|
46
|
-
*/
|
|
47
|
-
export function getFileQuery(value: string, cursorPos: number): string | null {
|
|
48
|
-
// Search backwards from cursor to find @
|
|
49
|
-
const textBeforeCursor = value.substring(0, cursorPos);
|
|
50
|
-
|
|
51
|
-
// Find the last @ before cursor that starts a mention
|
|
52
|
-
// A mention starts after whitespace, at start of string, or after certain punctuation
|
|
53
|
-
const mentionMatch = textBeforeCursor.match(/(?:^|[\s(])@(\S*)$/);
|
|
54
|
-
if (mentionMatch) {
|
|
55
|
-
const query = mentionMatch[1];
|
|
56
|
-
|
|
57
|
-
// Trigger file autocomplete only for unambiguous file patterns:
|
|
58
|
-
// 1. Contains path separator: @src/components, @./file
|
|
59
|
-
if (query.includes('/')) {
|
|
60
|
-
return query;
|
|
61
|
-
}
|
|
62
|
-
|
|
63
|
-
// 2. Has file extension (dot followed by 1-10 chars): @file.ts, @package.json
|
|
64
|
-
// But not just a trailing dot (user still typing)
|
|
65
|
-
if (/\.[a-zA-Z0-9]{1,10}$/.test(query)) {
|
|
66
|
-
return query;
|
|
67
|
-
}
|
|
68
|
-
}
|
|
69
|
-
return null;
|
|
70
|
-
}
|
|
71
|
-
|
|
72
|
-
/**
|
|
73
|
-
* Complete a file path in the input value at the cursor position.
|
|
74
|
-
*/
|
|
75
|
-
export function completeFileInValue(
|
|
76
|
-
value: string,
|
|
77
|
-
filePath: string,
|
|
78
|
-
cursorPos: number
|
|
79
|
-
): string {
|
|
80
|
-
const textBeforeCursor = value.substring(0, cursorPos);
|
|
81
|
-
const textAfterCursor = value.substring(cursorPos);
|
|
82
|
-
|
|
83
|
-
// Find the @ and partial text before cursor
|
|
84
|
-
const mentionMatch = textBeforeCursor.match(/(?:^|[\s(])@(\S*)$/);
|
|
85
|
-
if (mentionMatch) {
|
|
86
|
-
// Calculate where the @ starts (accounting for whitespace/punctuation before it)
|
|
87
|
-
const matchStart = mentionMatch.index || 0;
|
|
88
|
-
const prefixChar = mentionMatch[0].charAt(0);
|
|
89
|
-
const atStart = prefixChar === '@' ? matchStart : matchStart + 1;
|
|
90
|
-
|
|
91
|
-
// Build the new value
|
|
92
|
-
const beforeMention = value.substring(0, atStart);
|
|
93
|
-
const completedFile = `@${filePath} `;
|
|
94
|
-
return beforeMention + completedFile + textAfterCursor;
|
|
95
|
-
}
|
|
96
|
-
return value;
|
|
97
|
-
}
|
|
98
|
-
|
|
99
|
-
// Cache for file search results
|
|
100
|
-
const fileCache = new Map<string, { files: FileOption[]; timestamp: number }>();
|
|
101
|
-
const CACHE_TTL_MS = 30000; // 30 seconds cache
|
|
102
|
-
|
|
103
|
-
export function FileAutocomplete({
|
|
104
|
-
inputValue,
|
|
105
|
-
cursorPosition,
|
|
106
|
-
onSelect,
|
|
107
|
-
onClose,
|
|
108
|
-
isVisible,
|
|
109
|
-
apiBase = '',
|
|
110
|
-
}: FileAutocompleteProps) {
|
|
111
|
-
const [selectedIndex, setSelectedIndex] = useState(0);
|
|
112
|
-
const [files, setFiles] = useState<FileOption[]>([]);
|
|
113
|
-
const [isLoading, setIsLoading] = useState(false);
|
|
114
|
-
const [error, setError] = useState<string | null>(null);
|
|
115
|
-
const listRef = useRef<HTMLDivElement>(null);
|
|
116
|
-
const abortControllerRef = useRef<AbortController | null>(null);
|
|
117
|
-
|
|
118
|
-
// Get the current file query
|
|
119
|
-
const query = useMemo(
|
|
120
|
-
() => getFileQuery(inputValue, cursorPosition),
|
|
121
|
-
[inputValue, cursorPosition]
|
|
122
|
-
);
|
|
123
|
-
|
|
124
|
-
// Fetch files when query changes
|
|
125
|
-
useEffect(() => {
|
|
126
|
-
if (!isVisible || query === null) {
|
|
127
|
-
setFiles([]);
|
|
128
|
-
return;
|
|
129
|
-
}
|
|
130
|
-
|
|
131
|
-
// Check cache first
|
|
132
|
-
const cacheKey = query || '__root__';
|
|
133
|
-
const cached = fileCache.get(cacheKey);
|
|
134
|
-
if (cached && Date.now() - cached.timestamp < CACHE_TTL_MS) {
|
|
135
|
-
setFiles(cached.files);
|
|
136
|
-
setError(null);
|
|
137
|
-
return;
|
|
138
|
-
}
|
|
139
|
-
|
|
140
|
-
// Cancel previous request
|
|
141
|
-
if (abortControllerRef.current) {
|
|
142
|
-
abortControllerRef.current.abort();
|
|
143
|
-
}
|
|
144
|
-
|
|
145
|
-
const controller = new AbortController();
|
|
146
|
-
abortControllerRef.current = controller;
|
|
147
|
-
|
|
148
|
-
const fetchFiles = async () => {
|
|
149
|
-
setIsLoading(true);
|
|
150
|
-
setError(null);
|
|
151
|
-
|
|
152
|
-
try {
|
|
153
|
-
const searchQuery = query || '';
|
|
154
|
-
// Use getApiUrl for cloud mode support (routes through workspace proxy)
|
|
155
|
-
const url = apiBase
|
|
156
|
-
? `${apiBase}/api/files?q=${encodeURIComponent(searchQuery)}&limit=15`
|
|
157
|
-
: getApiUrl(`/api/files?q=${encodeURIComponent(searchQuery)}&limit=15`);
|
|
158
|
-
const response = await fetch(url, { signal: controller.signal });
|
|
159
|
-
|
|
160
|
-
if (!response.ok) {
|
|
161
|
-
throw new Error('Failed to fetch files');
|
|
162
|
-
}
|
|
163
|
-
|
|
164
|
-
const data = await response.json();
|
|
165
|
-
const fileList: FileOption[] = (data.files || []).map((f: { path: string; name: string; isDirectory?: boolean }) => ({
|
|
166
|
-
path: f.path,
|
|
167
|
-
name: f.name,
|
|
168
|
-
isDirectory: f.isDirectory || false,
|
|
169
|
-
}));
|
|
170
|
-
|
|
171
|
-
// Update cache
|
|
172
|
-
fileCache.set(cacheKey, { files: fileList, timestamp: Date.now() });
|
|
173
|
-
setFiles(fileList);
|
|
174
|
-
} catch (err: unknown) {
|
|
175
|
-
if (err instanceof Error && err.name === 'AbortError') return;
|
|
176
|
-
setError('Failed to load files');
|
|
177
|
-
setFiles([]);
|
|
178
|
-
} finally {
|
|
179
|
-
setIsLoading(false);
|
|
180
|
-
}
|
|
181
|
-
};
|
|
182
|
-
|
|
183
|
-
// Debounce the search
|
|
184
|
-
const timeoutId = setTimeout(fetchFiles, 150);
|
|
185
|
-
return () => {
|
|
186
|
-
clearTimeout(timeoutId);
|
|
187
|
-
controller.abort();
|
|
188
|
-
};
|
|
189
|
-
}, [isVisible, query, apiBase]);
|
|
190
|
-
|
|
191
|
-
// Reset selection when files change
|
|
192
|
-
useEffect(() => {
|
|
193
|
-
setSelectedIndex(0);
|
|
194
|
-
}, [files.length]);
|
|
195
|
-
|
|
196
|
-
// Scroll selected item into view
|
|
197
|
-
useEffect(() => {
|
|
198
|
-
if (!listRef.current) return;
|
|
199
|
-
const selected = listRef.current.querySelector('[data-selected="true"]');
|
|
200
|
-
if (selected) {
|
|
201
|
-
selected.scrollIntoView({ block: 'nearest' });
|
|
202
|
-
}
|
|
203
|
-
}, [selectedIndex]);
|
|
204
|
-
|
|
205
|
-
// Handle keyboard navigation
|
|
206
|
-
const handleKeyDown = useCallback(
|
|
207
|
-
(e: KeyboardEvent) => {
|
|
208
|
-
if (!isVisible || files.length === 0) return;
|
|
209
|
-
|
|
210
|
-
switch (e.key) {
|
|
211
|
-
case 'ArrowDown':
|
|
212
|
-
e.preventDefault();
|
|
213
|
-
setSelectedIndex((prev) => (prev + 1) % files.length);
|
|
214
|
-
break;
|
|
215
|
-
case 'ArrowUp':
|
|
216
|
-
e.preventDefault();
|
|
217
|
-
setSelectedIndex((prev) => (prev - 1 + files.length) % files.length);
|
|
218
|
-
break;
|
|
219
|
-
case 'Enter':
|
|
220
|
-
case 'Tab':
|
|
221
|
-
e.preventDefault();
|
|
222
|
-
const selected = files[selectedIndex];
|
|
223
|
-
if (selected) {
|
|
224
|
-
const newValue = completeFileInValue(inputValue, selected.path, cursorPosition);
|
|
225
|
-
onSelect(selected.path, newValue);
|
|
226
|
-
}
|
|
227
|
-
break;
|
|
228
|
-
case 'Escape':
|
|
229
|
-
e.preventDefault();
|
|
230
|
-
onClose();
|
|
231
|
-
break;
|
|
232
|
-
}
|
|
233
|
-
},
|
|
234
|
-
[isVisible, files, selectedIndex, inputValue, cursorPosition, onSelect, onClose]
|
|
235
|
-
);
|
|
236
|
-
|
|
237
|
-
// Register keyboard listener
|
|
238
|
-
useEffect(() => {
|
|
239
|
-
if (isVisible) {
|
|
240
|
-
window.addEventListener('keydown', handleKeyDown);
|
|
241
|
-
return () => window.removeEventListener('keydown', handleKeyDown);
|
|
242
|
-
}
|
|
243
|
-
}, [isVisible, handleKeyDown]);
|
|
244
|
-
|
|
245
|
-
// Handle click on option
|
|
246
|
-
const handleClick = useCallback(
|
|
247
|
-
(file: FileOption) => {
|
|
248
|
-
const newValue = completeFileInValue(inputValue, file.path, cursorPosition);
|
|
249
|
-
onSelect(file.path, newValue);
|
|
250
|
-
},
|
|
251
|
-
[inputValue, cursorPosition, onSelect]
|
|
252
|
-
);
|
|
253
|
-
|
|
254
|
-
if (!isVisible || (files.length === 0 && !isLoading && !error)) {
|
|
255
|
-
return null;
|
|
256
|
-
}
|
|
257
|
-
|
|
258
|
-
return (
|
|
259
|
-
<div
|
|
260
|
-
className="absolute bottom-full left-0 right-0 max-h-[240px] overflow-y-auto bg-[#1a1d21] border border-white/10 rounded-lg shadow-[0_-4px_20px_rgba(0,0,0,0.4)] z-[100] mb-1"
|
|
261
|
-
ref={listRef}
|
|
262
|
-
>
|
|
263
|
-
{/* Header */}
|
|
264
|
-
<div className="px-3 py-1.5 text-[10px] uppercase tracking-wider text-[#8d8d8e] border-b border-white/5 flex items-center gap-2">
|
|
265
|
-
<svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2">
|
|
266
|
-
<path d="M13 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V9z" />
|
|
267
|
-
<polyline points="13 2 13 9 20 9" />
|
|
268
|
-
</svg>
|
|
269
|
-
Files {query && <span className="text-[#6b6b6c]">matching "{query}"</span>}
|
|
270
|
-
</div>
|
|
271
|
-
|
|
272
|
-
{/* Loading state */}
|
|
273
|
-
{isLoading && files.length === 0 && (
|
|
274
|
-
<div className="px-3 py-4 text-sm text-[#8d8d8e] text-center">
|
|
275
|
-
<svg className="animate-spin mx-auto mb-2" width="16" height="16" viewBox="0 0 24 24">
|
|
276
|
-
<circle cx="12" cy="12" r="10" stroke="currentColor" strokeWidth="2" fill="none" strokeDasharray="32" strokeLinecap="round" />
|
|
277
|
-
</svg>
|
|
278
|
-
Searching files...
|
|
279
|
-
</div>
|
|
280
|
-
)}
|
|
281
|
-
|
|
282
|
-
{/* Error state */}
|
|
283
|
-
{error && (
|
|
284
|
-
<div className="px-3 py-4 text-sm text-red-400 text-center">
|
|
285
|
-
{error}
|
|
286
|
-
</div>
|
|
287
|
-
)}
|
|
288
|
-
|
|
289
|
-
{/* File list */}
|
|
290
|
-
{files.map((file, index) => (
|
|
291
|
-
<div
|
|
292
|
-
key={file.path}
|
|
293
|
-
data-selected={index === selectedIndex}
|
|
294
|
-
className={`flex items-center gap-2.5 py-2 px-3 cursor-pointer transition-colors duration-150 ${
|
|
295
|
-
index === selectedIndex ? 'bg-white/[0.08]' : 'hover:bg-white/[0.08]'
|
|
296
|
-
}`}
|
|
297
|
-
onClick={() => handleClick(file)}
|
|
298
|
-
onMouseEnter={() => setSelectedIndex(index)}
|
|
299
|
-
>
|
|
300
|
-
{/* File/Folder icon */}
|
|
301
|
-
<div
|
|
302
|
-
className="w-7 h-7 rounded-md flex items-center justify-center text-[#8d8d8e]"
|
|
303
|
-
style={{ background: file.isDirectory ? 'rgba(251, 191, 36, 0.15)' : 'rgba(96, 165, 250, 0.15)' }}
|
|
304
|
-
>
|
|
305
|
-
{file.isDirectory ? (
|
|
306
|
-
<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="#fbbf24" strokeWidth="2">
|
|
307
|
-
<path d="M22 19a2 2 0 0 1-2 2H4a2 2 0 0 1-2-2V5a2 2 0 0 1 2-2h5l2 3h9a2 2 0 0 1 2 2z" />
|
|
308
|
-
</svg>
|
|
309
|
-
) : (
|
|
310
|
-
<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="#60a5fa" strokeWidth="2">
|
|
311
|
-
<path d="M13 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V9z" />
|
|
312
|
-
<polyline points="13 2 13 9 20 9" />
|
|
313
|
-
</svg>
|
|
314
|
-
)}
|
|
315
|
-
</div>
|
|
316
|
-
<div className="flex flex-col gap-0.5 min-w-0 flex-1">
|
|
317
|
-
<span className="text-sm font-medium text-[#d1d2d3] truncate">{file.name}</span>
|
|
318
|
-
<span className="text-xs text-[#6b6b6c] truncate">{file.path}</span>
|
|
319
|
-
</div>
|
|
320
|
-
</div>
|
|
321
|
-
))}
|
|
322
|
-
|
|
323
|
-
{/* Empty state (no results after search) */}
|
|
324
|
-
{!isLoading && !error && files.length === 0 && query && (
|
|
325
|
-
<div className="px-3 py-4 text-sm text-[#8d8d8e] text-center">
|
|
326
|
-
No files found matching "{query}"
|
|
327
|
-
</div>
|
|
328
|
-
)}
|
|
329
|
-
</div>
|
|
330
|
-
);
|
|
331
|
-
}
|
|
332
|
-
|
|
333
|
-
/**
|
|
334
|
-
* Hook to manage file autocomplete state
|
|
335
|
-
*/
|
|
336
|
-
export function useFileAutocomplete() {
|
|
337
|
-
const [isVisible, setIsVisible] = useState(false);
|
|
338
|
-
const [inputValue, setInputValue] = useState('');
|
|
339
|
-
const [cursorPosition, setCursorPosition] = useState(0);
|
|
340
|
-
|
|
341
|
-
const handleInputChange = useCallback((value: string, cursorPos: number) => {
|
|
342
|
-
setInputValue(value);
|
|
343
|
-
setCursorPosition(cursorPos);
|
|
344
|
-
|
|
345
|
-
// Show autocomplete if typing @file pattern
|
|
346
|
-
const query = getFileQuery(value, cursorPos);
|
|
347
|
-
setIsVisible(query !== null);
|
|
348
|
-
}, []);
|
|
349
|
-
|
|
350
|
-
const handleSelect = useCallback((filePath: string, newValue: string) => {
|
|
351
|
-
setInputValue(newValue);
|
|
352
|
-
setCursorPosition(newValue.indexOf(' ') + 1);
|
|
353
|
-
setIsVisible(false);
|
|
354
|
-
}, []);
|
|
355
|
-
|
|
356
|
-
const handleClose = useCallback(() => {
|
|
357
|
-
setIsVisible(false);
|
|
358
|
-
}, []);
|
|
359
|
-
|
|
360
|
-
return {
|
|
361
|
-
isVisible,
|
|
362
|
-
inputValue,
|
|
363
|
-
cursorPosition,
|
|
364
|
-
setInputValue: handleInputChange,
|
|
365
|
-
handleSelect,
|
|
366
|
-
handleClose,
|
|
367
|
-
};
|
|
368
|
-
}
|