@pennyfarthing/core 11.3.8 → 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 +4 -1
- 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/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/lib/find-root.sh +1 -1
- package/pennyfarthing-dist/scripts/misc/migrate_bmad_workflow.py +0 -1
- package/pennyfarthing-dist/scripts/portraits/generate-portraits.py +13 -13
- package/pennyfarthing-dist/scripts/workflow/check.py +4 -6
- package/pennyfarthing-dist/scripts/workflow/complete-step.py +2 -2
- package/pennyfarthing-dist/skills/skill-registry.yaml +19 -0
- 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
|
@@ -0,0 +1,93 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* ACPanel - Acceptance criteria checklist panel
|
|
3
|
+
*
|
|
4
|
+
* Extracted from ProgressPanel as part of MSSCI-14188.
|
|
5
|
+
* Shows acceptance criteria with completion status and progress bar.
|
|
6
|
+
*
|
|
7
|
+
* Story: MSSCI-14188 - Split Progress panel into Workflow, AC, and Todo panels
|
|
8
|
+
* Epic: epic-76 (Dockview Panel Migration)
|
|
9
|
+
*/
|
|
10
|
+
|
|
11
|
+
import React from 'react';
|
|
12
|
+
import { Skeleton } from '@/components/ui/skeleton';
|
|
13
|
+
import { useStory } from '../../hooks/useStory';
|
|
14
|
+
import type { CriteriaItem } from '../../../story-parser.js';
|
|
15
|
+
|
|
16
|
+
// =============================================================================
|
|
17
|
+
// Criteria Item Component
|
|
18
|
+
// =============================================================================
|
|
19
|
+
|
|
20
|
+
function CriteriaItemView({ item }: { item: CriteriaItem }): React.ReactElement {
|
|
21
|
+
const statusClass = item.completed ? 'ac-item ac-done' : 'ac-item';
|
|
22
|
+
const icon = item.completed ? '\u2713' : '\u25CB';
|
|
23
|
+
|
|
24
|
+
return (
|
|
25
|
+
<div className={statusClass}>
|
|
26
|
+
<span className="ac-icon">{icon}</span>
|
|
27
|
+
<span className="ac-text">{item.text}</span>
|
|
28
|
+
</div>
|
|
29
|
+
);
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
// =============================================================================
|
|
33
|
+
// ACPanel Component
|
|
34
|
+
// =============================================================================
|
|
35
|
+
|
|
36
|
+
export function ACPanel(): React.ReactElement {
|
|
37
|
+
const { story, isLoading, error } = useStory();
|
|
38
|
+
|
|
39
|
+
if (isLoading) {
|
|
40
|
+
return (
|
|
41
|
+
<div className="ac-panel loading" data-testid="ac-panel">
|
|
42
|
+
<div className="space-y-2 p-2">
|
|
43
|
+
<Skeleton className="h-3 w-full" />
|
|
44
|
+
<Skeleton className="h-4 w-3/4" />
|
|
45
|
+
<Skeleton className="h-4 w-5/6" />
|
|
46
|
+
<Skeleton className="h-4 w-2/3" />
|
|
47
|
+
</div>
|
|
48
|
+
</div>
|
|
49
|
+
);
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
if (error) {
|
|
53
|
+
return (
|
|
54
|
+
<div className="ac-panel error" data-testid="ac-panel">
|
|
55
|
+
<div className="error-message">{error.message}</div>
|
|
56
|
+
</div>
|
|
57
|
+
);
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
const criteria = story?.criteria ?? null;
|
|
61
|
+
|
|
62
|
+
if (!criteria || criteria.length === 0) {
|
|
63
|
+
return (
|
|
64
|
+
<div className="ac-panel" data-testid="ac-panel">
|
|
65
|
+
<div className="placeholder">No acceptance criteria</div>
|
|
66
|
+
</div>
|
|
67
|
+
);
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
const completedCount = criteria.filter(c => c.completed).length;
|
|
71
|
+
const totalCount = criteria.length;
|
|
72
|
+
|
|
73
|
+
return (
|
|
74
|
+
<div className="ac-panel" data-testid="ac-panel">
|
|
75
|
+
<div className="ac-content">
|
|
76
|
+
<span className="progress-text">{completedCount}/{totalCount}</span>
|
|
77
|
+
<div className="progress-bar-container">
|
|
78
|
+
<div
|
|
79
|
+
className="progress-bar"
|
|
80
|
+
style={{ width: `${(completedCount / totalCount) * 100}%` }}
|
|
81
|
+
/>
|
|
82
|
+
</div>
|
|
83
|
+
<div className="ac-list">
|
|
84
|
+
{criteria.map((item, index) => (
|
|
85
|
+
<CriteriaItemView key={index} item={item} />
|
|
86
|
+
))}
|
|
87
|
+
</div>
|
|
88
|
+
</div>
|
|
89
|
+
</div>
|
|
90
|
+
);
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
export default ACPanel;
|
|
@@ -0,0 +1,104 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* AcceptanceCriteriaPanel - Display acceptance criteria checklist
|
|
3
|
+
*
|
|
4
|
+
* Story MSSCI-12849 - Missing AC & BikeLane panels in Progress tab
|
|
5
|
+
*
|
|
6
|
+
* Reference: Deleted vanilla JS in commit 9aea4f371
|
|
7
|
+
* - js/sidebar/acceptance-criteria.js
|
|
8
|
+
*/
|
|
9
|
+
|
|
10
|
+
import React from 'react';
|
|
11
|
+
import { Skeleton } from '@/components/ui/skeleton';
|
|
12
|
+
import type { CriteriaItem } from '../../../story-parser.js';
|
|
13
|
+
import { useStory } from '../../hooks/useStory.js';
|
|
14
|
+
|
|
15
|
+
export interface AcceptanceCriteriaPanelProps {
|
|
16
|
+
criteria: CriteriaItem[] | null;
|
|
17
|
+
collapsed?: boolean;
|
|
18
|
+
onToggle?: () => void;
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
/**
|
|
22
|
+
* Individual acceptance criteria item
|
|
23
|
+
*/
|
|
24
|
+
function CriteriaItemView({ item }: { item: CriteriaItem }): React.ReactElement {
|
|
25
|
+
const statusClass = `todo-item ${item.completed ? 'todo-completed' : ''}`;
|
|
26
|
+
return (
|
|
27
|
+
<div className={statusClass}>
|
|
28
|
+
<span className="todo-status">{item.completed ? '\u2713' : '\u25CB'}</span>
|
|
29
|
+
<span className="todo-subject">{item.text}</span>
|
|
30
|
+
</div>
|
|
31
|
+
);
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
/**
|
|
35
|
+
* AcceptanceCriteriaPanel - Displays acceptance criteria checklist with progress
|
|
36
|
+
*/
|
|
37
|
+
export function AcceptanceCriteriaPanel({
|
|
38
|
+
criteria,
|
|
39
|
+
}: AcceptanceCriteriaPanelProps): React.ReactElement {
|
|
40
|
+
// Handle empty state
|
|
41
|
+
if (!criteria || criteria.length === 0) {
|
|
42
|
+
return (
|
|
43
|
+
<div className="todo-panel" data-testid="ac-panel">
|
|
44
|
+
<div className="placeholder">No acceptance criteria</div>
|
|
45
|
+
</div>
|
|
46
|
+
);
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
// Calculate progress
|
|
50
|
+
const completedCount = criteria.filter(c => c.completed).length;
|
|
51
|
+
const totalCount = criteria.length;
|
|
52
|
+
const progressPercent = totalCount > 0 ? Math.round((completedCount / totalCount) * 100) : 0;
|
|
53
|
+
|
|
54
|
+
return (
|
|
55
|
+
<div className="todo-panel" data-testid="ac-panel">
|
|
56
|
+
<div className="progress-bar-container">
|
|
57
|
+
<div
|
|
58
|
+
className="progress-bar"
|
|
59
|
+
style={{ width: `${progressPercent}%` }}
|
|
60
|
+
/>
|
|
61
|
+
<span className="progress-text">{completedCount}/{totalCount}</span>
|
|
62
|
+
</div>
|
|
63
|
+
<div className="todo-section">
|
|
64
|
+
{criteria.map((item, index) => (
|
|
65
|
+
<CriteriaItemView key={index} item={item} />
|
|
66
|
+
))}
|
|
67
|
+
</div>
|
|
68
|
+
</div>
|
|
69
|
+
);
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
/**
|
|
73
|
+
* ConnectedAcceptanceCriteriaPanel - Self-contained panel that fetches its own data
|
|
74
|
+
*
|
|
75
|
+
* Used by DockingWorkspace via registerPanelComponent
|
|
76
|
+
*/
|
|
77
|
+
export function ConnectedAcceptanceCriteriaPanel(): React.ReactElement {
|
|
78
|
+
const { story, isLoading, error } = useStory();
|
|
79
|
+
|
|
80
|
+
if (isLoading) {
|
|
81
|
+
return (
|
|
82
|
+
<div className="todo-panel loading" data-testid="ac-panel">
|
|
83
|
+
<div className="space-y-2 p-2">
|
|
84
|
+
<Skeleton className="h-3 w-full" />
|
|
85
|
+
<Skeleton className="h-4 w-3/4" />
|
|
86
|
+
<Skeleton className="h-4 w-5/6" />
|
|
87
|
+
<Skeleton className="h-4 w-2/3" />
|
|
88
|
+
</div>
|
|
89
|
+
</div>
|
|
90
|
+
);
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
if (error) {
|
|
94
|
+
return (
|
|
95
|
+
<div className="todo-panel error" data-testid="ac-panel">
|
|
96
|
+
<div className="error-message">{error.message}</div>
|
|
97
|
+
</div>
|
|
98
|
+
);
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
return <AcceptanceCriteriaPanel criteria={story?.criteria ?? null} />;
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
export default AcceptanceCriteriaPanel;
|
|
@@ -0,0 +1,489 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* AuditLogPanel - Tool execution audit log viewer
|
|
3
|
+
*
|
|
4
|
+
* Features:
|
|
5
|
+
* - View tool events with filtering by type and status
|
|
6
|
+
* - Real-time updates via WebSocket
|
|
7
|
+
* - Export as JSON or CSV
|
|
8
|
+
* - Statistics display
|
|
9
|
+
*/
|
|
10
|
+
|
|
11
|
+
import React, { useState, useEffect, useCallback, useRef } from 'react';
|
|
12
|
+
import { Button } from '@/components/ui/button';
|
|
13
|
+
import { Tooltip, TooltipContent, TooltipProvider, TooltipTrigger } from '@/components/ui/tooltip';
|
|
14
|
+
import { Separator } from '@/components/ui/separator';
|
|
15
|
+
import { Skeleton } from '@/components/ui/skeleton';
|
|
16
|
+
|
|
17
|
+
// =============================================================================
|
|
18
|
+
// Types
|
|
19
|
+
// =============================================================================
|
|
20
|
+
|
|
21
|
+
interface DiffSummary {
|
|
22
|
+
added: number;
|
|
23
|
+
removed: number;
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
interface OutputSummary {
|
|
27
|
+
firstLines: string[];
|
|
28
|
+
lastLines: string[];
|
|
29
|
+
totalLines: number;
|
|
30
|
+
truncated: boolean;
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
interface ToolEvent {
|
|
34
|
+
toolName: string;
|
|
35
|
+
input?: string;
|
|
36
|
+
output?: string;
|
|
37
|
+
durationMs?: number;
|
|
38
|
+
success: boolean;
|
|
39
|
+
error?: string;
|
|
40
|
+
timestamp: number;
|
|
41
|
+
// File enrichment (Read/Edit/Write)
|
|
42
|
+
filePath?: string;
|
|
43
|
+
fileSize?: number;
|
|
44
|
+
lineCount?: number;
|
|
45
|
+
language?: string;
|
|
46
|
+
gitStatus?: 'clean' | 'modified' | 'new' | 'untracked' | null;
|
|
47
|
+
diff?: DiffSummary;
|
|
48
|
+
// Bash enrichment
|
|
49
|
+
command?: string;
|
|
50
|
+
exitCode?: number | null;
|
|
51
|
+
outputSummary?: OutputSummary;
|
|
52
|
+
workingDirectory?: string;
|
|
53
|
+
// Task enrichment
|
|
54
|
+
subagentType?: string;
|
|
55
|
+
promptSummary?: string;
|
|
56
|
+
resultSummary?: string;
|
|
57
|
+
isBackground?: boolean;
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
interface AuditLogStats {
|
|
61
|
+
total: number;
|
|
62
|
+
byType: Record<string, number>;
|
|
63
|
+
successCount: number;
|
|
64
|
+
errorCount: number;
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
// =============================================================================
|
|
68
|
+
// Helper Functions
|
|
69
|
+
// =============================================================================
|
|
70
|
+
|
|
71
|
+
function formatTimestamp(timestamp: number): string {
|
|
72
|
+
return new Date(timestamp).toLocaleTimeString();
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
function formatDuration(ms: number | undefined): string {
|
|
76
|
+
if (ms === undefined) return '-';
|
|
77
|
+
if (ms < 1000) return `${ms}ms`;
|
|
78
|
+
return `${(ms / 1000).toFixed(1)}s`;
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
function truncateInput(input: string | undefined, maxLength = 60): string {
|
|
82
|
+
if (!input) return '-';
|
|
83
|
+
if (input.length <= maxLength) return input;
|
|
84
|
+
return input.substring(0, maxLength) + '...';
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
function formatBytes(bytes: number | undefined): string {
|
|
88
|
+
if (bytes === undefined) return '';
|
|
89
|
+
if (bytes < 1024) return `${bytes} B`;
|
|
90
|
+
if (bytes < 1024 * 1024) return `${(bytes / 1024).toFixed(1)} KB`;
|
|
91
|
+
return `${(bytes / (1024 * 1024)).toFixed(1)} MB`;
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
function formatFilePath(path: string | undefined): string {
|
|
95
|
+
if (!path) return '';
|
|
96
|
+
// Show last 3 segments for brevity
|
|
97
|
+
const parts = path.split('/');
|
|
98
|
+
if (parts.length <= 3) return path;
|
|
99
|
+
return '.../' + parts.slice(-3).join('/');
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
/** Render enrichment detail rows as label/value pairs */
|
|
103
|
+
function DetailRow({ label, value, mono }: { label: string; value: React.ReactNode; mono?: boolean }) {
|
|
104
|
+
if (value === undefined || value === null || value === '') return null;
|
|
105
|
+
return (
|
|
106
|
+
<div className="detail-row">
|
|
107
|
+
<span className="detail-label">{label}</span>
|
|
108
|
+
<span className={mono ? 'detail-value font-mono' : 'detail-value'}>{value}</span>
|
|
109
|
+
</div>
|
|
110
|
+
);
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
// =============================================================================
|
|
114
|
+
// Component
|
|
115
|
+
// =============================================================================
|
|
116
|
+
|
|
117
|
+
export function AuditLogPanel(): React.ReactElement {
|
|
118
|
+
const [entries, setEntries] = useState<ToolEvent[]>([]);
|
|
119
|
+
const [stats, setStats] = useState<AuditLogStats | null>(null);
|
|
120
|
+
const [toolTypes, setToolTypes] = useState<string[]>([]);
|
|
121
|
+
const [selectedType, setSelectedType] = useState<string>('');
|
|
122
|
+
const [selectedStatus, setSelectedStatus] = useState<string>('');
|
|
123
|
+
const [loading, setLoading] = useState(true);
|
|
124
|
+
const [error, setError] = useState<string | null>(null);
|
|
125
|
+
const [expandedEntry, setExpandedEntry] = useState<number | null>(null);
|
|
126
|
+
|
|
127
|
+
// Auto-scroll refs (using ref instead of state for synchronous updates)
|
|
128
|
+
const autoScrollRef = useRef(true);
|
|
129
|
+
const topRef = useRef<HTMLDivElement>(null);
|
|
130
|
+
const scrollContainerRef = useRef<HTMLDivElement>(null);
|
|
131
|
+
|
|
132
|
+
// Detect manual scroll to pause/resume auto-scroll
|
|
133
|
+
// Re-runs when loading changes so the listener attaches after the scroll container renders
|
|
134
|
+
useEffect(() => {
|
|
135
|
+
const el = scrollContainerRef.current;
|
|
136
|
+
if (!el) return;
|
|
137
|
+
const handler = () => {
|
|
138
|
+
autoScrollRef.current = el.scrollTop === 0;
|
|
139
|
+
};
|
|
140
|
+
el.addEventListener('scroll', handler);
|
|
141
|
+
return () => el.removeEventListener('scroll', handler);
|
|
142
|
+
}, [loading]);
|
|
143
|
+
|
|
144
|
+
// Auto-scroll to newest entry when entries change
|
|
145
|
+
useEffect(() => {
|
|
146
|
+
if (autoScrollRef.current && topRef.current && entries.length > 0) {
|
|
147
|
+
topRef.current.scrollIntoView();
|
|
148
|
+
}
|
|
149
|
+
}, [entries]);
|
|
150
|
+
|
|
151
|
+
// Fetch entries
|
|
152
|
+
const fetchEntries = useCallback(async () => {
|
|
153
|
+
try {
|
|
154
|
+
const params = new URLSearchParams();
|
|
155
|
+
if (selectedType) params.set('toolType', selectedType);
|
|
156
|
+
if (selectedStatus) params.set('status', selectedStatus);
|
|
157
|
+
params.set('limit', '200');
|
|
158
|
+
|
|
159
|
+
const response = await fetch(`/api/audit-log?${params}`);
|
|
160
|
+
if (!response.ok) throw new Error('Failed to fetch audit log');
|
|
161
|
+
const data = await response.json();
|
|
162
|
+
setEntries(data.entries || []);
|
|
163
|
+
setError(null);
|
|
164
|
+
} catch (err) {
|
|
165
|
+
setError(err instanceof Error ? err.message : 'Unknown error');
|
|
166
|
+
}
|
|
167
|
+
}, [selectedType, selectedStatus]);
|
|
168
|
+
|
|
169
|
+
// Fetch stats
|
|
170
|
+
const fetchStats = useCallback(async () => {
|
|
171
|
+
try {
|
|
172
|
+
const response = await fetch('/api/audit-log/stats');
|
|
173
|
+
if (!response.ok) throw new Error('Failed to fetch stats');
|
|
174
|
+
const data = await response.json();
|
|
175
|
+
setStats(data);
|
|
176
|
+
} catch (err) {
|
|
177
|
+
console.error('[AuditLogPanel] Stats error:', err);
|
|
178
|
+
}
|
|
179
|
+
}, []);
|
|
180
|
+
|
|
181
|
+
// Fetch tool types
|
|
182
|
+
const fetchTypes = useCallback(async () => {
|
|
183
|
+
try {
|
|
184
|
+
const response = await fetch('/api/audit-log/types');
|
|
185
|
+
if (!response.ok) throw new Error('Failed to fetch types');
|
|
186
|
+
const data = await response.json();
|
|
187
|
+
setToolTypes(data.types || []);
|
|
188
|
+
} catch (err) {
|
|
189
|
+
console.error('[AuditLogPanel] Types error:', err);
|
|
190
|
+
}
|
|
191
|
+
}, []);
|
|
192
|
+
|
|
193
|
+
// Initial load and WebSocket subscription
|
|
194
|
+
useEffect(() => {
|
|
195
|
+
setLoading(true);
|
|
196
|
+
Promise.all([fetchEntries(), fetchStats(), fetchTypes()]).finally(() => {
|
|
197
|
+
setLoading(false);
|
|
198
|
+
});
|
|
199
|
+
|
|
200
|
+
// Subscribe to real-time updates via spans WebSocket
|
|
201
|
+
const ws = new WebSocket(`ws://${window.location.host}/ws/spans`);
|
|
202
|
+
ws.onmessage = (event) => {
|
|
203
|
+
try {
|
|
204
|
+
const data = JSON.parse(event.data);
|
|
205
|
+
if (data.type === 'span' && data.span) {
|
|
206
|
+
// Add new entry at the beginning
|
|
207
|
+
setEntries(prev => [data.span, ...prev].slice(0, 200));
|
|
208
|
+
// Refresh stats
|
|
209
|
+
fetchStats();
|
|
210
|
+
}
|
|
211
|
+
} catch (err) {
|
|
212
|
+
console.error('[AuditLogPanel] WebSocket parse error:', err);
|
|
213
|
+
}
|
|
214
|
+
};
|
|
215
|
+
|
|
216
|
+
return () => ws.close();
|
|
217
|
+
}, [fetchEntries, fetchStats, fetchTypes]);
|
|
218
|
+
|
|
219
|
+
// Refetch when filters change
|
|
220
|
+
useEffect(() => {
|
|
221
|
+
fetchEntries();
|
|
222
|
+
}, [selectedType, selectedStatus, fetchEntries]);
|
|
223
|
+
|
|
224
|
+
// Export handler
|
|
225
|
+
const handleExport = async (format: 'json' | 'csv') => {
|
|
226
|
+
try {
|
|
227
|
+
const params = new URLSearchParams();
|
|
228
|
+
params.set('format', format);
|
|
229
|
+
params.set('download', 'true');
|
|
230
|
+
if (selectedType) params.set('toolType', selectedType);
|
|
231
|
+
|
|
232
|
+
const response = await fetch(`/api/audit-log/export?${params}`);
|
|
233
|
+
if (!response.ok) throw new Error('Export failed');
|
|
234
|
+
|
|
235
|
+
const blob = await response.blob();
|
|
236
|
+
const url = URL.createObjectURL(blob);
|
|
237
|
+
const a = document.createElement('a');
|
|
238
|
+
a.href = url;
|
|
239
|
+
a.download = `audit-log.${format}`;
|
|
240
|
+
a.click();
|
|
241
|
+
URL.revokeObjectURL(url);
|
|
242
|
+
} catch (err) {
|
|
243
|
+
console.error('[AuditLogPanel] Export error:', err);
|
|
244
|
+
}
|
|
245
|
+
};
|
|
246
|
+
|
|
247
|
+
// Clear handler
|
|
248
|
+
const handleClear = async () => {
|
|
249
|
+
if (!confirm('Clear all audit log entries?')) return;
|
|
250
|
+
try {
|
|
251
|
+
const response = await fetch('/api/audit-log', { method: 'DELETE' });
|
|
252
|
+
if (!response.ok) throw new Error('Clear failed');
|
|
253
|
+
setEntries([]);
|
|
254
|
+
fetchStats();
|
|
255
|
+
} catch (err) {
|
|
256
|
+
console.error('[AuditLogPanel] Clear error:', err);
|
|
257
|
+
}
|
|
258
|
+
};
|
|
259
|
+
|
|
260
|
+
if (loading) {
|
|
261
|
+
return (
|
|
262
|
+
<div className="audit-log-panel p-4 space-y-3">
|
|
263
|
+
<div className="flex gap-4">
|
|
264
|
+
<Skeleton className="h-4 w-20" />
|
|
265
|
+
<Skeleton className="h-4 w-24" />
|
|
266
|
+
<Skeleton className="h-4 w-20" />
|
|
267
|
+
</div>
|
|
268
|
+
<Separator />
|
|
269
|
+
<div className="space-y-2">
|
|
270
|
+
<Skeleton className="h-6 w-full" />
|
|
271
|
+
<Skeleton className="h-6 w-full" />
|
|
272
|
+
<Skeleton className="h-6 w-full" />
|
|
273
|
+
<Skeleton className="h-6 w-3/4" />
|
|
274
|
+
</div>
|
|
275
|
+
</div>
|
|
276
|
+
);
|
|
277
|
+
}
|
|
278
|
+
|
|
279
|
+
if (error) {
|
|
280
|
+
return (
|
|
281
|
+
<div className="audit-log-panel p-4">
|
|
282
|
+
<div className="text-error">Error: {error}</div>
|
|
283
|
+
</div>
|
|
284
|
+
);
|
|
285
|
+
}
|
|
286
|
+
|
|
287
|
+
return (
|
|
288
|
+
<div className="audit-log-panel flex flex-col h-full">
|
|
289
|
+
<TooltipProvider delayDuration={300}>
|
|
290
|
+
{/* Stats Bar */}
|
|
291
|
+
{stats && (
|
|
292
|
+
<>
|
|
293
|
+
<div className="audit-log-stats flex gap-4 p-2 text-sm">
|
|
294
|
+
<span className="text-muted">Total: <strong>{stats.total}</strong></span>
|
|
295
|
+
<span className="text-success">Success: <strong>{stats.successCount}</strong></span>
|
|
296
|
+
<span className="text-error">Errors: <strong>{stats.errorCount}</strong></span>
|
|
297
|
+
</div>
|
|
298
|
+
<Separator />
|
|
299
|
+
</>
|
|
300
|
+
)}
|
|
301
|
+
|
|
302
|
+
{/* Filters and Actions */}
|
|
303
|
+
<div className="audit-log-toolbar flex p-2 items-center">
|
|
304
|
+
<select
|
|
305
|
+
value={selectedType}
|
|
306
|
+
onChange={(e) => setSelectedType(e.target.value)}
|
|
307
|
+
>
|
|
308
|
+
<option value="">All Tools</option>
|
|
309
|
+
{toolTypes.map(type => (
|
|
310
|
+
<option key={type} value={type}>{type}</option>
|
|
311
|
+
))}
|
|
312
|
+
</select>
|
|
313
|
+
|
|
314
|
+
<select
|
|
315
|
+
value={selectedStatus}
|
|
316
|
+
onChange={(e) => setSelectedStatus(e.target.value)}
|
|
317
|
+
>
|
|
318
|
+
<option value="">All Status</option>
|
|
319
|
+
<option value="success">Success</option>
|
|
320
|
+
<option value="error">Errors</option>
|
|
321
|
+
</select>
|
|
322
|
+
|
|
323
|
+
<div className="flex-1" />
|
|
324
|
+
|
|
325
|
+
<Tooltip>
|
|
326
|
+
<TooltipTrigger asChild>
|
|
327
|
+
<Button variant="ghost" size="sm" className="audit-log-btn" onClick={() => handleExport('json')}>
|
|
328
|
+
JSON
|
|
329
|
+
</Button>
|
|
330
|
+
</TooltipTrigger>
|
|
331
|
+
<TooltipContent>Export as JSON</TooltipContent>
|
|
332
|
+
</Tooltip>
|
|
333
|
+
<Tooltip>
|
|
334
|
+
<TooltipTrigger asChild>
|
|
335
|
+
<Button variant="ghost" size="sm" className="audit-log-btn" onClick={() => handleExport('csv')}>
|
|
336
|
+
CSV
|
|
337
|
+
</Button>
|
|
338
|
+
</TooltipTrigger>
|
|
339
|
+
<TooltipContent>Export as CSV</TooltipContent>
|
|
340
|
+
</Tooltip>
|
|
341
|
+
<Tooltip>
|
|
342
|
+
<TooltipTrigger asChild>
|
|
343
|
+
<Button variant="ghost" size="sm" className="audit-log-btn audit-log-btn-clear" onClick={handleClear}>
|
|
344
|
+
Clear
|
|
345
|
+
</Button>
|
|
346
|
+
</TooltipTrigger>
|
|
347
|
+
<TooltipContent>Clear audit log</TooltipContent>
|
|
348
|
+
</Tooltip>
|
|
349
|
+
</div>
|
|
350
|
+
|
|
351
|
+
{/* Entries List */}
|
|
352
|
+
<div ref={scrollContainerRef} className="audit-log-entries flex-1" style={{ overflow: 'auto' }}>
|
|
353
|
+
<div ref={topRef} />
|
|
354
|
+
{entries.length === 0 ? (
|
|
355
|
+
<div className="p-4 text-muted text-center">No entries</div>
|
|
356
|
+
) : (
|
|
357
|
+
<table>
|
|
358
|
+
<thead>
|
|
359
|
+
<tr>
|
|
360
|
+
<th className="text-left">Time</th>
|
|
361
|
+
<th className="text-left">Tool</th>
|
|
362
|
+
<th className="text-left">Input</th>
|
|
363
|
+
<th className="text-right">Duration</th>
|
|
364
|
+
<th className="text-center">Status</th>
|
|
365
|
+
</tr>
|
|
366
|
+
</thead>
|
|
367
|
+
<tbody>
|
|
368
|
+
{entries.map((entry, idx) => (
|
|
369
|
+
<React.Fragment key={`${entry.timestamp}-${idx}`}>
|
|
370
|
+
<tr
|
|
371
|
+
className={expandedEntry === idx ? 'expanded' : ''}
|
|
372
|
+
onClick={() => setExpandedEntry(expandedEntry === idx ? null : idx)}
|
|
373
|
+
>
|
|
374
|
+
<td className="whitespace-nowrap">
|
|
375
|
+
{formatTimestamp(entry.timestamp)}
|
|
376
|
+
</td>
|
|
377
|
+
<td className="tool-name font-mono" data-tool={entry.toolName.toLowerCase()}>{entry.toolName}</td>
|
|
378
|
+
<td className="truncate max-w-[200px]">
|
|
379
|
+
{entry.input ? (
|
|
380
|
+
<Tooltip>
|
|
381
|
+
<TooltipTrigger asChild>
|
|
382
|
+
<span>{truncateInput(entry.input)}</span>
|
|
383
|
+
</TooltipTrigger>
|
|
384
|
+
<TooltipContent>{entry.input}</TooltipContent>
|
|
385
|
+
</Tooltip>
|
|
386
|
+
) : (
|
|
387
|
+
truncateInput(entry.input)
|
|
388
|
+
)}
|
|
389
|
+
</td>
|
|
390
|
+
<td className="text-right whitespace-nowrap">
|
|
391
|
+
{formatDuration(entry.durationMs)}
|
|
392
|
+
</td>
|
|
393
|
+
<td className="text-center">
|
|
394
|
+
{entry.success ? (
|
|
395
|
+
<span className="status-ok">✓</span>
|
|
396
|
+
) : (
|
|
397
|
+
<Tooltip>
|
|
398
|
+
<TooltipTrigger asChild>
|
|
399
|
+
<span className="status-err">✗</span>
|
|
400
|
+
</TooltipTrigger>
|
|
401
|
+
<TooltipContent>{entry.error}</TooltipContent>
|
|
402
|
+
</Tooltip>
|
|
403
|
+
)}
|
|
404
|
+
</td>
|
|
405
|
+
</tr>
|
|
406
|
+
{expandedEntry === idx && (
|
|
407
|
+
<tr className="expanded-detail">
|
|
408
|
+
<td colSpan={5}>
|
|
409
|
+
<div className="detail-grid">
|
|
410
|
+
{/* Common fields */}
|
|
411
|
+
<DetailRow label="Tool" value={entry.toolName} />
|
|
412
|
+
<DetailRow label="Duration" value={entry.durationMs !== undefined ? formatDuration(entry.durationMs) : undefined} />
|
|
413
|
+
{entry.exitCode !== undefined && entry.exitCode !== null && (
|
|
414
|
+
<DetailRow label="Exit Code" value={entry.exitCode} mono />
|
|
415
|
+
)}
|
|
416
|
+
|
|
417
|
+
{/* File enrichment (Read/Edit/Write) */}
|
|
418
|
+
<DetailRow label="File" value={entry.filePath} mono />
|
|
419
|
+
{entry.language && <DetailRow label="Language" value={entry.language} />}
|
|
420
|
+
{entry.fileSize !== undefined && <DetailRow label="Size" value={formatBytes(entry.fileSize)} />}
|
|
421
|
+
{entry.lineCount !== undefined && <DetailRow label="Lines" value={entry.lineCount.toLocaleString()} />}
|
|
422
|
+
{entry.gitStatus && <DetailRow label="Git" value={entry.gitStatus} />}
|
|
423
|
+
{entry.diff && (
|
|
424
|
+
<DetailRow label="Diff" value={
|
|
425
|
+
<span>
|
|
426
|
+
<span style={{color: 'var(--success, #22c55e)'}}>+{entry.diff.added}</span>
|
|
427
|
+
{' '}
|
|
428
|
+
<span style={{color: 'var(--error, #ef4444)'}}>-{entry.diff.removed}</span>
|
|
429
|
+
</span>
|
|
430
|
+
} />
|
|
431
|
+
)}
|
|
432
|
+
|
|
433
|
+
{/* Bash enrichment */}
|
|
434
|
+
{entry.command && <DetailRow label="Command" value={entry.command} mono />}
|
|
435
|
+
{entry.workingDirectory && <DetailRow label="CWD" value={formatFilePath(entry.workingDirectory)} mono />}
|
|
436
|
+
|
|
437
|
+
{/* Task enrichment */}
|
|
438
|
+
{entry.subagentType && <DetailRow label="Agent" value={entry.subagentType} />}
|
|
439
|
+
{entry.isBackground && <DetailRow label="Background" value="Yes" />}
|
|
440
|
+
{entry.promptSummary && <DetailRow label="Prompt" value={entry.promptSummary} />}
|
|
441
|
+
{entry.resultSummary && <DetailRow label="Result" value={entry.resultSummary} />}
|
|
442
|
+
</div>
|
|
443
|
+
|
|
444
|
+
{/* Output summary (Bash) */}
|
|
445
|
+
{entry.outputSummary && entry.outputSummary.totalLines > 0 && (
|
|
446
|
+
<div className="detail-output">
|
|
447
|
+
<div className="detail-label">Output ({entry.outputSummary.totalLines} lines{entry.outputSummary.truncated ? ', truncated' : ''})</div>
|
|
448
|
+
<pre>{entry.outputSummary.firstLines.join('\n')}{entry.outputSummary.truncated && entry.outputSummary.lastLines.length > 0 ? '\n...\n' + entry.outputSummary.lastLines.join('\n') : ''}</pre>
|
|
449
|
+
</div>
|
|
450
|
+
)}
|
|
451
|
+
|
|
452
|
+
{/* Raw input (fallback when no enrichment) */}
|
|
453
|
+
{entry.input && !entry.command && !entry.filePath && !entry.subagentType && (
|
|
454
|
+
<div className="detail-output">
|
|
455
|
+
<div className="detail-label">Input</div>
|
|
456
|
+
<pre>{entry.input}</pre>
|
|
457
|
+
</div>
|
|
458
|
+
)}
|
|
459
|
+
|
|
460
|
+
{/* Raw output (when no outputSummary) */}
|
|
461
|
+
{entry.output && !entry.outputSummary && (
|
|
462
|
+
<div className="detail-output">
|
|
463
|
+
<div className="detail-label">Output</div>
|
|
464
|
+
<pre>{entry.output.substring(0, 500)}{entry.output.length > 500 && '...'}</pre>
|
|
465
|
+
</div>
|
|
466
|
+
)}
|
|
467
|
+
|
|
468
|
+
{/* Error */}
|
|
469
|
+
{entry.error && (
|
|
470
|
+
<div className="detail-output">
|
|
471
|
+
<div className="detail-label status-err">Error</div>
|
|
472
|
+
<pre className="status-err">{entry.error}</pre>
|
|
473
|
+
</div>
|
|
474
|
+
)}
|
|
475
|
+
</td>
|
|
476
|
+
</tr>
|
|
477
|
+
)}
|
|
478
|
+
</React.Fragment>
|
|
479
|
+
))}
|
|
480
|
+
</tbody>
|
|
481
|
+
</table>
|
|
482
|
+
)}
|
|
483
|
+
</div>
|
|
484
|
+
</TooltipProvider>
|
|
485
|
+
</div>
|
|
486
|
+
);
|
|
487
|
+
}
|
|
488
|
+
|
|
489
|
+
export default AuditLogPanel;
|