@productbrain/cli 0.1.0-beta.9 → 0.1.0-beta.90

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 (643) hide show
  1. package/README.md +77 -92
  2. package/dist/__tests__/adapters.test.d.ts +2 -0
  3. package/dist/__tests__/adapters.test.d.ts.map +1 -0
  4. package/dist/__tests__/adapters.test.js +417 -0
  5. package/dist/__tests__/adapters.test.js.map +1 -0
  6. package/dist/__tests__/audit.test.d.ts +2 -0
  7. package/dist/__tests__/audit.test.d.ts.map +1 -0
  8. package/dist/__tests__/audit.test.js +394 -0
  9. package/dist/__tests__/audit.test.js.map +1 -0
  10. package/dist/__tests__/batch-transformations.test.d.ts +2 -0
  11. package/dist/__tests__/batch-transformations.test.d.ts.map +1 -0
  12. package/dist/__tests__/batch-transformations.test.js +263 -0
  13. package/dist/__tests__/batch-transformations.test.js.map +1 -0
  14. package/dist/__tests__/capture.test.js +309 -18
  15. package/dist/__tests__/capture.test.js.map +1 -1
  16. package/dist/__tests__/config.test.d.ts +8 -0
  17. package/dist/__tests__/config.test.d.ts.map +1 -0
  18. package/dist/__tests__/config.test.js +166 -0
  19. package/dist/__tests__/config.test.js.map +1 -0
  20. package/dist/__tests__/constants.test.d.ts +2 -0
  21. package/dist/__tests__/constants.test.d.ts.map +1 -0
  22. package/dist/__tests__/constants.test.js +141 -0
  23. package/dist/__tests__/constants.test.js.map +1 -0
  24. package/dist/__tests__/constellation.test.js +20 -26
  25. package/dist/__tests__/constellation.test.js.map +1 -1
  26. package/dist/__tests__/context-strategy.test.d.ts +2 -0
  27. package/dist/__tests__/context-strategy.test.d.ts.map +1 -0
  28. package/dist/__tests__/context-strategy.test.js +79 -0
  29. package/dist/__tests__/context-strategy.test.js.map +1 -0
  30. package/dist/__tests__/errors.test.d.ts +2 -0
  31. package/dist/__tests__/errors.test.d.ts.map +1 -0
  32. package/dist/__tests__/errors.test.js +117 -0
  33. package/dist/__tests__/errors.test.js.map +1 -0
  34. package/dist/__tests__/experiment.test.d.ts +6 -0
  35. package/dist/__tests__/experiment.test.d.ts.map +1 -0
  36. package/dist/__tests__/experiment.test.js +69 -0
  37. package/dist/__tests__/experiment.test.js.map +1 -0
  38. package/dist/__tests__/fields.test.js +68 -68
  39. package/dist/__tests__/fields.test.js.map +1 -1
  40. package/dist/__tests__/glossary.test.d.ts +2 -0
  41. package/dist/__tests__/glossary.test.d.ts.map +1 -0
  42. package/dist/__tests__/glossary.test.js +32 -0
  43. package/dist/__tests__/glossary.test.js.map +1 -0
  44. package/dist/__tests__/handshake-preview.test.d.ts +2 -0
  45. package/dist/__tests__/handshake-preview.test.d.ts.map +1 -0
  46. package/dist/__tests__/handshake-preview.test.js +156 -0
  47. package/dist/__tests__/handshake-preview.test.js.map +1 -0
  48. package/dist/__tests__/handshake.test.js +27 -18
  49. package/dist/__tests__/handshake.test.js.map +1 -1
  50. package/dist/__tests__/hook-intents.test.d.ts +2 -0
  51. package/dist/__tests__/hook-intents.test.d.ts.map +1 -0
  52. package/dist/__tests__/hook-intents.test.js +184 -0
  53. package/dist/__tests__/hook-intents.test.js.map +1 -0
  54. package/dist/__tests__/ingest.test.js +110 -12
  55. package/dist/__tests__/ingest.test.js.map +1 -1
  56. package/dist/__tests__/init.test.d.ts +7 -0
  57. package/dist/__tests__/init.test.d.ts.map +1 -0
  58. package/dist/__tests__/init.test.js +146 -0
  59. package/dist/__tests__/init.test.js.map +1 -0
  60. package/dist/__tests__/login.test.d.ts +2 -0
  61. package/dist/__tests__/login.test.d.ts.map +1 -0
  62. package/dist/__tests__/login.test.js +167 -0
  63. package/dist/__tests__/login.test.js.map +1 -0
  64. package/dist/__tests__/manifest.test.d.ts +6 -0
  65. package/dist/__tests__/manifest.test.d.ts.map +1 -0
  66. package/dist/__tests__/manifest.test.js +138 -0
  67. package/dist/__tests__/manifest.test.js.map +1 -0
  68. package/dist/__tests__/method-registry.integration.test.d.ts +6 -0
  69. package/dist/__tests__/method-registry.integration.test.d.ts.map +1 -0
  70. package/dist/__tests__/method-registry.integration.test.js +18 -0
  71. package/dist/__tests__/method-registry.integration.test.js.map +1 -0
  72. package/dist/__tests__/method-registry.test.d.ts +14 -0
  73. package/dist/__tests__/method-registry.test.d.ts.map +1 -0
  74. package/dist/__tests__/method-registry.test.js +134 -0
  75. package/dist/__tests__/method-registry.test.js.map +1 -0
  76. package/dist/__tests__/onboarding-path-b.test.d.ts +2 -0
  77. package/dist/__tests__/onboarding-path-b.test.d.ts.map +1 -0
  78. package/dist/__tests__/onboarding-path-b.test.js +46 -0
  79. package/dist/__tests__/onboarding-path-b.test.js.map +1 -0
  80. package/dist/__tests__/onboarding.test.d.ts +6 -0
  81. package/dist/__tests__/onboarding.test.d.ts.map +1 -0
  82. package/dist/__tests__/onboarding.test.js +347 -0
  83. package/dist/__tests__/onboarding.test.js.map +1 -0
  84. package/dist/__tests__/orient.test.d.ts +2 -0
  85. package/dist/__tests__/orient.test.d.ts.map +1 -0
  86. package/dist/__tests__/orient.test.js +143 -0
  87. package/dist/__tests__/orient.test.js.map +1 -0
  88. package/dist/__tests__/personal-layer.test.d.ts +12 -0
  89. package/dist/__tests__/personal-layer.test.d.ts.map +1 -0
  90. package/dist/__tests__/personal-layer.test.js +304 -0
  91. package/dist/__tests__/personal-layer.test.js.map +1 -0
  92. package/dist/__tests__/profiles.test.d.ts +2 -0
  93. package/dist/__tests__/profiles.test.d.ts.map +1 -0
  94. package/dist/__tests__/profiles.test.js +168 -0
  95. package/dist/__tests__/profiles.test.js.map +1 -0
  96. package/dist/__tests__/promote.test.d.ts +2 -0
  97. package/dist/__tests__/promote.test.d.ts.map +1 -0
  98. package/dist/__tests__/promote.test.js +161 -0
  99. package/dist/__tests__/promote.test.js.map +1 -0
  100. package/dist/__tests__/prompts.test.d.ts +6 -0
  101. package/dist/__tests__/prompts.test.d.ts.map +1 -0
  102. package/dist/__tests__/prompts.test.js +146 -0
  103. package/dist/__tests__/prompts.test.js.map +1 -0
  104. package/dist/__tests__/proposals.test.d.ts +2 -0
  105. package/dist/__tests__/proposals.test.d.ts.map +1 -0
  106. package/dist/__tests__/proposals.test.js +167 -0
  107. package/dist/__tests__/proposals.test.js.map +1 -0
  108. package/dist/__tests__/relate.test.js +57 -33
  109. package/dist/__tests__/relate.test.js.map +1 -1
  110. package/dist/__tests__/repo-detect.test.js +97 -1
  111. package/dist/__tests__/repo-detect.test.js.map +1 -1
  112. package/dist/__tests__/runner.test.js +19 -15
  113. package/dist/__tests__/runner.test.js.map +1 -1
  114. package/dist/__tests__/session-state-machine.test.d.ts +2 -0
  115. package/dist/__tests__/session-state-machine.test.d.ts.map +1 -0
  116. package/dist/__tests__/session-state-machine.test.js +154 -0
  117. package/dist/__tests__/session-state-machine.test.js.map +1 -0
  118. package/dist/__tests__/session-touch.test.js +11 -11
  119. package/dist/__tests__/session-touch.test.js.map +1 -1
  120. package/dist/__tests__/session.test.js +4 -10
  121. package/dist/__tests__/session.test.js.map +1 -1
  122. package/dist/__tests__/setup.test.d.ts +2 -0
  123. package/dist/__tests__/setup.test.d.ts.map +1 -0
  124. package/dist/__tests__/setup.test.js +141 -0
  125. package/dist/__tests__/setup.test.js.map +1 -0
  126. package/dist/__tests__/spinner-labels.test.d.ts +2 -0
  127. package/dist/__tests__/spinner-labels.test.d.ts.map +1 -0
  128. package/dist/__tests__/spinner-labels.test.js +23 -0
  129. package/dist/__tests__/spinner-labels.test.js.map +1 -0
  130. package/dist/__tests__/state.test.d.ts +6 -0
  131. package/dist/__tests__/state.test.d.ts.map +1 -0
  132. package/dist/__tests__/state.test.js +97 -0
  133. package/dist/__tests__/state.test.js.map +1 -0
  134. package/dist/__tests__/strip.test.js +2 -2
  135. package/dist/__tests__/strip.test.js.map +1 -1
  136. package/dist/__tests__/surface-profiles.test.d.ts +2 -0
  137. package/dist/__tests__/surface-profiles.test.d.ts.map +1 -0
  138. package/dist/__tests__/surface-profiles.test.js +233 -0
  139. package/dist/__tests__/surface-profiles.test.js.map +1 -0
  140. package/dist/__tests__/surfaces.test.d.ts +2 -0
  141. package/dist/__tests__/surfaces.test.d.ts.map +1 -0
  142. package/dist/__tests__/surfaces.test.js +46 -0
  143. package/dist/__tests__/surfaces.test.js.map +1 -0
  144. package/dist/__tests__/update.test.js +123 -43
  145. package/dist/__tests__/update.test.js.map +1 -1
  146. package/dist/__tests__/workspace.test.d.ts +2 -0
  147. package/dist/__tests__/workspace.test.d.ts.map +1 -0
  148. package/dist/__tests__/workspace.test.js +308 -0
  149. package/dist/__tests__/workspace.test.js.map +1 -0
  150. package/dist/commands/accept.d.ts +18 -0
  151. package/dist/commands/accept.d.ts.map +1 -0
  152. package/dist/commands/accept.js +76 -0
  153. package/dist/commands/accept.js.map +1 -0
  154. package/dist/commands/admin/cockpit.d.ts +90 -0
  155. package/dist/commands/admin/cockpit.d.ts.map +1 -0
  156. package/dist/commands/admin/cockpit.js +618 -0
  157. package/dist/commands/admin/cockpit.js.map +1 -0
  158. package/dist/commands/admin/index.d.ts +21 -0
  159. package/dist/commands/admin/index.d.ts.map +1 -0
  160. package/dist/commands/admin/index.js +256 -0
  161. package/dist/commands/admin/index.js.map +1 -0
  162. package/dist/commands/admin/inspect.d.ts +30 -0
  163. package/dist/commands/admin/inspect.d.ts.map +1 -0
  164. package/dist/commands/admin/inspect.js +555 -0
  165. package/dist/commands/admin/inspect.js.map +1 -0
  166. package/dist/commands/admin/inspect.test.d.ts +7 -0
  167. package/dist/commands/admin/inspect.test.d.ts.map +1 -0
  168. package/dist/commands/admin/inspect.test.js +90 -0
  169. package/dist/commands/admin/inspect.test.js.map +1 -0
  170. package/dist/commands/admin/manage.d.ts +8 -0
  171. package/dist/commands/admin/manage.d.ts.map +1 -0
  172. package/dist/commands/admin/manage.js +76 -0
  173. package/dist/commands/admin/manage.js.map +1 -0
  174. package/dist/commands/admin/seed.d.ts +32 -0
  175. package/dist/commands/admin/seed.d.ts.map +1 -0
  176. package/dist/commands/admin/seed.js +533 -0
  177. package/dist/commands/admin/seed.js.map +1 -0
  178. package/dist/commands/admin/seed.test.d.ts +6 -0
  179. package/dist/commands/admin/seed.test.d.ts.map +1 -0
  180. package/dist/commands/admin/seed.test.js +65 -0
  181. package/dist/commands/admin/seed.test.js.map +1 -0
  182. package/dist/commands/audit.d.ts +25 -0
  183. package/dist/commands/audit.d.ts.map +1 -0
  184. package/dist/commands/audit.js +188 -0
  185. package/dist/commands/audit.js.map +1 -0
  186. package/dist/commands/brand-pack.d.ts +2 -0
  187. package/dist/commands/brand-pack.d.ts.map +1 -0
  188. package/dist/commands/brand-pack.js +25 -0
  189. package/dist/commands/brand-pack.js.map +1 -0
  190. package/dist/commands/brief.d.ts +28 -0
  191. package/dist/commands/brief.d.ts.map +1 -0
  192. package/dist/commands/brief.js +75 -0
  193. package/dist/commands/brief.js.map +1 -0
  194. package/dist/commands/capture.d.ts +9 -0
  195. package/dist/commands/capture.d.ts.map +1 -1
  196. package/dist/commands/capture.js +256 -17
  197. package/dist/commands/capture.js.map +1 -1
  198. package/dist/commands/chain-walk.d.ts +14 -0
  199. package/dist/commands/chain-walk.d.ts.map +1 -0
  200. package/dist/commands/chain-walk.js +38 -0
  201. package/dist/commands/chain-walk.js.map +1 -0
  202. package/dist/commands/changes.d.ts +11 -0
  203. package/dist/commands/changes.d.ts.map +1 -0
  204. package/dist/commands/changes.js +46 -0
  205. package/dist/commands/changes.js.map +1 -0
  206. package/dist/commands/codex-prep.d.ts +12 -0
  207. package/dist/commands/codex-prep.d.ts.map +1 -0
  208. package/dist/commands/codex-prep.js +122 -0
  209. package/dist/commands/codex-prep.js.map +1 -0
  210. package/dist/commands/collections.d.ts +22 -0
  211. package/dist/commands/collections.d.ts.map +1 -0
  212. package/dist/commands/collections.js +77 -0
  213. package/dist/commands/collections.js.map +1 -0
  214. package/dist/commands/connect-integration.test.d.ts +7 -0
  215. package/dist/commands/connect-integration.test.d.ts.map +1 -0
  216. package/dist/commands/connect-integration.test.js +195 -0
  217. package/dist/commands/connect-integration.test.js.map +1 -0
  218. package/dist/commands/connect-screens.d.ts +21 -0
  219. package/dist/commands/connect-screens.d.ts.map +1 -0
  220. package/dist/commands/connect-screens.js +79 -0
  221. package/dist/commands/connect-screens.js.map +1 -0
  222. package/dist/commands/connect.d.ts +21 -0
  223. package/dist/commands/connect.d.ts.map +1 -0
  224. package/dist/commands/connect.js +237 -0
  225. package/dist/commands/connect.js.map +1 -0
  226. package/dist/commands/connect.test.d.ts +6 -0
  227. package/dist/commands/connect.test.d.ts.map +1 -0
  228. package/dist/commands/connect.test.js +234 -0
  229. package/dist/commands/connect.test.js.map +1 -0
  230. package/dist/commands/constellation.d.ts.map +1 -1
  231. package/dist/commands/constellation.js +8 -3
  232. package/dist/commands/constellation.js.map +1 -1
  233. package/dist/commands/context.d.ts.map +1 -1
  234. package/dist/commands/context.js +8 -3
  235. package/dist/commands/context.js.map +1 -1
  236. package/dist/commands/cross-cut.d.ts +11 -0
  237. package/dist/commands/cross-cut.d.ts.map +1 -0
  238. package/dist/commands/cross-cut.js +23 -0
  239. package/dist/commands/cross-cut.js.map +1 -0
  240. package/dist/commands/doctor.d.ts +18 -0
  241. package/dist/commands/doctor.d.ts.map +1 -0
  242. package/dist/commands/doctor.js +232 -0
  243. package/dist/commands/doctor.js.map +1 -0
  244. package/dist/commands/doctor.test.d.ts +8 -0
  245. package/dist/commands/doctor.test.d.ts.map +1 -0
  246. package/dist/commands/doctor.test.js +311 -0
  247. package/dist/commands/doctor.test.js.map +1 -0
  248. package/dist/commands/fields.d.ts.map +1 -1
  249. package/dist/commands/fields.js +8 -4
  250. package/dist/commands/fields.js.map +1 -1
  251. package/dist/commands/get.d.ts.map +1 -1
  252. package/dist/commands/get.js +14 -6
  253. package/dist/commands/get.js.map +1 -1
  254. package/dist/commands/handshake.d.ts +12 -0
  255. package/dist/commands/handshake.d.ts.map +1 -1
  256. package/dist/commands/handshake.js +422 -74
  257. package/dist/commands/handshake.js.map +1 -1
  258. package/dist/commands/ingest.d.ts +7 -2
  259. package/dist/commands/ingest.d.ts.map +1 -1
  260. package/dist/commands/ingest.js +137 -31
  261. package/dist/commands/ingest.js.map +1 -1
  262. package/dist/commands/init.d.ts +14 -0
  263. package/dist/commands/init.d.ts.map +1 -0
  264. package/dist/commands/init.js +109 -0
  265. package/dist/commands/init.js.map +1 -0
  266. package/dist/commands/login.d.ts +4 -0
  267. package/dist/commands/login.d.ts.map +1 -1
  268. package/dist/commands/login.js +101 -38
  269. package/dist/commands/login.js.map +1 -1
  270. package/dist/commands/method.d.ts +99 -0
  271. package/dist/commands/method.d.ts.map +1 -0
  272. package/dist/commands/method.js +781 -0
  273. package/dist/commands/method.js.map +1 -0
  274. package/dist/commands/orient.d.ts +106 -1
  275. package/dist/commands/orient.d.ts.map +1 -1
  276. package/dist/commands/orient.js +13 -2
  277. package/dist/commands/orient.js.map +1 -1
  278. package/dist/commands/profile.d.ts +24 -0
  279. package/dist/commands/profile.d.ts.map +1 -0
  280. package/dist/commands/profile.js +82 -0
  281. package/dist/commands/profile.js.map +1 -0
  282. package/dist/commands/promote.d.ts +12 -0
  283. package/dist/commands/promote.d.ts.map +1 -0
  284. package/dist/commands/promote.js +90 -0
  285. package/dist/commands/promote.js.map +1 -0
  286. package/dist/commands/proposals.d.ts +9 -0
  287. package/dist/commands/proposals.d.ts.map +1 -0
  288. package/dist/commands/proposals.js +24 -0
  289. package/dist/commands/proposals.js.map +1 -0
  290. package/dist/commands/reject.d.ts +14 -0
  291. package/dist/commands/reject.d.ts.map +1 -0
  292. package/dist/commands/reject.js +43 -0
  293. package/dist/commands/reject.js.map +1 -0
  294. package/dist/commands/relate.d.ts +1 -0
  295. package/dist/commands/relate.d.ts.map +1 -1
  296. package/dist/commands/relate.js +41 -15
  297. package/dist/commands/relate.js.map +1 -1
  298. package/dist/commands/search.js +2 -2
  299. package/dist/commands/search.js.map +1 -1
  300. package/dist/commands/session.d.ts.map +1 -1
  301. package/dist/commands/session.js +85 -16
  302. package/dist/commands/session.js.map +1 -1
  303. package/dist/commands/setup.d.ts +15 -0
  304. package/dist/commands/setup.d.ts.map +1 -0
  305. package/dist/commands/setup.js +148 -0
  306. package/dist/commands/setup.js.map +1 -0
  307. package/dist/commands/update.d.ts +1 -0
  308. package/dist/commands/update.d.ts.map +1 -1
  309. package/dist/commands/update.js +102 -13
  310. package/dist/commands/update.js.map +1 -1
  311. package/dist/commands/usage.d.ts +40 -0
  312. package/dist/commands/usage.d.ts.map +1 -0
  313. package/dist/commands/usage.js +232 -0
  314. package/dist/commands/usage.js.map +1 -0
  315. package/dist/commands/verify.d.ts +13 -0
  316. package/dist/commands/verify.d.ts.map +1 -0
  317. package/dist/commands/verify.js +49 -0
  318. package/dist/commands/verify.js.map +1 -0
  319. package/dist/commands/welcome.d.ts +21 -0
  320. package/dist/commands/welcome.d.ts.map +1 -0
  321. package/dist/commands/welcome.js +50 -0
  322. package/dist/commands/welcome.js.map +1 -0
  323. package/dist/commands/workspace.d.ts +41 -0
  324. package/dist/commands/workspace.d.ts.map +1 -0
  325. package/dist/commands/workspace.js +239 -0
  326. package/dist/commands/workspace.js.map +1 -0
  327. package/dist/formatters/audit.d.ts +46 -0
  328. package/dist/formatters/audit.d.ts.map +1 -0
  329. package/dist/formatters/audit.js +81 -0
  330. package/dist/formatters/audit.js.map +1 -0
  331. package/dist/formatters/brief.d.ts +112 -0
  332. package/dist/formatters/brief.d.ts.map +1 -0
  333. package/dist/formatters/brief.js +179 -0
  334. package/dist/formatters/brief.js.map +1 -0
  335. package/dist/formatters/capture.d.ts +21 -3
  336. package/dist/formatters/capture.d.ts.map +1 -1
  337. package/dist/formatters/capture.js +20 -1
  338. package/dist/formatters/capture.js.map +1 -1
  339. package/dist/formatters/chain-walk.d.ts +33 -0
  340. package/dist/formatters/chain-walk.d.ts.map +1 -0
  341. package/dist/formatters/chain-walk.js +54 -0
  342. package/dist/formatters/chain-walk.js.map +1 -0
  343. package/dist/formatters/changes.d.ts +25 -0
  344. package/dist/formatters/changes.d.ts.map +1 -0
  345. package/dist/formatters/changes.js +60 -0
  346. package/dist/formatters/changes.js.map +1 -0
  347. package/dist/formatters/collections.d.ts +40 -0
  348. package/dist/formatters/collections.d.ts.map +1 -0
  349. package/dist/formatters/collections.js +93 -0
  350. package/dist/formatters/collections.js.map +1 -0
  351. package/dist/formatters/cross-cut.d.ts +21 -0
  352. package/dist/formatters/cross-cut.d.ts.map +1 -0
  353. package/dist/formatters/cross-cut.js +32 -0
  354. package/dist/formatters/cross-cut.js.map +1 -0
  355. package/dist/formatters/entry.d.ts +11 -4
  356. package/dist/formatters/entry.d.ts.map +1 -1
  357. package/dist/formatters/entry.js +24 -8
  358. package/dist/formatters/entry.js.map +1 -1
  359. package/dist/formatters/handshake.d.ts +29 -0
  360. package/dist/formatters/handshake.d.ts.map +1 -1
  361. package/dist/formatters/handshake.js +115 -3
  362. package/dist/formatters/handshake.js.map +1 -1
  363. package/dist/formatters/orient.d.ts +104 -1
  364. package/dist/formatters/orient.d.ts.map +1 -1
  365. package/dist/formatters/orient.js +140 -17
  366. package/dist/formatters/orient.js.map +1 -1
  367. package/dist/formatters/promote.d.ts +30 -0
  368. package/dist/formatters/promote.d.ts.map +1 -0
  369. package/dist/formatters/promote.js +39 -0
  370. package/dist/formatters/promote.js.map +1 -0
  371. package/dist/formatters/proposals.d.ts +45 -0
  372. package/dist/formatters/proposals.d.ts.map +1 -0
  373. package/dist/formatters/proposals.js +62 -0
  374. package/dist/formatters/proposals.js.map +1 -0
  375. package/dist/formatters/relate.d.ts +3 -0
  376. package/dist/formatters/relate.d.ts.map +1 -1
  377. package/dist/formatters/relate.js +6 -0
  378. package/dist/formatters/relate.js.map +1 -1
  379. package/dist/formatters/search.d.ts +0 -4
  380. package/dist/formatters/search.d.ts.map +1 -1
  381. package/dist/formatters/search.js +4 -1
  382. package/dist/formatters/search.js.map +1 -1
  383. package/dist/formatters/session.d.ts +1 -1
  384. package/dist/formatters/session.d.ts.map +1 -1
  385. package/dist/formatters/session.js +3 -1
  386. package/dist/formatters/session.js.map +1 -1
  387. package/dist/formatters/update.d.ts.map +1 -1
  388. package/dist/formatters/update.js +2 -0
  389. package/dist/formatters/update.js.map +1 -1
  390. package/dist/formatters/verify.d.ts +11 -0
  391. package/dist/formatters/verify.d.ts.map +1 -0
  392. package/dist/formatters/verify.js +11 -0
  393. package/dist/formatters/verify.js.map +1 -0
  394. package/dist/generators/__tests__/surface-profiles.test.d.ts +2 -0
  395. package/dist/generators/__tests__/surface-profiles.test.d.ts.map +1 -0
  396. package/dist/generators/__tests__/surface-profiles.test.js +89 -0
  397. package/dist/generators/__tests__/surface-profiles.test.js.map +1 -0
  398. package/dist/generators/adapters.d.ts +37 -3
  399. package/dist/generators/adapters.d.ts.map +1 -1
  400. package/dist/generators/adapters.js +193 -5
  401. package/dist/generators/adapters.js.map +1 -1
  402. package/dist/generators/adapters.test.d.ts +2 -0
  403. package/dist/generators/adapters.test.d.ts.map +1 -0
  404. package/dist/generators/adapters.test.js +27 -0
  405. package/dist/generators/adapters.test.js.map +1 -0
  406. package/dist/generators/archetypes.d.ts +52 -0
  407. package/dist/generators/archetypes.d.ts.map +1 -0
  408. package/dist/generators/archetypes.js +153 -0
  409. package/dist/generators/archetypes.js.map +1 -0
  410. package/dist/generators/archetypes.test.d.ts +2 -0
  411. package/dist/generators/archetypes.test.d.ts.map +1 -0
  412. package/dist/generators/archetypes.test.js +237 -0
  413. package/dist/generators/archetypes.test.js.map +1 -0
  414. package/dist/generators/chain-classifier.d.ts +49 -0
  415. package/dist/generators/chain-classifier.d.ts.map +1 -0
  416. package/dist/generators/chain-classifier.js +180 -0
  417. package/dist/generators/chain-classifier.js.map +1 -0
  418. package/dist/generators/chain-classifier.test.d.ts +2 -0
  419. package/dist/generators/chain-classifier.test.d.ts.map +1 -0
  420. package/dist/generators/chain-classifier.test.js +257 -0
  421. package/dist/generators/chain-classifier.test.js.map +1 -0
  422. package/dist/generators/chain-rules.d.ts +42 -0
  423. package/dist/generators/chain-rules.d.ts.map +1 -0
  424. package/dist/generators/chain-rules.js +144 -0
  425. package/dist/generators/chain-rules.js.map +1 -0
  426. package/dist/generators/chain-rules.test.d.ts +2 -0
  427. package/dist/generators/chain-rules.test.d.ts.map +1 -0
  428. package/dist/generators/chain-rules.test.js +179 -0
  429. package/dist/generators/chain-rules.test.js.map +1 -0
  430. package/dist/generators/context-md.d.ts +1 -1
  431. package/dist/generators/context-md.d.ts.map +1 -1
  432. package/dist/generators/context-md.js +12 -1
  433. package/dist/generators/context-md.js.map +1 -1
  434. package/dist/generators/handshake-diff.d.ts +67 -0
  435. package/dist/generators/handshake-diff.d.ts.map +1 -0
  436. package/dist/generators/handshake-diff.js +183 -0
  437. package/dist/generators/handshake-diff.js.map +1 -0
  438. package/dist/generators/handshake-diff.test.d.ts +2 -0
  439. package/dist/generators/handshake-diff.test.d.ts.map +1 -0
  440. package/dist/generators/handshake-diff.test.js +264 -0
  441. package/dist/generators/handshake-diff.test.js.map +1 -0
  442. package/dist/generators/manifest.d.ts +37 -0
  443. package/dist/generators/manifest.d.ts.map +1 -0
  444. package/dist/generators/manifest.js +166 -0
  445. package/dist/generators/manifest.js.map +1 -0
  446. package/dist/generators/portable-knowledge.d.ts +102 -9
  447. package/dist/generators/portable-knowledge.d.ts.map +1 -1
  448. package/dist/generators/portable-knowledge.js +384 -17
  449. package/dist/generators/portable-knowledge.js.map +1 -1
  450. package/dist/generators/portable-knowledge.test.js +529 -1
  451. package/dist/generators/portable-knowledge.test.js.map +1 -1
  452. package/dist/generators/surface-profiles.d.ts +49 -0
  453. package/dist/generators/surface-profiles.d.ts.map +1 -0
  454. package/dist/generators/surface-profiles.js +98 -0
  455. package/dist/generators/surface-profiles.js.map +1 -0
  456. package/dist/index.js +618 -138
  457. package/dist/index.js.map +1 -1
  458. package/dist/lib/activation.d.ts +28 -0
  459. package/dist/lib/activation.d.ts.map +1 -0
  460. package/dist/lib/activation.js +57 -0
  461. package/dist/lib/activation.js.map +1 -0
  462. package/dist/lib/activation.test.d.ts +6 -0
  463. package/dist/lib/activation.test.d.ts.map +1 -0
  464. package/dist/lib/activation.test.js +121 -0
  465. package/dist/lib/activation.test.js.map +1 -0
  466. package/dist/lib/client.d.ts +45 -3
  467. package/dist/lib/client.d.ts.map +1 -1
  468. package/dist/lib/client.js +188 -36
  469. package/dist/lib/client.js.map +1 -1
  470. package/dist/lib/collectionRegistry.d.ts +38 -0
  471. package/dist/lib/collectionRegistry.d.ts.map +1 -0
  472. package/dist/lib/collectionRegistry.js +112 -0
  473. package/dist/lib/collectionRegistry.js.map +1 -0
  474. package/dist/lib/config.d.ts +84 -4
  475. package/dist/lib/config.d.ts.map +1 -1
  476. package/dist/lib/config.js +322 -42
  477. package/dist/lib/config.js.map +1 -1
  478. package/dist/lib/constants.d.ts +42 -0
  479. package/dist/lib/constants.d.ts.map +1 -0
  480. package/dist/lib/constants.js +76 -0
  481. package/dist/lib/constants.js.map +1 -0
  482. package/dist/lib/conversation-engine.d.ts +45 -0
  483. package/dist/lib/conversation-engine.d.ts.map +1 -0
  484. package/dist/lib/conversation-engine.js +112 -0
  485. package/dist/lib/conversation-engine.js.map +1 -0
  486. package/dist/lib/conversation-phases.d.ts +59 -0
  487. package/dist/lib/conversation-phases.d.ts.map +1 -0
  488. package/dist/lib/conversation-phases.js +11 -0
  489. package/dist/lib/conversation-phases.js.map +1 -0
  490. package/dist/lib/conversation-signals.d.ts +30 -0
  491. package/dist/lib/conversation-signals.d.ts.map +1 -0
  492. package/dist/lib/conversation-signals.js +64 -0
  493. package/dist/lib/conversation-signals.js.map +1 -0
  494. package/dist/lib/deployment.d.ts +23 -0
  495. package/dist/lib/deployment.d.ts.map +1 -0
  496. package/dist/lib/deployment.js +78 -0
  497. package/dist/lib/deployment.js.map +1 -0
  498. package/dist/lib/deployment.test.d.ts +5 -0
  499. package/dist/lib/deployment.test.d.ts.map +1 -0
  500. package/dist/lib/deployment.test.js +54 -0
  501. package/dist/lib/deployment.test.js.map +1 -0
  502. package/dist/lib/errors.d.ts +58 -0
  503. package/dist/lib/errors.d.ts.map +1 -0
  504. package/dist/lib/errors.js +67 -0
  505. package/dist/lib/errors.js.map +1 -0
  506. package/dist/lib/experiment.d.ts +18 -0
  507. package/dist/lib/experiment.d.ts.map +1 -0
  508. package/dist/lib/experiment.js +28 -0
  509. package/dist/lib/experiment.js.map +1 -0
  510. package/dist/lib/format.d.ts +10 -0
  511. package/dist/lib/format.d.ts.map +1 -0
  512. package/dist/lib/format.js +27 -0
  513. package/dist/lib/format.js.map +1 -0
  514. package/dist/lib/glossary.d.ts +19 -0
  515. package/dist/lib/glossary.d.ts.map +1 -0
  516. package/dist/lib/glossary.js +53 -0
  517. package/dist/lib/glossary.js.map +1 -0
  518. package/dist/lib/hook-intents.d.ts +51 -0
  519. package/dist/lib/hook-intents.d.ts.map +1 -0
  520. package/dist/lib/hook-intents.js +85 -0
  521. package/dist/lib/hook-intents.js.map +1 -0
  522. package/dist/lib/inferSourceDate.d.ts +12 -0
  523. package/dist/lib/inferSourceDate.d.ts.map +1 -0
  524. package/dist/lib/inferSourceDate.js +44 -0
  525. package/dist/lib/inferSourceDate.js.map +1 -0
  526. package/dist/lib/method-registry.d.ts +32 -0
  527. package/dist/lib/method-registry.d.ts.map +1 -0
  528. package/dist/lib/method-registry.js +53 -0
  529. package/dist/lib/method-registry.js.map +1 -0
  530. package/dist/lib/onboarding-path-b.d.ts +10 -0
  531. package/dist/lib/onboarding-path-b.d.ts.map +1 -0
  532. package/dist/lib/onboarding-path-b.js +214 -0
  533. package/dist/lib/onboarding-path-b.js.map +1 -0
  534. package/dist/lib/onboarding-phases.d.ts +9 -0
  535. package/dist/lib/onboarding-phases.d.ts.map +1 -0
  536. package/dist/lib/onboarding-phases.js +120 -0
  537. package/dist/lib/onboarding-phases.js.map +1 -0
  538. package/dist/lib/onboarding-shared.d.ts +81 -0
  539. package/dist/lib/onboarding-shared.d.ts.map +1 -0
  540. package/dist/lib/onboarding-shared.js +190 -0
  541. package/dist/lib/onboarding-shared.js.map +1 -0
  542. package/dist/lib/onboarding-topics.d.ts +27 -0
  543. package/dist/lib/onboarding-topics.d.ts.map +1 -0
  544. package/dist/lib/onboarding-topics.js +57 -0
  545. package/dist/lib/onboarding-topics.js.map +1 -0
  546. package/dist/lib/onboarding.d.ts +17 -0
  547. package/dist/lib/onboarding.d.ts.map +1 -0
  548. package/dist/lib/onboarding.js +350 -0
  549. package/dist/lib/onboarding.js.map +1 -0
  550. package/dist/lib/profiles.d.ts +39 -0
  551. package/dist/lib/profiles.d.ts.map +1 -0
  552. package/dist/lib/profiles.js +185 -0
  553. package/dist/lib/profiles.js.map +1 -0
  554. package/dist/lib/prompts.d.ts +65 -0
  555. package/dist/lib/prompts.d.ts.map +1 -0
  556. package/dist/lib/prompts.js +132 -0
  557. package/dist/lib/prompts.js.map +1 -0
  558. package/dist/lib/repo-detect.d.ts +19 -0
  559. package/dist/lib/repo-detect.d.ts.map +1 -1
  560. package/dist/lib/repo-detect.js +25 -0
  561. package/dist/lib/repo-detect.js.map +1 -1
  562. package/dist/lib/runner.d.ts +2 -0
  563. package/dist/lib/runner.d.ts.map +1 -1
  564. package/dist/lib/runner.js +21 -7
  565. package/dist/lib/runner.js.map +1 -1
  566. package/dist/lib/session.d.ts +25 -0
  567. package/dist/lib/session.d.ts.map +1 -1
  568. package/dist/lib/session.js +70 -4
  569. package/dist/lib/session.js.map +1 -1
  570. package/dist/lib/spinner.d.ts +27 -0
  571. package/dist/lib/spinner.d.ts.map +1 -0
  572. package/dist/lib/spinner.js +76 -0
  573. package/dist/lib/spinner.js.map +1 -0
  574. package/dist/lib/spinner.test.d.ts +2 -0
  575. package/dist/lib/spinner.test.d.ts.map +1 -0
  576. package/dist/lib/spinner.test.js +39 -0
  577. package/dist/lib/spinner.test.js.map +1 -0
  578. package/dist/lib/state.d.ts +51 -0
  579. package/dist/lib/state.d.ts.map +1 -0
  580. package/dist/lib/state.js +90 -0
  581. package/dist/lib/state.js.map +1 -0
  582. package/dist/lib/strip.d.ts +1 -0
  583. package/dist/lib/strip.d.ts.map +1 -1
  584. package/dist/lib/strip.js +15 -0
  585. package/dist/lib/strip.js.map +1 -1
  586. package/dist/lib/style.d.ts +96 -0
  587. package/dist/lib/style.d.ts.map +1 -0
  588. package/dist/lib/style.js +169 -0
  589. package/dist/lib/style.js.map +1 -0
  590. package/dist/lib/style.test.d.ts +7 -0
  591. package/dist/lib/style.test.d.ts.map +1 -0
  592. package/dist/lib/style.test.js +263 -0
  593. package/dist/lib/style.test.js.map +1 -0
  594. package/dist/lib/telemetry.d.ts +15 -0
  595. package/dist/lib/telemetry.d.ts.map +1 -0
  596. package/dist/lib/telemetry.js +47 -0
  597. package/dist/lib/telemetry.js.map +1 -0
  598. package/dist/lib/tokenConstants.d.ts +17 -0
  599. package/dist/lib/tokenConstants.d.ts.map +1 -0
  600. package/dist/lib/tokenConstants.js +17 -0
  601. package/dist/lib/tokenConstants.js.map +1 -0
  602. package/dist/lib/update-check.d.ts +21 -0
  603. package/dist/lib/update-check.d.ts.map +1 -0
  604. package/dist/lib/update-check.js +145 -0
  605. package/dist/lib/update-check.js.map +1 -0
  606. package/dist/lib/wizard-surfaces.d.ts +47 -0
  607. package/dist/lib/wizard-surfaces.d.ts.map +1 -0
  608. package/dist/lib/wizard-surfaces.js +176 -0
  609. package/dist/lib/wizard-surfaces.js.map +1 -0
  610. package/dist/lib/wizard-surfaces.test.d.ts +2 -0
  611. package/dist/lib/wizard-surfaces.test.d.ts.map +1 -0
  612. package/dist/lib/wizard-surfaces.test.js +127 -0
  613. package/dist/lib/wizard-surfaces.test.js.map +1 -0
  614. package/dist/lib/wizard-trust.d.ts +31 -0
  615. package/dist/lib/wizard-trust.d.ts.map +1 -0
  616. package/dist/lib/wizard-trust.js +66 -0
  617. package/dist/lib/wizard-trust.js.map +1 -0
  618. package/dist/lib/wizard-trust.test.d.ts +2 -0
  619. package/dist/lib/wizard-trust.test.d.ts.map +1 -0
  620. package/dist/lib/wizard-trust.test.js +32 -0
  621. package/dist/lib/wizard-trust.test.js.map +1 -0
  622. package/dist/lib/workspace-probe.d.ts +19 -0
  623. package/dist/lib/workspace-probe.d.ts.map +1 -0
  624. package/dist/lib/workspace-probe.js +27 -0
  625. package/dist/lib/workspace-probe.js.map +1 -0
  626. package/dist/surfaces/registry.d.ts +20 -0
  627. package/dist/surfaces/registry.d.ts.map +1 -0
  628. package/dist/surfaces/registry.js +42 -0
  629. package/dist/surfaces/registry.js.map +1 -0
  630. package/package.json +9 -3
  631. package/templates/archetypes/boundary.md +23 -0
  632. package/templates/archetypes/constraint.md +23 -0
  633. package/templates/archetypes/convention.md +23 -0
  634. package/templates/archetypes/policy.md +23 -0
  635. package/templates/archetypes/quality-gate.md +23 -0
  636. package/templates/archetypes/workflow.md +23 -0
  637. package/templates/general/code-integrity.md +11 -0
  638. package/templates/general/getting-started.md +12 -0
  639. package/templates/method-registry.json +16 -0
  640. package/templates/node-ts/code-integrity.md +13 -0
  641. package/templates/node-ts/testing.md +12 -0
  642. package/templates/python/code-integrity.md +13 -0
  643. package/templates/python/testing.md +12 -0
