principles-disciple 1.72.0 → 1.74.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (319) hide show
  1. package/INSTALL.md +1 -3
  2. package/openclaw.plugin.json +10 -5
  3. package/package.json +17 -19
  4. package/scripts/acceptance-test.mjs +16 -73
  5. package/scripts/sync-plugin.mjs +382 -77
  6. package/src/commands/archive-impl.ts +2 -1
  7. package/src/commands/capabilities.ts +2 -2
  8. package/src/commands/context.ts +2 -2
  9. package/src/commands/disable-impl.ts +2 -1
  10. package/src/commands/evolution-status.ts +16 -16
  11. package/src/commands/export.ts +12 -67
  12. package/src/commands/pain.ts +91 -1
  13. package/src/commands/principle-rollback.ts +2 -1
  14. package/src/commands/promote-impl.ts +7 -43
  15. package/src/commands/rollback-impl.ts +2 -1
  16. package/src/commands/rollback.ts +2 -1
  17. package/src/commands/samples.ts +2 -1
  18. package/src/commands/thinking-os.ts +2 -1
  19. package/src/config/errors.ts +18 -2
  20. package/src/constants/diagnostician.ts +2 -2
  21. package/src/constants/tools.ts +2 -1
  22. package/src/core/__tests__/focus-history.test.ts +210 -0
  23. package/src/core/config.ts +1 -1
  24. package/src/core/correction-cue-learner.ts +2 -136
  25. package/src/core/correction-types.ts +16 -88
  26. package/src/core/dictionary.ts +19 -20
  27. package/src/core/empathy-keyword-matcher.ts +17 -289
  28. package/src/core/empathy-types.ts +18 -229
  29. package/src/core/event-log.ts +29 -132
  30. package/src/core/evolution-reducer.ts +21 -2
  31. package/src/core/evolution-types.ts +76 -464
  32. package/src/core/file-store.ts +80 -0
  33. package/src/core/focus-history.ts +228 -955
  34. package/src/core/local-worker-routing.ts +34 -314
  35. package/src/core/merge-gate-audit.ts +0 -195
  36. package/src/core/migration.ts +0 -1
  37. package/src/core/pain-diagnostic-gate.ts +154 -0
  38. package/src/core/pain-signal.ts +21 -138
  39. package/src/core/pain.ts +15 -88
  40. package/src/core/path-resolver.ts +0 -1
  41. package/src/core/paths.ts +0 -1
  42. package/src/core/pd-task-reconciler.ts +26 -115
  43. package/src/core/pd-task-service.ts +9 -9
  44. package/src/core/pd-task-types.ts +23 -127
  45. package/src/core/principle-compiler/__tests__/compiler-replay-gate.test.ts +174 -0
  46. package/src/core/principle-compiler/code-validator.ts +15 -42
  47. package/src/core/principle-compiler/compiler.ts +100 -15
  48. package/src/core/principle-compiler/index.ts +5 -2
  49. package/src/core/principle-compiler/template-generator.ts +4 -104
  50. package/src/core/principle-injection.ts +10 -202
  51. package/src/core/principle-internalization/filesystem-lifecycle-datasource.ts +42 -0
  52. package/src/core/principle-internalization/lifecycle-read-model.ts +39 -242
  53. package/src/core/principle-internalization/principle-lifecycle-service.ts +12 -10
  54. package/src/core/principle-tree-ledger-adapter.ts +145 -0
  55. package/src/core/principle-tree-ledger.ts +8 -6
  56. package/src/core/reflection/reflection-context.ts +14 -109
  57. package/src/core/replay-engine.ts +8 -500
  58. package/src/core/rule-host-helpers.ts +5 -35
  59. package/src/core/rule-host-types.ts +10 -82
  60. package/src/core/rule-host.ts +6 -63
  61. package/src/core/runtime-v2-prompt-activation-reader.ts +231 -0
  62. package/src/core/session-tracker.ts +87 -101
  63. package/src/core/shadow-observation-registry.ts +19 -48
  64. package/src/core/trajectory.ts +3 -1
  65. package/src/core/workflow-funnel-loader.ts +62 -68
  66. package/src/core/workspace-context.ts +46 -0
  67. package/src/core/workspace-dir-service.ts +1 -1
  68. package/src/core/workspace-dir-validation.ts +18 -9
  69. package/src/hooks/AGENTS.md +1 -1
  70. package/src/hooks/gate-block-helper.ts +71 -64
  71. package/src/hooks/gate.ts +183 -31
  72. package/src/hooks/lifecycle.ts +30 -32
  73. package/src/hooks/llm.ts +60 -32
  74. package/src/hooks/pain.ts +297 -103
  75. package/src/hooks/prompt.ts +400 -440
  76. package/src/hooks/subagent.ts +2 -29
  77. package/src/i18n/commands.ts +2 -10
  78. package/src/index.ts +95 -85
  79. package/src/openclaw-sdk.ts +311 -0
  80. package/src/service/central-database.ts +8 -4
  81. package/src/service/evolution-queue-migration.ts +2 -1
  82. package/src/service/evolution-worker.ts +163 -1786
  83. package/src/service/internalization-trigger-adapter.ts +302 -0
  84. package/src/service/keyword-optimization-service.ts +4 -4
  85. package/src/service/monitoring-query-service.ts +1 -215
  86. package/src/service/queue-io.ts +60 -331
  87. package/src/service/runtime-summary-service.ts +59 -16
  88. package/src/service/subagent-workflow/index.ts +0 -41
  89. package/src/service/subagent-workflow/types.ts +9 -120
  90. package/src/service/subagent-workflow/workflow-store.ts +2 -119
  91. package/src/service/workflow-watchdog.ts +0 -43
  92. package/src/types/event-payload.ts +16 -74
  93. package/src/types/event-types.ts +38 -547
  94. package/src/types/hygiene-types.ts +7 -30
  95. package/src/types/principle-tree-schema.ts +20 -222
  96. package/src/types/queue.ts +15 -70
  97. package/src/types/runtime-summary.ts +5 -49
  98. package/src/utils/io.ts +8 -20
  99. package/src/utils/retry.ts +1 -1
  100. package/src/utils/shadow-fingerprint.ts +2 -2
  101. package/src/utils/workspace-resolver.ts +50 -0
  102. package/templates/langs/en/core/AGENTS.md +7 -7
  103. package/templates/langs/en/core/BOOT.md +1 -1
  104. package/templates/langs/en/core/HEARTBEAT.md +2 -2
  105. package/templates/langs/en/principles/THINKING_OS.md +3 -2
  106. package/templates/langs/en/skills/ai-sprint-orchestration/references/agent-registry.json +1 -72
  107. package/templates/langs/en/skills/ai-sprint-orchestration/references/specs/bugfix-complex-template.json +6 -6
  108. package/templates/langs/en/skills/ai-sprint-orchestration/references/specs/feature-complex-template.json +6 -6
  109. package/templates/langs/en/skills/ai-sprint-orchestration/references/specs/workflow-validation-minimal-verify.json +2 -12
  110. package/templates/langs/en/skills/ai-sprint-orchestration/references/specs/workflow-validation-minimal.json +2 -12
  111. package/templates/langs/en/skills/ai-sprint-orchestration/scripts/run.mjs +51 -15
  112. package/templates/langs/en/skills/evolve-task/SKILL.md +3 -3
  113. package/templates/langs/en/skills/pd-cli-operator/SKILL.md +67 -0
  114. package/templates/langs/en/skills/pd-diagnostician/SKILL.md +1 -1
  115. package/templates/langs/en/skills/pd-mentor/SKILL.md +2 -3
  116. package/templates/langs/en/skills/pd-pain-signal/SKILL.md +17 -39
  117. package/templates/langs/en/skills/pd-runtime-v2/SKILL.md +61 -0
  118. package/templates/langs/zh/core/AGENTS.md +7 -7
  119. package/templates/langs/zh/core/BOOT.md +1 -1
  120. package/templates/langs/zh/core/HEARTBEAT.md +2 -2
  121. package/templates/langs/zh/principles/THINKING_OS.md +3 -2
  122. package/templates/langs/zh/skills/ai-sprint-orchestration/references/agent-registry.json +1 -72
  123. package/templates/langs/zh/skills/ai-sprint-orchestration/references/specs/bugfix-complex-template.json +6 -6
  124. package/templates/langs/zh/skills/ai-sprint-orchestration/references/specs/feature-complex-template.json +6 -6
  125. package/templates/langs/zh/skills/ai-sprint-orchestration/references/specs/nocturnal-trinity-quality-enhancement.json +8 -8
  126. package/templates/langs/zh/skills/ai-sprint-orchestration/references/specs/workflow-validation-minimal-verify.json +2 -12
  127. package/templates/langs/zh/skills/ai-sprint-orchestration/references/specs/workflow-validation-minimal.json +2 -12
  128. package/templates/langs/zh/skills/ai-sprint-orchestration/scripts/run.mjs +51 -15
  129. package/templates/langs/zh/skills/ai-sprint-orchestration/test/run.test.mjs +21 -5
  130. package/templates/langs/zh/skills/evolve-task/SKILL.md +4 -4
  131. package/templates/langs/zh/skills/pd-cli-operator/SKILL.md +67 -0
  132. package/templates/langs/zh/skills/pd-diagnostician/SKILL.md +1 -1
  133. package/templates/langs/zh/skills/pd-mentor/SKILL.md +2 -3
  134. package/templates/langs/zh/skills/pd-pain-signal/SKILL.md +17 -38
  135. package/templates/langs/zh/skills/pd-runtime-v2/SKILL.md +61 -0
  136. package/tests/build-artifacts.test.ts +1 -3
  137. package/tests/commands/evolution-status.test.ts +0 -118
  138. package/tests/core/bootstrap-rules.test.ts +1 -1
  139. package/tests/core/config.test.ts +1 -1
  140. package/tests/core/event-log.test.ts +35 -0
  141. package/tests/core/evolution-engine.test.ts +610 -0
  142. package/tests/core/file-store.test.ts +102 -0
  143. package/tests/core/focus-history.test.ts +203 -11
  144. package/tests/core/merge-gate-audit.test.ts +2 -169
  145. package/tests/core/migration.test.ts +7 -7
  146. package/tests/core/model-deployment-registry.test.ts +7 -1
  147. package/tests/core/model-training-registry.test.ts +19 -0
  148. package/tests/core/observability.test.ts +0 -1
  149. package/tests/core/pain-diagnostic-gate.test.ts +498 -0
  150. package/tests/core/pain.test.ts +0 -1
  151. package/tests/core/path-resolver.test.ts +1 -1
  152. package/tests/core/paths-refactor.test.ts +0 -22
  153. package/tests/core/principle-internalization/deprecated-readiness.test.ts +2 -2
  154. package/tests/core/principle-internalization/lifecycle-metrics.test.ts +2 -2
  155. package/tests/core/principle-internalization/{internalization-routing-policy.test.ts → lifecycle-routing-policy.test.ts} +6 -6
  156. package/tests/core/principle-internalization/lineage-source-retired.test.ts +56 -0
  157. package/tests/core/principle-internalization/principle-lifecycle-service.test.ts +1 -23
  158. package/tests/core/principle-tree-ledger-adapter.test.ts +253 -0
  159. package/tests/core/reflection-context.test.ts +0 -14
  160. package/tests/core/replay-engine.test.ts +127 -215
  161. package/tests/core/rule-host-helpers.test.ts +2 -2
  162. package/tests/core/rule-implementation-runtime.test.ts +0 -27
  163. package/tests/core/workflow-funnel-loader.test.ts +162 -0
  164. package/tests/core/workspace-context.test.ts +2 -2
  165. package/tests/core/workspace-dir-validation.test.ts +8 -1
  166. package/tests/core-anti-growth.test.ts +191 -0
  167. package/tests/hook-workspace-nextaction-contract.test.ts +42 -0
  168. package/tests/hooks/confirm-first-removal.test.ts +188 -0
  169. package/tests/hooks/gate-auto-correct-shadow.test.ts +310 -0
  170. package/tests/hooks/gate-auto-correct.test.ts +665 -0
  171. package/tests/hooks/gate-no-path-write-tool.test.ts +172 -0
  172. package/tests/hooks/gate-rule-host-pipeline.test.ts +2 -1
  173. package/tests/hooks/pain.test.ts +269 -12
  174. package/tests/hooks/prompt-characterization.test.ts +500 -0
  175. package/tests/hooks/prompt-size-guard.test.ts +32 -17
  176. package/tests/hooks/runtime-v2-prompt-activation.test.ts +869 -0
  177. package/tests/index.test.ts +94 -1
  178. package/tests/integration/auto-entry-gate.test.ts +248 -0
  179. package/tests/integration/internalization-trigger-guard.test.ts +69 -0
  180. package/tests/integration/m8-legacy-paths.test.ts +63 -0
  181. package/tests/integration/runtime-v2-pain-guard.test.ts +125 -0
  182. package/tests/plugin-config-resolution-cutover.test.ts +359 -0
  183. package/tests/runtime-v2-discovery-guard.test.ts +154 -0
  184. package/tests/service/central-database.test.ts +457 -0
  185. package/tests/service/evolution-worker.correction-observer.test.ts +173 -0
  186. package/tests/service/evolution-worker.timeout.test.ts +11 -129
  187. package/tests/service/internalization-trigger-adapter.test.ts +251 -0
  188. package/tests/service/monitoring-query-service.test.ts +1 -47
  189. package/tests/service/queue-io.test.ts +1 -62
  190. package/tests/service/runtime-summary-service.test.ts +3 -1
  191. package/tests/service/workflow-watchdog.test.ts +0 -91
  192. package/tests/utils/file-lock.test.ts +5 -3
  193. package/tests/utils/session-key.test.ts +52 -0
  194. package/tests/utils/subagent-probe.test.ts +48 -1
  195. package/vitest.config.ts +4 -11
  196. package/.planning/codebase/ARCHITECTURE.md +0 -157
  197. package/.planning/codebase/CONCERNS.md +0 -145
  198. package/.planning/codebase/CONVENTIONS.md +0 -148
  199. package/.planning/codebase/INTEGRATIONS.md +0 -81
  200. package/.planning/codebase/STACK.md +0 -87
  201. package/.planning/codebase/STRUCTURE.md +0 -193
  202. package/.planning/codebase/TESTING.md +0 -243
  203. package/.planning/phases/01-basic-visualization/01-GAP-CLOSURE-VERIFICATION.md +0 -113
  204. package/docs/COMMAND_REFERENCE.md +0 -76
  205. package/docs/COMMAND_REFERENCE_EN.md +0 -79
  206. package/scripts/build-web.mjs +0 -46
  207. package/scripts/diagnose-nocturnal.mjs +0 -537
  208. package/scripts/seed-nocturnal-scenarios.mjs +0 -384
  209. package/src/commands/nocturnal-review.ts +0 -322
  210. package/src/commands/nocturnal-rollout.ts +0 -790
  211. package/src/commands/nocturnal-train.ts +0 -986
  212. package/src/commands/pd-reflect.ts +0 -88
  213. package/src/core/adaptive-thresholds.ts +0 -478
  214. package/src/core/diagnostician-task-store.ts +0 -192
  215. package/src/core/nocturnal-arbiter.ts +0 -715
  216. package/src/core/nocturnal-artifact-lineage.ts +0 -116
  217. package/src/core/nocturnal-artificer.ts +0 -257
  218. package/src/core/nocturnal-candidate-scoring.ts +0 -530
  219. package/src/core/nocturnal-compliance.ts +0 -1146
  220. package/src/core/nocturnal-dataset.ts +0 -763
  221. package/src/core/nocturnal-executability.ts +0 -428
  222. package/src/core/nocturnal-export.ts +0 -499
  223. package/src/core/nocturnal-paths.ts +0 -240
  224. package/src/core/nocturnal-reasoning-deriver.ts +0 -343
  225. package/src/core/nocturnal-rule-implementation-validator.ts +0 -246
  226. package/src/core/nocturnal-snapshot-contract.ts +0 -99
  227. package/src/core/nocturnal-trajectory-extractor.ts +0 -512
  228. package/src/core/nocturnal-trinity-types.ts +0 -218
  229. package/src/core/nocturnal-trinity.ts +0 -2680
  230. package/src/core/principle-internalization/deprecated-readiness.ts +0 -93
  231. package/src/core/principle-internalization/internalization-routing-policy.ts +0 -208
  232. package/src/core/principle-internalization/lifecycle-metrics.ts +0 -152
  233. package/src/http/principles-console-route.ts +0 -709
  234. package/src/service/central-health-service.ts +0 -49
  235. package/src/service/central-overview-service.ts +0 -138
  236. package/src/service/control-ui-query-service.ts +0 -900
  237. package/src/service/cooldown-strategy.ts +0 -97
  238. package/src/service/evolution-pain-context.ts +0 -79
  239. package/src/service/evolution-query-service.ts +0 -407
  240. package/src/service/health-query-service.ts +0 -1038
  241. package/src/service/nocturnal-config.ts +0 -214
  242. package/src/service/nocturnal-runtime.ts +0 -734
  243. package/src/service/nocturnal-service.ts +0 -1605
  244. package/src/service/nocturnal-target-selector.ts +0 -545
  245. package/src/service/sleep-cycle.ts +0 -157
  246. package/src/service/startup-reconciler.ts +0 -112
  247. package/src/service/subagent-workflow/correction-observer-types.ts +0 -82
  248. package/src/service/subagent-workflow/correction-observer-workflow-manager.ts +0 -250
  249. package/src/service/subagent-workflow/deep-reflect-workflow-manager.ts +0 -1
  250. package/src/service/subagent-workflow/dynamic-timeout.ts +0 -30
  251. package/src/service/subagent-workflow/empathy-observer-workflow-manager.ts +0 -268
  252. package/src/service/subagent-workflow/nocturnal-workflow-manager.ts +0 -795
  253. package/src/service/subagent-workflow/runtime-direct-driver.ts +0 -268
  254. package/src/service/subagent-workflow/workflow-manager-base.ts +0 -580
  255. package/src/tools/write-pain-flag.ts +0 -215
  256. package/templates/langs/en/skills/plan-script/SKILL.md +0 -32
  257. package/templates/langs/zh/skills/plan-script/SKILL.md +0 -32
  258. package/tests/commands/nocturnal-review.test.ts +0 -448
  259. package/tests/commands/nocturnal-train.test.ts +0 -97
  260. package/tests/commands/pd-reflect.test.ts +0 -49
  261. package/tests/core/adaptive-thresholds.test.ts +0 -261
  262. package/tests/core/nocturnal-arbiter.test.ts +0 -559
  263. package/tests/core/nocturnal-artifact-lineage.test.ts +0 -53
  264. package/tests/core/nocturnal-artificer.test.ts +0 -241
  265. package/tests/core/nocturnal-candidate-scoring.test.ts +0 -532
  266. package/tests/core/nocturnal-compliance-p-principles.test.ts +0 -133
  267. package/tests/core/nocturnal-compliance.test.ts +0 -646
  268. package/tests/core/nocturnal-dataset.test.ts +0 -892
  269. package/tests/core/nocturnal-e2e.test.ts +0 -234
  270. package/tests/core/nocturnal-executability.test.ts +0 -357
  271. package/tests/core/nocturnal-export.test.ts +0 -517
  272. package/tests/core/nocturnal-reasoning-deriver.test.ts +0 -372
  273. package/tests/core/nocturnal-reviewed-subset-comparison.test.ts +0 -428
  274. package/tests/core/nocturnal-rule-implementation-validator.test.ts +0 -127
  275. package/tests/core/nocturnal-snapshot-contract.test.ts +0 -121
  276. package/tests/core/nocturnal-trajectory-extractor.test.ts +0 -634
  277. package/tests/core/nocturnal-trinity.test.ts +0 -2053
  278. package/tests/core/pain-auto-repair.test.ts +0 -96
  279. package/tests/core/pain-integration.test.ts +0 -510
  280. package/tests/fixtures/nocturnal-reviewed-subset.json +0 -183
  281. package/tests/http/principles-console-route.test.ts +0 -162
  282. package/tests/integration/chaos-resilience.test.ts +0 -348
  283. package/tests/integration/empathy-workflow-integration.test.ts +0 -626
  284. package/tests/integration/pain-diagnostician-loop.e2e.test.ts +0 -380
  285. package/tests/service/control-ui-query-service.test.ts +0 -121
  286. package/tests/service/cooldown-strategy.test.ts +0 -164
  287. package/tests/service/data-endpoints-regression.test.ts +0 -834
  288. package/tests/service/empathy-observer-workflow-manager.test.ts +0 -175
  289. package/tests/service/evolution-worker.nocturnal.test.ts +0 -601
  290. package/tests/service/nocturnal-runtime-hardening.test.ts +0 -118
  291. package/tests/service/nocturnal-runtime.test.ts +0 -473
  292. package/tests/service/nocturnal-service-code-candidate.test.ts +0 -330
  293. package/tests/service/nocturnal-target-selector.test.ts +0 -615
  294. package/tests/service/startup-reconciler.test.ts +0 -148
  295. package/tests/tools/write-pain-flag.test.ts +0 -358
  296. package/ui/src/App.tsx +0 -45
  297. package/ui/src/api.ts +0 -220
  298. package/ui/src/charts.tsx +0 -955
  299. package/ui/src/components/ErrorState.tsx +0 -6
  300. package/ui/src/components/Loading.tsx +0 -13
  301. package/ui/src/components/ProtectedRoute.tsx +0 -12
  302. package/ui/src/components/Shell.tsx +0 -91
  303. package/ui/src/components/WorkspaceConfig.tsx +0 -178
  304. package/ui/src/components/index.ts +0 -5
  305. package/ui/src/context/auth.tsx +0 -80
  306. package/ui/src/context/theme.tsx +0 -66
  307. package/ui/src/hooks/useAutoRefresh.ts +0 -39
  308. package/ui/src/i18n/ui.ts +0 -473
  309. package/ui/src/main.tsx +0 -16
  310. package/ui/src/pages/EvolutionPage.tsx +0 -333
  311. package/ui/src/pages/FeedbackPage.tsx +0 -138
  312. package/ui/src/pages/GateMonitorPage.tsx +0 -136
  313. package/ui/src/pages/LoginPage.tsx +0 -89
  314. package/ui/src/pages/OverviewPage.tsx +0 -599
  315. package/ui/src/pages/SamplesPage.tsx +0 -174
  316. package/ui/src/pages/ThinkingModelsPage.tsx +0 -702
  317. package/ui/src/styles.css +0 -2020
  318. package/ui/src/types.ts +0 -384
  319. package/ui/src/utils/format.ts +0 -15
