@pennyfarthing/core 11.0.0 → 11.1.1
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 +81 -23
- package/package.json +1 -1
- package/packages/core/dist/cli/utils/010-detect-remove-old-packages.test.d.ts +20 -0
- package/packages/core/dist/cli/utils/010-detect-remove-old-packages.test.d.ts.map +1 -0
- package/packages/core/dist/cli/utils/010-detect-remove-old-packages.test.js +278 -0
- package/packages/core/dist/cli/utils/010-detect-remove-old-packages.test.js.map +1 -0
- package/packages/core/dist/cli/utils/constants.d.ts +8 -2
- package/packages/core/dist/cli/utils/constants.d.ts.map +1 -1
- package/packages/core/dist/cli/utils/constants.js +4 -1
- package/packages/core/dist/cli/utils/constants.js.map +1 -1
- package/packages/core/dist/cli/utils/constants.test.d.ts +10 -0
- package/packages/core/dist/cli/utils/constants.test.d.ts.map +1 -0
- package/packages/core/dist/cli/utils/constants.test.js +38 -0
- package/packages/core/dist/cli/utils/constants.test.js.map +1 -0
- package/packages/core/dist/consultation/consultation-protocol.d.ts +139 -0
- package/packages/core/dist/consultation/consultation-protocol.d.ts.map +1 -0
- package/packages/core/dist/consultation/consultation-protocol.js +178 -0
- package/packages/core/dist/consultation/consultation-protocol.js.map +1 -0
- package/packages/core/dist/consultation/consultation-protocol.test.d.ts +20 -0
- package/packages/core/dist/consultation/consultation-protocol.test.d.ts.map +1 -0
- package/packages/core/dist/consultation/consultation-protocol.test.js +474 -0
- package/packages/core/dist/consultation/consultation-protocol.test.js.map +1 -0
- package/packages/core/dist/consultation/dialogue-manager.d.ts +75 -0
- package/packages/core/dist/consultation/dialogue-manager.d.ts.map +1 -0
- package/packages/core/dist/consultation/dialogue-manager.js +334 -0
- package/packages/core/dist/consultation/dialogue-manager.js.map +1 -0
- package/packages/core/dist/consultation/dialogue-manager.test.d.ts +19 -0
- package/packages/core/dist/consultation/dialogue-manager.test.d.ts.map +1 -0
- package/packages/core/dist/consultation/dialogue-manager.test.js +444 -0
- package/packages/core/dist/consultation/dialogue-manager.test.js.map +1 -0
- package/packages/core/dist/public/js/react/react.js +3 -3
- package/packages/core/dist/scripts/theme-detail.test.d.ts +10 -0
- package/packages/core/dist/scripts/theme-detail.test.js +199 -0
- package/packages/core/dist/server/api/git.d.ts +13 -1
- package/packages/core/dist/server/api/git.d.ts.map +1 -1
- package/packages/core/dist/server/api/git.js +53 -34
- package/packages/core/dist/server/api/git.js.map +1 -1
- package/packages/core/dist/server/api/health-score.d.ts.map +1 -1
- package/packages/core/dist/server/api/health-score.js +25 -1
- package/packages/core/dist/server/api/health-score.js.map +1 -1
- package/packages/core/dist/server/api/settings.d.ts.map +1 -1
- package/packages/core/dist/server/api/settings.js +63 -1
- package/packages/core/dist/server/api/settings.js.map +1 -1
- package/packages/core/dist/server/api/theme-agents.d.ts.map +1 -1
- package/packages/core/dist/server/api/theme-agents.js +61 -0
- package/packages/core/dist/server/api/theme-agents.js.map +1 -1
- package/packages/core/dist/server/server.d.ts.map +1 -1
- package/packages/core/dist/server/server.js +17 -12
- package/packages/core/dist/server/server.js.map +1 -1
- package/packages/core/dist/shared/skill-search.test.js +2 -2
- package/packages/core/dist/workflow/gate-file-validation.d.ts +49 -0
- package/packages/core/dist/workflow/gate-file-validation.d.ts.map +1 -0
- package/packages/core/dist/workflow/gate-file-validation.js +157 -0
- package/packages/core/dist/workflow/gate-file-validation.js.map +1 -0
- package/packages/core/dist/workflow/gate-file-validation.test.d.ts +19 -0
- package/packages/core/dist/workflow/gate-file-validation.test.d.ts.map +1 -0
- package/packages/core/dist/workflow/gate-file-validation.test.js +536 -0
- package/packages/core/dist/workflow/gate-file-validation.test.js.map +1 -0
- package/packages/core/dist/workflow/gate-schema-validation.test.d.ts +14 -0
- package/packages/core/dist/workflow/gate-schema-validation.test.d.ts.map +1 -0
- package/packages/core/dist/workflow/gate-schema-validation.test.js +339 -0
- package/packages/core/dist/workflow/gate-schema-validation.test.js.map +1 -0
- package/packages/core/dist/workflow/handoff.js +2 -2
- package/packages/core/dist/workflow/handoff.js.map +1 -1
- package/packages/core/dist/workflow/handoff.test.js +16 -0
- package/packages/core/dist/workflow/handoff.test.js.map +1 -1
- package/packages/core/dist/workflow/workflow-schema.d.ts +4 -2
- package/packages/core/dist/workflow/workflow-schema.d.ts.map +1 -1
- package/packages/core/dist/workflow/workflow-schema.js +43 -8
- package/packages/core/dist/workflow/workflow-schema.js.map +1 -1
- package/pennyfarthing-dist/agents/README.md +6 -14
- package/pennyfarthing-dist/agents/architect.md +43 -29
- package/pennyfarthing-dist/agents/ba.md +30 -29
- package/pennyfarthing-dist/agents/dev.md +32 -43
- package/pennyfarthing-dist/agents/devops.md +57 -21
- package/pennyfarthing-dist/agents/orchestrator.md +3 -10
- package/pennyfarthing-dist/agents/pm.md +45 -31
- package/pennyfarthing-dist/agents/reviewer.md +20 -66
- package/pennyfarthing-dist/agents/sm-setup.md +2 -2
- package/pennyfarthing-dist/agents/sm.md +8 -30
- package/pennyfarthing-dist/agents/tea.md +25 -41
- package/pennyfarthing-dist/agents/tech-writer.md +33 -90
- package/pennyfarthing-dist/agents/ux-designer.md +39 -39
- package/pennyfarthing-dist/commands/benchmark-control.md +8 -64
- package/pennyfarthing-dist/commands/benchmark.md +8 -480
- package/pennyfarthing-dist/commands/job-fair.md +8 -97
- package/pennyfarthing-dist/commands/pf-benchmark-control.md +70 -0
- package/pennyfarthing-dist/commands/pf-benchmark.md +486 -0
- package/pennyfarthing-dist/commands/pf-chore.md +4 -4
- package/pennyfarthing-dist/commands/pf-ci.md +40 -0
- package/pennyfarthing-dist/commands/pf-close-epic.md +9 -27
- package/pennyfarthing-dist/commands/pf-continue-session.md +9 -213
- package/pennyfarthing-dist/commands/pf-create-branches-from-story.md +11 -353
- package/pennyfarthing-dist/commands/pf-docs.md +28 -0
- package/pennyfarthing-dist/commands/pf-epic.md +67 -0
- package/pennyfarthing-dist/commands/pf-git-cleanup.md +11 -52
- package/pennyfarthing-dist/commands/pf-git.md +75 -0
- package/pennyfarthing-dist/commands/pf-help.md +110 -128
- package/pennyfarthing-dist/commands/pf-job-fair.md +102 -0
- package/pennyfarthing-dist/commands/pf-new-work.md +9 -18
- package/pennyfarthing-dist/commands/pf-parallel-work.md +6 -66
- package/pennyfarthing-dist/commands/pf-release.md +11 -76
- package/pennyfarthing-dist/commands/pf-repo-status.md +11 -44
- package/pennyfarthing-dist/commands/pf-run-ci.md +8 -111
- package/pennyfarthing-dist/commands/pf-session.md +51 -0
- package/pennyfarthing-dist/commands/pf-solo.md +447 -0
- package/pennyfarthing-dist/commands/pf-sprint-planning.md +8 -104
- package/pennyfarthing-dist/commands/pf-standalone.md +1 -1
- package/pennyfarthing-dist/commands/pf-start-epic.md +9 -163
- package/pennyfarthing-dist/commands/pf-sync-epic-to-jira.md +8 -179
- package/pennyfarthing-dist/commands/pf-sync-work-with-sprint.md +8 -368
- package/pennyfarthing-dist/commands/pf-update-domain-docs.md +8 -78
- package/pennyfarthing-dist/commands/solo.md +8 -442
- package/pennyfarthing-dist/guides/agent-behavior.md +13 -13
- package/pennyfarthing-dist/guides/agent-coordination.md +7 -7
- package/pennyfarthing-dist/guides/agent-tag-taxonomy.md +6 -5
- package/pennyfarthing-dist/guides/bikerack.md +128 -0
- package/pennyfarthing-dist/guides/brownfield-tools.md +133 -0
- package/pennyfarthing-dist/guides/command-tag-taxonomy.md +2 -2
- package/pennyfarthing-dist/guides/gate-schema.md +227 -0
- package/pennyfarthing-dist/guides/gates.md +120 -0
- package/pennyfarthing-dist/guides/handoff-cli.md +116 -0
- package/pennyfarthing-dist/guides/hooks.md +86 -4
- package/pennyfarthing-dist/guides/output-styles.md +65 -0
- package/pennyfarthing-dist/guides/patterns/approval-gates-pattern.md +5 -5
- package/pennyfarthing-dist/guides/patterns/tdd-flow-pattern.md +4 -4
- package/pennyfarthing-dist/guides/prompt-patterns.md +5 -5
- package/pennyfarthing-dist/guides/reflector.md +4 -4
- package/pennyfarthing-dist/guides/session-artifacts.md +1 -1
- package/pennyfarthing-dist/guides/skill-schema.md +1 -1
- package/pennyfarthing-dist/guides/tandem-protocol.md +13 -1
- package/pennyfarthing-dist/guides/worktree-mode.md +3 -3
- package/pennyfarthing-dist/guides/xml-tags.md +5 -4
- package/pennyfarthing-dist/personas/themes/hogans-heroes.yaml +11 -22
- package/pennyfarthing-dist/personas/themes/stephen-king.yaml +13 -24
- package/pennyfarthing-dist/scripts/core/dialogue-manager.sh +322 -0
- package/pennyfarthing-dist/scripts/core/phase-check-start.sh +1 -1
- package/pennyfarthing-dist/scripts/hooks/otel-auto-config.sh +19 -14
- package/pennyfarthing-dist/scripts/portraits/generate-portraits.py +191 -57
- package/pennyfarthing-dist/scripts/portraits/generate-portraits.sh +26 -10
- package/pennyfarthing-dist/skills/pf-changelog/SKILL.md +4 -4
- package/pennyfarthing-dist/skills/pf-sprint/skill.md +1 -1
- package/pennyfarthing-dist/skills/skill-registry.schema.json +4 -0
- package/pennyfarthing-dist/skills/skill-registry.yaml +5 -0
- package/pennyfarthing-dist/workflows/2party-tdd.yaml +11 -0
- package/pennyfarthing-dist/workflows/agent-docs.yaml +2 -0
- package/pennyfarthing-dist/workflows/bdd-tandem.yaml +4 -0
- package/pennyfarthing-dist/workflows/bdd.yaml +4 -0
- package/pennyfarthing-dist/workflows/git-cleanup/steps/step-05-complete.md +1 -1
- package/pennyfarthing-dist/workflows/tdd-tandem.yaml +3 -0
- package/pennyfarthing-dist/workflows/tdd.yaml +3 -0
- package/pennyfarthing-dist/workflows/trivial.yaml +2 -0
- package/pennyfarthing_scripts/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/__pycache__/bellmode_hook.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/__pycache__/config.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/__pycache__/context.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/__pycache__/hooks.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/__pycache__/jira_bidirectional_sync.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/__pycache__/jira_epic_creation.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/__pycache__/jira_sync.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/__pycache__/jira_sync_story.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/__pycache__/output.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/__pycache__/patch_mode.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/__pycache__/pretooluse_hook.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/__pycache__/schema_validation_hook.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/__pycache__/session_start_hook.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/__pycache__/workflow.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/bc/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/bc/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/bc/__pycache__/focus.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/bikerack/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/bikerack/__pycache__/__main__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/bikerack/__pycache__/background_panel.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/bikerack/__pycache__/base_panel.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/bikerack/__pycache__/changed_panel.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/bikerack/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/bikerack/__pycache__/debug_panel.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/bikerack/__pycache__/diffs_panel.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/bikerack/__pycache__/git_panel.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/bikerack/__pycache__/launcher.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/bikerack/__pycache__/sprint_panel.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/bikerack/__pycache__/tui.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/bikerack/__pycache__/ws_client.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/bikerack/cli.py +10 -11
- package/pennyfarthing_scripts/bikerack/debug_panel.py +218 -0
- package/pennyfarthing_scripts/bikerack/diffs_panel.py +203 -27
- package/pennyfarthing_scripts/brownfield/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/brownfield/__pycache__/__main__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/brownfield/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/brownfield/__pycache__/discover.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/cli.py +114 -0
- package/pennyfarthing_scripts/codemarkers/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/codemarkers/__pycache__/__main__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/codemarkers/__pycache__/analyze.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/codemarkers/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/codemarkers/__pycache__/formatters.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/codemarkers/__pycache__/models.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/common/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/common/__pycache__/config.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/common/__pycache__/output.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/common/__pycache__/themes.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/complexity/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/complexity/__pycache__/__main__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/complexity/__pycache__/analyze.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/complexity/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/complexity/__pycache__/formatters.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/complexity/__pycache__/models.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/deadcode/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/deadcode/__pycache__/__main__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/deadcode/__pycache__/analyze.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/deadcode/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/deadcode/__pycache__/formatters.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/deadcode/__pycache__/models.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/dependencies/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/dependencies/__pycache__/__main__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/dependencies/__pycache__/analyze.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/dependencies/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/dependencies/__pycache__/formatters.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/dependencies/__pycache__/models.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/epic/__init__.py +0 -0
- package/pennyfarthing_scripts/epic/cli.py +64 -0
- package/pennyfarthing_scripts/gate/__init__.py +1 -0
- package/pennyfarthing_scripts/gate/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/gate/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/gate/__pycache__/validate.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/gate/cli.py +56 -0
- package/pennyfarthing_scripts/gate/validate.py +266 -0
- package/pennyfarthing_scripts/git/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/git/__pycache__/create_branches.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/git/__pycache__/status_all.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/git_group/__init__.py +0 -0
- package/pennyfarthing_scripts/git_group/cli.py +100 -0
- package/pennyfarthing_scripts/handoff/__init__.py +1 -0
- package/pennyfarthing_scripts/handoff/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/handoff/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/handoff/__pycache__/complete_phase.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/handoff/__pycache__/gate_file.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/handoff/__pycache__/gate_runner.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/handoff/__pycache__/marker.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/handoff/__pycache__/resolve_gate.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/handoff/cli.py +120 -0
- package/pennyfarthing_scripts/handoff/complete_phase.py +155 -0
- package/pennyfarthing_scripts/handoff/gate_file.py +105 -0
- package/pennyfarthing_scripts/handoff/gate_runner.py +152 -0
- package/pennyfarthing_scripts/handoff/marker.py +109 -0
- package/pennyfarthing_scripts/handoff/resolve_gate.py +152 -0
- package/pennyfarthing_scripts/healthscore/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/healthscore/__pycache__/__main__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/healthscore/__pycache__/analyze.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/healthscore/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/healthscore/__pycache__/formatters.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/healthscore/__pycache__/models.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/hotspots/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/hotspots/__pycache__/__main__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/hotspots/__pycache__/analyze.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/hotspots/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/hotspots/__pycache__/formatters.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/hotspots/__pycache__/models.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/jira/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/jira/__pycache__/__main__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/jira/__pycache__/bidirectional.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/jira/__pycache__/claim.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/jira/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/jira/__pycache__/client.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/jira/__pycache__/create.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/jira/__pycache__/epic.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/jira/__pycache__/operations.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/jira/__pycache__/reconcile.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/jira/__pycache__/story.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/jira/__pycache__/sync.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/launch/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/launch/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/migration/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/migration/__pycache__/session.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/migration/__pycache__/skill.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/migration/__pycache__/step.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/migration/__pycache__/validate.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/preflight/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/preflight/__pycache__/__main__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/preflight/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/preflight/__pycache__/finish.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/prime/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/prime/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/prime/__pycache__/loader.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/prime/__pycache__/models.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/prime/__pycache__/persona.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/prime/__pycache__/session.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/prime/__pycache__/tiers.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/prime/__pycache__/version_sentinel.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/prime/__pycache__/workflow.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/prime/workflow.py +39 -0
- package/pennyfarthing_scripts/session/__init__.py +0 -0
- package/pennyfarthing_scripts/session/cli.py +87 -0
- package/pennyfarthing_scripts/sprint/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/sprint/__pycache__/__main__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/sprint/__pycache__/archive.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/sprint/__pycache__/archive_epic.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/sprint/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/sprint/__pycache__/epic_add.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/sprint/__pycache__/epic_update.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/sprint/__pycache__/import_epic.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/sprint/__pycache__/loader.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/sprint/__pycache__/status.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/sprint/__pycache__/story_add.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/sprint/__pycache__/story_finish.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/sprint/__pycache__/story_update.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/sprint/__pycache__/validate_cmd.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/sprint/__pycache__/validator.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/sprint/__pycache__/work.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/sprint/__pycache__/yaml_io.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/sprint/story_finish.py +14 -0
- package/pennyfarthing_scripts/story/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/story/__pycache__/__main__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/story/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/story/__pycache__/create.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/story/__pycache__/size.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/story/__pycache__/template.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/conftest.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_108_2_remove_handoff_fallback.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_archive_epic.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_bc.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_bikerack.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_brownfield.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_cli_modules.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_cli_normalization.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_codemarkers.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_common.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_epic_shard_validation.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_gate_file_resolution.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_gate_runner.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_git_utils.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_handoff_cli.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_handoff_e2e.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_healthscore.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_jira_package.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_package_structure.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_patch_mode.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_prime.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_resolve_gate_file_field.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_sprint_package.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_sprint_panel.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_sprint_validator.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_story_add.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_story_package.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_story_update.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_tiers.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_token_counting.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_topology_loader.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_tui_focus.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_tui_panel_persistence.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_validate_cmd.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_version_sentinel.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_workflow_check.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_yaml_io.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing_scripts/tests/test_108_1_gate_migration.py +540 -0
- package/pennyfarthing_scripts/tests/test_108_2_remove_handoff_fallback.py +339 -0
- package/pennyfarthing_scripts/tests/test_confidence_sm_evaluation.py +253 -0
- package/pennyfarthing_scripts/tests/test_confidence_sm_gate.py +315 -0
- package/pennyfarthing_scripts/tests/test_gate_file_resolution.py +341 -0
- package/pennyfarthing_scripts/tests/test_gate_runner.py +620 -0
- package/pennyfarthing_scripts/tests/test_handoff_cli.py +929 -0
- package/pennyfarthing_scripts/tests/test_handoff_e2e.py +454 -0
- package/pennyfarthing_scripts/tests/test_resolve_gate_file_field.py +464 -0
- package/pennyfarthing_scripts/theme/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/theme/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/validate/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/validate/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/validate/adapters/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/validate/adapters/__pycache__/agent.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/validate/adapters/__pycache__/schema.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/validate/adapters/__pycache__/skill_command.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/validate/adapters/__pycache__/sprint.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/validate/adapters/__pycache__/workflow.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/validate/adapters/skill_command.py +200 -0
- package/pennyfarthing_scripts/validate/adapters/workflow.py +64 -0
- package/pennyfarthing_scripts/validate/cli.py +15 -4
- package/packages/core/dist/scripts/benchmark-integration.d.ts +0 -182
- package/packages/core/dist/scripts/benchmark-integration.d.ts.map +0 -1
- package/packages/core/dist/scripts/benchmark-integration.js +0 -691
- package/packages/core/dist/scripts/benchmark-integration.js.map +0 -1
- package/packages/core/dist/scripts/job-fair-aggregator.d.ts +0 -150
- package/packages/core/dist/scripts/job-fair-aggregator.d.ts.map +0 -1
- package/packages/core/dist/scripts/job-fair-aggregator.js +0 -547
- package/packages/core/dist/scripts/job-fair-aggregator.js.map +0 -1
- package/pennyfarthing-dist/agents/handoff.md +0 -250
- package/pennyfarthing-dist/agents/sm-handoff.md +0 -152
- package/pennyfarthing-dist/scripts/core/handoff-marker.sh +0 -112
- package/pennyfarthing-dist/scripts/hooks/__pycache__/question_reflector_check.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/__pycache__/__init__.cpython-311.pyc +0 -0
- package/pennyfarthing_scripts/__pycache__/jira.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/__pycache__/sprint.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/__pycache__/workflow.cpython-311.pyc +0 -0
- package/pennyfarthing_scripts/jira/__pycache__/compat.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/jira/__pycache__/mappings.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/jira/__pycache__/models.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/migration/__pycache__/__main__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/migration/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/prime/__pycache__/__main__.cpython-314.pyc +0 -0
- package/pennyfarthing_scripts/tests/__pycache__/test_workflow_cli.cpython-314-pytest-9.0.2.pyc +0 -0
|
@@ -21,6 +21,7 @@ workflow:
|
|
|
21
21
|
input: [session_file, story_context]
|
|
22
22
|
output: [design_spec, user_flows, wireframes, behavior_scenarios]
|
|
23
23
|
gate:
|
|
24
|
+
file: gates/design-review
|
|
24
25
|
type: design_review
|
|
25
26
|
condition: UX spec defines user flows and acceptance behaviors
|
|
26
27
|
tandem:
|
|
@@ -32,6 +33,7 @@ workflow:
|
|
|
32
33
|
input: [design_spec, behavior_scenarios, story_context]
|
|
33
34
|
output: [failing_tests]
|
|
34
35
|
gate:
|
|
36
|
+
file: gates/tests-fail
|
|
35
37
|
type: tests_fail
|
|
36
38
|
condition: Behavior scenarios have test coverage
|
|
37
39
|
|
|
@@ -40,6 +42,7 @@ workflow:
|
|
|
40
42
|
input: [failing_tests, design_spec, story_context]
|
|
41
43
|
output: [implementation, passing_tests]
|
|
42
44
|
gate:
|
|
45
|
+
file: gates/tests-pass
|
|
43
46
|
type: tests_pass
|
|
44
47
|
condition: All tests passing, UX spec implemented
|
|
45
48
|
tandem:
|
|
@@ -51,6 +54,7 @@ workflow:
|
|
|
51
54
|
input: [implementation, passing_tests, design_spec]
|
|
52
55
|
output: [approval]
|
|
53
56
|
gate:
|
|
57
|
+
file: gates/approval
|
|
54
58
|
type: approval
|
|
55
59
|
condition: Code review approved, UX requirements met
|
|
56
60
|
tandem:
|
|
@@ -19,6 +19,7 @@ workflow:
|
|
|
19
19
|
input: [session_file, story_context]
|
|
20
20
|
output: [design_spec, user_flows, wireframes, behavior_scenarios]
|
|
21
21
|
gate:
|
|
22
|
+
file: gates/design-review
|
|
22
23
|
type: design_review
|
|
23
24
|
condition: UX spec defines user flows and acceptance behaviors
|
|
24
25
|
|
|
@@ -27,6 +28,7 @@ workflow:
|
|
|
27
28
|
input: [design_spec, behavior_scenarios, story_context]
|
|
28
29
|
output: [failing_tests]
|
|
29
30
|
gate:
|
|
31
|
+
file: gates/tests-fail
|
|
30
32
|
type: tests_fail
|
|
31
33
|
condition: Behavior scenarios have test coverage
|
|
32
34
|
|
|
@@ -35,6 +37,7 @@ workflow:
|
|
|
35
37
|
input: [failing_tests, design_spec, story_context]
|
|
36
38
|
output: [implementation, passing_tests]
|
|
37
39
|
gate:
|
|
40
|
+
file: gates/tests-pass
|
|
38
41
|
type: tests_pass
|
|
39
42
|
condition: All tests passing, UX spec implemented
|
|
40
43
|
|
|
@@ -43,6 +46,7 @@ workflow:
|
|
|
43
46
|
input: [implementation, passing_tests, design_spec]
|
|
44
47
|
output: [approval]
|
|
45
48
|
gate:
|
|
49
|
+
file: gates/approval
|
|
46
50
|
type: approval
|
|
47
51
|
condition: Code review approved, UX requirements met
|
|
48
52
|
|
|
@@ -21,6 +21,7 @@ workflow:
|
|
|
21
21
|
input: [session_file, story_context]
|
|
22
22
|
output: [failing_tests]
|
|
23
23
|
gate:
|
|
24
|
+
file: gates/tests-fail
|
|
24
25
|
type: tests_fail
|
|
25
26
|
condition: All acceptance criteria have test coverage
|
|
26
27
|
tandem:
|
|
@@ -32,6 +33,7 @@ workflow:
|
|
|
32
33
|
input: [failing_tests, story_context]
|
|
33
34
|
output: [implementation, passing_tests]
|
|
34
35
|
gate:
|
|
36
|
+
file: gates/tests-pass
|
|
35
37
|
type: tests_pass
|
|
36
38
|
condition: All tests passing, no skipped tests
|
|
37
39
|
tandem:
|
|
@@ -43,6 +45,7 @@ workflow:
|
|
|
43
45
|
input: [implementation, passing_tests]
|
|
44
46
|
output: [approval]
|
|
45
47
|
gate:
|
|
48
|
+
file: gates/approval
|
|
46
49
|
type: approval
|
|
47
50
|
condition: Code review approved, no blocking issues
|
|
48
51
|
tandem:
|
|
@@ -19,6 +19,7 @@ workflow:
|
|
|
19
19
|
input: [session_file, story_context]
|
|
20
20
|
output: [failing_tests]
|
|
21
21
|
gate:
|
|
22
|
+
file: gates/tests-fail
|
|
22
23
|
type: tests_fail
|
|
23
24
|
condition: All acceptance criteria have test coverage
|
|
24
25
|
|
|
@@ -27,6 +28,7 @@ workflow:
|
|
|
27
28
|
input: [failing_tests, story_context]
|
|
28
29
|
output: [implementation, passing_tests]
|
|
29
30
|
gate:
|
|
31
|
+
file: gates/tests-pass
|
|
30
32
|
type: tests_pass
|
|
31
33
|
condition: All tests passing, no skipped tests
|
|
32
34
|
|
|
@@ -35,6 +37,7 @@ workflow:
|
|
|
35
37
|
input: [implementation, passing_tests]
|
|
36
38
|
output: [approval]
|
|
37
39
|
gate:
|
|
40
|
+
file: gates/approval
|
|
38
41
|
type: approval
|
|
39
42
|
condition: Code review approved, no blocking issues
|
|
40
43
|
|
|
@@ -19,6 +19,7 @@ workflow:
|
|
|
19
19
|
input: [session_file]
|
|
20
20
|
output: [implementation]
|
|
21
21
|
gate:
|
|
22
|
+
file: gates/tests-pass
|
|
22
23
|
type: tests_pass
|
|
23
24
|
condition: Existing tests still pass
|
|
24
25
|
|
|
@@ -27,6 +28,7 @@ workflow:
|
|
|
27
28
|
input: [implementation]
|
|
28
29
|
output: [approval]
|
|
29
30
|
gate:
|
|
31
|
+
file: gates/approval
|
|
30
32
|
type: approval
|
|
31
33
|
|
|
32
34
|
- name: finish
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
@@ -64,12 +64,12 @@ def start(project_dir, dry_run):
|
|
|
64
64
|
|
|
65
65
|
running, pid, port = is_already_running(project_dir)
|
|
66
66
|
if running:
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
)
|
|
71
|
-
click.echo("
|
|
72
|
-
|
|
67
|
+
# Idempotent: WheelHub already up, just exec Claude with OTEL env
|
|
68
|
+
click.echo(f"BikeRack already running (PID {pid}, port {port})")
|
|
69
|
+
otel_env = build_otel_env(port)
|
|
70
|
+
click.echo(f"Dashboard: http://localhost:{port}/bikerack")
|
|
71
|
+
click.echo("Starting Claude CLI...")
|
|
72
|
+
exec_claude(otel_env, project_dir)
|
|
73
73
|
|
|
74
74
|
click.echo("Starting BikeRack mode...")
|
|
75
75
|
try:
|
|
@@ -116,11 +116,10 @@ def stop(project_dir, dry_run):
|
|
|
116
116
|
|
|
117
117
|
result = stop_bikerack(project_dir)
|
|
118
118
|
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
sys.exit(1)
|
|
119
|
+
click.echo(result["message"])
|
|
120
|
+
if not result["success"]:
|
|
121
|
+
# "Not running" is not an error for stop — idempotent
|
|
122
|
+
sys.exit(0)
|
|
124
123
|
|
|
125
124
|
|
|
126
125
|
@bikerack.command()
|
|
@@ -0,0 +1,218 @@
|
|
|
1
|
+
"""DebugPanel — Context usage and token stats for BikeRack TUI.
|
|
2
|
+
|
|
3
|
+
Story 103-17: Port of the React DebugPanel. Subscribes to /ws/context
|
|
4
|
+
and /ws/token-stats, renders context usage (tokens, percent, tier) and
|
|
5
|
+
token consumption stats (input, output, cache, cost).
|
|
6
|
+
"""
|
|
7
|
+
|
|
8
|
+
from __future__ import annotations
|
|
9
|
+
|
|
10
|
+
from typing import Any
|
|
11
|
+
|
|
12
|
+
from rich.console import Group
|
|
13
|
+
from rich.table import Table
|
|
14
|
+
from rich.text import Text
|
|
15
|
+
|
|
16
|
+
from pennyfarthing_scripts.bikerack.base_panel import PANEL_ICONS, BasePanel
|
|
17
|
+
|
|
18
|
+
# Tier → Rich style mapping
|
|
19
|
+
_TIER_STYLES: dict[str, str] = {
|
|
20
|
+
"FULL": "bold green",
|
|
21
|
+
"REFRESH": "bold yellow",
|
|
22
|
+
"HANDOFF": "bold cyan",
|
|
23
|
+
"MINIMAL": "bold red",
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
|
|
27
|
+
def _safe_int(value: Any) -> int | None:
|
|
28
|
+
"""Safely convert a value to int, returning None on failure."""
|
|
29
|
+
if value is None:
|
|
30
|
+
return None
|
|
31
|
+
try:
|
|
32
|
+
return int(value)
|
|
33
|
+
except (ValueError, TypeError):
|
|
34
|
+
return None
|
|
35
|
+
|
|
36
|
+
|
|
37
|
+
def _safe_float(value: Any) -> float | None:
|
|
38
|
+
"""Safely convert a value to float, returning None on failure."""
|
|
39
|
+
if value is None:
|
|
40
|
+
return None
|
|
41
|
+
try:
|
|
42
|
+
return float(value)
|
|
43
|
+
except (ValueError, TypeError):
|
|
44
|
+
return None
|
|
45
|
+
|
|
46
|
+
|
|
47
|
+
def _format_tokens(value: Any) -> str:
|
|
48
|
+
"""Format a token count with comma separators."""
|
|
49
|
+
n = _safe_int(value)
|
|
50
|
+
if n is None:
|
|
51
|
+
return "—"
|
|
52
|
+
return f"{n:,}"
|
|
53
|
+
|
|
54
|
+
|
|
55
|
+
class DebugPanel(BasePanel):
|
|
56
|
+
"""Context usage and token stats panel.
|
|
57
|
+
|
|
58
|
+
Subscribes to both ``context`` and ``token-stats`` WebSocket channels.
|
|
59
|
+
Renders context usage (tokens, percentage, tier) and token consumption
|
|
60
|
+
stats (input, output, cache read/write, cost).
|
|
61
|
+
"""
|
|
62
|
+
|
|
63
|
+
channel: str = "context"
|
|
64
|
+
panel_name: str = "Debug"
|
|
65
|
+
icon: str = PANEL_ICONS["debug"][0]
|
|
66
|
+
|
|
67
|
+
def __init__(self, client: Any = None, **kwargs: Any) -> None:
|
|
68
|
+
super().__init__(client=client, **kwargs)
|
|
69
|
+
self._context_data: dict[str, Any] | None = None
|
|
70
|
+
self._token_stats: dict[str, Any] | None = None
|
|
71
|
+
|
|
72
|
+
def on_mount(self) -> None:
|
|
73
|
+
"""Subscribe to both context and token-stats channels."""
|
|
74
|
+
self._mounted = True
|
|
75
|
+
if self._client is not None:
|
|
76
|
+
self._client.subscribe("context", self._handle_context_message)
|
|
77
|
+
self._client.subscribe("token-stats", self._handle_token_stats_message)
|
|
78
|
+
|
|
79
|
+
def _handle_context_message(self, message: dict[str, Any] | None) -> None:
|
|
80
|
+
"""Handle incoming context channel message."""
|
|
81
|
+
if message is None:
|
|
82
|
+
return
|
|
83
|
+
ctx = message.get("context")
|
|
84
|
+
if isinstance(ctx, dict):
|
|
85
|
+
self._context_data = ctx
|
|
86
|
+
else:
|
|
87
|
+
self._context_data = {}
|
|
88
|
+
self._rerender()
|
|
89
|
+
|
|
90
|
+
def _handle_token_stats_message(self, message: dict[str, Any] | None) -> None:
|
|
91
|
+
"""Handle incoming token-stats channel message."""
|
|
92
|
+
if message is None:
|
|
93
|
+
return
|
|
94
|
+
self._token_stats = message
|
|
95
|
+
self._rerender()
|
|
96
|
+
|
|
97
|
+
def _rerender(self) -> None:
|
|
98
|
+
"""Re-render with the latest data from both channels."""
|
|
99
|
+
rendered = self.render_panel(self._context_data or {})
|
|
100
|
+
try:
|
|
101
|
+
self.update(rendered)
|
|
102
|
+
except Exception:
|
|
103
|
+
pass
|
|
104
|
+
|
|
105
|
+
def render_panel(self, payload: dict[str, Any]) -> Any:
|
|
106
|
+
"""Render combined context usage and token stats."""
|
|
107
|
+
parts: list[Any] = []
|
|
108
|
+
|
|
109
|
+
# --- Context Usage Section ---
|
|
110
|
+
ctx = self._context_data
|
|
111
|
+
if ctx:
|
|
112
|
+
parts.append(_render_context(ctx))
|
|
113
|
+
elif not self._token_stats:
|
|
114
|
+
return Text("No context data", style="dim italic")
|
|
115
|
+
|
|
116
|
+
# --- Token Stats Section ---
|
|
117
|
+
if self._token_stats:
|
|
118
|
+
if parts:
|
|
119
|
+
parts.append(Text("")) # spacer
|
|
120
|
+
parts.append(_render_token_stats(self._token_stats))
|
|
121
|
+
|
|
122
|
+
if not parts:
|
|
123
|
+
return Text("No context data", style="dim italic")
|
|
124
|
+
|
|
125
|
+
return Group(*parts)
|
|
126
|
+
|
|
127
|
+
|
|
128
|
+
def _render_context(ctx: dict[str, Any]) -> Any:
|
|
129
|
+
"""Render context usage section."""
|
|
130
|
+
parts: list[Any] = []
|
|
131
|
+
|
|
132
|
+
# Tier badge
|
|
133
|
+
tier = ctx.get("tier")
|
|
134
|
+
if tier and isinstance(tier, str):
|
|
135
|
+
style = _TIER_STYLES.get(tier, "bold")
|
|
136
|
+
tier_text = Text()
|
|
137
|
+
tier_text.append(tier, style=style)
|
|
138
|
+
savings = _tier_savings(tier)
|
|
139
|
+
if savings > 0:
|
|
140
|
+
tier_text.append(f" {savings}% savings", style="dim")
|
|
141
|
+
parts.append(tier_text)
|
|
142
|
+
|
|
143
|
+
# Token usage
|
|
144
|
+
tokens = _safe_int(ctx.get("tokens"))
|
|
145
|
+
percent = _safe_int(ctx.get("percent"))
|
|
146
|
+
baseline = _safe_int(ctx.get("baseline"))
|
|
147
|
+
available = _safe_int(ctx.get("available"))
|
|
148
|
+
|
|
149
|
+
if tokens is not None:
|
|
150
|
+
usage_text = Text()
|
|
151
|
+
usage_text.append(f"{tokens:,}", style="bold")
|
|
152
|
+
if baseline is not None and available is not None:
|
|
153
|
+
total = baseline + available
|
|
154
|
+
usage_text.append(f" / {total:,}")
|
|
155
|
+
usage_text.append(" tokens")
|
|
156
|
+
if percent is not None:
|
|
157
|
+
usage_text.append(f" ({percent}%)")
|
|
158
|
+
parts.append(usage_text)
|
|
159
|
+
|
|
160
|
+
# Breakdown: baseline / conversation / available
|
|
161
|
+
if baseline is not None:
|
|
162
|
+
breakdown = Table(show_header=False, show_edge=False, pad_edge=False, box=None)
|
|
163
|
+
breakdown.add_column("Label", style="dim")
|
|
164
|
+
breakdown.add_column("Value", justify="right")
|
|
165
|
+
breakdown.add_row("System Prompt", _format_tokens(baseline))
|
|
166
|
+
usable = _safe_int(ctx.get("usableTokens"))
|
|
167
|
+
if usable is not None:
|
|
168
|
+
breakdown.add_row("Conversation", _format_tokens(usable))
|
|
169
|
+
if available is not None:
|
|
170
|
+
breakdown.add_row("Available", _format_tokens(available))
|
|
171
|
+
parts.append(breakdown)
|
|
172
|
+
|
|
173
|
+
if not parts:
|
|
174
|
+
return Text("No context data", style="dim italic")
|
|
175
|
+
|
|
176
|
+
return Group(*parts)
|
|
177
|
+
|
|
178
|
+
|
|
179
|
+
def _render_token_stats(stats: dict[str, Any]) -> Any:
|
|
180
|
+
"""Render token stats section."""
|
|
181
|
+
table = Table(show_header=False, show_edge=False, pad_edge=False, box=None)
|
|
182
|
+
table.add_column("Stat", style="dim")
|
|
183
|
+
table.add_column("Value", justify="right")
|
|
184
|
+
|
|
185
|
+
rows: list[tuple[str, str]] = []
|
|
186
|
+
|
|
187
|
+
input_t = _safe_int(stats.get("inputTokens"))
|
|
188
|
+
if input_t is not None:
|
|
189
|
+
rows.append(("Input", f"{input_t:,}"))
|
|
190
|
+
|
|
191
|
+
output_t = _safe_int(stats.get("outputTokens"))
|
|
192
|
+
if output_t is not None:
|
|
193
|
+
rows.append(("Output", f"{output_t:,}"))
|
|
194
|
+
|
|
195
|
+
cache_read = _safe_int(stats.get("cacheReadTokens"))
|
|
196
|
+
if cache_read is not None:
|
|
197
|
+
rows.append(("Cache Read", f"{cache_read:,}"))
|
|
198
|
+
|
|
199
|
+
cache_write = _safe_int(stats.get("cacheCreationTokens"))
|
|
200
|
+
if cache_write is not None:
|
|
201
|
+
rows.append(("Cache Write", f"{cache_write:,}"))
|
|
202
|
+
|
|
203
|
+
cost = _safe_float(stats.get("totalCostUsd"))
|
|
204
|
+
if cost is not None and cost > 0:
|
|
205
|
+
rows.append(("Cost", f"${cost:.4f}"))
|
|
206
|
+
|
|
207
|
+
if not rows:
|
|
208
|
+
return Text("No token stats", style="dim italic")
|
|
209
|
+
|
|
210
|
+
for label, value in rows:
|
|
211
|
+
table.add_row(label, value)
|
|
212
|
+
|
|
213
|
+
return table
|
|
214
|
+
|
|
215
|
+
|
|
216
|
+
def _tier_savings(tier: str) -> int:
|
|
217
|
+
"""Calculate token savings percentage for a tier vs FULL."""
|
|
218
|
+
return {"FULL": 0, "REFRESH": 85, "HANDOFF": 82, "MINIMAL": 95}.get(tier, 0)
|