genoma-evolution 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (445) hide show
  1. package/.brv/.obsidian/app.json +1 -0
  2. package/.brv/.obsidian/appearance.json +1 -0
  3. package/.brv/.obsidian/core-plugins.json +33 -0
  4. package/.brv/.obsidian/graph.json +22 -0
  5. package/.brv/.obsidian/workspace.json +195 -0
  6. package/.brv/Sin ti/314/201tulo 1.canvas" +1 -0
  7. package/.brv/Sin ti/314/201tulo 2.canvas" +1 -0
  8. package/.brv/Sin ti/314/201tulo.canvas" +1 -0
  9. package/.brv/_queue_status.json +1 -0
  10. package/.brv/config.json +5 -0
  11. package/.brv/context-tree/_index.md +60 -0
  12. package/.brv/context-tree/_manifest.json +165 -0
  13. package/.brv/context-tree/backend/_index.md +24 -0
  14. package/.brv/context-tree/backend/backend/_index.md +40 -0
  15. package/.brv/context-tree/backend/backend/init.abstract.md +0 -0
  16. package/.brv/context-tree/backend/backend/init.md +27 -0
  17. package/.brv/context-tree/backend/backend/init.overview.md +29 -0
  18. package/.brv/context-tree/backend/backend/job_tracker.abstract.md +1 -0
  19. package/.brv/context-tree/backend/backend/job_tracker.md +273 -0
  20. package/.brv/context-tree/backend/backend/job_tracker.overview.md +31 -0
  21. package/.brv/context-tree/backend/backend/main.abstract.md +0 -0
  22. package/.brv/context-tree/backend/backend/main.md +1292 -0
  23. package/.brv/context-tree/backend/backend/main.overview.md +30 -0
  24. package/.brv/context-tree/backend/backend/requirements.abstract.md +1 -0
  25. package/.brv/context-tree/backend/backend/requirements.md +37 -0
  26. package/.brv/context-tree/backend/backend/requirements.overview.md +28 -0
  27. package/.brv/context-tree/docs/_index.md +37 -0
  28. package/.brv/context-tree/docs/api/_index.md +54 -0
  29. package/.brv/context-tree/docs/api/context.md +11 -0
  30. package/.brv/context-tree/docs/api/hermes_api_openapi_specification.abstract.md +0 -0
  31. package/.brv/context-tree/docs/api/hermes_api_openapi_specification.md +468 -0
  32. package/.brv/context-tree/docs/api/hermes_api_openapi_specification.overview.md +44 -0
  33. package/.brv/context-tree/frontend/_index.md +48 -0
  34. package/.brv/context-tree/frontend/hermes_dashboard/_index.md +31 -0
  35. package/.brv/context-tree/frontend/hermes_dashboard/architecture_overview.abstract.md +0 -0
  36. package/.brv/context-tree/frontend/hermes_dashboard/architecture_overview.md +41 -0
  37. package/.brv/context-tree/frontend/hermes_dashboard/architecture_overview.overview.md +34 -0
  38. package/.brv/context-tree/frontend/src/_index.md +53 -0
  39. package/.brv/context-tree/frontend/src/components/_index.md +52 -0
  40. package/.brv/context-tree/frontend/src/components/sidebar_navigation_component.abstract.md +0 -0
  41. package/.brv/context-tree/frontend/src/components/sidebar_navigation_component.md +161 -0
  42. package/.brv/context-tree/frontend/src/components/sidebar_navigation_component.overview.md +32 -0
  43. package/.brv/context-tree/frontend/src/context.md +10 -0
  44. package/.brv/context-tree/frontend/src/functioncallingpage.abstract.md +0 -0
  45. package/.brv/context-tree/frontend/src/functioncallingpage.md +34 -0
  46. package/.brv/context-tree/frontend/src/functioncallingpage.overview.md +26 -0
  47. package/.brv/context-tree/frontend/src/lib/_index.md +48 -0
  48. package/.brv/context-tree/frontend/src/lib/api_client_library.abstract.md +1 -0
  49. package/.brv/context-tree/frontend/src/lib/api_client_library.md +403 -0
  50. package/.brv/context-tree/frontend/src/lib/api_client_library.overview.md +69 -0
  51. package/.brv/context-tree/frontend/src/page.abstract.md +0 -0
  52. package/.brv/context-tree/frontend/src/page.md +103 -0
  53. package/.brv/context-tree/frontend/src/page.overview.md +7 -0
  54. package/.brv/context-tree/frontend/src/settingspage.abstract.md +0 -0
  55. package/.brv/context-tree/frontend/src/settingspage.md +124 -0
  56. package/.brv/context-tree/frontend/src/settingspage.overview.md +34 -0
  57. package/.brv/context-tree/frontend/src/sidebar.abstract.md +0 -0
  58. package/.brv/context-tree/frontend/src/sidebar.md +170 -0
  59. package/.brv/context-tree/frontend/src/sidebar.overview.md +25 -0
  60. package/.brv/context-tree/meta/_index.md +24 -0
  61. package/.brv/context-tree/meta/curation_context/_index.md +24 -0
  62. package/.brv/context-tree/meta/curation_context/empty_context.abstract.md +4 -0
  63. package/.brv/context-tree/meta/curation_context/empty_context.md +35 -0
  64. package/.brv/context-tree/meta/curation_context/empty_context.overview.md +20 -0
  65. package/.brv/dream-log/drm-1777341062653.json +33 -0
  66. package/.brv/dream-state.json +8 -0
  67. package/.brv/dream.lock +0 -0
  68. package/.brv/review-backups/docs/api/hermes_api_openapi_specification.md +468 -0
  69. package/.claude/settings.local.json +7 -0
  70. package/.claude/worktrees/phase-2-mcp/.brv/.obsidian/app.json +1 -0
  71. package/.claude/worktrees/phase-2-mcp/.brv/.obsidian/appearance.json +1 -0
  72. package/.claude/worktrees/phase-2-mcp/.brv/.obsidian/core-plugins.json +33 -0
  73. package/.claude/worktrees/phase-2-mcp/.brv/.obsidian/graph.json +22 -0
  74. package/.claude/worktrees/phase-2-mcp/.brv/.obsidian/workspace.json +195 -0
  75. package/.claude/worktrees/phase-2-mcp/.brv/Sin t/303/255tulo 1.canvas" +1 -0
  76. package/.claude/worktrees/phase-2-mcp/.brv/Sin t/303/255tulo 2.canvas" +1 -0
  77. package/.claude/worktrees/phase-2-mcp/.brv/Sin t/303/255tulo.canvas" +1 -0
  78. package/.claude/worktrees/phase-2-mcp/.brv/_queue_status.json +1 -0
  79. package/.claude/worktrees/phase-2-mcp/.brv/config.json +5 -0
  80. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/_index.md +60 -0
  81. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/_manifest.json +165 -0
  82. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/backend/_index.md +24 -0
  83. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/backend/backend/_index.md +40 -0
  84. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/backend/backend/init.abstract.md +0 -0
  85. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/backend/backend/init.md +27 -0
  86. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/backend/backend/init.overview.md +29 -0
  87. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/backend/backend/job_tracker.abstract.md +1 -0
  88. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/backend/backend/job_tracker.md +273 -0
  89. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/backend/backend/job_tracker.overview.md +31 -0
  90. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/backend/backend/main.abstract.md +0 -0
  91. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/backend/backend/main.md +1292 -0
  92. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/backend/backend/main.overview.md +30 -0
  93. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/backend/backend/requirements.abstract.md +1 -0
  94. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/backend/backend/requirements.md +37 -0
  95. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/backend/backend/requirements.overview.md +28 -0
  96. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/docs/_index.md +37 -0
  97. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/docs/api/_index.md +54 -0
  98. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/docs/api/context.md +11 -0
  99. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/docs/api/hermes_api_openapi_specification.abstract.md +0 -0
  100. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/docs/api/hermes_api_openapi_specification.md +468 -0
  101. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/docs/api/hermes_api_openapi_specification.overview.md +44 -0
  102. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/frontend/_index.md +48 -0
  103. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/frontend/hermes_dashboard/_index.md +31 -0
  104. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/frontend/hermes_dashboard/architecture_overview.abstract.md +0 -0
  105. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/frontend/hermes_dashboard/architecture_overview.md +41 -0
  106. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/frontend/hermes_dashboard/architecture_overview.overview.md +34 -0
  107. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/frontend/src/_index.md +53 -0
  108. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/frontend/src/components/_index.md +52 -0
  109. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/frontend/src/components/sidebar_navigation_component.abstract.md +0 -0
  110. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/frontend/src/components/sidebar_navigation_component.md +161 -0
  111. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/frontend/src/components/sidebar_navigation_component.overview.md +32 -0
  112. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/frontend/src/context.md +10 -0
  113. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/frontend/src/functioncallingpage.abstract.md +0 -0
  114. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/frontend/src/functioncallingpage.md +34 -0
  115. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/frontend/src/functioncallingpage.overview.md +26 -0
  116. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/frontend/src/lib/_index.md +48 -0
  117. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/frontend/src/lib/api_client_library.abstract.md +1 -0
  118. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/frontend/src/lib/api_client_library.md +403 -0
  119. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/frontend/src/lib/api_client_library.overview.md +69 -0
  120. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/frontend/src/page.abstract.md +0 -0
  121. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/frontend/src/page.md +103 -0
  122. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/frontend/src/page.overview.md +7 -0
  123. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/frontend/src/settingspage.abstract.md +0 -0
  124. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/frontend/src/settingspage.md +124 -0
  125. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/frontend/src/settingspage.overview.md +34 -0
  126. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/frontend/src/sidebar.abstract.md +0 -0
  127. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/frontend/src/sidebar.md +170 -0
  128. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/frontend/src/sidebar.overview.md +25 -0
  129. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/meta/_index.md +24 -0
  130. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/meta/curation_context/_index.md +24 -0
  131. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/meta/curation_context/empty_context.abstract.md +4 -0
  132. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/meta/curation_context/empty_context.md +35 -0
  133. package/.claude/worktrees/phase-2-mcp/.brv/context-tree/meta/curation_context/empty_context.overview.md +20 -0
  134. package/.claude/worktrees/phase-2-mcp/.brv/dream-log/drm-1777341062653.json +33 -0
  135. package/.claude/worktrees/phase-2-mcp/.brv/dream-state.json +8 -0
  136. package/.claude/worktrees/phase-2-mcp/.brv/dream.lock +0 -0
  137. package/.claude/worktrees/phase-2-mcp/.brv/review-backups/docs/api/hermes_api_openapi_specification.md +468 -0
  138. package/.claude/worktrees/phase-2-mcp/.claude/settings.local.json +13 -0
  139. package/.claude/worktrees/phase-2-mcp/.kilocode/package-lock.json +378 -0
  140. package/.claude/worktrees/phase-2-mcp/.kilocode/package.json +5 -0
  141. package/.claude/worktrees/phase-2-mcp/AGENTS.md +5 -0
  142. package/.claude/worktrees/phase-2-mcp/CLAUDE.md +29 -0
  143. package/.claude/worktrees/phase-2-mcp/QA_AUDIT_PLAN.md +156 -0
  144. package/.claude/worktrees/phase-2-mcp/README.md +316 -0
  145. package/.claude/worktrees/phase-2-mcp/agent-agnostic-evolution-dashboard.md +405 -0
  146. package/.claude/worktrees/phase-2-mcp/backend/__init__.py +0 -0
  147. package/.claude/worktrees/phase-2-mcp/backend/collectors/__init__.py +0 -0
  148. package/.claude/worktrees/phase-2-mcp/backend/collectors/claude_code_collector.py +277 -0
  149. package/.claude/worktrees/phase-2-mcp/backend/collectors/hermes_collector.py +68 -0
  150. package/.claude/worktrees/phase-2-mcp/backend/curator.py +512 -0
  151. package/.claude/worktrees/phase-2-mcp/backend/eval/__init__.py +19 -0
  152. package/.claude/worktrees/phase-2-mcp/backend/eval/engine.py +116 -0
  153. package/.claude/worktrees/phase-2-mcp/backend/eval/scorers.py +201 -0
  154. package/.claude/worktrees/phase-2-mcp/backend/generate_dataset.py +86 -0
  155. package/.claude/worktrees/phase-2-mcp/backend/job_tracker.py +232 -0
  156. package/.claude/worktrees/phase-2-mcp/backend/main.py +1746 -0
  157. package/.claude/worktrees/phase-2-mcp/backend/mcp_server.py +250 -0
  158. package/.claude/worktrees/phase-2-mcp/backend/promethean/__init__.py +24 -0
  159. package/.claude/worktrees/phase-2-mcp/backend/promethean/cycle_orchestrator.py +270 -0
  160. package/.claude/worktrees/phase-2-mcp/backend/promethean/delta_validator.py +191 -0
  161. package/.claude/worktrees/phase-2-mcp/backend/promethean/dspy_compiler.py +315 -0
  162. package/.claude/worktrees/phase-2-mcp/backend/promethean/gepa_strategist.py +213 -0
  163. package/.claude/worktrees/phase-2-mcp/backend/promethean/models.py +260 -0
  164. package/.claude/worktrees/phase-2-mcp/backend/promethean/skill_deployer.py +195 -0
  165. package/.claude/worktrees/phase-2-mcp/backend/promethean/trace_ingestion.py +142 -0
  166. package/.claude/worktrees/phase-2-mcp/backend/requirements.txt +6 -0
  167. package/.claude/worktrees/phase-2-mcp/backend/sdd_evolve.py +459 -0
  168. package/.claude/worktrees/phase-2-mcp/backend/skill_detector.py +227 -0
  169. package/.claude/worktrees/phase-2-mcp/backend/skill_registry.py +289 -0
  170. package/.claude/worktrees/phase-2-mcp/backend/storage/__init__.py +5 -0
  171. package/.claude/worktrees/phase-2-mcp/backend/storage/run_store.py +393 -0
  172. package/.claude/worktrees/phase-2-mcp/backend/storage/schema.sql +99 -0
  173. package/.claude/worktrees/phase-2-mcp/backend/validate_evolution.py +267 -0
  174. package/.claude/worktrees/phase-2-mcp/components.json +28 -0
  175. package/.claude/worktrees/phase-2-mcp/docs/api/hermes-api.openapi.yaml +438 -0
  176. package/.claude/worktrees/phase-2-mcp/docs/hero.svg +148 -0
  177. package/.claude/worktrees/phase-2-mcp/eslint.config.mjs +18 -0
  178. package/.claude/worktrees/phase-2-mcp/install.sh +245 -0
  179. package/.claude/worktrees/phase-2-mcp/next-env.d.ts +6 -0
  180. package/.claude/worktrees/phase-2-mcp/next.config.ts +32 -0
  181. package/.claude/worktrees/phase-2-mcp/package-lock.json +11936 -0
  182. package/.claude/worktrees/phase-2-mcp/package.json +41 -0
  183. package/.claude/worktrees/phase-2-mcp/pnpm-workspace.yaml +4 -0
  184. package/.claude/worktrees/phase-2-mcp/postcss.config.mjs +7 -0
  185. package/.claude/worktrees/phase-2-mcp/public/file.svg +1 -0
  186. package/.claude/worktrees/phase-2-mcp/public/fonts/SF-Pro-Display-Bold.otf +0 -0
  187. package/.claude/worktrees/phase-2-mcp/public/fonts/SF-Pro-Display-Heavy.otf +0 -0
  188. package/.claude/worktrees/phase-2-mcp/public/fonts/SF-Pro-Display-Medium.otf +0 -0
  189. package/.claude/worktrees/phase-2-mcp/public/fonts/SF-Pro-Display-Regular.otf +0 -0
  190. package/.claude/worktrees/phase-2-mcp/public/fonts/SF-Pro-Display-Semibold.otf +0 -0
  191. package/.claude/worktrees/phase-2-mcp/public/fonts/SF-Pro-Text-Bold.otf +0 -0
  192. package/.claude/worktrees/phase-2-mcp/public/fonts/SF-Pro-Text-Heavy.otf +0 -0
  193. package/.claude/worktrees/phase-2-mcp/public/fonts/SF-Pro-Text-Medium.otf +0 -0
  194. package/.claude/worktrees/phase-2-mcp/public/fonts/SF-Pro-Text-Regular.otf +0 -0
  195. package/.claude/worktrees/phase-2-mcp/public/fonts/SF-Pro-Text-Semibold.otf +0 -0
  196. package/.claude/worktrees/phase-2-mcp/public/globe.svg +1 -0
  197. package/.claude/worktrees/phase-2-mcp/public/next.svg +1 -0
  198. package/.claude/worktrees/phase-2-mcp/public/theme-preview.html +257 -0
  199. package/.claude/worktrees/phase-2-mcp/public/vercel.svg +1 -0
  200. package/.claude/worktrees/phase-2-mcp/public/window.svg +1 -0
  201. package/.claude/worktrees/phase-2-mcp/run.sh +26 -0
  202. package/.claude/worktrees/phase-2-mcp/skills-lock.json +10 -0
  203. package/.claude/worktrees/phase-2-mcp/specs/event-schema.md +223 -0
  204. package/.claude/worktrees/phase-2-mcp/specs/examples/run.jsonl +3 -0
  205. package/.claude/worktrees/phase-2-mcp/src/app/api/[...path]/route.ts +55 -0
  206. package/.claude/worktrees/phase-2-mcp/src/app/api/auth/token/route.ts +22 -0
  207. package/.claude/worktrees/phase-2-mcp/src/app/evolution/page.tsx +589 -0
  208. package/.claude/worktrees/phase-2-mcp/src/app/favicon.ico +0 -0
  209. package/.claude/worktrees/phase-2-mcp/src/app/globals.css +321 -0
  210. package/.claude/worktrees/phase-2-mcp/src/app/layout.tsx +63 -0
  211. package/.claude/worktrees/phase-2-mcp/src/app/page.tsx +70 -0
  212. package/.claude/worktrees/phase-2-mcp/src/app/skills/page.tsx +369 -0
  213. package/.claude/worktrees/phase-2-mcp/src/components/ApiConfigCard.tsx +199 -0
  214. package/.claude/worktrees/phase-2-mcp/src/components/ColorBends.css +1 -0
  215. package/.claude/worktrees/phase-2-mcp/src/components/ColorBends.d.ts +1 -0
  216. package/.claude/worktrees/phase-2-mcp/src/components/ColorBends.jsx +1 -0
  217. package/.claude/worktrees/phase-2-mcp/src/components/CoreLoopToggle.tsx +111 -0
  218. package/.claude/worktrees/phase-2-mcp/src/components/EnvironmentStatus.tsx +176 -0
  219. package/.claude/worktrees/phase-2-mcp/src/components/EvolutionBackground.tsx +1 -0
  220. package/.claude/worktrees/phase-2-mcp/src/components/ReactQueryProvider.tsx +24 -0
  221. package/.claude/worktrees/phase-2-mcp/src/components/Sidebar.tsx +247 -0
  222. package/.claude/worktrees/phase-2-mcp/src/components/SkillDiffViewer.tsx +154 -0
  223. package/.claude/worktrees/phase-2-mcp/src/components/ThemeAwareBackground.tsx +67 -0
  224. package/.claude/worktrees/phase-2-mcp/src/components/ThemeToggle.tsx +54 -0
  225. package/.claude/worktrees/phase-2-mcp/src/components/WelcomeHero.tsx +77 -0
  226. package/.claude/worktrees/phase-2-mcp/src/components/bits/ClickSpark.tsx +116 -0
  227. package/.claude/worktrees/phase-2-mcp/src/components/bits/CountUp.tsx +98 -0
  228. package/.claude/worktrees/phase-2-mcp/src/components/bits/DarkSelect.tsx +95 -0
  229. package/.claude/worktrees/phase-2-mcp/src/components/bits/DecryptedText.tsx +161 -0
  230. package/.claude/worktrees/phase-2-mcp/src/components/bits/ElectricBorder.tsx +184 -0
  231. package/.claude/worktrees/phase-2-mcp/src/components/bits/GlitchText.tsx +34 -0
  232. package/.claude/worktrees/phase-2-mcp/src/components/bits/ShinyText.tsx +55 -0
  233. package/.claude/worktrees/phase-2-mcp/src/components/bits/SpotlightCard.tsx +42 -0
  234. package/.claude/worktrees/phase-2-mcp/src/components/bits/TextType.tsx +95 -0
  235. package/.claude/worktrees/phase-2-mcp/src/components/bits/index.ts +9 -0
  236. package/.claude/worktrees/phase-2-mcp/src/components/pages/CuratorPage.tsx +632 -0
  237. package/.claude/worktrees/phase-2-mcp/src/components/pages/DatasetPage.tsx +271 -0
  238. package/.claude/worktrees/phase-2-mcp/src/components/pages/EvolutionPage.tsx +676 -0
  239. package/.claude/worktrees/phase-2-mcp/src/components/pages/FunctionCallingPage.tsx +1 -0
  240. package/.claude/worktrees/phase-2-mcp/src/components/pages/LogsPage.tsx +272 -0
  241. package/.claude/worktrees/phase-2-mcp/src/components/pages/MetricsPage.tsx +246 -0
  242. package/.claude/worktrees/phase-2-mcp/src/components/pages/OverviewPage.tsx +420 -0
  243. package/.claude/worktrees/phase-2-mcp/src/components/pages/SettingsPage.tsx +88 -0
  244. package/.claude/worktrees/phase-2-mcp/src/components/pages/SkillStudioPage.tsx +376 -0
  245. package/.claude/worktrees/phase-2-mcp/src/components/ui/animated-theme-toggler.tsx +97 -0
  246. package/.claude/worktrees/phase-2-mcp/src/components/ui/button.tsx +67 -0
  247. package/.claude/worktrees/phase-2-mcp/src/components/ui/card.tsx +103 -0
  248. package/.claude/worktrees/phase-2-mcp/src/components/ui/input.tsx +19 -0
  249. package/.claude/worktrees/phase-2-mcp/src/components/ui/separator.tsx +28 -0
  250. package/.claude/worktrees/phase-2-mcp/src/components/ui/sheet.tsx +147 -0
  251. package/.claude/worktrees/phase-2-mcp/src/components/ui/sidebar.tsx +702 -0
  252. package/.claude/worktrees/phase-2-mcp/src/components/ui/skeleton.tsx +13 -0
  253. package/.claude/worktrees/phase-2-mcp/src/components/ui/theme-toggle.tsx +272 -0
  254. package/.claude/worktrees/phase-2-mcp/src/components/ui/tooltip.tsx +57 -0
  255. package/.claude/worktrees/phase-2-mcp/src/hooks/use-mobile.ts +19 -0
  256. package/.claude/worktrees/phase-2-mcp/src/lib/api.ts +455 -0
  257. package/.claude/worktrees/phase-2-mcp/src/lib/queryClient.ts +12 -0
  258. package/.claude/worktrees/phase-2-mcp/src/lib/utils.ts +6 -0
  259. package/.claude/worktrees/phase-2-mcp/stitch/agent_dashboard/DESIGN_SPEC.md +521 -0
  260. package/.claude/worktrees/phase-2-mcp/stitch/agent_dashboard/prototype.html +676 -0
  261. package/.claude/worktrees/phase-2-mcp/stitch/curator_workspace/code.html +448 -0
  262. package/.claude/worktrees/phase-2-mcp/stitch/curator_workspace/screen.png +0 -0
  263. package/.claude/worktrees/phase-2-mcp/stitch/datasets/code.html +479 -0
  264. package/.claude/worktrees/phase-2-mcp/stitch/datasets/screen.png +0 -0
  265. package/.claude/worktrees/phase-2-mcp/stitch/evolution_history/code.html +461 -0
  266. package/.claude/worktrees/phase-2-mcp/stitch/evolution_history/screen.png +0 -0
  267. package/.claude/worktrees/phase-2-mcp/stitch/hermes_dashboard/DESIGN.md +192 -0
  268. package/.claude/worktrees/phase-2-mcp/stitch/hermes_dashboard/DESIGN_SPEC.md +455 -0
  269. package/.claude/worktrees/phase-2-mcp/stitch/hermes_overview/code.html +399 -0
  270. package/.claude/worktrees/phase-2-mcp/stitch/hermes_overview/screen.png +0 -0
  271. package/.claude/worktrees/phase-2-mcp/stitch/live_logs/code.html +324 -0
  272. package/.claude/worktrees/phase-2-mcp/stitch/live_logs/screen.png +0 -0
  273. package/.claude/worktrees/phase-2-mcp/stitch/skill_hub/code.html +596 -0
  274. package/.claude/worktrees/phase-2-mcp/stitch/skill_hub/screen.png +0 -0
  275. package/.claude/worktrees/phase-2-mcp/stitch/system_metrics/code.html +527 -0
  276. package/.claude/worktrees/phase-2-mcp/stitch/system_metrics/screen.png +0 -0
  277. package/.claude/worktrees/phase-2-mcp/stitch/system_settings/code.html +257 -0
  278. package/.claude/worktrees/phase-2-mcp/stitch/system_settings/screen.png +0 -0
  279. package/.claude/worktrees/phase-2-mcp/test_dashboard.py +201 -0
  280. package/.claude/worktrees/phase-2-mcp/tests/collectors/__init__.py +0 -0
  281. package/.claude/worktrees/phase-2-mcp/tests/collectors/fixtures/sample_session.jsonl +7 -0
  282. package/.claude/worktrees/phase-2-mcp/tests/collectors/test_claude_code_collector.py +171 -0
  283. package/.claude/worktrees/phase-2-mcp/tests/collectors/test_hermes_collector.py +167 -0
  284. package/.claude/worktrees/phase-2-mcp/tests/eval/test_engine.py +234 -0
  285. package/.claude/worktrees/phase-2-mcp/tests/eval/test_scorers.py +249 -0
  286. package/.claude/worktrees/phase-2-mcp/tests/storage/__init__.py +0 -0
  287. package/.claude/worktrees/phase-2-mcp/tests/storage/test_run_store.py +359 -0
  288. package/.claude/worktrees/phase-2-mcp/tests/test_curator.py +559 -0
  289. package/.claude/worktrees/phase-2-mcp/tests/test_mcp_server.py +114 -0
  290. package/.claude/worktrees/phase-2-mcp/tsconfig.json +34 -0
  291. package/.env.example +72 -0
  292. package/.kilocode/package-lock.json +378 -0
  293. package/.kilocode/package.json +5 -0
  294. package/AGENTS.md +5 -0
  295. package/CLAUDE.md +29 -0
  296. package/QA_AUDIT_PLAN.md +156 -0
  297. package/README.md +355 -0
  298. package/agent-agnostic-evolution-dashboard.md +405 -0
  299. package/backend/__init__.py +0 -0
  300. package/backend/collectors/__init__.py +0 -0
  301. package/backend/collectors/claude_code_collector.py +277 -0
  302. package/backend/collectors/hermes_collector.py +68 -0
  303. package/backend/curator.py +512 -0
  304. package/backend/eval/__init__.py +19 -0
  305. package/backend/eval/engine.py +116 -0
  306. package/backend/eval/scorers.py +201 -0
  307. package/backend/generate_dataset.py +86 -0
  308. package/backend/job_tracker.py +232 -0
  309. package/backend/main.py +1746 -0
  310. package/backend/mcp_server.py +250 -0
  311. package/backend/promethean/__init__.py +24 -0
  312. package/backend/promethean/cycle_orchestrator.py +270 -0
  313. package/backend/promethean/delta_validator.py +191 -0
  314. package/backend/promethean/dspy_compiler.py +315 -0
  315. package/backend/promethean/gepa_strategist.py +213 -0
  316. package/backend/promethean/models.py +260 -0
  317. package/backend/promethean/skill_deployer.py +195 -0
  318. package/backend/promethean/trace_ingestion.py +142 -0
  319. package/backend/requirements.txt +6 -0
  320. package/backend/sdd_evolve.py +459 -0
  321. package/backend/skill_detector.py +227 -0
  322. package/backend/skill_registry.py +289 -0
  323. package/backend/storage/__init__.py +5 -0
  324. package/backend/storage/run_store.py +393 -0
  325. package/backend/storage/schema.sql +99 -0
  326. package/backend/validate_evolution.py +267 -0
  327. package/bin/genoma.js +250 -0
  328. package/components.json +28 -0
  329. package/docs/api/hermes-api.openapi.yaml +438 -0
  330. package/docs/hero.svg +148 -0
  331. package/eslint.config.mjs +18 -0
  332. package/install.sh +245 -0
  333. package/next-env.d.ts +6 -0
  334. package/next.config.ts +32 -0
  335. package/package.json +46 -0
  336. package/pnpm-workspace.yaml +4 -0
  337. package/postcss.config.mjs +7 -0
  338. package/public/file.svg +1 -0
  339. package/public/fonts/SF-Pro-Display-Bold.otf +0 -0
  340. package/public/fonts/SF-Pro-Display-Heavy.otf +0 -0
  341. package/public/fonts/SF-Pro-Display-Medium.otf +0 -0
  342. package/public/fonts/SF-Pro-Display-Regular.otf +0 -0
  343. package/public/fonts/SF-Pro-Display-Semibold.otf +0 -0
  344. package/public/fonts/SF-Pro-Text-Bold.otf +0 -0
  345. package/public/fonts/SF-Pro-Text-Heavy.otf +0 -0
  346. package/public/fonts/SF-Pro-Text-Medium.otf +0 -0
  347. package/public/fonts/SF-Pro-Text-Regular.otf +0 -0
  348. package/public/fonts/SF-Pro-Text-Semibold.otf +0 -0
  349. package/public/globe.svg +1 -0
  350. package/public/next.svg +1 -0
  351. package/public/theme-preview.html +257 -0
  352. package/public/vercel.svg +1 -0
  353. package/public/window.svg +1 -0
  354. package/run.sh +26 -0
  355. package/scripts/postinstall.js +50 -0
  356. package/skills-lock.json +10 -0
  357. package/specs/event-schema.md +223 -0
  358. package/specs/examples/run.jsonl +3 -0
  359. package/src/app/api/[...path]/route.ts +55 -0
  360. package/src/app/api/auth/token/route.ts +22 -0
  361. package/src/app/evolution/page.tsx +589 -0
  362. package/src/app/favicon.ico +0 -0
  363. package/src/app/globals.css +321 -0
  364. package/src/app/layout.tsx +63 -0
  365. package/src/app/page.tsx +70 -0
  366. package/src/app/skills/page.tsx +369 -0
  367. package/src/components/ApiConfigCard.tsx +199 -0
  368. package/src/components/ColorBends.css +1 -0
  369. package/src/components/ColorBends.d.ts +1 -0
  370. package/src/components/ColorBends.jsx +1 -0
  371. package/src/components/CoreLoopToggle.tsx +111 -0
  372. package/src/components/EnvironmentStatus.tsx +176 -0
  373. package/src/components/EvolutionBackground.tsx +1 -0
  374. package/src/components/ReactQueryProvider.tsx +24 -0
  375. package/src/components/Sidebar.tsx +247 -0
  376. package/src/components/SkillDiffViewer.tsx +154 -0
  377. package/src/components/ThemeAwareBackground.tsx +67 -0
  378. package/src/components/ThemeToggle.tsx +54 -0
  379. package/src/components/WelcomeHero.tsx +77 -0
  380. package/src/components/bits/ClickSpark.tsx +116 -0
  381. package/src/components/bits/CountUp.tsx +98 -0
  382. package/src/components/bits/DarkSelect.tsx +95 -0
  383. package/src/components/bits/DecryptedText.tsx +161 -0
  384. package/src/components/bits/ElectricBorder.tsx +184 -0
  385. package/src/components/bits/GlitchText.tsx +34 -0
  386. package/src/components/bits/ShinyText.tsx +55 -0
  387. package/src/components/bits/SpotlightCard.tsx +42 -0
  388. package/src/components/bits/TextType.tsx +95 -0
  389. package/src/components/bits/index.ts +9 -0
  390. package/src/components/pages/CuratorPage.tsx +632 -0
  391. package/src/components/pages/DatasetPage.tsx +271 -0
  392. package/src/components/pages/EvolutionPage.tsx +676 -0
  393. package/src/components/pages/FunctionCallingPage.tsx +1 -0
  394. package/src/components/pages/LogsPage.tsx +272 -0
  395. package/src/components/pages/MetricsPage.tsx +246 -0
  396. package/src/components/pages/OverviewPage.tsx +420 -0
  397. package/src/components/pages/SettingsPage.tsx +88 -0
  398. package/src/components/pages/SkillStudioPage.tsx +376 -0
  399. package/src/components/ui/animated-theme-toggler.tsx +97 -0
  400. package/src/components/ui/button.tsx +67 -0
  401. package/src/components/ui/card.tsx +103 -0
  402. package/src/components/ui/input.tsx +19 -0
  403. package/src/components/ui/separator.tsx +28 -0
  404. package/src/components/ui/sheet.tsx +147 -0
  405. package/src/components/ui/sidebar.tsx +702 -0
  406. package/src/components/ui/skeleton.tsx +13 -0
  407. package/src/components/ui/theme-toggle.tsx +272 -0
  408. package/src/components/ui/tooltip.tsx +57 -0
  409. package/src/hooks/use-mobile.ts +19 -0
  410. package/src/lib/api.ts +455 -0
  411. package/src/lib/queryClient.ts +12 -0
  412. package/src/lib/utils.ts +6 -0
  413. package/stitch/agent_dashboard/DESIGN_SPEC.md +521 -0
  414. package/stitch/agent_dashboard/prototype.html +676 -0
  415. package/stitch/curator_workspace/code.html +448 -0
  416. package/stitch/curator_workspace/screen.png +0 -0
  417. package/stitch/datasets/code.html +479 -0
  418. package/stitch/datasets/screen.png +0 -0
  419. package/stitch/evolution_history/code.html +461 -0
  420. package/stitch/evolution_history/screen.png +0 -0
  421. package/stitch/hermes_dashboard/DESIGN.md +192 -0
  422. package/stitch/hermes_dashboard/DESIGN_SPEC.md +455 -0
  423. package/stitch/hermes_overview/code.html +399 -0
  424. package/stitch/hermes_overview/screen.png +0 -0
  425. package/stitch/live_logs/code.html +324 -0
  426. package/stitch/live_logs/screen.png +0 -0
  427. package/stitch/skill_hub/code.html +596 -0
  428. package/stitch/skill_hub/screen.png +0 -0
  429. package/stitch/system_metrics/code.html +527 -0
  430. package/stitch/system_metrics/screen.png +0 -0
  431. package/stitch/system_settings/code.html +257 -0
  432. package/stitch/system_settings/screen.png +0 -0
  433. package/test_dashboard.py +201 -0
  434. package/tests/collectors/__init__.py +0 -0
  435. package/tests/collectors/fixtures/sample_session.jsonl +7 -0
  436. package/tests/collectors/test_claude_code_collector.py +171 -0
  437. package/tests/collectors/test_hermes_collector.py +167 -0
  438. package/tests/eval/test_engine.py +234 -0
  439. package/tests/eval/test_scorers.py +249 -0
  440. package/tests/storage/__init__.py +0 -0
  441. package/tests/storage/test_run_store.py +359 -0
  442. package/tests/test_curator.py +559 -0
  443. package/tests/test_e2e_npm.py +621 -0
  444. package/tests/test_mcp_server.py +114 -0
  445. package/tsconfig.json +34 -0
