qualia-framework 2.5.1 → 3.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.
Files changed (327) hide show
  1. package/CLAUDE.md +63 -0
  2. package/README.md +108 -30
  3. package/agents/builder.md +110 -0
  4. package/agents/planner.md +186 -0
  5. package/agents/qa-browser.md +186 -0
  6. package/agents/verifier.md +369 -0
  7. package/bin/cli.js +706 -417
  8. package/bin/install.js +622 -0
  9. package/bin/qualia-ui.js +284 -0
  10. package/bin/state.js +824 -0
  11. package/bin/statusline.js +252 -0
  12. package/docs/erp-contract.md +161 -0
  13. package/guide.md +63 -0
  14. package/hooks/auto-update.js +117 -0
  15. package/hooks/block-env-edit.js +52 -0
  16. package/hooks/branch-guard.js +68 -0
  17. package/hooks/migration-guard.js +83 -0
  18. package/hooks/pre-compact.js +52 -0
  19. package/hooks/pre-deploy-gate.js +149 -0
  20. package/hooks/pre-push.js +53 -0
  21. package/hooks/session-start.js +126 -0
  22. package/package.json +31 -17
  23. package/rules/design-reference.md +179 -0
  24. package/rules/frontend.md +126 -0
  25. package/rules/infrastructure.md +87 -0
  26. package/skills/qualia/SKILL.md +88 -0
  27. package/skills/qualia-build/SKILL.md +115 -0
  28. package/skills/qualia-debug/SKILL.md +87 -0
  29. package/skills/qualia-design/SKILL.md +99 -0
  30. package/skills/qualia-handoff/SKILL.md +66 -0
  31. package/skills/qualia-help/SKILL.md +60 -0
  32. package/skills/qualia-idk/SKILL.md +8 -0
  33. package/skills/qualia-learn/SKILL.md +111 -0
  34. package/skills/qualia-new/SKILL.md +323 -0
  35. package/skills/qualia-pause/SKILL.md +63 -0
  36. package/skills/qualia-plan/SKILL.md +101 -0
  37. package/skills/qualia-polish/SKILL.md +207 -0
  38. package/skills/qualia-quick/SKILL.md +37 -0
  39. package/skills/qualia-report/SKILL.md +114 -0
  40. package/skills/qualia-resume/SKILL.md +49 -0
  41. package/skills/qualia-review/SKILL.md +161 -0
  42. package/skills/qualia-ship/SKILL.md +90 -0
  43. package/skills/qualia-skill-new/SKILL.md +167 -0
  44. package/skills/qualia-task/SKILL.md +91 -0
  45. package/skills/qualia-test/SKILL.md +134 -0
  46. package/skills/qualia-verify/SKILL.md +113 -0
  47. package/templates/DESIGN.md +475 -0
  48. package/templates/help.html +476 -0
  49. package/templates/plan.md +42 -0
  50. package/templates/project.md +22 -0
  51. package/templates/state.md +27 -0
  52. package/templates/tracking.json +20 -0
  53. package/tests/bin.test.sh +687 -0
  54. package/tests/hooks.test.sh +384 -0
  55. package/tests/runner.js +1956 -0
  56. package/tests/state.test.sh +713 -0
  57. package/tests/statusline.test.sh +243 -0
  58. package/bin/collect-metrics.sh +0 -62
  59. package/framework/.claudeignore +0 -51
  60. package/framework/CLAUDE.md +0 -51
  61. package/framework/MCP_SETUP.md +0 -229
  62. package/framework/agents/architecture-strategist.md +0 -53
  63. package/framework/agents/backend-agent.md +0 -150
  64. package/framework/agents/code-simplicity-reviewer.md +0 -86
  65. package/framework/agents/frontend-agent.md +0 -111
  66. package/framework/agents/kieran-typescript-reviewer.md +0 -96
  67. package/framework/agents/performance-oracle.md +0 -111
  68. package/framework/agents/qualia-codebase-mapper.md +0 -761
  69. package/framework/agents/qualia-debugger.md +0 -1204
  70. package/framework/agents/qualia-executor.md +0 -882
  71. package/framework/agents/qualia-integration-checker.md +0 -424
  72. package/framework/agents/qualia-phase-researcher.md +0 -457
  73. package/framework/agents/qualia-plan-checker.md +0 -700
  74. package/framework/agents/qualia-planner.md +0 -1245
  75. package/framework/agents/qualia-project-researcher.md +0 -603
  76. package/framework/agents/qualia-research-synthesizer.md +0 -200
  77. package/framework/agents/qualia-roadmapper.md +0 -606
  78. package/framework/agents/qualia-verifier.md +0 -686
  79. package/framework/agents/red-team-qa.md +0 -130
  80. package/framework/agents/security-auditor.md +0 -72
  81. package/framework/agents/team-orchestrator.md +0 -229
  82. package/framework/agents/teams/framework-audit-team.md +0 -66
  83. package/framework/agents/teams/full-stack-team.md +0 -48
  84. package/framework/agents/teams/optimize-team.md +0 -53
  85. package/framework/agents/teams/review-team.md +0 -70
  86. package/framework/agents/teams/ship-team.md +0 -86
  87. package/framework/agents/test-agent.md +0 -182
  88. package/framework/hooks/auto-format.sh +0 -54
  89. package/framework/hooks/block-env-edit.sh +0 -42
  90. package/framework/hooks/branch-guard.sh +0 -43
  91. package/framework/hooks/confirm-delete.sh +0 -59
  92. package/framework/hooks/migration-validate.sh +0 -77
  93. package/framework/hooks/notification-speak.sh +0 -16
  94. package/framework/hooks/pre-commit.sh +0 -100
  95. package/framework/hooks/pre-compact.sh +0 -56
  96. package/framework/hooks/pre-deploy-gate.sh +0 -160
  97. package/framework/hooks/qualia-colors.sh +0 -32
  98. package/framework/hooks/retention-cleanup.sh +0 -62
  99. package/framework/hooks/save-session-state.sh +0 -185
  100. package/framework/hooks/session-context-loader.sh +0 -96
  101. package/framework/hooks/session-learn.sh +0 -32
  102. package/framework/hooks/skill-announce.sh +0 -123
  103. package/framework/hooks/tool-error-announce.sh +0 -27
  104. package/framework/install.ps1 +0 -323
  105. package/framework/install.sh +0 -313
  106. package/framework/qualia-framework/VERSION +0 -1
  107. package/framework/qualia-framework/assets/qualia-logo.png +0 -0
  108. package/framework/qualia-framework/bin/collect-metrics.sh +0 -67
  109. package/framework/qualia-framework/bin/generate-report-docx.py +0 -429
  110. package/framework/qualia-framework/bin/qualia-tools.js +0 -2201
  111. package/framework/qualia-framework/bin/qualia-tools.test.js +0 -1054
  112. package/framework/qualia-framework/references/checkpoints.md +0 -775
  113. package/framework/qualia-framework/references/completion-checklists.md +0 -359
  114. package/framework/qualia-framework/references/continuation-format.md +0 -249
  115. package/framework/qualia-framework/references/continuation-prompt.md +0 -97
  116. package/framework/qualia-framework/references/decimal-phase-calculation.md +0 -65
  117. package/framework/qualia-framework/references/design-quality.md +0 -56
  118. package/framework/qualia-framework/references/employee-guide.md +0 -167
  119. package/framework/qualia-framework/references/git-integration.md +0 -254
  120. package/framework/qualia-framework/references/git-planning-commit.md +0 -50
  121. package/framework/qualia-framework/references/model-profile-resolution.md +0 -32
  122. package/framework/qualia-framework/references/model-profiles.md +0 -73
  123. package/framework/qualia-framework/references/phase-argument-parsing.md +0 -61
  124. package/framework/qualia-framework/references/planning-config.md +0 -195
  125. package/framework/qualia-framework/references/questioning.md +0 -141
  126. package/framework/qualia-framework/references/tdd.md +0 -263
  127. package/framework/qualia-framework/references/ui-brand.md +0 -160
  128. package/framework/qualia-framework/references/verification-patterns.md +0 -612
  129. package/framework/qualia-framework/templates/DEBUG.md +0 -159
  130. package/framework/qualia-framework/templates/DESIGN.md +0 -81
  131. package/framework/qualia-framework/templates/UAT.md +0 -247
  132. package/framework/qualia-framework/templates/codebase/architecture.md +0 -255
  133. package/framework/qualia-framework/templates/codebase/concerns.md +0 -310
  134. package/framework/qualia-framework/templates/codebase/conventions.md +0 -307
  135. package/framework/qualia-framework/templates/codebase/integrations.md +0 -280
  136. package/framework/qualia-framework/templates/codebase/stack.md +0 -186
  137. package/framework/qualia-framework/templates/codebase/structure.md +0 -285
  138. package/framework/qualia-framework/templates/codebase/testing.md +0 -480
  139. package/framework/qualia-framework/templates/config.json +0 -35
  140. package/framework/qualia-framework/templates/context.md +0 -283
  141. package/framework/qualia-framework/templates/continue-here.md +0 -78
  142. package/framework/qualia-framework/templates/debug-subagent-prompt.md +0 -91
  143. package/framework/qualia-framework/templates/discovery.md +0 -146
  144. package/framework/qualia-framework/templates/lab-notes.md +0 -16
  145. package/framework/qualia-framework/templates/milestone-archive.md +0 -123
  146. package/framework/qualia-framework/templates/milestone.md +0 -115
  147. package/framework/qualia-framework/templates/phase-prompt.md +0 -567
  148. package/framework/qualia-framework/templates/planner-subagent-prompt.md +0 -117
  149. package/framework/qualia-framework/templates/project.md +0 -184
  150. package/framework/qualia-framework/templates/projects/ai-agent.md +0 -156
  151. package/framework/qualia-framework/templates/projects/mobile-app.md +0 -181
  152. package/framework/qualia-framework/templates/projects/voice-agent.md +0 -134
  153. package/framework/qualia-framework/templates/projects/website.md +0 -137
  154. package/framework/qualia-framework/templates/requirements.md +0 -231
  155. package/framework/qualia-framework/templates/research-project/ARCHITECTURE.md +0 -204
  156. package/framework/qualia-framework/templates/research-project/FEATURES.md +0 -147
  157. package/framework/qualia-framework/templates/research-project/PITFALLS.md +0 -200
  158. package/framework/qualia-framework/templates/research-project/STACK.md +0 -120
  159. package/framework/qualia-framework/templates/research-project/SUMMARY.md +0 -170
  160. package/framework/qualia-framework/templates/research.md +0 -552
  161. package/framework/qualia-framework/templates/roadmap.md +0 -206
  162. package/framework/qualia-framework/templates/state.md +0 -179
  163. package/framework/qualia-framework/templates/summary-complex.md +0 -59
  164. package/framework/qualia-framework/templates/summary-minimal.md +0 -41
  165. package/framework/qualia-framework/templates/summary-standard.md +0 -48
  166. package/framework/qualia-framework/templates/summary.md +0 -246
  167. package/framework/qualia-framework/templates/user-setup.md +0 -311
  168. package/framework/qualia-framework/templates/verification-report.md +0 -322
  169. package/framework/qualia-framework/workflows/add-phase.md +0 -179
  170. package/framework/qualia-framework/workflows/add-todo.md +0 -157
  171. package/framework/qualia-framework/workflows/audit-milestone.md +0 -241
  172. package/framework/qualia-framework/workflows/check-todos.md +0 -176
  173. package/framework/qualia-framework/workflows/complete-milestone.md +0 -858
  174. package/framework/qualia-framework/workflows/diagnose-issues.md +0 -219
  175. package/framework/qualia-framework/workflows/discovery-phase.md +0 -289
  176. package/framework/qualia-framework/workflows/discuss-phase.md +0 -534
  177. package/framework/qualia-framework/workflows/execute-phase.md +0 -559
  178. package/framework/qualia-framework/workflows/execute-plan.md +0 -438
  179. package/framework/qualia-framework/workflows/help.md +0 -470
  180. package/framework/qualia-framework/workflows/insert-phase.md +0 -220
  181. package/framework/qualia-framework/workflows/list-phase-assumptions.md +0 -178
  182. package/framework/qualia-framework/workflows/map-codebase.md +0 -327
  183. package/framework/qualia-framework/workflows/new-milestone.md +0 -363
  184. package/framework/qualia-framework/workflows/new-project.md +0 -982
  185. package/framework/qualia-framework/workflows/pause-work.md +0 -122
  186. package/framework/qualia-framework/workflows/plan-milestone-gaps.md +0 -256
  187. package/framework/qualia-framework/workflows/plan-phase.md +0 -422
  188. package/framework/qualia-framework/workflows/progress.md +0 -389
  189. package/framework/qualia-framework/workflows/quick.md +0 -252
  190. package/framework/qualia-framework/workflows/remove-phase.md +0 -326
  191. package/framework/qualia-framework/workflows/research-phase.md +0 -74
  192. package/framework/qualia-framework/workflows/resume-project.md +0 -306
  193. package/framework/qualia-framework/workflows/set-profile.md +0 -80
  194. package/framework/qualia-framework/workflows/settings.md +0 -145
  195. package/framework/qualia-framework/workflows/transition.md +0 -556
  196. package/framework/qualia-framework/workflows/update.md +0 -197
  197. package/framework/qualia-framework/workflows/verify-phase.md +0 -195
  198. package/framework/qualia-framework/workflows/verify-work.md +0 -625
  199. package/framework/rules/context7.md +0 -14
  200. package/framework/rules/frontend.md +0 -33
  201. package/framework/rules/speed.md +0 -23
  202. package/framework/scripts/__pycache__/say.cpython-314.pyc +0 -0
  203. package/framework/scripts/apply-retention.sh +0 -120
  204. package/framework/scripts/bootstrap-pop-os.sh +0 -354
  205. package/framework/scripts/claude-voice +0 -13
  206. package/framework/scripts/cleanup.sh +0 -131
  207. package/framework/scripts/cowork-mode.sh +0 -141
  208. package/framework/scripts/generate-project-claude-md.sh +0 -153
  209. package/framework/scripts/load-test-webhook.js +0 -172
  210. package/framework/scripts/say.py +0 -236
  211. package/framework/scripts/showcase-video-recorder/ffmpeg-builder.js +0 -167
  212. package/framework/scripts/showcase-video-recorder/playwright-helpers.js +0 -216
  213. package/framework/scripts/speak.py +0 -55
  214. package/framework/scripts/speak.sh +0 -18
  215. package/framework/scripts/status.sh +0 -138
  216. package/framework/scripts/sync-to-framework.sh +0 -65
  217. package/framework/scripts/voice-hotkey.py +0 -227
  218. package/framework/scripts/voice-input.sh +0 -51
  219. package/framework/skills/animate/SKILL.md +0 -202
  220. package/framework/skills/bolder/SKILL.md +0 -144
  221. package/framework/skills/browser-qa/SKILL.md +0 -536
  222. package/framework/skills/clarify/SKILL.md +0 -179
  223. package/framework/skills/client-handoff/SKILL.md +0 -135
  224. package/framework/skills/collab-onboard/SKILL.md +0 -111
  225. package/framework/skills/colorize/SKILL.md +0 -170
  226. package/framework/skills/critique/SKILL.md +0 -126
  227. package/framework/skills/deep-research/SKILL.md +0 -240
  228. package/framework/skills/delight/SKILL.md +0 -329
  229. package/framework/skills/deploy/SKILL.md +0 -261
  230. package/framework/skills/deploy-verify/SKILL.md +0 -377
  231. package/framework/skills/deploy-verify/scripts/canary-check.sh +0 -206
  232. package/framework/skills/deploy-verify/scripts/check-console-errors.js +0 -147
  233. package/framework/skills/deploy-verify/scripts/check-cwv.js +0 -139
  234. package/framework/skills/deploy-verify/scripts/project-detect.sh +0 -84
  235. package/framework/skills/deploy-verify/scripts/verify.sh +0 -548
  236. package/framework/skills/design-quieter/SKILL.md +0 -130
  237. package/framework/skills/distill/SKILL.md +0 -149
  238. package/framework/skills/docs-lookup/SKILL.md +0 -79
  239. package/framework/skills/fcm-notifications/SKILL.md +0 -125
  240. package/framework/skills/financial-ledger/SKILL.md +0 -1039
  241. package/framework/skills/frontend-master/NOTICE.md +0 -4
  242. package/framework/skills/frontend-master/SKILL.md +0 -127
  243. package/framework/skills/frontend-master/reference/color-and-contrast.md +0 -132
  244. package/framework/skills/frontend-master/reference/interaction-design.md +0 -123
  245. package/framework/skills/frontend-master/reference/motion-design.md +0 -99
  246. package/framework/skills/frontend-master/reference/responsive-design.md +0 -114
  247. package/framework/skills/frontend-master/reference/spatial-design.md +0 -100
  248. package/framework/skills/frontend-master/reference/typography.md +0 -131
  249. package/framework/skills/frontend-master/reference/ux-writing.md +0 -107
  250. package/framework/skills/harden/SKILL.md +0 -357
  251. package/framework/skills/i18n-rtl/SKILL.md +0 -752
  252. package/framework/skills/learn/SKILL.md +0 -95
  253. package/framework/skills/memory/SKILL.md +0 -50
  254. package/framework/skills/mobile-expo/SKILL.md +0 -977
  255. package/framework/skills/mobile-expo/references/store-checklist.md +0 -550
  256. package/framework/skills/nestjs-backend/README.md +0 -73
  257. package/framework/skills/nestjs-backend/SKILL.md +0 -446
  258. package/framework/skills/nestjs-backend/references/templates.md +0 -1173
  259. package/framework/skills/normalize/SKILL.md +0 -79
  260. package/framework/skills/onboard/SKILL.md +0 -242
  261. package/framework/skills/openrouter-agent/SKILL.md +0 -922
  262. package/framework/skills/polish/SKILL.md +0 -209
  263. package/framework/skills/pr/SKILL.md +0 -66
  264. package/framework/skills/qualia/SKILL.md +0 -199
  265. package/framework/skills/qualia-add-todo/SKILL.md +0 -68
  266. package/framework/skills/qualia-audit-milestone/SKILL.md +0 -95
  267. package/framework/skills/qualia-check-todos/SKILL.md +0 -55
  268. package/framework/skills/qualia-complete-milestone/SKILL.md +0 -134
  269. package/framework/skills/qualia-debug/SKILL.md +0 -149
  270. package/framework/skills/qualia-design/SKILL.md +0 -203
  271. package/framework/skills/qualia-discuss-phase/SKILL.md +0 -72
  272. package/framework/skills/qualia-evolve/SKILL.md +0 -200
  273. package/framework/skills/qualia-execute-phase/SKILL.md +0 -89
  274. package/framework/skills/qualia-framework-audit/SKILL.md +0 -604
  275. package/framework/skills/qualia-guide/SKILL.md +0 -32
  276. package/framework/skills/qualia-help/SKILL.md +0 -114
  277. package/framework/skills/qualia-idk/SKILL.md +0 -352
  278. package/framework/skills/qualia-list-phase-assumptions/SKILL.md +0 -67
  279. package/framework/skills/qualia-new-milestone/SKILL.md +0 -72
  280. package/framework/skills/qualia-new-project/SKILL.md +0 -232
  281. package/framework/skills/qualia-optimize/SKILL.md +0 -417
  282. package/framework/skills/qualia-pause-work/SKILL.md +0 -96
  283. package/framework/skills/qualia-plan-milestone-gaps/SKILL.md +0 -57
  284. package/framework/skills/qualia-plan-phase/SKILL.md +0 -104
  285. package/framework/skills/qualia-production-check/SKILL.md +0 -0
  286. package/framework/skills/qualia-progress/SKILL.md +0 -53
  287. package/framework/skills/qualia-quick/SKILL.md +0 -89
  288. package/framework/skills/qualia-report/SKILL.md +0 -166
  289. package/framework/skills/qualia-research-phase/SKILL.md +0 -88
  290. package/framework/skills/qualia-resume-work/SKILL.md +0 -62
  291. package/framework/skills/qualia-review/SKILL.md +0 -263
  292. package/framework/skills/qualia-start/SKILL.md +0 -161
  293. package/framework/skills/qualia-verify-work/SKILL.md +0 -132
  294. package/framework/skills/rag/SKILL.md +0 -750
  295. package/framework/skills/responsive/SKILL.md +0 -231
  296. package/framework/skills/retro/SKILL.md +0 -284
  297. package/framework/skills/sakani-conventions/SKILL.md +0 -136
  298. package/framework/skills/sakani-conventions/evals/evals.json +0 -23
  299. package/framework/skills/sakani-conventions/references/entities.md +0 -365
  300. package/framework/skills/sakani-conventions/references/error-codes.md +0 -95
  301. package/framework/skills/seo-master/SKILL.md +0 -490
  302. package/framework/skills/seo-master/references/checklist.md +0 -199
  303. package/framework/skills/seo-master/references/structured-data.md +0 -609
  304. package/framework/skills/ship/SKILL.md +0 -239
  305. package/framework/skills/stack-researcher/SKILL.md +0 -215
  306. package/framework/skills/status/SKILL.md +0 -154
  307. package/framework/skills/status/scripts/health-check.sh +0 -562
  308. package/framework/skills/subscription-payments/SKILL.md +0 -250
  309. package/framework/skills/supabase/SKILL.md +0 -973
  310. package/framework/skills/supabase/references/templates.md +0 -159
  311. package/framework/skills/team/SKILL.md +0 -67
  312. package/framework/skills/test-runner/SKILL.md +0 -202
  313. package/framework/skills/voice-agent/SKILL.md +0 -1312
  314. package/framework/skills/zoho-workflow/SKILL.md +0 -51
  315. package/framework/statusline-command.sh +0 -117
  316. package/framework/teams/default/inboxes/plan-04.json +0 -9
  317. package/framework/teams/review-team.md +0 -75
  318. package/framework/teams/ship-team.md +0 -86
  319. package/profiles/fawzi.json +0 -16
  320. package/profiles/hasan.json +0 -16
  321. package/profiles/moayad.json +0 -16
  322. package/templates/CLAUDE-owner.md +0 -52
  323. package/templates/CLAUDE.md.hbs +0 -58
  324. package/templates/env.claude.template +0 -12
  325. package/templates/settings.json +0 -172
  326. /package/{framework/rules → rules}/deployment.md +0 -0
  327. /package/{framework/rules → rules}/security.md +0 -0
