@rubix0270/arboris 1.0.2 → 1.0.3

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/package.json +8 -20
  2. package/run.mjs +10 -0
  3. package/dist/cli.mjs +0 -383
  4. package/manifest.json +0 -323
  5. package/prisma/skills/accessibility/SKILL.md +0 -147
  6. package/prisma/skills/agent-architecture-audit/SKILL.md +0 -257
  7. package/prisma/skills/agent-eval/SKILL.md +0 -146
  8. package/prisma/skills/agent-harness-construction/SKILL.md +0 -74
  9. package/prisma/skills/agent-introspection-debugging/SKILL.md +0 -154
  10. package/prisma/skills/agent-payment-x402/SKILL.md +0 -225
  11. package/prisma/skills/agent-self-evaluation/SKILL.md +0 -182
  12. package/prisma/skills/agent-self-evaluation/examples/high-score-example.md +0 -87
  13. package/prisma/skills/agent-self-evaluation/examples/low-score-example.md +0 -86
  14. package/prisma/skills/agent-self-evaluation/references/evaluation-criteria.md +0 -71
  15. package/prisma/skills/agent-self-evaluation/references/hook-integration.md +0 -64
  16. package/prisma/skills/agent-self-evaluation/scripts/evaluate.py +0 -408
  17. package/prisma/skills/agent-self-evaluation/templates/evaluation-report.md +0 -86
  18. package/prisma/skills/agent-sort/SKILL.md +0 -216
  19. package/prisma/skills/agentic-engineering/SKILL.md +0 -64
  20. package/prisma/skills/agentic-os/SKILL.md +0 -388
  21. package/prisma/skills/ai-first-engineering/SKILL.md +0 -52
  22. package/prisma/skills/ai-regression-testing/SKILL.md +0 -386
  23. package/prisma/skills/android-clean-architecture/SKILL.md +0 -340
  24. package/prisma/skills/angular-developer/SKILL.md +0 -155
  25. package/prisma/skills/angular-developer/references/angular-animations.md +0 -160
  26. package/prisma/skills/angular-developer/references/angular-aria.md +0 -410
  27. package/prisma/skills/angular-developer/references/cli.md +0 -86
  28. package/prisma/skills/angular-developer/references/component-harnesses.md +0 -59
  29. package/prisma/skills/angular-developer/references/component-styling.md +0 -91
  30. package/prisma/skills/angular-developer/references/components.md +0 -117
  31. package/prisma/skills/angular-developer/references/creating-services.md +0 -97
  32. package/prisma/skills/angular-developer/references/data-resolvers.md +0 -69
  33. package/prisma/skills/angular-developer/references/define-routes.md +0 -67
  34. package/prisma/skills/angular-developer/references/defining-providers.md +0 -72
  35. package/prisma/skills/angular-developer/references/di-fundamentals.md +0 -120
  36. package/prisma/skills/angular-developer/references/e2e-testing.md +0 -56
  37. package/prisma/skills/angular-developer/references/effects.md +0 -83
  38. package/prisma/skills/angular-developer/references/hierarchical-injectors.md +0 -43
  39. package/prisma/skills/angular-developer/references/host-elements.md +0 -80
  40. package/prisma/skills/angular-developer/references/injection-context.md +0 -63
  41. package/prisma/skills/angular-developer/references/inputs.md +0 -101
  42. package/prisma/skills/angular-developer/references/linked-signal.md +0 -59
  43. package/prisma/skills/angular-developer/references/loading-strategies.md +0 -61
  44. package/prisma/skills/angular-developer/references/mcp.md +0 -108
  45. package/prisma/skills/angular-developer/references/navigate-to-routes.md +0 -69
  46. package/prisma/skills/angular-developer/references/outputs.md +0 -86
  47. package/prisma/skills/angular-developer/references/reactive-forms.md +0 -122
  48. package/prisma/skills/angular-developer/references/rendering-strategies.md +0 -44
  49. package/prisma/skills/angular-developer/references/resource.md +0 -77
  50. package/prisma/skills/angular-developer/references/route-animations.md +0 -56
  51. package/prisma/skills/angular-developer/references/route-guards.md +0 -52
  52. package/prisma/skills/angular-developer/references/router-lifecycle.md +0 -45
  53. package/prisma/skills/angular-developer/references/router-testing.md +0 -87
  54. package/prisma/skills/angular-developer/references/show-routes-with-outlets.md +0 -68
  55. package/prisma/skills/angular-developer/references/signal-forms.md +0 -795
  56. package/prisma/skills/angular-developer/references/signals-overview.md +0 -94
  57. package/prisma/skills/angular-developer/references/tailwind-css.md +0 -69
  58. package/prisma/skills/angular-developer/references/template-driven-forms.md +0 -114
  59. package/prisma/skills/angular-developer/references/testing-fundamentals.md +0 -65
  60. package/prisma/skills/api-connector-builder/SKILL.md +0 -121
  61. package/prisma/skills/api-design/SKILL.md +0 -524
  62. package/prisma/skills/architecture-decision-records/SKILL.md +0 -180
  63. package/prisma/skills/article-writing/SKILL.md +0 -80
  64. package/prisma/skills/automation-audit-ops/SKILL.md +0 -143
  65. package/prisma/skills/autonomous-agent-harness/SKILL.md +0 -274
  66. package/prisma/skills/autonomous-loops/SKILL.md +0 -611
  67. package/prisma/skills/backend-patterns/SKILL.md +0 -562
  68. package/prisma/skills/benchmark/SKILL.md +0 -94
  69. package/prisma/skills/benchmark-methodology/SKILL.md +0 -190
  70. package/prisma/skills/benchmark-optimization-loop/SKILL.md +0 -70
  71. package/prisma/skills/blender-motion-state-inspection/SKILL.md +0 -165
  72. package/prisma/skills/blueprint/SKILL.md +0 -106
  73. package/prisma/skills/brand-discovery/SKILL.md +0 -145
  74. package/prisma/skills/brand-discovery/references/10_purpose-why.md +0 -40
  75. package/prisma/skills/brand-discovery/references/20_positioning.md +0 -44
  76. package/prisma/skills/brand-discovery/references/30_audience-niche.md +0 -52
  77. package/prisma/skills/brand-discovery/references/40_personality-archetype.md +0 -57
  78. package/prisma/skills/brand-discovery/references/50_voice-tone.md +0 -59
  79. package/prisma/skills/brand-discovery/references/60_narrative-story.md +0 -50
  80. package/prisma/skills/brand-discovery/references/70_founder-tension.md +0 -49
  81. package/prisma/skills/brand-discovery/references/90_SYNTHESIS.md +0 -133
  82. package/prisma/skills/brand-voice/SKILL.md +0 -98
  83. package/prisma/skills/brand-voice/references/voice-profile-schema.md +0 -55
  84. package/prisma/skills/browser-qa/SKILL.md +0 -105
  85. package/prisma/skills/bun-runtime/SKILL.md +0 -85
  86. package/prisma/skills/canary-watch/SKILL.md +0 -108
  87. package/prisma/skills/carrier-relationship-management/SKILL.md +0 -212
  88. package/prisma/skills/cisco-ios-patterns/SKILL.md +0 -164
  89. package/prisma/skills/ck/SKILL.md +0 -148
  90. package/prisma/skills/ck/commands/forget.mjs +0 -44
  91. package/prisma/skills/ck/commands/info.mjs +0 -24
  92. package/prisma/skills/ck/commands/init.mjs +0 -143
  93. package/prisma/skills/ck/commands/list.mjs +0 -40
  94. package/prisma/skills/ck/commands/migrate.mjs +0 -202
  95. package/prisma/skills/ck/commands/resume.mjs +0 -36
  96. package/prisma/skills/ck/commands/save.mjs +0 -210
  97. package/prisma/skills/ck/commands/shared.mjs +0 -387
  98. package/prisma/skills/ck/hooks/session-start.mjs +0 -224
  99. package/prisma/skills/claude-devfleet/SKILL.md +0 -112
  100. package/prisma/skills/click-path-audit/SKILL.md +0 -245
  101. package/prisma/skills/clickhouse-io/SKILL.md +0 -440
  102. package/prisma/skills/code-tour/SKILL.md +0 -254
  103. package/prisma/skills/codebase-onboarding/SKILL.md +0 -234
  104. package/prisma/skills/codehealth-mcp/SKILL.md +0 -167
  105. package/prisma/skills/coding-standards/SKILL.md +0 -551
  106. package/prisma/skills/competitive-platform-analysis/SKILL.md +0 -214
  107. package/prisma/skills/competitive-report-structure/SKILL.md +0 -162
  108. package/prisma/skills/compose-multiplatform-patterns/SKILL.md +0 -300
  109. package/prisma/skills/config-gc/SKILL.md +0 -120
  110. package/prisma/skills/configure-ecc/SKILL.md +0 -385
  111. package/prisma/skills/connections-optimizer/SKILL.md +0 -190
  112. package/prisma/skills/content-engine/SKILL.md +0 -132
  113. package/prisma/skills/content-hash-cache-pattern/SKILL.md +0 -162
  114. package/prisma/skills/context-budget/SKILL.md +0 -136
  115. package/prisma/skills/continuous-agent-loop/SKILL.md +0 -46
  116. package/prisma/skills/continuous-learning/SKILL.md +0 -132
  117. package/prisma/skills/continuous-learning/config.json +0 -18
  118. package/prisma/skills/continuous-learning/evaluate-session.sh +0 -69
  119. package/prisma/skills/continuous-learning-v2/SKILL.md +0 -361
  120. package/prisma/skills/continuous-learning-v2/agents/observer-loop.sh +0 -359
  121. package/prisma/skills/continuous-learning-v2/agents/observer.md +0 -189
  122. package/prisma/skills/continuous-learning-v2/agents/session-guardian.sh +0 -150
  123. package/prisma/skills/continuous-learning-v2/agents/start-observer.sh +0 -248
  124. package/prisma/skills/continuous-learning-v2/config.json +0 -8
  125. package/prisma/skills/continuous-learning-v2/hooks/observe.sh +0 -585
  126. package/prisma/skills/continuous-learning-v2/scripts/detect-project.sh +0 -322
  127. package/prisma/skills/continuous-learning-v2/scripts/instinct-cli.py +0 -1956
  128. package/prisma/skills/continuous-learning-v2/scripts/lib/homunculus-dir.sh +0 -31
  129. package/prisma/skills/continuous-learning-v2/scripts/migrate-homunculus.sh +0 -68
  130. package/prisma/skills/continuous-learning-v2/scripts/test_parse_instinct.py +0 -1421
  131. package/prisma/skills/cost-aware-llm-pipeline/SKILL.md +0 -184
  132. package/prisma/skills/cost-tracking/SKILL.md +0 -97
  133. package/prisma/skills/council/SKILL.md +0 -204
  134. package/prisma/skills/cpp-coding-standards/SKILL.md +0 -724
  135. package/prisma/skills/cpp-testing/SKILL.md +0 -325
  136. package/prisma/skills/crosspost/SKILL.md +0 -112
  137. package/prisma/skills/csharp-testing/SKILL.md +0 -322
  138. package/prisma/skills/customer-billing-ops/SKILL.md +0 -141
  139. package/prisma/skills/customs-trade-compliance/SKILL.md +0 -263
  140. package/prisma/skills/dart-flutter-patterns/SKILL.md +0 -564
  141. package/prisma/skills/dashboard-builder/SKILL.md +0 -109
  142. package/prisma/skills/data-scraper-agent/SKILL.md +0 -765
  143. package/prisma/skills/data-throughput-accelerator/SKILL.md +0 -73
  144. package/prisma/skills/database-migrations/SKILL.md +0 -430
  145. package/prisma/skills/deep-research/SKILL.md +0 -160
  146. package/prisma/skills/defi-amm-security/SKILL.md +0 -167
  147. package/prisma/skills/delivery-gate/SKILL.md +0 -126
  148. package/prisma/skills/delivery-gate/hooks/quality-gate.py +0 -220
  149. package/prisma/skills/deployment-patterns/SKILL.md +0 -428
  150. package/prisma/skills/design-system/SKILL.md +0 -83
  151. package/prisma/skills/django-celery/SKILL.md +0 -458
  152. package/prisma/skills/django-patterns/SKILL.md +0 -735
  153. package/prisma/skills/django-security/SKILL.md +0 -644
  154. package/prisma/skills/django-tdd/SKILL.md +0 -730
  155. package/prisma/skills/django-verification/SKILL.md +0 -470
  156. package/prisma/skills/dmux-workflows/SKILL.md +0 -192
  157. package/prisma/skills/docker-patterns/SKILL.md +0 -365
  158. package/prisma/skills/documentation-lookup/SKILL.md +0 -91
  159. package/prisma/skills/dotnet-patterns/SKILL.md +0 -322
  160. package/prisma/skills/dynamic-workflow-mode/SKILL.md +0 -124
  161. package/prisma/skills/e2e-testing/SKILL.md +0 -327
  162. package/prisma/skills/ecc-guide/SKILL.md +0 -190
  163. package/prisma/skills/ecc-recipes/SKILL.md +0 -149
  164. package/prisma/skills/ecc-tools-cost-audit/SKILL.md +0 -161
  165. package/prisma/skills/email-ops/SKILL.md +0 -122
  166. package/prisma/skills/energy-procurement/SKILL.md +0 -228
  167. package/prisma/skills/enterprise-agent-ops/SKILL.md +0 -51
  168. package/prisma/skills/error-handling/SKILL.md +0 -377
  169. package/prisma/skills/eval-harness/SKILL.md +0 -271
  170. package/prisma/skills/evm-token-decimals/SKILL.md +0 -131
  171. package/prisma/skills/exa-search/SKILL.md +0 -108
  172. package/prisma/skills/fal-ai-media/SKILL.md +0 -289
  173. package/prisma/skills/fastapi-patterns/SKILL.md +0 -514
  174. package/prisma/skills/finance-billing-ops/SKILL.md +0 -128
  175. package/prisma/skills/flox-environments/SKILL.md +0 -497
  176. package/prisma/skills/flutter-dart-code-review/SKILL.md +0 -436
  177. package/prisma/skills/foundation-models-on-device/SKILL.md +0 -243
  178. package/prisma/skills/frontend-a11y/SKILL.md +0 -446
  179. package/prisma/skills/frontend-design-direction/SKILL.md +0 -93
  180. package/prisma/skills/frontend-patterns/SKILL.md +0 -657
  181. package/prisma/skills/frontend-slides/SKILL.md +0 -185
  182. package/prisma/skills/frontend-slides/STYLE_PRESETS.md +0 -330
  183. package/prisma/skills/frontend-slides/animation-patterns.md +0 -122
  184. package/prisma/skills/frontend-slides/html-template.md +0 -419
  185. package/prisma/skills/frontend-slides/scripts/export-pdf.sh +0 -418
  186. package/prisma/skills/frontend-slides/scripts/extract-pptx.py +0 -96
  187. package/prisma/skills/frontend-slides/viewport-base.css +0 -153
  188. package/prisma/skills/fsharp-testing/SKILL.md +0 -281
  189. package/prisma/skills/gan-style-harness/SKILL.md +0 -279
  190. package/prisma/skills/gateguard/SKILL.md +0 -133
  191. package/prisma/skills/generating-python-installer/SKILL.md +0 -820
  192. package/prisma/skills/git-workflow/SKILL.md +0 -716
  193. package/prisma/skills/github-ops/SKILL.md +0 -145
  194. package/prisma/skills/golang-patterns/SKILL.md +0 -675
  195. package/prisma/skills/golang-testing/SKILL.md +0 -721
  196. package/prisma/skills/google-workspace-ops/SKILL.md +0 -96
  197. package/prisma/skills/growth-log/SKILL.md +0 -128
  198. package/prisma/skills/healthcare-cdss-patterns/SKILL.md +0 -246
  199. package/prisma/skills/healthcare-emr-patterns/SKILL.md +0 -160
  200. package/prisma/skills/healthcare-eval-harness/SKILL.md +0 -208
  201. package/prisma/skills/healthcare-phi-compliance/SKILL.md +0 -146
  202. package/prisma/skills/hermes-imports/SKILL.md +0 -89
  203. package/prisma/skills/hexagonal-architecture/SKILL.md +0 -277
  204. package/prisma/skills/hipaa-compliance/SKILL.md +0 -79
  205. package/prisma/skills/homelab-network-readiness/SKILL.md +0 -170
  206. package/prisma/skills/homelab-network-setup/SKILL.md +0 -130
  207. package/prisma/skills/homelab-pihole-dns/SKILL.md +0 -275
  208. package/prisma/skills/homelab-vlan-segmentation/SKILL.md +0 -312
  209. package/prisma/skills/homelab-wireguard-vpn/SKILL.md +0 -306
  210. package/prisma/skills/hookify-rules/SKILL.md +0 -128
  211. package/prisma/skills/inherit-legacy-style/SKILL.md +0 -157
  212. package/prisma/skills/intent-driven-development/SKILL.md +0 -360
  213. package/prisma/skills/inventory-demand-planning/SKILL.md +0 -247
  214. package/prisma/skills/investor-materials/SKILL.md +0 -97
  215. package/prisma/skills/investor-outreach/SKILL.md +0 -92
  216. package/prisma/skills/ios-icon-gen/SKILL.md +0 -158
  217. package/prisma/skills/ios-icon-gen/scripts/generate_icons.swift +0 -258
  218. package/prisma/skills/ios-icon-gen/scripts/iconify_gen.sh +0 -235
  219. package/prisma/skills/iterative-retrieval/SKILL.md +0 -212
  220. package/prisma/skills/ito-basket-compare/SKILL.md +0 -64
  221. package/prisma/skills/ito-data-atlas-agent/SKILL.md +0 -64
  222. package/prisma/skills/ito-market-intelligence/SKILL.md +0 -61
  223. package/prisma/skills/ito-trade-planner/SKILL.md +0 -68
  224. package/prisma/skills/java-coding-standards/SKILL.md +0 -384
  225. package/prisma/skills/jira-integration/SKILL.md +0 -303
  226. package/prisma/skills/jpa-patterns/SKILL.md +0 -152
  227. package/prisma/skills/knowledge-ops/SKILL.md +0 -155
  228. package/prisma/skills/kotlin-coroutines-flows/SKILL.md +0 -285
  229. package/prisma/skills/kotlin-exposed-patterns/SKILL.md +0 -720
  230. package/prisma/skills/kotlin-ktor-patterns/SKILL.md +0 -690
  231. package/prisma/skills/kotlin-patterns/SKILL.md +0 -712
  232. package/prisma/skills/kotlin-testing/SKILL.md +0 -825
  233. package/prisma/skills/kubernetes-patterns/SKILL.md +0 -756
  234. package/prisma/skills/laravel-patterns/SKILL.md +0 -416
  235. package/prisma/skills/laravel-plugin-discovery/SKILL.md +0 -230
  236. package/prisma/skills/laravel-security/SKILL.md +0 -948
  237. package/prisma/skills/laravel-tdd/SKILL.md +0 -675
  238. package/prisma/skills/laravel-verification/SKILL.md +0 -180
  239. package/prisma/skills/latency-critical-systems/SKILL.md +0 -74
  240. package/prisma/skills/lead-intelligence/SKILL.md +0 -322
  241. package/prisma/skills/lead-intelligence/agents/enrichment-agent.md +0 -85
  242. package/prisma/skills/lead-intelligence/agents/mutual-mapper.md +0 -75
  243. package/prisma/skills/lead-intelligence/agents/outreach-drafter.md +0 -98
  244. package/prisma/skills/lead-intelligence/agents/signal-scorer.md +0 -60
  245. package/prisma/skills/liquid-glass-design/SKILL.md +0 -279
  246. package/prisma/skills/llm-trading-agent-security/SKILL.md +0 -147
  247. package/prisma/skills/logistics-exception-management/SKILL.md +0 -222
  248. package/prisma/skills/loop-design-check/SKILL.md +0 -143
  249. package/prisma/skills/mailtrap-email-integration/SKILL.md +0 -77
  250. package/prisma/skills/make-interfaces-feel-better/SKILL.md +0 -152
  251. package/prisma/skills/manim-video/SKILL.md +0 -90
  252. package/prisma/skills/manim-video/assets/network_graph_scene.py +0 -52
  253. package/prisma/skills/market-research/SKILL.md +0 -76
  254. package/prisma/skills/marketing-campaign/SKILL.md +0 -114
  255. package/prisma/skills/mcp-server-patterns/SKILL.md +0 -70
  256. package/prisma/skills/messages-ops/SKILL.md +0 -105
  257. package/prisma/skills/ml-adoption-playbook/SKILL.md +0 -57
  258. package/prisma/skills/mle-workflow/SKILL.md +0 -347
  259. package/prisma/skills/motion-advanced/SKILL.md +0 -596
  260. package/prisma/skills/motion-foundations/SKILL.md +0 -299
  261. package/prisma/skills/motion-patterns/SKILL.md +0 -434
  262. package/prisma/skills/motion-ui/SKILL.md +0 -576
  263. package/prisma/skills/mysql-patterns/SKILL.md +0 -413
  264. package/prisma/skills/nanoclaw-repl/SKILL.md +0 -34
  265. package/prisma/skills/nestjs-patterns/SKILL.md +0 -231
  266. package/prisma/skills/netmiko-ssh-automation/SKILL.md +0 -174
  267. package/prisma/skills/network-bgp-diagnostics/SKILL.md +0 -168
  268. package/prisma/skills/network-config-validation/SKILL.md +0 -211
  269. package/prisma/skills/network-interface-health/SKILL.md +0 -153
  270. package/prisma/skills/nextjs-turbopack/SKILL.md +0 -58
  271. package/prisma/skills/nodejs-keccak256/SKILL.md +0 -103
  272. package/prisma/skills/nutrient-document-processing/SKILL.md +0 -168
  273. package/prisma/skills/nuxt4-patterns/SKILL.md +0 -101
  274. package/prisma/skills/openclaw-persona-forge/SKILL.md +0 -289
  275. package/prisma/skills/openclaw-persona-forge/gacha.py +0 -224
  276. package/prisma/skills/openclaw-persona-forge/gacha.sh +0 -5
  277. package/prisma/skills/openclaw-persona-forge/references/avatar-style.md +0 -124
  278. package/prisma/skills/openclaw-persona-forge/references/boundary-rules.md +0 -53
  279. package/prisma/skills/openclaw-persona-forge/references/error-handling.md +0 -53
  280. package/prisma/skills/openclaw-persona-forge/references/identity-tension.md +0 -48
  281. package/prisma/skills/openclaw-persona-forge/references/naming-system.md +0 -39
  282. package/prisma/skills/openclaw-persona-forge/references/output-template.md +0 -166
  283. package/prisma/skills/opensource-pipeline/SKILL.md +0 -256
  284. package/prisma/skills/orch-add-feature/SKILL.md +0 -45
  285. package/prisma/skills/orch-build-mvp/SKILL.md +0 -49
  286. package/prisma/skills/orch-change-feature/SKILL.md +0 -43
  287. package/prisma/skills/orch-fix-defect/SKILL.md +0 -43
  288. package/prisma/skills/orch-pipeline/SKILL.md +0 -121
  289. package/prisma/skills/orch-refine-code/SKILL.md +0 -44
  290. package/prisma/skills/parallel-execution-optimizer/SKILL.md +0 -73
  291. package/prisma/skills/perl-patterns/SKILL.md +0 -505
  292. package/prisma/skills/perl-security/SKILL.md +0 -504
  293. package/prisma/skills/perl-testing/SKILL.md +0 -476
  294. package/prisma/skills/plan-orchestrate/SKILL.md +0 -263
  295. package/prisma/skills/plankton-code-quality/SKILL.md +0 -237
  296. package/prisma/skills/postgres-patterns/SKILL.md +0 -148
  297. package/prisma/skills/prediction-market-oracle-research/SKILL.md +0 -64
  298. package/prisma/skills/prediction-market-risk-review/SKILL.md +0 -61
  299. package/prisma/skills/prisma-patterns/SKILL.md +0 -401
  300. package/prisma/skills/product-capability/SKILL.md +0 -142
  301. package/prisma/skills/product-lens/SKILL.md +0 -93
  302. package/prisma/skills/production-audit/SKILL.md +0 -207
  303. package/prisma/skills/production-scheduling/SKILL.md +0 -238
  304. package/prisma/skills/project-flow-ops/SKILL.md +0 -112
  305. package/prisma/skills/prompt-optimizer/SKILL.md +0 -398
  306. package/prisma/skills/python-patterns/SKILL.md +0 -751
  307. package/prisma/skills/python-testing/SKILL.md +0 -817
  308. package/prisma/skills/pytorch-patterns/SKILL.md +0 -397
  309. package/prisma/skills/quality-nonconformance/SKILL.md +0 -260
  310. package/prisma/skills/quarkus-patterns/SKILL.md +0 -723
  311. package/prisma/skills/quarkus-security/SKILL.md +0 -468
  312. package/prisma/skills/quarkus-tdd/SKILL.md +0 -812
  313. package/prisma/skills/quarkus-verification/SKILL.md +0 -480
  314. package/prisma/skills/ralphinho-rfc-pipeline/SKILL.md +0 -68
  315. package/prisma/skills/react-native-patterns/SKILL.md +0 -326
  316. package/prisma/skills/react-patterns/SKILL.md +0 -342
  317. package/prisma/skills/react-performance/SKILL.md +0 -575
  318. package/prisma/skills/react-testing/SKILL.md +0 -424
  319. package/prisma/skills/recsys-pipeline-architect/SKILL.md +0 -115
  320. package/prisma/skills/recursive-decision-ledger/SKILL.md +0 -80
  321. package/prisma/skills/redis-patterns/SKILL.md +0 -404
  322. package/prisma/skills/regex-vs-llm-structured-text/SKILL.md +0 -221
  323. package/prisma/skills/remotion-video-creation/SKILL.md +0 -43
  324. package/prisma/skills/remotion-video-creation/rules/3d.md +0 -86
  325. package/prisma/skills/remotion-video-creation/rules/animations.md +0 -29
  326. package/prisma/skills/remotion-video-creation/rules/assets/charts-bar-chart.tsx +0 -173
  327. package/prisma/skills/remotion-video-creation/rules/assets/text-animations-typewriter.tsx +0 -100
  328. package/prisma/skills/remotion-video-creation/rules/assets/text-animations-word-highlight.tsx +0 -108
  329. package/prisma/skills/remotion-video-creation/rules/assets.md +0 -78
  330. package/prisma/skills/remotion-video-creation/rules/audio.md +0 -172
  331. package/prisma/skills/remotion-video-creation/rules/calculate-metadata.md +0 -104
  332. package/prisma/skills/remotion-video-creation/rules/can-decode.md +0 -75
  333. package/prisma/skills/remotion-video-creation/rules/charts.md +0 -58
  334. package/prisma/skills/remotion-video-creation/rules/compositions.md +0 -146
  335. package/prisma/skills/remotion-video-creation/rules/display-captions.md +0 -126
  336. package/prisma/skills/remotion-video-creation/rules/extract-frames.md +0 -229
  337. package/prisma/skills/remotion-video-creation/rules/fonts.md +0 -152
  338. package/prisma/skills/remotion-video-creation/rules/get-audio-duration.md +0 -58
  339. package/prisma/skills/remotion-video-creation/rules/get-video-dimensions.md +0 -68
  340. package/prisma/skills/remotion-video-creation/rules/get-video-duration.md +0 -58
  341. package/prisma/skills/remotion-video-creation/rules/gifs.md +0 -138
  342. package/prisma/skills/remotion-video-creation/rules/images.md +0 -130
  343. package/prisma/skills/remotion-video-creation/rules/import-srt-captions.md +0 -67
  344. package/prisma/skills/remotion-video-creation/rules/lottie.md +0 -67
  345. package/prisma/skills/remotion-video-creation/rules/measuring-dom-nodes.md +0 -34
  346. package/prisma/skills/remotion-video-creation/rules/measuring-text.md +0 -143
  347. package/prisma/skills/remotion-video-creation/rules/sequencing.md +0 -106
  348. package/prisma/skills/remotion-video-creation/rules/tailwind.md +0 -11
  349. package/prisma/skills/remotion-video-creation/rules/text-animations.md +0 -20
  350. package/prisma/skills/remotion-video-creation/rules/timing.md +0 -179
  351. package/prisma/skills/remotion-video-creation/rules/transcribe-captions.md +0 -19
  352. package/prisma/skills/remotion-video-creation/rules/transitions.md +0 -122
  353. package/prisma/skills/remotion-video-creation/rules/trimming.md +0 -52
  354. package/prisma/skills/remotion-video-creation/rules/videos.md +0 -171
  355. package/prisma/skills/repo-scan/SKILL.md +0 -79
  356. package/prisma/skills/research-ops/SKILL.md +0 -113
  357. package/prisma/skills/returns-reverse-logistics/SKILL.md +0 -240
  358. package/prisma/skills/rules-distill/SKILL.md +0 -265
  359. package/prisma/skills/rules-distill/scripts/scan-rules.sh +0 -58
  360. package/prisma/skills/rules-distill/scripts/scan-skills.sh +0 -129
  361. package/prisma/skills/rust-patterns/SKILL.md +0 -500
  362. package/prisma/skills/rust-testing/SKILL.md +0 -501
  363. package/prisma/skills/safety-guard/SKILL.md +0 -76
  364. package/prisma/skills/santa-method/SKILL.md +0 -307
  365. package/prisma/skills/scientific-db-pubmed-database/SKILL.md +0 -176
  366. package/prisma/skills/scientific-db-uspto-database/SKILL.md +0 -178
  367. package/prisma/skills/scientific-pkg-gget/SKILL.md +0 -167
  368. package/prisma/skills/scientific-thinking-literature-review/SKILL.md +0 -193
  369. package/prisma/skills/scientific-thinking-scholar-evaluation/SKILL.md +0 -161
  370. package/prisma/skills/search-first/SKILL.md +0 -183
  371. package/prisma/skills/security-bounty-hunter/SKILL.md +0 -100
  372. package/prisma/skills/security-review/SKILL.md +0 -504
  373. package/prisma/skills/security-review/cloud-infrastructure-security.md +0 -361
  374. package/prisma/skills/security-scan/SKILL.md +0 -166
  375. package/prisma/skills/seo/SKILL.md +0 -155
  376. package/prisma/skills/skill-comply/SKILL.md +0 -59
  377. package/prisma/skills/skill-comply/fixtures/compliant_trace.jsonl +0 -5
  378. package/prisma/skills/skill-comply/fixtures/noncompliant_trace.jsonl +0 -3
  379. package/prisma/skills/skill-comply/fixtures/tdd_spec.yaml +0 -44
  380. package/prisma/skills/skill-comply/prompts/classifier.md +0 -24
  381. package/prisma/skills/skill-comply/prompts/scenario_generator.md +0 -62
  382. package/prisma/skills/skill-comply/prompts/spec_generator.md +0 -42
  383. package/prisma/skills/skill-comply/pyproject.toml +0 -15
  384. package/prisma/skills/skill-comply/scripts/__init__.py +0 -0
  385. package/prisma/skills/skill-comply/scripts/classifier.py +0 -85
  386. package/prisma/skills/skill-comply/scripts/grader.py +0 -124
  387. package/prisma/skills/skill-comply/scripts/parser.py +0 -107
  388. package/prisma/skills/skill-comply/scripts/report.py +0 -170
  389. package/prisma/skills/skill-comply/scripts/run.py +0 -127
  390. package/prisma/skills/skill-comply/scripts/runner.py +0 -194
  391. package/prisma/skills/skill-comply/scripts/scenario_generator.py +0 -70
  392. package/prisma/skills/skill-comply/scripts/spec_generator.py +0 -72
  393. package/prisma/skills/skill-comply/scripts/utils.py +0 -13
  394. package/prisma/skills/skill-comply/tests/test_grader.py +0 -197
  395. package/prisma/skills/skill-comply/tests/test_parser.py +0 -90
  396. package/prisma/skills/skill-comply/tests/test_runner.py +0 -172
  397. package/prisma/skills/skill-scout/SKILL.md +0 -141
  398. package/prisma/skills/skill-stocktake/SKILL.md +0 -195
  399. package/prisma/skills/skill-stocktake/scripts/quick-diff.sh +0 -87
  400. package/prisma/skills/skill-stocktake/scripts/save-results.sh +0 -56
  401. package/prisma/skills/skill-stocktake/scripts/scan.sh +0 -170
  402. package/prisma/skills/social-graph-ranker/SKILL.md +0 -155
  403. package/prisma/skills/social-publisher/SKILL.md +0 -130
  404. package/prisma/skills/springboot-patterns/SKILL.md +0 -315
  405. package/prisma/skills/springboot-security/SKILL.md +0 -273
  406. package/prisma/skills/springboot-tdd/SKILL.md +0 -159
  407. package/prisma/skills/springboot-verification/SKILL.md +0 -232
  408. package/prisma/skills/strategic-compact/SKILL.md +0 -136
  409. package/prisma/skills/swift-actor-persistence/SKILL.md +0 -144
  410. package/prisma/skills/swift-concurrency-6-2/SKILL.md +0 -216
  411. package/prisma/skills/swift-protocol-di-testing/SKILL.md +0 -191
  412. package/prisma/skills/swiftui-patterns/SKILL.md +0 -259
  413. package/prisma/skills/taste/SKILL.md +0 -264
  414. package/prisma/skills/taste/references/genre-taxonomy.md +0 -87
  415. package/prisma/skills/tdd-workflow/SKILL.md +0 -583
  416. package/prisma/skills/team-agent-orchestration/SKILL.md +0 -111
  417. package/prisma/skills/team-builder/SKILL.md +0 -169
  418. package/prisma/skills/terminal-ops/SKILL.md +0 -110
  419. package/prisma/skills/tinystruct-patterns/SKILL.md +0 -279
  420. package/prisma/skills/tinystruct-patterns/references/architecture.md +0 -90
  421. package/prisma/skills/tinystruct-patterns/references/data-handling.md +0 -60
  422. package/prisma/skills/tinystruct-patterns/references/database.md +0 -99
  423. package/prisma/skills/tinystruct-patterns/references/routing.md +0 -64
  424. package/prisma/skills/tinystruct-patterns/references/system-usage.md +0 -97
  425. package/prisma/skills/tinystruct-patterns/references/testing.md +0 -72
  426. package/prisma/skills/token-budget-advisor/SKILL.md +0 -134
  427. package/prisma/skills/ui-demo/SKILL.md +0 -466
  428. package/prisma/skills/ui-to-vue/SKILL.md +0 -135
  429. package/prisma/skills/uncloud/SKILL.md +0 -344
  430. package/prisma/skills/unified-notifications-ops/SKILL.md +0 -188
  431. package/prisma/skills/verification-loop/SKILL.md +0 -127
  432. package/prisma/skills/video-editing/SKILL.md +0 -311
  433. package/prisma/skills/videodb/SKILL.md +0 -375
  434. package/prisma/skills/videodb/reference/api-reference.md +0 -550
  435. package/prisma/skills/videodb/reference/capture-reference.md +0 -407
  436. package/prisma/skills/videodb/reference/capture.md +0 -101
  437. package/prisma/skills/videodb/reference/editor.md +0 -443
  438. package/prisma/skills/videodb/reference/generative.md +0 -331
  439. package/prisma/skills/videodb/reference/rtstream-reference.md +0 -564
  440. package/prisma/skills/videodb/reference/rtstream.md +0 -65
  441. package/prisma/skills/videodb/reference/search.md +0 -230
  442. package/prisma/skills/videodb/reference/streaming.md +0 -406
  443. package/prisma/skills/videodb/reference/use-cases.md +0 -118
  444. package/prisma/skills/videodb/scripts/ws_listener.py +0 -282
  445. package/prisma/skills/visa-doc-translate/README.md +0 -86
  446. package/prisma/skills/visa-doc-translate/SKILL.md +0 -117
  447. package/prisma/skills/vite-patterns/SKILL.md +0 -450
  448. package/prisma/skills/vue-patterns/SKILL.md +0 -471
  449. package/prisma/skills/windows-desktop-e2e/SKILL.md +0 -888
  450. package/prisma/skills/workspace-surface-audit/SKILL.md +0 -126
  451. package/prisma/skills/x-api/SKILL.md +0 -235
@@ -1,585 +0,0 @@
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