@heytherevibin/skillforge 0.2.1

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 (402) hide show
  1. package/CHANGELOG.md +16 -0
  2. package/CODE_OF_CONDUCT.md +34 -0
  3. package/CONTRIBUTING.md +38 -0
  4. package/LICENSE +21 -0
  5. package/README.md +337 -0
  6. package/RELEASING.md +93 -0
  7. package/SECURITY.md +31 -0
  8. package/STRATEGY.md +26 -0
  9. package/bin/cli.js +547 -0
  10. package/lib/packs.js +184 -0
  11. package/package.json +38 -0
  12. package/python/app/__init__.py +0 -0
  13. package/python/app/__pycache__/__init__.cpython-312.pyc +0 -0
  14. package/python/app/__pycache__/auth.cpython-312.pyc +0 -0
  15. package/python/app/__pycache__/main.cpython-312.pyc +0 -0
  16. package/python/app/auth.py +63 -0
  17. package/python/app/cli.py +78 -0
  18. package/python/app/db_paths.py +26 -0
  19. package/python/app/events_cli.py +175 -0
  20. package/python/app/main.py +647 -0
  21. package/python/app/materialize.py +138 -0
  22. package/python/app/mcp_server.py +610 -0
  23. package/python/app/route_cli.py +117 -0
  24. package/python/requirements-dev.txt +1 -0
  25. package/python/requirements.txt +7 -0
  26. package/python/tests/test_db_paths.py +41 -0
  27. package/skills/accessibility/SKILL.md +145 -0
  28. package/skills/agent-architecture-audit/SKILL.md +256 -0
  29. package/skills/agent-eval/SKILL.md +144 -0
  30. package/skills/agent-harness-construction/SKILL.md +72 -0
  31. package/skills/agent-introspection-debugging/SKILL.md +152 -0
  32. package/skills/agent-payment-x402/SKILL.md +224 -0
  33. package/skills/agent-sort/SKILL.md +214 -0
  34. package/skills/agentic-engineering/SKILL.md +62 -0
  35. package/skills/agentic-os/SKILL.md +386 -0
  36. package/skills/ai-first-engineering/SKILL.md +50 -0
  37. package/skills/ai-regression-testing/SKILL.md +384 -0
  38. package/skills/android-clean-architecture/SKILL.md +338 -0
  39. package/skills/angular-developer/SKILL.md +153 -0
  40. package/skills/angular-developer/references/angular-animations.md +160 -0
  41. package/skills/angular-developer/references/angular-aria.md +410 -0
  42. package/skills/angular-developer/references/cli.md +86 -0
  43. package/skills/angular-developer/references/component-harnesses.md +59 -0
  44. package/skills/angular-developer/references/component-styling.md +91 -0
  45. package/skills/angular-developer/references/components.md +117 -0
  46. package/skills/angular-developer/references/creating-services.md +97 -0
  47. package/skills/angular-developer/references/data-resolvers.md +69 -0
  48. package/skills/angular-developer/references/define-routes.md +67 -0
  49. package/skills/angular-developer/references/defining-providers.md +72 -0
  50. package/skills/angular-developer/references/di-fundamentals.md +120 -0
  51. package/skills/angular-developer/references/e2e-testing.md +56 -0
  52. package/skills/angular-developer/references/effects.md +83 -0
  53. package/skills/angular-developer/references/hierarchical-injectors.md +43 -0
  54. package/skills/angular-developer/references/host-elements.md +80 -0
  55. package/skills/angular-developer/references/injection-context.md +63 -0
  56. package/skills/angular-developer/references/inputs.md +101 -0
  57. package/skills/angular-developer/references/linked-signal.md +59 -0
  58. package/skills/angular-developer/references/loading-strategies.md +61 -0
  59. package/skills/angular-developer/references/mcp.md +108 -0
  60. package/skills/angular-developer/references/navigate-to-routes.md +69 -0
  61. package/skills/angular-developer/references/outputs.md +86 -0
  62. package/skills/angular-developer/references/reactive-forms.md +122 -0
  63. package/skills/angular-developer/references/rendering-strategies.md +44 -0
  64. package/skills/angular-developer/references/resource.md +77 -0
  65. package/skills/angular-developer/references/route-animations.md +56 -0
  66. package/skills/angular-developer/references/route-guards.md +52 -0
  67. package/skills/angular-developer/references/router-lifecycle.md +45 -0
  68. package/skills/angular-developer/references/router-testing.md +87 -0
  69. package/skills/angular-developer/references/show-routes-with-outlets.md +68 -0
  70. package/skills/angular-developer/references/signal-forms.md +795 -0
  71. package/skills/angular-developer/references/signals-overview.md +94 -0
  72. package/skills/angular-developer/references/tailwind-css.md +69 -0
  73. package/skills/angular-developer/references/template-driven-forms.md +114 -0
  74. package/skills/angular-developer/references/testing-fundamentals.md +65 -0
  75. package/skills/api-connector-builder/SKILL.md +120 -0
  76. package/skills/api-design/SKILL.md +522 -0
  77. package/skills/architecture-decision-records/SKILL.md +178 -0
  78. package/skills/article-writing/SKILL.md +78 -0
  79. package/skills/automation-audit-ops/SKILL.md +141 -0
  80. package/skills/autonomous-agent-harness/SKILL.md +272 -0
  81. package/skills/autonomous-loops/SKILL.md +609 -0
  82. package/skills/backend-patterns/SKILL.md +560 -0
  83. package/skills/benchmark/SKILL.md +92 -0
  84. package/skills/blueprint/SKILL.md +104 -0
  85. package/skills/browser-qa/SKILL.md +86 -0
  86. package/skills/bun-runtime/SKILL.md +83 -0
  87. package/skills/canary-watch/SKILL.md +98 -0
  88. package/skills/carrier-relationship-management/SKILL.md +211 -0
  89. package/skills/cisco-ios-patterns/SKILL.md +163 -0
  90. package/skills/ck/SKILL.md +147 -0
  91. package/skills/ck/commands/forget.mjs +44 -0
  92. package/skills/ck/commands/info.mjs +24 -0
  93. package/skills/ck/commands/init.mjs +143 -0
  94. package/skills/ck/commands/list.mjs +40 -0
  95. package/skills/ck/commands/migrate.mjs +202 -0
  96. package/skills/ck/commands/resume.mjs +36 -0
  97. package/skills/ck/commands/save.mjs +210 -0
  98. package/skills/ck/commands/shared.mjs +387 -0
  99. package/skills/ck/hooks/session-start.mjs +224 -0
  100. package/skills/claude-devfleet/SKILL.md +103 -0
  101. package/skills/click-path-audit/SKILL.md +244 -0
  102. package/skills/clickhouse-io/SKILL.md +438 -0
  103. package/skills/code-tour/SKILL.md +235 -0
  104. package/skills/codebase-onboarding/SKILL.md +232 -0
  105. package/skills/coding-standards/SKILL.md +548 -0
  106. package/skills/compose-multiplatform-patterns/SKILL.md +298 -0
  107. package/skills/connections-optimizer/SKILL.md +188 -0
  108. package/skills/content-engine/SKILL.md +126 -0
  109. package/skills/content-hash-cache-pattern/SKILL.md +160 -0
  110. package/skills/context-budget/SKILL.md +134 -0
  111. package/skills/continuous-agent-loop/SKILL.md +44 -0
  112. package/skills/continuous-learning/SKILL.md +129 -0
  113. package/skills/continuous-learning/config.json +18 -0
  114. package/skills/continuous-learning/evaluate-session.sh +69 -0
  115. package/skills/continuous-learning-v2/SKILL.md +358 -0
  116. package/skills/continuous-learning-v2/agents/observer-loop.sh +322 -0
  117. package/skills/continuous-learning-v2/agents/observer.md +198 -0
  118. package/skills/continuous-learning-v2/agents/session-guardian.sh +150 -0
  119. package/skills/continuous-learning-v2/agents/start-observer.sh +248 -0
  120. package/skills/continuous-learning-v2/config.json +8 -0
  121. package/skills/continuous-learning-v2/hooks/observe.sh +476 -0
  122. package/skills/continuous-learning-v2/scripts/detect-project.sh +288 -0
  123. package/skills/continuous-learning-v2/scripts/instinct-cli.py +1519 -0
  124. package/skills/continuous-learning-v2/scripts/lib/homunculus-dir.sh +31 -0
  125. package/skills/continuous-learning-v2/scripts/migrate-homunculus.sh +62 -0
  126. package/skills/continuous-learning-v2/scripts/test_parse_instinct.py +1018 -0
  127. package/skills/cost-aware-llm-pipeline/SKILL.md +182 -0
  128. package/skills/cost-tracking/SKILL.md +147 -0
  129. package/skills/council/SKILL.md +202 -0
  130. package/skills/cpp-coding-standards/SKILL.md +722 -0
  131. package/skills/cpp-testing/SKILL.md +323 -0
  132. package/skills/crosspost/SKILL.md +110 -0
  133. package/skills/csharp-testing/SKILL.md +320 -0
  134. package/skills/customer-billing-ops/SKILL.md +139 -0
  135. package/skills/customs-trade-compliance/SKILL.md +262 -0
  136. package/skills/dart-flutter-patterns/SKILL.md +562 -0
  137. package/skills/dashboard-builder/SKILL.md +108 -0
  138. package/skills/data-scraper-agent/SKILL.md +764 -0
  139. package/skills/database-migrations/SKILL.md +428 -0
  140. package/skills/deep-research/SKILL.md +158 -0
  141. package/skills/defi-amm-security/SKILL.md +166 -0
  142. package/skills/deployment-patterns/SKILL.md +426 -0
  143. package/skills/design-system/SKILL.md +81 -0
  144. package/skills/django-celery/SKILL.md +456 -0
  145. package/skills/django-patterns/SKILL.md +733 -0
  146. package/skills/django-security/SKILL.md +592 -0
  147. package/skills/django-tdd/SKILL.md +728 -0
  148. package/skills/django-verification/SKILL.md +468 -0
  149. package/skills/dmux-workflows/SKILL.md +190 -0
  150. package/skills/docker-patterns/SKILL.md +363 -0
  151. package/skills/documentation-lookup/SKILL.md +89 -0
  152. package/skills/dotnet-patterns/SKILL.md +320 -0
  153. package/skills/e2e-testing/SKILL.md +325 -0
  154. package/skills/email-ops/SKILL.md +120 -0
  155. package/skills/energy-procurement/SKILL.md +227 -0
  156. package/skills/enterprise-agent-ops/SKILL.md +49 -0
  157. package/skills/error-handling/SKILL.md +375 -0
  158. package/skills/eval-harness/SKILL.md +269 -0
  159. package/skills/evm-token-decimals/SKILL.md +130 -0
  160. package/skills/exa-search/SKILL.md +106 -0
  161. package/skills/fal-ai-media/SKILL.md +287 -0
  162. package/skills/fastapi-patterns/SKILL.md +327 -0
  163. package/skills/finance-billing-ops/SKILL.md +126 -0
  164. package/skills/flox-environments/SKILL.md +496 -0
  165. package/skills/flutter-dart-code-review/SKILL.md +434 -0
  166. package/skills/foundation-models-on-device/SKILL.md +243 -0
  167. package/skills/frontend-design-direction/SKILL.md +92 -0
  168. package/skills/frontend-patterns/SKILL.md +641 -0
  169. package/skills/frontend-slides/SKILL.md +183 -0
  170. package/skills/frontend-slides/STYLE_PRESETS.md +330 -0
  171. package/skills/frontend-slides/animation-patterns.md +122 -0
  172. package/skills/frontend-slides/html-template.md +419 -0
  173. package/skills/frontend-slides/scripts/export-pdf.sh +418 -0
  174. package/skills/frontend-slides/scripts/extract-pptx.py +96 -0
  175. package/skills/frontend-slides/viewport-base.css +153 -0
  176. package/skills/fsharp-testing/SKILL.md +279 -0
  177. package/skills/gan-style-harness/SKILL.md +278 -0
  178. package/skills/gateguard/SKILL.md +125 -0
  179. package/skills/git-workflow/SKILL.md +714 -0
  180. package/skills/github-ops/SKILL.md +143 -0
  181. package/skills/golang-patterns/SKILL.md +673 -0
  182. package/skills/golang-testing/SKILL.md +719 -0
  183. package/skills/google-workspace-ops/SKILL.md +94 -0
  184. package/skills/healthcare-cdss-patterns/SKILL.md +245 -0
  185. package/skills/healthcare-emr-patterns/SKILL.md +159 -0
  186. package/skills/healthcare-eval-harness/SKILL.md +207 -0
  187. package/skills/healthcare-phi-compliance/SKILL.md +145 -0
  188. package/skills/hermes-imports/SKILL.md +87 -0
  189. package/skills/hexagonal-architecture/SKILL.md +275 -0
  190. package/skills/hipaa-compliance/SKILL.md +78 -0
  191. package/skills/homelab-network-readiness/SKILL.md +169 -0
  192. package/skills/homelab-network-setup/SKILL.md +129 -0
  193. package/skills/homelab-pihole-dns/SKILL.md +274 -0
  194. package/skills/homelab-vlan-segmentation/SKILL.md +311 -0
  195. package/skills/homelab-wireguard-vpn/SKILL.md +305 -0
  196. package/skills/hookify-rules/SKILL.md +128 -0
  197. package/skills/inventory-demand-planning/SKILL.md +246 -0
  198. package/skills/investor-materials/SKILL.md +95 -0
  199. package/skills/investor-outreach/SKILL.md +90 -0
  200. package/skills/ios-icon-gen/SKILL.md +157 -0
  201. package/skills/ios-icon-gen/scripts/generate_icons.swift +258 -0
  202. package/skills/ios-icon-gen/scripts/iconify_gen.sh +235 -0
  203. package/skills/iterative-retrieval/SKILL.md +209 -0
  204. package/skills/java-coding-standards/SKILL.md +382 -0
  205. package/skills/jira-integration/SKILL.md +292 -0
  206. package/skills/jpa-patterns/SKILL.md +150 -0
  207. package/skills/knowledge-ops/SKILL.md +153 -0
  208. package/skills/kotlin-coroutines-flows/SKILL.md +283 -0
  209. package/skills/kotlin-exposed-patterns/SKILL.md +718 -0
  210. package/skills/kotlin-ktor-patterns/SKILL.md +688 -0
  211. package/skills/kotlin-patterns/SKILL.md +710 -0
  212. package/skills/kotlin-testing/SKILL.md +823 -0
  213. package/skills/laravel-patterns/SKILL.md +414 -0
  214. package/skills/laravel-plugin-discovery/SKILL.md +228 -0
  215. package/skills/laravel-security/SKILL.md +284 -0
  216. package/skills/laravel-tdd/SKILL.md +282 -0
  217. package/skills/laravel-verification/SKILL.md +178 -0
  218. package/skills/lead-intelligence/SKILL.md +320 -0
  219. package/skills/lead-intelligence/agents/enrichment-agent.md +85 -0
  220. package/skills/lead-intelligence/agents/mutual-mapper.md +75 -0
  221. package/skills/lead-intelligence/agents/outreach-drafter.md +98 -0
  222. package/skills/lead-intelligence/agents/signal-scorer.md +60 -0
  223. package/skills/liquid-glass-design/SKILL.md +279 -0
  224. package/skills/llm-trading-agent-security/SKILL.md +146 -0
  225. package/skills/logistics-exception-management/SKILL.md +221 -0
  226. package/skills/make-interfaces-feel-better/SKILL.md +151 -0
  227. package/skills/manim-video/SKILL.md +88 -0
  228. package/skills/manim-video/assets/network_graph_scene.py +52 -0
  229. package/skills/market-research/SKILL.md +74 -0
  230. package/skills/mcp-server-patterns/SKILL.md +68 -0
  231. package/skills/messages-ops/SKILL.md +103 -0
  232. package/skills/mle-workflow/SKILL.md +345 -0
  233. package/skills/motion-advanced/SKILL.md +596 -0
  234. package/skills/motion-foundations/SKILL.md +299 -0
  235. package/skills/motion-patterns/SKILL.md +435 -0
  236. package/skills/motion-ui/SKILL.md +574 -0
  237. package/skills/mysql-patterns/SKILL.md +411 -0
  238. package/skills/nanoclaw-repl/SKILL.md +32 -0
  239. package/skills/nestjs-patterns/SKILL.md +229 -0
  240. package/skills/netmiko-ssh-automation/SKILL.md +173 -0
  241. package/skills/network-bgp-diagnostics/SKILL.md +167 -0
  242. package/skills/network-config-validation/SKILL.md +210 -0
  243. package/skills/network-interface-health/SKILL.md +152 -0
  244. package/skills/nextjs-turbopack/SKILL.md +43 -0
  245. package/skills/nodejs-keccak256/SKILL.md +102 -0
  246. package/skills/nutrient-document-processing/SKILL.md +166 -0
  247. package/skills/nuxt4-patterns/SKILL.md +99 -0
  248. package/skills/openclaw-persona-forge/SKILL.md +288 -0
  249. package/skills/openclaw-persona-forge/gacha.py +224 -0
  250. package/skills/openclaw-persona-forge/gacha.sh +5 -0
  251. package/skills/openclaw-persona-forge/references/avatar-style.md +124 -0
  252. package/skills/openclaw-persona-forge/references/boundary-rules.md +53 -0
  253. package/skills/openclaw-persona-forge/references/error-handling.md +53 -0
  254. package/skills/openclaw-persona-forge/references/identity-tension.md +48 -0
  255. package/skills/openclaw-persona-forge/references/naming-system.md +39 -0
  256. package/skills/openclaw-persona-forge/references/output-template.md +166 -0
  257. package/skills/opensource-pipeline/SKILL.md +254 -0
  258. package/skills/perl-patterns/SKILL.md +503 -0
  259. package/skills/perl-security/SKILL.md +502 -0
  260. package/skills/perl-testing/SKILL.md +474 -0
  261. package/skills/plan-orchestrate/SKILL.md +253 -0
  262. package/skills/plankton-code-quality/SKILL.md +236 -0
  263. package/skills/postgres-patterns/SKILL.md +146 -0
  264. package/skills/product-capability/SKILL.md +140 -0
  265. package/skills/product-lens/SKILL.md +91 -0
  266. package/skills/production-audit/SKILL.md +206 -0
  267. package/skills/production-scheduling/SKILL.md +237 -0
  268. package/skills/project-flow-ops/SKILL.md +110 -0
  269. package/skills/prompt-optimizer/SKILL.md +398 -0
  270. package/skills/python-patterns/SKILL.md +749 -0
  271. package/skills/python-testing/SKILL.md +815 -0
  272. package/skills/pytorch-patterns/SKILL.md +395 -0
  273. package/skills/quality-nonconformance/SKILL.md +259 -0
  274. package/skills/quarkus-patterns/SKILL.md +721 -0
  275. package/skills/quarkus-security/SKILL.md +466 -0
  276. package/skills/quarkus-tdd/SKILL.md +810 -0
  277. package/skills/quarkus-verification/SKILL.md +478 -0
  278. package/skills/ralphinho-rfc-pipeline/SKILL.md +66 -0
  279. package/skills/redis-patterns/SKILL.md +402 -0
  280. package/skills/regex-vs-llm-structured-text/SKILL.md +219 -0
  281. package/skills/remotion-video-creation/SKILL.md +43 -0
  282. package/skills/remotion-video-creation/rules/3d.md +86 -0
  283. package/skills/remotion-video-creation/rules/animations.md +29 -0
  284. package/skills/remotion-video-creation/rules/assets/charts-bar-chart.tsx +173 -0
  285. package/skills/remotion-video-creation/rules/assets/text-animations-typewriter.tsx +100 -0
  286. package/skills/remotion-video-creation/rules/assets/text-animations-word-highlight.tsx +108 -0
  287. package/skills/remotion-video-creation/rules/assets.md +78 -0
  288. package/skills/remotion-video-creation/rules/audio.md +172 -0
  289. package/skills/remotion-video-creation/rules/calculate-metadata.md +104 -0
  290. package/skills/remotion-video-creation/rules/can-decode.md +75 -0
  291. package/skills/remotion-video-creation/rules/charts.md +58 -0
  292. package/skills/remotion-video-creation/rules/compositions.md +146 -0
  293. package/skills/remotion-video-creation/rules/display-captions.md +126 -0
  294. package/skills/remotion-video-creation/rules/extract-frames.md +229 -0
  295. package/skills/remotion-video-creation/rules/fonts.md +152 -0
  296. package/skills/remotion-video-creation/rules/get-audio-duration.md +58 -0
  297. package/skills/remotion-video-creation/rules/get-video-dimensions.md +68 -0
  298. package/skills/remotion-video-creation/rules/get-video-duration.md +58 -0
  299. package/skills/remotion-video-creation/rules/gifs.md +138 -0
  300. package/skills/remotion-video-creation/rules/images.md +130 -0
  301. package/skills/remotion-video-creation/rules/import-srt-captions.md +67 -0
  302. package/skills/remotion-video-creation/rules/lottie.md +67 -0
  303. package/skills/remotion-video-creation/rules/measuring-dom-nodes.md +34 -0
  304. package/skills/remotion-video-creation/rules/measuring-text.md +143 -0
  305. package/skills/remotion-video-creation/rules/sequencing.md +106 -0
  306. package/skills/remotion-video-creation/rules/tailwind.md +11 -0
  307. package/skills/remotion-video-creation/rules/text-animations.md +20 -0
  308. package/skills/remotion-video-creation/rules/timing.md +179 -0
  309. package/skills/remotion-video-creation/rules/transcribe-captions.md +19 -0
  310. package/skills/remotion-video-creation/rules/transitions.md +122 -0
  311. package/skills/remotion-video-creation/rules/trimming.md +52 -0
  312. package/skills/remotion-video-creation/rules/videos.md +171 -0
  313. package/skills/repo-scan/SKILL.md +78 -0
  314. package/skills/research-ops/SKILL.md +111 -0
  315. package/skills/returns-reverse-logistics/SKILL.md +239 -0
  316. package/skills/rules-distill/SKILL.md +263 -0
  317. package/skills/rules-distill/scripts/scan-rules.sh +58 -0
  318. package/skills/rules-distill/scripts/scan-skills.sh +129 -0
  319. package/skills/rust-patterns/SKILL.md +498 -0
  320. package/skills/rust-testing/SKILL.md +499 -0
  321. package/skills/safety-guard/SKILL.md +74 -0
  322. package/skills/santa-method/SKILL.md +306 -0
  323. package/skills/scientific-db-pubmed-database/SKILL.md +175 -0
  324. package/skills/scientific-db-uspto-database/SKILL.md +177 -0
  325. package/skills/scientific-pkg-gget/SKILL.md +166 -0
  326. package/skills/scientific-thinking-literature-review/SKILL.md +192 -0
  327. package/skills/scientific-thinking-scholar-evaluation/SKILL.md +160 -0
  328. package/skills/search-first/SKILL.md +181 -0
  329. package/skills/security-bounty-hunter/SKILL.md +99 -0
  330. package/skills/security-review/SKILL.md +502 -0
  331. package/skills/security-review/cloud-infrastructure-security.md +361 -0
  332. package/skills/seo/SKILL.md +153 -0
  333. package/skills/skill-comply/SKILL.md +57 -0
  334. package/skills/skill-comply/fixtures/compliant_trace.jsonl +5 -0
  335. package/skills/skill-comply/fixtures/noncompliant_trace.jsonl +3 -0
  336. package/skills/skill-comply/fixtures/tdd_spec.yaml +44 -0
  337. package/skills/skill-comply/prompts/classifier.md +24 -0
  338. package/skills/skill-comply/prompts/scenario_generator.md +62 -0
  339. package/skills/skill-comply/prompts/spec_generator.md +42 -0
  340. package/skills/skill-comply/pyproject.toml +15 -0
  341. package/skills/skill-comply/scripts/__init__.py +0 -0
  342. package/skills/skill-comply/scripts/classifier.py +85 -0
  343. package/skills/skill-comply/scripts/grader.py +124 -0
  344. package/skills/skill-comply/scripts/parser.py +107 -0
  345. package/skills/skill-comply/scripts/report.py +170 -0
  346. package/skills/skill-comply/scripts/run.py +127 -0
  347. package/skills/skill-comply/scripts/runner.py +186 -0
  348. package/skills/skill-comply/scripts/scenario_generator.py +70 -0
  349. package/skills/skill-comply/scripts/spec_generator.py +72 -0
  350. package/skills/skill-comply/scripts/utils.py +13 -0
  351. package/skills/skill-comply/tests/test_grader.py +197 -0
  352. package/skills/skill-comply/tests/test_parser.py +90 -0
  353. package/skills/skill-comply/tests/test_runner.py +172 -0
  354. package/skills/skill-scout/SKILL.md +139 -0
  355. package/skills/skill-stocktake/SKILL.md +193 -0
  356. package/skills/skill-stocktake/scripts/quick-diff.sh +87 -0
  357. package/skills/skill-stocktake/scripts/save-results.sh +56 -0
  358. package/skills/skill-stocktake/scripts/scan.sh +170 -0
  359. package/skills/social-graph-ranker/SKILL.md +153 -0
  360. package/skills/springboot-patterns/SKILL.md +313 -0
  361. package/skills/springboot-security/SKILL.md +271 -0
  362. package/skills/springboot-tdd/SKILL.md +157 -0
  363. package/skills/springboot-verification/SKILL.md +230 -0
  364. package/skills/strategic-compact/SKILL.md +129 -0
  365. package/skills/strategic-compact/suggest-compact.sh +54 -0
  366. package/skills/swift-actor-persistence/SKILL.md +142 -0
  367. package/skills/swift-concurrency-6-2/SKILL.md +216 -0
  368. package/skills/swift-protocol-di-testing/SKILL.md +189 -0
  369. package/skills/swiftui-patterns/SKILL.md +259 -0
  370. package/skills/tdd-workflow/SKILL.md +462 -0
  371. package/skills/team-builder/SKILL.md +166 -0
  372. package/skills/terminal-ops/SKILL.md +108 -0
  373. package/skills/tinystruct-patterns/SKILL.md +130 -0
  374. package/skills/tinystruct-patterns/references/architecture.md +77 -0
  375. package/skills/tinystruct-patterns/references/data-handling.md +35 -0
  376. package/skills/tinystruct-patterns/references/routing.md +57 -0
  377. package/skills/tinystruct-patterns/references/system-usage.md +74 -0
  378. package/skills/tinystruct-patterns/references/testing.md +59 -0
  379. package/skills/token-budget-advisor/SKILL.md +133 -0
  380. package/skills/ui-demo/SKILL.md +464 -0
  381. package/skills/ui-to-vue/SKILL.md +134 -0
  382. package/skills/unified-notifications-ops/SKILL.md +186 -0
  383. package/skills/verification-loop/SKILL.md +125 -0
  384. package/skills/video-editing/SKILL.md +309 -0
  385. package/skills/videodb/SKILL.md +373 -0
  386. package/skills/videodb/reference/api-reference.md +550 -0
  387. package/skills/videodb/reference/capture-reference.md +407 -0
  388. package/skills/videodb/reference/capture.md +101 -0
  389. package/skills/videodb/reference/editor.md +443 -0
  390. package/skills/videodb/reference/generative.md +331 -0
  391. package/skills/videodb/reference/rtstream-reference.md +564 -0
  392. package/skills/videodb/reference/rtstream.md +65 -0
  393. package/skills/videodb/reference/search.md +230 -0
  394. package/skills/videodb/reference/streaming.md +406 -0
  395. package/skills/videodb/reference/use-cases.md +118 -0
  396. package/skills/videodb/scripts/ws_listener.py +282 -0
  397. package/skills/visa-doc-translate/README.md +86 -0
  398. package/skills/visa-doc-translate/SKILL.md +117 -0
  399. package/skills/vite-patterns/SKILL.md +448 -0
  400. package/skills/windows-desktop-e2e/SKILL.md +787 -0
  401. package/skills/workspace-surface-audit/SKILL.md +124 -0
  402. package/skills/x-api/SKILL.md +233 -0
