qualia-framework 2.6.0 → 3.2.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.
Files changed (321) hide show
  1. package/CLAUDE.md +64 -0
  2. package/README.md +103 -30
  3. package/agents/builder.md +110 -0
  4. package/agents/planner.md +134 -0
  5. package/agents/qa-browser.md +186 -0
  6. package/agents/verifier.md +221 -0
  7. package/bin/cli.js +336 -531
  8. package/bin/install.js +570 -0
  9. package/bin/qualia-ui.js +299 -0
  10. package/bin/state.js +630 -0
  11. package/bin/statusline.js +252 -0
  12. package/guide.md +63 -0
  13. package/hooks/auto-update.js +139 -0
  14. package/hooks/branch-guard.js +47 -0
  15. package/hooks/migration-guard.js +60 -0
  16. package/hooks/pre-compact.js +32 -0
  17. package/hooks/pre-deploy-gate.js +110 -0
  18. package/hooks/pre-push.js +33 -0
  19. package/hooks/session-start.js +170 -0
  20. package/package.json +29 -20
  21. package/rules/design-reference.md +179 -0
  22. package/rules/frontend.md +126 -0
  23. package/skills/qualia/SKILL.md +87 -0
  24. package/skills/qualia-build/SKILL.md +97 -0
  25. package/skills/qualia-debug/SKILL.md +87 -0
  26. package/skills/qualia-design/SKILL.md +93 -0
  27. package/skills/qualia-handoff/SKILL.md +66 -0
  28. package/skills/qualia-idk/SKILL.md +8 -0
  29. package/skills/qualia-learn/SKILL.md +88 -0
  30. package/skills/qualia-new/SKILL.md +323 -0
  31. package/{framework/skills → skills}/qualia-optimize/SKILL.md +1 -1
  32. package/skills/qualia-pause/SKILL.md +63 -0
  33. package/skills/qualia-plan/SKILL.md +101 -0
  34. package/skills/qualia-polish/SKILL.md +157 -0
  35. package/skills/qualia-quick/SKILL.md +37 -0
  36. package/skills/qualia-report/SKILL.md +105 -0
  37. package/skills/qualia-resume/SKILL.md +49 -0
  38. package/skills/qualia-review/SKILL.md +76 -0
  39. package/skills/qualia-ship/SKILL.md +90 -0
  40. package/skills/qualia-skill-new/SKILL.md +167 -0
  41. package/skills/qualia-task/SKILL.md +91 -0
  42. package/skills/qualia-verify/SKILL.md +113 -0
  43. package/templates/DESIGN.md +137 -0
  44. package/templates/plan.md +28 -0
  45. package/templates/project.md +22 -0
  46. package/templates/state.md +27 -0
  47. package/templates/tracking.json +20 -0
  48. package/tests/bin.test.sh +673 -0
  49. package/tests/hooks.test.sh +315 -0
  50. package/tests/state.test.sh +535 -0
  51. package/tests/statusline.test.sh +243 -0
  52. package/bin/collect-metrics.sh +0 -62
  53. package/framework/.claudeignore +0 -51
  54. package/framework/CLAUDE.md +0 -51
  55. package/framework/MCP_SETUP.md +0 -229
  56. package/framework/agents/architecture-strategist.md +0 -53
  57. package/framework/agents/backend-agent.md +0 -150
  58. package/framework/agents/code-simplicity-reviewer.md +0 -86
  59. package/framework/agents/frontend-agent.md +0 -111
  60. package/framework/agents/kieran-typescript-reviewer.md +0 -96
  61. package/framework/agents/performance-oracle.md +0 -111
  62. package/framework/agents/qualia-codebase-mapper.md +0 -761
  63. package/framework/agents/qualia-debugger.md +0 -1204
  64. package/framework/agents/qualia-executor.md +0 -882
  65. package/framework/agents/qualia-integration-checker.md +0 -424
  66. package/framework/agents/qualia-phase-researcher.md +0 -457
  67. package/framework/agents/qualia-plan-checker.md +0 -700
  68. package/framework/agents/qualia-planner.md +0 -1245
  69. package/framework/agents/qualia-project-researcher.md +0 -603
  70. package/framework/agents/qualia-research-synthesizer.md +0 -200
  71. package/framework/agents/qualia-roadmapper.md +0 -606
  72. package/framework/agents/qualia-verifier.md +0 -686
  73. package/framework/agents/red-team-qa.md +0 -130
  74. package/framework/agents/security-auditor.md +0 -72
  75. package/framework/agents/team-orchestrator.md +0 -229
  76. package/framework/agents/teams/framework-audit-team.md +0 -66
  77. package/framework/agents/teams/full-stack-team.md +0 -48
  78. package/framework/agents/teams/optimize-team.md +0 -53
  79. package/framework/agents/teams/review-team.md +0 -70
  80. package/framework/agents/teams/ship-team.md +0 -86
  81. package/framework/agents/test-agent.md +0 -182
  82. package/framework/hooks/auto-format.sh +0 -54
  83. package/framework/hooks/block-env-edit.sh +0 -42
  84. package/framework/hooks/branch-guard.sh +0 -43
  85. package/framework/hooks/confirm-delete.sh +0 -59
  86. package/framework/hooks/migration-validate.sh +0 -77
  87. package/framework/hooks/notification-speak.sh +0 -16
  88. package/framework/hooks/pre-commit.sh +0 -100
  89. package/framework/hooks/pre-compact.sh +0 -56
  90. package/framework/hooks/pre-deploy-gate.sh +0 -160
  91. package/framework/hooks/qualia-colors.sh +0 -32
  92. package/framework/hooks/retention-cleanup.sh +0 -62
  93. package/framework/hooks/save-session-state.sh +0 -185
  94. package/framework/hooks/session-context-loader.sh +0 -96
  95. package/framework/hooks/session-learn.sh +0 -32
  96. package/framework/hooks/skill-announce.sh +0 -123
  97. package/framework/hooks/tool-error-announce.sh +0 -27
  98. package/framework/install.ps1 +0 -323
  99. package/framework/install.sh +0 -313
  100. package/framework/qualia-framework/VERSION +0 -1
  101. package/framework/qualia-framework/assets/qualia-logo.png +0 -0
  102. package/framework/qualia-framework/bin/collect-metrics.sh +0 -67
  103. package/framework/qualia-framework/bin/generate-report-docx.py +0 -429
  104. package/framework/qualia-framework/bin/qualia-tools.js +0 -2201
  105. package/framework/qualia-framework/bin/qualia-tools.test.js +0 -1054
  106. package/framework/qualia-framework/references/checkpoints.md +0 -775
  107. package/framework/qualia-framework/references/completion-checklists.md +0 -359
  108. package/framework/qualia-framework/references/continuation-format.md +0 -249
  109. package/framework/qualia-framework/references/continuation-prompt.md +0 -97
  110. package/framework/qualia-framework/references/decimal-phase-calculation.md +0 -65
  111. package/framework/qualia-framework/references/design-quality.md +0 -56
  112. package/framework/qualia-framework/references/employee-guide.md +0 -167
  113. package/framework/qualia-framework/references/git-integration.md +0 -254
  114. package/framework/qualia-framework/references/git-planning-commit.md +0 -50
  115. package/framework/qualia-framework/references/model-profile-resolution.md +0 -32
  116. package/framework/qualia-framework/references/model-profiles.md +0 -73
  117. package/framework/qualia-framework/references/phase-argument-parsing.md +0 -61
  118. package/framework/qualia-framework/references/planning-config.md +0 -195
  119. package/framework/qualia-framework/references/questioning.md +0 -141
  120. package/framework/qualia-framework/references/tdd.md +0 -263
  121. package/framework/qualia-framework/references/ui-brand.md +0 -160
  122. package/framework/qualia-framework/references/verification-patterns.md +0 -612
  123. package/framework/qualia-framework/templates/DEBUG.md +0 -159
  124. package/framework/qualia-framework/templates/DESIGN.md +0 -81
  125. package/framework/qualia-framework/templates/UAT.md +0 -247
  126. package/framework/qualia-framework/templates/codebase/architecture.md +0 -255
  127. package/framework/qualia-framework/templates/codebase/concerns.md +0 -310
  128. package/framework/qualia-framework/templates/codebase/conventions.md +0 -307
  129. package/framework/qualia-framework/templates/codebase/integrations.md +0 -280
  130. package/framework/qualia-framework/templates/codebase/stack.md +0 -186
  131. package/framework/qualia-framework/templates/codebase/structure.md +0 -285
  132. package/framework/qualia-framework/templates/codebase/testing.md +0 -480
  133. package/framework/qualia-framework/templates/config.json +0 -35
  134. package/framework/qualia-framework/templates/context.md +0 -283
  135. package/framework/qualia-framework/templates/continue-here.md +0 -78
  136. package/framework/qualia-framework/templates/debug-subagent-prompt.md +0 -91
  137. package/framework/qualia-framework/templates/discovery.md +0 -146
  138. package/framework/qualia-framework/templates/lab-notes.md +0 -16
  139. package/framework/qualia-framework/templates/milestone-archive.md +0 -123
  140. package/framework/qualia-framework/templates/milestone.md +0 -115
  141. package/framework/qualia-framework/templates/phase-prompt.md +0 -567
  142. package/framework/qualia-framework/templates/planner-subagent-prompt.md +0 -117
  143. package/framework/qualia-framework/templates/project.md +0 -184
  144. package/framework/qualia-framework/templates/projects/ai-agent.md +0 -156
  145. package/framework/qualia-framework/templates/projects/mobile-app.md +0 -181
  146. package/framework/qualia-framework/templates/projects/voice-agent.md +0 -134
  147. package/framework/qualia-framework/templates/projects/website.md +0 -137
  148. package/framework/qualia-framework/templates/requirements.md +0 -231
  149. package/framework/qualia-framework/templates/research-project/ARCHITECTURE.md +0 -204
  150. package/framework/qualia-framework/templates/research-project/FEATURES.md +0 -147
  151. package/framework/qualia-framework/templates/research-project/PITFALLS.md +0 -200
  152. package/framework/qualia-framework/templates/research-project/STACK.md +0 -120
  153. package/framework/qualia-framework/templates/research-project/SUMMARY.md +0 -170
  154. package/framework/qualia-framework/templates/research.md +0 -552
  155. package/framework/qualia-framework/templates/roadmap.md +0 -206
  156. package/framework/qualia-framework/templates/state.md +0 -179
  157. package/framework/qualia-framework/templates/summary-complex.md +0 -59
  158. package/framework/qualia-framework/templates/summary-minimal.md +0 -41
  159. package/framework/qualia-framework/templates/summary-standard.md +0 -48
  160. package/framework/qualia-framework/templates/summary.md +0 -246
  161. package/framework/qualia-framework/templates/user-setup.md +0 -311
  162. package/framework/qualia-framework/templates/verification-report.md +0 -322
  163. package/framework/qualia-framework/workflows/add-phase.md +0 -179
  164. package/framework/qualia-framework/workflows/add-todo.md +0 -157
  165. package/framework/qualia-framework/workflows/audit-milestone.md +0 -241
  166. package/framework/qualia-framework/workflows/check-todos.md +0 -176
  167. package/framework/qualia-framework/workflows/complete-milestone.md +0 -858
  168. package/framework/qualia-framework/workflows/diagnose-issues.md +0 -219
  169. package/framework/qualia-framework/workflows/discovery-phase.md +0 -289
  170. package/framework/qualia-framework/workflows/discuss-phase.md +0 -534
  171. package/framework/qualia-framework/workflows/execute-phase.md +0 -559
  172. package/framework/qualia-framework/workflows/execute-plan.md +0 -438
  173. package/framework/qualia-framework/workflows/help.md +0 -470
  174. package/framework/qualia-framework/workflows/insert-phase.md +0 -220
  175. package/framework/qualia-framework/workflows/list-phase-assumptions.md +0 -178
  176. package/framework/qualia-framework/workflows/map-codebase.md +0 -327
  177. package/framework/qualia-framework/workflows/new-milestone.md +0 -363
  178. package/framework/qualia-framework/workflows/new-project.md +0 -982
  179. package/framework/qualia-framework/workflows/pause-work.md +0 -122
  180. package/framework/qualia-framework/workflows/plan-milestone-gaps.md +0 -256
  181. package/framework/qualia-framework/workflows/plan-phase.md +0 -422
  182. package/framework/qualia-framework/workflows/progress.md +0 -389
  183. package/framework/qualia-framework/workflows/quick.md +0 -252
  184. package/framework/qualia-framework/workflows/remove-phase.md +0 -326
  185. package/framework/qualia-framework/workflows/research-phase.md +0 -74
  186. package/framework/qualia-framework/workflows/resume-project.md +0 -306
  187. package/framework/qualia-framework/workflows/set-profile.md +0 -80
  188. package/framework/qualia-framework/workflows/settings.md +0 -145
  189. package/framework/qualia-framework/workflows/transition.md +0 -556
  190. package/framework/qualia-framework/workflows/update.md +0 -197
  191. package/framework/qualia-framework/workflows/verify-phase.md +0 -195
  192. package/framework/qualia-framework/workflows/verify-work.md +0 -625
  193. package/framework/rules/context7.md +0 -14
  194. package/framework/rules/frontend.md +0 -33
  195. package/framework/rules/speed.md +0 -23
  196. package/framework/scripts/__pycache__/say.cpython-314.pyc +0 -0
  197. package/framework/scripts/apply-retention.sh +0 -120
  198. package/framework/scripts/bootstrap-pop-os.sh +0 -354
  199. package/framework/scripts/claude-voice +0 -13
  200. package/framework/scripts/cleanup.sh +0 -131
  201. package/framework/scripts/cowork-mode.sh +0 -141
  202. package/framework/scripts/generate-project-claude-md.sh +0 -153
  203. package/framework/scripts/load-test-webhook.js +0 -172
  204. package/framework/scripts/say.py +0 -236
  205. package/framework/scripts/showcase-video-recorder/ffmpeg-builder.js +0 -167
  206. package/framework/scripts/showcase-video-recorder/playwright-helpers.js +0 -216
  207. package/framework/scripts/speak.py +0 -55
  208. package/framework/scripts/speak.sh +0 -18
  209. package/framework/scripts/status.sh +0 -138
  210. package/framework/scripts/sync-to-framework.sh +0 -65
  211. package/framework/scripts/voice-hotkey.py +0 -227
  212. package/framework/scripts/voice-input.sh +0 -51
  213. package/framework/skills/animate/SKILL.md +0 -202
  214. package/framework/skills/bolder/SKILL.md +0 -144
  215. package/framework/skills/browser-qa/SKILL.md +0 -536
  216. package/framework/skills/clarify/SKILL.md +0 -179
  217. package/framework/skills/client-handoff/SKILL.md +0 -135
  218. package/framework/skills/collab-onboard/SKILL.md +0 -111
  219. package/framework/skills/colorize/SKILL.md +0 -170
  220. package/framework/skills/critique/SKILL.md +0 -126
  221. package/framework/skills/deep-research/SKILL.md +0 -240
  222. package/framework/skills/delight/SKILL.md +0 -329
  223. package/framework/skills/deploy/SKILL.md +0 -261
  224. package/framework/skills/deploy-verify/SKILL.md +0 -377
  225. package/framework/skills/deploy-verify/scripts/canary-check.sh +0 -206
  226. package/framework/skills/deploy-verify/scripts/check-console-errors.js +0 -147
  227. package/framework/skills/deploy-verify/scripts/check-cwv.js +0 -139
  228. package/framework/skills/deploy-verify/scripts/project-detect.sh +0 -84
  229. package/framework/skills/deploy-verify/scripts/verify.sh +0 -548
  230. package/framework/skills/design-quieter/SKILL.md +0 -130
  231. package/framework/skills/distill/SKILL.md +0 -149
  232. package/framework/skills/docs-lookup/SKILL.md +0 -79
  233. package/framework/skills/fcm-notifications/SKILL.md +0 -125
  234. package/framework/skills/financial-ledger/SKILL.md +0 -1039
  235. package/framework/skills/frontend-master/NOTICE.md +0 -4
  236. package/framework/skills/frontend-master/SKILL.md +0 -127
  237. package/framework/skills/frontend-master/reference/color-and-contrast.md +0 -132
  238. package/framework/skills/frontend-master/reference/interaction-design.md +0 -123
  239. package/framework/skills/frontend-master/reference/motion-design.md +0 -99
  240. package/framework/skills/frontend-master/reference/responsive-design.md +0 -114
  241. package/framework/skills/frontend-master/reference/spatial-design.md +0 -100
  242. package/framework/skills/frontend-master/reference/typography.md +0 -131
  243. package/framework/skills/frontend-master/reference/ux-writing.md +0 -107
  244. package/framework/skills/harden/SKILL.md +0 -357
  245. package/framework/skills/i18n-rtl/SKILL.md +0 -752
  246. package/framework/skills/learn/SKILL.md +0 -95
  247. package/framework/skills/memory/SKILL.md +0 -50
  248. package/framework/skills/mobile-expo/SKILL.md +0 -977
  249. package/framework/skills/mobile-expo/references/store-checklist.md +0 -550
  250. package/framework/skills/nestjs-backend/README.md +0 -73
  251. package/framework/skills/nestjs-backend/SKILL.md +0 -446
  252. package/framework/skills/nestjs-backend/references/templates.md +0 -1173
  253. package/framework/skills/normalize/SKILL.md +0 -79
  254. package/framework/skills/onboard/SKILL.md +0 -242
  255. package/framework/skills/openrouter-agent/SKILL.md +0 -922
  256. package/framework/skills/polish/SKILL.md +0 -209
  257. package/framework/skills/pr/SKILL.md +0 -66
  258. package/framework/skills/qualia/SKILL.md +0 -199
  259. package/framework/skills/qualia-add-todo/SKILL.md +0 -68
  260. package/framework/skills/qualia-audit-milestone/SKILL.md +0 -95
  261. package/framework/skills/qualia-check-todos/SKILL.md +0 -55
  262. package/framework/skills/qualia-complete-milestone/SKILL.md +0 -134
  263. package/framework/skills/qualia-debug/SKILL.md +0 -149
  264. package/framework/skills/qualia-design/SKILL.md +0 -203
  265. package/framework/skills/qualia-discuss-phase/SKILL.md +0 -72
  266. package/framework/skills/qualia-evolve/SKILL.md +0 -200
  267. package/framework/skills/qualia-execute-phase/SKILL.md +0 -89
  268. package/framework/skills/qualia-framework-audit/SKILL.md +0 -604
  269. package/framework/skills/qualia-guide/SKILL.md +0 -32
  270. package/framework/skills/qualia-help/SKILL.md +0 -114
  271. package/framework/skills/qualia-idk/SKILL.md +0 -352
  272. package/framework/skills/qualia-list-phase-assumptions/SKILL.md +0 -67
  273. package/framework/skills/qualia-new-milestone/SKILL.md +0 -72
  274. package/framework/skills/qualia-new-project/SKILL.md +0 -232
  275. package/framework/skills/qualia-pause-work/SKILL.md +0 -96
  276. package/framework/skills/qualia-plan-milestone-gaps/SKILL.md +0 -57
  277. package/framework/skills/qualia-plan-phase/SKILL.md +0 -104
  278. package/framework/skills/qualia-production-check/SKILL.md +0 -0
  279. package/framework/skills/qualia-progress/SKILL.md +0 -53
  280. package/framework/skills/qualia-quick/SKILL.md +0 -89
  281. package/framework/skills/qualia-report/SKILL.md +0 -166
  282. package/framework/skills/qualia-research-phase/SKILL.md +0 -88
  283. package/framework/skills/qualia-resume-work/SKILL.md +0 -62
  284. package/framework/skills/qualia-review/SKILL.md +0 -263
  285. package/framework/skills/qualia-start/SKILL.md +0 -161
  286. package/framework/skills/qualia-verify-work/SKILL.md +0 -132
  287. package/framework/skills/rag/SKILL.md +0 -750
  288. package/framework/skills/responsive/SKILL.md +0 -231
  289. package/framework/skills/retro/SKILL.md +0 -284
  290. package/framework/skills/sakani-conventions/SKILL.md +0 -136
  291. package/framework/skills/sakani-conventions/evals/evals.json +0 -23
  292. package/framework/skills/sakani-conventions/references/entities.md +0 -365
  293. package/framework/skills/sakani-conventions/references/error-codes.md +0 -95
  294. package/framework/skills/seo-master/SKILL.md +0 -490
  295. package/framework/skills/seo-master/references/checklist.md +0 -199
  296. package/framework/skills/seo-master/references/structured-data.md +0 -609
  297. package/framework/skills/ship/SKILL.md +0 -239
  298. package/framework/skills/stack-researcher/SKILL.md +0 -215
  299. package/framework/skills/status/SKILL.md +0 -154
  300. package/framework/skills/status/scripts/health-check.sh +0 -562
  301. package/framework/skills/subscription-payments/SKILL.md +0 -250
  302. package/framework/skills/supabase/SKILL.md +0 -973
  303. package/framework/skills/supabase/references/templates.md +0 -159
  304. package/framework/skills/team/SKILL.md +0 -67
  305. package/framework/skills/test-runner/SKILL.md +0 -202
  306. package/framework/skills/voice-agent/SKILL.md +0 -1312
  307. package/framework/skills/zoho-workflow/SKILL.md +0 -51
  308. package/framework/statusline-command.sh +0 -117
  309. package/framework/teams/default/inboxes/plan-04.json +0 -9
  310. package/framework/teams/review-team.md +0 -75
  311. package/framework/teams/ship-team.md +0 -86
  312. package/profiles/fawzi.json +0 -16
  313. package/profiles/hasan.json +0 -16
  314. package/profiles/moayad.json +0 -16
  315. package/templates/CLAUDE-owner.md +0 -52
  316. package/templates/CLAUDE.md.hbs +0 -58
  317. package/templates/env.claude.template +0 -12
  318. package/templates/settings.json +0 -172
  319. package/uninstall.sh +0 -90
  320. /package/{framework/rules → rules}/deployment.md +0 -0
  321. /package/{framework/rules → rules}/security.md +0 -0
