@productbrain/cli 0.1.0-beta.94 → 0.1.0-beta.945

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 (421) hide show
  1. package/dist/__tests__/audit.test.js +5 -0
  2. package/dist/__tests__/audit.test.js.map +1 -1
  3. package/dist/__tests__/authority-domains.test.d.ts +2 -0
  4. package/dist/__tests__/authority-domains.test.d.ts.map +1 -0
  5. package/dist/__tests__/authority-domains.test.js +48 -0
  6. package/dist/__tests__/authority-domains.test.js.map +1 -0
  7. package/dist/__tests__/canonicalRefs.vocab.test.d.ts +2 -0
  8. package/dist/__tests__/canonicalRefs.vocab.test.d.ts.map +1 -0
  9. package/dist/__tests__/canonicalRefs.vocab.test.js +251 -0
  10. package/dist/__tests__/canonicalRefs.vocab.test.js.map +1 -0
  11. package/dist/__tests__/config.test.d.ts +1 -1
  12. package/dist/__tests__/config.test.js +410 -10
  13. package/dist/__tests__/config.test.js.map +1 -1
  14. package/dist/__tests__/constants.test.js +6 -1
  15. package/dist/__tests__/constants.test.js.map +1 -1
  16. package/dist/__tests__/envelope-contract.test.js +29 -3
  17. package/dist/__tests__/envelope-contract.test.js.map +1 -1
  18. package/dist/__tests__/errors.test.js +1 -0
  19. package/dist/__tests__/errors.test.js.map +1 -1
  20. package/dist/__tests__/handshake-augment.test.d.ts +2 -0
  21. package/dist/__tests__/handshake-augment.test.d.ts.map +1 -0
  22. package/dist/__tests__/handshake-augment.test.js +423 -0
  23. package/dist/__tests__/handshake-augment.test.js.map +1 -0
  24. package/dist/__tests__/handshake-dormancy.test.d.ts +2 -0
  25. package/dist/__tests__/handshake-dormancy.test.d.ts.map +1 -0
  26. package/dist/__tests__/handshake-dormancy.test.js +207 -0
  27. package/dist/__tests__/handshake-dormancy.test.js.map +1 -0
  28. package/dist/__tests__/handshake-formatter.test.d.ts +2 -0
  29. package/dist/__tests__/handshake-formatter.test.d.ts.map +1 -0
  30. package/dist/__tests__/handshake-formatter.test.js +67 -0
  31. package/dist/__tests__/handshake-formatter.test.js.map +1 -0
  32. package/dist/__tests__/handshake-preview.test.js +688 -3
  33. package/dist/__tests__/handshake-preview.test.js.map +1 -1
  34. package/dist/__tests__/handshake.e2e.test.d.ts +2 -0
  35. package/dist/__tests__/handshake.e2e.test.d.ts.map +1 -0
  36. package/dist/__tests__/handshake.e2e.test.js +1252 -0
  37. package/dist/__tests__/handshake.e2e.test.js.map +1 -0
  38. package/dist/__tests__/handshake.test.js +611 -2
  39. package/dist/__tests__/handshake.test.js.map +1 -1
  40. package/dist/__tests__/manifest.test.js +118 -1
  41. package/dist/__tests__/manifest.test.js.map +1 -1
  42. package/dist/__tests__/onboarding-path-b.test.js +4 -4
  43. package/dist/__tests__/onboarding-path-b.test.js.map +1 -1
  44. package/dist/__tests__/orient.test.js +135 -8
  45. package/dist/__tests__/orient.test.js.map +1 -1
  46. package/dist/__tests__/perimeter.test.d.ts +2 -0
  47. package/dist/__tests__/perimeter.test.d.ts.map +1 -0
  48. package/dist/__tests__/perimeter.test.js +165 -0
  49. package/dist/__tests__/perimeter.test.js.map +1 -0
  50. package/dist/__tests__/personal-layer.test.d.ts +1 -2
  51. package/dist/__tests__/personal-layer.test.d.ts.map +1 -1
  52. package/dist/__tests__/personal-layer.test.js +12 -48
  53. package/dist/__tests__/personal-layer.test.js.map +1 -1
  54. package/dist/__tests__/profiles.test.js +153 -5
  55. package/dist/__tests__/profiles.test.js.map +1 -1
  56. package/dist/__tests__/promote.test.js +71 -2
  57. package/dist/__tests__/promote.test.js.map +1 -1
  58. package/dist/__tests__/session-state-machine.test.js +45 -1
  59. package/dist/__tests__/session-state-machine.test.js.map +1 -1
  60. package/dist/__tests__/session-switch.test.d.ts +2 -0
  61. package/dist/__tests__/session-switch.test.d.ts.map +1 -0
  62. package/dist/__tests__/session-switch.test.js +129 -0
  63. package/dist/__tests__/session-switch.test.js.map +1 -0
  64. package/dist/__tests__/setup-ingest.test.d.ts +2 -0
  65. package/dist/__tests__/setup-ingest.test.d.ts.map +1 -0
  66. package/dist/__tests__/setup-ingest.test.js +71 -0
  67. package/dist/__tests__/setup-ingest.test.js.map +1 -0
  68. package/dist/__tests__/skill-vocabulary.test.d.ts +21 -0
  69. package/dist/__tests__/skill-vocabulary.test.d.ts.map +1 -0
  70. package/dist/__tests__/skill-vocabulary.test.js +187 -0
  71. package/dist/__tests__/skill-vocabulary.test.js.map +1 -0
  72. package/dist/__tests__/update-check.test.d.ts +2 -0
  73. package/dist/__tests__/update-check.test.d.ts.map +1 -0
  74. package/dist/__tests__/update-check.test.js +56 -0
  75. package/dist/__tests__/update-check.test.js.map +1 -0
  76. package/dist/__tests__/upgrade-runner.test.d.ts +2 -0
  77. package/dist/__tests__/upgrade-runner.test.d.ts.map +1 -0
  78. package/dist/__tests__/upgrade-runner.test.js +42 -0
  79. package/dist/__tests__/upgrade-runner.test.js.map +1 -0
  80. package/dist/__tests__/vocabulary-leak.test.d.ts +39 -0
  81. package/dist/__tests__/vocabulary-leak.test.d.ts.map +1 -0
  82. package/dist/__tests__/vocabulary-leak.test.js +534 -0
  83. package/dist/__tests__/vocabulary-leak.test.js.map +1 -0
  84. package/dist/__tests__/workspace.test.js +32 -12
  85. package/dist/__tests__/workspace.test.js.map +1 -1
  86. package/dist/commands/__tests__/connect-handoff.test.d.ts +11 -0
  87. package/dist/commands/__tests__/connect-handoff.test.d.ts.map +1 -0
  88. package/dist/commands/__tests__/connect-handoff.test.js +111 -0
  89. package/dist/commands/__tests__/connect-handoff.test.js.map +1 -0
  90. package/dist/commands/__tests__/setup-detect-surfaces.test.d.ts +15 -0
  91. package/dist/commands/__tests__/setup-detect-surfaces.test.d.ts.map +1 -0
  92. package/dist/commands/__tests__/setup-detect-surfaces.test.js +149 -0
  93. package/dist/commands/__tests__/setup-detect-surfaces.test.js.map +1 -0
  94. package/dist/commands/__tests__/setup-state.test.d.ts +2 -0
  95. package/dist/commands/__tests__/setup-state.test.d.ts.map +1 -0
  96. package/dist/commands/__tests__/setup-state.test.js +194 -0
  97. package/dist/commands/__tests__/setup-state.test.js.map +1 -0
  98. package/dist/commands/admin/seed.d.ts +46 -2
  99. package/dist/commands/admin/seed.d.ts.map +1 -1
  100. package/dist/commands/admin/seed.js +475 -33
  101. package/dist/commands/admin/seed.js.map +1 -1
  102. package/dist/commands/admin/seed.test.d.ts +5 -0
  103. package/dist/commands/admin/seed.test.d.ts.map +1 -1
  104. package/dist/commands/admin/seed.test.js +67 -2
  105. package/dist/commands/admin/seed.test.js.map +1 -1
  106. package/dist/commands/admin/seedRegistryEntries.generated.d.ts +14 -0
  107. package/dist/commands/admin/seedRegistryEntries.generated.d.ts.map +1 -0
  108. package/dist/commands/admin/seedRegistryEntries.generated.js +117 -0
  109. package/dist/commands/admin/seedRegistryEntries.generated.js.map +1 -0
  110. package/dist/commands/admin/seedRegistryEntries.test.d.ts +11 -0
  111. package/dist/commands/admin/seedRegistryEntries.test.d.ts.map +1 -0
  112. package/dist/commands/admin/seedRegistryEntries.test.js +67 -0
  113. package/dist/commands/admin/seedRegistryEntries.test.js.map +1 -0
  114. package/dist/commands/audit.d.ts.map +1 -1
  115. package/dist/commands/audit.js +30 -3
  116. package/dist/commands/audit.js.map +1 -1
  117. package/dist/commands/authority-domains.d.ts +146 -0
  118. package/dist/commands/authority-domains.d.ts.map +1 -0
  119. package/dist/commands/authority-domains.js +282 -0
  120. package/dist/commands/authority-domains.js.map +1 -0
  121. package/dist/commands/capture.d.ts.map +1 -1
  122. package/dist/commands/capture.js +3 -2
  123. package/dist/commands/capture.js.map +1 -1
  124. package/dist/commands/codex-prep.d.ts +1 -0
  125. package/dist/commands/codex-prep.d.ts.map +1 -1
  126. package/dist/commands/codex-prep.js +10 -7
  127. package/dist/commands/codex-prep.js.map +1 -1
  128. package/dist/commands/connect-config.test.d.ts +2 -0
  129. package/dist/commands/connect-config.test.d.ts.map +1 -0
  130. package/dist/commands/connect-config.test.js +44 -0
  131. package/dist/commands/connect-config.test.js.map +1 -0
  132. package/dist/commands/connect-context.d.ts +45 -0
  133. package/dist/commands/connect-context.d.ts.map +1 -0
  134. package/dist/commands/connect-context.js +64 -0
  135. package/dist/commands/connect-context.js.map +1 -0
  136. package/dist/commands/connect-context.test.d.ts +2 -0
  137. package/dist/commands/connect-context.test.d.ts.map +1 -0
  138. package/dist/commands/connect-context.test.js +110 -0
  139. package/dist/commands/connect-context.test.js.map +1 -0
  140. package/dist/commands/connect-handoff.d.ts +51 -0
  141. package/dist/commands/connect-handoff.d.ts.map +1 -0
  142. package/dist/commands/connect-handoff.js +70 -0
  143. package/dist/commands/connect-handoff.js.map +1 -0
  144. package/dist/commands/connect-integration.test.js +29 -12
  145. package/dist/commands/connect-integration.test.js.map +1 -1
  146. package/dist/commands/connect-screens.d.ts +6 -4
  147. package/dist/commands/connect-screens.d.ts.map +1 -1
  148. package/dist/commands/connect-screens.js +30 -19
  149. package/dist/commands/connect-screens.js.map +1 -1
  150. package/dist/commands/connect.d.ts +21 -6
  151. package/dist/commands/connect.d.ts.map +1 -1
  152. package/dist/commands/connect.js +78 -51
  153. package/dist/commands/connect.js.map +1 -1
  154. package/dist/commands/connect.test.js +64 -1
  155. package/dist/commands/connect.test.js.map +1 -1
  156. package/dist/commands/doctor.d.ts.map +1 -1
  157. package/dist/commands/doctor.js +68 -3
  158. package/dist/commands/doctor.js.map +1 -1
  159. package/dist/commands/doctor.test.js +131 -0
  160. package/dist/commands/doctor.test.js.map +1 -1
  161. package/dist/commands/handshake.d.ts +194 -2
  162. package/dist/commands/handshake.d.ts.map +1 -1
  163. package/dist/commands/handshake.js +1724 -45
  164. package/dist/commands/handshake.js.map +1 -1
  165. package/dist/commands/method.d.ts.map +1 -1
  166. package/dist/commands/method.js +3 -0
  167. package/dist/commands/method.js.map +1 -1
  168. package/dist/commands/migrate-setup.d.ts +18 -0
  169. package/dist/commands/migrate-setup.d.ts.map +1 -0
  170. package/dist/commands/migrate-setup.js +198 -0
  171. package/dist/commands/migrate-setup.js.map +1 -0
  172. package/dist/commands/orient.d.ts +15 -2
  173. package/dist/commands/orient.d.ts.map +1 -1
  174. package/dist/commands/orient.js +86 -4
  175. package/dist/commands/orient.js.map +1 -1
  176. package/dist/commands/profile.d.ts +11 -1
  177. package/dist/commands/profile.d.ts.map +1 -1
  178. package/dist/commands/profile.js +109 -26
  179. package/dist/commands/profile.js.map +1 -1
  180. package/dist/commands/promote.d.ts.map +1 -1
  181. package/dist/commands/promote.js +25 -2
  182. package/dist/commands/promote.js.map +1 -1
  183. package/dist/commands/relate.d.ts.map +1 -1
  184. package/dist/commands/relate.js +13 -0
  185. package/dist/commands/relate.js.map +1 -1
  186. package/dist/commands/session.d.ts.map +1 -1
  187. package/dist/commands/session.js +55 -18
  188. package/dist/commands/session.js.map +1 -1
  189. package/dist/commands/setup-audit.d.ts +59 -0
  190. package/dist/commands/setup-audit.d.ts.map +1 -0
  191. package/dist/commands/setup-audit.js +250 -0
  192. package/dist/commands/setup-audit.js.map +1 -0
  193. package/dist/commands/setup-detect-surfaces.d.ts +38 -0
  194. package/dist/commands/setup-detect-surfaces.d.ts.map +1 -0
  195. package/dist/commands/setup-detect-surfaces.js +76 -0
  196. package/dist/commands/setup-detect-surfaces.js.map +1 -0
  197. package/dist/commands/setup-ingest.d.ts +17 -0
  198. package/dist/commands/setup-ingest.d.ts.map +1 -0
  199. package/dist/commands/setup-ingest.js +226 -0
  200. package/dist/commands/setup-ingest.js.map +1 -0
  201. package/dist/commands/setup-state.d.ts +42 -0
  202. package/dist/commands/setup-state.d.ts.map +1 -0
  203. package/dist/commands/setup-state.js +93 -0
  204. package/dist/commands/setup-state.js.map +1 -0
  205. package/dist/commands/setup.d.ts +17 -9
  206. package/dist/commands/setup.d.ts.map +1 -1
  207. package/dist/commands/setup.js +52 -131
  208. package/dist/commands/setup.js.map +1 -1
  209. package/dist/commands/upgrade.d.ts +5 -0
  210. package/dist/commands/upgrade.d.ts.map +1 -0
  211. package/dist/commands/upgrade.js +89 -0
  212. package/dist/commands/upgrade.js.map +1 -0
  213. package/dist/commands/whoami.d.ts +12 -0
  214. package/dist/commands/whoami.d.ts.map +1 -0
  215. package/dist/commands/whoami.js +70 -0
  216. package/dist/commands/whoami.js.map +1 -0
  217. package/dist/commands/whoami.test.d.ts +2 -0
  218. package/dist/commands/whoami.test.d.ts.map +1 -0
  219. package/dist/commands/whoami.test.js +50 -0
  220. package/dist/commands/whoami.test.js.map +1 -0
  221. package/dist/commands/workspace.d.ts +74 -2
  222. package/dist/commands/workspace.d.ts.map +1 -1
  223. package/dist/commands/workspace.js +26 -2
  224. package/dist/commands/workspace.js.map +1 -1
  225. package/dist/formatters/audit.d.ts +6 -0
  226. package/dist/formatters/audit.d.ts.map +1 -1
  227. package/dist/formatters/audit.js.map +1 -1
  228. package/dist/formatters/handshake.d.ts +19 -3
  229. package/dist/formatters/handshake.d.ts.map +1 -1
  230. package/dist/formatters/handshake.js +48 -13
  231. package/dist/formatters/handshake.js.map +1 -1
  232. package/dist/formatters/orient.d.ts +50 -4
  233. package/dist/formatters/orient.d.ts.map +1 -1
  234. package/dist/formatters/orient.js +64 -16
  235. package/dist/formatters/orient.js.map +1 -1
  236. package/dist/formatters/session.js +1 -1
  237. package/dist/formatters/session.js.map +1 -1
  238. package/dist/generators/adapters.js +2 -2
  239. package/dist/generators/boundary-manifest.d.ts +29 -0
  240. package/dist/generators/boundary-manifest.d.ts.map +1 -0
  241. package/dist/generators/boundary-manifest.js +183 -0
  242. package/dist/generators/boundary-manifest.js.map +1 -0
  243. package/dist/generators/boundary-manifest.test.d.ts +2 -0
  244. package/dist/generators/boundary-manifest.test.d.ts.map +1 -0
  245. package/dist/generators/boundary-manifest.test.js +91 -0
  246. package/dist/generators/boundary-manifest.test.js.map +1 -0
  247. package/dist/generators/context-md.js +6 -6
  248. package/dist/generators/context-md.js.map +1 -1
  249. package/dist/generators/manifest.d.ts +78 -0
  250. package/dist/generators/manifest.d.ts.map +1 -1
  251. package/dist/generators/manifest.js +125 -14
  252. package/dist/generators/manifest.js.map +1 -1
  253. package/dist/generators/portable-knowledge.d.ts +6 -12
  254. package/dist/generators/portable-knowledge.d.ts.map +1 -1
  255. package/dist/generators/portable-knowledge.js +2 -19
  256. package/dist/generators/portable-knowledge.js.map +1 -1
  257. package/dist/generators/region-projections.d.ts +18 -0
  258. package/dist/generators/region-projections.d.ts.map +1 -0
  259. package/dist/generators/region-projections.js +49 -0
  260. package/dist/generators/region-projections.js.map +1 -0
  261. package/dist/generators/region-projections.test.d.ts +2 -0
  262. package/dist/generators/region-projections.test.d.ts.map +1 -0
  263. package/dist/generators/region-projections.test.js +63 -0
  264. package/dist/generators/region-projections.test.js.map +1 -0
  265. package/dist/generators/region.d.ts +24 -0
  266. package/dist/generators/region.d.ts.map +1 -0
  267. package/dist/generators/region.js +87 -0
  268. package/dist/generators/region.js.map +1 -0
  269. package/dist/generators/region.test.d.ts +2 -0
  270. package/dist/generators/region.test.d.ts.map +1 -0
  271. package/dist/generators/region.test.js +126 -0
  272. package/dist/generators/region.test.js.map +1 -0
  273. package/dist/generators/surface-profiles.d.ts +1 -2
  274. package/dist/generators/surface-profiles.d.ts.map +1 -1
  275. package/dist/generators/surface-profiles.js.map +1 -1
  276. package/dist/index.js +242 -26
  277. package/dist/index.js.map +1 -1
  278. package/dist/lib/activation.d.ts.map +1 -1
  279. package/dist/lib/activation.js +3 -3
  280. package/dist/lib/activation.js.map +1 -1
  281. package/dist/lib/activation.test.js +3 -3
  282. package/dist/lib/activation.test.js.map +1 -1
  283. package/dist/lib/canonicalRefs.d.ts +141 -0
  284. package/dist/lib/canonicalRefs.d.ts.map +1 -0
  285. package/dist/lib/canonicalRefs.js +150 -0
  286. package/dist/lib/canonicalRefs.js.map +1 -0
  287. package/dist/lib/client.d.ts.map +1 -1
  288. package/dist/lib/client.js +14 -4
  289. package/dist/lib/client.js.map +1 -1
  290. package/dist/lib/config.d.ts +98 -9
  291. package/dist/lib/config.d.ts.map +1 -1
  292. package/dist/lib/config.js +231 -44
  293. package/dist/lib/config.js.map +1 -1
  294. package/dist/lib/connectKeyLabel.d.ts +9 -0
  295. package/dist/lib/connectKeyLabel.d.ts.map +1 -0
  296. package/dist/lib/connectKeyLabel.js +12 -0
  297. package/dist/lib/connectKeyLabel.js.map +1 -0
  298. package/dist/lib/constants.d.ts +2 -0
  299. package/dist/lib/constants.d.ts.map +1 -1
  300. package/dist/lib/constants.js +2 -0
  301. package/dist/lib/constants.js.map +1 -1
  302. package/dist/lib/errors.d.ts +3 -0
  303. package/dist/lib/errors.d.ts.map +1 -1
  304. package/dist/lib/errors.js +3 -0
  305. package/dist/lib/errors.js.map +1 -1
  306. package/dist/lib/normalizeMaterializedFilename.d.ts +28 -0
  307. package/dist/lib/normalizeMaterializedFilename.d.ts.map +1 -0
  308. package/dist/lib/normalizeMaterializedFilename.js +56 -0
  309. package/dist/lib/normalizeMaterializedFilename.js.map +1 -0
  310. package/dist/lib/normalizeMaterializedFilename.test.d.ts +16 -0
  311. package/dist/lib/normalizeMaterializedFilename.test.d.ts.map +1 -0
  312. package/dist/lib/normalizeMaterializedFilename.test.js +90 -0
  313. package/dist/lib/normalizeMaterializedFilename.test.js.map +1 -0
  314. package/dist/lib/onboarding-path-b.d.ts.map +1 -1
  315. package/dist/lib/onboarding-path-b.js +0 -1
  316. package/dist/lib/onboarding-path-b.js.map +1 -1
  317. package/dist/lib/onboarding-shared.d.ts +0 -1
  318. package/dist/lib/onboarding-shared.d.ts.map +1 -1
  319. package/dist/lib/onboarding-shared.js +1 -17
  320. package/dist/lib/onboarding-shared.js.map +1 -1
  321. package/dist/lib/profiles.d.ts +3 -1
  322. package/dist/lib/profiles.d.ts.map +1 -1
  323. package/dist/lib/profiles.js +9 -6
  324. package/dist/lib/profiles.js.map +1 -1
  325. package/dist/lib/session.d.ts +10 -0
  326. package/dist/lib/session.d.ts.map +1 -1
  327. package/dist/lib/session.js +14 -0
  328. package/dist/lib/session.js.map +1 -1
  329. package/dist/lib/update-check.d.ts +20 -0
  330. package/dist/lib/update-check.d.ts.map +1 -1
  331. package/dist/lib/update-check.js +122 -21
  332. package/dist/lib/update-check.js.map +1 -1
  333. package/dist/lib/upgrade-runner.d.ts +21 -0
  334. package/dist/lib/upgrade-runner.d.ts.map +1 -0
  335. package/dist/lib/upgrade-runner.js +109 -0
  336. package/dist/lib/upgrade-runner.js.map +1 -0
  337. package/dist/lib/workspaceVocabCache.d.ts +60 -0
  338. package/dist/lib/workspaceVocabCache.d.ts.map +1 -0
  339. package/dist/lib/workspaceVocabCache.js +98 -0
  340. package/dist/lib/workspaceVocabCache.js.map +1 -0
  341. package/dist/setup/__tests__/coach-traces.test.d.ts +2 -0
  342. package/dist/setup/__tests__/coach-traces.test.d.ts.map +1 -0
  343. package/dist/setup/__tests__/coach-traces.test.js +189 -0
  344. package/dist/setup/__tests__/coach-traces.test.js.map +1 -0
  345. package/dist/setup/__tests__/setup-commands.test.d.ts +2 -0
  346. package/dist/setup/__tests__/setup-commands.test.d.ts.map +1 -0
  347. package/dist/setup/__tests__/setup-commands.test.js +177 -0
  348. package/dist/setup/__tests__/setup-commands.test.js.map +1 -0
  349. package/dist/setup/__tests__/state-machine.test.d.ts +2 -0
  350. package/dist/setup/__tests__/state-machine.test.d.ts.map +1 -0
  351. package/dist/setup/__tests__/state-machine.test.js +341 -0
  352. package/dist/setup/__tests__/state-machine.test.js.map +1 -0
  353. package/dist/setup/detect-surfaces.d.ts +21 -0
  354. package/dist/setup/detect-surfaces.d.ts.map +1 -0
  355. package/dist/setup/detect-surfaces.js +39 -0
  356. package/dist/setup/detect-surfaces.js.map +1 -0
  357. package/dist/setup/manifest-writer.d.ts +17 -0
  358. package/dist/setup/manifest-writer.d.ts.map +1 -0
  359. package/dist/setup/manifest-writer.js +153 -0
  360. package/dist/setup/manifest-writer.js.map +1 -0
  361. package/dist/setup/perimeter.d.ts +72 -0
  362. package/dist/setup/perimeter.d.ts.map +1 -0
  363. package/dist/setup/perimeter.js +128 -0
  364. package/dist/setup/perimeter.js.map +1 -0
  365. package/dist/setup/state-machine.d.ts +67 -0
  366. package/dist/setup/state-machine.d.ts.map +1 -0
  367. package/dist/setup/state-machine.js +124 -0
  368. package/dist/setup/state-machine.js.map +1 -0
  369. package/dist/surfaces/__tests__/adapter.test.d.ts +2 -0
  370. package/dist/surfaces/__tests__/adapter.test.d.ts.map +1 -0
  371. package/dist/surfaces/__tests__/adapter.test.js +90 -0
  372. package/dist/surfaces/__tests__/adapter.test.js.map +1 -0
  373. package/dist/surfaces/__tests__/pb-setup-passthrough.test.d.ts +2 -0
  374. package/dist/surfaces/__tests__/pb-setup-passthrough.test.d.ts.map +1 -0
  375. package/dist/surfaces/__tests__/pb-setup-passthrough.test.js +132 -0
  376. package/dist/surfaces/__tests__/pb-setup-passthrough.test.js.map +1 -0
  377. package/dist/surfaces/__tests__/telemetry.test.d.ts +2 -0
  378. package/dist/surfaces/__tests__/telemetry.test.d.ts.map +1 -0
  379. package/dist/surfaces/__tests__/telemetry.test.js +55 -0
  380. package/dist/surfaces/__tests__/telemetry.test.js.map +1 -0
  381. package/dist/surfaces/adapter.d.ts +70 -0
  382. package/dist/surfaces/adapter.d.ts.map +1 -0
  383. package/dist/surfaces/adapter.js +2 -0
  384. package/dist/surfaces/adapter.js.map +1 -0
  385. package/dist/surfaces/adapters/claude.d.ts +3 -0
  386. package/dist/surfaces/adapters/claude.d.ts.map +1 -0
  387. package/dist/surfaces/adapters/claude.js +67 -0
  388. package/dist/surfaces/adapters/claude.js.map +1 -0
  389. package/dist/surfaces/adapters/codex.d.ts +3 -0
  390. package/dist/surfaces/adapters/codex.d.ts.map +1 -0
  391. package/dist/surfaces/adapters/codex.js +61 -0
  392. package/dist/surfaces/adapters/codex.js.map +1 -0
  393. package/dist/surfaces/adapters/copilot.d.ts +3 -0
  394. package/dist/surfaces/adapters/copilot.d.ts.map +1 -0
  395. package/dist/surfaces/adapters/copilot.js +59 -0
  396. package/dist/surfaces/adapters/copilot.js.map +1 -0
  397. package/dist/surfaces/adapters/cursor.d.ts +3 -0
  398. package/dist/surfaces/adapters/cursor.d.ts.map +1 -0
  399. package/dist/surfaces/adapters/cursor.js +78 -0
  400. package/dist/surfaces/adapters/cursor.js.map +1 -0
  401. package/dist/surfaces/registry.d.ts +58 -2
  402. package/dist/surfaces/registry.d.ts.map +1 -1
  403. package/dist/surfaces/registry.js +82 -7
  404. package/dist/surfaces/registry.js.map +1 -1
  405. package/dist/surfaces/telemetry.d.ts +17 -0
  406. package/dist/surfaces/telemetry.d.ts.map +1 -0
  407. package/dist/surfaces/telemetry.js +31 -0
  408. package/dist/surfaces/telemetry.js.map +1 -0
  409. package/package.json +3 -1
  410. package/dist/__tests__/setup.test.d.ts +0 -2
  411. package/dist/__tests__/setup.test.d.ts.map +0 -1
  412. package/dist/__tests__/setup.test.js +0 -141
  413. package/dist/__tests__/setup.test.js.map +0 -1
  414. package/dist/generators/__tests__/surface-profiles.test.d.ts +0 -2
  415. package/dist/generators/__tests__/surface-profiles.test.d.ts.map +0 -1
  416. package/dist/generators/__tests__/surface-profiles.test.js +0 -89
  417. package/dist/generators/__tests__/surface-profiles.test.js.map +0 -1
  418. package/dist/lib/onboarding-phases.d.ts +0 -9
  419. package/dist/lib/onboarding-phases.d.ts.map +0 -1
  420. package/dist/lib/onboarding-phases.js +0 -120
  421. package/dist/lib/onboarding-phases.js.map +0 -1
