qualia-framework 2.6.0 → 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.
- package/CLAUDE.md +63 -0
- package/README.md +108 -30
- package/agents/builder.md +110 -0
- package/agents/planner.md +186 -0
- package/agents/qa-browser.md +186 -0
- package/agents/verifier.md +369 -0
- package/bin/cli.js +691 -492
- package/bin/install.js +622 -0
- package/bin/qualia-ui.js +284 -0
- package/bin/state.js +824 -0
- package/bin/statusline.js +252 -0
- package/docs/erp-contract.md +161 -0
- package/guide.md +63 -0
- package/hooks/auto-update.js +117 -0
- package/hooks/block-env-edit.js +52 -0
- package/hooks/branch-guard.js +68 -0
- package/hooks/migration-guard.js +83 -0
- package/hooks/pre-compact.js +52 -0
- package/hooks/pre-deploy-gate.js +149 -0
- package/hooks/pre-push.js +53 -0
- package/hooks/session-start.js +126 -0
- package/package.json +30 -20
- package/rules/design-reference.md +179 -0
- package/rules/frontend.md +126 -0
- package/rules/infrastructure.md +87 -0
- package/skills/qualia/SKILL.md +88 -0
- package/skills/qualia-build/SKILL.md +115 -0
- package/skills/qualia-debug/SKILL.md +87 -0
- package/skills/qualia-design/SKILL.md +99 -0
- package/skills/qualia-handoff/SKILL.md +66 -0
- package/skills/qualia-help/SKILL.md +60 -0
- package/skills/qualia-idk/SKILL.md +8 -0
- package/skills/qualia-learn/SKILL.md +111 -0
- package/skills/qualia-new/SKILL.md +323 -0
- package/skills/qualia-pause/SKILL.md +63 -0
- package/skills/qualia-plan/SKILL.md +101 -0
- package/skills/qualia-polish/SKILL.md +207 -0
- package/skills/qualia-quick/SKILL.md +37 -0
- package/skills/qualia-report/SKILL.md +114 -0
- package/skills/qualia-resume/SKILL.md +49 -0
- package/skills/qualia-review/SKILL.md +161 -0
- package/skills/qualia-ship/SKILL.md +90 -0
- package/skills/qualia-skill-new/SKILL.md +167 -0
- package/skills/qualia-task/SKILL.md +91 -0
- package/skills/qualia-test/SKILL.md +134 -0
- package/skills/qualia-verify/SKILL.md +113 -0
- package/templates/DESIGN.md +475 -0
- package/templates/help.html +476 -0
- package/templates/plan.md +42 -0
- package/templates/project.md +22 -0
- package/templates/state.md +27 -0
- package/templates/tracking.json +20 -0
- package/tests/bin.test.sh +687 -0
- package/tests/hooks.test.sh +384 -0
- package/tests/runner.js +1956 -0
- package/tests/state.test.sh +713 -0
- package/tests/statusline.test.sh +243 -0
- package/bin/collect-metrics.sh +0 -62
- package/framework/.claudeignore +0 -51
- package/framework/CLAUDE.md +0 -51
- package/framework/MCP_SETUP.md +0 -229
- package/framework/agents/architecture-strategist.md +0 -53
- package/framework/agents/backend-agent.md +0 -150
- package/framework/agents/code-simplicity-reviewer.md +0 -86
- package/framework/agents/frontend-agent.md +0 -111
- package/framework/agents/kieran-typescript-reviewer.md +0 -96
- package/framework/agents/performance-oracle.md +0 -111
- package/framework/agents/qualia-codebase-mapper.md +0 -761
- package/framework/agents/qualia-debugger.md +0 -1204
- package/framework/agents/qualia-executor.md +0 -882
- package/framework/agents/qualia-integration-checker.md +0 -424
- package/framework/agents/qualia-phase-researcher.md +0 -457
- package/framework/agents/qualia-plan-checker.md +0 -700
- package/framework/agents/qualia-planner.md +0 -1245
- package/framework/agents/qualia-project-researcher.md +0 -603
- package/framework/agents/qualia-research-synthesizer.md +0 -200
- package/framework/agents/qualia-roadmapper.md +0 -606
- package/framework/agents/qualia-verifier.md +0 -686
- package/framework/agents/red-team-qa.md +0 -130
- package/framework/agents/security-auditor.md +0 -72
- package/framework/agents/team-orchestrator.md +0 -229
- package/framework/agents/teams/framework-audit-team.md +0 -66
- package/framework/agents/teams/full-stack-team.md +0 -48
- package/framework/agents/teams/optimize-team.md +0 -53
- package/framework/agents/teams/review-team.md +0 -70
- package/framework/agents/teams/ship-team.md +0 -86
- package/framework/agents/test-agent.md +0 -182
- package/framework/hooks/auto-format.sh +0 -54
- package/framework/hooks/block-env-edit.sh +0 -42
- package/framework/hooks/branch-guard.sh +0 -43
- package/framework/hooks/confirm-delete.sh +0 -59
- package/framework/hooks/migration-validate.sh +0 -77
- package/framework/hooks/notification-speak.sh +0 -16
- package/framework/hooks/pre-commit.sh +0 -100
- package/framework/hooks/pre-compact.sh +0 -56
- package/framework/hooks/pre-deploy-gate.sh +0 -160
- package/framework/hooks/qualia-colors.sh +0 -32
- package/framework/hooks/retention-cleanup.sh +0 -62
- package/framework/hooks/save-session-state.sh +0 -185
- package/framework/hooks/session-context-loader.sh +0 -96
- package/framework/hooks/session-learn.sh +0 -32
- package/framework/hooks/skill-announce.sh +0 -123
- package/framework/hooks/tool-error-announce.sh +0 -27
- package/framework/install.ps1 +0 -323
- package/framework/install.sh +0 -313
- package/framework/qualia-framework/VERSION +0 -1
- package/framework/qualia-framework/assets/qualia-logo.png +0 -0
- package/framework/qualia-framework/bin/collect-metrics.sh +0 -67
- package/framework/qualia-framework/bin/generate-report-docx.py +0 -429
- package/framework/qualia-framework/bin/qualia-tools.js +0 -2201
- package/framework/qualia-framework/bin/qualia-tools.test.js +0 -1054
- package/framework/qualia-framework/references/checkpoints.md +0 -775
- package/framework/qualia-framework/references/completion-checklists.md +0 -359
- package/framework/qualia-framework/references/continuation-format.md +0 -249
- package/framework/qualia-framework/references/continuation-prompt.md +0 -97
- package/framework/qualia-framework/references/decimal-phase-calculation.md +0 -65
- package/framework/qualia-framework/references/design-quality.md +0 -56
- package/framework/qualia-framework/references/employee-guide.md +0 -167
- package/framework/qualia-framework/references/git-integration.md +0 -254
- package/framework/qualia-framework/references/git-planning-commit.md +0 -50
- package/framework/qualia-framework/references/model-profile-resolution.md +0 -32
- package/framework/qualia-framework/references/model-profiles.md +0 -73
- package/framework/qualia-framework/references/phase-argument-parsing.md +0 -61
- package/framework/qualia-framework/references/planning-config.md +0 -195
- package/framework/qualia-framework/references/questioning.md +0 -141
- package/framework/qualia-framework/references/tdd.md +0 -263
- package/framework/qualia-framework/references/ui-brand.md +0 -160
- package/framework/qualia-framework/references/verification-patterns.md +0 -612
- package/framework/qualia-framework/templates/DEBUG.md +0 -159
- package/framework/qualia-framework/templates/DESIGN.md +0 -81
- package/framework/qualia-framework/templates/UAT.md +0 -247
- package/framework/qualia-framework/templates/codebase/architecture.md +0 -255
- package/framework/qualia-framework/templates/codebase/concerns.md +0 -310
- package/framework/qualia-framework/templates/codebase/conventions.md +0 -307
- package/framework/qualia-framework/templates/codebase/integrations.md +0 -280
- package/framework/qualia-framework/templates/codebase/stack.md +0 -186
- package/framework/qualia-framework/templates/codebase/structure.md +0 -285
- package/framework/qualia-framework/templates/codebase/testing.md +0 -480
- package/framework/qualia-framework/templates/config.json +0 -35
- package/framework/qualia-framework/templates/context.md +0 -283
- package/framework/qualia-framework/templates/continue-here.md +0 -78
- package/framework/qualia-framework/templates/debug-subagent-prompt.md +0 -91
- package/framework/qualia-framework/templates/discovery.md +0 -146
- package/framework/qualia-framework/templates/lab-notes.md +0 -16
- package/framework/qualia-framework/templates/milestone-archive.md +0 -123
- package/framework/qualia-framework/templates/milestone.md +0 -115
- package/framework/qualia-framework/templates/phase-prompt.md +0 -567
- package/framework/qualia-framework/templates/planner-subagent-prompt.md +0 -117
- package/framework/qualia-framework/templates/project.md +0 -184
- package/framework/qualia-framework/templates/projects/ai-agent.md +0 -156
- package/framework/qualia-framework/templates/projects/mobile-app.md +0 -181
- package/framework/qualia-framework/templates/projects/voice-agent.md +0 -134
- package/framework/qualia-framework/templates/projects/website.md +0 -137
- package/framework/qualia-framework/templates/requirements.md +0 -231
- package/framework/qualia-framework/templates/research-project/ARCHITECTURE.md +0 -204
- package/framework/qualia-framework/templates/research-project/FEATURES.md +0 -147
- package/framework/qualia-framework/templates/research-project/PITFALLS.md +0 -200
- package/framework/qualia-framework/templates/research-project/STACK.md +0 -120
- package/framework/qualia-framework/templates/research-project/SUMMARY.md +0 -170
- package/framework/qualia-framework/templates/research.md +0 -552
- package/framework/qualia-framework/templates/roadmap.md +0 -206
- package/framework/qualia-framework/templates/state.md +0 -179
- package/framework/qualia-framework/templates/summary-complex.md +0 -59
- package/framework/qualia-framework/templates/summary-minimal.md +0 -41
- package/framework/qualia-framework/templates/summary-standard.md +0 -48
- package/framework/qualia-framework/templates/summary.md +0 -246
- package/framework/qualia-framework/templates/user-setup.md +0 -311
- package/framework/qualia-framework/templates/verification-report.md +0 -322
- package/framework/qualia-framework/workflows/add-phase.md +0 -179
- package/framework/qualia-framework/workflows/add-todo.md +0 -157
- package/framework/qualia-framework/workflows/audit-milestone.md +0 -241
- package/framework/qualia-framework/workflows/check-todos.md +0 -176
- package/framework/qualia-framework/workflows/complete-milestone.md +0 -858
- package/framework/qualia-framework/workflows/diagnose-issues.md +0 -219
- package/framework/qualia-framework/workflows/discovery-phase.md +0 -289
- package/framework/qualia-framework/workflows/discuss-phase.md +0 -534
- package/framework/qualia-framework/workflows/execute-phase.md +0 -559
- package/framework/qualia-framework/workflows/execute-plan.md +0 -438
- package/framework/qualia-framework/workflows/help.md +0 -470
- package/framework/qualia-framework/workflows/insert-phase.md +0 -220
- package/framework/qualia-framework/workflows/list-phase-assumptions.md +0 -178
- package/framework/qualia-framework/workflows/map-codebase.md +0 -327
- package/framework/qualia-framework/workflows/new-milestone.md +0 -363
- package/framework/qualia-framework/workflows/new-project.md +0 -982
- package/framework/qualia-framework/workflows/pause-work.md +0 -122
- package/framework/qualia-framework/workflows/plan-milestone-gaps.md +0 -256
- package/framework/qualia-framework/workflows/plan-phase.md +0 -422
- package/framework/qualia-framework/workflows/progress.md +0 -389
- package/framework/qualia-framework/workflows/quick.md +0 -252
- package/framework/qualia-framework/workflows/remove-phase.md +0 -326
- package/framework/qualia-framework/workflows/research-phase.md +0 -74
- package/framework/qualia-framework/workflows/resume-project.md +0 -306
- package/framework/qualia-framework/workflows/set-profile.md +0 -80
- package/framework/qualia-framework/workflows/settings.md +0 -145
- package/framework/qualia-framework/workflows/transition.md +0 -556
- package/framework/qualia-framework/workflows/update.md +0 -197
- package/framework/qualia-framework/workflows/verify-phase.md +0 -195
- package/framework/qualia-framework/workflows/verify-work.md +0 -625
- package/framework/rules/context7.md +0 -14
- package/framework/rules/frontend.md +0 -33
- package/framework/rules/speed.md +0 -23
- package/framework/scripts/__pycache__/say.cpython-314.pyc +0 -0
- package/framework/scripts/apply-retention.sh +0 -120
- package/framework/scripts/bootstrap-pop-os.sh +0 -354
- package/framework/scripts/claude-voice +0 -13
- package/framework/scripts/cleanup.sh +0 -131
- package/framework/scripts/cowork-mode.sh +0 -141
- package/framework/scripts/generate-project-claude-md.sh +0 -153
- package/framework/scripts/load-test-webhook.js +0 -172
- package/framework/scripts/say.py +0 -236
- package/framework/scripts/showcase-video-recorder/ffmpeg-builder.js +0 -167
- package/framework/scripts/showcase-video-recorder/playwright-helpers.js +0 -216
- package/framework/scripts/speak.py +0 -55
- package/framework/scripts/speak.sh +0 -18
- package/framework/scripts/status.sh +0 -138
- package/framework/scripts/sync-to-framework.sh +0 -65
- package/framework/scripts/voice-hotkey.py +0 -227
- package/framework/scripts/voice-input.sh +0 -51
- package/framework/skills/animate/SKILL.md +0 -202
- package/framework/skills/bolder/SKILL.md +0 -144
- package/framework/skills/browser-qa/SKILL.md +0 -536
- package/framework/skills/clarify/SKILL.md +0 -179
- package/framework/skills/client-handoff/SKILL.md +0 -135
- package/framework/skills/collab-onboard/SKILL.md +0 -111
- package/framework/skills/colorize/SKILL.md +0 -170
- package/framework/skills/critique/SKILL.md +0 -126
- package/framework/skills/deep-research/SKILL.md +0 -240
- package/framework/skills/delight/SKILL.md +0 -329
- package/framework/skills/deploy/SKILL.md +0 -261
- package/framework/skills/deploy-verify/SKILL.md +0 -377
- package/framework/skills/deploy-verify/scripts/canary-check.sh +0 -206
- package/framework/skills/deploy-verify/scripts/check-console-errors.js +0 -147
- package/framework/skills/deploy-verify/scripts/check-cwv.js +0 -139
- package/framework/skills/deploy-verify/scripts/project-detect.sh +0 -84
- package/framework/skills/deploy-verify/scripts/verify.sh +0 -548
- package/framework/skills/design-quieter/SKILL.md +0 -130
- package/framework/skills/distill/SKILL.md +0 -149
- package/framework/skills/docs-lookup/SKILL.md +0 -79
- package/framework/skills/fcm-notifications/SKILL.md +0 -125
- package/framework/skills/financial-ledger/SKILL.md +0 -1039
- package/framework/skills/frontend-master/NOTICE.md +0 -4
- package/framework/skills/frontend-master/SKILL.md +0 -127
- package/framework/skills/frontend-master/reference/color-and-contrast.md +0 -132
- package/framework/skills/frontend-master/reference/interaction-design.md +0 -123
- package/framework/skills/frontend-master/reference/motion-design.md +0 -99
- package/framework/skills/frontend-master/reference/responsive-design.md +0 -114
- package/framework/skills/frontend-master/reference/spatial-design.md +0 -100
- package/framework/skills/frontend-master/reference/typography.md +0 -131
- package/framework/skills/frontend-master/reference/ux-writing.md +0 -107
- package/framework/skills/harden/SKILL.md +0 -357
- package/framework/skills/i18n-rtl/SKILL.md +0 -752
- package/framework/skills/learn/SKILL.md +0 -95
- package/framework/skills/memory/SKILL.md +0 -50
- package/framework/skills/mobile-expo/SKILL.md +0 -977
- package/framework/skills/mobile-expo/references/store-checklist.md +0 -550
- package/framework/skills/nestjs-backend/README.md +0 -73
- package/framework/skills/nestjs-backend/SKILL.md +0 -446
- package/framework/skills/nestjs-backend/references/templates.md +0 -1173
- package/framework/skills/normalize/SKILL.md +0 -79
- package/framework/skills/onboard/SKILL.md +0 -242
- package/framework/skills/openrouter-agent/SKILL.md +0 -922
- package/framework/skills/polish/SKILL.md +0 -209
- package/framework/skills/pr/SKILL.md +0 -66
- package/framework/skills/qualia/SKILL.md +0 -199
- package/framework/skills/qualia-add-todo/SKILL.md +0 -68
- package/framework/skills/qualia-audit-milestone/SKILL.md +0 -95
- package/framework/skills/qualia-check-todos/SKILL.md +0 -55
- package/framework/skills/qualia-complete-milestone/SKILL.md +0 -134
- package/framework/skills/qualia-debug/SKILL.md +0 -149
- package/framework/skills/qualia-design/SKILL.md +0 -203
- package/framework/skills/qualia-discuss-phase/SKILL.md +0 -72
- package/framework/skills/qualia-evolve/SKILL.md +0 -200
- package/framework/skills/qualia-execute-phase/SKILL.md +0 -89
- package/framework/skills/qualia-framework-audit/SKILL.md +0 -604
- package/framework/skills/qualia-guide/SKILL.md +0 -32
- package/framework/skills/qualia-help/SKILL.md +0 -114
- package/framework/skills/qualia-idk/SKILL.md +0 -352
- package/framework/skills/qualia-list-phase-assumptions/SKILL.md +0 -67
- package/framework/skills/qualia-new-milestone/SKILL.md +0 -72
- package/framework/skills/qualia-new-project/SKILL.md +0 -232
- package/framework/skills/qualia-optimize/SKILL.md +0 -417
- package/framework/skills/qualia-pause-work/SKILL.md +0 -96
- package/framework/skills/qualia-plan-milestone-gaps/SKILL.md +0 -57
- package/framework/skills/qualia-plan-phase/SKILL.md +0 -104
- package/framework/skills/qualia-production-check/SKILL.md +0 -0
- package/framework/skills/qualia-progress/SKILL.md +0 -53
- package/framework/skills/qualia-quick/SKILL.md +0 -89
- package/framework/skills/qualia-report/SKILL.md +0 -166
- package/framework/skills/qualia-research-phase/SKILL.md +0 -88
- package/framework/skills/qualia-resume-work/SKILL.md +0 -62
- package/framework/skills/qualia-review/SKILL.md +0 -263
- package/framework/skills/qualia-start/SKILL.md +0 -161
- package/framework/skills/qualia-verify-work/SKILL.md +0 -132
- package/framework/skills/rag/SKILL.md +0 -750
- package/framework/skills/responsive/SKILL.md +0 -231
- package/framework/skills/retro/SKILL.md +0 -284
- package/framework/skills/sakani-conventions/SKILL.md +0 -136
- package/framework/skills/sakani-conventions/evals/evals.json +0 -23
- package/framework/skills/sakani-conventions/references/entities.md +0 -365
- package/framework/skills/sakani-conventions/references/error-codes.md +0 -95
- package/framework/skills/seo-master/SKILL.md +0 -490
- package/framework/skills/seo-master/references/checklist.md +0 -199
- package/framework/skills/seo-master/references/structured-data.md +0 -609
- package/framework/skills/ship/SKILL.md +0 -239
- package/framework/skills/stack-researcher/SKILL.md +0 -215
- package/framework/skills/status/SKILL.md +0 -154
- package/framework/skills/status/scripts/health-check.sh +0 -562
- package/framework/skills/subscription-payments/SKILL.md +0 -250
- package/framework/skills/supabase/SKILL.md +0 -973
- package/framework/skills/supabase/references/templates.md +0 -159
- package/framework/skills/team/SKILL.md +0 -67
- package/framework/skills/test-runner/SKILL.md +0 -202
- package/framework/skills/voice-agent/SKILL.md +0 -1312
- package/framework/skills/zoho-workflow/SKILL.md +0 -51
- package/framework/statusline-command.sh +0 -117
- package/framework/teams/default/inboxes/plan-04.json +0 -9
- package/framework/teams/review-team.md +0 -75
- package/framework/teams/ship-team.md +0 -86
- package/profiles/fawzi.json +0 -16
- package/profiles/hasan.json +0 -16
- package/profiles/moayad.json +0 -16
- package/templates/CLAUDE-owner.md +0 -52
- package/templates/CLAUDE.md.hbs +0 -58
- package/templates/env.claude.template +0 -12
- package/templates/settings.json +0 -172
- package/uninstall.sh +0 -90
- /package/{framework/rules → rules}/deployment.md +0 -0
- /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,33 +1,43 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "qualia-framework",
|
|
3
|
-
"version": "
|
|
4
|
-
"description": "Qualia Solutions
|
|
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
|
+
"keywords": [
|
|
9
|
+
"claude-code",
|
|
10
|
+
"claude",
|
|
11
|
+
"ai",
|
|
12
|
+
"framework",
|
|
13
|
+
"workflow",
|
|
14
|
+
"qualia",
|
|
15
|
+
"agents",
|
|
16
|
+
"automation"
|
|
17
|
+
],
|
|
18
|
+
"author": "Qualia Solutions <hello@qualia.solutions>",
|
|
19
|
+
"license": "MIT",
|
|
20
|
+
"repository": {
|
|
21
|
+
"type": "git",
|
|
22
|
+
"url": "git+https://github.com/Qualiasolutions/qualia-framework.git"
|
|
23
|
+
},
|
|
24
|
+
"homepage": "https://github.com/Qualiasolutions/qualia-framework#readme",
|
|
8
25
|
"scripts": {
|
|
9
|
-
"
|
|
26
|
+
"test": "node --test tests/runner.js"
|
|
10
27
|
},
|
|
11
28
|
"files": [
|
|
12
29
|
"bin/",
|
|
13
|
-
"
|
|
14
|
-
"
|
|
30
|
+
"agents/",
|
|
31
|
+
"hooks/",
|
|
32
|
+
"rules/",
|
|
33
|
+
"skills/",
|
|
15
34
|
"templates/",
|
|
16
|
-
"
|
|
35
|
+
"tests/",
|
|
36
|
+
"docs/",
|
|
37
|
+
"CLAUDE.md",
|
|
38
|
+
"guide.md"
|
|
17
39
|
],
|
|
18
40
|
"engines": {
|
|
19
|
-
"node": ">=18
|
|
20
|
-
}
|
|
21
|
-
"repository": {
|
|
22
|
-
"type": "git",
|
|
23
|
-
"url": "git+https://github.com/Qualiasolutions/qualia-framework.git"
|
|
24
|
-
},
|
|
25
|
-
"keywords": [
|
|
26
|
-
"claude-code",
|
|
27
|
-
"qualia",
|
|
28
|
-
"framework",
|
|
29
|
-
"ai-dev"
|
|
30
|
-
],
|
|
31
|
-
"author": "Qualia Solutions",
|
|
32
|
-
"license": "UNLICENSED"
|
|
41
|
+
"node": ">=18"
|
|
42
|
+
}
|
|
33
43
|
}
|