@@ -1,88 +1,12 @@
1
-
1
+
2
2
  import * as fs from 'fs';
3
3
  import * as path from 'path';
4
- import { withLock } from '../utils/file-lock.js';
5
- import { normalizePath, isRisky, planStatus as getPlanStatus, atomicWriteFileSync } from '../utils/io.js';
6
- import {
7
- listSamplesByClassification,
8
- loadSampleContent,
9
- } from './nocturnal-dataset.js';
10
- import {
11
- getImplementationAssetRoot,
12
- loadEntrySource,
13
- } from './code-implementation-storage.js';
14
- import type {
15
- NocturnalDatasetRecord,
16
- SampleClassification,
17
- } from './nocturnal-dataset.js';
4
+ import { getImplementationAssetRoot } from './code-implementation-storage.js';
18
5
  import { loadLedger } from './principle-tree-ledger.js';
19
6
  import type { Implementation } from '../types/principle-tree-schema.js';
20
- import type { RuleHostHelpers } from './rule-host-helpers.js';
21
- import { createRuleHostHelpers } from './rule-host-helpers.js';
22
- import type { RuleHostInput, RuleHostResult } from './rule-host-types.js';
23
- import { loadRuleImplementationModule } from './rule-implementation-runtime.js';
24
- import {
25
- getNocturnalSessionSnapshot,
26
- type NocturnalGateBlock,
27
- type NocturnalSessionSnapshot,
28
- type NocturnalToolCall,
29
- } from './nocturnal-trajectory-extractor.js';
30
- import { TrajectoryRegistry } from './trajectory.js';
31
-
32
- export interface ReplaySample {
33
- fingerprint: string;
34
- classification: SampleClassification;
35
- content: unknown;
36
- expectedOutcome: {
37
- shouldBlock?: boolean;
38
- shouldPass?: boolean;
39
- expectedPrinciple?: string;
40
- };
41
- record: NocturnalDatasetRecord;
42
- }
7
+ import type { ReplayReport, ClassificationSummary, ReplayResult } from '@principles/core/runtime-v2';
43
8
 
