@pennyfarthing/core 11.3.7 → 11.4.0
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/README.md +1 -1
- package/package.json +17 -16
- package/packages/core/dist/public/css/react.css +1 -1
- package/packages/core/dist/public/js/react/react.js +24 -24
- package/packages/core/src/public/App.tsx +356 -0
- package/packages/core/src/public/components/AgentLoadDialog.tsx +202 -0
- package/packages/core/src/public/components/AgentPopup.tsx +308 -0
- package/packages/core/src/public/components/ApprovalModal/ApprovalModal.css +35 -0
- package/packages/core/src/public/components/ApprovalModal/index.tsx +632 -0
- package/packages/core/src/public/components/BikeRackIndex.tsx +53 -0
- package/packages/core/src/public/components/BikeRackWorkspace.tsx +217 -0
- package/packages/core/src/public/components/CommandPalette.tsx +554 -0
- package/packages/core/src/public/components/ConfirmDialog.tsx +168 -0
- package/packages/core/src/public/components/ContextIndicator/ContextIndicator.css +85 -0
- package/packages/core/src/public/components/ContextIndicator/index.tsx +330 -0
- package/packages/core/src/public/components/ContextSparkline.tsx +56 -0
- package/packages/core/src/public/components/ControlBar.tsx +636 -0
- package/packages/core/src/public/components/DeadCodeDialog.tsx +169 -0
- package/packages/core/src/public/components/DiffViewer.tsx +585 -0
- package/packages/core/src/public/components/DockviewWorkspace.tsx +749 -0
- package/packages/core/src/public/components/Editor.tsx +630 -0
- package/packages/core/src/public/components/ErrorBoundary.tsx +67 -0
- package/packages/core/src/public/components/FileTree.tsx +379 -0
- package/packages/core/src/public/components/FontPicker/FontPicker.css +276 -0
- package/packages/core/src/public/components/FontPicker/index.tsx +430 -0
- package/packages/core/src/public/components/FullFileTree.tsx +237 -0
- package/packages/core/src/public/components/HealthGauge.tsx +181 -0
- package/packages/core/src/public/components/Message.tsx +225 -0
- package/packages/core/src/public/components/MessageList.tsx +98 -0
- package/packages/core/src/public/components/MessageView.tsx +400 -0
- package/packages/core/src/public/components/ModeSwitch/ModeSwitch.css +165 -0
- package/packages/core/src/public/components/ModeSwitch/index.tsx +372 -0
- package/packages/core/src/public/components/PersonaHeader.tsx +242 -0
- package/packages/core/src/public/components/ProjectInfoBar.tsx +45 -0
- package/packages/core/src/public/components/QuickActions.tsx +267 -0
- package/packages/core/src/public/components/SpanTimeline.tsx +352 -0
- package/packages/core/src/public/components/StandalonePanel.tsx +82 -0
- package/packages/core/src/public/components/StatsStrip.tsx +162 -0
- package/packages/core/src/public/components/StreamingContent.tsx +77 -0
- package/packages/core/src/public/components/SubagentSpan.tsx +180 -0
- package/packages/core/src/public/components/TandemPortrait.tsx +72 -0
- package/packages/core/src/public/components/ThemePalette/ThemePalette.css +179 -0
- package/packages/core/src/public/components/ThemePalette/index.tsx +326 -0
- package/packages/core/src/public/components/ToolCallBlock.tsx +252 -0
- package/packages/core/src/public/components/ToolStack.tsx +209 -0
- package/packages/core/src/public/components/ToolStatus.tsx +57 -0
- package/packages/core/src/public/components/dialogs/CodeMarkersDialog.tsx +169 -0
- package/packages/core/src/public/components/dialogs/ComplexityDialog.tsx +163 -0
- package/packages/core/src/public/components/dialogs/DependenciesDialog.tsx +120 -0
- package/packages/core/src/public/components/dialogs/HotspotsDialog.tsx +451 -0
- package/packages/core/src/public/components/dialogs/ToolDialog.tsx +43 -0
- package/packages/core/src/public/components/panel-registry.ts +13 -0
- package/packages/core/src/public/components/panels/ACPanel.tsx +93 -0
- package/packages/core/src/public/components/panels/AcceptanceCriteriaPanel.tsx +104 -0
- package/packages/core/src/public/components/panels/AuditLogPanel.tsx +489 -0
- package/packages/core/src/public/components/panels/BackgroundPanel.tsx +115 -0
- package/packages/core/src/public/components/panels/BikeLanePanel.tsx +214 -0
- package/packages/core/src/public/components/panels/DebugPanel.tsx +344 -0
- package/packages/core/src/public/components/panels/DiffView.tsx +109 -0
- package/packages/core/src/public/components/panels/DiffsPanel.tsx +56 -0
- package/packages/core/src/public/components/panels/GitPanel.tsx +260 -0
- package/packages/core/src/public/components/panels/HotspotsPanel.tsx +365 -0
- package/packages/core/src/public/components/panels/MessageFeed.tsx +39 -0
- package/packages/core/src/public/components/panels/MessagePanel.tsx +497 -0
- package/packages/core/src/public/components/panels/ProgressPanel.tsx +189 -0
- package/packages/core/src/public/components/panels/SettingsPanel.tsx +361 -0
- package/packages/core/src/public/components/panels/SprintPanel.tsx +723 -0
- package/packages/core/src/public/components/panels/TandemPanel.tsx +104 -0
- package/packages/core/src/public/components/panels/TaskTracker.tsx +48 -0
- package/packages/core/src/public/components/panels/TeamPanel.tsx +64 -0
- package/packages/core/src/public/components/panels/TeamRoster.tsx +67 -0
- package/packages/core/src/public/components/panels/TodoPanel.tsx +142 -0
- package/packages/core/src/public/components/panels/WorkflowPanel.tsx +224 -0
- package/packages/core/src/public/components/panels/index.ts +24 -0
- package/packages/core/src/public/components/ui/alert-dialog.tsx +139 -0
- package/packages/core/src/public/components/ui/badge.tsx +36 -0
- package/packages/core/src/public/components/ui/button.tsx +57 -0
- package/packages/core/src/public/components/ui/checkbox.tsx +28 -0
- package/packages/core/src/public/components/ui/collapsible.tsx +9 -0
- package/packages/core/src/public/components/ui/command.tsx +151 -0
- package/packages/core/src/public/components/ui/dialog.tsx +120 -0
- package/packages/core/src/public/components/ui/popover.tsx +31 -0
- package/packages/core/src/public/components/ui/progress.tsx +28 -0
- package/packages/core/src/public/components/ui/scroll-area.tsx +46 -0
- package/packages/core/src/public/components/ui/select.tsx +157 -0
- package/packages/core/src/public/components/ui/separator.tsx +29 -0
- package/packages/core/src/public/components/ui/skeleton.tsx +15 -0
- package/packages/core/src/public/components/ui/switch.tsx +27 -0
- package/packages/core/src/public/components/ui/toggle-group.tsx +59 -0
- package/packages/core/src/public/components/ui/toggle.tsx +43 -0
- package/packages/core/src/public/components/ui/tooltip.tsx +30 -0
- package/packages/core/src/public/contexts/ClaudeContext.tsx +311 -0
- package/packages/core/src/public/contexts/MessageQueueContext.tsx +143 -0
- package/packages/core/src/public/css/theme-browser.css +550 -0
- package/packages/core/src/public/css/theme-system.css +630 -0
- package/packages/core/src/public/hooks/index.ts +49 -0
- package/packages/core/src/public/hooks/useAgentLoad.ts +105 -0
- package/packages/core/src/public/hooks/useBackgroundTasks.ts +131 -0
- package/packages/core/src/public/hooks/useClaude.ts +234 -0
- package/packages/core/src/public/hooks/useCodeMarkers.ts +101 -0
- package/packages/core/src/public/hooks/useColorScheme.ts +42 -0
- package/packages/core/src/public/hooks/useCommandHistory.ts +99 -0
- package/packages/core/src/public/hooks/useComplexity.ts +80 -0
- package/packages/core/src/public/hooks/useDeadCode.ts +99 -0
- package/packages/core/src/public/hooks/useDependencies.ts +82 -0
- package/packages/core/src/public/hooks/useDiffs.ts +143 -0
- package/packages/core/src/public/hooks/useFileBrowser.ts +73 -0
- package/packages/core/src/public/hooks/useFocusPanel.ts +137 -0
- package/packages/core/src/public/hooks/useGitStatus.ts +233 -0
- package/packages/core/src/public/hooks/useHealthScore.ts +71 -0
- package/packages/core/src/public/hooks/useHotspots.ts +123 -0
- package/packages/core/src/public/hooks/useLayoutPersistence.ts +141 -0
- package/packages/core/src/public/hooks/useMarkdownParser.ts +36 -0
- package/packages/core/src/public/hooks/useMarkerActions.ts +234 -0
- package/packages/core/src/public/hooks/useMessageQueue.ts +380 -0
- package/packages/core/src/public/hooks/useMessageStream.ts +131 -0
- package/packages/core/src/public/hooks/usePersona.ts +112 -0
- package/packages/core/src/public/hooks/usePlanModeExit.ts +105 -0
- package/packages/core/src/public/hooks/useResponsiveLayout.ts +173 -0
- package/packages/core/src/public/hooks/useSprint.ts +157 -0
- package/packages/core/src/public/hooks/useStatsStrip.ts +204 -0
- package/packages/core/src/public/hooks/useStory.ts +135 -0
- package/packages/core/src/public/hooks/useSubagentHelper.ts +64 -0
- package/packages/core/src/public/hooks/useSyntaxHighlighter.ts +52 -0
- package/packages/core/src/public/hooks/useTabCompletion.ts +124 -0
- package/packages/core/src/public/hooks/useTandemObservations.ts +165 -0
- package/packages/core/src/public/hooks/useTeamMembers.ts +273 -0
- package/packages/core/src/public/hooks/useTodos.ts +93 -0
- package/packages/core/src/public/hooks/useUserAvatar.ts +54 -0
- package/packages/core/src/public/images/cyclist-dark.png +0 -0
- package/packages/core/src/public/images/cyclist-light.png +0 -0
- package/packages/core/src/public/index.html +14 -0
- package/packages/core/src/public/index.tsx +10 -0
- package/packages/core/src/public/lib/utils.ts +6 -0
- package/packages/core/src/public/styles/dockview-theme.css +376 -0
- package/packages/core/src/public/styles/tailwind.css +4454 -0
- package/packages/core/src/public/types/message.ts +51 -0
- package/packages/core/src/public/utils/avatar-service.ts +73 -0
- package/packages/core/src/public/utils/color-presets.ts +940 -0
- package/packages/core/src/public/utils/font-presets.ts +362 -0
- package/packages/core/src/public/utils/formatDuration.ts +14 -0
- package/packages/core/src/public/utils/markdown.ts +249 -0
- package/packages/core/src/public/utils/messageFilters.ts +128 -0
- package/packages/core/src/public/utils/slash-commands.ts +341 -0
- package/packages/core/src/public/utils/subagent-display.ts +146 -0
- package/packages/core/src/public/utils/syntax.ts +219 -0
- package/packages/core/src/public/utils/toolIntentSummarizer.ts +199 -0
- package/packages/core/src/public/utils/toolStackGrouper.ts +106 -0
- package/packages/core/src/public/utils/toolTypeColors.ts +45 -0
- package/pennyfarthing-dist/pf/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/__pycache__/context.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bc/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bc/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bc/__pycache__/focus.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bc/__pycache__/split.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bc/cli.py +0 -1
- package/pennyfarthing-dist/pf/bc/focus.py +0 -1
- package/pennyfarthing-dist/pf/bikerack/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bikerack/__pycache__/base_panel.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bikerack/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bikerack/__pycache__/git_panel.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bikerack/__pycache__/launcher.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bikerack/__pycache__/portrait_resolver.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bikerack/__pycache__/sprint_panel.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bikerack/__pycache__/story_detail_data.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bikerack/__pycache__/story_detail_screen.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bikerack/__pycache__/tui.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bikerack/base_panel.py +0 -1
- package/pennyfarthing-dist/pf/bikerack/events.py +1 -7
- package/pennyfarthing-dist/pf/bikerack/git_panel.py +273 -10
- package/pennyfarthing-dist/pf/bikerack/portrait_resolver.py +21 -0
- package/pennyfarthing-dist/pf/bikerack/sprint_panel.py +58 -1
- package/pennyfarthing-dist/pf/bikerack/tui.py +5 -20
- package/pennyfarthing-dist/pf/bmad/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bmad/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bmad/__pycache__/parser.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bmad/parser.py +15 -9
- package/pennyfarthing-dist/pf/codemarkers/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/codemarkers/__pycache__/analyze.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/codemarkers/__pycache__/models.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/common/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/common/__pycache__/config.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/common/__pycache__/output.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/common/__pycache__/pr_config.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/common/__pycache__/themes.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/common/pr_config.py +27 -2
- package/pennyfarthing-dist/pf/complexity/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/complexity/__pycache__/analyze.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/complexity/__pycache__/models.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/consultation/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/consultation/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/deadcode/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/deadcode/__pycache__/analyze.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/deadcode/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/deadcode/__pycache__/models.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/dependencies/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/dependencies/__pycache__/analyze.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/dependencies/__pycache__/models.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/epic/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/epic/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/git_group/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/git_group/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/handoff/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/handoff/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/handoff/__pycache__/complete_phase.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/handoff/__pycache__/marker.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/handoff/__pycache__/phase_check.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/handoff/__pycache__/resolve_gate.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/healthscore/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/healthscore/__pycache__/__main__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/healthscore/__pycache__/analyze.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/healthscore/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/healthscore/__pycache__/formatters.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/healthscore/__pycache__/models.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hooks/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hooks/__pycache__/bell_mode.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hooks/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hooks/__pycache__/context_breaker.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hooks/__pycache__/context_warning.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hooks/__pycache__/cyclist_pretooluse.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hooks/__pycache__/pre_edit_check.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hooks/__pycache__/reflector_check.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hooks/__pycache__/schema_validation.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hooks/__pycache__/session_start.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hooks/__pycache__/session_stop.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hooks/__pycache__/sprint_yaml_validation.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hooks/__pycache__/statusline.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hooks/cyclist-pretooluse-hook.sh +0 -0
- package/pennyfarthing-dist/pf/hotspots/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hotspots/__pycache__/analyze.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hotspots/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hotspots/__pycache__/models.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/jira/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/jira/__pycache__/bidirectional.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/jira/__pycache__/claim.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/jira/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/jira/__pycache__/client.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/jira/__pycache__/create.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/jira/__pycache__/epic.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/jira/__pycache__/operations.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/jira/__pycache__/reconcile.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/jira/__pycache__/story.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/jira/__pycache__/sync.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/launch/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/launch/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/prime/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/prime/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/prime/__pycache__/loader.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/prime/__pycache__/models.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/prime/__pycache__/persona.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/prime/__pycache__/session.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/prime/__pycache__/tiers.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/prime/__pycache__/workflow.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/session/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/session/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/settings/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/settings/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/settings/__pycache__/settings.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/settings/settings.py +44 -8
- package/pennyfarthing-dist/pf/sprint/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/sprint/__pycache__/archive.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/sprint/__pycache__/archive_epic.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/sprint/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/sprint/__pycache__/epic_add.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/sprint/__pycache__/epic_update.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/sprint/__pycache__/loader.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/sprint/__pycache__/status.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/sprint/__pycache__/story_add.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/sprint/__pycache__/story_finish.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/sprint/__pycache__/story_update.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/sprint/__pycache__/validate_cmd.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/sprint/__pycache__/validator.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/sprint/__pycache__/work.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/sprint/__pycache__/yaml_io.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/sprint/story_finish.py +14 -2
- package/pennyfarthing-dist/pf/sprint/validator.py +7 -7
- package/pennyfarthing-dist/pf/tests/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/tests/__pycache__/conftest.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing-dist/pf/tests/__pycache__/test_sprint_validator.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing-dist/pf/tests/test_sprint_validator.py +44 -0
- package/pennyfarthing-dist/pf/theme/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/theme/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/validate/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/validate/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/workflow/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/workflow/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/workflow/__pycache__/helpers.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/workflow/__pycache__/scale.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/workflow/__pycache__/state.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/scripts/core/agent-session.sh +0 -0
- package/pennyfarthing-dist/scripts/core/check-context.sh +0 -0
- package/pennyfarthing-dist/scripts/core/dialogue-manager.sh +0 -0
- package/pennyfarthing-dist/scripts/core/pf.sh +0 -0
- package/pennyfarthing-dist/scripts/core/phase-check-start.sh +0 -0
- package/pennyfarthing-dist/scripts/core/prime.sh +0 -0
- package/pennyfarthing-dist/scripts/cyclist/is-cyclist.sh +0 -0
- package/pennyfarthing-dist/scripts/git/create-feature-branches.sh +0 -0
- package/pennyfarthing-dist/scripts/git/git-status-all.sh +0 -0
- package/pennyfarthing-dist/scripts/git/install-git-hooks.sh +0 -0
- package/pennyfarthing-dist/scripts/git/release.sh +0 -0
- package/pennyfarthing-dist/scripts/git/worktree-manager.sh +0 -0
- package/pennyfarthing-dist/scripts/health/drift-detection.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/bell-mode-hook.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/context-circuit-breaker.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/context-warning.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/cyclist-pretooluse-hook.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/dispatcher-template.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/otel-auto-config.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/post-merge.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/pre-commit.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/pre-edit-check.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/pre-push.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/question-reflector-check.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/question_reflector_check.py +0 -0
- package/pennyfarthing-dist/scripts/hooks/schema-validation.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/session-start.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/session-stop.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/sprint-yaml-validation.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/welcome-hook.sh +0 -0
- package/pennyfarthing-dist/scripts/jira/create-jira-epic.sh +0 -0
- package/pennyfarthing-dist/scripts/jira/create-jira-story.sh +0 -0
- package/pennyfarthing-dist/scripts/jira/jira-claim-story.sh +0 -0
- package/pennyfarthing-dist/scripts/jira/jira-reconcile.sh +0 -0
- package/pennyfarthing-dist/scripts/jira/jira-sync-story.sh +0 -0
- package/pennyfarthing-dist/scripts/jira/sync-epic-jira.sh +0 -0
- package/pennyfarthing-dist/scripts/lib/background-tasks.sh +0 -0
- package/pennyfarthing-dist/scripts/lib/checkpoint.sh +0 -0
- package/pennyfarthing-dist/scripts/lib/common.sh +0 -0
- package/pennyfarthing-dist/scripts/lib/env.sh +0 -0
- package/pennyfarthing-dist/scripts/lib/file-lock.sh +0 -0
- package/pennyfarthing-dist/scripts/lib/find-root.sh +1 -1
- package/pennyfarthing-dist/scripts/lib/logging.sh +0 -0
- package/pennyfarthing-dist/scripts/lib/retry.sh +0 -0
- package/pennyfarthing-dist/scripts/lib/run-pf.sh +0 -0
- package/pennyfarthing-dist/scripts/maintenance/migrate-theme-schema.mjs +0 -0
- package/pennyfarthing-dist/scripts/maintenance/sidecar-health.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/add-short-names.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/add_short_names.py +0 -0
- package/pennyfarthing-dist/scripts/misc/backlog.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/check-status.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/find-related-work.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/generate-skill-docs.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/log-skill-usage.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/migrate-bmad-workflow.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/migrate_bmad_workflow.py +0 -1
- package/pennyfarthing-dist/scripts/misc/repo-scan.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/repo-utils.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/run-ci.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/run-timestamp.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/session-cleanup.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/skill-usage-report.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/statusline.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/uninstall.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/validate-subagent-frontmatter.sh +0 -0
- package/pennyfarthing-dist/scripts/portraits/generate-portraits.py +13 -13
- package/pennyfarthing-dist/scripts/portraits/generate-portraits.sh +0 -0
- package/pennyfarthing-dist/scripts/portraits/generate-tandem-portraits.sh +0 -0
- package/pennyfarthing-dist/scripts/story/create-story.sh +0 -0
- package/pennyfarthing-dist/scripts/story/size-story.sh +0 -0
- package/pennyfarthing-dist/scripts/story/story-template.sh +0 -0
- package/pennyfarthing-dist/scripts/tests/check.test.sh +0 -0
- package/pennyfarthing-dist/scripts/tests/dev-story-workflow-import.test.sh +0 -0
- package/pennyfarthing-dist/scripts/tests/epics-and-stories-workflow-import.test.sh +0 -0
- package/pennyfarthing-dist/scripts/tests/handoff-phase-update.test.sh +0 -0
- package/pennyfarthing-dist/scripts/tests/implementation-readiness-workflow-import.test.sh +0 -0
- package/pennyfarthing-dist/scripts/tests/migrate-bmad-workflow.test.sh +0 -0
- package/pennyfarthing-dist/scripts/tests/prd-workflow-import.test.sh +0 -0
- package/pennyfarthing-dist/scripts/tests/project-context-workflow-import.test.sh +0 -0
- package/pennyfarthing-dist/scripts/tests/test-character-voice.sh +0 -0
- package/pennyfarthing-dist/scripts/tests/test-drift-detection.sh +0 -0
- package/pennyfarthing-dist/scripts/tests/test-post-merge-hook.sh +0 -0
- package/pennyfarthing-dist/scripts/tests/test-session-checkpoint.sh +0 -0
- package/pennyfarthing-dist/scripts/tests/test-solo-command.sh +0 -0
- package/pennyfarthing-dist/scripts/tests/ux-design-workflow-import.test.sh +0 -0
- package/pennyfarthing-dist/scripts/theme/list-themes.sh +0 -0
- package/pennyfarthing-dist/scripts/validation/validate-agent-schema.sh +0 -0
- package/pennyfarthing-dist/scripts/workflow/check.py +4 -6
- package/pennyfarthing-dist/scripts/workflow/check.sh +0 -0
- package/pennyfarthing-dist/scripts/workflow/complete-step.py +2 -2
- package/pennyfarthing-dist/scripts/workflow/finish-story.sh +0 -0
- package/pennyfarthing-dist/scripts/workflow/fix-session-phase.sh +0 -0
- package/pennyfarthing-dist/scripts/workflow/get-workflow-type.py +0 -0
- package/pennyfarthing-dist/scripts/workflow/get-workflow-type.sh +0 -0
- package/pennyfarthing-dist/scripts/workflow/list-workflows.sh +0 -0
- package/pennyfarthing-dist/scripts/workflow/phase-owner.sh +0 -0
- package/pennyfarthing-dist/scripts/workflow/resume-workflow.sh +0 -0
- package/pennyfarthing-dist/scripts/workflow/show-workflow.sh +0 -0
- package/pennyfarthing-dist/scripts/workflow/start-workflow.sh +0 -0
- package/pennyfarthing-dist/scripts/workflow/workflow-status.sh +0 -0
- package/pennyfarthing-dist/skills/pf-story/scripts/create-story.sh +0 -0
- package/pennyfarthing-dist/skills/pf-story/scripts/size-story.sh +0 -0
- package/pennyfarthing-dist/skills/pf-story/scripts/story-template.sh +0 -0
- package/pennyfarthing-dist/skills/skill-registry.yaml +19 -0
- package/pennyfarthing-dist/workflows/release/steps/step-10-publish.md +41 -9
- package/pennyfarthing-dist/workflows/tdd-tandem.yaml +15 -2
- package/packages/core/dist/workflow/__test_context_watch__/.session/.tandem-turn-counter +0 -1
- package/packages/core/dist/workflow/__test_context_watch__/.session/95-6-session.md +0 -3
- package/packages/core/dist/workflow/__test_context_watch__/.session/95-6-tandem-architect.md +0 -6
- package/packages/core/dist/workflow/__test_file_watch__/.session/95-4-tandem-architect.md +0 -6
- package/packages/core/dist/workflow/__test_file_watch__/workdir/trigger.ts +0 -1
- package/packages/core/dist/workflow/__test_tool_watch__/.session/95-5-tandem-architect.md +0 -6
- package/packages/core/dist/workflow/__test_tool_watch__/.session/95-5-tandem-toolcalls.jsonl +0 -1
- package/pennyfarthing-dist/pf/bikerack/changed_panel.py +0 -201
- package/scripts/README.md +0 -41
|
@@ -0,0 +1,632 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* ApprovalModal Component
|
|
3
|
+
*
|
|
4
|
+
* Tool permission modal with command preview.
|
|
5
|
+
* Story MSSCI-12713 - ApprovalModal Component
|
|
6
|
+
*
|
|
7
|
+
* Features:
|
|
8
|
+
* - Command preview for different tool types
|
|
9
|
+
* - Keyboard-first interaction (Enter approve, Escape reject)
|
|
10
|
+
* - Red accent styling for destructive actions
|
|
11
|
+
* - "Always allow" checkbox for persistent permissions
|
|
12
|
+
* - Non-blocking overlay (click outside to dismiss)
|
|
13
|
+
* - Accessible with ARIA attributes
|
|
14
|
+
*
|
|
15
|
+
* Uses shadcn Dialog, Checkbox, and Button primitives.
|
|
16
|
+
*/
|
|
17
|
+
|
|
18
|
+
import React, { useState, useEffect, useCallback } from 'react';
|
|
19
|
+
import {
|
|
20
|
+
Dialog,
|
|
21
|
+
DialogContent,
|
|
22
|
+
DialogHeader,
|
|
23
|
+
DialogTitle,
|
|
24
|
+
DialogDescription,
|
|
25
|
+
DialogFooter,
|
|
26
|
+
} from '@/components/ui/dialog';
|
|
27
|
+
import { Checkbox } from '@/components/ui/checkbox';
|
|
28
|
+
import { Button } from '@/components/ui/button';
|
|
29
|
+
import { cn } from '@/lib/utils';
|
|
30
|
+
import './ApprovalModal.css';
|
|
31
|
+
|
|
32
|
+
// ============================================================================
|
|
33
|
+
// Constants - Test IDs
|
|
34
|
+
// ============================================================================
|
|
35
|
+
|
|
36
|
+
export const MODAL_TESTID = 'approval-modal';
|
|
37
|
+
export const OVERLAY_TESTID = 'approval-modal-overlay';
|
|
38
|
+
export const COMMAND_PREVIEW_TESTID = 'command-preview';
|
|
39
|
+
export const TOOL_NAME_TESTID = 'tool-name';
|
|
40
|
+
export const APPROVE_BUTTON_TESTID = 'approve-button';
|
|
41
|
+
export const REJECT_BUTTON_TESTID = 'reject-button';
|
|
42
|
+
export const ALWAYS_ALLOW_TESTID = 'always-allow-checkbox';
|
|
43
|
+
export const WARNING_TESTID = 'approval-modal-warning';
|
|
44
|
+
|
|
45
|
+
// ============================================================================
|
|
46
|
+
// Constants - Keyboard Shortcuts
|
|
47
|
+
// ============================================================================
|
|
48
|
+
|
|
49
|
+
export const KEYBOARD_SHORTCUTS = {
|
|
50
|
+
APPROVE: 'Enter',
|
|
51
|
+
REJECT: 'Escape',
|
|
52
|
+
} as const;
|
|
53
|
+
|
|
54
|
+
export const INITIAL_FOCUS_ELEMENT = 'approve-button';
|
|
55
|
+
|
|
56
|
+
// ============================================================================
|
|
57
|
+
// Constants - Grant Scopes
|
|
58
|
+
// ============================================================================
|
|
59
|
+
|
|
60
|
+
export const GRANT_SCOPES = {
|
|
61
|
+
ONCE: 'once',
|
|
62
|
+
SESSION: 'session',
|
|
63
|
+
ALWAYS: 'always',
|
|
64
|
+
} as const;
|
|
65
|
+
|
|
66
|
+
export type GrantScope = typeof GRANT_SCOPES[keyof typeof GRANT_SCOPES];
|
|
67
|
+
|
|
68
|
+
// ============================================================================
|
|
69
|
+
// Constants - Severity Styling
|
|
70
|
+
// ============================================================================
|
|
71
|
+
|
|
72
|
+
export const SEVERITY_CLASSNAMES = {
|
|
73
|
+
destructive: 'severity-destructive',
|
|
74
|
+
normal: 'severity-normal',
|
|
75
|
+
safe: 'severity-safe',
|
|
76
|
+
} as const;
|
|
77
|
+
|
|
78
|
+
export type ActionSeverity = keyof typeof SEVERITY_CLASSNAMES;
|
|
79
|
+
|
|
80
|
+
// ============================================================================
|
|
81
|
+
// Constants - Accessibility
|
|
82
|
+
// ============================================================================
|
|
83
|
+
|
|
84
|
+
export const MODAL_ROLE = 'dialog';
|
|
85
|
+
export const ARIA_MODAL = true;
|
|
86
|
+
export const TITLE_ID = 'approval-modal-title';
|
|
87
|
+
export const DESCRIPTION_ID = 'approval-modal-description';
|
|
88
|
+
export const MODAL_TITLE = 'Permission Required';
|
|
89
|
+
|
|
90
|
+
// ============================================================================
|
|
91
|
+
// Constants - Labels
|
|
92
|
+
// ============================================================================
|
|
93
|
+
|
|
94
|
+
export const ALWAYS_ALLOW_LABEL = 'Always allow this action';
|
|
95
|
+
export const ALWAYS_ALLOW_ARIA_LABEL = 'Check to always allow this type of action without asking';
|
|
96
|
+
|
|
97
|
+
// ============================================================================
|
|
98
|
+
// Constants - Behavior
|
|
99
|
+
// ============================================================================
|
|
100
|
+
|
|
101
|
+
export const IS_NON_BLOCKING = true;
|
|
102
|
+
export const COMPONENT_CLASSNAME = 'approval-modal';
|
|
103
|
+
|
|
104
|
+
// ============================================================================
|
|
105
|
+
// Types
|
|
106
|
+
// ============================================================================
|
|
107
|
+
|
|
108
|
+
export interface ToolInput {
|
|
109
|
+
command?: string;
|
|
110
|
+
file_path?: string;
|
|
111
|
+
old_string?: string;
|
|
112
|
+
new_string?: string;
|
|
113
|
+
content?: string;
|
|
114
|
+
url?: string;
|
|
115
|
+
prompt?: string;
|
|
116
|
+
[key: string]: unknown;
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
export interface ApprovalRequest {
|
|
120
|
+
toolId: string;
|
|
121
|
+
toolName: string;
|
|
122
|
+
input: ToolInput;
|
|
123
|
+
reason?: string;
|
|
124
|
+
severity?: ActionSeverity;
|
|
125
|
+
warning?: string;
|
|
126
|
+
agent?: string;
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
export interface ApprovalResponse {
|
|
130
|
+
toolId: string;
|
|
131
|
+
approved: boolean;
|
|
132
|
+
grantScope?: GrantScope;
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
export interface ApprovalModalProps {
|
|
136
|
+
/** Whether the modal is visible */
|
|
137
|
+
isOpen: boolean;
|
|
138
|
+
/** Tool name (Bash, Edit, Write, WebFetch, etc.) */
|
|
139
|
+
toolName: string;
|
|
140
|
+
/** Tool ID for response */
|
|
141
|
+
toolId: string;
|
|
142
|
+
/** Tool input parameters */
|
|
143
|
+
input: ToolInput;
|
|
144
|
+
/** Called when user approves */
|
|
145
|
+
onApprove: (grantScope: GrantScope) => void;
|
|
146
|
+
/** Called when user rejects */
|
|
147
|
+
onReject: () => void;
|
|
148
|
+
/** Called when user dismisses (clicks overlay) */
|
|
149
|
+
onDismiss?: () => void;
|
|
150
|
+
/** Additional CSS class name */
|
|
151
|
+
className?: string;
|
|
152
|
+
/** Server-provided severity classification (MSSCI-14323) */
|
|
153
|
+
severity?: ActionSeverity;
|
|
154
|
+
/** Server-provided warning text for destructive operations (MSSCI-14323) */
|
|
155
|
+
warning?: string;
|
|
156
|
+
/** Agent name requesting permission (MSSCI-14392) */
|
|
157
|
+
agent?: string;
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
interface UseApprovalModalResult {
|
|
161
|
+
request: ApprovalRequest | null;
|
|
162
|
+
isOpen: boolean;
|
|
163
|
+
show: (request: ApprovalRequest) => void;
|
|
164
|
+
hide: () => void;
|
|
165
|
+
approve: (grantScope: GrantScope) => void;
|
|
166
|
+
reject: () => void;
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
// ============================================================================
|
|
170
|
+
// Utility Functions
|
|
171
|
+
// ============================================================================
|
|
172
|
+
|
|
173
|
+
/**
|
|
174
|
+
* Format command preview based on tool type.
|
|
175
|
+
*/
|
|
176
|
+
export function formatCommandPreview(toolName: string, input: ToolInput): string {
|
|
177
|
+
switch (toolName) {
|
|
178
|
+
case 'Bash':
|
|
179
|
+
return input.command ?? '';
|
|
180
|
+
case 'Edit':
|
|
181
|
+
return input.file_path ?? '';
|
|
182
|
+
case 'Write':
|
|
183
|
+
return input.file_path ?? '';
|
|
184
|
+
case 'WebFetch':
|
|
185
|
+
return input.url ?? '';
|
|
186
|
+
default:
|
|
187
|
+
return JSON.stringify(input);
|
|
188
|
+
}
|
|
189
|
+
}
|
|
190
|
+
|
|
191
|
+
/**
|
|
192
|
+
* Get icon identifier for tool type.
|
|
193
|
+
*/
|
|
194
|
+
export function getToolIcon(toolName: string): string {
|
|
195
|
+
const icons: Record<string, string> = {
|
|
196
|
+
Bash: 'terminal',
|
|
197
|
+
Edit: 'edit',
|
|
198
|
+
Write: 'file-plus',
|
|
199
|
+
WebFetch: 'globe',
|
|
200
|
+
Read: 'file',
|
|
201
|
+
Grep: 'search',
|
|
202
|
+
Glob: 'folder-search',
|
|
203
|
+
};
|
|
204
|
+
return icons[toolName] ?? 'tool';
|
|
205
|
+
}
|
|
206
|
+
|
|
207
|
+
/**
|
|
208
|
+
* Classify action severity based on tool and input.
|
|
209
|
+
*/
|
|
210
|
+
export function classifyActionSeverity(toolName: string, input: ToolInput): ActionSeverity {
|
|
211
|
+
if (toolName === 'Bash' && input.command) {
|
|
212
|
+
const cmd = input.command;
|
|
213
|
+
|
|
214
|
+
// Destructive patterns
|
|
215
|
+
if (
|
|
216
|
+
/rm\s+(-[rf]+\s+)*/.test(cmd) ||
|
|
217
|
+
/git\s+(reset\s+--hard|push\s+--force|clean\s+-[fd])/.test(cmd) ||
|
|
218
|
+
/drop\s+database/i.test(cmd) ||
|
|
219
|
+
/truncate\s+table/i.test(cmd)
|
|
220
|
+
) {
|
|
221
|
+
return 'destructive';
|
|
222
|
+
}
|
|
223
|
+
|
|
224
|
+
// Safe patterns (read-only commands)
|
|
225
|
+
if (
|
|
226
|
+
/^(ls|cat|head|tail|grep|find|pwd|echo|which|type|file|stat|wc|diff)\b/.test(cmd) ||
|
|
227
|
+
/^git\s+(status|log|diff|show|branch|remote)\b/.test(cmd)
|
|
228
|
+
) {
|
|
229
|
+
return 'safe';
|
|
230
|
+
}
|
|
231
|
+
}
|
|
232
|
+
|
|
233
|
+
return 'normal';
|
|
234
|
+
}
|
|
235
|
+
|
|
236
|
+
/**
|
|
237
|
+
* Get grant scope based on checkbox state.
|
|
238
|
+
*/
|
|
239
|
+
export function getGrantScope(alwaysAllow: boolean): GrantScope {
|
|
240
|
+
return alwaysAllow ? GRANT_SCOPES.ALWAYS : GRANT_SCOPES.ONCE;
|
|
241
|
+
}
|
|
242
|
+
|
|
243
|
+
/**
|
|
244
|
+
* Create approval response object.
|
|
245
|
+
*/
|
|
246
|
+
export function createApprovalResponse(
|
|
247
|
+
toolId: string,
|
|
248
|
+
approved: boolean,
|
|
249
|
+
grantScope?: GrantScope
|
|
250
|
+
): ApprovalResponse {
|
|
251
|
+
const response: ApprovalResponse = { toolId, approved };
|
|
252
|
+
if (approved && grantScope) {
|
|
253
|
+
response.grantScope = grantScope;
|
|
254
|
+
}
|
|
255
|
+
return response;
|
|
256
|
+
}
|
|
257
|
+
|
|
258
|
+
// ============================================================================
|
|
259
|
+
// Event Handlers
|
|
260
|
+
// ============================================================================
|
|
261
|
+
|
|
262
|
+
/**
|
|
263
|
+
* Handle keyboard events for modal.
|
|
264
|
+
*/
|
|
265
|
+
export function handleKeyDown(
|
|
266
|
+
event: KeyboardEvent,
|
|
267
|
+
onApprove: () => void,
|
|
268
|
+
onReject: () => void
|
|
269
|
+
): void {
|
|
270
|
+
if (event.key === KEYBOARD_SHORTCUTS.APPROVE) {
|
|
271
|
+
event.preventDefault();
|
|
272
|
+
onApprove();
|
|
273
|
+
} else if (event.key === KEYBOARD_SHORTCUTS.REJECT) {
|
|
274
|
+
event.preventDefault();
|
|
275
|
+
onReject();
|
|
276
|
+
}
|
|
277
|
+
}
|
|
278
|
+
|
|
279
|
+
/**
|
|
280
|
+
* Handle overlay click for non-blocking dismiss.
|
|
281
|
+
*/
|
|
282
|
+
export function handleOverlayClick(
|
|
283
|
+
event: React.MouseEvent,
|
|
284
|
+
onDismiss: () => void
|
|
285
|
+
): void {
|
|
286
|
+
// Only dismiss if clicking directly on the overlay, not its children
|
|
287
|
+
if (event.target === event.currentTarget) {
|
|
288
|
+
onDismiss();
|
|
289
|
+
}
|
|
290
|
+
}
|
|
291
|
+
|
|
292
|
+
// ============================================================================
|
|
293
|
+
// Hooks
|
|
294
|
+
// ============================================================================
|
|
295
|
+
|
|
296
|
+
/**
|
|
297
|
+
* Hook for managing focus trap within modal.
|
|
298
|
+
* Note: Kept for backwards compatibility but shadcn Dialog handles focus trapping
|
|
299
|
+
* automatically via Radix UI primitives.
|
|
300
|
+
*/
|
|
301
|
+
export function useFocusTrap(isOpen: boolean, modalRef: React.RefObject<HTMLDivElement>): void {
|
|
302
|
+
useEffect(() => {
|
|
303
|
+
if (!isOpen || !modalRef.current) return;
|
|
304
|
+
|
|
305
|
+
const focusableElements = modalRef.current.querySelectorAll<HTMLElement>(
|
|
306
|
+
'button, [href], input, select, textarea, [tabindex]:not([tabindex="-1"])'
|
|
307
|
+
);
|
|
308
|
+
|
|
309
|
+
const firstElement = focusableElements[0];
|
|
310
|
+
const lastElement = focusableElements[focusableElements.length - 1];
|
|
311
|
+
|
|
312
|
+
// Focus first element (approve button)
|
|
313
|
+
firstElement?.focus();
|
|
314
|
+
|
|
315
|
+
const handleTabKey = (e: KeyboardEvent) => {
|
|
316
|
+
if (e.key !== 'Tab') return;
|
|
317
|
+
|
|
318
|
+
if (e.shiftKey) {
|
|
319
|
+
if (document.activeElement === firstElement) {
|
|
320
|
+
e.preventDefault();
|
|
321
|
+
lastElement?.focus();
|
|
322
|
+
}
|
|
323
|
+
} else {
|
|
324
|
+
if (document.activeElement === lastElement) {
|
|
325
|
+
e.preventDefault();
|
|
326
|
+
firstElement?.focus();
|
|
327
|
+
}
|
|
328
|
+
}
|
|
329
|
+
};
|
|
330
|
+
|
|
331
|
+
document.addEventListener('keydown', handleTabKey);
|
|
332
|
+
return () => document.removeEventListener('keydown', handleTabKey);
|
|
333
|
+
}, [isOpen, modalRef]);
|
|
334
|
+
}
|
|
335
|
+
|
|
336
|
+
/**
|
|
337
|
+
* Hook for managing approval modal state.
|
|
338
|
+
*/
|
|
339
|
+
export function useApprovalModal(): UseApprovalModalResult {
|
|
340
|
+
const [request, setRequest] = useState<ApprovalRequest | null>(null);
|
|
341
|
+
const [isOpen, setIsOpen] = useState(false);
|
|
342
|
+
|
|
343
|
+
const show = useCallback((req: ApprovalRequest) => {
|
|
344
|
+
setRequest(req);
|
|
345
|
+
setIsOpen(true);
|
|
346
|
+
}, []);
|
|
347
|
+
|
|
348
|
+
const hide = useCallback(() => {
|
|
349
|
+
setIsOpen(false);
|
|
350
|
+
setRequest(null);
|
|
351
|
+
}, []);
|
|
352
|
+
|
|
353
|
+
const approve = useCallback((grantScope: GrantScope) => {
|
|
354
|
+
hide();
|
|
355
|
+
}, [hide]);
|
|
356
|
+
|
|
357
|
+
const reject = useCallback(() => {
|
|
358
|
+
hide();
|
|
359
|
+
}, [hide]);
|
|
360
|
+
|
|
361
|
+
return { request, isOpen, show, hide, approve, reject };
|
|
362
|
+
}
|
|
363
|
+
|
|
364
|
+
// ============================================================================
|
|
365
|
+
// WebSocket Functions (Phase 1 Migration - MSSCI-12860)
|
|
366
|
+
// ============================================================================
|
|
367
|
+
|
|
368
|
+
/** WebSocket message format from /ws/hooks */
|
|
369
|
+
interface HookRequestMessage {
|
|
370
|
+
type: 'hook-request';
|
|
371
|
+
toolId: string;
|
|
372
|
+
toolName: string;
|
|
373
|
+
input: Record<string, unknown>;
|
|
374
|
+
severity?: 'safe' | 'normal' | 'destructive';
|
|
375
|
+
warning?: string;
|
|
376
|
+
agent?: string;
|
|
377
|
+
context?: {
|
|
378
|
+
percentage: number;
|
|
379
|
+
isHigh: boolean;
|
|
380
|
+
isCritical: boolean;
|
|
381
|
+
};
|
|
382
|
+
}
|
|
383
|
+
|
|
384
|
+
/** WebSocket response format to /ws/hooks */
|
|
385
|
+
interface HookResponseMessage {
|
|
386
|
+
type: 'hook-response';
|
|
387
|
+
toolId: string;
|
|
388
|
+
approved: boolean;
|
|
389
|
+
data?: {
|
|
390
|
+
grantScope?: GrantScope;
|
|
391
|
+
};
|
|
392
|
+
}
|
|
393
|
+
|
|
394
|
+
/** Global WebSocket instance for permission requests */
|
|
395
|
+
let hooksWs: WebSocket | null = null;
|
|
396
|
+
let reconnectTimeout: ReturnType<typeof setTimeout> | undefined;
|
|
397
|
+
|
|
398
|
+
/**
|
|
399
|
+
* Get or create WebSocket connection to /ws/hooks.
|
|
400
|
+
*/
|
|
401
|
+
function getHooksWebSocket(): WebSocket | null {
|
|
402
|
+
if (hooksWs && hooksWs.readyState === WebSocket.OPEN) {
|
|
403
|
+
return hooksWs;
|
|
404
|
+
}
|
|
405
|
+
return null;
|
|
406
|
+
}
|
|
407
|
+
|
|
408
|
+
/**
|
|
409
|
+
* Subscribe to permission requests via WebSocket.
|
|
410
|
+
*/
|
|
411
|
+
export function subscribeToPermissionRequests(
|
|
412
|
+
callback: (request: ApprovalRequest) => void
|
|
413
|
+
): () => void {
|
|
414
|
+
const protocol = window.location.protocol === 'https:' ? 'wss:' : 'ws:';
|
|
415
|
+
const wsUrl = `${protocol}//${window.location.host}/ws/hooks`;
|
|
416
|
+
|
|
417
|
+
const connect = () => {
|
|
418
|
+
try {
|
|
419
|
+
hooksWs = new WebSocket(wsUrl);
|
|
420
|
+
|
|
421
|
+
hooksWs.onopen = () => {
|
|
422
|
+
console.debug('[ApprovalModal] WebSocket connected to /ws/hooks');
|
|
423
|
+
};
|
|
424
|
+
|
|
425
|
+
hooksWs.onmessage = (event) => {
|
|
426
|
+
try {
|
|
427
|
+
const msg = JSON.parse(event.data) as HookRequestMessage;
|
|
428
|
+
if (msg.type === 'hook-request') {
|
|
429
|
+
callback({
|
|
430
|
+
toolId: msg.toolId,
|
|
431
|
+
toolName: msg.toolName,
|
|
432
|
+
input: msg.input as ToolInput,
|
|
433
|
+
severity: msg.severity as ActionSeverity | undefined,
|
|
434
|
+
warning: msg.warning,
|
|
435
|
+
agent: msg.agent,
|
|
436
|
+
});
|
|
437
|
+
}
|
|
438
|
+
} catch (err) {
|
|
439
|
+
console.error('[ApprovalModal] Failed to parse message:', err);
|
|
440
|
+
}
|
|
441
|
+
};
|
|
442
|
+
|
|
443
|
+
hooksWs.onclose = () => {
|
|
444
|
+
console.debug('[ApprovalModal] WebSocket closed, reconnecting...');
|
|
445
|
+
reconnectTimeout = setTimeout(connect, 2000);
|
|
446
|
+
};
|
|
447
|
+
|
|
448
|
+
hooksWs.onerror = (err) => {
|
|
449
|
+
console.error('[ApprovalModal] WebSocket error:', err);
|
|
450
|
+
};
|
|
451
|
+
} catch (err) {
|
|
452
|
+
console.error('[ApprovalModal] WebSocket init failed:', err);
|
|
453
|
+
}
|
|
454
|
+
};
|
|
455
|
+
|
|
456
|
+
connect();
|
|
457
|
+
|
|
458
|
+
return () => {
|
|
459
|
+
if (reconnectTimeout) {
|
|
460
|
+
clearTimeout(reconnectTimeout);
|
|
461
|
+
}
|
|
462
|
+
if (hooksWs) {
|
|
463
|
+
hooksWs.close();
|
|
464
|
+
hooksWs = null;
|
|
465
|
+
}
|
|
466
|
+
};
|
|
467
|
+
}
|
|
468
|
+
|
|
469
|
+
/**
|
|
470
|
+
* Send permission response via WebSocket.
|
|
471
|
+
*/
|
|
472
|
+
export function sendPermissionResponse(response: ApprovalResponse): void {
|
|
473
|
+
const ws = getHooksWebSocket();
|
|
474
|
+
if (ws) {
|
|
475
|
+
const msg: HookResponseMessage = {
|
|
476
|
+
type: 'hook-response',
|
|
477
|
+
toolId: response.toolId,
|
|
478
|
+
approved: response.approved,
|
|
479
|
+
data: response.grantScope ? { grantScope: response.grantScope } : undefined,
|
|
480
|
+
};
|
|
481
|
+
ws.send(JSON.stringify(msg));
|
|
482
|
+
} else {
|
|
483
|
+
console.warn('[ApprovalModal] WebSocket not connected, cannot send response');
|
|
484
|
+
}
|
|
485
|
+
}
|
|
486
|
+
|
|
487
|
+
// ============================================================================
|
|
488
|
+
// Component
|
|
489
|
+
// ============================================================================
|
|
490
|
+
|
|
491
|
+
/**
|
|
492
|
+
* ApprovalModal Component
|
|
493
|
+
*
|
|
494
|
+
* Modal dialog for tool permission approval.
|
|
495
|
+
* Uses shadcn Dialog, Checkbox, and Button primitives.
|
|
496
|
+
*/
|
|
497
|
+
export default function ApprovalModal({
|
|
498
|
+
isOpen,
|
|
499
|
+
toolName,
|
|
500
|
+
toolId,
|
|
501
|
+
input,
|
|
502
|
+
onApprove,
|
|
503
|
+
onReject,
|
|
504
|
+
onDismiss,
|
|
505
|
+
className = '',
|
|
506
|
+
severity: serverSeverity,
|
|
507
|
+
warning,
|
|
508
|
+
agent,
|
|
509
|
+
}: ApprovalModalProps): React.ReactElement {
|
|
510
|
+
const [alwaysAllow, setAlwaysAllow] = useState(false);
|
|
511
|
+
|
|
512
|
+
// Keyboard handler for Enter to approve
|
|
513
|
+
useEffect(() => {
|
|
514
|
+
if (!isOpen) return;
|
|
515
|
+
|
|
516
|
+
const handleKey = (e: KeyboardEvent) => {
|
|
517
|
+
if (e.key === KEYBOARD_SHORTCUTS.APPROVE) {
|
|
518
|
+
e.preventDefault();
|
|
519
|
+
onApprove(getGrantScope(alwaysAllow));
|
|
520
|
+
}
|
|
521
|
+
// Note: Escape is handled natively by Radix Dialog (triggers onOpenChange)
|
|
522
|
+
};
|
|
523
|
+
|
|
524
|
+
document.addEventListener('keydown', handleKey);
|
|
525
|
+
return () => document.removeEventListener('keydown', handleKey);
|
|
526
|
+
}, [isOpen, alwaysAllow, onApprove]);
|
|
527
|
+
|
|
528
|
+
const severity = serverSeverity ?? classifyActionSeverity(toolName, input);
|
|
529
|
+
const severityClass = SEVERITY_CLASSNAMES[severity];
|
|
530
|
+
const preview = formatCommandPreview(toolName, input);
|
|
531
|
+
const icon = getToolIcon(toolName);
|
|
532
|
+
|
|
533
|
+
const handleOpenChange = (open: boolean) => {
|
|
534
|
+
if (!open) {
|
|
535
|
+
// Dialog is closing (Escape key or overlay click)
|
|
536
|
+
if (onDismiss) {
|
|
537
|
+
onDismiss();
|
|
538
|
+
} else {
|
|
539
|
+
onReject();
|
|
540
|
+
}
|
|
541
|
+
}
|
|
542
|
+
};
|
|
543
|
+
|
|
544
|
+
return (
|
|
545
|
+
<Dialog open={isOpen} onOpenChange={handleOpenChange}>
|
|
546
|
+
<DialogContent
|
|
547
|
+
data-testid={MODAL_TESTID}
|
|
548
|
+
className={cn('max-w-[480px]', severityClass, className)}
|
|
549
|
+
onPointerDownOutside={(e) => {
|
|
550
|
+
// Non-blocking overlay: allow dismiss on outside click
|
|
551
|
+
if (!onDismiss) {
|
|
552
|
+
e.preventDefault();
|
|
553
|
+
}
|
|
554
|
+
}}
|
|
555
|
+
>
|
|
556
|
+
<DialogHeader>
|
|
557
|
+
<DialogTitle
|
|
558
|
+
className={cn(
|
|
559
|
+
severity === 'destructive' && 'text-destructive'
|
|
560
|
+
)}
|
|
561
|
+
>
|
|
562
|
+
{MODAL_TITLE}
|
|
563
|
+
</DialogTitle>
|
|
564
|
+
<DialogDescription asChild>
|
|
565
|
+
<div>
|
|
566
|
+
<div className="flex items-center gap-2 mb-3 text-sm text-muted-foreground">
|
|
567
|
+
<span className="approval-modal__icon" data-icon={icon} />
|
|
568
|
+
{agent && <span data-testid="agent-name" className="font-medium">{agent}</span>}
|
|
569
|
+
{agent && <span className="text-muted-foreground/50">/</span>}
|
|
570
|
+
<span data-testid={TOOL_NAME_TESTID}>{toolName}</span>
|
|
571
|
+
</div>
|
|
572
|
+
|
|
573
|
+
<pre
|
|
574
|
+
className={cn(
|
|
575
|
+
'approval-modal__preview',
|
|
576
|
+
severity === 'safe' && 'border-l-[3px] border-l-green-500',
|
|
577
|
+
severity === 'normal' && 'border-l-[3px] border-l-primary',
|
|
578
|
+
severity === 'destructive' && 'border-l-[3px] border-l-destructive bg-destructive/10'
|
|
579
|
+
)}
|
|
580
|
+
data-testid={COMMAND_PREVIEW_TESTID}
|
|
581
|
+
>
|
|
582
|
+
{preview}
|
|
583
|
+
</pre>
|
|
584
|
+
</div>
|
|
585
|
+
</DialogDescription>
|
|
586
|
+
</DialogHeader>
|
|
587
|
+
|
|
588
|
+
{warning && (
|
|
589
|
+
<div
|
|
590
|
+
data-testid={WARNING_TESTID}
|
|
591
|
+
className="text-sm text-destructive font-medium"
|
|
592
|
+
>
|
|
593
|
+
{warning}
|
|
594
|
+
</div>
|
|
595
|
+
)}
|
|
596
|
+
|
|
597
|
+
<div className="flex items-center gap-2 text-sm text-muted-foreground">
|
|
598
|
+
<Checkbox
|
|
599
|
+
id="always-allow"
|
|
600
|
+
data-testid={ALWAYS_ALLOW_TESTID}
|
|
601
|
+
checked={alwaysAllow}
|
|
602
|
+
onCheckedChange={(checked) => setAlwaysAllow(checked === true)}
|
|
603
|
+
aria-label={ALWAYS_ALLOW_ARIA_LABEL}
|
|
604
|
+
/>
|
|
605
|
+
<label
|
|
606
|
+
htmlFor="always-allow"
|
|
607
|
+
className="cursor-pointer select-none hover:text-foreground transition-colors"
|
|
608
|
+
>
|
|
609
|
+
{ALWAYS_ALLOW_LABEL}
|
|
610
|
+
</label>
|
|
611
|
+
</div>
|
|
612
|
+
|
|
613
|
+
<DialogFooter>
|
|
614
|
+
<Button
|
|
615
|
+
data-testid={REJECT_BUTTON_TESTID}
|
|
616
|
+
variant="destructive"
|
|
617
|
+
onClick={onReject}
|
|
618
|
+
>
|
|
619
|
+
Reject
|
|
620
|
+
</Button>
|
|
621
|
+
<Button
|
|
622
|
+
data-testid={APPROVE_BUTTON_TESTID}
|
|
623
|
+
variant="default"
|
|
624
|
+
onClick={() => onApprove(getGrantScope(alwaysAllow))}
|
|
625
|
+
>
|
|
626
|
+
Approve
|
|
627
|
+
</Button>
|
|
628
|
+
</DialogFooter>
|
|
629
|
+
</DialogContent>
|
|
630
|
+
</Dialog>
|
|
631
|
+
);
|
|
632
|
+
}
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* BikeRackIndex - Index page listing all available panels
|
|
3
|
+
*
|
|
4
|
+
* Story MSSCI-14822: BikeRackIndex panel listing page
|
|
5
|
+
* Epic: 101 (BikeRack Mode)
|
|
6
|
+
*
|
|
7
|
+
* Lists all 13 panels with links to standalone mode via ?panel=X.
|
|
8
|
+
* Styled with Tailwind dark mode, consistent with Cyclist.
|
|
9
|
+
*
|
|
10
|
+
* Rules:
|
|
11
|
+
* - No dockview-react imports (Rule 7)
|
|
12
|
+
* - No BikeRack-specific props (Rule 2)
|
|
13
|
+
* - URL-based detection only (Rule 10)
|
|
14
|
+
*/
|
|
15
|
+
|
|
16
|
+
import React from 'react';
|
|
17
|
+
|
|
18
|
+
const PANELS = [
|
|
19
|
+
{ id: 'sprint', label: 'Sprint', description: 'Story tracking and sprint progress' },
|
|
20
|
+
{ id: 'git', label: 'Git', description: 'Repository status and branches' },
|
|
21
|
+
{ id: 'diffs', label: 'Diffs', description: 'Code changes and diffs' },
|
|
22
|
+
{ id: 'todos', label: 'Todos', description: 'Acceptance criteria tracking' },
|
|
23
|
+
{ id: 'workflow', label: 'Workflow', description: 'Current workflow state' },
|
|
24
|
+
{ id: 'background', label: 'Background', description: 'Background tasks' },
|
|
25
|
+
{ id: 'audit', label: 'Audit', description: 'OTEL spans and logs' },
|
|
26
|
+
{ id: 'ac', label: 'AC', description: 'Acceptance criteria detail' },
|
|
27
|
+
{ id: 'debug', label: 'Debug', description: 'Debug information' },
|
|
28
|
+
{ id: 'bikelane', label: 'BikeLane', description: 'Workflow visualization' },
|
|
29
|
+
{ id: 'settings', label: 'Settings', description: 'Theme, fonts, and display preferences' },
|
|
30
|
+
{ id: 'progress', label: 'Progress', description: 'At-a-glance story dashboard' },
|
|
31
|
+
] as const;
|
|
32
|
+
|
|
33
|
+
export function BikeRackIndex(): React.ReactElement {
|
|
34
|
+
return (
|
|
35
|
+
<div className="bg-slate-900 min-h-screen text-gray-200 p-8">
|
|
36
|
+
<div className="max-w-4xl mx-auto">
|
|
37
|
+
<h1 className="text-3xl font-bold text-white mb-8">BikeRack</h1>
|
|
38
|
+
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4">
|
|
39
|
+
{PANELS.map((panel) => (
|
|
40
|
+
<a
|
|
41
|
+
key={panel.id}
|
|
42
|
+
href={`/?panel=${panel.id}`}
|
|
43
|
+
className="block border border-slate-700 rounded-lg p-4 hover:bg-slate-800 hover:text-cyan-400 transition-colors"
|
|
44
|
+
>
|
|
45
|
+
<div className="text-lg font-semibold text-white mb-1">{panel.label}</div>
|
|
46
|
+
<p className="text-sm text-gray-400">{panel.description}</p>
|
|
47
|
+
</a>
|
|
48
|
+
))}
|
|
49
|
+
</div>
|
|
50
|
+
</div>
|
|
51
|
+
</div>
|
|
52
|
+
);
|
|
53
|
+
}
|