@@ -0,0 +1,1018 @@
1
+ """Tests for continuous-learning-v2 instinct-cli.py
2
+
3
+ Covers:
4
+ - parse_instinct_file() — content preservation, edge cases
5
+ - _validate_file_path() — path traversal blocking
6
+ - detect_project() — project detection with mocked git/env
7
+ - load_all_instincts() — loading from project + global dirs, dedup
8
+ - _load_instincts_from_dir() — directory scanning
9
+ - cmd_projects() — listing projects from registry
10
+ - cmd_status() — status display
11
+ - _promote_specific() — single instinct promotion
12
+ - _promote_auto() — auto-promotion across projects
13
+ """
14
+
15
+ import importlib.util
16
+ import io
17
+ import json
18
+ import os
19
+ import sys
20
+ from pathlib import Path
21
+ from types import SimpleNamespace
22
+ from unittest import mock
23
+
24
+ import pytest
25
+
26
+ # Load instinct-cli.py (hyphenated filename requires importlib)
27
+ _spec = importlib.util.spec_from_file_location(
28
+ "instinct_cli",
29
+ os.path.join(os.path.dirname(__file__), "instinct-cli.py"),
30
+ )
31
+ _mod = importlib.util.module_from_spec(_spec)
32
+ _spec.loader.exec_module(_mod)
33
+
34
+ parse_instinct_file = _mod.parse_instinct_file
35
+ _validate_file_path = _mod._validate_file_path
36
+ detect_project = _mod.detect_project
37
+ load_all_instincts = _mod.load_all_instincts
38
+ load_project_only_instincts = _mod.load_project_only_instincts
39
+ _load_instincts_from_dir = _mod._load_instincts_from_dir
40
+ cmd_status = _mod.cmd_status
41
+ cmd_projects = _mod.cmd_projects
42
+ _promote_specific = _mod._promote_specific
43
+ _promote_auto = _mod._promote_auto
44
+ _find_cross_project_instincts = _mod._find_cross_project_instincts
45
+ load_registry = _mod.load_registry
46
+ _validate_instinct_id = _mod._validate_instinct_id
47
+ _update_registry = _mod._update_registry
48
+ _confidence_bar = _mod._confidence_bar
49
+
50
+
51
+ # ─────────────────────────────────────────────
52
+ # Fixtures
53
+ # ─────────────────────────────────────────────
54
+
55
+ SAMPLE_INSTINCT_YAML = """\
56
+ ---
57
+ id: test-instinct
58
+ trigger: "when writing tests"
59
+ confidence: 0.8
60
+ domain: testing
61
+ scope: project
62
+ ---
63
+
64
+ ## Action
65
+ Always write tests first.
66
+
67
+ ## Evidence
68
+ TDD leads to better design.
69
+ """
70
+
71
+ SAMPLE_GLOBAL_INSTINCT_YAML = """\
72
+ ---
73
+ id: global-instinct
74
+ trigger: "always"
75
+ confidence: 0.9
76
+ domain: security
77
+ scope: global
78
+ ---
79
+
80
+ ## Action
81
+ Validate all user input.
82
+ """
83
+
84
+
85
+ @pytest.fixture
86
+ def project_tree(tmp_path):
87
+ """Create a realistic project directory tree for testing."""
88
+ homunculus = tmp_path / ".claude" / "homunculus"
89
+ projects_dir = homunculus / "projects"
90
+ global_personal = homunculus / "instincts" / "personal"
91
+ global_inherited = homunculus / "instincts" / "inherited"
92
+ global_evolved = homunculus / "evolved"
93
+
94
+ for d in [
95
+ global_personal, global_inherited,
96
+ global_evolved / "skills", global_evolved / "commands", global_evolved / "agents",
97
+ projects_dir,
98
+ ]:
99
+ d.mkdir(parents=True, exist_ok=True)
100
+
101
+ return {
102
+ "root": tmp_path,
103
+ "homunculus": homunculus,
104
+ "projects_dir": projects_dir,
105
+ "global_personal": global_personal,
106
+ "global_inherited": global_inherited,
107
+ "global_evolved": global_evolved,
108
+ "registry_file": homunculus / "projects.json",
109
+ }
110
+
111
+
112
+ @pytest.fixture
113
+ def patch_globals(project_tree, monkeypatch):
114
+ """Patch module-level globals to use tmp_path-based directories."""
115
+ monkeypatch.setattr(_mod, "HOMUNCULUS_DIR", project_tree["homunculus"])
116
+ monkeypatch.setattr(_mod, "PROJECTS_DIR", project_tree["projects_dir"])
117
+ monkeypatch.setattr(_mod, "REGISTRY_FILE", project_tree["registry_file"])
118
+ monkeypatch.setattr(_mod, "GLOBAL_PERSONAL_DIR", project_tree["global_personal"])
119
+ monkeypatch.setattr(_mod, "GLOBAL_INHERITED_DIR", project_tree["global_inherited"])
120
+ monkeypatch.setattr(_mod, "GLOBAL_EVOLVED_DIR", project_tree["global_evolved"])
121
+ monkeypatch.setattr(_mod, "GLOBAL_OBSERVATIONS_FILE", project_tree["homunculus"] / "observations.jsonl")
122
+ return project_tree
123
+
124
+
125
+ def _make_project(tree, pid="abc123", pname="test-project"):
126
+ """Create project directory structure and return a project dict."""
127
+ project_dir = tree["projects_dir"] / pid
128
+ personal_dir = project_dir / "instincts" / "personal"
129
+ inherited_dir = project_dir / "instincts" / "inherited"
130
+ for d in [personal_dir, inherited_dir,
131
+ project_dir / "evolved" / "skills",
132
+ project_dir / "evolved" / "commands",
133
+ project_dir / "evolved" / "agents",
134
+ project_dir / "observations.archive"]:
135
+ d.mkdir(parents=True, exist_ok=True)
136
+
137
+ return {
138
+ "id": pid,
139
+ "name": pname,
140
+ "root": str(tree["root"] / "fake-repo"),
141
+ "remote": "https://github.com/test/test-project.git",
142
+ "project_dir": project_dir,
143
+ "instincts_personal": personal_dir,
144
+ "instincts_inherited": inherited_dir,
145
+ "evolved_dir": project_dir / "evolved",
146
+ "observations_file": project_dir / "observations.jsonl",
147
+ }
148
+
149
+
150
+ # ─────────────────────────────────────────────
151
+ # parse_instinct_file tests
152
+ # ─────────────────────────────────────────────
153
+
154
+ MULTI_SECTION = """\
155
+ ---
156
+ id: instinct-a
157
+ trigger: "when coding"
158
+ confidence: 0.9
159
+ domain: general
160
+ ---
161
+
162
+ ## Action
163
+ Do thing A.
164
+
165
+ ## Examples
166
+ - Example A1
167
+
168
+ ---
169
+ id: instinct-b
170
+ trigger: "when testing"
171
+ confidence: 0.7
172
+ domain: testing
173
+ ---
174
+
175
+ ## Action
176
+ Do thing B.
177
+ """
178
+
179
+
180
+ def test_multiple_instincts_preserve_content():
181
+ result = parse_instinct_file(MULTI_SECTION)
182
+ assert len(result) == 2
183
+ assert "Do thing A." in result[0]["content"]
184
+ assert "Example A1" in result[0]["content"]
185
+ assert "Do thing B." in result[1]["content"]
186
+
187
+
188
+ def test_single_instinct_preserves_content():
189
+ content = """\
190
+ ---
191
+ id: solo
192
+ trigger: "when reviewing"
193
+ confidence: 0.8
194
+ domain: review
195
+ ---
196
+
197
+ ## Action
198
+ Check for security issues.
199
+
200
+ ## Evidence
201
+ Prevents vulnerabilities.
202
+ """
203
+ result = parse_instinct_file(content)
204
+ assert len(result) == 1
205
+ assert "Check for security issues." in result[0]["content"]
206
+ assert "Prevents vulnerabilities." in result[0]["content"]
207
+
208
+
209
+ def test_empty_content_no_error():
210
+ content = """\
211
+ ---
212
+ id: empty
213
+ trigger: "placeholder"
214
+ confidence: 0.5
215
+ domain: general
216
+ ---
217
+ """
218
+ result = parse_instinct_file(content)
219
+ assert len(result) == 1
220
+ assert result[0]["content"] == ""
221
+
222
+
223
+ def test_parse_no_id_skipped():
224
+ """Instincts without an 'id' field should be silently dropped."""
225
+ content = """\
226
+ ---
227
+ trigger: "when doing nothing"
228
+ confidence: 0.5
229
+ ---
230
+
231
+ No id here.
232
+ """
233
+ result = parse_instinct_file(content)
234
+ assert len(result) == 0
235
+
236
+
237
+ def test_parse_confidence_is_float():
238
+ content = """\
239
+ ---
240
+ id: float-check
241
+ trigger: "when parsing"
242
+ confidence: 0.42
243
+ domain: general
244
+ ---
245
+
246
+ Body.
247
+ """
248
+ result = parse_instinct_file(content)
249
+ assert isinstance(result[0]["confidence"], float)
250
+ assert result[0]["confidence"] == pytest.approx(0.42)
251
+
252
+
253
+ def test_parse_trigger_strips_quotes():
254
+ content = """\
255
+ ---
256
+ id: quote-check
257
+ trigger: "when quoting"
258
+ confidence: 0.5
259
+ domain: general
260
+ ---
261
+
262
+ Body.
263
+ """
264
+ result = parse_instinct_file(content)
265
+ assert result[0]["trigger"] == "when quoting"
266
+
267
+
268
+ def test_parse_empty_string():
269
+ result = parse_instinct_file("")
270
+ assert result == []
271
+
272
+
273
+ def test_parse_garbage_input():
274
+ result = parse_instinct_file("this is not yaml at all\nno frontmatter here")
275
+ assert result == []
276
+
277
+
278
+ # ─────────────────────────────────────────────
279
+ # _validate_file_path tests
280
+ # ─────────────────────────────────────────────
281
+
282
+ def test_validate_normal_path(tmp_path):
283
+ test_file = tmp_path / "test.yaml"
284
+ test_file.write_text("hello")
285
+ result = _validate_file_path(str(test_file), must_exist=True)
286
+ assert result == test_file.resolve()
287
+
288
+
289
+ def test_validate_rejects_etc():
290
+ with pytest.raises(ValueError, match="system directory"):
291
+ _validate_file_path("/etc/passwd")
292
+
293
+
294
+ def test_validate_rejects_var_log():
295
+ with pytest.raises(ValueError, match="system directory"):
296
+ _validate_file_path("/var/log/syslog")
297
+
298
+
299
+ def test_validate_rejects_usr():
300
+ with pytest.raises(ValueError, match="system directory"):
301
+ _validate_file_path("/usr/local/bin/foo")
302
+
303
+
304
+ def test_validate_rejects_proc():
305
+ with pytest.raises(ValueError, match="system directory"):
306
+ _validate_file_path("/proc/self/status")
307
+
308
+
309
+ def test_validate_must_exist_fails(tmp_path):
310
+ with pytest.raises(ValueError, match="does not exist"):
311
+ _validate_file_path(str(tmp_path / "nonexistent.yaml"), must_exist=True)
312
+
313
+
314
+ def test_validate_home_expansion(tmp_path):
315
+ """Tilde expansion should work."""
316
+ result = _validate_file_path("~/test.yaml")
317
+ assert str(result).startswith(str(Path.home()))
318
+
319
+
320
+ def test_validate_relative_path(tmp_path, monkeypatch):
321
+ """Relative paths should be resolved."""
322
+ monkeypatch.chdir(tmp_path)
323
+ test_file = tmp_path / "rel.yaml"
324
+ test_file.write_text("content")
325
+ result = _validate_file_path("rel.yaml", must_exist=True)
326
+ assert result == test_file.resolve()
327
+
328
+
329
+ # ─────────────────────────────────────────────
330
+ # detect_project tests
331
+ # ─────────────────────────────────────────────
332
+
333
+ def test_detect_project_global_fallback(patch_globals, monkeypatch):
334
+ """When no git and no env var, should return global project."""
335
+ monkeypatch.delenv("CLAUDE_PROJECT_DIR", raising=False)
336
+
337
+ # Mock subprocess.run to simulate git not available
338
+ def mock_run(*args, **kwargs):
339
+ raise FileNotFoundError("git not found")
340
+
341
+ monkeypatch.setattr("subprocess.run", mock_run)
342
+
343
+ project = detect_project()
344
+ assert project["id"] == "global"
345
+ assert project["name"] == "global"
346
+
347
+
348
+ def test_detect_project_from_env(patch_globals, monkeypatch, tmp_path):
349
+ """CLAUDE_PROJECT_DIR env var should be used as project root."""
350
+ fake_repo = tmp_path / "my-repo"
351
+ fake_repo.mkdir()
352
+ monkeypatch.setenv("CLAUDE_PROJECT_DIR", str(fake_repo))
353
+
354
+ # Mock git remote to return a URL
355
+ def mock_run(cmd, **kwargs):
356
+ if "rev-parse" in cmd:
357
+ return SimpleNamespace(returncode=0, stdout=str(fake_repo) + "\n", stderr="")
358
+ if "get-url" in cmd:
359
+ return SimpleNamespace(returncode=0, stdout="https://github.com/test/my-repo.git\n", stderr="")
360
+ return SimpleNamespace(returncode=1, stdout="", stderr="")
361
+
362
+ monkeypatch.setattr("subprocess.run", mock_run)
363
+
364
+ project = detect_project()
365
+ assert project["id"] != "global"
366
+ assert project["name"] == "my-repo"
367
+
368
+
369
+ def test_detect_project_git_timeout(patch_globals, monkeypatch):
370
+ """Git timeout should fall through to global."""
371
+ monkeypatch.delenv("CLAUDE_PROJECT_DIR", raising=False)
372
+ import subprocess as sp
373
+
374
+ def mock_run(cmd, **kwargs):
375
+ raise sp.TimeoutExpired(cmd, 5)
376
+
377
+ monkeypatch.setattr("subprocess.run", mock_run)
378
+
379
+ project = detect_project()
380
+ assert project["id"] == "global"
381
+
382
+
383
+ def test_detect_project_creates_directories(patch_globals, monkeypatch, tmp_path):
384
+ """detect_project should create the project dir structure."""
385
+ fake_repo = tmp_path / "structured-repo"
386
+ fake_repo.mkdir()
387
+ monkeypatch.setenv("CLAUDE_PROJECT_DIR", str(fake_repo))
388
+
389
+ def mock_run(cmd, **kwargs):
390
+ if "rev-parse" in cmd:
391
+ return SimpleNamespace(returncode=0, stdout=str(fake_repo) + "\n", stderr="")
392
+ if "get-url" in cmd:
393
+ return SimpleNamespace(returncode=1, stdout="", stderr="no remote")
394
+ return SimpleNamespace(returncode=1, stdout="", stderr="")
395
+
396
+ monkeypatch.setattr("subprocess.run", mock_run)
397
+
398
+ project = detect_project()
399
+ assert project["instincts_personal"].exists()
400
+ assert project["instincts_inherited"].exists()
401
+ assert (project["evolved_dir"] / "skills").exists()
402
+
403
+
404
+ # ─────────────────────────────────────────────
405
+ # _load_instincts_from_dir tests
406
+ # ─────────────────────────────────────────────
407
+
408
+ def test_load_from_empty_dir(tmp_path):
409
+ result = _load_instincts_from_dir(tmp_path, "personal", "project")
410
+ assert result == []
411
+
412
+
413
+ def test_load_from_nonexistent_dir(tmp_path):
414
+ result = _load_instincts_from_dir(tmp_path / "does-not-exist", "personal", "project")
415
+ assert result == []
416
+
417
+
418
+ def test_load_annotates_metadata(tmp_path):
419
+ """Loaded instincts should have _source_file, _source_type, _scope_label."""
420
+ yaml_file = tmp_path / "test.yaml"
421
+ yaml_file.write_text(SAMPLE_INSTINCT_YAML)
422
+
423
+ result = _load_instincts_from_dir(tmp_path, "personal", "project")
424
+ assert len(result) == 1
425
+ assert result[0]["_source_file"] == str(yaml_file)
426
+ assert result[0]["_source_type"] == "personal"
427
+ assert result[0]["_scope_label"] == "project"
428
+
429
+
430
+ def test_load_defaults_scope_from_label(tmp_path):
431
+ """If an instinct has no 'scope' in frontmatter, it should default to scope_label."""
432
+ no_scope_yaml = """\
433
+ ---
434
+ id: no-scope
435
+ trigger: "test"
436
+ confidence: 0.5
437
+ domain: general
438
+ ---
439
+
440
+ Body.
441
+ """
442
+ (tmp_path / "no-scope.yaml").write_text(no_scope_yaml)
443
+ result = _load_instincts_from_dir(tmp_path, "inherited", "global")
444
+ assert result[0]["scope"] == "global"
445
+
446
+
447
+ def test_load_preserves_explicit_scope(tmp_path):
448
+ """If frontmatter has explicit scope, it should be preserved."""
449
+ yaml_file = tmp_path / "test.yaml"
450
+ yaml_file.write_text(SAMPLE_INSTINCT_YAML)
451
+
452
+ result = _load_instincts_from_dir(tmp_path, "personal", "global")
453
+ # Frontmatter says scope: project, scope_label is global
454
+ # The explicit scope should be preserved (not overwritten)
455
+ assert result[0]["scope"] == "project"
456
+
457
+
458
+ def test_load_handles_corrupt_file(tmp_path, capsys):
459
+ """Corrupt YAML files should be warned about but not crash."""
460
+ # A file that will cause parse_instinct_file to return empty
461
+ (tmp_path / "good.yaml").write_text(SAMPLE_INSTINCT_YAML)
462
+ (tmp_path / "bad.yaml").write_text("not yaml\nno frontmatter")
463
+
464
+ result = _load_instincts_from_dir(tmp_path, "personal", "project")
465
+ # bad.yaml has no valid instincts (no id), so only good.yaml contributes
466
+ assert len(result) == 1
467
+ assert result[0]["id"] == "test-instinct"
468
+
469
+
470
+ def test_load_supports_yml_extension(tmp_path):
471
+ yml_file = tmp_path / "test.yml"
472
+ yml_file.write_text(SAMPLE_INSTINCT_YAML)
473
+
474
+ result = _load_instincts_from_dir(tmp_path, "personal", "project")
475
+ ids = {i["id"] for i in result}
476
+ assert "test-instinct" in ids
477
+
478
+
479
+ def test_load_supports_md_extension(tmp_path):
480
+ md_file = tmp_path / "legacy-instinct.md"
481
+ md_file.write_text(SAMPLE_INSTINCT_YAML)
482
+
483
+ result = _load_instincts_from_dir(tmp_path, "personal", "project")
484
+ ids = {i["id"] for i in result}
485
+ assert "test-instinct" in ids
486
+
487
+
488
+ def test_load_instincts_from_dir_uses_utf8_encoding(tmp_path, monkeypatch):
489
+ yaml_file = tmp_path / "test.yaml"
490
+ yaml_file.write_text("placeholder")
491
+ calls = []
492
+
493
+ def fake_read_text(self, *args, **kwargs):
494
+ calls.append(kwargs.get("encoding"))
495
+ return SAMPLE_INSTINCT_YAML
496
+
497
+ monkeypatch.setattr(Path, "read_text", fake_read_text)
498
+ result = _load_instincts_from_dir(tmp_path, "personal", "project")
499
+ assert result[0]["id"] == "test-instinct"
500
+ assert calls == ["utf-8"]
501
+
502
+
503
+ # ─────────────────────────────────────────────
504
+ # load_all_instincts tests
505
+ # ─────────────────────────────────────────────
506
+
507
+ def test_load_all_project_and_global(patch_globals):
508
+ """Should load from both project and global directories."""
509
+ tree = patch_globals
510
+ project = _make_project(tree)
511
+
512
+ # Write a project instinct
513
+ (project["instincts_personal"] / "proj.yaml").write_text(SAMPLE_INSTINCT_YAML)
514
+ # Write a global instinct
515
+ (tree["global_personal"] / "glob.yaml").write_text(SAMPLE_GLOBAL_INSTINCT_YAML)
516
+
517
+ result = load_all_instincts(project)
518
+ ids = {i["id"] for i in result}
519
+ assert "test-instinct" in ids
520
+ assert "global-instinct" in ids
521
+
522
+
523
+ def test_load_all_project_overrides_global(patch_globals):
524
+ """When project and global have same ID, project wins."""
525
+ tree = patch_globals
526
+ project = _make_project(tree)
527
+
528
+ # Same ID but different confidence
529
+ proj_yaml = SAMPLE_INSTINCT_YAML.replace("id: test-instinct", "id: shared-id")
530
+ proj_yaml = proj_yaml.replace("confidence: 0.8", "confidence: 0.9")
531
+ glob_yaml = SAMPLE_GLOBAL_INSTINCT_YAML.replace("id: global-instinct", "id: shared-id")
532
+ glob_yaml = glob_yaml.replace("confidence: 0.9", "confidence: 0.3")
533
+
534
+ (project["instincts_personal"] / "shared.yaml").write_text(proj_yaml)
535
+ (tree["global_personal"] / "shared.yaml").write_text(glob_yaml)
536
+
537
+ result = load_all_instincts(project)
538
+ shared = [i for i in result if i["id"] == "shared-id"]
539
+ assert len(shared) == 1
540
+ assert shared[0]["_scope_label"] == "project"
541
+ assert shared[0]["confidence"] == 0.9
542
+
543
+
544
+ def test_load_all_global_only(patch_globals):
545
+ """Global project should only load global instincts."""
546
+ tree = patch_globals
547
+ (tree["global_personal"] / "glob.yaml").write_text(SAMPLE_GLOBAL_INSTINCT_YAML)
548
+
549
+ global_project = {
550
+ "id": "global",
551
+ "name": "global",
552
+ "root": "",
553
+ "project_dir": tree["homunculus"],
554
+ "instincts_personal": tree["global_personal"],
555
+ "instincts_inherited": tree["global_inherited"],
556
+ "evolved_dir": tree["global_evolved"],
557
+ "observations_file": tree["homunculus"] / "observations.jsonl",
558
+ }
559
+
560
+ result = load_all_instincts(global_project)
561
+ assert len(result) == 1
562
+ assert result[0]["id"] == "global-instinct"
563
+
564
+
565
+ def test_load_project_only_excludes_global(patch_globals):
566
+ """load_project_only_instincts should NOT include global instincts."""
567
+ tree = patch_globals
568
+ project = _make_project(tree)
569
+
570
+ (project["instincts_personal"] / "proj.yaml").write_text(SAMPLE_INSTINCT_YAML)
571
+ (tree["global_personal"] / "glob.yaml").write_text(SAMPLE_GLOBAL_INSTINCT_YAML)
572
+
573
+ result = load_project_only_instincts(project)
574
+ ids = {i["id"] for i in result}
575
+ assert "test-instinct" in ids
576
+ assert "global-instinct" not in ids
577
+
578
+
579
+ def test_load_project_only_global_fallback_loads_global(patch_globals):
580
+ """Global fallback should return global instincts for project-only queries."""
581
+ tree = patch_globals
582
+ (tree["global_personal"] / "glob.yaml").write_text(SAMPLE_GLOBAL_INSTINCT_YAML)
583
+
584
+ global_project = {
585
+ "id": "global",
586
+ "name": "global",
587
+ "root": "",
588
+ "project_dir": tree["homunculus"],
589
+ "instincts_personal": tree["global_personal"],
590
+ "instincts_inherited": tree["global_inherited"],
591
+ "evolved_dir": tree["global_evolved"],
592
+ "observations_file": tree["homunculus"] / "observations.jsonl",
593
+ }
594
+
595
+ result = load_project_only_instincts(global_project)
596
+ assert len(result) == 1
597
+ assert result[0]["id"] == "global-instinct"
598
+
599
+
600
+ def test_load_all_empty(patch_globals):
601
+ """No instincts at all should return empty list."""
602
+ tree = patch_globals
603
+ project = _make_project(tree)
604
+
605
+ result = load_all_instincts(project)
606
+ assert result == []
607
+
608
+
609
+ # ─────────────────────────────────────────────
610
+ # cmd_status tests
611
+ # ─────────────────────────────────────────────
612
+
613
+ def test_cmd_status_no_instincts(patch_globals, monkeypatch, capsys):
614
+ """Status with no instincts should print fallback message."""
615
+ tree = patch_globals
616
+ project = _make_project(tree)
617
+ monkeypatch.setattr(_mod, "detect_project", lambda: project)
618
+
619
+ args = SimpleNamespace()
620
+ ret = cmd_status(args)
621
+ assert ret == 0
622
+ out = capsys.readouterr().out
623
+ assert "No instincts found." in out
624
+
625
+
626
+ def test_cmd_status_with_instincts(patch_globals, monkeypatch, capsys):
627
+ """Status should show project and global instinct counts."""
628
+ tree = patch_globals
629
+ project = _make_project(tree)
630
+ monkeypatch.setattr(_mod, "detect_project", lambda: project)
631
+
632
+ (project["instincts_personal"] / "proj.yaml").write_text(SAMPLE_INSTINCT_YAML)
633
+ (tree["global_personal"] / "glob.yaml").write_text(SAMPLE_GLOBAL_INSTINCT_YAML)
634
+
635
+ args = SimpleNamespace()
636
+ ret = cmd_status(args)
637
+ assert ret == 0
638
+ out = capsys.readouterr().out
639
+ assert "INSTINCT STATUS" in out
640
+ assert "Project instincts: 1" in out
641
+ assert "Global instincts: 1" in out
642
+ assert "PROJECT-SCOPED" in out
643
+ assert "GLOBAL" in out
644
+
645
+
646
+ def test_confidence_bar_uses_unicode_when_supported():
647
+ """Confidence bars should retain block glyphs on UTF-8 streams."""
648
+ stream = SimpleNamespace(encoding="utf-8")
649
+ assert _confidence_bar(0.8, stream=stream) == "\u2588" * 8 + "\u2591" * 2
650
+
651
+
652
+ def test_confidence_bar_uses_ascii_when_stream_rejects_block_glyphs():
653
+ """Windows cp1252 streams cannot encode block glyphs."""
654
+ stream = SimpleNamespace(encoding="cp1252")
655
+ assert _confidence_bar(0.8, stream=stream) == "########.."
656
+
657
+
658
+ def test_print_instincts_by_domain_is_cp1252_safe(monkeypatch):
659
+ """Status rendering should not crash on Windows cp1252 stdout."""
660
+ raw = io.BytesIO()
661
+ stream = io.TextIOWrapper(raw, encoding="cp1252")
662
+ monkeypatch.setattr(_mod.sys, "stdout", stream)
663
+
664
+ _mod._print_instincts_by_domain([{
665
+ "id": "windows-safe",
666
+ "trigger": "when stdout uses cp1252",
667
+ "confidence": 0.8,
668
+ "domain": "platform",
669
+ "scope": "project",
670
+ }])
671
+
672
+ stream.flush()
673
+ out = raw.getvalue().decode("cp1252")
674
+ assert "########.." in out
675
+ assert "\u2588" not in out
676
+ assert "\u2591" not in out
677
+
678
+
679
+ def test_cmd_status_returns_int(patch_globals, monkeypatch):
680
+ """cmd_status should always return an int."""
681
+ tree = patch_globals
682
+ project = _make_project(tree)
683
+ monkeypatch.setattr(_mod, "detect_project", lambda: project)
684
+
685
+ args = SimpleNamespace()
686
+ ret = cmd_status(args)
687
+ assert isinstance(ret, int)
688
+
689
+
690
+ # ─────────────────────────────────────────────
691
+ # cmd_projects tests
692
+ # ─────────────────────────────────────────────
693
+
694
+ def test_cmd_projects_empty_registry(patch_globals, capsys):
695
+ """No projects should print helpful message."""
696
+ args = SimpleNamespace()
697
+ ret = cmd_projects(args)
698
+ assert ret == 0
699
+ out = capsys.readouterr().out
700
+ assert "No projects registered yet." in out
701
+
702
+
703
+ def test_cmd_projects_with_registry(patch_globals, capsys):
704
+ """Should list projects from registry."""
705
+ tree = patch_globals
706
+
707
+ # Create a project dir with instincts
708
+ pid = "test123abc"
709
+ project = _make_project(tree, pid=pid, pname="my-app")
710
+ (project["instincts_personal"] / "inst.yaml").write_text(SAMPLE_INSTINCT_YAML)
711
+
712
+ # Write registry
713
+ registry = {
714
+ pid: {
715
+ "name": "my-app",
716
+ "root": "/home/user/my-app",
717
+ "remote": "https://github.com/user/my-app.git",
718
+ "last_seen": "2025-01-15T12:00:00Z",
719
+ }
720
+ }
721
+ tree["registry_file"].write_text(json.dumps(registry))
722
+
723
+ args = SimpleNamespace()
724
+ ret = cmd_projects(args)
725
+ assert ret == 0
726
+ out = capsys.readouterr().out
727
+ assert "my-app" in out
728
+ assert pid in out
729
+ assert "1 personal" in out
730
+
731
+
732
+ # ─────────────────────────────────────────────
733
+ # _promote_specific tests
734
+ # ─────────────────────────────────────────────
735
+
736
+ def test_promote_specific_not_found(patch_globals, capsys):
737
+ """Promoting nonexistent instinct should fail."""
738
+ tree = patch_globals
739
+ project = _make_project(tree)
740
+
741
+ ret = _promote_specific(project, "nonexistent", force=True)
742
+ assert ret == 1
743
+ out = capsys.readouterr().out
744
+ assert "not found" in out
745
+
746
+
747
+ def test_promote_specific_rejects_invalid_id(patch_globals, capsys):
748
+ """Path-like instinct IDs should be rejected before file writes."""
749
+ tree = patch_globals
750
+ project = _make_project(tree)
751
+
752
+ ret = _promote_specific(project, "../escape", force=True)
753
+ assert ret == 1
754
+ err = capsys.readouterr().err
755
+ assert "Invalid instinct ID" in err
756
+
757
+
758
+ def test_promote_specific_already_global(patch_globals, capsys):
759
+ """Promoting an instinct that already exists globally should fail."""
760
+ tree = patch_globals
761
+ project = _make_project(tree)
762
+
763
+ # Write same-id instinct in both project and global
764
+ (project["instincts_personal"] / "shared.yaml").write_text(SAMPLE_INSTINCT_YAML)
765
+ global_yaml = SAMPLE_INSTINCT_YAML # same id: test-instinct
766
+ (tree["global_personal"] / "shared.yaml").write_text(global_yaml)
767
+
768
+ ret = _promote_specific(project, "test-instinct", force=True)
769
+ assert ret == 1
770
+ out = capsys.readouterr().out
771
+ assert "already exists in global" in out
772
+
773
+
774
+ def test_promote_specific_success(patch_globals, capsys):
775
+ """Promote a project instinct to global with --force."""
776
+ tree = patch_globals
777
+ project = _make_project(tree)
778
+
779
+ (project["instincts_personal"] / "inst.yaml").write_text(SAMPLE_INSTINCT_YAML)
780
+
781
+ ret = _promote_specific(project, "test-instinct", force=True)
782
+ assert ret == 0
783
+ out = capsys.readouterr().out
784
+ assert "Promoted" in out
785
+
786
+ # Verify file was created in global dir
787
+ promoted_file = tree["global_personal"] / "test-instinct.yaml"
788
+ assert promoted_file.exists()
789
+ content = promoted_file.read_text()
790
+ assert "scope: global" in content
791
+ assert "promoted_from: abc123" in content
792
+
793
+
794
+ # ─────────────────────────────────────────────
795
+ # _promote_auto tests
796
+ # ─────────────────────────────────────────────
797
+
798
+ def test_promote_auto_no_candidates(patch_globals, capsys):
799
+ """Auto-promote with no cross-project instincts should say so."""
800
+ tree = patch_globals
801
+ project = _make_project(tree)
802
+
803
+ # Empty registry
804
+ tree["registry_file"].write_text("{}")
805
+
806
+ ret = _promote_auto(project, force=True, dry_run=False)
807
+ assert ret == 0
808
+ out = capsys.readouterr().out
809
+ assert "No instincts qualify" in out
810
+
811
+
812
+ def test_promote_auto_dry_run(patch_globals, capsys):
813
+ """Dry run should list candidates but not write files."""
814
+ tree = patch_globals
815
+
816
+ # Create two projects with the same high-confidence instinct
817
+ p1 = _make_project(tree, pid="proj1", pname="project-one")
818
+ p2 = _make_project(tree, pid="proj2", pname="project-two")
819
+
820
+ high_conf_yaml = """\
821
+ ---
822
+ id: cross-project-instinct
823
+ trigger: "when reviewing"
824
+ confidence: 0.95
825
+ domain: security
826
+ scope: project
827
+ ---
828
+
829
+ ## Action
830
+ Always review for injection.
831
+ """
832
+ (p1["instincts_personal"] / "cross.yaml").write_text(high_conf_yaml)
833
+ (p2["instincts_personal"] / "cross.yaml").write_text(high_conf_yaml)
834
+
835
+ # Write registry
836
+ registry = {
837
+ "proj1": {"name": "project-one", "root": "/a", "remote": "", "last_seen": "2025-01-01T00:00:00Z"},
838
+ "proj2": {"name": "project-two", "root": "/b", "remote": "", "last_seen": "2025-01-01T00:00:00Z"},
839
+ }
840
+ tree["registry_file"].write_text(json.dumps(registry))
841
+
842
+ project = p1
843
+ ret = _promote_auto(project, force=True, dry_run=True)
844
+ assert ret == 0
845
+ out = capsys.readouterr().out
846
+ assert "DRY RUN" in out
847
+ assert "cross-project-instinct" in out
848
+
849
+ # Verify no file was created
850
+ assert not (tree["global_personal"] / "cross-project-instinct.yaml").exists()
851
+
852
+
853
+ def test_promote_auto_writes_file(patch_globals, capsys):
854
+ """Auto-promote with force should write global instinct file."""
855
+ tree = patch_globals
856
+
857
+ p1 = _make_project(tree, pid="proj1", pname="project-one")
858
+ p2 = _make_project(tree, pid="proj2", pname="project-two")
859
+
860
+ high_conf_yaml = """\
861
+ ---
862
+ id: universal-pattern
863
+ trigger: "when coding"
864
+ confidence: 0.85
865
+ domain: general
866
+ scope: project
867
+ ---
868
+
869
+ ## Action
870
+ Use descriptive variable names.
871
+ """
872
+ (p1["instincts_personal"] / "uni.yaml").write_text(high_conf_yaml)
873
+ (p2["instincts_personal"] / "uni.yaml").write_text(high_conf_yaml)
874
+
875
+ registry = {
876
+ "proj1": {"name": "project-one", "root": "/a", "remote": "", "last_seen": "2025-01-01T00:00:00Z"},
877
+ "proj2": {"name": "project-two", "root": "/b", "remote": "", "last_seen": "2025-01-01T00:00:00Z"},
878
+ }
879
+ tree["registry_file"].write_text(json.dumps(registry))
880
+
881
+ ret = _promote_auto(p1, force=True, dry_run=False)
882
+ assert ret == 0
883
+
884
+ promoted = tree["global_personal"] / "universal-pattern.yaml"
885
+ assert promoted.exists()
886
+ content = promoted.read_text()
887
+ assert "scope: global" in content
888
+ assert "auto-promoted" in content
889
+
890
+
891
+ def test_promote_auto_skips_invalid_id(patch_globals, capsys):
892
+ tree = patch_globals
893
+
894
+ p1 = _make_project(tree, pid="proj1", pname="project-one")
895
+ p2 = _make_project(tree, pid="proj2", pname="project-two")
896
+
897
+ bad_id_yaml = """\
898
+ ---
899
+ id: ../escape
900
+ trigger: "when coding"
901
+ confidence: 0.9
902
+ domain: general
903
+ scope: project
904
+ ---
905
+
906
+ ## Action
907
+ Invalid id should be skipped.
908
+ """
909
+ (p1["instincts_personal"] / "bad.yaml").write_text(bad_id_yaml)
910
+ (p2["instincts_personal"] / "bad.yaml").write_text(bad_id_yaml)
911
+
912
+ registry = {
913
+ "proj1": {"name": "project-one", "root": "/a", "remote": "", "last_seen": "2025-01-01T00:00:00Z"},
914
+ "proj2": {"name": "project-two", "root": "/b", "remote": "", "last_seen": "2025-01-01T00:00:00Z"},
915
+ }
916
+ tree["registry_file"].write_text(json.dumps(registry))
917
+
918
+ ret = _promote_auto(p1, force=True, dry_run=False)
919
+ assert ret == 0
920
+ err = capsys.readouterr().err
921
+ assert "Skipping invalid instinct ID" in err
922
+ assert not (tree["global_personal"] / "../escape.yaml").exists()
923
+
924
+
925
+ # ─────────────────────────────────────────────
926
+ # _find_cross_project_instincts tests
927
+ # ─────────────────────────────────────────────
928
+
929
+ def test_find_cross_project_empty_registry(patch_globals):
930
+ tree = patch_globals
931
+ tree["registry_file"].write_text("{}")
932
+ result = _find_cross_project_instincts()
933
+ assert result == {}
934
+
935
+
936
+ def test_find_cross_project_single_project(patch_globals):
937
+ """Single project should return nothing (need 2+)."""
938
+ tree = patch_globals
939
+ p1 = _make_project(tree, pid="proj1", pname="project-one")
940
+ (p1["instincts_personal"] / "inst.yaml").write_text(SAMPLE_INSTINCT_YAML)
941
+
942
+ registry = {"proj1": {"name": "project-one", "root": "/a", "remote": "", "last_seen": "2025-01-01T00:00:00Z"}}
943
+ tree["registry_file"].write_text(json.dumps(registry))
944
+
945
+ result = _find_cross_project_instincts()
946
+ assert result == {}
947
+
948
+
949
+ def test_find_cross_project_shared_instinct(patch_globals):
950
+ """Same instinct ID in 2 projects should be found."""
951
+ tree = patch_globals
952
+ p1 = _make_project(tree, pid="proj1", pname="project-one")
953
+ p2 = _make_project(tree, pid="proj2", pname="project-two")
954
+
955
+ (p1["instincts_personal"] / "shared.yaml").write_text(SAMPLE_INSTINCT_YAML)
956
+ (p2["instincts_personal"] / "shared.yaml").write_text(SAMPLE_INSTINCT_YAML)
957
+
958
+ registry = {
959
+ "proj1": {"name": "project-one", "root": "/a", "remote": "", "last_seen": "2025-01-01T00:00:00Z"},
960
+ "proj2": {"name": "project-two", "root": "/b", "remote": "", "last_seen": "2025-01-01T00:00:00Z"},
961
+ }
962
+ tree["registry_file"].write_text(json.dumps(registry))
963
+
964
+ result = _find_cross_project_instincts()
965
+ assert "test-instinct" in result
966
+ assert len(result["test-instinct"]) == 2
967
+
968
+
969
+ # ─────────────────────────────────────────────
970
+ # load_registry tests
971
+ # ─────────────────────────────────────────────
972
+
973
+ def test_load_registry_missing_file(patch_globals):
974
+ result = load_registry()
975
+ assert result == {}
976
+
977
+
978
+ def test_load_registry_corrupt_json(patch_globals):
979
+ tree = patch_globals
980
+ tree["registry_file"].write_text("not json at all {{{")
981
+ result = load_registry()
982
+ assert result == {}
983
+
984
+
985
+ def test_load_registry_valid(patch_globals):
986
+ tree = patch_globals
987
+ data = {"abc": {"name": "test", "root": "/test"}}
988
+ tree["registry_file"].write_text(json.dumps(data))
989
+ result = load_registry()
990
+ assert result == data
991
+
992
+
993
+ def test_load_registry_uses_utf8_encoding(monkeypatch):
994
+ calls = []
995
+
996
+ def fake_open(path, mode="r", *args, **kwargs):
997
+ calls.append(kwargs.get("encoding"))
998
+ return io.StringIO("{}")
999
+
1000
+ monkeypatch.setattr(_mod, "open", fake_open, raising=False)
1001
+ assert load_registry() == {}
1002
+ assert calls == ["utf-8"]
1003
+
1004
+
1005
+ def test_validate_instinct_id():
1006
+ assert _validate_instinct_id("good-id_1.0")
1007
+ assert not _validate_instinct_id("../bad")
1008
+ assert not _validate_instinct_id("bad/name")
1009
+ assert not _validate_instinct_id(".hidden")
1010
+
1011
+
1012
+ def test_update_registry_atomic_replaces_file(patch_globals):
1013
+ tree = patch_globals
1014
+ _update_registry("abc123", "demo", "/repo", "https://example.com/repo.git")
1015
+ data = json.loads(tree["registry_file"].read_text())
1016
+ assert "abc123" in data
1017
+ leftovers = list(tree["registry_file"].parent.glob(".projects.json.tmp.*"))
1018
+ assert leftovers == []