@butlerw/vellum 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (446) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +411 -0
  3. package/__fixtures__/responses/code-generation.json +42 -0
  4. package/__fixtures__/responses/error-response.json +20 -0
  5. package/__fixtures__/responses/hello-world.json +32 -0
  6. package/dist/auth-6MCXESOH.js +26 -0
  7. package/dist/chunk-SECXJGWA.js +597 -0
  8. package/dist/index.js +34023 -0
  9. package/package.json +67 -0
  10. package/src/__tests__/commands.e2e.test.ts +728 -0
  11. package/src/__tests__/credentials.test.ts +713 -0
  12. package/src/__tests__/mode-e2e.test.ts +391 -0
  13. package/src/__tests__/tui-integration.test.tsx +1271 -0
  14. package/src/agents/__tests__/task-persistence.test.ts +235 -0
  15. package/src/agents/commands/delegate.ts +240 -0
  16. package/src/agents/commands/index.ts +10 -0
  17. package/src/agents/commands/resume.ts +335 -0
  18. package/src/agents/index.ts +29 -0
  19. package/src/agents/task-persistence.ts +272 -0
  20. package/src/agents/task-resumption.ts +242 -0
  21. package/src/app.tsx +4737 -0
  22. package/src/commands/__tests__/.gitkeep +1 -0
  23. package/src/commands/__tests__/agents.test.ts +606 -0
  24. package/src/commands/__tests__/auth.test.ts +626 -0
  25. package/src/commands/__tests__/autocomplete.test.ts +683 -0
  26. package/src/commands/__tests__/batch.test.ts +287 -0
  27. package/src/commands/__tests__/chain-pipe-parser.test.ts +654 -0
  28. package/src/commands/__tests__/completion.test.ts +238 -0
  29. package/src/commands/__tests__/core.test.ts +363 -0
  30. package/src/commands/__tests__/executor.test.ts +496 -0
  31. package/src/commands/__tests__/exit-codes.test.ts +220 -0
  32. package/src/commands/__tests__/init.test.ts +243 -0
  33. package/src/commands/__tests__/language.test.ts +353 -0
  34. package/src/commands/__tests__/mode-cli.test.ts +667 -0
  35. package/src/commands/__tests__/model.test.ts +277 -0
  36. package/src/commands/__tests__/parser.test.ts +493 -0
  37. package/src/commands/__tests__/performance.bench.ts +380 -0
  38. package/src/commands/__tests__/registry.test.ts +534 -0
  39. package/src/commands/__tests__/resume.test.ts +449 -0
  40. package/src/commands/__tests__/security.test.ts +845 -0
  41. package/src/commands/__tests__/stream-json.test.ts +372 -0
  42. package/src/commands/__tests__/user-commands.test.ts +597 -0
  43. package/src/commands/adapters.ts +267 -0
  44. package/src/commands/agent.ts +395 -0
  45. package/src/commands/agents/generate.ts +506 -0
  46. package/src/commands/agents/index.ts +272 -0
  47. package/src/commands/agents/show.ts +271 -0
  48. package/src/commands/agents/validate.ts +387 -0
  49. package/src/commands/auth.ts +883 -0
  50. package/src/commands/autocomplete.ts +480 -0
  51. package/src/commands/batch/command.ts +388 -0
  52. package/src/commands/batch/executor.ts +361 -0
  53. package/src/commands/batch/index.ts +12 -0
  54. package/src/commands/commit.ts +235 -0
  55. package/src/commands/completion/index.ts +371 -0
  56. package/src/commands/condense.ts +191 -0
  57. package/src/commands/config.ts +344 -0
  58. package/src/commands/context-provider.ts +173 -0
  59. package/src/commands/copy.ts +329 -0
  60. package/src/commands/core/clear.ts +38 -0
  61. package/src/commands/core/exit.ts +43 -0
  62. package/src/commands/core/help.ts +354 -0
  63. package/src/commands/core/index.ts +15 -0
  64. package/src/commands/cost.ts +179 -0
  65. package/src/commands/credentials.tsx +618 -0
  66. package/src/commands/custom-agents/__tests__/custom-agents.test.ts +709 -0
  67. package/src/commands/custom-agents/create.ts +377 -0
  68. package/src/commands/custom-agents/export.ts +135 -0
  69. package/src/commands/custom-agents/import.ts +199 -0
  70. package/src/commands/custom-agents/index.ts +372 -0
  71. package/src/commands/custom-agents/info.ts +318 -0
  72. package/src/commands/custom-agents/list.ts +267 -0
  73. package/src/commands/custom-agents/validate.ts +388 -0
  74. package/src/commands/diff-mode.ts +241 -0
  75. package/src/commands/env.ts +53 -0
  76. package/src/commands/executor.ts +579 -0
  77. package/src/commands/exit-codes.ts +202 -0
  78. package/src/commands/index.ts +701 -0
  79. package/src/commands/init/index.ts +15 -0
  80. package/src/commands/init/prompts.ts +366 -0
  81. package/src/commands/init/templates/commands-readme.md +80 -0
  82. package/src/commands/init/templates/example-command.md +79 -0
  83. package/src/commands/init/templates/example-skill.md +168 -0
  84. package/src/commands/init/templates/example-workflow.md +101 -0
  85. package/src/commands/init/templates/prompts-readme.md +52 -0
  86. package/src/commands/init/templates/rules-readme.md +63 -0
  87. package/src/commands/init/templates/skills-readme.md +83 -0
  88. package/src/commands/init/templates/workflows-readme.md +94 -0
  89. package/src/commands/init.ts +391 -0
  90. package/src/commands/install.ts +90 -0
  91. package/src/commands/language.ts +191 -0
  92. package/src/commands/loaders/.gitkeep +1 -0
  93. package/src/commands/lsp.ts +199 -0
  94. package/src/commands/markdown-commands.ts +253 -0
  95. package/src/commands/mcp.ts +588 -0
  96. package/src/commands/memory/export.ts +341 -0
  97. package/src/commands/memory/index.ts +148 -0
  98. package/src/commands/memory/list.ts +261 -0
  99. package/src/commands/memory/search.ts +346 -0
  100. package/src/commands/memory/utils.ts +15 -0
  101. package/src/commands/metrics.ts +75 -0
  102. package/src/commands/migrate/index.ts +16 -0
  103. package/src/commands/migrate/prompts.ts +477 -0
  104. package/src/commands/mode.ts +331 -0
  105. package/src/commands/model.ts +298 -0
  106. package/src/commands/onboard.ts +205 -0
  107. package/src/commands/open.ts +169 -0
  108. package/src/commands/output/stream-json.ts +373 -0
  109. package/src/commands/parser/chain-parser.ts +370 -0
  110. package/src/commands/parser/index.ts +29 -0
  111. package/src/commands/parser/pipe-parser.ts +480 -0
  112. package/src/commands/parser.ts +588 -0
  113. package/src/commands/persistence.ts +355 -0
  114. package/src/commands/progress.ts +18 -0
  115. package/src/commands/prompt/index.ts +17 -0
  116. package/src/commands/prompt/validate.ts +621 -0
  117. package/src/commands/prompt-priority.ts +401 -0
  118. package/src/commands/registry.ts +374 -0
  119. package/src/commands/sandbox/index.ts +131 -0
  120. package/src/commands/security/index.ts +21 -0
  121. package/src/commands/security/input-sanitizer.ts +168 -0
  122. package/src/commands/security/permission-checker.ts +456 -0
  123. package/src/commands/security/sensitive-data.ts +350 -0
  124. package/src/commands/session/delete.ts +38 -0
  125. package/src/commands/session/export.ts +39 -0
  126. package/src/commands/session/index.ts +26 -0
  127. package/src/commands/session/list.ts +26 -0
  128. package/src/commands/session/resume.ts +562 -0
  129. package/src/commands/session/search.ts +434 -0
  130. package/src/commands/session/show.ts +26 -0
  131. package/src/commands/settings.ts +368 -0
  132. package/src/commands/setup.ts +23 -0
  133. package/src/commands/shell/index.ts +16 -0
  134. package/src/commands/shell/setup.ts +422 -0
  135. package/src/commands/shell-init.ts +50 -0
  136. package/src/commands/shell-integration/index.ts +194 -0
  137. package/src/commands/skill.ts +1220 -0
  138. package/src/commands/spec.ts +558 -0
  139. package/src/commands/status.ts +246 -0
  140. package/src/commands/theme.ts +211 -0
  141. package/src/commands/think.ts +551 -0
  142. package/src/commands/trust.ts +211 -0
  143. package/src/commands/tutorial.ts +522 -0
  144. package/src/commands/types.ts +512 -0
  145. package/src/commands/update.ts +274 -0
  146. package/src/commands/usage.ts +213 -0
  147. package/src/commands/user-commands.ts +630 -0
  148. package/src/commands/utils.ts +142 -0
  149. package/src/commands/vim.ts +152 -0
  150. package/src/commands/workflow.ts +257 -0
  151. package/src/components/header.tsx +25 -0
  152. package/src/components/input.tsx +25 -0
  153. package/src/components/message-list.tsx +32 -0
  154. package/src/components/status-bar.tsx +23 -0
  155. package/src/index.tsx +614 -0
  156. package/src/onboarding/__tests__/tutorial.test.ts +740 -0
  157. package/src/onboarding/index.ts +69 -0
  158. package/src/onboarding/tips/index.ts +9 -0
  159. package/src/onboarding/tips/tip-engine.ts +459 -0
  160. package/src/onboarding/tutorial/index.ts +88 -0
  161. package/src/onboarding/tutorial/lessons/basics.ts +151 -0
  162. package/src/onboarding/tutorial/lessons/index.ts +151 -0
  163. package/src/onboarding/tutorial/lessons/modes.ts +230 -0
  164. package/src/onboarding/tutorial/lessons/tools.ts +172 -0
  165. package/src/onboarding/tutorial/progress-tracker.ts +350 -0
  166. package/src/onboarding/tutorial/storage.ts +249 -0
  167. package/src/onboarding/tutorial/tutorial-system.ts +462 -0
  168. package/src/onboarding/tutorial/types.ts +310 -0
  169. package/src/orchestrator-singleton.ts +129 -0
  170. package/src/shutdown.ts +33 -0
  171. package/src/test/e2e/assertions.ts +267 -0
  172. package/src/test/e2e/fixtures.ts +204 -0
  173. package/src/test/e2e/harness.ts +575 -0
  174. package/src/test/e2e/index.ts +57 -0
  175. package/src/test/e2e/types.ts +228 -0
  176. package/src/test/fixtures/__tests__/fake-response-loader.test.ts +314 -0
  177. package/src/test/fixtures/fake-response-loader.ts +314 -0
  178. package/src/test/fixtures/index.ts +20 -0
  179. package/src/tui/__tests__/mcp-panel.test.tsx +82 -0
  180. package/src/tui/__tests__/mcp-wiring.test.tsx +78 -0
  181. package/src/tui/__tests__/mode-components.test.tsx +395 -0
  182. package/src/tui/__tests__/permission-ask-flow.test.tsx +138 -0
  183. package/src/tui/__tests__/sidebar-panel-data.test.tsx +148 -0
  184. package/src/tui/__tests__/tools-panel-hotkeys.test.tsx +41 -0
  185. package/src/tui/adapters/agent-adapter.ts +1008 -0
  186. package/src/tui/adapters/index.ts +48 -0
  187. package/src/tui/adapters/message-adapter.ts +315 -0
  188. package/src/tui/adapters/persistence-bridge.ts +331 -0
  189. package/src/tui/adapters/session-adapter.ts +419 -0
  190. package/src/tui/buffered-stdout.ts +223 -0
  191. package/src/tui/components/AgentProgress.tsx +424 -0
  192. package/src/tui/components/Banner/AsciiArt.ts +160 -0
  193. package/src/tui/components/Banner/Banner.tsx +355 -0
  194. package/src/tui/components/Banner/ShimmerContext.tsx +131 -0
  195. package/src/tui/components/Banner/ShimmerText.tsx +193 -0
  196. package/src/tui/components/Banner/TypeWriterGradient.tsx +321 -0
  197. package/src/tui/components/Banner/index.ts +61 -0
  198. package/src/tui/components/Banner/useShimmer.ts +241 -0
  199. package/src/tui/components/ChatView.tsx +11 -0
  200. package/src/tui/components/Checkpoint/CheckpointDiffView.tsx +371 -0
  201. package/src/tui/components/Checkpoint/SnapshotCheckpointPanel.tsx +440 -0
  202. package/src/tui/components/Checkpoint/index.ts +19 -0
  203. package/src/tui/components/CostDisplay.tsx +226 -0
  204. package/src/tui/components/InitErrorBanner.tsx +122 -0
  205. package/src/tui/components/Input/Autocomplete.tsx +603 -0
  206. package/src/tui/components/Input/EnhancedCommandInput.tsx +471 -0
  207. package/src/tui/components/Input/HighlightedText.tsx +236 -0
  208. package/src/tui/components/Input/MentionAutocomplete.tsx +375 -0
  209. package/src/tui/components/Input/TextInput.tsx +1002 -0
  210. package/src/tui/components/Input/__tests__/Autocomplete.test.tsx +374 -0
  211. package/src/tui/components/Input/__tests__/TextInput.test.tsx +241 -0
  212. package/src/tui/components/Input/__tests__/highlight.test.ts +219 -0
  213. package/src/tui/components/Input/__tests__/slash-command-utils.test.ts +104 -0
  214. package/src/tui/components/Input/highlight.ts +362 -0
  215. package/src/tui/components/Input/index.ts +36 -0
  216. package/src/tui/components/Input/slash-command-utils.ts +135 -0
  217. package/src/tui/components/Layout.tsx +432 -0
  218. package/src/tui/components/McpPanel.tsx +137 -0
  219. package/src/tui/components/MemoryPanel.tsx +448 -0
  220. package/src/tui/components/Messages/CodeBlock.tsx +527 -0
  221. package/src/tui/components/Messages/DiffView.tsx +679 -0
  222. package/src/tui/components/Messages/ImageReference.tsx +89 -0
  223. package/src/tui/components/Messages/MarkdownBlock.tsx +228 -0
  224. package/src/tui/components/Messages/MarkdownRenderer.tsx +498 -0
  225. package/src/tui/components/Messages/MessageBubble.tsx +270 -0
  226. package/src/tui/components/Messages/MessageList.tsx +1719 -0
  227. package/src/tui/components/Messages/StreamingText.tsx +216 -0
  228. package/src/tui/components/Messages/ThinkingBlock.tsx +408 -0
  229. package/src/tui/components/Messages/ToolResultPreview.tsx +243 -0
  230. package/src/tui/components/Messages/__tests__/CodeBlock.test.tsx +296 -0
  231. package/src/tui/components/Messages/__tests__/DiffView.test.tsx +239 -0
  232. package/src/tui/components/Messages/__tests__/MarkdownRenderer.test.tsx +303 -0
  233. package/src/tui/components/Messages/__tests__/MessageBubble.test.tsx +268 -0
  234. package/src/tui/components/Messages/__tests__/MessageList.test.tsx +324 -0
  235. package/src/tui/components/Messages/__tests__/StreamingText.test.tsx +215 -0
  236. package/src/tui/components/Messages/index.ts +25 -0
  237. package/src/tui/components/ModeIndicator.tsx +177 -0
  238. package/src/tui/components/ModeSelector.tsx +216 -0
  239. package/src/tui/components/ModelSelector.tsx +339 -0
  240. package/src/tui/components/OnboardingWizard.tsx +670 -0
  241. package/src/tui/components/PhaseProgressIndicator.tsx +270 -0
  242. package/src/tui/components/RateLimitIndicator.tsx +82 -0
  243. package/src/tui/components/ScreenReaderLayout.tsx +295 -0
  244. package/src/tui/components/SettingsPanel.tsx +643 -0
  245. package/src/tui/components/Sidebar/SystemStatusPanel.tsx +284 -0
  246. package/src/tui/components/Sidebar/index.ts +9 -0
  247. package/src/tui/components/Status/ModelStatusBar.tsx +270 -0
  248. package/src/tui/components/Status/index.ts +12 -0
  249. package/src/tui/components/StatusBar/AgentModeIndicator.tsx +257 -0
  250. package/src/tui/components/StatusBar/ContextProgress.tsx +167 -0
  251. package/src/tui/components/StatusBar/FileChangesIndicator.tsx +62 -0
  252. package/src/tui/components/StatusBar/GitIndicator.tsx +89 -0
  253. package/src/tui/components/StatusBar/HeaderBar.tsx +126 -0
  254. package/src/tui/components/StatusBar/ModelIndicator.tsx +157 -0
  255. package/src/tui/components/StatusBar/PersistenceStatusIndicator.tsx +210 -0
  256. package/src/tui/components/StatusBar/ResilienceIndicator.tsx +106 -0
  257. package/src/tui/components/StatusBar/SandboxIndicator.tsx +167 -0
  258. package/src/tui/components/StatusBar/StatusBar.tsx +368 -0
  259. package/src/tui/components/StatusBar/ThinkingModeIndicator.tsx +170 -0
  260. package/src/tui/components/StatusBar/TokenBreakdown.tsx +246 -0
  261. package/src/tui/components/StatusBar/TokenCounter.tsx +135 -0
  262. package/src/tui/components/StatusBar/TrustModeIndicator.tsx +130 -0
  263. package/src/tui/components/StatusBar/WorkspaceIndicator.tsx +86 -0
  264. package/src/tui/components/StatusBar/__tests__/AgentModeIndicator.test.tsx +193 -0
  265. package/src/tui/components/StatusBar/__tests__/StatusBar.test.tsx +729 -0
  266. package/src/tui/components/StatusBar/index.ts +60 -0
  267. package/src/tui/components/TipBanner.tsx +115 -0
  268. package/src/tui/components/TodoItem.tsx +208 -0
  269. package/src/tui/components/TodoPanel.tsx +455 -0
  270. package/src/tui/components/Tools/ApprovalQueue.tsx +407 -0
  271. package/src/tui/components/Tools/OptionSelector.tsx +160 -0
  272. package/src/tui/components/Tools/PermissionDialog.tsx +286 -0
  273. package/src/tui/components/Tools/ToolParams.tsx +483 -0
  274. package/src/tui/components/Tools/ToolsPanel.tsx +178 -0
  275. package/src/tui/components/Tools/__tests__/PermissionDialog.test.tsx +510 -0
  276. package/src/tui/components/Tools/__tests__/ToolParams.test.tsx +432 -0
  277. package/src/tui/components/Tools/index.ts +21 -0
  278. package/src/tui/components/TrustPrompt.tsx +279 -0
  279. package/src/tui/components/UpdateBanner.tsx +166 -0
  280. package/src/tui/components/VimModeIndicator.tsx +112 -0
  281. package/src/tui/components/backtrack/BacktrackControls.tsx +402 -0
  282. package/src/tui/components/backtrack/index.ts +13 -0
  283. package/src/tui/components/common/AutoApprovalStatus.tsx +251 -0
  284. package/src/tui/components/common/CostWarning.tsx +294 -0
  285. package/src/tui/components/common/DynamicShortcutHints.tsx +209 -0
  286. package/src/tui/components/common/EnhancedLoadingIndicator.tsx +305 -0
  287. package/src/tui/components/common/ErrorBoundary.tsx +140 -0
  288. package/src/tui/components/common/GradientText.tsx +224 -0
  289. package/src/tui/components/common/HotkeyHelpModal.tsx +193 -0
  290. package/src/tui/components/common/HotkeyHints.tsx +70 -0
  291. package/src/tui/components/common/MaxSizedBox.tsx +354 -0
  292. package/src/tui/components/common/NewMessagesBadge.tsx +65 -0
  293. package/src/tui/components/common/ProtectedFileLegend.tsx +89 -0
  294. package/src/tui/components/common/ScrollIndicator.tsx +160 -0
  295. package/src/tui/components/common/Spinner.tsx +342 -0
  296. package/src/tui/components/common/StreamingIndicator.tsx +316 -0
  297. package/src/tui/components/common/VirtualizedList/VirtualizedList.tsx +428 -0
  298. package/src/tui/components/common/VirtualizedList/hooks/index.ts +19 -0
  299. package/src/tui/components/common/VirtualizedList/hooks/useBatchedScroll.ts +64 -0
  300. package/src/tui/components/common/VirtualizedList/hooks/useScrollAnchor.ts +290 -0
  301. package/src/tui/components/common/VirtualizedList/hooks/useVirtualization.ts +340 -0
  302. package/src/tui/components/common/VirtualizedList/index.ts +30 -0
  303. package/src/tui/components/common/VirtualizedList/types.ts +107 -0
  304. package/src/tui/components/common/__tests__/NewMessagesBadge.test.tsx +74 -0
  305. package/src/tui/components/common/__tests__/ScrollIndicator.test.tsx +193 -0
  306. package/src/tui/components/common/index.ts +110 -0
  307. package/src/tui/components/index.ts +79 -0
  308. package/src/tui/components/session/CheckpointPanel.tsx +323 -0
  309. package/src/tui/components/session/RollbackDialog.tsx +169 -0
  310. package/src/tui/components/session/SessionItem.tsx +136 -0
  311. package/src/tui/components/session/SessionListPanel.tsx +252 -0
  312. package/src/tui/components/session/SessionPicker.tsx +449 -0
  313. package/src/tui/components/session/SessionPreview.tsx +240 -0
  314. package/src/tui/components/session/__tests__/session.test.tsx +408 -0
  315. package/src/tui/components/session/index.ts +28 -0
  316. package/src/tui/components/session/types.ts +116 -0
  317. package/src/tui/components/theme/__tests__/tokens.test.ts +471 -0
  318. package/src/tui/components/theme/index.ts +227 -0
  319. package/src/tui/components/theme/tokens.ts +484 -0
  320. package/src/tui/config/defaults.ts +134 -0
  321. package/src/tui/config/index.ts +17 -0
  322. package/src/tui/context/AnimationContext.tsx +284 -0
  323. package/src/tui/context/AppContext.tsx +349 -0
  324. package/src/tui/context/BracketedPasteContext.tsx +372 -0
  325. package/src/tui/context/LspContext.tsx +192 -0
  326. package/src/tui/context/McpContext.tsx +325 -0
  327. package/src/tui/context/MessagesContext.tsx +870 -0
  328. package/src/tui/context/OverflowContext.tsx +213 -0
  329. package/src/tui/context/RateLimitContext.tsx +108 -0
  330. package/src/tui/context/ResilienceContext.tsx +275 -0
  331. package/src/tui/context/RootProvider.tsx +136 -0
  332. package/src/tui/context/ScrollContext.tsx +331 -0
  333. package/src/tui/context/ToolsContext.tsx +702 -0
  334. package/src/tui/context/__tests__/BracketedPasteContext.test.tsx +416 -0
  335. package/src/tui/context/index.ts +140 -0
  336. package/src/tui/enterprise-integration.ts +282 -0
  337. package/src/tui/hooks/__tests__/useBacktrack.test.tsx +138 -0
  338. package/src/tui/hooks/__tests__/useBracketedPaste.test.tsx +222 -0
  339. package/src/tui/hooks/__tests__/useCopyMode.test.tsx +336 -0
  340. package/src/tui/hooks/__tests__/useHotkeys.ctrl-input.test.tsx +96 -0
  341. package/src/tui/hooks/__tests__/useHotkeys.test.tsx +454 -0
  342. package/src/tui/hooks/__tests__/useInputHistory.test.tsx +660 -0
  343. package/src/tui/hooks/__tests__/useLineBuffer.test.ts +295 -0
  344. package/src/tui/hooks/__tests__/useModeController.test.ts +137 -0
  345. package/src/tui/hooks/__tests__/useModeShortcuts.test.tsx +142 -0
  346. package/src/tui/hooks/__tests__/useScrollController.test.ts +464 -0
  347. package/src/tui/hooks/__tests__/useVim.test.tsx +531 -0
  348. package/src/tui/hooks/index.ts +252 -0
  349. package/src/tui/hooks/useAgentLoop.ts +712 -0
  350. package/src/tui/hooks/useAlternateBuffer.ts +398 -0
  351. package/src/tui/hooks/useAnimatedScrollbar.ts +241 -0
  352. package/src/tui/hooks/useBacktrack.ts +443 -0
  353. package/src/tui/hooks/useBracketedPaste.ts +104 -0
  354. package/src/tui/hooks/useCollapsible.ts +240 -0
  355. package/src/tui/hooks/useCopyMode.ts +382 -0
  356. package/src/tui/hooks/useCostSummary.ts +75 -0
  357. package/src/tui/hooks/useDesktopNotification.ts +414 -0
  358. package/src/tui/hooks/useDiffMode.ts +44 -0
  359. package/src/tui/hooks/useFileChangeStats.ts +110 -0
  360. package/src/tui/hooks/useFileSuggestions.ts +284 -0
  361. package/src/tui/hooks/useFlickerDetector.ts +250 -0
  362. package/src/tui/hooks/useGitStatus.ts +200 -0
  363. package/src/tui/hooks/useHotkeys.ts +579 -0
  364. package/src/tui/hooks/useImagePaste.ts +114 -0
  365. package/src/tui/hooks/useInputHighlight.ts +145 -0
  366. package/src/tui/hooks/useInputHistory.ts +246 -0
  367. package/src/tui/hooks/useKeyboardScroll.ts +209 -0
  368. package/src/tui/hooks/useLineBuffer.ts +356 -0
  369. package/src/tui/hooks/useMentionAutocomplete.ts +235 -0
  370. package/src/tui/hooks/useModeController.ts +167 -0
  371. package/src/tui/hooks/useModeShortcuts.ts +196 -0
  372. package/src/tui/hooks/usePermissionHandler.ts +146 -0
  373. package/src/tui/hooks/usePersistence.ts +480 -0
  374. package/src/tui/hooks/usePersistenceShortcuts.ts +225 -0
  375. package/src/tui/hooks/usePlaceholderRotation.ts +143 -0
  376. package/src/tui/hooks/useProviderStatus.ts +270 -0
  377. package/src/tui/hooks/useRateLimitStatus.ts +90 -0
  378. package/src/tui/hooks/useScreenReader.ts +315 -0
  379. package/src/tui/hooks/useScrollController.ts +450 -0
  380. package/src/tui/hooks/useScrollEventBatcher.ts +185 -0
  381. package/src/tui/hooks/useSidebarPanelData.ts +115 -0
  382. package/src/tui/hooks/useSmoothScroll.ts +202 -0
  383. package/src/tui/hooks/useSnapshots.ts +300 -0
  384. package/src/tui/hooks/useStateAndRef.ts +50 -0
  385. package/src/tui/hooks/useTerminalSize.ts +206 -0
  386. package/src/tui/hooks/useToolApprovalController.ts +91 -0
  387. package/src/tui/hooks/useVim.ts +334 -0
  388. package/src/tui/hooks/useWorkspace.ts +56 -0
  389. package/src/tui/i18n/__tests__/init.test.ts +278 -0
  390. package/src/tui/i18n/__tests__/language-config.test.ts +199 -0
  391. package/src/tui/i18n/__tests__/locale-detection.test.ts +250 -0
  392. package/src/tui/i18n/__tests__/settings-integration.test.ts +262 -0
  393. package/src/tui/i18n/index.ts +72 -0
  394. package/src/tui/i18n/init.ts +131 -0
  395. package/src/tui/i18n/language-config.ts +106 -0
  396. package/src/tui/i18n/locale-detection.ts +173 -0
  397. package/src/tui/i18n/settings-integration.ts +557 -0
  398. package/src/tui/i18n/tui-namespace.ts +538 -0
  399. package/src/tui/i18n/types.ts +312 -0
  400. package/src/tui/index.ts +43 -0
  401. package/src/tui/lsp-integration.ts +409 -0
  402. package/src/tui/metrics-integration.ts +366 -0
  403. package/src/tui/plugins.ts +383 -0
  404. package/src/tui/resilience.ts +342 -0
  405. package/src/tui/sandbox-integration.ts +317 -0
  406. package/src/tui/services/clipboard.ts +348 -0
  407. package/src/tui/services/fuzzy-search.ts +441 -0
  408. package/src/tui/services/index.ts +72 -0
  409. package/src/tui/services/markdown-renderer.ts +565 -0
  410. package/src/tui/services/open-external.ts +247 -0
  411. package/src/tui/services/syntax-highlighter.ts +483 -0
  412. package/src/tui/slash-commands.ts +12 -0
  413. package/src/tui/theme/index.ts +15 -0
  414. package/src/tui/theme/provider.tsx +206 -0
  415. package/src/tui/tip-integration.ts +300 -0
  416. package/src/tui/types/__tests__/ink-extended.test.ts +121 -0
  417. package/src/tui/types/ink-extended.ts +87 -0
  418. package/src/tui/utils/__tests__/bracketedPaste.test.ts +231 -0
  419. package/src/tui/utils/__tests__/heightEstimator.test.ts +157 -0
  420. package/src/tui/utils/__tests__/text-width.test.ts +158 -0
  421. package/src/tui/utils/__tests__/textSanitizer.test.ts +266 -0
  422. package/src/tui/utils/__tests__/ui-sizing.test.ts +169 -0
  423. package/src/tui/utils/bracketedPaste.ts +107 -0
  424. package/src/tui/utils/cursor-manager.ts +131 -0
  425. package/src/tui/utils/detectTerminal.ts +596 -0
  426. package/src/tui/utils/findLastSafeSplitPoint.ts +92 -0
  427. package/src/tui/utils/heightEstimator.ts +198 -0
  428. package/src/tui/utils/index.ts +91 -0
  429. package/src/tui/utils/isNarrowWidth.ts +52 -0
  430. package/src/tui/utils/stdoutGuard.ts +90 -0
  431. package/src/tui/utils/synchronized-update.ts +70 -0
  432. package/src/tui/utils/text-width.ts +225 -0
  433. package/src/tui/utils/textSanitizer.ts +225 -0
  434. package/src/tui/utils/textUtils.ts +114 -0
  435. package/src/tui/utils/ui-sizing.ts +192 -0
  436. package/src/tui-blessed/app.ts +160 -0
  437. package/src/tui-blessed/index.ts +2 -0
  438. package/src/tui-blessed/neo-blessed.d.ts +6 -0
  439. package/src/tui-blessed/test.ts +21 -0
  440. package/src/tui-blessed/types.ts +14 -0
  441. package/src/utils/icons.ts +130 -0
  442. package/src/utils/index.ts +33 -0
  443. package/src/utils/resume-hint.ts +86 -0
  444. package/src/version.ts +1 -0
  445. package/tsconfig.json +8 -0
  446. package/vitest.config.ts +35 -0
