principles-disciple 1.8.1 → 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 (470) 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/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 -129
  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 -101
  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 -13
  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 -52
  403. package/dist/hooks/progressive-trust-gate.js +0 -134
  404. package/dist/hooks/prompt.d.ts +0 -49
  405. package/dist/hooks/prompt.js +0 -905
  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 -681
  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 -88
  423. package/dist/service/empathy-observer-manager.js +0 -414
  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 -975
  428. package/dist/service/health-query-service.d.ts +0 -170
  429. package/dist/service/health-query-service.js +0 -662
  430. package/dist/service/nocturnal-runtime.d.ts +0 -183
  431. package/dist/service/nocturnal-service.d.ts +0 -163
  432. package/dist/service/nocturnal-service.js +0 -787
  433. package/dist/service/nocturnal-target-selector.d.ts +0 -145
  434. package/dist/service/nocturnal-target-selector.js +0 -315
  435. package/dist/service/phase3-input-filter.d.ts +0 -73
  436. package/dist/service/phase3-input-filter.js +0 -172
  437. package/dist/service/runtime-summary-service.d.ts +0 -122
  438. package/dist/service/runtime-summary-service.js +0 -485
  439. package/dist/service/subagent-workflow/empathy-observer-workflow-manager.d.ts +0 -48
  440. package/dist/service/subagent-workflow/index.d.ts +0 -4
  441. package/dist/service/subagent-workflow/index.js +0 -3
  442. package/dist/service/subagent-workflow/runtime-direct-driver.d.ts +0 -77
  443. package/dist/service/subagent-workflow/runtime-direct-driver.js +0 -75
  444. package/dist/service/subagent-workflow/types.js +0 -11
  445. package/dist/service/subagent-workflow/workflow-store.d.ts +0 -26
  446. package/dist/service/subagent-workflow/workflow-store.js +0 -165
  447. package/dist/service/trajectory-service.d.ts +0 -2
  448. package/dist/service/trajectory-service.js +0 -15
  449. package/dist/tools/critique-prompt.d.ts +0 -14
  450. package/dist/tools/deep-reflect.d.ts +0 -39
  451. package/dist/tools/deep-reflect.js +0 -350
  452. package/dist/tools/model-index.d.ts +0 -9
  453. package/dist/types/event-types.d.ts +0 -306
  454. package/dist/types/event-types.js +0 -106
  455. package/dist/types/hygiene-types.d.ts +0 -20
  456. package/dist/types/hygiene-types.js +0 -12
  457. package/dist/types/runtime-summary.d.ts +0 -47
  458. package/dist/types/runtime-summary.js +0 -1
  459. package/dist/types.d.ts +0 -50
  460. package/dist/types.js +0 -22
  461. package/dist/utils/file-lock.d.ts +0 -71
  462. package/dist/utils/file-lock.js +0 -309
  463. package/dist/utils/glob-match.d.ts +0 -28
  464. package/dist/utils/hashing.d.ts +0 -9
  465. package/dist/utils/io.d.ts +0 -6
  466. package/dist/utils/io.js +0 -106
  467. package/dist/utils/nlp.d.ts +0 -9
  468. package/dist/utils/plugin-logger.d.ts +0 -39
  469. package/dist/utils/subagent-probe.d.ts +0 -34
  470. package/dist/utils/subagent-probe.js +0 -81
