cleargate 0.2.0 → 0.3.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/LICENSE +21 -0
- package/dist/MANIFEST.json +59 -17
- package/dist/admin-api/index.cjs +88 -1
- package/dist/admin-api/index.cjs.map +1 -1
- package/dist/admin-api/index.d.cts +105 -1
- package/dist/admin-api/index.d.ts +105 -1
- package/dist/admin-api/index.js +77 -1
- package/dist/admin-api/index.js.map +1 -1
- package/dist/bootstrap-root-FGWDICDT.js +130 -0
- package/dist/bootstrap-root-FGWDICDT.js.map +1 -0
- package/dist/chunk-OM4FAEA7.js +184 -0
- package/dist/chunk-OM4FAEA7.js.map +1 -0
- package/dist/cli.cjs +7995 -3984
- package/dist/cli.cjs.map +1 -1
- package/dist/cli.js +4062 -561
- package/dist/cli.js.map +1 -1
- package/dist/templates/cleargate-planning/.claude/agents/architect.md +72 -0
- package/dist/templates/cleargate-planning/.claude/agents/developer.md +45 -3
- package/dist/templates/cleargate-planning/.claude/agents/qa.md +7 -3
- package/dist/templates/cleargate-planning/.claude/agents/reporter.md +72 -75
- package/dist/templates/cleargate-planning/.claude/hooks/pending-task-sentinel.sh +204 -0
- package/dist/templates/cleargate-planning/.claude/hooks/pre-commit-surface-gate.sh +10 -0
- package/dist/templates/cleargate-planning/.claude/hooks/pre-commit-test-ratchet.sh +58 -0
- package/dist/templates/cleargate-planning/.claude/hooks/pre-commit.sh +19 -0
- package/dist/templates/cleargate-planning/.claude/hooks/session-start.sh +51 -0
- package/dist/templates/cleargate-planning/.claude/hooks/token-ledger.sh +1 -1
- package/dist/templates/cleargate-planning/.claude/settings.json +11 -0
- package/dist/templates/cleargate-planning/.claude/skills/flashcard/SKILL.md +31 -12
- package/dist/templates/cleargate-planning/.cleargate/FLASHCARD.md +2 -1
- package/dist/templates/cleargate-planning/.cleargate/config.example.yml +37 -0
- package/dist/templates/cleargate-planning/.cleargate/knowledge/cleargate-protocol.md +407 -0
- package/dist/templates/cleargate-planning/.cleargate/scripts/assert_story_files.mjs +146 -0
- package/dist/templates/cleargate-planning/.cleargate/scripts/close_sprint.mjs +250 -0
- package/dist/templates/cleargate-planning/.cleargate/scripts/constants.mjs +57 -0
- package/dist/templates/cleargate-planning/.cleargate/scripts/file_surface_diff.sh +320 -0
- package/dist/templates/cleargate-planning/.cleargate/scripts/gate-checks.json +15 -0
- package/dist/templates/cleargate-planning/.cleargate/scripts/init_gate_config.sh +38 -0
- package/dist/templates/cleargate-planning/.cleargate/scripts/init_sprint.mjs +187 -0
- package/dist/templates/cleargate-planning/.cleargate/scripts/pre_gate_common.sh +132 -0
- package/dist/templates/cleargate-planning/.cleargate/scripts/pre_gate_runner.sh +307 -0
- package/dist/templates/cleargate-planning/.cleargate/scripts/prefill_report.mjs +280 -0
- package/dist/templates/cleargate-planning/.cleargate/scripts/run_script.sh +123 -0
- package/dist/templates/cleargate-planning/.cleargate/scripts/state.schema.json +110 -0
- package/dist/templates/cleargate-planning/.cleargate/scripts/suggest_improvements.mjs +247 -0
- package/dist/templates/cleargate-planning/.cleargate/scripts/surface-whitelist.txt +27 -0
- package/dist/templates/cleargate-planning/.cleargate/scripts/test/test_assert_story_files.sh +261 -0
- package/dist/templates/cleargate-planning/.cleargate/scripts/test/test_file_surface.sh +210 -0
- package/dist/templates/cleargate-planning/.cleargate/scripts/test/test_flashcard_gate.sh +190 -0
- package/dist/templates/cleargate-planning/.cleargate/scripts/test/test_test_ratchet.sh +327 -0
- package/dist/templates/cleargate-planning/.cleargate/scripts/test_ratchet.mjs +261 -0
- package/dist/templates/cleargate-planning/.cleargate/scripts/update_state.mjs +154 -0
- package/dist/templates/cleargate-planning/.cleargate/scripts/validate_bounce_readiness.mjs +111 -0
- package/dist/templates/cleargate-planning/.cleargate/scripts/validate_state.mjs +164 -0
- package/dist/templates/cleargate-planning/.cleargate/templates/Bug.md +9 -0
- package/dist/templates/cleargate-planning/.cleargate/templates/CR.md +9 -0
- package/dist/templates/cleargate-planning/.cleargate/templates/Sprint Plan Template.md +29 -3
- package/dist/templates/cleargate-planning/.cleargate/templates/epic.md +9 -0
- package/dist/templates/cleargate-planning/.cleargate/templates/proposal.md +9 -0
- package/dist/templates/cleargate-planning/.cleargate/templates/sprint_context.md +42 -0
- package/dist/templates/cleargate-planning/.cleargate/templates/sprint_report.md +175 -0
- package/dist/templates/cleargate-planning/.cleargate/templates/story.md +29 -0
- package/dist/templates/cleargate-planning/CLAUDE.md +3 -0
- package/dist/templates/cleargate-planning/MANIFEST.json +59 -17
- package/dist/whoami-CX7CXJD5.js +76 -0
- package/dist/whoami-CX7CXJD5.js.map +1 -0
- package/package.json +6 -2
- package/templates/cleargate-planning/.claude/agents/architect.md +72 -0
- package/templates/cleargate-planning/.claude/agents/developer.md +45 -3
- package/templates/cleargate-planning/.claude/agents/qa.md +7 -3
- package/templates/cleargate-planning/.claude/agents/reporter.md +72 -75
- package/templates/cleargate-planning/.claude/hooks/pending-task-sentinel.sh +204 -0
- package/templates/cleargate-planning/.claude/hooks/pre-commit-surface-gate.sh +10 -0
- package/templates/cleargate-planning/.claude/hooks/pre-commit-test-ratchet.sh +58 -0
- package/templates/cleargate-planning/.claude/hooks/pre-commit.sh +19 -0
- package/templates/cleargate-planning/.claude/hooks/session-start.sh +51 -0
- package/templates/cleargate-planning/.claude/hooks/token-ledger.sh +1 -1
- package/templates/cleargate-planning/.claude/settings.json +11 -0
- package/templates/cleargate-planning/.claude/skills/flashcard/SKILL.md +31 -12
- package/templates/cleargate-planning/.cleargate/FLASHCARD.md +2 -1
- package/templates/cleargate-planning/.cleargate/config.example.yml +37 -0
- package/templates/cleargate-planning/.cleargate/knowledge/cleargate-protocol.md +407 -0
- package/templates/cleargate-planning/.cleargate/scripts/assert_story_files.mjs +146 -0
- package/templates/cleargate-planning/.cleargate/scripts/close_sprint.mjs +250 -0
- package/templates/cleargate-planning/.cleargate/scripts/constants.mjs +57 -0
- package/templates/cleargate-planning/.cleargate/scripts/file_surface_diff.sh +320 -0
- package/templates/cleargate-planning/.cleargate/scripts/gate-checks.json +15 -0
- package/templates/cleargate-planning/.cleargate/scripts/init_gate_config.sh +38 -0
- package/templates/cleargate-planning/.cleargate/scripts/init_sprint.mjs +187 -0
- package/templates/cleargate-planning/.cleargate/scripts/pre_gate_common.sh +132 -0
- package/templates/cleargate-planning/.cleargate/scripts/pre_gate_runner.sh +307 -0
- package/templates/cleargate-planning/.cleargate/scripts/prefill_report.mjs +280 -0
- package/templates/cleargate-planning/.cleargate/scripts/run_script.sh +123 -0
- package/templates/cleargate-planning/.cleargate/scripts/state.schema.json +110 -0
- package/templates/cleargate-planning/.cleargate/scripts/suggest_improvements.mjs +247 -0
- package/templates/cleargate-planning/.cleargate/scripts/surface-whitelist.txt +27 -0
- package/templates/cleargate-planning/.cleargate/scripts/test/test_assert_story_files.sh +261 -0
- package/templates/cleargate-planning/.cleargate/scripts/test/test_file_surface.sh +210 -0
- package/templates/cleargate-planning/.cleargate/scripts/test/test_flashcard_gate.sh +190 -0
- package/templates/cleargate-planning/.cleargate/scripts/test/test_test_ratchet.sh +327 -0
- package/templates/cleargate-planning/.cleargate/scripts/test_ratchet.mjs +261 -0
- package/templates/cleargate-planning/.cleargate/scripts/update_state.mjs +154 -0
- package/templates/cleargate-planning/.cleargate/scripts/validate_bounce_readiness.mjs +111 -0
- package/templates/cleargate-planning/.cleargate/scripts/validate_state.mjs +164 -0
- package/templates/cleargate-planning/.cleargate/templates/Bug.md +9 -0
- package/templates/cleargate-planning/.cleargate/templates/CR.md +9 -0
- package/templates/cleargate-planning/.cleargate/templates/Sprint Plan Template.md +29 -3
- package/templates/cleargate-planning/.cleargate/templates/epic.md +9 -0
- package/templates/cleargate-planning/.cleargate/templates/proposal.md +9 -0
- package/templates/cleargate-planning/.cleargate/templates/sprint_context.md +42 -0
- package/templates/cleargate-planning/.cleargate/templates/sprint_report.md +175 -0
- package/templates/cleargate-planning/.cleargate/templates/story.md +29 -0
- package/templates/cleargate-planning/CLAUDE.md +3 -0
- package/templates/cleargate-planning/MANIFEST.json +59 -17
|
@@ -0,0 +1,210 @@
|
|
|
1
|
+
#!/usr/bin/env bash
|
|
2
|
+
# test_file_surface.sh — Gherkin-style tests for file_surface_diff.sh
|
|
3
|
+
# Tests all 4 scenarios from STORY-014-01 §2.1
|
|
4
|
+
#
|
|
5
|
+
# Usage: bash .cleargate/scripts/test/test_file_surface.sh
|
|
6
|
+
# Exit: 0 if all pass, 1 if any fail
|
|
7
|
+
|
|
8
|
+
set -euo pipefail
|
|
9
|
+
|
|
10
|
+
# Navigate up 3 levels: test/ -> scripts/ -> .cleargate/ -> repo-root/
|
|
11
|
+
SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
|
|
12
|
+
REPO_ROOT="$(cd "${SCRIPT_DIR}/../../.." && pwd)"
|
|
13
|
+
SCRIPT="${REPO_ROOT}/.cleargate/scripts/file_surface_diff.sh"
|
|
14
|
+
PASS=0
|
|
15
|
+
FAIL=0
|
|
16
|
+
TOTAL=0
|
|
17
|
+
|
|
18
|
+
pass() { echo " PASS: $1"; PASS=$((PASS+1)); TOTAL=$((TOTAL+1)); }
|
|
19
|
+
fail() { echo " FAIL: $1 --- $2"; FAIL=$((FAIL+1)); TOTAL=$((TOTAL+1)); }
|
|
20
|
+
|
|
21
|
+
echo "REPO_ROOT: ${REPO_ROOT}"
|
|
22
|
+
echo "SCRIPT: ${SCRIPT}"
|
|
23
|
+
if [[ ! -f "${SCRIPT}" ]]; then
|
|
24
|
+
echo "ERROR: file_surface_diff.sh not found at ${SCRIPT}"
|
|
25
|
+
exit 1
|
|
26
|
+
fi
|
|
27
|
+
|
|
28
|
+
# ============================================================================
|
|
29
|
+
# Setup helpers
|
|
30
|
+
# ============================================================================
|
|
31
|
+
|
|
32
|
+
setup_git_repo() {
|
|
33
|
+
local dir="$1"
|
|
34
|
+
git init -q "${dir}"
|
|
35
|
+
git -C "${dir}" config user.email "test@test.com"
|
|
36
|
+
git -C "${dir}" config user.name "Test"
|
|
37
|
+
git -C "${dir}" commit -q --allow-empty -m "init"
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
create_story_file() {
|
|
41
|
+
local dir="$1"
|
|
42
|
+
shift
|
|
43
|
+
# remaining args are file paths to declare in §3.1
|
|
44
|
+
mkdir -p "${dir}/.cleargate/delivery/pending-sync"
|
|
45
|
+
local story="${dir}/.cleargate/delivery/pending-sync/STORY-014-01_Test.md"
|
|
46
|
+
{
|
|
47
|
+
echo "---"
|
|
48
|
+
echo "story_id: STORY-014-01"
|
|
49
|
+
echo "---"
|
|
50
|
+
echo ""
|
|
51
|
+
echo "# Test Story"
|
|
52
|
+
echo ""
|
|
53
|
+
echo "## 3. Implementation Guide"
|
|
54
|
+
echo ""
|
|
55
|
+
echo "### 3.1 Context & Files"
|
|
56
|
+
echo ""
|
|
57
|
+
echo "| Item | Value |"
|
|
58
|
+
echo "|---|---|"
|
|
59
|
+
for f in "$@"; do
|
|
60
|
+
echo "| New script | \`${f}\` |"
|
|
61
|
+
done
|
|
62
|
+
echo ""
|
|
63
|
+
echo "### 3.2 Technical Logic"
|
|
64
|
+
echo "test"
|
|
65
|
+
} > "${story}"
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
create_sprint_state() {
|
|
69
|
+
local dir="$1"
|
|
70
|
+
local mode="${2:-v2}"
|
|
71
|
+
mkdir -p "${dir}/.cleargate/sprint-runs/SPRINT-10"
|
|
72
|
+
echo "SPRINT-10" > "${dir}/.cleargate/sprint-runs/.active"
|
|
73
|
+
echo "{\"execution_mode\":\"${mode}\",\"stories\":{\"STORY-014-01\":{\"state\":\"In Progress\",\"updated_at\":\"2026-04-21T12:00:00Z\"}}}" > "${dir}/.cleargate/sprint-runs/SPRINT-10/state.json"
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
# ============================================================================
|
|
77
|
+
# Scenario 1: Gate catches off-surface edit
|
|
78
|
+
# ============================================================================
|
|
79
|
+
|
|
80
|
+
echo ""
|
|
81
|
+
echo "Scenario 1: Gate catches off-surface edit"
|
|
82
|
+
|
|
83
|
+
TMPDIR1="$(mktemp -d)"
|
|
84
|
+
setup_git_repo "${TMPDIR1}"
|
|
85
|
+
create_story_file "${TMPDIR1}" "hello.mjs" "README.md"
|
|
86
|
+
create_sprint_state "${TMPDIR1}" "v2"
|
|
87
|
+
|
|
88
|
+
touch "${TMPDIR1}/hello.mjs" "${TMPDIR1}/README.md" "${TMPDIR1}/unrelated.txt"
|
|
89
|
+
git -C "${TMPDIR1}" add hello.mjs README.md unrelated.txt
|
|
90
|
+
|
|
91
|
+
EXIT_CODE=0
|
|
92
|
+
STDERR_OUT="$(CLEARGATE_REPO_ROOT="${TMPDIR1}" bash "${SCRIPT}" 2>&1 >/dev/null)" || EXIT_CODE=$?
|
|
93
|
+
|
|
94
|
+
if [[ "${EXIT_CODE}" -ne 0 ]]; then
|
|
95
|
+
pass "exit code is non-zero"
|
|
96
|
+
else
|
|
97
|
+
fail "exit code is non-zero" "got exit 0"
|
|
98
|
+
fi
|
|
99
|
+
|
|
100
|
+
if echo "${STDERR_OUT}" | grep -q "unrelated.txt"; then
|
|
101
|
+
pass "stderr lists unrelated.txt as off-surface"
|
|
102
|
+
else
|
|
103
|
+
fail "stderr lists unrelated.txt as off-surface" "stderr was: ${STDERR_OUT}"
|
|
104
|
+
fi
|
|
105
|
+
|
|
106
|
+
rm -rf "${TMPDIR1}"
|
|
107
|
+
|
|
108
|
+
# ============================================================================
|
|
109
|
+
# Scenario 2: Gate passes when staged files match surface
|
|
110
|
+
# ============================================================================
|
|
111
|
+
|
|
112
|
+
echo ""
|
|
113
|
+
echo "Scenario 2: Gate passes when staged files match surface"
|
|
114
|
+
|
|
115
|
+
TMPDIR2="$(mktemp -d)"
|
|
116
|
+
setup_git_repo "${TMPDIR2}"
|
|
117
|
+
create_story_file "${TMPDIR2}" "hello.mjs" "README.md"
|
|
118
|
+
create_sprint_state "${TMPDIR2}" "v2"
|
|
119
|
+
|
|
120
|
+
touch "${TMPDIR2}/hello.mjs" "${TMPDIR2}/README.md"
|
|
121
|
+
git -C "${TMPDIR2}" add hello.mjs README.md
|
|
122
|
+
|
|
123
|
+
EXIT_CODE=0
|
|
124
|
+
CLEARGATE_REPO_ROOT="${TMPDIR2}" bash "${SCRIPT}" 2>/dev/null || EXIT_CODE=$?
|
|
125
|
+
|
|
126
|
+
if [[ "${EXIT_CODE}" -eq 0 ]]; then
|
|
127
|
+
pass "exit code is 0"
|
|
128
|
+
else
|
|
129
|
+
fail "exit code is 0" "got exit ${EXIT_CODE}"
|
|
130
|
+
fi
|
|
131
|
+
|
|
132
|
+
rm -rf "${TMPDIR2}"
|
|
133
|
+
|
|
134
|
+
# ============================================================================
|
|
135
|
+
# Scenario 3: Whitelist admits generated files
|
|
136
|
+
# ============================================================================
|
|
137
|
+
|
|
138
|
+
echo ""
|
|
139
|
+
echo "Scenario 3: Whitelist admits generated files"
|
|
140
|
+
|
|
141
|
+
TMPDIR3="$(mktemp -d)"
|
|
142
|
+
setup_git_repo "${TMPDIR3}"
|
|
143
|
+
create_story_file "${TMPDIR3}" "hello.mjs"
|
|
144
|
+
create_sprint_state "${TMPDIR3}" "v2"
|
|
145
|
+
|
|
146
|
+
mkdir -p "${TMPDIR3}/.cleargate/scripts"
|
|
147
|
+
{
|
|
148
|
+
echo "cleargate-planning/MANIFEST.json"
|
|
149
|
+
echo ".cleargate/hook-log/*"
|
|
150
|
+
} > "${TMPDIR3}/.cleargate/scripts/surface-whitelist.txt"
|
|
151
|
+
|
|
152
|
+
touch "${TMPDIR3}/hello.mjs"
|
|
153
|
+
mkdir -p "${TMPDIR3}/cleargate-planning"
|
|
154
|
+
touch "${TMPDIR3}/cleargate-planning/MANIFEST.json"
|
|
155
|
+
git -C "${TMPDIR3}" add hello.mjs cleargate-planning/MANIFEST.json
|
|
156
|
+
|
|
157
|
+
EXIT_CODE=0
|
|
158
|
+
CLEARGATE_REPO_ROOT="${TMPDIR3}" bash "${SCRIPT}" 2>/dev/null || EXIT_CODE=$?
|
|
159
|
+
|
|
160
|
+
if [[ "${EXIT_CODE}" -eq 0 ]]; then
|
|
161
|
+
pass "MANIFEST.json not flagged (whitelisted)"
|
|
162
|
+
else
|
|
163
|
+
fail "MANIFEST.json not flagged (whitelisted)" "got exit ${EXIT_CODE}"
|
|
164
|
+
fi
|
|
165
|
+
|
|
166
|
+
rm -rf "${TMPDIR3}"
|
|
167
|
+
|
|
168
|
+
# ============================================================================
|
|
169
|
+
# Scenario 4: v1 mode is advisory
|
|
170
|
+
# ============================================================================
|
|
171
|
+
|
|
172
|
+
echo ""
|
|
173
|
+
echo "Scenario 4: v1 mode is advisory"
|
|
174
|
+
|
|
175
|
+
TMPDIR4="$(mktemp -d)"
|
|
176
|
+
setup_git_repo "${TMPDIR4}"
|
|
177
|
+
create_story_file "${TMPDIR4}" "hello.mjs"
|
|
178
|
+
create_sprint_state "${TMPDIR4}" "v1"
|
|
179
|
+
|
|
180
|
+
touch "${TMPDIR4}/hello.mjs" "${TMPDIR4}/unrelated.txt"
|
|
181
|
+
git -C "${TMPDIR4}" add hello.mjs unrelated.txt
|
|
182
|
+
|
|
183
|
+
EXIT_CODE=0
|
|
184
|
+
STDERR_OUT="$(CLEARGATE_REPO_ROOT="${TMPDIR4}" bash "${SCRIPT}" 2>&1 >/dev/null)" || EXIT_CODE=$?
|
|
185
|
+
|
|
186
|
+
if [[ "${EXIT_CODE}" -eq 0 ]]; then
|
|
187
|
+
pass "v1 mode exits 0 (advisory)"
|
|
188
|
+
else
|
|
189
|
+
fail "v1 mode exits 0" "got exit ${EXIT_CODE}"
|
|
190
|
+
fi
|
|
191
|
+
|
|
192
|
+
if echo "${STDERR_OUT}" | grep -qi "warning"; then
|
|
193
|
+
pass "v1 mode prints warning"
|
|
194
|
+
else
|
|
195
|
+
fail "v1 mode prints warning" "stderr was: ${STDERR_OUT}"
|
|
196
|
+
fi
|
|
197
|
+
|
|
198
|
+
rm -rf "${TMPDIR4}"
|
|
199
|
+
|
|
200
|
+
# ============================================================================
|
|
201
|
+
# Summary
|
|
202
|
+
# ============================================================================
|
|
203
|
+
|
|
204
|
+
echo ""
|
|
205
|
+
echo "Results: ${PASS}/${TOTAL} passed, ${FAIL} failed"
|
|
206
|
+
|
|
207
|
+
if [[ "${FAIL}" -gt 0 ]]; then
|
|
208
|
+
exit 1
|
|
209
|
+
fi
|
|
210
|
+
exit 0
|
|
@@ -0,0 +1,190 @@
|
|
|
1
|
+
#!/usr/bin/env bash
|
|
2
|
+
# test_flashcard_gate.sh — STORY-013-06: Immediate Flashcard Hard-Gate
|
|
3
|
+
# Gherkin scenario: "Gate blocks next worktree creation until flashcards processed"
|
|
4
|
+
#
|
|
5
|
+
# Strategy: grep-based. Creates a synthetic dev-report fixture with
|
|
6
|
+
# flashcards_flagged, simulates a mock worktree-creation step, and asserts
|
|
7
|
+
# the gate contract documented in protocol §18 and the agent output-shape
|
|
8
|
+
# blocks in developer.md + qa.md.
|
|
9
|
+
#
|
|
10
|
+
# Usage: bash .cleargate/scripts/test/test_flashcard_gate.sh
|
|
11
|
+
# Exit: 0 if all assertions pass, 1 on first failure.
|
|
12
|
+
|
|
13
|
+
set -euo pipefail
|
|
14
|
+
|
|
15
|
+
REPO_ROOT="${CLEARGATE_REPO_ROOT:-$(cd "$(dirname "$0")/../../.." && pwd)}"
|
|
16
|
+
PASS=0
|
|
17
|
+
FAIL=0
|
|
18
|
+
|
|
19
|
+
pass() { echo "PASS: $1"; PASS=$((PASS + 1)); }
|
|
20
|
+
fail() { echo "FAIL: $1"; FAIL=$((FAIL + 1)); }
|
|
21
|
+
|
|
22
|
+
# ---------------------------------------------------------------------------
|
|
23
|
+
# Scenario: Gate blocks next worktree creation until flashcards processed
|
|
24
|
+
# ---------------------------------------------------------------------------
|
|
25
|
+
# §2.1 Gherkin from STORY-013-06:
|
|
26
|
+
# Given STORY-014-05 has just merged into sprint/S-XX
|
|
27
|
+
# And STORY-014-05-dev.md has flashcards_flagged: [...]
|
|
28
|
+
# When the orchestrator attempts to create .worktrees/STORY-014-06/
|
|
29
|
+
# Then the gate presents each flagged flashcard to the user for approval
|
|
30
|
+
# And upon approval each card is appended to .cleargate/FLASHCARD.md
|
|
31
|
+
# And only then does worktree creation proceed
|
|
32
|
+
#
|
|
33
|
+
# This script validates the contract (protocol §18 + output-shape fields)
|
|
34
|
+
# that makes the scenario enforceable. The gate logic itself is orchestrator-
|
|
35
|
+
# out-of-band (v1: informational; v2: mandatory). The test therefore checks:
|
|
36
|
+
# (a) developer.md output-shape contains flashcards_flagged field
|
|
37
|
+
# (b) qa.md output-shape contains flashcards_flagged field
|
|
38
|
+
# (c) protocol.md §18 exists with the correct heading
|
|
39
|
+
# (d) §18 specifies the approve/reject processing rule
|
|
40
|
+
# (e) §18 specifies the worktree creation gate
|
|
41
|
+
# (f) live vs mirror diff is empty for all three files
|
|
42
|
+
|
|
43
|
+
DEV_MD="$REPO_ROOT/.claude/agents/developer.md"
|
|
44
|
+
QA_MD="$REPO_ROOT/.claude/agents/qa.md"
|
|
45
|
+
PROTOCOL_MD="$REPO_ROOT/.cleargate/knowledge/cleargate-protocol.md"
|
|
46
|
+
|
|
47
|
+
PLANNING_DEV_MD="$REPO_ROOT/cleargate-planning/.claude/agents/developer.md"
|
|
48
|
+
PLANNING_QA_MD="$REPO_ROOT/cleargate-planning/.claude/agents/qa.md"
|
|
49
|
+
PLANNING_PROTOCOL_MD="$REPO_ROOT/cleargate-planning/.cleargate/knowledge/cleargate-protocol.md"
|
|
50
|
+
|
|
51
|
+
# (a) developer.md has flashcards_flagged in output-shape block
|
|
52
|
+
if grep -q "flashcards_flagged:" "$DEV_MD"; then
|
|
53
|
+
pass "developer.md output-shape contains flashcards_flagged field"
|
|
54
|
+
else
|
|
55
|
+
fail "developer.md missing flashcards_flagged field in output-shape"
|
|
56
|
+
fi
|
|
57
|
+
|
|
58
|
+
# (b) qa.md has flashcards_flagged in output-shape block
|
|
59
|
+
if grep -q "flashcards_flagged:" "$QA_MD"; then
|
|
60
|
+
pass "qa.md output-shape contains flashcards_flagged field"
|
|
61
|
+
else
|
|
62
|
+
fail "qa.md missing flashcards_flagged field in output-shape"
|
|
63
|
+
fi
|
|
64
|
+
|
|
65
|
+
# (c) protocol.md §18 heading exists
|
|
66
|
+
if grep -q "^## 18. Immediate Flashcard Gate (v2)" "$PROTOCOL_MD"; then
|
|
67
|
+
pass "protocol.md contains ## 18. Immediate Flashcard Gate (v2)"
|
|
68
|
+
else
|
|
69
|
+
fail "protocol.md missing ## 18. Immediate Flashcard Gate (v2)"
|
|
70
|
+
fi
|
|
71
|
+
|
|
72
|
+
# (d) §18 specifies approve + reject processing
|
|
73
|
+
if grep -q "Approve" "$PROTOCOL_MD" && grep -q "Reject" "$PROTOCOL_MD"; then
|
|
74
|
+
pass "protocol §18 documents Approve/Reject processing rule"
|
|
75
|
+
else
|
|
76
|
+
fail "protocol §18 missing Approve/Reject processing rule"
|
|
77
|
+
fi
|
|
78
|
+
|
|
79
|
+
# (e) §18 specifies the worktree creation gate
|
|
80
|
+
if grep -q "Worktree creation gate\|worktree creation gate\|MUST NOT.*worktree\|worktree.*MUST NOT" "$PROTOCOL_MD"; then
|
|
81
|
+
pass "protocol §18 documents worktree creation gate"
|
|
82
|
+
else
|
|
83
|
+
fail "protocol §18 missing worktree creation gate rule"
|
|
84
|
+
fi
|
|
85
|
+
|
|
86
|
+
# (f) qa.md says QA list is additive to Developer's
|
|
87
|
+
if grep -q "additive" "$QA_MD"; then
|
|
88
|
+
pass "qa.md notes that flashcards_flagged list is additive to Developer's"
|
|
89
|
+
else
|
|
90
|
+
fail "qa.md missing note that flashcards_flagged is additive"
|
|
91
|
+
fi
|
|
92
|
+
|
|
93
|
+
# (g) developer.md references protocol §18
|
|
94
|
+
if grep -q "protocol §18\|§18" "$DEV_MD"; then
|
|
95
|
+
pass "developer.md references protocol §18"
|
|
96
|
+
else
|
|
97
|
+
fail "developer.md does not reference protocol §18"
|
|
98
|
+
fi
|
|
99
|
+
|
|
100
|
+
# (h) qa.md references protocol §18
|
|
101
|
+
if grep -q "protocol §18\|§18" "$QA_MD"; then
|
|
102
|
+
pass "qa.md references protocol §18"
|
|
103
|
+
else
|
|
104
|
+
fail "qa.md does not reference protocol §18"
|
|
105
|
+
fi
|
|
106
|
+
|
|
107
|
+
# (i) three-surface diff: live developer.md vs mirror
|
|
108
|
+
if diff -q "$DEV_MD" "$PLANNING_DEV_MD" > /dev/null 2>&1; then
|
|
109
|
+
pass "developer.md live vs mirror diff is empty"
|
|
110
|
+
else
|
|
111
|
+
fail "developer.md live vs mirror diff is NOT empty"
|
|
112
|
+
fi
|
|
113
|
+
|
|
114
|
+
# (j) three-surface diff: live qa.md vs mirror
|
|
115
|
+
if diff -q "$QA_MD" "$PLANNING_QA_MD" > /dev/null 2>&1; then
|
|
116
|
+
pass "qa.md live vs mirror diff is empty"
|
|
117
|
+
else
|
|
118
|
+
fail "qa.md live vs mirror diff is NOT empty"
|
|
119
|
+
fi
|
|
120
|
+
|
|
121
|
+
# (k) three-surface diff: live protocol.md vs mirror
|
|
122
|
+
if diff -q "$PROTOCOL_MD" "$PLANNING_PROTOCOL_MD" > /dev/null 2>&1; then
|
|
123
|
+
pass "cleargate-protocol.md live vs mirror diff is empty"
|
|
124
|
+
else
|
|
125
|
+
fail "cleargate-protocol.md live vs mirror diff is NOT empty"
|
|
126
|
+
fi
|
|
127
|
+
|
|
128
|
+
# (l) synthetic fixture: dev report with flashcards_flagged triggers gate
|
|
129
|
+
# Simulate: write a temp dev report with flashcards_flagged, grep for it
|
|
130
|
+
TMPDIR_FIXTURE="$(mktemp -d)"
|
|
131
|
+
trap 'rm -rf "$TMPDIR_FIXTURE"' EXIT
|
|
132
|
+
|
|
133
|
+
cat > "$TMPDIR_FIXTURE/STORY-014-05-dev.md" << 'FIXTURE'
|
|
134
|
+
STORY: STORY-014-05
|
|
135
|
+
STATUS: done
|
|
136
|
+
COMMIT: abc1234
|
|
137
|
+
TYPECHECK: pass
|
|
138
|
+
TESTS: 5 passed, 0 failed
|
|
139
|
+
FILES_CHANGED: src/foo.ts
|
|
140
|
+
NOTES: all green
|
|
141
|
+
flashcards_flagged:
|
|
142
|
+
- "2026-04-22 · #test-harness · vitest fake-timers conflict with worker.spawn"
|
|
143
|
+
FIXTURE
|
|
144
|
+
|
|
145
|
+
# Assert the fixture has the flashcard entry
|
|
146
|
+
if grep -q "flashcards_flagged:" "$TMPDIR_FIXTURE/STORY-014-05-dev.md"; then
|
|
147
|
+
pass "synthetic dev-report fixture contains flashcards_flagged field"
|
|
148
|
+
else
|
|
149
|
+
fail "synthetic dev-report fixture missing flashcards_flagged field"
|
|
150
|
+
fi
|
|
151
|
+
|
|
152
|
+
# Assert extraction of the flagged card matches the FLASHCARD.md format
|
|
153
|
+
EXTRACTED=$(grep -A1 "flashcards_flagged:" "$TMPDIR_FIXTURE/STORY-014-05-dev.md" | tail -1 | sed 's/^[[:space:]]*- "//' | sed 's/"$//')
|
|
154
|
+
if echo "$EXTRACTED" | grep -qE "^[0-9]{4}-[0-9]{2}-[0-9]{2} · #[a-z-]+ · .+"; then
|
|
155
|
+
pass "extracted flashcard matches YYYY-MM-DD · #tag · lesson format"
|
|
156
|
+
else
|
|
157
|
+
fail "extracted flashcard does NOT match expected format: '$EXTRACTED'"
|
|
158
|
+
fi
|
|
159
|
+
|
|
160
|
+
# Mock worktree guard: confirm the gate would block by checking the field is non-empty
|
|
161
|
+
FLAGGED_COUNT=$(grep -c " - " "$TMPDIR_FIXTURE/STORY-014-05-dev.md" || true)
|
|
162
|
+
if [ "$FLAGGED_COUNT" -gt 0 ]; then
|
|
163
|
+
pass "gate detects non-empty flashcards_flagged — worktree creation would be blocked"
|
|
164
|
+
else
|
|
165
|
+
fail "gate failed to detect non-empty flashcards_flagged list"
|
|
166
|
+
fi
|
|
167
|
+
|
|
168
|
+
# Mock approval: append to a temp FLASHCARD.md and confirm it was written
|
|
169
|
+
TEMP_FLASHCARD="$TMPDIR_FIXTURE/FLASHCARD.md"
|
|
170
|
+
echo "# ClearGate Flashcards" > "$TEMP_FLASHCARD"
|
|
171
|
+
echo "" >> "$TEMP_FLASHCARD"
|
|
172
|
+
echo "$EXTRACTED" >> "$TEMP_FLASHCARD"
|
|
173
|
+
|
|
174
|
+
if grep -q "vitest fake-timers conflict" "$TEMP_FLASHCARD"; then
|
|
175
|
+
pass "approved flashcard appended to FLASHCARD.md mock"
|
|
176
|
+
else
|
|
177
|
+
fail "approved flashcard NOT found in FLASHCARD.md mock"
|
|
178
|
+
fi
|
|
179
|
+
|
|
180
|
+
# ---------------------------------------------------------------------------
|
|
181
|
+
# Summary
|
|
182
|
+
# ---------------------------------------------------------------------------
|
|
183
|
+
TOTAL=$((PASS + FAIL))
|
|
184
|
+
echo ""
|
|
185
|
+
echo "Results: $PASS/$TOTAL passed"
|
|
186
|
+
|
|
187
|
+
if [ "$FAIL" -gt 0 ]; then
|
|
188
|
+
exit 1
|
|
189
|
+
fi
|
|
190
|
+
exit 0
|