@@ -0,0 +1,575 @@
1
+ /**
2
+ * CLI Test Harness
3
+ *
4
+ * Spawns CLI processes, captures output, and handles interactive input.
5
+ *
6
+ * @module cli/test/e2e/harness
7
+ */
8
+
9
+ import { type ChildProcess, spawn } from "node:child_process";
10
+ import { resolve } from "node:path";
11
+ import { fileURLToPath } from "node:url";
12
+
13
+ import type {
14
+ AssertionFailure,
15
+ BatchResult,
16
+ CLIResult,
17
+ HarnessConfig,
18
+ HarnessEvent,
19
+ HarnessEventListener,
20
+ InputStep,
21
+ OutputMatcher,
22
+ ResolvedHarnessConfig,
23
+ ScenarioResult,
24
+ TestScenario,
25
+ } from "./types.js";
26
+
27
+ const __dirname = fileURLToPath(new URL(".", import.meta.url));
28
+
29
+ /**
30
+ * Assert CLI output against expected matchers (inline to avoid circular import)
31
+ */
32
+ // biome-ignore lint/complexity/noExcessiveCognitiveComplexity: Assertion function needs to check multiple output conditions
33
+ function assertOutput(result: CLIResult, expected: OutputMatcher): AssertionFailure[] {
34
+ const failures: AssertionFailure[] = [];
35
+
36
+ if (expected.exitCode !== undefined && result.exitCode !== expected.exitCode) {
37
+ failures.push({
38
+ type: "exitCode",
39
+ expected: expected.exitCode,
40
+ actual: result.exitCode,
41
+ message: `Expected exit code ${expected.exitCode}, got ${result.exitCode}`,
42
+ });
43
+ }
44
+
45
+ if (expected.stdout !== undefined) {
46
+ if (typeof expected.stdout === "string") {
47
+ if (!result.stdout.includes(expected.stdout)) {
48
+ failures.push({
49
+ type: "stdout",
50
+ expected: expected.stdout,
51
+ actual: result.stdout,
52
+ message: `stdout does not contain expected string: "${expected.stdout}"`,
53
+ });
54
+ }
55
+ } else if (!expected.stdout.test(result.stdout)) {
56
+ failures.push({
57
+ type: "stdout",
58
+ expected: expected.stdout.toString(),
59
+ actual: result.stdout,
60
+ message: `stdout does not match pattern: ${expected.stdout.toString()}`,
61
+ });
62
+ }
63
+ }
64
+
65
+ if (expected.stderr !== undefined) {
66
+ if (typeof expected.stderr === "string") {
67
+ if (!result.stderr.includes(expected.stderr)) {
68
+ failures.push({
69
+ type: "stderr",
70
+ expected: expected.stderr,
71
+ actual: result.stderr,
72
+ message: `stderr does not contain expected string: "${expected.stderr}"`,
73
+ });
74
+ }
75
+ } else if (!expected.stderr.test(result.stderr)) {
76
+ failures.push({
77
+ type: "stderr",
78
+ expected: expected.stderr.toString(),
79
+ actual: result.stderr,
80
+ message: `stderr does not match pattern: ${expected.stderr.toString()}`,
81
+ });
82
+ }
83
+ }
84
+
85
+ if (expected.contains) {
86
+ const combined = result.stdout + result.stderr;
87
+ for (const text of expected.contains) {
88
+ if (!combined.includes(text)) {
89
+ failures.push({
90
+ type: "contains",
91
+ expected: text,
92
+ actual: combined,
93
+ message: `Output does not contain: "${text}"`,
94
+ });
95
+ }
96
+ }
97
+ }
98
+
99
+ if (expected.excludes) {
100
+ const combined = result.stdout + result.stderr;
101
+ for (const text of expected.excludes) {
102
+ if (combined.includes(text)) {
103
+ failures.push({
104
+ type: "excludes",
105
+ expected: `not "${text}"`,
106
+ actual: combined,
107
+ message: `Output should NOT contain: "${text}"`,
108
+ });
109
+ }
110
+ }
111
+ }
112
+
113
+ if (result.timedOut) {
114
+ failures.push({
115
+ type: "timeout",
116
+ expected: "no timeout",
117
+ actual: "timed out",
118
+ message: `Process timed out after ${result.duration}ms`,
119
+ });
120
+ }
121
+
122
+ return failures;
123
+ }
124
+
125
+ /**
126
+ * Default harness configuration
127
+ */
128
+ const DEFAULT_CONFIG: ResolvedHarnessConfig = {
129
+ cliPath: resolve(__dirname, "../../../dist/index.js"),
130
+ timeout: 30000,
131
+ cwd: process.cwd(),
132
+ env: {},
133
+ inheritEnv: true,
134
+ encoding: "utf-8",
135
+ debug: false,
136
+ };
137
+
138
+ /**
139
+ * CLI Test Harness for E2E testing
140
+ */
141
+ export class CLITestHarness {
142
+ private readonly config: ResolvedHarnessConfig;
143
+ private readonly listeners: Set<HarnessEventListener> = new Set();
144
+ private activeProcess: ChildProcess | null = null;
145
+
146
+ constructor(config: HarnessConfig = {}) {
147
+ this.config = {
148
+ ...DEFAULT_CONFIG,
149
+ ...config,
150
+ env: { ...DEFAULT_CONFIG.env, ...config.env },
151
+ };
152
+ }
153
+
154
+ /**
155
+ * Add event listener
156
+ */
157
+ on(listener: HarnessEventListener): () => void {
158
+ this.listeners.add(listener);
159
+ return () => this.listeners.delete(listener);
160
+ }
161
+
162
+ /**
163
+ * Emit event to all listeners
164
+ */
165
+ private emit(event: HarnessEvent): void {
166
+ for (const listener of this.listeners) {
167
+ try {
168
+ listener(event);
169
+ } catch {
170
+ // Ignore listener errors
171
+ }
172
+ }
173
+ }
174
+
175
+ /**
176
+ * Debug log if enabled
177
+ */
178
+ private debug(...args: unknown[]): void {
179
+ if (this.config.debug) {
180
+ console.log("[CLITestHarness]", ...args);
181
+ }
182
+ }
183
+
184
+ /**
185
+ * Run CLI with arguments and capture output
186
+ */
187
+ async run(args: string[], options?: Partial<HarnessConfig>): Promise<CLIResult> {
188
+ const config = { ...this.config, ...options };
189
+ const startTime = Date.now();
190
+
191
+ const stdoutChunks: string[] = [];
192
+ const stderrChunks: string[] = [];
193
+ let timedOut = false;
194
+
195
+ return new Promise<CLIResult>((resolve, reject) => {
196
+ this.debug("Spawning:", "node", config.cliPath, ...args);
197
+
198
+ const env = config.inheritEnv ? { ...process.env, ...config.env } : config.env;
199
+
200
+ const proc = spawn("node", [config.cliPath, ...args], {
201
+ cwd: config.cwd,
202
+ env,
203
+ stdio: ["pipe", "pipe", "pipe"],
204
+ });
205
+
206
+ this.activeProcess = proc;
207
+
208
+ if (proc.pid !== undefined) {
209
+ this.emit({ type: "spawn", pid: proc.pid });
210
+ }
211
+
212
+ // Timeout handling
213
+ const timeoutId = setTimeout(() => {
214
+ timedOut = true;
215
+ this.emit({ type: "timeout" });
216
+ proc.kill("SIGTERM");
217
+
218
+ // Force kill after grace period
219
+ setTimeout(() => {
220
+ if (!proc.killed) {
221
+ proc.kill("SIGKILL");
222
+ }
223
+ }, 1000);
224
+ }, config.timeout);
225
+
226
+ // Capture stdout
227
+ proc.stdout?.setEncoding(config.encoding);
228
+ proc.stdout?.on("data", (data: string) => {
229
+ this.debug("stdout:", data);
230
+ stdoutChunks.push(data);
231
+ this.emit({ type: "stdout", data });
232
+ });
233
+
234
+ // Capture stderr
235
+ proc.stderr?.setEncoding(config.encoding);
236
+ proc.stderr?.on("data", (data: string) => {
237
+ this.debug("stderr:", data);
238
+ stderrChunks.push(data);
239
+ this.emit({ type: "stderr", data });
240
+ });
241
+
242
+ // Handle errors
243
+ proc.on("error", (error) => {
244
+ clearTimeout(timeoutId);
245
+ this.activeProcess = null;
246
+ this.emit({ type: "error", error });
247
+ reject(error);
248
+ });
249
+
250
+ // Handle exit
251
+ proc.on("close", (code, signal) => {
252
+ clearTimeout(timeoutId);
253
+ this.activeProcess = null;
254
+ this.emit({ type: "exit", code, signal });
255
+
256
+ resolve({
257
+ stdout: stdoutChunks.join(""),
258
+ stderr: stderrChunks.join(""),
259
+ exitCode: code,
260
+ signal,
261
+ duration: Date.now() - startTime,
262
+ timedOut,
263
+ stdoutChunks,
264
+ stderrChunks,
265
+ });
266
+ });
267
+ });
268
+ }
269
+
270
+ /**
271
+ * Run CLI with interactive input
272
+ */
273
+ async runInteractive(
274
+ args: string[],
275
+ inputs: InputStep[],
276
+ options?: Partial<HarnessConfig>
277
+ ): Promise<CLIResult> {
278
+ const config = { ...this.config, ...options };
279
+ const startTime = Date.now();
280
+
281
+ const stdoutChunks: string[] = [];
282
+ const stderrChunks: string[] = [];
283
+ let timedOut = false;
284
+
285
+ return new Promise<CLIResult>((resolve, reject) => {
286
+ this.debug("Spawning interactive:", "node", config.cliPath, ...args);
287
+
288
+ const env = config.inheritEnv ? { ...process.env, ...config.env } : config.env;
289
+
290
+ const proc = spawn("node", [config.cliPath, ...args], {
291
+ cwd: config.cwd,
292
+ env,
293
+ stdio: ["pipe", "pipe", "pipe"],
294
+ });
295
+
296
+ this.activeProcess = proc;
297
+
298
+ if (proc.pid !== undefined) {
299
+ this.emit({ type: "spawn", pid: proc.pid });
300
+ }
301
+
302
+ let outputBuffer = "";
303
+ let inputIndex = 0;
304
+
305
+ // Process input steps
306
+ // biome-ignore lint/complexity/noExcessiveCognitiveComplexity: Input processing requires handling multiple step types and timing conditions
307
+ const processNextInput = async (): Promise<void> => {
308
+ if (inputIndex >= inputs.length) return;
309
+
310
+ const step = inputs[inputIndex];
311
+ if (!step) return;
312
+
313
+ // Wait for specific output if specified
314
+ if (step.waitFor) {
315
+ const waitTimeout = step.waitTimeout ?? 5000;
316
+ const waitStart = Date.now();
317
+
318
+ const checkOutput = (): boolean => {
319
+ if (typeof step.waitFor === "string") {
320
+ return outputBuffer.includes(step.waitFor);
321
+ }
322
+ return step.waitFor?.test(outputBuffer) ?? false;
323
+ };
324
+
325
+ while (!checkOutput()) {
326
+ if (Date.now() - waitStart > waitTimeout) {
327
+ this.debug(`Wait timeout for: ${String(step.waitFor)}`);
328
+ break;
329
+ }
330
+ await new Promise((r) => setTimeout(r, 50));
331
+ }
332
+ }
333
+
334
+ // Delay before input
335
+ if (step.delay) {
336
+ await new Promise((r) => setTimeout(r, step.delay));
337
+ }
338
+
339
+ // Send input
340
+ this.debug("Sending input:", step.input);
341
+ proc.stdin?.write(step.input);
342
+ this.emit({ type: "input", data: step.input });
343
+
344
+ inputIndex++;
345
+
346
+ // Schedule next input
347
+ if (inputIndex < inputs.length) {
348
+ setTimeout(() => void processNextInput(), 100);
349
+ }
350
+ };
351
+
352
+ // Timeout handling
353
+ const timeoutId = setTimeout(() => {
354
+ timedOut = true;
355
+ this.emit({ type: "timeout" });
356
+ proc.kill("SIGTERM");
357
+
358
+ setTimeout(() => {
359
+ if (!proc.killed) {
360
+ proc.kill("SIGKILL");
361
+ }
362
+ }, 1000);
363
+ }, config.timeout);
364
+
365
+ // Capture stdout
366
+ proc.stdout?.setEncoding(config.encoding);
367
+ proc.stdout?.on("data", (data: string) => {
368
+ this.debug("stdout:", data);
369
+ stdoutChunks.push(data);
370
+ outputBuffer += data;
371
+ this.emit({ type: "stdout", data });
372
+ });
373
+
374
+ // Capture stderr
375
+ proc.stderr?.setEncoding(config.encoding);
376
+ proc.stderr?.on("data", (data: string) => {
377
+ this.debug("stderr:", data);
378
+ stderrChunks.push(data);
379
+ outputBuffer += data;
380
+ this.emit({ type: "stderr", data });
381
+ });
382
+
383
+ // Start processing inputs after process spawns
384
+ proc.on("spawn", () => {
385
+ void processNextInput();
386
+ });
387
+
388
+ // Handle errors
389
+ proc.on("error", (error) => {
390
+ clearTimeout(timeoutId);
391
+ this.activeProcess = null;
392
+ this.emit({ type: "error", error });
393
+ reject(error);
394
+ });
395
+
396
+ // Handle exit
397
+ proc.on("close", (code, signal) => {
398
+ clearTimeout(timeoutId);
399
+ this.activeProcess = null;
400
+ this.emit({ type: "exit", code, signal });
401
+
402
+ resolve({
403
+ stdout: stdoutChunks.join(""),
404
+ stderr: stderrChunks.join(""),
405
+ exitCode: code,
406
+ signal,
407
+ duration: Date.now() - startTime,
408
+ timedOut,
409
+ stdoutChunks,
410
+ stderrChunks,
411
+ });
412
+ });
413
+ });
414
+ }
415
+
416
+ /**
417
+ * Execute a test scenario
418
+ */
419
+ async runScenario(scenario: TestScenario): Promise<ScenarioResult> {
420
+ if (scenario.skip) {
421
+ return {
422
+ scenario,
423
+ result: {
424
+ stdout: "",
425
+ stderr: "",
426
+ exitCode: null,
427
+ signal: null,
428
+ duration: 0,
429
+ timedOut: false,
430
+ stdoutChunks: [],
431
+ stderrChunks: [],
432
+ },
433
+ passed: false,
434
+ failures: [
435
+ {
436
+ type: "exitCode",
437
+ expected: "not skipped",
438
+ actual: "skipped",
439
+ message: "Scenario was skipped",
440
+ },
441
+ ],
442
+ };
443
+ }
444
+
445
+ try {
446
+ // Run setup
447
+ if (scenario.setup) {
448
+ await scenario.setup();
449
+ }
450
+
451
+ // Execute CLI
452
+ const options: Partial<HarnessConfig> = {
453
+ timeout: scenario.timeout,
454
+ env: scenario.env,
455
+ cwd: scenario.cwd,
456
+ };
457
+
458
+ const result = scenario.inputs?.length
459
+ ? await this.runInteractive(scenario.args, scenario.inputs, options)
460
+ : await this.run(scenario.args, options);
461
+
462
+ // Assert output
463
+ const failures = assertOutput(result, scenario.expected);
464
+
465
+ // Run teardown
466
+ if (scenario.teardown) {
467
+ await scenario.teardown();
468
+ }
469
+
470
+ return {
471
+ scenario,
472
+ result,
473
+ passed: failures.length === 0,
474
+ failures,
475
+ };
476
+ } catch (error) {
477
+ return {
478
+ scenario,
479
+ result: {
480
+ stdout: "",
481
+ stderr: "",
482
+ exitCode: null,
483
+ signal: null,
484
+ duration: 0,
485
+ timedOut: false,
486
+ stdoutChunks: [],
487
+ stderrChunks: [],
488
+ },
489
+ passed: false,
490
+ failures: [],
491
+ error: error instanceof Error ? error : new Error(String(error)),
492
+ };
493
+ }
494
+ }
495
+
496
+ /**
497
+ * Run multiple scenarios
498
+ */
499
+ async runBatch(scenarios: TestScenario[]): Promise<BatchResult> {
500
+ const startTime = Date.now();
501
+ const results: ScenarioResult[] = [];
502
+
503
+ // Check for .only scenarios
504
+ const onlyScenarios = scenarios.filter((s) => s.only);
505
+ const toRun = onlyScenarios.length > 0 ? onlyScenarios : scenarios;
506
+
507
+ let passed = 0;
508
+ let failed = 0;
509
+ let skipped = 0;
510
+
511
+ for (const scenario of toRun) {
512
+ if (scenario.skip) {
513
+ skipped++;
514
+ continue;
515
+ }
516
+
517
+ const result = await this.runScenario(scenario);
518
+ results.push(result);
519
+
520
+ if (result.passed) {
521
+ passed++;
522
+ } else {
523
+ failed++;
524
+ }
525
+ }
526
+
527
+ // Count skipped from original list
528
+ skipped += scenarios.filter((s) => !toRun.includes(s)).length;
529
+
530
+ return {
531
+ total: scenarios.length,
532
+ passed,
533
+ failed,
534
+ skipped,
535
+ results,
536
+ duration: Date.now() - startTime,
537
+ };
538
+ }
539
+
540
+ /**
541
+ * Kill active process if any
542
+ */
543
+ kill(signal: NodeJS.Signals = "SIGTERM"): boolean {
544
+ if (this.activeProcess && !this.activeProcess.killed) {
545
+ this.activeProcess.kill(signal);
546
+ return true;
547
+ }
548
+ return false;
549
+ }
550
+
551
+ /**
552
+ * Create a new harness with merged config
553
+ */
554
+ withConfig(config: Partial<HarnessConfig>): CLITestHarness {
555
+ return new CLITestHarness({
556
+ ...this.config,
557
+ ...config,
558
+ });
559
+ }
560
+ }
561
+
562
+ /**
563
+ * Create a pre-configured harness instance
564
+ */
565
+ export function createHarness(config?: HarnessConfig): CLITestHarness {
566
+ return new CLITestHarness(config);
567
+ }
568
+
569
+ /**
570
+ * Quick run helper for simple test cases
571
+ */
572
+ export async function runCLI(args: string[], options?: HarnessConfig): Promise<CLIResult> {
573
+ const harness = new CLITestHarness(options);
574
+ return harness.run(args);
575
+ }
@@ -0,0 +1,57 @@
1
+ /**
2
+ * CLI E2E Test Harness
3
+ *
4
+ * Provides utilities for end-to-end testing of the Vellum CLI.
5
+ *
6
+ * @module cli/test/e2e
7
+ *
8
+ * @example
9
+ * ```typescript
10
+ * import { createHarness, TempDir, assertSuccess } from '../test/e2e/index.js';
11
+ *
12
+ * const harness = createHarness({ timeout: 10000 });
13
+ * const result = await harness.run(['--version']);
14
+ * assertSuccess(result);
15
+ * ```
16
+ */
17
+
18
+ // Assertions
19
+ export {
20
+ assertContains,
21
+ assertExitCode,
22
+ assertFailure,
23
+ assertJSON,
24
+ assertLinesInOrder,
25
+ assertNoTimeout,
26
+ assertNotContains,
27
+ assertOutput,
28
+ assertStderr,
29
+ assertStdout,
30
+ assertSuccess,
31
+ cliMatchers,
32
+ } from "./assertions.js";
33
+ // Fixtures
34
+ export type { TempDirOptions } from "./fixtures.js";
35
+ export {
36
+ createGitFixture,
37
+ createProjectFixture,
38
+ FixtureManager,
39
+ TempDir,
40
+ useFixtures,
41
+ } from "./fixtures.js";
42
+ // Harness
43
+ export { CLITestHarness, createHarness, runCLI } from "./harness.js";
44
+ // Types
45
+ export type {
46
+ AssertionFailure,
47
+ BatchResult,
48
+ CLIResult,
49
+ HarnessConfig,
50
+ HarnessEvent,
51
+ HarnessEventListener,
52
+ InputStep,
53
+ OutputMatcher,
54
+ ResolvedHarnessConfig,
55
+ ScenarioResult,
56
+ TestScenario,
57
+ } from "./types.js";