@@ -0,0 +1,263 @@
1
+ /**
2
+ * BET-270 S1: Batch transformation tests for surface-optimized skill projection.
3
+ * Tests applyBatchTransformations + generateCodexSkill batch mode + Claude/Cursor unchanged.
4
+ */
5
+ import { describe, expect, it } from 'vitest';
6
+ import { applyBatchTransformations, generateCodexSkill, generateCursorSkill, generateClaudeRule, generateClaudeSkillRouter, } from '../generators/portable-knowledge.js';
7
+ import { SURFACE_PROFILES } from '../generators/surface-profiles.js';
8
+ describe('applyBatchTransformations', () => {
9
+ it('strips sub-agent spawn patterns and replaces with inline step instruction', () => {
10
+ const body = `# My Skill
11
+
12
+ Do some work.
13
+ Spawn a sub-agent to handle the review.
14
+ Continue with the result.`;
15
+ const result = applyBatchTransformations(body);
16
+ expect(result).not.toContain('Spawn a sub-agent');
17
+ expect(result).toContain('Execute as inline steps within this session.');
18
+ expect(result).toContain('Do some work.');
19
+ expect(result).toContain('Continue with the result.');
20
+ });
21
+ it('strips SPAWN build sub-agent pattern (uppercase SPAWN)', () => {
22
+ const body = `# Heading
23
+
24
+ SPAWN build sub-agent with primer.
25
+ Other content.`;
26
+ const result = applyBatchTransformations(body);
27
+ expect(result).not.toContain('SPAWN build sub-agent');
28
+ expect(result).toContain('Execute as inline steps within this session.');
29
+ });
30
+ it('strips SPAWN review sub-agent pattern', () => {
31
+ const body = `# Heading
32
+
33
+ SPAWN review sub-agent for verification.
34
+ Done.`;
35
+ const result = applyBatchTransformations(body);
36
+ expect(result).not.toContain('SPAWN review sub-agent');
37
+ expect(result).toContain('Execute as inline steps within this session.');
38
+ });
39
+ it('strips "fresh conversation handoff" pattern', () => {
40
+ const body = `# Heading
41
+
42
+ Use fresh conversation handoff for next slice.
43
+ Continue.`;
44
+ const result = applyBatchTransformations(body);
45
+ expect(result).not.toContain('fresh conversation handoff');
46
+ expect(result).toContain('Execute as inline steps within this session.');
47
+ });
48
+ it('strips user confirmation gates — "Ask the user"', () => {
49
+ const body = `# Skill
50
+
51
+ Do work.
52
+ Ask the user before proceeding.
53
+ Continue.`;
54
+ const result = applyBatchTransformations(body);
55
+ expect(result).not.toContain('Ask the user');
56
+ expect(result).toContain('Proceed autonomously. Log decision rationale.');
57
+ expect(result).toContain('Do work.');
58
+ expect(result).toContain('Continue.');
59
+ });
60
+ it('strips user confirmation gates — "Wait for user input"', () => {
61
+ const body = `# Skill
62
+
63
+ Step one done.
64
+ Wait for user input before continuing.
65
+ Step two.`;
66
+ const result = applyBatchTransformations(body);
67
+ expect(result).not.toContain('Wait for user input');
68
+ expect(result).toContain('Proceed autonomously. Log decision rationale.');
69
+ });
70
+ it('strips user confirmation gates — "Present gate to user"', () => {
71
+ const body = `# Skill
72
+
73
+ Build the feature.
74
+ Present gate to user before merging.
75
+ Merge.`;
76
+ const result = applyBatchTransformations(body);
77
+ expect(result).not.toContain('Present gate to user');
78
+ expect(result).toContain('Proceed autonomously. Log decision rationale.');
79
+ });
80
+ it('strips user confirmation gates — "STOP and ask"', () => {
81
+ const body = `# Skill
82
+
83
+ Analyze problem.
84
+ STOP and ask the user whether to continue.
85
+ Then proceed.`;
86
+ const result = applyBatchTransformations(body);
87
+ expect(result).not.toContain('STOP and ask');
88
+ expect(result).toContain('Proceed autonomously. Log decision rationale.');
89
+ });
90
+ it('adds batch governance header after the first # heading', () => {
91
+ const body = `# My Skill
92
+
93
+ Some content here.`;
94
+ const result = applyBatchTransformations(body);
95
+ expect(result).toContain('> Batch execution mode: This skill runs autonomously without interactive gates. Decisions are logged, not prompted.');
96
+ // Header must appear right after the heading
97
+ const lines = result.split('\n');
98
+ const headingIdx = lines.findIndex((l) => l.startsWith('# My Skill'));
99
+ const batchHeaderIdx = lines.findIndex((l) => l.includes('Batch execution mode:'));
100
+ expect(headingIdx).toBeGreaterThanOrEqual(0);
101
+ expect(batchHeaderIdx).toBeGreaterThan(headingIdx);
102
+ expect(batchHeaderIdx - headingIdx).toBeLessThanOrEqual(3);
103
+ });
104
+ it('preserves non-interactive content unchanged', () => {
105
+ const body = `# Clean Skill
106
+
107
+ This skill has no interactive patterns.
108
+ It describes steps the agent should follow.
109
+ Use grep to find the file.
110
+ Run npm run build.`;
111
+ const result = applyBatchTransformations(body);
112
+ expect(result).toContain('This skill has no interactive patterns.');
113
+ expect(result).toContain('It describes steps the agent should follow.');
114
+ expect(result).toContain('Use grep to find the file.');
115
+ expect(result).toContain('Run npm run build.');
116
+ expect(result).toContain('Batch execution mode:');
117
+ });
118
+ it('preserves interactive patterns inside fenced code blocks', () => {
119
+ const body = `# Skill With Examples
120
+
121
+ Here is how it works in interactive mode:
122
+
123
+ \`\`\`
124
+ Spawn a sub-agent to handle the review.
125
+ Ask the user before proceeding.
126
+ Wait for user input.
127
+ \`\`\`
128
+
129
+ Normal prose outside the fence.
130
+ Spawn a sub-agent for the next step.`;
131
+ const result = applyBatchTransformations(body);
132
+ // Patterns inside the code fence must be preserved verbatim
133
+ expect(result).toContain('Spawn a sub-agent to handle the review.');
134
+ expect(result).toContain('Ask the user before proceeding.');
135
+ expect(result).toContain('Wait for user input.');
136
+ // Pattern outside the code fence must be transformed
137
+ expect(result).not.toContain('Spawn a sub-agent for the next step.');
138
+ expect(result).toContain('Execute as inline steps within this session.');
139
+ });
140
+ it('preserves patterns inside tilde-fenced code blocks', () => {
141
+ const body = `# Skill
142
+
143
+ ~~~bash
144
+ Ask the user for confirmation.
145
+ ~~~
146
+
147
+ Ask the user to approve.`;
148
+ const result = applyBatchTransformations(body);
149
+ // Inside fence: preserved
150
+ expect(result).toContain('Ask the user for confirmation.');
151
+ // Outside fence: transformed
152
+ expect(result).not.toContain('Ask the user to approve.');
153
+ expect(result).toContain('Proceed autonomously. Log decision rationale.');
154
+ });
155
+ it('does not transform every mention of "user" — only clear interactive gate patterns', () => {
156
+ const body = `# Skill
157
+
158
+ The user should be aware of this tradeoff.
159
+ User data must be validated before insert.`;
160
+ const result = applyBatchTransformations(body);
161
+ expect(result).toContain('The user should be aware of this tradeoff.');
162
+ expect(result).toContain('User data must be validated before insert.');
163
+ expect(result).not.toContain('Proceed autonomously. Log decision rationale.');
164
+ });
165
+ });
166
+ describe('generateCodexSkill — batch mode (BET-270 S1)', () => {
167
+ const baseSkill = {
168
+ name: 'orchestrate',
169
+ description: 'Orchestrate a bet slice.',
170
+ triggers: ['orchestrate'],
171
+ body: `# Orchestrate
172
+
173
+ Do planning.
174
+ Spawn a sub-agent to execute.
175
+ Ask the user to confirm the plan.
176
+ Finalize.`,
177
+ sourcePath: '/tmp/.productbrain/skills/orchestrate.md',
178
+ };
179
+ it('applies batch transformations when profile.executionModel is batch', () => {
180
+ const output = generateCodexSkill(baseSkill, SURFACE_PROFILES.codex);
181
+ expect(output).not.toContain('Spawn a sub-agent');
182
+ expect(output).not.toContain('Ask the user');
183
+ expect(output).toContain('Execute as inline steps within this session.');
184
+ expect(output).toContain('Proceed autonomously. Log decision rationale.');
185
+ expect(output).toContain('Batch execution mode:');
186
+ });
187
+ it('produces CURRENT output (no batch transforms) when profile is undefined — backward compat', () => {
188
+ const output = generateCodexSkill(baseSkill, undefined);
189
+ expect(output).toContain('Spawn a sub-agent to execute.');
190
+ expect(output).toContain('Ask the user to confirm the plan.');
191
+ expect(output).not.toContain('Batch execution mode:');
192
+ });
193
+ it('includes skill name, description, triggers in both modes', () => {
194
+ const withProfile = generateCodexSkill(baseSkill, SURFACE_PROFILES.codex);
195
+ const withoutProfile = generateCodexSkill(baseSkill, undefined);
196
+ for (const output of [withProfile, withoutProfile]) {
197
+ expect(output).toContain('# orchestrate');
198
+ expect(output).toContain('Orchestrate a bet slice.');
199
+ expect(output).toContain('`orchestrate`');
200
+ }
201
+ });
202
+ it('still performs path replacement before batch transforms', () => {
203
+ const skillWithPath = {
204
+ ...baseSkill,
205
+ body: `# Path Skill
206
+
207
+ Read \`.productbrain/skills/preflight.md\` before starting.
208
+ Ask the user for approval.`,
209
+ };
210
+ const output = generateCodexSkill(skillWithPath, SURFACE_PROFILES.codex);
211
+ expect(output).toContain('.codex/skills/preflight.md');
212
+ expect(output).not.toContain('.productbrain/skills/preflight.md');
213
+ expect(output).not.toContain('Ask the user');
214
+ });
215
+ });
216
+ describe('generateCursorSkill / generateClaudeRule / generateClaudeSkillRouter — profile ignored (BET-270 S1)', () => {
217
+ const skill = {
218
+ name: 'test-skill',
219
+ description: 'Test skill',
220
+ triggers: ['test'],
221
+ body: `# Test Skill
222
+
223
+ Ask the user to confirm.
224
+ Spawn a sub-agent.
225
+ Normal content.`,
226
+ sourcePath: '/tmp/.productbrain/skills/test-skill.md',
227
+ };
228
+ const rule = {
229
+ name: 'test-rule',
230
+ description: 'A test rule',
231
+ autoApply: true,
232
+ body: '# Rule body\n\nAsk the user to confirm.\nNormal content.',
233
+ sourcePath: '/tmp/.productbrain/rules/test-rule.md',
234
+ };
235
+ it('generateCursorSkill preserves interactive content regardless of profile', () => {
236
+ const withProfile = generateCursorSkill(skill, SURFACE_PROFILES.cursor);
237
+ const withCodexProfile = generateCursorSkill(skill, SURFACE_PROFILES.codex);
238
+ for (const output of [withProfile, withCodexProfile]) {
239
+ expect(output).toContain('Ask the user to confirm.');
240
+ expect(output).toContain('Spawn a sub-agent.');
241
+ expect(output).not.toContain('Batch execution mode:');
242
+ expect(output).not.toContain('Proceed autonomously');
243
+ }
244
+ });
245
+ it('generateClaudeRule preserves interactive content regardless of profile', () => {
246
+ const withProfile = generateClaudeRule(rule, SURFACE_PROFILES.claude);
247
+ const withCodexProfile = generateClaudeRule(rule, SURFACE_PROFILES.codex);
248
+ for (const output of [withProfile, withCodexProfile]) {
249
+ expect(output).toContain('Ask the user to confirm.');
250
+ expect(output).not.toContain('Batch execution mode:');
251
+ expect(output).not.toContain('Proceed autonomously');
252
+ }
253
+ });
254
+ it('generateClaudeSkillRouter preserves interactive content regardless of profile', () => {
255
+ const withProfile = generateClaudeSkillRouter([skill], SURFACE_PROFILES.claude);
256
+ const withCodexProfile = generateClaudeSkillRouter([skill], SURFACE_PROFILES.codex);
257
+ for (const output of [withProfile, withCodexProfile]) {
258
+ expect(output).toContain('test-skill');
259
+ expect(output).toContain('Skill Router');
260
+ }
261
+ });
262
+ });
263
+ //# sourceMappingURL=batch-transformations.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"batch-transformations.test.js","sourceRoot":"","sources":["../../src/__tests__/batch-transformations.test.ts"],"names":[],"mappings":"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"}
@@ -1,12 +1,13 @@
1
1
  /**
2
- * pb capture extensions — --link, --json flags.
2
+ * pb capture extensions — --link, --json flags, WP-319 grounding.
3
3
  * TEN-341 test contract: capture --link calls createEntryRelation, capture --json outputs JSON.
4
+ * WP-319 S3: grounding fires in parallel, fail-open, --suggest-only, --strict, PB_GROUNDING_MODE.
4
5
  */
