principles-disciple 1.8.1 → 1.8.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (508) hide show
  1. package/ADVANCED_CONFIG_ZH.md +97 -0
  2. package/AGENT_INSTALL.md +173 -0
  3. package/AGENT_INSTALL_EN.md +173 -0
  4. package/INSTALL.md +256 -0
  5. package/SKILL.md +63 -0
  6. package/docs/COMMAND_REFERENCE.md +76 -0
  7. package/docs/COMMAND_REFERENCE_EN.md +79 -0
  8. package/esbuild.config.js +75 -0
  9. package/openclaw.plugin.json +4 -4
  10. package/package.json +11 -13
  11. package/scripts/build-web.mjs +46 -0
  12. package/scripts/install-dependencies.cjs +47 -0
  13. package/scripts/sync-plugin.mjs +802 -0
  14. package/scripts/verify-build.mjs +109 -0
  15. package/src/agents/nocturnal-dreamer.md +152 -0
  16. package/src/agents/nocturnal-philosopher.md +138 -0
  17. package/src/agents/nocturnal-reflector.md +126 -0
  18. package/src/agents/nocturnal-scribe.md +164 -0
  19. package/src/commands/capabilities.ts +85 -0
  20. package/{dist/commands/context.js → src/commands/context.ts} +78 -38
  21. package/src/commands/evolution-status.ts +146 -0
  22. package/src/commands/export.ts +111 -0
  23. package/src/commands/focus.ts +533 -0
  24. package/src/commands/nocturnal-review.ts +311 -0
  25. package/src/commands/nocturnal-rollout.ts +763 -0
  26. package/src/commands/nocturnal-train.ts +1002 -0
  27. package/{dist/commands/pain.js → src/commands/pain.ts} +68 -49
  28. package/src/commands/principle-rollback.ts +27 -0
  29. package/{dist/commands/rollback.js → src/commands/rollback.ts} +44 -12
  30. package/src/commands/samples.ts +60 -0
  31. package/src/commands/strategy.ts +38 -0
  32. package/{dist/commands/thinking-os.js → src/commands/thinking-os.ts} +59 -36
  33. package/src/commands/workflow-debug.ts +128 -0
  34. package/{dist/config/defaults/runtime.js → src/config/defaults/runtime.ts} +12 -5
  35. package/src/config/errors.ts +163 -0
  36. package/{dist/config/index.d.ts → src/config/index.ts} +2 -1
  37. package/src/constants/diagnostician.ts +66 -0
  38. package/src/constants/tools.ts +62 -0
  39. package/src/core/adaptive-thresholds.ts +476 -0
  40. package/{dist/core/config-service.js → src/core/config-service.ts} +7 -4
  41. package/{dist/core/config.js → src/core/config.ts} +158 -46
  42. package/src/core/control-ui-db.ts +435 -0
  43. package/{dist/core/detection-funnel.js → src/core/detection-funnel.ts} +36 -21
  44. package/{dist/core/detection-service.js → src/core/detection-service.ts} +7 -4
  45. package/{dist/core/dictionary-service.js → src/core/dictionary-service.ts} +7 -4
  46. package/{dist/core/dictionary.js → src/core/dictionary.ts} +57 -34
  47. package/src/core/empathy-keyword-matcher.ts +327 -0
  48. package/src/core/empathy-types.ts +218 -0
  49. package/src/core/event-log.ts +544 -0
  50. package/src/core/evolution-engine.ts +612 -0
  51. package/src/core/evolution-logger.ts +353 -0
  52. package/src/core/evolution-migration.ts +77 -0
  53. package/src/core/evolution-reducer.ts +731 -0
  54. package/src/core/evolution-types.ts +456 -0
  55. package/src/core/external-training-contract.ts +527 -0
  56. package/src/core/focus-history.ts +1458 -0
  57. package/src/core/hygiene/tracker.ts +117 -0
  58. package/{dist/core/init.js → src/core/init.ts} +39 -26
  59. package/src/core/local-worker-routing.ts +617 -0
  60. package/{dist/core/migration.js → src/core/migration.ts} +18 -11
  61. package/src/core/model-deployment-registry.ts +722 -0
  62. package/src/core/model-training-registry.ts +813 -0
  63. package/src/core/nocturnal-arbiter.ts +706 -0
  64. package/src/core/nocturnal-candidate-scoring.ts +392 -0
  65. package/src/core/nocturnal-compliance.ts +1075 -0
  66. package/src/core/nocturnal-dataset.ts +668 -0
  67. package/src/core/nocturnal-executability.ts +428 -0
  68. package/src/core/nocturnal-export.ts +390 -0
  69. package/{dist/core/nocturnal-paths.js → src/core/nocturnal-paths.ts} +49 -23
  70. package/src/core/nocturnal-trajectory-extractor.ts +484 -0
  71. package/src/core/nocturnal-trinity.ts +1384 -0
  72. package/src/core/pain.ts +122 -0
  73. package/{dist/core/path-resolver.js → src/core/path-resolver.ts} +157 -36
  74. package/{dist/core/paths.js → src/core/paths.ts} +13 -4
  75. package/src/core/principle-training-state.ts +450 -0
  76. package/src/core/profile.ts +226 -0
  77. package/src/core/promotion-gate.ts +822 -0
  78. package/{dist/core/risk-calculator.js → src/core/risk-calculator.ts} +42 -16
  79. package/{dist/core/session-tracker.js → src/core/session-tracker.ts} +175 -62
  80. package/src/core/shadow-observation-registry.ts +534 -0
  81. package/{dist/core/system-logger.js → src/core/system-logger.ts} +9 -5
  82. package/src/core/thinking-models.ts +217 -0
  83. package/src/core/training-program.ts +630 -0
  84. package/src/core/trajectory-types.ts +243 -0
  85. package/src/core/trajectory.ts +1673 -0
  86. package/{dist/core/workspace-context.js → src/core/workspace-context.ts} +57 -32
  87. package/src/hooks/bash-risk.ts +171 -0
  88. package/src/hooks/edit-verification.ts +295 -0
  89. package/src/hooks/gate-block-helper.ts +160 -0
  90. package/src/hooks/gate.ts +210 -0
  91. package/src/hooks/gfi-gate.ts +177 -0
  92. package/src/hooks/lifecycle.ts +326 -0
  93. package/{dist/hooks/llm.js → src/hooks/llm.ts} +160 -80
  94. package/src/hooks/message-sanitize.ts +45 -0
  95. package/src/hooks/pain.ts +384 -0
  96. package/src/hooks/progressive-trust-gate.ts +174 -0
  97. package/src/hooks/prompt.ts +920 -0
  98. package/src/hooks/subagent.ts +207 -0
  99. package/src/hooks/thinking-checkpoint.ts +73 -0
  100. package/src/hooks/trajectory-collector.ts +290 -0
  101. package/src/http/principles-console-route.ts +716 -0
  102. package/src/i18n/commands.ts +117 -0
  103. package/src/index.ts +694 -0
  104. package/src/service/central-database.ts +831 -0
  105. package/src/service/control-ui-query-service.ts +888 -0
  106. package/src/service/evolution-query-service.ts +405 -0
  107. package/src/service/evolution-worker.ts +1646 -0
  108. package/src/service/health-query-service.ts +836 -0
  109. package/{dist/service/nocturnal-runtime.js → src/service/nocturnal-runtime.ts} +235 -79
  110. package/src/service/nocturnal-service.ts +1015 -0
  111. package/src/service/nocturnal-target-selector.ts +532 -0
  112. package/src/service/phase3-input-filter.ts +237 -0
  113. package/src/service/runtime-summary-service.ts +757 -0
  114. package/src/service/subagent-workflow/deep-reflect-workflow-manager.ts +513 -0
  115. package/{dist/service/subagent-workflow/empathy-observer-workflow-manager.js → src/service/subagent-workflow/empathy-observer-workflow-manager.ts} +240 -117
  116. package/src/service/subagent-workflow/index.ts +51 -0
  117. package/src/service/subagent-workflow/nocturnal-workflow-manager.ts +856 -0
  118. package/src/service/subagent-workflow/runtime-direct-driver.ts +166 -0
  119. package/{dist/service/subagent-workflow/types.d.ts → src/service/subagent-workflow/types.ts} +137 -18
  120. package/src/service/subagent-workflow/workflow-store.ts +328 -0
  121. package/src/service/trajectory-service.ts +15 -0
  122. package/{dist/tools/critique-prompt.js → src/tools/critique-prompt.ts} +25 -8
  123. package/src/tools/deep-reflect.ts +349 -0
  124. package/{dist/tools/model-index.js → src/tools/model-index.ts} +33 -17
  125. package/src/types/event-types.ts +453 -0
  126. package/src/types/hygiene-types.ts +31 -0
  127. package/src/types/principle-tree-schema.ts +244 -0
  128. package/src/types/runtime-summary.ts +49 -0
  129. package/src/types.ts +74 -0
  130. package/src/utils/file-lock.ts +391 -0
  131. package/{dist/utils/glob-match.js → src/utils/glob-match.ts} +21 -20
  132. package/{dist/utils/hashing.js → src/utils/hashing.ts} +6 -4
  133. package/src/utils/io.ts +110 -0
  134. package/{dist/utils/nlp.js → src/utils/nlp.ts} +19 -12
  135. package/{dist/utils/plugin-logger.js → src/utils/plugin-logger.ts} +33 -8
  136. package/src/utils/subagent-probe.ts +94 -0
  137. package/templates/langs/en/skills/ai-sprint-orchestration/EXAMPLES.md +63 -0
  138. package/templates/langs/en/skills/ai-sprint-orchestration/REFERENCE.md +136 -0
  139. package/templates/langs/en/skills/ai-sprint-orchestration/SKILL.md +67 -0
  140. package/templates/langs/en/skills/ai-sprint-orchestration/references/agent-registry.json +214 -0
  141. package/templates/langs/en/skills/ai-sprint-orchestration/references/specs/bugfix-complex-template.json +107 -0
  142. package/templates/langs/en/skills/ai-sprint-orchestration/references/specs/feature-complex-template.json +107 -0
  143. package/templates/langs/en/skills/ai-sprint-orchestration/references/specs/workflow-validation-minimal-verify.json +105 -0
  144. package/templates/langs/en/skills/ai-sprint-orchestration/references/specs/workflow-validation-minimal.json +108 -0
  145. package/templates/langs/en/skills/ai-sprint-orchestration/references/workflow-v1-acceptance-checklist.md +58 -0
  146. package/templates/langs/en/skills/ai-sprint-orchestration/references/workflow-v1.4-work-unit-handoff.md +190 -0
  147. package/templates/langs/en/skills/ai-sprint-orchestration/runtime/.gitignore +2 -0
  148. package/templates/langs/en/skills/ai-sprint-orchestration/scripts/lib/archive.mjs +310 -0
  149. package/templates/langs/en/skills/ai-sprint-orchestration/scripts/lib/contract-enforcement.mjs +683 -0
  150. package/templates/langs/en/skills/ai-sprint-orchestration/scripts/lib/decision.mjs +604 -0
  151. package/templates/langs/en/skills/ai-sprint-orchestration/scripts/lib/state-store.mjs +32 -0
  152. package/templates/langs/en/skills/ai-sprint-orchestration/scripts/lib/task-specs.mjs +707 -0
  153. package/templates/langs/en/skills/ai-sprint-orchestration/scripts/run.mjs +3419 -0
  154. package/templates/langs/zh/skills/ai-sprint-orchestration/EXAMPLES.md +63 -0
  155. package/templates/langs/zh/skills/ai-sprint-orchestration/REFERENCE.md +136 -0
  156. package/templates/langs/zh/skills/ai-sprint-orchestration/SKILL.md +67 -0
  157. package/templates/langs/zh/skills/ai-sprint-orchestration/references/agent-registry.json +214 -0
  158. package/templates/langs/zh/skills/ai-sprint-orchestration/references/specs/bugfix-complex-template.json +107 -0
  159. package/templates/langs/zh/skills/ai-sprint-orchestration/references/specs/feature-complex-template.json +107 -0
  160. package/templates/langs/zh/skills/ai-sprint-orchestration/references/specs/workflow-validation-minimal-verify.json +105 -0
  161. package/templates/langs/zh/skills/ai-sprint-orchestration/references/specs/workflow-validation-minimal.json +108 -0
  162. package/templates/langs/zh/skills/ai-sprint-orchestration/references/workflow-v1-acceptance-checklist.md +58 -0
  163. package/templates/langs/zh/skills/ai-sprint-orchestration/references/workflow-v1.4-work-unit-handoff.md +190 -0
  164. package/templates/langs/zh/skills/ai-sprint-orchestration/runtime/.gitignore +2 -0
  165. package/templates/langs/zh/skills/ai-sprint-orchestration/scripts/lib/archive.mjs +310 -0
  166. package/templates/langs/zh/skills/ai-sprint-orchestration/scripts/lib/contract-enforcement.mjs +683 -0
  167. package/templates/langs/zh/skills/ai-sprint-orchestration/scripts/lib/decision.mjs +604 -0
  168. package/templates/langs/zh/skills/ai-sprint-orchestration/scripts/lib/state-store.mjs +32 -0
  169. package/templates/langs/zh/skills/ai-sprint-orchestration/scripts/lib/task-specs.mjs +707 -0
  170. package/templates/langs/zh/skills/ai-sprint-orchestration/scripts/run.mjs +3419 -0
  171. package/templates/langs/zh/skills/ai-sprint-orchestration/test/archive.test.mjs +230 -0
  172. package/templates/langs/zh/skills/ai-sprint-orchestration/test/contract-enforcement.test.mjs +672 -0
  173. package/templates/langs/zh/skills/ai-sprint-orchestration/test/decision.test.mjs +1321 -0
  174. package/templates/langs/zh/skills/ai-sprint-orchestration/test/run.test.mjs +1419 -0
  175. package/templates/langs/zh/skills/pd-diagnostician/SKILL.md +70 -1
  176. package/templates/pain_settings.json +2 -1
  177. package/tests/README.md +120 -0
  178. package/tests/build-artifacts.test.ts +111 -0
  179. package/tests/commands/evolution-status.test.ts +222 -0
  180. package/tests/commands/evolver.test.ts +22 -0
  181. package/tests/commands/export.test.ts +78 -0
  182. package/tests/commands/nocturnal-review.test.ts +448 -0
  183. package/tests/commands/nocturnal-train.test.ts +97 -0
  184. package/tests/commands/pain.test.ts +108 -0
  185. package/tests/commands/samples.test.ts +65 -0
  186. package/tests/commands/strategy.test.ts +34 -0
  187. package/tests/commands/thinking-os.test.ts +88 -0
  188. package/tests/core/adaptive-thresholds.test.ts +261 -0
  189. package/tests/core/config-service.test.ts +89 -0
  190. package/tests/core/config.test.ts +90 -0
  191. package/tests/core/control-ui-db.test.ts +75 -0
  192. package/tests/core/core-template-guidance.test.ts +21 -0
  193. package/tests/core/detection-funnel.test.ts +63 -0
  194. package/tests/core/detection-service.test.ts +50 -0
  195. package/tests/core/dictionary-service.test.ts +116 -0
  196. package/tests/core/dictionary.test.ts +168 -0
  197. package/tests/core/empathy-keyword-matcher.test.ts +209 -0
  198. package/tests/core/event-log.test.ts +181 -0
  199. package/tests/core/evolution-e2e.test.ts +58 -0
  200. package/tests/core/evolution-engine-gate-integration.test.ts +543 -0
  201. package/tests/core/evolution-engine.test.ts +562 -0
  202. package/tests/core/evolution-logger.test.ts +148 -0
  203. package/tests/core/evolution-migration.test.ts +50 -0
  204. package/tests/core/evolution-paths.test.ts +21 -0
  205. package/tests/core/evolution-reducer.detector-metadata.test.ts +602 -0
  206. package/tests/core/evolution-reducer.test.ts +180 -0
  207. package/tests/core/evolution-types-loop.test.ts +48 -0
  208. package/tests/core/evolution-user-stories.e2e.test.ts +249 -0
  209. package/tests/core/external-training-contract.test.ts +463 -0
  210. package/tests/core/focus-history.test.ts +682 -0
  211. package/tests/core/init-flatten.test.ts +69 -0
  212. package/tests/core/init-refactor.test.ts +87 -0
  213. package/tests/core/init-v1.3.test.ts +46 -0
  214. package/tests/core/init.test.ts +190 -0
  215. package/tests/core/local-worker-routing.test.ts +757 -0
  216. package/tests/core/migration.test.ts +84 -0
  217. package/tests/core/model-deployment-registry.test.ts +845 -0
  218. package/tests/core/model-training-registry.test.ts +889 -0
  219. package/tests/core/nocturnal-arbiter.test.ts +494 -0
  220. package/tests/core/nocturnal-candidate-scoring.test.ts +400 -0
  221. package/tests/core/nocturnal-compliance.test.ts +646 -0
  222. package/tests/core/nocturnal-dataset.test.ts +892 -0
  223. package/tests/core/nocturnal-executability.test.ts +357 -0
  224. package/tests/core/nocturnal-export.test.ts +462 -0
  225. package/tests/core/nocturnal-reviewed-subset-comparison.test.ts +428 -0
  226. package/tests/core/nocturnal-trajectory-extractor.test.ts +634 -0
  227. package/tests/core/nocturnal-trinity.test.ts +953 -0
  228. package/tests/core/pain.test.ts +33 -0
  229. package/tests/core/path-resolver.test.ts +57 -0
  230. package/tests/core/paths-refactor.test.ts +42 -0
  231. package/tests/core/phase7-rollout-integration.test.ts +477 -0
  232. package/tests/core/principle-training-state.test.ts +712 -0
  233. package/tests/core/profile.test.ts +56 -0
  234. package/tests/core/promotion-gate.test.ts +556 -0
  235. package/tests/core/risk-calculator.test.ts +168 -0
  236. package/tests/core/session-tracker.test.ts +191 -0
  237. package/tests/core/training-program.test.ts +472 -0
  238. package/tests/core/trajectory.test.ts +265 -0
  239. package/tests/core/workspace-context-factory.test.ts +18 -0
  240. package/tests/core/workspace-context.test.ts +134 -0
  241. package/tests/fixtures/nocturnal-reviewed-subset.json +183 -0
  242. package/tests/fixtures/production-compatibility.test.ts +147 -0
  243. package/tests/fixtures/production-mock-generator.ts +282 -0
  244. package/tests/hooks/bash-risk-integration.test.ts +137 -0
  245. package/tests/hooks/bash-risk.test.ts +81 -0
  246. package/tests/hooks/edit-verification.test.ts +678 -0
  247. package/tests/hooks/gate-edit-verification-p1.test.ts +632 -0
  248. package/tests/hooks/gate-edit-verification.test.ts +435 -0
  249. package/tests/hooks/gate-pipeline-integration.test.ts +404 -0
  250. package/tests/hooks/gate.test.ts +271 -0
  251. package/tests/hooks/gfi-gate-unit.test.ts +422 -0
  252. package/tests/hooks/gfi-gate.test.ts +669 -0
  253. package/tests/hooks/lifecycle.test.ts +248 -0
  254. package/tests/hooks/llm.test.ts +308 -0
  255. package/tests/hooks/message-sanitize.test.ts +36 -0
  256. package/tests/hooks/pain.test.ts +141 -0
  257. package/tests/hooks/progressive-trust-gate.test.ts +277 -0
  258. package/tests/hooks/prompt.test.ts +1411 -0
  259. package/tests/hooks/subagent.test.ts +467 -0
  260. package/tests/hooks/thinking-gate.test.ts +313 -0
  261. package/tests/http/principles-console-route.test.ts +140 -0
  262. package/tests/hygiene-tracker.test.ts +77 -0
  263. package/tests/index.integration.test.ts +179 -0
  264. package/tests/index.shadow-routing.integration.test.ts +140 -0
  265. package/tests/index.test.ts +9 -0
  266. package/tests/integration/empathy-workflow-integration.test.ts +627 -0
  267. package/tests/service/control-ui-query-service.test.ts +121 -0
  268. package/tests/service/empathy-observer-workflow-manager.test.ts +176 -0
  269. package/tests/service/evolution-worker.test.ts +585 -0
  270. package/tests/service/nocturnal-runtime.test.ts +470 -0
  271. package/tests/service/nocturnal-service.test.ts +577 -0
  272. package/tests/service/nocturnal-target-selector.test.ts +615 -0
  273. package/tests/service/nocturnal-workflow-manager.test.ts +439 -0
  274. package/tests/service/phase3-input-filter.test.ts +289 -0
  275. package/tests/service/runtime-summary-service.test.ts +919 -0
  276. package/tests/task-compliance.test.ts +166 -0
  277. package/tests/test-utils.ts +48 -0
  278. package/tests/tools/critique-prompt.test.ts +260 -0
  279. package/tests/tools/deep-reflect.test.ts +232 -0
  280. package/tests/tools/model-index.test.ts +246 -0
  281. package/tests/ui/app.test.tsx +114 -0
  282. package/tests/utils/file-lock.test.ts +407 -0
  283. package/tests/utils/hashing.test.ts +32 -0
  284. package/tests/utils/io.test.ts +39 -0
  285. package/tests/utils/nlp.test.ts +53 -0
  286. package/tests/utils/plugin-logger.test.ts +156 -0
  287. package/tsconfig.json +16 -0
  288. package/tsconfig.tsbuildinfo +1 -0
  289. package/ui/src/App.tsx +45 -0
  290. package/ui/src/api.ts +216 -0
  291. package/ui/src/charts.tsx +586 -0
  292. package/ui/src/components/ErrorState.tsx +6 -0
  293. package/ui/src/components/Loading.tsx +13 -0
  294. package/ui/src/components/ProtectedRoute.tsx +12 -0
  295. package/ui/src/components/Shell.tsx +91 -0
  296. package/ui/src/components/WorkspaceConfig.tsx +146 -0
  297. package/ui/src/components/index.ts +5 -0
  298. package/ui/src/context/auth.tsx +80 -0
  299. package/ui/src/context/theme.tsx +66 -0
  300. package/ui/src/hooks/useAutoRefresh.ts +39 -0
  301. package/ui/src/i18n/ui.ts +363 -0
  302. package/ui/src/main.tsx +16 -0
  303. package/ui/src/pages/EvolutionPage.tsx +352 -0
  304. package/ui/src/pages/FeedbackPage.tsx +140 -0
  305. package/ui/src/pages/GateMonitorPage.tsx +136 -0
  306. package/ui/src/pages/LoginPage.tsx +88 -0
  307. package/ui/src/pages/OverviewPage.tsx +238 -0
  308. package/ui/src/pages/SamplesPage.tsx +174 -0
  309. package/ui/src/pages/ThinkingModelsPage.tsx +127 -0
  310. package/ui/src/styles.css +1661 -0
  311. package/ui/src/types.ts +368 -0
  312. package/ui/src/utils/format.ts +15 -0
  313. package/vitest.config.ts +23 -0
  314. package/dist/commands/capabilities.d.ts +0 -3
  315. package/dist/commands/capabilities.js +0 -73
  316. package/dist/commands/context.d.ts +0 -5
  317. package/dist/commands/evolution-status.d.ts +0 -4
  318. package/dist/commands/evolution-status.js +0 -117
  319. package/dist/commands/evolver.d.ts +0 -9
  320. package/dist/commands/evolver.js +0 -26
  321. package/dist/commands/export.d.ts +0 -2
  322. package/dist/commands/export.js +0 -98
  323. package/dist/commands/focus.d.ts +0 -14
  324. package/dist/commands/focus.js +0 -457
  325. package/dist/commands/nocturnal-review.d.ts +0 -24
  326. package/dist/commands/nocturnal-review.js +0 -265
  327. package/dist/commands/nocturnal-rollout.d.ts +0 -27
  328. package/dist/commands/nocturnal-rollout.js +0 -671
  329. package/dist/commands/nocturnal-train.d.ts +0 -25
  330. package/dist/commands/nocturnal-train.js +0 -919
  331. package/dist/commands/pain.d.ts +0 -5
  332. package/dist/commands/principle-rollback.d.ts +0 -4
  333. package/dist/commands/principle-rollback.js +0 -22
  334. package/dist/commands/rollback.d.ts +0 -19
  335. package/dist/commands/samples.d.ts +0 -2
  336. package/dist/commands/samples.js +0 -55
  337. package/dist/commands/strategy.d.ts +0 -3
  338. package/dist/commands/strategy.js +0 -29
  339. package/dist/commands/thinking-os.d.ts +0 -2
  340. package/dist/config/defaults/runtime.d.ts +0 -40
  341. package/dist/config/errors.d.ts +0 -84
  342. package/dist/config/errors.js +0 -94
  343. package/dist/config/index.js +0 -7
  344. package/dist/constants/diagnostician.d.ts +0 -12
  345. package/dist/constants/diagnostician.js +0 -56
  346. package/dist/constants/tools.d.ts +0 -17
  347. package/dist/constants/tools.js +0 -54
  348. package/dist/core/adaptive-thresholds.d.ts +0 -186
  349. package/dist/core/adaptive-thresholds.js +0 -300
  350. package/dist/core/config-service.d.ts +0 -15
  351. package/dist/core/config.d.ts +0 -129
  352. package/dist/core/control-ui-db.d.ts +0 -95
  353. package/dist/core/control-ui-db.js +0 -292
  354. package/dist/core/detection-funnel.d.ts +0 -33
  355. package/dist/core/detection-service.d.ts +0 -15
  356. package/dist/core/dictionary-service.d.ts +0 -15
  357. package/dist/core/dictionary.d.ts +0 -38
  358. package/dist/core/event-log.d.ts +0 -82
  359. package/dist/core/event-log.js +0 -463
  360. package/dist/core/evolution-engine.d.ts +0 -118
  361. package/dist/core/evolution-engine.js +0 -464
  362. package/dist/core/evolution-logger.d.ts +0 -137
  363. package/dist/core/evolution-logger.js +0 -256
  364. package/dist/core/evolution-migration.d.ts +0 -5
  365. package/dist/core/evolution-migration.js +0 -65
  366. package/dist/core/evolution-reducer.d.ts +0 -98
  367. package/dist/core/evolution-reducer.js +0 -465
  368. package/dist/core/evolution-types.d.ts +0 -287
  369. package/dist/core/evolution-types.js +0 -78
  370. package/dist/core/external-training-contract.d.ts +0 -276
  371. package/dist/core/external-training-contract.js +0 -269
  372. package/dist/core/focus-history.d.ts +0 -210
  373. package/dist/core/focus-history.js +0 -1185
  374. package/dist/core/hygiene/tracker.d.ts +0 -22
  375. package/dist/core/hygiene/tracker.js +0 -106
  376. package/dist/core/init.d.ts +0 -12
  377. package/dist/core/local-worker-routing.d.ts +0 -175
  378. package/dist/core/local-worker-routing.js +0 -525
  379. package/dist/core/migration.d.ts +0 -6
  380. package/dist/core/model-deployment-registry.d.ts +0 -218
  381. package/dist/core/model-deployment-registry.js +0 -503
  382. package/dist/core/model-training-registry.d.ts +0 -295
  383. package/dist/core/model-training-registry.js +0 -475
  384. package/dist/core/nocturnal-arbiter.d.ts +0 -159
  385. package/dist/core/nocturnal-arbiter.js +0 -534
  386. package/dist/core/nocturnal-candidate-scoring.d.ts +0 -137
  387. package/dist/core/nocturnal-candidate-scoring.js +0 -266
  388. package/dist/core/nocturnal-compliance.d.ts +0 -175
  389. package/dist/core/nocturnal-compliance.js +0 -824
  390. package/dist/core/nocturnal-dataset.d.ts +0 -224
  391. package/dist/core/nocturnal-dataset.js +0 -443
  392. package/dist/core/nocturnal-executability.d.ts +0 -85
  393. package/dist/core/nocturnal-executability.js +0 -331
  394. package/dist/core/nocturnal-export.d.ts +0 -124
  395. package/dist/core/nocturnal-export.js +0 -275
  396. package/dist/core/nocturnal-paths.d.ts +0 -124
  397. package/dist/core/nocturnal-trajectory-extractor.d.ts +0 -242
  398. package/dist/core/nocturnal-trajectory-extractor.js +0 -307
  399. package/dist/core/nocturnal-trinity.d.ts +0 -311
  400. package/dist/core/nocturnal-trinity.js +0 -880
  401. package/dist/core/pain.d.ts +0 -4
  402. package/dist/core/pain.js +0 -70
  403. package/dist/core/path-resolver.d.ts +0 -46
  404. package/dist/core/paths.d.ts +0 -65
  405. package/dist/core/principle-training-state.d.ts +0 -121
  406. package/dist/core/principle-training-state.js +0 -321
  407. package/dist/core/profile.d.ts +0 -62
  408. package/dist/core/profile.js +0 -210
  409. package/dist/core/promotion-gate.d.ts +0 -238
  410. package/dist/core/promotion-gate.js +0 -529
  411. package/dist/core/risk-calculator.d.ts +0 -22
  412. package/dist/core/session-tracker.d.ts +0 -101
  413. package/dist/core/shadow-observation-registry.d.ts +0 -217
  414. package/dist/core/shadow-observation-registry.js +0 -308
  415. package/dist/core/system-logger.d.ts +0 -8
  416. package/dist/core/thinking-models.d.ts +0 -38
  417. package/dist/core/thinking-models.js +0 -170
  418. package/dist/core/training-program.d.ts +0 -233
  419. package/dist/core/training-program.js +0 -433
  420. package/dist/core/trajectory.d.ts +0 -411
  421. package/dist/core/trajectory.js +0 -1307
  422. package/dist/core/workspace-context.d.ts +0 -71
  423. package/dist/hooks/bash-risk.d.ts +0 -57
  424. package/dist/hooks/bash-risk.js +0 -137
  425. package/dist/hooks/edit-verification.d.ts +0 -62
  426. package/dist/hooks/edit-verification.js +0 -256
  427. package/dist/hooks/gate-block-helper.d.ts +0 -44
  428. package/dist/hooks/gate-block-helper.js +0 -119
  429. package/dist/hooks/gate.d.ts +0 -24
  430. package/dist/hooks/gate.js +0 -173
  431. package/dist/hooks/gfi-gate.d.ts +0 -40
  432. package/dist/hooks/gfi-gate.js +0 -113
  433. package/dist/hooks/lifecycle.d.ts +0 -5
  434. package/dist/hooks/lifecycle.js +0 -284
  435. package/dist/hooks/llm.d.ts +0 -13
  436. package/dist/hooks/message-sanitize.d.ts +0 -3
  437. package/dist/hooks/message-sanitize.js +0 -37
  438. package/dist/hooks/pain.d.ts +0 -5
  439. package/dist/hooks/pain.js +0 -301
  440. package/dist/hooks/progressive-trust-gate.d.ts +0 -52
  441. package/dist/hooks/progressive-trust-gate.js +0 -134
  442. package/dist/hooks/prompt.d.ts +0 -49
  443. package/dist/hooks/prompt.js +0 -905
  444. package/dist/hooks/subagent.d.ts +0 -10
  445. package/dist/hooks/subagent.js +0 -387
  446. package/dist/hooks/thinking-checkpoint.d.ts +0 -37
  447. package/dist/hooks/thinking-checkpoint.js +0 -51
  448. package/dist/hooks/trajectory-collector.d.ts +0 -32
  449. package/dist/hooks/trajectory-collector.js +0 -256
  450. package/dist/http/principles-console-route.d.ts +0 -9
  451. package/dist/http/principles-console-route.js +0 -681
  452. package/dist/i18n/commands.d.ts +0 -26
  453. package/dist/i18n/commands.js +0 -116
  454. package/dist/index.d.ts +0 -7
  455. package/dist/index.js +0 -581
  456. package/dist/service/central-database.d.ts +0 -104
  457. package/dist/service/central-database.js +0 -649
  458. package/dist/service/control-ui-query-service.d.ts +0 -221
  459. package/dist/service/control-ui-query-service.js +0 -543
  460. package/dist/service/empathy-observer-manager.d.ts +0 -88
  461. package/dist/service/empathy-observer-manager.js +0 -414
  462. package/dist/service/evolution-query-service.d.ts +0 -155
  463. package/dist/service/evolution-query-service.js +0 -258
  464. package/dist/service/evolution-worker.d.ts +0 -101
  465. package/dist/service/evolution-worker.js +0 -975
  466. package/dist/service/health-query-service.d.ts +0 -170
  467. package/dist/service/health-query-service.js +0 -662
  468. package/dist/service/nocturnal-runtime.d.ts +0 -183
  469. package/dist/service/nocturnal-service.d.ts +0 -163
  470. package/dist/service/nocturnal-service.js +0 -787
  471. package/dist/service/nocturnal-target-selector.d.ts +0 -145
  472. package/dist/service/nocturnal-target-selector.js +0 -315
  473. package/dist/service/phase3-input-filter.d.ts +0 -73
  474. package/dist/service/phase3-input-filter.js +0 -172
  475. package/dist/service/runtime-summary-service.d.ts +0 -122
  476. package/dist/service/runtime-summary-service.js +0 -485
  477. package/dist/service/subagent-workflow/empathy-observer-workflow-manager.d.ts +0 -48
  478. package/dist/service/subagent-workflow/index.d.ts +0 -4
  479. package/dist/service/subagent-workflow/index.js +0 -3
  480. package/dist/service/subagent-workflow/runtime-direct-driver.d.ts +0 -77
  481. package/dist/service/subagent-workflow/runtime-direct-driver.js +0 -75
  482. package/dist/service/subagent-workflow/types.js +0 -11
  483. package/dist/service/subagent-workflow/workflow-store.d.ts +0 -26
  484. package/dist/service/subagent-workflow/workflow-store.js +0 -165
  485. package/dist/service/trajectory-service.d.ts +0 -2
  486. package/dist/service/trajectory-service.js +0 -15
  487. package/dist/tools/critique-prompt.d.ts +0 -14
  488. package/dist/tools/deep-reflect.d.ts +0 -39
  489. package/dist/tools/deep-reflect.js +0 -350
  490. package/dist/tools/model-index.d.ts +0 -9
  491. package/dist/types/event-types.d.ts +0 -306
  492. package/dist/types/event-types.js +0 -106
  493. package/dist/types/hygiene-types.d.ts +0 -20
  494. package/dist/types/hygiene-types.js +0 -12
  495. package/dist/types/runtime-summary.d.ts +0 -47
  496. package/dist/types/runtime-summary.js +0 -1
  497. package/dist/types.d.ts +0 -50
  498. package/dist/types.js +0 -22
  499. package/dist/utils/file-lock.d.ts +0 -71
  500. package/dist/utils/file-lock.js +0 -309
  501. package/dist/utils/glob-match.d.ts +0 -28
  502. package/dist/utils/hashing.d.ts +0 -9
  503. package/dist/utils/io.d.ts +0 -6
  504. package/dist/utils/io.js +0 -106
  505. package/dist/utils/nlp.d.ts +0 -9
  506. package/dist/utils/plugin-logger.d.ts +0 -39
  507. package/dist/utils/subagent-probe.d.ts +0 -34
  508. package/dist/utils/subagent-probe.js +0 -81