44
- export interface ReplayResult {
45
- sampleFingerprint: string;
46
- classification: SampleClassification;
47
- passed: boolean;
48
- reason?: string;
49
- decision: string;
50
- }
51
-
52
- export interface ClassificationSummary {
53
- total: number;
54
- passed: number;
55
- failed: number;
56
- details: ReplayResult[];
57
- }
58
-
59
- export interface ReplayReport {
60
- overallDecision: 'pass' | 'fail' | 'needs-review';
61
- replayResults: {
62
- painNegative: ClassificationSummary;
63
- successPositive: ClassificationSummary;
64
- principleAnchor: ClassificationSummary;
65
- };
66
- blockers: string[];
67
- evidenceSummary: {
68
- evidenceStatus: 'observed' | 'empty';
69
- totalSamples: number;
70
- classifiedCounts: {
71
- painNegative: number;
72
- successPositive: number;
73
- principleAnchor: number;
74
- };
75
- };
76
- generatedAt: string;
77
- implementationId: string;
78
- sampleFingerprints: string[];
79
- }
80
-
81
- export interface CandidateEvaluator {
82
-
83
- evaluate(sample: unknown): { passed: boolean; reason?: string; decision: string };
84
-
85
- }
9
+ export type { ReplayReport, ClassificationSummary, ReplayResult };
86
10
 