5
6
  import { beforeEach, describe, expect, it, vi } from 'vitest';
6
7
  import { runCapture } from '../commands/capture.js';
7
- const mcpCallWithSessionMock = vi.fn();
8
+ const kernelCallWithSessionMock = vi.fn();
8
9
  vi.mock('../lib/client.js', () => ({
9
- mcpCallWithSession: (...args) => mcpCallWithSessionMock(...args),
10
+ kernelCallWithSession: (...args) => kernelCallWithSessionMock(...args),
10
11
  }));
11
12
  vi.mock('../lib/config.js', () => ({
12
13
  getConfigOrGuide: vi.fn(() => Promise.resolve({ apiKey: 'pb_sk_test', siteUrl: 'https://test.convex.site' })),
@@ -18,17 +19,30 @@ vi.mock('../lib/session.js', () => ({
18
19
  }));
19
20
  describe('runCapture --link', () => {
20
21
  beforeEach(() => {
21
- mcpCallWithSessionMock.mockReset();
22
+ kernelCallWithSessionMock.mockReset();
22
23
  mockSession = { sessionId: 'sess-test' };
23
24
  });
24
25
  it('creates entry then links to target with default relation type', async () => {
25
- mcpCallWithSessionMock
26
- .mockResolvedValueOnce({ collection: 'tensions', confidence: 0.9, tier: 'high' })
27
- .mockResolvedValueOnce({ docId: 'doc-1', entryId: 'TEN-710' })
28
- .mockResolvedValueOnce({});
26
+ // Prefix "TEN:" is extracted → skips classification, routes directly to tensions
27
+ kernelCallWithSessionMock.mockImplementation((fn) => {
28
+ if (fn === 'chain.suggestLinksForCapture')
29
+ return Promise.resolve([]);
30
+ if (fn === 'chain.createEntry')
31
+ return Promise.resolve({ docId: 'doc-1', entryId: 'TEN-710' });
32
+ if (fn === 'chain.createEntryRelation')
33
+ return Promise.resolve({});
34
+ return Promise.resolve({});
35
+ });
29
36
  const writeSpy = vi.spyOn(process.stdout, 'write').mockImplementation(() => true);
30
37
  await runCapture({ text: 'TEN: missing validation', link: 'BET-151' });
31
- expect(mcpCallWithSessionMock).toHaveBeenCalledWith('chain.createEntryRelation', {
38
+ // Verify classification was skipped (no resolveCollection call)
39
+ expect(kernelCallWithSessionMock).not.toHaveBeenCalledWith('chain.resolveCollection', expect.anything());
40
+ // Verify prefix was stripped from entry name and routed to correct collection
41
+ expect(kernelCallWithSessionMock).toHaveBeenCalledWith('chain.createEntry', expect.objectContaining({
42
+ collectionSlug: 'tensions',
43
+ name: 'missing validation',
44
+ }));
45
+ expect(kernelCallWithSessionMock).toHaveBeenCalledWith('chain.createEntryRelation', {
32
46
  fromEntryId: 'TEN-710',
33
47
  toEntryId: 'BET-151',
34
48
  type: 'surfaces_tension_in',
@@ -36,13 +50,20 @@ describe('runCapture --link', () => {
36
50
  writeSpy.mockRestore();
37
51
  });
38
52
  it('uses custom relation type when --type provided', async () => {
39
- mcpCallWithSessionMock
40
- .mockResolvedValueOnce({ collection: 'insights', confidence: 0.8, tier: 'high' })
41
- .mockResolvedValueOnce({ docId: 'doc-2', entryId: 'INS-50' })
42
- .mockResolvedValueOnce({});
53
+ // Prefix "INS:" is extracted → skips classification, routes directly to insights
54
+ kernelCallWithSessionMock.mockImplementation((fn) => {
55
+ if (fn === 'chain.suggestLinksForCapture')
56
+ return Promise.resolve([]);
57
+ if (fn === 'chain.createEntry')
58
+ return Promise.resolve({ docId: 'doc-2', entryId: 'INS-50' });
59
+ if (fn === 'chain.createEntryRelation')
60
+ return Promise.resolve({});
61
+ return Promise.resolve({});
62
+ });
43
63
  const writeSpy = vi.spyOn(process.stdout, 'write').mockImplementation(() => true);
44
64
  await runCapture({ text: 'INS: discovered X', link: 'DEC-264', type: 'informed_by' });
45
- expect(mcpCallWithSessionMock).toHaveBeenCalledWith('chain.createEntryRelation', {
65
+ expect(kernelCallWithSessionMock).not.toHaveBeenCalledWith('chain.resolveCollection', expect.anything());
66
+ expect(kernelCallWithSessionMock).toHaveBeenCalledWith('chain.createEntryRelation', {
46
67
  fromEntryId: 'INS-50',
47
68
  toEntryId: 'DEC-264',
48
69
  type: 'informed_by',
@@ -52,13 +73,18 @@ describe('runCapture --link', () => {
52
73
  });
53
74
  describe('runCapture --json', () => {
54
75
  beforeEach(() => {
55
- mcpCallWithSessionMock.mockReset();
76
+ kernelCallWithSessionMock.mockReset();
56
77
  mockSession = { sessionId: 'sess-test' };
57
78
  });
58
79
  it('outputs valid JSON with id field and no other stdout', async () => {
59
- mcpCallWithSessionMock
60
- .mockResolvedValueOnce({ collection: 'tensions', confidence: 0.9, tier: 'high' })
61
- .mockResolvedValueOnce({ docId: 'doc-1', entryId: 'TEN-711' });
80
+ // Prefix "TEN:" is extracted → skips classification
81
+ kernelCallWithSessionMock.mockImplementation((fn) => {
82
+ if (fn === 'chain.suggestLinksForCapture')
83
+ return Promise.resolve([]);
84
+ if (fn === 'chain.createEntry')
85
+ return Promise.resolve({ docId: 'doc-1', entryId: 'TEN-711' });
86
+ return Promise.resolve({});
87
+ });
62
88
  let stdoutOutput = '';
63
89
  const writeSpy = vi.spyOn(process.stdout, 'write').mockImplementation((chunk) => {
64
90
  stdoutOutput += String(chunk);
@@ -83,4 +109,269 @@ describe('runCapture --json', () => {
83
109
  expect(output).toHaveProperty('confidence');
84
110
  });
85
111
  });
112
+ describe('runCapture description field routing', () => {
113
+ beforeEach(() => {
114
+ kernelCallWithSessionMock.mockReset();
115
+ mockSession = { sessionId: 'sess-test' };
116
+ });
117
+ it('passes description key to createEntry for assumptions (TEN-1258 — remap happens in Convex)', async () => {
118
+ // The CLI always sends `description` — the Convex write gate remaps it to `belief`
119
+ // for the assumptions collection (TEN-1258). Verify the CLI routes correctly and
120
+ // sends the text as description so the Convex remap can fire.
121
+ kernelCallWithSessionMock.mockImplementation((fn) => {
122
+ if (fn === 'chain.suggestLinksForCapture')
123
+ return Promise.resolve([]);
124
+ if (fn === 'chain.createEntry')
125
+ return Promise.resolve({ docId: 'doc-5', entryId: 'ASM-1' });
126
+ return Promise.resolve({});
127
+ });
128
+ const writeSpy = vi.spyOn(process.stdout, 'write').mockImplementation(() => true);
129
+ await runCapture({ text: 'ASM: users won\'t pay before seeing value' });
130
+ const createCall = kernelCallWithSessionMock.mock.calls.find(([fn]) => fn === 'chain.createEntry');
131
+ expect(createCall?.[1]).toMatchObject({
132
+ collectionSlug: 'assumptions',
133
+ name: "users won't pay before seeing value",
134
+ data: { description: "users won't pay before seeing value" },
135
+ });
136
+ writeSpy.mockRestore();
137
+ });
138
+ });
139
+ describe('runCapture classification guard', () => {
140
+ beforeEach(() => {
141
+ kernelCallWithSessionMock.mockReset();
142
+ mockSession = { sessionId: 'sess-test' };
143
+ });
144
+ it('throws CLIError when classification requires review', async () => {
145
+ kernelCallWithSessionMock.mockResolvedValueOnce(null);
146
+ await expect(runCapture({ text: 'Ambiguous capture' })).rejects.toThrow('Classification needs review before capture can continue.');
147
+ });
148
+ });
149
+ describe('runCapture date inference', () => {
150
+ beforeEach(() => {
151
+ kernelCallWithSessionMock.mockReset();
152
+ mockSession = { sessionId: 'sess-test' };
153
+ });
154
+ it('does not default date to today when no source date is present', async () => {
155
+ kernelCallWithSessionMock.mockImplementation((fn) => {
156
+ if (fn === 'chain.suggestLinksForCapture')
157
+ return Promise.resolve([]);
158
+ if (fn === 'chain.resolveCollection')
159
+ return Promise.resolve({ collection: 'decisions', confidence: 0.9, tier: 'high' });
160
+ if (fn === 'chain.createEntry')
161
+ return Promise.resolve({ docId: 'doc-3', entryId: 'DEC-10' });
162
+ return Promise.resolve({});
163
+ });
164
+ const writeSpy = vi.spyOn(process.stdout, 'write').mockImplementation(() => true);
165
+ await runCapture({ text: 'Decision without explicit source date' });
166
+ const createCall = kernelCallWithSessionMock.mock.calls.find(([fn]) => fn === 'chain.createEntry');
167
+ expect(createCall?.[1]).toMatchObject({
168
+ collectionSlug: 'decisions',
169
+ data: { description: 'Decision without explicit source date' },
170
+ });
171
+ expect((createCall?.[1]).data).not.toHaveProperty('date');
172
+ writeSpy.mockRestore();
173
+ });
174
+ it('uses inferred source date when the capture text includes one', async () => {
175
+ kernelCallWithSessionMock.mockImplementation((fn) => {
176
+ if (fn === 'chain.suggestLinksForCapture')
177
+ return Promise.resolve([]);
178
+ if (fn === 'chain.resolveCollection')
179
+ return Promise.resolve({ collection: 'decisions', confidence: 0.9, tier: 'high' });
180
+ if (fn === 'chain.createEntry')
181
+ return Promise.resolve({ docId: 'doc-4', entryId: 'DEC-11' });
182
+ return Promise.resolve({});
183
+ });
184
+ const writeSpy = vi.spyOn(process.stdout, 'write').mockImplementation(() => true);
185
+ await runCapture({ text: 'Decision from April 1, 2026 steering meeting' });
186
+ const createCall = kernelCallWithSessionMock.mock.calls.find(([fn]) => fn === 'chain.createEntry');
187
+ expect((createCall?.[1]).data.date).toBe('2026-04-01');
188
+ writeSpy.mockRestore();
189
+ });
190
+ });
191
+ // ---------------------------------------------------------------------------
192
+ // WP-319 S3: Pre-write grounding tests
193
+ // ---------------------------------------------------------------------------
194
+ describe('runCapture WP-319 grounding', () => {
195
+ const emptyGrounding = { index: 0, related: [], duplicates: [], governance: [], v: 1 };
196
+ const groundingWithDuplicate = {
197
+ index: 0,
198
+ related: [],
199
+ duplicates: [{ entryId: 'TEN-5', name: 'Similar tension', collectionSlug: 'tensions', matchType: 'possible_duplicate', overlapRatio: 0.9 }],
200
+ governance: [],
201
+ v: 1,
202
+ };
203
+ beforeEach(() => {
204
+ kernelCallWithSessionMock.mockReset();
205
+ mockSession = { sessionId: 'sess-test' };
206
+ delete process.env.PB_GROUNDING_MODE;
207
+ });
208
+ it('fires chain.suggestLinksForCapture in parallel and still calls createEntry', async () => {
209
+ // Prefix-based route skips classification; grounding fires for TEN prefix
210
+ kernelCallWithSessionMock
211
+ .mockImplementation((fn) => {
212
+ if (fn === 'chain.suggestLinksForCapture')
213
+ return Promise.resolve([emptyGrounding]);
214
+ if (fn === 'chain.createEntry')
215
+ return Promise.resolve({ docId: 'doc-1', entryId: 'TEN-720' });
216
+ return Promise.resolve({});
217
+ });
218
+ const writeSpy = vi.spyOn(process.stdout, 'write').mockImplementation(() => true);
219
+ const stderrSpy = vi.spyOn(process.stderr, 'write').mockImplementation(() => true);
220
+ await runCapture({ text: 'TEN: grounding test tension' });
221
+ expect(kernelCallWithSessionMock).toHaveBeenCalledWith('chain.suggestLinksForCapture', expect.objectContaining({
222
+ entries: expect.arrayContaining([expect.objectContaining({ name: 'grounding test tension' })]),
223
+ }));
224
+ expect(kernelCallWithSessionMock).toHaveBeenCalledWith('chain.createEntry', expect.anything());
225
+ writeSpy.mockRestore();
226
+ stderrSpy.mockRestore();
227
+ });
228
+ it('does NOT block capture when grounding call fails', async () => {
229
+ kernelCallWithSessionMock
230
+ .mockImplementation((fn) => {
231
+ if (fn === 'chain.suggestLinksForCapture')
232
+ return Promise.reject(new Error('network error'));
233
+ if (fn === 'chain.createEntry')
234
+ return Promise.resolve({ docId: 'doc-2', entryId: 'TEN-721' });
235
+ return Promise.resolve({});
236
+ });
237
+ const writeSpy = vi.spyOn(process.stdout, 'write').mockImplementation(() => true);
238
+ const stderrSpy = vi.spyOn(process.stderr, 'write').mockImplementation(() => true);
239
+ // Must not throw — grounding failure is fail-open
240
+ await expect(runCapture({ text: 'TEN: grounding failure test' })).resolves.toBeUndefined();
241
+ expect(kernelCallWithSessionMock).toHaveBeenCalledWith('chain.createEntry', expect.anything());
242
+ writeSpy.mockRestore();
243
+ stderrSpy.mockRestore();
244
+ });
245
+ it('--suggest-only does NOT call createEntry and outputs grounding data', async () => {
246
+ // In test env (non-TTY), isJsonMode() returns true → JSON path is taken.
247
+ kernelCallWithSessionMock
248
+ .mockImplementation((fn) => {
249
+ if (fn === 'chain.suggestLinksForCapture')
250
+ return Promise.resolve([groundingWithDuplicate]);
251
+ return Promise.resolve({});
252
+ });
253
+ let stdoutOutput = '';
254
+ const writeSpy = vi.spyOn(process.stdout, 'write').mockImplementation((chunk) => {
255
+ stdoutOutput += String(chunk);
256
+ return true;
257
+ });
258
+ const stderrSpy = vi.spyOn(process.stderr, 'write').mockImplementation(() => true);
259
+ await runCapture({ text: 'TEN: duplicate candidate', suggestOnly: true });
260
+ expect(kernelCallWithSessionMock).not.toHaveBeenCalledWith('chain.createEntry', expect.anything());
261
+ // JSON output contains suggestOnly flag and grounding report
262
+ const jsonLines = stdoutOutput.split('\n').filter((l) => { try {
263
+ JSON.parse(l);
264
+ return true;
265
+ }
266
+ catch {
267
+ return false;
268
+ } });
269
+ expect(jsonLines).toHaveLength(1);
270
+ const parsed = JSON.parse(jsonLines[0]);
271
+ expect(parsed.suggestOnly).toBe(true);
272
+ expect(parsed.groundingReport).toBeDefined();
273
+ writeSpy.mockRestore();
274
+ stderrSpy.mockRestore();
275
+ });
276
+ it('--suggest-only with --json outputs machine-readable object without calling createEntry', async () => {
277
+ kernelCallWithSessionMock
278
+ .mockImplementation((fn) => {
279
+ if (fn === 'chain.suggestLinksForCapture')
280
+ return Promise.resolve([emptyGrounding]);
281
+ return Promise.resolve({});
282
+ });
283
+ let stdoutOutput = '';
284
+ const writeSpy = vi.spyOn(process.stdout, 'write').mockImplementation((chunk) => {
285
+ stdoutOutput += String(chunk);
286
+ return true;
287
+ });
288
+ const stderrSpy = vi.spyOn(process.stderr, 'write').mockImplementation(() => true);
289
+ await runCapture({ text: 'TEN: dry run json', json: true, suggestOnly: true });
290
+ expect(kernelCallWithSessionMock).not.toHaveBeenCalledWith('chain.createEntry', expect.anything());
291
+ const jsonLines = stdoutOutput.split('\n').filter((l) => { try {
292
+ JSON.parse(l);
293
+ return true;
294
+ }
295
+ catch {
296
+ return false;
297
+ } });
298
+ expect(jsonLines).toHaveLength(1);
299
+ const parsed = JSON.parse(jsonLines[0]);
300
+ expect(parsed.suggestOnly).toBe(true);
301
+ writeSpy.mockRestore();
302
+ stderrSpy.mockRestore();
303
+ });
304
+ it('banner is suppressed when grounding has no matches (zero-noise floor)', async () => {
305
+ kernelCallWithSessionMock
306
+ .mockImplementation((fn) => {
307
+ if (fn === 'chain.suggestLinksForCapture')
308
+ return Promise.resolve([emptyGrounding]);
309
+ if (fn === 'chain.createEntry')
310
+ return Promise.resolve({ docId: 'doc-3', entryId: 'TEN-722' });
311
+ return Promise.resolve({});
312
+ });
313
+ const writeSpy = vi.spyOn(process.stdout, 'write').mockImplementation(() => true);
314
+ const stderrLines = [];
315
+ const stderrSpy = vi.spyOn(process.stderr, 'write').mockImplementation((chunk) => {
316
+ stderrLines.push(String(chunk));
317
+ return true;
318
+ });
319
+ await runCapture({ text: 'TEN: no matches tension' });
320
+ // No "Grounding" banner in stderr when zero matches
321
+ const groundingInStderr = stderrLines.some((l) => l.includes('Grounding'));
322
+ expect(groundingInStderr).toBe(false);
323
+ writeSpy.mockRestore();
324
+ stderrSpy.mockRestore();
325
+ });
326
+ it('PB_GROUNDING_MODE=off skips grounding call entirely', async () => {
327
+ process.env.PB_GROUNDING_MODE = 'off';
328
+ kernelCallWithSessionMock
329
+ .mockImplementation((fn) => {
330
+ if (fn === 'chain.createEntry')
331
+ return Promise.resolve({ docId: 'doc-4', entryId: 'TEN-723' });
332
+ return Promise.resolve({});
333
+ });
334
+ const writeSpy = vi.spyOn(process.stdout, 'write').mockImplementation(() => true);
335
+ const stderrSpy = vi.spyOn(process.stderr, 'write').mockImplementation(() => true);
336
+ await runCapture({ text: 'TEN: no grounding env off' });
337
+ expect(kernelCallWithSessionMock).not.toHaveBeenCalledWith('chain.suggestLinksForCapture', expect.anything());
338
+ expect(kernelCallWithSessionMock).toHaveBeenCalledWith('chain.createEntry', expect.anything());
339
+ delete process.env.PB_GROUNDING_MODE;
340
+ writeSpy.mockRestore();
341
+ stderrSpy.mockRestore();
342
+ });
343
+ it('--json output stdout is clean (no grounding on stdout)', async () => {
344
+ kernelCallWithSessionMock
345
+ .mockImplementation((fn) => {
346
+ if (fn === 'chain.suggestLinksForCapture')
347
+ return Promise.resolve([groundingWithDuplicate]);
348
+ if (fn === 'chain.createEntry')
349
+ return Promise.resolve({ docId: 'doc-5', entryId: 'TEN-724' });
350
+ return Promise.resolve({});
351
+ });
352
+ let stdoutOutput = '';
353
+ const writeSpy = vi.spyOn(process.stdout, 'write').mockImplementation((chunk) => {
354
+ stdoutOutput += String(chunk);
355
+ return true;
356
+ });
357
+ const stderrSpy = vi.spyOn(process.stderr, 'write').mockImplementation(() => true);
358
+ await runCapture({ text: 'TEN: json clean stdout', json: true });
359
+ // Stdout must be exactly one valid JSON line (jq .id must work)
360
+ const jsonLines = stdoutOutput.split('\n').filter((l) => { try {
361
+ JSON.parse(l);
362
+ return true;
363
+ }
364
+ catch {
365
+ return false;
366
+ } });
367
+ expect(jsonLines).toHaveLength(1);
368
+ const parsed = JSON.parse(jsonLines[0]);
369
+ expect(parsed.id).toBe('TEN-724');
370
+ // No grounding content in stdout
371
+ expect(stdoutOutput).not.toContain('Grounding');
372
+ expect(stdoutOutput).not.toContain('duplicate');
373
+ writeSpy.mockRestore();
374
+ stderrSpy.mockRestore();
375
+ });
376
+ });
86
377
  //# sourceMappingURL=capture.test.js.map