agim-cli 1.2.147 → 1.2.148

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 (347) hide show
  1. package/CHANGELOG.md +58 -0
  2. package/dist/core/skills/builtin/ECC_LICENSE +21 -0
  3. package/dist/core/skills/builtin/ECC_NOTICE.md +22 -0
  4. package/dist/core/skills/builtin/accessibility/SKILL.md +146 -0
  5. package/dist/core/skills/builtin/agent-eval/SKILL.md +145 -0
  6. package/dist/core/skills/builtin/agent-harness-construction/SKILL.md +73 -0
  7. package/dist/core/skills/builtin/agent-introspection-debugging/SKILL.md +153 -0
  8. package/dist/core/skills/builtin/agentic-engineering/SKILL.md +63 -0
  9. package/dist/core/skills/builtin/ai-first-engineering/SKILL.md +51 -0
  10. package/dist/core/skills/builtin/ai-regression-testing/SKILL.md +385 -0
  11. package/dist/core/skills/builtin/android-clean-architecture/SKILL.md +339 -0
  12. package/dist/core/skills/builtin/angular-developer/SKILL.md +154 -0
  13. package/dist/core/skills/builtin/angular-developer/references/angular-animations.md +160 -0
  14. package/dist/core/skills/builtin/angular-developer/references/angular-aria.md +410 -0
  15. package/dist/core/skills/builtin/angular-developer/references/cli.md +86 -0
  16. package/dist/core/skills/builtin/angular-developer/references/component-harnesses.md +59 -0
  17. package/dist/core/skills/builtin/angular-developer/references/component-styling.md +91 -0
  18. package/dist/core/skills/builtin/angular-developer/references/components.md +117 -0
  19. package/dist/core/skills/builtin/angular-developer/references/creating-services.md +97 -0
  20. package/dist/core/skills/builtin/angular-developer/references/data-resolvers.md +69 -0
  21. package/dist/core/skills/builtin/angular-developer/references/define-routes.md +67 -0
  22. package/dist/core/skills/builtin/angular-developer/references/defining-providers.md +72 -0
  23. package/dist/core/skills/builtin/angular-developer/references/di-fundamentals.md +120 -0
  24. package/dist/core/skills/builtin/angular-developer/references/e2e-testing.md +56 -0
  25. package/dist/core/skills/builtin/angular-developer/references/effects.md +83 -0
  26. package/dist/core/skills/builtin/angular-developer/references/hierarchical-injectors.md +43 -0
  27. package/dist/core/skills/builtin/angular-developer/references/host-elements.md +80 -0
  28. package/dist/core/skills/builtin/angular-developer/references/injection-context.md +63 -0
  29. package/dist/core/skills/builtin/angular-developer/references/inputs.md +101 -0
  30. package/dist/core/skills/builtin/angular-developer/references/linked-signal.md +59 -0
  31. package/dist/core/skills/builtin/angular-developer/references/loading-strategies.md +61 -0
  32. package/dist/core/skills/builtin/angular-developer/references/mcp.md +108 -0
  33. package/dist/core/skills/builtin/angular-developer/references/navigate-to-routes.md +69 -0
  34. package/dist/core/skills/builtin/angular-developer/references/outputs.md +86 -0
  35. package/dist/core/skills/builtin/angular-developer/references/reactive-forms.md +122 -0
  36. package/dist/core/skills/builtin/angular-developer/references/rendering-strategies.md +44 -0
  37. package/dist/core/skills/builtin/angular-developer/references/resource.md +77 -0
  38. package/dist/core/skills/builtin/angular-developer/references/route-animations.md +56 -0
  39. package/dist/core/skills/builtin/angular-developer/references/route-guards.md +52 -0
  40. package/dist/core/skills/builtin/angular-developer/references/router-lifecycle.md +45 -0
  41. package/dist/core/skills/builtin/angular-developer/references/router-testing.md +87 -0
  42. package/dist/core/skills/builtin/angular-developer/references/show-routes-with-outlets.md +68 -0
  43. package/dist/core/skills/builtin/angular-developer/references/signal-forms.md +795 -0
  44. package/dist/core/skills/builtin/angular-developer/references/signals-overview.md +94 -0
  45. package/dist/core/skills/builtin/angular-developer/references/tailwind-css.md +69 -0
  46. package/dist/core/skills/builtin/angular-developer/references/template-driven-forms.md +114 -0
  47. package/dist/core/skills/builtin/angular-developer/references/testing-fundamentals.md +65 -0
  48. package/dist/core/skills/builtin/api-connector-builder/SKILL.md +120 -0
  49. package/dist/core/skills/builtin/api-design/SKILL.md +523 -0
  50. package/dist/core/skills/builtin/architecture-decision-records/SKILL.md +179 -0
  51. package/dist/core/skills/builtin/article-writing/SKILL.md +79 -0
  52. package/dist/core/skills/builtin/automation-audit-ops/SKILL.md +142 -0
  53. package/dist/core/skills/builtin/autonomous-agent-harness/SKILL.md +273 -0
  54. package/dist/core/skills/builtin/autonomous-loops/SKILL.md +610 -0
  55. package/dist/core/skills/builtin/backend-patterns/SKILL.md +561 -0
  56. package/dist/core/skills/builtin/benchmark/SKILL.md +93 -0
  57. package/dist/core/skills/builtin/benchmark-optimization-loop/SKILL.md +69 -0
  58. package/dist/core/skills/builtin/blueprint/SKILL.md +105 -0
  59. package/dist/core/skills/builtin/browser-qa/SKILL.md +87 -0
  60. package/dist/core/skills/builtin/bun-runtime/SKILL.md +84 -0
  61. package/dist/core/skills/builtin/cisco-ios-patterns/SKILL.md +163 -0
  62. package/dist/core/skills/builtin/claude-devfleet/SKILL.md +111 -0
  63. package/dist/core/skills/builtin/click-path-audit/SKILL.md +244 -0
  64. package/dist/core/skills/builtin/clickhouse-io/SKILL.md +439 -0
  65. package/dist/core/skills/builtin/code-tour/SKILL.md +236 -0
  66. package/dist/core/skills/builtin/codebase-onboarding/SKILL.md +233 -0
  67. package/dist/core/skills/builtin/codehealth-mcp/SKILL.md +166 -0
  68. package/dist/core/skills/builtin/coding-standards/SKILL.md +550 -0
  69. package/dist/core/skills/builtin/compose-multiplatform-patterns/SKILL.md +299 -0
  70. package/dist/core/skills/builtin/config-gc/SKILL.md +119 -0
  71. package/dist/core/skills/builtin/content-hash-cache-pattern/SKILL.md +161 -0
  72. package/dist/core/skills/builtin/context-budget/SKILL.md +135 -0
  73. package/dist/core/skills/builtin/continuous-agent-loop/SKILL.md +45 -0
  74. package/dist/core/skills/builtin/continuous-learning/SKILL.md +131 -0
  75. package/dist/core/skills/builtin/continuous-learning/config.json +18 -0
  76. package/dist/core/skills/builtin/continuous-learning/evaluate-session.sh +69 -0
  77. package/dist/core/skills/builtin/continuous-learning-v2/SKILL.md +360 -0
  78. package/dist/core/skills/builtin/continuous-learning-v2/agents/observer-loop.sh +335 -0
  79. package/dist/core/skills/builtin/continuous-learning-v2/agents/observer.md +198 -0
  80. package/dist/core/skills/builtin/continuous-learning-v2/agents/session-guardian.sh +150 -0
  81. package/dist/core/skills/builtin/continuous-learning-v2/agents/start-observer.sh +248 -0
  82. package/dist/core/skills/builtin/continuous-learning-v2/config.json +8 -0
  83. package/dist/core/skills/builtin/continuous-learning-v2/hooks/observe.sh +498 -0
  84. package/dist/core/skills/builtin/continuous-learning-v2/scripts/detect-project.sh +322 -0
  85. package/dist/core/skills/builtin/continuous-learning-v2/scripts/instinct-cli.py +1914 -0
  86. package/dist/core/skills/builtin/continuous-learning-v2/scripts/lib/homunculus-dir.sh +31 -0
  87. package/dist/core/skills/builtin/continuous-learning-v2/scripts/migrate-homunculus.sh +62 -0
  88. package/dist/core/skills/builtin/continuous-learning-v2/scripts/test_parse_instinct.py +1045 -0
  89. package/dist/core/skills/builtin/cost-aware-llm-pipeline/SKILL.md +183 -0
  90. package/dist/core/skills/builtin/cost-tracking/SKILL.md +147 -0
  91. package/dist/core/skills/builtin/council/SKILL.md +203 -0
  92. package/dist/core/skills/builtin/cpp-coding-standards/SKILL.md +723 -0
  93. package/dist/core/skills/builtin/cpp-testing/SKILL.md +324 -0
  94. package/dist/core/skills/builtin/crosspost/SKILL.md +111 -0
  95. package/dist/core/skills/builtin/csharp-testing/SKILL.md +321 -0
  96. package/dist/core/skills/builtin/customs-trade-compliance/SKILL.md +263 -0
  97. package/dist/core/skills/builtin/dart-flutter-patterns/SKILL.md +563 -0
  98. package/dist/core/skills/builtin/dashboard-builder/SKILL.md +108 -0
  99. package/dist/core/skills/builtin/data-scraper-agent/SKILL.md +764 -0
  100. package/dist/core/skills/builtin/data-throughput-accelerator/SKILL.md +72 -0
  101. package/dist/core/skills/builtin/database-migrations/SKILL.md +429 -0
  102. package/dist/core/skills/builtin/deep-research/SKILL.md +159 -0
  103. package/dist/core/skills/builtin/defi-amm-security/SKILL.md +166 -0
  104. package/dist/core/skills/builtin/deployment-patterns/SKILL.md +427 -0
  105. package/dist/core/skills/builtin/design-system/SKILL.md +82 -0
  106. package/dist/core/skills/builtin/django-celery/SKILL.md +457 -0
  107. package/dist/core/skills/builtin/django-patterns/SKILL.md +734 -0
  108. package/dist/core/skills/builtin/django-security/SKILL.md +593 -0
  109. package/dist/core/skills/builtin/django-tdd/SKILL.md +729 -0
  110. package/dist/core/skills/builtin/django-verification/SKILL.md +469 -0
  111. package/dist/core/skills/builtin/dmux-workflows/SKILL.md +191 -0
  112. package/dist/core/skills/builtin/docker-patterns/SKILL.md +364 -0
  113. package/dist/core/skills/builtin/documentation-lookup/SKILL.md +90 -0
  114. package/dist/core/skills/builtin/dotnet-patterns/SKILL.md +321 -0
  115. package/dist/core/skills/builtin/dynamic-workflow-mode/SKILL.md +123 -0
  116. package/dist/core/skills/builtin/e2e-testing/SKILL.md +326 -0
  117. package/dist/core/skills/builtin/email-ops/SKILL.md +121 -0
  118. package/dist/core/skills/builtin/energy-procurement/SKILL.md +228 -0
  119. package/dist/core/skills/builtin/enterprise-agent-ops/SKILL.md +50 -0
  120. package/dist/core/skills/builtin/error-handling/SKILL.md +376 -0
  121. package/dist/core/skills/builtin/eval-harness/SKILL.md +270 -0
  122. package/dist/core/skills/builtin/evm-token-decimals/SKILL.md +130 -0
  123. package/dist/core/skills/builtin/exa-search/SKILL.md +107 -0
  124. package/dist/core/skills/builtin/fal-ai-media/SKILL.md +288 -0
  125. package/dist/core/skills/builtin/fastapi-patterns/SKILL.md +513 -0
  126. package/dist/core/skills/builtin/finance-billing-ops/SKILL.md +127 -0
  127. package/dist/core/skills/builtin/flox-environments/SKILL.md +496 -0
  128. package/dist/core/skills/builtin/flutter-dart-code-review/SKILL.md +435 -0
  129. package/dist/core/skills/builtin/foundation-models-on-device/SKILL.md +243 -0
  130. package/dist/core/skills/builtin/frontend-a11y/SKILL.md +445 -0
  131. package/dist/core/skills/builtin/frontend-design-direction/SKILL.md +92 -0
  132. package/dist/core/skills/builtin/frontend-patterns/SKILL.md +656 -0
  133. package/dist/core/skills/builtin/frontend-slides/SKILL.md +184 -0
  134. package/dist/core/skills/builtin/frontend-slides/STYLE_PRESETS.md +330 -0
  135. package/dist/core/skills/builtin/frontend-slides/animation-patterns.md +122 -0
  136. package/dist/core/skills/builtin/frontend-slides/html-template.md +419 -0
  137. package/dist/core/skills/builtin/frontend-slides/scripts/export-pdf.sh +418 -0
  138. package/dist/core/skills/builtin/frontend-slides/scripts/extract-pptx.py +96 -0
  139. package/dist/core/skills/builtin/frontend-slides/viewport-base.css +153 -0
  140. package/dist/core/skills/builtin/fsharp-testing/SKILL.md +280 -0
  141. package/dist/core/skills/builtin/gan-style-harness/SKILL.md +278 -0
  142. package/dist/core/skills/builtin/gateguard/SKILL.md +132 -0
  143. package/dist/core/skills/builtin/git-workflow/SKILL.md +715 -0
  144. package/dist/core/skills/builtin/github-ops/SKILL.md +144 -0
  145. package/dist/core/skills/builtin/golang-patterns/SKILL.md +674 -0
  146. package/dist/core/skills/builtin/golang-testing/SKILL.md +720 -0
  147. package/dist/core/skills/builtin/healthcare-cdss-patterns/SKILL.md +245 -0
  148. package/dist/core/skills/builtin/healthcare-emr-patterns/SKILL.md +159 -0
  149. package/dist/core/skills/builtin/healthcare-eval-harness/SKILL.md +207 -0
  150. package/dist/core/skills/builtin/healthcare-phi-compliance/SKILL.md +145 -0
  151. package/dist/core/skills/builtin/hermes-imports/SKILL.md +88 -0
  152. package/dist/core/skills/builtin/hexagonal-architecture/SKILL.md +276 -0
  153. package/dist/core/skills/builtin/hipaa-compliance/SKILL.md +78 -0
  154. package/dist/core/skills/builtin/hookify-rules/SKILL.md +128 -0
  155. package/dist/core/skills/builtin/inherit-legacy-style/SKILL.md +156 -0
  156. package/dist/core/skills/builtin/intent-driven-development/SKILL.md +360 -0
  157. package/dist/core/skills/builtin/inventory-demand-planning/SKILL.md +247 -0
  158. package/dist/core/skills/builtin/ios-icon-gen/SKILL.md +157 -0
  159. package/dist/core/skills/builtin/ios-icon-gen/scripts/generate_icons.swift +258 -0
  160. package/dist/core/skills/builtin/ios-icon-gen/scripts/iconify_gen.sh +235 -0
  161. package/dist/core/skills/builtin/iterative-retrieval/SKILL.md +211 -0
  162. package/dist/core/skills/builtin/java-coding-standards/SKILL.md +383 -0
  163. package/dist/core/skills/builtin/jira-integration/SKILL.md +302 -0
  164. package/dist/core/skills/builtin/jpa-patterns/SKILL.md +151 -0
  165. package/dist/core/skills/builtin/knowledge-ops/SKILL.md +154 -0
  166. package/dist/core/skills/builtin/kotlin-coroutines-flows/SKILL.md +284 -0
  167. package/dist/core/skills/builtin/kotlin-exposed-patterns/SKILL.md +719 -0
  168. package/dist/core/skills/builtin/kotlin-ktor-patterns/SKILL.md +689 -0
  169. package/dist/core/skills/builtin/kotlin-patterns/SKILL.md +711 -0
  170. package/dist/core/skills/builtin/kotlin-testing/SKILL.md +824 -0
  171. package/dist/core/skills/builtin/kubernetes-patterns/SKILL.md +755 -0
  172. package/dist/core/skills/builtin/laravel-patterns/SKILL.md +415 -0
  173. package/dist/core/skills/builtin/laravel-plugin-discovery/SKILL.md +229 -0
  174. package/dist/core/skills/builtin/laravel-security/SKILL.md +947 -0
  175. package/dist/core/skills/builtin/laravel-tdd/SKILL.md +674 -0
  176. package/dist/core/skills/builtin/laravel-verification/SKILL.md +179 -0
  177. package/dist/core/skills/builtin/latency-critical-systems/SKILL.md +73 -0
  178. package/dist/core/skills/builtin/lead-intelligence/SKILL.md +321 -0
  179. package/dist/core/skills/builtin/lead-intelligence/agents/enrichment-agent.md +85 -0
  180. package/dist/core/skills/builtin/lead-intelligence/agents/mutual-mapper.md +75 -0
  181. package/dist/core/skills/builtin/lead-intelligence/agents/outreach-drafter.md +98 -0
  182. package/dist/core/skills/builtin/lead-intelligence/agents/signal-scorer.md +60 -0
  183. package/dist/core/skills/builtin/liquid-glass-design/SKILL.md +279 -0
  184. package/dist/core/skills/builtin/llm-trading-agent-security/SKILL.md +146 -0
  185. package/dist/core/skills/builtin/logistics-exception-management/SKILL.md +222 -0
  186. package/dist/core/skills/builtin/make-interfaces-feel-better/SKILL.md +151 -0
  187. package/dist/core/skills/builtin/market-research/SKILL.md +75 -0
  188. package/dist/core/skills/builtin/marketing-campaign/SKILL.md +113 -0
  189. package/dist/core/skills/builtin/mcp-server-patterns/SKILL.md +69 -0
  190. package/dist/core/skills/builtin/messages-ops/SKILL.md +104 -0
  191. package/dist/core/skills/builtin/mle-workflow/SKILL.md +346 -0
  192. package/dist/core/skills/builtin/motion-advanced/SKILL.md +596 -0
  193. package/dist/core/skills/builtin/motion-foundations/SKILL.md +299 -0
  194. package/dist/core/skills/builtin/motion-patterns/SKILL.md +434 -0
  195. package/dist/core/skills/builtin/motion-ui/SKILL.md +575 -0
  196. package/dist/core/skills/builtin/mysql-patterns/SKILL.md +412 -0
  197. package/dist/core/skills/builtin/nanoclaw-repl/SKILL.md +33 -0
  198. package/dist/core/skills/builtin/nestjs-patterns/SKILL.md +230 -0
  199. package/dist/core/skills/builtin/netmiko-ssh-automation/SKILL.md +173 -0
  200. package/dist/core/skills/builtin/network-bgp-diagnostics/SKILL.md +167 -0
  201. package/dist/core/skills/builtin/network-config-validation/SKILL.md +210 -0
  202. package/dist/core/skills/builtin/network-interface-health/SKILL.md +152 -0
  203. package/dist/core/skills/builtin/nextjs-turbopack/SKILL.md +57 -0
  204. package/dist/core/skills/builtin/nodejs-keccak256/SKILL.md +102 -0
  205. package/dist/core/skills/builtin/nutrient-document-processing/SKILL.md +167 -0
  206. package/dist/core/skills/builtin/nuxt4-patterns/SKILL.md +100 -0
  207. package/dist/core/skills/builtin/openclaw-persona-forge/SKILL.md +288 -0
  208. package/dist/core/skills/builtin/openclaw-persona-forge/gacha.py +224 -0
  209. package/dist/core/skills/builtin/openclaw-persona-forge/gacha.sh +5 -0
  210. package/dist/core/skills/builtin/openclaw-persona-forge/references/avatar-style.md +124 -0
  211. package/dist/core/skills/builtin/openclaw-persona-forge/references/boundary-rules.md +53 -0
  212. package/dist/core/skills/builtin/openclaw-persona-forge/references/error-handling.md +53 -0
  213. package/dist/core/skills/builtin/openclaw-persona-forge/references/identity-tension.md +48 -0
  214. package/dist/core/skills/builtin/openclaw-persona-forge/references/naming-system.md +39 -0
  215. package/dist/core/skills/builtin/openclaw-persona-forge/references/output-template.md +166 -0
  216. package/dist/core/skills/builtin/opensource-pipeline/SKILL.md +255 -0
  217. package/dist/core/skills/builtin/orch-add-feature/SKILL.md +44 -0
  218. package/dist/core/skills/builtin/orch-build-mvp/SKILL.md +48 -0
  219. package/dist/core/skills/builtin/orch-change-feature/SKILL.md +42 -0
  220. package/dist/core/skills/builtin/orch-fix-defect/SKILL.md +42 -0
  221. package/dist/core/skills/builtin/orch-pipeline/SKILL.md +120 -0
  222. package/dist/core/skills/builtin/orch-refine-code/SKILL.md +43 -0
  223. package/dist/core/skills/builtin/parallel-execution-optimizer/SKILL.md +72 -0
  224. package/dist/core/skills/builtin/perl-patterns/SKILL.md +504 -0
  225. package/dist/core/skills/builtin/perl-security/SKILL.md +503 -0
  226. package/dist/core/skills/builtin/perl-testing/SKILL.md +475 -0
  227. package/dist/core/skills/builtin/plan-orchestrate/SKILL.md +262 -0
  228. package/dist/core/skills/builtin/plankton-code-quality/SKILL.md +236 -0
  229. package/dist/core/skills/builtin/postgres-patterns/SKILL.md +147 -0
  230. package/dist/core/skills/builtin/prediction-market-oracle-research/SKILL.md +63 -0
  231. package/dist/core/skills/builtin/prediction-market-risk-review/SKILL.md +60 -0
  232. package/dist/core/skills/builtin/prisma-patterns/SKILL.md +371 -0
  233. package/dist/core/skills/builtin/product-capability/SKILL.md +141 -0
  234. package/dist/core/skills/builtin/product-lens/SKILL.md +92 -0
  235. package/dist/core/skills/builtin/production-audit/SKILL.md +206 -0
  236. package/dist/core/skills/builtin/production-scheduling/SKILL.md +238 -0
  237. package/dist/core/skills/builtin/prompt-optimizer/SKILL.md +398 -0
  238. package/dist/core/skills/builtin/python-patterns/SKILL.md +750 -0
  239. package/dist/core/skills/builtin/python-testing/SKILL.md +816 -0
  240. package/dist/core/skills/builtin/pytorch-patterns/SKILL.md +396 -0
  241. package/dist/core/skills/builtin/quality-nonconformance/SKILL.md +260 -0
  242. package/dist/core/skills/builtin/quarkus-patterns/SKILL.md +722 -0
  243. package/dist/core/skills/builtin/quarkus-security/SKILL.md +467 -0
  244. package/dist/core/skills/builtin/quarkus-tdd/SKILL.md +811 -0
  245. package/dist/core/skills/builtin/quarkus-verification/SKILL.md +479 -0
  246. package/dist/core/skills/builtin/ralphinho-rfc-pipeline/SKILL.md +67 -0
  247. package/dist/core/skills/builtin/react-patterns/SKILL.md +341 -0
  248. package/dist/core/skills/builtin/react-performance/SKILL.md +574 -0
  249. package/dist/core/skills/builtin/react-testing/SKILL.md +423 -0
  250. package/dist/core/skills/builtin/recsys-pipeline-architect/SKILL.md +114 -0
  251. package/dist/core/skills/builtin/recursive-decision-ledger/SKILL.md +79 -0
  252. package/dist/core/skills/builtin/redis-patterns/SKILL.md +403 -0
  253. package/dist/core/skills/builtin/regex-vs-llm-structured-text/SKILL.md +220 -0
  254. package/dist/core/skills/builtin/repo-scan/SKILL.md +78 -0
  255. package/dist/core/skills/builtin/research-ops/SKILL.md +112 -0
  256. package/dist/core/skills/builtin/returns-reverse-logistics/SKILL.md +240 -0
  257. package/dist/core/skills/builtin/rules-distill/SKILL.md +264 -0
  258. package/dist/core/skills/builtin/rules-distill/scripts/scan-rules.sh +58 -0
  259. package/dist/core/skills/builtin/rules-distill/scripts/scan-skills.sh +129 -0
  260. package/dist/core/skills/builtin/rust-patterns/SKILL.md +499 -0
  261. package/dist/core/skills/builtin/rust-testing/SKILL.md +500 -0
  262. package/dist/core/skills/builtin/safety-guard/SKILL.md +75 -0
  263. package/dist/core/skills/builtin/santa-method/SKILL.md +306 -0
  264. package/dist/core/skills/builtin/scientific-db-pubmed-database/SKILL.md +175 -0
  265. package/dist/core/skills/builtin/scientific-db-uspto-database/SKILL.md +177 -0
  266. package/dist/core/skills/builtin/scientific-pkg-gget/SKILL.md +166 -0
  267. package/dist/core/skills/builtin/scientific-thinking-literature-review/SKILL.md +192 -0
  268. package/dist/core/skills/builtin/scientific-thinking-scholar-evaluation/SKILL.md +160 -0
  269. package/dist/core/skills/builtin/search-first/SKILL.md +182 -0
  270. package/dist/core/skills/builtin/security-bounty-hunter/SKILL.md +99 -0
  271. package/dist/core/skills/builtin/security-review/SKILL.md +503 -0
  272. package/dist/core/skills/builtin/security-review/cloud-infrastructure-security.md +361 -0
  273. package/dist/core/skills/builtin/security-scan/SKILL.md +165 -0
  274. package/dist/core/skills/builtin/seo/SKILL.md +154 -0
  275. package/dist/core/skills/builtin/skill-comply/SKILL.md +58 -0
  276. package/dist/core/skills/builtin/skill-comply/fixtures/compliant_trace.jsonl +5 -0
  277. package/dist/core/skills/builtin/skill-comply/fixtures/noncompliant_trace.jsonl +3 -0
  278. package/dist/core/skills/builtin/skill-comply/fixtures/tdd_spec.yaml +44 -0
  279. package/dist/core/skills/builtin/skill-comply/prompts/classifier.md +24 -0
  280. package/dist/core/skills/builtin/skill-comply/prompts/scenario_generator.md +62 -0
  281. package/dist/core/skills/builtin/skill-comply/prompts/spec_generator.md +42 -0
  282. package/dist/core/skills/builtin/skill-comply/pyproject.toml +15 -0
  283. package/dist/core/skills/builtin/skill-comply/scripts/__init__.py +0 -0
  284. package/dist/core/skills/builtin/skill-comply/scripts/classifier.py +85 -0
  285. package/dist/core/skills/builtin/skill-comply/scripts/grader.py +124 -0
  286. package/dist/core/skills/builtin/skill-comply/scripts/parser.py +107 -0
  287. package/dist/core/skills/builtin/skill-comply/scripts/report.py +170 -0
  288. package/dist/core/skills/builtin/skill-comply/scripts/run.py +127 -0
  289. package/dist/core/skills/builtin/skill-comply/scripts/runner.py +186 -0
  290. package/dist/core/skills/builtin/skill-comply/scripts/scenario_generator.py +70 -0
  291. package/dist/core/skills/builtin/skill-comply/scripts/spec_generator.py +72 -0
  292. package/dist/core/skills/builtin/skill-comply/scripts/utils.py +13 -0
  293. package/dist/core/skills/builtin/skill-comply/tests/test_grader.py +197 -0
  294. package/dist/core/skills/builtin/skill-comply/tests/test_parser.py +90 -0
  295. package/dist/core/skills/builtin/skill-comply/tests/test_runner.py +172 -0
  296. package/dist/core/skills/builtin/skill-scout/SKILL.md +140 -0
  297. package/dist/core/skills/builtin/skill-stocktake/SKILL.md +194 -0
  298. package/dist/core/skills/builtin/skill-stocktake/scripts/quick-diff.sh +87 -0
  299. package/dist/core/skills/builtin/skill-stocktake/scripts/save-results.sh +56 -0
  300. package/dist/core/skills/builtin/skill-stocktake/scripts/scan.sh +170 -0
  301. package/dist/core/skills/builtin/springboot-patterns/SKILL.md +314 -0
  302. package/dist/core/skills/builtin/springboot-security/SKILL.md +272 -0
  303. package/dist/core/skills/builtin/springboot-tdd/SKILL.md +158 -0
  304. package/dist/core/skills/builtin/springboot-verification/SKILL.md +231 -0
  305. package/dist/core/skills/builtin/strategic-compact/SKILL.md +135 -0
  306. package/dist/core/skills/builtin/swift-actor-persistence/SKILL.md +143 -0
  307. package/dist/core/skills/builtin/swift-concurrency-6-2/SKILL.md +216 -0
  308. package/dist/core/skills/builtin/swift-protocol-di-testing/SKILL.md +190 -0
  309. package/dist/core/skills/builtin/swiftui-patterns/SKILL.md +259 -0
  310. package/dist/core/skills/builtin/tdd-workflow/SKILL.md +463 -0
  311. package/dist/core/skills/builtin/team-agent-orchestration/SKILL.md +110 -0
  312. package/dist/core/skills/builtin/team-builder/SKILL.md +168 -0
  313. package/dist/core/skills/builtin/terminal-ops/SKILL.md +109 -0
  314. package/dist/core/skills/builtin/tinystruct-patterns/SKILL.md +203 -0
  315. package/dist/core/skills/builtin/tinystruct-patterns/references/architecture.md +90 -0
  316. package/dist/core/skills/builtin/tinystruct-patterns/references/data-handling.md +60 -0
  317. package/dist/core/skills/builtin/tinystruct-patterns/references/database.md +99 -0
  318. package/dist/core/skills/builtin/tinystruct-patterns/references/routing.md +64 -0
  319. package/dist/core/skills/builtin/tinystruct-patterns/references/system-usage.md +97 -0
  320. package/dist/core/skills/builtin/tinystruct-patterns/references/testing.md +72 -0
  321. package/dist/core/skills/builtin/token-budget-advisor/SKILL.md +133 -0
  322. package/dist/core/skills/builtin/ui-demo/SKILL.md +465 -0
  323. package/dist/core/skills/builtin/ui-to-vue/SKILL.md +134 -0
  324. package/dist/core/skills/builtin/uncloud/SKILL.md +343 -0
  325. package/dist/core/skills/builtin/unified-notifications-ops/SKILL.md +187 -0
  326. package/dist/core/skills/builtin/verification-loop/SKILL.md +126 -0
  327. package/dist/core/skills/builtin/video-editing/SKILL.md +310 -0
  328. package/dist/core/skills/builtin/videodb/SKILL.md +374 -0
  329. package/dist/core/skills/builtin/videodb/reference/api-reference.md +550 -0
  330. package/dist/core/skills/builtin/videodb/reference/capture-reference.md +407 -0
  331. package/dist/core/skills/builtin/videodb/reference/capture.md +101 -0
  332. package/dist/core/skills/builtin/videodb/reference/editor.md +443 -0
  333. package/dist/core/skills/builtin/videodb/reference/generative.md +331 -0
  334. package/dist/core/skills/builtin/videodb/reference/rtstream-reference.md +564 -0
  335. package/dist/core/skills/builtin/videodb/reference/rtstream.md +65 -0
  336. package/dist/core/skills/builtin/videodb/reference/search.md +230 -0
  337. package/dist/core/skills/builtin/videodb/reference/streaming.md +406 -0
  338. package/dist/core/skills/builtin/videodb/reference/use-cases.md +118 -0
  339. package/dist/core/skills/builtin/videodb/scripts/ws_listener.py +282 -0
  340. package/dist/core/skills/builtin/visa-doc-translate/README.md +86 -0
  341. package/dist/core/skills/builtin/visa-doc-translate/SKILL.md +117 -0
  342. package/dist/core/skills/builtin/vite-patterns/SKILL.md +449 -0
  343. package/dist/core/skills/builtin/windows-desktop-e2e/SKILL.md +887 -0
  344. package/dist/core/skills/builtin/x-api/SKILL.md +234 -0
  345. package/dist/core/skills/loader.js +11 -0
  346. package/dist/core/skills/loader.js.map +1 -1
  347. package/package.json +1 -1