87
11
  export class ReplayEngine {
88
12
  private readonly workspaceDir: string;
@@ -93,79 +17,6 @@ export class ReplayEngine {
93
17
  this.stateDir = stateDir;
94
18
  }
95
19
 
96
- loadSamples(classifications: SampleClassification[]): ReplaySample[] {
97
- const samples: ReplaySample[] = [];
98
-
99
- for (const classification of classifications) {
100
- const records = listSamplesByClassification(this.workspaceDir, classification);
101
-
102
- for (const record of records) {
103
- try {
104
- const content = loadSampleContent(this.workspaceDir, record);
105
- const expectedOutcome = this._deriveExpectedOutcome(record);
106
-
107
- samples.push({
108
- fingerprint: record.sampleFingerprint,
109
- classification,
110
- content,
111
- expectedOutcome,
112
- record,
113
- });
114
- } catch (err) {
115
- console.warn(
116
- `[ReplayEngine] Skipping sample ${record.sampleFingerprint}: ${String(err)}`
117
- );
118
- }
119
- }
120
- }
121
-
122
- return samples;
123
- }
124
-
125
-
126
-
127
- runSingleSample(sample: ReplaySample, evaluator: CandidateEvaluator): ReplayResult {
128
- const evaluation = evaluator.evaluate(sample);
129
- return {
130
- sampleFingerprint: sample.fingerprint,
131
- classification: sample.classification,
132
- passed: evaluation.passed,
133
- reason: evaluation.passed ? undefined : evaluation.reason,
134
- decision: evaluation.decision,
135
- };
136
- }
137
-
138
- runReplay(
139
- candidateImplId: string,
140
- evaluator: CandidateEvaluator,
141
- classifications?: SampleClassification[]
142
- ): ReplayReport {
143
- const selectedClassifications: SampleClassification[] = classifications ?? [
144
- 'pain-negative',
145
- 'success-positive',
146
- 'principle-anchor',
147
- ];
148
-
149
- const samples = this.loadSamples(selectedClassifications);
150
- const allResults = samples.map((sample) => this.runSingleSample(sample, evaluator));
151
- const report = this._buildReport(candidateImplId, allResults);
152
- this._persistReport(report);
153
- return report;
154
- }
155
-
156
- runReplayForImplementation(
157
- implementationId: string,
158
- classifications?: SampleClassification[],
159
- ): ReplayReport {
160
- const implementation = this._getImplementationById(implementationId);
161
- if (!implementation) {
162
- throw new Error(`Implementation not found: ${implementationId}`);
163
- }
164
-
165
- const evaluator = this._createEvaluatorForImplementation(implementation);
166
- return this.runReplay(implementationId, evaluator, classifications);
167
- }
168
-
169
20
  listReports(implementationId: string): ReplayReport[] {
170
21
  const reportDir = path.join(
171
22
  getImplementationAssetRoot(this.stateDir, implementationId),
@@ -183,9 +34,10 @@ export class ReplayEngine {
183
34
  const content = fs.readFileSync(path.join(reportDir, file), 'utf-8');
184
35
  return JSON.parse(content) as ReplayReport;
185
36
  });
186
- } catch {
187
- return [];
188
- }
37
+ } catch (err) {
38
+ console.warn(`[ReplayEngine] Failed to read reports in ${reportDir}: ${err instanceof Error ? err.message : String(err)}`);
39
+ return [];
40
+ }
189
41
  }
