Files
agentic-dev-template/scripts/work/state-sync-guard.mjs
Danijel Martinek bae4b66fa4 refactor(work): drop date prefixes + move _state.json into _system/
Convention shift: epic folders + PRD filenames + frontmatter id
fields are now bare slugs. The created: timestamp (Phase 2) carries
the date; folder names don't repeat it. A future <task-id>-<slug>
shape (e.g. ClickUp) lands cleanly when that integration ships.

Renames (git mv preserves history):
- docs/work/2026-05-13-binder-wrap-helper/
    -> docs/work/binder-wrap-helper/
- docs/work/2026-05-14-library-evaluation-policy/
    -> docs/work/library-evaluation-policy/
- docs/work/2026-05-14-ci-security-and-supply-chain/
    -> docs/work/ci-security-and-supply-chain/
- docs/work/prds/2026-05-13-binder-wrap-helper.prd.md
    -> docs/work/prds/binder-wrap-helper.prd.md
- docs/work/prds/2026-05-13-coverage-architecture.prd.md
    -> docs/work/prds/coverage-architecture.prd.md
- docs/work/prds/2026-05-14-library-evaluation-policy.prd.md
    -> docs/work/prds/library-evaluation-policy.prd.md
- docs/work/prds/2026-05-14-ci-security-and-supply-chain.prd.md
    -> docs/work/prds/ci-security-and-supply-chain.prd.md

Frontmatter updates inside the renamed files: epic id, epic prd,
story epic, PRD id, PRD builds-on all drop date prefixes.

System folder + state file move:
- New docs/work/_system/ holds framework-managed state.
- docs/work/_state.json -> docs/work/_system/_state.json.
- state-builder.mjs adds _system to SKIP_FOLDERS.
- cli.mjs + state-sync-guard.mjs + .husky/pre-commit point at the
  new path.

template-reset-v1 epic deleted entirely (one-off cleanup epic from
the pre-date-convention era; status was already done).

Generator-template updates (so new artifacts ship in the right
shape):
- .sandcastle/decomposer.prompt.md emits bare-slug folder names +
  ISO created: timestamp.
- .claude/skills/to-prd/SKILL.md template uses bare-slug filename +
  bare-slug id field + ISO created: timestamp.

Doc reference updates: glossary, runbook, agent-first-workflow-
and-conformance, reviewer prompt, ADR-020, ADR-022, ADR-023 all
point at the new paths/slugs.
2026-05-14 21:16:51 +02:00

69 lines
2.3 KiB
JavaScript

#!/usr/bin/env node
/**
* Pre-commit guard: refuses the commit if docs/work/_system/_state.json is staged
* but is not byte-identical to what `pnpm work rebuild-state` would emit
* given the current markdown content of docs/work/.
*
* Run from .husky/pre-commit AFTER lint-staged and AFTER the conditional
* rebuild-state + re-stage step. By that point the staged _state.json
* should already match the rebuild output. This script is the safety net
* for the case where someone hand-edits _state.json without going through
* rebuild-state.
*
* Exit codes:
* 0 — _state.json is in sync (or not staged at all)
* 1 — _state.json is staged but out of sync
*/
import fs from "node:fs";
import path from "node:path";
import { fileURLToPath } from "node:url";
import { execSync } from "node:child_process";
import { buildState } from "./state-builder.mjs";
const __dirname = path.dirname(fileURLToPath(import.meta.url));
const REPO_ROOT = path.resolve(__dirname, "..", "..");
const WORK_ROOT = path.join(REPO_ROOT, "docs", "work");
const STATE_FILE = path.join(WORK_ROOT, "_system", "_state.json");
function stagedFiles() {
const out = execSync("git diff --cached --name-only", {
cwd: REPO_ROOT,
encoding: "utf8",
});
return out.split("\n").filter(Boolean);
}
function main() {
const staged = stagedFiles();
const stateRel = path.relative(REPO_ROOT, STATE_FILE);
if (
!staged.includes(stateRel) &&
!staged.some((f) => f.startsWith("docs/work/") && f.endsWith(".md"))
) {
process.exit(0);
}
if (!fs.existsSync(STATE_FILE)) {
console.error(
"✗ state-sync-guard: docs/work/_system/_state.json missing. Run `pnpm work rebuild-state`.",
);
process.exit(1);
}
const onDisk = fs.readFileSync(STATE_FILE, "utf8");
const fresh = JSON.stringify(buildState(WORK_ROOT), null, 2) + "\n";
// The `updated_at` timestamp will always differ. Strip it from both sides
// before comparing.
const stripUpdatedAt = (s) =>
s.replace(/"updated_at":\s*"[^"]+",?\s*\n?/, "");
if (stripUpdatedAt(onDisk) === stripUpdatedAt(fresh)) {
process.exit(0);
}
console.error(
"✗ state-sync-guard: docs/work/_system/_state.json is out of sync with markdown.",
);
console.error(" Run: pnpm work rebuild-state");
console.error(" Then: git add docs/work/_system/_state.json");
process.exit(1);
}
main();