@paths.design/caws-cli 10.2.0 → 11.0.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 +125 -374
- package/dist/index.js +43 -785
- package/dist/shell/binding/resolve-binding.d.ts +4 -0
- package/dist/shell/binding/resolve-binding.d.ts.map +1 -0
- package/dist/shell/binding/resolve-binding.js +228 -0
- package/dist/shell/binding/resolve-binding.js.map +1 -0
- package/dist/shell/binding/types.d.ts +42 -0
- package/dist/shell/binding/types.d.ts.map +1 -0
- package/dist/shell/binding/types.js +21 -0
- package/dist/shell/binding/types.js.map +1 -0
- package/dist/shell/commands/claim.d.ts +14 -0
- package/dist/shell/commands/claim.d.ts.map +1 -0
- package/dist/shell/commands/claim.js +197 -0
- package/dist/shell/commands/claim.js.map +1 -0
- package/dist/shell/commands/doctor.d.ts +13 -0
- package/dist/shell/commands/doctor.d.ts.map +1 -0
- package/dist/shell/commands/doctor.js +97 -0
- package/dist/shell/commands/doctor.js.map +1 -0
- package/dist/shell/commands/evidence.d.ts +28 -0
- package/dist/shell/commands/evidence.d.ts.map +1 -0
- package/dist/shell/commands/evidence.js +166 -0
- package/dist/shell/commands/evidence.js.map +1 -0
- package/dist/shell/commands/gates.d.ts +19 -0
- package/dist/shell/commands/gates.d.ts.map +1 -0
- package/dist/shell/commands/gates.js +181 -0
- package/dist/shell/commands/gates.js.map +1 -0
- package/dist/shell/commands/init.d.ts +8 -0
- package/dist/shell/commands/init.d.ts.map +1 -0
- package/dist/shell/commands/init.js +64 -0
- package/dist/shell/commands/init.js.map +1 -0
- package/dist/shell/commands/scope.d.ts +11 -0
- package/dist/shell/commands/scope.d.ts.map +1 -0
- package/dist/shell/commands/scope.js +92 -0
- package/dist/shell/commands/scope.js.map +1 -0
- package/dist/shell/commands/status.d.ts +15 -0
- package/dist/shell/commands/status.d.ts.map +1 -0
- package/dist/shell/commands/status.js +106 -0
- package/dist/shell/commands/status.js.map +1 -0
- package/dist/shell/commands/waiver.d.ts +38 -0
- package/dist/shell/commands/waiver.d.ts.map +1 -0
- package/dist/shell/commands/waiver.js +240 -0
- package/dist/shell/commands/waiver.js.map +1 -0
- package/dist/shell/gates/disposition.d.ts +23 -0
- package/dist/shell/gates/disposition.d.ts.map +1 -0
- package/dist/shell/gates/disposition.js +87 -0
- package/dist/shell/gates/disposition.js.map +1 -0
- package/dist/shell/gates/gate-result-contract.d.ts +39 -0
- package/dist/shell/gates/gate-result-contract.d.ts.map +1 -0
- package/dist/shell/gates/gate-result-contract.js +150 -0
- package/dist/shell/gates/gate-result-contract.js.map +1 -0
- package/dist/shell/gates/quality-gates-adapter.d.ts +55 -0
- package/dist/shell/gates/quality-gates-adapter.d.ts.map +1 -0
- package/dist/shell/gates/quality-gates-adapter.js +161 -0
- package/dist/shell/gates/quality-gates-adapter.js.map +1 -0
- package/dist/shell/gates/waiver-filter.d.ts +58 -0
- package/dist/shell/gates/waiver-filter.d.ts.map +1 -0
- package/dist/shell/gates/waiver-filter.js +119 -0
- package/dist/shell/gates/waiver-filter.js.map +1 -0
- package/dist/shell/index.d.ts +50 -0
- package/dist/shell/index.d.ts.map +1 -0
- package/dist/shell/index.js +73 -0
- package/dist/shell/index.js.map +1 -0
- package/dist/shell/register.d.ts +11 -0
- package/dist/shell/register.d.ts.map +1 -0
- package/dist/shell/register.js +274 -0
- package/dist/shell/register.js.map +1 -0
- package/dist/shell/render/claim.d.ts +22 -0
- package/dist/shell/render/claim.d.ts.map +1 -0
- package/dist/shell/render/claim.js +75 -0
- package/dist/shell/render/claim.js.map +1 -0
- package/dist/shell/render/decision.d.ts +15 -0
- package/dist/shell/render/decision.d.ts.map +1 -0
- package/dist/shell/render/decision.js +66 -0
- package/dist/shell/render/decision.js.map +1 -0
- package/dist/shell/render/diagnostic.d.ts +19 -0
- package/dist/shell/render/diagnostic.d.ts.map +1 -0
- package/dist/shell/render/diagnostic.js +76 -0
- package/dist/shell/render/diagnostic.js.map +1 -0
- package/dist/shell/render/finding.d.ts +15 -0
- package/dist/shell/render/finding.d.ts.map +1 -0
- package/dist/shell/render/finding.js +57 -0
- package/dist/shell/render/finding.js.map +1 -0
- package/dist/shell/render/gates.d.ts +3 -0
- package/dist/shell/render/gates.d.ts.map +1 -0
- package/dist/shell/render/gates.js +56 -0
- package/dist/shell/render/gates.js.map +1 -0
- package/dist/shell/render/init.d.ts +11 -0
- package/dist/shell/render/init.d.ts.map +1 -0
- package/dist/shell/render/init.js +32 -0
- package/dist/shell/render/init.js.map +1 -0
- package/dist/shell/render/status.d.ts +26 -0
- package/dist/shell/render/status.d.ts.map +1 -0
- package/dist/shell/render/status.js +143 -0
- package/dist/shell/render/status.js.map +1 -0
- package/dist/shell/render/waiver.d.ts +21 -0
- package/dist/shell/render/waiver.d.ts.map +1 -0
- package/dist/shell/render/waiver.js +94 -0
- package/dist/shell/render/waiver.js.map +1 -0
- package/dist/shell/rules.d.ts +37 -0
- package/dist/shell/rules.d.ts.map +1 -0
- package/dist/shell/rules.js +51 -0
- package/dist/shell/rules.js.map +1 -0
- package/dist/shell/session/actor.d.ts +14 -0
- package/dist/shell/session/actor.d.ts.map +1 -0
- package/dist/shell/session/actor.js +34 -0
- package/dist/shell/session/actor.js.map +1 -0
- package/dist/shell/session/resolve-session.d.ts +5 -0
- package/dist/shell/session/resolve-session.d.ts.map +1 -0
- package/dist/shell/session/resolve-session.js +239 -0
- package/dist/shell/session/resolve-session.js.map +1 -0
- package/dist/shell/session/types.d.ts +56 -0
- package/dist/shell/session/types.d.ts.map +1 -0
- package/dist/shell/session/types.js +15 -0
- package/dist/shell/session/types.js.map +1 -0
- package/dist/store/agents-store.d.ts +3 -0
- package/dist/store/agents-store.d.ts.map +1 -0
- package/dist/store/agents-store.js +63 -0
- package/dist/store/agents-store.js.map +1 -0
- package/dist/store/apply-patch.d.ts +16 -0
- package/dist/store/apply-patch.d.ts.map +1 -0
- package/dist/store/apply-patch.js +191 -0
- package/dist/store/apply-patch.js.map +1 -0
- package/dist/store/atomic-write.d.ts +16 -0
- package/dist/store/atomic-write.d.ts.map +1 -0
- package/dist/store/atomic-write.js +132 -0
- package/dist/store/atomic-write.js.map +1 -0
- package/dist/store/doctor-snapshot.d.ts +20 -0
- package/dist/store/doctor-snapshot.d.ts.map +1 -0
- package/dist/store/doctor-snapshot.js +176 -0
- package/dist/store/doctor-snapshot.js.map +1 -0
- package/dist/store/events-store.d.ts +33 -0
- package/dist/store/events-store.d.ts.map +1 -0
- package/dist/store/events-store.js +297 -0
- package/dist/store/events-store.js.map +1 -0
- package/dist/store/index.d.ts +21 -0
- package/dist/store/index.d.ts.map +1 -0
- package/dist/store/index.js +47 -0
- package/dist/store/index.js.map +1 -0
- package/dist/store/init-store.d.ts +21 -0
- package/dist/store/init-store.d.ts.map +1 -0
- package/dist/store/init-store.js +295 -0
- package/dist/store/init-store.js.map +1 -0
- package/dist/store/json-store.d.ts +3 -0
- package/dist/store/json-store.d.ts.map +1 -0
- package/dist/store/json-store.js +65 -0
- package/dist/store/json-store.js.map +1 -0
- package/dist/store/policy-store.d.ts +3 -0
- package/dist/store/policy-store.d.ts.map +1 -0
- package/dist/store/policy-store.js +65 -0
- package/dist/store/policy-store.js.map +1 -0
- package/dist/store/repo-root.d.ts +46 -0
- package/dist/store/repo-root.d.ts.map +1 -0
- package/dist/store/repo-root.js +145 -0
- package/dist/store/repo-root.js.map +1 -0
- package/dist/store/rules.d.ts +53 -0
- package/dist/store/rules.d.ts.map +1 -0
- package/dist/store/rules.js +78 -0
- package/dist/store/rules.js.map +1 -0
- package/dist/store/specs-store.d.ts +3 -0
- package/dist/store/specs-store.d.ts.map +1 -0
- package/dist/store/specs-store.js +131 -0
- package/dist/store/specs-store.js.map +1 -0
- package/dist/store/types.d.ts +84 -0
- package/dist/store/types.d.ts.map +1 -0
- package/dist/store/types.js +14 -0
- package/dist/store/types.js.map +1 -0
- package/dist/store/waivers-store.d.ts +25 -0
- package/dist/store/waivers-store.d.ts.map +1 -0
- package/dist/store/waivers-store.js +232 -0
- package/dist/store/waivers-store.js.map +1 -0
- package/dist/store/worktrees-store.d.ts +3 -0
- package/dist/store/worktrees-store.d.ts.map +1 -0
- package/dist/store/worktrees-store.js +62 -0
- package/dist/store/worktrees-store.js.map +1 -0
- package/dist/store/yaml-store.d.ts +9 -0
- package/dist/store/yaml-store.d.ts.map +1 -0
- package/dist/store/yaml-store.js +121 -0
- package/dist/store/yaml-store.js.map +1 -0
- package/package.json +15 -13
- package/dist/budget-derivation.js +0 -751
- package/dist/cicd-optimizer.js +0 -504
- package/dist/commands/agents.js +0 -124
- package/dist/commands/archive.js +0 -500
- package/dist/commands/burnup.js +0 -198
- package/dist/commands/diagnose.js +0 -525
- package/dist/commands/evaluate.js +0 -314
- package/dist/commands/gates.js +0 -149
- package/dist/commands/init.js +0 -857
- package/dist/commands/iterate.js +0 -417
- package/dist/commands/mode.js +0 -269
- package/dist/commands/parallel.js +0 -242
- package/dist/commands/plan.js +0 -438
- package/dist/commands/provenance.js +0 -1143
- package/dist/commands/quality-monitor.js +0 -284
- package/dist/commands/scope.js +0 -264
- package/dist/commands/session.js +0 -312
- package/dist/commands/sidecar.js +0 -74
- package/dist/commands/specs.js +0 -1656
- package/dist/commands/status.js +0 -1172
- package/dist/commands/templates.js +0 -237
- package/dist/commands/tool.js +0 -136
- package/dist/commands/tutorial.js +0 -480
- package/dist/commands/validate.js +0 -357
- package/dist/commands/verify-acs.js +0 -443
- package/dist/commands/waivers.js +0 -599
- package/dist/commands/workflow.js +0 -243
- package/dist/commands/worktree.js +0 -502
- package/dist/config/lite-scope.js +0 -158
- package/dist/config/modes.js +0 -347
- package/dist/constants/spec-types.js +0 -65
- package/dist/gates/budget-limit.js +0 -121
- package/dist/gates/feedback.js +0 -260
- package/dist/gates/format.js +0 -179
- package/dist/gates/god-object.js +0 -117
- package/dist/gates/pipeline.js +0 -167
- package/dist/gates/scope-boundary.js +0 -112
- package/dist/gates/spec-completeness.js +0 -109
- package/dist/gates/todo-detection.js +0 -205
- package/dist/generators/jest-config-generator.js +0 -242
- package/dist/generators/working-spec.js +0 -237
- package/dist/minimal-cli.js +0 -88
- package/dist/parallel/parallel-manager.js +0 -433
- package/dist/policy/PolicyManager.js +0 -470
- package/dist/scaffold/claude-hooks.js +0 -443
- package/dist/scaffold/cursor-hooks.js +0 -177
- package/dist/scaffold/git-hooks.js +0 -928
- package/dist/scaffold/index.js +0 -794
- package/dist/session/session-manager.js +0 -653
- package/dist/sidecars/index.js +0 -33
- package/dist/sidecars/listeners.js +0 -40
- package/dist/sidecars/provenance-summary.js +0 -238
- package/dist/sidecars/quality-gaps.js +0 -258
- package/dist/sidecars/schema.js +0 -149
- package/dist/sidecars/spec-drift.js +0 -151
- package/dist/sidecars/waiver-draft.js +0 -176
- package/dist/spec/SpecFileManager.js +0 -419
- package/dist/templates/.caws/schemas/policy.schema.json +0 -117
- package/dist/templates/.caws/schemas/scope.schema.json +0 -52
- package/dist/templates/.caws/schemas/waivers.schema.json +0 -106
- package/dist/templates/.caws/schemas/working-spec.schema.json +0 -340
- package/dist/templates/.caws/schemas/worktrees.schema.json +0 -38
- package/dist/templates/.caws/templates/working-spec.template.yml +0 -80
- package/dist/templates/.caws/tools/README.md +0 -18
- package/dist/templates/.caws/tools/scope-guard.js +0 -203
- package/dist/templates/.caws/tools-allow.json +0 -331
- package/dist/templates/.caws/waivers.yml +0 -19
- package/dist/templates/.claude/README.md +0 -190
- package/dist/templates/.claude/hooks/audit.sh +0 -121
- package/dist/templates/.claude/hooks/block-dangerous.sh +0 -203
- package/dist/templates/.claude/hooks/classify_command.py +0 -592
- package/dist/templates/.claude/hooks/doc-frontmatter-check.sh +0 -173
- package/dist/templates/.claude/hooks/lite-sprawl-check.sh +0 -145
- package/dist/templates/.claude/hooks/naming-check.sh +0 -100
- package/dist/templates/.claude/hooks/protected-paths.sh +0 -39
- package/dist/templates/.claude/hooks/quality-check.sh +0 -81
- package/dist/templates/.claude/hooks/scan-secrets.sh +0 -85
- package/dist/templates/.claude/hooks/scope-guard.sh +0 -381
- package/dist/templates/.claude/hooks/session-caws-status.sh +0 -117
- package/dist/templates/.claude/hooks/session-log.sh +0 -634
- package/dist/templates/.claude/hooks/simplification-guard.sh +0 -92
- package/dist/templates/.claude/hooks/stop-worktree-check.sh +0 -46
- package/dist/templates/.claude/hooks/test_classify_command.py +0 -370
- package/dist/templates/.claude/hooks/test_wrapper_smoke.sh +0 -96
- package/dist/templates/.claude/hooks/validate-spec.sh +0 -76
- package/dist/templates/.claude/hooks/worktree-guard.sh +0 -220
- package/dist/templates/.claude/hooks/worktree-write-guard.sh +0 -190
- package/dist/templates/.claude/rules/git-safety.md +0 -26
- package/dist/templates/.claude/rules/worktree-isolation.md +0 -101
- package/dist/templates/.claude/settings.json +0 -141
- package/dist/templates/.cursor/README.md +0 -299
- package/dist/templates/.cursor/hooks/audit.sh +0 -55
- package/dist/templates/.cursor/hooks/block-dangerous.sh +0 -84
- package/dist/templates/.cursor/hooks/caws-quality-check.sh +0 -52
- package/dist/templates/.cursor/hooks/caws-scope-guard.sh +0 -130
- package/dist/templates/.cursor/hooks/format.sh +0 -38
- package/dist/templates/.cursor/hooks/naming-check.sh +0 -64
- package/dist/templates/.cursor/hooks/scan-secrets.sh +0 -51
- package/dist/templates/.cursor/hooks/scope-guard.sh +0 -52
- package/dist/templates/.cursor/hooks/session-log.sh +0 -924
- package/dist/templates/.cursor/hooks/validate-spec.sh +0 -83
- package/dist/templates/.cursor/hooks.json +0 -76
- package/dist/templates/.cursor/rules/00-claims-verification.mdc +0 -144
- package/dist/templates/.cursor/rules/01-working-style.mdc +0 -50
- package/dist/templates/.cursor/rules/02-quality-gates.mdc +0 -368
- package/dist/templates/.cursor/rules/03-naming-and-refactor.mdc +0 -33
- package/dist/templates/.cursor/rules/04-logging-language-style.mdc +0 -23
- package/dist/templates/.cursor/rules/05-safe-defaults-guards.mdc +0 -23
- package/dist/templates/.cursor/rules/06-typescript-conventions.mdc +0 -36
- package/dist/templates/.cursor/rules/07-process-ops.mdc +0 -20
- package/dist/templates/.cursor/rules/08-solid-and-architecture.mdc +0 -16
- package/dist/templates/.cursor/rules/09-docstrings.mdc +0 -89
- package/dist/templates/.cursor/rules/10-documentation-quality-standards.mdc +0 -385
- package/dist/templates/.cursor/rules/11-scope-management-waivers.mdc +0 -381
- package/dist/templates/.cursor/rules/12-implementation-completeness.mdc +0 -516
- package/dist/templates/.cursor/rules/13-language-agnostic-standards.mdc +0 -578
- package/dist/templates/.cursor/rules/README.md +0 -148
- package/dist/templates/.github/copilot-instructions.md +0 -82
- package/dist/templates/.idea/runConfigurations/CAWS_Evaluate.xml +0 -5
- package/dist/templates/.idea/runConfigurations/CAWS_Validate.xml +0 -5
- package/dist/templates/.junie/guidelines.md +0 -73
- package/dist/templates/.vscode/launch.json +0 -17
- package/dist/templates/.vscode/settings.json +0 -95
- package/dist/templates/.windsurf/rules/caws-quality-standards.md +0 -54
- package/dist/templates/.windsurf/workflows/caws-guided-development.md +0 -92
- package/dist/templates/CLAUDE.md +0 -196
- package/dist/templates/COMMIT_CONVENTIONS.md +0 -86
- package/dist/templates/OIDC_SETUP.md +0 -300
- package/dist/templates/agents.md +0 -171
- package/dist/templates/codemod/README.md +0 -1
- package/dist/templates/codemod/test.js +0 -93
- package/dist/templates/docs/README.md +0 -151
- package/dist/templates/scripts/new_feature.sh +0 -80
- package/dist/templates/scripts/quality-gates/check-god-objects.js +0 -146
- package/dist/templates/scripts/quality-gates/run-quality-gates.js +0 -50
- package/dist/templates/scripts/v3/analysis/todo_analyzer.py +0 -1997
- package/dist/test-analysis.js +0 -786
- package/dist/tool-interface.js +0 -314
- package/dist/tool-loader.js +0 -303
- package/dist/tool-validator.js +0 -393
- package/dist/utils/agent-display.js +0 -210
- package/dist/utils/agent-session.js +0 -344
- package/dist/utils/async-utils.js +0 -188
- package/dist/utils/command-wrapper.js +0 -200
- package/dist/utils/event-log.js +0 -584
- package/dist/utils/event-renderer.js +0 -521
- package/dist/utils/finalization.js +0 -230
- package/dist/utils/git-lock.js +0 -119
- package/dist/utils/gitignore-updater.js +0 -158
- package/dist/utils/ide-detection.js +0 -133
- package/dist/utils/lifecycle-events.js +0 -94
- package/dist/utils/project-analysis.js +0 -367
- package/dist/utils/promise-utils.js +0 -72
- package/dist/utils/quality-gates-errors.js +0 -520
- package/dist/utils/quality-gates-utils.js +0 -387
- package/dist/utils/schema-validator.js +0 -50
- package/dist/utils/spec-resolver.js +0 -711
- package/dist/utils/typescript-detector.js +0 -369
- package/dist/utils/working-state.js +0 -530
- package/dist/utils/yaml-validation.js +0 -156
- package/dist/validation/spec-validation.js +0 -924
- package/dist/waivers-manager.js +0 -732
- package/dist/worktree/worktree-manager.js +0 -1735
- package/templates/.caws/schemas/policy.schema.json +0 -117
- package/templates/.caws/schemas/scope.schema.json +0 -52
- package/templates/.caws/schemas/waivers.schema.json +0 -106
- package/templates/.caws/schemas/working-spec.schema.json +0 -340
- package/templates/.caws/schemas/worktrees.schema.json +0 -38
- package/templates/.caws/templates/working-spec.template.yml +0 -80
- package/templates/.caws/tools/README.md +0 -18
- package/templates/.caws/tools/scope-guard.js +0 -203
- package/templates/.caws/tools-allow.json +0 -331
- package/templates/.caws/waivers.yml +0 -19
- package/templates/.claude/README.md +0 -190
- package/templates/.claude/hooks/audit.sh +0 -121
- package/templates/.claude/hooks/block-dangerous.sh +0 -203
- package/templates/.claude/hooks/classify_command.py +0 -592
- package/templates/.claude/hooks/doc-frontmatter-check.sh +0 -173
- package/templates/.claude/hooks/lite-sprawl-check.sh +0 -145
- package/templates/.claude/hooks/naming-check.sh +0 -100
- package/templates/.claude/hooks/protected-paths.sh +0 -39
- package/templates/.claude/hooks/quality-check.sh +0 -81
- package/templates/.claude/hooks/scan-secrets.sh +0 -85
- package/templates/.claude/hooks/scope-guard.sh +0 -381
- package/templates/.claude/hooks/session-caws-status.sh +0 -117
- package/templates/.claude/hooks/session-log.sh +0 -634
- package/templates/.claude/hooks/simplification-guard.sh +0 -92
- package/templates/.claude/hooks/stop-worktree-check.sh +0 -46
- package/templates/.claude/hooks/test_classify_command.py +0 -370
- package/templates/.claude/hooks/test_wrapper_smoke.sh +0 -96
- package/templates/.claude/hooks/validate-spec.sh +0 -76
- package/templates/.claude/hooks/worktree-guard.sh +0 -220
- package/templates/.claude/hooks/worktree-write-guard.sh +0 -190
- package/templates/.claude/rules/git-safety.md +0 -26
- package/templates/.claude/rules/worktree-isolation.md +0 -101
- package/templates/.claude/settings.json +0 -141
- package/templates/.cursor/README.md +0 -299
- package/templates/.cursor/hooks/audit.sh +0 -55
- package/templates/.cursor/hooks/block-dangerous.sh +0 -84
- package/templates/.cursor/hooks/caws-quality-check.sh +0 -52
- package/templates/.cursor/hooks/caws-scope-guard.sh +0 -130
- package/templates/.cursor/hooks/format.sh +0 -38
- package/templates/.cursor/hooks/naming-check.sh +0 -64
- package/templates/.cursor/hooks/scan-secrets.sh +0 -51
- package/templates/.cursor/hooks/scope-guard.sh +0 -52
- package/templates/.cursor/hooks/session-log.sh +0 -924
- package/templates/.cursor/hooks/validate-spec.sh +0 -83
- package/templates/.cursor/hooks.json +0 -76
- package/templates/.cursor/rules/00-claims-verification.mdc +0 -144
- package/templates/.cursor/rules/01-working-style.mdc +0 -50
- package/templates/.cursor/rules/02-quality-gates.mdc +0 -368
- package/templates/.cursor/rules/03-naming-and-refactor.mdc +0 -33
- package/templates/.cursor/rules/04-logging-language-style.mdc +0 -23
- package/templates/.cursor/rules/05-safe-defaults-guards.mdc +0 -23
- package/templates/.cursor/rules/06-typescript-conventions.mdc +0 -36
- package/templates/.cursor/rules/07-process-ops.mdc +0 -20
- package/templates/.cursor/rules/08-solid-and-architecture.mdc +0 -16
- package/templates/.cursor/rules/09-docstrings.mdc +0 -89
- package/templates/.cursor/rules/10-documentation-quality-standards.mdc +0 -385
- package/templates/.cursor/rules/11-scope-management-waivers.mdc +0 -381
- package/templates/.cursor/rules/12-implementation-completeness.mdc +0 -516
- package/templates/.cursor/rules/13-language-agnostic-standards.mdc +0 -578
- package/templates/.cursor/rules/README.md +0 -148
- package/templates/.github/copilot-instructions.md +0 -82
- package/templates/.idea/runConfigurations/CAWS_Evaluate.xml +0 -5
- package/templates/.idea/runConfigurations/CAWS_Validate.xml +0 -5
- package/templates/.junie/guidelines.md +0 -73
- package/templates/.vscode/launch.json +0 -17
- package/templates/.vscode/settings.json +0 -95
- package/templates/.windsurf/rules/caws-quality-standards.md +0 -54
- package/templates/.windsurf/workflows/caws-guided-development.md +0 -92
- package/templates/CLAUDE.md +0 -196
- package/templates/COMMIT_CONVENTIONS.md +0 -86
- package/templates/OIDC_SETUP.md +0 -300
- package/templates/agents.md +0 -171
- package/templates/codemod/README.md +0 -1
- package/templates/codemod/test.js +0 -93
- package/templates/docs/README.md +0 -151
- package/templates/scripts/new_feature.sh +0 -80
- package/templates/scripts/quality-gates/check-god-objects.js +0 -146
- package/templates/scripts/quality-gates/run-quality-gates.js +0 -50
- package/templates/scripts/v3/analysis/todo_analyzer.py +0 -1997
|
@@ -1,653 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @fileoverview CAWS Session Capsule Manager
|
|
3
|
-
* Manages session lifecycle and capsule persistence for multi-agent coordination.
|
|
4
|
-
* Each session produces a structured capsule that captures baseline state on entry
|
|
5
|
-
* and work summary + verification evidence on exit.
|
|
6
|
-
* @author @darianrosebrook
|
|
7
|
-
*/
|
|
8
|
-
|
|
9
|
-
const { execFileSync } = require('child_process');
|
|
10
|
-
const fs = require('fs-extra');
|
|
11
|
-
const path = require('path');
|
|
12
|
-
const crypto = require('crypto');
|
|
13
|
-
|
|
14
|
-
const { mergeFilesTouched } = require('../utils/working-state');
|
|
15
|
-
const { appendEventSync } = require('../utils/event-log');
|
|
16
|
-
|
|
17
|
-
const SESSIONS_DIR = '.caws/sessions';
|
|
18
|
-
const REGISTRY_FILE = '.caws/sessions.json';
|
|
19
|
-
const CAPSULE_SCHEMA_VERSION = 'caws.capsule.v1';
|
|
20
|
-
|
|
21
|
-
/**
|
|
22
|
-
* Get the git repository root
|
|
23
|
-
* @returns {string} Absolute path to repo root
|
|
24
|
-
*/
|
|
25
|
-
function getRepoRoot() {
|
|
26
|
-
return execFileSync('git', ['rev-parse', '--show-toplevel'], {
|
|
27
|
-
encoding: 'utf8',
|
|
28
|
-
}).trim();
|
|
29
|
-
}
|
|
30
|
-
|
|
31
|
-
/**
|
|
32
|
-
* Get current HEAD revision (short hash)
|
|
33
|
-
* @param {string} cwd - Working directory
|
|
34
|
-
* @returns {string}
|
|
35
|
-
*/
|
|
36
|
-
function getHeadRev(cwd) {
|
|
37
|
-
try {
|
|
38
|
-
return execFileSync('git', ['rev-parse', '--short', 'HEAD'], {
|
|
39
|
-
encoding: 'utf8',
|
|
40
|
-
cwd,
|
|
41
|
-
}).trim();
|
|
42
|
-
} catch {
|
|
43
|
-
return 'unknown';
|
|
44
|
-
}
|
|
45
|
-
}
|
|
46
|
-
|
|
47
|
-
/**
|
|
48
|
-
* Get current branch name
|
|
49
|
-
* @param {string} cwd - Working directory
|
|
50
|
-
* @returns {string}
|
|
51
|
-
*/
|
|
52
|
-
function getCurrentBranch(cwd) {
|
|
53
|
-
try {
|
|
54
|
-
return execFileSync('git', ['rev-parse', '--abbrev-ref', 'HEAD'], {
|
|
55
|
-
encoding: 'utf8',
|
|
56
|
-
cwd,
|
|
57
|
-
}).trim();
|
|
58
|
-
} catch {
|
|
59
|
-
return 'detached';
|
|
60
|
-
}
|
|
61
|
-
}
|
|
62
|
-
|
|
63
|
-
/**
|
|
64
|
-
* Get dirty files in working tree
|
|
65
|
-
* @param {string} cwd - Working directory
|
|
66
|
-
* @returns {{ paths: string[], dirty: boolean }}
|
|
67
|
-
*/
|
|
68
|
-
function getWorkspaceFingerprint(cwd) {
|
|
69
|
-
try {
|
|
70
|
-
const output = execFileSync('git', ['status', '--porcelain'], {
|
|
71
|
-
encoding: 'utf8',
|
|
72
|
-
cwd,
|
|
73
|
-
});
|
|
74
|
-
const paths = output
|
|
75
|
-
.split('\n')
|
|
76
|
-
.filter(Boolean)
|
|
77
|
-
.map((line) => line.substring(3).trim());
|
|
78
|
-
return { paths_touched: paths, dirty: paths.length > 0 };
|
|
79
|
-
} catch {
|
|
80
|
-
return { paths_touched: [], dirty: false };
|
|
81
|
-
}
|
|
82
|
-
}
|
|
83
|
-
|
|
84
|
-
/**
|
|
85
|
-
* Load the best available spec synchronously (feature specs first, then legacy).
|
|
86
|
-
* @param {string} root - Repository root
|
|
87
|
-
* @param {string} [specId] - Optional specific spec ID
|
|
88
|
-
* @returns {object|null} Parsed spec object or null
|
|
89
|
-
*/
|
|
90
|
-
function loadBestSpecSync(root, specId) {
|
|
91
|
-
const yaml = require('js-yaml');
|
|
92
|
-
|
|
93
|
-
// If a specific spec ID is requested, load it directly
|
|
94
|
-
if (specId) {
|
|
95
|
-
for (const ext of ['.yaml', '.yml']) {
|
|
96
|
-
const p = path.join(root, '.caws/specs', `${specId}${ext}`);
|
|
97
|
-
if (fs.existsSync(p)) {
|
|
98
|
-
try { return yaml.load(fs.readFileSync(p, 'utf8')); } catch { /* skip */ }
|
|
99
|
-
}
|
|
100
|
-
}
|
|
101
|
-
}
|
|
102
|
-
|
|
103
|
-
// Check registry for active feature specs
|
|
104
|
-
const registryPath = path.join(root, '.caws/specs/registry.json');
|
|
105
|
-
if (fs.existsSync(registryPath)) {
|
|
106
|
-
try {
|
|
107
|
-
const registry = JSON.parse(fs.readFileSync(registryPath, 'utf8'));
|
|
108
|
-
const specs = registry.specs || {};
|
|
109
|
-
const activeIds = Object.keys(specs).filter(
|
|
110
|
-
id => specs[id].status !== 'closed' && specs[id].status !== 'archived'
|
|
111
|
-
);
|
|
112
|
-
for (const id of activeIds) {
|
|
113
|
-
const specEntry = specs[id];
|
|
114
|
-
const p = path.join(root, '.caws/specs', specEntry.path || `${id}.yaml`);
|
|
115
|
-
if (fs.existsSync(p)) {
|
|
116
|
-
try { return yaml.load(fs.readFileSync(p, 'utf8')); } catch { /* skip */ }
|
|
117
|
-
}
|
|
118
|
-
}
|
|
119
|
-
} catch { /* fall through */ }
|
|
120
|
-
}
|
|
121
|
-
|
|
122
|
-
// Legacy fallback
|
|
123
|
-
const legacyPath = path.join(root, '.caws/working-spec.yaml');
|
|
124
|
-
if (fs.existsSync(legacyPath)) {
|
|
125
|
-
try { return yaml.load(fs.readFileSync(legacyPath, 'utf8')); } catch { /* skip */ }
|
|
126
|
-
}
|
|
127
|
-
|
|
128
|
-
return null;
|
|
129
|
-
}
|
|
130
|
-
|
|
131
|
-
/**
|
|
132
|
-
* Get project name from best available spec or directory
|
|
133
|
-
* @param {string} root - Repository root
|
|
134
|
-
* @param {string} [specId] - Optional specific spec ID
|
|
135
|
-
* @returns {string}
|
|
136
|
-
*/
|
|
137
|
-
function getProjectName(root, specId) {
|
|
138
|
-
try {
|
|
139
|
-
const spec = loadBestSpecSync(root, specId);
|
|
140
|
-
if (spec) {
|
|
141
|
-
return spec.title || spec.id || path.basename(root);
|
|
142
|
-
}
|
|
143
|
-
} catch {
|
|
144
|
-
// Fall through
|
|
145
|
-
}
|
|
146
|
-
return path.basename(root);
|
|
147
|
-
}
|
|
148
|
-
|
|
149
|
-
/**
|
|
150
|
-
* Get skein ID from best available spec
|
|
151
|
-
* @param {string} root - Repository root
|
|
152
|
-
* @param {string} [specId] - Optional specific spec ID
|
|
153
|
-
* @returns {string}
|
|
154
|
-
*/
|
|
155
|
-
function getSkeinId(root, specId) {
|
|
156
|
-
try {
|
|
157
|
-
const spec = loadBestSpecSync(root, specId);
|
|
158
|
-
if (spec) {
|
|
159
|
-
return spec.id || 'unknown';
|
|
160
|
-
}
|
|
161
|
-
} catch {
|
|
162
|
-
// Fall through
|
|
163
|
-
}
|
|
164
|
-
return 'unknown';
|
|
165
|
-
}
|
|
166
|
-
|
|
167
|
-
/**
|
|
168
|
-
* Load the session registry
|
|
169
|
-
* @param {string} root - Repository root
|
|
170
|
-
* @returns {Object} Registry object
|
|
171
|
-
*/
|
|
172
|
-
function loadRegistry(root) {
|
|
173
|
-
const registryPath = path.join(root, REGISTRY_FILE);
|
|
174
|
-
try {
|
|
175
|
-
if (fs.existsSync(registryPath)) {
|
|
176
|
-
return JSON.parse(fs.readFileSync(registryPath, 'utf8'));
|
|
177
|
-
}
|
|
178
|
-
} catch {
|
|
179
|
-
// Corrupted registry, start fresh
|
|
180
|
-
}
|
|
181
|
-
return { version: 1, sessions: {} };
|
|
182
|
-
}
|
|
183
|
-
|
|
184
|
-
/**
|
|
185
|
-
* Save the session registry
|
|
186
|
-
* @param {string} root - Repository root
|
|
187
|
-
* @param {Object} registry - Registry object
|
|
188
|
-
*/
|
|
189
|
-
function saveRegistry(root, registry) {
|
|
190
|
-
const registryPath = path.join(root, REGISTRY_FILE);
|
|
191
|
-
fs.ensureDirSync(path.dirname(registryPath));
|
|
192
|
-
fs.writeFileSync(registryPath, JSON.stringify(registry, null, 2));
|
|
193
|
-
}
|
|
194
|
-
|
|
195
|
-
/**
|
|
196
|
-
* Generate a deterministic session ID
|
|
197
|
-
* @returns {string}
|
|
198
|
-
*/
|
|
199
|
-
function generateSessionId() {
|
|
200
|
-
const timestamp = new Date().toISOString().replace(/[:.]/g, '-');
|
|
201
|
-
const suffix = crypto.randomBytes(4).toString('hex');
|
|
202
|
-
return `${timestamp}__${suffix}`;
|
|
203
|
-
}
|
|
204
|
-
|
|
205
|
-
/**
|
|
206
|
-
* Start a new session, creating the initial capsule with baseline state
|
|
207
|
-
* @param {Object} options - Session options
|
|
208
|
-
* @param {string} [options.role] - Agent role (worker, integrator, qa)
|
|
209
|
-
* @param {string} [options.specId] - Associated feature spec ID
|
|
210
|
-
* @param {string[]} [options.allowedGlobs] - Allowed file patterns
|
|
211
|
-
* @param {string[]} [options.forbiddenGlobs] - Forbidden file patterns
|
|
212
|
-
* @param {string} [options.intent] - What this session intends to accomplish
|
|
213
|
-
* @returns {Object} Created capsule
|
|
214
|
-
*/
|
|
215
|
-
function startSession(options = {}) {
|
|
216
|
-
const root = getRepoRoot();
|
|
217
|
-
const registry = loadRegistry(root);
|
|
218
|
-
const sessionId = generateSessionId();
|
|
219
|
-
|
|
220
|
-
const {
|
|
221
|
-
role = 'worker',
|
|
222
|
-
specId,
|
|
223
|
-
allowedGlobs = [],
|
|
224
|
-
forbiddenGlobs = [],
|
|
225
|
-
intent = '',
|
|
226
|
-
} = options;
|
|
227
|
-
|
|
228
|
-
// Build scope from spec if available and no explicit globs provided
|
|
229
|
-
let scope = {
|
|
230
|
-
allowed_globs: allowedGlobs,
|
|
231
|
-
forbidden_globs: forbiddenGlobs,
|
|
232
|
-
};
|
|
233
|
-
|
|
234
|
-
if (specId && allowedGlobs.length === 0) {
|
|
235
|
-
try {
|
|
236
|
-
const yaml = require('js-yaml');
|
|
237
|
-
const specPath = path.join(root, `.caws/specs/${specId}.yaml`);
|
|
238
|
-
if (fs.existsSync(specPath)) {
|
|
239
|
-
const spec = yaml.load(fs.readFileSync(specPath, 'utf8'));
|
|
240
|
-
if (spec.scope) {
|
|
241
|
-
scope.allowed_globs = spec.scope.in || [];
|
|
242
|
-
scope.forbidden_globs = spec.scope.out || [];
|
|
243
|
-
}
|
|
244
|
-
}
|
|
245
|
-
} catch {
|
|
246
|
-
// Non-fatal: scope stays as provided
|
|
247
|
-
}
|
|
248
|
-
}
|
|
249
|
-
|
|
250
|
-
const capsule = {
|
|
251
|
-
schema: CAPSULE_SCHEMA_VERSION,
|
|
252
|
-
project: getProjectName(root, specId),
|
|
253
|
-
skein_id: getSkeinId(root, specId),
|
|
254
|
-
session_id: sessionId,
|
|
255
|
-
role,
|
|
256
|
-
spec_id: specId || null,
|
|
257
|
-
scope,
|
|
258
|
-
base_state: {
|
|
259
|
-
head_rev: getHeadRev(root),
|
|
260
|
-
branch: getCurrentBranch(root),
|
|
261
|
-
workspace_fingerprint: getWorkspaceFingerprint(root),
|
|
262
|
-
},
|
|
263
|
-
started_at: new Date().toISOString(),
|
|
264
|
-
ended_at: null,
|
|
265
|
-
work_summary: {
|
|
266
|
-
intent: intent || '',
|
|
267
|
-
paths_touched: [],
|
|
268
|
-
artifacts_written: [],
|
|
269
|
-
commits: [],
|
|
270
|
-
},
|
|
271
|
-
verification: {
|
|
272
|
-
tests_run: [],
|
|
273
|
-
determinism_checks: [],
|
|
274
|
-
},
|
|
275
|
-
known_issues: [],
|
|
276
|
-
handoff: {
|
|
277
|
-
next_actions: [],
|
|
278
|
-
risk_notes: [],
|
|
279
|
-
},
|
|
280
|
-
};
|
|
281
|
-
|
|
282
|
-
// Persist capsule
|
|
283
|
-
const sessionsDir = path.join(root, SESSIONS_DIR);
|
|
284
|
-
fs.ensureDirSync(sessionsDir);
|
|
285
|
-
const capsulePath = path.join(sessionsDir, `${sessionId}.json`);
|
|
286
|
-
fs.writeFileSync(capsulePath, JSON.stringify(capsule, null, 2));
|
|
287
|
-
|
|
288
|
-
// Update registry
|
|
289
|
-
registry.sessions[sessionId] = {
|
|
290
|
-
path: `${sessionId}.json`,
|
|
291
|
-
role,
|
|
292
|
-
spec_id: specId || null,
|
|
293
|
-
status: 'active',
|
|
294
|
-
started_at: capsule.started_at,
|
|
295
|
-
ended_at: null,
|
|
296
|
-
head_rev: capsule.base_state.head_rev,
|
|
297
|
-
branch: capsule.base_state.branch,
|
|
298
|
-
};
|
|
299
|
-
saveRegistry(root, registry);
|
|
300
|
-
|
|
301
|
-
// EVLOG-001: emit session_started event via the sync append path.
|
|
302
|
-
// spec_id is optional for this event type; we include it only when
|
|
303
|
-
// the session is bound to a spec.
|
|
304
|
-
const sessionStartedEvent = {
|
|
305
|
-
actor: 'session',
|
|
306
|
-
event: 'session_started',
|
|
307
|
-
data: {
|
|
308
|
-
session_id: sessionId,
|
|
309
|
-
role,
|
|
310
|
-
branch: capsule.base_state.branch,
|
|
311
|
-
head_rev: capsule.base_state.head_rev,
|
|
312
|
-
},
|
|
313
|
-
};
|
|
314
|
-
if (specId) sessionStartedEvent.spec_id = specId;
|
|
315
|
-
appendEventSync(sessionStartedEvent, { projectRoot: root, session_id: sessionId });
|
|
316
|
-
|
|
317
|
-
return capsule;
|
|
318
|
-
}
|
|
319
|
-
|
|
320
|
-
/**
|
|
321
|
-
* Add a checkpoint to the current (most recent active) session
|
|
322
|
-
* @param {Object} data - Checkpoint data
|
|
323
|
-
* @param {string} [data.sessionId] - Specific session ID (uses latest active if omitted)
|
|
324
|
-
* @param {string[]} [data.pathsTouched] - Files changed
|
|
325
|
-
* @param {string[]} [data.artifactsWritten] - Generated artifacts
|
|
326
|
-
* @param {Object[]} [data.testsRun] - Test results { name, status, evidence }
|
|
327
|
-
* @param {Object[]} [data.determinismChecks] - Determinism checks { name, status, total }
|
|
328
|
-
* @param {Object[]} [data.knownIssues] - Issues discovered { type, description }
|
|
329
|
-
* @param {string} [data.intent] - Updated intent description
|
|
330
|
-
* @returns {Object} Updated capsule
|
|
331
|
-
*/
|
|
332
|
-
function checkpointSession(data = {}) {
|
|
333
|
-
const root = getRepoRoot();
|
|
334
|
-
const registry = loadRegistry(root);
|
|
335
|
-
|
|
336
|
-
// Find session
|
|
337
|
-
const sessionId = data.sessionId || findActiveSession(registry);
|
|
338
|
-
if (!sessionId) {
|
|
339
|
-
throw new Error('No active session found. Start one with: caws session start');
|
|
340
|
-
}
|
|
341
|
-
|
|
342
|
-
const capsulePath = path.join(root, SESSIONS_DIR, `${sessionId}.json`);
|
|
343
|
-
if (!fs.existsSync(capsulePath)) {
|
|
344
|
-
throw new Error(`Session capsule not found: ${sessionId}`);
|
|
345
|
-
}
|
|
346
|
-
|
|
347
|
-
const capsule = JSON.parse(fs.readFileSync(capsulePath, 'utf8'));
|
|
348
|
-
|
|
349
|
-
// Merge checkpoint data
|
|
350
|
-
if (data.intent) {
|
|
351
|
-
capsule.work_summary.intent = data.intent;
|
|
352
|
-
}
|
|
353
|
-
if (data.pathsTouched) {
|
|
354
|
-
const existing = new Set(capsule.work_summary.paths_touched);
|
|
355
|
-
for (const p of data.pathsTouched) existing.add(p);
|
|
356
|
-
capsule.work_summary.paths_touched = [...existing];
|
|
357
|
-
}
|
|
358
|
-
if (data.artifactsWritten) {
|
|
359
|
-
const existing = new Set(capsule.work_summary.artifacts_written);
|
|
360
|
-
for (const a of data.artifactsWritten) existing.add(a);
|
|
361
|
-
capsule.work_summary.artifacts_written = [...existing];
|
|
362
|
-
}
|
|
363
|
-
if (data.testsRun) {
|
|
364
|
-
capsule.verification.tests_run.push(...data.testsRun);
|
|
365
|
-
}
|
|
366
|
-
if (data.determinismChecks) {
|
|
367
|
-
capsule.verification.determinism_checks.push(...data.determinismChecks);
|
|
368
|
-
}
|
|
369
|
-
if (data.knownIssues) {
|
|
370
|
-
capsule.known_issues.push(...data.knownIssues);
|
|
371
|
-
}
|
|
372
|
-
|
|
373
|
-
// Record current commit as a checkpoint
|
|
374
|
-
const currentRev = getHeadRev(root);
|
|
375
|
-
if (currentRev !== capsule.base_state.head_rev) {
|
|
376
|
-
capsule.work_summary.commits.push({
|
|
377
|
-
rev: currentRev,
|
|
378
|
-
checkpoint_at: new Date().toISOString(),
|
|
379
|
-
});
|
|
380
|
-
}
|
|
381
|
-
|
|
382
|
-
// Write updated capsule
|
|
383
|
-
fs.writeFileSync(capsulePath, JSON.stringify(capsule, null, 2));
|
|
384
|
-
|
|
385
|
-
// Bridge to working state (per-spec)
|
|
386
|
-
if (capsule.spec_id && capsule.work_summary.paths_touched.length > 0) {
|
|
387
|
-
try { mergeFilesTouched(capsule.spec_id, capsule.work_summary.paths_touched, root); } catch { /* non-fatal */ }
|
|
388
|
-
}
|
|
389
|
-
|
|
390
|
-
return capsule;
|
|
391
|
-
}
|
|
392
|
-
|
|
393
|
-
/**
|
|
394
|
-
* End a session, finalizing the capsule with handoff information
|
|
395
|
-
* @param {Object} data - End session data
|
|
396
|
-
* @param {string} [data.sessionId] - Specific session ID (uses latest active if omitted)
|
|
397
|
-
* @param {string[]} [data.nextActions] - What the next session should do
|
|
398
|
-
* @param {string[]} [data.riskNotes] - Risk notes for handoff
|
|
399
|
-
* @returns {Object} Finalized capsule
|
|
400
|
-
*/
|
|
401
|
-
function endSession(data = {}) {
|
|
402
|
-
const root = getRepoRoot();
|
|
403
|
-
const registry = loadRegistry(root);
|
|
404
|
-
|
|
405
|
-
const sessionId = data.sessionId || findActiveSession(registry);
|
|
406
|
-
if (!sessionId) {
|
|
407
|
-
throw new Error('No active session found.');
|
|
408
|
-
}
|
|
409
|
-
|
|
410
|
-
const capsulePath = path.join(root, SESSIONS_DIR, `${sessionId}.json`);
|
|
411
|
-
if (!fs.existsSync(capsulePath)) {
|
|
412
|
-
throw new Error(`Session capsule not found: ${sessionId}`);
|
|
413
|
-
}
|
|
414
|
-
|
|
415
|
-
const capsule = JSON.parse(fs.readFileSync(capsulePath, 'utf8'));
|
|
416
|
-
|
|
417
|
-
// Finalize
|
|
418
|
-
capsule.ended_at = new Date().toISOString();
|
|
419
|
-
|
|
420
|
-
// Capture final workspace state
|
|
421
|
-
const fingerprint = getWorkspaceFingerprint(root);
|
|
422
|
-
capsule.work_summary.paths_touched = [
|
|
423
|
-
...new Set([...capsule.work_summary.paths_touched, ...fingerprint.paths_touched]),
|
|
424
|
-
];
|
|
425
|
-
|
|
426
|
-
// Record final commit
|
|
427
|
-
const finalRev = getHeadRev(root);
|
|
428
|
-
if (
|
|
429
|
-
finalRev !== capsule.base_state.head_rev &&
|
|
430
|
-
!capsule.work_summary.commits.some((c) => c.rev === finalRev)
|
|
431
|
-
) {
|
|
432
|
-
capsule.work_summary.commits.push({
|
|
433
|
-
rev: finalRev,
|
|
434
|
-
checkpoint_at: new Date().toISOString(),
|
|
435
|
-
});
|
|
436
|
-
}
|
|
437
|
-
|
|
438
|
-
// Handoff
|
|
439
|
-
if (data.nextActions) {
|
|
440
|
-
capsule.handoff.next_actions = data.nextActions;
|
|
441
|
-
}
|
|
442
|
-
if (data.riskNotes) {
|
|
443
|
-
capsule.handoff.risk_notes = data.riskNotes;
|
|
444
|
-
}
|
|
445
|
-
|
|
446
|
-
// Flag if dirty
|
|
447
|
-
if (fingerprint.dirty) {
|
|
448
|
-
capsule.known_issues.push({
|
|
449
|
-
type: 'warning',
|
|
450
|
-
description: `Session ended with ${fingerprint.paths_touched.length} uncommitted file(s).`,
|
|
451
|
-
});
|
|
452
|
-
}
|
|
453
|
-
|
|
454
|
-
// Write finalized capsule
|
|
455
|
-
fs.writeFileSync(capsulePath, JSON.stringify(capsule, null, 2));
|
|
456
|
-
|
|
457
|
-
// Bridge to working state (per-spec)
|
|
458
|
-
if (capsule.spec_id && capsule.work_summary.paths_touched.length > 0) {
|
|
459
|
-
try { mergeFilesTouched(capsule.spec_id, capsule.work_summary.paths_touched, root); } catch { /* non-fatal */ }
|
|
460
|
-
}
|
|
461
|
-
|
|
462
|
-
// Update registry
|
|
463
|
-
registry.sessions[sessionId].status = 'completed';
|
|
464
|
-
registry.sessions[sessionId].ended_at = capsule.ended_at;
|
|
465
|
-
saveRegistry(root, registry);
|
|
466
|
-
|
|
467
|
-
// EVLOG-001: emit session_ended event with final files_touched list.
|
|
468
|
-
// The renderer uses this to merge file touches into the spec view.
|
|
469
|
-
const sessionEndedEvent = {
|
|
470
|
-
actor: 'session',
|
|
471
|
-
event: 'session_ended',
|
|
472
|
-
data: {
|
|
473
|
-
session_id: sessionId,
|
|
474
|
-
files_touched: capsule.work_summary.paths_touched || [],
|
|
475
|
-
outcome: capsule.known_issues.some((i) => i.type === 'error') ? 'error' : 'success',
|
|
476
|
-
},
|
|
477
|
-
};
|
|
478
|
-
if (capsule.spec_id) {
|
|
479
|
-
sessionEndedEvent.spec_id = capsule.spec_id;
|
|
480
|
-
sessionEndedEvent.data.spec_id = capsule.spec_id;
|
|
481
|
-
}
|
|
482
|
-
appendEventSync(sessionEndedEvent, { projectRoot: root, session_id: sessionId });
|
|
483
|
-
|
|
484
|
-
return capsule;
|
|
485
|
-
}
|
|
486
|
-
|
|
487
|
-
/**
|
|
488
|
-
* List all sessions
|
|
489
|
-
* @param {Object} [options] - List options
|
|
490
|
-
* @param {string} [options.status] - Filter by status (active, completed)
|
|
491
|
-
* @param {number} [options.limit] - Max entries to return
|
|
492
|
-
* @returns {Object[]} Session entries
|
|
493
|
-
*/
|
|
494
|
-
function listSessions(options = {}) {
|
|
495
|
-
const root = getRepoRoot();
|
|
496
|
-
const registry = loadRegistry(root);
|
|
497
|
-
|
|
498
|
-
let entries = Object.entries(registry.sessions).map(([id, meta]) => ({
|
|
499
|
-
id,
|
|
500
|
-
...meta,
|
|
501
|
-
}));
|
|
502
|
-
|
|
503
|
-
if (options.status) {
|
|
504
|
-
entries = entries.filter((e) => e.status === options.status);
|
|
505
|
-
}
|
|
506
|
-
|
|
507
|
-
// Sort by started_at descending (most recent first)
|
|
508
|
-
entries.sort((a, b) => new Date(b.started_at) - new Date(a.started_at));
|
|
509
|
-
|
|
510
|
-
if (options.limit) {
|
|
511
|
-
entries = entries.slice(0, options.limit);
|
|
512
|
-
}
|
|
513
|
-
|
|
514
|
-
return entries;
|
|
515
|
-
}
|
|
516
|
-
|
|
517
|
-
/**
|
|
518
|
-
* Show a specific session's full capsule
|
|
519
|
-
* @param {string} sessionId - Session ID (or "latest" for most recent)
|
|
520
|
-
* @returns {Object} Full capsule
|
|
521
|
-
*/
|
|
522
|
-
function showSession(sessionId) {
|
|
523
|
-
const root = getRepoRoot();
|
|
524
|
-
|
|
525
|
-
if (sessionId === 'latest') {
|
|
526
|
-
const registry = loadRegistry(root);
|
|
527
|
-
const active = findActiveSession(registry);
|
|
528
|
-
if (active) {
|
|
529
|
-
sessionId = active;
|
|
530
|
-
} else {
|
|
531
|
-
// Find most recent completed
|
|
532
|
-
const entries = Object.entries(registry.sessions).sort(
|
|
533
|
-
(a, b) => new Date(b[1].started_at) - new Date(a[1].started_at)
|
|
534
|
-
);
|
|
535
|
-
if (entries.length === 0) throw new Error('No sessions found.');
|
|
536
|
-
sessionId = entries[0][0];
|
|
537
|
-
}
|
|
538
|
-
}
|
|
539
|
-
|
|
540
|
-
const capsulePath = path.join(root, SESSIONS_DIR, `${sessionId}.json`);
|
|
541
|
-
if (!fs.existsSync(capsulePath)) {
|
|
542
|
-
throw new Error(`Session capsule not found: ${sessionId}`);
|
|
543
|
-
}
|
|
544
|
-
|
|
545
|
-
return JSON.parse(fs.readFileSync(capsulePath, 'utf8'));
|
|
546
|
-
}
|
|
547
|
-
|
|
548
|
-
/**
|
|
549
|
-
* Briefing output for session start hooks - returns structured text
|
|
550
|
-
* @returns {string} Briefing text
|
|
551
|
-
*/
|
|
552
|
-
function getBriefing() {
|
|
553
|
-
const root = getRepoRoot();
|
|
554
|
-
const registry = loadRegistry(root);
|
|
555
|
-
|
|
556
|
-
const lines = [];
|
|
557
|
-
lines.push('--- CAWS Session Briefing ---');
|
|
558
|
-
|
|
559
|
-
// Git baseline
|
|
560
|
-
const headRev = getHeadRev(root);
|
|
561
|
-
const branch = getCurrentBranch(root);
|
|
562
|
-
const fingerprint = getWorkspaceFingerprint(root);
|
|
563
|
-
lines.push(`Git: ${branch} @ ${headRev} (${fingerprint.paths_touched.length} dirty files)`);
|
|
564
|
-
|
|
565
|
-
if (fingerprint.dirty) {
|
|
566
|
-
lines.push('WARNING: Working tree has uncommitted changes from a prior session.');
|
|
567
|
-
}
|
|
568
|
-
|
|
569
|
-
// Active sessions
|
|
570
|
-
const activeSessions = Object.entries(registry.sessions)
|
|
571
|
-
.filter(([, meta]) => meta.status === 'active')
|
|
572
|
-
.map(([id, meta]) => ({ id, ...meta }));
|
|
573
|
-
|
|
574
|
-
if (activeSessions.length > 0) {
|
|
575
|
-
lines.push(`Active sessions: ${activeSessions.length}`);
|
|
576
|
-
for (const s of activeSessions) {
|
|
577
|
-
lines.push(` - ${s.id} (${s.role}, spec: ${s.spec_id || 'none'})`);
|
|
578
|
-
}
|
|
579
|
-
}
|
|
580
|
-
|
|
581
|
-
// Last completed session handoff
|
|
582
|
-
const completedSessions = Object.entries(registry.sessions)
|
|
583
|
-
.filter(([, meta]) => meta.status === 'completed')
|
|
584
|
-
.sort((a, b) => new Date(b[1].ended_at) - new Date(a[1].ended_at));
|
|
585
|
-
|
|
586
|
-
if (completedSessions.length > 0) {
|
|
587
|
-
const [lastId] = completedSessions[0];
|
|
588
|
-
try {
|
|
589
|
-
const capsule = showSession(lastId);
|
|
590
|
-
if (capsule.handoff.next_actions.length > 0) {
|
|
591
|
-
lines.push('Handoff from prior session:');
|
|
592
|
-
for (const action of capsule.handoff.next_actions) {
|
|
593
|
-
lines.push(` - ${action}`);
|
|
594
|
-
}
|
|
595
|
-
}
|
|
596
|
-
if (capsule.known_issues.length > 0) {
|
|
597
|
-
lines.push('Known issues from prior session:');
|
|
598
|
-
for (const issue of capsule.known_issues) {
|
|
599
|
-
lines.push(` - [${issue.type}] ${issue.description}`);
|
|
600
|
-
}
|
|
601
|
-
}
|
|
602
|
-
} catch {
|
|
603
|
-
// Non-fatal
|
|
604
|
-
}
|
|
605
|
-
}
|
|
606
|
-
|
|
607
|
-
lines.push('---');
|
|
608
|
-
lines.push("Run 'caws session start' to begin a tracked session.");
|
|
609
|
-
lines.push('--- End CAWS Briefing ---');
|
|
610
|
-
|
|
611
|
-
return lines.join('\n');
|
|
612
|
-
}
|
|
613
|
-
|
|
614
|
-
/**
|
|
615
|
-
* Find the most recent active session
|
|
616
|
-
* @param {Object} registry - Session registry
|
|
617
|
-
* @returns {string|null} Session ID or null
|
|
618
|
-
*/
|
|
619
|
-
function findActiveSession(registry) {
|
|
620
|
-
const active = Object.entries(registry.sessions)
|
|
621
|
-
.filter(([, meta]) => meta.status === 'active')
|
|
622
|
-
.sort((a, b) => new Date(b[1].started_at) - new Date(a[1].started_at));
|
|
623
|
-
|
|
624
|
-
return active.length > 0 ? active[0][0] : null;
|
|
625
|
-
}
|
|
626
|
-
|
|
627
|
-
/**
|
|
628
|
-
* Find all active sessions on a specific branch
|
|
629
|
-
* @param {string} branch - Branch name to search
|
|
630
|
-
* @returns {Object[]} Active sessions on that branch with id and metadata
|
|
631
|
-
*/
|
|
632
|
-
function findActiveSessionsOnBranch(branch) {
|
|
633
|
-
const root = getRepoRoot();
|
|
634
|
-
const registry = loadRegistry(root);
|
|
635
|
-
return Object.entries(registry.sessions)
|
|
636
|
-
.filter(([, meta]) => meta.status === 'active' && meta.branch === branch)
|
|
637
|
-
.map(([id, meta]) => ({ id, ...meta }));
|
|
638
|
-
}
|
|
639
|
-
|
|
640
|
-
module.exports = {
|
|
641
|
-
startSession,
|
|
642
|
-
checkpointSession,
|
|
643
|
-
endSession,
|
|
644
|
-
listSessions,
|
|
645
|
-
showSession,
|
|
646
|
-
getBriefing,
|
|
647
|
-
loadRegistry,
|
|
648
|
-
getRepoRoot,
|
|
649
|
-
SESSIONS_DIR,
|
|
650
|
-
REGISTRY_FILE,
|
|
651
|
-
CAPSULE_SCHEMA_VERSION,
|
|
652
|
-
findActiveSessionsOnBranch,
|
|
653
|
-
};
|
package/dist/sidecars/index.js
DELETED
|
@@ -1,33 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @fileoverview Sidecar Registry
|
|
3
|
-
* Central registry of all bounded governance sidecars.
|
|
4
|
-
* @author @darianrosebrook
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
const { analyzeSpecDrift } = require('./spec-drift');
|
|
8
|
-
const { diagnoseQualityGaps } = require('./quality-gaps');
|
|
9
|
-
const { draftWaiver } = require('./waiver-draft');
|
|
10
|
-
const { summarizeProvenance } = require('./provenance-summary');
|
|
11
|
-
const { createSidecarOutput, createNoStateOutput, formatSidecarText } = require('./schema');
|
|
12
|
-
|
|
13
|
-
/**
|
|
14
|
-
* Registry of available sidecars.
|
|
15
|
-
* Each entry has a function and a short description for help text.
|
|
16
|
-
*/
|
|
17
|
-
const SIDECARS = {
|
|
18
|
-
drift: { fn: analyzeSpecDrift, description: 'Analyze spec drift vs implementation evidence' },
|
|
19
|
-
gaps: { fn: diagnoseQualityGaps, description: 'Diagnose quality gaps preventing phase advancement' },
|
|
20
|
-
'waiver-draft': { fn: draftWaiver, description: 'Generate pre-filled waiver templates from gate failures' },
|
|
21
|
-
provenance: { fn: summarizeProvenance, description: 'Summarize work provenance for merge readiness' },
|
|
22
|
-
};
|
|
23
|
-
|
|
24
|
-
module.exports = {
|
|
25
|
-
SIDECARS,
|
|
26
|
-
analyzeSpecDrift,
|
|
27
|
-
diagnoseQualityGaps,
|
|
28
|
-
draftWaiver,
|
|
29
|
-
summarizeProvenance,
|
|
30
|
-
createSidecarOutput,
|
|
31
|
-
createNoStateOutput,
|
|
32
|
-
formatSidecarText,
|
|
33
|
-
};
|