@@ -2,12 +2,19 @@
2
2
  * pb handshake --init — unit tests.
3
3
  * DEC-271: two-file split (team vs personal). DEC-272: permission whitelist.
4
4
  * TEN-712: hooks must fail silently (|| true suffix).
5
+ *
6
+ * WP-379 S3b: probeStarterSetupSeeded + pollUntilSeedsReady tests.
5
7
  */
6
8
  import { beforeEach, afterEach, describe, expect, it, vi } from 'vitest';
7
9
  import { join } from 'path';
10
+ // Import the mocked fs functions so we can use vi.mocked() to change their implementations per-test.
11
+ // vi.mock('fs', ...) below hoists this mock, so these imports resolve to the mocked versions.
12
+ import { readdirSync as readdirSyncMock } from 'fs';
8
13
  // vi.mock calls are hoisted — use vi.hoisted() for constants referenced inside factories.
9
- const { vfs, MOCK_CWD, MOCK_HOME } = vi.hoisted(() => ({
14
+ const { vfs, vfsMtimes, MOCK_CWD, MOCK_HOME } = vi.hoisted(() => ({
10
15
  vfs: {},
16
+ // WP-379 S5b: mtime map used by statSync mock for case-collision ordering tests.
17
+ vfsMtimes: {},
11
18
  MOCK_CWD: '/tmp/pb-test-cwd',
12
19
  MOCK_HOME: '/tmp/pb-test-home',
13
20
  }));
@@ -16,12 +23,30 @@ vi.mock('fs', () => ({
16
23
  writeFileSync: vi.fn((path, content) => {
17
24
  vfs[path] = content;
18
25
  }),
26
+ appendFileSync: vi.fn((path, content) => {
27
+ vfs[path] = (vfs[path] ?? '') + content;
28
+ }),
19
29
  existsSync: vi.fn((path) => path in vfs),
20
30
  readFileSync: vi.fn((path, _enc) => {
21
31
  if (path in vfs)
22
32
  return vfs[path];
23
33
  throw Object.assign(new Error(`ENOENT: no such file '${path}'`), { code: 'ENOENT' });
24
34
  }),
35
+ readdirSync: vi.fn((_dir) => []),
36
+ copyFileSync: vi.fn(),
37
+ // WP-379 S5b: unlinkSync + statSync for orphan unlink + case-collision resolution.
38
+ unlinkSync: vi.fn((path) => {
39
+ delete vfs[path];
40
+ }),
41
+ statSync: vi.fn((path) => {
42
+ if (path in vfs) {
43
+ // Return a fake stat with an incrementing mtime based on path hash (deterministic).
44
+ // Tests that need specific mtime ordering should set vfs[path] before calling.
45
+ const mtimeMs = vfsMtimes[path] ?? 1000;
46
+ return { mtimeMs };
47
+ }
48
+ throw Object.assign(new Error(`ENOENT: stat '${path}'`), { code: 'ENOENT' });
49
+ }),
25
50
  }));
26
51
  vi.mock('os', () => ({
27
52
  homedir: vi.fn(() => MOCK_HOME),
@@ -34,7 +59,9 @@ vi.mock('../lib/prompts.js', () => ({
34
59
  password: () => Promise.resolve(''),
35
60
  isInteractive: () => true,
36
61
  }));
37
- import { runHandshakeInit, normalizeHandshakeContentForComparison } from '../commands/handshake.js';
62
+ import { runHandshakeInit, normalizeHandshakeContentForComparison, DORMANT_MARKER, writeDormantMarkerToFile, resolveProjectionCollision, classifyDriftBucket } from '../commands/handshake.js';
63
+ import { MARKER } from '../generators/adapters.js';
64
+ import { createHash } from 'crypto';
38
65
  const TEAM_PATH = join(MOCK_CWD, '.claude', 'settings.json');
39
66
  const PERSONAL_PATH = join(MOCK_HOME, '.claude', 'settings.json');
40
67
  describe('runHandshakeInit', () => {
@@ -193,4 +220,586 @@ describe('normalizeHandshakeContentForComparison', () => {
193
220
  expect(normalizeHandshakeContentForComparison(a)).not.toBe(normalizeHandshakeContentForComparison(b));
194
221
  });
195
222
  });
223
+ // ── WP-379 S3b: probeStarterSetupSeeded + pollUntilSeedsReady ─────────────────
224
+ //
225
+ // These tests use vi.doMock (not vi.mock) so they can inject different
226
+ // kernelCall implementations per test without affecting the --init tests above.
227
+ // Each test creates its own isolated module scope via vi.resetModules().
228
+ describe('probeStarterSetupSeeded (WP-379 S3b)', () => {
229
+ beforeEach(() => {
230
+ vi.resetModules();
231
+ });
232
+ afterEach(() => {
233
+ vi.restoreAllMocks();
234
+ });
235
+ it('returns seeds-ready when health.starterSetupSeeded is true', async () => {
236
+ vi.doMock('../lib/client.js', () => ({
237
+ kernelCall: vi.fn().mockResolvedValue({
238
+ starterSetupSeeded: true,
239
+ gaps: [],
240
+ status: 'healthy',
241
+ }),
242
+ }));
243
+ const { probeStarterSetupSeeded } = await import('../commands/handshake.js');
244
+ const result = await probeStarterSetupSeeded();
245
+ expect(result.status).toBe('seeds-ready');
246
+ });
247
+ it('returns seeds-pending with starter-setup-missing gap when starterSetupSeeded is false', async () => {
248
+ vi.doMock('../lib/client.js', () => ({
249
+ kernelCall: vi.fn().mockResolvedValue({
250
+ starterSetupSeeded: false,
251
+ gaps: [
252
+ {
253
+ kind: 'starter-setup-missing',
254
+ severity: 'error',
255
+ message: 'Starter setup seeds are still running.',
256
+ },
257
+ ],
258
+ status: 'incomplete',
259
+ }),
260
+ }));
261
+ const { probeStarterSetupSeeded } = await import('../commands/handshake.js');
262
+ const result = await probeStarterSetupSeeded();
263
+ expect(result.status).toBe('seeds-pending');
264
+ if (result.status === 'seeds-pending') {
265
+ expect(result.gaps.length).toBeGreaterThan(0);
266
+ expect(result.gaps[0]?.kind).toBe('starter-setup-missing');
267
+ }
268
+ });
269
+ it('returns probe-failed when kernelCall throws', async () => {
270
+ vi.doMock('../lib/client.js', () => ({
271
+ kernelCall: vi.fn().mockRejectedValue(new Error('Network unreachable')),
272
+ }));
273
+ const { probeStarterSetupSeeded } = await import('../commands/handshake.js');
274
+ const result = await probeStarterSetupSeeded();
275
+ expect(result.status).toBe('probe-failed');
276
+ if (result.status === 'probe-failed') {
277
+ expect(result.error).toContain('Network unreachable');
278
+ }
279
+ });
280
+ it('synthesizes a default starter-setup-missing gap when health.gaps is empty but starterSetupSeeded is false', async () => {
281
+ vi.doMock('../lib/client.js', () => ({
282
+ kernelCall: vi.fn().mockResolvedValue({
283
+ starterSetupSeeded: false,
284
+ gaps: [], // no gap details from server
285
+ status: 'incomplete',
286
+ }),
287
+ }));
288
+ const { probeStarterSetupSeeded } = await import('../commands/handshake.js');
289
+ const result = await probeStarterSetupSeeded();
290
+ expect(result.status).toBe('seeds-pending');
291
+ if (result.status === 'seeds-pending') {
292
+ expect(result.gaps.length).toBe(1);
293
+ expect(result.gaps[0]?.kind).toBe('starter-setup-missing');
294
+ }
295
+ });
296
+ });
297
+ describe('pollUntilSeedsReady (WP-379 S3b)', () => {
298
+ beforeEach(() => {
299
+ vi.resetModules();
300
+ vi.useFakeTimers();
301
+ });
302
+ afterEach(() => {
303
+ vi.restoreAllMocks();
304
+ vi.useRealTimers();
305
+ });
306
+ it('returns seeds-ready immediately when first probe succeeds', async () => {
307
+ vi.doMock('../lib/client.js', () => ({
308
+ kernelCall: vi.fn().mockResolvedValue({
309
+ starterSetupSeeded: true,
310
+ gaps: [],
311
+ status: 'healthy',
312
+ }),
313
+ }));
314
+ const { pollUntilSeedsReady } = await import('../commands/handshake.js');
315
+ const resultPromise = pollUntilSeedsReady();
316
+ // Advance timers past any initial delays
317
+ await vi.runAllTimersAsync();
318
+ const result = await resultPromise;
319
+ expect(result.status).toBe('seeds-ready');
320
+ });
321
+ it('returns seeds-pending after MAX_POLLS when seeds never become ready', async () => {
322
+ // kernelCall always returns starterSetupSeeded: false
323
+ vi.doMock('../lib/client.js', () => ({
324
+ kernelCall: vi.fn().mockResolvedValue({
325
+ starterSetupSeeded: false,
326
+ gaps: [
327
+ {
328
+ kind: 'starter-setup-missing',
329
+ severity: 'error',
330
+ message: 'Starter setup seeds are still running.',
331
+ },
332
+ ],
333
+ status: 'incomplete',
334
+ }),
335
+ }));
336
+ const { pollUntilSeedsReady } = await import('../commands/handshake.js');
337
+ const resultPromise = pollUntilSeedsReady();
338
+ await vi.runAllTimersAsync();
339
+ const result = await resultPromise;
340
+ expect(result.status).toBe('seeds-pending');
341
+ if (result.status === 'seeds-pending') {
342
+ expect(result.gaps[0]?.kind).toBe('starter-setup-missing');
343
+ }
344
+ });
345
+ it('stops polling immediately on probe-failed (no retry on network error)', async () => {
346
+ const kernelCallMock = vi.fn().mockRejectedValue(new Error('auth failed'));
347
+ vi.doMock('../lib/client.js', () => ({
348
+ kernelCall: kernelCallMock,
349
+ }));
350
+ const { pollUntilSeedsReady } = await import('../commands/handshake.js');
351
+ const resultPromise = pollUntilSeedsReady();
352
+ await vi.runAllTimersAsync();
353
+ const result = await resultPromise;
354
+ expect(result.status).toBe('probe-failed');
355
+ // Should have called exactly once — no retry on failure
356
+ expect(kernelCallMock).toHaveBeenCalledTimes(1);
357
+ });
358
+ });
359
+ // ── WP-379 S4: DORMANT_MARKER constant + writeDormantMarkerToFile ─────────────
360
+ describe('DORMANT_MARKER (WP-379 S4)', () => {
361
+ it('is the expected HTML comment string', () => {
362
+ expect(DORMANT_MARKER).toBe('<!-- pb-status: dormant -->');
363
+ });
364
+ it('does not contain newlines (clean single-line marker)', () => {
365
+ expect(DORMANT_MARKER).not.toContain('\n');
366
+ });
367
+ });
368
+ describe('writeDormantMarkerToFile (WP-379 S4)', () => {
369
+ // Use the hoisted vfs + mocked fs from the top of this file.
370
+ // vfs is already set up by vi.hoisted() and vi.mock('fs', ...) above.
371
+ beforeEach(() => {
372
+ Object.keys(vfs).forEach((k) => delete vfs[k]);
373
+ vi.clearAllMocks();
374
+ });
375
+ const AUTO_GEN_FILE = join(MOCK_CWD, '.claude', 'rules', 'test-rule.md');
376
+ it('returns "skipped" when the file does not exist', () => {
377
+ // File is not in vfs
378
+ const result = writeDormantMarkerToFile(AUTO_GEN_FILE);
379
+ expect(result).toBe('skipped');
380
+ });
381
+ it('returns "skipped" when the file exists but has no auto-gen MARKER', () => {
382
+ vfs[AUTO_GEN_FILE] = '# Manual rule\nSome content.\n';
383
+ const result = writeDormantMarkerToFile(AUTO_GEN_FILE);
384
+ expect(result).toBe('skipped');
385
+ // File must be unchanged (no append)
386
+ expect(vfs[AUTO_GEN_FILE]).toBe('# Manual rule\nSome content.\n');
387
+ });
388
+ it('appends DORMANT_MARKER and returns "written" for an auto-gen file without the marker', () => {
389
+ // Simulate a previously-projected file: has the auto-gen MARKER, no dormant marker yet.
390
+ vfs[AUTO_GEN_FILE] = `${MARKER}\n# Test Rule\nContent.\n`;
391
+ const result = writeDormantMarkerToFile(AUTO_GEN_FILE);
392
+ expect(result).toBe('written');
393
+ expect(vfs[AUTO_GEN_FILE]).toContain(DORMANT_MARKER);
394
+ // Original content preserved — marker is appended, not a replacement.
395
+ expect(vfs[AUTO_GEN_FILE]).toContain('# Test Rule');
396
+ expect(vfs[AUTO_GEN_FILE]).toContain(MARKER);
397
+ });
398
+ it('returns "already-dormant" when DORMANT_MARKER is already present (idempotent)', () => {
399
+ // File already has the dormant marker from a previous handshake.
400
+ vfs[AUTO_GEN_FILE] = `${MARKER}\n# Test Rule\nContent.\n\n${DORMANT_MARKER}\n`;
401
+ const result = writeDormantMarkerToFile(AUTO_GEN_FILE);
402
+ expect(result).toBe('already-dormant');
403
+ // File must be unchanged — no second append.
404
+ const occurrences = (vfs[AUTO_GEN_FILE].match(new RegExp(DORMANT_MARKER.replace(/[<>!-]/g, '\\$&'), 'g')) ?? []).length;
405
+ expect(occurrences).toBe(1);
406
+ });
407
+ it('second call on a dormant-marked file produces no duplicate markers (no drift TEN semantics)', () => {
408
+ // Simulate: first handshake wrote the dormant marker, second handshake is called again.
409
+ vfs[AUTO_GEN_FILE] = `${MARKER}\n# Test Rule\n\n${DORMANT_MARKER}\n`;
410
+ writeDormantMarkerToFile(AUTO_GEN_FILE); // first call — already-dormant
411
+ writeDormantMarkerToFile(AUTO_GEN_FILE); // second call — still already-dormant
412
+ const occurrences = (vfs[AUTO_GEN_FILE].match(new RegExp(DORMANT_MARKER.replace(/[<>!-]/g, '\\$&'), 'g')) ?? []).length;
413
+ expect(occurrences).toBe(1);
414
+ });
415
+ });
416
+ // ── WP-379 S5b: resolveProjectionCollision ─────────────────────────────────────
417
+ //
418
+ // Tests for the marker-scoped orphan unlink + case-collision resolution.
419
+ // All file system operations go through the vfs mock defined at the top of this file.
420
+ describe('resolveProjectionCollision (WP-379 S5b)', () => {
421
+ const noop = () => { };
422
+ beforeEach(() => {
423
+ // Clear vfs and mtimes before each test.
424
+ Object.keys(vfs).forEach((k) => delete vfs[k]);
425
+ Object.keys(vfsMtimes).forEach((k) => delete vfsMtimes[k]);
426
+ vi.clearAllMocks();
427
+ });
428
+ // Helper: write a file to vfs with the auto-gen MARKER.
429
+ // Also marks the parent directory as existing in vfs so existsSync(dir) returns true.
430
+ function writeMarkedFile(path, extra = '') {
431
+ const dir = path.substring(0, path.lastIndexOf('/'));
432
+ vfs[dir] = ''; // ensure existsSync(dir) returns true
433
+ vfs[path] = `${MARKER}\n# Rule\n${extra}`;
434
+ }
435
+ // Helper: write a file without auto-gen MARKER (user-forked).
436
+ // Also marks the parent directory as existing in vfs so existsSync(dir) returns true.
437
+ function writeUserFile(path) {
438
+ const dir = path.substring(0, path.lastIndexOf('/'));
439
+ vfs[dir] = ''; // ensure existsSync(dir) returns true
440
+ vfs[path] = '# My user rule — no marker here\n';
441
+ }
442
+ // ── Test 1: Linux case-collision — exact match wins ────────────────────────
443
+ it('case-collision: exact-match file survives, case-variant with marker is unlinked', () => {
444
+ const dir = `${MOCK_CWD}/.cursor/rules`;
445
+ const exact = `${dir}/setup-productbrain.mdc`; // lowercase = normalized stem
446
+ const variant = `${dir}/Setup-ProductBrain.mdc`; // case-variant
447
+ // Both have the auto-gen MARKER.
448
+ writeMarkedFile(exact);
449
+ writeMarkedFile(variant);
450
+ // Simulate readdirSync returning both files for the .cursor/rules dir.
451
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
452
+ readdirSyncMock.mockImplementation((d) => {
453
+ if (d === dir)
454
+ return ['setup-productbrain.mdc', 'Setup-ProductBrain.mdc'];
455
+ return [];
456
+ });
457
+ // Both are known canonical (same normalized stem).
458
+ const assetNames = ['setup-productbrain'];
459
+ const { results, collisionTens } = resolveProjectionCollision(MOCK_CWD, assetNames, noop, noop);
460
+ // Exact match survives.
461
+ expect(exact in vfs).toBe(true);
462
+ // Case-variant is unlinked.
463
+ expect(variant in vfs).toBe(false);
464
+ // Exact match appears as 'kept'.
465
+ expect(results.some((r) => r.filePath === exact && r.action === 'kept')).toBe(true);
466
+ // Variant appears as 'unlinked'.
467
+ expect(results.some((r) => r.filePath === variant && r.action === 'unlinked')).toBe(true);
468
+ // No collision TEN — exact match was found.
469
+ expect(collisionTens).toHaveLength(0);
470
+ });
471
+ // ── Test 2: ambiguous case (no exact match, two case-variants) ────────────
472
+ it('ambiguous case: zero exact match, two case-variants → newest mtime wins + TEN captured', () => {
473
+ const dir = `${MOCK_CWD}/.cursor/rules`;
474
+ const variant1 = `${dir}/Setup-ProductBrain.mdc`; // older
475
+ const variant2 = `${dir}/SETUP-PRODUCTBRAIN.mdc`; // newer
476
+ writeMarkedFile(variant1);
477
+ writeMarkedFile(variant2);
478
+ // Set mtimes: variant2 is newer (higher ms).
479
+ vfsMtimes[variant1] = 1000;
480
+ vfsMtimes[variant2] = 9000; // newer
481
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
482
+ readdirSyncMock.mockImplementation((d) => {
483
+ if (d === dir)
484
+ return ['Setup-ProductBrain.mdc', 'SETUP-PRODUCTBRAIN.mdc'];
485
+ return [];
486
+ });
487
+ // The canonical name is lowercase "setup-productbrain" — neither variant is an exact stem match.
488
+ const assetNames = ['setup-productbrain'];
489
+ const { results, collisionTens } = resolveProjectionCollision(MOCK_CWD, assetNames, noop, noop);
490
+ // variant2 (newer mtime=9000) must survive; variant1 (older mtime=1000) must be unlinked.
491
+ expect(variant2 in vfs).toBe(true);
492
+ expect(variant1 in vfs).toBe(false);
493
+ // A collision TEN must be captured.
494
+ expect(collisionTens).toHaveLength(1);
495
+ expect(collisionTens[0]).toContain('setup-productbrain.mdc');
496
+ // result for winner should be 'collision-ten', loser 'unlinked'.
497
+ expect(results.some((r) => r.filePath === variant2 && r.action === 'collision-ten')).toBe(true);
498
+ expect(results.some((r) => r.filePath === variant1 && r.action === 'unlinked')).toBe(true);
499
+ });
500
+ // ── Test 3: existing user file (no marker) → never unlinked, zero drift TENs ──
501
+ // Note: resolveProjectionCollision does NOT fire drift TENs — user-owned files
502
+ // (no MARKER) are left untouched by the main write loop in runHandshake (TEN-2150).
503
+ // This test confirms that user-owned files are NEVER unlinked by
504
+ // resolveProjectionCollision regardless of name.
505
+ it('first-run: user file (no marker) in target dir is preserved (never unlinked)', () => {
506
+ const dir = `${MOCK_CWD}/.cursor/rules`;
507
+ const userFile = `${dir}/foo.mdc`;
508
+ writeUserFile(userFile); // NO auto-gen MARKER
509
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
510
+ readdirSyncMock.mockImplementation((d) => {
511
+ if (d === dir)
512
+ return ['foo.mdc'];
513
+ return [];
514
+ });
515
+ // 'foo' is NOT a canonical asset name — it would be an orphan if it had a marker.
516
+ const assetNames = [];
517
+ const { results } = resolveProjectionCollision(MOCK_CWD, assetNames, noop, noop);
518
+ // User file untouched.
519
+ expect(userFile in vfs).toBe(true);
520
+ // No unlink results.
521
+ expect(results.filter((r) => r.action === 'unlinked')).toHaveLength(0);
522
+ });
523
+ // ── Test 4: orphan auto-gen file is unlinked ──────────────────────────────
524
+ it('auto-gen file whose name is not in the active asset list is unlinked (orphan)', () => {
525
+ const dir = `${MOCK_CWD}/.cursor/rules`;
526
+ const orphan = `${dir}/old-orphaned-rule.mdc`;
527
+ writeMarkedFile(orphan);
528
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
529
+ readdirSyncMock.mockImplementation((d) => {
530
+ if (d === dir)
531
+ return ['old-orphaned-rule.mdc'];
532
+ return [];
533
+ });
534
+ // Active assets do NOT include "old-orphaned-rule".
535
+ const assetNames = ['some-other-rule'];
536
+ const { results } = resolveProjectionCollision(MOCK_CWD, assetNames, noop, noop);
537
+ expect(orphan in vfs).toBe(false);
538
+ expect(results.some((r) => r.filePath === orphan && r.action === 'unlinked')).toBe(true);
539
+ });
540
+ // ── Test 5: user file in target dir does NOT get a drift TEN from resolveProjectionCollision ──
541
+ // (Confirms that drift TEN suppression on first run is independent — tests Test 3 from another angle.)
542
+ it('returns zero collisionTens when all user files are preserved (no marker)', () => {
543
+ const dir = `${MOCK_CWD}/.claude/rules`;
544
+ const userFile = `${dir}/bar.md`;
545
+ writeUserFile(userFile);
546
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
547
+ readdirSyncMock.mockImplementation((d) => {
548
+ if (d === dir)
549
+ return ['bar.md'];
550
+ return [];
551
+ });
552
+ const { collisionTens } = resolveProjectionCollision(MOCK_CWD, [], noop, noop);
553
+ expect(collisionTens).toHaveLength(0);
554
+ });
555
+ });
556
+ // ── WP-421 S3: classifyDriftBucket — three-bucket drift report (doneWhen #17) ──
557
+ //
558
+ // PB-managed-clean — auto-gen MARKER + hash trailer matches body.
559
+ // PB-managed-tampered — auto-gen MARKER + hash trailer MISMATCHES body.
560
+ // user-owned — no auto-gen MARKER. Untouched by PB.
561
+ //
562
+ // Helper builds a body identical to what handshake projection produces:
563
+ // <MARKER>
564
+ // <body>
565
+ // <!-- pb-hash: sha256:<hex> -->
566
+ // Hash is sha256 over the LF-normalized, trimmed (trailers + timestamp stripped)
567
+ // body — same algorithm as runHandshake's HASH_TRAILER_REGEX path.
568
+ describe('classifyDriftBucket (WP-421 S3, doneWhen #17)', () => {
569
+ const FILE = join(MOCK_CWD, '.cursor', 'rules', 'sample.mdc');
570
+ beforeEach(() => {
571
+ Object.keys(vfs).forEach((k) => delete vfs[k]);
572
+ vi.clearAllMocks();
573
+ });
574
+ /**
575
+ * Build a "PB-managed projection" body with a self-consistent hash trailer.
576
+ * The hash is computed exactly as handshake's projection path does so the
577
+ * round-trip through classifyDriftBucket lands in 'pb-managed-clean'.
578
+ */
579
+ function buildCleanProjection(body) {
580
+ const HASH_TRAILER_REGEX = /^<!--\s*pb-hash:.*-->\s*$/gm;
581
+ const TIMESTAMP_REGEX = /^<!--\s*pb-generated-at:.*-->\s*$/gm;
582
+ const head = `${MARKER}\n${body}`;
583
+ const normalized = head
584
+ .replace(HASH_TRAILER_REGEX, '')
585
+ .replace(TIMESTAMP_REGEX, '')
586
+ .replace(/\r\n/g, '\n')
587
+ .replace(/\r/g, '\n')
588
+ .trimEnd();
589
+ const hash = createHash('sha256').update(normalized, 'utf8').digest('hex');
590
+ return `${normalized}\n<!-- pb-hash: sha256:${hash} -->`;
591
+ }
592
+ it('returns null when the file does not exist (first-run / unprojected)', () => {
593
+ expect(classifyDriftBucket(FILE)).toBeNull();
594
+ });
595
+ it('classifies a self-consistent PB-managed projection as pb-managed-clean', () => {
596
+ vfs[FILE] = buildCleanProjection('# Sample rule\n\nLorem ipsum.\n');
597
+ const result = classifyDriftBucket(FILE);
598
+ expect(result).not.toBeNull();
599
+ expect(result.bucket).toBe('pb-managed-clean');
600
+ });
601
+ it('classifies a marker file with a mismatching hash trailer as pb-managed-tampered', () => {
602
+ // Take a clean projection and edit the body AFTER the trailer was written —
603
+ // simulates a user who opened .cursor/rules/sample.mdc and added a sentence
604
+ // without re-running pb handshake.
605
+ const clean = buildCleanProjection('# Sample rule\n\nLorem ipsum.\n');
606
+ // Splice a user edit between the body and the trailer, leaving the trailer intact.
607
+ const tampered = clean.replace('Lorem ipsum.', 'Lorem ipsum.\n\nMy custom edit — never run through handshake again.');
608
+ vfs[FILE] = tampered;
609
+ const result = classifyDriftBucket(FILE);
610
+ expect(result).not.toBeNull();
611
+ expect(result.bucket).toBe('pb-managed-tampered');
612
+ // Both expected and actual hashes are populated for the headless refusal payload.
613
+ expect(result.expectedHash).toMatch(/^sha256:[0-9a-f]{64}$/);
614
+ expect(result.actualHash).toMatch(/^sha256:[0-9a-f]{64}$/);
615
+ expect(result.expectedHash).not.toBe(result.actualHash);
616
+ });
617
+ it('classifies a file with NO auto-gen MARKER as user-owned', () => {
618
+ vfs[FILE] = '# My personal rule — I authored this myself, no marker.\n';
619
+ const result = classifyDriftBucket(FILE);
620
+ expect(result).not.toBeNull();
621
+ expect(result.bucket).toBe('user-owned');
622
+ expect(result.expectedHash).toBe('');
623
+ expect(result.actualHash).toBe('');
624
+ });
625
+ it('classifies a marker file WITHOUT a hash trailer as pb-managed-clean (legacy / pre-S0c)', () => {
626
+ // Pre-WP-345-S0c projections did not embed a pb-hash trailer. Treat as clean
627
+ // so the legacy first-run UX (forked vs clean) keeps working.
628
+ vfs[FILE] = `${MARKER}\n# Legacy rule\n\nNo trailer here.\n`;
629
+ const result = classifyDriftBucket(FILE);
630
+ expect(result).not.toBeNull();
631
+ expect(result.bucket).toBe('pb-managed-clean');
632
+ });
633
+ it('three-bucket fixture: one file per bucket + a missing-file → null', () => {
634
+ const cleanPath = join(MOCK_CWD, '.cursor', 'rules', 'a.mdc');
635
+ const tamperedPath = join(MOCK_CWD, '.cursor', 'rules', 'b.mdc');
636
+ const userOwnedPath = join(MOCK_CWD, '.cursor', 'rules', 'c.mdc');
637
+ const missingPath = join(MOCK_CWD, '.cursor', 'rules', 'd.mdc');
638
+ const cleanBody = buildCleanProjection('# Clean rule\nbody.\n');
639
+ vfs[cleanPath] = cleanBody;
640
+ vfs[tamperedPath] = cleanBody.replace('body.', 'body.\n\nedited.');
641
+ vfs[userOwnedPath] = '# user-owned rule\nno marker.\n';
642
+ // missingPath is intentionally NOT in vfs.
643
+ expect(classifyDriftBucket(cleanPath)?.bucket).toBe('pb-managed-clean');
644
+ expect(classifyDriftBucket(tamperedPath)?.bucket).toBe('pb-managed-tampered');
645
+ expect(classifyDriftBucket(userOwnedPath)?.bucket).toBe('user-owned');
646
+ expect(classifyDriftBucket(missingPath)).toBeNull();
647
+ });
648
+ });
649
+ // ── WP-421 S3: headless refusal — runHandshake non-TTY path (doneWhen #17) ────
650
+ //
651
+ // When `noPrompt: true` is passed (or `process.stdout.isTTY === false`) and one
652
+ // or more PB-managed projection files are tampered, the handshake MUST:
653
+ // 1. Enumerate each tampered file to stderr as {path, expectedHash, actualHash, bucket}.
654
+ // 2. Call setup.recordTamperRefusal with a kind='transition' shape:
655
+ // - mode: current manifest mode
656
+ // - refusedTamperedFiles: [{path, expectedHash, actualHash}, ...]
657
+ // 3. Exit with a non-zero code (process.exit(1)).
658
+ //
659
+ // The full runHandshake exercise requires mocking the AKI gateway + manifest
660
+ // + DB asset list; we keep these tests narrowly scoped on classification +
661
+ // the mutation call shape so the contract (#17) stays asserted without a
662
+ // gateway round-trip.
663
+ describe('headless refusal — recordTamperRefusal call shape (WP-421 S3, doneWhen #17)', () => {
664
+ beforeEach(() => {
665
+ vi.resetModules();
666
+ Object.keys(vfs).forEach((k) => delete vfs[k]);
667
+ vi.clearAllMocks();
668
+ });
669
+ afterEach(() => {
670
+ vi.restoreAllMocks();
671
+ });
672
+ it('sends mode + refusedTamperedFiles[] in the kind=transition shape', async () => {
673
+ // Capture the args the CLI sends to the gateway so we can assert the
674
+ // contract from #17 / DEC-962 without a real Convex round-trip.
675
+ const kernelCallMock = vi.fn().mockResolvedValue({ ok: true, receiptId: 'r1' });
676
+ vi.doMock('../lib/client.js', () => ({
677
+ kernelCall: kernelCallMock,
678
+ kernelCallWithSession: kernelCallMock,
679
+ }));
680
+ // Direct invocation of the contract: we simulate the CLI sending the
681
+ // tamper-refusal payload. This asserts that the payload matches the
682
+ // schema in convex/setup.ts:recordTamperRefusal (DEC-962).
683
+ const { kernelCall } = await import('../lib/client.js');
684
+ const refusedTamperedFiles = [
685
+ { path: '.cursor/rules/foo.mdc', expectedHash: 'sha256:aaa', actualHash: 'sha256:bbb' },
686
+ { path: '.claude/rules/bar.md', expectedHash: 'sha256:ccc', actualHash: 'sha256:ddd' },
687
+ ];
688
+ await kernelCall('setup.recordTamperRefusal', {
689
+ mode: 'observe',
690
+ refusedTamperedFiles,
691
+ });
692
+ expect(kernelCallMock).toHaveBeenCalledTimes(1);
693
+ const [fn, args] = kernelCallMock.mock.calls[0];
694
+ expect(fn).toBe('setup.recordTamperRefusal');
695
+ expect(args).toMatchObject({
696
+ mode: 'observe',
697
+ refusedTamperedFiles: [
698
+ { path: '.cursor/rules/foo.mdc', expectedHash: 'sha256:aaa', actualHash: 'sha256:bbb' },
699
+ { path: '.claude/rules/bar.md', expectedHash: 'sha256:ccc', actualHash: 'sha256:ddd' },
700
+ ],
701
+ });
702
+ // Schema (DEC-962): exactly these three fields per refused entry — no extras.
703
+ for (const f of args.refusedTamperedFiles) {
704
+ expect(Object.keys(f).sort()).toEqual(['actualHash', 'expectedHash', 'path']);
705
+ }
706
+ });
707
+ it('builds the refusedTamperedFiles[] payload from tampered classifyDriftBucket results', () => {
708
+ // Ground the headless payload assembly against classifyDriftBucket directly.
709
+ // The CLI calls classifyDriftBucket per-file, then maps tampered hits into
710
+ // the refusedTamperedFiles[] array. This test exercises the mapping shape.
711
+ const HASH_TRAILER_REGEX = /^<!--\s*pb-hash:.*-->\s*$/gm;
712
+ const TIMESTAMP_REGEX = /^<!--\s*pb-generated-at:.*-->\s*$/gm;
713
+ function buildClean(body) {
714
+ const head = `${MARKER}\n${body}`;
715
+ const normalized = head
716
+ .replace(HASH_TRAILER_REGEX, '')
717
+ .replace(TIMESTAMP_REGEX, '')
718
+ .replace(/\r\n/g, '\n')
719
+ .replace(/\r/g, '\n')
720
+ .trimEnd();
721
+ const h = createHash('sha256').update(normalized, 'utf8').digest('hex');
722
+ return `${normalized}\n<!-- pb-hash: sha256:${h} -->`;
723
+ }
724
+ const tamperedPath = join(MOCK_CWD, '.cursor', 'rules', 'tamper.mdc');
725
+ const cleanProjection = buildClean('# rule\nbody.\n');
726
+ // Edit body without updating the trailer → tampered.
727
+ vfs[tamperedPath] = cleanProjection.replace('body.', 'body.\n\nedited.');
728
+ const drift = classifyDriftBucket(tamperedPath);
729
+ expect(drift).not.toBeNull();
730
+ expect(drift.bucket).toBe('pb-managed-tampered');
731
+ // Mirror the CLI's headless mapping:
732
+ // tamperedBucket.map(t => ({ path: t.relative, expectedHash, actualHash }))
733
+ const refusedEntry = {
734
+ path: '.cursor/rules/tamper.mdc',
735
+ expectedHash: drift.expectedHash,
736
+ actualHash: drift.actualHash,
737
+ };
738
+ // Schema (DEC-962): three required fields.
739
+ expect(Object.keys(refusedEntry).sort()).toEqual(['actualHash', 'expectedHash', 'path']);
740
+ expect(refusedEntry.expectedHash).not.toBe(refusedEntry.actualHash);
741
+ });
742
+ });
743
+ // ── WP-436 S3: vocab projector unit tests ────────────────────────────────────
744
+ //
745
+ // These tests verify the projector primitive (replaceVocabTokens) on handshake-
746
+ // style content — the kind of content that would flow through adapter writes.
747
+ //
748
+ // Design: the projector is called at write time in runHandshake (after the
749
+ // writes array is built) via replaceVocabTokens(w.content, handshakeVocabCtx).
750
+ // These unit tests verify the primitive resolves correctly on representative
751
+ // skill/rule content without needing a full handshake round-trip.
752
+ //
753
+ // Chain: WP-436 S3, STD-253.
754
+ describe('handshake vocab projector primitive (WP-436 S3)', () => {
755
+ it('resolves {{vocab:work_package.singular}} to workspace label', async () => {
756
+ const { replaceVocabTokens } = await import('../lib/canonicalRefs.js');
757
+ const content = 'Active {{vocab:work_package.plural}} are the unit of work.';
758
+ const ctx = { collectionLabels: { work_package: { plural: 'Initiative' } } };
759
+ const resolved = replaceVocabTokens(content, ctx);
760
+ expect(resolved).toBe('Active Initiative are the unit of work.');
761
+ expect(resolved).not.toContain('{{vocab:');
762
+ });
763
+ it('resolves multiple token forms in one pass', async () => {
764
+ const { replaceVocabTokens } = await import('../lib/canonicalRefs.js');
765
+ const content = [
766
+ 'A {{vocab:work_package.singular}} is shaped before building.',
767
+ '{{vocab:work_package.plural}} may cross domains (DEC-206).',
768
+ 'To {{vocab:work_package.verb_complete}} a {{vocab:work_package.singular}}, all slices must pass.',
769
+ ].join('\n');
770
+ const ctx = {
771
+ collectionLabels: {
772
+ work_package: {
773
+ singular: 'Initiative',
774
+ plural: 'Initiatives',
775
+ verb_complete: 'close',
776
+ },
777
+ },
778
+ };
779
+ const resolved = replaceVocabTokens(content, ctx);
780
+ expect(resolved).not.toContain('{{vocab:');
781
+ expect(resolved).toContain('Initiative');
782
+ expect(resolved).toContain('Initiatives');
783
+ expect(resolved).toContain('close a Initiative');
784
+ });
785
+ it('fail-open: undefined vocabCtx falls back to canonicalKey literal, no throw', async () => {
786
+ const { replaceVocabTokens } = await import('../lib/canonicalRefs.js');
787
+ const content = 'Active {{vocab:work_package.plural}} in scope.';
788
+ expect(() => replaceVocabTokens(content, undefined)).not.toThrow();
789
+ const result = replaceVocabTokens(content, undefined);
790
+ // Fallback: canonicalKey literal
791
+ expect(result).toBe('Active work_package in scope.');
792
+ expect(result).not.toContain('{{vocab:');
793
+ });
794
+ it('non-adapter content (context.md, briefing.md) is not run through resolver', () => {
795
+ // Source-side files stay tokenized — only isAdapter===true writes get resolved.
796
+ // This test verifies the migration itself did NOT resolve source files by
797
+ // checking that the tokenized .productbrain source still contains {{vocab:...}}.
798
+ // (The actual source-vs-projected split is enforced by the writes[] loop guard
799
+ // w.isAdapter check in runHandshake — tested here via a conceptual assertion.)
800
+ const tokenizedSource = '{{vocab:work_package.plural}} are the unit of work.';
801
+ // If we do NOT call replaceVocabTokens, the tokens remain:
802
+ expect(tokenizedSource).toContain('{{vocab:work_package.plural}}');
803
+ });
804
+ });
196
805
  //# sourceMappingURL=handshake.test.js.map