@@ -0,0 +1,831 @@
1
+ import Database from 'better-sqlite3';
2
+ import fs from 'fs';
3
+ import path from 'path';
4
+ import os from 'os';
5
+ import { WorkspaceNotFoundError } from '../config/index.js';
6
+
7
+ const CENTRAL_DB_DIR = '.central';
8
+ const CENTRAL_DB_NAME = 'aggregated.db';
9
+
10
+ export interface WorkspaceInfo {
11
+ name: string;
12
+ path: string;
13
+ lastSync: string | null;
14
+ }
15
+
16
+ /**
17
+ * Central database that aggregates data from all agent workspaces.
18
+ * Stored in ~/.openclaw/.central/ (NOT in memory/ which is for embeddings)
19
+ */
20
+ export class CentralDatabase {
21
+ private readonly dbPath: string;
22
+ private readonly db: Database.Database;
23
+ private readonly workspaces: WorkspaceInfo[] = [];
24
+
25
+ constructor() {
26
+ const openClawDir = os.homedir();
27
+ this.dbPath = path.join(openClawDir, '.openclaw', CENTRAL_DB_DIR, CENTRAL_DB_NAME);
28
+
29
+ // Ensure directory exists
30
+ fs.mkdirSync(path.dirname(this.dbPath), { recursive: true });
31
+
32
+ this.db = new Database(this.dbPath);
33
+ this.db.pragma('journal_mode = WAL');
34
+ this.db.pragma('synchronous = NORMAL');
35
+
36
+ this.initSchema();
37
+ this.discoverWorkspaces();
38
+ }
39
+
40
+ dispose(): void {
41
+ this.db.close();
42
+ }
43
+
44
+ private tableExists(db: Database.Database, tableName: string): boolean {
45
+ const result = db.prepare(`
46
+ SELECT name FROM sqlite_master WHERE type='table' AND name=?
47
+ `).get(tableName);
48
+ return !!result;
49
+ }
50
+
51
+ private initSchema(): void {
52
+ this.db.exec(`
53
+ CREATE TABLE IF NOT EXISTS schema_version (
54
+ version INTEGER NOT NULL
55
+ );
56
+
57
+ CREATE TABLE IF NOT EXISTS workspaces (
58
+ name TEXT PRIMARY KEY,
59
+ path TEXT NOT NULL,
60
+ last_sync TEXT
61
+ );
62
+
63
+ CREATE TABLE IF NOT EXISTS workspace_config (
64
+ workspace_name TEXT PRIMARY KEY,
65
+ enabled INTEGER NOT NULL DEFAULT 1,
66
+ display_name TEXT,
67
+ sync_enabled INTEGER NOT NULL DEFAULT 1,
68
+ created_at TEXT NOT NULL DEFAULT (datetime('now')),
69
+ updated_at TEXT NOT NULL DEFAULT (datetime('now'))
70
+ );
71
+
72
+ CREATE TABLE IF NOT EXISTS global_config (
73
+ key TEXT PRIMARY KEY,
74
+ value TEXT NOT NULL,
75
+ updated_at TEXT NOT NULL DEFAULT (datetime('now'))
76
+ );
77
+
78
+ CREATE TABLE IF NOT EXISTS aggregated_sessions (
79
+ session_id TEXT PRIMARY KEY,
80
+ workspace TEXT NOT NULL,
81
+ started_at TEXT NOT NULL,
82
+ updated_at TEXT NOT NULL
83
+ );
84
+
85
+ CREATE TABLE IF NOT EXISTS aggregated_tool_calls (
86
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
87
+ workspace TEXT NOT NULL,
88
+ session_id TEXT NOT NULL,
89
+ tool_name TEXT NOT NULL,
90
+ outcome TEXT NOT NULL,
91
+ duration_ms INTEGER,
92
+ error_type TEXT,
93
+ error_message TEXT,
94
+ created_at TEXT NOT NULL
95
+ );
96
+
97
+ CREATE TABLE IF NOT EXISTS aggregated_pain_events (
98
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
99
+ workspace TEXT NOT NULL,
100
+ session_id TEXT NOT NULL,
101
+ source TEXT NOT NULL,
102
+ score REAL NOT NULL,
103
+ reason TEXT,
104
+ created_at TEXT NOT NULL
105
+ );
106
+
107
+ CREATE TABLE IF NOT EXISTS aggregated_user_corrections (
108
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
109
+ workspace TEXT NOT NULL,
110
+ session_id TEXT NOT NULL,
111
+ correction_cue TEXT,
112
+ created_at TEXT NOT NULL
113
+ );
114
+
115
+ CREATE TABLE IF NOT EXISTS aggregated_principle_events (
116
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
117
+ workspace TEXT NOT NULL,
118
+ principle_id TEXT,
119
+ event_type TEXT NOT NULL,
120
+ created_at TEXT NOT NULL
121
+ );
122
+
123
+ CREATE TABLE IF NOT EXISTS aggregated_thinking_events (
124
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
125
+ workspace TEXT NOT NULL,
126
+ session_id TEXT NOT NULL,
127
+ model_id TEXT NOT NULL,
128
+ matched_pattern TEXT NOT NULL,
129
+ created_at TEXT NOT NULL
130
+ );
131
+
132
+ CREATE TABLE IF NOT EXISTS aggregated_correction_samples (
133
+ sample_id TEXT PRIMARY KEY,
134
+ workspace TEXT NOT NULL,
135
+ session_id TEXT NOT NULL,
136
+ bad_assistant_turn_id INTEGER NOT NULL,
137
+ quality_score REAL,
138
+ review_status TEXT,
139
+ created_at TEXT NOT NULL
140
+ );
141
+
142
+ CREATE TABLE IF NOT EXISTS aggregated_task_outcomes (
143
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
144
+ workspace TEXT NOT NULL,
145
+ session_id TEXT NOT NULL,
146
+ task_id TEXT,
147
+ outcome TEXT NOT NULL,
148
+ created_at TEXT NOT NULL
149
+ );
150
+
151
+ CREATE TABLE IF NOT EXISTS sync_log (
152
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
153
+ workspace TEXT NOT NULL,
154
+ synced_at TEXT NOT NULL,
155
+ records_synced INTEGER NOT NULL
156
+ );
157
+
158
+ -- Indexes for performance
159
+ CREATE INDEX IF NOT EXISTS idx_tool_calls_workspace ON aggregated_tool_calls(workspace);
160
+ CREATE INDEX IF NOT EXISTS idx_tool_calls_outcome ON aggregated_tool_calls(outcome);
161
+ CREATE INDEX IF NOT EXISTS idx_tool_calls_created ON aggregated_tool_calls(created_at);
162
+ CREATE INDEX IF NOT EXISTS idx_pain_workspace ON aggregated_pain_events(workspace);
163
+ CREATE INDEX IF NOT EXISTS idx_pain_created ON aggregated_pain_events(created_at);
164
+ CREATE INDEX IF NOT EXISTS idx_thinking_workspace ON aggregated_thinking_events(workspace);
165
+ CREATE INDEX IF NOT EXISTS idx_thinking_model ON aggregated_thinking_events(model_id);
166
+ CREATE INDEX IF NOT EXISTS idx_corrections_workspace ON aggregated_correction_samples(workspace);
167
+ CREATE INDEX IF NOT EXISTS idx_sessions_workspace ON aggregated_sessions(workspace);
168
+ `);
169
+ }
170
+
171
+ private discoverWorkspaces(): void {
172
+ const openClawDir = os.homedir();
173
+ const workspacesDir = path.join(openClawDir, '.openclaw');
174
+
175
+ this.workspaces.length = 0;
176
+
177
+ const entries = fs.readdirSync(workspacesDir);
178
+ for (const entry of entries) {
179
+ if (entry.startsWith('workspace-') && entry !== 'workspace') {
180
+ const workspacePath = path.join(workspacesDir, entry);
181
+ const stat = fs.statSync(workspacePath);
182
+ if (stat.isDirectory()) {
183
+ this.workspaces.push({
184
+ name: entry,
185
+ path: workspacePath,
186
+ lastSync: null,
187
+ });
188
+ }
189
+ }
190
+ }
191
+ }
192
+
193
+ /**
194
+ * Sync data from a single workspace into the central database
195
+ */
196
+ syncWorkspace(workspaceName: string): number {
197
+ const workspace = this.workspaces.find(w => w.name === workspaceName);
198
+ if (!workspace) {
199
+ throw new WorkspaceNotFoundError(workspaceName);
200
+ }
201
+
202
+ const trajectoryDbPath = path.join(workspace.path, '.state', 'trajectory.db');
203
+ if (!fs.existsSync(trajectoryDbPath)) {
204
+ return 0;
205
+ }
206
+
207
+ const sourceDb = new Database(trajectoryDbPath, { readonly: true });
208
+ let totalSynced = 0;
209
+
210
+ try {
211
+ // Sync sessions
212
+ const sessions = sourceDb.prepare(`
213
+ SELECT session_id, started_at, updated_at FROM sessions
214
+ `).all() as Array<{session_id: string; started_at: string; updated_at: string}>;
215
+
216
+ const insertSession = this.db.prepare(`
217
+ INSERT OR REPLACE INTO aggregated_sessions (session_id, workspace, started_at, updated_at)
218
+ VALUES (?, ?, ?, ?)
219
+ `);
220
+
221
+ for (const s of sessions) {
222
+ insertSession.run(s.session_id, workspaceName, s.started_at, s.updated_at);
223
+ totalSynced++;
224
+ }
225
+
226
+ // Sync tool_calls
227
+ const toolCalls = sourceDb.prepare(`
228
+ SELECT session_id, tool_name, outcome, duration_ms, error_type, error_message, created_at
229
+ FROM tool_calls
230
+ `).all() as Array<{
231
+ session_id: string; tool_name: string; outcome: string;
232
+ duration_ms: number | null; error_type: string | null;
233
+ error_message: string | null; created_at: string
234
+ }>;
235
+
236
+ const insertTool = this.db.prepare(`
237
+ INSERT INTO aggregated_tool_calls
238
+ (workspace, session_id, tool_name, outcome, duration_ms, error_type, error_message, created_at)
239
+ VALUES (?, ?, ?, ?, ?, ?, ?, ?)
240
+ `);
241
+
242
+ for (const t of toolCalls) {
243
+ insertTool.run(
244
+ workspaceName, t.session_id, t.tool_name, t.outcome,
245
+ t.duration_ms, t.error_type, t.error_message, t.created_at
246
+ );
247
+ totalSynced++;
248
+ }
249
+
250
+ // Sync pain_events
251
+ const painEvents = sourceDb.prepare(`
252
+ SELECT session_id, source, score, reason, created_at FROM pain_events
253
+ `).all() as Array<{
254
+ session_id: string; source: string; score: number;
255
+ reason: string | null; created_at: string
256
+ }>;
257
+
258
+ const insertPain = this.db.prepare(`
259
+ INSERT INTO aggregated_pain_events (workspace, session_id, source, score, reason, created_at)
260
+ VALUES (?, ?, ?, ?, ?, ?)
261
+ `);
262
+
263
+ for (const p of painEvents) {
264
+ insertPain.run(workspaceName, p.session_id, p.source, p.score, p.reason, p.created_at);
265
+ totalSynced++;
266
+ }
267
+
268
+ // Sync user corrections
269
+ const corrections = sourceDb.prepare(`
270
+ SELECT session_id, correction_cue, created_at FROM user_turns
271
+ WHERE correction_detected = 1
272
+ `).all() as Array<{
273
+ session_id: string; correction_cue: string | null; created_at: string
274
+ }>;
275
+
276
+ const insertCorr = this.db.prepare(`
277
+ INSERT INTO aggregated_user_corrections (workspace, session_id, correction_cue, created_at)
278
+ VALUES (?, ?, ?, ?)
279
+ `);
280
+
281
+ for (const c of corrections) {
282
+ insertCorr.run(workspaceName, c.session_id, c.correction_cue, c.created_at);
283
+ totalSynced++;
284
+ }
285
+
286
+ // Sync principle_events
287
+ const principles = sourceDb.prepare(`
288
+ SELECT principle_id, event_type, created_at FROM principle_events
289
+ `).all() as Array<{
290
+ principle_id: string | null; event_type: string; created_at: string
291
+ }>;
292
+
293
+ const insertPrinciple = this.db.prepare(`
294
+ INSERT INTO aggregated_principle_events (workspace, principle_id, event_type, created_at)
295
+ VALUES (?, ?, ?, ?)
296
+ `);
297
+
298
+ for (const p of principles) {
299
+ insertPrinciple.run(workspaceName, p.principle_id, p.event_type, p.created_at);
300
+ totalSynced++;
301
+ }
302
+
303
+ // Sync thinking_model_events (may not exist in older workspaces)
304
+ if (this.tableExists(sourceDb, 'thinking_model_events')) {
305
+ const thinking = sourceDb.prepare(`
306
+ SELECT session_id, model_id, matched_pattern, created_at FROM thinking_model_events
307
+ `).all() as Array<{
308
+ session_id: string; model_id: string; matched_pattern: string; created_at: string
309
+ }>;
310
+
311
+ const insertThinking = this.db.prepare(`
312
+ INSERT INTO aggregated_thinking_events (workspace, session_id, model_id, matched_pattern, created_at)
313
+ VALUES (?, ?, ?, ?, ?)
314
+ `);
315
+
316
+ for (const t of thinking) {
317
+ insertThinking.run(workspaceName, t.session_id, t.model_id, t.matched_pattern, t.created_at);
318
+ totalSynced++;
319
+ }
320
+ }
321
+
322
+ // Sync correction_samples
323
+ const samples = sourceDb.prepare(`
324
+ SELECT sample_id, session_id, bad_assistant_turn_id, quality_score, review_status, created_at
325
+ FROM correction_samples
326
+ `).all() as Array<{
327
+ sample_id: string; session_id: string; bad_assistant_turn_id: number;
328
+ quality_score: number | null; review_status: string | null; created_at: string
329
+ }>;
330
+
331
+ const insertSample = this.db.prepare(`
332
+ INSERT OR REPLACE INTO aggregated_correction_samples
333
+ (sample_id, workspace, session_id, bad_assistant_turn_id, quality_score, review_status, created_at)
334
+ VALUES (?, ?, ?, ?, ?, ?, ?)
335
+ `);
336
+
337
+ for (const s of samples) {
338
+ insertSample.run(
339
+ s.sample_id, workspaceName, s.session_id, s.bad_assistant_turn_id,
340
+ s.quality_score, s.review_status, s.created_at
341
+ );
342
+ totalSynced++;
343
+ }
344
+
345
+ // Sync task_outcomes
346
+ const outcomes = sourceDb.prepare(`
347
+ SELECT session_id, task_id, outcome, created_at FROM task_outcomes
348
+ `).all() as Array<{
349
+ session_id: string; task_id: string | null; outcome: string; created_at: string
350
+ }>;
351
+
352
+ const insertOutcome = this.db.prepare(`
353
+ INSERT INTO aggregated_task_outcomes (workspace, session_id, task_id, outcome, created_at)
354
+ VALUES (?, ?, ?, ?, ?)
355
+ `);
356
+
357
+ for (const o of outcomes) {
358
+ insertOutcome.run(workspaceName, o.session_id, o.task_id, o.outcome, o.created_at);
359
+ totalSynced++;
360
+ }
361
+
362
+ // Update last sync time
363
+ this.db.prepare(`
364
+ INSERT OR REPLACE INTO workspaces (name, path, last_sync)
365
+ VALUES (?, ?, datetime('now'))
366
+ `).run(workspaceName, workspace.path);
367
+
368
+ // Log sync
369
+ this.db.prepare(`
370
+ INSERT INTO sync_log (workspace, synced_at, records_synced)
371
+ VALUES (?, datetime('now'), ?)
372
+ `).run(workspaceName, totalSynced);
373
+
374
+ return totalSynced;
375
+ } finally {
376
+ sourceDb.close();
377
+ }
378
+ }
379
+
380
+ syncEnabled(): Map<string, number> {
381
+ const results = new Map<string, number>();
382
+ for (const ws of this.getEnabledWorkspaces()) {
383
+ try {
384
+ const count = this.syncWorkspace(ws.name);
385
+ results.set(ws.name, count);
386
+ } catch (error) {
387
+ console.error(`Failed to sync workspace ${ws.name}:`, error);
388
+ results.set(ws.name, 0);
389
+ }
390
+ }
391
+ return results;
392
+ }
393
+
394
+ /**
395
+ * Sync all workspaces (legacy method - syncs all regardless of config)
396
+ */
397
+ syncAll(): Map<string, number> {
398
+ const results = new Map<string, number>();
399
+ for (const ws of this.workspaces) {
400
+ try {
401
+ const count = this.syncWorkspace(ws.name);
402
+ results.set(ws.name, count);
403
+ } catch (error) {
404
+ console.error(`Failed to sync workspace ${ws.name}:`, error);
405
+ results.set(ws.name, 0);
406
+ }
407
+ }
408
+ return results;
409
+ }
410
+
411
+ private getEnabledWorkspaceFilter(): string {
412
+ const enabled = this.getWorkspaceConfigs().filter(c => c.enabled && c.syncEnabled);
413
+ if (enabled.length === 0) return "''";
414
+ return enabled.map(c => `'${c.workspaceName.replace(/'/g, "''")}'`).join(', ');
415
+ }
416
+
417
+ /**
418
+ * Get aggregated overview stats (only from enabled workspaces)
419
+ */
420
+ getOverviewStats(): {
421
+ totalSessions: number;
422
+ totalToolCalls: number;
423
+ totalFailures: number;
424
+ totalPainEvents: number;
425
+ totalCorrections: number;
426
+ totalThinkingEvents: number;
427
+ totalSamples: number;
428
+ pendingSamples: number;
429
+ approvedSamples: number;
430
+ rejectedSamples: number;
431
+ workspaceCount: number;
432
+ enabledWorkspaceCount: number;
433
+ workspaceNames: string[];
434
+ enabledWorkspaceNames: string[];
435
+ } {
436
+ const filter = this.getEnabledWorkspaceFilter();
437
+
438
+ const totalSessions = this.db.prepare(`
439
+ SELECT COUNT(DISTINCT session_id) as count FROM aggregated_sessions
440
+ WHERE workspace IN (${filter})
441
+ `).get() as { count: number };
442
+
443
+ const toolStats = this.db.prepare(`
444
+ SELECT
445
+ COUNT(*) as total,
446
+ SUM(CASE WHEN outcome = 'failure' THEN 1 ELSE 0 END) as failures
447
+ FROM aggregated_tool_calls
448
+ WHERE workspace IN (${filter})
449
+ `).get() as { total: number; failures: number };
450
+
451
+ const painEvents = this.db.prepare(`
452
+ SELECT COUNT(*) as count FROM aggregated_pain_events
453
+ WHERE workspace IN (${filter})
454
+ `).get() as { count: number };
455
+
456
+ const corrections = this.db.prepare(`
457
+ SELECT COUNT(*) as count FROM aggregated_user_corrections
458
+ WHERE workspace IN (${filter})
459
+ `).get() as { count: number };
460
+
461
+ const thinkingEvents = this.db.prepare(`
462
+ SELECT COUNT(*) as count FROM aggregated_thinking_events
463
+ WHERE workspace IN (${filter})
464
+ `).get() as { count: number };
465
+
466
+ const sampleStats = this.db.prepare(`
467
+ SELECT
468
+ COUNT(*) as total,
469
+ SUM(CASE WHEN review_status = 'pending' THEN 1 ELSE 0 END) as pending,
470
+ SUM(CASE WHEN review_status = 'approved' THEN 1 ELSE 0 END) as approved,
471
+ SUM(CASE WHEN review_status = 'rejected' THEN 1 ELSE 0 END) as rejected
472
+ FROM aggregated_correction_samples
473
+ WHERE workspace IN (${filter})
474
+ `).get() as { total: number; pending: number; approved: number; rejected: number };
475
+
476
+ const workspaces = this.db.prepare(`
477
+ SELECT name FROM workspaces ORDER BY name
478
+ `).all() as Array<{ name: string }>;
479
+
480
+ const enabledConfigs = this.getWorkspaceConfigs().filter(c => c.enabled && c.syncEnabled);
481
+ const enabledWorkspaceNames = enabledConfigs.map(c => c.workspaceName);
482
+
483
+ return {
484
+ totalSessions: totalSessions.count,
485
+ totalToolCalls: toolStats.total,
486
+ totalFailures: toolStats.failures || 0,
487
+ totalPainEvents: painEvents.count,
488
+ totalCorrections: corrections.count,
489
+ totalThinkingEvents: thinkingEvents.count,
490
+ totalSamples: sampleStats.total,
491
+ pendingSamples: sampleStats.pending || 0,
492
+ approvedSamples: sampleStats.approved || 0,
493
+ rejectedSamples: sampleStats.rejected || 0,
494
+ workspaceCount: workspaces.length,
495
+ enabledWorkspaceCount: enabledConfigs.length,
496
+ workspaceNames: workspaces.map(w => w.name),
497
+ enabledWorkspaceNames,
498
+ };
499
+ }
500
+
501
+ /**
502
+ * Get daily trend data
503
+ */
504
+ getDailyTrend(days: number = 7): Array<{
505
+ day: string;
506
+ toolCalls: number;
507
+ failures: number;
508
+ userCorrections: number;
509
+ thinkingTurns: number;
510
+ }> {
511
+ const cutoffDate = new Date();
512
+ cutoffDate.setDate(cutoffDate.getDate() - days);
513
+ const cutoffStr = cutoffDate.toISOString().split('T')[0];
514
+
515
+ const toolDaily = this.db.prepare(`
516
+ SELECT
517
+ substr(created_at, 1, 10) as day,
518
+ COUNT(*) as tool_calls,
519
+ SUM(CASE WHEN outcome = 'failure' THEN 1 ELSE 0 END) as failures
520
+ FROM aggregated_tool_calls
521
+ WHERE substr(created_at, 1, 10) >= ?
522
+ GROUP BY substr(created_at, 1, 10)
523
+ ORDER BY day
524
+ `).all(cutoffStr) as Array<{
525
+ day: string; tool_calls: number; failures: number
526
+ }>;
527
+
528
+ const correctionsDaily = this.db.prepare(`
529
+ SELECT
530
+ substr(created_at, 1, 10) as day,
531
+ COUNT(*) as corrections
532
+ FROM aggregated_user_corrections
533
+ WHERE substr(created_at, 1, 10) >= ?
534
+ GROUP BY substr(created_at, 1, 10)
535
+ `).all(cutoffStr) as Array<{
536
+ day: string; corrections: number
537
+ }>;
538
+
539
+ const thinkingDaily = this.db.prepare(`
540
+ SELECT
541
+ substr(created_at, 1, 10) as day,
542
+ COUNT(*) as thinking_turns
543
+ FROM aggregated_thinking_events
544
+ WHERE substr(created_at, 1, 10) >= ?
545
+ GROUP BY substr(created_at, 1, 10)
546
+ `).all(cutoffStr) as Array<{
547
+ day: string; thinking_turns: number
548
+ }>;
549
+
550
+ // Merge all trends
551
+ const dayMap = new Map<string, {
552
+ day: string;
553
+ toolCalls: number;
554
+ failures: number;
555
+ userCorrections: number;
556
+ thinkingTurns: number;
557
+ }>();
558
+
559
+ for (const t of toolDaily) {
560
+ dayMap.set(t.day, {
561
+ day: t.day,
562
+ toolCalls: t.tool_calls,
563
+ failures: t.failures || 0,
564
+ userCorrections: 0,
565
+ thinkingTurns: 0,
566
+ });
567
+ }
568
+
569
+ for (const c of correctionsDaily) {
570
+ const existing = dayMap.get(c.day);
571
+ if (existing) {
572
+ existing.userCorrections = c.corrections;
573
+ } else {
574
+ dayMap.set(c.day, {
575
+ day: c.day,
576
+ toolCalls: 0,
577
+ failures: 0,
578
+ userCorrections: c.corrections,
579
+ thinkingTurns: 0,
580
+ });
581
+ }
582
+ }
583
+
584
+ for (const t of thinkingDaily) {
585
+ const existing = dayMap.get(t.day);
586
+ if (existing) {
587
+ existing.thinkingTurns = t.thinking_turns;
588
+ } else {
589
+ dayMap.set(t.day, {
590
+ day: t.day,
591
+ toolCalls: 0,
592
+ failures: 0,
593
+ userCorrections: 0,
594
+ thinkingTurns: t.thinking_turns,
595
+ });
596
+ }
597
+ }
598
+
599
+ return Array.from(dayMap.values()).sort((a, b) => a.day.localeCompare(b.day));
600
+ }
601
+
602
+ /**
603
+ * Get top regressions
604
+ */
605
+ getTopRegressions(limit: number = 5): Array<{
606
+ toolName: string;
607
+ errorType: string;
608
+ occurrences: number;
609
+ }> {
610
+ return this.db.prepare(`
611
+ SELECT
612
+ tool_name as toolName,
613
+ error_type as errorType,
614
+ COUNT(*) as occurrences
615
+ FROM aggregated_tool_calls
616
+ WHERE outcome = 'failure' AND error_type IS NOT NULL
617
+ GROUP BY tool_name, error_type
618
+ ORDER BY occurrences DESC
619
+ LIMIT ?
620
+ `).all(limit) as Array<{
621
+ toolName: string;
622
+ errorType: string;
623
+ occurrences: number;
624
+ }>;
625
+ }
626
+
627
+ /**
628
+ * Get thinking model stats
629
+ */
630
+ getThinkingModelStats(): {
631
+ totalModels: number;
632
+ activeModels: number;
633
+ models: Array<{
634
+ modelId: string;
635
+ hits: number;
636
+ coverageRate: number;
637
+ }>;
638
+ } {
639
+ const totalModels = this.db.prepare(`
640
+ SELECT COUNT(DISTINCT model_id) as count FROM aggregated_thinking_events
641
+ `).get() as { count: number };
642
+
643
+ // Consider a model "active" if it has events in the last 7 days
644
+ const recentDate = new Date();
645
+ recentDate.setDate(recentDate.getDate() - 7);
646
+ const recentStr = recentDate.toISOString();
647
+
648
+ const activeModels = this.db.prepare(`
649
+ SELECT COUNT(DISTINCT model_id) as count FROM aggregated_thinking_events
650
+ WHERE created_at >= ?
651
+ `).get(recentStr) as { count: number };
652
+
653
+ const totalToolCalls = this.db.prepare(`
654
+ SELECT COUNT(*) as count FROM aggregated_tool_calls
655
+ `).get() as { count: number };
656
+
657
+ const models = this.db.prepare(`
658
+ SELECT
659
+ model_id as modelId,
660
+ COUNT(*) as hits
661
+ FROM aggregated_thinking_events
662
+ GROUP BY model_id
663
+ ORDER BY hits DESC
664
+ `).all() as Array<{ modelId: string; hits: number }>;
665
+
666
+ const coverageRate = totalToolCalls.count > 0
667
+ ? models.reduce((sum, m) => sum + m.hits, 0) / totalToolCalls.count
668
+ : 0;
669
+
670
+ return {
671
+ totalModels: totalModels.count,
672
+ activeModels: activeModels.count,
673
+ models: models.map(m => ({
674
+ ...m,
675
+ coverageRate: totalToolCalls.count > 0 ? m.hits / totalToolCalls.count : 0,
676
+ })),
677
+ };
678
+ }
679
+
680
+ /**
681
+ * Get workspace list
682
+ */
683
+ getWorkspaces(): WorkspaceInfo[] {
684
+ return this.db.prepare(`
685
+ SELECT name, path, last_sync as lastSync FROM workspaces ORDER BY name
686
+ `).all() as WorkspaceInfo[];
687
+ }
688
+
689
+ getWorkspaceConfigs(): Array<{
690
+ workspaceName: string;
691
+ enabled: boolean;
692
+ displayName: string | null;
693
+ syncEnabled: boolean;
694
+ }> {
695
+ const configs = this.db.prepare(`
696
+ SELECT workspace_name, enabled, display_name, sync_enabled
697
+ FROM workspace_config
698
+ ORDER BY workspace_name
699
+ `).all() as Array<{
700
+ workspace_name: string;
701
+ enabled: number;
702
+ display_name: string | null;
703
+ sync_enabled: number;
704
+ }>;
705
+
706
+ return configs.map(c => ({
707
+ workspaceName: c.workspace_name,
708
+ enabled: c.enabled === 1,
709
+ displayName: c.display_name,
710
+ syncEnabled: c.sync_enabled === 1,
711
+ }));
712
+ }
713
+
714
+ updateWorkspaceConfig(
715
+ workspaceName: string,
716
+ updates: {
717
+ enabled?: boolean;
718
+ displayName?: string | null;
719
+ syncEnabled?: boolean;
720
+ }
721
+ ): void {
722
+ const existing = this.db.prepare(`
723
+ SELECT workspace_name FROM workspace_config WHERE workspace_name = ?
724
+ `).get(workspaceName);
725
+
726
+ if (existing) {
727
+ const setClauses: string[] = ['updated_at = datetime(\'now\')'];
728
+ const params: unknown[] = [];
729
+
730
+ if (updates.enabled !== undefined) {
731
+ setClauses.push('enabled = ?');
732
+ params.push(updates.enabled ? 1 : 0);
733
+ }
734
+ if (updates.displayName !== undefined) {
735
+ setClauses.push('display_name = ?');
736
+ params.push(updates.displayName);
737
+ }
738
+ if (updates.syncEnabled !== undefined) {
739
+ setClauses.push('sync_enabled = ?');
740
+ params.push(updates.syncEnabled ? 1 : 0);
741
+ }
742
+
743
+ params.push(workspaceName);
744
+ this.db.prepare(`
745
+ UPDATE workspace_config SET ${setClauses.join(', ')} WHERE workspace_name = ?
746
+ `).run(...params);
747
+ } else {
748
+ this.db.prepare(`
749
+ INSERT INTO workspace_config (workspace_name, enabled, display_name, sync_enabled)
750
+ VALUES (?, ?, ?, ?)
751
+ `).run(
752
+ workspaceName,
753
+ updates.enabled !== undefined ? (updates.enabled ? 1 : 0) : 1,
754
+ updates.displayName ?? null,
755
+ updates.syncEnabled !== undefined ? (updates.syncEnabled ? 1 : 0) : 1
756
+ );
757
+ }
758
+ }
759
+
760
+ isWorkspaceEnabled(workspaceName: string): boolean {
761
+ const config = this.db.prepare(`
762
+ SELECT enabled, sync_enabled FROM workspace_config WHERE workspace_name = ?
763
+ `).get(workspaceName) as { enabled: number; sync_enabled: number } | undefined;
764
+
765
+ if (!config) return true;
766
+ return config.enabled === 1 && config.sync_enabled === 1;
767
+ }
768
+
769
+ getEnabledWorkspaces(): WorkspaceInfo[] {
770
+ return this.workspaces.filter(ws => this.isWorkspaceEnabled(ws.name));
771
+ }
772
+
773
+ addCustomWorkspace(name: string, workspacePath: string): void {
774
+ if (!this.workspaces.find(ws => ws.name === name)) {
775
+ this.workspaces.push({ name, path: workspacePath, lastSync: null });
776
+ this.db.prepare(`
777
+ INSERT INTO workspaces (name, path, last_sync) VALUES (?, ?, NULL)
778
+ `).run(name, workspacePath);
779
+ this.db.prepare(`
780
+ INSERT INTO workspace_config (workspace_name, enabled, display_name, sync_enabled)
781
+ VALUES (?, 1, ?, 1)
782
+ `).run(name, name);
783
+ }
784
+ }
785
+
786
+ removeWorkspace(workspaceName: string): void {
787
+ this.updateWorkspaceConfig(workspaceName, { enabled: false, syncEnabled: false });
788
+ }
789
+
790
+ getGlobalConfig(key: string): string | null {
791
+ const result = this.db.prepare(`
792
+ SELECT value FROM global_config WHERE key = ?
793
+ `).get(key) as { value: string } | undefined;
794
+ return result?.value ?? null;
795
+ }
796
+
797
+ setGlobalConfig(key: string, value: string): void {
798
+ this.db.prepare(`
799
+ INSERT OR REPLACE INTO global_config (key, value, updated_at)
800
+ VALUES (?, ?, datetime('now'))
801
+ `).run(key, value);
802
+ }
803
+
804
+ /**
805
+ * Clear all aggregated data (for testing/reset)
806
+ */
807
+ clearAll(): void {
808
+ this.db.exec(`
809
+ DELETE FROM aggregated_sessions;
810
+ DELETE FROM aggregated_tool_calls;
811
+ DELETE FROM aggregated_pain_events;
812
+ DELETE FROM aggregated_user_corrections;
813
+ DELETE FROM aggregated_principle_events;
814
+ DELETE FROM aggregated_thinking_events;
815
+ DELETE FROM aggregated_correction_samples;
816
+ DELETE FROM aggregated_task_outcomes;
817
+ DELETE FROM workspaces;
818
+ DELETE FROM sync_log;
819
+ `);
820
+ }
821
+ }
822
+
823
+ // Singleton instance
824
+ let centralDbInstance: CentralDatabase | null = null;
825
+
826
+ export function getCentralDatabase(): CentralDatabase {
827
+ if (!centralDbInstance) {
828
+ centralDbInstance = new CentralDatabase();
829
+ }
830
+ return centralDbInstance;
831
+ }