@paths.design/caws-cli 9.3.2 → 10.1.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 +71 -32
- package/dist/budget-derivation.js +221 -74
- package/dist/commands/archive.js +67 -28
- package/dist/commands/burnup.js +20 -11
- package/dist/commands/diagnose.js +34 -22
- package/dist/commands/evaluate.js +41 -15
- package/dist/commands/gates.js +149 -0
- package/dist/commands/init.js +150 -19
- package/dist/commands/iterate.js +81 -4
- package/dist/commands/parallel.js +4 -0
- package/dist/commands/plan.js +9 -19
- package/dist/commands/provenance.js +53 -17
- package/dist/commands/quality-monitor.js +64 -45
- package/dist/commands/scope.js +264 -0
- package/dist/commands/sidecar.js +74 -0
- package/dist/commands/specs.js +381 -45
- package/dist/commands/status.js +117 -9
- package/dist/commands/templates.js +0 -8
- package/dist/commands/tutorial.js +10 -9
- package/dist/commands/validate.js +70 -6
- package/dist/commands/verify-acs.js +48 -76
- package/dist/commands/waivers.js +212 -13
- package/dist/commands/worktree.js +131 -26
- package/dist/error-handler.js +2 -13
- package/dist/gates/budget-limit.js +121 -0
- package/dist/gates/feedback.js +260 -0
- package/dist/gates/format.js +179 -0
- package/dist/gates/god-object.js +117 -0
- package/dist/gates/pipeline.js +167 -0
- package/dist/gates/scope-boundary.js +93 -0
- package/dist/gates/spec-completeness.js +109 -0
- package/dist/gates/todo-detection.js +205 -0
- package/dist/index.js +157 -151
- package/dist/parallel/parallel-manager.js +3 -3
- package/dist/policy/PolicyManager.js +51 -17
- package/dist/scaffold/claude-hooks.js +24 -1
- package/dist/scaffold/git-hooks.js +45 -102
- package/dist/scaffold/index.js +4 -3
- package/dist/session/session-manager.js +105 -14
- package/dist/sidecars/index.js +33 -0
- package/dist/sidecars/listeners.js +40 -0
- package/dist/sidecars/provenance-summary.js +238 -0
- package/dist/sidecars/quality-gaps.js +258 -0
- package/dist/sidecars/schema.js +149 -0
- package/dist/sidecars/spec-drift.js +151 -0
- package/dist/sidecars/waiver-draft.js +176 -0
- package/dist/templates/.caws/schemas/policy.schema.json +112 -0
- package/dist/templates/.caws/schemas/scope.schema.json +3 -3
- package/dist/templates/.caws/schemas/waivers.schema.json +96 -20
- package/dist/templates/.caws/schemas/working-spec.schema.json +264 -57
- package/dist/templates/.caws/schemas/worktrees.schema.json +3 -1
- package/dist/templates/.caws/templates/working-spec.template.yml +10 -4
- package/dist/templates/.caws/tools/scope-guard.js +66 -15
- package/dist/templates/.claude/README.md +1 -1
- package/dist/templates/.claude/hooks/audit.sh +0 -0
- package/dist/templates/.claude/hooks/block-dangerous.sh +52 -11
- package/dist/templates/.claude/hooks/classify_command.py +592 -0
- package/dist/templates/.claude/hooks/doc-frontmatter-check.sh +173 -0
- package/dist/templates/.claude/hooks/protected-paths.sh +39 -0
- package/dist/templates/.claude/hooks/quality-check.sh +23 -10
- package/dist/templates/.claude/hooks/scope-guard.sh +136 -55
- package/dist/templates/.claude/hooks/session-caws-status.sh +2 -2
- package/dist/templates/.claude/hooks/session-log.sh +76 -3
- package/dist/templates/.claude/hooks/stop-worktree-check.sh +1 -1
- package/dist/templates/.claude/hooks/test_classify_command.py +370 -0
- package/dist/templates/.claude/hooks/test_wrapper_smoke.sh +96 -0
- package/dist/templates/.claude/hooks/worktree-guard.sh +2 -2
- package/dist/templates/.claude/hooks/worktree-write-guard.sh +97 -4
- package/dist/templates/.claude/settings.json +31 -0
- package/dist/templates/.cursor/hooks/caws-quality-check.sh +4 -4
- package/dist/templates/.cursor/hooks/caws-scope-guard.sh +1 -1
- package/dist/templates/.cursor/hooks/session-log.sh +924 -0
- package/dist/templates/.cursor/hooks.json +25 -0
- package/dist/templates/.cursor/rules/02-quality-gates.mdc +3 -5
- package/dist/templates/.cursor/rules/10-documentation-quality-standards.mdc +6 -11
- package/dist/templates/.cursor/rules/11-scope-management-waivers.mdc +14 -18
- package/dist/templates/.cursor/rules/12-implementation-completeness.mdc +4 -4
- package/dist/templates/.cursor/rules/13-language-agnostic-standards.mdc +3 -13
- package/dist/templates/.github/copilot-instructions.md +5 -5
- package/dist/templates/.idea/runConfigurations/CAWS_Evaluate.xml +1 -1
- package/dist/templates/.junie/guidelines.md +2 -2
- package/dist/templates/.vscode/settings.json +3 -1
- package/dist/templates/.windsurf/rules/caws-quality-standards.md +2 -2
- package/dist/templates/.windsurf/workflows/caws-guided-development.md +3 -3
- package/dist/templates/CLAUDE.md +77 -8
- package/dist/templates/agents.md +50 -9
- package/dist/templates/docs/README.md +8 -7
- package/dist/templates/scripts/new_feature.sh +80 -0
- package/dist/test-analysis.js +43 -30
- package/dist/tool-loader.js +1 -1
- package/dist/utils/agent-session.js +202 -0
- package/dist/utils/detection.js +8 -2
- package/dist/utils/event-log.js +584 -0
- package/dist/utils/event-renderer.js +521 -0
- package/dist/utils/finalization.js +7 -6
- package/dist/utils/gitignore-updater.js +3 -0
- package/dist/utils/lifecycle-events.js +94 -0
- package/dist/utils/quality-gates-utils.js +29 -44
- package/dist/utils/schema-validator.js +50 -0
- package/dist/utils/spec-resolver.js +93 -21
- package/dist/utils/working-state.js +530 -0
- package/dist/validation/spec-validation.js +191 -31
- package/dist/waivers-manager.js +144 -6
- package/dist/worktree/worktree-manager.js +598 -95
- package/package.json +9 -8
- package/templates/.caws/schemas/policy.schema.json +112 -0
- package/templates/.caws/schemas/scope.schema.json +3 -3
- package/templates/.caws/schemas/waivers.schema.json +96 -20
- package/templates/.caws/schemas/working-spec.schema.json +264 -57
- package/templates/.caws/schemas/worktrees.schema.json +3 -1
- package/templates/.caws/templates/working-spec.template.yml +10 -4
- package/templates/.caws/tools/scope-guard.js +66 -15
- package/templates/.claude/README.md +1 -1
- package/templates/.claude/hooks/block-dangerous.sh +52 -11
- package/templates/.claude/hooks/classify_command.py +592 -0
- package/templates/.claude/hooks/doc-frontmatter-check.sh +173 -0
- package/templates/.claude/hooks/protected-paths.sh +39 -0
- package/templates/.claude/hooks/quality-check.sh +23 -10
- package/templates/.claude/hooks/scope-guard.sh +136 -55
- package/templates/.claude/hooks/session-caws-status.sh +2 -2
- package/templates/.claude/hooks/session-log.sh +76 -3
- package/templates/.claude/hooks/stop-worktree-check.sh +1 -1
- package/templates/.claude/hooks/test_classify_command.py +370 -0
- package/templates/.claude/hooks/test_wrapper_smoke.sh +96 -0
- package/templates/.claude/hooks/worktree-guard.sh +2 -2
- package/templates/.claude/hooks/worktree-write-guard.sh +97 -4
- package/templates/.claude/settings.json +31 -0
- package/templates/.cursor/hooks/caws-quality-check.sh +4 -4
- package/templates/.cursor/hooks/caws-scope-guard.sh +1 -1
- package/templates/.cursor/hooks/session-log.sh +924 -0
- package/templates/.cursor/hooks.json +25 -0
- package/templates/.cursor/rules/02-quality-gates.mdc +3 -5
- package/templates/.cursor/rules/10-documentation-quality-standards.mdc +6 -11
- package/templates/.cursor/rules/11-scope-management-waivers.mdc +14 -18
- package/templates/.cursor/rules/12-implementation-completeness.mdc +4 -4
- package/templates/.cursor/rules/13-language-agnostic-standards.mdc +3 -13
- package/templates/.github/copilot-instructions.md +5 -5
- package/templates/.idea/runConfigurations/CAWS_Evaluate.xml +1 -1
- package/templates/.junie/guidelines.md +2 -2
- package/templates/.vscode/settings.json +3 -1
- package/templates/.windsurf/rules/caws-quality-standards.md +2 -2
- package/templates/.windsurf/workflows/caws-guided-development.md +3 -3
- package/templates/CLAUDE.md +77 -8
- package/templates/{AGENTS.md → agents.md} +50 -9
- package/templates/docs/README.md +8 -7
- package/templates/scripts/new_feature.sh +80 -0
- package/dist/budget-derivation.d.ts +0 -74
- package/dist/budget-derivation.d.ts.map +0 -1
- package/dist/cicd-optimizer.d.ts +0 -142
- package/dist/cicd-optimizer.d.ts.map +0 -1
- package/dist/commands/archive.d.ts +0 -51
- package/dist/commands/archive.d.ts.map +0 -1
- package/dist/commands/burnup.d.ts +0 -6
- package/dist/commands/burnup.d.ts.map +0 -1
- package/dist/commands/diagnose.d.ts +0 -52
- package/dist/commands/diagnose.d.ts.map +0 -1
- package/dist/commands/evaluate.d.ts +0 -8
- package/dist/commands/evaluate.d.ts.map +0 -1
- package/dist/commands/init.d.ts +0 -5
- package/dist/commands/init.d.ts.map +0 -1
- package/dist/commands/iterate.d.ts +0 -8
- package/dist/commands/iterate.d.ts.map +0 -1
- package/dist/commands/mode.d.ts +0 -25
- package/dist/commands/mode.d.ts.map +0 -1
- package/dist/commands/parallel.d.ts +0 -7
- package/dist/commands/parallel.d.ts.map +0 -1
- package/dist/commands/plan.d.ts +0 -49
- package/dist/commands/plan.d.ts.map +0 -1
- package/dist/commands/provenance.d.ts +0 -32
- package/dist/commands/provenance.d.ts.map +0 -1
- package/dist/commands/quality-gates.d.ts +0 -6
- package/dist/commands/quality-gates.d.ts.map +0 -1
- package/dist/commands/quality-gates.js +0 -444
- package/dist/commands/quality-monitor.d.ts +0 -17
- package/dist/commands/quality-monitor.d.ts.map +0 -1
- package/dist/commands/session.d.ts +0 -7
- package/dist/commands/session.d.ts.map +0 -1
- package/dist/commands/specs.d.ts +0 -77
- package/dist/commands/specs.d.ts.map +0 -1
- package/dist/commands/status.d.ts +0 -44
- package/dist/commands/status.d.ts.map +0 -1
- package/dist/commands/templates.d.ts +0 -74
- package/dist/commands/templates.d.ts.map +0 -1
- package/dist/commands/tool.d.ts +0 -13
- package/dist/commands/tool.d.ts.map +0 -1
- package/dist/commands/troubleshoot.d.ts +0 -8
- package/dist/commands/troubleshoot.d.ts.map +0 -1
- package/dist/commands/troubleshoot.js +0 -104
- package/dist/commands/tutorial.d.ts +0 -55
- package/dist/commands/tutorial.d.ts.map +0 -1
- package/dist/commands/validate.d.ts +0 -15
- package/dist/commands/validate.d.ts.map +0 -1
- package/dist/commands/waivers.d.ts +0 -8
- package/dist/commands/waivers.d.ts.map +0 -1
- package/dist/commands/workflow.d.ts +0 -85
- package/dist/commands/workflow.d.ts.map +0 -1
- package/dist/commands/worktree.d.ts +0 -7
- package/dist/commands/worktree.d.ts.map +0 -1
- package/dist/config/index.d.ts +0 -29
- package/dist/config/index.d.ts.map +0 -1
- package/dist/config/lite-scope.d.ts +0 -33
- package/dist/config/lite-scope.d.ts.map +0 -1
- package/dist/config/modes.d.ts +0 -264
- package/dist/config/modes.d.ts.map +0 -1
- package/dist/constants/spec-types.d.ts +0 -93
- package/dist/constants/spec-types.d.ts.map +0 -1
- package/dist/error-handler.d.ts +0 -151
- package/dist/error-handler.d.ts.map +0 -1
- package/dist/generators/jest-config-generator.d.ts +0 -32
- package/dist/generators/jest-config-generator.d.ts.map +0 -1
- package/dist/generators/jest-config.d.ts +0 -32
- package/dist/generators/jest-config.d.ts.map +0 -1
- package/dist/generators/jest-config.js +0 -242
- package/dist/generators/working-spec.d.ts +0 -13
- package/dist/generators/working-spec.d.ts.map +0 -1
- package/dist/index-new.d.ts +0 -5
- package/dist/index-new.d.ts.map +0 -1
- package/dist/index-new.js +0 -317
- package/dist/index.d.ts +0 -5
- package/dist/index.d.ts.map +0 -1
- package/dist/index.js.backup +0 -4711
- package/dist/minimal-cli.d.ts +0 -3
- package/dist/minimal-cli.d.ts.map +0 -1
- package/dist/parallel/parallel-manager.d.ts +0 -67
- package/dist/parallel/parallel-manager.d.ts.map +0 -1
- package/dist/policy/PolicyManager.d.ts +0 -104
- package/dist/policy/PolicyManager.d.ts.map +0 -1
- package/dist/scaffold/claude-hooks.d.ts +0 -28
- package/dist/scaffold/claude-hooks.d.ts.map +0 -1
- package/dist/scaffold/cursor-hooks.d.ts +0 -7
- package/dist/scaffold/cursor-hooks.d.ts.map +0 -1
- package/dist/scaffold/git-hooks.d.ts +0 -38
- package/dist/scaffold/git-hooks.d.ts.map +0 -1
- package/dist/scaffold/index.d.ts +0 -17
- package/dist/scaffold/index.d.ts.map +0 -1
- package/dist/session/session-manager.d.ts +0 -94
- package/dist/session/session-manager.d.ts.map +0 -1
- package/dist/spec/SpecFileManager.d.ts +0 -146
- package/dist/spec/SpecFileManager.d.ts.map +0 -1
- package/dist/templates/.cursor/hooks/caws-tool-validation.sh +0 -121
- package/dist/templates/.github/copilot/instructions.md +0 -311
- package/dist/test-analysis.d.ts +0 -231
- package/dist/test-analysis.d.ts.map +0 -1
- package/dist/tool-interface.d.ts +0 -236
- package/dist/tool-interface.d.ts.map +0 -1
- package/dist/tool-loader.d.ts +0 -77
- package/dist/tool-loader.d.ts.map +0 -1
- package/dist/tool-validator.d.ts +0 -72
- package/dist/tool-validator.d.ts.map +0 -1
- package/dist/utils/async-utils.d.ts +0 -73
- package/dist/utils/async-utils.d.ts.map +0 -1
- package/dist/utils/command-wrapper.d.ts +0 -66
- package/dist/utils/command-wrapper.d.ts.map +0 -1
- package/dist/utils/detection.d.ts +0 -14
- package/dist/utils/detection.d.ts.map +0 -1
- package/dist/utils/error-categories.d.ts +0 -52
- package/dist/utils/error-categories.d.ts.map +0 -1
- package/dist/utils/finalization.d.ts +0 -17
- package/dist/utils/finalization.d.ts.map +0 -1
- package/dist/utils/git-lock.d.ts +0 -13
- package/dist/utils/git-lock.d.ts.map +0 -1
- package/dist/utils/gitignore-updater.d.ts +0 -39
- package/dist/utils/gitignore-updater.d.ts.map +0 -1
- package/dist/utils/ide-detection.d.ts +0 -89
- package/dist/utils/ide-detection.d.ts.map +0 -1
- package/dist/utils/project-analysis.d.ts +0 -34
- package/dist/utils/project-analysis.d.ts.map +0 -1
- package/dist/utils/promise-utils.d.ts +0 -30
- package/dist/utils/promise-utils.d.ts.map +0 -1
- package/dist/utils/quality-gates-utils.d.ts +0 -49
- package/dist/utils/quality-gates-utils.d.ts.map +0 -1
- package/dist/utils/quality-gates.d.ts +0 -49
- package/dist/utils/quality-gates.d.ts.map +0 -1
- package/dist/utils/quality-gates.js +0 -402
- package/dist/utils/spec-resolver.d.ts +0 -80
- package/dist/utils/spec-resolver.d.ts.map +0 -1
- package/dist/utils/typescript-detector.d.ts +0 -66
- package/dist/utils/typescript-detector.d.ts.map +0 -1
- package/dist/utils/yaml-validation.d.ts +0 -32
- package/dist/utils/yaml-validation.d.ts.map +0 -1
- package/dist/validation/spec-validation.d.ts +0 -43
- package/dist/validation/spec-validation.d.ts.map +0 -1
- package/dist/waivers-manager.d.ts +0 -167
- package/dist/waivers-manager.d.ts.map +0 -1
- package/dist/worktree/worktree-manager.d.ts +0 -54
- package/dist/worktree/worktree-manager.d.ts.map +0 -1
|
@@ -0,0 +1,151 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @fileoverview Spec Drift Analysis Sidecar
|
|
3
|
+
* Compares implementation evidence (files_touched, AC results, gate results)
|
|
4
|
+
* against spec scope and acceptance criteria.
|
|
5
|
+
* @author @darianrosebrook
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
const _minimatch = require('minimatch');
|
|
9
|
+
const minimatch = typeof _minimatch === 'function'
|
|
10
|
+
? _minimatch
|
|
11
|
+
: (_minimatch.minimatch || (() => { throw new Error('minimatch export not found — expected v3 default or v5+ named export'); }));
|
|
12
|
+
const { createSidecarOutput, createNoStateOutput } = require('./schema');
|
|
13
|
+
const { getRecurrence } = require('../gates/feedback');
|
|
14
|
+
|
|
15
|
+
// ---------------------------------------------------------------------------
|
|
16
|
+
// Helpers
|
|
17
|
+
// ---------------------------------------------------------------------------
|
|
18
|
+
|
|
19
|
+
/**
|
|
20
|
+
* Check whether a file path is within the spec's declared scope.
|
|
21
|
+
* A file is in scope if it matches at least one scope.in pattern
|
|
22
|
+
* and does not match any scope.out pattern.
|
|
23
|
+
* @param {string} file
|
|
24
|
+
* @param {string[]} scopeIn
|
|
25
|
+
* @param {string[]} scopeOut
|
|
26
|
+
* @returns {boolean}
|
|
27
|
+
*/
|
|
28
|
+
function isInScope(file, scopeIn, scopeOut) {
|
|
29
|
+
if (scopeIn.length === 0) return true; // No scope.in means everything is allowed
|
|
30
|
+
const matchesIn = scopeIn.some(pattern => minimatch(file, pattern));
|
|
31
|
+
if (!matchesIn) return false;
|
|
32
|
+
const matchesOut = scopeOut.some(pattern => minimatch(file, pattern));
|
|
33
|
+
return !matchesOut;
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
/**
|
|
37
|
+
* Build a set of file paths that are referenced by any acceptance criterion.
|
|
38
|
+
* This is a rough heuristic: ACs that have results with file references, or
|
|
39
|
+
* whose descriptions mention paths present in files_touched.
|
|
40
|
+
* @param {object[]} acResults - state.acceptance_criteria.results
|
|
41
|
+
* @param {string[]} filesTouched
|
|
42
|
+
* @returns {Set<string>}
|
|
43
|
+
*/
|
|
44
|
+
function filesWithACCoverage(acResults, filesTouched) {
|
|
45
|
+
const covered = new Set();
|
|
46
|
+
if (!acResults || !filesTouched) return covered;
|
|
47
|
+
|
|
48
|
+
for (const result of acResults) {
|
|
49
|
+
// If a result references files explicitly, mark them covered
|
|
50
|
+
if (result.files) {
|
|
51
|
+
result.files.forEach(f => covered.add(f));
|
|
52
|
+
}
|
|
53
|
+
// Heuristic: any file_touched whose basename appears in the AC description
|
|
54
|
+
// is considered loosely covered
|
|
55
|
+
if (result.description) {
|
|
56
|
+
for (const file of filesTouched) {
|
|
57
|
+
const basename = file.split('/').pop();
|
|
58
|
+
if (result.description.includes(basename)) {
|
|
59
|
+
covered.add(file);
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
return covered;
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
/**
|
|
68
|
+
* Gather gate corroboration: how many scope_boundary failures exist in history.
|
|
69
|
+
* @param {object|null} state
|
|
70
|
+
* @returns {{ scope_failures: number, last_failure: string|null }}
|
|
71
|
+
*/
|
|
72
|
+
function getGateCorroboration(state) {
|
|
73
|
+
const recurrence = getRecurrence('scope_boundary', state);
|
|
74
|
+
if (!recurrence) return { scope_failures: 0, last_failure: null };
|
|
75
|
+
return { scope_failures: recurrence.count, last_failure: recurrence.lastSeen };
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
// ---------------------------------------------------------------------------
|
|
79
|
+
// Main analysis
|
|
80
|
+
// ---------------------------------------------------------------------------
|
|
81
|
+
|
|
82
|
+
/**
|
|
83
|
+
* Analyze spec drift by comparing implementation evidence against spec intent.
|
|
84
|
+
* Pure function -- no side effects, no writes.
|
|
85
|
+
* @param {object|null} state - Working state (from loadState)
|
|
86
|
+
* @param {object} spec - Resolved spec object
|
|
87
|
+
* @returns {object} Sidecar output envelope
|
|
88
|
+
*/
|
|
89
|
+
function analyzeSpecDrift(state, spec) {
|
|
90
|
+
if (!state) {
|
|
91
|
+
return createNoStateOutput('drift', spec.id);
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
const scopeIn = (spec.scope && spec.scope.in) || [];
|
|
95
|
+
const scopeOut = (spec.scope && spec.scope.out) || [];
|
|
96
|
+
const filesTouched = state.files_touched || [];
|
|
97
|
+
const acceptance = spec.acceptance || [];
|
|
98
|
+
const acResults = (state.acceptance_criteria && state.acceptance_criteria.results) || [];
|
|
99
|
+
|
|
100
|
+
// 1. Scope analysis -- find files outside declared scope
|
|
101
|
+
const outOfScopeFiles = filesTouched.filter(f => !isInScope(f, scopeIn, scopeOut));
|
|
102
|
+
|
|
103
|
+
// 2. Acceptance criteria cross-reference
|
|
104
|
+
const acResultMap = new Map();
|
|
105
|
+
for (const r of acResults) {
|
|
106
|
+
acResultMap.set(r.id, r);
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
const failingCriteria = [];
|
|
110
|
+
const missingEvidence = [];
|
|
111
|
+
|
|
112
|
+
for (const ac of acceptance) {
|
|
113
|
+
const result = acResultMap.get(ac.id);
|
|
114
|
+
if (!result || result.status === 'UNCHECKED') {
|
|
115
|
+
missingEvidence.push({ id: ac.id, description: ac.description || ac.text || '' });
|
|
116
|
+
} else if (result.status === 'FAIL') {
|
|
117
|
+
failingCriteria.push({ id: ac.id, description: ac.description || ac.text || '' });
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
// 3. Scope creep -- files with no AC coverage
|
|
122
|
+
const covered = filesWithACCoverage(acResults, filesTouched);
|
|
123
|
+
const scopeCreepFiles = filesTouched.filter(f => !covered.has(f));
|
|
124
|
+
|
|
125
|
+
// 4. Gate corroboration
|
|
126
|
+
const gateCorroboration = getGateCorroboration(state);
|
|
127
|
+
|
|
128
|
+
// 5. Drift detection
|
|
129
|
+
const driftDetected = outOfScopeFiles.length > 0 ||
|
|
130
|
+
failingCriteria.length > 0 ||
|
|
131
|
+
missingEvidence.length > 0;
|
|
132
|
+
|
|
133
|
+
// 6. Summary
|
|
134
|
+
const parts = [];
|
|
135
|
+
if (outOfScopeFiles.length > 0) parts.push(`${outOfScopeFiles.length} file(s) outside scope`);
|
|
136
|
+
if (failingCriteria.length > 0) parts.push(`${failingCriteria.length} AC failing`);
|
|
137
|
+
if (missingEvidence.length > 0) parts.push(`${missingEvidence.length} AC unchecked`);
|
|
138
|
+
const summary = parts.length > 0 ? parts.join(', ') : 'No drift detected';
|
|
139
|
+
|
|
140
|
+
return createSidecarOutput('drift', spec.id, {
|
|
141
|
+
drift_detected: driftDetected,
|
|
142
|
+
out_of_scope_files: outOfScopeFiles,
|
|
143
|
+
failing_criteria: failingCriteria,
|
|
144
|
+
missing_evidence: missingEvidence,
|
|
145
|
+
scope_creep_files: scopeCreepFiles,
|
|
146
|
+
gate_corroboration: gateCorroboration,
|
|
147
|
+
summary,
|
|
148
|
+
});
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
module.exports = { analyzeSpecDrift };
|
|
@@ -0,0 +1,176 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @fileoverview Waiver Drafting Assistance Sidecar
|
|
3
|
+
* Generates pre-filled waiver templates from gate failure context.
|
|
4
|
+
* Reduces boilerplate for the human reviewer.
|
|
5
|
+
* @author @darianrosebrook
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
const yaml = require('js-yaml');
|
|
9
|
+
const { getRecurrence, GATE_CATEGORIES } = require('../gates/feedback');
|
|
10
|
+
const { createSidecarOutput, createNoStateOutput } = require('./schema');
|
|
11
|
+
|
|
12
|
+
// ---------------------------------------------------------------------------
|
|
13
|
+
// Category → reason mapping
|
|
14
|
+
// ---------------------------------------------------------------------------
|
|
15
|
+
|
|
16
|
+
const CATEGORY_REASON_MAP = {
|
|
17
|
+
scope: 'third_party_constraint',
|
|
18
|
+
policy: 'infrastructure_limitation',
|
|
19
|
+
quality: 'experimental_feature',
|
|
20
|
+
architectural: 'legacy_integration',
|
|
21
|
+
};
|
|
22
|
+
|
|
23
|
+
// ---------------------------------------------------------------------------
|
|
24
|
+
// Impact level from recurrence count
|
|
25
|
+
// ---------------------------------------------------------------------------
|
|
26
|
+
|
|
27
|
+
function impactFromRecurrence(count) {
|
|
28
|
+
if (count <= 1) return 'low';
|
|
29
|
+
if (count <= 3) return 'medium';
|
|
30
|
+
return 'high';
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
// ---------------------------------------------------------------------------
|
|
34
|
+
// Template generation
|
|
35
|
+
// ---------------------------------------------------------------------------
|
|
36
|
+
|
|
37
|
+
/**
|
|
38
|
+
* Pad a description to meet the 50-char minimum.
|
|
39
|
+
* @param {string} desc
|
|
40
|
+
* @returns {string}
|
|
41
|
+
*/
|
|
42
|
+
function padDescription(desc) {
|
|
43
|
+
if (desc.length >= 50) return desc;
|
|
44
|
+
return desc + ' '.repeat(1) + 'This waiver allows temporary bypass while the issue is addressed.';
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
/**
|
|
48
|
+
* Clamp a string to max length.
|
|
49
|
+
* @param {string} str
|
|
50
|
+
* @param {number} max
|
|
51
|
+
* @returns {string}
|
|
52
|
+
*/
|
|
53
|
+
function clamp(str, max) {
|
|
54
|
+
if (str.length <= max) return str;
|
|
55
|
+
return str.slice(0, max - 3) + '...';
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
/**
|
|
59
|
+
* Build a waiver template for one failing gate.
|
|
60
|
+
* @param {object} gate - Gate result object
|
|
61
|
+
* @param {object} spec - Spec object
|
|
62
|
+
* @param {object} state - Working state (for recurrence)
|
|
63
|
+
* @returns {object} Draft object with gate, category, recurrence, template, yaml
|
|
64
|
+
*/
|
|
65
|
+
function buildDraft(gate, spec, state) {
|
|
66
|
+
const gateName = gate.name;
|
|
67
|
+
const category = GATE_CATEGORIES[gateName] || 'quality';
|
|
68
|
+
const reason = CATEGORY_REASON_MAP[category] || 'other';
|
|
69
|
+
|
|
70
|
+
const recurrenceInfo = getRecurrence(gateName, state);
|
|
71
|
+
const recurrenceCount = recurrenceInfo ? recurrenceInfo.count : 0;
|
|
72
|
+
const impact = impactFromRecurrence(recurrenceCount);
|
|
73
|
+
|
|
74
|
+
const now = new Date();
|
|
75
|
+
const expiresAt = new Date(now.getTime() + 7 * 24 * 60 * 60 * 1000);
|
|
76
|
+
|
|
77
|
+
// Build description from gate messages
|
|
78
|
+
const specTitle = spec?.title || spec?.id || 'unknown';
|
|
79
|
+
let rawTitle = `Waive ${gateName} for ${specTitle}`;
|
|
80
|
+
// Clamp title to 10-200 chars, pad if too short
|
|
81
|
+
if (rawTitle.length < 10) {
|
|
82
|
+
rawTitle = rawTitle + ' — temporary bypass';
|
|
83
|
+
}
|
|
84
|
+
const title = clamp(rawTitle, 200);
|
|
85
|
+
|
|
86
|
+
// Build description from messages
|
|
87
|
+
const messageText = gate.messages && gate.messages.length > 0
|
|
88
|
+
? gate.messages.join('. ')
|
|
89
|
+
: `Gate ${gateName} is failing`;
|
|
90
|
+
let description = `Gate ${gateName} is blocking: ${messageText}. This waiver allows temporary bypass while ${gateName} is addressed.`;
|
|
91
|
+
description = padDescription(description);
|
|
92
|
+
description = clamp(description, 1000);
|
|
93
|
+
|
|
94
|
+
const template = {
|
|
95
|
+
id: 'WV-XXXX',
|
|
96
|
+
title,
|
|
97
|
+
reason,
|
|
98
|
+
description,
|
|
99
|
+
gates: [gateName],
|
|
100
|
+
risk_assessment: {
|
|
101
|
+
impact_level: impact,
|
|
102
|
+
mitigation_plan: '[REQUIRED: Describe how you will address the underlying issue within the waiver period]',
|
|
103
|
+
review_required: impact === 'high' || impact === 'critical',
|
|
104
|
+
},
|
|
105
|
+
expires_at: expiresAt.toISOString(),
|
|
106
|
+
approved_by: '[REQUIRED]',
|
|
107
|
+
created_at: now.toISOString(),
|
|
108
|
+
metadata: { environment: 'development' },
|
|
109
|
+
};
|
|
110
|
+
|
|
111
|
+
const yamlStr = yaml.dump(template, { lineWidth: 120, noRefs: true });
|
|
112
|
+
|
|
113
|
+
return {
|
|
114
|
+
gate: gateName,
|
|
115
|
+
category,
|
|
116
|
+
recurrence: recurrenceCount,
|
|
117
|
+
template,
|
|
118
|
+
yaml: yamlStr,
|
|
119
|
+
};
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
// ---------------------------------------------------------------------------
|
|
123
|
+
// Main entry point
|
|
124
|
+
// ---------------------------------------------------------------------------
|
|
125
|
+
|
|
126
|
+
/**
|
|
127
|
+
* Generate waiver draft templates from gate failures.
|
|
128
|
+
* @param {object|null} state - Working state
|
|
129
|
+
* @param {object} [spec] - Spec object
|
|
130
|
+
* @param {object} [options={}] - Options
|
|
131
|
+
* @param {string} [options.gateName] - Filter to a specific gate
|
|
132
|
+
* @returns {object} Sidecar envelope
|
|
133
|
+
*/
|
|
134
|
+
function draftWaiver(state, spec, options = {}) {
|
|
135
|
+
const specId = spec?.id || 'unknown';
|
|
136
|
+
|
|
137
|
+
if (!state) {
|
|
138
|
+
return createNoStateOutput('waiver-draft', specId);
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
if (!state.gates || !state.gates.results) {
|
|
142
|
+
return createSidecarOutput('waiver-draft', specId, {
|
|
143
|
+
drafts: [],
|
|
144
|
+
instructions: '',
|
|
145
|
+
summary: 'No gate results available',
|
|
146
|
+
});
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
// Find failing gates
|
|
150
|
+
let failingGates = state.gates.results.filter(g => g.status === 'fail');
|
|
151
|
+
|
|
152
|
+
if (options.gateName) {
|
|
153
|
+
failingGates = failingGates.filter(g => g.name === options.gateName);
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
if (failingGates.length === 0) {
|
|
157
|
+
return createSidecarOutput('waiver-draft', specId, {
|
|
158
|
+
drafts: [],
|
|
159
|
+
instructions: '',
|
|
160
|
+
summary: 'No failing gates found',
|
|
161
|
+
});
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
const drafts = failingGates.map(gate => buildDraft(gate, spec, state));
|
|
165
|
+
|
|
166
|
+
const gateNames = drafts.map(d => d.gate).join(', ');
|
|
167
|
+
const summary = `${drafts.length} waiver draft${drafts.length !== 1 ? 's' : ''} generated for ${gateNames}`;
|
|
168
|
+
|
|
169
|
+
return createSidecarOutput('waiver-draft', specId, {
|
|
170
|
+
drafts,
|
|
171
|
+
instructions: 'Review and fill [REQUIRED] fields, then run: caws waivers create --file <path>',
|
|
172
|
+
summary,
|
|
173
|
+
});
|
|
174
|
+
}
|
|
175
|
+
|
|
176
|
+
module.exports = { draftWaiver };
|
|
@@ -0,0 +1,112 @@
|
|
|
1
|
+
{
|
|
2
|
+
"$schema": "http://json-schema.org/draft-07/schema#",
|
|
3
|
+
"type": "object",
|
|
4
|
+
"required": [
|
|
5
|
+
"version",
|
|
6
|
+
"risk_tiers",
|
|
7
|
+
"edit_rules"
|
|
8
|
+
],
|
|
9
|
+
"properties": {
|
|
10
|
+
"version": {
|
|
11
|
+
"type": "integer",
|
|
12
|
+
"enum": [
|
|
13
|
+
1
|
|
14
|
+
],
|
|
15
|
+
"description": "Policy schema version"
|
|
16
|
+
},
|
|
17
|
+
"risk_tiers": {
|
|
18
|
+
"type": "object",
|
|
19
|
+
"patternProperties": {
|
|
20
|
+
"^[1-3]$": {
|
|
21
|
+
"type": "object",
|
|
22
|
+
"required": [
|
|
23
|
+
"max_files",
|
|
24
|
+
"max_loc"
|
|
25
|
+
],
|
|
26
|
+
"properties": {
|
|
27
|
+
"max_files": {
|
|
28
|
+
"type": "integer",
|
|
29
|
+
"minimum": 1,
|
|
30
|
+
"description": "Maximum files allowed for this risk tier"
|
|
31
|
+
},
|
|
32
|
+
"max_loc": {
|
|
33
|
+
"type": "integer",
|
|
34
|
+
"minimum": 1,
|
|
35
|
+
"description": "Maximum lines of code allowed for this risk tier"
|
|
36
|
+
},
|
|
37
|
+
"description": {
|
|
38
|
+
"type": "string",
|
|
39
|
+
"description": "Human-readable description of the tier"
|
|
40
|
+
}
|
|
41
|
+
},
|
|
42
|
+
"additionalProperties": false
|
|
43
|
+
}
|
|
44
|
+
},
|
|
45
|
+
"additionalProperties": false,
|
|
46
|
+
"description": "Risk tier definitions with budget limits"
|
|
47
|
+
},
|
|
48
|
+
"edit_rules": {
|
|
49
|
+
"type": "object",
|
|
50
|
+
"required": [
|
|
51
|
+
"policy_and_code_same_pr",
|
|
52
|
+
"min_approvers_for_budget_raise"
|
|
53
|
+
],
|
|
54
|
+
"properties": {
|
|
55
|
+
"policy_and_code_same_pr": {
|
|
56
|
+
"type": "boolean",
|
|
57
|
+
"description": "Whether policy and code changes can be in the same PR"
|
|
58
|
+
},
|
|
59
|
+
"min_approvers_for_budget_raise": {
|
|
60
|
+
"type": "integer",
|
|
61
|
+
"minimum": 1,
|
|
62
|
+
"description": "Minimum approvers required for budget increases"
|
|
63
|
+
},
|
|
64
|
+
"require_signed_commits": {
|
|
65
|
+
"type": "boolean",
|
|
66
|
+
"description": "Whether signed commits are required for policy changes"
|
|
67
|
+
}
|
|
68
|
+
},
|
|
69
|
+
"additionalProperties": false,
|
|
70
|
+
"description": "Rules governing policy file edits"
|
|
71
|
+
},
|
|
72
|
+
"gates": {
|
|
73
|
+
"type": "object",
|
|
74
|
+
"patternProperties": {
|
|
75
|
+
"^.*$": {
|
|
76
|
+
"type": "object",
|
|
77
|
+
"required": [
|
|
78
|
+
"enabled"
|
|
79
|
+
],
|
|
80
|
+
"properties": {
|
|
81
|
+
"enabled": {
|
|
82
|
+
"type": "boolean",
|
|
83
|
+
"description": "Whether this gate is active"
|
|
84
|
+
},
|
|
85
|
+
"mode": {
|
|
86
|
+
"type": "string",
|
|
87
|
+
"enum": [
|
|
88
|
+
"warn",
|
|
89
|
+
"block",
|
|
90
|
+
"skip"
|
|
91
|
+
],
|
|
92
|
+
"description": "How the gate reports failures: warn, block, or skip entirely"
|
|
93
|
+
},
|
|
94
|
+
"description": {
|
|
95
|
+
"type": "string",
|
|
96
|
+
"description": "Human-readable description of the gate"
|
|
97
|
+
},
|
|
98
|
+
"thresholds": {
|
|
99
|
+
"type": "object",
|
|
100
|
+
"description": "Gate-specific thresholds (e.g. warning/critical limits)"
|
|
101
|
+
}
|
|
102
|
+
},
|
|
103
|
+
"additionalProperties": false
|
|
104
|
+
}
|
|
105
|
+
},
|
|
106
|
+
"additionalProperties": false,
|
|
107
|
+
"description": "Quality gate configurations"
|
|
108
|
+
}
|
|
109
|
+
},
|
|
110
|
+
"additionalProperties": false,
|
|
111
|
+
"title": "CAWS Policy"
|
|
112
|
+
}
|
|
@@ -1,14 +1,14 @@
|
|
|
1
1
|
{
|
|
2
2
|
"$schema": "https://json-schema.org/draft/2020-12/schema",
|
|
3
3
|
"title": "CAWS Lite Scope Configuration",
|
|
4
|
-
"description": "Scope configuration for CAWS lite mode — guardrails without YAML specs",
|
|
4
|
+
"description": "Scope configuration for CAWS lite mode — guardrails without YAML specs. This schema governs the standalone .caws/scope.json file ONLY; inline scope: blocks inside working-spec.yaml or feature specs are governed by the working-spec schema's scope sub-schema and do NOT invoke this schema. See CAWSFIX-11.",
|
|
5
5
|
"type": "object",
|
|
6
|
-
"required": ["
|
|
6
|
+
"required": ["allowedDirectories"],
|
|
7
7
|
"properties": {
|
|
8
8
|
"version": {
|
|
9
9
|
"type": "integer",
|
|
10
10
|
"const": 1,
|
|
11
|
-
"description": "Schema version"
|
|
11
|
+
"description": "Schema version. Optional for back-compat with scope.json files that predate versioning; the runtime (src/config/lite-scope.js) defaults to 1 when missing. If present, must be exactly 1. CAWSFIX-11 lifted `version` from the required list because no code path enforces a version mismatch — only the schema did, producing spurious warnings for pre-versioning scope.json files."
|
|
12
12
|
},
|
|
13
13
|
"allowedDirectories": {
|
|
14
14
|
"type": "array",
|
|
@@ -1,29 +1,105 @@
|
|
|
1
1
|
{
|
|
2
|
-
"$schema": "
|
|
3
|
-
"title": "CAWS
|
|
2
|
+
"$schema": "http://json-schema.org/draft-07/schema#",
|
|
3
|
+
"title": "CAWS Waiver",
|
|
4
|
+
"description": "Individual waiver file created by caws waivers create",
|
|
4
5
|
"type": "object",
|
|
5
|
-
"
|
|
6
|
-
|
|
6
|
+
"required": ["id", "applies_to", "gates", "delta", "reason_code", "expires_at", "risk_owner", "approvers", "status"],
|
|
7
|
+
"properties": {
|
|
8
|
+
"id": {
|
|
9
|
+
"type": "string",
|
|
10
|
+
"pattern": "^WV-\\d{4}$",
|
|
11
|
+
"description": "Waiver ID in format WV-XXXX"
|
|
12
|
+
},
|
|
13
|
+
"applies_to": {
|
|
14
|
+
"type": "string",
|
|
15
|
+
"description": "Spec ID or PR number this waiver applies to"
|
|
16
|
+
},
|
|
17
|
+
"gates": {
|
|
18
|
+
"type": "array",
|
|
19
|
+
"items": {
|
|
20
|
+
"type": "string",
|
|
21
|
+
"enum": ["budget_limit", "spec_completeness", "contract_compliance", "coverage_threshold", "mutation_threshold", "security_scan", "accessibility_check", "performance_budget", "scope_boundary"]
|
|
22
|
+
},
|
|
23
|
+
"minItems": 1,
|
|
24
|
+
"description": "Quality gates to waive"
|
|
25
|
+
},
|
|
26
|
+
"delta": {
|
|
7
27
|
"type": "object",
|
|
8
|
-
"
|
|
28
|
+
"description": "Additive budget deltas (only positive values allowed)",
|
|
9
29
|
"properties": {
|
|
10
|
-
"
|
|
11
|
-
"type": "
|
|
12
|
-
"
|
|
30
|
+
"max_files": {
|
|
31
|
+
"type": "integer",
|
|
32
|
+
"minimum": 0,
|
|
33
|
+
"description": "Additional files allowed"
|
|
13
34
|
},
|
|
14
|
-
"
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
"ticket_url": { "type": "string", "format": "uri" },
|
|
19
|
-
"approved_by": { "type": "string" },
|
|
20
|
-
"created_at": { "type": "string", "format": "date-time" },
|
|
21
|
-
"status": {
|
|
22
|
-
"type": "string",
|
|
23
|
-
"enum": ["active", "expired", "revoked"],
|
|
24
|
-
"default": "active"
|
|
35
|
+
"max_loc": {
|
|
36
|
+
"type": "integer",
|
|
37
|
+
"minimum": 0,
|
|
38
|
+
"description": "Additional lines of code allowed"
|
|
25
39
|
}
|
|
26
|
-
}
|
|
40
|
+
},
|
|
41
|
+
"additionalProperties": false
|
|
42
|
+
},
|
|
43
|
+
"reason_code": {
|
|
44
|
+
"type": "string",
|
|
45
|
+
"enum": ["emergency_hotfix", "legacy_integration", "experimental_feature", "third_party_constraint", "performance_critical", "security_patch", "infrastructure_limitation", "architectural_refactor", "other"],
|
|
46
|
+
"description": "Controlled vocabulary for waiver reasons"
|
|
47
|
+
},
|
|
48
|
+
"description": {
|
|
49
|
+
"type": "string",
|
|
50
|
+
"minLength": 50,
|
|
51
|
+
"maxLength": 1000,
|
|
52
|
+
"description": "Detailed explanation of why waiver is needed"
|
|
53
|
+
},
|
|
54
|
+
"mitigation": {
|
|
55
|
+
"type": "string",
|
|
56
|
+
"minLength": 50,
|
|
57
|
+
"description": "Plan to address the underlying issue"
|
|
58
|
+
},
|
|
59
|
+
"expires_at": {
|
|
60
|
+
"type": "string",
|
|
61
|
+
"format": "date-time",
|
|
62
|
+
"description": "ISO 8601 datetime when waiver expires"
|
|
63
|
+
},
|
|
64
|
+
"risk_owner": {
|
|
65
|
+
"type": "string",
|
|
66
|
+
"description": "Person/entity responsible for managing this risk"
|
|
67
|
+
},
|
|
68
|
+
"approvers": {
|
|
69
|
+
"type": "array",
|
|
70
|
+
"items": {
|
|
71
|
+
"type": "object",
|
|
72
|
+
"required": ["handle"],
|
|
73
|
+
"properties": {
|
|
74
|
+
"handle": {
|
|
75
|
+
"type": "string",
|
|
76
|
+
"description": "GitHub handle or email of approver"
|
|
77
|
+
},
|
|
78
|
+
"approved_at": {
|
|
79
|
+
"type": "string",
|
|
80
|
+
"format": "date-time",
|
|
81
|
+
"description": "When this approval was given"
|
|
82
|
+
}
|
|
83
|
+
},
|
|
84
|
+
"additionalProperties": false
|
|
85
|
+
},
|
|
86
|
+
"minItems": 1,
|
|
87
|
+
"description": "List of people who approved this waiver"
|
|
88
|
+
},
|
|
89
|
+
"status": {
|
|
90
|
+
"type": "string",
|
|
91
|
+
"enum": ["proposed", "active", "expired", "revoked"],
|
|
92
|
+
"description": "Current status of the waiver"
|
|
93
|
+
},
|
|
94
|
+
"metadata": {
|
|
95
|
+
"type": "object",
|
|
96
|
+
"properties": {
|
|
97
|
+
"related_pr": { "type": "string" },
|
|
98
|
+
"related_issue": { "type": "string" },
|
|
99
|
+
"environment": { "type": "string", "enum": ["development", "staging", "production"] },
|
|
100
|
+
"urgency": { "type": "string", "enum": ["low", "normal", "high", "critical"] }
|
|
101
|
+
},
|
|
102
|
+
"additionalProperties": false
|
|
27
103
|
}
|
|
28
104
|
},
|
|
29
105
|
"additionalProperties": false
|