gsd-pi 2.44.0-dev.848dd4c → 2.44.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 +12 -30
- package/dist/resources/extensions/gsd/auto-start.js +0 -10
- package/dist/resources/extensions/gsd/commands/handlers/workflow.js +0 -5
- package/dist/web/standalone/.next/BUILD_ID +1 -1
- package/dist/web/standalone/.next/app-path-routes-manifest.json +14 -14
- package/dist/web/standalone/.next/build-manifest.json +3 -3
- package/dist/web/standalone/.next/prerender-manifest.json +3 -3
- package/dist/web/standalone/.next/required-server-files.json +3 -3
- package/dist/web/standalone/.next/server/app/_global-error/page.js +3 -3
- package/dist/web/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.html +2 -2
- package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found/page.js +2 -2
- package/dist/web/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.rsc +3 -3
- package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +3 -3
- package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +3 -3
- package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/api/boot/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/boot/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js +2 -2
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/browse-directories/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/browse-directories/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/captures/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/captures/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/cleanup/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/cleanup/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/dev-mode/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/dev-mode/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/doctor/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/doctor/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/export-data/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/export-data/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/files/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/files/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/forensics/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/forensics/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/git/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/git/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/history/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/history/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/hooks/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/hooks/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/inspect/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/inspect/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/knowledge/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/knowledge/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/live-state/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/live-state/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/onboarding/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/onboarding/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/preferences/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/preferences/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/projects/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/projects/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/recovery/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/recovery/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/remote-questions/route.js +5 -5
- package/dist/web/standalone/.next/server/app/api/remote-questions/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/session/browser/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/session/browser/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/session/command/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/session/command/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/session/events/route.js +2 -2
- package/dist/web/standalone/.next/server/app/api/session/events/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/session/manage/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/session/manage/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/settings-data/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/settings-data/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/shutdown/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/shutdown/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/skill-health/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/skill-health/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/steer/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/steer/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/switch-root/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/switch-root/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/input/route.js +2 -2
- package/dist/web/standalone/.next/server/app/api/terminal/input/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/resize/route.js +2 -2
- package/dist/web/standalone/.next/server/app/api/terminal/resize/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js +2 -2
- package/dist/web/standalone/.next/server/app/api/terminal/sessions/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js +4 -4
- package/dist/web/standalone/.next/server/app/api/terminal/stream/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/upload/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/upload/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/undo/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/undo/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/update/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/update/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/visualizer/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/visualizer/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/index.html +1 -1
- package/dist/web/standalone/.next/server/app/index.rsc +4 -4
- package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +2 -2
- package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +4 -4
- package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +3 -3
- package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/page.js +2 -2
- package/dist/web/standalone/.next/server/app/page_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app-paths-manifest.json +14 -14
- package/dist/web/standalone/.next/server/chunks/229.js +1 -1
- package/dist/web/standalone/.next/server/chunks/471.js +3 -3
- package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
- package/dist/web/standalone/.next/server/middleware.js +2 -2
- package/dist/web/standalone/.next/server/next-font-manifest.js +1 -1
- package/dist/web/standalone/.next/server/next-font-manifest.json +1 -1
- package/dist/web/standalone/.next/server/pages/404.html +1 -1
- package/dist/web/standalone/.next/server/pages/500.html +2 -2
- package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
- package/dist/web/standalone/.next/static/chunks/app/_not-found/{page-f2a7482d42a5614b.js → page-2f24283c162b6ab3.js} +1 -1
- package/dist/web/standalone/.next/static/chunks/app/{layout-a16c7a7ecdf0c2cf.js → layout-9ecfd95f343793f0.js} +1 -1
- package/dist/web/standalone/.next/static/chunks/app/page-7e9530a7122506c5.js +1 -0
- package/dist/web/standalone/.next/static/chunks/main-app-d3d4c336195465f9.js +1 -0
- package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/global-error-ab5a8926e07ec673.js +1 -0
- package/dist/web/standalone/node_modules/node-pty/build/Makefile +2 -2
- package/dist/web/standalone/node_modules/node-pty/build/Release/pty.node +0 -0
- package/dist/web/standalone/node_modules/node-pty/build/pty.target.mk +14 -14
- package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api.target.mk +14 -14
- package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api_except.target.mk +14 -14
- package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api_maybe.target.mk +14 -14
- package/dist/web/standalone/server.js +1 -1
- package/package.json +1 -1
- package/packages/pi-coding-agent/dist/core/auth-storage.test.js +8 -6
- package/packages/pi-coding-agent/dist/core/auth-storage.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/runner.test.js +26 -24
- package/packages/pi-coding-agent/dist/core/extensions/runner.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/fs-utils.test.js +48 -29
- package/packages/pi-coding-agent/dist/core/fs-utils.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/resolve-config-value.test.js +44 -34
- package/packages/pi-coding-agent/dist/core/resolve-config-value.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/session-manager.test.js +34 -30
- package/packages/pi-coding-agent/dist/core/session-manager.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/tools/edit-diff.test.js +12 -10
- package/packages/pi-coding-agent/dist/core/tools/edit-diff.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/resources/extensions/memory/storage.test.js +47 -43
- package/packages/pi-coding-agent/dist/resources/extensions/memory/storage.test.js.map +1 -1
- package/packages/pi-coding-agent/src/core/auth-storage.test.ts +7 -7
- package/packages/pi-coding-agent/src/core/extensions/runner.test.ts +26 -26
- package/packages/pi-coding-agent/src/core/fs-utils.test.ts +43 -31
- package/packages/pi-coding-agent/src/core/resolve-config-value.test.ts +45 -40
- package/packages/pi-coding-agent/src/core/session-manager.test.ts +33 -33
- package/packages/pi-coding-agent/src/core/tools/edit-diff.test.ts +17 -17
- package/packages/pi-coding-agent/src/resources/extensions/memory/storage.test.ts +74 -74
- package/src/resources/extensions/gsd/auto-start.ts +0 -14
- package/src/resources/extensions/gsd/commands/handlers/workflow.ts +0 -8
- package/src/resources/extensions/gsd/tests/all-milestones-complete-merge.test.ts +99 -99
- package/src/resources/extensions/gsd/tests/auto-lock-creation.test.ts +16 -14
- package/src/resources/extensions/gsd/tests/auto-paused-session-validation.test.ts +57 -43
- package/src/resources/extensions/gsd/tests/auto-preflight.test.ts +13 -11
- package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +523 -465
- package/src/resources/extensions/gsd/tests/auto-secrets-gate.test.ts +75 -73
- package/src/resources/extensions/gsd/tests/auto-start-needs-discussion.test.ts +56 -34
- package/src/resources/extensions/gsd/tests/auto-worktree-milestone-merge.test.ts +656 -533
- package/src/resources/extensions/gsd/tests/auto-worktree.test.ts +143 -165
- package/src/resources/extensions/gsd/tests/cache-staleness-regression.test.ts +52 -29
- package/src/resources/extensions/gsd/tests/captures.test.ts +176 -148
- package/src/resources/extensions/gsd/tests/claude-import-tui.test.ts +33 -32
- package/src/resources/extensions/gsd/tests/collect-from-manifest.test.ts +143 -141
- package/src/resources/extensions/gsd/tests/commands-inspect-open-db.test.ts +25 -25
- package/src/resources/extensions/gsd/tests/commands-logs.test.ts +81 -81
- package/src/resources/extensions/gsd/tests/complete-milestone.test.ts +59 -38
- package/src/resources/extensions/gsd/tests/complete-slice.test.ts +263 -228
- package/src/resources/extensions/gsd/tests/complete-task.test.ts +302 -250
- package/src/resources/extensions/gsd/tests/context-store.test.ts +367 -354
- package/src/resources/extensions/gsd/tests/continue-here.test.ts +72 -68
- package/src/resources/extensions/gsd/tests/cost-projection.test.ts +106 -92
- package/src/resources/extensions/gsd/tests/crash-recovery.test.ts +35 -27
- package/src/resources/extensions/gsd/tests/dashboard-budget.test.ts +237 -220
- package/src/resources/extensions/gsd/tests/db-writer.test.ts +420 -390
- package/src/resources/extensions/gsd/tests/definition-loader.test.ts +92 -76
- package/src/resources/extensions/gsd/tests/derive-state-crossval.test.ts +83 -68
- package/src/resources/extensions/gsd/tests/derive-state-db.test.ts +183 -152
- package/src/resources/extensions/gsd/tests/derive-state-deps.test.ts +101 -78
- package/src/resources/extensions/gsd/tests/derive-state.test.ts +227 -192
- package/src/resources/extensions/gsd/tests/detection.test.ts +278 -232
- package/src/resources/extensions/gsd/tests/dev-engine-wrapper.test.ts +34 -30
- package/src/resources/extensions/gsd/tests/dispatch-guard.test.ts +180 -164
- package/src/resources/extensions/gsd/tests/dispatch-missing-task-plans.test.ts +49 -43
- package/src/resources/extensions/gsd/tests/dispatch-uat-last-completed.test.ts +32 -28
- package/src/resources/extensions/gsd/tests/doctor-completion-deferral.test.ts +29 -27
- package/src/resources/extensions/gsd/tests/doctor-delimiter-fix.test.ts +38 -34
- package/src/resources/extensions/gsd/tests/doctor-enhancements.test.ts +75 -54
- package/src/resources/extensions/gsd/tests/doctor-environment-worktree.test.ts +32 -21
- package/src/resources/extensions/gsd/tests/doctor-environment.test.ts +97 -72
- package/src/resources/extensions/gsd/tests/doctor-fixlevel.test.ts +44 -38
- package/src/resources/extensions/gsd/tests/doctor-git.test.ts +145 -104
- package/src/resources/extensions/gsd/tests/doctor-proactive.test.ts +106 -84
- package/src/resources/extensions/gsd/tests/doctor-roadmap-summary-atomicity.test.ts +60 -54
- package/src/resources/extensions/gsd/tests/doctor-runtime.test.ts +93 -72
- package/src/resources/extensions/gsd/tests/doctor.test.ts +134 -104
- package/src/resources/extensions/gsd/tests/ensure-db-open.test.ts +131 -123
- package/src/resources/extensions/gsd/tests/exit-command.test.ts +24 -20
- package/src/resources/extensions/gsd/tests/feature-branch-lifecycle-integration.test.ts +57 -48
- package/src/resources/extensions/gsd/tests/files-loadfile-eisdir.test.ts +7 -5
- package/src/resources/extensions/gsd/tests/flag-file-db.test.ts +42 -30
- package/src/resources/extensions/gsd/tests/freeform-decisions.test.ts +206 -198
- package/src/resources/extensions/gsd/tests/git-locale.test.ts +27 -13
- package/src/resources/extensions/gsd/tests/git-service.test.ts +388 -285
- package/src/resources/extensions/gsd/tests/gitignore-tracked-gsd.test.ts +39 -31
- package/src/resources/extensions/gsd/tests/graph-operations.test.ts +69 -63
- package/src/resources/extensions/gsd/tests/gsd-db.test.ts +264 -255
- package/src/resources/extensions/gsd/tests/gsd-inspect.test.ts +119 -108
- package/src/resources/extensions/gsd/tests/gsd-recover.test.ts +103 -81
- package/src/resources/extensions/gsd/tests/gsd-tools.test.ts +262 -229
- package/src/resources/extensions/gsd/tests/headless-answers.test.ts +13 -13
- package/src/resources/extensions/gsd/tests/health-widget.test.ts +37 -29
- package/src/resources/extensions/gsd/tests/idle-recovery.test.ts +102 -81
- package/src/resources/extensions/gsd/tests/init-wizard.test.ts +18 -16
- package/src/resources/extensions/gsd/tests/integration-edge.test.ts +46 -41
- package/src/resources/extensions/gsd/tests/integration-lifecycle.test.ts +53 -42
- package/src/resources/extensions/gsd/tests/integration-mixed-milestones.test.ts +91 -75
- package/src/resources/extensions/gsd/tests/integration-proof.test.ts +18 -18
- package/src/resources/extensions/gsd/tests/markdown-renderer.test.ts +194 -150
- package/src/resources/extensions/gsd/tests/md-importer.test.ts +125 -101
- package/src/resources/extensions/gsd/tests/memory-extractor.test.ts +54 -45
- package/src/resources/extensions/gsd/tests/memory-store.test.ts +93 -80
- package/src/resources/extensions/gsd/tests/migrate-command.test.ts +66 -57
- package/src/resources/extensions/gsd/tests/migrate-hierarchy.test.ts +93 -83
- package/src/resources/extensions/gsd/tests/migrate-parser.test.ts +170 -161
- package/src/resources/extensions/gsd/tests/migrate-transformer.test.ts +141 -125
- package/src/resources/extensions/gsd/tests/migrate-validator-parsers.test.ts +131 -107
- package/src/resources/extensions/gsd/tests/migrate-writer-integration.test.ts +96 -87
- package/src/resources/extensions/gsd/tests/migrate-writer.test.ts +164 -125
- package/src/resources/extensions/gsd/tests/must-have-parser.test.ts +94 -81
- package/src/resources/extensions/gsd/tests/none-mode-gates.test.ts +36 -35
- package/src/resources/extensions/gsd/tests/overrides.test.ts +106 -99
- package/src/resources/extensions/gsd/tests/parallel-crash-recovery.test.ts +47 -40
- package/src/resources/extensions/gsd/tests/parallel-worker-monitoring.test.ts +28 -25
- package/src/resources/extensions/gsd/tests/parallel-workers-multi-milestone-e2e.test.ts +83 -66
- package/src/resources/extensions/gsd/tests/park-edge-cases.test.ts +77 -54
- package/src/resources/extensions/gsd/tests/park-milestone.test.ts +115 -68
- package/src/resources/extensions/gsd/tests/parsers.test.ts +611 -546
- package/src/resources/extensions/gsd/tests/paths.test.ts +87 -72
- package/src/resources/extensions/gsd/tests/post-unit-hooks.test.ts +117 -77
- package/src/resources/extensions/gsd/tests/prompt-db.test.ts +56 -56
- package/src/resources/extensions/gsd/tests/queue-draft-detection.test.ts +119 -93
- package/src/resources/extensions/gsd/tests/queue-order.test.ts +82 -70
- package/src/resources/extensions/gsd/tests/queue-reorder-e2e.test.ts +55 -42
- package/src/resources/extensions/gsd/tests/quick-branch-lifecycle.test.ts +73 -45
- package/src/resources/extensions/gsd/tests/reassess-prompt.test.ts +38 -28
- package/src/resources/extensions/gsd/tests/replan-slice.test.ts +80 -73
- package/src/resources/extensions/gsd/tests/repo-identity-worktree.test.ts +74 -71
- package/src/resources/extensions/gsd/tests/requirements.test.ts +75 -70
- package/src/resources/extensions/gsd/tests/retry-state-reset.test.ts +66 -44
- package/src/resources/extensions/gsd/tests/roadmap-parse-regression.test.ts +181 -114
- package/src/resources/extensions/gsd/tests/rule-registry.test.ts +65 -63
- package/src/resources/extensions/gsd/tests/run-uat.test.ts +128 -66
- package/src/resources/extensions/gsd/tests/session-lock-multipath.test.ts +25 -18
- package/src/resources/extensions/gsd/tests/session-lock-regression.test.ts +44 -37
- package/src/resources/extensions/gsd/tests/shared-wal.test.ts +26 -19
- package/src/resources/extensions/gsd/tests/stalled-tool-recovery.test.ts +8 -6
- package/src/resources/extensions/gsd/tests/symlink-numbered-variants.test.ts +28 -22
- package/src/resources/extensions/gsd/tests/token-savings.test.ts +56 -54
- package/src/resources/extensions/gsd/tests/tool-call-loop-guard.test.ts +25 -23
- package/src/resources/extensions/gsd/tests/tool-naming.test.ts +11 -9
- package/src/resources/extensions/gsd/tests/unique-milestone-ids.test.ts +82 -66
- package/src/resources/extensions/gsd/tests/unit-runtime.test.ts +47 -46
- package/src/resources/extensions/gsd/tests/visualizer-critical-path.test.ts +22 -20
- package/src/resources/extensions/gsd/tests/visualizer-data.test.ts +86 -84
- package/src/resources/extensions/gsd/tests/visualizer-overlay.test.ts +43 -41
- package/src/resources/extensions/gsd/tests/visualizer-views.test.ts +96 -94
- package/src/resources/extensions/gsd/tests/windows-path-normalization.test.ts +13 -11
- package/src/resources/extensions/gsd/tests/worker-registry.test.ts +29 -27
- package/src/resources/extensions/gsd/tests/workflow-templates.test.ts +52 -50
- package/src/resources/extensions/gsd/tests/worktree-bugfix.test.ts +13 -10
- package/src/resources/extensions/gsd/tests/worktree-db-integration.test.ts +18 -14
- package/src/resources/extensions/gsd/tests/worktree-db.test.ts +39 -38
- package/src/resources/extensions/gsd/tests/worktree-e2e.test.ts +21 -17
- package/src/resources/extensions/gsd/tests/worktree-health.test.ts +30 -25
- package/src/resources/extensions/gsd/tests/worktree-integration.test.ts +37 -30
- package/src/resources/extensions/gsd/tests/worktree-symlink-removal.test.ts +22 -15
- package/src/resources/extensions/gsd/tests/worktree-sync-milestones.test.ts +66 -59
- package/src/resources/extensions/gsd/tests/worktree.test.ts +50 -44
- package/dist/web/standalone/.next/static/chunks/app/page-b9367c5ae13b99c6.js +0 -1
- package/dist/web/standalone/.next/static/chunks/main-app-fdab67f7802d7832.js +0 -1
- package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/global-error-459824ffb8c323dd.js +0 -1
- package/src/resources/extensions/gsd/tests/quick-auto-guard.test.ts +0 -100
- package/src/resources/extensions/gsd/tests/sqlite-unavailable-gate.test.ts +0 -63
- /package/dist/web/standalone/.next/static/{-zps1Q9mQmioAKLcQiCr8 → mgkxN0mGP6gSUmGPEzbk_}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{-zps1Q9mQmioAKLcQiCr8 → mgkxN0mGP6gSUmGPEzbk_}/_ssgManifest.js +0 -0
|
@@ -1,5 +1,3 @@
|
|
|
1
|
-
import { describe, test } from 'node:test';
|
|
2
|
-
import assert from 'node:assert/strict';
|
|
3
1
|
/**
|
|
4
2
|
* Tests for dashboard budget indicator rendering.
|
|
5
3
|
*
|
|
@@ -20,6 +18,10 @@ import {
|
|
|
20
18
|
getProjectTotals,
|
|
21
19
|
formatTokenCount,
|
|
22
20
|
} from "../metrics.js";
|
|
21
|
+
import { createTestContext } from './test-helpers.ts';
|
|
22
|
+
|
|
23
|
+
const { assertEq, assertTrue, assertMatch, assertNoMatch, report } = createTestContext();
|
|
24
|
+
|
|
23
25
|
// ─── Test helpers ─────────────────────────────────────────────────────────────
|
|
24
26
|
|
|
25
27
|
function makeUnit(overrides: Partial<UnitMetrics> = {}): UnitMetrics {
|
|
@@ -100,230 +102,245 @@ function renderModelContextWindow(units: UnitMetrics[], modelName: string): stri
|
|
|
100
102
|
|
|
101
103
|
// ─── Completed section: budget indicators ─────────────────────────────────────
|
|
102
104
|
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
{
|
|
118
|
-
// Unit with truncation only — no wrap-up marker
|
|
119
|
-
const ledgerUnits = [
|
|
120
|
-
makeUnit({ type: "execute-task", id: "M001/S01/T01", truncationSections: 5, continueHereFired: false }),
|
|
121
|
-
];
|
|
122
|
-
const markers = renderCompletedBudgetMarkers(
|
|
123
|
-
{ type: "execute-task", id: "M001/S01/T01" },
|
|
124
|
-
ledgerUnits,
|
|
125
|
-
);
|
|
126
|
-
assert.match(markers, /▼5/, "completed: shows ▼5 truncation only");
|
|
127
|
-
assert.doesNotMatch(markers, /wrap-up/, "completed: no wrap-up when continueHereFired=false");
|
|
128
|
-
}
|
|
105
|
+
console.log("\n=== Completed section: truncation + continue-here markers ===");
|
|
106
|
+
|
|
107
|
+
{
|
|
108
|
+
// Unit with truncation and continue-here — both markers appear
|
|
109
|
+
const ledgerUnits = [
|
|
110
|
+
makeUnit({ type: "execute-task", id: "M001/S01/T01", truncationSections: 3, continueHereFired: true }),
|
|
111
|
+
];
|
|
112
|
+
const markers = renderCompletedBudgetMarkers(
|
|
113
|
+
{ type: "execute-task", id: "M001/S01/T01" },
|
|
114
|
+
ledgerUnits,
|
|
115
|
+
);
|
|
116
|
+
assertMatch(markers, /▼3/, "completed: shows ▼3 for 3 truncation sections");
|
|
117
|
+
assertMatch(markers, /→ wrap-up/, "completed: shows → wrap-up when continueHereFired");
|
|
118
|
+
}
|
|
129
119
|
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
120
|
+
{
|
|
121
|
+
// Unit with truncation only — no wrap-up marker
|
|
122
|
+
const ledgerUnits = [
|
|
123
|
+
makeUnit({ type: "execute-task", id: "M001/S01/T01", truncationSections: 5, continueHereFired: false }),
|
|
124
|
+
];
|
|
125
|
+
const markers = renderCompletedBudgetMarkers(
|
|
126
|
+
{ type: "execute-task", id: "M001/S01/T01" },
|
|
127
|
+
ledgerUnits,
|
|
128
|
+
);
|
|
129
|
+
assertMatch(markers, /▼5/, "completed: shows ▼5 truncation only");
|
|
130
|
+
assertNoMatch(markers, /wrap-up/, "completed: no wrap-up when continueHereFired=false");
|
|
131
|
+
}
|
|
142
132
|
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
});
|
|
156
|
-
|
|
157
|
-
{
|
|
158
|
-
// Empty ledger — no crash, no markers
|
|
159
|
-
const markers = renderCompletedBudgetMarkers(
|
|
160
|
-
{ type: "execute-task", id: "M001/S01/T01" },
|
|
161
|
-
[],
|
|
162
|
-
);
|
|
163
|
-
assert.deepStrictEqual(markers, "", "empty ledger: empty markers");
|
|
164
|
-
}
|
|
133
|
+
{
|
|
134
|
+
// Unit with continue-here only — no truncation marker
|
|
135
|
+
const ledgerUnits = [
|
|
136
|
+
makeUnit({ type: "execute-task", id: "M001/S01/T01", truncationSections: 0, continueHereFired: true }),
|
|
137
|
+
];
|
|
138
|
+
const markers = renderCompletedBudgetMarkers(
|
|
139
|
+
{ type: "execute-task", id: "M001/S01/T01" },
|
|
140
|
+
ledgerUnits,
|
|
141
|
+
);
|
|
142
|
+
assertNoMatch(markers, /▼/, "completed: no ▼ when truncationSections=0");
|
|
143
|
+
assertMatch(markers, /→ wrap-up/, "completed: shows → wrap-up");
|
|
144
|
+
}
|
|
165
145
|
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
test('Completed section: retry handling', () => {
|
|
169
|
-
// Two ledger entries for same unit (retry) — last entry wins
|
|
170
|
-
const ledgerUnits = [
|
|
171
|
-
makeUnit({ type: "execute-task", id: "M001/S01/T01", truncationSections: 1 }),
|
|
172
|
-
makeUnit({ type: "execute-task", id: "M001/S01/T01", truncationSections: 7 }),
|
|
173
|
-
];
|
|
174
|
-
const markers = renderCompletedBudgetMarkers(
|
|
175
|
-
{ type: "execute-task", id: "M001/S01/T01" },
|
|
176
|
-
ledgerUnits,
|
|
177
|
-
);
|
|
178
|
-
assert.match(markers, /▼7/, "retry: last entry's truncation count (7) wins over first (1)");
|
|
179
|
-
assert.doesNotMatch(markers, /▼1/, "retry: first entry's count (1) is not shown");
|
|
180
|
-
});
|
|
181
|
-
|
|
182
|
-
// ─── By Model section: context window display ─────────────────────────────────
|
|
183
|
-
|
|
184
|
-
test('By Model section: context window', () => {
|
|
185
|
-
// Model with context window — shows formatted token count
|
|
186
|
-
const units = [
|
|
187
|
-
makeUnit({ model: "claude-sonnet-4-20250514", contextWindowTokens: 200000 }),
|
|
188
|
-
];
|
|
189
|
-
const label = renderModelContextWindow(units, "claude-sonnet-4-20250514");
|
|
190
|
-
assert.deepStrictEqual(label, "[200.0k]", "by model: shows [200.0k] for 200000 context window");
|
|
191
|
-
});
|
|
192
|
-
|
|
193
|
-
{
|
|
194
|
-
// Model without context window — no label
|
|
195
|
-
const units = [
|
|
196
|
-
makeUnit({ model: "claude-sonnet-4-20250514" }),
|
|
197
|
-
];
|
|
198
|
-
const label = renderModelContextWindow(units, "claude-sonnet-4-20250514");
|
|
199
|
-
assert.deepStrictEqual(label, null, "by model: null when no contextWindowTokens");
|
|
200
|
-
}
|
|
146
|
+
// ─── Completed section: missing ledger match ──────────────────────────────────
|
|
201
147
|
|
|
202
|
-
|
|
203
|
-
// Multiple models — each gets its own context window
|
|
204
|
-
const units = [
|
|
205
|
-
makeUnit({ model: "claude-sonnet-4-20250514", contextWindowTokens: 200000, cost: 0.05 }),
|
|
206
|
-
makeUnit({ model: "claude-opus-4-20250514", contextWindowTokens: 200000, cost: 0.30 }),
|
|
207
|
-
];
|
|
208
|
-
const sonnetLabel = renderModelContextWindow(units, "claude-sonnet-4-20250514");
|
|
209
|
-
const opusLabel = renderModelContextWindow(units, "claude-opus-4-20250514");
|
|
210
|
-
assert.deepStrictEqual(sonnetLabel, "[200.0k]", "by model multi: sonnet has context window");
|
|
211
|
-
assert.deepStrictEqual(opusLabel, "[200.0k]", "by model multi: opus has context window");
|
|
212
|
-
}
|
|
148
|
+
console.log("\n=== Completed section: missing ledger match ===");
|
|
213
149
|
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
// The guard `models.length >= 1` (changed from > 1) means this section now renders
|
|
226
|
-
assert.ok(models.length >= 1, "single model: passes >= 1 guard (section will render)");
|
|
227
|
-
});
|
|
228
|
-
|
|
229
|
-
// ─── Cost & Usage: aggregate budget line ──────────────────────────────────────
|
|
230
|
-
|
|
231
|
-
test('Cost & Usage: aggregate budget line', () => {
|
|
232
|
-
// Units with truncation and continue-here — both stats appear
|
|
233
|
-
const units = [
|
|
234
|
-
makeUnit({ truncationSections: 3, continueHereFired: true }),
|
|
235
|
-
makeUnit({ truncationSections: 2, continueHereFired: false }),
|
|
236
|
-
makeUnit({ truncationSections: 1, continueHereFired: true }),
|
|
237
|
-
];
|
|
238
|
-
const line = renderCostBudgetLine(units);
|
|
239
|
-
assert.ok(line !== null, "cost budget: line rendered when budget data exists");
|
|
240
|
-
assert.match(line!, /6 sections truncated/, "cost budget: shows total truncation count (3+2+1=6)");
|
|
241
|
-
assert.match(line!, /2 continue-here fired/, "cost budget: shows continue-here count");
|
|
242
|
-
});
|
|
243
|
-
|
|
244
|
-
{
|
|
245
|
-
// Only truncation, no continue-here
|
|
246
|
-
const units = [
|
|
247
|
-
makeUnit({ truncationSections: 4, continueHereFired: false }),
|
|
248
|
-
];
|
|
249
|
-
const line = renderCostBudgetLine(units);
|
|
250
|
-
assert.ok(line !== null, "cost budget truncation-only: line rendered");
|
|
251
|
-
assert.match(line!, /4 sections truncated/, "cost budget truncation-only: shows count");
|
|
252
|
-
assert.doesNotMatch(line!, /continue-here/, "cost budget truncation-only: no continue-here text");
|
|
253
|
-
}
|
|
150
|
+
{
|
|
151
|
+
// Completed unit with no matching ledger entry — no crash, no markers
|
|
152
|
+
const ledgerUnits = [
|
|
153
|
+
makeUnit({ type: "execute-task", id: "M001/S01/T99", truncationSections: 3 }),
|
|
154
|
+
];
|
|
155
|
+
const markers = renderCompletedBudgetMarkers(
|
|
156
|
+
{ type: "execute-task", id: "M001/S01/T01" },
|
|
157
|
+
ledgerUnits,
|
|
158
|
+
);
|
|
159
|
+
assertEq(markers, "", "missing match: empty markers when no ledger entry matches");
|
|
160
|
+
}
|
|
254
161
|
|
|
255
|
-
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
]
|
|
260
|
-
|
|
261
|
-
|
|
262
|
-
|
|
263
|
-
assert.match(line!, /1 continue-here fired/, "cost budget continue-only: shows count");
|
|
264
|
-
}
|
|
162
|
+
{
|
|
163
|
+
// Empty ledger — no crash, no markers
|
|
164
|
+
const markers = renderCompletedBudgetMarkers(
|
|
165
|
+
{ type: "execute-task", id: "M001/S01/T01" },
|
|
166
|
+
[],
|
|
167
|
+
);
|
|
168
|
+
assertEq(markers, "", "empty ledger: empty markers");
|
|
169
|
+
}
|
|
265
170
|
|
|
266
|
-
|
|
267
|
-
|
|
268
|
-
|
|
269
|
-
|
|
270
|
-
|
|
271
|
-
|
|
272
|
-
|
|
273
|
-
|
|
274
|
-
|
|
275
|
-
|
|
276
|
-
|
|
277
|
-
|
|
278
|
-
|
|
279
|
-
|
|
280
|
-
|
|
281
|
-
|
|
282
|
-
|
|
283
|
-
|
|
284
|
-
|
|
285
|
-
|
|
286
|
-
|
|
287
|
-
|
|
288
|
-
|
|
289
|
-
|
|
290
|
-
|
|
291
|
-
|
|
292
|
-
|
|
293
|
-
|
|
294
|
-
|
|
295
|
-
|
|
296
|
-
|
|
297
|
-
|
|
298
|
-
|
|
299
|
-
|
|
300
|
-
|
|
301
|
-
|
|
302
|
-
|
|
303
|
-
|
|
304
|
-
|
|
305
|
-
|
|
306
|
-
|
|
307
|
-
|
|
308
|
-
|
|
309
|
-
|
|
310
|
-
|
|
311
|
-
|
|
312
|
-
|
|
313
|
-
|
|
314
|
-
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
318
|
-
|
|
319
|
-
|
|
320
|
-
|
|
321
|
-
|
|
322
|
-
|
|
323
|
-
|
|
324
|
-
|
|
325
|
-
|
|
171
|
+
// ─── Completed section: retry handling (last entry wins) ──────────────────────
|
|
172
|
+
|
|
173
|
+
console.log("\n=== Completed section: retry handling ===");
|
|
174
|
+
|
|
175
|
+
{
|
|
176
|
+
// Two ledger entries for same unit (retry) — last entry wins
|
|
177
|
+
const ledgerUnits = [
|
|
178
|
+
makeUnit({ type: "execute-task", id: "M001/S01/T01", truncationSections: 1 }),
|
|
179
|
+
makeUnit({ type: "execute-task", id: "M001/S01/T01", truncationSections: 7 }),
|
|
180
|
+
];
|
|
181
|
+
const markers = renderCompletedBudgetMarkers(
|
|
182
|
+
{ type: "execute-task", id: "M001/S01/T01" },
|
|
183
|
+
ledgerUnits,
|
|
184
|
+
);
|
|
185
|
+
assertMatch(markers, /▼7/, "retry: last entry's truncation count (7) wins over first (1)");
|
|
186
|
+
assertNoMatch(markers, /▼1/, "retry: first entry's count (1) is not shown");
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
// ─── By Model section: context window display ─────────────────────────────────
|
|
190
|
+
|
|
191
|
+
console.log("\n=== By Model section: context window ===");
|
|
192
|
+
|
|
193
|
+
{
|
|
194
|
+
// Model with context window — shows formatted token count
|
|
195
|
+
const units = [
|
|
196
|
+
makeUnit({ model: "claude-sonnet-4-20250514", contextWindowTokens: 200000 }),
|
|
197
|
+
];
|
|
198
|
+
const label = renderModelContextWindow(units, "claude-sonnet-4-20250514");
|
|
199
|
+
assertEq(label, "[200.0k]", "by model: shows [200.0k] for 200000 context window");
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
{
|
|
203
|
+
// Model without context window — no label
|
|
204
|
+
const units = [
|
|
205
|
+
makeUnit({ model: "claude-sonnet-4-20250514" }),
|
|
206
|
+
];
|
|
207
|
+
const label = renderModelContextWindow(units, "claude-sonnet-4-20250514");
|
|
208
|
+
assertEq(label, null, "by model: null when no contextWindowTokens");
|
|
209
|
+
}
|
|
210
|
+
|
|
211
|
+
{
|
|
212
|
+
// Multiple models — each gets its own context window
|
|
213
|
+
const units = [
|
|
214
|
+
makeUnit({ model: "claude-sonnet-4-20250514", contextWindowTokens: 200000, cost: 0.05 }),
|
|
215
|
+
makeUnit({ model: "claude-opus-4-20250514", contextWindowTokens: 200000, cost: 0.30 }),
|
|
216
|
+
];
|
|
217
|
+
const sonnetLabel = renderModelContextWindow(units, "claude-sonnet-4-20250514");
|
|
218
|
+
const opusLabel = renderModelContextWindow(units, "claude-opus-4-20250514");
|
|
219
|
+
assertEq(sonnetLabel, "[200.0k]", "by model multi: sonnet has context window");
|
|
220
|
+
assertEq(opusLabel, "[200.0k]", "by model multi: opus has context window");
|
|
221
|
+
}
|
|
222
|
+
|
|
223
|
+
// ─── By Model section: single model visibility ───────────────────────────────
|
|
224
|
+
|
|
225
|
+
console.log("\n=== By Model section: single model visibility ===");
|
|
226
|
+
|
|
227
|
+
{
|
|
228
|
+
// With guard changed to >= 1, single model aggregation should produce results
|
|
229
|
+
const units = [
|
|
230
|
+
makeUnit({ model: "claude-sonnet-4-20250514" }),
|
|
231
|
+
];
|
|
232
|
+
const models = aggregateByModel(units);
|
|
233
|
+
assertTrue(models.length >= 1, "single model: aggregateByModel returns >= 1 entry");
|
|
234
|
+
assertEq(models.length, 1, "single model: exactly 1 model aggregate");
|
|
235
|
+
assertEq(models[0].model, "claude-sonnet-4-20250514", "single model: correct model name");
|
|
236
|
+
// The guard `models.length >= 1` (changed from > 1) means this section now renders
|
|
237
|
+
assertTrue(models.length >= 1, "single model: passes >= 1 guard (section will render)");
|
|
238
|
+
}
|
|
239
|
+
|
|
240
|
+
// ─── Cost & Usage: aggregate budget line ──────────────────────────────────────
|
|
241
|
+
|
|
242
|
+
console.log("\n=== Cost & Usage: aggregate budget line ===");
|
|
243
|
+
|
|
244
|
+
{
|
|
245
|
+
// Units with truncation and continue-here — both stats appear
|
|
246
|
+
const units = [
|
|
247
|
+
makeUnit({ truncationSections: 3, continueHereFired: true }),
|
|
248
|
+
makeUnit({ truncationSections: 2, continueHereFired: false }),
|
|
249
|
+
makeUnit({ truncationSections: 1, continueHereFired: true }),
|
|
250
|
+
];
|
|
251
|
+
const line = renderCostBudgetLine(units);
|
|
252
|
+
assertTrue(line !== null, "cost budget: line rendered when budget data exists");
|
|
253
|
+
assertMatch(line!, /6 sections truncated/, "cost budget: shows total truncation count (3+2+1=6)");
|
|
254
|
+
assertMatch(line!, /2 continue-here fired/, "cost budget: shows continue-here count");
|
|
255
|
+
}
|
|
256
|
+
|
|
257
|
+
{
|
|
258
|
+
// Only truncation, no continue-here
|
|
259
|
+
const units = [
|
|
260
|
+
makeUnit({ truncationSections: 4, continueHereFired: false }),
|
|
261
|
+
];
|
|
262
|
+
const line = renderCostBudgetLine(units);
|
|
263
|
+
assertTrue(line !== null, "cost budget truncation-only: line rendered");
|
|
264
|
+
assertMatch(line!, /4 sections truncated/, "cost budget truncation-only: shows count");
|
|
265
|
+
assertNoMatch(line!, /continue-here/, "cost budget truncation-only: no continue-here text");
|
|
266
|
+
}
|
|
267
|
+
|
|
268
|
+
{
|
|
269
|
+
// Only continue-here, no truncation
|
|
270
|
+
const units = [
|
|
271
|
+
makeUnit({ truncationSections: 0, continueHereFired: true }),
|
|
272
|
+
];
|
|
273
|
+
const line = renderCostBudgetLine(units);
|
|
274
|
+
assertTrue(line !== null, "cost budget continue-only: line rendered");
|
|
275
|
+
assertNoMatch(line!, /truncated/, "cost budget continue-only: no truncation text");
|
|
276
|
+
assertMatch(line!, /1 continue-here fired/, "cost budget continue-only: shows count");
|
|
277
|
+
}
|
|
278
|
+
|
|
279
|
+
// ─── Backward compat: no budget fields ────────────────────────────────────────
|
|
280
|
+
|
|
281
|
+
console.log("\n=== Backward compat: no budget data ===");
|
|
282
|
+
|
|
283
|
+
{
|
|
284
|
+
// Old-format units without budget fields — no indicators anywhere
|
|
285
|
+
const oldUnits = [
|
|
286
|
+
makeUnit(), // no budget fields
|
|
287
|
+
makeUnit({ id: "M001/S01/T02" }),
|
|
288
|
+
];
|
|
289
|
+
|
|
290
|
+
// Completed section: no markers
|
|
291
|
+
const markers = renderCompletedBudgetMarkers(
|
|
292
|
+
{ type: "execute-task", id: "M001/S01/T01" },
|
|
293
|
+
oldUnits,
|
|
294
|
+
);
|
|
295
|
+
assertNoMatch(markers, /▼/, "backward compat completed: no truncation marker");
|
|
296
|
+
assertNoMatch(markers, /wrap-up/, "backward compat completed: no wrap-up marker");
|
|
297
|
+
assertEq(markers, "", "backward compat completed: empty markers string");
|
|
298
|
+
|
|
299
|
+
// By Model section: no context window label
|
|
300
|
+
const label = renderModelContextWindow(oldUnits, "claude-sonnet-4-20250514");
|
|
301
|
+
assertEq(label, null, "backward compat by-model: no context window label");
|
|
302
|
+
|
|
303
|
+
// Cost & Usage: no budget line
|
|
304
|
+
const line = renderCostBudgetLine(oldUnits);
|
|
305
|
+
assertEq(line, null, "backward compat cost: no budget summary line");
|
|
306
|
+
|
|
307
|
+
// Aggregation still works
|
|
308
|
+
const totals = getProjectTotals(oldUnits);
|
|
309
|
+
assertEq(totals.totalTruncationSections, 0, "backward compat: truncation total = 0");
|
|
310
|
+
assertEq(totals.continueHereFiredCount, 0, "backward compat: continueHere count = 0");
|
|
311
|
+
assertEq(totals.units, 2, "backward compat: unit count correct");
|
|
312
|
+
}
|
|
313
|
+
|
|
314
|
+
// ─── Edge cases ───────────────────────────────────────────────────────────────
|
|
315
|
+
|
|
316
|
+
console.log("\n=== Edge cases ===");
|
|
317
|
+
|
|
318
|
+
{
|
|
319
|
+
// formatTokenCount for context window values
|
|
320
|
+
assertEq(formatTokenCount(200000), "200.0k", "format: 200000 → 200.0k");
|
|
321
|
+
assertEq(formatTokenCount(128000), "128.0k", "format: 128000 → 128.0k");
|
|
322
|
+
assertEq(formatTokenCount(1000000), "1.00M", "format: 1000000 → 1.00M");
|
|
323
|
+
assertEq(formatTokenCount(32000), "32.0k", "format: 32000 → 32.0k");
|
|
324
|
+
}
|
|
325
|
+
|
|
326
|
+
{
|
|
327
|
+
// Completed unit key includes type — different types don't collide
|
|
328
|
+
const ledgerUnits = [
|
|
329
|
+
makeUnit({ type: "research-slice", id: "M001/S01", truncationSections: 2 }),
|
|
330
|
+
makeUnit({ type: "plan-slice", id: "M001/S01", truncationSections: 5 }),
|
|
331
|
+
];
|
|
332
|
+
const researchMarkers = renderCompletedBudgetMarkers(
|
|
333
|
+
{ type: "research-slice", id: "M001/S01" },
|
|
334
|
+
ledgerUnits,
|
|
335
|
+
);
|
|
336
|
+
const planMarkers = renderCompletedBudgetMarkers(
|
|
337
|
+
{ type: "plan-slice", id: "M001/S01" },
|
|
338
|
+
ledgerUnits,
|
|
339
|
+
);
|
|
340
|
+
assertMatch(researchMarkers, /▼2/, "type-keying: research unit gets its own truncation count");
|
|
341
|
+
assertMatch(planMarkers, /▼5/, "type-keying: plan unit gets its own truncation count");
|
|
342
|
+
}
|
|
326
343
|
|
|
327
|
-
|
|
344
|
+
// ─── Summary ──────────────────────────────────────────────────────────────────
|
|
328
345
|
|
|
329
|
-
|
|
346
|
+
report();
|