@@ -0,0 +1,722 @@
1
+ /**
2
+ * Model Deployment Registry — Worker Profile → Checkpoint Binding & Routing Control
3
+ * ===============================================================================
4
+ *
5
+ * PURPOSE: Establish auditable, reversible bindings between worker profiles and
6
+ * trained model checkpoints so that routing decisions are code-governed and
7
+ * rollback-safe.
8
+ *
9
+ * ARCHITECTURE:
10
+ * - Registry file: {stateDir}/.state/nocturnal/deployment-registry.json
11
+ * - File locking on all write operations
12
+ * - Immutable deployment records — rollback uses previousCheckpointId
13
+ * - Tight integration with model-training-registry for checkpoint validation
14
+ *
15
+ * PROFILE CONSTRAINTS (Phase 5 only):
16
+ * - local-reader → must bind a checkpoint whose targetModelFamily is a "reader" family
17
+ * - local-editor → must bind a checkpoint whose targetModelFamily is an "editor" family
18
+ * - No other profiles are accepted
19
+ *
20
+ * BINDING RULES:
21
+ * - Only a deployable checkpoint can be bound
22
+ * - The checkpoint's targetModelFamily must satisfy the profile's family constraint
23
+ * - binding sets routingEnabled = false; enableRoutingForProfile() must be called explicitly
24
+ * - rollbackDeployment() returns to previousCheckpointId (if any)
25
+ *
26
+ * DESIGN CONSTRAINTS:
27
+ * - No actual task routing execution (Phase 5 only)
28
+ * - No automatic promotion or failover
29
+ * - Registry is append-only for deployments; rollback creates a new binding
30
+ */
31
+
32
+ import * as fs from 'fs';
33
+ import * as path from 'path';
34
+ import * as crypto from 'crypto';
35
+ import { withLock } from '../utils/file-lock.js';
36
+ import type { Checkpoint } from './model-training-registry.js';
37
+ import {
38
+ getCheckpoint,
39
+ isCheckpointDeployable,
40
+ } from './model-training-registry.js';
41
+ import { getPromotionState } from './promotion-gate.js';
42
+
43
+ // ---------------------------------------------------------------------------
44
+ // Constants
45
+ // ---------------------------------------------------------------------------
46
+
47
+ const REGISTRY_FILE = '.state/nocturnal/deployment-registry.json';
48
+
49
+ /**
50
+ * Worker profiles supported in Phase 5.
51
+ * Only these two profiles may be registered.
52
+ */
53
+ export type WorkerProfile = 'local-reader' | 'local-editor';
54
+
55
+ /**
56
+ * The set of valid Phase 5 worker profile names.
57
+ */
58
+ export const SUPPORTED_PROFILES: readonly WorkerProfile[] = ['local-reader', 'local-editor'];
59
+
60
+ // ---------------------------------------------------------------------------
61
+ // Profile–Family Constraint System
62
+ // ---------------------------------------------------------------------------
63
+
64
+ /**
65
+ * Known model family name prefixes/suffixes recognized as "reader" families.
66
+ * Any targetModelFamily string containing one of these tokens is considered a reader family.
67
+ *
68
+ * This is a first-iteration heuristic. Real systems may use explicit tag registries.
69
+ */
70
+ const READER_FAMILY_KEYWORDS = ['reader', 'read', 'claude-haiku', 'qwen-lite', 'phi-mini'];
71
+
72
+ /**
73
+ * Known model family name prefixes/suffixes recognized as "editor" families.
74
+ * Any targetModelFamily string containing one of these tokens is considered an editor family.
75
+ */
76
+ const EDITOR_FAMILY_KEYWORDS = ['editor', 'edit', 'code', 'claude-sonnet', 'gpt-4o-mini'];
77
+
78
+ /**
79
+ * Determine whether a target model family name qualifies as a "reader" family.
80
+ * Returns true if any known reader keyword appears in the family name (case-insensitive).
81
+ */
82
+ function isReaderFamily(targetModelFamily: string): boolean {
83
+ const lower = targetModelFamily.toLowerCase();
84
+ return READER_FAMILY_KEYWORDS.some((kw) => lower.includes(kw));
85
+ }
86
+
87
+ /**
88
+ * Determine whether a target model family name qualifies as an "editor" family.
89
+ * Returns true if any known editor keyword appears in the family name (case-insensitive).
90
+ */
91
+ function isEditorFamily(targetModelFamily: string): boolean {
92
+ const lower = targetModelFamily.toLowerCase();
93
+ return EDITOR_FAMILY_KEYWORDS.some((kw) => lower.includes(kw));
94
+ }
95
+
96
+ /**
97
+ * Validate that a given targetModelFamily satisfies a worker's family constraint.
98
+ *
99
+ * @param profile - The worker profile requesting the binding
100
+ * @param targetModelFamily - The checkpoint's target model family
101
+ * @throws Error if the family is incompatible with the profile
102
+ */
103
+ function validateProfileFamilyConstraint(profile: WorkerProfile, targetModelFamily: string): void {
104
+ if (profile === 'local-reader') {
105
+ if (!isReaderFamily(targetModelFamily)) {
106
+ throw new Error(
107
+ `Family constraint violated: profile "${profile}" requires a reader-family checkpoint ` +
108
+ `but checkpoint targets "${targetModelFamily}". ` +
109
+ `Reader families must contain one of: ${READER_FAMILY_KEYWORDS.join(', ')}. ` +
110
+ `If you are deploying a new model family, update the READER_FAMILY_KEYWORDS or ` +
111
+ `EDITOR_FAMILY_KEYWORDS constants in model-deployment-registry.ts.`
112
+ );
113
+ }
114
+ } else if (profile === 'local-editor') {
115
+ if (!isEditorFamily(targetModelFamily)) {
116
+ throw new Error(
117
+ `Family constraint violated: profile "${profile}" requires an editor-family checkpoint ` +
118
+ `but checkpoint targets "${targetModelFamily}". ` +
119
+ `Editor families must contain one of: ${EDITOR_FAMILY_KEYWORDS.join(', ')}. ` +
120
+ `If you are deploying a new model family, update the EDITOR_FAMILY_KEYWORDS constant ` +
121
+ `in model-deployment-registry.ts.`
122
+ );
123
+ }
124
+ }
125
+ }
126
+
127
+ // ---------------------------------------------------------------------------
128
+ // Types
129
+ // ---------------------------------------------------------------------------
130
+
131
+ /**
132
+ * A deployment record — binds a worker profile to a specific checkpoint
133
+ * and controls whether routing is enabled for that profile.
134
+ */
135
+ export interface Deployment {
136
+ /** Unique identifier for this deployment record */
137
+ deploymentId: string;
138
+
139
+ /** Worker profile this deployment targets (local-reader | local-editor) */
140
+ workerProfile: WorkerProfile;
141
+
142
+ /**
143
+ * The model family this deployment targets.
144
+ * Derived from the bound checkpoint at bind time; stored for quick queries.
145
+ */
146
+ targetModelFamily: string;
147
+
148
+ /**
149
+ * The currently active checkpoint for this profile.
150
+ * null means the profile is bound but no checkpoint is active (e.g., after rollback).
151
+ */
152
+ activeCheckpointId: string | null;
153
+
154
+ /**
155
+ * The previously active checkpoint (before the current activeCheckpointId).
156
+ * Used for rollback. null if no previous checkpoint exists.
157
+ */
158
+ previousCheckpointId: string | null;
159
+
160
+ /**
161
+ * Whether routing to this worker profile is currently permitted.
162
+ * Must be explicitly enabled via enableRoutingForProfile().
163
+ * Cannot be true if activeCheckpointId is null.
164
+ */
165
+ routingEnabled: boolean;
166
+
167
+ /** ISO-8601 timestamp — when this binding was first created */
168
+ deployedAt: string;
169
+
170
+ /** ISO-8601 timestamp — when this binding was last updated (checkpoint change or flag toggle) */
171
+ updatedAt: string;
172
+
173
+ /**
174
+ * Optional human-readable note about this deployment.
175
+ * E.g., "initial deployment", "rollback from eval failure", "promoted after 30-day holdout".
176
+ */
177
+ note?: string;
178
+ }
179
+
180
+ /**
181
+ * The complete deployment registry — all deployment records in one store.
182
+ */
183
+ export interface ModelDeploymentRegistry {
184
+ deployments: Deployment[];
185
+ }
186
+
187
+ // ---------------------------------------------------------------------------
188
+ // Registry Path
189
+ // ---------------------------------------------------------------------------
190
+
191
+ function getRegistryPath(stateDir: string): string {
192
+ return path.join(stateDir, REGISTRY_FILE);
193
+ }
194
+
195
+ /**
196
+ * Ensure the registry directory exists.
197
+ */
198
+ function ensureRegistryDir(stateDir: string): void {
199
+ const registryPath = getRegistryPath(stateDir);
200
+ const dir = path.dirname(registryPath);
201
+ if (!fs.existsSync(dir)) {
202
+ fs.mkdirSync(dir, { recursive: true });
203
+ }
204
+ }
205
+
206
+ // ---------------------------------------------------------------------------
207
+ // File Operations
208
+ // ---------------------------------------------------------------------------
209
+
210
+ /**
211
+ * Read the registry from disk. Returns empty registry if missing.
212
+ * Throws if the file exists but contains invalid JSON — fail-closed
213
+ * to prevent silent data loss when a corrupt registry would otherwise
214
+ * be overwritten on the next write.
215
+ */
216
+ function readRegistry(stateDir: string): ModelDeploymentRegistry {
217
+ const registryPath = getRegistryPath(stateDir);
218
+ if (!fs.existsSync(registryPath)) {
219
+ return { deployments: [] };
220
+ }
221
+ try {
222
+ const content = fs.readFileSync(registryPath, 'utf-8');
223
+ const registry = JSON.parse(content) as ModelDeploymentRegistry;
224
+ // Validate required structure — fail closed on malformed registry
225
+ if (!Array.isArray(registry.deployments)) {
226
+ throw new Error(`Corrupt deployment registry at "${registryPath}": missing or invalid "deployments" field`);
227
+ }
228
+ return registry;
229
+ } catch (err) {
230
+ if (err instanceof SyntaxError || err instanceof Error) {
231
+ throw new Error(`Failed to read deployment registry from "${registryPath}": ${err.message}`);
232
+ }
233
+ throw err;
234
+ }
235
+ }
236
+
237
+ /**
238
+ * Write the registry to disk atomically.
239
+ * Caller must hold the registry lock.
240
+ */
241
+ function writeRegistry(stateDir: string, registry: ModelDeploymentRegistry): void {
242
+ ensureRegistryDir(stateDir);
243
+ const registryPath = getRegistryPath(stateDir);
244
+ const tmpPath = `${registryPath}.tmp`;
245
+ fs.writeFileSync(tmpPath, JSON.stringify(registry, null, 2), 'utf-8');
246
+ fs.renameSync(tmpPath, registryPath);
247
+ }
248
+
249
+ /**
250
+ * Execute a read-modify-write under an exclusive file lock.
251
+ */
252
+ function withDeploymentRegistryLock<T>(
253
+ stateDir: string,
254
+ fn: (registry: ModelDeploymentRegistry) => T
255
+ ): T {
256
+ const registryPath = getRegistryPath(stateDir);
257
+ return withLock(registryPath, () => {
258
+ const registry = readRegistry(stateDir);
259
+ return fn(registry);
260
+ });
261
+ }
262
+
263
+ // ---------------------------------------------------------------------------
264
+ // Profile Validation
265
+ // ---------------------------------------------------------------------------
266
+
267
+ /**
268
+ * Validate that a worker profile name is supported in Phase 5.
269
+ *
270
+ * @throws Error if the profile is not in SUPPORTED_PROFILES
271
+ */
272
+ export function assertSupportedProfile(profile: string): asserts profile is WorkerProfile {
273
+ if (!SUPPORTED_PROFILES.includes(profile as WorkerProfile)) {
274
+ throw new Error(
275
+ `Unsupported worker profile: "${profile}". ` +
276
+ `Phase 5 only supports: ${SUPPORTED_PROFILES.join(', ')}. ` +
277
+ `Do not add new profiles in Phase 5.`
278
+ );
279
+ }
280
+ }
281
+
282
+ // ---------------------------------------------------------------------------
283
+ // Promotion Gate Integration (Phase 7)
284
+ // ---------------------------------------------------------------------------
285
+
286
+ /**
287
+ * Check if a checkpoint has passed the promotion gate and can be deployed.
288
+ *
289
+ * This function checks:
290
+ * 1. The checkpoint has an eval summary attached (lineage complete)
291
+ * 2. The promotion state is 'shadow_ready' or 'promotable' (gate passed)
292
+ *
293
+ * @param stateDir - Workspace state directory
294
+ * @param checkpointId - Checkpoint to verify
295
+ * @returns true if the checkpoint can be deployed, false otherwise
296
+ */
297
+ export function hasPassedPromotionGate(stateDir: string, checkpointId: string): boolean {
298
+ const checkpoint = getCheckpoint(stateDir, checkpointId);
299
+ if (!checkpoint) return false;
300
+
301
+ // Must have eval summary attached
302
+ if (!checkpoint.lastEvalSummaryRef) return false;
303
+
304
+ // Check promotion state
305
+ const state = getPromotionState(stateDir, checkpointId);
306
+ return state === 'shadow_ready' || state === 'promotable';
307
+ }
308
+
309
+ /**
310
+ * Assert that a checkpoint has passed the promotion gate.
311
+ * Throws if the checkpoint cannot be deployed.
312
+ *
313
+ * @param stateDir - Workspace state directory
314
+ * @param checkpointId - Checkpoint to verify
315
+ * @throws Error if the checkpoint has not passed the promotion gate
316
+ */
317
+ export function assertPromotionGatePassed(stateDir: string, checkpointId: string): void {
318
+ if (!hasPassedPromotionGate(stateDir, checkpointId)) {
319
+ throw new Error(
320
+ `Checkpoint "${checkpointId}" has not passed the promotion gate. ` +
321
+ `A checkpoint must pass the promotion gate (state: shadow_ready or promotable) ` +
322
+ `before it can be bound to a worker profile. ` +
323
+ `Ensure the promotion gate has been evaluated and approved.`
324
+ );
325
+ }
326
+ }
327
+
328
+ // ---------------------------------------------------------------------------
329
+ // Core Operations
330
+ // ---------------------------------------------------------------------------
331
+
332
+ /**
333
+ * Bind a checkpoint to a worker profile, creating or updating the deployment record.
334
+ *
335
+ * BINDING RULE (fail-closed):
336
+ * Only a checkpoint that is marked deployable in the training registry may be bound.
337
+ *
338
+ * PROFILE-FAMILY CONSTRAINT:
339
+ * The checkpoint's targetModelFamily must satisfy the profile's family keyword constraint.
340
+ * See: validateProfileFamilyConstraint()
341
+ *
342
+ * @param stateDir - Workspace state directory
343
+ * @param workerProfile - Target worker profile (local-reader | local-editor)
344
+ * @param checkpointId - Checkpoint to bind (must be deployable)
345
+ * @param note - Optional human-readable note
346
+ * @returns The new or updated Deployment record
347
+ *
348
+ * @throws Error if checkpoint is not found or not deployable
349
+ * @throws Error if checkpoint's targetModelFamily violates profile constraints
350
+ */
351
+ export function bindCheckpointToWorkerProfile(
352
+ stateDir: string,
353
+ workerProfile: WorkerProfile,
354
+ checkpointId: string,
355
+ note?: string
356
+ ): Deployment {
357
+ assertSupportedProfile(workerProfile);
358
+
359
+ return withDeploymentRegistryLock(stateDir, (registry) => {
360
+ const now = new Date().toISOString();
361
+
362
+ // --- Validate checkpoint exists and is deployable ---
363
+ const checkpoint = getCheckpoint(stateDir, checkpointId);
364
+ if (!checkpoint) {
365
+ throw new Error(
366
+ `bindCheckpointToWorkerProfile failed: checkpoint "${checkpointId}" not found ` +
367
+ `in training registry. Ensure the checkpoint has been registered via ` +
368
+ `registerCheckpoint() in model-training-registry.ts first.`
369
+ );
370
+ }
371
+
372
+ if (!isCheckpointDeployable(stateDir, checkpointId)) {
373
+ throw new Error(
374
+ `bindCheckpointToWorkerProfile failed: checkpoint "${checkpointId}" is not deployable. ` +
375
+ `Only checkpoints that have passed evaluation may be bound to a worker profile. ` +
376
+ `Use markCheckpointDeployable() in model-training-registry.ts after a successful eval.`
377
+ );
378
+ }
379
+
380
+ // --- Phase 7: Validate promotion gate has passed ---
381
+ assertPromotionGatePassed(stateDir, checkpointId);
382
+
383
+ // --- Validate profile-family constraint ---
384
+ validateProfileFamilyConstraint(workerProfile, checkpoint.targetModelFamily);
385
+
386
+ // --- Find existing deployment for this profile (if any) ---
387
+ const existingIdx = registry.deployments.findIndex(
388
+ (d) => d.workerProfile === workerProfile
389
+ );
390
+
391
+ const deploymentId = existingIdx >= 0
392
+ ? registry.deployments[existingIdx].deploymentId
393
+ : crypto.randomUUID();
394
+
395
+ const previousCheckpointId = existingIdx >= 0
396
+ ? registry.deployments[existingIdx].activeCheckpointId ?? null
397
+ : null;
398
+
399
+ const newDeployment: Deployment = {
400
+ deploymentId,
401
+ workerProfile,
402
+ targetModelFamily: checkpoint.targetModelFamily,
403
+ activeCheckpointId: checkpointId,
404
+ // When re-binding (updating checkpoint), the old active becomes previous
405
+ previousCheckpointId,
406
+ // routingEnabled starts false — must be explicitly enabled
407
+ routingEnabled: false,
408
+ deployedAt: existingIdx >= 0
409
+ ? registry.deployments[existingIdx].deployedAt
410
+ : now,
411
+ updatedAt: now,
412
+ note: note ?? (existingIdx >= 0 ? registry.deployments[existingIdx].note : undefined),
413
+ };
414
+
415
+ if (existingIdx >= 0) {
416
+ registry.deployments[existingIdx] = newDeployment;
417
+ } else {
418
+ registry.deployments.push(newDeployment);
419
+ }
420
+
421
+ writeRegistry(stateDir, registry);
422
+ return newDeployment;
423
+ });
424
+ }
425
+
426
+ /**
427
+ * Retrieve the deployment record for a worker profile.
428
+ *
429
+ * @returns Deployment if found, null otherwise
430
+ */
431
+ export function getDeployment(
432
+ stateDir: string,
433
+ workerProfile: WorkerProfile
434
+ ): Deployment | null {
435
+ assertSupportedProfile(workerProfile);
436
+ const registry = readRegistry(stateDir);
437
+ return registry.deployments.find((d) => d.workerProfile === workerProfile) ?? null;
438
+ }
439
+
440
+ /**
441
+ * List all deployments, optionally filtered.
442
+ *
443
+ * @param stateDir - Workspace state directory
444
+ * @param filter - Optional filter criteria
445
+ */
446
+ export function listDeployments(
447
+ stateDir: string,
448
+ filter?: {
449
+ workerProfile?: WorkerProfile;
450
+ routingEnabled?: boolean;
451
+ }
452
+ ): Deployment[] {
453
+ const registry = readRegistry(stateDir);
454
+ let deployments = registry.deployments;
455
+
456
+ if (filter?.workerProfile) {
457
+ deployments = deployments.filter((d) => d.workerProfile === filter.workerProfile);
458
+ }
459
+ if (filter?.routingEnabled !== undefined) {
460
+ deployments = deployments.filter((d) => d.routingEnabled === filter.routingEnabled);
461
+ }
462
+
463
+ return deployments.sort(
464
+ (a, b) => new Date(b.updatedAt).getTime() - new Date(a.updatedAt).getTime()
465
+ );
466
+ }
467
+
468
+ /**
469
+ * Enable routing for a worker profile.
470
+ *
471
+ * PRECONDITIONS (fail-closed):
472
+ * 1. A deployment record must exist for this profile
473
+ * 2. activeCheckpointId must not be null
474
+ *
475
+ * @throws Error if no deployment exists
476
+ * @throws Error if activeCheckpointId is null (nothing to route to)
477
+ */
478
+ export function enableRoutingForProfile(
479
+ stateDir: string,
480
+ workerProfile: WorkerProfile
481
+ ): Deployment {
482
+ assertSupportedProfile(workerProfile);
483
+
484
+ return withDeploymentRegistryLock(stateDir, (registry) => {
485
+ const idx = registry.deployments.findIndex(
486
+ (d) => d.workerProfile === workerProfile
487
+ );
488
+
489
+ if (idx === -1) {
490
+ throw new Error(
491
+ `enableRoutingForProfile failed: no deployment found for profile "${workerProfile}". ` +
492
+ `Bind a checkpoint first using bindCheckpointToWorkerProfile().`
493
+ );
494
+ }
495
+
496
+ const deployment = registry.deployments[idx];
497
+
498
+ if (!deployment.activeCheckpointId) {
499
+ throw new Error(
500
+ `enableRoutingForProfile failed: deployment for "${workerProfile}" has no ` +
501
+ `active checkpoint (activeCheckpointId is null). ` +
502
+ `Bind a checkpoint before enabling routing.`
503
+ );
504
+ }
505
+
506
+ // Double-check the active checkpoint is still deployable
507
+ if (!isCheckpointDeployable(stateDir, deployment.activeCheckpointId)) {
508
+ throw new Error(
509
+ `enableRoutingForProfile failed: active checkpoint "${deployment.activeCheckpointId}" ` +
510
+ `for profile "${workerProfile}" is no longer marked deployable. ` +
511
+ `Revoke deployment or bind a new checkpoint before enabling routing.`
512
+ );
513
+ }
514
+
515
+ registry.deployments[idx] = {
516
+ ...deployment,
517
+ routingEnabled: true,
518
+ updatedAt: new Date().toISOString(),
519
+ };
520
+
521
+ writeRegistry(stateDir, registry);
522
+ return registry.deployments[idx];
523
+ });
524
+ }
525
+
526
+ /**
527
+ * Disable routing for a worker profile.
528
+ * This is always safe — it does not unbind the checkpoint.
529
+ *
530
+ * @throws Error if no deployment exists for the profile
531
+ */
532
+ export function disableRoutingForProfile(
533
+ stateDir: string,
534
+ workerProfile: WorkerProfile
535
+ ): Deployment {
536
+ assertSupportedProfile(workerProfile);
537
+
538
+ return withDeploymentRegistryLock(stateDir, (registry) => {
539
+ const idx = registry.deployments.findIndex(
540
+ (d) => d.workerProfile === workerProfile
541
+ );
542
+
543
+ if (idx === -1) {
544
+ throw new Error(
545
+ `disableRoutingForProfile failed: no deployment found for profile "${workerProfile}".`
546
+ );
547
+ }
548
+
549
+ registry.deployments[idx] = {
550
+ ...registry.deployments[idx],
551
+ routingEnabled: false,
552
+ updatedAt: new Date().toISOString(),
553
+ };
554
+
555
+ writeRegistry(stateDir, registry);
556
+ return registry.deployments[idx];
557
+ });
558
+ }
559
+
560
+ /**
561
+ * Roll back the deployment for a worker profile to its previous checkpoint.
562
+ *
563
+ * ROLLBACK RULE:
564
+ * - Can only roll back if previousCheckpointId is not null
565
+ * - Sets activeCheckpointId = previousCheckpointId
566
+ * - The old activeCheckpointId becomes the new previousCheckpointId
567
+ * - routingEnabled is set to false (must be re-enabled explicitly)
568
+ *
569
+ * @throws Error if no deployment exists
570
+ * @throws Error if no previous checkpoint is available
571
+ */
572
+ export function rollbackDeployment(
573
+ stateDir: string,
574
+ workerProfile: WorkerProfile,
575
+ note?: string
576
+ ): Deployment {
577
+ assertSupportedProfile(workerProfile);
578
+
579
+ return withDeploymentRegistryLock(stateDir, (registry) => {
580
+ const idx = registry.deployments.findIndex(
581
+ (d) => d.workerProfile === workerProfile
582
+ );
583
+
584
+ if (idx === -1) {
585
+ throw new Error(
586
+ `rollbackDeployment failed: no deployment found for profile "${workerProfile}".`
587
+ );
588
+ }
589
+
590
+ const deployment = registry.deployments[idx];
591
+
592
+ if (!deployment.previousCheckpointId) {
593
+ throw new Error(
594
+ `rollbackDeployment failed: no previous checkpoint available for profile ` +
595
+ `"${workerProfile}". The current deployment has no rollback target. ` +
596
+ `(activeCheckpointId="${deployment.activeCheckpointId}", ` +
597
+ `previousCheckpointId=null)`
598
+ );
599
+ }
600
+
601
+ // Verify the rollback target checkpoint still exists and is deployable
602
+ const rollbackTarget = getCheckpoint(stateDir, deployment.previousCheckpointId);
603
+ if (!rollbackTarget) {
604
+ throw new Error(
605
+ `rollbackDeployment failed: previous checkpoint "${deployment.previousCheckpointId}" ` +
606
+ `no longer exists in the training registry. Cannot roll back to a deleted checkpoint.`
607
+ );
608
+ }
609
+ if (!isCheckpointDeployable(stateDir, deployment.previousCheckpointId)) {
610
+ throw new Error(
611
+ `rollbackDeployment failed: previous checkpoint "${deployment.previousCheckpointId}" ` +
612
+ `is no longer deployable. Roll back to a passing checkpoint or re-bind a new one.`
613
+ );
614
+ }
615
+
616
+ const now = new Date().toISOString();
617
+
618
+ // Chain the rollback: the old active becomes the new previous
619
+ const newPreviousCheckpointId = deployment.activeCheckpointId;
620
+
621
+ const rolledBack: Deployment = {
622
+ ...deployment,
623
+ activeCheckpointId: deployment.previousCheckpointId,
624
+ previousCheckpointId: newPreviousCheckpointId,
625
+ routingEnabled: false, // Always disable routing after rollback — must re-enable
626
+ updatedAt: now,
627
+ note: note ?? `Rollback to ${deployment.previousCheckpointId}`,
628
+ };
629
+
630
+ registry.deployments[idx] = rolledBack;
631
+ writeRegistry(stateDir, registry);
632
+ return rolledBack;
633
+ });
634
+ }
635
+
636
+ // ---------------------------------------------------------------------------
637
+ // Read-Only Query Helpers
638
+ // ---------------------------------------------------------------------------
639
+
640
+ /**
641
+ * Check whether a worker profile currently has an enabled deployment
642
+ * with an active checkpoint that is still deployable.
643
+ *
644
+ * GOVERNANCE: Even if routing was previously enabled, a checkpoint that
645
+ * has been revoked (marked non-deployable via markCheckpointDeployable(false))
646
+ * must not be used for routing. This prevents routing traffic to a
647
+ * checkpoint that has been superseded or failed re-evaluation.
648
+ */
649
+ export function isRoutingEnabledForProfile(
650
+ stateDir: string,
651
+ workerProfile: WorkerProfile
652
+ ): boolean {
653
+ const deployment = getDeployment(stateDir, workerProfile);
654
+ if (!deployment?.routingEnabled) return false;
655
+ if (!deployment.activeCheckpointId) return false;
656
+ // Re-check deployability on every routing decision — checkpoint may have been revoked
657
+ return isCheckpointDeployable(stateDir, deployment.activeCheckpointId);
658
+ }
659
+
660
+ /**
661
+ * Get the active checkpoint ID for a worker profile.
662
+ * Returns null if no deployment or no active checkpoint.
663
+ */
664
+ export function getActiveCheckpointForProfile(
665
+ stateDir: string,
666
+ workerProfile: WorkerProfile
667
+ ): string | null {
668
+ const deployment = getDeployment(stateDir, workerProfile);
669
+ return deployment?.activeCheckpointId ?? null;
670
+ }
671
+
672
+ /**
673
+ * Get the full deployment record with lineage context.
674
+ * Returns null if no deployment exists.
675
+ *
676
+ * Lineage includes: deployment record, active checkpoint, parent training run, eval summary.
677
+ */
678
+ export function getDeploymentLineage(
679
+ stateDir: string,
680
+ workerProfile: WorkerProfile
681
+ ): {
682
+ deployment: Deployment;
683
+ activeCheckpoint: Checkpoint | null;
684
+ } | null {
685
+ const deployment = getDeployment(stateDir, workerProfile);
686
+ if (!deployment) return null;
687
+
688
+ const activeCheckpoint = deployment.activeCheckpointId
689
+ ? getCheckpoint(stateDir, deployment.activeCheckpointId)
690
+ : null;
691
+
692
+ return { deployment, activeCheckpoint };
693
+ }
694
+
695
+ /**
696
+ * Get the complete deployment registry (for debugging/admin purposes).
697
+ */
698
+ export function getFullDeploymentRegistry(stateDir: string): ModelDeploymentRegistry {
699
+ return readRegistry(stateDir);
700
+ }
701
+
702
+ /**
703
+ * Compute stats for the deployment registry.
704
+ */
705
+ export function getDeploymentRegistryStats(
706
+ stateDir: string
707
+ ): {
708
+ totalDeployments: number;
709
+ activeDeployments: number; // deployments with routingEnabled === true
710
+ profilesWithBindings: number;
711
+ profilesWithRoutingEnabled: number;
712
+ } {
713
+ const registry = readRegistry(stateDir);
714
+ const deployments = registry.deployments;
715
+
716
+ return {
717
+ totalDeployments: deployments.length,
718
+ activeDeployments: deployments.filter((d) => d.routingEnabled).length,
719
+ profilesWithBindings: new Set(deployments.map((d) => d.workerProfile)).size,
720
+ profilesWithRoutingEnabled: deployments.filter((d) => d.routingEnabled).length,
721
+ };
722
+ }