@xelauvas/xela-cli 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 (1920) hide show
  1. package/README.md +200 -0
  2. package/bin/xela +100 -0
  3. package/package.json +88 -0
  4. package/src/QueryEngine.ts +1295 -0
  5. package/src/Task.ts +125 -0
  6. package/src/Tool.ts +792 -0
  7. package/src/_shims/_generated_stubs/_universal_stub.mjs +168 -0
  8. package/src/_shims/_generated_stubs/text_stub.mjs +1 -0
  9. package/src/_shims/bun_bundle.js +9 -0
  10. package/src/_shims/cjs_stub.cjs +23 -0
  11. package/src/_shims/empty_stub.js +33 -0
  12. package/src/_shims/loader.js +352 -0
  13. package/src/_shims/openai_adapter.ts +486 -0
  14. package/src/_shims/react_compiler_runtime.js +17 -0
  15. package/src/_shims/register.js +148 -0
  16. package/src/assistant/sessionHistory.ts +87 -0
  17. package/src/bootstrap/state.ts +1759 -0
  18. package/src/bridge/bridgeApi.ts +539 -0
  19. package/src/bridge/bridgeConfig.ts +48 -0
  20. package/src/bridge/bridgeDebug.ts +135 -0
  21. package/src/bridge/bridgeEnabled.ts +202 -0
  22. package/src/bridge/bridgeMain.ts +2999 -0
  23. package/src/bridge/bridgeMessaging.ts +461 -0
  24. package/src/bridge/bridgePermissionCallbacks.ts +43 -0
  25. package/src/bridge/bridgePointer.ts +210 -0
  26. package/src/bridge/bridgeStatusUtil.ts +163 -0
  27. package/src/bridge/bridgeUI.ts +530 -0
  28. package/src/bridge/capacityWake.ts +56 -0
  29. package/src/bridge/codeSessionApi.ts +168 -0
  30. package/src/bridge/createSession.ts +384 -0
  31. package/src/bridge/debugUtils.ts +141 -0
  32. package/src/bridge/envLessBridgeConfig.ts +165 -0
  33. package/src/bridge/flushGate.ts +71 -0
  34. package/src/bridge/inboundAttachments.ts +175 -0
  35. package/src/bridge/inboundMessages.ts +80 -0
  36. package/src/bridge/initReplBridge.ts +569 -0
  37. package/src/bridge/jwtUtils.ts +256 -0
  38. package/src/bridge/pollConfig.ts +110 -0
  39. package/src/bridge/pollConfigDefaults.ts +82 -0
  40. package/src/bridge/remoteBridgeCore.ts +1008 -0
  41. package/src/bridge/replBridge.ts +2406 -0
  42. package/src/bridge/replBridgeHandle.ts +36 -0
  43. package/src/bridge/replBridgeTransport.ts +370 -0
  44. package/src/bridge/sessionIdCompat.ts +57 -0
  45. package/src/bridge/sessionRunner.ts +550 -0
  46. package/src/bridge/trustedDevice.ts +210 -0
  47. package/src/bridge/types.ts +262 -0
  48. package/src/bridge/workSecret.ts +127 -0
  49. package/src/buddy/CompanionSprite.tsx +371 -0
  50. package/src/buddy/companion.ts +133 -0
  51. package/src/buddy/prompt.ts +36 -0
  52. package/src/buddy/sprites.ts +514 -0
  53. package/src/buddy/types.ts +148 -0
  54. package/src/buddy/useBuddyNotification.tsx +98 -0
  55. package/src/cli/exit.ts +31 -0
  56. package/src/cli/handlers/agents.ts +70 -0
  57. package/src/cli/handlers/auth.ts +330 -0
  58. package/src/cli/handlers/autoMode.ts +170 -0
  59. package/src/cli/handlers/mcp.tsx +362 -0
  60. package/src/cli/handlers/plugins.ts +878 -0
  61. package/src/cli/handlers/util.tsx +110 -0
  62. package/src/cli/ndjsonSafeStringify.ts +32 -0
  63. package/src/cli/print.ts +5594 -0
  64. package/src/cli/remoteIO.ts +255 -0
  65. package/src/cli/structuredIO.ts +859 -0
  66. package/src/cli/transports/HybridTransport.ts +282 -0
  67. package/src/cli/transports/SSETransport.ts +711 -0
  68. package/src/cli/transports/SerialBatchEventUploader.ts +275 -0
  69. package/src/cli/transports/WebSocketTransport.ts +800 -0
  70. package/src/cli/transports/WorkerStateUploader.ts +131 -0
  71. package/src/cli/transports/ccrClient.ts +998 -0
  72. package/src/cli/transports/transportUtils.ts +45 -0
  73. package/src/cli/update.ts +422 -0
  74. package/src/commands/add-dir/add-dir.tsx +126 -0
  75. package/src/commands/add-dir/index.ts +11 -0
  76. package/src/commands/add-dir/validation.ts +110 -0
  77. package/src/commands/advisor.ts +109 -0
  78. package/src/commands/agents/agents.tsx +12 -0
  79. package/src/commands/agents/index.ts +10 -0
  80. package/src/commands/ant-trace/index.js +1 -0
  81. package/src/commands/autofix-pr/index.js +1 -0
  82. package/src/commands/backfill-sessions/index.js +1 -0
  83. package/src/commands/branch/branch.ts +296 -0
  84. package/src/commands/branch/index.ts +14 -0
  85. package/src/commands/break-cache/index.js +1 -0
  86. package/src/commands/bridge/bridge.tsx +509 -0
  87. package/src/commands/bridge/index.ts +26 -0
  88. package/src/commands/bridge-kick.ts +200 -0
  89. package/src/commands/brief.ts +130 -0
  90. package/src/commands/btw/btw.tsx +243 -0
  91. package/src/commands/btw/index.ts +13 -0
  92. package/src/commands/bughunter/index.js +1 -0
  93. package/src/commands/chrome/chrome.tsx +285 -0
  94. package/src/commands/chrome/index.ts +13 -0
  95. package/src/commands/clear/caches.ts +144 -0
  96. package/src/commands/clear/clear.ts +7 -0
  97. package/src/commands/clear/conversation.ts +251 -0
  98. package/src/commands/clear/index.ts +19 -0
  99. package/src/commands/color/color.ts +93 -0
  100. package/src/commands/color/index.ts +16 -0
  101. package/src/commands/commit-push-pr.ts +158 -0
  102. package/src/commands/commit.ts +92 -0
  103. package/src/commands/compact/compact.ts +287 -0
  104. package/src/commands/compact/index.ts +15 -0
  105. package/src/commands/config/config.tsx +7 -0
  106. package/src/commands/config/index.ts +11 -0
  107. package/src/commands/context/context-noninteractive.ts +325 -0
  108. package/src/commands/context/context.tsx +64 -0
  109. package/src/commands/context/index.ts +24 -0
  110. package/src/commands/copy/copy.tsx +371 -0
  111. package/src/commands/copy/index.ts +15 -0
  112. package/src/commands/cost/cost.ts +24 -0
  113. package/src/commands/cost/index.ts +23 -0
  114. package/src/commands/createMovedToPluginCommand.ts +65 -0
  115. package/src/commands/ctx_viz/index.js +1 -0
  116. package/src/commands/debug-tool-call/index.js +1 -0
  117. package/src/commands/desktop/desktop.tsx +9 -0
  118. package/src/commands/desktop/index.ts +26 -0
  119. package/src/commands/diff/diff.tsx +9 -0
  120. package/src/commands/diff/index.ts +8 -0
  121. package/src/commands/doctor/doctor.tsx +7 -0
  122. package/src/commands/doctor/index.ts +12 -0
  123. package/src/commands/effort/effort.tsx +183 -0
  124. package/src/commands/effort/index.ts +13 -0
  125. package/src/commands/env/index.js +1 -0
  126. package/src/commands/exit/exit.tsx +33 -0
  127. package/src/commands/exit/index.ts +12 -0
  128. package/src/commands/export/export.tsx +91 -0
  129. package/src/commands/export/index.ts +11 -0
  130. package/src/commands/extra-usage/extra-usage-core.ts +118 -0
  131. package/src/commands/extra-usage/extra-usage-noninteractive.ts +16 -0
  132. package/src/commands/extra-usage/extra-usage.tsx +17 -0
  133. package/src/commands/extra-usage/index.ts +31 -0
  134. package/src/commands/fast/fast.tsx +269 -0
  135. package/src/commands/fast/index.ts +26 -0
  136. package/src/commands/feedback/feedback.tsx +25 -0
  137. package/src/commands/feedback/index.ts +26 -0
  138. package/src/commands/files/files.ts +19 -0
  139. package/src/commands/files/index.ts +12 -0
  140. package/src/commands/good-claude/index.js +1 -0
  141. package/src/commands/heapdump/heapdump.ts +17 -0
  142. package/src/commands/heapdump/index.ts +12 -0
  143. package/src/commands/help/help.tsx +11 -0
  144. package/src/commands/help/index.ts +10 -0
  145. package/src/commands/hooks/hooks.tsx +13 -0
  146. package/src/commands/hooks/index.ts +11 -0
  147. package/src/commands/ide/ide.tsx +646 -0
  148. package/src/commands/ide/index.ts +11 -0
  149. package/src/commands/init-verifiers.ts +262 -0
  150. package/src/commands/init.ts +256 -0
  151. package/src/commands/insights.ts +3200 -0
  152. package/src/commands/install-github-app/ApiKeyStep.tsx +231 -0
  153. package/src/commands/install-github-app/CheckExistingSecretStep.tsx +190 -0
  154. package/src/commands/install-github-app/CheckGitHubStep.tsx +15 -0
  155. package/src/commands/install-github-app/ChooseRepoStep.tsx +211 -0
  156. package/src/commands/install-github-app/CreatingStep.tsx +65 -0
  157. package/src/commands/install-github-app/ErrorStep.tsx +85 -0
  158. package/src/commands/install-github-app/ExistingWorkflowStep.tsx +103 -0
  159. package/src/commands/install-github-app/InstallAppStep.tsx +94 -0
  160. package/src/commands/install-github-app/OAuthFlowStep.tsx +276 -0
  161. package/src/commands/install-github-app/SuccessStep.tsx +96 -0
  162. package/src/commands/install-github-app/WarningsStep.tsx +73 -0
  163. package/src/commands/install-github-app/index.ts +13 -0
  164. package/src/commands/install-github-app/install-github-app.tsx +587 -0
  165. package/src/commands/install-github-app/setupGitHubActions.ts +325 -0
  166. package/src/commands/install-slack-app/index.ts +12 -0
  167. package/src/commands/install-slack-app/install-slack-app.ts +30 -0
  168. package/src/commands/install.tsx +300 -0
  169. package/src/commands/issue/index.js +1 -0
  170. package/src/commands/keybindings/index.ts +13 -0
  171. package/src/commands/keybindings/keybindings.ts +53 -0
  172. package/src/commands/login/index.ts +14 -0
  173. package/src/commands/login/login.tsx +104 -0
  174. package/src/commands/logout/index.ts +10 -0
  175. package/src/commands/logout/logout.tsx +82 -0
  176. package/src/commands/mcp/addCommand.ts +280 -0
  177. package/src/commands/mcp/index.ts +12 -0
  178. package/src/commands/mcp/mcp.tsx +85 -0
  179. package/src/commands/mcp/xaaIdpCommand.ts +266 -0
  180. package/src/commands/memory/index.ts +10 -0
  181. package/src/commands/memory/memory.tsx +90 -0
  182. package/src/commands/mobile/index.ts +11 -0
  183. package/src/commands/mobile/mobile.tsx +274 -0
  184. package/src/commands/mock-limits/index.js +1 -0
  185. package/src/commands/model/index.ts +16 -0
  186. package/src/commands/model/model.tsx +297 -0
  187. package/src/commands/oauth-refresh/index.js +1 -0
  188. package/src/commands/onboarding/index.js +1 -0
  189. package/src/commands/output-style/index.ts +11 -0
  190. package/src/commands/output-style/output-style.tsx +7 -0
  191. package/src/commands/passes/index.ts +22 -0
  192. package/src/commands/passes/passes.tsx +24 -0
  193. package/src/commands/perf-issue/index.js +1 -0
  194. package/src/commands/permissions/index.ts +11 -0
  195. package/src/commands/permissions/permissions.tsx +10 -0
  196. package/src/commands/plan/index.ts +11 -0
  197. package/src/commands/plan/plan.tsx +122 -0
  198. package/src/commands/plugin/AddMarketplace.tsx +162 -0
  199. package/src/commands/plugin/BrowseMarketplace.tsx +802 -0
  200. package/src/commands/plugin/DiscoverPlugins.tsx +781 -0
  201. package/src/commands/plugin/ManageMarketplaces.tsx +838 -0
  202. package/src/commands/plugin/ManagePlugins.tsx +2215 -0
  203. package/src/commands/plugin/PluginErrors.tsx +124 -0
  204. package/src/commands/plugin/PluginOptionsDialog.tsx +357 -0
  205. package/src/commands/plugin/PluginOptionsFlow.tsx +135 -0
  206. package/src/commands/plugin/PluginSettings.tsx +1072 -0
  207. package/src/commands/plugin/PluginTrustWarning.tsx +32 -0
  208. package/src/commands/plugin/UnifiedInstalledCell.tsx +565 -0
  209. package/src/commands/plugin/ValidatePlugin.tsx +98 -0
  210. package/src/commands/plugin/index.tsx +11 -0
  211. package/src/commands/plugin/parseArgs.ts +103 -0
  212. package/src/commands/plugin/plugin.tsx +7 -0
  213. package/src/commands/plugin/pluginDetailsHelpers.tsx +117 -0
  214. package/src/commands/plugin/usePagination.ts +171 -0
  215. package/src/commands/pr_comments/index.ts +50 -0
  216. package/src/commands/privacy-settings/index.ts +14 -0
  217. package/src/commands/privacy-settings/privacy-settings.tsx +58 -0
  218. package/src/commands/rate-limit-options/index.ts +19 -0
  219. package/src/commands/rate-limit-options/rate-limit-options.tsx +210 -0
  220. package/src/commands/release-notes/index.ts +11 -0
  221. package/src/commands/release-notes/release-notes.ts +50 -0
  222. package/src/commands/reload-plugins/index.ts +18 -0
  223. package/src/commands/reload-plugins/reload-plugins.ts +61 -0
  224. package/src/commands/remote-env/index.ts +15 -0
  225. package/src/commands/remote-env/remote-env.tsx +7 -0
  226. package/src/commands/remote-setup/api.ts +182 -0
  227. package/src/commands/remote-setup/index.ts +20 -0
  228. package/src/commands/remote-setup/remote-setup.tsx +187 -0
  229. package/src/commands/rename/generateSessionName.ts +67 -0
  230. package/src/commands/rename/index.ts +12 -0
  231. package/src/commands/rename/rename.ts +87 -0
  232. package/src/commands/reset-limits/index.js +4 -0
  233. package/src/commands/resume/index.ts +12 -0
  234. package/src/commands/resume/resume.tsx +275 -0
  235. package/src/commands/review/UltrareviewOverageDialog.tsx +96 -0
  236. package/src/commands/review/reviewRemote.ts +316 -0
  237. package/src/commands/review/ultrareviewCommand.tsx +58 -0
  238. package/src/commands/review/ultrareviewEnabled.ts +14 -0
  239. package/src/commands/review.ts +57 -0
  240. package/src/commands/rewind/index.ts +13 -0
  241. package/src/commands/rewind/rewind.ts +13 -0
  242. package/src/commands/sandbox-toggle/index.ts +50 -0
  243. package/src/commands/sandbox-toggle/sandbox-toggle.tsx +83 -0
  244. package/src/commands/security-review.ts +243 -0
  245. package/src/commands/session/index.ts +16 -0
  246. package/src/commands/session/session.tsx +140 -0
  247. package/src/commands/share/index.js +1 -0
  248. package/src/commands/skills/index.ts +10 -0
  249. package/src/commands/skills/skills.tsx +8 -0
  250. package/src/commands/stats/index.ts +10 -0
  251. package/src/commands/stats/stats.tsx +7 -0
  252. package/src/commands/status/index.ts +12 -0
  253. package/src/commands/status/status.tsx +8 -0
  254. package/src/commands/statusline.tsx +24 -0
  255. package/src/commands/stickers/index.ts +11 -0
  256. package/src/commands/stickers/stickers.ts +16 -0
  257. package/src/commands/summary/index.js +1 -0
  258. package/src/commands/tag/index.ts +12 -0
  259. package/src/commands/tag/tag.tsx +215 -0
  260. package/src/commands/tasks/index.ts +11 -0
  261. package/src/commands/tasks/tasks.tsx +8 -0
  262. package/src/commands/teleport/index.js +1 -0
  263. package/src/commands/terminalSetup/index.ts +23 -0
  264. package/src/commands/terminalSetup/terminalSetup.tsx +531 -0
  265. package/src/commands/theme/index.ts +10 -0
  266. package/src/commands/theme/theme.tsx +57 -0
  267. package/src/commands/thinkback/index.ts +13 -0
  268. package/src/commands/thinkback/thinkback.tsx +554 -0
  269. package/src/commands/thinkback-play/index.ts +17 -0
  270. package/src/commands/thinkback-play/thinkback-play.ts +43 -0
  271. package/src/commands/ultraplan.tsx +471 -0
  272. package/src/commands/upgrade/index.ts +16 -0
  273. package/src/commands/upgrade/upgrade.tsx +38 -0
  274. package/src/commands/usage/index.ts +9 -0
  275. package/src/commands/usage/usage.tsx +7 -0
  276. package/src/commands/version.ts +22 -0
  277. package/src/commands/vim/index.ts +11 -0
  278. package/src/commands/vim/vim.ts +38 -0
  279. package/src/commands/voice/index.ts +20 -0
  280. package/src/commands/voice/voice.ts +150 -0
  281. package/src/commands.ts +754 -0
  282. package/src/components/AgentProgressLine.tsx +136 -0
  283. package/src/components/App.tsx +56 -0
  284. package/src/components/ApproveApiKey.tsx +123 -0
  285. package/src/components/AutoModeOptInDialog.tsx +142 -0
  286. package/src/components/AutoUpdater.tsx +198 -0
  287. package/src/components/AutoUpdaterWrapper.tsx +91 -0
  288. package/src/components/AwsAuthStatusBox.tsx +82 -0
  289. package/src/components/BaseTextInput.tsx +136 -0
  290. package/src/components/BashModeProgress.tsx +56 -0
  291. package/src/components/BridgeDialog.tsx +401 -0
  292. package/src/components/BypassPermissionsModeDialog.tsx +87 -0
  293. package/src/components/ChannelDowngradeDialog.tsx +102 -0
  294. package/src/components/ClaudeCodeHint/PluginHintMenu.tsx +78 -0
  295. package/src/components/ClaudeInChromeOnboarding.tsx +121 -0
  296. package/src/components/ClaudeMdExternalIncludesDialog.tsx +137 -0
  297. package/src/components/ClickableImageRef.tsx +73 -0
  298. package/src/components/CompactSummary.tsx +118 -0
  299. package/src/components/ConfigurableShortcutHint.tsx +57 -0
  300. package/src/components/ConsoleOAuthFlow.tsx +631 -0
  301. package/src/components/ContextSuggestions.tsx +47 -0
  302. package/src/components/ContextVisualization.tsx +489 -0
  303. package/src/components/CoordinatorAgentStatus.tsx +273 -0
  304. package/src/components/CostThresholdDialog.tsx +50 -0
  305. package/src/components/CtrlOToExpand.tsx +51 -0
  306. package/src/components/CustomSelect/SelectMulti.tsx +213 -0
  307. package/src/components/CustomSelect/index.ts +3 -0
  308. package/src/components/CustomSelect/option-map.ts +50 -0
  309. package/src/components/CustomSelect/select-input-option.tsx +488 -0
  310. package/src/components/CustomSelect/select-option.tsx +68 -0
  311. package/src/components/CustomSelect/select.tsx +690 -0
  312. package/src/components/CustomSelect/use-multi-select-state.ts +414 -0
  313. package/src/components/CustomSelect/use-select-input.ts +287 -0
  314. package/src/components/CustomSelect/use-select-navigation.ts +653 -0
  315. package/src/components/CustomSelect/use-select-state.ts +157 -0
  316. package/src/components/DesktopHandoff.tsx +193 -0
  317. package/src/components/DesktopUpsell/DesktopUpsellStartup.tsx +171 -0
  318. package/src/components/DevBar.tsx +49 -0
  319. package/src/components/DevChannelsDialog.tsx +105 -0
  320. package/src/components/DiagnosticsDisplay.tsx +95 -0
  321. package/src/components/EffortCallout.tsx +265 -0
  322. package/src/components/EffortIndicator.ts +42 -0
  323. package/src/components/ExitFlow.tsx +48 -0
  324. package/src/components/ExportDialog.tsx +128 -0
  325. package/src/components/FallbackToolUseErrorMessage.tsx +116 -0
  326. package/src/components/FallbackToolUseRejectedMessage.tsx +16 -0
  327. package/src/components/FastIcon.tsx +46 -0
  328. package/src/components/Feedback.tsx +592 -0
  329. package/src/components/FeedbackSurvey/FeedbackSurvey.tsx +174 -0
  330. package/src/components/FeedbackSurvey/FeedbackSurveyView.tsx +108 -0
  331. package/src/components/FeedbackSurvey/TranscriptSharePrompt.tsx +88 -0
  332. package/src/components/FeedbackSurvey/submitTranscriptShare.ts +112 -0
  333. package/src/components/FeedbackSurvey/useDebouncedDigitInput.ts +82 -0
  334. package/src/components/FeedbackSurvey/useFeedbackSurvey.tsx +296 -0
  335. package/src/components/FeedbackSurvey/useMemorySurvey.tsx +213 -0
  336. package/src/components/FeedbackSurvey/usePostCompactSurvey.tsx +206 -0
  337. package/src/components/FeedbackSurvey/useSurveyState.tsx +100 -0
  338. package/src/components/FileEditToolDiff.tsx +181 -0
  339. package/src/components/FileEditToolUpdatedMessage.tsx +124 -0
  340. package/src/components/FileEditToolUseRejectedMessage.tsx +170 -0
  341. package/src/components/FilePathLink.tsx +43 -0
  342. package/src/components/FullscreenLayout.tsx +637 -0
  343. package/src/components/GlobalSearchDialog.tsx +343 -0
  344. package/src/components/HelpV2/Commands.tsx +82 -0
  345. package/src/components/HelpV2/General.tsx +23 -0
  346. package/src/components/HelpV2/HelpV2.tsx +184 -0
  347. package/src/components/HighlightedCode/Fallback.tsx +193 -0
  348. package/src/components/HighlightedCode.tsx +190 -0
  349. package/src/components/HistorySearchDialog.tsx +118 -0
  350. package/src/components/IdeAutoConnectDialog.tsx +154 -0
  351. package/src/components/IdeOnboardingDialog.tsx +167 -0
  352. package/src/components/IdeStatusIndicator.tsx +58 -0
  353. package/src/components/IdleReturnDialog.tsx +118 -0
  354. package/src/components/InterruptedByUser.tsx +15 -0
  355. package/src/components/InvalidConfigDialog.tsx +156 -0
  356. package/src/components/InvalidSettingsDialog.tsx +89 -0
  357. package/src/components/KeybindingWarnings.tsx +55 -0
  358. package/src/components/LanguagePicker.tsx +86 -0
  359. package/src/components/LogSelector.tsx +1575 -0
  360. package/src/components/LogoV2/AnimatedAsterisk.tsx +50 -0
  361. package/src/components/LogoV2/AnimatedClawd.tsx +124 -0
  362. package/src/components/LogoV2/ChannelsNotice.tsx +266 -0
  363. package/src/components/LogoV2/Clawd.tsx +240 -0
  364. package/src/components/LogoV2/CondensedLogo.tsx +161 -0
  365. package/src/components/LogoV2/EmergencyTip.tsx +58 -0
  366. package/src/components/LogoV2/Feed.tsx +112 -0
  367. package/src/components/LogoV2/FeedColumn.tsx +59 -0
  368. package/src/components/LogoV2/GuestPassesUpsell.tsx +70 -0
  369. package/src/components/LogoV2/LogoV2.tsx +543 -0
  370. package/src/components/LogoV2/Opus1mMergeNotice.tsx +55 -0
  371. package/src/components/LogoV2/OverageCreditUpsell.tsx +166 -0
  372. package/src/components/LogoV2/VoiceModeNotice.tsx +68 -0
  373. package/src/components/LogoV2/WelcomeV2.tsx +433 -0
  374. package/src/components/LogoV2/feedConfigs.tsx +92 -0
  375. package/src/components/LspRecommendation/LspRecommendationMenu.tsx +88 -0
  376. package/src/components/MCPServerApprovalDialog.tsx +115 -0
  377. package/src/components/MCPServerDesktopImportDialog.tsx +203 -0
  378. package/src/components/MCPServerDialogCopy.tsx +15 -0
  379. package/src/components/MCPServerMultiselectDialog.tsx +133 -0
  380. package/src/components/ManagedSettingsSecurityDialog/ManagedSettingsSecurityDialog.tsx +149 -0
  381. package/src/components/ManagedSettingsSecurityDialog/utils.ts +144 -0
  382. package/src/components/Markdown.tsx +236 -0
  383. package/src/components/MarkdownTable.tsx +322 -0
  384. package/src/components/MemoryUsageIndicator.tsx +37 -0
  385. package/src/components/Message.tsx +627 -0
  386. package/src/components/MessageModel.tsx +43 -0
  387. package/src/components/MessageResponse.tsx +78 -0
  388. package/src/components/MessageRow.tsx +383 -0
  389. package/src/components/MessageSelector.tsx +831 -0
  390. package/src/components/MessageTimestamp.tsx +63 -0
  391. package/src/components/Messages.tsx +834 -0
  392. package/src/components/ModelPicker.tsx +448 -0
  393. package/src/components/NativeAutoUpdater.tsx +193 -0
  394. package/src/components/NotebookEditToolUseRejectedMessage.tsx +92 -0
  395. package/src/components/OffscreenFreeze.tsx +44 -0
  396. package/src/components/Onboarding.tsx +244 -0
  397. package/src/components/OutputStylePicker.tsx +112 -0
  398. package/src/components/PackageManagerAutoUpdater.tsx +104 -0
  399. package/src/components/Passes/Passes.tsx +184 -0
  400. package/src/components/PrBadge.tsx +97 -0
  401. package/src/components/PressEnterToContinue.tsx +15 -0
  402. package/src/components/PromptInput/HistorySearchInput.tsx +51 -0
  403. package/src/components/PromptInput/IssueFlagBanner.tsx +12 -0
  404. package/src/components/PromptInput/Notifications.tsx +332 -0
  405. package/src/components/PromptInput/PromptInput.tsx +2339 -0
  406. package/src/components/PromptInput/PromptInputFooter.tsx +191 -0
  407. package/src/components/PromptInput/PromptInputFooterLeftSide.tsx +517 -0
  408. package/src/components/PromptInput/PromptInputFooterSuggestions.tsx +293 -0
  409. package/src/components/PromptInput/PromptInputHelpMenu.tsx +358 -0
  410. package/src/components/PromptInput/PromptInputModeIndicator.tsx +93 -0
  411. package/src/components/PromptInput/PromptInputQueuedCommands.tsx +117 -0
  412. package/src/components/PromptInput/PromptInputStashNotice.tsx +25 -0
  413. package/src/components/PromptInput/SandboxPromptFooterHint.tsx +64 -0
  414. package/src/components/PromptInput/ShimmeredInput.tsx +143 -0
  415. package/src/components/PromptInput/VoiceIndicator.tsx +137 -0
  416. package/src/components/PromptInput/inputModes.ts +33 -0
  417. package/src/components/PromptInput/inputPaste.ts +90 -0
  418. package/src/components/PromptInput/useMaybeTruncateInput.ts +58 -0
  419. package/src/components/PromptInput/usePromptInputPlaceholder.ts +76 -0
  420. package/src/components/PromptInput/useShowFastIconHint.ts +31 -0
  421. package/src/components/PromptInput/useSwarmBanner.ts +155 -0
  422. package/src/components/PromptInput/utils.ts +60 -0
  423. package/src/components/QuickOpenDialog.tsx +244 -0
  424. package/src/components/RemoteCallout.tsx +76 -0
  425. package/src/components/RemoteEnvironmentDialog.tsx +340 -0
  426. package/src/components/ResumeTask.tsx +268 -0
  427. package/src/components/SandboxViolationExpandedView.tsx +99 -0
  428. package/src/components/ScrollKeybindingHandler.tsx +1012 -0
  429. package/src/components/SearchBox.tsx +72 -0
  430. package/src/components/SentryErrorBoundary.ts +28 -0
  431. package/src/components/SessionBackgroundHint.tsx +108 -0
  432. package/src/components/SessionPreview.tsx +194 -0
  433. package/src/components/Settings/Config.tsx +1822 -0
  434. package/src/components/Settings/Settings.tsx +137 -0
  435. package/src/components/Settings/Status.tsx +241 -0
  436. package/src/components/Settings/Usage.tsx +377 -0
  437. package/src/components/ShowInIDEPrompt.tsx +170 -0
  438. package/src/components/SkillImprovementSurvey.tsx +152 -0
  439. package/src/components/Spinner/FlashingChar.tsx +61 -0
  440. package/src/components/Spinner/GlimmerMessage.tsx +328 -0
  441. package/src/components/Spinner/ShimmerChar.tsx +36 -0
  442. package/src/components/Spinner/SpinnerAnimationRow.tsx +265 -0
  443. package/src/components/Spinner/SpinnerGlyph.tsx +80 -0
  444. package/src/components/Spinner/TeammateSpinnerLine.tsx +233 -0
  445. package/src/components/Spinner/TeammateSpinnerTree.tsx +272 -0
  446. package/src/components/Spinner/index.ts +10 -0
  447. package/src/components/Spinner/teammateSelectHint.ts +1 -0
  448. package/src/components/Spinner/useShimmerAnimation.ts +31 -0
  449. package/src/components/Spinner/useStalledAnimation.ts +75 -0
  450. package/src/components/Spinner/utils.ts +84 -0
  451. package/src/components/Spinner.tsx +562 -0
  452. package/src/components/Stats.tsx +1228 -0
  453. package/src/components/StatusLine.tsx +324 -0
  454. package/src/components/StatusNotices.tsx +55 -0
  455. package/src/components/StructuredDiff/Fallback.tsx +487 -0
  456. package/src/components/StructuredDiff/colorDiff.ts +37 -0
  457. package/src/components/StructuredDiff.tsx +190 -0
  458. package/src/components/StructuredDiffList.tsx +30 -0
  459. package/src/components/TagTabs.tsx +139 -0
  460. package/src/components/TaskListV2.tsx +378 -0
  461. package/src/components/TeammateViewHeader.tsx +82 -0
  462. package/src/components/TeleportError.tsx +189 -0
  463. package/src/components/TeleportProgress.tsx +140 -0
  464. package/src/components/TeleportRepoMismatchDialog.tsx +104 -0
  465. package/src/components/TeleportResumeWrapper.tsx +167 -0
  466. package/src/components/TeleportStash.tsx +116 -0
  467. package/src/components/TextInput.tsx +124 -0
  468. package/src/components/ThemePicker.tsx +333 -0
  469. package/src/components/ThinkingToggle.tsx +153 -0
  470. package/src/components/TokenWarning.tsx +179 -0
  471. package/src/components/ToolUseLoader.tsx +42 -0
  472. package/src/components/TrustDialog/TrustDialog.tsx +290 -0
  473. package/src/components/TrustDialog/utils.ts +245 -0
  474. package/src/components/ValidationErrorsList.tsx +148 -0
  475. package/src/components/VimTextInput.tsx +140 -0
  476. package/src/components/VirtualMessageList.tsx +1082 -0
  477. package/src/components/WorkflowMultiselectDialog.tsx +128 -0
  478. package/src/components/WorktreeExitDialog.tsx +231 -0
  479. package/src/components/agents/AgentDetail.tsx +220 -0
  480. package/src/components/agents/AgentEditor.tsx +178 -0
  481. package/src/components/agents/AgentNavigationFooter.tsx +26 -0
  482. package/src/components/agents/AgentsList.tsx +440 -0
  483. package/src/components/agents/AgentsMenu.tsx +800 -0
  484. package/src/components/agents/ColorPicker.tsx +112 -0
  485. package/src/components/agents/ModelSelector.tsx +68 -0
  486. package/src/components/agents/ToolSelector.tsx +562 -0
  487. package/src/components/agents/agentFileUtils.ts +272 -0
  488. package/src/components/agents/generateAgent.ts +197 -0
  489. package/src/components/agents/new-agent-creation/CreateAgentWizard.tsx +97 -0
  490. package/src/components/agents/new-agent-creation/wizard-steps/ColorStep.tsx +84 -0
  491. package/src/components/agents/new-agent-creation/wizard-steps/ConfirmStep.tsx +378 -0
  492. package/src/components/agents/new-agent-creation/wizard-steps/ConfirmStepWrapper.tsx +74 -0
  493. package/src/components/agents/new-agent-creation/wizard-steps/DescriptionStep.tsx +123 -0
  494. package/src/components/agents/new-agent-creation/wizard-steps/GenerateStep.tsx +143 -0
  495. package/src/components/agents/new-agent-creation/wizard-steps/LocationStep.tsx +80 -0
  496. package/src/components/agents/new-agent-creation/wizard-steps/MemoryStep.tsx +113 -0
  497. package/src/components/agents/new-agent-creation/wizard-steps/MethodStep.tsx +80 -0
  498. package/src/components/agents/new-agent-creation/wizard-steps/ModelStep.tsx +52 -0
  499. package/src/components/agents/new-agent-creation/wizard-steps/PromptStep.tsx +128 -0
  500. package/src/components/agents/new-agent-creation/wizard-steps/ToolsStep.tsx +61 -0
  501. package/src/components/agents/new-agent-creation/wizard-steps/TypeStep.tsx +103 -0
  502. package/src/components/agents/types.ts +27 -0
  503. package/src/components/agents/utils.ts +18 -0
  504. package/src/components/agents/validateAgent.ts +109 -0
  505. package/src/components/design-system/Byline.tsx +77 -0
  506. package/src/components/design-system/Dialog.tsx +138 -0
  507. package/src/components/design-system/Divider.tsx +149 -0
  508. package/src/components/design-system/FuzzyPicker.tsx +312 -0
  509. package/src/components/design-system/KeyboardShortcutHint.tsx +81 -0
  510. package/src/components/design-system/ListItem.tsx +244 -0
  511. package/src/components/design-system/LoadingState.tsx +94 -0
  512. package/src/components/design-system/Pane.tsx +77 -0
  513. package/src/components/design-system/ProgressBar.tsx +86 -0
  514. package/src/components/design-system/Ratchet.tsx +80 -0
  515. package/src/components/design-system/StatusIcon.tsx +95 -0
  516. package/src/components/design-system/Tabs.tsx +340 -0
  517. package/src/components/design-system/ThemeProvider.tsx +174 -0
  518. package/src/components/design-system/ThemedBox.tsx +156 -0
  519. package/src/components/design-system/ThemedText.tsx +124 -0
  520. package/src/components/design-system/color.ts +30 -0
  521. package/src/components/diff/DiffDetailView.tsx +281 -0
  522. package/src/components/diff/DiffDialog.tsx +383 -0
  523. package/src/components/diff/DiffFileList.tsx +292 -0
  524. package/src/components/grove/Grove.tsx +463 -0
  525. package/src/components/hooks/HooksConfigMenu.tsx +578 -0
  526. package/src/components/hooks/PromptDialog.tsx +90 -0
  527. package/src/components/hooks/SelectEventMode.tsx +127 -0
  528. package/src/components/hooks/SelectHookMode.tsx +112 -0
  529. package/src/components/hooks/SelectMatcherMode.tsx +144 -0
  530. package/src/components/hooks/ViewHookMode.tsx +199 -0
  531. package/src/components/mcp/CapabilitiesSection.tsx +61 -0
  532. package/src/components/mcp/ElicitationDialog.tsx +1169 -0
  533. package/src/components/mcp/MCPAgentServerMenu.tsx +183 -0
  534. package/src/components/mcp/MCPListPanel.tsx +504 -0
  535. package/src/components/mcp/MCPReconnect.tsx +167 -0
  536. package/src/components/mcp/MCPRemoteServerMenu.tsx +649 -0
  537. package/src/components/mcp/MCPSettings.tsx +398 -0
  538. package/src/components/mcp/MCPStdioServerMenu.tsx +177 -0
  539. package/src/components/mcp/MCPToolDetailView.tsx +212 -0
  540. package/src/components/mcp/MCPToolListView.tsx +141 -0
  541. package/src/components/mcp/McpParsingWarnings.tsx +213 -0
  542. package/src/components/mcp/index.ts +9 -0
  543. package/src/components/mcp/utils/reconnectHelpers.tsx +49 -0
  544. package/src/components/memory/MemoryFileSelector.tsx +438 -0
  545. package/src/components/memory/MemoryUpdateNotification.tsx +45 -0
  546. package/src/components/messageActions.tsx +450 -0
  547. package/src/components/messages/AdvisorMessage.tsx +158 -0
  548. package/src/components/messages/AssistantRedactedThinkingMessage.tsx +31 -0
  549. package/src/components/messages/AssistantTextMessage.tsx +270 -0
  550. package/src/components/messages/AssistantThinkingMessage.tsx +86 -0
  551. package/src/components/messages/AssistantToolUseMessage.tsx +368 -0
  552. package/src/components/messages/AttachmentMessage.tsx +536 -0
  553. package/src/components/messages/CollapsedReadSearchContent.tsx +484 -0
  554. package/src/components/messages/CompactBoundaryMessage.tsx +18 -0
  555. package/src/components/messages/GroupedToolUseContent.tsx +58 -0
  556. package/src/components/messages/HighlightedThinkingText.tsx +162 -0
  557. package/src/components/messages/HookProgressMessage.tsx +116 -0
  558. package/src/components/messages/PlanApprovalMessage.tsx +222 -0
  559. package/src/components/messages/RateLimitMessage.tsx +161 -0
  560. package/src/components/messages/ShutdownMessage.tsx +132 -0
  561. package/src/components/messages/SystemAPIErrorMessage.tsx +141 -0
  562. package/src/components/messages/SystemTextMessage.tsx +827 -0
  563. package/src/components/messages/TaskAssignmentMessage.tsx +76 -0
  564. package/src/components/messages/UserAgentNotificationMessage.tsx +83 -0
  565. package/src/components/messages/UserBashInputMessage.tsx +58 -0
  566. package/src/components/messages/UserBashOutputMessage.tsx +54 -0
  567. package/src/components/messages/UserChannelMessage.tsx +137 -0
  568. package/src/components/messages/UserCommandMessage.tsx +108 -0
  569. package/src/components/messages/UserImageMessage.tsx +59 -0
  570. package/src/components/messages/UserLocalCommandOutputMessage.tsx +167 -0
  571. package/src/components/messages/UserMemoryInputMessage.tsx +75 -0
  572. package/src/components/messages/UserPlanMessage.tsx +42 -0
  573. package/src/components/messages/UserPromptMessage.tsx +80 -0
  574. package/src/components/messages/UserResourceUpdateMessage.tsx +121 -0
  575. package/src/components/messages/UserTeammateMessage.tsx +206 -0
  576. package/src/components/messages/UserTextMessage.tsx +275 -0
  577. package/src/components/messages/UserToolResultMessage/RejectedPlanMessage.tsx +31 -0
  578. package/src/components/messages/UserToolResultMessage/RejectedToolUseMessage.tsx +16 -0
  579. package/src/components/messages/UserToolResultMessage/UserToolCanceledMessage.tsx +16 -0
  580. package/src/components/messages/UserToolResultMessage/UserToolErrorMessage.tsx +103 -0
  581. package/src/components/messages/UserToolResultMessage/UserToolRejectMessage.tsx +95 -0
  582. package/src/components/messages/UserToolResultMessage/UserToolResultMessage.tsx +106 -0
  583. package/src/components/messages/UserToolResultMessage/UserToolSuccessMessage.tsx +104 -0
  584. package/src/components/messages/UserToolResultMessage/utils.tsx +44 -0
  585. package/src/components/messages/nullRenderingAttachments.ts +70 -0
  586. package/src/components/messages/teamMemCollapsed.tsx +140 -0
  587. package/src/components/messages/teamMemSaved.ts +19 -0
  588. package/src/components/permissions/AskUserQuestionPermissionRequest/AskUserQuestionPermissionRequest.tsx +645 -0
  589. package/src/components/permissions/AskUserQuestionPermissionRequest/PreviewBox.tsx +229 -0
  590. package/src/components/permissions/AskUserQuestionPermissionRequest/PreviewQuestionView.tsx +328 -0
  591. package/src/components/permissions/AskUserQuestionPermissionRequest/QuestionNavigationBar.tsx +178 -0
  592. package/src/components/permissions/AskUserQuestionPermissionRequest/QuestionView.tsx +465 -0
  593. package/src/components/permissions/AskUserQuestionPermissionRequest/SubmitQuestionsView.tsx +144 -0
  594. package/src/components/permissions/AskUserQuestionPermissionRequest/use-multiple-choice-state.ts +179 -0
  595. package/src/components/permissions/BashPermissionRequest/BashPermissionRequest.tsx +482 -0
  596. package/src/components/permissions/BashPermissionRequest/bashToolUseOptions.tsx +147 -0
  597. package/src/components/permissions/ComputerUseApproval/ComputerUseApproval.tsx +441 -0
  598. package/src/components/permissions/EnterPlanModePermissionRequest/EnterPlanModePermissionRequest.tsx +122 -0
  599. package/src/components/permissions/ExitPlanModePermissionRequest/ExitPlanModePermissionRequest.tsx +768 -0
  600. package/src/components/permissions/FallbackPermissionRequest.tsx +333 -0
  601. package/src/components/permissions/FileEditPermissionRequest/FileEditPermissionRequest.tsx +182 -0
  602. package/src/components/permissions/FilePermissionDialog/FilePermissionDialog.tsx +204 -0
  603. package/src/components/permissions/FilePermissionDialog/ideDiffConfig.ts +42 -0
  604. package/src/components/permissions/FilePermissionDialog/permissionOptions.tsx +177 -0
  605. package/src/components/permissions/FilePermissionDialog/useFilePermissionDialog.ts +212 -0
  606. package/src/components/permissions/FilePermissionDialog/usePermissionHandler.ts +185 -0
  607. package/src/components/permissions/FileWritePermissionRequest/FileWritePermissionRequest.tsx +161 -0
  608. package/src/components/permissions/FileWritePermissionRequest/FileWriteToolDiff.tsx +89 -0
  609. package/src/components/permissions/FilesystemPermissionRequest/FilesystemPermissionRequest.tsx +115 -0
  610. package/src/components/permissions/NotebookEditPermissionRequest/NotebookEditPermissionRequest.tsx +166 -0
  611. package/src/components/permissions/NotebookEditPermissionRequest/NotebookEditToolDiff.tsx +235 -0
  612. package/src/components/permissions/PermissionDecisionDebugInfo.tsx +460 -0
  613. package/src/components/permissions/PermissionDialog.tsx +72 -0
  614. package/src/components/permissions/PermissionExplanation.tsx +272 -0
  615. package/src/components/permissions/PermissionPrompt.tsx +336 -0
  616. package/src/components/permissions/PermissionRequest.tsx +217 -0
  617. package/src/components/permissions/PermissionRequestTitle.tsx +66 -0
  618. package/src/components/permissions/PermissionRuleExplanation.tsx +121 -0
  619. package/src/components/permissions/PowerShellPermissionRequest/PowerShellPermissionRequest.tsx +235 -0
  620. package/src/components/permissions/PowerShellPermissionRequest/powershellToolUseOptions.tsx +91 -0
  621. package/src/components/permissions/SandboxPermissionRequest.tsx +163 -0
  622. package/src/components/permissions/SedEditPermissionRequest/SedEditPermissionRequest.tsx +230 -0
  623. package/src/components/permissions/SkillPermissionRequest/SkillPermissionRequest.tsx +369 -0
  624. package/src/components/permissions/WebFetchPermissionRequest/WebFetchPermissionRequest.tsx +258 -0
  625. package/src/components/permissions/WorkerBadge.tsx +49 -0
  626. package/src/components/permissions/WorkerPendingPermission.tsx +105 -0
  627. package/src/components/permissions/hooks.ts +209 -0
  628. package/src/components/permissions/rules/AddPermissionRules.tsx +180 -0
  629. package/src/components/permissions/rules/AddWorkspaceDirectory.tsx +340 -0
  630. package/src/components/permissions/rules/PermissionRuleDescription.tsx +76 -0
  631. package/src/components/permissions/rules/PermissionRuleInput.tsx +138 -0
  632. package/src/components/permissions/rules/PermissionRuleList.tsx +1179 -0
  633. package/src/components/permissions/rules/RecentDenialsTab.tsx +207 -0
  634. package/src/components/permissions/rules/RemoveWorkspaceDirectory.tsx +110 -0
  635. package/src/components/permissions/rules/WorkspaceTab.tsx +150 -0
  636. package/src/components/permissions/shellPermissionHelpers.tsx +164 -0
  637. package/src/components/permissions/useShellPermissionFeedback.ts +148 -0
  638. package/src/components/permissions/utils.ts +25 -0
  639. package/src/components/sandbox/SandboxConfigTab.tsx +45 -0
  640. package/src/components/sandbox/SandboxDependenciesTab.tsx +120 -0
  641. package/src/components/sandbox/SandboxDoctorSection.tsx +46 -0
  642. package/src/components/sandbox/SandboxOverridesTab.tsx +193 -0
  643. package/src/components/sandbox/SandboxSettings.tsx +296 -0
  644. package/src/components/shell/ExpandShellOutputContext.tsx +36 -0
  645. package/src/components/shell/OutputLine.tsx +118 -0
  646. package/src/components/shell/ShellProgressMessage.tsx +150 -0
  647. package/src/components/shell/ShellTimeDisplay.tsx +74 -0
  648. package/src/components/skills/SkillsMenu.tsx +237 -0
  649. package/src/components/tasks/AsyncAgentDetailDialog.tsx +229 -0
  650. package/src/components/tasks/BackgroundTask.tsx +345 -0
  651. package/src/components/tasks/BackgroundTaskStatus.tsx +429 -0
  652. package/src/components/tasks/BackgroundTasksDialog.tsx +652 -0
  653. package/src/components/tasks/DreamDetailDialog.tsx +251 -0
  654. package/src/components/tasks/InProcessTeammateDetailDialog.tsx +266 -0
  655. package/src/components/tasks/RemoteSessionDetailDialog.tsx +904 -0
  656. package/src/components/tasks/RemoteSessionProgress.tsx +243 -0
  657. package/src/components/tasks/ShellDetailDialog.tsx +404 -0
  658. package/src/components/tasks/ShellProgress.tsx +87 -0
  659. package/src/components/tasks/renderToolActivity.tsx +33 -0
  660. package/src/components/tasks/taskStatusUtils.tsx +107 -0
  661. package/src/components/teams/TeamStatus.tsx +80 -0
  662. package/src/components/teams/TeamsDialog.tsx +715 -0
  663. package/src/components/ui/OrderedList.tsx +71 -0
  664. package/src/components/ui/OrderedListItem.tsx +45 -0
  665. package/src/components/ui/TreeSelect.tsx +397 -0
  666. package/src/components/wizard/WizardDialogLayout.tsx +65 -0
  667. package/src/components/wizard/WizardNavigationFooter.tsx +24 -0
  668. package/src/components/wizard/WizardProvider.tsx +213 -0
  669. package/src/components/wizard/index.ts +9 -0
  670. package/src/components/wizard/useWizard.ts +13 -0
  671. package/src/constants/apiLimits.ts +94 -0
  672. package/src/constants/betas.ts +52 -0
  673. package/src/constants/common.ts +33 -0
  674. package/src/constants/cyberRiskInstruction.ts +24 -0
  675. package/src/constants/errorIds.ts +15 -0
  676. package/src/constants/figures.ts +45 -0
  677. package/src/constants/files.ts +156 -0
  678. package/src/constants/github-app.ts +144 -0
  679. package/src/constants/keys.ts +11 -0
  680. package/src/constants/messages.ts +1 -0
  681. package/src/constants/oauth.ts +234 -0
  682. package/src/constants/outputStyles.ts +216 -0
  683. package/src/constants/product.ts +76 -0
  684. package/src/constants/prompts.ts +914 -0
  685. package/src/constants/spinnerVerbs.ts +204 -0
  686. package/src/constants/system.ts +95 -0
  687. package/src/constants/systemPromptSections.ts +68 -0
  688. package/src/constants/toolLimits.ts +56 -0
  689. package/src/constants/tools.ts +112 -0
  690. package/src/constants/turnCompletionVerbs.ts +12 -0
  691. package/src/constants/xml.ts +86 -0
  692. package/src/context/QueuedMessageContext.tsx +63 -0
  693. package/src/context/fpsMetrics.tsx +30 -0
  694. package/src/context/mailbox.tsx +38 -0
  695. package/src/context/modalContext.tsx +58 -0
  696. package/src/context/notifications.tsx +240 -0
  697. package/src/context/overlayContext.tsx +151 -0
  698. package/src/context/promptOverlayContext.tsx +125 -0
  699. package/src/context/stats.tsx +220 -0
  700. package/src/context/voice.tsx +88 -0
  701. package/src/context.ts +189 -0
  702. package/src/coordinator/coordinatorMode.ts +369 -0
  703. package/src/cost-tracker.ts +323 -0
  704. package/src/costHook.ts +22 -0
  705. package/src/dialogLaunchers.tsx +133 -0
  706. package/src/entrypoints/agentSdkTypes.ts +443 -0
  707. package/src/entrypoints/cli.tsx +303 -0
  708. package/src/entrypoints/init.ts +340 -0
  709. package/src/entrypoints/mcp.ts +196 -0
  710. package/src/entrypoints/sandboxTypes.ts +156 -0
  711. package/src/entrypoints/sdk/controlSchemas.ts +663 -0
  712. package/src/entrypoints/sdk/coreSchemas.ts +1889 -0
  713. package/src/entrypoints/sdk/coreTypes.ts +62 -0
  714. package/src/history.ts +464 -0
  715. package/src/hooks/fileSuggestions.ts +811 -0
  716. package/src/hooks/notifs/useAutoModeUnavailableNotification.ts +56 -0
  717. package/src/hooks/notifs/useCanSwitchToExistingSubscription.tsx +60 -0
  718. package/src/hooks/notifs/useDeprecationWarningNotification.tsx +44 -0
  719. package/src/hooks/notifs/useFastModeNotification.tsx +162 -0
  720. package/src/hooks/notifs/useIDEStatusIndicator.tsx +186 -0
  721. package/src/hooks/notifs/useInstallMessages.tsx +26 -0
  722. package/src/hooks/notifs/useLspInitializationNotification.tsx +143 -0
  723. package/src/hooks/notifs/useMcpConnectivityStatus.tsx +88 -0
  724. package/src/hooks/notifs/useModelMigrationNotifications.tsx +52 -0
  725. package/src/hooks/notifs/useNpmDeprecationNotification.tsx +25 -0
  726. package/src/hooks/notifs/usePluginAutoupdateNotification.tsx +83 -0
  727. package/src/hooks/notifs/usePluginInstallationStatus.tsx +128 -0
  728. package/src/hooks/notifs/useRateLimitWarningNotification.tsx +114 -0
  729. package/src/hooks/notifs/useSettingsErrors.tsx +69 -0
  730. package/src/hooks/notifs/useStartupNotification.ts +41 -0
  731. package/src/hooks/notifs/useTeammateShutdownNotification.ts +78 -0
  732. package/src/hooks/renderPlaceholder.ts +51 -0
  733. package/src/hooks/toolPermission/PermissionContext.ts +388 -0
  734. package/src/hooks/toolPermission/handlers/coordinatorHandler.ts +65 -0
  735. package/src/hooks/toolPermission/handlers/interactiveHandler.ts +536 -0
  736. package/src/hooks/toolPermission/handlers/swarmWorkerHandler.ts +159 -0
  737. package/src/hooks/toolPermission/permissionLogging.ts +238 -0
  738. package/src/hooks/unifiedSuggestions.ts +202 -0
  739. package/src/hooks/useAfterFirstRender.ts +17 -0
  740. package/src/hooks/useApiKeyVerification.ts +84 -0
  741. package/src/hooks/useArrowKeyHistory.tsx +229 -0
  742. package/src/hooks/useAssistantHistory.ts +250 -0
  743. package/src/hooks/useAwaySummary.ts +125 -0
  744. package/src/hooks/useBackgroundTaskNavigation.ts +251 -0
  745. package/src/hooks/useBlink.ts +34 -0
  746. package/src/hooks/useCanUseTool.tsx +204 -0
  747. package/src/hooks/useCancelRequest.ts +276 -0
  748. package/src/hooks/useChromeExtensionNotification.tsx +50 -0
  749. package/src/hooks/useClaudeCodeHintRecommendation.tsx +129 -0
  750. package/src/hooks/useClipboardImageHint.ts +77 -0
  751. package/src/hooks/useCommandKeybindings.tsx +108 -0
  752. package/src/hooks/useCommandQueue.ts +15 -0
  753. package/src/hooks/useCopyOnSelect.ts +98 -0
  754. package/src/hooks/useDeferredHookMessages.ts +46 -0
  755. package/src/hooks/useDiffData.ts +110 -0
  756. package/src/hooks/useDiffInIDE.ts +379 -0
  757. package/src/hooks/useDirectConnect.ts +229 -0
  758. package/src/hooks/useDoublePress.ts +62 -0
  759. package/src/hooks/useDynamicConfig.ts +22 -0
  760. package/src/hooks/useElapsedTime.ts +37 -0
  761. package/src/hooks/useExitOnCtrlCD.ts +95 -0
  762. package/src/hooks/useExitOnCtrlCDWithKeybindings.ts +24 -0
  763. package/src/hooks/useFileHistorySnapshotInit.ts +25 -0
  764. package/src/hooks/useGlobalKeybindings.tsx +249 -0
  765. package/src/hooks/useHistorySearch.ts +303 -0
  766. package/src/hooks/useIDEIntegration.tsx +70 -0
  767. package/src/hooks/useIdeAtMentioned.ts +76 -0
  768. package/src/hooks/useIdeConnectionStatus.ts +33 -0
  769. package/src/hooks/useIdeLogging.ts +41 -0
  770. package/src/hooks/useIdeSelection.ts +150 -0
  771. package/src/hooks/useInboxPoller.ts +969 -0
  772. package/src/hooks/useInputBuffer.ts +132 -0
  773. package/src/hooks/useIssueFlagBanner.ts +133 -0
  774. package/src/hooks/useLogMessages.ts +119 -0
  775. package/src/hooks/useLspPluginRecommendation.tsx +194 -0
  776. package/src/hooks/useMailboxBridge.ts +21 -0
  777. package/src/hooks/useMainLoopModel.ts +34 -0
  778. package/src/hooks/useManagePlugins.ts +304 -0
  779. package/src/hooks/useMemoryUsage.ts +39 -0
  780. package/src/hooks/useMergedClients.ts +23 -0
  781. package/src/hooks/useMergedCommands.ts +15 -0
  782. package/src/hooks/useMergedTools.ts +44 -0
  783. package/src/hooks/useMinDisplayTime.ts +35 -0
  784. package/src/hooks/useNotifyAfterTimeout.ts +65 -0
  785. package/src/hooks/useOfficialMarketplaceNotification.tsx +48 -0
  786. package/src/hooks/usePasteHandler.ts +285 -0
  787. package/src/hooks/usePluginRecommendationBase.tsx +105 -0
  788. package/src/hooks/usePrStatus.ts +106 -0
  789. package/src/hooks/usePromptSuggestion.ts +177 -0
  790. package/src/hooks/usePromptsFromClaudeInChrome.tsx +71 -0
  791. package/src/hooks/useQueueProcessor.ts +68 -0
  792. package/src/hooks/useRemoteSession.ts +605 -0
  793. package/src/hooks/useReplBridge.tsx +723 -0
  794. package/src/hooks/useSSHSession.ts +241 -0
  795. package/src/hooks/useScheduledTasks.ts +139 -0
  796. package/src/hooks/useSearchInput.ts +364 -0
  797. package/src/hooks/useSessionBackgrounding.ts +158 -0
  798. package/src/hooks/useSettings.ts +17 -0
  799. package/src/hooks/useSettingsChange.ts +25 -0
  800. package/src/hooks/useSkillImprovementSurvey.ts +105 -0
  801. package/src/hooks/useSkillsChange.ts +62 -0
  802. package/src/hooks/useSwarmInitialization.ts +81 -0
  803. package/src/hooks/useSwarmPermissionPoller.ts +330 -0
  804. package/src/hooks/useTaskListWatcher.ts +221 -0
  805. package/src/hooks/useTasksV2.ts +250 -0
  806. package/src/hooks/useTeammateViewAutoExit.ts +63 -0
  807. package/src/hooks/useTeleportResume.tsx +85 -0
  808. package/src/hooks/useTerminalSize.ts +15 -0
  809. package/src/hooks/useTextInput.ts +529 -0
  810. package/src/hooks/useTimeout.ts +14 -0
  811. package/src/hooks/useTurnDiffs.ts +213 -0
  812. package/src/hooks/useTypeahead.tsx +1385 -0
  813. package/src/hooks/useUpdateNotification.ts +34 -0
  814. package/src/hooks/useVimInput.ts +316 -0
  815. package/src/hooks/useVirtualScroll.ts +721 -0
  816. package/src/hooks/useVoice.ts +1144 -0
  817. package/src/hooks/useVoiceEnabled.ts +25 -0
  818. package/src/hooks/useVoiceIntegration.tsx +677 -0
  819. package/src/ink/Ansi.tsx +292 -0
  820. package/src/ink/bidi.ts +139 -0
  821. package/src/ink/clearTerminal.ts +74 -0
  822. package/src/ink/colorize.ts +231 -0
  823. package/src/ink/components/AlternateScreen.tsx +80 -0
  824. package/src/ink/components/App.tsx +659 -0
  825. package/src/ink/components/AppContext.ts +21 -0
  826. package/src/ink/components/Box.tsx +214 -0
  827. package/src/ink/components/Button.tsx +192 -0
  828. package/src/ink/components/ClockContext.tsx +112 -0
  829. package/src/ink/components/CursorDeclarationContext.ts +32 -0
  830. package/src/ink/components/ErrorOverview.tsx +109 -0
  831. package/src/ink/components/Link.tsx +42 -0
  832. package/src/ink/components/Newline.tsx +39 -0
  833. package/src/ink/components/NoSelect.tsx +68 -0
  834. package/src/ink/components/RawAnsi.tsx +57 -0
  835. package/src/ink/components/ScrollBox.tsx +237 -0
  836. package/src/ink/components/Spacer.tsx +20 -0
  837. package/src/ink/components/StdinContext.ts +49 -0
  838. package/src/ink/components/TerminalFocusContext.tsx +52 -0
  839. package/src/ink/components/TerminalSizeContext.tsx +7 -0
  840. package/src/ink/components/Text.tsx +254 -0
  841. package/src/ink/constants.ts +2 -0
  842. package/src/ink/dom.ts +484 -0
  843. package/src/ink/events/click-event.ts +38 -0
  844. package/src/ink/events/dispatcher.ts +233 -0
  845. package/src/ink/events/emitter.ts +39 -0
  846. package/src/ink/events/event-handlers.ts +73 -0
  847. package/src/ink/events/event.ts +11 -0
  848. package/src/ink/events/focus-event.ts +21 -0
  849. package/src/ink/events/input-event.ts +205 -0
  850. package/src/ink/events/keyboard-event.ts +51 -0
  851. package/src/ink/events/terminal-event.ts +107 -0
  852. package/src/ink/events/terminal-focus-event.ts +19 -0
  853. package/src/ink/focus.ts +181 -0
  854. package/src/ink/frame.ts +124 -0
  855. package/src/ink/get-max-width.ts +27 -0
  856. package/src/ink/hit-test.ts +130 -0
  857. package/src/ink/hooks/use-animation-frame.ts +57 -0
  858. package/src/ink/hooks/use-app.ts +8 -0
  859. package/src/ink/hooks/use-declared-cursor.ts +73 -0
  860. package/src/ink/hooks/use-input.ts +92 -0
  861. package/src/ink/hooks/use-interval.ts +67 -0
  862. package/src/ink/hooks/use-search-highlight.ts +53 -0
  863. package/src/ink/hooks/use-selection.ts +104 -0
  864. package/src/ink/hooks/use-stdin.ts +8 -0
  865. package/src/ink/hooks/use-tab-status.ts +72 -0
  866. package/src/ink/hooks/use-terminal-focus.ts +16 -0
  867. package/src/ink/hooks/use-terminal-title.ts +31 -0
  868. package/src/ink/hooks/use-terminal-viewport.ts +96 -0
  869. package/src/ink/ink.tsx +1728 -0
  870. package/src/ink/instances.ts +10 -0
  871. package/src/ink/layout/engine.ts +6 -0
  872. package/src/ink/layout/geometry.ts +97 -0
  873. package/src/ink/layout/node.ts +152 -0
  874. package/src/ink/layout/yoga.ts +308 -0
  875. package/src/ink/line-width-cache.ts +24 -0
  876. package/src/ink/log-update.ts +773 -0
  877. package/src/ink/measure-element.ts +23 -0
  878. package/src/ink/measure-text.ts +47 -0
  879. package/src/ink/node-cache.ts +54 -0
  880. package/src/ink/optimizer.ts +93 -0
  881. package/src/ink/output.ts +797 -0
  882. package/src/ink/parse-keypress.ts +801 -0
  883. package/src/ink/reconciler.ts +512 -0
  884. package/src/ink/render-border.ts +231 -0
  885. package/src/ink/render-node-to-output.ts +1462 -0
  886. package/src/ink/render-to-screen.ts +231 -0
  887. package/src/ink/renderer.ts +178 -0
  888. package/src/ink/root.ts +184 -0
  889. package/src/ink/screen.ts +1486 -0
  890. package/src/ink/searchHighlight.ts +93 -0
  891. package/src/ink/selection.ts +917 -0
  892. package/src/ink/squash-text-nodes.ts +92 -0
  893. package/src/ink/stringWidth.ts +222 -0
  894. package/src/ink/styles.ts +771 -0
  895. package/src/ink/supports-hyperlinks.ts +57 -0
  896. package/src/ink/tabstops.ts +46 -0
  897. package/src/ink/terminal-focus-state.ts +47 -0
  898. package/src/ink/terminal-querier.ts +212 -0
  899. package/src/ink/terminal.ts +248 -0
  900. package/src/ink/termio/ansi.ts +75 -0
  901. package/src/ink/termio/csi.ts +319 -0
  902. package/src/ink/termio/dec.ts +60 -0
  903. package/src/ink/termio/esc.ts +67 -0
  904. package/src/ink/termio/osc.ts +493 -0
  905. package/src/ink/termio/parser.ts +394 -0
  906. package/src/ink/termio/sgr.ts +308 -0
  907. package/src/ink/termio/tokenize.ts +319 -0
  908. package/src/ink/termio/types.ts +236 -0
  909. package/src/ink/termio.ts +42 -0
  910. package/src/ink/useTerminalNotification.ts +126 -0
  911. package/src/ink/warn.ts +9 -0
  912. package/src/ink/widest-line.ts +19 -0
  913. package/src/ink/wrap-text.ts +74 -0
  914. package/src/ink/wrapAnsi.ts +20 -0
  915. package/src/ink.ts +85 -0
  916. package/src/interactiveHelpers.tsx +367 -0
  917. package/src/keybindings/KeybindingContext.tsx +243 -0
  918. package/src/keybindings/KeybindingProviderSetup.tsx +308 -0
  919. package/src/keybindings/defaultBindings.ts +340 -0
  920. package/src/keybindings/loadUserBindings.ts +472 -0
  921. package/src/keybindings/match.ts +120 -0
  922. package/src/keybindings/parser.ts +203 -0
  923. package/src/keybindings/reservedShortcuts.ts +127 -0
  924. package/src/keybindings/resolver.ts +244 -0
  925. package/src/keybindings/schema.ts +236 -0
  926. package/src/keybindings/shortcutFormat.ts +63 -0
  927. package/src/keybindings/template.ts +52 -0
  928. package/src/keybindings/useKeybinding.ts +196 -0
  929. package/src/keybindings/useShortcutDisplay.ts +59 -0
  930. package/src/keybindings/validate.ts +498 -0
  931. package/src/main.tsx +4684 -0
  932. package/src/memdir/findRelevantMemories.ts +141 -0
  933. package/src/memdir/memdir.ts +507 -0
  934. package/src/memdir/memoryAge.ts +53 -0
  935. package/src/memdir/memoryScan.ts +94 -0
  936. package/src/memdir/memoryTypes.ts +271 -0
  937. package/src/memdir/paths.ts +278 -0
  938. package/src/memdir/teamMemPaths.ts +292 -0
  939. package/src/memdir/teamMemPrompts.ts +100 -0
  940. package/src/migrations/migrateAutoUpdatesToSettings.ts +61 -0
  941. package/src/migrations/migrateBypassPermissionsAcceptedToSettings.ts +40 -0
  942. package/src/migrations/migrateEnableAllProjectMcpServersToSettings.ts +118 -0
  943. package/src/migrations/migrateFennecToOpus.ts +45 -0
  944. package/src/migrations/migrateLegacyOpusToCurrent.ts +57 -0
  945. package/src/migrations/migrateOpusToOpus1m.ts +43 -0
  946. package/src/migrations/migrateReplBridgeEnabledToRemoteControlAtStartup.ts +22 -0
  947. package/src/migrations/migrateSonnet1mToSonnet45.ts +48 -0
  948. package/src/migrations/migrateSonnet45ToSonnet46.ts +67 -0
  949. package/src/migrations/resetAutoModeOptInForDefaultOffer.ts +51 -0
  950. package/src/migrations/resetProToOpusDefault.ts +51 -0
  951. package/src/moreright/useMoreRight.tsx +26 -0
  952. package/src/native-ts/color-diff/index.ts +999 -0
  953. package/src/native-ts/file-index/index.ts +370 -0
  954. package/src/native-ts/yoga-layout/enums.ts +134 -0
  955. package/src/native-ts/yoga-layout/index.ts +2578 -0
  956. package/src/outputStyles/loadOutputStylesDir.ts +98 -0
  957. package/src/plugins/builtinPlugins.ts +159 -0
  958. package/src/plugins/bundled/index.ts +23 -0
  959. package/src/projectOnboardingState.ts +83 -0
  960. package/src/query/config.ts +46 -0
  961. package/src/query/deps.ts +40 -0
  962. package/src/query/stopHooks.ts +473 -0
  963. package/src/query/tokenBudget.ts +93 -0
  964. package/src/query.ts +1729 -0
  965. package/src/remote/RemoteSessionManager.ts +343 -0
  966. package/src/remote/SessionsWebSocket.ts +404 -0
  967. package/src/remote/remotePermissionBridge.ts +78 -0
  968. package/src/remote/sdkMessageAdapter.ts +302 -0
  969. package/src/replLauncher.tsx +23 -0
  970. package/src/schemas/hooks.ts +222 -0
  971. package/src/screens/Doctor.tsx +575 -0
  972. package/src/screens/REPL.tsx +5006 -0
  973. package/src/screens/ResumeConversation.tsx +399 -0
  974. package/src/server/createDirectConnectSession.ts +88 -0
  975. package/src/server/directConnectManager.ts +213 -0
  976. package/src/server/types.ts +57 -0
  977. package/src/services/AgentSummary/agentSummary.ts +179 -0
  978. package/src/services/MagicDocs/magicDocs.ts +254 -0
  979. package/src/services/MagicDocs/prompts.ts +127 -0
  980. package/src/services/PromptSuggestion/promptSuggestion.ts +523 -0
  981. package/src/services/PromptSuggestion/speculation.ts +991 -0
  982. package/src/services/SessionMemory/prompts.ts +324 -0
  983. package/src/services/SessionMemory/sessionMemory.ts +495 -0
  984. package/src/services/SessionMemory/sessionMemoryUtils.ts +207 -0
  985. package/src/services/analytics/config.ts +38 -0
  986. package/src/services/analytics/datadog.ts +307 -0
  987. package/src/services/analytics/firstPartyEventLogger.ts +449 -0
  988. package/src/services/analytics/firstPartyEventLoggingExporter.ts +806 -0
  989. package/src/services/analytics/growthbook.ts +1157 -0
  990. package/src/services/analytics/index.ts +173 -0
  991. package/src/services/analytics/metadata.ts +973 -0
  992. package/src/services/analytics/sink.ts +114 -0
  993. package/src/services/analytics/sinkKillswitch.ts +25 -0
  994. package/src/services/api/adminRequests.ts +119 -0
  995. package/src/services/api/bootstrap.ts +142 -0
  996. package/src/services/api/claude.ts +3433 -0
  997. package/src/services/api/client.ts +395 -0
  998. package/src/services/api/dumpPrompts.ts +226 -0
  999. package/src/services/api/emptyUsage.ts +22 -0
  1000. package/src/services/api/errorUtils.ts +260 -0
  1001. package/src/services/api/errors.ts +1207 -0
  1002. package/src/services/api/filesApi.ts +748 -0
  1003. package/src/services/api/firstTokenDate.ts +60 -0
  1004. package/src/services/api/grove.ts +357 -0
  1005. package/src/services/api/logging.ts +788 -0
  1006. package/src/services/api/metricsOptOut.ts +159 -0
  1007. package/src/services/api/overageCreditGrant.ts +137 -0
  1008. package/src/services/api/promptCacheBreakDetection.ts +727 -0
  1009. package/src/services/api/referral.ts +281 -0
  1010. package/src/services/api/sessionIngress.ts +514 -0
  1011. package/src/services/api/ultrareviewQuota.ts +38 -0
  1012. package/src/services/api/usage.ts +63 -0
  1013. package/src/services/api/withRetry.ts +826 -0
  1014. package/src/services/autoDream/autoDream.ts +324 -0
  1015. package/src/services/autoDream/config.ts +21 -0
  1016. package/src/services/autoDream/consolidationLock.ts +140 -0
  1017. package/src/services/autoDream/consolidationPrompt.ts +65 -0
  1018. package/src/services/awaySummary.ts +74 -0
  1019. package/src/services/claudeAiLimits.ts +515 -0
  1020. package/src/services/claudeAiLimitsHook.ts +23 -0
  1021. package/src/services/compact/apiMicrocompact.ts +153 -0
  1022. package/src/services/compact/autoCompact.ts +351 -0
  1023. package/src/services/compact/compact.ts +1705 -0
  1024. package/src/services/compact/compactWarningHook.ts +16 -0
  1025. package/src/services/compact/compactWarningState.ts +18 -0
  1026. package/src/services/compact/grouping.ts +63 -0
  1027. package/src/services/compact/microCompact.ts +530 -0
  1028. package/src/services/compact/postCompactCleanup.ts +77 -0
  1029. package/src/services/compact/prompt.ts +374 -0
  1030. package/src/services/compact/sessionMemoryCompact.ts +630 -0
  1031. package/src/services/compact/timeBasedMCConfig.ts +43 -0
  1032. package/src/services/diagnosticTracking.ts +397 -0
  1033. package/src/services/extractMemories/extractMemories.ts +615 -0
  1034. package/src/services/extractMemories/prompts.ts +154 -0
  1035. package/src/services/internalLogging.ts +90 -0
  1036. package/src/services/lsp/LSPClient.ts +447 -0
  1037. package/src/services/lsp/LSPDiagnosticRegistry.ts +386 -0
  1038. package/src/services/lsp/LSPServerInstance.ts +511 -0
  1039. package/src/services/lsp/LSPServerManager.ts +420 -0
  1040. package/src/services/lsp/config.ts +79 -0
  1041. package/src/services/lsp/manager.ts +289 -0
  1042. package/src/services/lsp/passiveFeedback.ts +328 -0
  1043. package/src/services/mcp/InProcessTransport.ts +63 -0
  1044. package/src/services/mcp/MCPConnectionManager.tsx +73 -0
  1045. package/src/services/mcp/SdkControlTransport.ts +136 -0
  1046. package/src/services/mcp/auth.ts +2465 -0
  1047. package/src/services/mcp/channelAllowlist.ts +76 -0
  1048. package/src/services/mcp/channelNotification.ts +316 -0
  1049. package/src/services/mcp/channelPermissions.ts +240 -0
  1050. package/src/services/mcp/claudeai.ts +164 -0
  1051. package/src/services/mcp/client.ts +3348 -0
  1052. package/src/services/mcp/config.ts +1578 -0
  1053. package/src/services/mcp/elicitationHandler.ts +313 -0
  1054. package/src/services/mcp/envExpansion.ts +38 -0
  1055. package/src/services/mcp/headersHelper.ts +138 -0
  1056. package/src/services/mcp/mcpStringUtils.ts +106 -0
  1057. package/src/services/mcp/normalization.ts +23 -0
  1058. package/src/services/mcp/oauthPort.ts +78 -0
  1059. package/src/services/mcp/officialRegistry.ts +72 -0
  1060. package/src/services/mcp/types.ts +258 -0
  1061. package/src/services/mcp/useManageMCPConnections.ts +1141 -0
  1062. package/src/services/mcp/utils.ts +575 -0
  1063. package/src/services/mcp/vscodeSdkMcp.ts +112 -0
  1064. package/src/services/mcp/xaa.ts +511 -0
  1065. package/src/services/mcp/xaaIdpLogin.ts +487 -0
  1066. package/src/services/mcpServerApproval.tsx +41 -0
  1067. package/src/services/mockRateLimits.ts +882 -0
  1068. package/src/services/notifier.ts +156 -0
  1069. package/src/services/oauth/auth-code-listener.ts +211 -0
  1070. package/src/services/oauth/client.ts +566 -0
  1071. package/src/services/oauth/crypto.ts +23 -0
  1072. package/src/services/oauth/getOauthProfile.ts +53 -0
  1073. package/src/services/oauth/index.ts +198 -0
  1074. package/src/services/plugins/PluginInstallationManager.ts +184 -0
  1075. package/src/services/plugins/pluginCliCommands.ts +344 -0
  1076. package/src/services/plugins/pluginOperations.ts +1088 -0
  1077. package/src/services/policyLimits/index.ts +664 -0
  1078. package/src/services/policyLimits/types.ts +27 -0
  1079. package/src/services/preventSleep.ts +165 -0
  1080. package/src/services/rateLimitMessages.ts +344 -0
  1081. package/src/services/rateLimitMocking.ts +144 -0
  1082. package/src/services/remoteManagedSettings/index.ts +639 -0
  1083. package/src/services/remoteManagedSettings/securityCheck.tsx +74 -0
  1084. package/src/services/remoteManagedSettings/syncCache.ts +112 -0
  1085. package/src/services/remoteManagedSettings/syncCacheState.ts +96 -0
  1086. package/src/services/remoteManagedSettings/types.ts +31 -0
  1087. package/src/services/settingsSync/index.ts +581 -0
  1088. package/src/services/settingsSync/types.ts +67 -0
  1089. package/src/services/teamMemorySync/index.ts +1256 -0
  1090. package/src/services/teamMemorySync/secretScanner.ts +324 -0
  1091. package/src/services/teamMemorySync/teamMemSecretGuard.ts +44 -0
  1092. package/src/services/teamMemorySync/types.ts +156 -0
  1093. package/src/services/teamMemorySync/watcher.ts +387 -0
  1094. package/src/services/tips/tipHistory.ts +17 -0
  1095. package/src/services/tips/tipRegistry.ts +686 -0
  1096. package/src/services/tips/tipScheduler.ts +58 -0
  1097. package/src/services/tokenEstimation.ts +495 -0
  1098. package/src/services/toolUseSummary/toolUseSummaryGenerator.ts +112 -0
  1099. package/src/services/tools/StreamingToolExecutor.ts +530 -0
  1100. package/src/services/tools/toolExecution.ts +1745 -0
  1101. package/src/services/tools/toolHooks.ts +650 -0
  1102. package/src/services/tools/toolOrchestration.ts +188 -0
  1103. package/src/services/vcr.ts +406 -0
  1104. package/src/services/voice.ts +525 -0
  1105. package/src/services/voiceKeyterms.ts +106 -0
  1106. package/src/services/voiceStreamSTT.ts +544 -0
  1107. package/src/setup.ts +477 -0
  1108. package/src/skills/bundled/batch.ts +124 -0
  1109. package/src/skills/bundled/claudeApi.ts +196 -0
  1110. package/src/skills/bundled/claudeApiContent.ts +75 -0
  1111. package/src/skills/bundled/claudeInChrome.ts +34 -0
  1112. package/src/skills/bundled/debug.ts +103 -0
  1113. package/src/skills/bundled/index.ts +79 -0
  1114. package/src/skills/bundled/keybindings.ts +339 -0
  1115. package/src/skills/bundled/loop.ts +92 -0
  1116. package/src/skills/bundled/loremIpsum.ts +282 -0
  1117. package/src/skills/bundled/remember.ts +82 -0
  1118. package/src/skills/bundled/scheduleRemoteAgents.ts +447 -0
  1119. package/src/skills/bundled/simplify.ts +69 -0
  1120. package/src/skills/bundled/skillify.ts +197 -0
  1121. package/src/skills/bundled/stuck.ts +79 -0
  1122. package/src/skills/bundled/updateConfig.ts +475 -0
  1123. package/src/skills/bundled/verify.ts +30 -0
  1124. package/src/skills/bundled/verifyContent.ts +13 -0
  1125. package/src/skills/bundledSkills.ts +220 -0
  1126. package/src/skills/loadSkillsDir.ts +1086 -0
  1127. package/src/skills/mcpSkillBuilders.ts +44 -0
  1128. package/src/state/AppState.tsx +200 -0
  1129. package/src/state/AppStateStore.ts +569 -0
  1130. package/src/state/onChangeAppState.ts +171 -0
  1131. package/src/state/selectors.ts +76 -0
  1132. package/src/state/store.ts +34 -0
  1133. package/src/state/teammateViewHelpers.ts +141 -0
  1134. package/src/tasks/DreamTask/DreamTask.ts +157 -0
  1135. package/src/tasks/InProcessTeammateTask/InProcessTeammateTask.tsx +126 -0
  1136. package/src/tasks/InProcessTeammateTask/types.ts +121 -0
  1137. package/src/tasks/LocalAgentTask/LocalAgentTask.tsx +683 -0
  1138. package/src/tasks/LocalMainSessionTask.ts +479 -0
  1139. package/src/tasks/LocalShellTask/LocalShellTask.tsx +523 -0
  1140. package/src/tasks/LocalShellTask/guards.ts +41 -0
  1141. package/src/tasks/LocalShellTask/killShellTasks.ts +76 -0
  1142. package/src/tasks/RemoteAgentTask/RemoteAgentTask.tsx +856 -0
  1143. package/src/tasks/pillLabel.ts +82 -0
  1144. package/src/tasks/stopTask.ts +100 -0
  1145. package/src/tasks/types.ts +46 -0
  1146. package/src/tasks.ts +39 -0
  1147. package/src/tools/AgentTool/AgentTool.tsx +1398 -0
  1148. package/src/tools/AgentTool/UI.tsx +872 -0
  1149. package/src/tools/AgentTool/agentColorManager.ts +66 -0
  1150. package/src/tools/AgentTool/agentDisplay.ts +104 -0
  1151. package/src/tools/AgentTool/agentMemory.ts +177 -0
  1152. package/src/tools/AgentTool/agentMemorySnapshot.ts +197 -0
  1153. package/src/tools/AgentTool/agentToolUtils.ts +686 -0
  1154. package/src/tools/AgentTool/built-in/claudeCodeGuideAgent.ts +205 -0
  1155. package/src/tools/AgentTool/built-in/exploreAgent.ts +83 -0
  1156. package/src/tools/AgentTool/built-in/generalPurposeAgent.ts +34 -0
  1157. package/src/tools/AgentTool/built-in/planAgent.ts +92 -0
  1158. package/src/tools/AgentTool/built-in/statuslineSetup.ts +144 -0
  1159. package/src/tools/AgentTool/built-in/verificationAgent.ts +152 -0
  1160. package/src/tools/AgentTool/builtInAgents.ts +72 -0
  1161. package/src/tools/AgentTool/constants.ts +12 -0
  1162. package/src/tools/AgentTool/forkSubagent.ts +210 -0
  1163. package/src/tools/AgentTool/loadAgentsDir.ts +755 -0
  1164. package/src/tools/AgentTool/prompt.ts +287 -0
  1165. package/src/tools/AgentTool/resumeAgent.ts +265 -0
  1166. package/src/tools/AgentTool/runAgent.ts +973 -0
  1167. package/src/tools/AskUserQuestionTool/AskUserQuestionTool.tsx +266 -0
  1168. package/src/tools/AskUserQuestionTool/prompt.ts +44 -0
  1169. package/src/tools/BashTool/BashTool.tsx +1144 -0
  1170. package/src/tools/BashTool/BashToolResultMessage.tsx +191 -0
  1171. package/src/tools/BashTool/UI.tsx +185 -0
  1172. package/src/tools/BashTool/bashCommandHelpers.ts +265 -0
  1173. package/src/tools/BashTool/bashPermissions.ts +2621 -0
  1174. package/src/tools/BashTool/bashSecurity.ts +2592 -0
  1175. package/src/tools/BashTool/commandSemantics.ts +140 -0
  1176. package/src/tools/BashTool/commentLabel.ts +13 -0
  1177. package/src/tools/BashTool/destructiveCommandWarning.ts +102 -0
  1178. package/src/tools/BashTool/modeValidation.ts +115 -0
  1179. package/src/tools/BashTool/pathValidation.ts +1303 -0
  1180. package/src/tools/BashTool/prompt.ts +369 -0
  1181. package/src/tools/BashTool/readOnlyValidation.ts +1990 -0
  1182. package/src/tools/BashTool/sedEditParser.ts +322 -0
  1183. package/src/tools/BashTool/sedValidation.ts +684 -0
  1184. package/src/tools/BashTool/shouldUseSandbox.ts +153 -0
  1185. package/src/tools/BashTool/toolName.ts +2 -0
  1186. package/src/tools/BashTool/utils.ts +223 -0
  1187. package/src/tools/BriefTool/BriefTool.ts +204 -0
  1188. package/src/tools/BriefTool/UI.tsx +101 -0
  1189. package/src/tools/BriefTool/attachments.ts +110 -0
  1190. package/src/tools/BriefTool/prompt.ts +22 -0
  1191. package/src/tools/BriefTool/upload.ts +174 -0
  1192. package/src/tools/ConfigTool/ConfigTool.ts +467 -0
  1193. package/src/tools/ConfigTool/UI.tsx +38 -0
  1194. package/src/tools/ConfigTool/constants.ts +1 -0
  1195. package/src/tools/ConfigTool/prompt.ts +93 -0
  1196. package/src/tools/ConfigTool/supportedSettings.ts +211 -0
  1197. package/src/tools/EnterPlanModeTool/EnterPlanModeTool.ts +126 -0
  1198. package/src/tools/EnterPlanModeTool/UI.tsx +33 -0
  1199. package/src/tools/EnterPlanModeTool/constants.ts +1 -0
  1200. package/src/tools/EnterPlanModeTool/prompt.ts +170 -0
  1201. package/src/tools/EnterWorktreeTool/EnterWorktreeTool.ts +127 -0
  1202. package/src/tools/EnterWorktreeTool/UI.tsx +20 -0
  1203. package/src/tools/EnterWorktreeTool/constants.ts +1 -0
  1204. package/src/tools/EnterWorktreeTool/prompt.ts +30 -0
  1205. package/src/tools/ExitPlanModeTool/ExitPlanModeV2Tool.ts +493 -0
  1206. package/src/tools/ExitPlanModeTool/UI.tsx +82 -0
  1207. package/src/tools/ExitPlanModeTool/constants.ts +2 -0
  1208. package/src/tools/ExitPlanModeTool/prompt.ts +29 -0
  1209. package/src/tools/ExitWorktreeTool/ExitWorktreeTool.ts +329 -0
  1210. package/src/tools/ExitWorktreeTool/UI.tsx +25 -0
  1211. package/src/tools/ExitWorktreeTool/constants.ts +1 -0
  1212. package/src/tools/ExitWorktreeTool/prompt.ts +32 -0
  1213. package/src/tools/FileEditTool/FileEditTool.ts +625 -0
  1214. package/src/tools/FileEditTool/UI.tsx +289 -0
  1215. package/src/tools/FileEditTool/constants.ts +11 -0
  1216. package/src/tools/FileEditTool/prompt.ts +28 -0
  1217. package/src/tools/FileEditTool/types.ts +85 -0
  1218. package/src/tools/FileEditTool/utils.ts +775 -0
  1219. package/src/tools/FileReadTool/FileReadTool.ts +1183 -0
  1220. package/src/tools/FileReadTool/UI.tsx +185 -0
  1221. package/src/tools/FileReadTool/imageProcessor.ts +94 -0
  1222. package/src/tools/FileReadTool/limits.ts +92 -0
  1223. package/src/tools/FileReadTool/prompt.ts +49 -0
  1224. package/src/tools/FileWriteTool/FileWriteTool.ts +434 -0
  1225. package/src/tools/FileWriteTool/UI.tsx +405 -0
  1226. package/src/tools/FileWriteTool/prompt.ts +18 -0
  1227. package/src/tools/GlobTool/GlobTool.ts +198 -0
  1228. package/src/tools/GlobTool/UI.tsx +63 -0
  1229. package/src/tools/GlobTool/prompt.ts +7 -0
  1230. package/src/tools/GrepTool/GrepTool.ts +577 -0
  1231. package/src/tools/GrepTool/UI.tsx +201 -0
  1232. package/src/tools/GrepTool/prompt.ts +18 -0
  1233. package/src/tools/LSPTool/LSPTool.ts +860 -0
  1234. package/src/tools/LSPTool/UI.tsx +228 -0
  1235. package/src/tools/LSPTool/formatters.ts +592 -0
  1236. package/src/tools/LSPTool/prompt.ts +21 -0
  1237. package/src/tools/LSPTool/schemas.ts +215 -0
  1238. package/src/tools/LSPTool/symbolContext.ts +90 -0
  1239. package/src/tools/ListMcpResourcesTool/ListMcpResourcesTool.ts +123 -0
  1240. package/src/tools/ListMcpResourcesTool/UI.tsx +29 -0
  1241. package/src/tools/ListMcpResourcesTool/prompt.ts +20 -0
  1242. package/src/tools/MCPTool/MCPTool.ts +77 -0
  1243. package/src/tools/MCPTool/UI.tsx +403 -0
  1244. package/src/tools/MCPTool/classifyForCollapse.ts +604 -0
  1245. package/src/tools/MCPTool/prompt.ts +3 -0
  1246. package/src/tools/McpAuthTool/McpAuthTool.ts +215 -0
  1247. package/src/tools/NotebookEditTool/NotebookEditTool.ts +490 -0
  1248. package/src/tools/NotebookEditTool/UI.tsx +93 -0
  1249. package/src/tools/NotebookEditTool/constants.ts +2 -0
  1250. package/src/tools/NotebookEditTool/prompt.ts +3 -0
  1251. package/src/tools/PowerShellTool/PowerShellTool.tsx +1001 -0
  1252. package/src/tools/PowerShellTool/UI.tsx +131 -0
  1253. package/src/tools/PowerShellTool/clmTypes.ts +211 -0
  1254. package/src/tools/PowerShellTool/commandSemantics.ts +142 -0
  1255. package/src/tools/PowerShellTool/commonParameters.ts +30 -0
  1256. package/src/tools/PowerShellTool/destructiveCommandWarning.ts +109 -0
  1257. package/src/tools/PowerShellTool/gitSafety.ts +176 -0
  1258. package/src/tools/PowerShellTool/modeValidation.ts +404 -0
  1259. package/src/tools/PowerShellTool/pathValidation.ts +2049 -0
  1260. package/src/tools/PowerShellTool/powershellPermissions.ts +1648 -0
  1261. package/src/tools/PowerShellTool/powershellSecurity.ts +1090 -0
  1262. package/src/tools/PowerShellTool/prompt.ts +145 -0
  1263. package/src/tools/PowerShellTool/readOnlyValidation.ts +1823 -0
  1264. package/src/tools/PowerShellTool/toolName.ts +2 -0
  1265. package/src/tools/REPLTool/constants.ts +46 -0
  1266. package/src/tools/REPLTool/primitiveTools.ts +39 -0
  1267. package/src/tools/ReadMcpResourceTool/ReadMcpResourceTool.ts +158 -0
  1268. package/src/tools/ReadMcpResourceTool/UI.tsx +37 -0
  1269. package/src/tools/ReadMcpResourceTool/prompt.ts +16 -0
  1270. package/src/tools/RemoteTriggerTool/RemoteTriggerTool.ts +161 -0
  1271. package/src/tools/RemoteTriggerTool/UI.tsx +17 -0
  1272. package/src/tools/RemoteTriggerTool/prompt.ts +15 -0
  1273. package/src/tools/ScheduleCronTool/CronCreateTool.ts +157 -0
  1274. package/src/tools/ScheduleCronTool/CronDeleteTool.ts +95 -0
  1275. package/src/tools/ScheduleCronTool/CronListTool.ts +97 -0
  1276. package/src/tools/ScheduleCronTool/UI.tsx +60 -0
  1277. package/src/tools/ScheduleCronTool/prompt.ts +135 -0
  1278. package/src/tools/SendMessageTool/SendMessageTool.ts +917 -0
  1279. package/src/tools/SendMessageTool/UI.tsx +31 -0
  1280. package/src/tools/SendMessageTool/constants.ts +1 -0
  1281. package/src/tools/SendMessageTool/prompt.ts +49 -0
  1282. package/src/tools/SkillTool/SkillTool.ts +1108 -0
  1283. package/src/tools/SkillTool/UI.tsx +128 -0
  1284. package/src/tools/SkillTool/constants.ts +1 -0
  1285. package/src/tools/SkillTool/prompt.ts +241 -0
  1286. package/src/tools/SleepTool/prompt.ts +17 -0
  1287. package/src/tools/SyntheticOutputTool/SyntheticOutputTool.ts +163 -0
  1288. package/src/tools/TaskCreateTool/TaskCreateTool.ts +138 -0
  1289. package/src/tools/TaskCreateTool/constants.ts +1 -0
  1290. package/src/tools/TaskCreateTool/prompt.ts +56 -0
  1291. package/src/tools/TaskGetTool/TaskGetTool.ts +128 -0
  1292. package/src/tools/TaskGetTool/constants.ts +1 -0
  1293. package/src/tools/TaskGetTool/prompt.ts +24 -0
  1294. package/src/tools/TaskListTool/TaskListTool.ts +116 -0
  1295. package/src/tools/TaskListTool/constants.ts +1 -0
  1296. package/src/tools/TaskListTool/prompt.ts +49 -0
  1297. package/src/tools/TaskOutputTool/TaskOutputTool.tsx +584 -0
  1298. package/src/tools/TaskOutputTool/constants.ts +1 -0
  1299. package/src/tools/TaskStopTool/TaskStopTool.ts +131 -0
  1300. package/src/tools/TaskStopTool/UI.tsx +41 -0
  1301. package/src/tools/TaskStopTool/prompt.ts +8 -0
  1302. package/src/tools/TaskUpdateTool/TaskUpdateTool.ts +406 -0
  1303. package/src/tools/TaskUpdateTool/constants.ts +1 -0
  1304. package/src/tools/TaskUpdateTool/prompt.ts +77 -0
  1305. package/src/tools/TeamCreateTool/TeamCreateTool.ts +240 -0
  1306. package/src/tools/TeamCreateTool/UI.tsx +6 -0
  1307. package/src/tools/TeamCreateTool/constants.ts +1 -0
  1308. package/src/tools/TeamCreateTool/prompt.ts +113 -0
  1309. package/src/tools/TeamDeleteTool/TeamDeleteTool.ts +139 -0
  1310. package/src/tools/TeamDeleteTool/UI.tsx +20 -0
  1311. package/src/tools/TeamDeleteTool/constants.ts +1 -0
  1312. package/src/tools/TeamDeleteTool/prompt.ts +16 -0
  1313. package/src/tools/TodoWriteTool/TodoWriteTool.ts +115 -0
  1314. package/src/tools/TodoWriteTool/constants.ts +1 -0
  1315. package/src/tools/TodoWriteTool/prompt.ts +184 -0
  1316. package/src/tools/ToolSearchTool/ToolSearchTool.ts +471 -0
  1317. package/src/tools/ToolSearchTool/constants.ts +1 -0
  1318. package/src/tools/ToolSearchTool/prompt.ts +121 -0
  1319. package/src/tools/TungstenTool/TungstenTool.js +2 -0
  1320. package/src/tools/TungstenTool/TungstenTool.ts +1 -0
  1321. package/src/tools/WebFetchTool/UI.tsx +72 -0
  1322. package/src/tools/WebFetchTool/WebFetchTool.ts +318 -0
  1323. package/src/tools/WebFetchTool/preapproved.ts +166 -0
  1324. package/src/tools/WebFetchTool/prompt.ts +46 -0
  1325. package/src/tools/WebFetchTool/utils.ts +530 -0
  1326. package/src/tools/WebSearchTool/UI.tsx +101 -0
  1327. package/src/tools/WebSearchTool/WebSearchTool.ts +435 -0
  1328. package/src/tools/WebSearchTool/prompt.ts +34 -0
  1329. package/src/tools/shared/gitOperationTracking.ts +277 -0
  1330. package/src/tools/shared/spawnMultiAgent.ts +1093 -0
  1331. package/src/tools/testing/TestingPermissionTool.tsx +74 -0
  1332. package/src/tools/utils.ts +40 -0
  1333. package/src/tools.ts +389 -0
  1334. package/src/types/command.ts +216 -0
  1335. package/src/types/connectorText.js +5 -0
  1336. package/src/types/connectorText.ts +1 -0
  1337. package/src/types/generated/events_mono/claude_code/v1/claude_code_internal_event.ts +865 -0
  1338. package/src/types/generated/events_mono/common/v1/auth.ts +100 -0
  1339. package/src/types/generated/events_mono/growthbook/v1/growthbook_experiment_event.ts +223 -0
  1340. package/src/types/generated/google/protobuf/timestamp.ts +187 -0
  1341. package/src/types/hooks.ts +290 -0
  1342. package/src/types/ids.ts +44 -0
  1343. package/src/types/logs.ts +330 -0
  1344. package/src/types/permissions.ts +441 -0
  1345. package/src/types/plugin.ts +363 -0
  1346. package/src/types/textInputTypes.ts +387 -0
  1347. package/src/upstreamproxy/relay.ts +455 -0
  1348. package/src/upstreamproxy/upstreamproxy.ts +285 -0
  1349. package/src/utils/CircularBuffer.ts +84 -0
  1350. package/src/utils/Cursor.ts +1530 -0
  1351. package/src/utils/QueryGuard.ts +121 -0
  1352. package/src/utils/Shell.ts +474 -0
  1353. package/src/utils/ShellCommand.ts +465 -0
  1354. package/src/utils/abortController.ts +99 -0
  1355. package/src/utils/activityManager.ts +164 -0
  1356. package/src/utils/advisor.ts +145 -0
  1357. package/src/utils/agentContext.ts +178 -0
  1358. package/src/utils/agentId.ts +99 -0
  1359. package/src/utils/agentSwarmsEnabled.ts +44 -0
  1360. package/src/utils/agenticSessionSearch.ts +307 -0
  1361. package/src/utils/analyzeContext.ts +1382 -0
  1362. package/src/utils/ansiToPng.ts +334 -0
  1363. package/src/utils/ansiToSvg.ts +272 -0
  1364. package/src/utils/api.ts +718 -0
  1365. package/src/utils/apiPreconnect.ts +72 -0
  1366. package/src/utils/appleTerminalBackup.ts +124 -0
  1367. package/src/utils/argumentSubstitution.ts +145 -0
  1368. package/src/utils/array.ts +13 -0
  1369. package/src/utils/asciicast.ts +239 -0
  1370. package/src/utils/attachments.ts +3997 -0
  1371. package/src/utils/attribution.ts +393 -0
  1372. package/src/utils/auth.ts +2007 -0
  1373. package/src/utils/authFileDescriptor.ts +196 -0
  1374. package/src/utils/authPortable.ts +19 -0
  1375. package/src/utils/autoModeDenials.ts +26 -0
  1376. package/src/utils/autoRunIssue.tsx +122 -0
  1377. package/src/utils/autoUpdater.ts +562 -0
  1378. package/src/utils/aws.ts +74 -0
  1379. package/src/utils/awsAuthStatusManager.ts +81 -0
  1380. package/src/utils/background/remote/preconditions.ts +235 -0
  1381. package/src/utils/background/remote/remoteSession.ts +98 -0
  1382. package/src/utils/backgroundHousekeeping.ts +94 -0
  1383. package/src/utils/bash/ParsedCommand.ts +318 -0
  1384. package/src/utils/bash/ShellSnapshot.ts +582 -0
  1385. package/src/utils/bash/ast.ts +2679 -0
  1386. package/src/utils/bash/bashParser.ts +4436 -0
  1387. package/src/utils/bash/bashPipeCommand.ts +294 -0
  1388. package/src/utils/bash/commands.ts +1339 -0
  1389. package/src/utils/bash/heredoc.ts +733 -0
  1390. package/src/utils/bash/parser.ts +230 -0
  1391. package/src/utils/bash/prefix.ts +204 -0
  1392. package/src/utils/bash/registry.ts +53 -0
  1393. package/src/utils/bash/shellCompletion.ts +259 -0
  1394. package/src/utils/bash/shellPrefix.ts +28 -0
  1395. package/src/utils/bash/shellQuote.ts +304 -0
  1396. package/src/utils/bash/shellQuoting.ts +128 -0
  1397. package/src/utils/bash/specs/alias.ts +14 -0
  1398. package/src/utils/bash/specs/index.ts +18 -0
  1399. package/src/utils/bash/specs/nohup.ts +13 -0
  1400. package/src/utils/bash/specs/pyright.ts +91 -0
  1401. package/src/utils/bash/specs/sleep.ts +13 -0
  1402. package/src/utils/bash/specs/srun.ts +31 -0
  1403. package/src/utils/bash/specs/time.ts +13 -0
  1404. package/src/utils/bash/specs/timeout.ts +20 -0
  1405. package/src/utils/bash/treeSitterAnalysis.ts +506 -0
  1406. package/src/utils/betas.ts +438 -0
  1407. package/src/utils/billing.ts +78 -0
  1408. package/src/utils/binaryCheck.ts +53 -0
  1409. package/src/utils/browser.ts +68 -0
  1410. package/src/utils/bufferedWriter.ts +100 -0
  1411. package/src/utils/bundledMode.ts +22 -0
  1412. package/src/utils/caCerts.ts +115 -0
  1413. package/src/utils/caCertsConfig.ts +88 -0
  1414. package/src/utils/cachePaths.ts +38 -0
  1415. package/src/utils/classifierApprovals.ts +88 -0
  1416. package/src/utils/classifierApprovalsHook.ts +17 -0
  1417. package/src/utils/claudeCodeHints.ts +193 -0
  1418. package/src/utils/claudeDesktop.ts +152 -0
  1419. package/src/utils/claudeInChrome/chromeNativeHost.ts +527 -0
  1420. package/src/utils/claudeInChrome/common.ts +540 -0
  1421. package/src/utils/claudeInChrome/mcpServer.ts +293 -0
  1422. package/src/utils/claudeInChrome/prompt.ts +83 -0
  1423. package/src/utils/claudeInChrome/setup.ts +400 -0
  1424. package/src/utils/claudeInChrome/setupPortable.ts +233 -0
  1425. package/src/utils/claudeInChrome/toolRendering.tsx +262 -0
  1426. package/src/utils/claudemd.ts +1479 -0
  1427. package/src/utils/cleanup.ts +602 -0
  1428. package/src/utils/cleanupRegistry.ts +25 -0
  1429. package/src/utils/cliArgs.ts +60 -0
  1430. package/src/utils/cliHighlight.ts +54 -0
  1431. package/src/utils/codeIndexing.ts +206 -0
  1432. package/src/utils/collapseBackgroundBashNotifications.ts +84 -0
  1433. package/src/utils/collapseHookSummaries.ts +59 -0
  1434. package/src/utils/collapseReadSearch.ts +1109 -0
  1435. package/src/utils/collapseTeammateShutdowns.ts +55 -0
  1436. package/src/utils/combinedAbortSignal.ts +47 -0
  1437. package/src/utils/commandLifecycle.ts +21 -0
  1438. package/src/utils/commitAttribution.ts +961 -0
  1439. package/src/utils/completionCache.ts +166 -0
  1440. package/src/utils/computerUse/appNames.ts +196 -0
  1441. package/src/utils/computerUse/cleanup.ts +86 -0
  1442. package/src/utils/computerUse/common.ts +61 -0
  1443. package/src/utils/computerUse/computerUseLock.ts +215 -0
  1444. package/src/utils/computerUse/drainRunLoop.ts +79 -0
  1445. package/src/utils/computerUse/escHotkey.ts +54 -0
  1446. package/src/utils/computerUse/executor.ts +658 -0
  1447. package/src/utils/computerUse/gates.ts +72 -0
  1448. package/src/utils/computerUse/hostAdapter.ts +69 -0
  1449. package/src/utils/computerUse/inputLoader.ts +30 -0
  1450. package/src/utils/computerUse/mcpServer.ts +106 -0
  1451. package/src/utils/computerUse/setup.ts +53 -0
  1452. package/src/utils/computerUse/swiftLoader.ts +23 -0
  1453. package/src/utils/computerUse/toolRendering.tsx +125 -0
  1454. package/src/utils/computerUse/wrapper.tsx +336 -0
  1455. package/src/utils/concurrentSessions.ts +204 -0
  1456. package/src/utils/config.ts +1817 -0
  1457. package/src/utils/configConstants.ts +21 -0
  1458. package/src/utils/contentArray.ts +51 -0
  1459. package/src/utils/context.ts +221 -0
  1460. package/src/utils/contextAnalysis.ts +272 -0
  1461. package/src/utils/contextSuggestions.ts +235 -0
  1462. package/src/utils/controlMessageCompat.ts +32 -0
  1463. package/src/utils/conversationRecovery.ts +597 -0
  1464. package/src/utils/cron.ts +308 -0
  1465. package/src/utils/cronJitterConfig.ts +75 -0
  1466. package/src/utils/cronScheduler.ts +565 -0
  1467. package/src/utils/cronTasks.ts +458 -0
  1468. package/src/utils/cronTasksLock.ts +195 -0
  1469. package/src/utils/crossProjectResume.ts +75 -0
  1470. package/src/utils/crypto.ts +13 -0
  1471. package/src/utils/cwd.ts +32 -0
  1472. package/src/utils/debug.ts +268 -0
  1473. package/src/utils/debugFilter.ts +157 -0
  1474. package/src/utils/deepLink/banner.ts +123 -0
  1475. package/src/utils/deepLink/parseDeepLink.ts +170 -0
  1476. package/src/utils/deepLink/protocolHandler.ts +136 -0
  1477. package/src/utils/deepLink/registerProtocol.ts +348 -0
  1478. package/src/utils/deepLink/terminalLauncher.ts +557 -0
  1479. package/src/utils/deepLink/terminalPreference.ts +54 -0
  1480. package/src/utils/desktopDeepLink.ts +236 -0
  1481. package/src/utils/detectRepository.ts +178 -0
  1482. package/src/utils/diagLogs.ts +94 -0
  1483. package/src/utils/diff.ts +177 -0
  1484. package/src/utils/directMemberMessage.ts +69 -0
  1485. package/src/utils/displayTags.ts +51 -0
  1486. package/src/utils/doctorContextWarnings.ts +265 -0
  1487. package/src/utils/doctorDiagnostic.ts +625 -0
  1488. package/src/utils/dxt/helpers.ts +88 -0
  1489. package/src/utils/dxt/zip.ts +226 -0
  1490. package/src/utils/earlyInput.ts +191 -0
  1491. package/src/utils/editor.ts +183 -0
  1492. package/src/utils/effort.ts +329 -0
  1493. package/src/utils/embeddedTools.ts +29 -0
  1494. package/src/utils/env.ts +347 -0
  1495. package/src/utils/envDynamic.ts +151 -0
  1496. package/src/utils/envUtils.ts +183 -0
  1497. package/src/utils/envValidation.ts +38 -0
  1498. package/src/utils/errorLogSink.ts +235 -0
  1499. package/src/utils/errors.ts +238 -0
  1500. package/src/utils/exampleCommands.ts +184 -0
  1501. package/src/utils/execFileNoThrow.ts +150 -0
  1502. package/src/utils/execFileNoThrowPortable.ts +89 -0
  1503. package/src/utils/execSyncWrapper.ts +38 -0
  1504. package/src/utils/exportRenderer.tsx +98 -0
  1505. package/src/utils/extraUsage.ts +23 -0
  1506. package/src/utils/fastMode.ts +532 -0
  1507. package/src/utils/file.ts +584 -0
  1508. package/src/utils/fileHistory.ts +1115 -0
  1509. package/src/utils/fileOperationAnalytics.ts +71 -0
  1510. package/src/utils/filePersistence/filePersistence.ts +287 -0
  1511. package/src/utils/filePersistence/outputsScanner.ts +126 -0
  1512. package/src/utils/fileRead.ts +102 -0
  1513. package/src/utils/fileReadCache.ts +96 -0
  1514. package/src/utils/fileStateCache.ts +142 -0
  1515. package/src/utils/findExecutable.ts +17 -0
  1516. package/src/utils/fingerprint.ts +76 -0
  1517. package/src/utils/forkedAgent.ts +689 -0
  1518. package/src/utils/format.ts +308 -0
  1519. package/src/utils/formatBriefTimestamp.ts +81 -0
  1520. package/src/utils/fpsTracker.ts +47 -0
  1521. package/src/utils/frontmatterParser.ts +370 -0
  1522. package/src/utils/fsOperations.ts +770 -0
  1523. package/src/utils/fullscreen.ts +202 -0
  1524. package/src/utils/generatedFiles.ts +136 -0
  1525. package/src/utils/generators.ts +88 -0
  1526. package/src/utils/genericProcessUtils.ts +184 -0
  1527. package/src/utils/getWorktreePaths.ts +70 -0
  1528. package/src/utils/getWorktreePathsPortable.ts +27 -0
  1529. package/src/utils/ghPrStatus.ts +106 -0
  1530. package/src/utils/git/gitConfigParser.ts +277 -0
  1531. package/src/utils/git/gitFilesystem.ts +699 -0
  1532. package/src/utils/git/gitignore.ts +99 -0
  1533. package/src/utils/git.ts +926 -0
  1534. package/src/utils/gitDiff.ts +532 -0
  1535. package/src/utils/gitSettings.ts +18 -0
  1536. package/src/utils/github/ghAuthStatus.ts +29 -0
  1537. package/src/utils/githubRepoPathMapping.ts +162 -0
  1538. package/src/utils/glob.ts +130 -0
  1539. package/src/utils/gracefulShutdown.ts +529 -0
  1540. package/src/utils/groupToolUses.ts +182 -0
  1541. package/src/utils/handlePromptSubmit.ts +610 -0
  1542. package/src/utils/hash.ts +46 -0
  1543. package/src/utils/headlessProfiler.ts +178 -0
  1544. package/src/utils/heapDumpService.ts +303 -0
  1545. package/src/utils/heatmap.ts +198 -0
  1546. package/src/utils/highlightMatch.tsx +28 -0
  1547. package/src/utils/hooks/AsyncHookRegistry.ts +309 -0
  1548. package/src/utils/hooks/apiQueryHookHelper.ts +141 -0
  1549. package/src/utils/hooks/execAgentHook.ts +339 -0
  1550. package/src/utils/hooks/execHttpHook.ts +242 -0
  1551. package/src/utils/hooks/execPromptHook.ts +211 -0
  1552. package/src/utils/hooks/fileChangedWatcher.ts +191 -0
  1553. package/src/utils/hooks/hookEvents.ts +192 -0
  1554. package/src/utils/hooks/hookHelpers.ts +83 -0
  1555. package/src/utils/hooks/hooksConfigManager.ts +400 -0
  1556. package/src/utils/hooks/hooksConfigSnapshot.ts +133 -0
  1557. package/src/utils/hooks/hooksSettings.ts +271 -0
  1558. package/src/utils/hooks/postSamplingHooks.ts +70 -0
  1559. package/src/utils/hooks/registerFrontmatterHooks.ts +67 -0
  1560. package/src/utils/hooks/registerSkillHooks.ts +64 -0
  1561. package/src/utils/hooks/sessionHooks.ts +447 -0
  1562. package/src/utils/hooks/skillImprovement.ts +267 -0
  1563. package/src/utils/hooks/ssrfGuard.ts +294 -0
  1564. package/src/utils/hooks.ts +5022 -0
  1565. package/src/utils/horizontalScroll.ts +137 -0
  1566. package/src/utils/http.ts +136 -0
  1567. package/src/utils/hyperlink.ts +39 -0
  1568. package/src/utils/iTermBackup.ts +73 -0
  1569. package/src/utils/ide.ts +1494 -0
  1570. package/src/utils/idePathConversion.ts +90 -0
  1571. package/src/utils/idleTimeout.ts +53 -0
  1572. package/src/utils/imagePaste.ts +416 -0
  1573. package/src/utils/imageResizer.ts +880 -0
  1574. package/src/utils/imageStore.ts +167 -0
  1575. package/src/utils/imageValidation.ts +104 -0
  1576. package/src/utils/immediateCommand.ts +15 -0
  1577. package/src/utils/inProcessTeammateHelpers.ts +102 -0
  1578. package/src/utils/ink.ts +26 -0
  1579. package/src/utils/intl.ts +94 -0
  1580. package/src/utils/jetbrains.ts +191 -0
  1581. package/src/utils/json.ts +277 -0
  1582. package/src/utils/jsonRead.ts +16 -0
  1583. package/src/utils/keyboardShortcuts.ts +14 -0
  1584. package/src/utils/lazySchema.ts +8 -0
  1585. package/src/utils/listSessionsImpl.ts +454 -0
  1586. package/src/utils/localInstaller.ts +162 -0
  1587. package/src/utils/lockfile.ts +43 -0
  1588. package/src/utils/log.ts +362 -0
  1589. package/src/utils/logoV2Utils.ts +350 -0
  1590. package/src/utils/mailbox.ts +73 -0
  1591. package/src/utils/managedEnv.ts +199 -0
  1592. package/src/utils/managedEnvConstants.ts +191 -0
  1593. package/src/utils/markdown.ts +381 -0
  1594. package/src/utils/markdownConfigLoader.ts +600 -0
  1595. package/src/utils/mcp/dateTimeParser.ts +121 -0
  1596. package/src/utils/mcp/elicitationValidation.ts +336 -0
  1597. package/src/utils/mcpInstructionsDelta.ts +130 -0
  1598. package/src/utils/mcpOutputStorage.ts +189 -0
  1599. package/src/utils/mcpValidation.ts +208 -0
  1600. package/src/utils/mcpWebSocketTransport.ts +200 -0
  1601. package/src/utils/memoize.ts +269 -0
  1602. package/src/utils/memory/types.ts +12 -0
  1603. package/src/utils/memory/versions.ts +8 -0
  1604. package/src/utils/memoryFileDetection.ts +289 -0
  1605. package/src/utils/messagePredicates.ts +8 -0
  1606. package/src/utils/messageQueueManager.ts +547 -0
  1607. package/src/utils/messages/mappers.ts +290 -0
  1608. package/src/utils/messages/systemInit.ts +96 -0
  1609. package/src/utils/messages.ts +5512 -0
  1610. package/src/utils/model/agent.ts +157 -0
  1611. package/src/utils/model/aliases.ts +25 -0
  1612. package/src/utils/model/antModels.ts +64 -0
  1613. package/src/utils/model/bedrock.ts +265 -0
  1614. package/src/utils/model/check1mAccess.ts +72 -0
  1615. package/src/utils/model/configs.ts +118 -0
  1616. package/src/utils/model/contextWindowUpgradeCheck.ts +47 -0
  1617. package/src/utils/model/deprecation.ts +101 -0
  1618. package/src/utils/model/model.ts +634 -0
  1619. package/src/utils/model/modelAllowlist.ts +170 -0
  1620. package/src/utils/model/modelCapabilities.ts +118 -0
  1621. package/src/utils/model/modelOptions.ts +540 -0
  1622. package/src/utils/model/modelStrings.ts +166 -0
  1623. package/src/utils/model/modelSupportOverrides.ts +50 -0
  1624. package/src/utils/model/providers.ts +46 -0
  1625. package/src/utils/model/validateModel.ts +159 -0
  1626. package/src/utils/modelCost.ts +235 -0
  1627. package/src/utils/modifiers.ts +36 -0
  1628. package/src/utils/mtls.ts +179 -0
  1629. package/src/utils/nativeInstaller/download.ts +523 -0
  1630. package/src/utils/nativeInstaller/index.ts +18 -0
  1631. package/src/utils/nativeInstaller/installer.ts +1708 -0
  1632. package/src/utils/nativeInstaller/packageManagers.ts +336 -0
  1633. package/src/utils/nativeInstaller/pidLock.ts +433 -0
  1634. package/src/utils/notebook.ts +224 -0
  1635. package/src/utils/objectGroupBy.ts +18 -0
  1636. package/src/utils/pasteStore.ts +104 -0
  1637. package/src/utils/path.ts +155 -0
  1638. package/src/utils/pdf.ts +300 -0
  1639. package/src/utils/pdfUtils.ts +70 -0
  1640. package/src/utils/peerAddress.ts +21 -0
  1641. package/src/utils/permissions/PermissionMode.ts +141 -0
  1642. package/src/utils/permissions/PermissionPromptToolResultSchema.ts +127 -0
  1643. package/src/utils/permissions/PermissionResult.ts +35 -0
  1644. package/src/utils/permissions/PermissionRule.ts +40 -0
  1645. package/src/utils/permissions/PermissionUpdate.ts +389 -0
  1646. package/src/utils/permissions/PermissionUpdateSchema.ts +78 -0
  1647. package/src/utils/permissions/autoModeState.ts +39 -0
  1648. package/src/utils/permissions/bashClassifier.ts +61 -0
  1649. package/src/utils/permissions/bypassPermissionsKillswitch.ts +155 -0
  1650. package/src/utils/permissions/classifierDecision.ts +98 -0
  1651. package/src/utils/permissions/classifierShared.ts +39 -0
  1652. package/src/utils/permissions/dangerousPatterns.ts +80 -0
  1653. package/src/utils/permissions/denialTracking.ts +45 -0
  1654. package/src/utils/permissions/filesystem.ts +1777 -0
  1655. package/src/utils/permissions/getNextPermissionMode.ts +101 -0
  1656. package/src/utils/permissions/pathValidation.ts +485 -0
  1657. package/src/utils/permissions/permissionExplainer.ts +250 -0
  1658. package/src/utils/permissions/permissionRuleParser.ts +198 -0
  1659. package/src/utils/permissions/permissionSetup.ts +1532 -0
  1660. package/src/utils/permissions/permissions.ts +1486 -0
  1661. package/src/utils/permissions/permissionsLoader.ts +296 -0
  1662. package/src/utils/permissions/shadowedRuleDetection.ts +234 -0
  1663. package/src/utils/permissions/shellRuleMatching.ts +228 -0
  1664. package/src/utils/permissions/yoloClassifier.ts +1495 -0
  1665. package/src/utils/planModeV2.ts +95 -0
  1666. package/src/utils/plans.ts +397 -0
  1667. package/src/utils/platform.ts +150 -0
  1668. package/src/utils/plugins/addDirPluginSettings.ts +71 -0
  1669. package/src/utils/plugins/cacheUtils.ts +196 -0
  1670. package/src/utils/plugins/dependencyResolver.ts +305 -0
  1671. package/src/utils/plugins/fetchTelemetry.ts +135 -0
  1672. package/src/utils/plugins/gitAvailability.ts +69 -0
  1673. package/src/utils/plugins/headlessPluginInstall.ts +174 -0
  1674. package/src/utils/plugins/hintRecommendation.ts +164 -0
  1675. package/src/utils/plugins/installCounts.ts +292 -0
  1676. package/src/utils/plugins/installedPluginsManager.ts +1268 -0
  1677. package/src/utils/plugins/loadPluginAgents.ts +348 -0
  1678. package/src/utils/plugins/loadPluginCommands.ts +946 -0
  1679. package/src/utils/plugins/loadPluginHooks.ts +287 -0
  1680. package/src/utils/plugins/loadPluginOutputStyles.ts +178 -0
  1681. package/src/utils/plugins/lspPluginIntegration.ts +387 -0
  1682. package/src/utils/plugins/lspRecommendation.ts +374 -0
  1683. package/src/utils/plugins/managedPlugins.ts +27 -0
  1684. package/src/utils/plugins/marketplaceHelpers.ts +592 -0
  1685. package/src/utils/plugins/marketplaceManager.ts +2643 -0
  1686. package/src/utils/plugins/mcpPluginIntegration.ts +634 -0
  1687. package/src/utils/plugins/mcpbHandler.ts +968 -0
  1688. package/src/utils/plugins/officialMarketplace.ts +25 -0
  1689. package/src/utils/plugins/officialMarketplaceGcs.ts +216 -0
  1690. package/src/utils/plugins/officialMarketplaceStartupCheck.ts +439 -0
  1691. package/src/utils/plugins/orphanedPluginFilter.ts +114 -0
  1692. package/src/utils/plugins/parseMarketplaceInput.ts +162 -0
  1693. package/src/utils/plugins/performStartupChecks.tsx +70 -0
  1694. package/src/utils/plugins/pluginAutoupdate.ts +284 -0
  1695. package/src/utils/plugins/pluginBlocklist.ts +127 -0
  1696. package/src/utils/plugins/pluginDirectories.ts +178 -0
  1697. package/src/utils/plugins/pluginFlagging.ts +208 -0
  1698. package/src/utils/plugins/pluginIdentifier.ts +123 -0
  1699. package/src/utils/plugins/pluginInstallationHelpers.ts +595 -0
  1700. package/src/utils/plugins/pluginLoader.ts +3302 -0
  1701. package/src/utils/plugins/pluginOptionsStorage.ts +400 -0
  1702. package/src/utils/plugins/pluginPolicy.ts +20 -0
  1703. package/src/utils/plugins/pluginStartupCheck.ts +341 -0
  1704. package/src/utils/plugins/pluginVersioning.ts +157 -0
  1705. package/src/utils/plugins/reconciler.ts +265 -0
  1706. package/src/utils/plugins/refresh.ts +215 -0
  1707. package/src/utils/plugins/schemas.ts +1681 -0
  1708. package/src/utils/plugins/validatePlugin.ts +903 -0
  1709. package/src/utils/plugins/walkPluginMarkdown.ts +69 -0
  1710. package/src/utils/plugins/zipCache.ts +406 -0
  1711. package/src/utils/plugins/zipCacheAdapters.ts +164 -0
  1712. package/src/utils/powershell/dangerousCmdlets.ts +185 -0
  1713. package/src/utils/powershell/parser.ts +1804 -0
  1714. package/src/utils/powershell/staticPrefix.ts +316 -0
  1715. package/src/utils/preflightChecks.tsx +151 -0
  1716. package/src/utils/privacyLevel.ts +55 -0
  1717. package/src/utils/process.ts +68 -0
  1718. package/src/utils/processUserInput/processBashCommand.tsx +140 -0
  1719. package/src/utils/processUserInput/processSlashCommand.tsx +922 -0
  1720. package/src/utils/processUserInput/processTextPrompt.ts +100 -0
  1721. package/src/utils/processUserInput/processUserInput.ts +605 -0
  1722. package/src/utils/profilerBase.ts +46 -0
  1723. package/src/utils/promptCategory.ts +49 -0
  1724. package/src/utils/promptEditor.ts +188 -0
  1725. package/src/utils/promptShellExecution.ts +183 -0
  1726. package/src/utils/proxy.ts +426 -0
  1727. package/src/utils/queryContext.ts +179 -0
  1728. package/src/utils/queryHelpers.ts +552 -0
  1729. package/src/utils/queryProfiler.ts +301 -0
  1730. package/src/utils/queueProcessor.ts +95 -0
  1731. package/src/utils/readEditContext.ts +227 -0
  1732. package/src/utils/readFileInRange.ts +383 -0
  1733. package/src/utils/releaseNotes.ts +360 -0
  1734. package/src/utils/renderOptions.ts +77 -0
  1735. package/src/utils/ripgrep.ts +679 -0
  1736. package/src/utils/sandbox/sandbox-adapter.ts +985 -0
  1737. package/src/utils/sandbox/sandbox-ui-utils.ts +12 -0
  1738. package/src/utils/sanitization.ts +91 -0
  1739. package/src/utils/screenshotClipboard.ts +121 -0
  1740. package/src/utils/sdkEventQueue.ts +134 -0
  1741. package/src/utils/secureStorage/fallbackStorage.ts +70 -0
  1742. package/src/utils/secureStorage/index.ts +17 -0
  1743. package/src/utils/secureStorage/keychainPrefetch.ts +116 -0
  1744. package/src/utils/secureStorage/macOsKeychainHelpers.ts +111 -0
  1745. package/src/utils/secureStorage/macOsKeychainStorage.ts +231 -0
  1746. package/src/utils/secureStorage/plainTextStorage.ts +84 -0
  1747. package/src/utils/semanticBoolean.ts +29 -0
  1748. package/src/utils/semanticNumber.ts +36 -0
  1749. package/src/utils/semver.ts +59 -0
  1750. package/src/utils/sequential.ts +56 -0
  1751. package/src/utils/sessionActivity.ts +133 -0
  1752. package/src/utils/sessionEnvVars.ts +22 -0
  1753. package/src/utils/sessionEnvironment.ts +166 -0
  1754. package/src/utils/sessionFileAccessHooks.ts +250 -0
  1755. package/src/utils/sessionIngressAuth.ts +140 -0
  1756. package/src/utils/sessionRestore.ts +551 -0
  1757. package/src/utils/sessionStart.ts +232 -0
  1758. package/src/utils/sessionState.ts +150 -0
  1759. package/src/utils/sessionStorage.ts +5105 -0
  1760. package/src/utils/sessionStoragePortable.ts +793 -0
  1761. package/src/utils/sessionTitle.ts +129 -0
  1762. package/src/utils/sessionUrl.ts +64 -0
  1763. package/src/utils/set.ts +53 -0
  1764. package/src/utils/settings/allErrors.ts +32 -0
  1765. package/src/utils/settings/applySettingsChange.ts +92 -0
  1766. package/src/utils/settings/changeDetector.ts +488 -0
  1767. package/src/utils/settings/constants.ts +202 -0
  1768. package/src/utils/settings/internalWrites.ts +37 -0
  1769. package/src/utils/settings/managedPath.ts +34 -0
  1770. package/src/utils/settings/mdm/constants.ts +81 -0
  1771. package/src/utils/settings/mdm/rawRead.ts +130 -0
  1772. package/src/utils/settings/mdm/settings.ts +316 -0
  1773. package/src/utils/settings/permissionValidation.ts +262 -0
  1774. package/src/utils/settings/pluginOnlyPolicy.ts +60 -0
  1775. package/src/utils/settings/schemaOutput.ts +8 -0
  1776. package/src/utils/settings/settings.ts +1015 -0
  1777. package/src/utils/settings/settingsCache.ts +80 -0
  1778. package/src/utils/settings/toolValidationConfig.ts +103 -0
  1779. package/src/utils/settings/types.ts +1148 -0
  1780. package/src/utils/settings/validateEditTool.ts +45 -0
  1781. package/src/utils/settings/validation.ts +265 -0
  1782. package/src/utils/settings/validationTips.ts +164 -0
  1783. package/src/utils/shell/bashProvider.ts +255 -0
  1784. package/src/utils/shell/outputLimits.ts +14 -0
  1785. package/src/utils/shell/powershellDetection.ts +107 -0
  1786. package/src/utils/shell/powershellProvider.ts +123 -0
  1787. package/src/utils/shell/prefix.ts +367 -0
  1788. package/src/utils/shell/readOnlyCommandValidation.ts +1893 -0
  1789. package/src/utils/shell/resolveDefaultShell.ts +14 -0
  1790. package/src/utils/shell/shellProvider.ts +33 -0
  1791. package/src/utils/shell/shellToolUtils.ts +22 -0
  1792. package/src/utils/shell/specPrefix.ts +241 -0
  1793. package/src/utils/shellConfig.ts +167 -0
  1794. package/src/utils/sideQuery.ts +222 -0
  1795. package/src/utils/sideQuestion.ts +155 -0
  1796. package/src/utils/signal.ts +43 -0
  1797. package/src/utils/sinks.ts +16 -0
  1798. package/src/utils/skills/skillChangeDetector.ts +311 -0
  1799. package/src/utils/slashCommandParsing.ts +60 -0
  1800. package/src/utils/sleep.ts +84 -0
  1801. package/src/utils/sliceAnsi.ts +91 -0
  1802. package/src/utils/slowOperations.ts +286 -0
  1803. package/src/utils/standaloneAgent.ts +23 -0
  1804. package/src/utils/startupProfiler.ts +194 -0
  1805. package/src/utils/staticRender.tsx +116 -0
  1806. package/src/utils/stats.ts +1061 -0
  1807. package/src/utils/statsCache.ts +434 -0
  1808. package/src/utils/status.tsx +362 -0
  1809. package/src/utils/statusNoticeDefinitions.tsx +198 -0
  1810. package/src/utils/statusNoticeHelpers.ts +20 -0
  1811. package/src/utils/stream.ts +76 -0
  1812. package/src/utils/streamJsonStdoutGuard.ts +123 -0
  1813. package/src/utils/streamlinedTransform.ts +201 -0
  1814. package/src/utils/stringUtils.ts +235 -0
  1815. package/src/utils/subprocessEnv.ts +99 -0
  1816. package/src/utils/suggestions/commandSuggestions.ts +567 -0
  1817. package/src/utils/suggestions/directoryCompletion.ts +263 -0
  1818. package/src/utils/suggestions/shellHistoryCompletion.ts +119 -0
  1819. package/src/utils/suggestions/skillUsageTracking.ts +55 -0
  1820. package/src/utils/suggestions/slackChannelSuggestions.ts +209 -0
  1821. package/src/utils/swarm/It2SetupPrompt.tsx +380 -0
  1822. package/src/utils/swarm/backends/ITermBackend.ts +370 -0
  1823. package/src/utils/swarm/backends/InProcessBackend.ts +339 -0
  1824. package/src/utils/swarm/backends/PaneBackendExecutor.ts +354 -0
  1825. package/src/utils/swarm/backends/TmuxBackend.ts +764 -0
  1826. package/src/utils/swarm/backends/detection.ts +128 -0
  1827. package/src/utils/swarm/backends/it2Setup.ts +245 -0
  1828. package/src/utils/swarm/backends/registry.ts +464 -0
  1829. package/src/utils/swarm/backends/teammateModeSnapshot.ts +87 -0
  1830. package/src/utils/swarm/backends/types.ts +311 -0
  1831. package/src/utils/swarm/constants.ts +33 -0
  1832. package/src/utils/swarm/inProcessRunner.ts +1552 -0
  1833. package/src/utils/swarm/leaderPermissionBridge.ts +54 -0
  1834. package/src/utils/swarm/permissionSync.ts +928 -0
  1835. package/src/utils/swarm/reconnection.ts +119 -0
  1836. package/src/utils/swarm/spawnInProcess.ts +328 -0
  1837. package/src/utils/swarm/spawnUtils.ts +146 -0
  1838. package/src/utils/swarm/teamHelpers.ts +683 -0
  1839. package/src/utils/swarm/teammateInit.ts +129 -0
  1840. package/src/utils/swarm/teammateLayoutManager.ts +107 -0
  1841. package/src/utils/swarm/teammateModel.ts +10 -0
  1842. package/src/utils/swarm/teammatePromptAddendum.ts +18 -0
  1843. package/src/utils/systemDirectories.ts +74 -0
  1844. package/src/utils/systemPrompt.ts +123 -0
  1845. package/src/utils/systemPromptType.ts +14 -0
  1846. package/src/utils/systemTheme.ts +119 -0
  1847. package/src/utils/taggedId.ts +54 -0
  1848. package/src/utils/task/TaskOutput.ts +390 -0
  1849. package/src/utils/task/diskOutput.ts +451 -0
  1850. package/src/utils/task/framework.ts +308 -0
  1851. package/src/utils/task/outputFormatting.ts +38 -0
  1852. package/src/utils/task/sdkProgress.ts +36 -0
  1853. package/src/utils/tasks.ts +862 -0
  1854. package/src/utils/teamDiscovery.ts +81 -0
  1855. package/src/utils/teamMemoryOps.ts +88 -0
  1856. package/src/utils/teammate.ts +292 -0
  1857. package/src/utils/teammateContext.ts +96 -0
  1858. package/src/utils/teammateMailbox.ts +1183 -0
  1859. package/src/utils/telemetry/betaSessionTracing.ts +491 -0
  1860. package/src/utils/telemetry/bigqueryExporter.ts +252 -0
  1861. package/src/utils/telemetry/events.ts +75 -0
  1862. package/src/utils/telemetry/instrumentation.ts +825 -0
  1863. package/src/utils/telemetry/logger.ts +26 -0
  1864. package/src/utils/telemetry/perfettoTracing.ts +1120 -0
  1865. package/src/utils/telemetry/pluginTelemetry.ts +289 -0
  1866. package/src/utils/telemetry/sessionTracing.ts +927 -0
  1867. package/src/utils/telemetry/skillLoadedEvent.ts +39 -0
  1868. package/src/utils/telemetryAttributes.ts +71 -0
  1869. package/src/utils/teleport/api.ts +466 -0
  1870. package/src/utils/teleport/environmentSelection.ts +77 -0
  1871. package/src/utils/teleport/environments.ts +120 -0
  1872. package/src/utils/teleport/gitBundle.ts +292 -0
  1873. package/src/utils/teleport.tsx +1226 -0
  1874. package/src/utils/tempfile.ts +31 -0
  1875. package/src/utils/terminal.ts +131 -0
  1876. package/src/utils/terminalPanel.ts +191 -0
  1877. package/src/utils/textHighlighting.ts +166 -0
  1878. package/src/utils/theme.ts +639 -0
  1879. package/src/utils/thinking.ts +162 -0
  1880. package/src/utils/timeouts.ts +39 -0
  1881. package/src/utils/tmuxSocket.ts +427 -0
  1882. package/src/utils/todo/types.ts +18 -0
  1883. package/src/utils/tokenBudget.ts +73 -0
  1884. package/src/utils/tokens.ts +261 -0
  1885. package/src/utils/toolErrors.ts +132 -0
  1886. package/src/utils/toolPool.ts +79 -0
  1887. package/src/utils/toolResultStorage.ts +1040 -0
  1888. package/src/utils/toolSchemaCache.ts +26 -0
  1889. package/src/utils/toolSearch.ts +756 -0
  1890. package/src/utils/transcriptSearch.ts +202 -0
  1891. package/src/utils/treeify.ts +170 -0
  1892. package/src/utils/truncate.ts +179 -0
  1893. package/src/utils/ultraplan/ccrSession.ts +349 -0
  1894. package/src/utils/ultraplan/keyword.ts +127 -0
  1895. package/src/utils/ultraplan/prompt.txt +1 -0
  1896. package/src/utils/unaryLogging.ts +39 -0
  1897. package/src/utils/undercover.ts +89 -0
  1898. package/src/utils/user.ts +194 -0
  1899. package/src/utils/userAgent.ts +10 -0
  1900. package/src/utils/userPromptKeywords.ts +27 -0
  1901. package/src/utils/uuid.ts +27 -0
  1902. package/src/utils/warningHandler.ts +121 -0
  1903. package/src/utils/which.ts +82 -0
  1904. package/src/utils/windowsPaths.ts +173 -0
  1905. package/src/utils/withResolvers.ts +13 -0
  1906. package/src/utils/words.ts +800 -0
  1907. package/src/utils/workloadContext.ts +57 -0
  1908. package/src/utils/worktree.ts +1519 -0
  1909. package/src/utils/worktreeModeEnabled.ts +11 -0
  1910. package/src/utils/xdg.ts +65 -0
  1911. package/src/utils/xml.ts +16 -0
  1912. package/src/utils/yaml.ts +15 -0
  1913. package/src/utils/zodToJsonSchema.ts +23 -0
  1914. package/src/vim/motions.ts +82 -0
  1915. package/src/vim/operators.ts +556 -0
  1916. package/src/vim/textObjects.ts +186 -0
  1917. package/src/vim/transitions.ts +490 -0
  1918. package/src/vim/types.ts +199 -0
  1919. package/src/voice/voiceModeEnabled.ts +54 -0
  1920. package/start.js +1 -0
@@ -0,0 +1,4436 @@
1
+ /**
2
+ * Pure-TypeScript bash parser producing tree-sitter-bash-compatible ASTs.
3
+ *
4
+ * Downstream code in parser.ts, ast.ts, prefix.ts, ParsedCommand.ts walks this
5
+ * by field name. startIndex/endIndex are UTF-8 BYTE offsets (not JS string
6
+ * indices).
7
+ *
8
+ * Grammar reference: tree-sitter-bash. Validated against a 3449-input golden
9
+ * corpus generated from the WASM parser.
10
+ */
11
+
12
+ export type TsNode = {
13
+ type: string
14
+ text: string
15
+ startIndex: number
16
+ endIndex: number
17
+ children: TsNode[]
18
+ }
19
+
20
+ type ParserModule = {
21
+ parse: (source: string, timeoutMs?: number) => TsNode | null
22
+ }
23
+
24
+ /**
25
+ * 50ms wall-clock cap — bails out on pathological/adversarial input.
26
+ * Pass `Infinity` via `parse(src, Infinity)` to disable (e.g. correctness
27
+ * tests, where CI jitter would otherwise cause spurious null returns).
28
+ */
29
+ const PARSE_TIMEOUT_MS = 50
30
+
31
+ /** Node budget cap — bails out before OOM on deeply nested input. */
32
+ const MAX_NODES = 50_000
33
+
34
+ const MODULE: ParserModule = { parse: parseSource }
35
+
36
+ const READY = Promise.resolve()
37
+
38
+ /** No-op: pure-TS parser needs no async init. Kept for API compatibility. */
39
+ export function ensureParserInitialized(): Promise<void> {
40
+ return READY
41
+ }
42
+
43
+ /** Always succeeds — pure-TS needs no init. */
44
+ export function getParserModule(): ParserModule | null {
45
+ return MODULE
46
+ }
47
+
48
+ // ───────────────────────────── Tokenizer ─────────────────────────────
49
+
50
+ type TokenType =
51
+ | 'WORD'
52
+ | 'NUMBER'
53
+ | 'OP'
54
+ | 'NEWLINE'
55
+ | 'COMMENT'
56
+ | 'DQUOTE'
57
+ | 'SQUOTE'
58
+ | 'ANSI_C'
59
+ | 'DOLLAR'
60
+ | 'DOLLAR_PAREN'
61
+ | 'DOLLAR_BRACE'
62
+ | 'DOLLAR_DPAREN'
63
+ | 'BACKTICK'
64
+ | 'LT_PAREN'
65
+ | 'GT_PAREN'
66
+ | 'EOF'
67
+
68
+ type Token = {
69
+ type: TokenType
70
+ value: string
71
+ /** UTF-8 byte offset of first char */
72
+ start: number
73
+ /** UTF-8 byte offset one past last char */
74
+ end: number
75
+ }
76
+
77
+ const SPECIAL_VARS = new Set(['?', '$', '@', '*', '#', '-', '!', '_'])
78
+
79
+ const DECL_KEYWORDS = new Set([
80
+ 'export',
81
+ 'declare',
82
+ 'typeset',
83
+ 'readonly',
84
+ 'local',
85
+ ])
86
+
87
+ export const SHELL_KEYWORDS = new Set([
88
+ 'if',
89
+ 'then',
90
+ 'elif',
91
+ 'else',
92
+ 'fi',
93
+ 'while',
94
+ 'until',
95
+ 'for',
96
+ 'in',
97
+ 'do',
98
+ 'done',
99
+ 'case',
100
+ 'esac',
101
+ 'function',
102
+ 'select',
103
+ ])
104
+
105
+ /**
106
+ * Lexer state. Tracks both JS-string index (for charAt) and UTF-8 byte offset
107
+ * (for TsNode positions). ASCII fast path: byte == char index. Non-ASCII
108
+ * advances byte count per-codepoint.
109
+ */
110
+ type Lexer = {
111
+ src: string
112
+ len: number
113
+ /** JS string index */
114
+ i: number
115
+ /** UTF-8 byte offset */
116
+ b: number
117
+ /** Pending heredoc delimiters awaiting body scan at next newline */
118
+ heredocs: HeredocPending[]
119
+ /** Precomputed byte offset for each char index (lazy for non-ASCII) */
120
+ byteTable: Uint32Array | null
121
+ }
122
+
123
+ type HeredocPending = {
124
+ delim: string
125
+ stripTabs: boolean
126
+ quoted: boolean
127
+ /** Filled after body scan */
128
+ bodyStart: number
129
+ bodyEnd: number
130
+ endStart: number
131
+ endEnd: number
132
+ }
133
+
134
+ function makeLexer(src: string): Lexer {
135
+ return {
136
+ src,
137
+ len: src.length,
138
+ i: 0,
139
+ b: 0,
140
+ heredocs: [],
141
+ byteTable: null,
142
+ }
143
+ }
144
+
145
+ /** Advance one JS char, updating byte offset for UTF-8. */
146
+ function advance(L: Lexer): void {
147
+ const c = L.src.charCodeAt(L.i)
148
+ L.i++
149
+ if (c < 0x80) {
150
+ L.b++
151
+ } else if (c < 0x800) {
152
+ L.b += 2
153
+ } else if (c >= 0xd800 && c <= 0xdbff) {
154
+ // High surrogate — next char completes the pair, total 4 UTF-8 bytes
155
+ L.b += 4
156
+ L.i++
157
+ } else {
158
+ L.b += 3
159
+ }
160
+ }
161
+
162
+ function peek(L: Lexer, off = 0): string {
163
+ return L.i + off < L.len ? L.src[L.i + off]! : ''
164
+ }
165
+
166
+ function byteAt(L: Lexer, charIdx: number): number {
167
+ // Fast path: ASCII-only prefix means char idx == byte idx
168
+ if (L.byteTable) return L.byteTable[charIdx]!
169
+ // Build table on first non-trivial lookup
170
+ const t = new Uint32Array(L.len + 1)
171
+ let b = 0
172
+ let i = 0
173
+ while (i < L.len) {
174
+ t[i] = b
175
+ const c = L.src.charCodeAt(i)
176
+ if (c < 0x80) {
177
+ b++
178
+ i++
179
+ } else if (c < 0x800) {
180
+ b += 2
181
+ i++
182
+ } else if (c >= 0xd800 && c <= 0xdbff) {
183
+ t[i + 1] = b + 2
184
+ b += 4
185
+ i += 2
186
+ } else {
187
+ b += 3
188
+ i++
189
+ }
190
+ }
191
+ t[L.len] = b
192
+ L.byteTable = t
193
+ return t[charIdx]!
194
+ }
195
+
196
+ function isWordChar(c: string): boolean {
197
+ // Bash word chars: alphanumeric + various punctuation that doesn't start operators
198
+ return (
199
+ (c >= 'a' && c <= 'z') ||
200
+ (c >= 'A' && c <= 'Z') ||
201
+ (c >= '0' && c <= '9') ||
202
+ c === '_' ||
203
+ c === '/' ||
204
+ c === '.' ||
205
+ c === '-' ||
206
+ c === '+' ||
207
+ c === ':' ||
208
+ c === '@' ||
209
+ c === '%' ||
210
+ c === ',' ||
211
+ c === '~' ||
212
+ c === '^' ||
213
+ c === '?' ||
214
+ c === '*' ||
215
+ c === '!' ||
216
+ c === '=' ||
217
+ c === '[' ||
218
+ c === ']'
219
+ )
220
+ }
221
+
222
+ function isWordStart(c: string): boolean {
223
+ return isWordChar(c) || c === '\\'
224
+ }
225
+
226
+ function isIdentStart(c: string): boolean {
227
+ return (c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z') || c === '_'
228
+ }
229
+
230
+ function isIdentChar(c: string): boolean {
231
+ return isIdentStart(c) || (c >= '0' && c <= '9')
232
+ }
233
+
234
+ function isDigit(c: string): boolean {
235
+ return c >= '0' && c <= '9'
236
+ }
237
+
238
+ function isHexDigit(c: string): boolean {
239
+ return isDigit(c) || (c >= 'a' && c <= 'f') || (c >= 'A' && c <= 'F')
240
+ }
241
+
242
+ function isBaseDigit(c: string): boolean {
243
+ // Bash BASE#DIGITS: digits, letters, @ and _ (up to base 64)
244
+ return isIdentChar(c) || c === '@'
245
+ }
246
+
247
+ /**
248
+ * Unquoted heredoc delimiter chars. Bash accepts most non-metacharacters —
249
+ * not just identifiers. Stop at whitespace, redirects, pipe/list operators,
250
+ * and structural tokens. Allows !, -, ., +, etc. (e.g. <<!HEREDOC!).
251
+ */
252
+ function isHeredocDelimChar(c: string): boolean {
253
+ return (
254
+ c !== '' &&
255
+ c !== ' ' &&
256
+ c !== '\t' &&
257
+ c !== '\n' &&
258
+ c !== '<' &&
259
+ c !== '>' &&
260
+ c !== '|' &&
261
+ c !== '&' &&
262
+ c !== ';' &&
263
+ c !== '(' &&
264
+ c !== ')' &&
265
+ c !== "'" &&
266
+ c !== '"' &&
267
+ c !== '`' &&
268
+ c !== '\\'
269
+ )
270
+ }
271
+
272
+ function skipBlanks(L: Lexer): void {
273
+ while (L.i < L.len) {
274
+ const c = L.src[L.i]!
275
+ if (c === ' ' || c === '\t' || c === '\r') {
276
+ // \r is whitespace per tree-sitter-bash extras /\s/ — handles CRLF inputs
277
+ advance(L)
278
+ } else if (c === '\\') {
279
+ const nx = L.src[L.i + 1]
280
+ if (nx === '\n' || (nx === '\r' && L.src[L.i + 2] === '\n')) {
281
+ // Line continuation — tree-sitter extras: /\\\r?\n/
282
+ advance(L)
283
+ advance(L)
284
+ if (nx === '\r') advance(L)
285
+ } else if (nx === ' ' || nx === '\t') {
286
+ // \<space> or \<tab> — tree-sitter's _whitespace is /\\?[ \t\v]+/
287
+ advance(L)
288
+ advance(L)
289
+ } else {
290
+ break
291
+ }
292
+ } else {
293
+ break
294
+ }
295
+ }
296
+ }
297
+
298
+ /**
299
+ * Scan next token. Context-sensitive: `cmd` mode treats [ as operator (test
300
+ * command start), `arg` mode treats [ as word char (glob/subscript).
301
+ */
302
+ function nextToken(L: Lexer, ctx: 'cmd' | 'arg' = 'arg'): Token {
303
+ skipBlanks(L)
304
+ const start = L.b
305
+ if (L.i >= L.len) return { type: 'EOF', value: '', start, end: start }
306
+
307
+ const c = L.src[L.i]!
308
+ const c1 = peek(L, 1)
309
+ const c2 = peek(L, 2)
310
+
311
+ if (c === '\n') {
312
+ advance(L)
313
+ return { type: 'NEWLINE', value: '\n', start, end: L.b }
314
+ }
315
+
316
+ if (c === '#') {
317
+ const si = L.i
318
+ while (L.i < L.len && L.src[L.i] !== '\n') advance(L)
319
+ return {
320
+ type: 'COMMENT',
321
+ value: L.src.slice(si, L.i),
322
+ start,
323
+ end: L.b,
324
+ }
325
+ }
326
+
327
+ // Multi-char operators (longest match first)
328
+ if (c === '&' && c1 === '&') {
329
+ advance(L)
330
+ advance(L)
331
+ return { type: 'OP', value: '&&', start, end: L.b }
332
+ }
333
+ if (c === '|' && c1 === '|') {
334
+ advance(L)
335
+ advance(L)
336
+ return { type: 'OP', value: '||', start, end: L.b }
337
+ }
338
+ if (c === '|' && c1 === '&') {
339
+ advance(L)
340
+ advance(L)
341
+ return { type: 'OP', value: '|&', start, end: L.b }
342
+ }
343
+ if (c === ';' && c1 === ';' && c2 === '&') {
344
+ advance(L)
345
+ advance(L)
346
+ advance(L)
347
+ return { type: 'OP', value: ';;&', start, end: L.b }
348
+ }
349
+ if (c === ';' && c1 === ';') {
350
+ advance(L)
351
+ advance(L)
352
+ return { type: 'OP', value: ';;', start, end: L.b }
353
+ }
354
+ if (c === ';' && c1 === '&') {
355
+ advance(L)
356
+ advance(L)
357
+ return { type: 'OP', value: ';&', start, end: L.b }
358
+ }
359
+ if (c === '>' && c1 === '>') {
360
+ advance(L)
361
+ advance(L)
362
+ return { type: 'OP', value: '>>', start, end: L.b }
363
+ }
364
+ if (c === '>' && c1 === '&' && c2 === '-') {
365
+ advance(L)
366
+ advance(L)
367
+ advance(L)
368
+ return { type: 'OP', value: '>&-', start, end: L.b }
369
+ }
370
+ if (c === '>' && c1 === '&') {
371
+ advance(L)
372
+ advance(L)
373
+ return { type: 'OP', value: '>&', start, end: L.b }
374
+ }
375
+ if (c === '>' && c1 === '|') {
376
+ advance(L)
377
+ advance(L)
378
+ return { type: 'OP', value: '>|', start, end: L.b }
379
+ }
380
+ if (c === '&' && c1 === '>' && c2 === '>') {
381
+ advance(L)
382
+ advance(L)
383
+ advance(L)
384
+ return { type: 'OP', value: '&>>', start, end: L.b }
385
+ }
386
+ if (c === '&' && c1 === '>') {
387
+ advance(L)
388
+ advance(L)
389
+ return { type: 'OP', value: '&>', start, end: L.b }
390
+ }
391
+ if (c === '<' && c1 === '<' && c2 === '<') {
392
+ advance(L)
393
+ advance(L)
394
+ advance(L)
395
+ return { type: 'OP', value: '<<<', start, end: L.b }
396
+ }
397
+ if (c === '<' && c1 === '<' && c2 === '-') {
398
+ advance(L)
399
+ advance(L)
400
+ advance(L)
401
+ return { type: 'OP', value: '<<-', start, end: L.b }
402
+ }
403
+ if (c === '<' && c1 === '<') {
404
+ advance(L)
405
+ advance(L)
406
+ return { type: 'OP', value: '<<', start, end: L.b }
407
+ }
408
+ if (c === '<' && c1 === '&' && c2 === '-') {
409
+ advance(L)
410
+ advance(L)
411
+ advance(L)
412
+ return { type: 'OP', value: '<&-', start, end: L.b }
413
+ }
414
+ if (c === '<' && c1 === '&') {
415
+ advance(L)
416
+ advance(L)
417
+ return { type: 'OP', value: '<&', start, end: L.b }
418
+ }
419
+ if (c === '<' && c1 === '(') {
420
+ advance(L)
421
+ advance(L)
422
+ return { type: 'LT_PAREN', value: '<(', start, end: L.b }
423
+ }
424
+ if (c === '>' && c1 === '(') {
425
+ advance(L)
426
+ advance(L)
427
+ return { type: 'GT_PAREN', value: '>(', start, end: L.b }
428
+ }
429
+ if (c === '(' && c1 === '(') {
430
+ advance(L)
431
+ advance(L)
432
+ return { type: 'OP', value: '((', start, end: L.b }
433
+ }
434
+ if (c === ')' && c1 === ')') {
435
+ advance(L)
436
+ advance(L)
437
+ return { type: 'OP', value: '))', start, end: L.b }
438
+ }
439
+
440
+ if (c === '|' || c === '&' || c === ';' || c === '>' || c === '<') {
441
+ advance(L)
442
+ return { type: 'OP', value: c, start, end: L.b }
443
+ }
444
+ if (c === '(' || c === ')') {
445
+ advance(L)
446
+ return { type: 'OP', value: c, start, end: L.b }
447
+ }
448
+
449
+ // In cmd position, [ [[ { start test/group; in arg position they're word chars
450
+ if (ctx === 'cmd') {
451
+ if (c === '[' && c1 === '[') {
452
+ advance(L)
453
+ advance(L)
454
+ return { type: 'OP', value: '[[', start, end: L.b }
455
+ }
456
+ if (c === '[') {
457
+ advance(L)
458
+ return { type: 'OP', value: '[', start, end: L.b }
459
+ }
460
+ if (c === '{' && (c1 === ' ' || c1 === '\t' || c1 === '\n')) {
461
+ advance(L)
462
+ return { type: 'OP', value: '{', start, end: L.b }
463
+ }
464
+ if (c === '}') {
465
+ advance(L)
466
+ return { type: 'OP', value: '}', start, end: L.b }
467
+ }
468
+ if (c === '!' && (c1 === ' ' || c1 === '\t')) {
469
+ advance(L)
470
+ return { type: 'OP', value: '!', start, end: L.b }
471
+ }
472
+ }
473
+
474
+ if (c === '"') {
475
+ advance(L)
476
+ return { type: 'DQUOTE', value: '"', start, end: L.b }
477
+ }
478
+ if (c === "'") {
479
+ const si = L.i
480
+ advance(L)
481
+ while (L.i < L.len && L.src[L.i] !== "'") advance(L)
482
+ if (L.i < L.len) advance(L)
483
+ return {
484
+ type: 'SQUOTE',
485
+ value: L.src.slice(si, L.i),
486
+ start,
487
+ end: L.b,
488
+ }
489
+ }
490
+
491
+ if (c === '$') {
492
+ if (c1 === '(' && c2 === '(') {
493
+ advance(L)
494
+ advance(L)
495
+ advance(L)
496
+ return { type: 'DOLLAR_DPAREN', value: '$((', start, end: L.b }
497
+ }
498
+ if (c1 === '(') {
499
+ advance(L)
500
+ advance(L)
501
+ return { type: 'DOLLAR_PAREN', value: '$(', start, end: L.b }
502
+ }
503
+ if (c1 === '{') {
504
+ advance(L)
505
+ advance(L)
506
+ return { type: 'DOLLAR_BRACE', value: '${', start, end: L.b }
507
+ }
508
+ if (c1 === "'") {
509
+ // ANSI-C string $'...'
510
+ const si = L.i
511
+ advance(L)
512
+ advance(L)
513
+ while (L.i < L.len && L.src[L.i] !== "'") {
514
+ if (L.src[L.i] === '\\' && L.i + 1 < L.len) advance(L)
515
+ advance(L)
516
+ }
517
+ if (L.i < L.len) advance(L)
518
+ return {
519
+ type: 'ANSI_C',
520
+ value: L.src.slice(si, L.i),
521
+ start,
522
+ end: L.b,
523
+ }
524
+ }
525
+ advance(L)
526
+ return { type: 'DOLLAR', value: '$', start, end: L.b }
527
+ }
528
+
529
+ if (c === '`') {
530
+ advance(L)
531
+ return { type: 'BACKTICK', value: '`', start, end: L.b }
532
+ }
533
+
534
+ // File descriptor before redirect: digit+ immediately followed by > or <
535
+ if (isDigit(c)) {
536
+ let j = L.i
537
+ while (j < L.len && isDigit(L.src[j]!)) j++
538
+ const after = j < L.len ? L.src[j]! : ''
539
+ if (after === '>' || after === '<') {
540
+ const si = L.i
541
+ while (L.i < j) advance(L)
542
+ return {
543
+ type: 'WORD',
544
+ value: L.src.slice(si, L.i),
545
+ start,
546
+ end: L.b,
547
+ }
548
+ }
549
+ }
550
+
551
+ // Word / number
552
+ if (isWordStart(c) || c === '{' || c === '}') {
553
+ const si = L.i
554
+ while (L.i < L.len) {
555
+ const ch = L.src[L.i]!
556
+ if (ch === '\\') {
557
+ if (L.i + 1 >= L.len) {
558
+ // Trailing `\` at EOF — tree-sitter excludes it from the word and
559
+ // emits a sibling ERROR. Stop here so the word ends before `\`.
560
+ break
561
+ }
562
+ // Escape next char (including \n for line continuation mid-word)
563
+ if (L.src[L.i + 1] === '\n') {
564
+ advance(L)
565
+ advance(L)
566
+ continue
567
+ }
568
+ advance(L)
569
+ advance(L)
570
+ continue
571
+ }
572
+ if (!isWordChar(ch) && ch !== '{' && ch !== '}') {
573
+ break
574
+ }
575
+ advance(L)
576
+ }
577
+ if (L.i > si) {
578
+ const v = L.src.slice(si, L.i)
579
+ // Number: optional sign then digits only
580
+ if (/^-?\d+$/.test(v)) {
581
+ return { type: 'NUMBER', value: v, start, end: L.b }
582
+ }
583
+ return { type: 'WORD', value: v, start, end: L.b }
584
+ }
585
+ // Empty word (lone `\` at EOF) — fall through to single-char consumer
586
+ }
587
+
588
+ // Unknown char — consume as single-char word
589
+ advance(L)
590
+ return { type: 'WORD', value: c, start, end: L.b }
591
+ }
592
+
593
+ // ───────────────────────────── Parser ─────────────────────────────
594
+
595
+ type ParseState = {
596
+ L: Lexer
597
+ src: string
598
+ srcBytes: number
599
+ /** True when byte offsets == char indices (no multi-byte UTF-8) */
600
+ isAscii: boolean
601
+ nodeCount: number
602
+ deadline: number
603
+ aborted: boolean
604
+ /** Depth of backtick nesting — inside `...`, ` terminates words */
605
+ inBacktick: number
606
+ /** When set, parseSimpleCommand stops at this token (for `[` backtrack) */
607
+ stopToken: string | null
608
+ }
609
+
610
+ function parseSource(source: string, timeoutMs?: number): TsNode | null {
611
+ const L = makeLexer(source)
612
+ const srcBytes = byteLengthUtf8(source)
613
+ const P: ParseState = {
614
+ L,
615
+ src: source,
616
+ srcBytes,
617
+ isAscii: srcBytes === source.length,
618
+ nodeCount: 0,
619
+ deadline: performance.now() + (timeoutMs ?? PARSE_TIMEOUT_MS),
620
+ aborted: false,
621
+ inBacktick: 0,
622
+ stopToken: null,
623
+ }
624
+ try {
625
+ const program = parseProgram(P)
626
+ if (P.aborted) return null
627
+ return program
628
+ } catch {
629
+ return null
630
+ }
631
+ }
632
+
633
+ function byteLengthUtf8(s: string): number {
634
+ let b = 0
635
+ for (let i = 0; i < s.length; i++) {
636
+ const c = s.charCodeAt(i)
637
+ if (c < 0x80) b++
638
+ else if (c < 0x800) b += 2
639
+ else if (c >= 0xd800 && c <= 0xdbff) {
640
+ b += 4
641
+ i++
642
+ } else b += 3
643
+ }
644
+ return b
645
+ }
646
+
647
+ function checkBudget(P: ParseState): void {
648
+ P.nodeCount++
649
+ if (P.nodeCount > MAX_NODES) {
650
+ P.aborted = true
651
+ throw new Error('budget')
652
+ }
653
+ if ((P.nodeCount & 0x7f) === 0 && performance.now() > P.deadline) {
654
+ P.aborted = true
655
+ throw new Error('timeout')
656
+ }
657
+ }
658
+
659
+ /** Build a node. Slices text from source by byte range via char-index lookup. */
660
+ function mk(
661
+ P: ParseState,
662
+ type: string,
663
+ start: number,
664
+ end: number,
665
+ children: TsNode[],
666
+ ): TsNode {
667
+ checkBudget(P)
668
+ return {
669
+ type,
670
+ text: sliceBytes(P, start, end),
671
+ startIndex: start,
672
+ endIndex: end,
673
+ children,
674
+ }
675
+ }
676
+
677
+ function sliceBytes(P: ParseState, startByte: number, endByte: number): string {
678
+ if (P.isAscii) return P.src.slice(startByte, endByte)
679
+ // Find char indices for byte offsets. Build byte table if needed.
680
+ const L = P.L
681
+ if (!L.byteTable) byteAt(L, 0)
682
+ const t = L.byteTable!
683
+ // Binary search for char index where byte offset matches
684
+ let lo = 0
685
+ let hi = P.src.length
686
+ while (lo < hi) {
687
+ const m = (lo + hi) >>> 1
688
+ if (t[m]! < startByte) lo = m + 1
689
+ else hi = m
690
+ }
691
+ const sc = lo
692
+ lo = sc
693
+ hi = P.src.length
694
+ while (lo < hi) {
695
+ const m = (lo + hi) >>> 1
696
+ if (t[m]! < endByte) lo = m + 1
697
+ else hi = m
698
+ }
699
+ return P.src.slice(sc, lo)
700
+ }
701
+
702
+ function leaf(P: ParseState, type: string, tok: Token): TsNode {
703
+ return mk(P, type, tok.start, tok.end, [])
704
+ }
705
+
706
+ function parseProgram(P: ParseState): TsNode {
707
+ const children: TsNode[] = []
708
+ // Skip leading whitespace & newlines — program start is first content byte
709
+ skipBlanks(P.L)
710
+ while (true) {
711
+ const save = saveLex(P.L)
712
+ const t = nextToken(P.L, 'cmd')
713
+ if (t.type === 'NEWLINE') {
714
+ skipBlanks(P.L)
715
+ continue
716
+ }
717
+ restoreLex(P.L, save)
718
+ break
719
+ }
720
+ const progStart = P.L.b
721
+ while (P.L.i < P.L.len) {
722
+ const save = saveLex(P.L)
723
+ const t = nextToken(P.L, 'cmd')
724
+ if (t.type === 'EOF') break
725
+ if (t.type === 'NEWLINE') continue
726
+ if (t.type === 'COMMENT') {
727
+ children.push(leaf(P, 'comment', t))
728
+ continue
729
+ }
730
+ restoreLex(P.L, save)
731
+ const stmts = parseStatements(P, null)
732
+ for (const s of stmts) children.push(s)
733
+ if (stmts.length === 0) {
734
+ // Couldn't parse — emit ERROR and skip one token
735
+ const errTok = nextToken(P.L, 'cmd')
736
+ if (errTok.type === 'EOF') break
737
+ // Stray `;;` at program level (e.g., `var=;;` outside case) — tree-sitter
738
+ // silently elides. Keep leading `;` as ERROR (security: paste artifact).
739
+ if (
740
+ errTok.type === 'OP' &&
741
+ errTok.value === ';;' &&
742
+ children.length > 0
743
+ ) {
744
+ continue
745
+ }
746
+ children.push(mk(P, 'ERROR', errTok.start, errTok.end, []))
747
+ }
748
+ }
749
+ // tree-sitter includes trailing whitespace in program extent
750
+ const progEnd = children.length > 0 ? P.srcBytes : progStart
751
+ return mk(P, 'program', progStart, progEnd, children)
752
+ }
753
+
754
+ /** Packed as (b << 16) | i — avoids heap alloc on every backtrack. */
755
+ type LexSave = number
756
+ function saveLex(L: Lexer): LexSave {
757
+ return L.b * 0x10000 + L.i
758
+ }
759
+ function restoreLex(L: Lexer, s: LexSave): void {
760
+ L.i = s & 0xffff
761
+ L.b = s >>> 16
762
+ }
763
+
764
+ /**
765
+ * Parse a sequence of statements separated by ; & newline. Returns a flat list
766
+ * where ; and & are sibling leaves (NOT wrapped in 'list' — only && || get
767
+ * that). Stops at terminator or EOF.
768
+ */
769
+ function parseStatements(P: ParseState, terminator: string | null): TsNode[] {
770
+ const out: TsNode[] = []
771
+ while (true) {
772
+ skipBlanks(P.L)
773
+ const save = saveLex(P.L)
774
+ const t = nextToken(P.L, 'cmd')
775
+ if (t.type === 'EOF') {
776
+ restoreLex(P.L, save)
777
+ break
778
+ }
779
+ if (t.type === 'NEWLINE') {
780
+ // Process pending heredocs
781
+ if (P.L.heredocs.length > 0) {
782
+ scanHeredocBodies(P)
783
+ }
784
+ continue
785
+ }
786
+ if (t.type === 'COMMENT') {
787
+ out.push(leaf(P, 'comment', t))
788
+ continue
789
+ }
790
+ if (terminator && t.type === 'OP' && t.value === terminator) {
791
+ restoreLex(P.L, save)
792
+ break
793
+ }
794
+ if (
795
+ t.type === 'OP' &&
796
+ (t.value === ')' ||
797
+ t.value === '}' ||
798
+ t.value === ';;' ||
799
+ t.value === ';&' ||
800
+ t.value === ';;&' ||
801
+ t.value === '))' ||
802
+ t.value === ']]' ||
803
+ t.value === ']')
804
+ ) {
805
+ restoreLex(P.L, save)
806
+ break
807
+ }
808
+ if (t.type === 'BACKTICK' && P.inBacktick > 0) {
809
+ restoreLex(P.L, save)
810
+ break
811
+ }
812
+ if (
813
+ t.type === 'WORD' &&
814
+ (t.value === 'then' ||
815
+ t.value === 'elif' ||
816
+ t.value === 'else' ||
817
+ t.value === 'fi' ||
818
+ t.value === 'do' ||
819
+ t.value === 'done' ||
820
+ t.value === 'esac')
821
+ ) {
822
+ restoreLex(P.L, save)
823
+ break
824
+ }
825
+ restoreLex(P.L, save)
826
+ const stmt = parseAndOr(P)
827
+ if (!stmt) break
828
+ out.push(stmt)
829
+ // Look for separator
830
+ skipBlanks(P.L)
831
+ const save2 = saveLex(P.L)
832
+ const sep = nextToken(P.L, 'cmd')
833
+ if (sep.type === 'OP' && (sep.value === ';' || sep.value === '&')) {
834
+ // Check if terminator follows — if so, emit separator but stop
835
+ const save3 = saveLex(P.L)
836
+ const after = nextToken(P.L, 'cmd')
837
+ restoreLex(P.L, save3)
838
+ out.push(leaf(P, sep.value, sep))
839
+ if (
840
+ after.type === 'EOF' ||
841
+ (after.type === 'OP' &&
842
+ (after.value === ')' ||
843
+ after.value === '}' ||
844
+ after.value === ';;' ||
845
+ after.value === ';&' ||
846
+ after.value === ';;&')) ||
847
+ (after.type === 'WORD' &&
848
+ (after.value === 'then' ||
849
+ after.value === 'elif' ||
850
+ after.value === 'else' ||
851
+ after.value === 'fi' ||
852
+ after.value === 'do' ||
853
+ after.value === 'done' ||
854
+ after.value === 'esac'))
855
+ ) {
856
+ // Trailing separator — don't include it at program level unless
857
+ // there's content after. But at inner levels we keep it.
858
+ continue
859
+ }
860
+ } else if (sep.type === 'NEWLINE') {
861
+ if (P.L.heredocs.length > 0) {
862
+ scanHeredocBodies(P)
863
+ }
864
+ continue
865
+ } else {
866
+ restoreLex(P.L, save2)
867
+ }
868
+ }
869
+ // Trim trailing separator if at program level
870
+ return out
871
+ }
872
+
873
+ /**
874
+ * Parse pipeline chains joined by && ||. Left-associative nesting.
875
+ * tree-sitter quirk: trailing redirect on the last pipeline wraps the ENTIRE
876
+ * list in a redirected_statement — `a > x && b > y` becomes
877
+ * redirected_statement(list(redirected_statement(a,>x), &&, b), >y).
878
+ */
879
+ function parseAndOr(P: ParseState): TsNode | null {
880
+ let left = parsePipeline(P)
881
+ if (!left) return null
882
+ while (true) {
883
+ const save = saveLex(P.L)
884
+ const t = nextToken(P.L, 'cmd')
885
+ if (t.type === 'OP' && (t.value === '&&' || t.value === '||')) {
886
+ const op = leaf(P, t.value, t)
887
+ skipNewlines(P)
888
+ const right = parsePipeline(P)
889
+ if (!right) {
890
+ left = mk(P, 'list', left.startIndex, op.endIndex, [left, op])
891
+ break
892
+ }
893
+ // If right is a redirected_statement, hoist its redirects to wrap the list.
894
+ if (right.type === 'redirected_statement' && right.children.length >= 2) {
895
+ const inner = right.children[0]!
896
+ const redirs = right.children.slice(1)
897
+ const listNode = mk(P, 'list', left.startIndex, inner.endIndex, [
898
+ left,
899
+ op,
900
+ inner,
901
+ ])
902
+ const lastR = redirs[redirs.length - 1]!
903
+ left = mk(
904
+ P,
905
+ 'redirected_statement',
906
+ listNode.startIndex,
907
+ lastR.endIndex,
908
+ [listNode, ...redirs],
909
+ )
910
+ } else {
911
+ left = mk(P, 'list', left.startIndex, right.endIndex, [left, op, right])
912
+ }
913
+ } else {
914
+ restoreLex(P.L, save)
915
+ break
916
+ }
917
+ }
918
+ return left
919
+ }
920
+
921
+ function skipNewlines(P: ParseState): void {
922
+ while (true) {
923
+ const save = saveLex(P.L)
924
+ const t = nextToken(P.L, 'cmd')
925
+ if (t.type !== 'NEWLINE') {
926
+ restoreLex(P.L, save)
927
+ break
928
+ }
929
+ }
930
+ }
931
+
932
+ /**
933
+ * Parse commands joined by | or |&. Flat children with operator leaves.
934
+ * tree-sitter quirk: `a | b 2>nul | c` hoists the redirect on `b` to wrap
935
+ * the preceding pipeline fragment — pipeline(redirected_statement(
936
+ * pipeline(a,|,b), 2>nul), |, c).
937
+ */
938
+ function parsePipeline(P: ParseState): TsNode | null {
939
+ let first = parseCommand(P)
940
+ if (!first) return null
941
+ const parts: TsNode[] = [first]
942
+ while (true) {
943
+ const save = saveLex(P.L)
944
+ const t = nextToken(P.L, 'cmd')
945
+ if (t.type === 'OP' && (t.value === '|' || t.value === '|&')) {
946
+ const op = leaf(P, t.value, t)
947
+ skipNewlines(P)
948
+ const next = parseCommand(P)
949
+ if (!next) {
950
+ parts.push(op)
951
+ break
952
+ }
953
+ // Hoist trailing redirect on `next` to wrap current pipeline fragment
954
+ if (
955
+ next.type === 'redirected_statement' &&
956
+ next.children.length >= 2 &&
957
+ parts.length >= 1
958
+ ) {
959
+ const inner = next.children[0]!
960
+ const redirs = next.children.slice(1)
961
+ // Wrap existing parts + op + inner as a pipeline
962
+ const pipeKids = [...parts, op, inner]
963
+ const pipeNode = mk(
964
+ P,
965
+ 'pipeline',
966
+ pipeKids[0]!.startIndex,
967
+ inner.endIndex,
968
+ pipeKids,
969
+ )
970
+ const lastR = redirs[redirs.length - 1]!
971
+ const wrapped = mk(
972
+ P,
973
+ 'redirected_statement',
974
+ pipeNode.startIndex,
975
+ lastR.endIndex,
976
+ [pipeNode, ...redirs],
977
+ )
978
+ parts.length = 0
979
+ parts.push(wrapped)
980
+ first = wrapped
981
+ continue
982
+ }
983
+ parts.push(op, next)
984
+ } else {
985
+ restoreLex(P.L, save)
986
+ break
987
+ }
988
+ }
989
+ if (parts.length === 1) return parts[0]!
990
+ const last = parts[parts.length - 1]!
991
+ return mk(P, 'pipeline', parts[0]!.startIndex, last.endIndex, parts)
992
+ }
993
+
994
+ /** Parse a single command: simple, compound, or control structure. */
995
+ function parseCommand(P: ParseState): TsNode | null {
996
+ skipBlanks(P.L)
997
+ const save = saveLex(P.L)
998
+ const t = nextToken(P.L, 'cmd')
999
+
1000
+ if (t.type === 'EOF') {
1001
+ restoreLex(P.L, save)
1002
+ return null
1003
+ }
1004
+
1005
+ // Negation — tree-sitter wraps just the command, redirects go outside.
1006
+ // `! cmd > out` → redirected_statement(negated_command(!, cmd), >out)
1007
+ if (t.type === 'OP' && t.value === '!') {
1008
+ const bang = leaf(P, '!', t)
1009
+ const inner = parseCommand(P)
1010
+ if (!inner) {
1011
+ restoreLex(P.L, save)
1012
+ return null
1013
+ }
1014
+ // If inner is a redirected_statement, hoist redirects outside negation
1015
+ if (inner.type === 'redirected_statement' && inner.children.length >= 2) {
1016
+ const cmd = inner.children[0]!
1017
+ const redirs = inner.children.slice(1)
1018
+ const neg = mk(P, 'negated_command', bang.startIndex, cmd.endIndex, [
1019
+ bang,
1020
+ cmd,
1021
+ ])
1022
+ const lastR = redirs[redirs.length - 1]!
1023
+ return mk(P, 'redirected_statement', neg.startIndex, lastR.endIndex, [
1024
+ neg,
1025
+ ...redirs,
1026
+ ])
1027
+ }
1028
+ return mk(P, 'negated_command', bang.startIndex, inner.endIndex, [
1029
+ bang,
1030
+ inner,
1031
+ ])
1032
+ }
1033
+
1034
+ if (t.type === 'OP' && t.value === '(') {
1035
+ const open = leaf(P, '(', t)
1036
+ const body = parseStatements(P, ')')
1037
+ const closeTok = nextToken(P.L, 'cmd')
1038
+ const close =
1039
+ closeTok.type === 'OP' && closeTok.value === ')'
1040
+ ? leaf(P, ')', closeTok)
1041
+ : mk(P, ')', open.endIndex, open.endIndex, [])
1042
+ const node = mk(P, 'subshell', open.startIndex, close.endIndex, [
1043
+ open,
1044
+ ...body,
1045
+ close,
1046
+ ])
1047
+ return maybeRedirect(P, node)
1048
+ }
1049
+
1050
+ if (t.type === 'OP' && t.value === '((') {
1051
+ const open = leaf(P, '((', t)
1052
+ const exprs = parseArithCommaList(P, '))', 'var')
1053
+ const closeTok = nextToken(P.L, 'cmd')
1054
+ const close =
1055
+ closeTok.value === '))'
1056
+ ? leaf(P, '))', closeTok)
1057
+ : mk(P, '))', open.endIndex, open.endIndex, [])
1058
+ return mk(P, 'compound_statement', open.startIndex, close.endIndex, [
1059
+ open,
1060
+ ...exprs,
1061
+ close,
1062
+ ])
1063
+ }
1064
+
1065
+ if (t.type === 'OP' && t.value === '{') {
1066
+ const open = leaf(P, '{', t)
1067
+ const body = parseStatements(P, '}')
1068
+ const closeTok = nextToken(P.L, 'cmd')
1069
+ const close =
1070
+ closeTok.type === 'OP' && closeTok.value === '}'
1071
+ ? leaf(P, '}', closeTok)
1072
+ : mk(P, '}', open.endIndex, open.endIndex, [])
1073
+ const node = mk(P, 'compound_statement', open.startIndex, close.endIndex, [
1074
+ open,
1075
+ ...body,
1076
+ close,
1077
+ ])
1078
+ return maybeRedirect(P, node)
1079
+ }
1080
+
1081
+ if (t.type === 'OP' && (t.value === '[' || t.value === '[[')) {
1082
+ const open = leaf(P, t.value, t)
1083
+ const closer = t.value === '[' ? ']' : ']]'
1084
+ // Grammar: `[` can contain choice(_expression, redirected_statement).
1085
+ // Try _expression first; if we don't reach `]`, backtrack and parse as
1086
+ // redirected_statement (handles `[ ! cmd -v go &>/dev/null ]`).
1087
+ const exprSave = saveLex(P.L)
1088
+ let expr = parseTestExpr(P, closer)
1089
+ skipBlanks(P.L)
1090
+ if (t.value === '[' && peek(P.L) !== ']') {
1091
+ // Expression parse didn't reach `]` — try as redirected_statement.
1092
+ // Thread `]` stop-token so parseSimpleCommand doesn't eat it as arg.
1093
+ restoreLex(P.L, exprSave)
1094
+ const prevStop = P.stopToken
1095
+ P.stopToken = ']'
1096
+ const rstmt = parseCommand(P)
1097
+ P.stopToken = prevStop
1098
+ if (rstmt && rstmt.type === 'redirected_statement') {
1099
+ expr = rstmt
1100
+ } else {
1101
+ // Neither worked — restore and keep the expression result
1102
+ restoreLex(P.L, exprSave)
1103
+ expr = parseTestExpr(P, closer)
1104
+ }
1105
+ skipBlanks(P.L)
1106
+ }
1107
+ const closeTok = nextToken(P.L, 'arg')
1108
+ let close: TsNode
1109
+ if (closeTok.value === closer) {
1110
+ close = leaf(P, closer, closeTok)
1111
+ } else {
1112
+ close = mk(P, closer, open.endIndex, open.endIndex, [])
1113
+ }
1114
+ const kids = expr ? [open, expr, close] : [open, close]
1115
+ return mk(P, 'test_command', open.startIndex, close.endIndex, kids)
1116
+ }
1117
+
1118
+ if (t.type === 'WORD') {
1119
+ if (t.value === 'if') return maybeRedirect(P, parseIf(P, t), true)
1120
+ if (t.value === 'while' || t.value === 'until')
1121
+ return maybeRedirect(P, parseWhile(P, t), true)
1122
+ if (t.value === 'for') return maybeRedirect(P, parseFor(P, t), true)
1123
+ if (t.value === 'select') return maybeRedirect(P, parseFor(P, t), true)
1124
+ if (t.value === 'case') return maybeRedirect(P, parseCase(P, t), true)
1125
+ if (t.value === 'function') return parseFunction(P, t)
1126
+ if (DECL_KEYWORDS.has(t.value))
1127
+ return maybeRedirect(P, parseDeclaration(P, t))
1128
+ if (t.value === 'unset' || t.value === 'unsetenv') {
1129
+ return maybeRedirect(P, parseUnset(P, t))
1130
+ }
1131
+ }
1132
+
1133
+ restoreLex(P.L, save)
1134
+ return parseSimpleCommand(P)
1135
+ }
1136
+
1137
+ /**
1138
+ * Parse a simple command: [assignment]* word [arg|redirect]*
1139
+ * Returns variable_assignment if only one assignment and no command.
1140
+ */
1141
+ function parseSimpleCommand(P: ParseState): TsNode | null {
1142
+ const start = P.L.b
1143
+ const assignments: TsNode[] = []
1144
+ const preRedirects: TsNode[] = []
1145
+
1146
+ while (true) {
1147
+ skipBlanks(P.L)
1148
+ const a = tryParseAssignment(P)
1149
+ if (a) {
1150
+ assignments.push(a)
1151
+ continue
1152
+ }
1153
+ const r = tryParseRedirect(P)
1154
+ if (r) {
1155
+ preRedirects.push(r)
1156
+ continue
1157
+ }
1158
+ break
1159
+ }
1160
+
1161
+ skipBlanks(P.L)
1162
+ const save = saveLex(P.L)
1163
+ const nameTok = nextToken(P.L, 'cmd')
1164
+ if (
1165
+ nameTok.type === 'EOF' ||
1166
+ nameTok.type === 'NEWLINE' ||
1167
+ nameTok.type === 'COMMENT' ||
1168
+ (nameTok.type === 'OP' &&
1169
+ nameTok.value !== '{' &&
1170
+ nameTok.value !== '[' &&
1171
+ nameTok.value !== '[[') ||
1172
+ (nameTok.type === 'WORD' &&
1173
+ SHELL_KEYWORDS.has(nameTok.value) &&
1174
+ nameTok.value !== 'in')
1175
+ ) {
1176
+ restoreLex(P.L, save)
1177
+ // No command — standalone assignment(s) or redirect
1178
+ if (assignments.length === 1 && preRedirects.length === 0) {
1179
+ return assignments[0]!
1180
+ }
1181
+ if (preRedirects.length > 0 && assignments.length === 0) {
1182
+ // Bare redirect → redirected_statement with just file_redirect children
1183
+ const last = preRedirects[preRedirects.length - 1]!
1184
+ return mk(
1185
+ P,
1186
+ 'redirected_statement',
1187
+ preRedirects[0]!.startIndex,
1188
+ last.endIndex,
1189
+ preRedirects,
1190
+ )
1191
+ }
1192
+ if (assignments.length > 1 && preRedirects.length === 0) {
1193
+ // `A=1 B=2` with no command → variable_assignments (plural)
1194
+ const last = assignments[assignments.length - 1]!
1195
+ return mk(
1196
+ P,
1197
+ 'variable_assignments',
1198
+ assignments[0]!.startIndex,
1199
+ last.endIndex,
1200
+ assignments,
1201
+ )
1202
+ }
1203
+ if (assignments.length > 0 || preRedirects.length > 0) {
1204
+ const all = [...assignments, ...preRedirects]
1205
+ const last = all[all.length - 1]!
1206
+ return mk(P, 'command', start, last.endIndex, all)
1207
+ }
1208
+ return null
1209
+ }
1210
+ restoreLex(P.L, save)
1211
+
1212
+ // Check for function definition: name() { ... }
1213
+ const fnSave = saveLex(P.L)
1214
+ const nm = parseWord(P, 'cmd')
1215
+ if (nm && nm.type === 'word') {
1216
+ skipBlanks(P.L)
1217
+ if (peek(P.L) === '(' && peek(P.L, 1) === ')') {
1218
+ const oTok = nextToken(P.L, 'cmd')
1219
+ const cTok = nextToken(P.L, 'cmd')
1220
+ const oParen = leaf(P, '(', oTok)
1221
+ const cParen = leaf(P, ')', cTok)
1222
+ skipBlanks(P.L)
1223
+ skipNewlines(P)
1224
+ const body = parseCommand(P)
1225
+ if (body) {
1226
+ // If body is redirected_statement(compound_statement, file_redirect...),
1227
+ // hoist redirects to function_definition level per tree-sitter grammar
1228
+ let bodyKids: TsNode[] = [body]
1229
+ if (
1230
+ body.type === 'redirected_statement' &&
1231
+ body.children.length >= 2 &&
1232
+ body.children[0]!.type === 'compound_statement'
1233
+ ) {
1234
+ bodyKids = body.children
1235
+ }
1236
+ const last = bodyKids[bodyKids.length - 1]!
1237
+ return mk(P, 'function_definition', nm.startIndex, last.endIndex, [
1238
+ nm,
1239
+ oParen,
1240
+ cParen,
1241
+ ...bodyKids,
1242
+ ])
1243
+ }
1244
+ }
1245
+ }
1246
+ restoreLex(P.L, fnSave)
1247
+
1248
+ const nameArg = parseWord(P, 'cmd')
1249
+ if (!nameArg) {
1250
+ if (assignments.length === 1) return assignments[0]!
1251
+ return null
1252
+ }
1253
+
1254
+ const cmdName = mk(P, 'command_name', nameArg.startIndex, nameArg.endIndex, [
1255
+ nameArg,
1256
+ ])
1257
+
1258
+ const args: TsNode[] = []
1259
+ const redirects: TsNode[] = []
1260
+ let heredocRedirect: TsNode | null = null
1261
+
1262
+ while (true) {
1263
+ skipBlanks(P.L)
1264
+ // Post-command redirects are greedy (repeat1 $._literal) — once a redirect
1265
+ // appears after command_name, subsequent literals attach to it per grammar's
1266
+ // prec.left. `grep 2>/dev/null -q foo` → file_redirect eats `-q foo`.
1267
+ // Args parsed BEFORE the first redirect still go to command (cat a b > out).
1268
+ const r = tryParseRedirect(P, true)
1269
+ if (r) {
1270
+ if (r.type === 'heredoc_redirect') {
1271
+ heredocRedirect = r
1272
+ } else if (r.type === 'herestring_redirect') {
1273
+ args.push(r)
1274
+ } else {
1275
+ redirects.push(r)
1276
+ }
1277
+ continue
1278
+ }
1279
+ // Once a file_redirect has been seen, command args are done — grammar's
1280
+ // command rule doesn't allow file_redirect in its post-name choice, so
1281
+ // anything after belongs to redirected_statement's file_redirect children.
1282
+ if (redirects.length > 0) break
1283
+ // `[` test_command backtrack — stop at `]` so outer handler can consume it
1284
+ if (P.stopToken === ']' && peek(P.L) === ']') break
1285
+ const save2 = saveLex(P.L)
1286
+ const pk = nextToken(P.L, 'arg')
1287
+ if (
1288
+ pk.type === 'EOF' ||
1289
+ pk.type === 'NEWLINE' ||
1290
+ pk.type === 'COMMENT' ||
1291
+ (pk.type === 'OP' &&
1292
+ (pk.value === '|' ||
1293
+ pk.value === '|&' ||
1294
+ pk.value === '&&' ||
1295
+ pk.value === '||' ||
1296
+ pk.value === ';' ||
1297
+ pk.value === ';;' ||
1298
+ pk.value === ';&' ||
1299
+ pk.value === ';;&' ||
1300
+ pk.value === '&' ||
1301
+ pk.value === ')' ||
1302
+ pk.value === '}' ||
1303
+ pk.value === '))'))
1304
+ ) {
1305
+ restoreLex(P.L, save2)
1306
+ break
1307
+ }
1308
+ restoreLex(P.L, save2)
1309
+ const arg = parseWord(P, 'arg')
1310
+ if (!arg) {
1311
+ // Lone `(` in arg position — tree-sitter parses this as subshell arg
1312
+ // e.g., `echo =(cmd)` → command has ERROR(=), subshell(cmd) as args
1313
+ if (peek(P.L) === '(') {
1314
+ const oTok = nextToken(P.L, 'cmd')
1315
+ const open = leaf(P, '(', oTok)
1316
+ const body = parseStatements(P, ')')
1317
+ const cTok = nextToken(P.L, 'cmd')
1318
+ const close =
1319
+ cTok.type === 'OP' && cTok.value === ')'
1320
+ ? leaf(P, ')', cTok)
1321
+ : mk(P, ')', open.endIndex, open.endIndex, [])
1322
+ args.push(
1323
+ mk(P, 'subshell', open.startIndex, close.endIndex, [
1324
+ open,
1325
+ ...body,
1326
+ close,
1327
+ ]),
1328
+ )
1329
+ continue
1330
+ }
1331
+ break
1332
+ }
1333
+ // Lone `=` in arg position is a parse error in bash — tree-sitter wraps
1334
+ // it in ERROR for recovery. Happens in `echo =(cmd)` (zsh process-sub).
1335
+ if (arg.type === 'word' && arg.text === '=') {
1336
+ args.push(mk(P, 'ERROR', arg.startIndex, arg.endIndex, [arg]))
1337
+ continue
1338
+ }
1339
+ // Word immediately followed by `(` (no whitespace) is a parse error —
1340
+ // bash doesn't allow glob-then-subshell adjacency. tree-sitter wraps the
1341
+ // word in ERROR. Catches zsh glob qualifiers like `*.(e:'cmd':)`.
1342
+ if (
1343
+ (arg.type === 'word' || arg.type === 'concatenation') &&
1344
+ peek(P.L) === '(' &&
1345
+ P.L.b === arg.endIndex
1346
+ ) {
1347
+ args.push(mk(P, 'ERROR', arg.startIndex, arg.endIndex, [arg]))
1348
+ continue
1349
+ }
1350
+ args.push(arg)
1351
+ }
1352
+
1353
+ // preRedirects (e.g., `2>&1 cat`, `<<<str cmd`) go INSIDE the command node
1354
+ // before command_name per tree-sitter grammar, not in redirected_statement
1355
+ const cmdChildren = [...assignments, ...preRedirects, cmdName, ...args]
1356
+ const cmdEnd =
1357
+ cmdChildren.length > 0
1358
+ ? cmdChildren[cmdChildren.length - 1]!.endIndex
1359
+ : cmdName.endIndex
1360
+ const cmdStart = cmdChildren[0]!.startIndex
1361
+ const cmd = mk(P, 'command', cmdStart, cmdEnd, cmdChildren)
1362
+
1363
+ if (heredocRedirect) {
1364
+ // Scan heredoc body now
1365
+ scanHeredocBodies(P)
1366
+ const hd = P.L.heredocs.shift()
1367
+ if (hd && heredocRedirect.children.length >= 2) {
1368
+ const bodyNode = mk(
1369
+ P,
1370
+ 'heredoc_body',
1371
+ hd.bodyStart,
1372
+ hd.bodyEnd,
1373
+ hd.quoted ? [] : parseHeredocBodyContent(P, hd.bodyStart, hd.bodyEnd),
1374
+ )
1375
+ const endNode = mk(P, 'heredoc_end', hd.endStart, hd.endEnd, [])
1376
+ heredocRedirect.children.push(bodyNode, endNode)
1377
+ heredocRedirect.endIndex = hd.endEnd
1378
+ heredocRedirect.text = sliceBytes(
1379
+ P,
1380
+ heredocRedirect.startIndex,
1381
+ hd.endEnd,
1382
+ )
1383
+ }
1384
+ const allR = [...preRedirects, heredocRedirect, ...redirects]
1385
+ const rStart =
1386
+ preRedirects.length > 0
1387
+ ? Math.min(cmd.startIndex, preRedirects[0]!.startIndex)
1388
+ : cmd.startIndex
1389
+ return mk(P, 'redirected_statement', rStart, heredocRedirect.endIndex, [
1390
+ cmd,
1391
+ ...allR,
1392
+ ])
1393
+ }
1394
+
1395
+ if (redirects.length > 0) {
1396
+ const last = redirects[redirects.length - 1]!
1397
+ return mk(P, 'redirected_statement', cmd.startIndex, last.endIndex, [
1398
+ cmd,
1399
+ ...redirects,
1400
+ ])
1401
+ }
1402
+
1403
+ return cmd
1404
+ }
1405
+
1406
+ function maybeRedirect(
1407
+ P: ParseState,
1408
+ node: TsNode,
1409
+ allowHerestring = false,
1410
+ ): TsNode {
1411
+ const redirects: TsNode[] = []
1412
+ while (true) {
1413
+ skipBlanks(P.L)
1414
+ const save = saveLex(P.L)
1415
+ const r = tryParseRedirect(P)
1416
+ if (!r) break
1417
+ if (r.type === 'herestring_redirect' && !allowHerestring) {
1418
+ restoreLex(P.L, save)
1419
+ break
1420
+ }
1421
+ redirects.push(r)
1422
+ }
1423
+ if (redirects.length === 0) return node
1424
+ const last = redirects[redirects.length - 1]!
1425
+ return mk(P, 'redirected_statement', node.startIndex, last.endIndex, [
1426
+ node,
1427
+ ...redirects,
1428
+ ])
1429
+ }
1430
+
1431
+ function tryParseAssignment(P: ParseState): TsNode | null {
1432
+ const save = saveLex(P.L)
1433
+ skipBlanks(P.L)
1434
+ const startB = P.L.b
1435
+ // Must start with identifier
1436
+ if (!isIdentStart(peek(P.L))) {
1437
+ restoreLex(P.L, save)
1438
+ return null
1439
+ }
1440
+ while (isIdentChar(peek(P.L))) advance(P.L)
1441
+ const nameEnd = P.L.b
1442
+ // Optional subscript
1443
+ let subEnd = nameEnd
1444
+ if (peek(P.L) === '[') {
1445
+ advance(P.L)
1446
+ let depth = 1
1447
+ while (P.L.i < P.L.len && depth > 0) {
1448
+ const c = peek(P.L)
1449
+ if (c === '[') depth++
1450
+ else if (c === ']') depth--
1451
+ advance(P.L)
1452
+ }
1453
+ subEnd = P.L.b
1454
+ }
1455
+ const c = peek(P.L)
1456
+ const c1 = peek(P.L, 1)
1457
+ let op: string
1458
+ if (c === '=' && c1 !== '=') {
1459
+ op = '='
1460
+ } else if (c === '+' && c1 === '=') {
1461
+ op = '+='
1462
+ } else {
1463
+ restoreLex(P.L, save)
1464
+ return null
1465
+ }
1466
+ const nameNode = mk(P, 'variable_name', startB, nameEnd, [])
1467
+ // Subscript handling: wrap in subscript node if present
1468
+ let lhs: TsNode = nameNode
1469
+ if (subEnd > nameEnd) {
1470
+ const brOpen = mk(P, '[', nameEnd, nameEnd + 1, [])
1471
+ const idx = parseSubscriptIndex(P, nameEnd + 1, subEnd - 1)
1472
+ const brClose = mk(P, ']', subEnd - 1, subEnd, [])
1473
+ lhs = mk(P, 'subscript', startB, subEnd, [nameNode, brOpen, idx, brClose])
1474
+ }
1475
+ const opStart = P.L.b
1476
+ advance(P.L)
1477
+ if (op === '+=') advance(P.L)
1478
+ const opEnd = P.L.b
1479
+ const opNode = mk(P, op, opStart, opEnd, [])
1480
+ let val: TsNode | null = null
1481
+ if (peek(P.L) === '(') {
1482
+ // Array
1483
+ const aoTok = nextToken(P.L, 'cmd')
1484
+ const aOpen = leaf(P, '(', aoTok)
1485
+ const elems: TsNode[] = [aOpen]
1486
+ while (true) {
1487
+ skipBlanks(P.L)
1488
+ if (peek(P.L) === ')') break
1489
+ const e = parseWord(P, 'arg')
1490
+ if (!e) break
1491
+ elems.push(e)
1492
+ }
1493
+ const acTok = nextToken(P.L, 'cmd')
1494
+ const aClose =
1495
+ acTok.value === ')'
1496
+ ? leaf(P, ')', acTok)
1497
+ : mk(P, ')', aOpen.endIndex, aOpen.endIndex, [])
1498
+ elems.push(aClose)
1499
+ val = mk(P, 'array', aOpen.startIndex, aClose.endIndex, elems)
1500
+ } else {
1501
+ const c2 = peek(P.L)
1502
+ if (
1503
+ c2 &&
1504
+ c2 !== ' ' &&
1505
+ c2 !== '\t' &&
1506
+ c2 !== '\n' &&
1507
+ c2 !== ';' &&
1508
+ c2 !== '&' &&
1509
+ c2 !== '|' &&
1510
+ c2 !== ')' &&
1511
+ c2 !== '}'
1512
+ ) {
1513
+ val = parseWord(P, 'arg')
1514
+ }
1515
+ }
1516
+ const kids = val ? [lhs, opNode, val] : [lhs, opNode]
1517
+ const end = val ? val.endIndex : opEnd
1518
+ return mk(P, 'variable_assignment', startB, end, kids)
1519
+ }
1520
+
1521
+ /**
1522
+ * Parse subscript index content. Parsed arithmetically per tree-sitter grammar:
1523
+ * `${a[1+2]}` → binary_expression; `${a[++i]}` → unary_expression(word);
1524
+ * `${a[(($n+1))]}` → compound_statement(binary_expression). Falls back to
1525
+ * simple patterns (@, *) as word.
1526
+ */
1527
+ function parseSubscriptIndexInline(P: ParseState): TsNode | null {
1528
+ skipBlanks(P.L)
1529
+ const c = peek(P.L)
1530
+ // @ or * alone → word (associative array all-keys)
1531
+ if ((c === '@' || c === '*') && peek(P.L, 1) === ']') {
1532
+ const s = P.L.b
1533
+ advance(P.L)
1534
+ return mk(P, 'word', s, P.L.b, [])
1535
+ }
1536
+ // ((expr)) → compound_statement wrapping the inner arithmetic
1537
+ if (c === '(' && peek(P.L, 1) === '(') {
1538
+ const oStart = P.L.b
1539
+ advance(P.L)
1540
+ advance(P.L)
1541
+ const open = mk(P, '((', oStart, P.L.b, [])
1542
+ const inner = parseArithExpr(P, '))', 'var')
1543
+ skipBlanks(P.L)
1544
+ let close: TsNode
1545
+ if (peek(P.L) === ')' && peek(P.L, 1) === ')') {
1546
+ const cs = P.L.b
1547
+ advance(P.L)
1548
+ advance(P.L)
1549
+ close = mk(P, '))', cs, P.L.b, [])
1550
+ } else {
1551
+ close = mk(P, '))', P.L.b, P.L.b, [])
1552
+ }
1553
+ const kids = inner ? [open, inner, close] : [open, close]
1554
+ return mk(P, 'compound_statement', open.startIndex, close.endIndex, kids)
1555
+ }
1556
+ // Arithmetic — but bare identifiers in subscript use 'word' mode per
1557
+ // tree-sitter (${words[++counter]} → unary_expression(word)).
1558
+ return parseArithExpr(P, ']', 'word')
1559
+ }
1560
+
1561
+ /** Legacy byte-range subscript index parser — kept for callers that pre-scan. */
1562
+ function parseSubscriptIndex(
1563
+ P: ParseState,
1564
+ startB: number,
1565
+ endB: number,
1566
+ ): TsNode {
1567
+ const text = sliceBytes(P, startB, endB)
1568
+ if (/^\d+$/.test(text)) return mk(P, 'number', startB, endB, [])
1569
+ const m = /^\$([a-zA-Z_]\w*)$/.exec(text)
1570
+ if (m) {
1571
+ const dollar = mk(P, '$', startB, startB + 1, [])
1572
+ const vn = mk(P, 'variable_name', startB + 1, endB, [])
1573
+ return mk(P, 'simple_expansion', startB, endB, [dollar, vn])
1574
+ }
1575
+ if (text.length === 2 && text[0] === '$' && SPECIAL_VARS.has(text[1]!)) {
1576
+ const dollar = mk(P, '$', startB, startB + 1, [])
1577
+ const vn = mk(P, 'special_variable_name', startB + 1, endB, [])
1578
+ return mk(P, 'simple_expansion', startB, endB, [dollar, vn])
1579
+ }
1580
+ return mk(P, 'word', startB, endB, [])
1581
+ }
1582
+
1583
+ /**
1584
+ * Can the current position start a redirect destination literal?
1585
+ * Returns false at redirect ops, terminators, or file-descriptor-prefixed ops
1586
+ * so file_redirect's repeat1($._literal) stops at the right boundary.
1587
+ */
1588
+ function isRedirectLiteralStart(P: ParseState): boolean {
1589
+ const c = peek(P.L)
1590
+ if (c === '' || c === '\n') return false
1591
+ // Shell terminators and operators
1592
+ if (c === '|' || c === '&' || c === ';' || c === '(' || c === ')')
1593
+ return false
1594
+ // Redirect operators (< > with any suffix; <( >( handled by caller)
1595
+ if (c === '<' || c === '>') {
1596
+ // <( >( are process substitutions — those ARE literals
1597
+ return peek(P.L, 1) === '('
1598
+ }
1599
+ // N< N> file descriptor prefix — starts a new redirect, not a literal
1600
+ if (isDigit(c)) {
1601
+ let j = P.L.i
1602
+ while (j < P.L.len && isDigit(P.L.src[j]!)) j++
1603
+ const after = j < P.L.len ? P.L.src[j]! : ''
1604
+ if (after === '>' || after === '<') return false
1605
+ }
1606
+ // `}` only terminates if we're in a context where it's a closer — but
1607
+ // file_redirect sees `}` as word char (e.g., `>$HOME}` is valid path char).
1608
+ // Actually `}` at top level terminates compound_statement — need to stop.
1609
+ if (c === '}') return false
1610
+ // Test command closer — when parseSimpleCommand is called from `[` context,
1611
+ // `]` must terminate so parseCommand can return and `[` handler consume it.
1612
+ if (P.stopToken === ']' && c === ']') return false
1613
+ return true
1614
+ }
1615
+
1616
+ /**
1617
+ * Parse a redirect operator + destination(s).
1618
+ * @param greedy When true, file_redirect consumes repeat1($._literal) per
1619
+ * grammar's prec.left — `cmd >f a b c` attaches `a b c` to the redirect.
1620
+ * When false (preRedirect context), takes only 1 destination because
1621
+ * command's dynamic precedence beats redirected_statement's prec(-1).
1622
+ */
1623
+ function tryParseRedirect(P: ParseState, greedy = false): TsNode | null {
1624
+ const save = saveLex(P.L)
1625
+ skipBlanks(P.L)
1626
+ // File descriptor prefix?
1627
+ let fd: TsNode | null = null
1628
+ if (isDigit(peek(P.L))) {
1629
+ const startB = P.L.b
1630
+ let j = P.L.i
1631
+ while (j < P.L.len && isDigit(P.L.src[j]!)) j++
1632
+ const after = j < P.L.len ? P.L.src[j]! : ''
1633
+ if (after === '>' || after === '<') {
1634
+ while (P.L.i < j) advance(P.L)
1635
+ fd = mk(P, 'file_descriptor', startB, P.L.b, [])
1636
+ }
1637
+ }
1638
+ const t = nextToken(P.L, 'arg')
1639
+ if (t.type !== 'OP') {
1640
+ restoreLex(P.L, save)
1641
+ return null
1642
+ }
1643
+ const v = t.value
1644
+ if (v === '<<<') {
1645
+ const op = leaf(P, '<<<', t)
1646
+ skipBlanks(P.L)
1647
+ const target = parseWord(P, 'arg')
1648
+ const end = target ? target.endIndex : op.endIndex
1649
+ const kids = target ? [op, target] : [op]
1650
+ return mk(
1651
+ P,
1652
+ 'herestring_redirect',
1653
+ fd ? fd.startIndex : op.startIndex,
1654
+ end,
1655
+ fd ? [fd, ...kids] : kids,
1656
+ )
1657
+ }
1658
+ if (v === '<<' || v === '<<-') {
1659
+ const op = leaf(P, v, t)
1660
+ // Heredoc start — delimiter word (may be quoted)
1661
+ skipBlanks(P.L)
1662
+ const dStart = P.L.b
1663
+ let quoted = false
1664
+ let delim = ''
1665
+ const dc = peek(P.L)
1666
+ if (dc === "'" || dc === '"') {
1667
+ quoted = true
1668
+ advance(P.L)
1669
+ while (P.L.i < P.L.len && peek(P.L) !== dc) {
1670
+ delim += peek(P.L)
1671
+ advance(P.L)
1672
+ }
1673
+ if (P.L.i < P.L.len) advance(P.L)
1674
+ } else if (dc === '\\') {
1675
+ // Backslash-escaped delimiter: \X — exactly one escaped char, body is
1676
+ // quoted (literal). Covers <<\EOF <<\' <<\\ etc.
1677
+ quoted = true
1678
+ advance(P.L)
1679
+ if (P.L.i < P.L.len && peek(P.L) !== '\n') {
1680
+ delim += peek(P.L)
1681
+ advance(P.L)
1682
+ }
1683
+ // May be followed by more ident chars (e.g. <<\EOF → delim "EOF")
1684
+ while (P.L.i < P.L.len && isIdentChar(peek(P.L))) {
1685
+ delim += peek(P.L)
1686
+ advance(P.L)
1687
+ }
1688
+ } else {
1689
+ // Unquoted delimiter: bash accepts most non-metacharacters (not just
1690
+ // identifiers). Allow !, -, ., etc. — stop at shell metachars.
1691
+ while (P.L.i < P.L.len && isHeredocDelimChar(peek(P.L))) {
1692
+ delim += peek(P.L)
1693
+ advance(P.L)
1694
+ }
1695
+ }
1696
+ const dEnd = P.L.b
1697
+ const startNode = mk(P, 'heredoc_start', dStart, dEnd, [])
1698
+ // Register pending heredoc — body scanned at next newline
1699
+ P.L.heredocs.push({
1700
+ delim,
1701
+ stripTabs: v === '<<-',
1702
+ quoted,
1703
+ bodyStart: 0,
1704
+ bodyEnd: 0,
1705
+ endStart: 0,
1706
+ endEnd: 0,
1707
+ })
1708
+ const kids = fd ? [fd, op, startNode] : [op, startNode]
1709
+ const startIdx = fd ? fd.startIndex : op.startIndex
1710
+ // SECURITY: tree-sitter nests any pipeline/list/file_redirect appearing
1711
+ // between heredoc_start and the newline as a CHILD of heredoc_redirect.
1712
+ // `ls <<'EOF' | rm -rf /tmp/evil` must not silently drop the rm. Parse
1713
+ // trailing words and file_redirects properly (ast.ts walkHeredocRedirect
1714
+ // fails closed on any unrecognized child via tooComplex). Pipeline / list
1715
+ // operators (| && || ;) are structurally complex — emit ERROR so the same
1716
+ // fail-closed path rejects them.
1717
+ while (true) {
1718
+ skipBlanks(P.L)
1719
+ const tc = peek(P.L)
1720
+ if (tc === '\n' || tc === '' || P.L.i >= P.L.len) break
1721
+ // File redirect after delimiter: cat <<EOF > out.txt
1722
+ if (tc === '>' || tc === '<' || isDigit(tc)) {
1723
+ const rSave = saveLex(P.L)
1724
+ const r = tryParseRedirect(P)
1725
+ if (r && r.type === 'file_redirect') {
1726
+ kids.push(r)
1727
+ continue
1728
+ }
1729
+ restoreLex(P.L, rSave)
1730
+ }
1731
+ // Pipeline after heredoc_start: `one <<EOF | grep two` — tree-sitter
1732
+ // nests the pipeline as a child of heredoc_redirect. ast.ts
1733
+ // walkHeredocRedirect fails closed on pipeline/command via tooComplex.
1734
+ if (tc === '|' && peek(P.L, 1) !== '|') {
1735
+ advance(P.L)
1736
+ skipBlanks(P.L)
1737
+ const pipeCmds: TsNode[] = []
1738
+ while (true) {
1739
+ const cmd = parseCommand(P)
1740
+ if (!cmd) break
1741
+ pipeCmds.push(cmd)
1742
+ skipBlanks(P.L)
1743
+ if (peek(P.L) === '|' && peek(P.L, 1) !== '|') {
1744
+ const ps = P.L.b
1745
+ advance(P.L)
1746
+ pipeCmds.push(mk(P, '|', ps, P.L.b, []))
1747
+ skipBlanks(P.L)
1748
+ continue
1749
+ }
1750
+ break
1751
+ }
1752
+ if (pipeCmds.length > 0) {
1753
+ const pl = pipeCmds[pipeCmds.length - 1]!
1754
+ // tree-sitter always wraps in pipeline after `|`, even single command
1755
+ kids.push(
1756
+ mk(P, 'pipeline', pipeCmds[0]!.startIndex, pl.endIndex, pipeCmds),
1757
+ )
1758
+ }
1759
+ continue
1760
+ }
1761
+ // && / || after heredoc_start: `cat <<-EOF || die "..."` — tree-sitter
1762
+ // nests just the RHS command (not a list) as a child of heredoc_redirect.
1763
+ if (
1764
+ (tc === '&' && peek(P.L, 1) === '&') ||
1765
+ (tc === '|' && peek(P.L, 1) === '|')
1766
+ ) {
1767
+ advance(P.L)
1768
+ advance(P.L)
1769
+ skipBlanks(P.L)
1770
+ const rhs = parseCommand(P)
1771
+ if (rhs) kids.push(rhs)
1772
+ continue
1773
+ }
1774
+ // Terminator / unhandled metachar — consume rest of line as ERROR so
1775
+ // ast.ts rejects it. Covers ; & ( )
1776
+ if (tc === '&' || tc === ';' || tc === '(' || tc === ')') {
1777
+ const eStart = P.L.b
1778
+ while (P.L.i < P.L.len && peek(P.L) !== '\n') advance(P.L)
1779
+ kids.push(mk(P, 'ERROR', eStart, P.L.b, []))
1780
+ break
1781
+ }
1782
+ // Trailing word argument: newins <<-EOF - org.freedesktop.service
1783
+ const w = parseWord(P, 'arg')
1784
+ if (w) {
1785
+ kids.push(w)
1786
+ continue
1787
+ }
1788
+ // Unrecognized — consume rest of line as ERROR
1789
+ const eStart = P.L.b
1790
+ while (P.L.i < P.L.len && peek(P.L) !== '\n') advance(P.L)
1791
+ if (P.L.b > eStart) kids.push(mk(P, 'ERROR', eStart, P.L.b, []))
1792
+ break
1793
+ }
1794
+ return mk(P, 'heredoc_redirect', startIdx, P.L.b, kids)
1795
+ }
1796
+ // Close-fd variants: `<&-` `>&-` have OPTIONAL destination (0 or 1)
1797
+ if (v === '<&-' || v === '>&-') {
1798
+ const op = leaf(P, v, t)
1799
+ const kids: TsNode[] = []
1800
+ if (fd) kids.push(fd)
1801
+ kids.push(op)
1802
+ // Optional single destination — only consume if next is a literal
1803
+ skipBlanks(P.L)
1804
+ const dSave = saveLex(P.L)
1805
+ const dest = isRedirectLiteralStart(P) ? parseWord(P, 'arg') : null
1806
+ if (dest) {
1807
+ kids.push(dest)
1808
+ } else {
1809
+ restoreLex(P.L, dSave)
1810
+ }
1811
+ const startIdx = fd ? fd.startIndex : op.startIndex
1812
+ const end = dest ? dest.endIndex : op.endIndex
1813
+ return mk(P, 'file_redirect', startIdx, end, kids)
1814
+ }
1815
+ if (
1816
+ v === '>' ||
1817
+ v === '>>' ||
1818
+ v === '>&' ||
1819
+ v === '>|' ||
1820
+ v === '&>' ||
1821
+ v === '&>>' ||
1822
+ v === '<' ||
1823
+ v === '<&'
1824
+ ) {
1825
+ const op = leaf(P, v, t)
1826
+ const kids: TsNode[] = []
1827
+ if (fd) kids.push(fd)
1828
+ kids.push(op)
1829
+ // Grammar: destination is repeat1($._literal) — greedily consume literals
1830
+ // until a non-literal (redirect op, terminator, etc). tree-sitter's
1831
+ // prec.left makes `cmd >f a b c` attach `a b c` to the file_redirect,
1832
+ // NOT to the command. Structural quirk but required for corpus parity.
1833
+ // In preRedirect context (greedy=false), take only 1 literal because
1834
+ // command's dynamic precedence beats redirected_statement's prec(-1).
1835
+ let end = op.endIndex
1836
+ let taken = 0
1837
+ while (true) {
1838
+ skipBlanks(P.L)
1839
+ if (!isRedirectLiteralStart(P)) break
1840
+ if (!greedy && taken >= 1) break
1841
+ const tc = peek(P.L)
1842
+ const tc1 = peek(P.L, 1)
1843
+ let target: TsNode | null = null
1844
+ if ((tc === '<' || tc === '>') && tc1 === '(') {
1845
+ target = parseProcessSub(P)
1846
+ } else {
1847
+ target = parseWord(P, 'arg')
1848
+ }
1849
+ if (!target) break
1850
+ kids.push(target)
1851
+ end = target.endIndex
1852
+ taken++
1853
+ }
1854
+ const startIdx = fd ? fd.startIndex : op.startIndex
1855
+ return mk(P, 'file_redirect', startIdx, end, kids)
1856
+ }
1857
+ restoreLex(P.L, save)
1858
+ return null
1859
+ }
1860
+
1861
+ function parseProcessSub(P: ParseState): TsNode | null {
1862
+ const c = peek(P.L)
1863
+ if ((c !== '<' && c !== '>') || peek(P.L, 1) !== '(') return null
1864
+ const start = P.L.b
1865
+ advance(P.L)
1866
+ advance(P.L)
1867
+ const open = mk(P, c + '(', start, P.L.b, [])
1868
+ const body = parseStatements(P, ')')
1869
+ skipBlanks(P.L)
1870
+ let close: TsNode
1871
+ if (peek(P.L) === ')') {
1872
+ const cs = P.L.b
1873
+ advance(P.L)
1874
+ close = mk(P, ')', cs, P.L.b, [])
1875
+ } else {
1876
+ close = mk(P, ')', P.L.b, P.L.b, [])
1877
+ }
1878
+ return mk(P, 'process_substitution', start, close.endIndex, [
1879
+ open,
1880
+ ...body,
1881
+ close,
1882
+ ])
1883
+ }
1884
+
1885
+ function scanHeredocBodies(P: ParseState): void {
1886
+ // Skip to newline if not already there
1887
+ while (P.L.i < P.L.len && P.L.src[P.L.i] !== '\n') advance(P.L)
1888
+ if (P.L.i < P.L.len) advance(P.L)
1889
+ for (const hd of P.L.heredocs) {
1890
+ hd.bodyStart = P.L.b
1891
+ const delimLen = hd.delim.length
1892
+ while (P.L.i < P.L.len) {
1893
+ const lineStart = P.L.i
1894
+ const lineStartB = P.L.b
1895
+ // Skip leading tabs if <<-
1896
+ let checkI = lineStart
1897
+ if (hd.stripTabs) {
1898
+ while (checkI < P.L.len && P.L.src[checkI] === '\t') checkI++
1899
+ }
1900
+ // Check if this line is the delimiter
1901
+ if (
1902
+ P.L.src.startsWith(hd.delim, checkI) &&
1903
+ (checkI + delimLen >= P.L.len ||
1904
+ P.L.src[checkI + delimLen] === '\n' ||
1905
+ P.L.src[checkI + delimLen] === '\r')
1906
+ ) {
1907
+ hd.bodyEnd = lineStartB
1908
+ // Advance past tabs
1909
+ while (P.L.i < checkI) advance(P.L)
1910
+ hd.endStart = P.L.b
1911
+ // Advance past delimiter
1912
+ for (let k = 0; k < delimLen; k++) advance(P.L)
1913
+ hd.endEnd = P.L.b
1914
+ // Skip trailing newline
1915
+ if (P.L.i < P.L.len && P.L.src[P.L.i] === '\n') advance(P.L)
1916
+ return
1917
+ }
1918
+ // Consume line
1919
+ while (P.L.i < P.L.len && P.L.src[P.L.i] !== '\n') advance(P.L)
1920
+ if (P.L.i < P.L.len) advance(P.L)
1921
+ }
1922
+ // Unterminated
1923
+ hd.bodyEnd = P.L.b
1924
+ hd.endStart = P.L.b
1925
+ hd.endEnd = P.L.b
1926
+ }
1927
+ }
1928
+
1929
+ function parseHeredocBodyContent(
1930
+ P: ParseState,
1931
+ start: number,
1932
+ end: number,
1933
+ ): TsNode[] {
1934
+ // Parse expansions inside an unquoted heredoc body.
1935
+ const saved = saveLex(P.L)
1936
+ // Position lexer at body start
1937
+ restoreLexToByte(P, start)
1938
+ const out: TsNode[] = []
1939
+ let contentStart = P.L.b
1940
+ // tree-sitter-bash's heredoc_body rule hides the initial text segment
1941
+ // (_heredoc_body_beginning) — only content AFTER the first expansion is
1942
+ // emitted as heredoc_content. Track whether we've seen an expansion yet.
1943
+ let sawExpansion = false
1944
+ while (P.L.b < end) {
1945
+ const c = peek(P.L)
1946
+ // Backslash escapes suppress expansion: \$ \` stay literal in heredoc.
1947
+ if (c === '\\') {
1948
+ const nxt = peek(P.L, 1)
1949
+ if (nxt === '$' || nxt === '`' || nxt === '\\') {
1950
+ advance(P.L)
1951
+ advance(P.L)
1952
+ continue
1953
+ }
1954
+ advance(P.L)
1955
+ continue
1956
+ }
1957
+ if (c === '$' || c === '`') {
1958
+ const preB = P.L.b
1959
+ const exp = parseDollarLike(P)
1960
+ // Bare `$` followed by non-name (e.g. `$'` in a regex) returns a lone
1961
+ // '$' leaf, not an expansion — treat as literal content, don't split.
1962
+ if (
1963
+ exp &&
1964
+ (exp.type === 'simple_expansion' ||
1965
+ exp.type === 'expansion' ||
1966
+ exp.type === 'command_substitution' ||
1967
+ exp.type === 'arithmetic_expansion')
1968
+ ) {
1969
+ if (sawExpansion && preB > contentStart) {
1970
+ out.push(mk(P, 'heredoc_content', contentStart, preB, []))
1971
+ }
1972
+ out.push(exp)
1973
+ contentStart = P.L.b
1974
+ sawExpansion = true
1975
+ }
1976
+ continue
1977
+ }
1978
+ advance(P.L)
1979
+ }
1980
+ // Only emit heredoc_content children if there were expansions — otherwise
1981
+ // the heredoc_body is a leaf node (tree-sitter convention).
1982
+ if (sawExpansion) {
1983
+ out.push(mk(P, 'heredoc_content', contentStart, end, []))
1984
+ }
1985
+ restoreLex(P.L, saved)
1986
+ return out
1987
+ }
1988
+
1989
+ function restoreLexToByte(P: ParseState, targetByte: number): void {
1990
+ if (!P.L.byteTable) byteAt(P.L, 0)
1991
+ const t = P.L.byteTable!
1992
+ let lo = 0
1993
+ let hi = P.src.length
1994
+ while (lo < hi) {
1995
+ const m = (lo + hi) >>> 1
1996
+ if (t[m]! < targetByte) lo = m + 1
1997
+ else hi = m
1998
+ }
1999
+ P.L.i = lo
2000
+ P.L.b = targetByte
2001
+ }
2002
+
2003
+ /**
2004
+ * Parse a word-position element: bare word, string, expansion, or concatenation
2005
+ * thereof. Returns a single node; if multiple adjacent fragments, wraps in
2006
+ * concatenation.
2007
+ */
2008
+ function parseWord(P: ParseState, _ctx: 'cmd' | 'arg'): TsNode | null {
2009
+ skipBlanks(P.L)
2010
+ const parts: TsNode[] = []
2011
+ while (P.L.i < P.L.len) {
2012
+ const c = peek(P.L)
2013
+ if (
2014
+ c === ' ' ||
2015
+ c === '\t' ||
2016
+ c === '\n' ||
2017
+ c === '\r' ||
2018
+ c === '' ||
2019
+ c === '|' ||
2020
+ c === '&' ||
2021
+ c === ';' ||
2022
+ c === '(' ||
2023
+ c === ')'
2024
+ ) {
2025
+ break
2026
+ }
2027
+ // < > are redirect operators unless <( >( (process substitution)
2028
+ if (c === '<' || c === '>') {
2029
+ if (peek(P.L, 1) === '(') {
2030
+ const ps = parseProcessSub(P)
2031
+ if (ps) parts.push(ps)
2032
+ continue
2033
+ }
2034
+ break
2035
+ }
2036
+ if (c === '"') {
2037
+ parts.push(parseDoubleQuoted(P))
2038
+ continue
2039
+ }
2040
+ if (c === "'") {
2041
+ const tok = nextToken(P.L, 'arg')
2042
+ parts.push(leaf(P, 'raw_string', tok))
2043
+ continue
2044
+ }
2045
+ if (c === '$') {
2046
+ const c1 = peek(P.L, 1)
2047
+ if (c1 === "'") {
2048
+ const tok = nextToken(P.L, 'arg')
2049
+ parts.push(leaf(P, 'ansi_c_string', tok))
2050
+ continue
2051
+ }
2052
+ if (c1 === '"') {
2053
+ // Translated string: emit $ leaf + string node
2054
+ const dTok: Token = {
2055
+ type: 'DOLLAR',
2056
+ value: '$',
2057
+ start: P.L.b,
2058
+ end: P.L.b + 1,
2059
+ }
2060
+ advance(P.L)
2061
+ parts.push(leaf(P, '$', dTok))
2062
+ parts.push(parseDoubleQuoted(P))
2063
+ continue
2064
+ }
2065
+ if (c1 === '`') {
2066
+ // `$` followed by backtick — tree-sitter elides the $ entirely
2067
+ // and emits just (command_substitution). Consume $ and let next
2068
+ // iteration handle the backtick.
2069
+ advance(P.L)
2070
+ continue
2071
+ }
2072
+ const exp = parseDollarLike(P)
2073
+ if (exp) parts.push(exp)
2074
+ continue
2075
+ }
2076
+ if (c === '`') {
2077
+ if (P.inBacktick > 0) break
2078
+ const bt = parseBacktick(P)
2079
+ if (bt) parts.push(bt)
2080
+ continue
2081
+ }
2082
+ // Brace expression {1..5} or {a,b,c} — only if looks like one
2083
+ if (c === '{') {
2084
+ const be = tryParseBraceExpr(P)
2085
+ if (be) {
2086
+ parts.push(be)
2087
+ continue
2088
+ }
2089
+ // SECURITY: if `{` is immediately followed by a command terminator
2090
+ // (; | & newline or EOF), it's a standalone word — don't slurp the
2091
+ // rest of the line via tryParseBraceLikeCat. `echo {;touch /tmp/evil`
2092
+ // must split on `;` so the security walker sees `touch`.
2093
+ const nc = peek(P.L, 1)
2094
+ if (
2095
+ nc === ';' ||
2096
+ nc === '|' ||
2097
+ nc === '&' ||
2098
+ nc === '\n' ||
2099
+ nc === '' ||
2100
+ nc === ')' ||
2101
+ nc === ' ' ||
2102
+ nc === '\t'
2103
+ ) {
2104
+ const bStart = P.L.b
2105
+ advance(P.L)
2106
+ parts.push(mk(P, 'word', bStart, P.L.b, []))
2107
+ continue
2108
+ }
2109
+ // Otherwise treat { and } as word fragments
2110
+ const cat = tryParseBraceLikeCat(P)
2111
+ if (cat) {
2112
+ for (const p of cat) parts.push(p)
2113
+ continue
2114
+ }
2115
+ }
2116
+ // Standalone `}` in arg position is a word (e.g., `echo }foo`).
2117
+ // parseBareWord breaks on `}` so handle it here.
2118
+ if (c === '}') {
2119
+ const bStart = P.L.b
2120
+ advance(P.L)
2121
+ parts.push(mk(P, 'word', bStart, P.L.b, []))
2122
+ continue
2123
+ }
2124
+ // `[` and `]` are single-char word fragments (tree-sitter splits at
2125
+ // brackets: `[:lower:]` → `[` `:lower:` `]`, `{o[k]}` → 6 words).
2126
+ if (c === '[' || c === ']') {
2127
+ const bStart = P.L.b
2128
+ advance(P.L)
2129
+ parts.push(mk(P, 'word', bStart, P.L.b, []))
2130
+ continue
2131
+ }
2132
+ // Bare word fragment
2133
+ const frag = parseBareWord(P)
2134
+ if (!frag) break
2135
+ // `NN#${...}` or `NN#$(...)` → (number (expansion|command_substitution)).
2136
+ // Grammar: number can be seq(/-?(0x)?[0-9]+#/, choice(expansion, cmd_sub)).
2137
+ // `10#${cmd}` must NOT be concatenation — it's a single number node with
2138
+ // the expansion as child. Detect here: frag ends with `#`, next is $ {/(.
2139
+ if (
2140
+ frag.type === 'word' &&
2141
+ /^-?(0x)?[0-9]+#$/.test(frag.text) &&
2142
+ peek(P.L) === '$' &&
2143
+ (peek(P.L, 1) === '{' || peek(P.L, 1) === '(')
2144
+ ) {
2145
+ const exp = parseDollarLike(P)
2146
+ if (exp) {
2147
+ // Prefix `NN#` is an anonymous pattern in grammar — only the
2148
+ // expansion/cmd_sub is a named child.
2149
+ parts.push(mk(P, 'number', frag.startIndex, exp.endIndex, [exp]))
2150
+ continue
2151
+ }
2152
+ }
2153
+ parts.push(frag)
2154
+ }
2155
+ if (parts.length === 0) return null
2156
+ if (parts.length === 1) return parts[0]!
2157
+ // Concatenation
2158
+ const first = parts[0]!
2159
+ const last = parts[parts.length - 1]!
2160
+ return mk(P, 'concatenation', first.startIndex, last.endIndex, parts)
2161
+ }
2162
+
2163
+ function parseBareWord(P: ParseState): TsNode | null {
2164
+ const start = P.L.b
2165
+ const startI = P.L.i
2166
+ while (P.L.i < P.L.len) {
2167
+ const c = peek(P.L)
2168
+ if (c === '\\') {
2169
+ if (P.L.i + 1 >= P.L.len) {
2170
+ // Trailing unpaired `\` at true EOF — tree-sitter emits word WITHOUT
2171
+ // the `\` plus a sibling ERROR node. Stop here; caller emits ERROR.
2172
+ break
2173
+ }
2174
+ const nx = P.L.src[P.L.i + 1]
2175
+ if (nx === '\n' || (nx === '\r' && P.L.src[P.L.i + 2] === '\n')) {
2176
+ // Line continuation BREAKS the word (tree-sitter quirk) — handles \r?\n
2177
+ break
2178
+ }
2179
+ advance(P.L)
2180
+ advance(P.L)
2181
+ continue
2182
+ }
2183
+ if (
2184
+ c === ' ' ||
2185
+ c === '\t' ||
2186
+ c === '\n' ||
2187
+ c === '\r' ||
2188
+ c === '' ||
2189
+ c === '|' ||
2190
+ c === '&' ||
2191
+ c === ';' ||
2192
+ c === '(' ||
2193
+ c === ')' ||
2194
+ c === '<' ||
2195
+ c === '>' ||
2196
+ c === '"' ||
2197
+ c === "'" ||
2198
+ c === '$' ||
2199
+ c === '`' ||
2200
+ c === '{' ||
2201
+ c === '}' ||
2202
+ c === '[' ||
2203
+ c === ']'
2204
+ ) {
2205
+ break
2206
+ }
2207
+ advance(P.L)
2208
+ }
2209
+ if (P.L.b === start) return null
2210
+ const text = P.src.slice(startI, P.L.i)
2211
+ const type = /^-?\d+$/.test(text) ? 'number' : 'word'
2212
+ return mk(P, type, start, P.L.b, [])
2213
+ }
2214
+
2215
+ function tryParseBraceExpr(P: ParseState): TsNode | null {
2216
+ // {N..M} where N, M are numbers or single chars
2217
+ const save = saveLex(P.L)
2218
+ if (peek(P.L) !== '{') return null
2219
+ const oStart = P.L.b
2220
+ advance(P.L)
2221
+ const oEnd = P.L.b
2222
+ // First part
2223
+ const p1Start = P.L.b
2224
+ while (isDigit(peek(P.L)) || isIdentStart(peek(P.L))) advance(P.L)
2225
+ const p1End = P.L.b
2226
+ if (p1End === p1Start || peek(P.L) !== '.' || peek(P.L, 1) !== '.') {
2227
+ restoreLex(P.L, save)
2228
+ return null
2229
+ }
2230
+ const dotStart = P.L.b
2231
+ advance(P.L)
2232
+ advance(P.L)
2233
+ const dotEnd = P.L.b
2234
+ const p2Start = P.L.b
2235
+ while (isDigit(peek(P.L)) || isIdentStart(peek(P.L))) advance(P.L)
2236
+ const p2End = P.L.b
2237
+ if (p2End === p2Start || peek(P.L) !== '}') {
2238
+ restoreLex(P.L, save)
2239
+ return null
2240
+ }
2241
+ const cStart = P.L.b
2242
+ advance(P.L)
2243
+ const cEnd = P.L.b
2244
+ const p1Text = sliceBytes(P, p1Start, p1End)
2245
+ const p2Text = sliceBytes(P, p2Start, p2End)
2246
+ const p1IsNum = /^\d+$/.test(p1Text)
2247
+ const p2IsNum = /^\d+$/.test(p2Text)
2248
+ // Valid brace expression: both numbers OR both single chars. Mixed = reject.
2249
+ if (p1IsNum !== p2IsNum) {
2250
+ restoreLex(P.L, save)
2251
+ return null
2252
+ }
2253
+ if (!p1IsNum && (p1Text.length !== 1 || p2Text.length !== 1)) {
2254
+ restoreLex(P.L, save)
2255
+ return null
2256
+ }
2257
+ const p1Type = p1IsNum ? 'number' : 'word'
2258
+ const p2Type = p2IsNum ? 'number' : 'word'
2259
+ return mk(P, 'brace_expression', oStart, cEnd, [
2260
+ mk(P, '{', oStart, oEnd, []),
2261
+ mk(P, p1Type, p1Start, p1End, []),
2262
+ mk(P, '..', dotStart, dotEnd, []),
2263
+ mk(P, p2Type, p2Start, p2End, []),
2264
+ mk(P, '}', cStart, cEnd, []),
2265
+ ])
2266
+ }
2267
+
2268
+ function tryParseBraceLikeCat(P: ParseState): TsNode[] | null {
2269
+ // {a,b,c} or {} → split into word fragments like tree-sitter does
2270
+ if (peek(P.L) !== '{') return null
2271
+ const oStart = P.L.b
2272
+ advance(P.L)
2273
+ const oEnd = P.L.b
2274
+ const inner: TsNode[] = [mk(P, 'word', oStart, oEnd, [])]
2275
+ while (P.L.i < P.L.len) {
2276
+ const bc = peek(P.L)
2277
+ // SECURITY: stop at command terminators so `{foo;rm x` splits correctly.
2278
+ if (
2279
+ bc === '}' ||
2280
+ bc === '\n' ||
2281
+ bc === ';' ||
2282
+ bc === '|' ||
2283
+ bc === '&' ||
2284
+ bc === ' ' ||
2285
+ bc === '\t' ||
2286
+ bc === '<' ||
2287
+ bc === '>' ||
2288
+ bc === '(' ||
2289
+ bc === ')'
2290
+ ) {
2291
+ break
2292
+ }
2293
+ // `[` and `]` are single-char words: {o[k]} → { o [ k ] }
2294
+ if (bc === '[' || bc === ']') {
2295
+ const bStart = P.L.b
2296
+ advance(P.L)
2297
+ inner.push(mk(P, 'word', bStart, P.L.b, []))
2298
+ continue
2299
+ }
2300
+ const midStart = P.L.b
2301
+ while (P.L.i < P.L.len) {
2302
+ const mc = peek(P.L)
2303
+ if (
2304
+ mc === '}' ||
2305
+ mc === '\n' ||
2306
+ mc === ';' ||
2307
+ mc === '|' ||
2308
+ mc === '&' ||
2309
+ mc === ' ' ||
2310
+ mc === '\t' ||
2311
+ mc === '<' ||
2312
+ mc === '>' ||
2313
+ mc === '(' ||
2314
+ mc === ')' ||
2315
+ mc === '[' ||
2316
+ mc === ']'
2317
+ ) {
2318
+ break
2319
+ }
2320
+ advance(P.L)
2321
+ }
2322
+ const midEnd = P.L.b
2323
+ if (midEnd > midStart) {
2324
+ const midText = sliceBytes(P, midStart, midEnd)
2325
+ const midType = /^-?\d+$/.test(midText) ? 'number' : 'word'
2326
+ inner.push(mk(P, midType, midStart, midEnd, []))
2327
+ } else {
2328
+ break
2329
+ }
2330
+ }
2331
+ if (peek(P.L) === '}') {
2332
+ const cStart = P.L.b
2333
+ advance(P.L)
2334
+ inner.push(mk(P, 'word', cStart, P.L.b, []))
2335
+ }
2336
+ return inner
2337
+ }
2338
+
2339
+ function parseDoubleQuoted(P: ParseState): TsNode {
2340
+ const qStart = P.L.b
2341
+ advance(P.L)
2342
+ const qEnd = P.L.b
2343
+ const openQ = mk(P, '"', qStart, qEnd, [])
2344
+ const parts: TsNode[] = [openQ]
2345
+ let contentStart = P.L.b
2346
+ let contentStartI = P.L.i
2347
+ const flushContent = (): void => {
2348
+ if (P.L.b > contentStart) {
2349
+ // Tree-sitter's extras rule /\s/ has higher precedence than
2350
+ // string_content (prec -1), so whitespace-only segments are elided.
2351
+ // `" ${x} "` → (string (expansion)) not (string (string_content)(expansion)(string_content)).
2352
+ // Note: this intentionally diverges from preserving all content — cc
2353
+ // tests relying on whitespace-only string_content need updating
2354
+ // (CCReconcile).
2355
+ const txt = P.src.slice(contentStartI, P.L.i)
2356
+ if (!/^[ \t]+$/.test(txt)) {
2357
+ parts.push(mk(P, 'string_content', contentStart, P.L.b, []))
2358
+ }
2359
+ }
2360
+ }
2361
+ while (P.L.i < P.L.len) {
2362
+ const c = peek(P.L)
2363
+ if (c === '"') break
2364
+ if (c === '\\' && P.L.i + 1 < P.L.len) {
2365
+ advance(P.L)
2366
+ advance(P.L)
2367
+ continue
2368
+ }
2369
+ if (c === '\n') {
2370
+ // Split string_content at newline
2371
+ flushContent()
2372
+ advance(P.L)
2373
+ contentStart = P.L.b
2374
+ contentStartI = P.L.i
2375
+ continue
2376
+ }
2377
+ if (c === '$') {
2378
+ const c1 = peek(P.L, 1)
2379
+ if (
2380
+ c1 === '(' ||
2381
+ c1 === '{' ||
2382
+ isIdentStart(c1) ||
2383
+ SPECIAL_VARS.has(c1) ||
2384
+ isDigit(c1)
2385
+ ) {
2386
+ flushContent()
2387
+ const exp = parseDollarLike(P)
2388
+ if (exp) parts.push(exp)
2389
+ contentStart = P.L.b
2390
+ contentStartI = P.L.i
2391
+ continue
2392
+ }
2393
+ // Bare $ not at end-of-string: tree-sitter emits it as an anonymous
2394
+ // '$' token, which splits string_content. $ immediately before the
2395
+ // closing " is absorbed into the preceding string_content.
2396
+ if (c1 !== '"' && c1 !== '') {
2397
+ flushContent()
2398
+ const dS = P.L.b
2399
+ advance(P.L)
2400
+ parts.push(mk(P, '$', dS, P.L.b, []))
2401
+ contentStart = P.L.b
2402
+ contentStartI = P.L.i
2403
+ continue
2404
+ }
2405
+ }
2406
+ if (c === '`') {
2407
+ flushContent()
2408
+ const bt = parseBacktick(P)
2409
+ if (bt) parts.push(bt)
2410
+ contentStart = P.L.b
2411
+ contentStartI = P.L.i
2412
+ continue
2413
+ }
2414
+ advance(P.L)
2415
+ }
2416
+ flushContent()
2417
+ let close: TsNode
2418
+ if (peek(P.L) === '"') {
2419
+ const cStart = P.L.b
2420
+ advance(P.L)
2421
+ close = mk(P, '"', cStart, P.L.b, [])
2422
+ } else {
2423
+ close = mk(P, '"', P.L.b, P.L.b, [])
2424
+ }
2425
+ parts.push(close)
2426
+ return mk(P, 'string', qStart, close.endIndex, parts)
2427
+ }
2428
+
2429
+ function parseDollarLike(P: ParseState): TsNode | null {
2430
+ const c1 = peek(P.L, 1)
2431
+ const dStart = P.L.b
2432
+ if (c1 === '(' && peek(P.L, 2) === '(') {
2433
+ // $(( arithmetic ))
2434
+ advance(P.L)
2435
+ advance(P.L)
2436
+ advance(P.L)
2437
+ const open = mk(P, '$((', dStart, P.L.b, [])
2438
+ const exprs = parseArithCommaList(P, '))', 'var')
2439
+ skipBlanks(P.L)
2440
+ let close: TsNode
2441
+ if (peek(P.L) === ')' && peek(P.L, 1) === ')') {
2442
+ const cStart = P.L.b
2443
+ advance(P.L)
2444
+ advance(P.L)
2445
+ close = mk(P, '))', cStart, P.L.b, [])
2446
+ } else {
2447
+ close = mk(P, '))', P.L.b, P.L.b, [])
2448
+ }
2449
+ return mk(P, 'arithmetic_expansion', dStart, close.endIndex, [
2450
+ open,
2451
+ ...exprs,
2452
+ close,
2453
+ ])
2454
+ }
2455
+ if (c1 === '[') {
2456
+ // $[ arithmetic ] — legacy bash syntax, same as $((...))
2457
+ advance(P.L)
2458
+ advance(P.L)
2459
+ const open = mk(P, '$[', dStart, P.L.b, [])
2460
+ const exprs = parseArithCommaList(P, ']', 'var')
2461
+ skipBlanks(P.L)
2462
+ let close: TsNode
2463
+ if (peek(P.L) === ']') {
2464
+ const cStart = P.L.b
2465
+ advance(P.L)
2466
+ close = mk(P, ']', cStart, P.L.b, [])
2467
+ } else {
2468
+ close = mk(P, ']', P.L.b, P.L.b, [])
2469
+ }
2470
+ return mk(P, 'arithmetic_expansion', dStart, close.endIndex, [
2471
+ open,
2472
+ ...exprs,
2473
+ close,
2474
+ ])
2475
+ }
2476
+ if (c1 === '(') {
2477
+ advance(P.L)
2478
+ advance(P.L)
2479
+ const open = mk(P, '$(', dStart, P.L.b, [])
2480
+ let body = parseStatements(P, ')')
2481
+ skipBlanks(P.L)
2482
+ let close: TsNode
2483
+ if (peek(P.L) === ')') {
2484
+ const cStart = P.L.b
2485
+ advance(P.L)
2486
+ close = mk(P, ')', cStart, P.L.b, [])
2487
+ } else {
2488
+ close = mk(P, ')', P.L.b, P.L.b, [])
2489
+ }
2490
+ // $(< file) shorthand: unwrap redirected_statement → bare file_redirect
2491
+ // tree-sitter emits (command_substitution (file_redirect (word))) directly
2492
+ if (
2493
+ body.length === 1 &&
2494
+ body[0]!.type === 'redirected_statement' &&
2495
+ body[0]!.children.length === 1 &&
2496
+ body[0]!.children[0]!.type === 'file_redirect'
2497
+ ) {
2498
+ body = body[0]!.children
2499
+ }
2500
+ return mk(P, 'command_substitution', dStart, close.endIndex, [
2501
+ open,
2502
+ ...body,
2503
+ close,
2504
+ ])
2505
+ }
2506
+ if (c1 === '{') {
2507
+ advance(P.L)
2508
+ advance(P.L)
2509
+ const open = mk(P, '${', dStart, P.L.b, [])
2510
+ const inner = parseExpansionBody(P)
2511
+ let close: TsNode
2512
+ if (peek(P.L) === '}') {
2513
+ const cStart = P.L.b
2514
+ advance(P.L)
2515
+ close = mk(P, '}', cStart, P.L.b, [])
2516
+ } else {
2517
+ close = mk(P, '}', P.L.b, P.L.b, [])
2518
+ }
2519
+ return mk(P, 'expansion', dStart, close.endIndex, [open, ...inner, close])
2520
+ }
2521
+ // Simple expansion $VAR or $? $$ $@ etc
2522
+ advance(P.L)
2523
+ const dEnd = P.L.b
2524
+ const dollar = mk(P, '$', dStart, dEnd, [])
2525
+ const nc = peek(P.L)
2526
+ // $_ is special_variable_name only when not followed by more ident chars
2527
+ if (nc === '_' && !isIdentChar(peek(P.L, 1))) {
2528
+ const vStart = P.L.b
2529
+ advance(P.L)
2530
+ const vn = mk(P, 'special_variable_name', vStart, P.L.b, [])
2531
+ return mk(P, 'simple_expansion', dStart, P.L.b, [dollar, vn])
2532
+ }
2533
+ if (isIdentStart(nc)) {
2534
+ const vStart = P.L.b
2535
+ while (isIdentChar(peek(P.L))) advance(P.L)
2536
+ const vn = mk(P, 'variable_name', vStart, P.L.b, [])
2537
+ return mk(P, 'simple_expansion', dStart, P.L.b, [dollar, vn])
2538
+ }
2539
+ if (isDigit(nc)) {
2540
+ const vStart = P.L.b
2541
+ advance(P.L)
2542
+ const vn = mk(P, 'variable_name', vStart, P.L.b, [])
2543
+ return mk(P, 'simple_expansion', dStart, P.L.b, [dollar, vn])
2544
+ }
2545
+ if (SPECIAL_VARS.has(nc)) {
2546
+ const vStart = P.L.b
2547
+ advance(P.L)
2548
+ const vn = mk(P, 'special_variable_name', vStart, P.L.b, [])
2549
+ return mk(P, 'simple_expansion', dStart, P.L.b, [dollar, vn])
2550
+ }
2551
+ // Bare $ — just a $ leaf (tree-sitter treats trailing $ as literal)
2552
+ return dollar
2553
+ }
2554
+
2555
+ function parseExpansionBody(P: ParseState): TsNode[] {
2556
+ const out: TsNode[] = []
2557
+ skipBlanks(P.L)
2558
+ // Bizarre cases: ${#!} ${!#} ${!##} ${!# } ${!## } all emit empty (expansion)
2559
+ // — both # and ! become anonymous nodes when only combined with each other
2560
+ // and optional trailing space before }. Note ${!##/} does NOT match (has
2561
+ // content after), so it parses normally as (special_variable_name)(regex).
2562
+ {
2563
+ const c0 = peek(P.L)
2564
+ const c1 = peek(P.L, 1)
2565
+ if (c0 === '#' && c1 === '!' && peek(P.L, 2) === '}') {
2566
+ advance(P.L)
2567
+ advance(P.L)
2568
+ return out
2569
+ }
2570
+ if (c0 === '!' && c1 === '#') {
2571
+ // ${!#} ${!##} with optional trailing space then }
2572
+ let j = 2
2573
+ if (peek(P.L, j) === '#') j++
2574
+ if (peek(P.L, j) === ' ') j++
2575
+ if (peek(P.L, j) === '}') {
2576
+ while (j-- > 0) advance(P.L)
2577
+ return out
2578
+ }
2579
+ }
2580
+ }
2581
+ // Optional # prefix for length
2582
+ if (peek(P.L) === '#') {
2583
+ const s = P.L.b
2584
+ advance(P.L)
2585
+ out.push(mk(P, '#', s, P.L.b, []))
2586
+ }
2587
+ // Optional ! prefix for indirect expansion: ${!varname} ${!prefix*} ${!prefix@}
2588
+ // Only when followed by an identifier — ${!} alone is special var $!
2589
+ // Also = ~ prefixes (zsh-style ${=var} ${~var})
2590
+ const pc = peek(P.L)
2591
+ if (
2592
+ (pc === '!' || pc === '=' || pc === '~') &&
2593
+ (isIdentStart(peek(P.L, 1)) || isDigit(peek(P.L, 1)))
2594
+ ) {
2595
+ const s = P.L.b
2596
+ advance(P.L)
2597
+ out.push(mk(P, pc, s, P.L.b, []))
2598
+ }
2599
+ skipBlanks(P.L)
2600
+ // Variable name
2601
+ if (isIdentStart(peek(P.L))) {
2602
+ const s = P.L.b
2603
+ while (isIdentChar(peek(P.L))) advance(P.L)
2604
+ out.push(mk(P, 'variable_name', s, P.L.b, []))
2605
+ } else if (isDigit(peek(P.L))) {
2606
+ const s = P.L.b
2607
+ while (isDigit(peek(P.L))) advance(P.L)
2608
+ out.push(mk(P, 'variable_name', s, P.L.b, []))
2609
+ } else if (SPECIAL_VARS.has(peek(P.L))) {
2610
+ const s = P.L.b
2611
+ advance(P.L)
2612
+ out.push(mk(P, 'special_variable_name', s, P.L.b, []))
2613
+ }
2614
+ // Optional subscript [idx] — parsed arithmetically
2615
+ if (peek(P.L) === '[') {
2616
+ const varNode = out[out.length - 1]
2617
+ const brOpen = P.L.b
2618
+ advance(P.L)
2619
+ const brOpenNode = mk(P, '[', brOpen, P.L.b, [])
2620
+ const idx = parseSubscriptIndexInline(P)
2621
+ skipBlanks(P.L)
2622
+ const brClose = P.L.b
2623
+ if (peek(P.L) === ']') advance(P.L)
2624
+ const brCloseNode = mk(P, ']', brClose, P.L.b, [])
2625
+ if (varNode) {
2626
+ const kids = idx
2627
+ ? [varNode, brOpenNode, idx, brCloseNode]
2628
+ : [varNode, brOpenNode, brCloseNode]
2629
+ out[out.length - 1] = mk(P, 'subscript', varNode.startIndex, P.L.b, kids)
2630
+ }
2631
+ }
2632
+ skipBlanks(P.L)
2633
+ // Trailing * or @ for indirect expansion (${!prefix*} ${!prefix@}) or
2634
+ // @operator for parameter transformation (${var@U} ${var@Q}) — anonymous
2635
+ const tc = peek(P.L)
2636
+ if ((tc === '*' || tc === '@') && peek(P.L, 1) === '}') {
2637
+ const s = P.L.b
2638
+ advance(P.L)
2639
+ out.push(mk(P, tc, s, P.L.b, []))
2640
+ return out
2641
+ }
2642
+ if (tc === '@' && isIdentStart(peek(P.L, 1))) {
2643
+ // ${var@U} transformation — @ is anonymous, consume op char(s)
2644
+ const s = P.L.b
2645
+ advance(P.L)
2646
+ out.push(mk(P, '@', s, P.L.b, []))
2647
+ while (isIdentChar(peek(P.L))) advance(P.L)
2648
+ return out
2649
+ }
2650
+ // Operator :- := :? :+ - = ? + # ## % %% / // ^ ^^ , ,, etc.
2651
+ const c = peek(P.L)
2652
+ // Bare `:` substring operator ${var:off:len} — offset and length parsed
2653
+ // arithmetically. Must come BEFORE the generic operator handling so `(` after
2654
+ // `:` goes to parenthesized_expression not the array path. `:-` `:=` `:?`
2655
+ // `:+` (no space) remain default-value operators; `: -1` (with space before
2656
+ // -1) is substring with negative offset.
2657
+ if (c === ':') {
2658
+ const c1 = peek(P.L, 1)
2659
+ // `:\n` or `:}` — empty substring expansion, emits nothing (variable_name only)
2660
+ if (c1 === '\n' || c1 === '}') {
2661
+ advance(P.L)
2662
+ while (peek(P.L) === '\n') advance(P.L)
2663
+ return out
2664
+ }
2665
+ if (c1 !== '-' && c1 !== '=' && c1 !== '?' && c1 !== '+') {
2666
+ advance(P.L)
2667
+ skipBlanks(P.L)
2668
+ // Offset — arithmetic. `-N` at top level is a single number node per
2669
+ // tree-sitter; inside parens it's unary_expression(number).
2670
+ const offC = peek(P.L)
2671
+ let off: TsNode | null
2672
+ if (offC === '-' && isDigit(peek(P.L, 1))) {
2673
+ const ns = P.L.b
2674
+ advance(P.L)
2675
+ while (isDigit(peek(P.L))) advance(P.L)
2676
+ off = mk(P, 'number', ns, P.L.b, [])
2677
+ } else {
2678
+ off = parseArithExpr(P, ':}', 'var')
2679
+ }
2680
+ if (off) out.push(off)
2681
+ skipBlanks(P.L)
2682
+ if (peek(P.L) === ':') {
2683
+ advance(P.L)
2684
+ skipBlanks(P.L)
2685
+ const lenC = peek(P.L)
2686
+ let len: TsNode | null
2687
+ if (lenC === '-' && isDigit(peek(P.L, 1))) {
2688
+ const ns = P.L.b
2689
+ advance(P.L)
2690
+ while (isDigit(peek(P.L))) advance(P.L)
2691
+ len = mk(P, 'number', ns, P.L.b, [])
2692
+ } else {
2693
+ len = parseArithExpr(P, '}', 'var')
2694
+ }
2695
+ if (len) out.push(len)
2696
+ }
2697
+ return out
2698
+ }
2699
+ }
2700
+ if (
2701
+ c === ':' ||
2702
+ c === '#' ||
2703
+ c === '%' ||
2704
+ c === '/' ||
2705
+ c === '^' ||
2706
+ c === ',' ||
2707
+ c === '-' ||
2708
+ c === '=' ||
2709
+ c === '?' ||
2710
+ c === '+'
2711
+ ) {
2712
+ const s = P.L.b
2713
+ const c1 = peek(P.L, 1)
2714
+ let op = c
2715
+ if (c === ':' && (c1 === '-' || c1 === '=' || c1 === '?' || c1 === '+')) {
2716
+ advance(P.L)
2717
+ advance(P.L)
2718
+ op = c + c1
2719
+ } else if (
2720
+ (c === '#' || c === '%' || c === '/' || c === '^' || c === ',') &&
2721
+ c1 === c
2722
+ ) {
2723
+ // Doubled operators: ## %% // ^^ ,,
2724
+ advance(P.L)
2725
+ advance(P.L)
2726
+ op = c + c
2727
+ } else {
2728
+ advance(P.L)
2729
+ }
2730
+ out.push(mk(P, op, s, P.L.b, []))
2731
+ // Rest is the default/replacement — parse as word or regex until }
2732
+ // Pattern-matching operators (# ## % %% / // ^ ^^ , ,,) emit regex;
2733
+ // value-substitution operators (:- := :? :+ - = ? + :) emit word.
2734
+ // `/` and `//` split at next `/` into (regex)+(word) for pat/repl.
2735
+ const isPattern =
2736
+ op === '#' ||
2737
+ op === '##' ||
2738
+ op === '%' ||
2739
+ op === '%%' ||
2740
+ op === '/' ||
2741
+ op === '//' ||
2742
+ op === '^' ||
2743
+ op === '^^' ||
2744
+ op === ',' ||
2745
+ op === ',,'
2746
+ if (op === '/' || op === '//') {
2747
+ // Optional /# or /% anchor prefix — anonymous node
2748
+ const ac = peek(P.L)
2749
+ if (ac === '#' || ac === '%') {
2750
+ const aStart = P.L.b
2751
+ advance(P.L)
2752
+ out.push(mk(P, ac, aStart, P.L.b, []))
2753
+ }
2754
+ // Pattern: per grammar _expansion_regex_replacement, pattern is
2755
+ // choice(regex, string, cmd_sub, seq(string, regex)). If it STARTS
2756
+ // with ", emit (string) and any trailing chars become (regex).
2757
+ // `${v//"${old}"/}` → (string(expansion)); `${v//"${c}"\//}` →
2758
+ // (string)(regex).
2759
+ if (peek(P.L) === '"') {
2760
+ out.push(parseDoubleQuoted(P))
2761
+ const tail = parseExpansionRest(P, 'regex', true)
2762
+ if (tail) out.push(tail)
2763
+ } else {
2764
+ const regex = parseExpansionRest(P, 'regex', true)
2765
+ if (regex) out.push(regex)
2766
+ }
2767
+ if (peek(P.L) === '/') {
2768
+ const sepStart = P.L.b
2769
+ advance(P.L)
2770
+ out.push(mk(P, '/', sepStart, P.L.b, []))
2771
+ // Replacement: per grammar, choice includes `seq(cmd_sub, word)`
2772
+ // which emits TWO siblings (not concatenation). Also `(` at start
2773
+ // of replacement is a regular word char, NOT array — unlike `:-`
2774
+ // default-value context. `${v/(/(Gentoo ${x}, }` replacement
2775
+ // `(Gentoo ${x}, ` is (concatenation (word)(expansion)(word)).
2776
+ const repl = parseExpansionRest(P, 'replword', false)
2777
+ if (repl) {
2778
+ // seq(cmd_sub, word) special case → siblings. Detected when
2779
+ // replacement is a concatenation of exactly 2 parts with first
2780
+ // being command_substitution.
2781
+ if (
2782
+ repl.type === 'concatenation' &&
2783
+ repl.children.length === 2 &&
2784
+ repl.children[0]!.type === 'command_substitution'
2785
+ ) {
2786
+ out.push(repl.children[0]!)
2787
+ out.push(repl.children[1]!)
2788
+ } else {
2789
+ out.push(repl)
2790
+ }
2791
+ }
2792
+ }
2793
+ } else if (op === '#' || op === '##' || op === '%' || op === '%%') {
2794
+ // Pattern-removal: per grammar _expansion_regex, pattern is
2795
+ // repeat(choice(regex, string, raw_string, ')')). Each quote/string
2796
+ // is a SIBLING, not absorbed into one regex. `${f%'str'*}` →
2797
+ // (raw_string)(regex); `${f/'str'*}` (slash) stays single regex.
2798
+ for (const p of parseExpansionRegexSegmented(P)) out.push(p)
2799
+ } else {
2800
+ const rest = parseExpansionRest(P, isPattern ? 'regex' : 'word', false)
2801
+ if (rest) out.push(rest)
2802
+ }
2803
+ }
2804
+ return out
2805
+ }
2806
+
2807
+ function parseExpansionRest(
2808
+ P: ParseState,
2809
+ nodeType: string,
2810
+ stopAtSlash: boolean,
2811
+ ): TsNode | null {
2812
+ // Don't skipBlanks — `${var:- }` space IS the word. Stop at } or newline
2813
+ // (`${var:\n}` emits no word). stopAtSlash=true stops at `/` for pat/repl
2814
+ // split in ${var/pat/repl}. nodeType 'replword' is word-mode for the
2815
+ // replacement in `/` `//` — same as 'word' but `(` is NOT array.
2816
+ const start = P.L.b
2817
+ // Value-substitution RHS starting with `(` parses as array: ${var:-(x)} →
2818
+ // (expansion (variable_name) (array (word))). Only for 'word' context (not
2819
+ // pattern-matching operators which emit regex, and not 'replword' where `(`
2820
+ // is a regular char per grammar `_expansion_regex_replacement`).
2821
+ if (nodeType === 'word' && peek(P.L) === '(') {
2822
+ advance(P.L)
2823
+ const open = mk(P, '(', start, P.L.b, [])
2824
+ const elems: TsNode[] = [open]
2825
+ while (P.L.i < P.L.len) {
2826
+ skipBlanks(P.L)
2827
+ const c = peek(P.L)
2828
+ if (c === ')' || c === '}' || c === '\n' || c === '') break
2829
+ const wStart = P.L.b
2830
+ while (P.L.i < P.L.len) {
2831
+ const wc = peek(P.L)
2832
+ if (
2833
+ wc === ')' ||
2834
+ wc === '}' ||
2835
+ wc === ' ' ||
2836
+ wc === '\t' ||
2837
+ wc === '\n' ||
2838
+ wc === ''
2839
+ ) {
2840
+ break
2841
+ }
2842
+ advance(P.L)
2843
+ }
2844
+ if (P.L.b > wStart) elems.push(mk(P, 'word', wStart, P.L.b, []))
2845
+ else break
2846
+ }
2847
+ if (peek(P.L) === ')') {
2848
+ const cStart = P.L.b
2849
+ advance(P.L)
2850
+ elems.push(mk(P, ')', cStart, P.L.b, []))
2851
+ }
2852
+ while (peek(P.L) === '\n') advance(P.L)
2853
+ return mk(P, 'array', start, P.L.b, elems)
2854
+ }
2855
+ // REGEX mode: flat single-span scan. Quotes are opaque (skipped past so
2856
+ // `/` inside them doesn't break stopAtSlash), but NOT emitted as separate
2857
+ // nodes — the entire range becomes one regex node.
2858
+ if (nodeType === 'regex') {
2859
+ let braceDepth = 0
2860
+ while (P.L.i < P.L.len) {
2861
+ const c = peek(P.L)
2862
+ if (c === '\n') break
2863
+ if (braceDepth === 0) {
2864
+ if (c === '}') break
2865
+ if (stopAtSlash && c === '/') break
2866
+ }
2867
+ if (c === '\\' && P.L.i + 1 < P.L.len) {
2868
+ advance(P.L)
2869
+ advance(P.L)
2870
+ continue
2871
+ }
2872
+ if (c === '"' || c === "'") {
2873
+ advance(P.L)
2874
+ while (P.L.i < P.L.len && peek(P.L) !== c) {
2875
+ if (peek(P.L) === '\\' && P.L.i + 1 < P.L.len) advance(P.L)
2876
+ advance(P.L)
2877
+ }
2878
+ if (peek(P.L) === c) advance(P.L)
2879
+ continue
2880
+ }
2881
+ // Skip past nested ${...} $(...) $[...] so their } / don't terminate us
2882
+ if (c === '$') {
2883
+ const c1 = peek(P.L, 1)
2884
+ if (c1 === '{') {
2885
+ let d = 0
2886
+ advance(P.L)
2887
+ advance(P.L)
2888
+ d++
2889
+ while (P.L.i < P.L.len && d > 0) {
2890
+ const nc = peek(P.L)
2891
+ if (nc === '{') d++
2892
+ else if (nc === '}') d--
2893
+ advance(P.L)
2894
+ }
2895
+ continue
2896
+ }
2897
+ if (c1 === '(') {
2898
+ let d = 0
2899
+ advance(P.L)
2900
+ advance(P.L)
2901
+ d++
2902
+ while (P.L.i < P.L.len && d > 0) {
2903
+ const nc = peek(P.L)
2904
+ if (nc === '(') d++
2905
+ else if (nc === ')') d--
2906
+ advance(P.L)
2907
+ }
2908
+ continue
2909
+ }
2910
+ }
2911
+ if (c === '{') braceDepth++
2912
+ else if (c === '}' && braceDepth > 0) braceDepth--
2913
+ advance(P.L)
2914
+ }
2915
+ const end = P.L.b
2916
+ while (peek(P.L) === '\n') advance(P.L)
2917
+ if (end === start) return null
2918
+ return mk(P, 'regex', start, end, [])
2919
+ }
2920
+ // WORD mode: segmenting parser — recognize nested ${...}, $(...), $'...',
2921
+ // "...", '...', $ident, <(...)/>(...); bare chars accumulate into word
2922
+ // segments. Multiple parts → wrapped in concatenation.
2923
+ const parts: TsNode[] = []
2924
+ let segStart = P.L.b
2925
+ let braceDepth = 0
2926
+ const flushSeg = (): void => {
2927
+ if (P.L.b > segStart) {
2928
+ parts.push(mk(P, 'word', segStart, P.L.b, []))
2929
+ }
2930
+ }
2931
+ while (P.L.i < P.L.len) {
2932
+ const c = peek(P.L)
2933
+ if (c === '\n') break
2934
+ if (braceDepth === 0) {
2935
+ if (c === '}') break
2936
+ if (stopAtSlash && c === '/') break
2937
+ }
2938
+ if (c === '\\' && P.L.i + 1 < P.L.len) {
2939
+ advance(P.L)
2940
+ advance(P.L)
2941
+ continue
2942
+ }
2943
+ const c1 = peek(P.L, 1)
2944
+ if (c === '$') {
2945
+ if (c1 === '{' || c1 === '(' || c1 === '[') {
2946
+ flushSeg()
2947
+ const exp = parseDollarLike(P)
2948
+ if (exp) parts.push(exp)
2949
+ segStart = P.L.b
2950
+ continue
2951
+ }
2952
+ if (c1 === "'") {
2953
+ // $'...' ANSI-C string
2954
+ flushSeg()
2955
+ const aStart = P.L.b
2956
+ advance(P.L)
2957
+ advance(P.L)
2958
+ while (P.L.i < P.L.len && peek(P.L) !== "'") {
2959
+ if (peek(P.L) === '\\' && P.L.i + 1 < P.L.len) advance(P.L)
2960
+ advance(P.L)
2961
+ }
2962
+ if (peek(P.L) === "'") advance(P.L)
2963
+ parts.push(mk(P, 'ansi_c_string', aStart, P.L.b, []))
2964
+ segStart = P.L.b
2965
+ continue
2966
+ }
2967
+ if (isIdentStart(c1) || isDigit(c1) || SPECIAL_VARS.has(c1)) {
2968
+ flushSeg()
2969
+ const exp = parseDollarLike(P)
2970
+ if (exp) parts.push(exp)
2971
+ segStart = P.L.b
2972
+ continue
2973
+ }
2974
+ }
2975
+ if (c === '"') {
2976
+ flushSeg()
2977
+ parts.push(parseDoubleQuoted(P))
2978
+ segStart = P.L.b
2979
+ continue
2980
+ }
2981
+ if (c === "'") {
2982
+ flushSeg()
2983
+ const rStart = P.L.b
2984
+ advance(P.L)
2985
+ while (P.L.i < P.L.len && peek(P.L) !== "'") advance(P.L)
2986
+ if (peek(P.L) === "'") advance(P.L)
2987
+ parts.push(mk(P, 'raw_string', rStart, P.L.b, []))
2988
+ segStart = P.L.b
2989
+ continue
2990
+ }
2991
+ if ((c === '<' || c === '>') && c1 === '(') {
2992
+ flushSeg()
2993
+ const ps = parseProcessSub(P)
2994
+ if (ps) parts.push(ps)
2995
+ segStart = P.L.b
2996
+ continue
2997
+ }
2998
+ if (c === '`') {
2999
+ flushSeg()
3000
+ const bt = parseBacktick(P)
3001
+ if (bt) parts.push(bt)
3002
+ segStart = P.L.b
3003
+ continue
3004
+ }
3005
+ // Brace tracking so nested {a,b} brace-expansion chars don't prematurely
3006
+ // terminate (rare, but the `?` in `${cond}? (` should be treated as word).
3007
+ if (c === '{') braceDepth++
3008
+ else if (c === '}' && braceDepth > 0) braceDepth--
3009
+ advance(P.L)
3010
+ }
3011
+ flushSeg()
3012
+ // Consume trailing newlines before } so caller sees }
3013
+ while (peek(P.L) === '\n') advance(P.L)
3014
+ // Tree-sitter skips leading whitespace (extras) in expansion RHS when
3015
+ // there's content after: `${2+ ${2}}` → just (expansion). But `${v:- }`
3016
+ // (space-only RHS) keeps the space as (word). So drop leading whitespace-
3017
+ // only word segment if it's NOT the only part.
3018
+ if (
3019
+ parts.length > 1 &&
3020
+ parts[0]!.type === 'word' &&
3021
+ /^[ \t]+$/.test(parts[0]!.text)
3022
+ ) {
3023
+ parts.shift()
3024
+ }
3025
+ if (parts.length === 0) return null
3026
+ if (parts.length === 1) return parts[0]!
3027
+ // Multiple parts: wrap in concatenation (word mode keeps concat wrapping;
3028
+ // regex mode also concats per tree-sitter for mixed quote+glob patterns).
3029
+ const last = parts[parts.length - 1]!
3030
+ return mk(P, 'concatenation', parts[0]!.startIndex, last.endIndex, parts)
3031
+ }
3032
+
3033
+ // Pattern for # ## % %% operators — per grammar _expansion_regex:
3034
+ // repeat(choice(regex, string, raw_string, ')', /\s+/→regex)). Each quote
3035
+ // becomes a SIBLING node, not absorbed. `${f%'str'*}` → (raw_string)(regex).
3036
+ function parseExpansionRegexSegmented(P: ParseState): TsNode[] {
3037
+ const out: TsNode[] = []
3038
+ let segStart = P.L.b
3039
+ const flushRegex = (): void => {
3040
+ if (P.L.b > segStart) out.push(mk(P, 'regex', segStart, P.L.b, []))
3041
+ }
3042
+ while (P.L.i < P.L.len) {
3043
+ const c = peek(P.L)
3044
+ if (c === '}' || c === '\n') break
3045
+ if (c === '\\' && P.L.i + 1 < P.L.len) {
3046
+ advance(P.L)
3047
+ advance(P.L)
3048
+ continue
3049
+ }
3050
+ if (c === '"') {
3051
+ flushRegex()
3052
+ out.push(parseDoubleQuoted(P))
3053
+ segStart = P.L.b
3054
+ continue
3055
+ }
3056
+ if (c === "'") {
3057
+ flushRegex()
3058
+ const rStart = P.L.b
3059
+ advance(P.L)
3060
+ while (P.L.i < P.L.len && peek(P.L) !== "'") advance(P.L)
3061
+ if (peek(P.L) === "'") advance(P.L)
3062
+ out.push(mk(P, 'raw_string', rStart, P.L.b, []))
3063
+ segStart = P.L.b
3064
+ continue
3065
+ }
3066
+ // Nested ${...} $(...) — opaque scan so their } doesn't terminate us
3067
+ if (c === '$') {
3068
+ const c1 = peek(P.L, 1)
3069
+ if (c1 === '{') {
3070
+ let d = 1
3071
+ advance(P.L)
3072
+ advance(P.L)
3073
+ while (P.L.i < P.L.len && d > 0) {
3074
+ const nc = peek(P.L)
3075
+ if (nc === '{') d++
3076
+ else if (nc === '}') d--
3077
+ advance(P.L)
3078
+ }
3079
+ continue
3080
+ }
3081
+ if (c1 === '(') {
3082
+ let d = 1
3083
+ advance(P.L)
3084
+ advance(P.L)
3085
+ while (P.L.i < P.L.len && d > 0) {
3086
+ const nc = peek(P.L)
3087
+ if (nc === '(') d++
3088
+ else if (nc === ')') d--
3089
+ advance(P.L)
3090
+ }
3091
+ continue
3092
+ }
3093
+ }
3094
+ advance(P.L)
3095
+ }
3096
+ flushRegex()
3097
+ while (peek(P.L) === '\n') advance(P.L)
3098
+ return out
3099
+ }
3100
+
3101
+ function parseBacktick(P: ParseState): TsNode | null {
3102
+ const start = P.L.b
3103
+ advance(P.L)
3104
+ const open = mk(P, '`', start, P.L.b, [])
3105
+ P.inBacktick++
3106
+ // Parse statements inline — stop at closing backtick
3107
+ const body: TsNode[] = []
3108
+ while (true) {
3109
+ skipBlanks(P.L)
3110
+ if (peek(P.L) === '`' || peek(P.L) === '') break
3111
+ const save = saveLex(P.L)
3112
+ const t = nextToken(P.L, 'cmd')
3113
+ if (t.type === 'EOF' || t.type === 'BACKTICK') {
3114
+ restoreLex(P.L, save)
3115
+ break
3116
+ }
3117
+ if (t.type === 'NEWLINE') continue
3118
+ restoreLex(P.L, save)
3119
+ const stmt = parseAndOr(P)
3120
+ if (!stmt) break
3121
+ body.push(stmt)
3122
+ skipBlanks(P.L)
3123
+ if (peek(P.L) === '`') break
3124
+ const save2 = saveLex(P.L)
3125
+ const sep = nextToken(P.L, 'cmd')
3126
+ if (sep.type === 'OP' && (sep.value === ';' || sep.value === '&')) {
3127
+ body.push(leaf(P, sep.value, sep))
3128
+ } else if (sep.type !== 'NEWLINE') {
3129
+ restoreLex(P.L, save2)
3130
+ }
3131
+ }
3132
+ P.inBacktick--
3133
+ let close: TsNode
3134
+ if (peek(P.L) === '`') {
3135
+ const cStart = P.L.b
3136
+ advance(P.L)
3137
+ close = mk(P, '`', cStart, P.L.b, [])
3138
+ } else {
3139
+ close = mk(P, '`', P.L.b, P.L.b, [])
3140
+ }
3141
+ // Empty backticks (whitespace/newline only) are elided entirely by
3142
+ // tree-sitter — used as a line-continuation hack: "foo"`<newline>`"bar"
3143
+ // → (concatenation (string) (string)) with no command_substitution.
3144
+ if (body.length === 0) return null
3145
+ return mk(P, 'command_substitution', start, close.endIndex, [
3146
+ open,
3147
+ ...body,
3148
+ close,
3149
+ ])
3150
+ }
3151
+
3152
+ function parseIf(P: ParseState, ifTok: Token): TsNode {
3153
+ const ifKw = leaf(P, 'if', ifTok)
3154
+ const kids: TsNode[] = [ifKw]
3155
+ const cond = parseStatements(P, null)
3156
+ kids.push(...cond)
3157
+ consumeKeyword(P, 'then', kids)
3158
+ const body = parseStatements(P, null)
3159
+ kids.push(...body)
3160
+ while (true) {
3161
+ const save = saveLex(P.L)
3162
+ const t = nextToken(P.L, 'cmd')
3163
+ if (t.type === 'WORD' && t.value === 'elif') {
3164
+ const eKw = leaf(P, 'elif', t)
3165
+ const eCond = parseStatements(P, null)
3166
+ const eKids: TsNode[] = [eKw, ...eCond]
3167
+ consumeKeyword(P, 'then', eKids)
3168
+ const eBody = parseStatements(P, null)
3169
+ eKids.push(...eBody)
3170
+ const last = eKids[eKids.length - 1]!
3171
+ kids.push(mk(P, 'elif_clause', eKw.startIndex, last.endIndex, eKids))
3172
+ } else if (t.type === 'WORD' && t.value === 'else') {
3173
+ const elKw = leaf(P, 'else', t)
3174
+ const elBody = parseStatements(P, null)
3175
+ const last = elBody.length > 0 ? elBody[elBody.length - 1]! : elKw
3176
+ kids.push(
3177
+ mk(P, 'else_clause', elKw.startIndex, last.endIndex, [elKw, ...elBody]),
3178
+ )
3179
+ } else {
3180
+ restoreLex(P.L, save)
3181
+ break
3182
+ }
3183
+ }
3184
+ consumeKeyword(P, 'fi', kids)
3185
+ const last = kids[kids.length - 1]!
3186
+ return mk(P, 'if_statement', ifKw.startIndex, last.endIndex, kids)
3187
+ }
3188
+
3189
+ function parseWhile(P: ParseState, kwTok: Token): TsNode {
3190
+ const kw = leaf(P, kwTok.value, kwTok)
3191
+ const kids: TsNode[] = [kw]
3192
+ const cond = parseStatements(P, null)
3193
+ kids.push(...cond)
3194
+ const dg = parseDoGroup(P)
3195
+ if (dg) kids.push(dg)
3196
+ const last = kids[kids.length - 1]!
3197
+ return mk(P, 'while_statement', kw.startIndex, last.endIndex, kids)
3198
+ }
3199
+
3200
+ function parseFor(P: ParseState, forTok: Token): TsNode {
3201
+ const forKw = leaf(P, forTok.value, forTok)
3202
+ skipBlanks(P.L)
3203
+ // C-style for (( ; ; )) — only for `for`, not `select`
3204
+ if (forTok.value === 'for' && peek(P.L) === '(' && peek(P.L, 1) === '(') {
3205
+ const oStart = P.L.b
3206
+ advance(P.L)
3207
+ advance(P.L)
3208
+ const open = mk(P, '((', oStart, P.L.b, [])
3209
+ const kids: TsNode[] = [forKw, open]
3210
+ // init; cond; update — all three use 'assign' mode so `c = expr` emits
3211
+ // variable_assignment, while bare idents (c in `c<=5`) → word. Each
3212
+ // clause may be a comma-separated list.
3213
+ for (let k = 0; k < 3; k++) {
3214
+ skipBlanks(P.L)
3215
+ const es = parseArithCommaList(P, k < 2 ? ';' : '))', 'assign')
3216
+ kids.push(...es)
3217
+ if (k < 2) {
3218
+ if (peek(P.L) === ';') {
3219
+ const s = P.L.b
3220
+ advance(P.L)
3221
+ kids.push(mk(P, ';', s, P.L.b, []))
3222
+ }
3223
+ }
3224
+ }
3225
+ skipBlanks(P.L)
3226
+ if (peek(P.L) === ')' && peek(P.L, 1) === ')') {
3227
+ const cStart = P.L.b
3228
+ advance(P.L)
3229
+ advance(P.L)
3230
+ kids.push(mk(P, '))', cStart, P.L.b, []))
3231
+ }
3232
+ // Optional ; or newline
3233
+ const save = saveLex(P.L)
3234
+ const sep = nextToken(P.L, 'cmd')
3235
+ if (sep.type === 'OP' && sep.value === ';') {
3236
+ kids.push(leaf(P, ';', sep))
3237
+ } else if (sep.type !== 'NEWLINE') {
3238
+ restoreLex(P.L, save)
3239
+ }
3240
+ const dg = parseDoGroup(P)
3241
+ if (dg) {
3242
+ kids.push(dg)
3243
+ } else {
3244
+ // C-style for can also use `{ ... }` body instead of `do ... done`
3245
+ skipNewlines(P)
3246
+ skipBlanks(P.L)
3247
+ if (peek(P.L) === '{') {
3248
+ const bOpen = P.L.b
3249
+ advance(P.L)
3250
+ const brace = mk(P, '{', bOpen, P.L.b, [])
3251
+ const body = parseStatements(P, '}')
3252
+ let bClose: TsNode
3253
+ if (peek(P.L) === '}') {
3254
+ const cs = P.L.b
3255
+ advance(P.L)
3256
+ bClose = mk(P, '}', cs, P.L.b, [])
3257
+ } else {
3258
+ bClose = mk(P, '}', P.L.b, P.L.b, [])
3259
+ }
3260
+ kids.push(
3261
+ mk(P, 'compound_statement', brace.startIndex, bClose.endIndex, [
3262
+ brace,
3263
+ ...body,
3264
+ bClose,
3265
+ ]),
3266
+ )
3267
+ }
3268
+ }
3269
+ const last = kids[kids.length - 1]!
3270
+ return mk(P, 'c_style_for_statement', forKw.startIndex, last.endIndex, kids)
3271
+ }
3272
+ // Regular for VAR in words; do ... done
3273
+ const kids: TsNode[] = [forKw]
3274
+ const varTok = nextToken(P.L, 'arg')
3275
+ kids.push(mk(P, 'variable_name', varTok.start, varTok.end, []))
3276
+ skipBlanks(P.L)
3277
+ const save = saveLex(P.L)
3278
+ const inTok = nextToken(P.L, 'arg')
3279
+ if (inTok.type === 'WORD' && inTok.value === 'in') {
3280
+ kids.push(leaf(P, 'in', inTok))
3281
+ while (true) {
3282
+ skipBlanks(P.L)
3283
+ const c = peek(P.L)
3284
+ if (c === ';' || c === '\n' || c === '') break
3285
+ const w = parseWord(P, 'arg')
3286
+ if (!w) break
3287
+ kids.push(w)
3288
+ }
3289
+ } else {
3290
+ restoreLex(P.L, save)
3291
+ }
3292
+ // Separator
3293
+ const save2 = saveLex(P.L)
3294
+ const sep = nextToken(P.L, 'cmd')
3295
+ if (sep.type === 'OP' && sep.value === ';') {
3296
+ kids.push(leaf(P, ';', sep))
3297
+ } else if (sep.type !== 'NEWLINE') {
3298
+ restoreLex(P.L, save2)
3299
+ }
3300
+ const dg = parseDoGroup(P)
3301
+ if (dg) kids.push(dg)
3302
+ const last = kids[kids.length - 1]!
3303
+ return mk(P, 'for_statement', forKw.startIndex, last.endIndex, kids)
3304
+ }
3305
+
3306
+ function parseDoGroup(P: ParseState): TsNode | null {
3307
+ skipNewlines(P)
3308
+ const save = saveLex(P.L)
3309
+ const doTok = nextToken(P.L, 'cmd')
3310
+ if (doTok.type !== 'WORD' || doTok.value !== 'do') {
3311
+ restoreLex(P.L, save)
3312
+ return null
3313
+ }
3314
+ const doKw = leaf(P, 'do', doTok)
3315
+ const body = parseStatements(P, null)
3316
+ const kids: TsNode[] = [doKw, ...body]
3317
+ consumeKeyword(P, 'done', kids)
3318
+ const last = kids[kids.length - 1]!
3319
+ return mk(P, 'do_group', doKw.startIndex, last.endIndex, kids)
3320
+ }
3321
+
3322
+ function parseCase(P: ParseState, caseTok: Token): TsNode {
3323
+ const caseKw = leaf(P, 'case', caseTok)
3324
+ const kids: TsNode[] = [caseKw]
3325
+ skipBlanks(P.L)
3326
+ const word = parseWord(P, 'arg')
3327
+ if (word) kids.push(word)
3328
+ skipBlanks(P.L)
3329
+ consumeKeyword(P, 'in', kids)
3330
+ skipNewlines(P)
3331
+ while (true) {
3332
+ skipBlanks(P.L)
3333
+ skipNewlines(P)
3334
+ const save = saveLex(P.L)
3335
+ const t = nextToken(P.L, 'arg')
3336
+ if (t.type === 'WORD' && t.value === 'esac') {
3337
+ kids.push(leaf(P, 'esac', t))
3338
+ break
3339
+ }
3340
+ if (t.type === 'EOF') break
3341
+ restoreLex(P.L, save)
3342
+ const item = parseCaseItem(P)
3343
+ if (!item) break
3344
+ kids.push(item)
3345
+ }
3346
+ const last = kids[kids.length - 1]!
3347
+ return mk(P, 'case_statement', caseKw.startIndex, last.endIndex, kids)
3348
+ }
3349
+
3350
+ function parseCaseItem(P: ParseState): TsNode | null {
3351
+ skipBlanks(P.L)
3352
+ const start = P.L.b
3353
+ const kids: TsNode[] = []
3354
+ // Optional leading '(' before pattern — bash allows (pattern) syntax
3355
+ if (peek(P.L) === '(') {
3356
+ const s = P.L.b
3357
+ advance(P.L)
3358
+ kids.push(mk(P, '(', s, P.L.b, []))
3359
+ }
3360
+ // Pattern(s)
3361
+ let isFirstAlt = true
3362
+ while (true) {
3363
+ skipBlanks(P.L)
3364
+ const c = peek(P.L)
3365
+ if (c === ')' || c === '') break
3366
+ const pats = parseCasePattern(P)
3367
+ if (pats.length === 0) break
3368
+ // tree-sitter quirk: first alternative with quotes is inlined as flat
3369
+ // siblings; subsequent alternatives are wrapped in (concatenation) with
3370
+ // `word` instead of `extglob_pattern` for bare segments.
3371
+ if (!isFirstAlt && pats.length > 1) {
3372
+ const rewritten = pats.map(p =>
3373
+ p.type === 'extglob_pattern'
3374
+ ? mk(P, 'word', p.startIndex, p.endIndex, [])
3375
+ : p,
3376
+ )
3377
+ const first = rewritten[0]!
3378
+ const last = rewritten[rewritten.length - 1]!
3379
+ kids.push(
3380
+ mk(P, 'concatenation', first.startIndex, last.endIndex, rewritten),
3381
+ )
3382
+ } else {
3383
+ kids.push(...pats)
3384
+ }
3385
+ isFirstAlt = false
3386
+ skipBlanks(P.L)
3387
+ // \<newline> line continuation between alternatives
3388
+ if (peek(P.L) === '\\' && peek(P.L, 1) === '\n') {
3389
+ advance(P.L)
3390
+ advance(P.L)
3391
+ skipBlanks(P.L)
3392
+ }
3393
+ if (peek(P.L) === '|') {
3394
+ const s = P.L.b
3395
+ advance(P.L)
3396
+ kids.push(mk(P, '|', s, P.L.b, []))
3397
+ // \<newline> after | is also a line continuation
3398
+ if (peek(P.L) === '\\' && peek(P.L, 1) === '\n') {
3399
+ advance(P.L)
3400
+ advance(P.L)
3401
+ }
3402
+ } else {
3403
+ break
3404
+ }
3405
+ }
3406
+ if (peek(P.L) === ')') {
3407
+ const s = P.L.b
3408
+ advance(P.L)
3409
+ kids.push(mk(P, ')', s, P.L.b, []))
3410
+ }
3411
+ const body = parseStatements(P, null)
3412
+ kids.push(...body)
3413
+ const save = saveLex(P.L)
3414
+ const term = nextToken(P.L, 'cmd')
3415
+ if (
3416
+ term.type === 'OP' &&
3417
+ (term.value === ';;' || term.value === ';&' || term.value === ';;&')
3418
+ ) {
3419
+ kids.push(leaf(P, term.value, term))
3420
+ } else {
3421
+ restoreLex(P.L, save)
3422
+ }
3423
+ if (kids.length === 0) return null
3424
+ // tree-sitter quirk: case_item with EMPTY body and a single pattern matching
3425
+ // extglob-operator-char-prefix (no actual glob metachars) downgrades to word.
3426
+ // `-o) owner=$2 ;;` (has body) → extglob_pattern; `-g) ;;` (empty) → word.
3427
+ if (body.length === 0) {
3428
+ for (let i = 0; i < kids.length; i++) {
3429
+ const k = kids[i]!
3430
+ if (k.type !== 'extglob_pattern') continue
3431
+ const text = sliceBytes(P, k.startIndex, k.endIndex)
3432
+ if (/^[-+?*@!][a-zA-Z]/.test(text) && !/[*?(]/.test(text)) {
3433
+ kids[i] = mk(P, 'word', k.startIndex, k.endIndex, [])
3434
+ }
3435
+ }
3436
+ }
3437
+ const last = kids[kids.length - 1]!
3438
+ return mk(P, 'case_item', start, last.endIndex, kids)
3439
+ }
3440
+
3441
+ function parseCasePattern(P: ParseState): TsNode[] {
3442
+ skipBlanks(P.L)
3443
+ const save = saveLex(P.L)
3444
+ const start = P.L.b
3445
+ const startI = P.L.i
3446
+ let parenDepth = 0
3447
+ let hasDollar = false
3448
+ let hasBracketOutsideParen = false
3449
+ let hasQuote = false
3450
+ while (P.L.i < P.L.len) {
3451
+ const c = peek(P.L)
3452
+ if (c === '\\' && P.L.i + 1 < P.L.len) {
3453
+ // Escaped char — consume both (handles `bar\ baz` as single pattern)
3454
+ // \<newline> is a line continuation; eat it but stay in pattern.
3455
+ advance(P.L)
3456
+ advance(P.L)
3457
+ continue
3458
+ }
3459
+ if (c === '"' || c === "'") {
3460
+ hasQuote = true
3461
+ // Skip past the quoted segment so its content (spaces, |, etc.) doesn't
3462
+ // break the peek-ahead scan.
3463
+ advance(P.L)
3464
+ while (P.L.i < P.L.len && peek(P.L) !== c) {
3465
+ if (peek(P.L) === '\\' && P.L.i + 1 < P.L.len) advance(P.L)
3466
+ advance(P.L)
3467
+ }
3468
+ if (peek(P.L) === c) advance(P.L)
3469
+ continue
3470
+ }
3471
+ // Paren counting: any ( inside pattern opens a scope; don't break at ) or |
3472
+ // until balanced. Handles extglob *(a|b) and nested shapes *([0-9])([0-9]).
3473
+ if (c === '(') {
3474
+ parenDepth++
3475
+ advance(P.L)
3476
+ continue
3477
+ }
3478
+ if (parenDepth > 0) {
3479
+ if (c === ')') {
3480
+ parenDepth--
3481
+ advance(P.L)
3482
+ continue
3483
+ }
3484
+ if (c === '\n') break
3485
+ advance(P.L)
3486
+ continue
3487
+ }
3488
+ if (c === ')' || c === '|' || c === ' ' || c === '\t' || c === '\n') break
3489
+ if (c === '$') hasDollar = true
3490
+ if (c === '[') hasBracketOutsideParen = true
3491
+ advance(P.L)
3492
+ }
3493
+ if (P.L.b === start) return []
3494
+ const text = P.src.slice(startI, P.L.i)
3495
+ const hasExtglobParen = /[*?+@!]\(/.test(text)
3496
+ // Quoted segments in pattern: tree-sitter splits at quote boundaries into
3497
+ // multiple sibling nodes. `*"foo"*` → (extglob_pattern)(string)(extglob_pattern).
3498
+ // Re-scan with a segmenting pass.
3499
+ if (hasQuote && !hasExtglobParen) {
3500
+ restoreLex(P.L, save)
3501
+ return parseCasePatternSegmented(P)
3502
+ }
3503
+ // tree-sitter splits patterns with [ or $ into concatenation via word parsing
3504
+ // UNLESS pattern has extglob parens (those override and emit extglob_pattern).
3505
+ // `*.[1357]` → concat(word word number word); `${PN}.pot` → concat(expansion word);
3506
+ // but `*([0-9])` → extglob_pattern (has extglob paren).
3507
+ if (!hasExtglobParen && (hasDollar || hasBracketOutsideParen)) {
3508
+ restoreLex(P.L, save)
3509
+ const w = parseWord(P, 'arg')
3510
+ return w ? [w] : []
3511
+ }
3512
+ // Patterns starting with extglob operator chars (+ - ? * @ !) followed by
3513
+ // identifier chars are extglob_pattern per tree-sitter, even without parens
3514
+ // or glob metachars. `-o)` → extglob_pattern; plain `foo)` → word.
3515
+ const type =
3516
+ hasExtglobParen || /[*?]/.test(text) || /^[-+?*@!][a-zA-Z]/.test(text)
3517
+ ? 'extglob_pattern'
3518
+ : 'word'
3519
+ return [mk(P, type, start, P.L.b, [])]
3520
+ }
3521
+
3522
+ // Segmented scan for case patterns containing quotes: `*"foo"*` →
3523
+ // [extglob_pattern, string, extglob_pattern]. Bare segments → extglob_pattern
3524
+ // if they have */?, else word. Stops at ) | space tab newline outside quotes.
3525
+ function parseCasePatternSegmented(P: ParseState): TsNode[] {
3526
+ const parts: TsNode[] = []
3527
+ let segStart = P.L.b
3528
+ let segStartI = P.L.i
3529
+ const flushSeg = (): void => {
3530
+ if (P.L.i > segStartI) {
3531
+ const t = P.src.slice(segStartI, P.L.i)
3532
+ const type = /[*?]/.test(t) ? 'extglob_pattern' : 'word'
3533
+ parts.push(mk(P, type, segStart, P.L.b, []))
3534
+ }
3535
+ }
3536
+ while (P.L.i < P.L.len) {
3537
+ const c = peek(P.L)
3538
+ if (c === '\\' && P.L.i + 1 < P.L.len) {
3539
+ advance(P.L)
3540
+ advance(P.L)
3541
+ continue
3542
+ }
3543
+ if (c === '"') {
3544
+ flushSeg()
3545
+ parts.push(parseDoubleQuoted(P))
3546
+ segStart = P.L.b
3547
+ segStartI = P.L.i
3548
+ continue
3549
+ }
3550
+ if (c === "'") {
3551
+ flushSeg()
3552
+ const tok = nextToken(P.L, 'arg')
3553
+ parts.push(leaf(P, 'raw_string', tok))
3554
+ segStart = P.L.b
3555
+ segStartI = P.L.i
3556
+ continue
3557
+ }
3558
+ if (c === ')' || c === '|' || c === ' ' || c === '\t' || c === '\n') break
3559
+ advance(P.L)
3560
+ }
3561
+ flushSeg()
3562
+ return parts
3563
+ }
3564
+
3565
+ function parseFunction(P: ParseState, fnTok: Token): TsNode {
3566
+ const fnKw = leaf(P, 'function', fnTok)
3567
+ skipBlanks(P.L)
3568
+ const nameTok = nextToken(P.L, 'arg')
3569
+ const name = mk(P, 'word', nameTok.start, nameTok.end, [])
3570
+ const kids: TsNode[] = [fnKw, name]
3571
+ skipBlanks(P.L)
3572
+ if (peek(P.L) === '(' && peek(P.L, 1) === ')') {
3573
+ const o = nextToken(P.L, 'cmd')
3574
+ const c = nextToken(P.L, 'cmd')
3575
+ kids.push(leaf(P, '(', o))
3576
+ kids.push(leaf(P, ')', c))
3577
+ }
3578
+ skipBlanks(P.L)
3579
+ skipNewlines(P)
3580
+ const body = parseCommand(P)
3581
+ if (body) {
3582
+ // Hoist redirects from redirected_statement(compound_statement, ...) to
3583
+ // function_definition level per tree-sitter grammar
3584
+ if (
3585
+ body.type === 'redirected_statement' &&
3586
+ body.children.length >= 2 &&
3587
+ body.children[0]!.type === 'compound_statement'
3588
+ ) {
3589
+ kids.push(...body.children)
3590
+ } else {
3591
+ kids.push(body)
3592
+ }
3593
+ }
3594
+ const last = kids[kids.length - 1]!
3595
+ return mk(P, 'function_definition', fnKw.startIndex, last.endIndex, kids)
3596
+ }
3597
+
3598
+ function parseDeclaration(P: ParseState, kwTok: Token): TsNode {
3599
+ const kw = leaf(P, kwTok.value, kwTok)
3600
+ const kids: TsNode[] = [kw]
3601
+ while (true) {
3602
+ skipBlanks(P.L)
3603
+ const c = peek(P.L)
3604
+ if (
3605
+ c === '' ||
3606
+ c === '\n' ||
3607
+ c === ';' ||
3608
+ c === '&' ||
3609
+ c === '|' ||
3610
+ c === ')' ||
3611
+ c === '<' ||
3612
+ c === '>'
3613
+ ) {
3614
+ break
3615
+ }
3616
+ const a = tryParseAssignment(P)
3617
+ if (a) {
3618
+ kids.push(a)
3619
+ continue
3620
+ }
3621
+ // Quoted string or concatenation: `export "FOO=bar"`, `export 'X'`
3622
+ if (c === '"' || c === "'" || c === '$') {
3623
+ const w = parseWord(P, 'arg')
3624
+ if (w) {
3625
+ kids.push(w)
3626
+ continue
3627
+ }
3628
+ break
3629
+ }
3630
+ // Flag like -a or bare variable name
3631
+ const save = saveLex(P.L)
3632
+ const tok = nextToken(P.L, 'arg')
3633
+ if (tok.type === 'WORD' || tok.type === 'NUMBER') {
3634
+ if (tok.value.startsWith('-')) {
3635
+ kids.push(leaf(P, 'word', tok))
3636
+ } else if (isIdentStart(tok.value[0] ?? '')) {
3637
+ kids.push(mk(P, 'variable_name', tok.start, tok.end, []))
3638
+ } else {
3639
+ kids.push(leaf(P, 'word', tok))
3640
+ }
3641
+ } else {
3642
+ restoreLex(P.L, save)
3643
+ break
3644
+ }
3645
+ }
3646
+ const last = kids[kids.length - 1]!
3647
+ return mk(P, 'declaration_command', kw.startIndex, last.endIndex, kids)
3648
+ }
3649
+
3650
+ function parseUnset(P: ParseState, kwTok: Token): TsNode {
3651
+ const kw = leaf(P, 'unset', kwTok)
3652
+ const kids: TsNode[] = [kw]
3653
+ while (true) {
3654
+ skipBlanks(P.L)
3655
+ const c = peek(P.L)
3656
+ if (
3657
+ c === '' ||
3658
+ c === '\n' ||
3659
+ c === ';' ||
3660
+ c === '&' ||
3661
+ c === '|' ||
3662
+ c === ')' ||
3663
+ c === '<' ||
3664
+ c === '>'
3665
+ ) {
3666
+ break
3667
+ }
3668
+ // SECURITY: use parseWord (not raw nextToken) so quoted strings like
3669
+ // `unset 'a[$(id)]'` emit a raw_string child that ast.ts can reject.
3670
+ // Previously `break` silently dropped non-WORD args — hiding the
3671
+ // arithmetic-subscript code-exec vector from the security walker.
3672
+ const arg = parseWord(P, 'arg')
3673
+ if (!arg) break
3674
+ if (arg.type === 'word') {
3675
+ if (arg.text.startsWith('-')) {
3676
+ kids.push(arg)
3677
+ } else {
3678
+ kids.push(mk(P, 'variable_name', arg.startIndex, arg.endIndex, []))
3679
+ }
3680
+ } else {
3681
+ kids.push(arg)
3682
+ }
3683
+ }
3684
+ const last = kids[kids.length - 1]!
3685
+ return mk(P, 'unset_command', kw.startIndex, last.endIndex, kids)
3686
+ }
3687
+
3688
+ function consumeKeyword(P: ParseState, name: string, kids: TsNode[]): void {
3689
+ skipNewlines(P)
3690
+ const save = saveLex(P.L)
3691
+ const t = nextToken(P.L, 'cmd')
3692
+ if (t.type === 'WORD' && t.value === name) {
3693
+ kids.push(leaf(P, name, t))
3694
+ } else {
3695
+ restoreLex(P.L, save)
3696
+ }
3697
+ }
3698
+
3699
+ // ───────────────────── Test & Arithmetic Expressions ─────────────────────
3700
+
3701
+ function parseTestExpr(P: ParseState, closer: string): TsNode | null {
3702
+ return parseTestOr(P, closer)
3703
+ }
3704
+
3705
+ function parseTestOr(P: ParseState, closer: string): TsNode | null {
3706
+ let left = parseTestAnd(P, closer)
3707
+ if (!left) return null
3708
+ while (true) {
3709
+ skipBlanks(P.L)
3710
+ const save = saveLex(P.L)
3711
+ if (peek(P.L) === '|' && peek(P.L, 1) === '|') {
3712
+ const s = P.L.b
3713
+ advance(P.L)
3714
+ advance(P.L)
3715
+ const op = mk(P, '||', s, P.L.b, [])
3716
+ const right = parseTestAnd(P, closer)
3717
+ if (!right) {
3718
+ restoreLex(P.L, save)
3719
+ break
3720
+ }
3721
+ left = mk(P, 'binary_expression', left.startIndex, right.endIndex, [
3722
+ left,
3723
+ op,
3724
+ right,
3725
+ ])
3726
+ } else {
3727
+ break
3728
+ }
3729
+ }
3730
+ return left
3731
+ }
3732
+
3733
+ function parseTestAnd(P: ParseState, closer: string): TsNode | null {
3734
+ let left = parseTestUnary(P, closer)
3735
+ if (!left) return null
3736
+ while (true) {
3737
+ skipBlanks(P.L)
3738
+ if (peek(P.L) === '&' && peek(P.L, 1) === '&') {
3739
+ const s = P.L.b
3740
+ advance(P.L)
3741
+ advance(P.L)
3742
+ const op = mk(P, '&&', s, P.L.b, [])
3743
+ const right = parseTestUnary(P, closer)
3744
+ if (!right) break
3745
+ left = mk(P, 'binary_expression', left.startIndex, right.endIndex, [
3746
+ left,
3747
+ op,
3748
+ right,
3749
+ ])
3750
+ } else {
3751
+ break
3752
+ }
3753
+ }
3754
+ return left
3755
+ }
3756
+
3757
+ function parseTestUnary(P: ParseState, closer: string): TsNode | null {
3758
+ skipBlanks(P.L)
3759
+ const c = peek(P.L)
3760
+ if (c === '(') {
3761
+ const s = P.L.b
3762
+ advance(P.L)
3763
+ const open = mk(P, '(', s, P.L.b, [])
3764
+ const inner = parseTestOr(P, closer)
3765
+ skipBlanks(P.L)
3766
+ let close: TsNode
3767
+ if (peek(P.L) === ')') {
3768
+ const cs = P.L.b
3769
+ advance(P.L)
3770
+ close = mk(P, ')', cs, P.L.b, [])
3771
+ } else {
3772
+ close = mk(P, ')', P.L.b, P.L.b, [])
3773
+ }
3774
+ const kids = inner ? [open, inner, close] : [open, close]
3775
+ return mk(
3776
+ P,
3777
+ 'parenthesized_expression',
3778
+ open.startIndex,
3779
+ close.endIndex,
3780
+ kids,
3781
+ )
3782
+ }
3783
+ return parseTestBinary(P, closer)
3784
+ }
3785
+
3786
+ /**
3787
+ * Parse `!`-negated or test-operator (`-f`) or parenthesized primary — but NOT
3788
+ * a binary comparison. Used as LHS of binary_expression so `! x =~ y` binds
3789
+ * `!` to `x` only, not the whole `x =~ y`.
3790
+ */
3791
+ function parseTestNegatablePrimary(
3792
+ P: ParseState,
3793
+ closer: string,
3794
+ ): TsNode | null {
3795
+ skipBlanks(P.L)
3796
+ const c = peek(P.L)
3797
+ if (c === '!') {
3798
+ const s = P.L.b
3799
+ advance(P.L)
3800
+ const bang = mk(P, '!', s, P.L.b, [])
3801
+ const inner = parseTestNegatablePrimary(P, closer)
3802
+ if (!inner) return bang
3803
+ return mk(P, 'unary_expression', bang.startIndex, inner.endIndex, [
3804
+ bang,
3805
+ inner,
3806
+ ])
3807
+ }
3808
+ if (c === '-' && isIdentStart(peek(P.L, 1))) {
3809
+ const s = P.L.b
3810
+ advance(P.L)
3811
+ while (isIdentChar(peek(P.L))) advance(P.L)
3812
+ const op = mk(P, 'test_operator', s, P.L.b, [])
3813
+ skipBlanks(P.L)
3814
+ const arg = parseTestPrimary(P, closer)
3815
+ if (!arg) return op
3816
+ return mk(P, 'unary_expression', op.startIndex, arg.endIndex, [op, arg])
3817
+ }
3818
+ return parseTestPrimary(P, closer)
3819
+ }
3820
+
3821
+ function parseTestBinary(P: ParseState, closer: string): TsNode | null {
3822
+ skipBlanks(P.L)
3823
+ // `!` in test context binds tighter than =~/==.
3824
+ // `[[ ! "x" =~ y ]]` → (binary_expression (unary_expression (string)) (regex))
3825
+ // `[[ ! -f x ]]` → (unary_expression ! (unary_expression (test_operator) (word)))
3826
+ const left = parseTestNegatablePrimary(P, closer)
3827
+ if (!left) return null
3828
+ skipBlanks(P.L)
3829
+ // Binary comparison: == != =~ -eq -lt etc.
3830
+ const c = peek(P.L)
3831
+ const c1 = peek(P.L, 1)
3832
+ let op: TsNode | null = null
3833
+ const os = P.L.b
3834
+ if (c === '=' && c1 === '=') {
3835
+ advance(P.L)
3836
+ advance(P.L)
3837
+ op = mk(P, '==', os, P.L.b, [])
3838
+ } else if (c === '!' && c1 === '=') {
3839
+ advance(P.L)
3840
+ advance(P.L)
3841
+ op = mk(P, '!=', os, P.L.b, [])
3842
+ } else if (c === '=' && c1 === '~') {
3843
+ advance(P.L)
3844
+ advance(P.L)
3845
+ op = mk(P, '=~', os, P.L.b, [])
3846
+ } else if (c === '=' && c1 !== '=') {
3847
+ advance(P.L)
3848
+ op = mk(P, '=', os, P.L.b, [])
3849
+ } else if (c === '<' && c1 !== '<') {
3850
+ advance(P.L)
3851
+ op = mk(P, '<', os, P.L.b, [])
3852
+ } else if (c === '>' && c1 !== '>') {
3853
+ advance(P.L)
3854
+ op = mk(P, '>', os, P.L.b, [])
3855
+ } else if (c === '-' && isIdentStart(c1)) {
3856
+ advance(P.L)
3857
+ while (isIdentChar(peek(P.L))) advance(P.L)
3858
+ op = mk(P, 'test_operator', os, P.L.b, [])
3859
+ }
3860
+ if (!op) return left
3861
+ skipBlanks(P.L)
3862
+ // In [[ ]], RHS of ==/!=/=/=~ gets special pattern parsing: paren counting
3863
+ // so @(a|b|c) doesn't break on |, and segments become extglob_pattern/regex.
3864
+ if (closer === ']]') {
3865
+ const opText = op.type
3866
+ if (opText === '=~') {
3867
+ skipBlanks(P.L)
3868
+ // If the ENTIRE RHS is a quoted string, emit string/raw_string not
3869
+ // regex: `[[ "$x" =~ "$y" ]]` → (binary_expression (string) (string)).
3870
+ // If there's content after the quote (`' boop '(.*)$`), the whole RHS
3871
+ // stays a single (regex). Peek past the quote to check.
3872
+ const rc = peek(P.L)
3873
+ let rhs: TsNode | null = null
3874
+ if (rc === '"' || rc === "'") {
3875
+ const save = saveLex(P.L)
3876
+ const quoted =
3877
+ rc === '"'
3878
+ ? parseDoubleQuoted(P)
3879
+ : leaf(P, 'raw_string', nextToken(P.L, 'arg'))
3880
+ // Check if RHS ends here: only whitespace then ]] or &&/|| or newline
3881
+ let j = P.L.i
3882
+ while (j < P.L.len && (P.src[j] === ' ' || P.src[j] === '\t')) j++
3883
+ const nc = P.src[j] ?? ''
3884
+ const nc1 = P.src[j + 1] ?? ''
3885
+ if (
3886
+ (nc === ']' && nc1 === ']') ||
3887
+ (nc === '&' && nc1 === '&') ||
3888
+ (nc === '|' && nc1 === '|') ||
3889
+ nc === '\n' ||
3890
+ nc === ''
3891
+ ) {
3892
+ rhs = quoted
3893
+ } else {
3894
+ restoreLex(P.L, save)
3895
+ }
3896
+ }
3897
+ if (!rhs) rhs = parseTestRegexRhs(P)
3898
+ if (!rhs) return left
3899
+ return mk(P, 'binary_expression', left.startIndex, rhs.endIndex, [
3900
+ left,
3901
+ op,
3902
+ rhs,
3903
+ ])
3904
+ }
3905
+ // Single `=` emits (regex) per tree-sitter; `==` and `!=` emit extglob_pattern
3906
+ if (opText === '=') {
3907
+ const rhs = parseTestRegexRhs(P)
3908
+ if (!rhs) return left
3909
+ return mk(P, 'binary_expression', left.startIndex, rhs.endIndex, [
3910
+ left,
3911
+ op,
3912
+ rhs,
3913
+ ])
3914
+ }
3915
+ if (opText === '==' || opText === '!=') {
3916
+ const parts = parseTestExtglobRhs(P)
3917
+ if (parts.length === 0) return left
3918
+ const last = parts[parts.length - 1]!
3919
+ return mk(P, 'binary_expression', left.startIndex, last.endIndex, [
3920
+ left,
3921
+ op,
3922
+ ...parts,
3923
+ ])
3924
+ }
3925
+ }
3926
+ const right = parseTestPrimary(P, closer)
3927
+ if (!right) return left
3928
+ return mk(P, 'binary_expression', left.startIndex, right.endIndex, [
3929
+ left,
3930
+ op,
3931
+ right,
3932
+ ])
3933
+ }
3934
+
3935
+ // RHS of =~ in [[ ]] — scan as single (regex) node with paren/bracket counting
3936
+ // so | ( ) inside the regex don't break parsing. Stop at ]] or ws+&&/||.
3937
+ function parseTestRegexRhs(P: ParseState): TsNode | null {
3938
+ skipBlanks(P.L)
3939
+ const start = P.L.b
3940
+ let parenDepth = 0
3941
+ let bracketDepth = 0
3942
+ while (P.L.i < P.L.len) {
3943
+ const c = peek(P.L)
3944
+ if (c === '\\' && P.L.i + 1 < P.L.len) {
3945
+ advance(P.L)
3946
+ advance(P.L)
3947
+ continue
3948
+ }
3949
+ if (c === '\n') break
3950
+ if (parenDepth === 0 && bracketDepth === 0) {
3951
+ if (c === ']' && peek(P.L, 1) === ']') break
3952
+ if (c === ' ' || c === '\t') {
3953
+ // Peek past blanks for ]] or &&/||
3954
+ let j = P.L.i
3955
+ while (j < P.L.len && (P.L.src[j] === ' ' || P.L.src[j] === '\t')) j++
3956
+ const nc = P.L.src[j] ?? ''
3957
+ const nc1 = P.L.src[j + 1] ?? ''
3958
+ if (
3959
+ (nc === ']' && nc1 === ']') ||
3960
+ (nc === '&' && nc1 === '&') ||
3961
+ (nc === '|' && nc1 === '|')
3962
+ ) {
3963
+ break
3964
+ }
3965
+ advance(P.L)
3966
+ continue
3967
+ }
3968
+ }
3969
+ if (c === '(') parenDepth++
3970
+ else if (c === ')' && parenDepth > 0) parenDepth--
3971
+ else if (c === '[') bracketDepth++
3972
+ else if (c === ']' && bracketDepth > 0) bracketDepth--
3973
+ advance(P.L)
3974
+ }
3975
+ if (P.L.b === start) return null
3976
+ return mk(P, 'regex', start, P.L.b, [])
3977
+ }
3978
+
3979
+ // RHS of ==/!=/= in [[ ]] — returns array of parts. Bare text → extglob_pattern
3980
+ // (with paren counting for @(a|b)); $(...)/${}/quoted → proper node types.
3981
+ // Multiple parts become flat children of binary_expression per tree-sitter.
3982
+ function parseTestExtglobRhs(P: ParseState): TsNode[] {
3983
+ skipBlanks(P.L)
3984
+ const parts: TsNode[] = []
3985
+ let segStart = P.L.b
3986
+ let segStartI = P.L.i
3987
+ let parenDepth = 0
3988
+ const flushSeg = () => {
3989
+ if (P.L.i > segStartI) {
3990
+ const text = P.src.slice(segStartI, P.L.i)
3991
+ // Pure number stays number; everything else is extglob_pattern
3992
+ const type = /^\d+$/.test(text) ? 'number' : 'extglob_pattern'
3993
+ parts.push(mk(P, type, segStart, P.L.b, []))
3994
+ }
3995
+ }
3996
+ while (P.L.i < P.L.len) {
3997
+ const c = peek(P.L)
3998
+ if (c === '\\' && P.L.i + 1 < P.L.len) {
3999
+ advance(P.L)
4000
+ advance(P.L)
4001
+ continue
4002
+ }
4003
+ if (c === '\n') break
4004
+ if (parenDepth === 0) {
4005
+ if (c === ']' && peek(P.L, 1) === ']') break
4006
+ if (c === ' ' || c === '\t') {
4007
+ let j = P.L.i
4008
+ while (j < P.L.len && (P.L.src[j] === ' ' || P.L.src[j] === '\t')) j++
4009
+ const nc = P.L.src[j] ?? ''
4010
+ const nc1 = P.L.src[j + 1] ?? ''
4011
+ if (
4012
+ (nc === ']' && nc1 === ']') ||
4013
+ (nc === '&' && nc1 === '&') ||
4014
+ (nc === '|' && nc1 === '|')
4015
+ ) {
4016
+ break
4017
+ }
4018
+ advance(P.L)
4019
+ continue
4020
+ }
4021
+ }
4022
+ // $ " ' must be parsed even inside @( ) extglob parens — parseDollarLike
4023
+ // consumes matching ) so parenDepth stays consistent.
4024
+ if (c === '$') {
4025
+ const c1 = peek(P.L, 1)
4026
+ if (
4027
+ c1 === '(' ||
4028
+ c1 === '{' ||
4029
+ isIdentStart(c1) ||
4030
+ SPECIAL_VARS.has(c1)
4031
+ ) {
4032
+ flushSeg()
4033
+ const exp = parseDollarLike(P)
4034
+ if (exp) parts.push(exp)
4035
+ segStart = P.L.b
4036
+ segStartI = P.L.i
4037
+ continue
4038
+ }
4039
+ }
4040
+ if (c === '"') {
4041
+ flushSeg()
4042
+ parts.push(parseDoubleQuoted(P))
4043
+ segStart = P.L.b
4044
+ segStartI = P.L.i
4045
+ continue
4046
+ }
4047
+ if (c === "'") {
4048
+ flushSeg()
4049
+ const tok = nextToken(P.L, 'arg')
4050
+ parts.push(leaf(P, 'raw_string', tok))
4051
+ segStart = P.L.b
4052
+ segStartI = P.L.i
4053
+ continue
4054
+ }
4055
+ if (c === '(') parenDepth++
4056
+ else if (c === ')' && parenDepth > 0) parenDepth--
4057
+ advance(P.L)
4058
+ }
4059
+ flushSeg()
4060
+ return parts
4061
+ }
4062
+
4063
+ function parseTestPrimary(P: ParseState, closer: string): TsNode | null {
4064
+ skipBlanks(P.L)
4065
+ // Stop at closer
4066
+ if (closer === ']' && peek(P.L) === ']') return null
4067
+ if (closer === ']]' && peek(P.L) === ']' && peek(P.L, 1) === ']') return null
4068
+ return parseWord(P, 'arg')
4069
+ }
4070
+
4071
+ /**
4072
+ * Arithmetic context modes:
4073
+ * - 'var': bare identifiers → variable_name (default, used in $((..)), ((..)))
4074
+ * - 'word': bare identifiers → word (c-style for head condition/update clauses)
4075
+ * - 'assign': identifiers with = → variable_assignment (c-style for init clause)
4076
+ */
4077
+ type ArithMode = 'var' | 'word' | 'assign'
4078
+
4079
+ /** Operator precedence table (higher = tighter binding). */
4080
+ const ARITH_PREC: Record<string, number> = {
4081
+ '=': 2,
4082
+ '+=': 2,
4083
+ '-=': 2,
4084
+ '*=': 2,
4085
+ '/=': 2,
4086
+ '%=': 2,
4087
+ '<<=': 2,
4088
+ '>>=': 2,
4089
+ '&=': 2,
4090
+ '^=': 2,
4091
+ '|=': 2,
4092
+ '||': 4,
4093
+ '&&': 5,
4094
+ '|': 6,
4095
+ '^': 7,
4096
+ '&': 8,
4097
+ '==': 9,
4098
+ '!=': 9,
4099
+ '<': 10,
4100
+ '>': 10,
4101
+ '<=': 10,
4102
+ '>=': 10,
4103
+ '<<': 11,
4104
+ '>>': 11,
4105
+ '+': 12,
4106
+ '-': 12,
4107
+ '*': 13,
4108
+ '/': 13,
4109
+ '%': 13,
4110
+ '**': 14,
4111
+ }
4112
+
4113
+ /** Right-associative operators (assignment and exponent). */
4114
+ const ARITH_RIGHT_ASSOC = new Set([
4115
+ '=',
4116
+ '+=',
4117
+ '-=',
4118
+ '*=',
4119
+ '/=',
4120
+ '%=',
4121
+ '<<=',
4122
+ '>>=',
4123
+ '&=',
4124
+ '^=',
4125
+ '|=',
4126
+ '**',
4127
+ ])
4128
+
4129
+ function parseArithExpr(
4130
+ P: ParseState,
4131
+ stop: string,
4132
+ mode: ArithMode = 'var',
4133
+ ): TsNode | null {
4134
+ return parseArithTernary(P, stop, mode)
4135
+ }
4136
+
4137
+ /** Top-level: comma-separated list. arithmetic_expansion emits multiple children. */
4138
+ function parseArithCommaList(
4139
+ P: ParseState,
4140
+ stop: string,
4141
+ mode: ArithMode = 'var',
4142
+ ): TsNode[] {
4143
+ const out: TsNode[] = []
4144
+ while (true) {
4145
+ const e = parseArithTernary(P, stop, mode)
4146
+ if (e) out.push(e)
4147
+ skipBlanks(P.L)
4148
+ if (peek(P.L) === ',' && !isArithStop(P, stop)) {
4149
+ advance(P.L)
4150
+ continue
4151
+ }
4152
+ break
4153
+ }
4154
+ return out
4155
+ }
4156
+
4157
+ function parseArithTernary(
4158
+ P: ParseState,
4159
+ stop: string,
4160
+ mode: ArithMode,
4161
+ ): TsNode | null {
4162
+ const cond = parseArithBinary(P, stop, 0, mode)
4163
+ if (!cond) return null
4164
+ skipBlanks(P.L)
4165
+ if (peek(P.L) === '?') {
4166
+ const qs = P.L.b
4167
+ advance(P.L)
4168
+ const q = mk(P, '?', qs, P.L.b, [])
4169
+ const t = parseArithBinary(P, ':', 0, mode)
4170
+ skipBlanks(P.L)
4171
+ let colon: TsNode
4172
+ if (peek(P.L) === ':') {
4173
+ const cs = P.L.b
4174
+ advance(P.L)
4175
+ colon = mk(P, ':', cs, P.L.b, [])
4176
+ } else {
4177
+ colon = mk(P, ':', P.L.b, P.L.b, [])
4178
+ }
4179
+ const f = parseArithTernary(P, stop, mode)
4180
+ const last = f ?? colon
4181
+ const kids: TsNode[] = [cond, q]
4182
+ if (t) kids.push(t)
4183
+ kids.push(colon)
4184
+ if (f) kids.push(f)
4185
+ return mk(P, 'ternary_expression', cond.startIndex, last.endIndex, kids)
4186
+ }
4187
+ return cond
4188
+ }
4189
+
4190
+ /** Scan next arithmetic binary operator; returns [text, length] or null. */
4191
+ function scanArithOp(P: ParseState): [string, number] | null {
4192
+ const c = peek(P.L)
4193
+ const c1 = peek(P.L, 1)
4194
+ const c2 = peek(P.L, 2)
4195
+ // 3-char: <<= >>=
4196
+ if (c === '<' && c1 === '<' && c2 === '=') return ['<<=', 3]
4197
+ if (c === '>' && c1 === '>' && c2 === '=') return ['>>=', 3]
4198
+ // 2-char
4199
+ if (c === '*' && c1 === '*') return ['**', 2]
4200
+ if (c === '<' && c1 === '<') return ['<<', 2]
4201
+ if (c === '>' && c1 === '>') return ['>>', 2]
4202
+ if (c === '=' && c1 === '=') return ['==', 2]
4203
+ if (c === '!' && c1 === '=') return ['!=', 2]
4204
+ if (c === '<' && c1 === '=') return ['<=', 2]
4205
+ if (c === '>' && c1 === '=') return ['>=', 2]
4206
+ if (c === '&' && c1 === '&') return ['&&', 2]
4207
+ if (c === '|' && c1 === '|') return ['||', 2]
4208
+ if (c === '+' && c1 === '=') return ['+=', 2]
4209
+ if (c === '-' && c1 === '=') return ['-=', 2]
4210
+ if (c === '*' && c1 === '=') return ['*=', 2]
4211
+ if (c === '/' && c1 === '=') return ['/=', 2]
4212
+ if (c === '%' && c1 === '=') return ['%=', 2]
4213
+ if (c === '&' && c1 === '=') return ['&=', 2]
4214
+ if (c === '^' && c1 === '=') return ['^=', 2]
4215
+ if (c === '|' && c1 === '=') return ['|=', 2]
4216
+ // 1-char — but NOT ++ -- (those are pre/postfix)
4217
+ if (c === '+' && c1 !== '+') return ['+', 1]
4218
+ if (c === '-' && c1 !== '-') return ['-', 1]
4219
+ if (c === '*') return ['*', 1]
4220
+ if (c === '/') return ['/', 1]
4221
+ if (c === '%') return ['%', 1]
4222
+ if (c === '<') return ['<', 1]
4223
+ if (c === '>') return ['>', 1]
4224
+ if (c === '&') return ['&', 1]
4225
+ if (c === '|') return ['|', 1]
4226
+ if (c === '^') return ['^', 1]
4227
+ if (c === '=') return ['=', 1]
4228
+ return null
4229
+ }
4230
+
4231
+ /** Precedence-climbing binary expression parser. */
4232
+ function parseArithBinary(
4233
+ P: ParseState,
4234
+ stop: string,
4235
+ minPrec: number,
4236
+ mode: ArithMode,
4237
+ ): TsNode | null {
4238
+ let left = parseArithUnary(P, stop, mode)
4239
+ if (!left) return null
4240
+ while (true) {
4241
+ skipBlanks(P.L)
4242
+ if (isArithStop(P, stop)) break
4243
+ if (peek(P.L) === ',') break
4244
+ const opInfo = scanArithOp(P)
4245
+ if (!opInfo) break
4246
+ const [opText, opLen] = opInfo
4247
+ const prec = ARITH_PREC[opText]
4248
+ if (prec === undefined || prec < minPrec) break
4249
+ const os = P.L.b
4250
+ for (let k = 0; k < opLen; k++) advance(P.L)
4251
+ const op = mk(P, opText, os, P.L.b, [])
4252
+ const nextMin = ARITH_RIGHT_ASSOC.has(opText) ? prec : prec + 1
4253
+ const right = parseArithBinary(P, stop, nextMin, mode)
4254
+ if (!right) break
4255
+ left = mk(P, 'binary_expression', left.startIndex, right.endIndex, [
4256
+ left,
4257
+ op,
4258
+ right,
4259
+ ])
4260
+ }
4261
+ return left
4262
+ }
4263
+
4264
+ function parseArithUnary(
4265
+ P: ParseState,
4266
+ stop: string,
4267
+ mode: ArithMode,
4268
+ ): TsNode | null {
4269
+ skipBlanks(P.L)
4270
+ if (isArithStop(P, stop)) return null
4271
+ const c = peek(P.L)
4272
+ const c1 = peek(P.L, 1)
4273
+ // Prefix ++ --
4274
+ if ((c === '+' && c1 === '+') || (c === '-' && c1 === '-')) {
4275
+ const s = P.L.b
4276
+ advance(P.L)
4277
+ advance(P.L)
4278
+ const op = mk(P, c + c1, s, P.L.b, [])
4279
+ const inner = parseArithUnary(P, stop, mode)
4280
+ if (!inner) return op
4281
+ return mk(P, 'unary_expression', op.startIndex, inner.endIndex, [op, inner])
4282
+ }
4283
+ if (c === '-' || c === '+' || c === '!' || c === '~') {
4284
+ // In 'word'/'assign' mode (c-style for head), `-N` is a single number
4285
+ // literal per tree-sitter, not unary_expression. 'var' mode uses unary.
4286
+ if (mode !== 'var' && c === '-' && isDigit(c1)) {
4287
+ const s = P.L.b
4288
+ advance(P.L)
4289
+ while (isDigit(peek(P.L))) advance(P.L)
4290
+ return mk(P, 'number', s, P.L.b, [])
4291
+ }
4292
+ const s = P.L.b
4293
+ advance(P.L)
4294
+ const op = mk(P, c, s, P.L.b, [])
4295
+ const inner = parseArithUnary(P, stop, mode)
4296
+ if (!inner) return op
4297
+ return mk(P, 'unary_expression', op.startIndex, inner.endIndex, [op, inner])
4298
+ }
4299
+ return parseArithPostfix(P, stop, mode)
4300
+ }
4301
+
4302
+ function parseArithPostfix(
4303
+ P: ParseState,
4304
+ stop: string,
4305
+ mode: ArithMode,
4306
+ ): TsNode | null {
4307
+ const prim = parseArithPrimary(P, stop, mode)
4308
+ if (!prim) return null
4309
+ const c = peek(P.L)
4310
+ const c1 = peek(P.L, 1)
4311
+ if ((c === '+' && c1 === '+') || (c === '-' && c1 === '-')) {
4312
+ const s = P.L.b
4313
+ advance(P.L)
4314
+ advance(P.L)
4315
+ const op = mk(P, c + c1, s, P.L.b, [])
4316
+ return mk(P, 'postfix_expression', prim.startIndex, op.endIndex, [prim, op])
4317
+ }
4318
+ return prim
4319
+ }
4320
+
4321
+ function parseArithPrimary(
4322
+ P: ParseState,
4323
+ stop: string,
4324
+ mode: ArithMode,
4325
+ ): TsNode | null {
4326
+ skipBlanks(P.L)
4327
+ if (isArithStop(P, stop)) return null
4328
+ const c = peek(P.L)
4329
+ if (c === '(') {
4330
+ const s = P.L.b
4331
+ advance(P.L)
4332
+ const open = mk(P, '(', s, P.L.b, [])
4333
+ // Parenthesized expression may contain comma-separated exprs
4334
+ const inners = parseArithCommaList(P, ')', mode)
4335
+ skipBlanks(P.L)
4336
+ let close: TsNode
4337
+ if (peek(P.L) === ')') {
4338
+ const cs = P.L.b
4339
+ advance(P.L)
4340
+ close = mk(P, ')', cs, P.L.b, [])
4341
+ } else {
4342
+ close = mk(P, ')', P.L.b, P.L.b, [])
4343
+ }
4344
+ return mk(P, 'parenthesized_expression', open.startIndex, close.endIndex, [
4345
+ open,
4346
+ ...inners,
4347
+ close,
4348
+ ])
4349
+ }
4350
+ if (c === '"') {
4351
+ return parseDoubleQuoted(P)
4352
+ }
4353
+ if (c === '$') {
4354
+ return parseDollarLike(P)
4355
+ }
4356
+ if (isDigit(c)) {
4357
+ const s = P.L.b
4358
+ while (isDigit(peek(P.L))) advance(P.L)
4359
+ // Hex: 0x1f
4360
+ if (
4361
+ P.L.b - s === 1 &&
4362
+ c === '0' &&
4363
+ (peek(P.L) === 'x' || peek(P.L) === 'X')
4364
+ ) {
4365
+ advance(P.L)
4366
+ while (isHexDigit(peek(P.L))) advance(P.L)
4367
+ }
4368
+ // Base notation: BASE#DIGITS e.g. 2#1010, 16#ff
4369
+ else if (peek(P.L) === '#') {
4370
+ advance(P.L)
4371
+ while (isBaseDigit(peek(P.L))) advance(P.L)
4372
+ }
4373
+ return mk(P, 'number', s, P.L.b, [])
4374
+ }
4375
+ if (isIdentStart(c)) {
4376
+ const s = P.L.b
4377
+ while (isIdentChar(peek(P.L))) advance(P.L)
4378
+ const nc = peek(P.L)
4379
+ // Assignment in 'assign' mode (c-style for init): emit variable_assignment
4380
+ // so chained `a = b = c = 1` nests correctly. Other modes treat `=` as a
4381
+ // binary_expression operator via the precedence table.
4382
+ if (mode === 'assign') {
4383
+ skipBlanks(P.L)
4384
+ const ac = peek(P.L)
4385
+ const ac1 = peek(P.L, 1)
4386
+ if (ac === '=' && ac1 !== '=') {
4387
+ const vn = mk(P, 'variable_name', s, P.L.b, [])
4388
+ const es = P.L.b
4389
+ advance(P.L)
4390
+ const eq = mk(P, '=', es, P.L.b, [])
4391
+ // RHS may itself be another assignment (chained)
4392
+ const val = parseArithTernary(P, stop, mode)
4393
+ const end = val ? val.endIndex : eq.endIndex
4394
+ const kids = val ? [vn, eq, val] : [vn, eq]
4395
+ return mk(P, 'variable_assignment', s, end, kids)
4396
+ }
4397
+ }
4398
+ // Subscript
4399
+ if (nc === '[') {
4400
+ const vn = mk(P, 'variable_name', s, P.L.b, [])
4401
+ const brS = P.L.b
4402
+ advance(P.L)
4403
+ const brOpen = mk(P, '[', brS, P.L.b, [])
4404
+ const idx = parseArithTernary(P, ']', 'var') ?? parseDollarLike(P)
4405
+ skipBlanks(P.L)
4406
+ let brClose: TsNode
4407
+ if (peek(P.L) === ']') {
4408
+ const cs = P.L.b
4409
+ advance(P.L)
4410
+ brClose = mk(P, ']', cs, P.L.b, [])
4411
+ } else {
4412
+ brClose = mk(P, ']', P.L.b, P.L.b, [])
4413
+ }
4414
+ const kids = idx ? [vn, brOpen, idx, brClose] : [vn, brOpen, brClose]
4415
+ return mk(P, 'subscript', s, brClose.endIndex, kids)
4416
+ }
4417
+ // Bare identifier: variable_name in 'var' mode, word in 'word'/'assign' mode.
4418
+ // 'assign' mode falls through to word when no `=` follows (c-style for
4419
+ // cond/update clauses: `c<=5` → binary_expression(word, number)).
4420
+ const identType = mode === 'var' ? 'variable_name' : 'word'
4421
+ return mk(P, identType, s, P.L.b, [])
4422
+ }
4423
+ return null
4424
+ }
4425
+
4426
+ function isArithStop(P: ParseState, stop: string): boolean {
4427
+ const c = peek(P.L)
4428
+ if (stop === '))') return c === ')' && peek(P.L, 1) === ')'
4429
+ if (stop === ')') return c === ')'
4430
+ if (stop === ';') return c === ';'
4431
+ if (stop === ':') return c === ':'
4432
+ if (stop === ']') return c === ']'
4433
+ if (stop === '}') return c === '}'
4434
+ if (stop === ':}') return c === ':' || c === '}'
4435
+ return c === '' || c === '\n'
4436
+ }