@@ -0,0 +1,68 @@
1
+ #!/usr/bin/env node
2
+ // ~/.claude/hooks/branch-guard.js — block non-OWNER push to main/master.
3
+ // PreToolUse hook on `git push*` commands. Reads role from
4
+ // ~/.claude/.qualia-config.json (single source of truth).
5
+ // Exits 1 to BLOCK. Exits 0 to allow.
6
+ // Cross-platform (Windows/macOS/Linux).
7
+
8
+ const fs = require("fs");
9
+ const path = require("path");
10
+ const os = require("os");
11
+ const { spawnSync } = require("child_process");
12
+
13
+ const _traceStart = Date.now();
14
+
15
+ const CONFIG = path.join(os.homedir(), ".claude", ".qualia-config.json");
16
+
17
+ function _trace(hookName, result, extra) {
18
+ try {
19
+ const traceDir = path.join(os.homedir(), ".claude", ".qualia-traces");
20
+ if (!fs.existsSync(traceDir)) fs.mkdirSync(traceDir, { recursive: true });
21
+ const entry = {
22
+ hook: hookName,
23
+ result,
24
+ timestamp: new Date().toISOString(),
25
+ duration_ms: Date.now() - _traceStart,
26
+ ...extra,
27
+ };
28
+ const file = path.join(traceDir, `${new Date().toISOString().split("T")[0]}.jsonl`);
29
+ fs.appendFileSync(file, JSON.stringify(entry) + "\n");
30
+ } catch {}
31
+ }
32
+
33
+ function fail(msg) {
34
+ console.log(msg);
35
+ _trace("branch-guard", "block", { reason: msg });
36
+ process.exit(2);
37
+ }
38
+
39
+ let role = "";
40
+ try {
41
+ const cfg = JSON.parse(fs.readFileSync(CONFIG, "utf8"));
42
+ role = cfg.role || "";
43
+ } catch {
44
+ fail(`BLOCKED: ${CONFIG} missing or unreadable. Run: npx qualia-framework install`);
45
+ }
46
+
47
+ if (!role) {
48
+ fail(`BLOCKED: Cannot determine role from ${CONFIG}. Defaulting to deny.`);
49
+ }
50
+
51
+ // Ask git for the current branch --show-current. Works identically on Windows/macOS/Linux.
52
+ const r = spawnSync("git", ["branch", "--show-current"], {
53
+ encoding: "utf8",
54
+ timeout: 3000,
55
+ });
56
+ const branch = ((r.stdout || "").trim());
57
+
58
+ if (branch === "main" || branch === "master") {
59
+ if (role !== "OWNER") {
60
+ console.log(`BLOCKED: Employees cannot push to ${branch}. Create a feature branch first.`);
61
+ console.log("Run: git checkout -b feature/your-feature-name");
62
+ _trace("branch-guard", "block", { reason: `non-owner push to ${branch}` });
63
+ process.exit(2);
64
+ }
65
+ }
66
+
67
+ _trace("branch-guard", "allow");
68
+ process.exit(0);
@@ -0,0 +1,83 @@
1
+ #!/usr/bin/env node
2
+ // ~/.claude/hooks/migration-guard.js — catch dangerous SQL patterns in migrations.
3
+ // PreToolUse hook on Edit/Write tool calls. Reads tool input as JSON on stdin.
4
+ // Exits 2 to BLOCK. Exits 0 to allow.
5
+ // Cross-platform (Windows/macOS/Linux).
6
+
7
+ const fs = require("fs");
8
+
9
+ const _traceStart = Date.now();
10
+
11
+ function readInput() {
12
+ try {
13
+ const raw = fs.readFileSync(0, "utf8");
14
+ return JSON.parse(raw);
15
+ } catch {
16
+ return {};
17
+ }
18
+ }
19
+
20
+ const input = readInput();
21
+ const ti = input.tool_input || {};
22
+ const file = String(ti.file_path || "").replace(/\\/g, "/");
23
+ const content = String(ti.content || ti.new_string || "");
24
+
25
+ function _trace(hookName, result, extra) {
26
+ try {
27
+ const os = require("os");
28
+ const path = require("path");
29
+ const traceDir = path.join(os.homedir(), ".claude", ".qualia-traces");
30
+ if (!fs.existsSync(traceDir)) fs.mkdirSync(traceDir, { recursive: true });
31
+ const entry = {
32
+ hook: hookName,
33
+ result,
34
+ timestamp: new Date().toISOString(),
35
+ duration_ms: Date.now() - _traceStart,
36
+ ...extra,
37
+ };
38
+ const filePath = path.join(traceDir, `${new Date().toISOString().split("T")[0]}.jsonl`);
39
+ fs.appendFileSync(filePath, JSON.stringify(entry) + "\n");
40
+ } catch {}
41
+ }
42
+
43
+ // Only inspect migration/SQL files
44
+ if (!/migration|migrate|\.sql$/i.test(file)) {
45
+ _trace("migration-guard", "allow", { reason: "non-migration file" });
46
+ process.exit(0);
47
+ }
48
+
49
+ const errors = [];
50
+
51
+ // DROP TABLE without IF EXISTS
52
+ if (/DROP\s+TABLE/i.test(content) && !/IF\s+EXISTS/i.test(content)) {
53
+ errors.push("DROP TABLE without IF EXISTS");
54
+ }
55
+
56
+ // DELETE without WHERE
57
+ if (/DELETE\s+FROM/i.test(content) && !/WHERE/i.test(content)) {
58
+ errors.push("DELETE FROM without WHERE clause");
59
+ }
60
+
61
+ // TRUNCATE (almost always wrong in migrations)
62
+ if (/TRUNCATE/i.test(content)) {
63
+ errors.push("TRUNCATE detected — are you sure?");
64
+ }
65
+
66
+ // CREATE TABLE without RLS
67
+ if (/CREATE\s+TABLE/i.test(content) && !/ENABLE\s+ROW\s+LEVEL\s+SECURITY/i.test(content)) {
68
+ errors.push("CREATE TABLE without ENABLE ROW LEVEL SECURITY");
69
+ }
70
+
71
+ if (errors.length > 0) {
72
+ console.log("⬢ Migration guard — dangerous patterns found:");
73
+ for (const e of errors) {
74
+ console.log(` ✗ ${e}`);
75
+ }
76
+ console.log("");
77
+ console.log("Fix these before proceeding. If intentional, ask Fawzi to approve.");
78
+ _trace("migration-guard", "block", { errors });
79
+ process.exit(2);
80
+ }
81
+
82
+ _trace("migration-guard", "allow");
83
+ process.exit(0);
@@ -0,0 +1,52 @@
1
+ #!/usr/bin/env node
2
+ // ~/.claude/hooks/pre-compact.js — commit STATE.md before context compaction.
3
+ // PreCompact hook. Silent on failure — context compaction must never be blocked.
4
+ // Cross-platform (Windows/macOS/Linux).
5
+
6
+ const fs = require("fs");
7
+ const path = require("path");
8
+ const { spawnSync } = require("child_process");
9
+
10
+ const _traceStart = Date.now();
11
+
12
+ const STATE_FILE = path.join(".planning", "STATE.md");
13
+
14
+ try {
15
+ if (fs.existsSync(STATE_FILE)) {
16
+ console.log("QUALIA: Saving state before compaction...");
17
+ // Check if STATE.md has uncommitted changes
18
+ const diff = spawnSync("git", ["diff", "--name-only", STATE_FILE], {
19
+ encoding: "utf8",
20
+ timeout: 3000,
21
+ });
22
+ if ((diff.stdout || "").includes("STATE.md")) {
23
+ spawnSync("git", ["add", STATE_FILE], { timeout: 3000 });
24
+ spawnSync("git", ["commit", "-m", "state: pre-compaction save"], {
25
+ timeout: 5000,
26
+ stdio: "ignore",
27
+ });
28
+ }
29
+ }
30
+ } catch {
31
+ // Silent — never block compaction
32
+ }
33
+
34
+ function _trace(hookName, result, extra) {
35
+ try {
36
+ const os = require("os");
37
+ const traceDir = path.join(os.homedir(), ".claude", ".qualia-traces");
38
+ if (!fs.existsSync(traceDir)) fs.mkdirSync(traceDir, { recursive: true });
39
+ const entry = {
40
+ hook: hookName,
41
+ result,
42
+ timestamp: new Date().toISOString(),
43
+ duration_ms: Date.now() - _traceStart,
44
+ ...extra,
45
+ };
46
+ const file = path.join(traceDir, `${new Date().toISOString().split("T")[0]}.jsonl`);
47
+ fs.appendFileSync(file, JSON.stringify(entry) + "\n");
48
+ } catch {}
49
+ }
50
+
51
+ _trace("pre-compact", "allow");
52
+ process.exit(0);
@@ -0,0 +1,149 @@
1
+ #!/usr/bin/env node
2
+ // ~/.claude/hooks/pre-deploy-gate.js — quality gates before production deploy.
3
+ // PreToolUse hook on `vercel --prod*` commands. Runs tsc, lint, tests, build,
4
+ // then scans for service_role leaks in client code.
5
+ // Exits 1 to BLOCK deploy. Exits 0 to allow.
6
+ // Cross-platform (Windows/macOS/Linux). No `grep` or `find` — pure Node.
7
+
8
+ const fs = require("fs");
9
+ const path = require("path");
10
+ const { spawnSync } = require("child_process");
11
+
12
+ const _traceStart = Date.now();
13
+
14
+ function _trace(hookName, result, extra) {
15
+ try {
16
+ const os = require("os");
17
+ const traceDir = path.join(os.homedir(), ".claude", ".qualia-traces");
18
+ if (!fs.existsSync(traceDir)) fs.mkdirSync(traceDir, { recursive: true });
19
+ const entry = {
20
+ hook: hookName,
21
+ result,
22
+ timestamp: new Date().toISOString(),
23
+ duration_ms: Date.now() - _traceStart,
24
+ ...extra,
25
+ };
26
+ const file = path.join(traceDir, `${new Date().toISOString().split("T")[0]}.jsonl`);
27
+ fs.appendFileSync(file, JSON.stringify(entry) + "\n");
28
+ } catch {}
29
+ }
30
+
31
+ function runGate(label, cmd, args, { required = true } = {}) {
32
+ const r = spawnSync(cmd, args, {
33
+ stdio: "ignore",
34
+ timeout: 180000,
35
+ shell: process.platform === "win32",
36
+ });
37
+ if (r.status === 0) {
38
+ console.log(` ✓ ${label}`);
39
+ return true;
40
+ }
41
+ if (required) {
42
+ console.log(`BLOCKED: ${label} errors. Fix before deploying.`);
43
+ _trace("pre-deploy-gate", "block", { gate: label });
44
+ process.exit(1);
45
+ }
46
+ return false;
47
+ }
48
+
49
+ function hasScript(name) {
50
+ try {
51
+ const pkg = JSON.parse(fs.readFileSync("package.json", "utf8"));
52
+ return pkg.scripts && typeof pkg.scripts[name] === "string";
53
+ } catch {
54
+ return false;
55
+ }
56
+ }
57
+
58
+ function walk(dir, out = []) {
59
+ if (!fs.existsSync(dir)) return out;
60
+ let entries;
61
+ try {
62
+ entries = fs.readdirSync(dir, { withFileTypes: true });
63
+ } catch {
64
+ return out;
65
+ }
66
+ for (const e of entries) {
67
+ if (e.name === "node_modules" || e.name.startsWith(".")) continue;
68
+ const full = path.join(dir, e.name);
69
+ if (e.isDirectory()) {
70
+ walk(full, out);
71
+ } else if (/\.(ts|tsx|js|jsx|mjs|cjs)$/.test(e.name)) {
72
+ out.push(full);
73
+ }
74
+ }
75
+ return out;
76
+ }
77
+
78
+ function scanServiceRoleLeaks() {
79
+ const roots = ["app", "components", "src", "pages", "lib"];
80
+ const leaks = [];
81
+ for (const root of roots) {
82
+ for (const file of walk(root)) {
83
+ // --- Path-based skips (no I/O needed) ---
84
+
85
+ // Skip server-only files (convention: *.server.ts, server/ dirs)
86
+ if (/\.server\.|[\\/]server[\\/]/.test(file)) continue;
87
+
88
+ // Skip App Router route handlers (always server-side)
89
+ if (/[\\/]route\.(ts|tsx|js|jsx|mjs)$/.test(file)) continue;
90
+
91
+ // Skip middleware (always server-side)
92
+ if (/[\\/]middleware\.(ts|tsx|js|jsx|mjs)$/.test(file)) continue;
93
+
94
+ // Skip files in app/api/ directory (always server-side)
95
+ if (/[\\/]app[\\/]api[\\/]/.test(file)) continue;
96
+
97
+ // --- Content-based checks (requires reading file) ---
98
+ try {
99
+ const content = fs.readFileSync(file, "utf8");
100
+
101
+ // Skip files with "use server" directive (Server Actions / Server Components)
102
+ if (/^["']use server["']/m.test(content)) continue;
103
+
104
+ if (/service_role/.test(content)) {
105
+ leaks.push(file);
106
+ }
107
+ } catch {}
108
+ }
109
+ }
110
+ return leaks;
111
+ }
112
+
113
+ console.log("⬢ Pre-deploy gate...");
114
+
115
+ // TypeScript
116
+ if (fs.existsSync("tsconfig.json")) {
117
+ runGate("TypeScript", "npx", ["tsc", "--noEmit"]);
118
+ }
119
+
120
+ // Lint
121
+ if (hasScript("lint")) {
122
+ runGate("Lint", "npm", ["run", "lint"]);
123
+ }
124
+
125
+ // Tests
126
+ if (hasScript("test")) {
127
+ runGate("Tests", "npm", ["test"]);
128
+ }
129
+
130
+ // Build
131
+ if (hasScript("build")) {
132
+ runGate("Build", "npm", ["run", "build"]);
133
+ }
134
+
135
+ // Security: no service_role in client code
136
+ const leaks = scanServiceRoleLeaks();
137
+ if (leaks.length > 0) {
138
+ console.log("BLOCKED: service_role found in client code. Remove before deploying.");
139
+ for (const f of leaks.slice(0, 10)) {
140
+ console.log(` ✗ ${f}`);
141
+ }
142
+ _trace("pre-deploy-gate", "block", { gate: "security", leaks: leaks.slice(0, 10) });
143
+ process.exit(1);
144
+ }
145
+ console.log(" ✓ Security");
146
+ console.log("⬢ All gates passed.");
147
+
148
+ _trace("pre-deploy-gate", "allow");
149
+ process.exit(0);
@@ -0,0 +1,53 @@
1
+ #!/usr/bin/env node
2
+ // ~/.claude/hooks/pre-push.js — update tracking.json with last commit + timestamp.
3
+ // PreToolUse hook on `git push*` commands. state.js handles phase/status sync;
4
+ // this just stamps the file so the ERP sees fresh commit info on every push.
5
+ // Cross-platform (Windows/macOS/Linux).
6
+
7
+ const fs = require("fs");
8
+ const path = require("path");
9
+ const { spawnSync } = require("child_process");
10
+
11
+ const _traceStart = Date.now();
12
+
13
+ const TRACKING = path.join(".planning", "tracking.json");
14
+
15
+ try {
16
+ if (fs.existsSync(TRACKING)) {
17
+ const r = spawnSync("git", ["log", "--oneline", "-1", "--format=%h"], {
18
+ encoding: "utf8",
19
+ timeout: 3000,
20
+ });
21
+ const lastCommit = ((r.stdout || "").trim());
22
+ const now = new Date().toISOString().replace(/\.\d+Z$/, "Z");
23
+
24
+ const t = JSON.parse(fs.readFileSync(TRACKING, "utf8"));
25
+ if (lastCommit) t.last_commit = lastCommit;
26
+ t.last_updated = now;
27
+ fs.writeFileSync(TRACKING, JSON.stringify(t, null, 2) + "\n");
28
+
29
+ spawnSync("git", ["add", TRACKING], { timeout: 3000 });
30
+ }
31
+ } catch (err) {
32
+ process.stderr.write(`WARNING: tracking sync failed: ${err.message}\n`);
33
+ }
34
+
35
+ function _trace(hookName, result, extra) {
36
+ try {
37
+ const os = require("os");
38
+ const traceDir = path.join(os.homedir(), ".claude", ".qualia-traces");
39
+ if (!fs.existsSync(traceDir)) fs.mkdirSync(traceDir, { recursive: true });
40
+ const entry = {
41
+ hook: hookName,
42
+ result,
43
+ timestamp: new Date().toISOString(),
44
+ duration_ms: Date.now() - _traceStart,
45
+ ...extra,
46
+ };
47
+ const file = path.join(traceDir, `${new Date().toISOString().split("T")[0]}.jsonl`);
48
+ fs.appendFileSync(file, JSON.stringify(entry) + "\n");
49
+ } catch {}
50
+ }
51
+
52
+ _trace("pre-push", "allow");
53
+ process.exit(0);
@@ -0,0 +1,126 @@
1
+ #!/usr/bin/env node
2
+ // ~/.claude/hooks/session-start.js — branded context panel on session start.
3
+ // Cross-platform (Windows/macOS/Linux). Zero shell dependencies.
4
+ //
5
+ // CRITICAL: this hook must NEVER exit non-zero. Claude Code treats any non-zero
6
+ // exit from a SessionStart hook as a "hook error" and shows a red banner. The
7
+ // banner is purely informational — we'd rather render nothing than block a
8
+ // session. Every call is wrapped in try/catch and we always exit 0 at the end.
9
+
10
+ const fs = require("fs");
11
+ const path = require("path");
12
+ const os = require("os");
13
+ const { spawnSync } = require("child_process");
14
+
15
+ const _traceStart = Date.now();
16
+
17
+ const HOME = os.homedir();
18
+ const UI = path.join(HOME, ".claude", "bin", "qualia-ui.js");
19
+ const STATE_FILE = path.join(".planning", "STATE.md");
20
+ const CONTINUE_HERE = ".continue-here.md";
21
+
22
+ function runUi(...args) {
23
+ if (!fs.existsSync(UI)) return;
24
+ try {
25
+ spawnSync(process.execPath, [UI, ...args], {
26
+ stdio: "inherit",
27
+ timeout: 3000,
28
+ });
29
+ } catch {}
30
+ }
31
+
32
+ function getNextCommand() {
33
+ const stateJs = path.join(HOME, ".claude", "bin", "state.js");
34
+ if (!fs.existsSync(stateJs)) return "";
35
+ try {
36
+ const r = spawnSync(process.execPath, [stateJs, "check"], {
37
+ encoding: "utf8",
38
+ timeout: 3000,
39
+ });
40
+ if (!r.stdout) return "";
41
+ const j = JSON.parse(r.stdout);
42
+ return j.next_command || "";
43
+ } catch {
44
+ return "";
45
+ }
46
+ }
47
+
48
+ function fallbackText() {
49
+ // If qualia-ui.js is missing, emit plain text. Keeps the session informative
50
+ // even on a broken install.
51
+ if (fs.existsSync(STATE_FILE)) {
52
+ try {
53
+ const content = fs.readFileSync(STATE_FILE, "utf8");
54
+ const phase = (content.match(/^Phase:\s*(.+)$/m) || [])[1] || "—";
55
+ const status = (content.match(/^Status:\s*(.+)$/m) || [])[1] || "—";
56
+ console.log(`QUALIA: Project loaded. Phase: ${phase.trim()} | Status: ${status.trim()}`);
57
+ console.log("QUALIA: Run /qualia for next step.");
58
+ } catch {
59
+ console.log("QUALIA: Project detected but STATE.md could not be read.");
60
+ }
61
+ } else if (fs.existsSync(CONTINUE_HERE)) {
62
+ console.log("QUALIA: Handoff file found. Read .continue-here.md to resume.");
63
+ } else {
64
+ console.log("QUALIA: No project detected. Run /qualia-new to start.");
65
+ }
66
+ }
67
+
68
+ try {
69
+ if (!fs.existsSync(UI)) {
70
+ fallbackText();
71
+ } else if (fs.existsSync(STATE_FILE)) {
72
+ runUi("banner", "router");
73
+ const next = getNextCommand();
74
+ if (next) {
75
+ console.log("");
76
+ runUi("next", next);
77
+ }
78
+ } else if (fs.existsSync(CONTINUE_HERE)) {
79
+ runUi("banner", "resume");
80
+ runUi("warn", "Previous session found — type /qualia-resume to pick up where you left off");
81
+ console.log("");
82
+ } else {
83
+ // No project — show a welcoming first-run experience
84
+ const config = readConfig();
85
+ const name = config.installed_by || "";
86
+ runUi("banner", "router");
87
+ if (name) {
88
+ console.log(` ${TEAL}Ready when you are, ${name}.${RESET}`);
89
+ }
90
+ console.log("");
91
+ console.log(` ${DIM}Start here:${RESET}`);
92
+ console.log(` ${TEAL}/qualia-new${RESET} ${DIM}Set up a new project${RESET}`);
93
+ console.log(` ${TEAL}/qualia${RESET} ${DIM}What should I do next?${RESET}`);
94
+ console.log(` ${TEAL}/qualia-quick${RESET} ${DIM}Quick fix (skip planning)${RESET}`);
95
+ console.log("");
96
+ }
97
+ } catch {
98
+ // Deliberately silent — hook must never fail
99
+ }
100
+
101
+ function readConfig() {
102
+ try {
103
+ return JSON.parse(fs.readFileSync(path.join(HOME, ".claude", ".qualia-config.json"), "utf8"));
104
+ } catch {
105
+ return {};
106
+ }
107
+ }
108
+
109
+ function _trace(hookName, result, extra) {
110
+ try {
111
+ const traceDir = path.join(os.homedir(), ".claude", ".qualia-traces");
112
+ if (!fs.existsSync(traceDir)) fs.mkdirSync(traceDir, { recursive: true });
113
+ const entry = {
114
+ hook: hookName,
115
+ result,
116
+ timestamp: new Date().toISOString(),
117
+ duration_ms: Date.now() - _traceStart,
118
+ ...extra,
119
+ };
120
+ const file = path.join(traceDir, `${new Date().toISOString().split("T")[0]}.jsonl`);
121
+ fs.appendFileSync(file, JSON.stringify(entry) + "\n");
122
+ } catch {}
123
+ }
124
+
125
+ _trace("session-start", "allow");
126
+ process.exit(0);
package/package.json CHANGED
@@ -1,29 +1,43 @@
1
1
  {
2
2
  "name": "qualia-framework",
3
- "version": "2.5.1",
4
- "description": "Qualia Solutions Claude Code Framework",
3
+ "version": "3.1.0",
4
+ "description": "Claude Code workflow framework by Qualia Solutions. Plan, build, verify, ship.",
5
5
  "bin": {
6
6
  "qualia-framework": "./bin/cli.js"
7
7
  },
8
- "files": [
9
- "bin/",
10
- "framework/",
11
- "profiles/",
12
- "templates/"
8
+ "keywords": [
9
+ "claude-code",
10
+ "claude",
11
+ "ai",
12
+ "framework",
13
+ "workflow",
14
+ "qualia",
15
+ "agents",
16
+ "automation"
13
17
  ],
14
- "engines": {
15
- "node": ">=18.0.0"
16
- },
18
+ "author": "Qualia Solutions <hello@qualia.solutions>",
19
+ "license": "MIT",
17
20
  "repository": {
18
21
  "type": "git",
19
22
  "url": "git+https://github.com/Qualiasolutions/qualia-framework.git"
20
23
  },
21
- "keywords": [
22
- "claude-code",
23
- "qualia",
24
- "framework",
25
- "ai-dev"
24
+ "homepage": "https://github.com/Qualiasolutions/qualia-framework#readme",
25
+ "scripts": {
26
+ "test": "node --test tests/runner.js"
27
+ },
28
+ "files": [
29
+ "bin/",
30
+ "agents/",
31
+ "hooks/",
32
+ "rules/",
33
+ "skills/",
34
+ "templates/",
35
+ "tests/",
36
+ "docs/",
37
+ "CLAUDE.md",
38
+ "guide.md"
26
39
  ],
27
- "author": "Qualia Solutions",
28
- "license": "UNLICENSED"
40
+ "engines": {
41
+ "node": ">=18"
42
+ }
29
43
  }