principles-disciple 1.8.0 → 1.8.2

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 (460) 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 +6 -1
  10. package/package.json +13 -15
  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} +185 -63
  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} +166 -139
  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} +263 -36
  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/src/service/subagent-workflow/empathy-observer-workflow-manager.ts +603 -0
  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/src/service/subagent-workflow/types.ts +378 -0
  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/zh/skills/pd-diagnostician/SKILL.md +70 -1
  138. package/templates/pain_settings.json +2 -1
  139. package/tests/README.md +120 -0
  140. package/tests/build-artifacts.test.ts +111 -0
  141. package/tests/commands/evolution-status.test.ts +222 -0
  142. package/tests/commands/evolver.test.ts +22 -0
  143. package/tests/commands/export.test.ts +78 -0
  144. package/tests/commands/nocturnal-review.test.ts +448 -0
  145. package/tests/commands/nocturnal-train.test.ts +97 -0
  146. package/tests/commands/pain.test.ts +108 -0
  147. package/tests/commands/samples.test.ts +65 -0
  148. package/tests/commands/strategy.test.ts +34 -0
  149. package/tests/commands/thinking-os.test.ts +88 -0
  150. package/tests/core/adaptive-thresholds.test.ts +261 -0
  151. package/tests/core/config-service.test.ts +89 -0
  152. package/tests/core/config.test.ts +90 -0
  153. package/tests/core/control-ui-db.test.ts +75 -0
  154. package/tests/core/core-template-guidance.test.ts +21 -0
  155. package/tests/core/detection-funnel.test.ts +63 -0
  156. package/tests/core/detection-service.test.ts +50 -0
  157. package/tests/core/dictionary-service.test.ts +116 -0
  158. package/tests/core/dictionary.test.ts +168 -0
  159. package/tests/core/empathy-keyword-matcher.test.ts +209 -0
  160. package/tests/core/event-log.test.ts +181 -0
  161. package/tests/core/evolution-e2e.test.ts +58 -0
  162. package/tests/core/evolution-engine-gate-integration.test.ts +543 -0
  163. package/tests/core/evolution-engine.test.ts +562 -0
  164. package/tests/core/evolution-logger.test.ts +148 -0
  165. package/tests/core/evolution-migration.test.ts +50 -0
  166. package/tests/core/evolution-paths.test.ts +21 -0
  167. package/tests/core/evolution-reducer.detector-metadata.test.ts +602 -0
  168. package/tests/core/evolution-reducer.test.ts +180 -0
  169. package/tests/core/evolution-types-loop.test.ts +48 -0
  170. package/tests/core/evolution-user-stories.e2e.test.ts +249 -0
  171. package/tests/core/external-training-contract.test.ts +463 -0
  172. package/tests/core/focus-history.test.ts +682 -0
  173. package/tests/core/init-flatten.test.ts +69 -0
  174. package/tests/core/init-refactor.test.ts +87 -0
  175. package/tests/core/init-v1.3.test.ts +46 -0
  176. package/tests/core/init.test.ts +190 -0
  177. package/tests/core/local-worker-routing.test.ts +757 -0
  178. package/tests/core/migration.test.ts +84 -0
  179. package/tests/core/model-deployment-registry.test.ts +845 -0
  180. package/tests/core/model-training-registry.test.ts +889 -0
  181. package/tests/core/nocturnal-arbiter.test.ts +494 -0
  182. package/tests/core/nocturnal-candidate-scoring.test.ts +400 -0
  183. package/tests/core/nocturnal-compliance.test.ts +646 -0
  184. package/tests/core/nocturnal-dataset.test.ts +892 -0
  185. package/tests/core/nocturnal-executability.test.ts +357 -0
  186. package/tests/core/nocturnal-export.test.ts +462 -0
  187. package/tests/core/nocturnal-reviewed-subset-comparison.test.ts +428 -0
  188. package/tests/core/nocturnal-trajectory-extractor.test.ts +634 -0
  189. package/tests/core/nocturnal-trinity.test.ts +953 -0
  190. package/tests/core/pain.test.ts +33 -0
  191. package/tests/core/path-resolver.test.ts +57 -0
  192. package/tests/core/paths-refactor.test.ts +42 -0
  193. package/tests/core/phase7-rollout-integration.test.ts +477 -0
  194. package/tests/core/principle-training-state.test.ts +712 -0
  195. package/tests/core/profile.test.ts +56 -0
  196. package/tests/core/promotion-gate.test.ts +556 -0
  197. package/tests/core/risk-calculator.test.ts +168 -0
  198. package/tests/core/session-tracker.test.ts +191 -0
  199. package/tests/core/training-program.test.ts +472 -0
  200. package/tests/core/trajectory.test.ts +265 -0
  201. package/tests/core/workspace-context-factory.test.ts +18 -0
  202. package/tests/core/workspace-context.test.ts +134 -0
  203. package/tests/fixtures/nocturnal-reviewed-subset.json +183 -0
  204. package/tests/fixtures/production-compatibility.test.ts +147 -0
  205. package/tests/fixtures/production-mock-generator.ts +282 -0
  206. package/tests/hooks/bash-risk-integration.test.ts +137 -0
  207. package/tests/hooks/bash-risk.test.ts +81 -0
  208. package/tests/hooks/edit-verification.test.ts +678 -0
  209. package/tests/hooks/gate-edit-verification-p1.test.ts +632 -0
  210. package/tests/hooks/gate-edit-verification.test.ts +435 -0
  211. package/tests/hooks/gate-pipeline-integration.test.ts +404 -0
  212. package/tests/hooks/gate.test.ts +271 -0
  213. package/tests/hooks/gfi-gate-unit.test.ts +422 -0
  214. package/tests/hooks/gfi-gate.test.ts +669 -0
  215. package/tests/hooks/lifecycle.test.ts +248 -0
  216. package/tests/hooks/llm.test.ts +308 -0
  217. package/tests/hooks/message-sanitize.test.ts +36 -0
  218. package/tests/hooks/pain.test.ts +141 -0
  219. package/tests/hooks/progressive-trust-gate.test.ts +277 -0
  220. package/tests/hooks/prompt.test.ts +1411 -0
  221. package/tests/hooks/subagent.test.ts +467 -0
  222. package/tests/hooks/thinking-gate.test.ts +313 -0
  223. package/tests/http/principles-console-route.test.ts +140 -0
  224. package/tests/hygiene-tracker.test.ts +77 -0
  225. package/tests/index.integration.test.ts +179 -0
  226. package/tests/index.shadow-routing.integration.test.ts +140 -0
  227. package/tests/index.test.ts +9 -0
  228. package/tests/integration/empathy-workflow-integration.test.ts +627 -0
  229. package/tests/service/control-ui-query-service.test.ts +121 -0
  230. package/tests/service/empathy-observer-workflow-manager.test.ts +176 -0
  231. package/tests/service/evolution-worker.test.ts +585 -0
  232. package/tests/service/nocturnal-runtime.test.ts +470 -0
  233. package/tests/service/nocturnal-service.test.ts +577 -0
  234. package/tests/service/nocturnal-target-selector.test.ts +615 -0
  235. package/tests/service/nocturnal-workflow-manager.test.ts +439 -0
  236. package/tests/service/phase3-input-filter.test.ts +289 -0
  237. package/tests/service/runtime-summary-service.test.ts +919 -0
  238. package/tests/task-compliance.test.ts +166 -0
  239. package/tests/test-utils.ts +48 -0
  240. package/tests/tools/critique-prompt.test.ts +260 -0
  241. package/tests/tools/deep-reflect.test.ts +232 -0
  242. package/tests/tools/model-index.test.ts +246 -0
  243. package/tests/ui/app.test.tsx +114 -0
  244. package/tests/utils/file-lock.test.ts +407 -0
  245. package/tests/utils/hashing.test.ts +32 -0
  246. package/tests/utils/io.test.ts +39 -0
  247. package/tests/utils/nlp.test.ts +53 -0
  248. package/tests/utils/plugin-logger.test.ts +156 -0
  249. package/tsconfig.json +16 -0
  250. package/tsconfig.tsbuildinfo +1 -0
  251. package/ui/src/App.tsx +45 -0
  252. package/ui/src/api.ts +216 -0
  253. package/ui/src/charts.tsx +586 -0
  254. package/ui/src/components/ErrorState.tsx +6 -0
  255. package/ui/src/components/Loading.tsx +13 -0
  256. package/ui/src/components/ProtectedRoute.tsx +12 -0
  257. package/ui/src/components/Shell.tsx +91 -0
  258. package/ui/src/components/WorkspaceConfig.tsx +146 -0
  259. package/ui/src/components/index.ts +5 -0
  260. package/ui/src/context/auth.tsx +80 -0
  261. package/ui/src/context/theme.tsx +66 -0
  262. package/ui/src/hooks/useAutoRefresh.ts +39 -0
  263. package/ui/src/i18n/ui.ts +363 -0
  264. package/ui/src/main.tsx +16 -0
  265. package/ui/src/pages/EvolutionPage.tsx +352 -0
  266. package/ui/src/pages/FeedbackPage.tsx +140 -0
  267. package/ui/src/pages/GateMonitorPage.tsx +136 -0
  268. package/ui/src/pages/LoginPage.tsx +88 -0
  269. package/ui/src/pages/OverviewPage.tsx +238 -0
  270. package/ui/src/pages/SamplesPage.tsx +174 -0
  271. package/ui/src/pages/ThinkingModelsPage.tsx +127 -0
  272. package/ui/src/styles.css +1661 -0
  273. package/ui/src/types.ts +368 -0
  274. package/ui/src/utils/format.ts +15 -0
  275. package/vitest.config.ts +23 -0
  276. package/dist/commands/capabilities.d.ts +0 -3
  277. package/dist/commands/capabilities.js +0 -73
  278. package/dist/commands/context.d.ts +0 -5
  279. package/dist/commands/evolution-status.d.ts +0 -4
  280. package/dist/commands/evolution-status.js +0 -117
  281. package/dist/commands/evolver.d.ts +0 -9
  282. package/dist/commands/evolver.js +0 -26
  283. package/dist/commands/export.d.ts +0 -2
  284. package/dist/commands/export.js +0 -98
  285. package/dist/commands/focus.d.ts +0 -14
  286. package/dist/commands/focus.js +0 -457
  287. package/dist/commands/nocturnal-review.d.ts +0 -24
  288. package/dist/commands/nocturnal-review.js +0 -265
  289. package/dist/commands/nocturnal-rollout.d.ts +0 -27
  290. package/dist/commands/nocturnal-rollout.js +0 -671
  291. package/dist/commands/nocturnal-train.d.ts +0 -25
  292. package/dist/commands/nocturnal-train.js +0 -919
  293. package/dist/commands/pain.d.ts +0 -5
  294. package/dist/commands/principle-rollback.d.ts +0 -4
  295. package/dist/commands/principle-rollback.js +0 -22
  296. package/dist/commands/rollback.d.ts +0 -19
  297. package/dist/commands/samples.d.ts +0 -2
  298. package/dist/commands/samples.js +0 -55
  299. package/dist/commands/strategy.d.ts +0 -3
  300. package/dist/commands/strategy.js +0 -29
  301. package/dist/commands/thinking-os.d.ts +0 -2
  302. package/dist/config/defaults/runtime.d.ts +0 -40
  303. package/dist/config/errors.d.ts +0 -84
  304. package/dist/config/errors.js +0 -94
  305. package/dist/config/index.js +0 -7
  306. package/dist/constants/diagnostician.d.ts +0 -12
  307. package/dist/constants/diagnostician.js +0 -56
  308. package/dist/constants/tools.d.ts +0 -17
  309. package/dist/constants/tools.js +0 -54
  310. package/dist/core/adaptive-thresholds.d.ts +0 -186
  311. package/dist/core/adaptive-thresholds.js +0 -300
  312. package/dist/core/config-service.d.ts +0 -15
  313. package/dist/core/config.d.ts +0 -127
  314. package/dist/core/control-ui-db.d.ts +0 -95
  315. package/dist/core/control-ui-db.js +0 -292
  316. package/dist/core/detection-funnel.d.ts +0 -33
  317. package/dist/core/detection-service.d.ts +0 -15
  318. package/dist/core/dictionary-service.d.ts +0 -15
  319. package/dist/core/dictionary.d.ts +0 -38
  320. package/dist/core/event-log.d.ts +0 -82
  321. package/dist/core/event-log.js +0 -463
  322. package/dist/core/evolution-engine.d.ts +0 -118
  323. package/dist/core/evolution-engine.js +0 -464
  324. package/dist/core/evolution-logger.d.ts +0 -137
  325. package/dist/core/evolution-logger.js +0 -256
  326. package/dist/core/evolution-migration.d.ts +0 -5
  327. package/dist/core/evolution-migration.js +0 -65
  328. package/dist/core/evolution-reducer.d.ts +0 -98
  329. package/dist/core/evolution-reducer.js +0 -465
  330. package/dist/core/evolution-types.d.ts +0 -287
  331. package/dist/core/evolution-types.js +0 -78
  332. package/dist/core/external-training-contract.d.ts +0 -276
  333. package/dist/core/external-training-contract.js +0 -269
  334. package/dist/core/focus-history.d.ts +0 -210
  335. package/dist/core/focus-history.js +0 -1185
  336. package/dist/core/hygiene/tracker.d.ts +0 -22
  337. package/dist/core/hygiene/tracker.js +0 -106
  338. package/dist/core/init.d.ts +0 -12
  339. package/dist/core/local-worker-routing.d.ts +0 -175
  340. package/dist/core/local-worker-routing.js +0 -525
  341. package/dist/core/migration.d.ts +0 -6
  342. package/dist/core/model-deployment-registry.d.ts +0 -218
  343. package/dist/core/model-deployment-registry.js +0 -503
  344. package/dist/core/model-training-registry.d.ts +0 -295
  345. package/dist/core/model-training-registry.js +0 -475
  346. package/dist/core/nocturnal-arbiter.d.ts +0 -159
  347. package/dist/core/nocturnal-arbiter.js +0 -534
  348. package/dist/core/nocturnal-candidate-scoring.d.ts +0 -137
  349. package/dist/core/nocturnal-candidate-scoring.js +0 -266
  350. package/dist/core/nocturnal-compliance.d.ts +0 -175
  351. package/dist/core/nocturnal-compliance.js +0 -824
  352. package/dist/core/nocturnal-dataset.d.ts +0 -224
  353. package/dist/core/nocturnal-dataset.js +0 -443
  354. package/dist/core/nocturnal-executability.d.ts +0 -85
  355. package/dist/core/nocturnal-executability.js +0 -331
  356. package/dist/core/nocturnal-export.d.ts +0 -124
  357. package/dist/core/nocturnal-export.js +0 -275
  358. package/dist/core/nocturnal-paths.d.ts +0 -124
  359. package/dist/core/nocturnal-trajectory-extractor.d.ts +0 -242
  360. package/dist/core/nocturnal-trajectory-extractor.js +0 -307
  361. package/dist/core/nocturnal-trinity.d.ts +0 -311
  362. package/dist/core/nocturnal-trinity.js +0 -880
  363. package/dist/core/pain.d.ts +0 -4
  364. package/dist/core/pain.js +0 -70
  365. package/dist/core/path-resolver.d.ts +0 -46
  366. package/dist/core/paths.d.ts +0 -65
  367. package/dist/core/principle-training-state.d.ts +0 -121
  368. package/dist/core/principle-training-state.js +0 -321
  369. package/dist/core/profile.d.ts +0 -62
  370. package/dist/core/profile.js +0 -210
  371. package/dist/core/promotion-gate.d.ts +0 -238
  372. package/dist/core/promotion-gate.js +0 -529
  373. package/dist/core/risk-calculator.d.ts +0 -22
  374. package/dist/core/session-tracker.d.ts +0 -99
  375. package/dist/core/shadow-observation-registry.d.ts +0 -217
  376. package/dist/core/shadow-observation-registry.js +0 -308
  377. package/dist/core/system-logger.d.ts +0 -8
  378. package/dist/core/thinking-models.d.ts +0 -38
  379. package/dist/core/thinking-models.js +0 -170
  380. package/dist/core/training-program.d.ts +0 -233
  381. package/dist/core/training-program.js +0 -433
  382. package/dist/core/trajectory.d.ts +0 -411
  383. package/dist/core/trajectory.js +0 -1307
  384. package/dist/core/workspace-context.d.ts +0 -71
  385. package/dist/hooks/bash-risk.d.ts +0 -57
  386. package/dist/hooks/bash-risk.js +0 -137
  387. package/dist/hooks/edit-verification.d.ts +0 -62
  388. package/dist/hooks/edit-verification.js +0 -256
  389. package/dist/hooks/gate-block-helper.d.ts +0 -44
  390. package/dist/hooks/gate-block-helper.js +0 -119
  391. package/dist/hooks/gate.d.ts +0 -24
  392. package/dist/hooks/gate.js +0 -173
  393. package/dist/hooks/gfi-gate.d.ts +0 -40
  394. package/dist/hooks/gfi-gate.js +0 -113
  395. package/dist/hooks/lifecycle.d.ts +0 -5
  396. package/dist/hooks/lifecycle.js +0 -284
  397. package/dist/hooks/llm.d.ts +0 -12
  398. package/dist/hooks/message-sanitize.d.ts +0 -3
  399. package/dist/hooks/message-sanitize.js +0 -37
  400. package/dist/hooks/pain.d.ts +0 -5
  401. package/dist/hooks/pain.js +0 -301
  402. package/dist/hooks/progressive-trust-gate.d.ts +0 -51
  403. package/dist/hooks/progressive-trust-gate.js +0 -89
  404. package/dist/hooks/prompt.d.ts +0 -47
  405. package/dist/hooks/prompt.js +0 -884
  406. package/dist/hooks/subagent.d.ts +0 -10
  407. package/dist/hooks/subagent.js +0 -387
  408. package/dist/hooks/thinking-checkpoint.d.ts +0 -37
  409. package/dist/hooks/thinking-checkpoint.js +0 -51
  410. package/dist/hooks/trajectory-collector.d.ts +0 -32
  411. package/dist/hooks/trajectory-collector.js +0 -256
  412. package/dist/http/principles-console-route.d.ts +0 -9
  413. package/dist/http/principles-console-route.js +0 -567
  414. package/dist/i18n/commands.d.ts +0 -26
  415. package/dist/i18n/commands.js +0 -116
  416. package/dist/index.d.ts +0 -7
  417. package/dist/index.js +0 -581
  418. package/dist/service/central-database.d.ts +0 -104
  419. package/dist/service/central-database.js +0 -649
  420. package/dist/service/control-ui-query-service.d.ts +0 -221
  421. package/dist/service/control-ui-query-service.js +0 -543
  422. package/dist/service/empathy-observer-manager.d.ts +0 -52
  423. package/dist/service/empathy-observer-manager.js +0 -229
  424. package/dist/service/evolution-query-service.d.ts +0 -155
  425. package/dist/service/evolution-query-service.js +0 -258
  426. package/dist/service/evolution-worker.d.ts +0 -101
  427. package/dist/service/evolution-worker.js +0 -974
  428. package/dist/service/nocturnal-runtime.d.ts +0 -183
  429. package/dist/service/nocturnal-service.d.ts +0 -163
  430. package/dist/service/nocturnal-service.js +0 -787
  431. package/dist/service/nocturnal-target-selector.d.ts +0 -145
  432. package/dist/service/nocturnal-target-selector.js +0 -315
  433. package/dist/service/phase3-input-filter.d.ts +0 -73
  434. package/dist/service/phase3-input-filter.js +0 -172
  435. package/dist/service/runtime-summary-service.d.ts +0 -122
  436. package/dist/service/runtime-summary-service.js +0 -485
  437. package/dist/service/trajectory-service.d.ts +0 -2
  438. package/dist/service/trajectory-service.js +0 -15
  439. package/dist/tools/critique-prompt.d.ts +0 -14
  440. package/dist/tools/deep-reflect.d.ts +0 -39
  441. package/dist/tools/deep-reflect.js +0 -350
  442. package/dist/tools/model-index.d.ts +0 -9
  443. package/dist/types/event-types.d.ts +0 -306
  444. package/dist/types/event-types.js +0 -106
  445. package/dist/types/hygiene-types.d.ts +0 -20
  446. package/dist/types/hygiene-types.js +0 -12
  447. package/dist/types/runtime-summary.d.ts +0 -47
  448. package/dist/types/runtime-summary.js +0 -1
  449. package/dist/types.d.ts +0 -50
  450. package/dist/types.js +0 -22
  451. package/dist/utils/file-lock.d.ts +0 -71
  452. package/dist/utils/file-lock.js +0 -309
  453. package/dist/utils/glob-match.d.ts +0 -28
  454. package/dist/utils/hashing.d.ts +0 -9
  455. package/dist/utils/io.d.ts +0 -6
  456. package/dist/utils/io.js +0 -106
  457. package/dist/utils/nlp.d.ts +0 -9
  458. package/dist/utils/plugin-logger.d.ts +0 -39
  459. package/dist/utils/subagent-probe.d.ts +0 -34
  460. 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
+ }