190
42
 
191
43
  getLatestReport(implementationId: string): ReplayReport | null {
@@ -201,350 +53,6 @@ export class ReplayEngine {
201
53
  const ledger = loadLedger(this.stateDir);
202
54
  return ledger.tree.implementations[implementationId] ?? null;
203
55
  }
204
-
205
- private _createEvaluatorForImplementation(implementation: Implementation): CandidateEvaluator {
206
- const sourceCode = loadEntrySource(this.stateDir, implementation.id);
207
- if (!sourceCode) {
208
- throw new Error(`Implementation asset entry source missing: ${implementation.id}`);
209
- }
210
-
211
- const moduleExports = loadRuleImplementationModule(sourceCode, implementation.id);
212
- if (typeof moduleExports.evaluate !== 'function') {
213
- throw new Error(`Implementation ${implementation.id} does not export evaluate().`);
214
- }
215
-
216
-
217
- const evaluate = moduleExports.evaluate as (
218
- _input: RuleHostInput,
219
- _helpers: RuleHostHelpers,
220
- ) => RuleHostResult;
221
-
222
-
223
- return {
224
- evaluate: (sample: unknown) => {
225
- const replaySample = sample as ReplaySample;
226
- const input = this._buildRuleHostInput(replaySample);
227
- if (!input) {
228
- return {
229
- passed: false,
230
- reason: `Could not build replay input for sample ${replaySample.fingerprint}.`,
231
- decision: 'replay-input-missing',
232
- };
233
- }
234
-
235
- const result = evaluate(input, createRuleHostHelpers(input));
236
- return this._scoreEvaluation(replaySample, result);
237
- },
238
- };
239
- }
240
-
241
-
242
- private _buildRuleHostInput(sample: ReplaySample): RuleHostInput | null {
243
- const snapshot = getNocturnalSessionSnapshot(
244
- TrajectoryRegistry.get(this.workspaceDir),
245
- sample.record.sessionId,
246
- );
247
- if (!snapshot) {
248
- return null;
249
- }
250
-
251
- const toolCall = this._selectToolCall(snapshot, sample.classification);
252
- if (!toolCall) {
253
- return null;
254
- }
255
-
256
- const normalizedPath =
257
- typeof toolCall.filePath === 'string' && toolCall.filePath.length > 0
258
- ? normalizePath(toolCall.filePath, this.workspaceDir)
259
- : null;
260
- const matchedGateBlock = this._matchGateBlock(snapshot.gateBlocks, toolCall);
261
-
262
- return {
263
- action: {
264
- toolName: toolCall.toolName,
265
- normalizedPath,
266
- paramsSummary: {
267
- artifactId: sample.record.artifactId,
268
- sourceSnapshotRef: sample.record.sourceSnapshotRef,
269
- classification: sample.classification,
270
- },
271
- },
272
- workspace: {
273
- isRiskPath:
274
- Boolean(matchedGateBlock) ||
275
- (normalizedPath !== null && this._isRiskPath(normalizedPath)),
276
- planStatus:
277
- matchedGateBlock?.planStatus === 'READY' ||
278
- matchedGateBlock?.planStatus === 'DRAFT' ||
279
- matchedGateBlock?.planStatus === 'NONE'
280
- ? matchedGateBlock.planStatus
281
- : this._safePlanStatus(),
282
- hasPlanFile: fs.existsSync(path.join(this.workspaceDir, 'PLAN.md')),
283
- },
284
- session: {
285
- sessionId: sample.record.sessionId,
286
- currentGfi: 0,
287
- recentThinking: false,
288
- },
289
- evolution: {
290
- epTier: 0,
291
- },
292
- derived: {
293
- estimatedLineChanges: this._estimateLineChanges(toolCall),
294
- bashRisk: this._inferBashRisk(toolCall),
295
- },
296
- };
297
- }
298
-
299
-
300
-
301
- private _selectToolCall(
302
- snapshot: NocturnalSessionSnapshot,
303
- classification: SampleClassification,
304
- ): NocturnalToolCall | null {
305
- const byNewest = [...snapshot.toolCalls].sort(
306
- (left, right) => new Date(right.createdAt).getTime() - new Date(left.createdAt).getTime(),
307
- );
308
-
309
- if (classification === 'pain-negative') {
310
- return (
311
- byNewest.find((toolCall) => toolCall.outcome === 'blocked') ??
312
- byNewest.find((toolCall) => toolCall.outcome === 'failure') ??
313
- byNewest[0] ??
314
- null
315
- );
316
- }
317
-
318
- if (classification === 'success-positive' || classification === 'principle-anchor') {
319
- return (
320
- byNewest.find((toolCall) => toolCall.outcome === 'success') ??
321
- byNewest.find((toolCall) => toolCall.outcome === 'failure') ??
322
- byNewest[0] ??
323
- null
324
- );
325
- }
326
-
327
- return byNewest[0] ?? null;
328
- }
329
-
330
-
331
-
332
- private _matchGateBlock(
333
- gateBlocks: NocturnalGateBlock[],
334
- toolCall: NocturnalToolCall,
335
- ): NocturnalGateBlock | null {
336
- return (
337
- [...gateBlocks]
338
- .sort((left, right) => new Date(right.createdAt).getTime() - new Date(left.createdAt).getTime())
339
- .find((gateBlock) => gateBlock.toolName === toolCall.toolName) ?? null
340
- );
341
- }
342
-
343
- private _isRiskPath(normalizedPath: string): boolean {
344
- try {
345
- const profilePath = path.join(this.workspaceDir, 'PROFILE.json');
346
- const riskPaths =
347
- fs.existsSync(profilePath)
348
- ? (((JSON.parse(fs.readFileSync(profilePath, 'utf-8')) as { risk_paths?: unknown }).risk_paths as string[] | undefined) ?? [])
349
- : [];
350
- return isRisky(normalizedPath, riskPaths);
351
- } catch {
352
- return false;
353
- }
354
- }
355
-
356
- private _safePlanStatus(): 'NONE' | 'DRAFT' | 'READY' | 'UNKNOWN' {
357
- try {
358
- const status = getPlanStatus(this.workspaceDir);
359
- if (status === 'READY') return 'READY';
360
- if (status === 'DRAFT') return 'DRAFT';
361
- if (status === '') return 'NONE';
362
- return 'UNKNOWN';
363
- } catch {
364
- return 'UNKNOWN';
365
- }
366
- }
367
-
368
-
369
-
370
- private _estimateLineChanges(toolCall: NocturnalToolCall): number {
371
- if (toolCall.toolName === 'edit' || toolCall.toolName === 'write') {
372
- return 20;
373
- }
374
- return 0;
375
- }
376
-
377
-
378
-
379
- private _inferBashRisk(toolCall: NocturnalToolCall): 'safe' | 'normal' | 'dangerous' | 'unknown' {
380
- if (toolCall.toolName !== 'bash' && toolCall.toolName !== 'run_shell_command') {
381
- return 'unknown';
382
- }
383
- const errorText = `${toolCall.errorType ?? ''} ${toolCall.errorMessage ?? ''}`;
384
- if (/\brm\s+-rf\b|\bchmod\b|\bchown\b|>\s*\/dev\//.test(errorText)) {
385
- return 'dangerous';
386
- }
387
- return toolCall.outcome === 'success' ? 'safe' : 'normal';
388
- }
389
-
390
-
391
-
392
-
393
- private _scoreEvaluation(
394
- sample: ReplaySample,
395
- result: RuleHostResult,
396
- ): { passed: boolean; reason?: string; decision: string } {
397
- switch (sample.classification) {
398
- case 'pain-negative':
399
- return {
400
- passed: result.decision === 'block' || result.decision === 'requireApproval',
401
- reason:
402
- result.decision === 'block' || result.decision === 'requireApproval'
403
- ? undefined
404
- : `Expected block/requireApproval but received ${result.decision}.`,
405
- decision: result.decision,
406
- };
407
- case 'success-positive':
408
- return {
409
- passed: result.decision === 'allow' || !result.matched,
410
- reason:
411
- result.decision === 'allow' || !result.matched
412
- ? undefined
413
- : `Expected allow/no-match but received ${result.decision}.`,
414
- decision: result.decision,
415
- };
416
- case 'principle-anchor':
417
- return {
418
- passed: result.decision !== 'block',
419
- reason:
420
- result.decision !== 'block'
421
- ? undefined
422
- : 'Principle-anchor sample should not regress to a hard block.',
423
- decision: result.decision,
424
- };
425
- default:
426
- return {
427
- passed: false,
428
- reason: 'Unknown replay classification.',
429
- decision: result.decision,
430
- };
431
- }
432
- }
433
-
434
- private _buildReport(
435
- implementationId: string,
436
- results: ReplayResult[]
437
- ): ReplayReport {
438
- const painNegative = results.filter((result) => result.classification === 'pain-negative');
439
- const successPositive = results.filter((result) => result.classification === 'success-positive');
440
- const principleAnchor = results.filter((result) => result.classification === 'principle-anchor');
441
-
442
- const toSummary = (details: ReplayResult[]): ClassificationSummary => ({
443
- total: details.length,
444
- passed: details.filter((result) => result.passed).length,
445
- failed: details.filter((result) => !result.passed).length,
446
- details,
447
- });
448
-
449
- const painSummary = toSummary(painNegative);
450
- const successSummary = toSummary(successPositive);
451
- const anchorSummary = toSummary(principleAnchor);
452
- const blockers: string[] = [];
453
- const totalSamples = results.length;
454
-
455
- if (totalSamples === 0) {
456
- blockers.push('NO REPLAY EVIDENCE: No classified replay samples were available. Report cannot justify promotion-quality conclusions.');
457
- }
458
-
459
- for (const leak of painSummary.details.filter((result) => !result.passed)) {
460
- blockers.push(
461
- `PAIN-NEGATIVE LEAK: Sample ${leak.sampleFingerprint} was not blocked. ${leak.reason ?? ''}`
462
- );
463
- }
464
-
465
- for (const violation of anchorSummary.details.filter((result) => !result.passed)) {
466
- blockers.push(
467
- `PRINCIPLE-ANCHOR VIOLATION: Sample ${violation.sampleFingerprint} did not adhere. ${violation.reason ?? ''}`
468
- );
469
- }
470
-
471
- for (const falsePositive of successSummary.details.filter((result) => !result.passed)) {
472
- blockers.push(
473
- `FALSE POSITIVE: Sample ${falsePositive.sampleFingerprint} was incorrectly blocked. ${falsePositive.reason ?? ''}`
474
- );
475
- }
476
-
477
- return {
478
- overallDecision: this._determineDecision(painSummary, successSummary, anchorSummary),
479
- replayResults: {
480
- painNegative: painSummary,
481
- successPositive: successSummary,
482
- principleAnchor: anchorSummary,
483
- },
484
- blockers,
485
- evidenceSummary: {
486
- evidenceStatus: totalSamples > 0 ? 'observed' : 'empty',
487
- totalSamples,
488
- classifiedCounts: {
489
- painNegative: painSummary.total,
490
- successPositive: successSummary.total,
491
- principleAnchor: anchorSummary.total,
492
- },
493
- },
494
- generatedAt: new Date().toISOString(),
495
- implementationId,
496
- sampleFingerprints: results.map((result) => result.sampleFingerprint),
497
- };
498
- }
499
-
500
-
501
-
502
- private _determineDecision(
503
- pain: ClassificationSummary,
504
- success: ClassificationSummary,
505
- anchor: ClassificationSummary
506
- ): 'pass' | 'fail' | 'needs-review' {
507
- if (pain.total + success.total + anchor.total === 0) return 'needs-review';
508
- if (pain.failed > 0) return 'fail';
509
- if (anchor.failed > 0) return 'fail';
510
- if (success.failed > 0) return 'needs-review';
511
- return 'pass';
512
- }
513
-
514
- private _persistReport(report: ReplayReport): void {
515
- const reportDir = path.join(
516
- getImplementationAssetRoot(this.stateDir, report.implementationId),
517
- 'replays'
518
- );
519
-
520
- if (!fs.existsSync(reportDir)) {
521
- fs.mkdirSync(reportDir, { recursive: true });
522
- }
523
-
524
- const timestamp = new Date().toISOString().replace(/[:.]/g, '-');
525
- const reportPath = path.join(reportDir, `${timestamp}.json`);
526
-
527
- withLock(reportPath, () => {
528
- atomicWriteFileSync(reportPath, JSON.stringify(report, null, 2));
529
- });
530
- }
531
-
532
-
533
-
534
- private _deriveExpectedOutcome(
535
- record: NocturnalDatasetRecord,
536
- ): ReplaySample['expectedOutcome'] {
537
- switch (record.classification) {
538
- case 'pain-negative':
539
- return { shouldBlock: true };
540
- case 'success-positive':
541
- return { shouldPass: true };
542
- case 'principle-anchor':
543
- return { expectedPrinciple: record.principleId };
544
- default:
545
- return {};
546
- }
547
- }
548
56
  }
