akm-cli 0.6.1 → 0.7.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 (333) hide show
  1. package/CHANGELOG.md +66 -0
  2. package/dist/{cli.js → src/cli.js} +712 -34
  3. package/dist/{commands → src/commands}/config-cli.js +47 -4
  4. package/dist/src/commands/distill.js +283 -0
  5. package/dist/src/commands/events.js +108 -0
  6. package/dist/src/commands/history.js +191 -0
  7. package/dist/{commands → src/commands}/installed-stashes.js +1 -1
  8. package/dist/src/commands/proposal.js +119 -0
  9. package/dist/src/commands/propose.js +171 -0
  10. package/dist/src/commands/reflect.js +193 -0
  11. package/dist/{commands → src/commands}/registry-search.js +71 -7
  12. package/dist/{commands → src/commands}/remember.js +12 -0
  13. package/dist/{commands → src/commands}/search.js +104 -4
  14. package/dist/{commands → src/commands}/self-update.js +4 -3
  15. package/dist/{commands → src/commands}/show.js +73 -0
  16. package/dist/{commands → src/commands}/source-add.js +5 -1
  17. package/dist/{commands → src/commands}/source-manage.js +7 -1
  18. package/dist/{core → src/core}/asset-ref.js +5 -5
  19. package/dist/{core → src/core}/asset-spec.js +12 -0
  20. package/dist/{core → src/core}/common.js +1 -1
  21. package/dist/{core → src/core}/config.js +203 -121
  22. package/dist/{core → src/core}/errors.js +4 -0
  23. package/dist/src/core/events.js +239 -0
  24. package/dist/src/core/lesson-lint.js +86 -0
  25. package/dist/src/core/proposals.js +406 -0
  26. package/dist/src/core/warn.js +72 -0
  27. package/dist/{core → src/core}/write-source.js +80 -5
  28. package/dist/{indexer → src/indexer}/db-search.js +114 -24
  29. package/dist/{indexer → src/indexer}/db.js +76 -23
  30. package/dist/{indexer → src/indexer}/file-context.js +0 -3
  31. package/dist/src/indexer/graph-boost.js +179 -0
  32. package/dist/src/indexer/graph-extraction.js +212 -0
  33. package/dist/{indexer → src/indexer}/indexer.js +88 -7
  34. package/dist/{indexer → src/indexer}/matchers.js +1 -1
  35. package/dist/src/indexer/memory-inference.js +263 -0
  36. package/dist/{indexer → src/indexer}/metadata.js +111 -3
  37. package/dist/{indexer → src/indexer}/search-source.js +4 -2
  38. package/dist/src/integrations/agent/config.js +292 -0
  39. package/dist/src/integrations/agent/detect.js +94 -0
  40. package/dist/src/integrations/agent/index.js +17 -0
  41. package/dist/src/integrations/agent/profiles.js +65 -0
  42. package/dist/src/integrations/agent/prompts.js +167 -0
  43. package/dist/src/integrations/agent/spawn.js +272 -0
  44. package/dist/{integrations → src/integrations}/github.js +9 -3
  45. package/dist/{integrations → src/integrations}/lockfile.js +0 -26
  46. package/dist/{llm → src/llm}/client.js +33 -2
  47. package/dist/{llm → src/llm}/embedders/remote.js +37 -3
  48. package/dist/src/llm/feature-gate.js +108 -0
  49. package/dist/src/llm/graph-extract.js +107 -0
  50. package/dist/src/llm/index-passes.js +35 -0
  51. package/dist/src/llm/memory-infer.js +86 -0
  52. package/dist/{output → src/output}/cli-hints.js +15 -2
  53. package/dist/{output → src/output}/renderers.js +63 -2
  54. package/dist/src/output/shapes.js +523 -0
  55. package/dist/src/output/text.js +1116 -0
  56. package/dist/{registry → src/registry}/build-index.js +19 -8
  57. package/dist/{registry → src/registry}/factory.js +0 -8
  58. package/dist/{registry → src/registry}/providers/static-index.js +6 -3
  59. package/dist/{registry → src/registry}/resolve.js +68 -2
  60. package/dist/{setup → src/setup}/setup.js +52 -5
  61. package/dist/{sources → src/sources}/providers/git.js +7 -15
  62. package/dist/{wiki → src/wiki}/wiki.js +54 -6
  63. package/dist/{workflows → src/workflows}/runs.js +37 -3
  64. package/dist/tests/add-website-source.test.js +119 -0
  65. package/dist/tests/agent/agent-config-loader.test.js +70 -0
  66. package/dist/tests/agent/agent-config.test.js +221 -0
  67. package/dist/tests/agent/agent-detect.test.js +100 -0
  68. package/dist/tests/agent/agent-spawn.test.js +234 -0
  69. package/dist/tests/agent-output.test.js +186 -0
  70. package/dist/tests/architecture/agent-no-llm-sdk-guard.test.js +103 -0
  71. package/dist/tests/architecture/agent-spawn-seam.test.js +193 -0
  72. package/dist/tests/architecture/llm-stateless-seam.test.js +112 -0
  73. package/dist/tests/asset-ref.test.js +192 -0
  74. package/dist/tests/asset-registry.test.js +103 -0
  75. package/dist/tests/asset-spec.test.js +241 -0
  76. package/dist/tests/bench/attribution.test.js +996 -0
  77. package/dist/tests/bench/cleanup-sigint.test.js +83 -0
  78. package/dist/tests/bench/cleanup.js +234 -0
  79. package/dist/tests/bench/cleanup.test.js +166 -0
  80. package/dist/tests/bench/cli.js +1018 -0
  81. package/dist/tests/bench/cli.test.js +445 -0
  82. package/dist/tests/bench/compare.test.js +556 -0
  83. package/dist/tests/bench/corpus.js +317 -0
  84. package/dist/tests/bench/corpus.test.js +258 -0
  85. package/dist/tests/bench/doctor.js +525 -0
  86. package/dist/tests/bench/driver.js +401 -0
  87. package/dist/tests/bench/driver.test.js +584 -0
  88. package/dist/tests/bench/environment.js +233 -0
  89. package/dist/tests/bench/environment.test.js +199 -0
  90. package/dist/tests/bench/evolve-metrics.js +179 -0
  91. package/dist/tests/bench/evolve-metrics.test.js +187 -0
  92. package/dist/tests/bench/evolve.js +647 -0
  93. package/dist/tests/bench/evolve.test.js +624 -0
  94. package/dist/tests/bench/failure-modes.test.js +349 -0
  95. package/dist/tests/bench/feedback-integrity.test.js +457 -0
  96. package/dist/tests/bench/leakage.test.js +228 -0
  97. package/dist/tests/bench/learning-curve.test.js +134 -0
  98. package/dist/tests/bench/metrics.js +2395 -0
  99. package/dist/tests/bench/metrics.test.js +1150 -0
  100. package/dist/tests/bench/no-os-tmpdir-invariant.test.js +43 -0
  101. package/dist/tests/bench/opencode-config.js +194 -0
  102. package/dist/tests/bench/opencode-config.test.js +370 -0
  103. package/dist/tests/bench/report.js +1885 -0
  104. package/dist/tests/bench/report.test.js +1038 -0
  105. package/dist/tests/bench/run-config.js +355 -0
  106. package/dist/tests/bench/run-config.test.js +298 -0
  107. package/dist/tests/bench/run-curate-test.js +32 -0
  108. package/dist/tests/bench/run-failing-tasks.js +56 -0
  109. package/dist/tests/bench/run-full-bench.js +51 -0
  110. package/dist/tests/bench/run-items36-targeted.js +69 -0
  111. package/dist/tests/bench/run-nano-quick.js +42 -0
  112. package/dist/tests/bench/run-waveg-targeted.js +62 -0
  113. package/dist/tests/bench/runner.js +699 -0
  114. package/dist/tests/bench/runner.test.js +958 -0
  115. package/dist/tests/bench/search-bridge.test.js +331 -0
  116. package/dist/tests/bench/tmp.js +131 -0
  117. package/dist/tests/bench/trajectory.js +116 -0
  118. package/dist/tests/bench/trajectory.test.js +127 -0
  119. package/dist/tests/bench/verifier.js +114 -0
  120. package/dist/tests/bench/verifier.test.js +118 -0
  121. package/dist/tests/bench/workflow-evaluator.js +557 -0
  122. package/dist/tests/bench/workflow-evaluator.test.js +421 -0
  123. package/dist/tests/bench/workflow-spec.js +345 -0
  124. package/dist/tests/bench/workflow-spec.test.js +363 -0
  125. package/dist/tests/bench/workflow-trace.js +472 -0
  126. package/dist/tests/bench/workflow-trace.test.js +254 -0
  127. package/dist/tests/benchmark-search-quality.js +536 -0
  128. package/dist/tests/benchmark-suite.js +1441 -0
  129. package/dist/tests/capture-cli.test.js +112 -0
  130. package/dist/tests/cli-errors.test.js +204 -0
  131. package/dist/tests/commands/events.test.js +370 -0
  132. package/dist/tests/commands/history.test.js +418 -0
  133. package/dist/tests/commands/import.test.js +103 -0
  134. package/dist/tests/commands/proposal-cli.test.js +209 -0
  135. package/dist/tests/commands/reflect-propose-cli.test.js +333 -0
  136. package/dist/tests/commands/remember.test.js +97 -0
  137. package/dist/tests/commands/scope-flags.test.js +300 -0
  138. package/dist/tests/commands/search.test.js +537 -0
  139. package/dist/tests/commands/show-indexer-parity.test.js +117 -0
  140. package/dist/tests/commands/show.test.js +294 -0
  141. package/dist/tests/common.test.js +266 -0
  142. package/dist/tests/completions.test.js +142 -0
  143. package/dist/tests/config-cli.test.js +193 -0
  144. package/dist/tests/config-llm-features.test.js +139 -0
  145. package/dist/tests/config.test.js +569 -0
  146. package/dist/tests/contracts/migration-baseline.test.js +43 -0
  147. package/dist/tests/contracts/reflect-propose-envelope.test.js +139 -0
  148. package/dist/tests/contracts/spec-helpers.js +46 -0
  149. package/dist/tests/contracts/v1-spec-section-11-proposal-queue.test.js +228 -0
  150. package/dist/tests/contracts/v1-spec-section-12-agent-config.test.js +56 -0
  151. package/dist/tests/contracts/v1-spec-section-13-lesson-type.test.js +34 -0
  152. package/dist/tests/contracts/v1-spec-section-14-llm-features.test.js +94 -0
  153. package/dist/tests/contracts/v1-spec-section-4-1-asset-types.test.js +39 -0
  154. package/dist/tests/contracts/v1-spec-section-4-2-quality-rules.test.js +44 -0
  155. package/dist/tests/contracts/v1-spec-section-5-configuration.test.js +47 -0
  156. package/dist/tests/contracts/v1-spec-section-6-orchestration.test.js +40 -0
  157. package/dist/tests/contracts/v1-spec-section-7-module-layout.test.js +58 -0
  158. package/dist/tests/contracts/v1-spec-section-8-extension-points.test.js +34 -0
  159. package/dist/tests/contracts/v1-spec-section-9-4-cli-surface.test.js +75 -0
  160. package/dist/tests/contracts/v1-spec-section-9-7-llm-agent-boundary.test.js +36 -0
  161. package/dist/tests/core/write-source.test.js +366 -0
  162. package/dist/tests/curate-command.test.js +87 -0
  163. package/dist/tests/db-scoring.test.js +201 -0
  164. package/dist/tests/db.test.js +654 -0
  165. package/dist/tests/distill-cli-flag.test.js +208 -0
  166. package/dist/tests/distill.test.js +515 -0
  167. package/dist/tests/docker-install.test.js +120 -0
  168. package/dist/tests/e2e.test.js +1419 -0
  169. package/dist/tests/embedder.test.js +340 -0
  170. package/dist/tests/embedding-model-config.test.js +379 -0
  171. package/dist/tests/feedback-command.test.js +172 -0
  172. package/dist/tests/file-context.test.js +552 -0
  173. package/dist/tests/fixtures/scripts/git/summarize-diff.js +9 -0
  174. package/dist/tests/fixtures/scripts/lint/eslint-check.js +7 -0
  175. package/dist/tests/fixtures/stashes/load.js +166 -0
  176. package/dist/tests/fixtures/stashes/load.test.js +97 -0
  177. package/dist/tests/fixtures/stashes/ranking-baseline/scripts/mem0-search.js +12 -0
  178. package/dist/tests/frontmatter.test.js +190 -0
  179. package/dist/tests/fts-field-weighting.test.js +254 -0
  180. package/dist/tests/fuzzy-search.test.js +230 -0
  181. package/dist/tests/git-provider-clone.test.js +45 -0
  182. package/dist/tests/github.test.js +161 -0
  183. package/dist/tests/graph-boost-ranking.test.js +305 -0
  184. package/dist/tests/graph-extraction.test.js +282 -0
  185. package/dist/tests/helpers/usage-events.js +8 -0
  186. package/dist/tests/index-pass-llm.test.js +161 -0
  187. package/dist/tests/indexer.test.js +570 -0
  188. package/dist/tests/info-command.test.js +166 -0
  189. package/dist/tests/init.test.js +69 -0
  190. package/dist/tests/install-script.test.js +246 -0
  191. package/dist/tests/integration/agent-real-profile.test.js +94 -0
  192. package/dist/tests/issue-36-repro.test.js +304 -0
  193. package/dist/tests/issues-191-194.test.js +160 -0
  194. package/dist/tests/lesson-lint.test.js +111 -0
  195. package/dist/tests/llm-client.test.js +115 -0
  196. package/dist/tests/llm-feature-gate.test.js +151 -0
  197. package/dist/tests/llm.test.js +139 -0
  198. package/dist/tests/lockfile.test.js +216 -0
  199. package/dist/tests/manifest.test.js +205 -0
  200. package/dist/tests/markdown.test.js +126 -0
  201. package/dist/tests/matchers-unit.test.js +189 -0
  202. package/dist/tests/memory-inference.test.js +299 -0
  203. package/dist/tests/merge-scoring.test.js +136 -0
  204. package/dist/tests/metadata.test.js +313 -0
  205. package/dist/tests/migration-help.test.js +89 -0
  206. package/dist/tests/origin-resolve.test.js +124 -0
  207. package/dist/tests/output-baseline.test.js +218 -0
  208. package/dist/tests/output-shapes-unit.test.js +478 -0
  209. package/dist/tests/parallel-search.test.js +272 -0
  210. package/dist/tests/parameter-metadata.test.js +365 -0
  211. package/dist/tests/paths.test.js +177 -0
  212. package/dist/tests/progressive-disclosure.test.js +280 -0
  213. package/dist/tests/proposals.test.js +279 -0
  214. package/dist/tests/proposed-quality.test.js +271 -0
  215. package/dist/tests/provider-registry.test.js +32 -0
  216. package/dist/tests/ranking-regression.test.js +548 -0
  217. package/dist/tests/reflect-propose.test.js +455 -0
  218. package/dist/tests/registry-build-index.test.js +394 -0
  219. package/dist/tests/registry-cli.test.js +290 -0
  220. package/dist/tests/registry-index-v2.test.js +430 -0
  221. package/dist/tests/registry-install.test.js +728 -0
  222. package/dist/tests/registry-providers/parity.test.js +189 -0
  223. package/dist/tests/registry-providers/skills-sh.test.js +309 -0
  224. package/dist/tests/registry-providers/static-index.test.js +238 -0
  225. package/dist/tests/registry-resolve.test.js +126 -0
  226. package/dist/tests/registry-search.test.js +923 -0
  227. package/dist/tests/remember-frontmatter.test.js +378 -0
  228. package/dist/tests/remember-unit.test.js +123 -0
  229. package/dist/tests/ripgrep-install.test.js +251 -0
  230. package/dist/tests/ripgrep-resolve.test.js +108 -0
  231. package/dist/tests/ripgrep.test.js +163 -0
  232. package/dist/tests/save-command.test.js +94 -0
  233. package/dist/tests/save-trust-qa-fixes.test.js +270 -0
  234. package/dist/tests/scoring-pipeline.test.js +648 -0
  235. package/dist/tests/search-include-proposed-cli.test.js +118 -0
  236. package/dist/tests/self-update.test.js +442 -0
  237. package/dist/tests/semantic-search-e2e.test.js +512 -0
  238. package/dist/tests/semantic-status.test.js +471 -0
  239. package/dist/tests/setup-run.integration.js +877 -0
  240. package/dist/tests/setup-wizard.test.js +198 -0
  241. package/dist/tests/setup.test.js +131 -0
  242. package/dist/tests/source-add.test.js +11 -0
  243. package/dist/tests/source-clone.test.js +254 -0
  244. package/dist/tests/source-manage.test.js +366 -0
  245. package/dist/tests/source-providers/filesystem.test.js +82 -0
  246. package/dist/tests/source-providers/git.test.js +252 -0
  247. package/dist/tests/source-providers/website.test.js +128 -0
  248. package/dist/tests/source-qa-fixes.test.js +286 -0
  249. package/dist/tests/source-registry.test.js +350 -0
  250. package/dist/tests/source-resolve.test.js +100 -0
  251. package/dist/tests/source-source.test.js +281 -0
  252. package/dist/tests/source.test.js +533 -0
  253. package/dist/tests/tar-utils-scan.test.js +73 -0
  254. package/dist/tests/toggle-components.test.js +73 -0
  255. package/dist/tests/usage-telemetry.test.js +265 -0
  256. package/dist/tests/utility-scoring.test.js +558 -0
  257. package/dist/tests/vault-load-error.test.js +78 -0
  258. package/dist/tests/vault-qa-fixes.test.js +194 -0
  259. package/dist/tests/vault.test.js +429 -0
  260. package/dist/tests/vector-search.test.js +608 -0
  261. package/dist/tests/walker.test.js +252 -0
  262. package/dist/tests/wave2-cluster-bc.test.js +228 -0
  263. package/dist/tests/wave2-cluster-d.test.js +180 -0
  264. package/dist/tests/wave2-cluster-e.test.js +179 -0
  265. package/dist/tests/wiki-qa-fixes.test.js +270 -0
  266. package/dist/tests/wiki.test.js +529 -0
  267. package/dist/tests/workflow-cli.test.js +271 -0
  268. package/dist/tests/workflow-markdown.test.js +171 -0
  269. package/dist/tests/workflow-path-escape.test.js +132 -0
  270. package/dist/tests/workflow-qa-fixes.test.js +395 -0
  271. package/dist/tests/workflows/indexer-rejection.test.js +213 -0
  272. package/docs/README.md +8 -0
  273. package/docs/migration/release-notes/0.7.0.md +244 -0
  274. package/package.json +2 -2
  275. package/dist/core/warn.js +0 -27
  276. package/dist/output/shapes.js +0 -212
  277. package/dist/output/text.js +0 -520
  278. /package/dist/{commands → src/commands}/completions.js +0 -0
  279. /package/dist/{commands → src/commands}/curate.js +0 -0
  280. /package/dist/{commands → src/commands}/info.js +0 -0
  281. /package/dist/{commands → src/commands}/init.js +0 -0
  282. /package/dist/{commands → src/commands}/install-audit.js +0 -0
  283. /package/dist/{commands → src/commands}/migration-help.js +0 -0
  284. /package/dist/{commands → src/commands}/source-clone.js +0 -0
  285. /package/dist/{commands → src/commands}/vault.js +0 -0
  286. /package/dist/{core → src/core}/asset-registry.js +0 -0
  287. /package/dist/{core → src/core}/frontmatter.js +0 -0
  288. /package/dist/{core → src/core}/markdown.js +0 -0
  289. /package/dist/{core → src/core}/paths.js +0 -0
  290. /package/dist/{indexer → src/indexer}/manifest.js +0 -0
  291. /package/dist/{indexer → src/indexer}/search-fields.js +0 -0
  292. /package/dist/{indexer → src/indexer}/semantic-status.js +0 -0
  293. /package/dist/{indexer → src/indexer}/usage-events.js +0 -0
  294. /package/dist/{indexer → src/indexer}/walker.js +0 -0
  295. /package/dist/{llm → src/llm}/embedder.js +0 -0
  296. /package/dist/{llm → src/llm}/embedders/cache.js +0 -0
  297. /package/dist/{llm → src/llm}/embedders/local.js +0 -0
  298. /package/dist/{llm → src/llm}/embedders/types.js +0 -0
  299. /package/dist/{llm → src/llm}/metadata-enhance.js +0 -0
  300. /package/dist/{output → src/output}/context.js +0 -0
  301. /package/dist/{registry → src/registry}/create-provider-registry.js +0 -0
  302. /package/dist/{registry → src/registry}/origin-resolve.js +0 -0
  303. /package/dist/{registry → src/registry}/providers/index.js +0 -0
  304. /package/dist/{registry → src/registry}/providers/skills-sh.js +0 -0
  305. /package/dist/{registry → src/registry}/providers/types.js +0 -0
  306. /package/dist/{registry → src/registry}/types.js +0 -0
  307. /package/dist/{setup → src/setup}/detect.js +0 -0
  308. /package/dist/{setup → src/setup}/ripgrep-install.js +0 -0
  309. /package/dist/{setup → src/setup}/ripgrep-resolve.js +0 -0
  310. /package/dist/{setup → src/setup}/steps.js +0 -0
  311. /package/dist/{sources → src/sources}/include.js +0 -0
  312. /package/dist/{sources → src/sources}/provider-factory.js +0 -0
  313. /package/dist/{sources → src/sources}/provider.js +0 -0
  314. /package/dist/{sources → src/sources}/providers/filesystem.js +0 -0
  315. /package/dist/{sources → src/sources}/providers/index.js +0 -0
  316. /package/dist/{sources → src/sources}/providers/install-types.js +0 -0
  317. /package/dist/{sources → src/sources}/providers/npm.js +0 -0
  318. /package/dist/{sources → src/sources}/providers/provider-utils.js +0 -0
  319. /package/dist/{sources → src/sources}/providers/sync-from-ref.js +0 -0
  320. /package/dist/{sources → src/sources}/providers/tar-utils.js +0 -0
  321. /package/dist/{sources → src/sources}/providers/website.js +0 -0
  322. /package/dist/{sources → src/sources}/resolve.js +0 -0
  323. /package/dist/{sources → src/sources}/types.js +0 -0
  324. /package/dist/{templates → src/templates}/wiki-templates.js +0 -0
  325. /package/dist/{version.js → src/version.js} +0 -0
  326. /package/dist/{workflows → src/workflows}/authoring.js +0 -0
  327. /package/dist/{workflows → src/workflows}/cli.js +0 -0
  328. /package/dist/{workflows → src/workflows}/db.js +0 -0
  329. /package/dist/{workflows → src/workflows}/document-cache.js +0 -0
  330. /package/dist/{workflows → src/workflows}/parser.js +0 -0
  331. /package/dist/{workflows → src/workflows}/renderer.js +0 -0
  332. /package/dist/{workflows → src/workflows}/schema.js +0 -0
  333. /package/dist/{workflows → src/workflows}/validator.js +0 -0
