arboris-cli 1.0.0 → 1.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 (451) hide show
  1. package/dist/cli.mjs +382 -0
  2. package/manifest.json +323 -0
  3. package/package.json +22 -10
  4. package/prisma/skills/accessibility/SKILL.md +147 -0
  5. package/prisma/skills/agent-architecture-audit/SKILL.md +257 -0
  6. package/prisma/skills/agent-eval/SKILL.md +146 -0
  7. package/prisma/skills/agent-harness-construction/SKILL.md +74 -0
  8. package/prisma/skills/agent-introspection-debugging/SKILL.md +154 -0
  9. package/prisma/skills/agent-payment-x402/SKILL.md +225 -0
  10. package/prisma/skills/agent-self-evaluation/SKILL.md +182 -0
  11. package/prisma/skills/agent-self-evaluation/examples/high-score-example.md +87 -0
  12. package/prisma/skills/agent-self-evaluation/examples/low-score-example.md +86 -0
  13. package/prisma/skills/agent-self-evaluation/references/evaluation-criteria.md +71 -0
  14. package/prisma/skills/agent-self-evaluation/references/hook-integration.md +64 -0
  15. package/prisma/skills/agent-self-evaluation/scripts/evaluate.py +408 -0
  16. package/prisma/skills/agent-self-evaluation/templates/evaluation-report.md +86 -0
  17. package/prisma/skills/agent-sort/SKILL.md +216 -0
  18. package/prisma/skills/agentic-engineering/SKILL.md +64 -0
  19. package/prisma/skills/agentic-os/SKILL.md +388 -0
  20. package/prisma/skills/ai-first-engineering/SKILL.md +52 -0
  21. package/prisma/skills/ai-regression-testing/SKILL.md +386 -0
  22. package/prisma/skills/android-clean-architecture/SKILL.md +340 -0
  23. package/prisma/skills/angular-developer/SKILL.md +155 -0
  24. package/prisma/skills/angular-developer/references/angular-animations.md +160 -0
  25. package/prisma/skills/angular-developer/references/angular-aria.md +410 -0
  26. package/prisma/skills/angular-developer/references/cli.md +86 -0
  27. package/prisma/skills/angular-developer/references/component-harnesses.md +59 -0
  28. package/prisma/skills/angular-developer/references/component-styling.md +91 -0
  29. package/prisma/skills/angular-developer/references/components.md +117 -0
  30. package/prisma/skills/angular-developer/references/creating-services.md +97 -0
  31. package/prisma/skills/angular-developer/references/data-resolvers.md +69 -0
  32. package/prisma/skills/angular-developer/references/define-routes.md +67 -0
  33. package/prisma/skills/angular-developer/references/defining-providers.md +72 -0
  34. package/prisma/skills/angular-developer/references/di-fundamentals.md +120 -0
  35. package/prisma/skills/angular-developer/references/e2e-testing.md +56 -0
  36. package/prisma/skills/angular-developer/references/effects.md +83 -0
  37. package/prisma/skills/angular-developer/references/hierarchical-injectors.md +43 -0
  38. package/prisma/skills/angular-developer/references/host-elements.md +80 -0
  39. package/prisma/skills/angular-developer/references/injection-context.md +63 -0
  40. package/prisma/skills/angular-developer/references/inputs.md +101 -0
  41. package/prisma/skills/angular-developer/references/linked-signal.md +59 -0
  42. package/prisma/skills/angular-developer/references/loading-strategies.md +61 -0
  43. package/prisma/skills/angular-developer/references/mcp.md +108 -0
  44. package/prisma/skills/angular-developer/references/navigate-to-routes.md +69 -0
  45. package/prisma/skills/angular-developer/references/outputs.md +86 -0
  46. package/prisma/skills/angular-developer/references/reactive-forms.md +122 -0
  47. package/prisma/skills/angular-developer/references/rendering-strategies.md +44 -0
  48. package/prisma/skills/angular-developer/references/resource.md +77 -0
  49. package/prisma/skills/angular-developer/references/route-animations.md +56 -0
  50. package/prisma/skills/angular-developer/references/route-guards.md +52 -0
  51. package/prisma/skills/angular-developer/references/router-lifecycle.md +45 -0
  52. package/prisma/skills/angular-developer/references/router-testing.md +87 -0
  53. package/prisma/skills/angular-developer/references/show-routes-with-outlets.md +68 -0
  54. package/prisma/skills/angular-developer/references/signal-forms.md +795 -0
  55. package/prisma/skills/angular-developer/references/signals-overview.md +94 -0
  56. package/prisma/skills/angular-developer/references/tailwind-css.md +69 -0
  57. package/prisma/skills/angular-developer/references/template-driven-forms.md +114 -0
  58. package/prisma/skills/angular-developer/references/testing-fundamentals.md +65 -0
  59. package/prisma/skills/api-connector-builder/SKILL.md +121 -0
  60. package/prisma/skills/api-design/SKILL.md +524 -0
  61. package/prisma/skills/architecture-decision-records/SKILL.md +180 -0
  62. package/prisma/skills/article-writing/SKILL.md +80 -0
  63. package/prisma/skills/automation-audit-ops/SKILL.md +143 -0
  64. package/prisma/skills/autonomous-agent-harness/SKILL.md +274 -0
  65. package/prisma/skills/autonomous-loops/SKILL.md +611 -0
  66. package/prisma/skills/backend-patterns/SKILL.md +562 -0
  67. package/prisma/skills/benchmark/SKILL.md +94 -0
  68. package/prisma/skills/benchmark-methodology/SKILL.md +190 -0
  69. package/prisma/skills/benchmark-optimization-loop/SKILL.md +70 -0
  70. package/prisma/skills/blender-motion-state-inspection/SKILL.md +165 -0
  71. package/prisma/skills/blueprint/SKILL.md +106 -0
  72. package/prisma/skills/brand-discovery/SKILL.md +145 -0
  73. package/prisma/skills/brand-discovery/references/10_purpose-why.md +40 -0
  74. package/prisma/skills/brand-discovery/references/20_positioning.md +44 -0
  75. package/prisma/skills/brand-discovery/references/30_audience-niche.md +52 -0
  76. package/prisma/skills/brand-discovery/references/40_personality-archetype.md +57 -0
  77. package/prisma/skills/brand-discovery/references/50_voice-tone.md +59 -0
  78. package/prisma/skills/brand-discovery/references/60_narrative-story.md +50 -0
  79. package/prisma/skills/brand-discovery/references/70_founder-tension.md +49 -0
  80. package/prisma/skills/brand-discovery/references/90_SYNTHESIS.md +133 -0
  81. package/prisma/skills/brand-voice/SKILL.md +98 -0
  82. package/prisma/skills/brand-voice/references/voice-profile-schema.md +55 -0
  83. package/prisma/skills/browser-qa/SKILL.md +105 -0
  84. package/prisma/skills/bun-runtime/SKILL.md +85 -0
  85. package/prisma/skills/canary-watch/SKILL.md +108 -0
  86. package/prisma/skills/carrier-relationship-management/SKILL.md +212 -0
  87. package/prisma/skills/cisco-ios-patterns/SKILL.md +164 -0
  88. package/prisma/skills/ck/SKILL.md +148 -0
  89. package/prisma/skills/ck/commands/forget.mjs +44 -0
  90. package/prisma/skills/ck/commands/info.mjs +24 -0
  91. package/prisma/skills/ck/commands/init.mjs +143 -0
  92. package/prisma/skills/ck/commands/list.mjs +40 -0
  93. package/prisma/skills/ck/commands/migrate.mjs +202 -0
  94. package/prisma/skills/ck/commands/resume.mjs +36 -0
  95. package/prisma/skills/ck/commands/save.mjs +210 -0
  96. package/prisma/skills/ck/commands/shared.mjs +387 -0
  97. package/prisma/skills/ck/hooks/session-start.mjs +224 -0
  98. package/prisma/skills/claude-devfleet/SKILL.md +112 -0
  99. package/prisma/skills/click-path-audit/SKILL.md +245 -0
  100. package/prisma/skills/clickhouse-io/SKILL.md +440 -0
  101. package/prisma/skills/code-tour/SKILL.md +254 -0
  102. package/prisma/skills/codebase-onboarding/SKILL.md +234 -0
  103. package/prisma/skills/codehealth-mcp/SKILL.md +167 -0
  104. package/prisma/skills/coding-standards/SKILL.md +551 -0
  105. package/prisma/skills/competitive-platform-analysis/SKILL.md +214 -0
  106. package/prisma/skills/competitive-report-structure/SKILL.md +162 -0
  107. package/prisma/skills/compose-multiplatform-patterns/SKILL.md +300 -0
  108. package/prisma/skills/config-gc/SKILL.md +120 -0
  109. package/prisma/skills/configure-ecc/SKILL.md +385 -0
  110. package/prisma/skills/connections-optimizer/SKILL.md +190 -0
  111. package/prisma/skills/content-engine/SKILL.md +132 -0
  112. package/prisma/skills/content-hash-cache-pattern/SKILL.md +162 -0
  113. package/prisma/skills/context-budget/SKILL.md +136 -0
  114. package/prisma/skills/continuous-agent-loop/SKILL.md +46 -0
  115. package/prisma/skills/continuous-learning/SKILL.md +132 -0
  116. package/prisma/skills/continuous-learning/config.json +18 -0
  117. package/prisma/skills/continuous-learning/evaluate-session.sh +69 -0
  118. package/prisma/skills/continuous-learning-v2/SKILL.md +361 -0
  119. package/prisma/skills/continuous-learning-v2/agents/observer-loop.sh +359 -0
  120. package/prisma/skills/continuous-learning-v2/agents/observer.md +189 -0
  121. package/prisma/skills/continuous-learning-v2/agents/session-guardian.sh +150 -0
  122. package/prisma/skills/continuous-learning-v2/agents/start-observer.sh +248 -0
  123. package/prisma/skills/continuous-learning-v2/config.json +8 -0
  124. package/prisma/skills/continuous-learning-v2/hooks/observe.sh +585 -0
  125. package/prisma/skills/continuous-learning-v2/scripts/detect-project.sh +322 -0
  126. package/prisma/skills/continuous-learning-v2/scripts/instinct-cli.py +1956 -0
  127. package/prisma/skills/continuous-learning-v2/scripts/lib/homunculus-dir.sh +31 -0
  128. package/prisma/skills/continuous-learning-v2/scripts/migrate-homunculus.sh +68 -0
  129. package/prisma/skills/continuous-learning-v2/scripts/test_parse_instinct.py +1421 -0
  130. package/prisma/skills/cost-aware-llm-pipeline/SKILL.md +184 -0
  131. package/prisma/skills/cost-tracking/SKILL.md +97 -0
  132. package/prisma/skills/council/SKILL.md +204 -0
  133. package/prisma/skills/cpp-coding-standards/SKILL.md +724 -0
  134. package/prisma/skills/cpp-testing/SKILL.md +325 -0
  135. package/prisma/skills/crosspost/SKILL.md +112 -0
  136. package/prisma/skills/csharp-testing/SKILL.md +322 -0
  137. package/prisma/skills/customer-billing-ops/SKILL.md +141 -0
  138. package/prisma/skills/customs-trade-compliance/SKILL.md +263 -0
  139. package/prisma/skills/dart-flutter-patterns/SKILL.md +564 -0
  140. package/prisma/skills/dashboard-builder/SKILL.md +109 -0
  141. package/prisma/skills/data-scraper-agent/SKILL.md +765 -0
  142. package/prisma/skills/data-throughput-accelerator/SKILL.md +73 -0
  143. package/prisma/skills/database-migrations/SKILL.md +430 -0
  144. package/prisma/skills/deep-research/SKILL.md +160 -0
  145. package/prisma/skills/defi-amm-security/SKILL.md +167 -0
  146. package/prisma/skills/delivery-gate/SKILL.md +126 -0
  147. package/prisma/skills/delivery-gate/hooks/quality-gate.py +220 -0
  148. package/prisma/skills/deployment-patterns/SKILL.md +428 -0
  149. package/prisma/skills/design-system/SKILL.md +83 -0
  150. package/prisma/skills/django-celery/SKILL.md +458 -0
  151. package/prisma/skills/django-patterns/SKILL.md +735 -0
  152. package/prisma/skills/django-security/SKILL.md +644 -0
  153. package/prisma/skills/django-tdd/SKILL.md +730 -0
  154. package/prisma/skills/django-verification/SKILL.md +470 -0
  155. package/prisma/skills/dmux-workflows/SKILL.md +192 -0
  156. package/prisma/skills/docker-patterns/SKILL.md +365 -0
  157. package/prisma/skills/documentation-lookup/SKILL.md +91 -0
  158. package/prisma/skills/dotnet-patterns/SKILL.md +322 -0
  159. package/prisma/skills/dynamic-workflow-mode/SKILL.md +124 -0
  160. package/prisma/skills/e2e-testing/SKILL.md +327 -0
  161. package/prisma/skills/ecc-guide/SKILL.md +190 -0
  162. package/prisma/skills/ecc-recipes/SKILL.md +149 -0
  163. package/prisma/skills/ecc-tools-cost-audit/SKILL.md +161 -0
  164. package/prisma/skills/email-ops/SKILL.md +122 -0
  165. package/prisma/skills/energy-procurement/SKILL.md +228 -0
  166. package/prisma/skills/enterprise-agent-ops/SKILL.md +51 -0
  167. package/prisma/skills/error-handling/SKILL.md +377 -0
  168. package/prisma/skills/eval-harness/SKILL.md +271 -0
  169. package/prisma/skills/evm-token-decimals/SKILL.md +131 -0
  170. package/prisma/skills/exa-search/SKILL.md +108 -0
  171. package/prisma/skills/fal-ai-media/SKILL.md +289 -0
  172. package/prisma/skills/fastapi-patterns/SKILL.md +514 -0
  173. package/prisma/skills/finance-billing-ops/SKILL.md +128 -0
  174. package/prisma/skills/flox-environments/SKILL.md +497 -0
  175. package/prisma/skills/flutter-dart-code-review/SKILL.md +436 -0
  176. package/prisma/skills/foundation-models-on-device/SKILL.md +243 -0
  177. package/prisma/skills/frontend-a11y/SKILL.md +446 -0
  178. package/prisma/skills/frontend-design-direction/SKILL.md +93 -0
  179. package/prisma/skills/frontend-patterns/SKILL.md +657 -0
  180. package/prisma/skills/frontend-slides/SKILL.md +185 -0
  181. package/prisma/skills/frontend-slides/STYLE_PRESETS.md +330 -0
  182. package/prisma/skills/frontend-slides/animation-patterns.md +122 -0
  183. package/prisma/skills/frontend-slides/html-template.md +419 -0
  184. package/prisma/skills/frontend-slides/scripts/export-pdf.sh +418 -0
  185. package/prisma/skills/frontend-slides/scripts/extract-pptx.py +96 -0
  186. package/prisma/skills/frontend-slides/viewport-base.css +153 -0
  187. package/prisma/skills/fsharp-testing/SKILL.md +281 -0
  188. package/prisma/skills/gan-style-harness/SKILL.md +279 -0
  189. package/prisma/skills/gateguard/SKILL.md +133 -0
  190. package/prisma/skills/generating-python-installer/SKILL.md +820 -0
  191. package/prisma/skills/git-workflow/SKILL.md +716 -0
  192. package/prisma/skills/github-ops/SKILL.md +145 -0
  193. package/prisma/skills/golang-patterns/SKILL.md +675 -0
  194. package/prisma/skills/golang-testing/SKILL.md +721 -0
  195. package/prisma/skills/google-workspace-ops/SKILL.md +96 -0
  196. package/prisma/skills/growth-log/SKILL.md +128 -0
  197. package/prisma/skills/healthcare-cdss-patterns/SKILL.md +246 -0
  198. package/prisma/skills/healthcare-emr-patterns/SKILL.md +160 -0
  199. package/prisma/skills/healthcare-eval-harness/SKILL.md +208 -0
  200. package/prisma/skills/healthcare-phi-compliance/SKILL.md +146 -0
  201. package/prisma/skills/hermes-imports/SKILL.md +89 -0
  202. package/prisma/skills/hexagonal-architecture/SKILL.md +277 -0
  203. package/prisma/skills/hipaa-compliance/SKILL.md +79 -0
  204. package/prisma/skills/homelab-network-readiness/SKILL.md +170 -0
  205. package/prisma/skills/homelab-network-setup/SKILL.md +130 -0
  206. package/prisma/skills/homelab-pihole-dns/SKILL.md +275 -0
  207. package/prisma/skills/homelab-vlan-segmentation/SKILL.md +312 -0
  208. package/prisma/skills/homelab-wireguard-vpn/SKILL.md +306 -0
  209. package/prisma/skills/hookify-rules/SKILL.md +128 -0
  210. package/prisma/skills/inherit-legacy-style/SKILL.md +157 -0
  211. package/prisma/skills/intent-driven-development/SKILL.md +360 -0
  212. package/prisma/skills/inventory-demand-planning/SKILL.md +247 -0
  213. package/prisma/skills/investor-materials/SKILL.md +97 -0
  214. package/prisma/skills/investor-outreach/SKILL.md +92 -0
  215. package/prisma/skills/ios-icon-gen/SKILL.md +158 -0
  216. package/prisma/skills/ios-icon-gen/scripts/generate_icons.swift +258 -0
  217. package/prisma/skills/ios-icon-gen/scripts/iconify_gen.sh +235 -0
  218. package/prisma/skills/iterative-retrieval/SKILL.md +212 -0
  219. package/prisma/skills/ito-basket-compare/SKILL.md +64 -0
  220. package/prisma/skills/ito-data-atlas-agent/SKILL.md +64 -0
  221. package/prisma/skills/ito-market-intelligence/SKILL.md +61 -0
  222. package/prisma/skills/ito-trade-planner/SKILL.md +68 -0
  223. package/prisma/skills/java-coding-standards/SKILL.md +384 -0
  224. package/prisma/skills/jira-integration/SKILL.md +303 -0
  225. package/prisma/skills/jpa-patterns/SKILL.md +152 -0
  226. package/prisma/skills/knowledge-ops/SKILL.md +155 -0
  227. package/prisma/skills/kotlin-coroutines-flows/SKILL.md +285 -0
  228. package/prisma/skills/kotlin-exposed-patterns/SKILL.md +720 -0
  229. package/prisma/skills/kotlin-ktor-patterns/SKILL.md +690 -0
  230. package/prisma/skills/kotlin-patterns/SKILL.md +712 -0
  231. package/prisma/skills/kotlin-testing/SKILL.md +825 -0
  232. package/prisma/skills/kubernetes-patterns/SKILL.md +756 -0
  233. package/prisma/skills/laravel-patterns/SKILL.md +416 -0
  234. package/prisma/skills/laravel-plugin-discovery/SKILL.md +230 -0
  235. package/prisma/skills/laravel-security/SKILL.md +948 -0
  236. package/prisma/skills/laravel-tdd/SKILL.md +675 -0
  237. package/prisma/skills/laravel-verification/SKILL.md +180 -0
  238. package/prisma/skills/latency-critical-systems/SKILL.md +74 -0
  239. package/prisma/skills/lead-intelligence/SKILL.md +322 -0
  240. package/prisma/skills/lead-intelligence/agents/enrichment-agent.md +85 -0
  241. package/prisma/skills/lead-intelligence/agents/mutual-mapper.md +75 -0
  242. package/prisma/skills/lead-intelligence/agents/outreach-drafter.md +98 -0
  243. package/prisma/skills/lead-intelligence/agents/signal-scorer.md +60 -0
  244. package/prisma/skills/liquid-glass-design/SKILL.md +279 -0
  245. package/prisma/skills/llm-trading-agent-security/SKILL.md +147 -0
  246. package/prisma/skills/logistics-exception-management/SKILL.md +222 -0
  247. package/prisma/skills/loop-design-check/SKILL.md +143 -0
  248. package/prisma/skills/mailtrap-email-integration/SKILL.md +77 -0
  249. package/prisma/skills/make-interfaces-feel-better/SKILL.md +152 -0
  250. package/prisma/skills/manim-video/SKILL.md +90 -0
  251. package/prisma/skills/manim-video/assets/network_graph_scene.py +52 -0
  252. package/prisma/skills/market-research/SKILL.md +76 -0
  253. package/prisma/skills/marketing-campaign/SKILL.md +114 -0
  254. package/prisma/skills/mcp-server-patterns/SKILL.md +70 -0
  255. package/prisma/skills/messages-ops/SKILL.md +105 -0
  256. package/prisma/skills/ml-adoption-playbook/SKILL.md +57 -0
  257. package/prisma/skills/mle-workflow/SKILL.md +347 -0
  258. package/prisma/skills/motion-advanced/SKILL.md +596 -0
  259. package/prisma/skills/motion-foundations/SKILL.md +299 -0
  260. package/prisma/skills/motion-patterns/SKILL.md +434 -0
  261. package/prisma/skills/motion-ui/SKILL.md +576 -0
  262. package/prisma/skills/mysql-patterns/SKILL.md +413 -0
  263. package/prisma/skills/nanoclaw-repl/SKILL.md +34 -0
  264. package/prisma/skills/nestjs-patterns/SKILL.md +231 -0
  265. package/prisma/skills/netmiko-ssh-automation/SKILL.md +174 -0
  266. package/prisma/skills/network-bgp-diagnostics/SKILL.md +168 -0
  267. package/prisma/skills/network-config-validation/SKILL.md +211 -0
  268. package/prisma/skills/network-interface-health/SKILL.md +153 -0
  269. package/prisma/skills/nextjs-turbopack/SKILL.md +58 -0
  270. package/prisma/skills/nodejs-keccak256/SKILL.md +103 -0
  271. package/prisma/skills/nutrient-document-processing/SKILL.md +168 -0
  272. package/prisma/skills/nuxt4-patterns/SKILL.md +101 -0
  273. package/prisma/skills/openclaw-persona-forge/SKILL.md +289 -0
  274. package/prisma/skills/openclaw-persona-forge/gacha.py +224 -0
  275. package/prisma/skills/openclaw-persona-forge/gacha.sh +5 -0
  276. package/prisma/skills/openclaw-persona-forge/references/avatar-style.md +124 -0
  277. package/prisma/skills/openclaw-persona-forge/references/boundary-rules.md +53 -0
  278. package/prisma/skills/openclaw-persona-forge/references/error-handling.md +53 -0
  279. package/prisma/skills/openclaw-persona-forge/references/identity-tension.md +48 -0
  280. package/prisma/skills/openclaw-persona-forge/references/naming-system.md +39 -0
  281. package/prisma/skills/openclaw-persona-forge/references/output-template.md +166 -0
  282. package/prisma/skills/opensource-pipeline/SKILL.md +256 -0
  283. package/prisma/skills/orch-add-feature/SKILL.md +45 -0
  284. package/prisma/skills/orch-build-mvp/SKILL.md +49 -0
  285. package/prisma/skills/orch-change-feature/SKILL.md +43 -0
  286. package/prisma/skills/orch-fix-defect/SKILL.md +43 -0
  287. package/prisma/skills/orch-pipeline/SKILL.md +121 -0
  288. package/prisma/skills/orch-refine-code/SKILL.md +44 -0
  289. package/prisma/skills/parallel-execution-optimizer/SKILL.md +73 -0
  290. package/prisma/skills/perl-patterns/SKILL.md +505 -0
  291. package/prisma/skills/perl-security/SKILL.md +504 -0
  292. package/prisma/skills/perl-testing/SKILL.md +476 -0
  293. package/prisma/skills/plan-orchestrate/SKILL.md +263 -0
  294. package/prisma/skills/plankton-code-quality/SKILL.md +237 -0
  295. package/prisma/skills/postgres-patterns/SKILL.md +148 -0
  296. package/prisma/skills/prediction-market-oracle-research/SKILL.md +64 -0
  297. package/prisma/skills/prediction-market-risk-review/SKILL.md +61 -0
  298. package/prisma/skills/prisma-patterns/SKILL.md +401 -0
  299. package/prisma/skills/product-capability/SKILL.md +142 -0
  300. package/prisma/skills/product-lens/SKILL.md +93 -0
  301. package/prisma/skills/production-audit/SKILL.md +207 -0
  302. package/prisma/skills/production-scheduling/SKILL.md +238 -0
  303. package/prisma/skills/project-flow-ops/SKILL.md +112 -0
  304. package/prisma/skills/prompt-optimizer/SKILL.md +398 -0
  305. package/prisma/skills/python-patterns/SKILL.md +751 -0
  306. package/prisma/skills/python-testing/SKILL.md +817 -0
  307. package/prisma/skills/pytorch-patterns/SKILL.md +397 -0
  308. package/prisma/skills/quality-nonconformance/SKILL.md +260 -0
  309. package/prisma/skills/quarkus-patterns/SKILL.md +723 -0
  310. package/prisma/skills/quarkus-security/SKILL.md +468 -0
  311. package/prisma/skills/quarkus-tdd/SKILL.md +812 -0
  312. package/prisma/skills/quarkus-verification/SKILL.md +480 -0
  313. package/prisma/skills/ralphinho-rfc-pipeline/SKILL.md +68 -0
  314. package/prisma/skills/react-native-patterns/SKILL.md +326 -0
  315. package/prisma/skills/react-patterns/SKILL.md +342 -0
  316. package/prisma/skills/react-performance/SKILL.md +575 -0
  317. package/prisma/skills/react-testing/SKILL.md +424 -0
  318. package/prisma/skills/recsys-pipeline-architect/SKILL.md +115 -0
  319. package/prisma/skills/recursive-decision-ledger/SKILL.md +80 -0
  320. package/prisma/skills/redis-patterns/SKILL.md +404 -0
  321. package/prisma/skills/regex-vs-llm-structured-text/SKILL.md +221 -0
  322. package/prisma/skills/remotion-video-creation/SKILL.md +43 -0
  323. package/prisma/skills/remotion-video-creation/rules/3d.md +86 -0
  324. package/prisma/skills/remotion-video-creation/rules/animations.md +29 -0
  325. package/prisma/skills/remotion-video-creation/rules/assets/charts-bar-chart.tsx +173 -0
  326. package/prisma/skills/remotion-video-creation/rules/assets/text-animations-typewriter.tsx +100 -0
  327. package/prisma/skills/remotion-video-creation/rules/assets/text-animations-word-highlight.tsx +108 -0
  328. package/prisma/skills/remotion-video-creation/rules/assets.md +78 -0
  329. package/prisma/skills/remotion-video-creation/rules/audio.md +172 -0
  330. package/prisma/skills/remotion-video-creation/rules/calculate-metadata.md +104 -0
  331. package/prisma/skills/remotion-video-creation/rules/can-decode.md +75 -0
  332. package/prisma/skills/remotion-video-creation/rules/charts.md +58 -0
  333. package/prisma/skills/remotion-video-creation/rules/compositions.md +146 -0
  334. package/prisma/skills/remotion-video-creation/rules/display-captions.md +126 -0
  335. package/prisma/skills/remotion-video-creation/rules/extract-frames.md +229 -0
  336. package/prisma/skills/remotion-video-creation/rules/fonts.md +152 -0
  337. package/prisma/skills/remotion-video-creation/rules/get-audio-duration.md +58 -0
  338. package/prisma/skills/remotion-video-creation/rules/get-video-dimensions.md +68 -0
  339. package/prisma/skills/remotion-video-creation/rules/get-video-duration.md +58 -0
  340. package/prisma/skills/remotion-video-creation/rules/gifs.md +138 -0
  341. package/prisma/skills/remotion-video-creation/rules/images.md +130 -0
  342. package/prisma/skills/remotion-video-creation/rules/import-srt-captions.md +67 -0
  343. package/prisma/skills/remotion-video-creation/rules/lottie.md +67 -0
  344. package/prisma/skills/remotion-video-creation/rules/measuring-dom-nodes.md +34 -0
  345. package/prisma/skills/remotion-video-creation/rules/measuring-text.md +143 -0
  346. package/prisma/skills/remotion-video-creation/rules/sequencing.md +106 -0
  347. package/prisma/skills/remotion-video-creation/rules/tailwind.md +11 -0
  348. package/prisma/skills/remotion-video-creation/rules/text-animations.md +20 -0
  349. package/prisma/skills/remotion-video-creation/rules/timing.md +179 -0
  350. package/prisma/skills/remotion-video-creation/rules/transcribe-captions.md +19 -0
  351. package/prisma/skills/remotion-video-creation/rules/transitions.md +122 -0
  352. package/prisma/skills/remotion-video-creation/rules/trimming.md +52 -0
  353. package/prisma/skills/remotion-video-creation/rules/videos.md +171 -0
  354. package/prisma/skills/repo-scan/SKILL.md +79 -0
  355. package/prisma/skills/research-ops/SKILL.md +113 -0
  356. package/prisma/skills/returns-reverse-logistics/SKILL.md +240 -0
  357. package/prisma/skills/rules-distill/SKILL.md +265 -0
  358. package/prisma/skills/rules-distill/scripts/scan-rules.sh +58 -0
  359. package/prisma/skills/rules-distill/scripts/scan-skills.sh +129 -0
  360. package/prisma/skills/rust-patterns/SKILL.md +500 -0
  361. package/prisma/skills/rust-testing/SKILL.md +501 -0
  362. package/prisma/skills/safety-guard/SKILL.md +76 -0
  363. package/prisma/skills/santa-method/SKILL.md +307 -0
  364. package/prisma/skills/scientific-db-pubmed-database/SKILL.md +176 -0
  365. package/prisma/skills/scientific-db-uspto-database/SKILL.md +178 -0
  366. package/prisma/skills/scientific-pkg-gget/SKILL.md +167 -0
  367. package/prisma/skills/scientific-thinking-literature-review/SKILL.md +193 -0
  368. package/prisma/skills/scientific-thinking-scholar-evaluation/SKILL.md +161 -0
  369. package/prisma/skills/search-first/SKILL.md +183 -0
  370. package/prisma/skills/security-bounty-hunter/SKILL.md +100 -0
  371. package/prisma/skills/security-review/SKILL.md +504 -0
  372. package/prisma/skills/security-review/cloud-infrastructure-security.md +361 -0
  373. package/prisma/skills/security-scan/SKILL.md +166 -0
  374. package/prisma/skills/seo/SKILL.md +155 -0
  375. package/prisma/skills/skill-comply/SKILL.md +59 -0
  376. package/prisma/skills/skill-comply/fixtures/compliant_trace.jsonl +5 -0
  377. package/prisma/skills/skill-comply/fixtures/noncompliant_trace.jsonl +3 -0
  378. package/prisma/skills/skill-comply/fixtures/tdd_spec.yaml +44 -0
  379. package/prisma/skills/skill-comply/prompts/classifier.md +24 -0
  380. package/prisma/skills/skill-comply/prompts/scenario_generator.md +62 -0
  381. package/prisma/skills/skill-comply/prompts/spec_generator.md +42 -0
  382. package/prisma/skills/skill-comply/pyproject.toml +15 -0
  383. package/prisma/skills/skill-comply/scripts/__init__.py +0 -0
  384. package/prisma/skills/skill-comply/scripts/classifier.py +85 -0
  385. package/prisma/skills/skill-comply/scripts/grader.py +124 -0
  386. package/prisma/skills/skill-comply/scripts/parser.py +107 -0
  387. package/prisma/skills/skill-comply/scripts/report.py +170 -0
  388. package/prisma/skills/skill-comply/scripts/run.py +127 -0
  389. package/prisma/skills/skill-comply/scripts/runner.py +194 -0
  390. package/prisma/skills/skill-comply/scripts/scenario_generator.py +70 -0
  391. package/prisma/skills/skill-comply/scripts/spec_generator.py +72 -0
  392. package/prisma/skills/skill-comply/scripts/utils.py +13 -0
  393. package/prisma/skills/skill-comply/tests/test_grader.py +197 -0
  394. package/prisma/skills/skill-comply/tests/test_parser.py +90 -0
  395. package/prisma/skills/skill-comply/tests/test_runner.py +172 -0
  396. package/prisma/skills/skill-scout/SKILL.md +141 -0
  397. package/prisma/skills/skill-stocktake/SKILL.md +195 -0
  398. package/prisma/skills/skill-stocktake/scripts/quick-diff.sh +87 -0
  399. package/prisma/skills/skill-stocktake/scripts/save-results.sh +56 -0
  400. package/prisma/skills/skill-stocktake/scripts/scan.sh +170 -0
  401. package/prisma/skills/social-graph-ranker/SKILL.md +155 -0
  402. package/prisma/skills/social-publisher/SKILL.md +130 -0
  403. package/prisma/skills/springboot-patterns/SKILL.md +315 -0
  404. package/prisma/skills/springboot-security/SKILL.md +273 -0
  405. package/prisma/skills/springboot-tdd/SKILL.md +159 -0
  406. package/prisma/skills/springboot-verification/SKILL.md +232 -0
  407. package/prisma/skills/strategic-compact/SKILL.md +136 -0
  408. package/prisma/skills/swift-actor-persistence/SKILL.md +144 -0
  409. package/prisma/skills/swift-concurrency-6-2/SKILL.md +216 -0
  410. package/prisma/skills/swift-protocol-di-testing/SKILL.md +191 -0
  411. package/prisma/skills/swiftui-patterns/SKILL.md +259 -0
  412. package/prisma/skills/taste/SKILL.md +264 -0
  413. package/prisma/skills/taste/references/genre-taxonomy.md +87 -0
  414. package/prisma/skills/tdd-workflow/SKILL.md +583 -0
  415. package/prisma/skills/team-agent-orchestration/SKILL.md +111 -0
  416. package/prisma/skills/team-builder/SKILL.md +169 -0
  417. package/prisma/skills/terminal-ops/SKILL.md +110 -0
  418. package/prisma/skills/tinystruct-patterns/SKILL.md +279 -0
  419. package/prisma/skills/tinystruct-patterns/references/architecture.md +90 -0
  420. package/prisma/skills/tinystruct-patterns/references/data-handling.md +60 -0
  421. package/prisma/skills/tinystruct-patterns/references/database.md +99 -0
  422. package/prisma/skills/tinystruct-patterns/references/routing.md +64 -0
  423. package/prisma/skills/tinystruct-patterns/references/system-usage.md +97 -0
  424. package/prisma/skills/tinystruct-patterns/references/testing.md +72 -0
  425. package/prisma/skills/token-budget-advisor/SKILL.md +134 -0
  426. package/prisma/skills/ui-demo/SKILL.md +466 -0
  427. package/prisma/skills/ui-to-vue/SKILL.md +135 -0
  428. package/prisma/skills/uncloud/SKILL.md +344 -0
  429. package/prisma/skills/unified-notifications-ops/SKILL.md +188 -0
  430. package/prisma/skills/verification-loop/SKILL.md +127 -0
  431. package/prisma/skills/video-editing/SKILL.md +311 -0
  432. package/prisma/skills/videodb/SKILL.md +375 -0
  433. package/prisma/skills/videodb/reference/api-reference.md +550 -0
  434. package/prisma/skills/videodb/reference/capture-reference.md +407 -0
  435. package/prisma/skills/videodb/reference/capture.md +101 -0
  436. package/prisma/skills/videodb/reference/editor.md +443 -0
  437. package/prisma/skills/videodb/reference/generative.md +331 -0
  438. package/prisma/skills/videodb/reference/rtstream-reference.md +564 -0
  439. package/prisma/skills/videodb/reference/rtstream.md +65 -0
  440. package/prisma/skills/videodb/reference/search.md +230 -0
  441. package/prisma/skills/videodb/reference/streaming.md +406 -0
  442. package/prisma/skills/videodb/reference/use-cases.md +118 -0
  443. package/prisma/skills/videodb/scripts/ws_listener.py +282 -0
  444. package/prisma/skills/visa-doc-translate/README.md +86 -0
  445. package/prisma/skills/visa-doc-translate/SKILL.md +117 -0
  446. package/prisma/skills/vite-patterns/SKILL.md +450 -0
  447. package/prisma/skills/vue-patterns/SKILL.md +471 -0
  448. package/prisma/skills/windows-desktop-e2e/SKILL.md +888 -0
  449. package/prisma/skills/workspace-surface-audit/SKILL.md +126 -0
  450. package/prisma/skills/x-api/SKILL.md +235 -0
  451. package/run.mjs +0 -10