549
57
 
550
58
  export function formatReplayReport(report: ReplayReport): string {
@@ -1,39 +1,9 @@
1
1
  /**
2
- * Rule Host Helpers — Minimal whitelisted helper surface for hosted implementations
2
+ * Rule Host Helpers — Re-exported from @principles/core (PRI-42)
3
3
  *
4
- * PURPOSE: Provide a constrained set of pure functions that hosted implementations
5
- * can call. All values are pre-computed from the frozen RuleHostInput snapshot.
6
- *
7
- * SECURITY:
8
- * - Helpers are a frozen object — cannot be modified by implementations
9
- * - No filesystem, process, require, dynamic import, eval, Function, or network access
10
- * - All functions are pure — no side effects, no external state access
4
+ * Canonical definitions moved to:
5
+ * packages/principles-core/src/runtime-v2/internalization/rule-host-helpers.ts
11
6
  */
12
7
 
13
- import type { RuleHostInput } from './rule-host-types.js';
14
-
15
- export interface RuleHostHelpers {
16
- isRiskPath(): boolean;
17
- getToolName(): string;
18
- getEstimatedLineChanges(): number;
19
- getBashRisk(): 'safe' | 'normal' | 'dangerous' | 'unknown';
20
- hasPlanFile(): boolean;
21
- getPlanStatus(): 'NONE' | 'DRAFT' | 'READY' | 'UNKNOWN';
22
- getCurrentEpiTier(): number;
23
- }
24
-
25
- /**
26
- * Create a frozen helper object from the pre-computed input snapshot.
27
- * Implementations receive this via the vm context — they cannot modify it.
28
- */
29
- export function createRuleHostHelpers(input: RuleHostInput): RuleHostHelpers {
30
- return Object.freeze({
31
- isRiskPath: () => input.workspace.isRiskPath,
32
- getToolName: () => input.action.toolName,
33
- getEstimatedLineChanges: () => input.derived.estimatedLineChanges,
34
- getBashRisk: () => input.derived.bashRisk,
35
- hasPlanFile: () => input.workspace.hasPlanFile,
36
- getPlanStatus: () => input.workspace.planStatus,
37
- getCurrentEpiTier: () => input.evolution.epTier,
38
- });
39
- }
8
+ export type { RuleHostHelpers } from '@principles/core/runtime-v2';
9
+ export { createRuleHostHelpers } from '@principles/core/runtime-v2';
@@ -1,86 +1,14 @@
1
1
  /**
2
- * Rule Host Types — Execution contracts for hosted code implementations
2
+ * Rule Host Types — Re-exported from @principles/core (PRI-42)
3
3
  *
4
- * PURPOSE: Define the constrained interface through which active code
5
- * implementations are executed. Implementations receive a frozen snapshot
6
- * of context and return one of three decisions.
7
- *
8
- * TRUST BOUNDARY:
9
- * - RuleHostInput is a frozen snapshot — no live workspace handles
10
- * - Implementations execute in a constrained vm context with minimal helpers
11
- * - No filesystem, process, require, dynamic import, eval, or network access
4
+ * Canonical definitions moved to:
5
+ * packages/principles-core/src/runtime-v2/internalization/rule-host-contracts.ts
12
6
  */
13
7
 
14
- // ---------------------------------------------------------------------------
15
- // Input: Frozen snapshot provided to implementations
16
- // ---------------------------------------------------------------------------
17
-
18
- export interface RuleHostInput {
19
- action: {
20
- toolName: string;
21
- normalizedPath: string | null;
22
- paramsSummary: Record<string, unknown>;
23
- };
24
- workspace: {
25
- isRiskPath: boolean;
26
- planStatus: 'NONE' | 'DRAFT' | 'READY' | 'UNKNOWN';
27
- hasPlanFile: boolean;
28
- };
29
- session: {
30
- sessionId?: string;
31
- currentGfi: number;
32
- recentThinking: boolean;
33
- };
34
- evolution: {
35
- epTier: number;
36
- };
37
- derived: {
38
- estimatedLineChanges: number;
39
- bashRisk: 'safe' | 'normal' | 'dangerous' | 'unknown';
40
- };
41
- }
42
-
43
- // ---------------------------------------------------------------------------
44
- // Decision: Limited to three outcomes
45
- // ---------------------------------------------------------------------------
46
-
47
- export type RuleHostDecision = 'allow' | 'block' | 'requireApproval';
48
-
49
- // ---------------------------------------------------------------------------
50
- // Meta: Exported by each implementation for identification
51
- // ---------------------------------------------------------------------------
52
-
53
- export interface RuleHostMeta {
54
- name: string;
55
- version: string;
56
- ruleId: string;
57
- coversCondition: string;
58
- }
59
-
60
- // ---------------------------------------------------------------------------
61
- // Result: Structured output from a single implementation evaluation
62
- // ---------------------------------------------------------------------------
63
-
64
- export interface RuleHostResult {
65
- decision: RuleHostDecision;
66
- matched: boolean;
67
- reason: string;
68
- diagnostics?: Record<string, unknown>;
69
- /** C: Rule ID that produced this result (for observability events) */
70
- ruleId?: string;
71
- /** C: Principle ID that this rule implements (for observability events) */
72
- principleId?: string;
73
- }
74
-
75
- // ---------------------------------------------------------------------------
76
- // LoadedImplementation: A successfully loaded active implementation
77
- // ---------------------------------------------------------------------------
78
-
79
- export interface LoadedImplementation {
80
- implId: string;
81
- ruleId: string;
82
- meta: RuleHostMeta;
83
-
84
- evaluate: (_input: RuleHostInput) => RuleHostResult;
85
-
86
- }
8
+ export type {
9
+ RuleHostInput,
10
+ RuleHostDecision,
11
+ RuleHostMeta,
12
+ RuleHostResult,
13
+ LoadedImplementation,
14
+ } from '@principles/core/runtime-v2';