@@ -0,0 +1,457 @@
1
+ /**
2
+ * Unit tests for §6.8 feedback-signal integrity (#244).
3
+ *
4
+ * Coverage:
5
+ * • All four 2×2 quadrants (TP, FP, TN, FN).
6
+ * • Per-asset breakdown when an asset has mixed signals across runs.
7
+ * • `feedback_agreement < 0.80` triggers the warning marker (markdown +
8
+ * structured `warnings[]` JSON entry).
9
+ * • `feedback_coverage` correctly counts runs with feedback dispatched
10
+ * vs total Phase 1 runs.
11
+ * • NaN-safety: zero-feedback asset emits all rates as `null`, never
12
+ * `0` or `NaN`.
13
+ * • Attribution rule (§6.8): a feedback event is attributed to the run
14
+ * that produced it, not to a later run touching the same asset.
15
+ *
16
+ * The metric is a pure function over RunResult[] + feedbackLog[]; no spawn
17
+ * fakes are needed. We build small synthetic streams directly.
18
+ */
19
+ import { describe, expect, test } from "bun:test";
20
+ import { computeFeedbackIntegrity } from "./metrics";
21
+ import { FEEDBACK_AGREEMENT_WARNING_THRESHOLD, renderEvolveReport, renderFeedbackIntegrityTable } from "./report";
22
+ function fakeRun(overrides) {
23
+ return {
24
+ schemaVersion: 1,
25
+ taskId: "t",
26
+ arm: "akm",
27
+ seed: 0,
28
+ model: "m",
29
+ outcome: "pass",
30
+ tokens: { input: 0, output: 0 },
31
+ wallclockMs: 0,
32
+ trajectory: { correctAssetLoaded: null, feedbackRecorded: null },
33
+ events: [],
34
+ verifierStdout: "",
35
+ verifierExitCode: 0,
36
+ assetsLoaded: [],
37
+ ...overrides,
38
+ };
39
+ }
40
+ function fb(overrides) {
41
+ return {
42
+ taskId: "t",
43
+ seed: 0,
44
+ goldRef: "skill:s",
45
+ signal: "positive",
46
+ ok: true,
47
+ ...overrides,
48
+ };
49
+ }
50
+ describe("computeFeedbackIntegrity — 2x2 quadrants", () => {
51
+ test("TP: feedback + on a passed run", () => {
52
+ const phase1 = { akmRuns: [fakeRun({ taskId: "t1", seed: 0, outcome: "pass" })] };
53
+ const feedbackLog = [fb({ taskId: "t1", seed: 0, goldRef: "skill:a", signal: "positive" })];
54
+ const m = computeFeedbackIntegrity({ phase1, feedbackLog });
55
+ expect(m.aggregate.truePositive).toBe(1);
56
+ expect(m.aggregate.falsePositive).toBe(0);
57
+ expect(m.aggregate.trueNegative).toBe(0);
58
+ expect(m.aggregate.falseNegative).toBe(0);
59
+ expect(m.aggregate.feedback_agreement).toBeCloseTo(1);
60
+ expect(m.aggregate.feedback_coverage).toBeCloseTo(1);
61
+ expect(m.perAsset).toHaveLength(1);
62
+ expect(m.perAsset[0].ref).toBe("skill:a");
63
+ expect(m.perAsset[0].truePositive).toBe(1);
64
+ expect(m.perAsset[0].feedback_agreement).toBeCloseTo(1);
65
+ });
66
+ test("FP: feedback + on a failed run", () => {
67
+ const phase1 = { akmRuns: [fakeRun({ taskId: "t1", seed: 0, outcome: "fail" })] };
68
+ const feedbackLog = [fb({ taskId: "t1", seed: 0, goldRef: "skill:a", signal: "positive" })];
69
+ const m = computeFeedbackIntegrity({ phase1, feedbackLog });
70
+ expect(m.aggregate.truePositive).toBe(0);
71
+ expect(m.aggregate.falsePositive).toBe(1);
72
+ expect(m.aggregate.trueNegative).toBe(0);
73
+ expect(m.aggregate.falseNegative).toBe(0);
74
+ expect(m.aggregate.feedback_agreement).toBeCloseTo(0);
75
+ expect(m.aggregate.false_positive_rate).toBeCloseTo(1);
76
+ expect(m.perAsset[0].falsePositive).toBe(1);
77
+ });
78
+ test("TN: feedback - on a failed run", () => {
79
+ const phase1 = { akmRuns: [fakeRun({ taskId: "t1", seed: 0, outcome: "fail" })] };
80
+ const feedbackLog = [fb({ taskId: "t1", seed: 0, goldRef: "skill:a", signal: "negative" })];
81
+ const m = computeFeedbackIntegrity({ phase1, feedbackLog });
82
+ expect(m.aggregate.trueNegative).toBe(1);
83
+ expect(m.aggregate.feedback_agreement).toBeCloseTo(1);
84
+ expect(m.aggregate.false_positive_rate).toBeCloseTo(0);
85
+ expect(m.perAsset[0].trueNegative).toBe(1);
86
+ });
87
+ test("FN: feedback - on a passed run", () => {
88
+ const phase1 = { akmRuns: [fakeRun({ taskId: "t1", seed: 0, outcome: "pass" })] };
89
+ const feedbackLog = [fb({ taskId: "t1", seed: 0, goldRef: "skill:a", signal: "negative" })];
90
+ const m = computeFeedbackIntegrity({ phase1, feedbackLog });
91
+ expect(m.aggregate.falseNegative).toBe(1);
92
+ expect(m.aggregate.feedback_agreement).toBeCloseTo(0);
93
+ expect(m.aggregate.false_negative_rate).toBeCloseTo(1);
94
+ expect(m.perAsset[0].falseNegative).toBe(1);
95
+ });
96
+ });
97
+ describe("computeFeedbackIntegrity — aggregate over mixed quadrants", () => {
98
+ test("computes feedback_agreement and rates correctly across mixed runs", () => {
99
+ // 4 runs covering all four quadrants — exactly one of each.
100
+ const phase1 = {
101
+ akmRuns: [
102
+ fakeRun({ taskId: "tp", seed: 0, outcome: "pass" }),
103
+ fakeRun({ taskId: "fp", seed: 0, outcome: "fail" }),
104
+ fakeRun({ taskId: "tn", seed: 0, outcome: "fail" }),
105
+ fakeRun({ taskId: "fn", seed: 0, outcome: "pass" }),
106
+ ],
107
+ };
108
+ const feedbackLog = [
109
+ fb({ taskId: "tp", seed: 0, goldRef: "skill:tp", signal: "positive" }),
110
+ fb({ taskId: "fp", seed: 0, goldRef: "skill:fp", signal: "positive" }),
111
+ fb({ taskId: "tn", seed: 0, goldRef: "skill:tn", signal: "negative" }),
112
+ fb({ taskId: "fn", seed: 0, goldRef: "skill:fn", signal: "negative" }),
113
+ ];
114
+ const m = computeFeedbackIntegrity({ phase1, feedbackLog });
115
+ expect(m.aggregate.truePositive).toBe(1);
116
+ expect(m.aggregate.falsePositive).toBe(1);
117
+ expect(m.aggregate.trueNegative).toBe(1);
118
+ expect(m.aggregate.falseNegative).toBe(1);
119
+ expect(m.aggregate.feedback_agreement).toBeCloseTo(0.5); // 2/4
120
+ expect(m.aggregate.false_positive_rate).toBeCloseTo(0.5); // 1 / (1+1)
121
+ expect(m.aggregate.false_negative_rate).toBeCloseTo(0.5); // 1 / (1+1)
122
+ expect(m.aggregate.feedback_coverage).toBeCloseTo(1);
123
+ expect(m.perAsset).toHaveLength(4);
124
+ // Per-asset rows should be sorted by ref
125
+ expect(m.perAsset.map((r) => r.ref)).toEqual(["skill:fn", "skill:fp", "skill:tn", "skill:tp"]);
126
+ });
127
+ });
128
+ describe("computeFeedbackIntegrity — per-asset mixed signals", () => {
129
+ test("aggregates correctly when one asset appears across multiple Phase 1 runs", () => {
130
+ // skill:shared has 2 TP, 1 FP, 1 TN, 1 FN across 5 runs.
131
+ const phase1 = {
132
+ akmRuns: [
133
+ fakeRun({ taskId: "t", seed: 0, outcome: "pass" }),
134
+ fakeRun({ taskId: "t", seed: 1, outcome: "pass" }),
135
+ fakeRun({ taskId: "t", seed: 2, outcome: "fail" }),
136
+ fakeRun({ taskId: "t", seed: 3, outcome: "fail" }),
137
+ fakeRun({ taskId: "t", seed: 4, outcome: "pass" }),
138
+ ],
139
+ };
140
+ const feedbackLog = [
141
+ fb({ taskId: "t", seed: 0, goldRef: "skill:shared", signal: "positive" }), // TP
142
+ fb({ taskId: "t", seed: 1, goldRef: "skill:shared", signal: "positive" }), // TP
143
+ fb({ taskId: "t", seed: 2, goldRef: "skill:shared", signal: "positive" }), // FP
144
+ fb({ taskId: "t", seed: 3, goldRef: "skill:shared", signal: "negative" }), // TN
145
+ fb({ taskId: "t", seed: 4, goldRef: "skill:shared", signal: "negative" }), // FN
146
+ ];
147
+ const m = computeFeedbackIntegrity({ phase1, feedbackLog });
148
+ expect(m.perAsset).toHaveLength(1);
149
+ const row = m.perAsset[0];
150
+ expect(row.ref).toBe("skill:shared");
151
+ expect(row.truePositive).toBe(2);
152
+ expect(row.falsePositive).toBe(1);
153
+ expect(row.trueNegative).toBe(1);
154
+ expect(row.falseNegative).toBe(1);
155
+ expect(row.feedback_agreement).toBeCloseTo(3 / 5);
156
+ expect(row.false_positive_rate).toBeCloseTo(1 / 2); // FP / (FP+TN) = 1/2
157
+ expect(row.false_negative_rate).toBeCloseTo(1 / 3); // FN / (FN+TP) = 1/3
158
+ });
159
+ });
160
+ describe("computeFeedbackIntegrity — attribution rule", () => {
161
+ test("attributes feedback to the run that produced it, not a later run touching the same asset", () => {
162
+ // skill:contested appears across two Phase 1 runs:
163
+ // run #0: passed, feedback + → TP
164
+ // run #1: failed, feedback + → FP
165
+ // The naive (wrong) implementation would conflate both events with
166
+ // run #1's outcome and label both as FP. The correct implementation
167
+ // joins each event to its own (taskId, seed) → gets one TP, one FP.
168
+ const phase1 = {
169
+ akmRuns: [fakeRun({ taskId: "t", seed: 0, outcome: "pass" }), fakeRun({ taskId: "t", seed: 1, outcome: "fail" })],
170
+ };
171
+ const feedbackLog = [
172
+ fb({ taskId: "t", seed: 0, goldRef: "skill:contested", signal: "positive" }),
173
+ fb({ taskId: "t", seed: 1, goldRef: "skill:contested", signal: "positive" }),
174
+ ];
175
+ const m = computeFeedbackIntegrity({ phase1, feedbackLog });
176
+ expect(m.aggregate.truePositive).toBe(1);
177
+ expect(m.aggregate.falsePositive).toBe(1);
178
+ expect(m.aggregate.trueNegative).toBe(0);
179
+ expect(m.aggregate.falseNegative).toBe(0);
180
+ expect(m.perAsset[0].truePositive).toBe(1);
181
+ expect(m.perAsset[0].falsePositive).toBe(1);
182
+ });
183
+ });
184
+ describe("computeFeedbackIntegrity — feedback_coverage", () => {
185
+ test("counts runs with feedback dispatched vs total Phase 1 runs", () => {
186
+ // 4 phase-1 runs, only 2 had feedback dispatched.
187
+ const phase1 = {
188
+ akmRuns: [
189
+ fakeRun({ taskId: "t", seed: 0, outcome: "pass" }),
190
+ fakeRun({ taskId: "t", seed: 1, outcome: "fail" }),
191
+ fakeRun({ taskId: "t", seed: 2, outcome: "harness_error" }),
192
+ fakeRun({ taskId: "t", seed: 3, outcome: "budget_exceeded" }),
193
+ ],
194
+ };
195
+ const feedbackLog = [
196
+ fb({ taskId: "t", seed: 0, goldRef: "skill:a", signal: "positive" }),
197
+ fb({ taskId: "t", seed: 1, goldRef: "skill:a", signal: "negative" }),
198
+ ];
199
+ const m = computeFeedbackIntegrity({ phase1, feedbackLog });
200
+ expect(m.aggregate.feedback_coverage).toBeCloseTo(0.5); // 2 of 4
201
+ });
202
+ test("zero coverage when no feedback dispatched", () => {
203
+ const phase1 = { akmRuns: [fakeRun({ taskId: "t", seed: 0, outcome: "pass" })] };
204
+ const m = computeFeedbackIntegrity({ phase1, feedbackLog: [] });
205
+ expect(m.aggregate.feedback_coverage).toBe(0);
206
+ expect(m.aggregate.feedback_agreement).toBe(0);
207
+ expect(m.perAsset).toEqual([]);
208
+ });
209
+ test("zero coverage and zero runs returns 0 (not NaN)", () => {
210
+ const m = computeFeedbackIntegrity({ phase1: { akmRuns: [] }, feedbackLog: [] });
211
+ expect(m.aggregate.feedback_coverage).toBe(0);
212
+ expect(m.aggregate.feedback_agreement).toBe(0);
213
+ expect(m.aggregate.false_positive_rate).toBe(0);
214
+ expect(m.aggregate.false_negative_rate).toBe(0);
215
+ expect(Number.isFinite(m.aggregate.feedback_coverage)).toBe(true);
216
+ expect(Number.isFinite(m.aggregate.feedback_agreement)).toBe(true);
217
+ });
218
+ });
219
+ describe("computeFeedbackIntegrity — NaN safety", () => {
220
+ test("per-asset row with FP+TN === 0 emits null false_positive_rate (only positive feedback on passes)", () => {
221
+ const phase1 = {
222
+ akmRuns: [fakeRun({ taskId: "t", seed: 0, outcome: "pass" }), fakeRun({ taskId: "t", seed: 1, outcome: "pass" })],
223
+ };
224
+ const feedbackLog = [
225
+ fb({ taskId: "t", seed: 0, goldRef: "skill:only-tp", signal: "positive" }),
226
+ fb({ taskId: "t", seed: 1, goldRef: "skill:only-tp", signal: "positive" }),
227
+ ];
228
+ const m = computeFeedbackIntegrity({ phase1, feedbackLog });
229
+ const row = m.perAsset[0];
230
+ expect(row.feedback_agreement).toBeCloseTo(1);
231
+ expect(row.false_positive_rate).toBeNull(); // FP+TN === 0
232
+ expect(row.false_negative_rate).toBeCloseTo(0); // FN/(FN+TP) = 0/2 = 0
233
+ });
234
+ test("per-asset row with FN+TP === 0 emits null false_negative_rate (only negative feedback on fails)", () => {
235
+ const phase1 = {
236
+ akmRuns: [fakeRun({ taskId: "t", seed: 0, outcome: "fail" }), fakeRun({ taskId: "t", seed: 1, outcome: "fail" })],
237
+ };
238
+ const feedbackLog = [
239
+ fb({ taskId: "t", seed: 0, goldRef: "skill:only-tn", signal: "negative" }),
240
+ fb({ taskId: "t", seed: 1, goldRef: "skill:only-tn", signal: "negative" }),
241
+ ];
242
+ const m = computeFeedbackIntegrity({ phase1, feedbackLog });
243
+ const row = m.perAsset[0];
244
+ expect(row.feedback_agreement).toBeCloseTo(1);
245
+ expect(row.false_negative_rate).toBeNull(); // FN+TP === 0
246
+ expect(row.false_positive_rate).toBeCloseTo(0); // FP/(FP+TN) = 0/2 = 0
247
+ });
248
+ test("ok=false feedback events are excluded from the matrix but still count toward coverage", () => {
249
+ const phase1 = {
250
+ akmRuns: [fakeRun({ taskId: "t", seed: 0, outcome: "pass" }), fakeRun({ taskId: "t", seed: 1, outcome: "fail" })],
251
+ };
252
+ const feedbackLog = [
253
+ fb({ taskId: "t", seed: 0, goldRef: "skill:a", signal: "positive", ok: true }),
254
+ fb({ taskId: "t", seed: 1, goldRef: "skill:a", signal: "negative", ok: false }),
255
+ ];
256
+ const m = computeFeedbackIntegrity({ phase1, feedbackLog });
257
+ // Only the ok=true entry contributes to the matrix (TP=1).
258
+ expect(m.aggregate.truePositive).toBe(1);
259
+ expect(m.aggregate.trueNegative).toBe(0);
260
+ // But coverage counts both attempts.
261
+ expect(m.aggregate.feedback_coverage).toBeCloseTo(1);
262
+ });
263
+ test("harness_error runs are excluded from the matrix even with a stamped feedback event", () => {
264
+ const phase1 = { akmRuns: [fakeRun({ taskId: "t", seed: 0, outcome: "harness_error" })] };
265
+ const feedbackLog = [fb({ taskId: "t", seed: 0, goldRef: "skill:a", signal: "positive" })];
266
+ const m = computeFeedbackIntegrity({ phase1, feedbackLog });
267
+ expect(m.aggregate.truePositive).toBe(0);
268
+ expect(m.aggregate.falsePositive).toBe(0);
269
+ expect(m.perAsset).toEqual([]);
270
+ });
271
+ test("feedback for a run not present in akmRuns is silently dropped", () => {
272
+ const phase1 = { akmRuns: [fakeRun({ taskId: "real", seed: 0, outcome: "pass" })] };
273
+ const feedbackLog = [fb({ taskId: "ghost", seed: 99, goldRef: "skill:a", signal: "positive" })];
274
+ const m = computeFeedbackIntegrity({ phase1, feedbackLog });
275
+ expect(m.aggregate.truePositive).toBe(0);
276
+ expect(m.perAsset).toEqual([]);
277
+ // Coverage still records the dispatch attempt — operator wanted feedback.
278
+ expect(m.aggregate.feedback_coverage).toBeCloseTo(1);
279
+ });
280
+ });
281
+ // ── Render-side coverage ───────────────────────────────────────────────────
282
+ function emptyUtilityReport() {
283
+ // Build a minimal §13.3-shaped utility report. The renderer reads
284
+ // many subfields; we stub them to safe zeros.
285
+ return {
286
+ timestamp: "2026-04-27T00:00:00Z",
287
+ branch: "test",
288
+ commit: "deadbee",
289
+ model: "m",
290
+ corpus: { domains: 0, tasks: 0, slice: "all", seedsPerArm: 1 },
291
+ aggregateNoakm: { passRate: 0, tokensPerPass: 0, tokensPerRun: null, wallclockMs: 0 },
292
+ aggregateAkm: { passRate: 0, tokensPerPass: 0, tokensPerRun: null, wallclockMs: 0 },
293
+ aggregateDelta: {
294
+ passRate: 0,
295
+ tokensPerPass: 0,
296
+ tokensPerRun: null,
297
+ wallclockMs: 0,
298
+ },
299
+ trajectoryAkm: {
300
+ correctAssetLoaded: null,
301
+ feedbackRecorded: 0,
302
+ },
303
+ failureModes: { byLabel: {}, byTask: {} },
304
+ tasks: [],
305
+ warnings: [],
306
+ akmRuns: [],
307
+ taskMetadata: [],
308
+ goldRankRecords: [],
309
+ };
310
+ }
311
+ function evolveInputWith(metrics) {
312
+ return {
313
+ timestamp: "2026-04-27T00:00:00Z",
314
+ branch: "test",
315
+ commit: "deadbee",
316
+ model: "m",
317
+ domain: "test",
318
+ seedsPerArm: 1,
319
+ proposals: { rows: [], totalProposals: 0, totalAccepted: 0, acceptanceRate: 0, lintPassRate: 0 },
320
+ longitudinal: {
321
+ improvementSlope: 0.1,
322
+ overSyntheticLift: 0.05,
323
+ degradationCount: 0,
324
+ degradations: [],
325
+ prePassRate: 0.5,
326
+ postPassRate: 0.6,
327
+ syntheticPassRate: 0.55,
328
+ },
329
+ arms: { pre: emptyUtilityReport(), post: emptyUtilityReport(), synthetic: emptyUtilityReport() },
330
+ warnings: [],
331
+ ...(metrics ? { feedbackIntegrity: metrics } : {}),
332
+ };
333
+ }
334
+ describe("renderFeedbackIntegrityTable", () => {
335
+ test("emits aggregate matrix + per-asset rows", () => {
336
+ const phase1 = {
337
+ akmRuns: [fakeRun({ taskId: "t", seed: 0, outcome: "pass" }), fakeRun({ taskId: "t", seed: 1, outcome: "fail" })],
338
+ };
339
+ const feedbackLog = [
340
+ fb({ taskId: "t", seed: 0, goldRef: "skill:a", signal: "positive" }),
341
+ fb({ taskId: "t", seed: 1, goldRef: "skill:a", signal: "negative" }),
342
+ ];
343
+ const m = computeFeedbackIntegrity({ phase1, feedbackLog });
344
+ const md = renderFeedbackIntegrityTable(m);
345
+ expect(md).toContain("Feedback-signal integrity");
346
+ expect(md).toContain("feedback_agreement | 1.00");
347
+ expect(md).toContain("feedback_coverage | 1.00");
348
+ expect(md).toContain("`skill:a`");
349
+ });
350
+ test("renders n/a when a per-asset rate is null", () => {
351
+ const phase1 = { akmRuns: [fakeRun({ taskId: "t", seed: 0, outcome: "pass" })] };
352
+ const feedbackLog = [fb({ taskId: "t", seed: 0, goldRef: "skill:a", signal: "positive" })];
353
+ const m = computeFeedbackIntegrity({ phase1, feedbackLog });
354
+ const md = renderFeedbackIntegrityTable(m);
355
+ // Only TP — false_positive_rate denom is 0 → null → "n/a".
356
+ expect(md).toContain("n/a");
357
+ });
358
+ test("renders 'No feedback events recorded' when perAsset is empty", () => {
359
+ const m = {
360
+ aggregate: {
361
+ truePositive: 0,
362
+ falsePositive: 0,
363
+ trueNegative: 0,
364
+ falseNegative: 0,
365
+ feedback_agreement: 0,
366
+ false_positive_rate: 0,
367
+ false_negative_rate: 0,
368
+ feedback_coverage: 0,
369
+ },
370
+ perAsset: [],
371
+ };
372
+ expect(renderFeedbackIntegrityTable(m)).toContain("No feedback events recorded");
373
+ });
374
+ });
375
+ describe("renderEvolveReport — feedback_agreement headline + warning marker", () => {
376
+ test("places real feedback_agreement after improvement_slope when metrics provided", () => {
377
+ const metrics = computeFeedbackIntegrity({
378
+ phase1: { akmRuns: [fakeRun({ taskId: "t", seed: 0, outcome: "pass" })] },
379
+ feedbackLog: [fb({ taskId: "t", seed: 0, goldRef: "skill:a", signal: "positive" })],
380
+ });
381
+ const { markdown, json } = renderEvolveReport(evolveInputWith(metrics));
382
+ // feedback_agreement is on a line directly after improvement_slope.
383
+ const slopeIdx = markdown.indexOf("improvement_slope:");
384
+ const agreementIdx = markdown.indexOf("feedback_agreement:");
385
+ expect(slopeIdx).toBeGreaterThanOrEqual(0);
386
+ expect(agreementIdx).toBeGreaterThan(slopeIdx);
387
+ expect(markdown).toContain("feedback_agreement: 1.00");
388
+ expect(markdown).not.toContain("pending (#244)");
389
+ // JSON envelope carries `feedback_integrity` as a top-level key.
390
+ const parsed = json;
391
+ expect(parsed.feedback_integrity).toBeDefined();
392
+ expect(parsed.warnings.some((w) => w.startsWith("feedback_agreement_below_threshold"))).toBe(false);
393
+ });
394
+ test("placeholder remains when metrics omitted (legacy path)", () => {
395
+ const { markdown, json } = renderEvolveReport(evolveInputWith(undefined));
396
+ expect(markdown).toContain("_feedback_agreement: pending (#244)_");
397
+ const parsed = json;
398
+ expect(parsed.feedback_integrity).toBeUndefined();
399
+ });
400
+ test("agreement < 0.80 prepends warning marker to markdown and structured warnings[]", () => {
401
+ // 1 TP + 4 FP → agreement = 1/5 = 0.20.
402
+ const phase1 = {
403
+ akmRuns: [
404
+ fakeRun({ taskId: "t", seed: 0, outcome: "pass" }),
405
+ fakeRun({ taskId: "t", seed: 1, outcome: "fail" }),
406
+ fakeRun({ taskId: "t", seed: 2, outcome: "fail" }),
407
+ fakeRun({ taskId: "t", seed: 3, outcome: "fail" }),
408
+ fakeRun({ taskId: "t", seed: 4, outcome: "fail" }),
409
+ ],
410
+ };
411
+ const feedbackLog = [
412
+ fb({ taskId: "t", seed: 0, goldRef: "skill:a", signal: "positive" }),
413
+ fb({ taskId: "t", seed: 1, goldRef: "skill:a", signal: "positive" }),
414
+ fb({ taskId: "t", seed: 2, goldRef: "skill:a", signal: "positive" }),
415
+ fb({ taskId: "t", seed: 3, goldRef: "skill:a", signal: "positive" }),
416
+ fb({ taskId: "t", seed: 4, goldRef: "skill:a", signal: "positive" }),
417
+ ];
418
+ const metrics = computeFeedbackIntegrity({ phase1, feedbackLog });
419
+ expect(metrics.aggregate.feedback_agreement).toBeCloseTo(0.2);
420
+ expect(metrics.aggregate.feedback_agreement).toBeLessThan(FEEDBACK_AGREEMENT_WARNING_THRESHOLD);
421
+ const { markdown, json } = renderEvolveReport(evolveInputWith(metrics));
422
+ // Marker appears above the headline, not after it.
423
+ const warnIdx = markdown.indexOf("feedback_agreement = 0.20");
424
+ const slopeIdx = markdown.indexOf("**improvement_slope:");
425
+ expect(warnIdx).toBeGreaterThanOrEqual(0);
426
+ expect(warnIdx).toBeLessThan(slopeIdx);
427
+ expect(markdown).toContain("Track B headline numbers");
428
+ // Structured warning surfaces in the JSON envelope.
429
+ const parsed = json;
430
+ expect(parsed.warnings.some((w) => w.startsWith("feedback_agreement_below_threshold"))).toBe(true);
431
+ });
432
+ test("agreement at exactly 0.80 does NOT trigger the warning marker", () => {
433
+ // 4 TP + 1 FP → agreement = 4/5 = 0.80 exactly.
434
+ const phase1 = {
435
+ akmRuns: [
436
+ fakeRun({ taskId: "t", seed: 0, outcome: "pass" }),
437
+ fakeRun({ taskId: "t", seed: 1, outcome: "pass" }),
438
+ fakeRun({ taskId: "t", seed: 2, outcome: "pass" }),
439
+ fakeRun({ taskId: "t", seed: 3, outcome: "pass" }),
440
+ fakeRun({ taskId: "t", seed: 4, outcome: "fail" }),
441
+ ],
442
+ };
443
+ const feedbackLog = [
444
+ fb({ taskId: "t", seed: 0, goldRef: "skill:a", signal: "positive" }),
445
+ fb({ taskId: "t", seed: 1, goldRef: "skill:a", signal: "positive" }),
446
+ fb({ taskId: "t", seed: 2, goldRef: "skill:a", signal: "positive" }),
447
+ fb({ taskId: "t", seed: 3, goldRef: "skill:a", signal: "positive" }),
448
+ fb({ taskId: "t", seed: 4, goldRef: "skill:a", signal: "positive" }),
449
+ ];
450
+ const metrics = computeFeedbackIntegrity({ phase1, feedbackLog });
451
+ expect(metrics.aggregate.feedback_agreement).toBeCloseTo(0.8);
452
+ const { markdown, json } = renderEvolveReport(evolveInputWith(metrics));
453
+ expect(markdown).not.toContain("Track B headline numbers");
454
+ const parsed = json;
455
+ expect(parsed.warnings.some((w) => w.startsWith("feedback_agreement_below_threshold"))).toBe(false);
456
+ });
457
+ });