package/bin/state.js ADDED
@@ -0,0 +1,630 @@
1
+ #!/usr/bin/env node
2
+ // Qualia State Machine — atomic state transitions with precondition validation
3
+ // No external dependencies. Node >= 18 only.
4
+
5
+ const fs = require("fs");
6
+ const path = require("path");
7
+
8
+ const PLANNING = ".planning";
9
+ const STATE_FILE = path.join(PLANNING, "STATE.md");
10
+ const TRACKING_FILE = path.join(PLANNING, "tracking.json");
11
+
12
+ // ─── Arg Parsing ─────────────────────────────────────────
13
+ function parseArgs(argv) {
14
+ const args = {};
15
+ for (let i = 0; i < argv.length; i++) {
16
+ if (argv[i].startsWith("--")) {
17
+ const key = argv[i].slice(2).replace(/-/g, "_");
18
+ const next = argv[i + 1];
19
+ if (!next || next.startsWith("--")) {
20
+ args[key] = true;
21
+ } else {
22
+ args[key] = next;
23
+ i++;
24
+ }
25
+ }
26
+ }
27
+ return args;
28
+ }
29
+
30
+ // ─── File I/O ────────────────────────────────────────────
31
+ function readTracking() {
32
+ try {
33
+ return JSON.parse(fs.readFileSync(TRACKING_FILE, "utf8"));
34
+ } catch {
35
+ return null;
36
+ }
37
+ }
38
+
39
+ function writeTracking(t) {
40
+ fs.writeFileSync(TRACKING_FILE, JSON.stringify(t, null, 2) + "\n");
41
+ }
42
+
43
+ function readState() {
44
+ try {
45
+ return fs.readFileSync(STATE_FILE, "utf8");
46
+ } catch {
47
+ return null;
48
+ }
49
+ }
50
+
51
+ // ─── STATE.md Parser ─────────────────────────────────────
52
+ function parseStateMd(content) {
53
+ if (!content) return null;
54
+ const schema_errors = [];
55
+ const get = (prefix) => {
56
+ const m = content.match(new RegExp(`^${prefix}:\\s*(.+)$`, "m"));
57
+ return m ? m[1].trim() : "";
58
+ };
59
+ const hasField = (prefix) =>
60
+ new RegExp(`^${prefix}:\\s*`, "m").test(content);
61
+
62
+ const phaseMatch = content.match(
63
+ /^Phase:\s*(\d+)\s+of\s+(\d+)\s*[—-]\s*(.+)$/m
64
+ );
65
+ if (!phaseMatch) {
66
+ schema_errors.push({
67
+ field: "phase_header",
68
+ message: 'Missing or malformed "Phase: N of M — Name" header',
69
+ severity: "error",
70
+ });
71
+ }
72
+
73
+ // Status field presence (independent of value)
74
+ if (!hasField("Status")) {
75
+ schema_errors.push({
76
+ field: "status_field",
77
+ message: "Missing Status: field",
78
+ severity: "warning",
79
+ });
80
+ }
81
+
82
+ // Parse roadmap table
83
+ const phases = [];
84
+ const tableHeaderRe = /\| # \| Phase \| Goal \| Status \|/;
85
+ const tableMatch = content.match(
86
+ /\| # \| Phase \| Goal \| Status \|\n\|[-|]+\|\n([\s\S]*?)(?=\n##|\n$|$)/
87
+ );
88
+ if (!tableHeaderRe.test(content)) {
89
+ schema_errors.push({
90
+ field: "roadmap_table",
91
+ message: "Roadmap table header not found",
92
+ severity: "error",
93
+ });
94
+ } else if (!tableMatch) {
95
+ // Header is there but the separator row or body is malformed
96
+ schema_errors.push({
97
+ field: "roadmap_table",
98
+ message: "Roadmap table is malformed (missing separator row or body)",
99
+ severity: "error",
100
+ });
101
+ } else {
102
+ for (const row of tableMatch[1].trim().split("\n")) {
103
+ const cols = row.split("|").map((c) => c.trim()).filter(Boolean);
104
+ if (cols.length >= 4) {
105
+ phases.push({
106
+ num: parseInt(cols[0]),
107
+ name: cols[1],
108
+ goal: cols[2],
109
+ status: cols[3],
110
+ });
111
+ }
112
+ }
113
+ }
114
+
115
+ // Row count vs header "of M"
116
+ if (phaseMatch) {
117
+ const declaredTotal = parseInt(phaseMatch[2]);
118
+ if (phases.length && phases.length !== declaredTotal) {
119
+ schema_errors.push({
120
+ field: "roadmap_rows",
121
+ message: `Expected ${declaredTotal} phases in roadmap, found ${phases.length}`,
122
+ severity: "warning",
123
+ });
124
+ }
125
+ }
126
+
127
+ return {
128
+ phase: phaseMatch ? parseInt(phaseMatch[1]) : 1,
129
+ total_phases: phaseMatch ? parseInt(phaseMatch[2]) : phases.length || 1,
130
+ phase_name: phaseMatch ? phaseMatch[3].trim() : "",
131
+ status: get("Status").toLowerCase().replace(/\s+/g, "_") || "setup",
132
+ assigned_to: get("Assigned to") || "",
133
+ phases,
134
+ schema_errors,
135
+ };
136
+ }
137
+
138
+ // ─── STATE.md Writer ─────────────────────────────────────
139
+ function writeStateMd(s) {
140
+ const phaseFrac = Math.round(((s.phase - 1) / s.total_phases) * 100);
141
+ const filled = Math.round(phaseFrac / 10);
142
+ const bar = "█".repeat(filled) + "░".repeat(10 - filled);
143
+ const now = new Date().toISOString().split("T")[0];
144
+
145
+ const roadmap = s.phases
146
+ .map((p) => `| ${p.num} | ${p.name} | ${p.goal} | ${p.status} |`)
147
+ .join("\n");
148
+
149
+ const md = `# Project State
150
+
151
+ ## Project
152
+ See: .planning/PROJECT.md
153
+
154
+ ## Current Position
155
+ Phase: ${s.phase} of ${s.total_phases} — ${s.phase_name}
156
+ Status: ${s.status}
157
+ Assigned to: ${s.assigned_to}
158
+ Last activity: ${now} — ${s.last_activity || "State updated"}
159
+
160
+ Progress: [${bar}] ${phaseFrac}%
161
+
162
+ ## Roadmap
163
+ | # | Phase | Goal | Status |
164
+ |---|-------|------|--------|
165
+ ${roadmap}
166
+
167
+ ## Blockers
168
+ ${s.blockers || "None."}
169
+
170
+ ## Session
171
+ Last session: ${now}
172
+ Last worked by: ${s.assigned_to}
173
+ Resume: ${s.resume || "—"}
174
+ `;
175
+ fs.writeFileSync(STATE_FILE, md);
176
+ }
177
+
178
+ // ─── Precondition Checks ─────────────────────────────────
179
+ const VALID_FROM = {
180
+ planned: ["setup", "verified"], // verified(fail) → planned = gap closure
181
+ built: ["planned"],
182
+ verified: ["built"],
183
+ polished: ["verified"],
184
+ shipped: ["polished"],
185
+ handed_off: ["shipped"],
186
+ done: ["handed_off"],
187
+ };
188
+
189
+ function checkPreconditions(current, target, opts) {
190
+ const phase = parseInt(opts.phase) || current.phase;
191
+
192
+ // Special transitions (no status gate)
193
+ if (target === "note" || target === "activity") return { ok: true };
194
+
195
+ // Check valid transition
196
+ const allowed = VALID_FROM[target];
197
+ if (!allowed) return fail("INVALID_STATUS", `Unknown status: ${target}`);
198
+ if (!allowed.includes(current.status)) {
199
+ return fail(
200
+ "PRECONDITION_FAILED",
201
+ `Cannot go from '${current.status}' to '${target}'. Allowed from: ${allowed.join(", ")}`
202
+ );
203
+ }
204
+
205
+ // File checks
206
+ if (target === "planned") {
207
+ const planFile = path.join(PLANNING, `phase-${phase}-plan.md`);
208
+ if (!fs.existsSync(planFile))
209
+ return fail("MISSING_FILE", `Plan file not found: ${planFile}`);
210
+ }
211
+
212
+ if (target === "verified") {
213
+ const vFile = path.join(PLANNING, `phase-${phase}-verification.md`);
214
+ if (!fs.existsSync(vFile))
215
+ return fail("MISSING_FILE", `Verification file not found: ${vFile}`);
216
+ if (!opts.verification || !["pass", "fail"].includes(opts.verification))
217
+ return fail("MISSING_ARG", "--verification must be 'pass' or 'fail'");
218
+ }
219
+
220
+ if (target === "shipped") {
221
+ if (!opts.deployed_url)
222
+ return fail("MISSING_ARG", "--deployed-url is required for 'shipped'");
223
+ }
224
+
225
+ if (target === "handed_off") {
226
+ const hFile = path.join(PLANNING, "HANDOFF.md");
227
+ if (!fs.existsSync(hFile))
228
+ return fail("MISSING_FILE", `Handoff file not found: ${hFile}`);
229
+ }
230
+
231
+ // Gap-closure circuit breaker
232
+ if (target === "planned" && current.status === "verified") {
233
+ const t = readTracking() || {};
234
+ const cycles = (t.gap_cycles || {})[String(phase)] || 0;
235
+ if (cycles >= 2) {
236
+ return fail(
237
+ "GAP_CYCLE_LIMIT",
238
+ `Phase ${phase} has failed verification ${cycles} times. Escalate to Fawzi or re-plan from scratch.`
239
+ );
240
+ }
241
+ }
242
+
243
+ return { ok: true };
244
+ }
245
+
246
+ function fail(error, message) {
247
+ return { ok: false, error, message };
248
+ }
249
+
250
+ // ─── Next Command Logic ──────────────────────────────────
251
+ function nextCommand(status, phase, totalPhases, verification) {
252
+ switch (status) {
253
+ case "setup":
254
+ return `/qualia-plan ${phase}`;
255
+ case "planned":
256
+ return `/qualia-build ${phase}`;
257
+ case "built":
258
+ return `/qualia-verify ${phase}`;
259
+ case "verified":
260
+ if (verification === "fail") return `/qualia-plan ${phase} --gaps`;
261
+ if (phase < totalPhases) return `/qualia-plan ${phase + 1}`;
262
+ return "/qualia-polish";
263
+ case "polished":
264
+ return "/qualia-ship";
265
+ case "shipped":
266
+ return "/qualia-handoff";
267
+ case "handed_off":
268
+ return "/qualia-report";
269
+ case "done":
270
+ return "Done.";
271
+ default:
272
+ return `/qualia`;
273
+ }
274
+ }
275
+
276
+ // ─── Commands ────────────────────────────────────────────
277
+
278
+ function cmdCheck(opts) {
279
+ const t = readTracking();
280
+ const s = parseStateMd(readState());
281
+ if (!t || !s) {
282
+ return output({
283
+ ok: false,
284
+ error: "NO_PROJECT",
285
+ message: "No .planning/ found. Run /qualia-new to start.",
286
+ });
287
+ }
288
+ output({
289
+ ok: true,
290
+ phase: s.phase,
291
+ phase_name: s.phase_name,
292
+ total_phases: s.total_phases,
293
+ status: s.status,
294
+ assigned_to: s.assigned_to,
295
+ verification: t.verification || "pending",
296
+ gap_cycles: (t.gap_cycles || {})[String(s.phase)] || 0,
297
+ tasks_done: t.tasks_done || 0,
298
+ tasks_total: t.tasks_total || 0,
299
+ deployed_url: t.deployed_url || "",
300
+ next_command: nextCommand(
301
+ s.status,
302
+ s.phase,
303
+ s.total_phases,
304
+ t.verification
305
+ ),
306
+ schema_errors: s.schema_errors && s.schema_errors.length ? s.schema_errors : undefined,
307
+ });
308
+ }
309
+
310
+ function cmdTransition(opts) {
311
+ const target = opts.to;
312
+ if (!target) return output(fail("MISSING_ARG", "--to is required"));
313
+
314
+ const t = readTracking();
315
+ const s = parseStateMd(readState());
316
+ if (!t || !s) {
317
+ return output(
318
+ fail("NO_PROJECT", "No .planning/ found. Run /qualia-new.")
319
+ );
320
+ }
321
+
322
+ // Refuse transitions if STATE.md has schema errors (severity=error)
323
+ if (s.schema_errors && s.schema_errors.some((e) => e.severity === "error")) {
324
+ return output(
325
+ fail(
326
+ "STATE_SCHEMA_ERROR",
327
+ "STATE.md is malformed. Run `node state.js check` to see errors. Consider `state.js fix` to rewrite canonically."
328
+ )
329
+ );
330
+ }
331
+
332
+ // Special: note/activity (no status change)
333
+ if (target === "note" || target === "activity") {
334
+ const now = new Date().toISOString().split("T")[0];
335
+ if (opts.notes) t.notes = opts.notes;
336
+ t.last_updated = new Date().toISOString();
337
+ writeTracking(t);
338
+ s.last_activity = opts.notes || "Activity logged";
339
+ writeStateMd(s);
340
+ return output({
341
+ ok: true,
342
+ phase: s.phase,
343
+ status: s.status,
344
+ action: target,
345
+ });
346
+ }
347
+
348
+ const phase = parseInt(opts.phase) || s.phase;
349
+
350
+ // Precondition check
351
+ const check = checkPreconditions(
352
+ { ...s, phase },
353
+ target,
354
+ { ...opts, phase }
355
+ );
356
+ if (!check.ok) return output(check);
357
+
358
+ const prevStatus = s.status;
359
+
360
+ // Apply transition
361
+ s.status = target;
362
+ s.last_activity = `${target} (phase ${phase})`;
363
+
364
+ // Update tracking fields
365
+ t.status = target;
366
+ t.phase = phase;
367
+ t.phase_name = s.phases[phase - 1]?.name || s.phase_name;
368
+ t.last_updated = new Date().toISOString();
369
+
370
+ if (target === "planned") {
371
+ // Gap closure: increment counter if coming from verified(fail)
372
+ if (prevStatus === "verified") {
373
+ if (!t.gap_cycles) t.gap_cycles = {};
374
+ t.gap_cycles[String(phase)] = (t.gap_cycles[String(phase)] || 0) + 1;
375
+ s.last_activity = `Gap closure #${t.gap_cycles[String(phase)]} planned (phase ${phase})`;
376
+ }
377
+ // Update roadmap
378
+ if (s.phases[phase - 1]) s.phases[phase - 1].status = "planned";
379
+ }
380
+
381
+ if (target === "built") {
382
+ t.tasks_done = parseInt(opts.tasks_done) || 0;
383
+ t.tasks_total = parseInt(opts.tasks_total) || 0;
384
+ t.wave = parseInt(opts.wave) || 0;
385
+ s.last_activity = `Phase ${phase} built (${t.tasks_done}/${t.tasks_total} tasks)`;
386
+ if (s.phases[phase - 1]) s.phases[phase - 1].status = "built";
387
+ }
388
+
389
+ if (target === "verified") {
390
+ t.verification = opts.verification;
391
+ s.last_activity = `Phase ${phase} verified — ${opts.verification}`;
392
+ if (s.phases[phase - 1])
393
+ s.phases[phase - 1].status =
394
+ opts.verification === "pass" ? "verified" : "failed";
395
+
396
+ // Auto-advance on pass
397
+ if (opts.verification === "pass") {
398
+ if (phase < s.total_phases) {
399
+ s.phase = phase + 1;
400
+ s.phase_name = s.phases[phase]?.name || `Phase ${phase + 1}`;
401
+ s.status = "setup";
402
+ t.phase = s.phase;
403
+ t.phase_name = s.phase_name;
404
+ t.status = "setup";
405
+ t.verification = "pending";
406
+ t.tasks_done = 0;
407
+ t.tasks_total = 0;
408
+ s.last_activity = `Phase ${phase} passed — advancing to phase ${s.phase}`;
409
+ }
410
+ // Reset gap counter for the passed phase
411
+ if (t.gap_cycles) t.gap_cycles[String(phase)] = 0;
412
+ }
413
+ }
414
+
415
+ if (target === "polished") {
416
+ if (s.phases[s.phases.length - 1])
417
+ s.phases[s.phases.length - 1].status = "verified";
418
+ }
419
+
420
+ if (target === "shipped") {
421
+ t.deployed_url = opts.deployed_url || "";
422
+ }
423
+
424
+ // Write both files
425
+ const backupState = readState();
426
+ try {
427
+ writeStateMd(s);
428
+ writeTracking(t);
429
+ } catch (e) {
430
+ // Revert STATE.md on failure
431
+ if (backupState) fs.writeFileSync(STATE_FILE, backupState);
432
+ return output(fail("WRITE_ERROR", e.message));
433
+ }
434
+
435
+ output({
436
+ ok: true,
437
+ phase: s.phase,
438
+ phase_name: s.phase_name,
439
+ status: s.status,
440
+ previous_status: prevStatus,
441
+ verification: t.verification,
442
+ gap_cycles: (t.gap_cycles || {})[String(s.phase)] || 0,
443
+ next_command: nextCommand(s.status, s.phase, s.total_phases, t.verification),
444
+ });
445
+ }
446
+
447
+ function cmdInit(opts) {
448
+ if (!opts.project) return output(fail("MISSING_ARG", "--project required"));
449
+
450
+ // Parse phases
451
+ let phases = [];
452
+ if (opts.phases) {
453
+ try {
454
+ phases = JSON.parse(opts.phases);
455
+ } catch {
456
+ return output(fail("INVALID_ARG", "--phases must be valid JSON array"));
457
+ }
458
+ }
459
+ const totalPhases = parseInt(opts.total_phases) || phases.length || 1;
460
+
461
+ // Ensure phases array has entries
462
+ while (phases.length < totalPhases) {
463
+ phases.push({
464
+ name: `Phase ${phases.length + 1}`,
465
+ goal: "TBD",
466
+ });
467
+ }
468
+
469
+ // Create .planning/ if needed
470
+ if (!fs.existsSync(PLANNING)) fs.mkdirSync(PLANNING, { recursive: true });
471
+
472
+ const now = new Date().toISOString();
473
+ const date = now.split("T")[0];
474
+
475
+ // Build state
476
+ const s = {
477
+ phase: 1,
478
+ total_phases: totalPhases,
479
+ phase_name: phases[0].name,
480
+ status: "setup",
481
+ assigned_to: opts.assigned_to || "",
482
+ last_activity: `Project initialized`,
483
+ phases: phases.map((p, i) => ({
484
+ num: i + 1,
485
+ name: p.name,
486
+ goal: p.goal,
487
+ status: i === 0 ? "ready" : "—",
488
+ })),
489
+ blockers: "None.",
490
+ resume: "—",
491
+ };
492
+
493
+ // Build tracking
494
+ const t = {
495
+ project: opts.project,
496
+ client: opts.client || "",
497
+ type: opts.type || "",
498
+ assigned_to: opts.assigned_to || "",
499
+ phase: 1,
500
+ phase_name: phases[0].name,
501
+ total_phases: totalPhases,
502
+ status: "setup",
503
+ wave: 0,
504
+ tasks_done: 0,
505
+ tasks_total: 0,
506
+ verification: "pending",
507
+ gap_cycles: {},
508
+ blockers: [],
509
+ last_updated: now,
510
+ last_commit: "",
511
+ deployed_url: "",
512
+ notes: "",
513
+ };
514
+
515
+ writeStateMd(s);
516
+ writeTracking(t);
517
+
518
+ output({
519
+ ok: true,
520
+ action: "init",
521
+ project: opts.project,
522
+ phase: 1,
523
+ total_phases: totalPhases,
524
+ status: "setup",
525
+ next_command: "/qualia-plan 1",
526
+ });
527
+ }
528
+
529
+ function cmdFix(opts) {
530
+ const raw = readState();
531
+ const t = readTracking();
532
+ if (!raw && !t) {
533
+ return output(
534
+ fail("NO_PROJECT", "No .planning/ found. Run /qualia-new.")
535
+ );
536
+ }
537
+ const parsed = parseStateMd(raw) || {
538
+ phase: 1,
539
+ total_phases: 1,
540
+ phase_name: "",
541
+ status: "setup",
542
+ assigned_to: "",
543
+ phases: [],
544
+ schema_errors: [
545
+ { field: "content", message: "STATE.md missing or empty", severity: "error" },
546
+ ],
547
+ };
548
+ const previousErrors = (parsed.schema_errors || []).length;
549
+
550
+ // Prefer tracking.json values when parsed fields are defaulted/missing
551
+ const tr = t || {};
552
+ const totalPhases =
553
+ parseInt(tr.total_phases) || parsed.total_phases || parsed.phases.length || 1;
554
+ const phaseNum = parseInt(tr.phase) || parsed.phase || 1;
555
+ const phaseName =
556
+ (parsed.phase_name && parsed.phase_name.trim()) ||
557
+ tr.phase_name ||
558
+ `Phase ${phaseNum}`;
559
+ const status = parsed.status || tr.status || "setup";
560
+ const assignedTo = parsed.assigned_to || tr.assigned_to || "";
561
+
562
+ // Build a phases array of the right length
563
+ const phases = [];
564
+ for (let i = 0; i < totalPhases; i++) {
565
+ const existing = parsed.phases[i];
566
+ phases.push({
567
+ num: i + 1,
568
+ name: existing?.name || `Phase ${i + 1}`,
569
+ goal: existing?.goal || "TBD",
570
+ status: existing?.status || (i === 0 ? "ready" : "—"),
571
+ });
572
+ }
573
+
574
+ const s = {
575
+ phase: phaseNum,
576
+ total_phases: totalPhases,
577
+ phase_name: phaseName,
578
+ status,
579
+ assigned_to: assignedTo,
580
+ last_activity: "STATE.md repaired by state.js fix",
581
+ phases,
582
+ blockers: "None.",
583
+ resume: "—",
584
+ };
585
+
586
+ try {
587
+ writeStateMd(s);
588
+ } catch (e) {
589
+ return output(fail("WRITE_ERROR", e.message));
590
+ }
591
+
592
+ output({
593
+ ok: true,
594
+ action: "fix",
595
+ previous_errors: previousErrors,
596
+ fixed: true,
597
+ });
598
+ }
599
+
600
+ // ─── Output ──────────────────────────────────────────────
601
+ function output(obj) {
602
+ console.log(JSON.stringify(obj, null, 2));
603
+ if (!obj.ok) process.exit(1);
604
+ }
605
+
606
+ // ─── Main ────────────────────────────────────────────────
607
+ const [cmd, ...rest] = process.argv.slice(2);
608
+ const opts = parseArgs(rest);
609
+
610
+ switch (cmd) {
611
+ case "check":
612
+ cmdCheck(opts);
613
+ break;
614
+ case "transition":
615
+ cmdTransition(opts);
616
+ break;
617
+ case "init":
618
+ cmdInit(opts);
619
+ break;
620
+ case "fix":
621
+ cmdFix(opts);
622
+ break;
623
+ default:
624
+ output(
625
+ fail(
626
+ "UNKNOWN_COMMAND",
627
+ `Usage: state.js <check|transition|init|fix> [--options]`
628
+ )
629
+ );
630
+ }