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.
Files changed (328) 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 +691 -492
  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 +30 -20
  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/uninstall.sh +0 -90
  327. /package/{framework/rules → rules}/deployment.md +0 -0
  328. /package/{framework/rules → rules}/security.md +0 -0
@@ -1,141 +0,0 @@
1
- #!/bin/bash
2
- # Toggle between full Claude Code config and lean Cowork config
3
- # Usage: cowork-mode.sh on|off|status
4
-
5
- set -e
6
-
7
- CLAUDE_DIR="$HOME/.claude"
8
- CONFIG="$CLAUDE_DIR/settings.json"
9
- SKILLS_DIR="$CLAUDE_DIR/skills"
10
- ARCHIVE_DIR="$CLAUDE_DIR/skills-archive"
11
- BACKUP="$CLAUDE_DIR/.settings-full.json"
12
- LOCK="$CLAUDE_DIR/.cowork-mode"
13
-
14
- # Skills to archive in Cowork (heavy/project-specific)
15
- HEAVY_SKILLS=(
16
- sakani-conventions
17
- nestjs-backend
18
- mobile-expo
19
- financial-ledger
20
- i18n-rtl
21
- fcm-notifications
22
- subscription-payments
23
- voice-agent
24
- zoho-workflow
25
- skill-creator
26
- deep-research
27
- test-runner
28
- git-worktree
29
- )
30
-
31
- # Plugins to disable in Cowork (keep: typescript-lsp, firecrawl, playwright)
32
- DISABLE_PLUGINS=(
33
- "code-review@claude-plugins-official"
34
- "skill-creator@claude-plugins-official"
35
- "claude-md-management@claude-plugins-official"
36
- "code-simplifier@claude-plugins-official"
37
- "cowork-plugin-management@knowledge-work-plugins"
38
- "supabase@claude-plugins-official"
39
- )
40
-
41
- # MCP servers to disable in Cowork (keep: supabase, firecrawl)
42
- DISABLE_MCP=(
43
- "zohomcp"
44
- "sentry"
45
- )
46
-
47
- on() {
48
- if [ -f "$LOCK" ]; then
49
- echo "Already in Cowork mode. Run 'cowork-mode.sh off' first."
50
- exit 1
51
- fi
52
-
53
- echo "Switching to Cowork mode (lean config)..."
54
-
55
- # Backup full settings
56
- cp "$CONFIG" "$BACKUP"
57
-
58
- # Archive heavy skills
59
- mkdir -p "$ARCHIVE_DIR"
60
- local count=0
61
- for skill in "${HEAVY_SKILLS[@]}"; do
62
- if [ -d "$SKILLS_DIR/$skill" ]; then
63
- mv "$SKILLS_DIR/$skill" "$ARCHIVE_DIR/$skill"
64
- count=$((count + 1))
65
- fi
66
- done
67
-
68
- # Disable plugins and MCP servers
69
- node -e "
70
- const fs = require('fs');
71
- const config = JSON.parse(fs.readFileSync(process.argv[1], 'utf8'));
72
-
73
- // Disable plugins
74
- const disablePlugins = process.argv.slice(3);
75
- for (const p of disablePlugins) {
76
- if (config.enabledPlugins && config.enabledPlugins[p] !== undefined) {
77
- config.enabledPlugins[p] = false;
78
- }
79
- }
80
-
81
- // Filter MCP servers
82
- const disableMcp = process.argv[2].split(',').filter(Boolean);
83
- if (config.enabledMcpjsonServers) {
84
- config.enabledMcpjsonServers = config.enabledMcpjsonServers.filter(s => !disableMcp.includes(s));
85
- }
86
-
87
- fs.writeFileSync(process.argv[1], JSON.stringify(config, null, 2));
88
- " "$CONFIG" "$(IFS=,; echo "${DISABLE_MCP[*]}")" "${DISABLE_PLUGINS[@]}"
89
-
90
- # Mark as active
91
- date > "$LOCK"
92
-
93
- echo "Done. Archived $count skills, disabled ${#DISABLE_PLUGINS[@]} plugins, ${#DISABLE_MCP[@]} MCP servers."
94
- echo ""
95
- echo "Now restart Claude desktop app to pick up changes."
96
- echo "Run 'cowork off' when done to restore full config."
97
- }
98
-
99
- off() {
100
- if [ ! -f "$LOCK" ]; then
101
- echo "Not in Cowork mode. Nothing to restore."
102
- exit 1
103
- fi
104
-
105
- echo "Restoring full Claude Code config..."
106
-
107
- # Restore skills from archive
108
- if [ -d "$ARCHIVE_DIR" ]; then
109
- for skill in "$ARCHIVE_DIR"/*/; do
110
- [ -d "$skill" ] || continue
111
- name=$(basename "$skill")
112
- mv "$skill" "$SKILLS_DIR/$name"
113
- done
114
- rmdir "$ARCHIVE_DIR" 2>/dev/null || true
115
- fi
116
-
117
- # Restore full settings
118
- if [ -f "$BACKUP" ]; then
119
- mv "$BACKUP" "$CONFIG"
120
- fi
121
-
122
- rm "$LOCK"
123
- echo "Done. Full config restored."
124
- echo "Restart Claude desktop app to pick up changes."
125
- }
126
-
127
- status() {
128
- if [ -f "$LOCK" ]; then
129
- echo "Cowork mode: ON (since $(cat "$LOCK"))"
130
- [ -d "$ARCHIVE_DIR" ] && echo "Archived skills: $(ls "$ARCHIVE_DIR" 2>/dev/null | wc -l)"
131
- else
132
- echo "Cowork mode: OFF (full config active)"
133
- fi
134
- }
135
-
136
- case "${1:-status}" in
137
- on) on ;;
138
- off) off ;;
139
- status) status ;;
140
- *) echo "Usage: cowork-mode.sh [on|off|status]" ;;
141
- esac
@@ -1,153 +0,0 @@
1
- #!/bin/bash
2
- # Generate CLAUDE.md for all Qualia projects that don't have one
3
- # Run from your main dev machine: bash ~/.claude/scripts/generate-project-claude-md.sh
4
-
5
- set -euo pipefail
6
- shopt -s nullglob
7
-
8
- PROJECTS_BASE="$HOME/Projects"
9
- # Note: Projects confirmed at ~/Projects/ (not ~/Desktop/Projects/)
10
- DEPLOYMENT_MAP="$HOME/.claude/knowledge/deployment-map.md"
11
- CLIENT_PREFS="$HOME/.claude/knowledge/client-prefs.md"
12
-
13
- GREEN='\033[0;32m'
14
- YELLOW='\033[1;33m'
15
- CYAN='\033[0;36m'
16
- NC='\033[0m'
17
-
18
- GENERATED=0
19
- SKIPPED=0
20
-
21
- # Supabase refs from deployment-map.md
22
- declare -A SUPABASE_REFS=(
23
- ["aibossbrainz"]="esymbjpzjjkffpfqukxw"
24
- ["aquador"]="hznpuxplqgszbacxzbhv"
25
- ["dababneh"]="toovladeybmlyzblqtbl"
26
- ["faris"]="lrgwxcqqnhxvvrruepcv"
27
- ["Hammah"]="ctcikdcjhxxqogwvtzrc"
28
- ["maud"]="vspyxscitcuwnrhchskl"
29
- ["mpm"]="llherorsfgbdyqkrrlpc"
30
- ["sofiatesting"]="vceeheaxcrhmpqueudqx"
31
- ["timo"]="qgaqvkmqterlljbegniu"
32
- ["under"]="vvppyijxpgeeijiozlve"
33
- ["vero"]="zskfdlqyzhkzefafqkpx"
34
- ["Zaid"]="uoqiwidqlsoamtugioik"
35
- )
36
-
37
- generate_claude_md() {
38
- local dir="$1"
39
- local name=$(basename "$dir")
40
- local category="$2"
41
-
42
- # Skip if CLAUDE.md already exists
43
- if [ -f "$dir/CLAUDE.md" ]; then
44
- printf "${YELLOW}SKIP${NC} %s (already has CLAUDE.md)\n" "$name"
45
- SKIPPED=$((SKIPPED + 1))
46
- return
47
- fi
48
-
49
- # Skip if not a real project (no package.json and no src/)
50
- if [ ! -f "$dir/package.json" ] && [ ! -d "$dir/src" ] && [ ! -f "$dir/requirements.txt" ]; then
51
- return
52
- fi
53
-
54
- # Detect stack from package.json
55
- local stack=""
56
- local scripts=""
57
- local nextjs_version=""
58
- if [ -f "$dir/package.json" ]; then
59
- nextjs_version=$(node -e "try{const p=require('$dir/package.json');console.log(p.dependencies?.next||p.devDependencies?.next||'')}catch(e){}" 2>/dev/null)
60
- local has_supabase=$(node -e "try{const p=require('$dir/package.json');console.log(p.dependencies?.['@supabase/supabase-js']?'yes':'')}catch(e){}" 2>/dev/null)
61
- local has_tailwind=$(node -e "try{const p=require('$dir/package.json');console.log(p.devDependencies?.tailwindcss||p.dependencies?.tailwindcss?'yes':'')}catch(e){}" 2>/dev/null)
62
-
63
- stack="Next.js${nextjs_version:+ $nextjs_version}, React, TypeScript"
64
- [ "$has_supabase" = "yes" ] && stack="$stack, Supabase"
65
- [ "$has_tailwind" = "yes" ] && stack="$stack, Tailwind"
66
-
67
- # Get npm scripts
68
- scripts=$(node -e "try{const p=require('$dir/package.json');const s=p.scripts||{};Object.keys(s).forEach(k=>console.log('- \`npm run '+k+'\` — '+s[k]))}catch(e){}" 2>/dev/null)
69
- fi
70
-
71
- # Python project detection
72
- if [ -f "$dir/requirements.txt" ]; then
73
- stack="Python"
74
- [ -f "$dir/pyproject.toml" ] && stack="$stack (pyproject)"
75
- fi
76
-
77
- # Supabase ref
78
- local supa_ref="${SUPABASE_REFS[$name]:-}"
79
- local supa_line=""
80
- [ -n "$supa_ref" ] && supa_line="- **Supabase:** \`$supa_ref\`"
81
-
82
- # Detect deploy target
83
- local deploy_target="Vercel"
84
- [ -f "$dir/wrangler.toml" ] || [ -f "$dir/wrangler.jsonc" ] && deploy_target="Cloudflare Workers"
85
-
86
- # Detect domain from vercel.json
87
- local domain=""
88
- if [ -f "$dir/vercel.json" ]; then
89
- domain=$(node -e "try{const v=require('$dir/vercel.json');console.log((v.alias||[])[0]||'')}catch(e){}" 2>/dev/null)
90
- fi
91
- [ -z "$domain" ] && domain="$name.vercel.app"
92
-
93
- # Build the CLAUDE.md
94
- cat > "$dir/CLAUDE.md" << CLAUDEMD
95
- # $name
96
-
97
- ## Purpose
98
- $category project. <!-- UPDATE: Add one-line description -->
99
-
100
- ## Stack
101
- $stack
102
-
103
- ## Config
104
- - **Deploy:** $deploy_target at $domain
105
- ${supa_line}
106
-
107
- ## Commands
108
- ${scripts:-"- \`npm run dev\` — local development
109
- - \`npm run build\` — production build"}
110
-
111
- ## Notes
112
- <!-- Add project-specific gotchas, client preferences, known issues -->
113
- CLAUDEMD
114
-
115
- printf "${GREEN}CREATED${NC} %s/CLAUDE.md\n" "$name"
116
- GENERATED=$((GENERATED + 1))
117
- }
118
-
119
- echo ""
120
- printf "${CYAN}Scanning projects...${NC}\n"
121
- echo ""
122
-
123
- # Live Projects
124
- if [ -d "$PROJECTS_BASE/Live-Projects" ]; then
125
- for dir in "$PROJECTS_BASE/Live-Projects"/*/; do
126
- [ -d "$dir" ] && generate_claude_md "$dir" "Client website"
127
- done
128
- fi
129
-
130
- # AI Agents
131
- if [ -d "$PROJECTS_BASE/aiagents" ]; then
132
- for dir in "$PROJECTS_BASE/aiagents"/*/; do
133
- [ -d "$dir" ] && generate_claude_md "$dir" "AI agent"
134
- done
135
- fi
136
-
137
- # Voice
138
- if [ -d "$PROJECTS_BASE/voice" ]; then
139
- for dir in "$PROJECTS_BASE/voice"/*/; do
140
- [ -d "$dir" ] && generate_claude_md "$dir" "Voice agent"
141
- done
142
- fi
143
-
144
- # Websites
145
- if [ -d "$PROJECTS_BASE/websites" ]; then
146
- for dir in "$PROJECTS_BASE/websites"/*/; do
147
- [ -d "$dir" ] && generate_claude_md "$dir" "Internal tool"
148
- done
149
- fi
150
-
151
- echo ""
152
- printf "${CYAN}Done.${NC} Generated: ${GREEN}$GENERATED${NC} | Skipped: ${YELLOW}$SKIPPED${NC}\n"
153
- printf "Review the generated files and fill in the <!-- UPDATE --> placeholders.\n"
@@ -1,172 +0,0 @@
1
- #!/usr/bin/env node
2
- /**
3
- * Simple load testing for webhook endpoints
4
- * Usage: node load-test-webhook.js <webhook-url> [concurrency=10] [total=100]
5
- */
6
-
7
- const https = require('https');
8
- const http = require('http');
9
-
10
- async function loadTest(webhookUrl, concurrency = 10, totalRequests = 100) {
11
- console.log(`🔥 Load testing ${webhookUrl}`);
12
- console.log(`📊 Concurrency: ${concurrency}, Total: ${totalRequests}\n`);
13
-
14
- const results = {
15
- success: 0,
16
- failed: 0,
17
- times: [],
18
- errors: []
19
- };
20
-
21
- const batches = Math.ceil(totalRequests / concurrency);
22
-
23
- for (let batch = 0; batch < batches; batch++) {
24
- const batchSize = Math.min(concurrency, totalRequests - batch * concurrency);
25
- const promises = [];
26
-
27
- console.log(`🚀 Batch ${batch + 1}/${batches} (${batchSize} requests)`);
28
-
29
- for (let i = 0; i < batchSize; i++) {
30
- promises.push(makeRequest(webhookUrl, batch * concurrency + i + 1));
31
- }
32
-
33
- const batchResults = await Promise.allSettled(promises);
34
-
35
- for (const result of batchResults) {
36
- if (result.status === 'fulfilled') {
37
- results.success++;
38
- results.times.push(result.value.duration);
39
- } else {
40
- results.failed++;
41
- results.errors.push(result.reason.message);
42
- }
43
- }
44
-
45
- // Brief pause between batches
46
- if (batch < batches - 1) {
47
- await sleep(100);
48
- }
49
- }
50
-
51
- printResults(results, totalRequests);
52
- }
53
-
54
- function makeRequest(url, requestId) {
55
- return new Promise((resolve, reject) => {
56
- const startTime = Date.now();
57
- const isHttps = url.startsWith('https');
58
- const client = isHttps ? https : http;
59
-
60
- // Sample WhatsApp webhook payload
61
- const payload = JSON.stringify({
62
- object: "whatsapp_business_account",
63
- entry: [{
64
- id: "123456789",
65
- changes: [{
66
- field: "messages",
67
- value: {
68
- messaging_product: "whatsapp",
69
- metadata: { display_phone_number: "15551234567" },
70
- contacts: [{ profile: { name: "Load Test User" }, wa_id: `loadtest${requestId}` }],
71
- messages: [{
72
- id: `loadtest_${requestId}_${Date.now()}`,
73
- from: `loadtest${requestId}`,
74
- timestamp: Math.floor(Date.now() / 1000).toString(),
75
- type: "text",
76
- text: { body: `Load test message ${requestId}` }
77
- }]
78
- }
79
- }]
80
- }]
81
- });
82
-
83
- const options = {
84
- method: 'POST',
85
- headers: {
86
- 'Content-Type': 'application/json',
87
- 'Content-Length': Buffer.byteLength(payload),
88
- 'User-Agent': 'WhatsApp/2.0 LoadTest'
89
- },
90
- timeout: 10000
91
- };
92
-
93
- const req = client.request(url, options, (res) => {
94
- let data = '';
95
- res.on('data', chunk => data += chunk);
96
- res.on('end', () => {
97
- const duration = Date.now() - startTime;
98
- if (res.statusCode >= 200 && res.statusCode < 300) {
99
- resolve({ duration, statusCode: res.statusCode });
100
- } else {
101
- reject(new Error(`HTTP ${res.statusCode}: ${data}`));
102
- }
103
- });
104
- });
105
-
106
- req.on('error', (error) => {
107
- reject(new Error(`Request failed: ${error.message}`));
108
- });
109
-
110
- req.on('timeout', () => {
111
- req.destroy();
112
- reject(new Error('Request timeout'));
113
- });
114
-
115
- req.write(payload);
116
- req.end();
117
- });
118
- }
119
-
120
- function printResults(results, totalRequests) {
121
- console.log('\n📈 RESULTS');
122
- console.log('=' .repeat(50));
123
-
124
- const successRate = (results.success / totalRequests * 100).toFixed(1);
125
- const avgTime = results.times.length ? (results.times.reduce((a, b) => a + b, 0) / results.times.length).toFixed(0) : 0;
126
- const p95Time = results.times.length ? results.times.sort((a, b) => a - b)[Math.floor(results.times.length * 0.95)] : 0;
127
-
128
- console.log(`✅ Success: ${results.success}/${totalRequests} (${successRate}%)`);
129
- console.log(`❌ Failed: ${results.failed}/${totalRequests}`);
130
- console.log(`⚡ Avg response: ${avgTime}ms`);
131
- console.log(`📊 95th percentile: ${p95Time}ms`);
132
-
133
- if (results.errors.length > 0) {
134
- console.log('\n🚨 ERRORS');
135
- const errorCounts = {};
136
- results.errors.forEach(error => {
137
- errorCounts[error] = (errorCounts[error] || 0) + 1;
138
- });
139
- Object.entries(errorCounts).forEach(([error, count]) => {
140
- console.log(` ${count}x ${error}`);
141
- });
142
- }
143
-
144
- // Production readiness assessment
145
- console.log('\n🎯 PRODUCTION READINESS');
146
- if (successRate >= 95 && avgTime < 5000) {
147
- console.log('✅ READY FOR PRODUCTION');
148
- } else {
149
- console.log('❌ NOT READY FOR PRODUCTION');
150
- if (successRate < 95) console.log(` - Success rate too low (${successRate}% < 95%)`);
151
- if (avgTime >= 5000) console.log(` - Response too slow (${avgTime}ms ≥ 5000ms)`);
152
- }
153
- }
154
-
155
- function sleep(ms) {
156
- return new Promise(resolve => setTimeout(resolve, ms));
157
- }
158
-
159
- // CLI execution
160
- if (require.main === module) {
161
- const [webhookUrl, concurrency = 10, total = 100] = process.argv.slice(2);
162
-
163
- if (!webhookUrl) {
164
- console.error('Usage: node load-test-webhook.js <webhook-url> [concurrency=10] [total=100]');
165
- process.exit(1);
166
- }
167
-
168
- loadTest(webhookUrl, parseInt(concurrency), parseInt(total))
169
- .catch(console.error);
170
- }
171
-
172
- module.exports = { loadTest };
@@ -1,236 +0,0 @@
1
- #!/usr/bin/env python3
2
- """
3
- Smart TTS wrapper with ElevenLabs best practices.
4
- - Filters code, errors, and system noise
5
- - Normalizes text for natural speech (numbers, URLs, abbreviations)
6
- - Only speaks natural conversation
7
-
8
- Usage: echo "message" | say.py OR say.py "message"
9
- """
10
-
11
- import sys
12
- import os
13
- import re
14
- import subprocess
15
- import tempfile
16
- import warnings
17
-
18
- warnings.filterwarnings("ignore")
19
-
20
- # ============================================================================
21
- # CONFIGURATION
22
- # ============================================================================
23
-
24
- VOLUME = 50 # 0-100, set to 0 to mute completely
25
- MAX_LENGTH = 500 # Max characters to speak
26
- MIN_LENGTH = 2 # Min characters
27
- MAX_NEWLINES = 4 # Max newlines allowed
28
-
29
- # Patterns that indicate non-conversational content (reject if matched)
30
- REJECT_PATTERNS = [
31
- r'```', # Code blocks
32
- r'^\s*import\s+', # Import statements
33
- r'^\s*export\s+', # Export statements
34
- r'^\s*const\s+\w+\s*=', # JS variable declarations
35
- r'^\s*let\s+\w+\s*=', # JS let declarations
36
- r'^\s*var\s+\w+\s*=', # JS var declarations
37
- r'^\s*function\s+\w+\s*\(', # JS function definitions
38
- r'^\s*def\s+\w+\s*\(', # Python functions
39
- r'^\s*class\s+\w+', # Class definitions
40
- r'^\s*async\s+', # Async declarations
41
- r'^\s*await\s+', # Await expressions
42
- r'\{\s*\n', # Opening braces with newline (code)
43
- r'^\s*<[a-zA-Z]', # HTML/JSX tags
44
- r'^\s*\d+[→│\|]\s*', # Line number prefixes (file reads)
45
- r'node_modules', # Path noise
46
- r'\.tsx?:\d+', # File:line references (file.ts:123)
47
- r'\.jsx?:\d+', # File:line references (file.js:123)
48
- r'\.py:\d+', # Python file:line references
49
- r'Error:|error:|ERROR|ERR!', # Error messages
50
- r'Warning:|warning:|WARN', # Warnings
51
- r'^\s*at\s+\w+\s*\(', # Stack traces
52
- r'Traceback \(most recent', # Python tracebacks
53
- r'^\s*\$\s+', # Shell prompts
54
- r'^\s*>\s+', # REPL prompts
55
- r'npm\s+(ERR|WARN|notice)', # npm messages
56
- r'^\s*[-*]\s+`', # Markdown list with code
57
- r'^\s*\d+\.\s+`', # Numbered list with code
58
- r'\|\s*\w+\s*\|', # Markdown tables
59
- r'={3,}|_{3,}|-{3,}', # Markdown separators
60
- ]
61
-
62
- # ============================================================================
63
- # TEXT NORMALIZATION (ElevenLabs best practices)
64
- # ============================================================================
65
-
66
- def normalize_numbers(text: str) -> str:
67
- """Convert numbers to speakable format."""
68
- # Simple number to words for common cases
69
- number_words = {
70
- '0': 'zero', '1': 'one', '2': 'two', '3': 'three', '4': 'four',
71
- '5': 'five', '6': 'six', '7': 'seven', '8': 'eight', '9': 'nine',
72
- '10': 'ten', '11': 'eleven', '12': 'twelve'
73
- }
74
-
75
- # Handle percentages: 50% -> fifty percent
76
- text = re.sub(r'(\d+)%', lambda m: f"{m.group(1)} percent", text)
77
-
78
- # Handle simple standalone small numbers
79
- for num, word in number_words.items():
80
- text = re.sub(rf'\b{num}\b', word, text)
81
-
82
- return text
83
-
84
- def normalize_urls(text: str) -> str:
85
- """Convert URLs to speakable format or remove."""
86
- # Remove full URLs - they're not speakable
87
- text = re.sub(r'https?://[^\s]+', '', text)
88
- # Simplify file paths
89
- text = re.sub(r'[~/][\w/.-]+\.(py|sh|js|ts|json|md)', 'the script', text)
90
- return text
91
-
92
- def expand_abbreviations(text: str) -> str:
93
- """Expand common abbreviations for clearer speech."""
94
- abbreviations = {
95
- r'\bTTS\b': 'text to speech',
96
- r'\bAPI\b': 'A P I',
97
- r'\bURL\b': 'U R L',
98
- r'\bUI\b': 'U I',
99
- r'\bAI\b': 'A I',
100
- r'\bOK\b': 'okay',
101
- r'\bok\b': 'okay',
102
- r'\bconfig\b': 'configuration',
103
- r'\brepo\b': 'repository',
104
- r'\binfo\b': 'information',
105
- r'\bdocs?\b': 'documentation',
106
- r'\bauth\b': 'authentication',
107
- r'\benv\b': 'environment',
108
- }
109
- for pattern, replacement in abbreviations.items():
110
- text = re.sub(pattern, replacement, text, flags=re.IGNORECASE)
111
- return text
112
-
113
- def clean_for_speech(text: str) -> str:
114
- """Clean and normalize text for natural speech."""
115
- # Remove markdown formatting
116
- text = re.sub(r'\*\*([^*]+)\*\*', r'\1', text) # Bold
117
- text = re.sub(r'\*([^*]+)\*', r'\1', text) # Italic
118
- text = re.sub(r'`([^`]+)`', r'\1', text) # Inline code
119
- text = re.sub(r'#{1,6}\s*', '', text) # Headers
120
-
121
- # Remove special characters that don't speak well
122
- text = re.sub(r'[→│\|•]', ' ', text)
123
- text = re.sub(r'[\[\]]', '', text) # Square brackets
124
-
125
- # Normalize whitespace
126
- text = re.sub(r'\s+', ' ', text)
127
-
128
- # Apply normalizations
129
- text = normalize_urls(text)
130
- text = expand_abbreviations(text)
131
- text = normalize_numbers(text)
132
-
133
- return text.strip()
134
-
135
- # ============================================================================
136
- # FILTERING
137
- # ============================================================================
138
-
139
- def should_speak(text: str) -> bool:
140
- """Return True if text is natural conversation worth speaking."""
141
- text = text.strip()
142
-
143
- # Length checks
144
- if len(text) < MIN_LENGTH or len(text) > MAX_LENGTH:
145
- return False
146
-
147
- # Too many newlines = probably not conversation
148
- if text.count('\n') > MAX_NEWLINES:
149
- return False
150
-
151
- # Check reject patterns
152
- for pattern in REJECT_PATTERNS:
153
- if re.search(pattern, text, re.MULTILINE | re.IGNORECASE):
154
- return False
155
-
156
- return True
157
-
158
- # ============================================================================
159
- # SPEECH OUTPUT
160
- # ============================================================================
161
-
162
- def speak(text: str):
163
- """Speak text via ElevenLabs with optimized settings."""
164
- if VOLUME <= 0:
165
- return
166
-
167
- try:
168
- from elevenlabs import ElevenLabs
169
- except ImportError:
170
- print("elevenlabs not installed", file=sys.stderr)
171
- return
172
-
173
- api_key = os.environ.get("ELEVEN_API_KEY")
174
- voice_id = os.environ.get("ELEVEN_VOICE_ID", "PB6BdkFkZLbI39GHdnbQ")
175
-
176
- if not api_key:
177
- return
178
-
179
- client = ElevenLabs(api_key=api_key)
180
-
181
- # Clean text for speech
182
- clean_text = clean_for_speech(text)
183
-
184
- if not clean_text or len(clean_text) < MIN_LENGTH:
185
- return
186
-
187
- try:
188
- audio = client.text_to_speech.convert(
189
- voice_id=voice_id,
190
- text=clean_text,
191
- model_id="eleven_turbo_v2_5", # Fast, good quality
192
- )
193
-
194
- with tempfile.NamedTemporaryFile(suffix=".mp3", delete=False) as f:
195
- for chunk in audio:
196
- f.write(chunk)
197
- f.flush()
198
- temp_path = f.name
199
-
200
- # Play audio
201
- try:
202
- subprocess.run(
203
- ["mpv", "--no-video", "--really-quiet", "--volume=50", temp_path],
204
- check=True,
205
- capture_output=True
206
- )
207
- except FileNotFoundError:
208
- try:
209
- subprocess.run(
210
- ["ffplay", "-nodisp", "-autoexit", "-loglevel", "quiet", "-volume", "50", temp_path],
211
- check=True,
212
- capture_output=True
213
- )
214
- except FileNotFoundError:
215
- print("No audio player found (mpv or ffplay)", file=sys.stderr)
216
-
217
- os.unlink(temp_path)
218
-
219
- except Exception as e:
220
- # Silently fail - don't interrupt workflow
221
- pass
222
-
223
- # ============================================================================
224
- # MAIN
225
- # ============================================================================
226
-
227
- if __name__ == "__main__":
228
- if len(sys.argv) > 1:
229
- text = " ".join(sys.argv[1:])
230
- else:
231
- text = sys.stdin.read()
232
-
233
- text = text.strip()
234
-
235
- if should_speak(text):
236
- speak(text)