@@ -0,0 +1,227 @@
1
+ """
2
+ Skill Detector — Escanea y detecta skills de múltiples proveedores
3
+ Proveedores soportados: Claude Code, OpenCode, Kilocode, Antigravity, Hermes
4
+ """
5
+
6
+ import json
7
+ from dataclasses import dataclass, asdict
8
+ from datetime import datetime
9
+ from pathlib import Path
10
+ from typing import List, Dict, Optional
11
+
12
+
13
+ @dataclass
14
+ class SkillInfo:
15
+ """Información de una skill detectada"""
16
+ name: str
17
+ provider: str
18
+ path: str
19
+ description: str = ""
20
+ enabled: bool = False
21
+ installed_at: Optional[str] = None
22
+ tags: List[str] = None
23
+
24
+ def __post_init__(self):
25
+ if self.tags is None:
26
+ self.tags = []
27
+
28
+ def to_dict(self):
29
+ return asdict(self)
30
+
31
+
32
+ class SkillDetector:
33
+ """Detector de skills multi-proveedor"""
34
+
35
+ PROVIDER_PATHS = {
36
+ "claude-code": Path.home() / ".claude" / "skills",
37
+ "opencode": Path.home() / ".opencode" / "skills",
38
+ "kilocode": Path.home() / ".kilocode" / "skills",
39
+ "antigravity": Path.home() / ".antigravity" / "providers",
40
+ "hermes": Path.home() / ".hermes" / "skills",
41
+ "agency": Path.home() / ".hermes" / "hermes-agent" / "skills" / "agency",
42
+ }
43
+
44
+ def __init__(self, config_path: Optional[Path] = None):
45
+ self.config_path = config_path or (Path.home() / ".hermes" / "memory" / "skills_config.json")
46
+ self.enabled_skills = self._load_enabled()
47
+
48
+ def _load_enabled(self) -> Dict[str, bool]:
49
+ if self.config_path.exists():
50
+ try:
51
+ data = json.loads(self.config_path.read_text())
52
+ return {k: v for k, v in data.items()}
53
+ except Exception:
54
+ pass
55
+ return {}
56
+
57
+ def _save_enabled(self):
58
+ self.config_path.parent.mkdir(parents=True, exist_ok=True)
59
+ self.config_path.write_text(json.dumps(self.enabled_skills, indent=2))
60
+
61
+ def detect_all(self) -> List[SkillInfo]:
62
+ skills = []
63
+ for provider, skills_dir in self.PROVIDER_PATHS.items():
64
+ if not skills_dir.exists():
65
+ continue
66
+ if provider == "claude-code":
67
+ skills.extend(self._detect_claude_skills(skills_dir))
68
+ elif provider == "opencode":
69
+ skills.extend(self._detect_opencode_skills(skills_dir))
70
+ elif provider == "kilocode":
71
+ skills.extend(self._detect_kilocode_skills(skills_dir))
72
+ elif provider == "antigravity":
73
+ skills.extend(self._detect_antigravity_skills(skills_dir))
74
+ elif provider == "hermes":
75
+ skills.extend(self._detect_hermes_skills(skills_dir))
76
+ elif provider == "agency":
77
+ skills.extend(self._detect_agency_skills(skills_dir))
78
+ return skills
79
+
80
+ def _detect_claude_skills(self, skills_dir: Path) -> List[SkillInfo]:
81
+ skills = []
82
+ for skill_dir in skills_dir.iterdir():
83
+ if not skill_dir.is_dir():
84
+ continue
85
+ skill_file = skill_dir / "SKILL.md"
86
+ if skill_file.exists():
87
+ name = skill_dir.name
88
+ description = self._parse_description(skill_file)
89
+ key = f"claude-code.{name}"
90
+ skills.append(SkillInfo(
91
+ name=name, provider="claude-code", path=str(skill_dir),
92
+ description=description, enabled=self.enabled_skills.get(key, False),
93
+ installed_at=str(skill_file.stat().st_mtime), tags=["claude", "ai-assistant"]
94
+ ))
95
+ return skills
96
+
97
+ def _detect_opencode_skills(self, skills_dir: Path) -> List[SkillInfo]:
98
+ skills = []
99
+ for skill_dir in skills_dir.iterdir():
100
+ if not skill_dir.is_dir():
101
+ continue
102
+ skill_file = skill_dir / "SKILL.md"
103
+ if skill_file.exists():
104
+ name = skill_dir.name
105
+ description = self._parse_description(skill_file)
106
+ key = f"opencode.{name}"
107
+ skills.append(SkillInfo(
108
+ name=name, provider="opencode", path=str(skill_dir),
109
+ description=description, enabled=self.enabled_skills.get(key, False),
110
+ installed_at=str(skill_file.stat().st_mtime), tags=["opencode", "coding"]
111
+ ))
112
+ return skills
113
+
114
+ def _detect_kilocode_skills(self, skills_dir: Path) -> List[SkillInfo]:
115
+ skills = []
116
+ for skill_dir in skills_dir.iterdir():
117
+ if not skill_dir.is_dir():
118
+ continue
119
+ skill_file = skill_dir / "SKILL.md"
120
+ if skill_file.exists():
121
+ name = skill_dir.name
122
+ description = self._parse_description(skill_file)
123
+ key = f"kilocode.{name}"
124
+ skills.append(SkillInfo(
125
+ name=name, provider="kilocode", path=str(skill_dir),
126
+ description=description, enabled=self.enabled_skills.get(key, False),
127
+ installed_at=str(skill_file.stat().st_mtime), tags=["kilocode"]
128
+ ))
129
+ return skills
130
+
131
+ def _detect_antigravity_skills(self, providers_dir: Path) -> List[SkillInfo]:
132
+ skills = []
133
+ if not providers_dir.exists():
134
+ return skills
135
+ for provider_dir in providers_dir.iterdir():
136
+ if not provider_dir.is_dir():
137
+ continue
138
+ for skill_dir in provider_dir.iterdir():
139
+ if not skill_dir.is_dir():
140
+ continue
141
+ skill_file = skill_dir / "SKILL.md"
142
+ if skill_file.exists():
143
+ name = f"{provider_dir.name}.{skill_dir.name}"
144
+ description = self._parse_description(skill_file)
145
+ key = f"antigravity.{provider_dir.name}.{skill_dir.name}"
146
+ skills.append(SkillInfo(
147
+ name=name, provider="antigravity", path=str(skill_dir),
148
+ description=description, enabled=self.enabled_skills.get(key, False),
149
+ installed_at=str(skill_file.stat().st_mtime),
150
+ tags=["antigravity", provider_dir.name]
151
+ ))
152
+ return skills
153
+
154
+ def _detect_agency_skills(self, skills_dir: Path) -> List[SkillInfo]:
155
+ skills = []
156
+ if not skills_dir.exists():
157
+ return skills
158
+ for skill_dir in skills_dir.iterdir():
159
+ if not skill_dir.is_dir():
160
+ continue
161
+ skill_file = skill_dir / "SKILL.md"
162
+ if skill_file.exists():
163
+ name = skill_dir.name
164
+ description = self._parse_description(skill_file)
165
+ key = f"agency.{name}"
166
+ skills.append(SkillInfo(
167
+ name=name, provider="agency", path=str(skill_dir),
168
+ description=description, enabled=self.enabled_skills.get(key, False),
169
+ installed_at=str(skill_file.stat().st_mtime), tags=["agency", "external"]
170
+ ))
171
+ return skills
172
+
173
+ def _detect_hermes_skills(self, skills_dir: Path) -> List[SkillInfo]:
174
+ skills = []
175
+ if not skills_dir.exists():
176
+ return skills
177
+ for skill_dir in skills_dir.iterdir():
178
+ if not skill_dir.is_dir():
179
+ continue
180
+ skill_file = skill_dir / "SKILL.md"
181
+ if skill_file.exists():
182
+ name = skill_dir.name
183
+ description = self._parse_description(skill_file)
184
+ key = f"hermes.{name}"
185
+ skills.append(SkillInfo(
186
+ name=name, provider="hermes", path=str(skill_dir),
187
+ description=description, enabled=self.enabled_skills.get(key, False),
188
+ installed_at=str(skill_file.stat().st_mtime), tags=["hermes", "core"]
189
+ ))
190
+ return skills
191
+
192
+ def _parse_description(self, skill_file: Path) -> str:
193
+ try:
194
+ content = skill_file.read_text()
195
+ lines = [l.strip() for l in content.split(chr(10)) if l.strip()]
196
+ for line in lines:
197
+ if not line.startswith('#') and len(line) > 10:
198
+ return line[:200]
199
+ except Exception:
200
+ pass
201
+ return "Skill descargada"
202
+
203
+ def get_providers(self) -> List[Dict]:
204
+ skills = self.detect_all()
205
+ providers = {}
206
+ for skill in skills:
207
+ if skill.provider not in providers:
208
+ providers[skill.provider] = {
209
+ "name": skill.provider, "total": 0, "enabled": 0, "skills": []
210
+ }
211
+ providers[skill.provider]["total"] += 1
212
+ if skill.enabled:
213
+ providers[skill.provider]["enabled"] += 1
214
+ providers[skill.provider]["skills"].append(skill.to_dict())
215
+ return list(providers.values())
216
+
217
+ def toggle_skill(self, provider: str, skill_name: str, enabled: bool) -> bool:
218
+ key = f"{provider}.{skill_name}"
219
+ self.enabled_skills[key] = enabled
220
+ self._save_enabled()
221
+ return True
222
+
223
+ def get_skill(self, provider: str, skill_name: str) -> Optional[SkillInfo]:
224
+ for s in self.detect_all():
225
+ if s.provider == provider and s.name == skill_name:
226
+ return s
227
+ return None
@@ -0,0 +1,289 @@
1
+
2
+ """
3
+ SkillRegistry: Versión simple — carga desde JSON al iniciar.
4
+ """
5
+
6
+ from pathlib import Path
7
+ import json, hashlib
8
+ from dataclasses import dataclass, asdict
9
+ from typing import Dict, List
10
+ import shutil
11
+
12
+ @dataclass
13
+ class GlobalSkill:
14
+ id: str
15
+ name: str
16
+ description: str
17
+ canonical_path: str
18
+ code_hash: str
19
+ tags: List
20
+ providers: List
21
+ created_at: str
22
+ is_fork: bool = False
23
+
24
+ class SimpleSkillRegistry:
25
+ GLOBAL_SKILLS_DIR = Path.home() / ".hermes" / "global_skills"
26
+ REGISTRY_DB = Path.home() / ".hermes" / "memory" / "skill_registry.json"
27
+
28
+ PROVIDER_SOURCES = {
29
+ "claude-code": Path.home() / ".claude" / "skills",
30
+ "opencode": Path.home() / ".opencode" / "skills",
31
+ "kilocode": Path.home() / ".kilocode" / "skills",
32
+ "antigravity": Path.home() / ".antigravity" / "providers",
33
+ "hermes": Path.home() / ".hermes" / "skills",
34
+ # "agency": Path.home() / ".hermes" / "hermes-agent" / "skills" / "agency",
35
+ }
36
+
37
+ def __init__(self):
38
+ self.global_skills: Dict[str, GlobalSkill] = {}
39
+ self.provider_index: Dict[str, str] = {}
40
+ self.load()
41
+
42
+ def _quick_hash(self, skill_dir: Path) -> str:
43
+ h = hashlib.sha256()
44
+ md = skill_dir / "SKILL.md"
45
+ if md.exists():
46
+ st = md.stat()
47
+ h.update(f"md:{st.st_size}:{int(st.st_mtime)}".encode())
48
+ for f in skill_dir.iterdir():
49
+ if f.is_file() and f.suffix in ['.cp', '.py']:
50
+ st = f.stat()
51
+ h.update(f"{f.name}:{st.st_size}:{int(st.st_mtime)}".encode())
52
+ break
53
+ return h.hexdigest()[:16]
54
+
55
+ def _quick_desc(self, md_path: Path) -> str:
56
+ try:
57
+ line = md_path.read_text().split('\n')[0].strip()
58
+ return line[:200] if line else "Skill"
59
+ except: return "Skill"
60
+
61
+ def load(self):
62
+ if self.REGISTRY_DB.exists():
63
+ try:
64
+ data = json.loads(self.REGISTRY_DB.read_text())
65
+ self.global_skills = {k: GlobalSkill(**v) for k, v in data["global_skills"].items()}
66
+ self.provider_index = data["provider_index"]
67
+ print(f"[Registry] Cargado: {len(self.global_skills)} skills")
68
+ return self
69
+ except Exception as e:
70
+ print(f"[Registry] Error: {e}")
71
+ print("[Registry] Escaneando...")
72
+ self._fast_scan()
73
+ self.save()
74
+ return self
75
+
76
+ def rescan(self):
77
+ self._fast_scan()
78
+ self.save()
79
+ return self
80
+
81
+ def _fast_scan(self):
82
+ self.global_skills.clear()
83
+ self.provider_index.clear()
84
+ self.GLOBAL_SKILLS_DIR.mkdir(parents=True, exist_ok=True)
85
+
86
+ for skill_dir in self.GLOBAL_SKILLS_DIR.iterdir():
87
+ if not skill_dir.is_dir() or not (skill_dir / "SKILL.md").exists():
88
+ continue
89
+ name = skill_dir.name
90
+ sid = self._quick_hash(skill_dir)
91
+ self.global_skills[sid] = GlobalSkill(
92
+ id=sid, name=name, description=self._quick_desc(skill_dir / "SKILL.md"),
93
+ canonical_path=str(skill_dir), code_hash=sid,
94
+ tags=[], providers=[], created_at="2024-01-01", is_fork=False
95
+ )
96
+
97
+ for provider, src in self.PROVIDER_SOURCES.items():
98
+ if not src.exists(): continue
99
+ # Collect all skill directories (direct + nested in categories)
100
+ skill_dirs = []
101
+ for item in src.iterdir():
102
+ if not item.is_dir():
103
+ continue
104
+ if (item / "SKILL.md").exists():
105
+ # Direct skill: ~/.hermes/skills/skillname/
106
+ skill_dirs.append((item.name, item, None))
107
+ else:
108
+ # Check subdirs for categorized skills: ~/.hermes/skills/category/skillname/
109
+ for sub in item.iterdir():
110
+ if sub.is_dir() and (sub / "SKILL.md").exists():
111
+ skill_dirs.append((sub.name, sub, item.name)) # (skill_name, path, category)
112
+
113
+ for name, skill_dir, category_hint in skill_dirs:
114
+ lhash = self._quick_hash(skill_dir)
115
+ matched = None
116
+ if skill_dir.is_symlink():
117
+ try:
118
+ target = str(skill_dir.resolve())
119
+ for gid, gs in self.global_skills.items():
120
+ if gs.name == name and gs.canonical_path == target:
121
+ matched = gid; break
122
+ except: pass
123
+ if matched is None:
124
+ for gid, gs in self.global_skills.items():
125
+ if gs.name == name and gs.code_hash == lhash:
126
+ matched = gid; break
127
+ key = f"{provider}.{name}"
128
+ if matched:
129
+ self.provider_index[key] = matched
130
+ self.global_skills[matched].providers.append({
131
+ "name": provider, "enabled": True, "overrides": {},
132
+ "local_path": str(skill_dir)
133
+ })
134
+ else:
135
+ # Skill existe solo en este provider → crear entrada global
136
+ sid = self._quick_hash(skill_dir)
137
+ self.global_skills[sid] = GlobalSkill(
138
+ id=sid, name=name, description=self._quick_desc(skill_dir / "SKILL.md"),
139
+ canonical_path=str(skill_dir), code_hash=sid,
140
+ tags=[], providers=[{
141
+ "name": provider, "enabled": True, "overrides": {},
142
+ "local_path": str(skill_dir)
143
+ }], created_at="2024-01-01", is_fork=False
144
+ )
145
+ self.provider_index[key] = sid
146
+ print(f"[Registry] Scan: {len(self.global_skills)} skills")
147
+
148
+ def save(self):
149
+ self.REGISTRY_DB.parent.mkdir(parents=True, exist_ok=True)
150
+ data = {
151
+ "global_skills": {k: asdict(v) for k, v in self.global_skills.items()},
152
+ "provider_index": self.provider_index,
153
+ "updated_at": "2024-01-01"
154
+ }
155
+ self.REGISTRY_DB.write_text(json.dumps(data, indent=2))
156
+
157
+ def _get_category(self, skill):
158
+ """Extrae categoría del path de la skill.
159
+
160
+ Reglas:
161
+ - skills/category/skillname/ → category
162
+ - skills/skillname/ → 'general' (skill directa, sin categoría)
163
+ - agency/skills/agency/skillname/ → 'agency' (path especial de agency)
164
+ """
165
+ for path in [skill.canonical_path] + [p.get('local_path', '') for p in skill.providers]:
166
+ if '/skills/' in path:
167
+ after_skills = path.split('/skills/')[1].strip('/')
168
+ if not after_skills:
169
+ continue
170
+ parts = after_skills.split('/')
171
+ # Si hay 2+ niveles → skills/category/skill/ → category es el primero
172
+ if len(parts) >= 2:
173
+ # Excepción: si el primer nivel es el mismo nombre que la skill,
174
+ # significa que es un path tipo ~/.claude/skills/skillname/
175
+ # donde skillname es la carpeta de la skill, no una categoría
176
+ if parts[0] == skill.name:
177
+ return 'general'
178
+ return parts[0]
179
+ # Si hay 1 nivel → skills/skillname/ → general
180
+ elif len(parts) == 1:
181
+ return 'general'
182
+ return 'uncategorized'
183
+
184
+ def get_providers(self):
185
+ providers = {}
186
+ for skill in self.global_skills.values():
187
+ # Skip agency skills entirely
188
+ if skill.name.startswith("agency-"):
189
+ continue
190
+ category = self._get_category(skill)
191
+ for p in skill.providers:
192
+ pn = p['name']
193
+ if pn not in providers:
194
+ providers[pn] = {"name": pn, "total": 0, "enabled": 0, "skills": []}
195
+ providers[pn]["total"] += 1
196
+ if p.get("enabled"): providers[pn]["enabled"] += 1
197
+ providers[pn]["skills"].append({
198
+ "name": skill.name, "description": skill.description,
199
+ "enabled": p.get("enabled", False), "tags": skill.tags,
200
+ "is_fork": skill.is_fork, "category": category,
201
+ })
202
+ return list(providers.values())
203
+
204
+ def get_global_skills(self):
205
+ return [{**asdict(s), 'canonical_path': s.canonical_path} for s in self.global_skills.values() if not s.name.startswith("agency-")]
206
+
207
+ def get_provider_skills(self, provider):
208
+ result = []
209
+ for skill in self.global_skills.values():
210
+ for p in skill.providers:
211
+ if p['name'] == provider:
212
+ result.append({
213
+ "global_id": skill.id, "name": skill.name,
214
+ "description": skill.description, "enabled": p['enabled'],
215
+ "tags": skill.tags, "is_fork": skill.is_fork,
216
+ "canonical_path": skill.canonical_path
217
+ })
218
+ break
219
+ return result
220
+
221
+ def get_skill_by_name(self, name):
222
+ for s in self.global_skills.values():
223
+ if s.name == name: return s
224
+ return None
225
+
226
+ def toggle_provider_skill(self, provider, skill_name, enabled):
227
+ key = f"{provider}.{skill_name}"
228
+ if key in self.provider_index:
229
+ gid = self.provider_index[key]
230
+ skill = self.global_skills[gid]
231
+ for p in skill.providers:
232
+ if p['name'] == provider:
233
+ p['enabled'] = enabled; self.save(); return True
234
+ for s in self.global_skills.values():
235
+ if s.name == skill_name and s.is_fork:
236
+ for p in s.providers:
237
+ if p['name'] == provider:
238
+ p['enabled'] = enabled; self.save(); return True
239
+ return False
240
+
241
+ def delete_provider_skill(self, provider, skill_name):
242
+ key = f"{provider}.{skill_name}"
243
+ if key in self.provider_index:
244
+ gid = self.provider_index[key]
245
+ skill = self.global_skills[gid]
246
+ skill.providers = [p for p in skill.providers if p['name'] != provider]
247
+ del self.provider_index[key]
248
+ self.save()
249
+ return True
250
+ for s in self.global_skills.values():
251
+ if s.name == skill_name:
252
+ for p in s.providers:
253
+ if p['name'] == provider:
254
+ s.providers = [pp for pp in s.providers if pp['name'] != provider]
255
+ self.save()
256
+ return True
257
+ return False
258
+
259
+ def delete_global_skill(self, skill_name):
260
+ for gid, skill in list(self.global_skills.items()):
261
+ if skill.name == skill_name and not skill.is_fork:
262
+ canon = Path(skill.canonical_path)
263
+ # 1. Eliminar symlinks PRIMERO
264
+ for provider, src_dir in self.PROVIDER_SOURCES.items():
265
+ if not src_dir.exists(): continue
266
+ link = src_dir / skill_name
267
+ if link.exists() and link.is_symlink():
268
+ try:
269
+ if str(link.resolve()) == skill.canonical_path:
270
+ link.unlink()
271
+ except: pass
272
+ # 2. Eliminar directorio canónico
273
+ if canon.exists():
274
+ shutil.rmtree(canon)
275
+ keys = [k for k, v in self.provider_index.items() if v == gid]
276
+ for k in keys: del self.provider_index[k]
277
+ del self.global_skills[gid]
278
+ self.save()
279
+ return True
280
+ return False
281
+
282
+ _registry = None
283
+ def get_registry():
284
+ global _registry
285
+ if _registry is None:
286
+ _registry = SimpleSkillRegistry()
287
+ return _registry
288
+
289
+ print("✅ SimpleSkillRegistry listo")
@@ -0,0 +1,5 @@
1
+ """Storage layer for CanonicalRun persistence."""
2
+
3
+ from backend.storage.run_store import RunStore
4
+
5
+ __all__ = ["RunStore"]