@@ -0,0 +1,498 @@
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
+ _SECRET_RE = re.compile(
272
+ r"(?i)(api[_-]?key|token|secret|password|authorization|credentials?|auth)"
273
+ r"""(["'"'"'\s:=]+)"""
274
+ r"([A-Za-z]+\s+)?"
275
+ r"([A-Za-z0-9_\-/.+=]{8,})"
276
+ )
277
+
278
+ raw = sys.stdin.read()[:2000]
279
+ raw = _SECRET_RE.sub(lambda m: m.group(1) + m.group(2) + (m.group(3) or "") + "[REDACTED]", raw)
280
+ print(json.dumps({"timestamp": os.environ["TIMESTAMP"], "event": "parse_error", "raw": raw}))
281
+ ' >> "$OBSERVATIONS_FILE"
282
+ exit 0
283
+ fi
284
+
285
+ # Archive if file too large (atomic: rename with unique suffix to avoid race)
286
+ if [ -f "$OBSERVATIONS_FILE" ]; then
287
+ file_size_mb=$(du -m "$OBSERVATIONS_FILE" 2>/dev/null | cut -f1)
288
+ if [ "${file_size_mb:-0}" -ge "$MAX_FILE_SIZE_MB" ]; then
289
+ archive_dir="${PROJECT_DIR}/observations.archive"
290
+ mkdir -p "$archive_dir"
291
+ mv "$OBSERVATIONS_FILE" "$archive_dir/observations-$(date +%Y%m%d-%H%M%S)-$$.jsonl" 2>/dev/null || true
292
+ fi
293
+ fi
294
+
295
+ # Build and write observation (now includes project context)
296
+ # Scrub common secret patterns from tool I/O before persisting
297
+ timestamp=$(date -u +"%Y-%m-%dT%H:%M:%SZ")
298
+
299
+ export PROJECT_ID_ENV="$PROJECT_ID"
300
+ export PROJECT_NAME_ENV="$PROJECT_NAME"
301
+ export TIMESTAMP="$timestamp"
302
+
303
+ echo "$PARSED" | "$PYTHON_CMD" -c '
304
+ import json, sys, os, re
305
+
306
+ parsed = json.load(sys.stdin)
307
+ observation = {
308
+ "timestamp": os.environ["TIMESTAMP"],
309
+ "event": parsed["event"],
310
+ "tool": parsed["tool"],
311
+ "session": parsed["session"],
312
+ "project_id": os.environ.get("PROJECT_ID_ENV", "global"),
313
+ "project_name": os.environ.get("PROJECT_NAME_ENV", "global")
314
+ }
315
+
316
+ # Scrub secrets: match common key=value, key: value, and key"value patterns
317
+ # Includes optional auth scheme (e.g., "Bearer", "Basic") before token
318
+ _SECRET_RE = re.compile(
319
+ r"(?i)(api[_-]?key|token|secret|password|authorization|credentials?|auth)"
320
+ r"""(["'"'"'\s:=]+)"""
321
+ r"([A-Za-z]+\s+)?"
322
+ r"([A-Za-z0-9_\-/.+=]{8,})"
323
+ )
324
+
325
+ def scrub(val):
326
+ if val is None:
327
+ return None
328
+ return _SECRET_RE.sub(lambda m: m.group(1) + m.group(2) + (m.group(3) or "") + "[REDACTED]", str(val))
329
+
330
+ if parsed["input"]:
331
+ observation["input"] = scrub(parsed["input"])
332
+ if parsed["output"] is not None:
333
+ observation["output"] = scrub(parsed["output"])
334
+
335
+ print(json.dumps(observation))
336
+ ' >> "$OBSERVATIONS_FILE"
337
+
338
+ # Lazy-start observer if enabled but not running (first-time setup)
339
+ # Use flock for atomic check-then-act to prevent race conditions
340
+ # Fallback for macOS (no flock): use lockfile or skip
341
+ LAZY_START_LOCK="${PROJECT_DIR}/.observer-start.lock"
342
+ _REMOVE_FILE_IF_PRESENT() {
343
+ local target="$1"
344
+ if [ -n "$target" ] && [ -e "$target" ]; then
345
+ rm -- "$target" 2>/dev/null || true
346
+ fi
347
+ }
348
+
349
+ _START_OBSERVER_LOGGED() {
350
+ local bootstrap_log="${PROJECT_DIR}/observer-start.log"
351
+ mkdir -p "$PROJECT_DIR"
352
+ "${SKILL_ROOT}/agents/start-observer.sh" start >> "$bootstrap_log" 2>&1 || true
353
+ }
354
+
355
+ _CHECK_OBSERVER_RUNNING() {
356
+ local pid_file="$1"
357
+ if [ -f "$pid_file" ]; then
358
+ local pid
359
+ pid=$(cat "$pid_file" 2>/dev/null)
360
+ # Validate PID is a positive integer (>1) to prevent signaling invalid targets
361
+ case "$pid" in
362
+ ''|*[!0-9]*|0|1)
363
+ _REMOVE_FILE_IF_PRESENT "$pid_file"
364
+ return 1
365
+ ;;
366
+ esac
367
+ if kill -0 "$pid" 2>/dev/null; then
368
+ return 0 # Process is alive
369
+ fi
370
+ # Stale PID file - remove it
371
+ _REMOVE_FILE_IF_PRESENT "$pid_file"
372
+ fi
373
+ return 1 # No PID file or process dead
374
+ }
375
+
376
+ if [ -f "${CONFIG_DIR}/disabled" ]; then
377
+ OBSERVER_ENABLED=false
378
+ else
379
+ OBSERVER_ENABLED=false
380
+ if [ -n "${CLV2_CONFIG:-}" ]; then
381
+ CONFIG_FILE="$CLV2_CONFIG"
382
+ elif [ -f "${CONFIG_DIR}/config.json" ]; then
383
+ CONFIG_FILE="${CONFIG_DIR}/config.json"
384
+ else
385
+ CONFIG_FILE="${SKILL_ROOT}/config.json"
386
+ fi
387
+ # Use effective config path for both existence check and reading
388
+ EFFECTIVE_CONFIG="$CONFIG_FILE"
389
+ if [ -f "$EFFECTIVE_CONFIG" ] && [ -n "$PYTHON_CMD" ]; then
390
+ _enabled=$(CLV2_CONFIG_PATH="$EFFECTIVE_CONFIG" "$PYTHON_CMD" -c "
391
+ import json, os
392
+ with open(os.environ['CLV2_CONFIG_PATH']) as f:
393
+ cfg = json.load(f)
394
+ print(str(cfg.get('observer', {}).get('enabled', False)).lower())
395
+ " 2>/dev/null || echo "false")
396
+ if [ "$_enabled" = "true" ]; then
397
+ OBSERVER_ENABLED=true
398
+ fi
399
+ fi
400
+ fi
401
+
402
+ # Check both project-scoped AND global PID files (with stale PID recovery)
403
+ if [ "$OBSERVER_ENABLED" = "true" ]; then
404
+ # Clean up stale PID files first
405
+ _CHECK_OBSERVER_RUNNING "${PROJECT_DIR}/.observer.pid" || true
406
+ _CHECK_OBSERVER_RUNNING "${CONFIG_DIR}/.observer.pid" || true
407
+
408
+ # Check if observer is now running after cleanup
409
+ if [ ! -f "${PROJECT_DIR}/.observer.pid" ] && [ ! -f "${CONFIG_DIR}/.observer.pid" ]; then
410
+ # Use flock if available (Linux), fallback for macOS
411
+ if command -v flock >/dev/null 2>&1; then
412
+ (
413
+ flock -n 9 || exit 0
414
+ # Double-check PID files after acquiring lock
415
+ _CHECK_OBSERVER_RUNNING "${PROJECT_DIR}/.observer.pid" || true
416
+ _CHECK_OBSERVER_RUNNING "${CONFIG_DIR}/.observer.pid" || true
417
+ if [ ! -f "${PROJECT_DIR}/.observer.pid" ] && [ ! -f "${CONFIG_DIR}/.observer.pid" ]; then
418
+ _START_OBSERVER_LOGGED
419
+ fi
420
+ ) 9>"$LAZY_START_LOCK"
421
+ else
422
+ # macOS fallback: use lockfile if available, otherwise mkdir-based lock
423
+ if command -v lockfile >/dev/null 2>&1; then
424
+ # Use subshell to isolate exit and add trap for cleanup
425
+ (
426
+ trap '_REMOVE_FILE_IF_PRESENT "$LAZY_START_LOCK"' EXIT
427
+ lockfile -r 1 -l 30 "$LAZY_START_LOCK" 2>/dev/null || exit 0
428
+ _CHECK_OBSERVER_RUNNING "${PROJECT_DIR}/.observer.pid" || true
429
+ _CHECK_OBSERVER_RUNNING "${CONFIG_DIR}/.observer.pid" || true
430
+ if [ ! -f "${PROJECT_DIR}/.observer.pid" ] && [ ! -f "${CONFIG_DIR}/.observer.pid" ]; then
431
+ _START_OBSERVER_LOGGED
432
+ fi
433
+ _REMOVE_FILE_IF_PRESENT "$LAZY_START_LOCK"
434
+ )
435
+ else
436
+ # POSIX fallback: mkdir is atomic -- fails if dir already exists
437
+ (
438
+ trap 'rmdir "${LAZY_START_LOCK}.d" 2>/dev/null || true' EXIT
439
+ mkdir "${LAZY_START_LOCK}.d" 2>/dev/null || exit 0
440
+ _CHECK_OBSERVER_RUNNING "${PROJECT_DIR}/.observer.pid" || true
441
+ _CHECK_OBSERVER_RUNNING "${CONFIG_DIR}/.observer.pid" || true
442
+ if [ ! -f "${PROJECT_DIR}/.observer.pid" ] && [ ! -f "${CONFIG_DIR}/.observer.pid" ]; then
443
+ _START_OBSERVER_LOGGED
444
+ fi
445
+ )
446
+ fi
447
+ fi
448
+ fi
449
+ fi
450
+
451
+ # Throttle SIGUSR1: only signal observer every N observations (#521)
452
+ # This prevents rapid signaling when tool calls fire every second,
453
+ # which caused runaway parallel Claude analysis processes.
454
+ SIGNAL_EVERY_N="${ECC_OBSERVER_SIGNAL_EVERY_N:-20}"
455
+ SIGNAL_COUNTER_FILE="${PROJECT_DIR}/.observer-signal-counter"
456
+ ACTIVITY_FILE="${PROJECT_DIR}/.observer-last-activity"
457
+
458
+ touch "$ACTIVITY_FILE" 2>/dev/null || true
459
+
460
+ should_signal=0
461
+ if [ -f "$SIGNAL_COUNTER_FILE" ]; then
462
+ counter=$(cat "$SIGNAL_COUNTER_FILE" 2>/dev/null || echo 0)
463
+ counter=$((counter + 1))
464
+ if [ "$counter" -ge "$SIGNAL_EVERY_N" ]; then
465
+ should_signal=1
466
+ counter=0
467
+ fi
468
+ echo "$counter" > "$SIGNAL_COUNTER_FILE"
469
+ else
470
+ echo "1" > "$SIGNAL_COUNTER_FILE"
471
+ fi
472
+
473
+ # Signal observer if running and throttle allows (check both project-scoped and global observer, deduplicate)
474
+ if [ "$should_signal" -eq 1 ]; then
475
+ signaled_pids=" "
476
+ for pid_file in "${PROJECT_DIR}/.observer.pid" "${CONFIG_DIR}/.observer.pid"; do
477
+ if [ -f "$pid_file" ]; then
478
+ observer_pid=$(cat "$pid_file" 2>/dev/null || true)
479
+ # Validate PID is a positive integer (>1)
480
+ case "$observer_pid" in
481
+ ''|*[!0-9]*|0|1)
482
+ _REMOVE_FILE_IF_PRESENT "$pid_file"
483
+ continue
484
+ ;;
485
+ esac
486
+ # Deduplicate: skip if already signaled this pass
487
+ case "$signaled_pids" in
488
+ *" $observer_pid "*) continue ;;
489
+ esac
490
+ if kill -0 "$observer_pid" 2>/dev/null; then
491
+ kill -USR1 "$observer_pid" 2>/dev/null || true
492
+ signaled_pids="${signaled_pids}${observer_pid} "
493
+ fi
494
+ fi
495
+ done
496
+ fi
497
+
498
+ exit 0