@@ -0,0 +1,585 @@
1
+ #!/bin/bash
2
+ # Continuous Learning v2 - Observation Hook
3
+ #
4
+ # Captures tool use events for pattern analysis.
5
+ # Claude Code passes hook data via stdin as JSON.
6
+ #
7
+ # v2.1: Project-scoped observations — detects current project context
8
+ # and writes observations to project-specific directory.
9
+ #
10
+ # Registered via plugin hooks/hooks.json (auto-loaded when plugin is enabled).
11
+ # Can also be registered manually in ~/.claude/settings.json.
12
+
13
+ set -e
14
+
15
+ # Hook phase from CLI argument: "pre" (PreToolUse) or "post" (PostToolUse).
16
+ # Manual settings.json installs can call this script without the plugin
17
+ # wrapper's positional phase argument, but Claude Code still exposes the hook
18
+ # event name in CLAUDE_HOOK_EVENT_NAME. Fall back to that env var before
19
+ # defaulting to post so manually registered PreToolUse hooks are recorded as
20
+ # tool_start instead of being silently misclassified as tool_complete.
21
+ HOOK_PHASE="${1:-}"
22
+ if [ -z "$HOOK_PHASE" ]; then
23
+ case "${CLAUDE_HOOK_EVENT_NAME:-}" in
24
+ PreToolUse|pretooluse|pre_tool_use|pre) HOOK_PHASE="pre" ;;
25
+ PostToolUse|posttooluse|post_tool_use|post) HOOK_PHASE="post" ;;
26
+ *) HOOK_PHASE="post" ;;
27
+ esac
28
+ fi
29
+
30
+ # ─────────────────────────────────────────────
31
+ # Read stdin first (before project detection)
32
+ # ─────────────────────────────────────────────
33
+
34
+ # Read JSON from stdin (Claude Code hook format)
35
+ INPUT_JSON=$(cat)
36
+
37
+ # Exit if no input
38
+ if [ -z "$INPUT_JSON" ]; then
39
+ exit 0
40
+ fi
41
+
42
+ _is_windows_app_installer_stub() {
43
+ # Windows 10/11 ships an "App Execution Alias" stub at
44
+ # %LOCALAPPDATA%\Microsoft\WindowsApps\python.exe
45
+ # %LOCALAPPDATA%\Microsoft\WindowsApps\python3.exe
46
+ # Both are symlinks to AppInstallerPythonRedirector.exe which, when Python
47
+ # is not installed from the Store, neither launches Python nor honors "-c".
48
+ # Calls to it hang or print a bare "Python " line, silently breaking every
49
+ # JSON-parsing step in this hook. Detect and skip such stubs here.
50
+ local _candidate="$1"
51
+ [ -z "$_candidate" ] && return 1
52
+ local _resolved
53
+ _resolved="$(command -v "$_candidate" 2>/dev/null || true)"
54
+ [ -z "$_resolved" ] && return 1
55
+ case "$_resolved" in
56
+ *AppInstallerPythonRedirector.exe|*AppInstallerPythonRedirector.EXE) return 0 ;;
57
+ esac
58
+ # Also resolve one level of symlink on POSIX-like shells (Git Bash, WSL).
59
+ if command -v readlink >/dev/null 2>&1; then
60
+ local _target
61
+ _target="$(readlink -f "$_resolved" 2>/dev/null || readlink "$_resolved" 2>/dev/null || true)"
62
+ case "$_target" in
63
+ *AppInstallerPythonRedirector.exe|*AppInstallerPythonRedirector.EXE) return 0 ;;
64
+ esac
65
+ fi
66
+ return 1
67
+ }
68
+
69
+ resolve_python_cmd() {
70
+ if [ -n "${CLV2_PYTHON_CMD:-}" ] && command -v "$CLV2_PYTHON_CMD" >/dev/null 2>&1; then
71
+ printf '%s\n' "$CLV2_PYTHON_CMD"
72
+ return 0
73
+ fi
74
+
75
+ if command -v python3 >/dev/null 2>&1 && ! _is_windows_app_installer_stub python3; then
76
+ printf '%s\n' python3
77
+ return 0
78
+ fi
79
+
80
+ if command -v python >/dev/null 2>&1 && ! _is_windows_app_installer_stub python; then
81
+ printf '%s\n' python
82
+ return 0
83
+ fi
84
+
85
+ return 1
86
+ }
87
+
88
+ PYTHON_CMD="$(resolve_python_cmd 2>/dev/null || true)"
89
+ if [ -z "$PYTHON_CMD" ]; then
90
+ echo "[observe] No python interpreter found, skipping observation" >&2
91
+ exit 0
92
+ fi
93
+
94
+ # Propagate our stub-aware selection so detect-project.sh (which is sourced
95
+ # below) does not re-resolve and silently fall back to the App Installer stub.
96
+ # detect-project.sh honors an already-set CLV2_PYTHON_CMD.
97
+ export CLV2_PYTHON_CMD="${CLV2_PYTHON_CMD:-$PYTHON_CMD}"
98
+
99
+ # ─────────────────────────────────────────────
100
+ # Extract cwd from stdin for project detection
101
+ # ─────────────────────────────────────────────
102
+
103
+ # Extract cwd from the hook JSON to use for project detection.
104
+ # If cwd is a subdirectory inside a git repo, resolve it to the repo root so
105
+ # observations attach to the project instead of a nested path.
106
+ STDIN_CWD=$(echo "$INPUT_JSON" | "$PYTHON_CMD" -c '
107
+ import json, sys
108
+ try:
109
+ data = json.load(sys.stdin)
110
+ cwd = data.get("cwd", "")
111
+ print(cwd)
112
+ except(KeyError, TypeError, ValueError):
113
+ print("")
114
+ ' 2>/dev/null || echo "")
115
+
116
+ # If cwd was provided in stdin, use it for project detection
117
+ if [ -n "$STDIN_CWD" ] && [ -d "$STDIN_CWD" ]; then
118
+ _GIT_ROOT=$(git -C "$STDIN_CWD" rev-parse --show-toplevel 2>/dev/null || true)
119
+ if [ -n "$_GIT_ROOT" ]; then
120
+ export CLAUDE_PROJECT_DIR="$_GIT_ROOT"
121
+ unset CLV2_NO_PROJECT
122
+ else
123
+ unset CLAUDE_PROJECT_DIR
124
+ export CLV2_NO_PROJECT=1
125
+ fi
126
+ fi
127
+
128
+ # ─────────────────────────────────────────────
129
+ # Lightweight config and automated session guards
130
+ # ─────────────────────────────────────────────
131
+ #
132
+ # IMPORTANT: keep these guards above detect-project.sh.
133
+ # Sourcing detect-project.sh creates project-scoped directories and updates
134
+ # projects.json, so automated sessions must return before that point.
135
+
136
+ # shellcheck disable=SC1091
137
+ . "$(dirname "$0")/../scripts/lib/homunculus-dir.sh"
138
+ CONFIG_DIR="$(_ecc_resolve_homunculus_dir)"
139
+
140
+ # Skip if disabled (check both default and CLV2_CONFIG-derived locations)
141
+ if [ -f "$CONFIG_DIR/disabled" ]; then
142
+ exit 0
143
+ fi
144
+ if [ -n "${CLV2_CONFIG:-}" ] && [ -f "$(dirname "$CLV2_CONFIG")/disabled" ]; then
145
+ exit 0
146
+ fi
147
+
148
+ # Prevent observe.sh from firing on non-human sessions to avoid:
149
+ # - ECC observing its own Haiku observer sessions (self-loop)
150
+ # - ECC observing other tools' automated sessions
151
+ # - automated sessions creating project-scoped homunculus metadata
152
+
153
+ # Layer 1: entrypoint. Only interactive terminal sessions should continue.
154
+ # sdk-ts: Agent SDK sessions can be human-interactive (e.g. via Happy).
155
+ # Non-interactive SDK automation is still filtered by Layers 2-5 below
156
+ # (ECC_HOOK_PROFILE=minimal, ECC_SKIP_OBSERVE=1, agent_id, path exclusions).
157
+ case "${CLAUDE_CODE_ENTRYPOINT:-cli}" in
158
+ cli|sdk-ts|claude-desktop|claude-vscode) ;;
159
+ *) exit 0 ;;
160
+ esac
161
+
162
+ # Layer 2: minimal hook profile suppresses non-essential hooks.
163
+ [ "${ECC_HOOK_PROFILE:-standard}" = "minimal" ] && exit 0
164
+
165
+ # Layer 3: cooperative skip env var for automated sessions.
166
+ [ "${ECC_SKIP_OBSERVE:-0}" = "1" ] && exit 0
167
+
168
+ # Layer 4: subagent sessions are automated by definition.
169
+ _ECC_AGENT_ID=$(echo "$INPUT_JSON" | "$PYTHON_CMD" -c "import json,sys; print(json.load(sys.stdin).get('agent_id',''))" 2>/dev/null || true)
170
+ [ -n "$_ECC_AGENT_ID" ] && exit 0
171
+
172
+ # Layer 5: known observer-session path exclusions.
173
+ _ECC_SKIP_PATHS="${ECC_OBSERVE_SKIP_PATHS:-observer-sessions,.claude-mem}"
174
+ if [ -n "$STDIN_CWD" ]; then
175
+ IFS=',' read -ra _ECC_SKIP_ARRAY <<< "$_ECC_SKIP_PATHS"
176
+ for _pattern in "${_ECC_SKIP_ARRAY[@]}"; do
177
+ _pattern="${_pattern#"${_pattern%%[![:space:]]*}"}"
178
+ _pattern="${_pattern%"${_pattern##*[![:space:]]}"}"
179
+ [ -z "$_pattern" ] && continue
180
+ case "$STDIN_CWD" in *"$_pattern"*) exit 0 ;; esac
181
+ done
182
+ fi
183
+
184
+ # ─────────────────────────────────────────────
185
+ # Project detection
186
+ # ─────────────────────────────────────────────
187
+
188
+ SCRIPT_DIR="$(cd "$(dirname "$0")" && pwd)"
189
+ SKILL_ROOT="$(cd "$SCRIPT_DIR/.." && pwd)"
190
+
191
+ # Source shared project detection helper
192
+ # This sets: PROJECT_ID, PROJECT_NAME, PROJECT_ROOT, PROJECT_DIR
193
+ source "${SKILL_ROOT}/scripts/detect-project.sh"
194
+ PYTHON_CMD="${CLV2_PYTHON_CMD:-$PYTHON_CMD}"
195
+
196
+ # ─────────────────────────────────────────────
197
+ # Configuration
198
+ # ─────────────────────────────────────────────
199
+
200
+ OBSERVATIONS_FILE="${PROJECT_DIR}/observations.jsonl"
201
+ MAX_FILE_SIZE_MB=10
202
+
203
+ # Auto-purge observation files older than 30 days (runs once per session)
204
+ PURGE_MARKER="${PROJECT_DIR}/.last-purge"
205
+ if [ ! -f "$PURGE_MARKER" ] || [ "$(find "$PURGE_MARKER" -mtime +1 2>/dev/null)" ]; then
206
+ find "${PROJECT_DIR}" -name "observations-*.jsonl" -mtime +30 -delete 2>/dev/null || true
207
+ touch "$PURGE_MARKER" 2>/dev/null || true
208
+ fi
209
+
210
+ # Parse using Python via stdin pipe (safe for all JSON payloads)
211
+ # Pass HOOK_PHASE via env var since Claude Code does not include hook type in stdin JSON
212
+ PARSED=$(echo "$INPUT_JSON" | HOOK_PHASE="$HOOK_PHASE" "$PYTHON_CMD" -c '
213
+ import json
214
+ import sys
215
+ import os
216
+
217
+ try:
218
+ data = json.load(sys.stdin)
219
+
220
+ # Determine event type from CLI argument passed via env var.
221
+ # Claude Code does NOT include a "hook_type" field in the stdin JSON,
222
+ # so we rely on the shell argument ("pre" or "post") instead.
223
+ hook_phase = os.environ.get("HOOK_PHASE", "post")
224
+ event = "tool_start" if hook_phase == "pre" else "tool_complete"
225
+
226
+ # Extract fields - Claude Code hook format
227
+ tool_name = data.get("tool_name", data.get("tool", "unknown"))
228
+ tool_input = data.get("tool_input", data.get("input", {}))
229
+ tool_output = data.get("tool_response")
230
+ if tool_output is None:
231
+ tool_output = data.get("tool_output", data.get("output", ""))
232
+ session_id = data.get("session_id", "unknown")
233
+ tool_use_id = data.get("tool_use_id", "")
234
+ cwd = data.get("cwd", "")
235
+
236
+ # Truncate large inputs/outputs
237
+ if isinstance(tool_input, dict):
238
+ tool_input_str = json.dumps(tool_input)[:5000]
239
+ else:
240
+ tool_input_str = str(tool_input)[:5000]
241
+
242
+ if isinstance(tool_output, dict):
243
+ tool_response_str = json.dumps(tool_output)[:5000]
244
+ else:
245
+ tool_response_str = str(tool_output)[:5000]
246
+
247
+ print(json.dumps({
248
+ "parsed": True,
249
+ "event": event,
250
+ "tool": tool_name,
251
+ "input": tool_input_str if event == "tool_start" else None,
252
+ "output": tool_response_str if event == "tool_complete" else None,
253
+ "session": session_id,
254
+ "tool_use_id": tool_use_id,
255
+ "cwd": cwd
256
+ }))
257
+ except Exception as e:
258
+ print(json.dumps({"parsed": False, "error": str(e)}))
259
+ ')
260
+
261
+ # Check if parsing succeeded
262
+ PARSED_OK=$(echo "$PARSED" | "$PYTHON_CMD" -c "import json,sys; print(json.load(sys.stdin).get('parsed', False))" 2>/dev/null || echo "False")
263
+
264
+ if [ "$PARSED_OK" != "True" ]; then
265
+ # Fallback: log raw input for debugging (scrub secrets before persisting)
266
+ timestamp=$(date -u +"%Y-%m-%dT%H:%M:%SZ")
267
+ export TIMESTAMP="$timestamp"
268
+ echo "$INPUT_JSON" | "$PYTHON_CMD" -c '
269
+ import json, sys, os, re
270
+
271
+ # Linear-time secret matcher. Bounded quantifiers and a fixed set of auth
272
+ # schemes (instead of a generic [A-Za-z]+\s+ that overlapped the value class)
273
+ # prevent the catastrophic backtracking that pegged python at 100% CPU (#2278).
274
+ _SECRET_RE = re.compile(
275
+ r"(?i)(api[_-]?key|token|secret|password|authorization|credentials?|auth)"
276
+ r"""(["'"'"'\s:=]{1,8})"""
277
+ r"((?:bearer|basic|token|bot)\s+)?"
278
+ r"([A-Za-z0-9_\-/.+=]{8,256})"
279
+ )
280
+
281
+ import signal
282
+ def _ecc_bail(*_):
283
+ print("[observe] SIGALRM timeout: parse-error fallback observation dropped before write (#2300)", file=sys.stderr)
284
+ sys.exit(0)
285
+ try:
286
+ signal.signal(signal.SIGALRM, _ecc_bail)
287
+ signal.alarm(8) # self-terminate before the async hook 10s timeout can orphan us (#2278)
288
+ except Exception:
289
+ pass
290
+
291
+ raw = sys.stdin.read()[:2000]
292
+ raw = _SECRET_RE.sub(lambda m: m.group(1) + m.group(2) + (m.group(3) or "") + "[REDACTED]", raw)
293
+ print(json.dumps({"timestamp": os.environ["TIMESTAMP"], "event": "parse_error", "raw": raw}))
294
+ ' >> "$OBSERVATIONS_FILE"
295
+ exit 0
296
+ fi
297
+
298
+ # Archive if file too large (atomic: rename with unique suffix to avoid race)
299
+ if [ -f "$OBSERVATIONS_FILE" ]; then
300
+ file_size_mb=$(du -m "$OBSERVATIONS_FILE" 2>/dev/null | cut -f1)
301
+ if [ "${file_size_mb:-0}" -ge "$MAX_FILE_SIZE_MB" ]; then
302
+ archive_dir="${PROJECT_DIR}/observations.archive"
303
+ mkdir -p "$archive_dir"
304
+ mv "$OBSERVATIONS_FILE" "$archive_dir/observations-$(date +%Y%m%d-%H%M%S)-$$.jsonl" 2>/dev/null || true
305
+ fi
306
+ fi
307
+
308
+ # Build and write observation (now includes project context)
309
+ # Scrub common secret patterns from tool I/O before persisting
310
+ timestamp=$(date -u +"%Y-%m-%dT%H:%M:%SZ")
311
+
312
+ export PROJECT_ID_ENV="$PROJECT_ID"
313
+ export PROJECT_NAME_ENV="$PROJECT_NAME"
314
+ export TIMESTAMP="$timestamp"
315
+
316
+ echo "$PARSED" | "$PYTHON_CMD" -c '
317
+ import json, sys, os, re
318
+ import signal
319
+
320
+ def _ecc_bail(*_):
321
+ print("[observe] SIGALRM timeout: in-flight observation dropped before write (#2300)", file=sys.stderr)
322
+ sys.exit(0)
323
+ try:
324
+ signal.signal(signal.SIGALRM, _ecc_bail)
325
+ signal.alarm(8) # self-terminate before the async hook 10s timeout can orphan us (#2278)
326
+ except Exception:
327
+ pass
328
+
329
+ parsed = json.load(sys.stdin)
330
+ observation = {
331
+ "timestamp": os.environ["TIMESTAMP"],
332
+ "event": parsed["event"],
333
+ "tool": parsed["tool"],
334
+ "session": parsed["session"],
335
+ "project_id": os.environ.get("PROJECT_ID_ENV", "global"),
336
+ "project_name": os.environ.get("PROJECT_NAME_ENV", "global")
337
+ }
338
+
339
+ # Scrub secrets: match common key=value, key: value, and key"value patterns
340
+ # Includes optional auth scheme (e.g., "Bearer", "Basic") before token
341
+ # Linear-time secret matcher. Bounded quantifiers and a fixed set of auth
342
+ # schemes (instead of a generic [A-Za-z]+\s+ that overlapped the value class)
343
+ # prevent the catastrophic backtracking that pegged python at 100% CPU (#2278).
344
+ _SECRET_RE = re.compile(
345
+ r"(?i)(api[_-]?key|token|secret|password|authorization|credentials?|auth)"
346
+ r"""(["'"'"'\s:=]{1,8})"""
347
+ r"((?:bearer|basic|token|bot)\s+)?"
348
+ r"([A-Za-z0-9_\-/.+=]{8,256})"
349
+ )
350
+
351
+ def scrub(val):
352
+ if val is None:
353
+ return None
354
+ return _SECRET_RE.sub(lambda m: m.group(1) + m.group(2) + (m.group(3) or "") + "[REDACTED]", str(val))
355
+
356
+ if parsed["input"]:
357
+ observation["input"] = scrub(parsed["input"])
358
+ if parsed["output"] is not None:
359
+ observation["output"] = scrub(parsed["output"])
360
+
361
+ print(json.dumps(observation))
362
+ ' >> "$OBSERVATIONS_FILE"
363
+
364
+ # Lazy-start observer if enabled but not running (first-time setup)
365
+ # Use flock for atomic check-then-act to prevent race conditions
366
+ # Fallback for macOS (no flock): use lockfile or skip
367
+ LAZY_START_LOCK="${PROJECT_DIR}/.observer-start.lock"
368
+ _REMOVE_FILE_IF_PRESENT() {
369
+ local target="$1"
370
+ if [ -n "$target" ] && [ -e "$target" ]; then
371
+ rm -- "$target" 2>/dev/null || true
372
+ fi
373
+ }
374
+
375
+ _START_OBSERVER_LOGGED() {
376
+ local bootstrap_log="${PROJECT_DIR}/observer-start.log"
377
+ mkdir -p "$PROJECT_DIR"
378
+ "${SKILL_ROOT}/agents/start-observer.sh" start >> "$bootstrap_log" 2>&1 || true
379
+ }
380
+
381
+ _CHECK_OBSERVER_RUNNING() {
382
+ local pid_file="$1"
383
+ if [ -f "$pid_file" ]; then
384
+ local pid
385
+ pid=$(cat "$pid_file" 2>/dev/null)
386
+ # Validate PID is a positive integer (>1) to prevent signaling invalid targets
387
+ case "$pid" in
388
+ ''|*[!0-9]*|0|1)
389
+ _REMOVE_FILE_IF_PRESENT "$pid_file"
390
+ return 1
391
+ ;;
392
+ esac
393
+ if kill -0 "$pid" 2>/dev/null; then
394
+ return 0 # Process is alive
395
+ fi
396
+ # Stale PID file - remove it
397
+ _REMOVE_FILE_IF_PRESENT "$pid_file"
398
+ fi
399
+ return 1 # No PID file or process dead
400
+ }
401
+
402
+ if [ -f "${CONFIG_DIR}/disabled" ]; then
403
+ OBSERVER_ENABLED=false
404
+ else
405
+ OBSERVER_ENABLED=false
406
+ if [ -n "${CLV2_CONFIG:-}" ]; then
407
+ CONFIG_FILE="$CLV2_CONFIG"
408
+ elif [ -f "${CONFIG_DIR}/config.json" ]; then
409
+ CONFIG_FILE="${CONFIG_DIR}/config.json"
410
+ else
411
+ CONFIG_FILE="${SKILL_ROOT}/config.json"
412
+ fi
413
+ # Use effective config path for both existence check and reading
414
+ EFFECTIVE_CONFIG="$CONFIG_FILE"
415
+ if [ -f "$EFFECTIVE_CONFIG" ] && [ -n "$PYTHON_CMD" ]; then
416
+ _enabled=$(CLV2_CONFIG_PATH="$EFFECTIVE_CONFIG" "$PYTHON_CMD" -c "
417
+ import json, os
418
+ with open(os.environ['CLV2_CONFIG_PATH']) as f:
419
+ cfg = json.load(f)
420
+ print(str(cfg.get('observer', {}).get('enabled', False)).lower())
421
+ " 2>/dev/null || echo "false")
422
+ if [ "$_enabled" = "true" ]; then
423
+ OBSERVER_ENABLED=true
424
+ fi
425
+ fi
426
+ fi
427
+
428
+ # Check both project-scoped AND global PID files (with stale PID recovery)
429
+ if [ "$OBSERVER_ENABLED" = "true" ]; then
430
+ # Clean up stale PID files first
431
+ _CHECK_OBSERVER_RUNNING "${PROJECT_DIR}/.observer.pid" || true
432
+ _CHECK_OBSERVER_RUNNING "${CONFIG_DIR}/.observer.pid" || true
433
+
434
+ # Check if observer is now running after cleanup
435
+ if [ ! -f "${PROJECT_DIR}/.observer.pid" ] && [ ! -f "${CONFIG_DIR}/.observer.pid" ]; then
436
+ # Use flock if available (Linux), fallback for macOS
437
+ if command -v flock >/dev/null 2>&1; then
438
+ (
439
+ flock -n 9 || exit 0
440
+ # Double-check PID files after acquiring lock
441
+ _CHECK_OBSERVER_RUNNING "${PROJECT_DIR}/.observer.pid" || true
442
+ _CHECK_OBSERVER_RUNNING "${CONFIG_DIR}/.observer.pid" || true
443
+ if [ ! -f "${PROJECT_DIR}/.observer.pid" ] && [ ! -f "${CONFIG_DIR}/.observer.pid" ]; then
444
+ _START_OBSERVER_LOGGED
445
+ fi
446
+ ) 9>"$LAZY_START_LOCK"
447
+ else
448
+ # macOS fallback: use lockfile if available, otherwise mkdir-based lock
449
+ if command -v lockfile >/dev/null 2>&1; then
450
+ # Use subshell to isolate exit and add trap for cleanup
451
+ (
452
+ trap '_REMOVE_FILE_IF_PRESENT "$LAZY_START_LOCK"' EXIT
453
+ lockfile -r 1 -l 30 "$LAZY_START_LOCK" 2>/dev/null || exit 0
454
+ _CHECK_OBSERVER_RUNNING "${PROJECT_DIR}/.observer.pid" || true
455
+ _CHECK_OBSERVER_RUNNING "${CONFIG_DIR}/.observer.pid" || true
456
+ if [ ! -f "${PROJECT_DIR}/.observer.pid" ] && [ ! -f "${CONFIG_DIR}/.observer.pid" ]; then
457
+ _START_OBSERVER_LOGGED
458
+ fi
459
+ _REMOVE_FILE_IF_PRESENT "$LAZY_START_LOCK"
460
+ )
461
+ else
462
+ # POSIX fallback: mkdir is atomic -- fails if dir already exists
463
+ (
464
+ trap 'rmdir "${LAZY_START_LOCK}.d" 2>/dev/null || true' EXIT
465
+ mkdir "${LAZY_START_LOCK}.d" 2>/dev/null || exit 0
466
+ _CHECK_OBSERVER_RUNNING "${PROJECT_DIR}/.observer.pid" || true
467
+ _CHECK_OBSERVER_RUNNING "${CONFIG_DIR}/.observer.pid" || true
468
+ if [ ! -f "${PROJECT_DIR}/.observer.pid" ] && [ ! -f "${CONFIG_DIR}/.observer.pid" ]; then
469
+ _START_OBSERVER_LOGGED
470
+ fi
471
+ )
472
+ fi
473
+ fi
474
+ fi
475
+ fi
476
+
477
+ # Throttle SIGUSR1: only signal observer every N observations (#521)
478
+ # This prevents rapid signaling when tool calls fire every second,
479
+ # which caused runaway parallel Claude analysis processes.
480
+ SIGNAL_EVERY_N="${ECC_OBSERVER_SIGNAL_EVERY_N:-20}"
481
+ SIGNAL_COUNTER_FILE="${PROJECT_DIR}/.observer-signal-counter"
482
+ SIGNAL_COUNTER_LOCK="${SIGNAL_COUNTER_FILE}.lock"
483
+ ACTIVITY_FILE="${PROJECT_DIR}/.observer-last-activity"
484
+
485
+ touch "$ACTIVITY_FILE" 2>/dev/null || true
486
+
487
+ # Serialize the throttle-counter read-modify-write. observe.sh runs on every
488
+ # tool call (which can fire every second), so concurrent invocations previously
489
+ # raced on this counter: both read the same value, both incremented, and one
490
+ # write was lost, signaling the observer at unpredictable intervals (#2296).
491
+ # Prefer flock (a kernel advisory lock the OS releases automatically if the hook
492
+ # is killed); fall back to the atomic mkdir lock this script already uses for
493
+ # the lazy-start path above. Both wrap the same read-modify-write below.
494
+ should_signal=0
495
+
496
+ _ecc_bump_signal_counter() {
497
+ if [ -f "$SIGNAL_COUNTER_FILE" ]; then
498
+ counter=$(cat "$SIGNAL_COUNTER_FILE" 2>/dev/null || echo 0)
499
+ # Guard against a corrupt counter file: a non-integer value would abort the
500
+ # hook under `set -e` at the arithmetic below.
501
+ case "$counter" in
502
+ ''|*[!0-9]*) counter=0 ;;
503
+ esac
504
+ counter=$((counter + 1))
505
+ if [ "$counter" -ge "$SIGNAL_EVERY_N" ]; then
506
+ should_signal=1
507
+ counter=0
508
+ fi
509
+ echo "$counter" > "$SIGNAL_COUNTER_FILE"
510
+ else
511
+ echo "1" > "$SIGNAL_COUNTER_FILE"
512
+ fi
513
+ }
514
+
515
+ if command -v flock >/dev/null 2>&1 && exec 8>"$SIGNAL_COUNTER_LOCK" 2>/dev/null; then
516
+ # flock is auto-released when fd 8 closes or the process dies, so there is no
517
+ # stale lock and no lost increment. Use a bounded -w wait so the hook never
518
+ # blocks indefinitely, and only bump the counter while the lock is held -- on
519
+ # a timeout we skip the tick rather than doing an unlocked read-modify-write.
520
+ if flock -w 2 8 2>/dev/null; then
521
+ _ecc_bump_signal_counter
522
+ flock -u 8 2>/dev/null || true
523
+ fi
524
+ exec 8>&- 2>/dev/null || true
525
+ else
526
+ # No flock (e.g. macOS): atomic mkdir lock with a bounded spin so the hook
527
+ # never blocks indefinitely. A trap releases the lock on every exit path --
528
+ # including the async-timeout SIGTERM -- so a killed hook does not strand the
529
+ # directory. We deliberately do NOT hand-roll PID-based stale reclaim:
530
+ # re-verifying then removing another process's lock is racy and can delete a
531
+ # live re-acquirer's directory, reintroducing the very race this fixes.
532
+ _signal_lock_held=0
533
+ _signal_lock_spins=0
534
+ while [ "$_signal_lock_spins" -lt 100 ]; do
535
+ if mkdir "$SIGNAL_COUNTER_LOCK" 2>/dev/null; then
536
+ # EXIT cleans up on normal completion. INT/TERM must release AND exit:
537
+ # a signal trap that only released the lock would otherwise fall through
538
+ # and continue the read-modify-write without ownership.
539
+ trap 'rmdir "$SIGNAL_COUNTER_LOCK" 2>/dev/null || true' EXIT
540
+ trap 'rmdir "$SIGNAL_COUNTER_LOCK" 2>/dev/null || true; exit 130' INT
541
+ trap 'rmdir "$SIGNAL_COUNTER_LOCK" 2>/dev/null || true; exit 143' TERM
542
+ _signal_lock_held=1
543
+ break
544
+ fi
545
+ _signal_lock_spins=$((_signal_lock_spins + 1))
546
+ sleep 0.02
547
+ done
548
+ if [ "$_signal_lock_held" -eq 1 ]; then
549
+ # Bump only under the held lock -- never an unlocked read-modify-write.
550
+ _ecc_bump_signal_counter
551
+ rmdir "$SIGNAL_COUNTER_LOCK" 2>/dev/null || true
552
+ trap - EXIT INT TERM
553
+ fi
554
+ # If the lock could not be acquired within the spin budget we skip this tick
555
+ # rather than racing on an unlocked counter. Dropping one throttle tick under
556
+ # extreme contention only delays the next observer signal slightly; it never
557
+ # corrupts the counter or signals spuriously.
558
+ fi
559
+
560
+ # Signal observer if running and throttle allows (check both project-scoped and global observer, deduplicate)
561
+ if [ "$should_signal" -eq 1 ]; then
562
+ signaled_pids=" "
563
+ for pid_file in "${PROJECT_DIR}/.observer.pid" "${CONFIG_DIR}/.observer.pid"; do
564
+ if [ -f "$pid_file" ]; then
565
+ observer_pid=$(cat "$pid_file" 2>/dev/null || true)
566
+ # Validate PID is a positive integer (>1)
567
+ case "$observer_pid" in
568
+ ''|*[!0-9]*|0|1)
569
+ _REMOVE_FILE_IF_PRESENT "$pid_file"
570
+ continue
571
+ ;;
572
+ esac
573
+ # Deduplicate: skip if already signaled this pass
574
+ case "$signaled_pids" in
575
+ *" $observer_pid "*) continue ;;
576
+ esac
577
+ if kill -0 "$observer_pid" 2>/dev/null; then
578
+ kill -USR1 "$observer_pid" 2>/dev/null || true
579
+ signaled_pids="${signaled_pids}${observer_pid} "
580
+ fi
581
+ fi
582
+ done
583
+ fi
584
+
585
+ exit 0