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,68 @@
1
+ """Hermes collector — converts TraceRecord to CanonicalRun schema."""
2
+
3
+ import json
4
+ from pathlib import Path
5
+ from typing import Optional
6
+ from backend.promethean.models import TraceRecord, CanonicalRun
7
+ from backend.promethean.trace_ingestion import TraceIngestor, INGESTED_DIR
8
+
9
+
10
+ class HermesCollector:
11
+ """Wraps TraceIngestor to emit CanonicalRun events in canonical schema."""
12
+
13
+ VERSION = "0.1.0"
14
+ AGENT_NAME = "hermes"
15
+
16
+ def __init__(self):
17
+ self.ingestor = TraceIngestor()
18
+
19
+ def collect_from_trace(self, trace: TraceRecord) -> CanonicalRun:
20
+ """Convert a single TraceRecord into a CanonicalRun."""
21
+ return trace.to_canonical()
22
+
23
+ def collect_batch(self, traces: list[TraceRecord]) -> list[CanonicalRun]:
24
+ """Convert a batch of TraceRecords into CanonicalRun events."""
25
+ return [self.collect_from_trace(t) for t in traces]
26
+
27
+ def load_from_disk(self, limit: int = 100) -> list[CanonicalRun]:
28
+ """Load all ingested traces from disk and convert to CanonicalRun."""
29
+ runs = []
30
+ ingested_path = Path(INGESTED_DIR)
31
+
32
+ if not ingested_path.exists():
33
+ return runs
34
+
35
+ # Read .json files from INGESTED_DIR, up to limit
36
+ count = 0
37
+ for json_file in sorted(ingested_path.glob("*.json"), reverse=True):
38
+ if count >= limit:
39
+ break
40
+ try:
41
+ trace_data = json.loads(json_file.read_text())
42
+ trace = TraceRecord.from_json(trace_data)
43
+ runs.append(self.collect_from_trace(trace))
44
+ count += 1
45
+ except Exception:
46
+ # Skip malformed traces
47
+ continue
48
+
49
+ return runs
50
+
51
+ def get_run_by_id(self, run_id: str) -> Optional[CanonicalRun]:
52
+ """Look up a run by trace_id (run_id in CanonicalRun schema)."""
53
+ ingested_path = Path(INGESTED_DIR)
54
+
55
+ if not ingested_path.exists():
56
+ return None
57
+
58
+ # Search for file matching the run_id
59
+ for json_file in ingested_path.glob(f"*{run_id}*.json"):
60
+ try:
61
+ trace_data = json.loads(json_file.read_text())
62
+ trace = TraceRecord.from_json(trace_data)
63
+ if trace.trace_id == run_id:
64
+ return self.collect_from_trace(trace)
65
+ except Exception:
66
+ continue
67
+
68
+ return None
@@ -0,0 +1,512 @@
1
+ """Curator Module — Skill Lifecycle Management for Hermes Dashboard
2
+
3
+ Compatible with the upstream Hermes Agent Curator format (issue #7816):
4
+ - ~/.hermes/skills/.usage.json — per-skill telemetry
5
+ - ~/.hermes/logs/curator/YYYYMMDD-HHMMSS/run.json + REPORT.md — per-run reports
6
+ - ~/.hermes/skills/.archive/ — archived skills
7
+
8
+ Operates independently so it works even if the upstream Curator isn't merged yet.
9
+ """
10
+
11
+ import json
12
+ import os
13
+ import shutil
14
+ import subprocess
15
+ import time
16
+ from datetime import datetime, timezone
17
+ from pathlib import Path
18
+ from typing import Optional
19
+
20
+ # ── Paths ──────────────────────────────────────────────────────────
21
+ USAGE_FILE = Path.home() / ".hermes" / "skills" / ".usage.json"
22
+ CURATOR_LOG_DIR = Path.home() / ".hermes" / "logs" / "curator"
23
+ SKILLS_DIR = Path.home() / ".hermes" / "skills"
24
+ ARCHIVE_DIR = SKILLS_DIR / ".archive"
25
+ BUNDLED_MANIFEST = SKILLS_DIR / ".bundled_manifest"
26
+ HUB_LOCK = SKILLS_DIR / ".hub" / "lock.json"
27
+
28
+ # ── Defaults (matching upstream) ───────────────────────────────────
29
+ STALE_AFTER_DAYS = 30
30
+ ARCHIVE_AFTER_DAYS = 90
31
+
32
+ # ── Usage telemetry ────────────────────────────────────────────────
33
+
34
+ def _load_usage() -> dict:
35
+ """Load usage telemetry from .usage.json, return empty dict if missing."""
36
+ if USAGE_FILE.exists():
37
+ try:
38
+ return json.loads(USAGE_FILE.read_text())
39
+ except (json.JSONDecodeError, OSError):
40
+ return {}
41
+ return {}
42
+
43
+
44
+ def _save_usage(usage: dict):
45
+ """Save usage telemetry to .usage.json."""
46
+ USAGE_FILE.parent.mkdir(parents=True, exist_ok=True)
47
+ USAGE_FILE.write_text(json.dumps(usage, indent=2, default=str))
48
+
49
+
50
+ def _is_agent_created(skill_name: str) -> bool:
51
+ """Check if a skill is agent-created (not bundled or hub-installed)."""
52
+ # Bundled check
53
+ if BUNDLED_MANIFEST.exists():
54
+ bundled = BUNDLED_MANIFEST.read_text().splitlines()
55
+ for line in bundled:
56
+ name = line.split(":")[0].strip()
57
+ if name == skill_name:
58
+ return False
59
+ # Hub-installed check
60
+ if HUB_LOCK.exists():
61
+ try:
62
+ hub = json.loads(HUB_LOCK.read_text())
63
+ if isinstance(hub, dict) and skill_name in hub:
64
+ return False
65
+ if isinstance(hub, list):
66
+ for entry in hub:
67
+ if isinstance(entry, dict) and entry.get("name") == skill_name:
68
+ return False
69
+ except (json.JSONDecodeError, OSError):
70
+ pass
71
+ return True
72
+
73
+
74
+ # Additional skill directories to search
75
+ EXTRA_SKILL_DIRS = [
76
+ Path.home() / ".claude" / "skills",
77
+ Path.home() / ".hermes" / "hermes-agent" / "skills",
78
+ ]
79
+
80
+
81
+ def _skill_exists(skill_name: str) -> bool:
82
+ """Check if a skill directory exists across all known skill locations."""
83
+ # Check primary skills dir
84
+ if (SKILLS_DIR / skill_name).exists():
85
+ return True
86
+ if (ARCHIVE_DIR / skill_name).exists():
87
+ return True
88
+ # Check extra dirs
89
+ for d in EXTRA_SKILL_DIRS:
90
+ if (d / skill_name).exists():
91
+ return True
92
+ # Check nested categories (e.g. ~/.claude/skills/backend-patterns/)
93
+ if (d / skill_name / "SKILL.md").exists():
94
+ return True
95
+ return False
96
+
97
+
98
+ # ── Public API ──────────────────────────────────────────────────────
99
+
100
+ def get_status() -> dict:
101
+ """Return curator status: last run, counts by state, pinned list."""
102
+ usage = _load_usage()
103
+
104
+ now = datetime.now(timezone.utc)
105
+
106
+ # Compute states
107
+ active = 0
108
+ stale = 0
109
+ archived = 0
110
+ pinned = []
111
+ lru = [] # Least recently used top 5
112
+
113
+ for skill_name, data in usage.items():
114
+ state = data.get("state", "active")
115
+ if state == "active":
116
+ active += 1
117
+ elif state == "stale":
118
+ stale += 1
119
+ elif state == "archived":
120
+ archived += 1
121
+
122
+ if data.get("pinned"):
123
+ pinned.append(skill_name)
124
+
125
+ # LRU sort
126
+ sorted_by_use = sorted(
127
+ [(name, data.get("last_used_at") or data.get("last_viewed_at") or "1970-01-01T00:00:00Z",
128
+ data.get("use_count", 0), data.get("view_count", 0))
129
+ for name, data in usage.items() if data.get("state") != "archived"],
130
+ key=lambda x: x[1]
131
+ )
132
+ lru = [{"name": s[0], "last_used": s[1], "use_count": s[2], "view_count": s[3]}
133
+ for s in sorted_by_use[:5]]
134
+
135
+ # Last run
136
+ last_run = None
137
+ newest_dir = None
138
+ if CURATOR_LOG_DIR.exists():
139
+ dirs = sorted(CURATOR_LOG_DIR.iterdir()) if CURATOR_LOG_DIR.exists() else []
140
+ if dirs:
141
+ newest_dir = dirs[-1].name
142
+ try:
143
+ run_data = json.loads((CURATOR_LOG_DIR / dirs[-1].name / "run.json").read_text())
144
+ last_run = run_data
145
+ except (json.JSONDecodeError, OSError, IndexError):
146
+ last_run = {"timestamp": dirs[-1].name}
147
+
148
+ return {
149
+ "status": "ok",
150
+ "last_run": last_run or {"timestamp": None},
151
+ "last_run_dir": newest_dir,
152
+ "stats": {
153
+ "active": active,
154
+ "stale": stale,
155
+ "archived": archived,
156
+ "total": active + stale + archived,
157
+ "pinned": len(pinned),
158
+ },
159
+ "pinned_skills": pinned,
160
+ "least_recently_used": lru,
161
+ }
162
+
163
+
164
+ def get_skills_usage() -> list[dict]:
165
+ """Return usage telemetry for all tracked skills."""
166
+ usage = _load_usage()
167
+ results = []
168
+
169
+ for skill_name, data in usage.items():
170
+ results.append({
171
+ "name": skill_name,
172
+ "state": data.get("state", "active"),
173
+ "pinned": data.get("pinned", False),
174
+ "use_count": data.get("use_count", 0),
175
+ "view_count": data.get("view_count", 0),
176
+ "patch_count": data.get("patch_count", 0),
177
+ "last_used_at": data.get("last_used_at"),
178
+ "last_viewed_at": data.get("last_viewed_at"),
179
+ "last_patched_at": data.get("last_patched_at"),
180
+ "created_at": data.get("created_at"),
181
+ "archived_at": data.get("archived_at"),
182
+ "agent_created": _is_agent_created(skill_name),
183
+ })
184
+
185
+ # Sort by use_count descending
186
+ results.sort(key=lambda x: x["use_count"], reverse=True)
187
+ return results
188
+
189
+
190
+ def record_use(skill_name: str, action: str = "use"):
191
+ """Record a usage/view/patch event for a skill.
192
+
193
+ Args:
194
+ skill_name: Name of the skill
195
+ action: One of "use", "view", "patch"
196
+ """
197
+ if not _skill_exists(skill_name):
198
+ return {"status": "error", "message": f"Skill '{skill_name}' not found"}
199
+
200
+ if not _is_agent_created(skill_name):
201
+ return {"status": "skipped", "message": f"Skill '{skill_name}' is bundled/hub-installed"}
202
+
203
+ usage = _load_usage()
204
+ now = datetime.now(timezone.utc).isoformat()
205
+
206
+ if skill_name not in usage:
207
+ usage[skill_name] = {
208
+ "use_count": 0,
209
+ "view_count": 0,
210
+ "patch_count": 0,
211
+ "state": "active",
212
+ "pinned": False,
213
+ "created_at": now,
214
+ "last_used_at": None,
215
+ "last_viewed_at": None,
216
+ "last_patched_at": None,
217
+ "archived_at": None,
218
+ }
219
+
220
+ entry = usage[skill_name]
221
+
222
+ if action == "use":
223
+ entry["use_count"] = entry.get("use_count", 0) + 1
224
+ entry["last_used_at"] = now
225
+ elif action == "view":
226
+ entry["view_count"] = entry.get("view_count", 0) + 1
227
+ entry["last_viewed_at"] = now
228
+ elif action == "patch":
229
+ entry["patch_count"] = entry.get("patch_count", 0) + 1
230
+ entry["last_patched_at"] = now
231
+
232
+ # Reset state to active on any activity
233
+ if entry.get("state") in ("stale",):
234
+ entry["state"] = "active"
235
+
236
+ _save_usage(usage)
237
+ return {"status": "ok", "action": action, "skill": skill_name}
238
+
239
+
240
+ def pin_skill(skill_name: str) -> dict:
241
+ """Pin a skill to protect it from curator auto-transitions."""
242
+ if not _skill_exists(skill_name):
243
+ return {"status": "error", "message": f"Skill '{skill_name}' not found"}
244
+
245
+ if not _is_agent_created(skill_name):
246
+ return {
247
+ "status": "error",
248
+ "message": f"Cannot pin '{skill_name}' — bundled and hub-installed skills are never touched by the curator"
249
+ }
250
+
251
+ usage = _load_usage()
252
+ if skill_name not in usage:
253
+ usage[skill_name] = {
254
+ "use_count": 0, "view_count": 0, "patch_count": 0,
255
+ "state": "active", "pinned": False,
256
+ "created_at": datetime.now(timezone.utc).isoformat(),
257
+ "last_used_at": None, "last_viewed_at": None,
258
+ "last_patched_at": None, "archived_at": None,
259
+ }
260
+
261
+ usage[skill_name]["pinned"] = True
262
+ _save_usage(usage)
263
+ return {"status": "ok", "skill": skill_name, "pinned": True}
264
+
265
+
266
+ def unpin_skill(skill_name: str) -> dict:
267
+ """Unpin a skill to allow curator auto-transitions."""
268
+ usage = _load_usage()
269
+ if skill_name in usage:
270
+ usage[skill_name]["pinned"] = False
271
+ _save_usage(usage)
272
+ return {"status": "ok", "skill": skill_name, "pinned": False}
273
+ return {"status": "error", "message": f"Skill '{skill_name}' not found in usage data"}
274
+
275
+
276
+ def restore_skill(skill_name: str) -> dict:
277
+ """Restore an archived skill back to active."""
278
+ archive_path = ARCHIVE_DIR / skill_name
279
+ active_path = SKILLS_DIR / skill_name
280
+
281
+ if not archive_path.exists():
282
+ return {"status": "error", "message": f"Archived skill '{skill_name}' not found"}
283
+
284
+ # Check if a bundled/hub skill shadows it
285
+ if active_path.exists():
286
+ return {
287
+ "status": "error",
288
+ "message": f"Cannot restore '{skill_name}' — a skill with that name already exists in active tree"
289
+ }
290
+
291
+ # Move back
292
+ shutil.move(str(archive_path), str(active_path))
293
+
294
+ # Update usage
295
+ usage = _load_usage()
296
+ if skill_name in usage:
297
+ usage[skill_name]["state"] = "active"
298
+ usage[skill_name]["archived_at"] = None
299
+ _save_usage(usage)
300
+
301
+ return {"status": "ok", "skill": skill_name, "restored": True}
302
+
303
+
304
+ def get_reports(limit: int = 10) -> list[dict]:
305
+ """List past curator run reports."""
306
+ if not CURATOR_LOG_DIR.exists():
307
+ return []
308
+
309
+ dirs = sorted(CURATOR_LOG_DIR.iterdir(), reverse=True)[:limit]
310
+ reports = []
311
+
312
+ for d in dirs:
313
+ report_path = d / "REPORT.md"
314
+ run_path = d / "run.json"
315
+ report = {
316
+ "id": d.name,
317
+ "timestamp": d.name,
318
+ "has_report": report_path.exists(),
319
+ "has_run_data": run_path.exists(),
320
+ }
321
+ if run_path.exists():
322
+ try:
323
+ run_data = json.loads(run_path.read_text())
324
+ report["summary"] = {
325
+ "skills_reviewed": run_data.get("skills_reviewed", 0),
326
+ "archived": run_data.get("archived", 0),
327
+ "patched": run_data.get("patched", 0),
328
+ "consolidated": run_data.get("consolidated", 0),
329
+ }
330
+ except (json.JSONDecodeError, OSError):
331
+ pass
332
+ reports.append(report)
333
+
334
+ return reports
335
+
336
+
337
+ def get_report_detail(report_id: str) -> dict:
338
+ """Get detailed report for a specific run."""
339
+ report_dir = CURATOR_LOG_DIR / report_id
340
+ if not report_dir.exists():
341
+ return {"status": "error", "message": f"Report '{report_id}' not found"}
342
+
343
+ result = {"id": report_id, "timestamp": report_id}
344
+
345
+ report_path = report_dir / "REPORT.md"
346
+ if report_path.exists():
347
+ result["report_md"] = report_path.read_text()
348
+
349
+ run_path = report_dir / "run.json"
350
+ if run_path.exists():
351
+ try:
352
+ result["run_data"] = json.loads(run_path.read_text())
353
+ except (json.JSONDecodeError, OSError):
354
+ pass
355
+
356
+ return result
357
+
358
+
359
+ def run_curator_review(sync: bool = False, python_bin: str = "python3") -> dict:
360
+ """Trigger a curator review pass.
361
+
362
+ In sync mode, runs inline (blocking). In background mode, returns immediately
363
+ and the curator runs as a subprocess.
364
+ """
365
+ # Check if there's curator CLI
366
+ curator_script = Path.home() / ".hermes" / "hermes-agent" / "hermes_cli" / "curator.py"
367
+
368
+ if curator_script.exists():
369
+ # Upstream curator available
370
+ cmd = [python_bin, str(curator_script), "run"]
371
+ if sync:
372
+ cmd.append("--sync")
373
+ try:
374
+ result = subprocess.run(
375
+ cmd,
376
+ capture_output=True,
377
+ text=True,
378
+ timeout=600 if sync else 30,
379
+ )
380
+ return {
381
+ "status": "ok" if result.returncode == 0 else "error",
382
+ "message": result.stdout or result.stderr,
383
+ "returncode": result.returncode,
384
+ }
385
+ except subprocess.TimeoutExpired:
386
+ return {"status": "timeout", "message": "Curator review timed out"}
387
+ except FileNotFoundError:
388
+ pass # Fall through to dashboard-native curator
389
+
390
+ # Dashboard-native curator (lightweight version)
391
+ usage = _load_usage()
392
+ now = datetime.now(timezone.utc)
393
+ transitions = {"stale": [], "archived": []}
394
+
395
+ for skill_name, data in list(usage.items()):
396
+ if data.get("pinned"):
397
+ continue
398
+ if not _is_agent_created(skill_name):
399
+ continue
400
+
401
+ state = data.get("state", "active")
402
+ last_used = data.get("last_used_at") or data.get("last_viewed_at") or data.get("created_at")
403
+
404
+ if not last_used:
405
+ continue
406
+
407
+ try:
408
+ last_dt = datetime.fromisoformat(last_used)
409
+ except (ValueError, TypeError):
410
+ continue
411
+
412
+ days_unused = (now - last_dt).days
413
+ current_state = state
414
+
415
+ # Determine new state
416
+ if days_unused >= ARCHIVE_AFTER_DAYS and state != "archived":
417
+ data["state"] = "archived"
418
+ data["archived_at"] = now.isoformat()
419
+ transitions["archived"].append(skill_name)
420
+ # Actually move the directory
421
+ src = SKILLS_DIR / skill_name
422
+ dst = ARCHIVE_DIR / skill_name
423
+ if src.exists():
424
+ dst.parent.mkdir(parents=True, exist_ok=True)
425
+ shutil.move(str(src), str(dst))
426
+ elif days_unused >= STALE_AFTER_DAYS and state not in ("stale", "archived"):
427
+ data["state"] = "stale"
428
+ transitions["stale"].append(skill_name)
429
+
430
+ _save_usage(usage)
431
+
432
+ # Write report
433
+ ts = datetime.now().strftime("%Y%m%d-%H%M%S")
434
+ report_dir = CURATOR_LOG_DIR / ts
435
+ report_dir.mkdir(parents=True, exist_ok=True)
436
+
437
+ run_data = {
438
+ "timestamp": ts,
439
+ "skills_reviewed": len(usage),
440
+ "stale": len(transitions["stale"]),
441
+ "archived": len(transitions["archived"]),
442
+ "patched": 0,
443
+ "consolidated": 0,
444
+ "transitions": transitions,
445
+ }
446
+ (report_dir / "run.json").write_text(json.dumps(run_data, indent=2))
447
+
448
+ report_md = [
449
+ f"# Curator Report — {ts}",
450
+ "",
451
+ f"**Skills reviewed:** {len(usage)}",
452
+ f"**Marked stale:** {len(transitions['stale'])}",
453
+ f"**Archived:** {len(transitions['archived'])}",
454
+ "",
455
+ "## Transitions",
456
+ ]
457
+ if transitions["stale"]:
458
+ report_md.append("\n### → Stale")
459
+ for s in transitions["stale"]:
460
+ report_md.append(f"- `{s}`")
461
+ if transitions["archived"]:
462
+ report_md.append("\n### → Archived")
463
+ for s in transitions["archived"]:
464
+ report_md.append(f"- `{s}`")
465
+ if not transitions["stale"] and not transitions["archived"]:
466
+ report_md.append("\nNo transitions.")
467
+
468
+ (report_dir / "REPORT.md").write_text("\n".join(report_md))
469
+
470
+ return {
471
+ "status": "ok",
472
+ "report_id": ts,
473
+ "transitions": transitions,
474
+ }
475
+
476
+
477
+ def run_auto_check():
478
+ """Run deterministic auto-transitions (no LLM). Called by cron."""
479
+ return run_curator_review(sync=True)
480
+
481
+
482
+ def get_all_skills_with_usage() -> list[dict]:
483
+ """Enrich skill registry with usage data. Merges registry + usage info."""
484
+ usage = _load_usage()
485
+
486
+ # Get basic skill list from registry
487
+ from .skill_registry import get_registry
488
+ registry = get_registry()
489
+
490
+ results = []
491
+ for skill in registry.get_global_skills():
492
+ name = skill["name"]
493
+ u = usage.get(name, {})
494
+ results.append({
495
+ "name": name,
496
+ "description": skill.get("description", ""),
497
+ "tags": skill.get("tags", []),
498
+ "category": skill.get("canonical_path", "").split("/skills/")[-1].split("/")[0] if "/skills/" in skill.get("canonical_path", "") else "uncategorized",
499
+ "providers": [p["name"] for p in skill.get("providers", [])],
500
+ "agent_created": _is_agent_created(name),
501
+ "curator": {
502
+ "state": u.get("state", "untracked"),
503
+ "pinned": u.get("pinned", False),
504
+ "use_count": u.get("use_count", 0),
505
+ "view_count": u.get("view_count", 0),
506
+ "patch_count": u.get("patch_count", 0),
507
+ "last_used_at": u.get("last_used_at"),
508
+ "created_at": u.get("created_at"),
509
+ }
510
+ })
511
+
512
+ return results
@@ -0,0 +1,19 @@
1
+ """Evaluation engine for canonical runs."""
2
+
3
+ from backend.eval.scorers import (
4
+ EvalScore,
5
+ OutcomeScorer,
6
+ ToolEfficiencyScorer,
7
+ TokenCostScorer,
8
+ ErrorRecoveryScorer,
9
+ )
10
+ from backend.eval.engine import EvaluationEngine
11
+
12
+ __all__ = [
13
+ "EvalScore",
14
+ "OutcomeScorer",
15
+ "ToolEfficiencyScorer",
16
+ "TokenCostScorer",
17
+ "ErrorRecoveryScorer",
18
+ "EvaluationEngine",
19
+ ]