@epoch-ai/cli 2.2.4

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 (710) hide show
  1. package/.artifacts/unit/junit.xml +2823 -0
  2. package/.project-map/backups/20260530_223453/.project-map.json +90101 -0
  3. package/.project-map/backups/20260530_223507/.project-map.json +90101 -0
  4. package/.project-map/backups/20260530_223512/.project-map.json +90101 -0
  5. package/.project-map/backups/20260530_223512/map.toon +666 -0
  6. package/.project-map/backups/20260530_223516/.project-map.json +90101 -0
  7. package/.project-map/backups/20260530_223516/map.toon +666 -0
  8. package/.project-map/backups/20260530_223520/.project-map.json +90101 -0
  9. package/.project-map/backups/20260530_223520/map.toon +666 -0
  10. package/AGENTS.md +47 -0
  11. package/BUN_SHELL_MIGRATION_PLAN.md +136 -0
  12. package/Dockerfile +18 -0
  13. package/README.md +15 -0
  14. package/bin/epochcli +179 -0
  15. package/bunfig.toml +7 -0
  16. package/drizzle.config.ts +10 -0
  17. package/git +0 -0
  18. package/migration/20260127222353_familiar_lady_ursula/migration.sql +90 -0
  19. package/migration/20260127222353_familiar_lady_ursula/snapshot.json +796 -0
  20. package/migration/20260211171708_add_project_commands/migration.sql +1 -0
  21. package/migration/20260211171708_add_project_commands/snapshot.json +806 -0
  22. package/migration/20260213144116_wakeful_the_professor/migration.sql +11 -0
  23. package/migration/20260213144116_wakeful_the_professor/snapshot.json +897 -0
  24. package/migration/20260225215848_workspace/migration.sql +7 -0
  25. package/migration/20260225215848_workspace/snapshot.json +959 -0
  26. package/migration/20260227213759_add_session_workspace_id/migration.sql +2 -0
  27. package/migration/20260227213759_add_session_workspace_id/snapshot.json +983 -0
  28. package/migration/20260228203230_blue_harpoon/migration.sql +17 -0
  29. package/migration/20260228203230_blue_harpoon/snapshot.json +1102 -0
  30. package/migration/20260303231226_add_workspace_fields/migration.sql +5 -0
  31. package/migration/20260303231226_add_workspace_fields/snapshot.json +1013 -0
  32. package/migration/20260309230000_move_org_to_state/migration.sql +3 -0
  33. package/migration/20260309230000_move_org_to_state/snapshot.json +1156 -0
  34. package/migration/20260312043431_session_message_cursor/migration.sql +4 -0
  35. package/migration/20260312043431_session_message_cursor/snapshot.json +1168 -0
  36. package/migration/20260323234822_events/migration.sql +13 -0
  37. package/migration/20260323234822_events/snapshot.json +1271 -0
  38. package/migration/20260418092949_add_yolo_to_session/migration.sql +2 -0
  39. package/migration/20260418092949_add_yolo_to_session/snapshot.json +1199 -0
  40. package/migration/20260419120000_add_intervention_to_session/migration.sql +2 -0
  41. package/package.json +186 -0
  42. package/parsers-config.ts +290 -0
  43. package/script/build-node.ts +71 -0
  44. package/script/build.ts +255 -0
  45. package/script/check-migrations.ts +16 -0
  46. package/script/fix-node-pty.ts +28 -0
  47. package/script/postinstall.mjs +131 -0
  48. package/script/publish.ts +184 -0
  49. package/script/schema.ts +63 -0
  50. package/script/seed-e2e.ts +60 -0
  51. package/script/upgrade-opentui.ts +64 -0
  52. package/specs/effect-migration.md +310 -0
  53. package/specs/tui-plugins.md +436 -0
  54. package/specs/v2.md +14 -0
  55. package/src/account/account.sql.ts +39 -0
  56. package/src/account/index.ts +465 -0
  57. package/src/account/repo.ts +163 -0
  58. package/src/account/schema.ts +91 -0
  59. package/src/acp/README.md +174 -0
  60. package/src/acp/agent.ts +1847 -0
  61. package/src/acp/session.ts +116 -0
  62. package/src/acp/types.ts +24 -0
  63. package/src/agent/agent.ts +445 -0
  64. package/src/agent/generate.txt +75 -0
  65. package/src/agent/prompt/compaction.txt +15 -0
  66. package/src/agent/prompt/explore.txt +9 -0
  67. package/src/agent/prompt/summary.txt +11 -0
  68. package/src/agent/prompt/title.txt +44 -0
  69. package/src/auth/index.ts +110 -0
  70. package/src/bus/bus-event.ts +40 -0
  71. package/src/bus/global.ts +10 -0
  72. package/src/bus/index.ts +232 -0
  73. package/src/cli/bootstrap.ts +17 -0
  74. package/src/cli/cmd/account.ts +257 -0
  75. package/src/cli/cmd/acp.ts +70 -0
  76. package/src/cli/cmd/agent.ts +245 -0
  77. package/src/cli/cmd/cmd.ts +7 -0
  78. package/src/cli/cmd/db.ts +119 -0
  79. package/src/cli/cmd/debug/agent.ts +167 -0
  80. package/src/cli/cmd/debug/config.ts +16 -0
  81. package/src/cli/cmd/debug/file.ts +97 -0
  82. package/src/cli/cmd/debug/index.ts +48 -0
  83. package/src/cli/cmd/debug/lsp.ts +53 -0
  84. package/src/cli/cmd/debug/ripgrep.ts +87 -0
  85. package/src/cli/cmd/debug/scrap.ts +16 -0
  86. package/src/cli/cmd/debug/skill.ts +16 -0
  87. package/src/cli/cmd/debug/snapshot.ts +52 -0
  88. package/src/cli/cmd/export.ts +89 -0
  89. package/src/cli/cmd/generate.ts +38 -0
  90. package/src/cli/cmd/github.ts +1639 -0
  91. package/src/cli/cmd/import.ts +169 -0
  92. package/src/cli/cmd/mcp.ts +754 -0
  93. package/src/cli/cmd/models.ts +78 -0
  94. package/src/cli/cmd/plug.ts +233 -0
  95. package/src/cli/cmd/pr.ts +127 -0
  96. package/src/cli/cmd/providers.ts +478 -0
  97. package/src/cli/cmd/run.ts +681 -0
  98. package/src/cli/cmd/serve.ts +24 -0
  99. package/src/cli/cmd/session.ts +159 -0
  100. package/src/cli/cmd/stats.ts +410 -0
  101. package/src/cli/cmd/tui/app.tsx +945 -0
  102. package/src/cli/cmd/tui/attach.ts +88 -0
  103. package/src/cli/cmd/tui/component/border.tsx +21 -0
  104. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  105. package/src/cli/cmd/tui/component/dialog-command.tsx +171 -0
  106. package/src/cli/cmd/tui/component/dialog-console-org.tsx +103 -0
  107. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  108. package/src/cli/cmd/tui/component/dialog-model.tsx +190 -0
  109. package/src/cli/cmd/tui/component/dialog-provider.tsx +364 -0
  110. package/src/cli/cmd/tui/component/dialog-session-list.tsx +108 -0
  111. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  112. package/src/cli/cmd/tui/component/dialog-skill.tsx +36 -0
  113. package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
  114. package/src/cli/cmd/tui/component/dialog-status.tsx +168 -0
  115. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  116. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  117. package/src/cli/cmd/tui/component/dialog-variant.tsx +39 -0
  118. package/src/cli/cmd/tui/component/dialog-workspace-list.tsx +320 -0
  119. package/src/cli/cmd/tui/component/error-component.tsx +92 -0
  120. package/src/cli/cmd/tui/component/logo.tsx +85 -0
  121. package/src/cli/cmd/tui/component/plugin-route-missing.tsx +14 -0
  122. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +672 -0
  123. package/src/cli/cmd/tui/component/prompt/frecency.tsx +90 -0
  124. package/src/cli/cmd/tui/component/prompt/history.tsx +109 -0
  125. package/src/cli/cmd/tui/component/prompt/index.tsx +1336 -0
  126. package/src/cli/cmd/tui/component/prompt/part.ts +16 -0
  127. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  128. package/src/cli/cmd/tui/component/spinner.tsx +24 -0
  129. package/src/cli/cmd/tui/component/startup-loading.tsx +63 -0
  130. package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
  131. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  132. package/src/cli/cmd/tui/component/workspace/dialog-session-list.tsx +151 -0
  133. package/src/cli/cmd/tui/context/args.tsx +15 -0
  134. package/src/cli/cmd/tui/context/directory.ts +13 -0
  135. package/src/cli/cmd/tui/context/exit.tsx +60 -0
  136. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  137. package/src/cli/cmd/tui/context/keybind.tsx +105 -0
  138. package/src/cli/cmd/tui/context/kv.tsx +52 -0
  139. package/src/cli/cmd/tui/context/local.tsx +456 -0
  140. package/src/cli/cmd/tui/context/plugin-keybinds.ts +41 -0
  141. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  142. package/src/cli/cmd/tui/context/route.tsx +52 -0
  143. package/src/cli/cmd/tui/context/sdk.tsx +115 -0
  144. package/src/cli/cmd/tui/context/sync.tsx +516 -0
  145. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  146. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  147. package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
  148. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  149. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  150. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  151. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  152. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  153. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  154. package/src/cli/cmd/tui/context/theme/epochcli.json +245 -0
  155. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  156. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  157. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  158. package/src/cli/cmd/tui/context/theme/gruvbox.json +242 -0
  159. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  160. package/src/cli/cmd/tui/context/theme/lucent-orng.json +237 -0
  161. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  162. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  163. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  164. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  165. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  166. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  167. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  168. package/src/cli/cmd/tui/context/theme/orng.json +249 -0
  169. package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
  170. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  171. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  172. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  173. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  174. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  175. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  176. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  177. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  178. package/src/cli/cmd/tui/context/theme.tsx +1236 -0
  179. package/src/cli/cmd/tui/context/tui-config.tsx +9 -0
  180. package/src/cli/cmd/tui/event.ts +48 -0
  181. package/src/cli/cmd/tui/feature-plugins/home/footer.tsx +93 -0
  182. package/src/cli/cmd/tui/feature-plugins/home/tips-view.tsx +145 -0
  183. package/src/cli/cmd/tui/feature-plugins/home/tips.tsx +50 -0
  184. package/src/cli/cmd/tui/feature-plugins/sidebar/context.tsx +63 -0
  185. package/src/cli/cmd/tui/feature-plugins/sidebar/files.tsx +62 -0
  186. package/src/cli/cmd/tui/feature-plugins/sidebar/footer.tsx +93 -0
  187. package/src/cli/cmd/tui/feature-plugins/sidebar/lsp.tsx +66 -0
  188. package/src/cli/cmd/tui/feature-plugins/sidebar/mcp.tsx +96 -0
  189. package/src/cli/cmd/tui/feature-plugins/sidebar/todo.tsx +48 -0
  190. package/src/cli/cmd/tui/feature-plugins/system/plugins.tsx +270 -0
  191. package/src/cli/cmd/tui/plugin/api.tsx +397 -0
  192. package/src/cli/cmd/tui/plugin/index.ts +3 -0
  193. package/src/cli/cmd/tui/plugin/internal.ts +27 -0
  194. package/src/cli/cmd/tui/plugin/runtime.ts +1031 -0
  195. package/src/cli/cmd/tui/plugin/slots.tsx +60 -0
  196. package/src/cli/cmd/tui/routes/home.tsx +84 -0
  197. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +65 -0
  198. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +110 -0
  199. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  200. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  201. package/src/cli/cmd/tui/routes/session/footer.tsx +91 -0
  202. package/src/cli/cmd/tui/routes/session/index.tsx +2161 -0
  203. package/src/cli/cmd/tui/routes/session/permission.tsx +691 -0
  204. package/src/cli/cmd/tui/routes/session/question.tsx +468 -0
  205. package/src/cli/cmd/tui/routes/session/sidebar.tsx +70 -0
  206. package/src/cli/cmd/tui/routes/session/subagent-footer.tsx +131 -0
  207. package/src/cli/cmd/tui/thread.ts +241 -0
  208. package/src/cli/cmd/tui/ui/dialog-alert.tsx +59 -0
  209. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +89 -0
  210. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +211 -0
  211. package/src/cli/cmd/tui/ui/dialog-help.tsx +40 -0
  212. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +115 -0
  213. package/src/cli/cmd/tui/ui/dialog-select.tsx +417 -0
  214. package/src/cli/cmd/tui/ui/dialog.tsx +192 -0
  215. package/src/cli/cmd/tui/ui/link.tsx +28 -0
  216. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  217. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  218. package/src/cli/cmd/tui/util/clipboard.ts +192 -0
  219. package/src/cli/cmd/tui/util/editor.ts +37 -0
  220. package/src/cli/cmd/tui/util/model.ts +23 -0
  221. package/src/cli/cmd/tui/util/provider-origin.ts +20 -0
  222. package/src/cli/cmd/tui/util/scroll.ts +23 -0
  223. package/src/cli/cmd/tui/util/selection.ts +25 -0
  224. package/src/cli/cmd/tui/util/signal.ts +7 -0
  225. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  226. package/src/cli/cmd/tui/util/transcript.ts +112 -0
  227. package/src/cli/cmd/tui/win32.ts +129 -0
  228. package/src/cli/cmd/tui/worker.ts +195 -0
  229. package/src/cli/cmd/uninstall.ts +353 -0
  230. package/src/cli/cmd/upgrade.ts +73 -0
  231. package/src/cli/cmd/web.ts +81 -0
  232. package/src/cli/effect/prompt.ts +25 -0
  233. package/src/cli/error.ts +46 -0
  234. package/src/cli/heap.ts +59 -0
  235. package/src/cli/logo.ts +6 -0
  236. package/src/cli/network.ts +60 -0
  237. package/src/cli/ui.ts +133 -0
  238. package/src/cli/upgrade.ts +31 -0
  239. package/src/command/index.ts +197 -0
  240. package/src/command/template/initialize.txt +66 -0
  241. package/src/command/template/review.txt +101 -0
  242. package/src/config/config.ts +1610 -0
  243. package/src/config/console-state.ts +15 -0
  244. package/src/config/markdown.ts +99 -0
  245. package/src/config/paths.ts +167 -0
  246. package/src/config/tui-migrate.ts +155 -0
  247. package/src/config/tui-schema.ts +37 -0
  248. package/src/config/tui.ts +179 -0
  249. package/src/config/validator.ts +52 -0
  250. package/src/control-plane/adaptors/index.ts +20 -0
  251. package/src/control-plane/adaptors/worktree.ts +42 -0
  252. package/src/control-plane/schema.ts +17 -0
  253. package/src/control-plane/sse.ts +66 -0
  254. package/src/control-plane/types.ts +32 -0
  255. package/src/control-plane/workspace.sql.ts +17 -0
  256. package/src/control-plane/workspace.ts +168 -0
  257. package/src/effect/cross-spawn-spawner.ts +502 -0
  258. package/src/effect/instance-ref.ts +6 -0
  259. package/src/effect/instance-registry.ts +12 -0
  260. package/src/effect/instance-state.ts +82 -0
  261. package/src/effect/run-service.ts +33 -0
  262. package/src/effect/runner.ts +216 -0
  263. package/src/env/index.ts +28 -0
  264. package/src/file/ignore.ts +82 -0
  265. package/src/file/index.ts +686 -0
  266. package/src/file/protected.ts +59 -0
  267. package/src/file/ripgrep.ts +376 -0
  268. package/src/file/time.ts +133 -0
  269. package/src/file/watcher.ts +172 -0
  270. package/src/filesystem/index.ts +236 -0
  271. package/src/flag/flag.ts +157 -0
  272. package/src/format/formatter.ts +413 -0
  273. package/src/format/index.ts +203 -0
  274. package/src/git/index.ts +303 -0
  275. package/src/global/index.ts +54 -0
  276. package/src/id/id.ts +85 -0
  277. package/src/ide/index.ts +74 -0
  278. package/src/index.ts +253 -0
  279. package/src/installation/index.ts +355 -0
  280. package/src/installation/meta.ts +7 -0
  281. package/src/lsp/client.ts +256 -0
  282. package/src/lsp/index.ts +558 -0
  283. package/src/lsp/language.ts +120 -0
  284. package/src/lsp/launch.ts +21 -0
  285. package/src/lsp/server.ts +1968 -0
  286. package/src/mcp/auth.ts +173 -0
  287. package/src/mcp/index.ts +1250 -0
  288. package/src/mcp/oauth-callback.ts +216 -0
  289. package/src/mcp/oauth-provider.ts +185 -0
  290. package/src/mcp/schema-loader.ts +82 -0
  291. package/src/node.ts +1 -0
  292. package/src/npm/index.ts +188 -0
  293. package/src/patch/index.ts +680 -0
  294. package/src/permission/arity.ts +163 -0
  295. package/src/permission/evaluate.ts +15 -0
  296. package/src/permission/index.ts +323 -0
  297. package/src/permission/schema.ts +17 -0
  298. package/src/plugin/cloudflare.ts +67 -0
  299. package/src/plugin/codex.ts +608 -0
  300. package/src/plugin/github-copilot/copilot.ts +361 -0
  301. package/src/plugin/github-copilot/models.ts +144 -0
  302. package/src/plugin/index.ts +288 -0
  303. package/src/plugin/install.ts +439 -0
  304. package/src/plugin/loader.ts +174 -0
  305. package/src/plugin/meta.ts +188 -0
  306. package/src/plugin/shared.ts +323 -0
  307. package/src/project/bootstrap.ts +29 -0
  308. package/src/project/instance.ts +175 -0
  309. package/src/project/project.sql.ts +16 -0
  310. package/src/project/project.ts +519 -0
  311. package/src/project/schema.ts +16 -0
  312. package/src/project/state.ts +70 -0
  313. package/src/project/vcs.ts +240 -0
  314. package/src/provider/auth.ts +253 -0
  315. package/src/provider/error.ts +297 -0
  316. package/src/provider/models.ts +162 -0
  317. package/src/provider/provider.ts +1776 -0
  318. package/src/provider/schema.ts +38 -0
  319. package/src/provider/sdk/copilot/README.md +5 -0
  320. package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +170 -0
  321. package/src/provider/sdk/copilot/chat/get-response-metadata.ts +15 -0
  322. package/src/provider/sdk/copilot/chat/map-openai-compatible-finish-reason.ts +19 -0
  323. package/src/provider/sdk/copilot/chat/openai-compatible-api-types.ts +64 -0
  324. package/src/provider/sdk/copilot/chat/openai-compatible-chat-language-model.ts +814 -0
  325. package/src/provider/sdk/copilot/chat/openai-compatible-chat-options.ts +28 -0
  326. package/src/provider/sdk/copilot/chat/openai-compatible-metadata-extractor.ts +44 -0
  327. package/src/provider/sdk/copilot/chat/openai-compatible-prepare-tools.ts +83 -0
  328. package/src/provider/sdk/copilot/copilot-provider.ts +100 -0
  329. package/src/provider/sdk/copilot/index.ts +2 -0
  330. package/src/provider/sdk/copilot/openai-compatible-error.ts +27 -0
  331. package/src/provider/sdk/copilot/responses/convert-to-openai-responses-input.ts +335 -0
  332. package/src/provider/sdk/copilot/responses/map-openai-responses-finish-reason.ts +22 -0
  333. package/src/provider/sdk/copilot/responses/openai-config.ts +18 -0
  334. package/src/provider/sdk/copilot/responses/openai-error.ts +22 -0
  335. package/src/provider/sdk/copilot/responses/openai-responses-api-types.ts +214 -0
  336. package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +1769 -0
  337. package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +173 -0
  338. package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +1 -0
  339. package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +87 -0
  340. package/src/provider/sdk/copilot/responses/tool/file-search.ts +127 -0
  341. package/src/provider/sdk/copilot/responses/tool/image-generation.ts +114 -0
  342. package/src/provider/sdk/copilot/responses/tool/local-shell.ts +64 -0
  343. package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +103 -0
  344. package/src/provider/sdk/copilot/responses/tool/web-search.ts +102 -0
  345. package/src/provider/transform.ts +1124 -0
  346. package/src/pty/index.ts +397 -0
  347. package/src/pty/pty.bun.ts +26 -0
  348. package/src/pty/pty.node.ts +27 -0
  349. package/src/pty/pty.ts +25 -0
  350. package/src/pty/schema.ts +17 -0
  351. package/src/question/index.ts +224 -0
  352. package/src/question/schema.ts +17 -0
  353. package/src/server/error.ts +36 -0
  354. package/src/server/event.ts +7 -0
  355. package/src/server/instance.ts +315 -0
  356. package/src/server/mdns.ts +60 -0
  357. package/src/server/middleware.ts +33 -0
  358. package/src/server/projectors.ts +28 -0
  359. package/src/server/proxy.ts +130 -0
  360. package/src/server/router.ts +105 -0
  361. package/src/server/routes/config.ts +92 -0
  362. package/src/server/routes/event.ts +83 -0
  363. package/src/server/routes/experimental.ts +374 -0
  364. package/src/server/routes/file.ts +197 -0
  365. package/src/server/routes/global.ts +312 -0
  366. package/src/server/routes/mcp.ts +225 -0
  367. package/src/server/routes/permission.ts +69 -0
  368. package/src/server/routes/project.ts +118 -0
  369. package/src/server/routes/provider.ts +171 -0
  370. package/src/server/routes/pty.ts +210 -0
  371. package/src/server/routes/question.ts +99 -0
  372. package/src/server/routes/session.ts +984 -0
  373. package/src/server/routes/tui.ts +378 -0
  374. package/src/server/routes/workspace.ts +94 -0
  375. package/src/server/server.ts +353 -0
  376. package/src/session/compaction.ts +86 -0
  377. package/src/session/index.ts +904 -0
  378. package/src/session/instruction.ts +261 -0
  379. package/src/session/llm/monitor.ts +87 -0
  380. package/src/session/llm.ts +1676 -0
  381. package/src/session/message-v2.ts +1082 -0
  382. package/src/session/message.ts +191 -0
  383. package/src/session/overflow.ts +35 -0
  384. package/src/session/processor.ts +635 -0
  385. package/src/session/projectors.ts +136 -0
  386. package/src/session/prompt/build-switch.txt +5 -0
  387. package/src/session/prompt/builder.ts +135 -0
  388. package/src/session/prompt/default.txt +11 -0
  389. package/src/session/prompt/engine.ts +1072 -0
  390. package/src/session/prompt/gemma4.txt +1 -0
  391. package/src/session/prompt/max-steps.txt +16 -0
  392. package/src/session/prompt/orchestrator.ts +426 -0
  393. package/src/session/prompt/plan.txt +28 -0
  394. package/src/session/prompt/qwen.txt +19 -0
  395. package/src/session/prompt/resolver.ts +670 -0
  396. package/src/session/prompt/router.ts +197 -0
  397. package/src/session/prompt/state.ts +96 -0
  398. package/src/session/prompt/types.ts +115 -0
  399. package/src/session/prompt/utils.ts +15 -0
  400. package/src/session/prompt.ts +362 -0
  401. package/src/session/retry.ts +106 -0
  402. package/src/session/revert.ts +176 -0
  403. package/src/session/sanitizer.ts +125 -0
  404. package/src/session/schema.ts +38 -0
  405. package/src/session/session.sql.ts +106 -0
  406. package/src/session/status.ts +102 -0
  407. package/src/session/summary.ts +183 -0
  408. package/src/session/system.ts +79 -0
  409. package/src/session/todo.ts +166 -0
  410. package/src/session/worker.ts +382 -0
  411. package/src/shell/shell.ts +110 -0
  412. package/src/skill/discovery.ts +116 -0
  413. package/src/skill/index.ts +287 -0
  414. package/src/snapshot/index.ts +726 -0
  415. package/src/sql.d.ts +4 -0
  416. package/src/storage/db.bun.ts +8 -0
  417. package/src/storage/db.node.ts +8 -0
  418. package/src/storage/db.ts +174 -0
  419. package/src/storage/json-migration.ts +387 -0
  420. package/src/storage/schema.sql.ts +10 -0
  421. package/src/storage/schema.ts +4 -0
  422. package/src/storage/storage.ts +353 -0
  423. package/src/sync/README.md +179 -0
  424. package/src/sync/event.sql.ts +16 -0
  425. package/src/sync/index.ts +263 -0
  426. package/src/sync/schema.ts +14 -0
  427. package/src/tool/apply_patch.ts +281 -0
  428. package/src/tool/apply_patch.txt +1 -0
  429. package/src/tool/arbitration.txt +5 -0
  430. package/src/tool/bash.ts +494 -0
  431. package/src/tool/bash.txt +2 -0
  432. package/src/tool/batch.ts +183 -0
  433. package/src/tool/batch.txt +1 -0
  434. package/src/tool/codesearch.ts +132 -0
  435. package/src/tool/codesearch.txt +1 -0
  436. package/src/tool/edit.ts +734 -0
  437. package/src/tool/edit.txt +1 -0
  438. package/src/tool/external-directory.ts +46 -0
  439. package/src/tool/glob.ts +73 -0
  440. package/src/tool/glob.txt +2 -0
  441. package/src/tool/grep.ts +156 -0
  442. package/src/tool/grep.txt +2 -0
  443. package/src/tool/invalid.ts +20 -0
  444. package/src/tool/ls.ts +121 -0
  445. package/src/tool/ls.txt +1 -0
  446. package/src/tool/lsp.ts +97 -0
  447. package/src/tool/lsp.txt +1 -0
  448. package/src/tool/multiedit.ts +46 -0
  449. package/src/tool/multiedit.txt +1 -0
  450. package/src/tool/plan-enter.txt +14 -0
  451. package/src/tool/plan-exit.txt +13 -0
  452. package/src/tool/plan.ts +131 -0
  453. package/src/tool/question.ts +46 -0
  454. package/src/tool/question.txt +10 -0
  455. package/src/tool/read.ts +332 -0
  456. package/src/tool/read.txt +1 -0
  457. package/src/tool/registry.ts +288 -0
  458. package/src/tool/revert.ts +37 -0
  459. package/src/tool/schema.ts +17 -0
  460. package/src/tool/skill.ts +105 -0
  461. package/src/tool/task.ts +150 -0
  462. package/src/tool/task.txt +3 -0
  463. package/src/tool/task_complete.ts +21 -0
  464. package/src/tool/tool.ts +112 -0
  465. package/src/tool/truncate.ts +144 -0
  466. package/src/tool/truncation-dir.ts +4 -0
  467. package/src/tool/webfetch.ts +206 -0
  468. package/src/tool/webfetch.txt +1 -0
  469. package/src/tool/websearch.ts +150 -0
  470. package/src/tool/websearch.txt +1 -0
  471. package/src/tool/write.ts +101 -0
  472. package/src/tool/write.txt +1 -0
  473. package/src/util/abort.ts +35 -0
  474. package/src/util/ai-sdk.ts +59 -0
  475. package/src/util/archive.ts +17 -0
  476. package/src/util/color.ts +19 -0
  477. package/src/util/context.ts +25 -0
  478. package/src/util/data-url.ts +9 -0
  479. package/src/util/defer.ts +12 -0
  480. package/src/util/effect-http-client.ts +11 -0
  481. package/src/util/effect-zod.ts +98 -0
  482. package/src/util/error.ts +77 -0
  483. package/src/util/filesystem.ts +245 -0
  484. package/src/util/flock.ts +333 -0
  485. package/src/util/fn.ts +21 -0
  486. package/src/util/format.ts +20 -0
  487. package/src/util/glob.ts +34 -0
  488. package/src/util/hash.ts +7 -0
  489. package/src/util/iife.ts +3 -0
  490. package/src/util/keybind.ts +103 -0
  491. package/src/util/lazy.ts +23 -0
  492. package/src/util/locale.ts +81 -0
  493. package/src/util/lock.ts +98 -0
  494. package/src/util/log-parser.ts +114 -0
  495. package/src/util/log.ts +250 -0
  496. package/src/util/network.ts +23 -0
  497. package/src/util/process.ts +176 -0
  498. package/src/util/queue.ts +32 -0
  499. package/src/util/record.ts +3 -0
  500. package/src/util/rpc.ts +66 -0
  501. package/src/util/schema.ts +53 -0
  502. package/src/util/scrap.ts +10 -0
  503. package/src/util/session-analyzer.ts +331 -0
  504. package/src/util/session-telemetry.ts +91 -0
  505. package/src/util/signal.ts +12 -0
  506. package/src/util/timeout.ts +14 -0
  507. package/src/util/token.ts +7 -0
  508. package/src/util/tokenizer.ts +50 -0
  509. package/src/util/toon.ts +45 -0
  510. package/src/util/update-schema.ts +13 -0
  511. package/src/util/which.ts +14 -0
  512. package/src/util/wildcard.ts +59 -0
  513. package/src/worktree/index.ts +612 -0
  514. package/sst-env.d.ts +10 -0
  515. package/test/AGENTS.md +81 -0
  516. package/test/account/repo.test.ts +326 -0
  517. package/test/account/service.test.ts +393 -0
  518. package/test/acp/agent-interface.test.ts +51 -0
  519. package/test/acp/event-subscription.test.ts +685 -0
  520. package/test/agent/agent.test.ts +716 -0
  521. package/test/auth/auth.test.ts +58 -0
  522. package/test/bus/bus-effect.test.ts +164 -0
  523. package/test/bus/bus-integration.test.ts +87 -0
  524. package/test/bus/bus.test.ts +219 -0
  525. package/test/cli/account.test.ts +26 -0
  526. package/test/cli/cmd/tui/prompt-part.test.ts +47 -0
  527. package/test/cli/github-action.test.ts +198 -0
  528. package/test/cli/github-remote.test.ts +80 -0
  529. package/test/cli/plugin-auth-picker.test.ts +120 -0
  530. package/test/cli/tui/keybind-plugin.test.ts +90 -0
  531. package/test/cli/tui/plugin-add.test.ts +107 -0
  532. package/test/cli/tui/plugin-install.test.ts +89 -0
  533. package/test/cli/tui/plugin-lifecycle.test.ts +225 -0
  534. package/test/cli/tui/plugin-loader-entrypoint.test.ts +492 -0
  535. package/test/cli/tui/plugin-loader-pure.test.ts +72 -0
  536. package/test/cli/tui/plugin-loader.test.ts +752 -0
  537. package/test/cli/tui/plugin-toggle.test.ts +159 -0
  538. package/test/cli/tui/slot-replace.test.tsx +47 -0
  539. package/test/cli/tui/theme-store.test.ts +51 -0
  540. package/test/cli/tui/thread.test.ts +128 -0
  541. package/test/cli/tui/transcript.test.ts +426 -0
  542. package/test/config/agent-color.test.ts +71 -0
  543. package/test/config/config.test.ts +2337 -0
  544. package/test/config/fixtures/empty-frontmatter.md +4 -0
  545. package/test/config/fixtures/frontmatter.md +28 -0
  546. package/test/config/fixtures/markdown-header.md +11 -0
  547. package/test/config/fixtures/no-frontmatter.md +1 -0
  548. package/test/config/fixtures/weird-model-id.md +13 -0
  549. package/test/config/markdown.test.ts +228 -0
  550. package/test/config/tui.test.ts +800 -0
  551. package/test/control-plane/sse.test.ts +56 -0
  552. package/test/effect/cross-spawn-spawner.test.ts +412 -0
  553. package/test/effect/instance-state.test.ts +482 -0
  554. package/test/effect/run-service.test.ts +46 -0
  555. package/test/effect/runner.test.ts +523 -0
  556. package/test/fake/provider.ts +82 -0
  557. package/test/file/fsmonitor.test.ts +62 -0
  558. package/test/file/ignore.test.ts +10 -0
  559. package/test/file/index.test.ts +946 -0
  560. package/test/file/path-traversal.test.ts +198 -0
  561. package/test/file/ripgrep.test.ts +54 -0
  562. package/test/file/time.test.ts +445 -0
  563. package/test/file/watcher.test.ts +247 -0
  564. package/test/filesystem/filesystem.test.ts +319 -0
  565. package/test/fixture/db.ts +11 -0
  566. package/test/fixture/fixture.test.ts +26 -0
  567. package/test/fixture/fixture.ts +172 -0
  568. package/test/fixture/flock-worker.ts +72 -0
  569. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  570. package/test/fixture/plug-worker.ts +93 -0
  571. package/test/fixture/plugin-meta-worker.ts +26 -0
  572. package/test/fixture/skills/agents-sdk/SKILL.md +152 -0
  573. package/test/fixture/skills/agents-sdk/references/callable.md +92 -0
  574. package/test/fixture/skills/cloudflare/SKILL.md +211 -0
  575. package/test/fixture/skills/index.json +6 -0
  576. package/test/fixture/tui-plugin.ts +328 -0
  577. package/test/fixture/tui-runtime.ts +27 -0
  578. package/test/format/format.test.ts +171 -0
  579. package/test/git/git.test.ts +128 -0
  580. package/test/ide/ide.test.ts +82 -0
  581. package/test/installation/installation.test.ts +152 -0
  582. package/test/keybind.test.ts +421 -0
  583. package/test/lib/effect.ts +53 -0
  584. package/test/lib/filesystem.ts +10 -0
  585. package/test/lib/llm-server.ts +794 -0
  586. package/test/lsp/client.test.ts +95 -0
  587. package/test/lsp/index.test.ts +133 -0
  588. package/test/lsp/launch.test.ts +22 -0
  589. package/test/lsp/lifecycle.test.ts +147 -0
  590. package/test/mcp/headers.test.ts +153 -0
  591. package/test/mcp/lifecycle.test.ts +750 -0
  592. package/test/mcp/oauth-auto-connect.test.ts +199 -0
  593. package/test/mcp/oauth-browser.test.ts +249 -0
  594. package/test/mcp/sc-approve-validator.test.ts +431 -0
  595. package/test/memory/abort-leak.test.ts +137 -0
  596. package/test/npm.test.ts +18 -0
  597. package/test/patch/patch.test.ts +348 -0
  598. package/test/permission/arity.test.ts +33 -0
  599. package/test/permission/next.test.ts +1123 -0
  600. package/test/permission-task.test.ts +323 -0
  601. package/test/plugin/auth-override.test.ts +74 -0
  602. package/test/plugin/codex.test.ts +123 -0
  603. package/test/plugin/github-copilot-models.test.ts +117 -0
  604. package/test/plugin/install-concurrency.test.ts +140 -0
  605. package/test/plugin/install.test.ts +570 -0
  606. package/test/plugin/loader-shared.test.ts +1136 -0
  607. package/test/plugin/meta.test.ts +137 -0
  608. package/test/plugin/shared.test.ts +88 -0
  609. package/test/plugin/trigger.test.ts +111 -0
  610. package/test/preload.ts +90 -0
  611. package/test/project/migrate-global.test.ts +140 -0
  612. package/test/project/project.test.ts +459 -0
  613. package/test/project/state.test.ts +115 -0
  614. package/test/project/vcs.test.ts +228 -0
  615. package/test/project/worktree-remove.test.ts +96 -0
  616. package/test/project/worktree.test.ts +173 -0
  617. package/test/provider/amazon-bedrock.test.ts +447 -0
  618. package/test/provider/copilot/convert-to-copilot-messages.test.ts +523 -0
  619. package/test/provider/copilot/copilot-chat-model.test.ts +592 -0
  620. package/test/provider/error.test.ts +49 -0
  621. package/test/provider/gitlab-duo.test.ts +412 -0
  622. package/test/provider/provider.test.ts +2494 -0
  623. package/test/provider/transform.test.ts +2944 -0
  624. package/test/pty/pty-output-isolation.test.ts +141 -0
  625. package/test/pty/pty-session.test.ts +92 -0
  626. package/test/pty/pty-shell.test.ts +59 -0
  627. package/test/question/question.test.ts +453 -0
  628. package/test/server/global-session-list.test.ts +89 -0
  629. package/test/server/project-init-git.test.ts +121 -0
  630. package/test/server/session-actions.test.ts +83 -0
  631. package/test/server/session-list.test.ts +98 -0
  632. package/test/server/session-messages.test.ts +159 -0
  633. package/test/server/session-select.test.ts +84 -0
  634. package/test/session/compaction.test.ts +683 -0
  635. package/test/session/continuity-handover.test.ts +620 -0
  636. package/test/session/deterministic-handover.test.ts +328 -0
  637. package/test/session/doom-protection.test.ts +247 -0
  638. package/test/session/hard-reset.test.ts +179 -0
  639. package/test/session/instruction.test.ts +286 -0
  640. package/test/session/llm/monitor.test.ts +53 -0
  641. package/test/session/llm-sanitizer.test.ts +90 -0
  642. package/test/session/llm-zones-e2e.test.ts +61 -0
  643. package/test/session/llm.test.ts +1308 -0
  644. package/test/session/mcpx-normalization.test.ts +86 -0
  645. package/test/session/mcpx-syntax-recovery.test.ts +28 -0
  646. package/test/session/message-v2.test.ts +957 -0
  647. package/test/session/messages-pagination.test.ts +885 -0
  648. package/test/session/processor-effect.test.ts +805 -0
  649. package/test/session/prompt/builder.test.ts +71 -0
  650. package/test/session/prompt/engine-loop.test.ts +80 -0
  651. package/test/session/prompt/orchestrator.test.ts +108 -0
  652. package/test/session/prompt/resolver.test.ts +211 -0
  653. package/test/session/prompt/router.test.ts +84 -0
  654. package/test/session/prompt/state.test.ts +57 -0
  655. package/test/session/prompt-effect.test.ts +1241 -0
  656. package/test/session/prompt.test.ts +522 -0
  657. package/test/session/refactor-system-zones.test.ts +241 -0
  658. package/test/session/retry.test.ts +232 -0
  659. package/test/session/revert-compact.test.ts +621 -0
  660. package/test/session/sanitizer.test.ts +61 -0
  661. package/test/session/session.test.ts +142 -0
  662. package/test/session/snapshot-tool-race.test.ts +242 -0
  663. package/test/session/structured-output-integration.test.ts +233 -0
  664. package/test/session/structured-output.test.ts +391 -0
  665. package/test/session/system.test.ts +59 -0
  666. package/test/session/telemetry.test.ts +35 -0
  667. package/test/shell/shell.test.ts +73 -0
  668. package/test/skill/discovery.test.ts +116 -0
  669. package/test/skill/skill.test.ts +392 -0
  670. package/test/snapshot/snapshot.test.ts +1404 -0
  671. package/test/storage/db.test.ts +14 -0
  672. package/test/storage/json-migration.test.ts +791 -0
  673. package/test/storage/storage.test.ts +295 -0
  674. package/test/sync/index.test.ts +191 -0
  675. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  676. package/test/tool/apply_patch.test.ts +567 -0
  677. package/test/tool/bash.test.ts +1099 -0
  678. package/test/tool/edit.test.ts +681 -0
  679. package/test/tool/external-directory.test.ts +198 -0
  680. package/test/tool/fixtures/large-image.png +0 -0
  681. package/test/tool/fixtures/models-api.json +65179 -0
  682. package/test/tool/grep.test.ts +111 -0
  683. package/test/tool/question.test.ts +126 -0
  684. package/test/tool/read.test.ts +468 -0
  685. package/test/tool/registry.test.ts +126 -0
  686. package/test/tool/skill.test.ts +167 -0
  687. package/test/tool/task.test.ts +49 -0
  688. package/test/tool/tool-define.test.ts +101 -0
  689. package/test/tool/truncation.test.ts +161 -0
  690. package/test/tool/webfetch.test.ts +101 -0
  691. package/test/tool/write.test.ts +354 -0
  692. package/test/util/data-url.test.ts +14 -0
  693. package/test/util/effect-zod.test.ts +61 -0
  694. package/test/util/error.test.ts +38 -0
  695. package/test/util/filesystem.test.ts +656 -0
  696. package/test/util/flock.test.ts +383 -0
  697. package/test/util/format.test.ts +59 -0
  698. package/test/util/glob.test.ts +164 -0
  699. package/test/util/iife.test.ts +36 -0
  700. package/test/util/lazy.test.ts +50 -0
  701. package/test/util/lock.test.ts +72 -0
  702. package/test/util/log-parser.test.ts +61 -0
  703. package/test/util/module.test.ts +59 -0
  704. package/test/util/process.test.ts +128 -0
  705. package/test/util/telemetry-integration.test.ts +104 -0
  706. package/test/util/timeout.test.ts +21 -0
  707. package/test/util/which.test.ts +100 -0
  708. package/test/util/wildcard.test.ts +90 -0
  709. package/test-regex.js +50 -0
  710. package/tsconfig.json +23 -0
@@ -0,0 +1,72 @@
1
+ import { describe, expect, test } from "bun:test"
2
+ import { Lock } from "../../src/util/lock"
3
+
4
+ function tick() {
5
+ return new Promise<void>((r) => queueMicrotask(r))
6
+ }
7
+
8
+ async function flush(n = 5) {
9
+ for (let i = 0; i < n; i++) await tick()
10
+ }
11
+
12
+ describe("util.lock", () => {
13
+ test("writer exclusivity: blocks reads and other writes while held", async () => {
14
+ const key = "lock:" + Math.random().toString(36).slice(2)
15
+
16
+ const state = {
17
+ writer2: false,
18
+ reader: false,
19
+ writers: 0,
20
+ }
21
+
22
+ // Acquire writer1
23
+ using writer1 = await Lock.write(key)
24
+ state.writers++
25
+ expect(state.writers).toBe(1)
26
+
27
+ // Start writer2 candidate (should block)
28
+ const writer2Task = (async () => {
29
+ const w = await Lock.write(key)
30
+ state.writers++
31
+ expect(state.writers).toBe(1)
32
+ state.writer2 = true
33
+ // Hold for a tick so reader cannot slip in
34
+ await tick()
35
+ return w
36
+ })()
37
+
38
+ // Start reader candidate (should block)
39
+ const readerTask = (async () => {
40
+ const r = await Lock.read(key)
41
+ state.reader = true
42
+ return r
43
+ })()
44
+
45
+ // Flush microtasks and assert neither acquired
46
+ await flush()
47
+ expect(state.writer2).toBe(false)
48
+ expect(state.reader).toBe(false)
49
+
50
+ // Release writer1
51
+ writer1[Symbol.dispose]()
52
+ state.writers--
53
+
54
+ // writer2 should acquire next
55
+ const writer2 = await writer2Task
56
+ expect(state.writer2).toBe(true)
57
+
58
+ // Reader still blocked while writer2 held
59
+ await flush()
60
+ expect(state.reader).toBe(false)
61
+
62
+ // Release writer2
63
+ writer2[Symbol.dispose]()
64
+ state.writers--
65
+
66
+ // Reader should now acquire
67
+ const reader = await readerTask
68
+ expect(state.reader).toBe(true)
69
+
70
+ reader[Symbol.dispose]()
71
+ })
72
+ })
@@ -0,0 +1,61 @@
1
+ import { describe, it, expect, afterAll, beforeAll } from "bun:test"
2
+ import * as fs from "fs/promises"
3
+ import * as path from "path"
4
+ import { LogParserTool } from "../../src/util/log-parser"
5
+
6
+ describe("LogParserTool - State Machine Overlap Detection", () => {
7
+ const tmpDir = path.join(process.cwd(), "test-logs-tmp")
8
+
9
+ beforeAll(async () => {
10
+ await fs.mkdir(tmpDir, { recursive: true })
11
+ })
12
+
13
+ afterAll(async () => {
14
+ await fs.rm(tmpDir, { recursive: true, force: true })
15
+ })
16
+
17
+ it("validates successful sequential execution", async () => {
18
+ const logPath = path.join(tmpDir, "valid.log")
19
+ const logContent = `
20
+ 2026-04-08T12:00:00 INFO {"mainEpochId":"123","event":"START_GENERATE","providerId":"local-side","phase":"Phase 1"}
21
+ 2026-04-08T12:00:01 INFO {"mainEpochId":"123","event":"END_GENERATE","providerId":"local-side","phase":"Phase 1","metrics":{"metrics":{"json_repaired":true}}}
22
+ 2026-04-08T12:00:02 INFO {"mainEpochId":"123","event":"START_GENERATE","providerId":"local-main","phase":"Phase 2"}
23
+ 2026-04-08T12:00:05 INFO {"mainEpochId":"123","event":"END_GENERATE","providerId":"local-main","phase":"Phase 2"}
24
+ `
25
+ await fs.writeFile(logPath, logContent.trim())
26
+
27
+ const result = await LogParserTool.verifySequentialExecution(logPath)
28
+ expect(result.valid).toBe(true)
29
+ expect(result.message).toContain("Total sequential handoffs: 2")
30
+ expect(result.message).toContain("JSON Repairs: 1")
31
+ })
32
+
33
+ it("flags overlapping execution when local-main starts while local-side is active", async () => {
34
+ const logPath = path.join(tmpDir, "overlap.log")
35
+ const logContent = `
36
+ 2026-04-08T12:00:00 INFO {"mainEpochId":"456","event":"START_GENERATE","providerId":"local-side","phase":"Phase 1"}
37
+ 2026-04-08T12:00:01 INFO {"mainEpochId":"456","event":"START_GENERATE","providerId":"local-main","phase":"Phase 2"}
38
+ `
39
+ await fs.writeFile(logPath, logContent.trim())
40
+
41
+ const result = await LogParserTool.verifySequentialExecution(logPath)
42
+ expect(result.valid).toBe(false)
43
+ expect(result.message).toContain(
44
+ "Concurrency Violation: local-main started while local-side was still active in epoch 456",
45
+ )
46
+ })
47
+
48
+ it("ignores malformed JSON and skips gracefully", async () => {
49
+ const logPath = path.join(tmpDir, "malformed.log")
50
+ const logContent = `
51
+ 2026-04-08T12:00:00 INFO {"mainEpochId":"789","event":"START_GENERATE","providerId":"local-side","phase":"Phase 1"}
52
+ Just a normal log line with { some invalid JSON
53
+ 2026-04-08T12:00:01 INFO {"mainEpochId":"789","event":"END_GENERATE","providerId":"local-side","phase":"Phase 1"}
54
+ `
55
+ await fs.writeFile(logPath, logContent.trim())
56
+
57
+ const result = await LogParserTool.verifySequentialExecution(logPath)
58
+ expect(result.valid).toBe(true)
59
+ expect(result.message).toContain("Total sequential handoffs: 1")
60
+ })
61
+ })
@@ -0,0 +1,59 @@
1
+ import { describe, expect, test } from "bun:test"
2
+ import path from "path"
3
+ import { Module } from "@epoch-ai/util/module"
4
+ import { Filesystem } from "../../src/util/filesystem"
5
+ import { tmpdir } from "../fixture/fixture"
6
+
7
+ describe("util.module", () => {
8
+ test("resolves package subpaths from the provided dir", async () => {
9
+ await using tmp = await tmpdir()
10
+ const root = path.join(tmp.path, "proj")
11
+ const file = path.join(root, "node_modules/typescript/lib/tsserver.js")
12
+ await Filesystem.write(file, "export {}\n")
13
+ await Filesystem.writeJson(path.join(root, "node_modules/typescript/package.json"), { name: "typescript" })
14
+
15
+ expect(Module.resolve("typescript/lib/tsserver.js", root)).toBe(file)
16
+ })
17
+
18
+ test("resolves packages through ancestor node_modules", async () => {
19
+ await using tmp = await tmpdir()
20
+ const root = path.join(tmp.path, "proj")
21
+ const cwd = path.join(root, "apps/web")
22
+ const file = path.join(root, "node_modules/eslint/lib/api.js")
23
+ await Filesystem.write(file, "export {}\n")
24
+ await Filesystem.writeJson(path.join(root, "node_modules/eslint/package.json"), {
25
+ name: "eslint",
26
+ main: "lib/api.js",
27
+ })
28
+ await Filesystem.write(path.join(cwd, ".keep"), "")
29
+
30
+ expect(Module.resolve("eslint", cwd)).toBe(file)
31
+ })
32
+
33
+ test("resolves relative to the provided dir", async () => {
34
+ await using tmp = await tmpdir()
35
+ const a = path.join(tmp.path, "a")
36
+ const b = path.join(tmp.path, "b")
37
+ const left = path.join(a, "node_modules/biome/index.js")
38
+ const right = path.join(b, "node_modules/biome/index.js")
39
+ await Filesystem.write(left, "export {}\n")
40
+ await Filesystem.write(right, "export {}\n")
41
+ await Filesystem.writeJson(path.join(a, "node_modules/biome/package.json"), {
42
+ name: "biome",
43
+ main: "index.js",
44
+ })
45
+ await Filesystem.writeJson(path.join(b, "node_modules/biome/package.json"), {
46
+ name: "biome",
47
+ main: "index.js",
48
+ })
49
+
50
+ expect(Module.resolve("biome", a)).toBe(left)
51
+ expect(Module.resolve("biome", b)).toBe(right)
52
+ expect(Module.resolve("biome", a)).not.toBe(Module.resolve("biome", b))
53
+ })
54
+
55
+ test("returns undefined when resolution fails", async () => {
56
+ await using tmp = await tmpdir()
57
+ expect(Module.resolve("missing-package", tmp.path)).toBeUndefined()
58
+ })
59
+ })
@@ -0,0 +1,128 @@
1
+ import { describe, expect, test } from "bun:test"
2
+ import fs from "fs/promises"
3
+ import path from "path"
4
+ import { Process } from "../../src/util/process"
5
+ import { tmpdir } from "../fixture/fixture"
6
+
7
+ function node(script: string) {
8
+ return [process.execPath, "-e", script]
9
+ }
10
+
11
+ describe("util.process", () => {
12
+ test("captures stdout and stderr", async () => {
13
+ const out = await Process.run(node('process.stdout.write("out");process.stderr.write("err")'))
14
+ expect(out.code).toBe(0)
15
+ expect(out.stdout.toString()).toBe("out")
16
+ expect(out.stderr.toString()).toBe("err")
17
+ })
18
+
19
+ test("returns code when nothrow is enabled", async () => {
20
+ const out = await Process.run(node("process.exit(7)"), { nothrow: true })
21
+ expect(out.code).toBe(7)
22
+ })
23
+
24
+ test("throws RunFailedError on non-zero exit", async () => {
25
+ const err = await Process.run(node('process.stderr.write("bad");process.exit(3)')).catch((error) => error)
26
+ expect(err).toBeInstanceOf(Process.RunFailedError)
27
+ if (!(err instanceof Process.RunFailedError)) throw err
28
+ expect(err.code).toBe(3)
29
+ expect(err.stderr.toString()).toBe("bad")
30
+ })
31
+
32
+ test("aborts a running process", async () => {
33
+ const abort = new AbortController()
34
+ const started = Date.now()
35
+ setTimeout(() => abort.abort(), 25)
36
+
37
+ const out = await Process.run(node("setInterval(() => {}, 1000)"), {
38
+ abort: abort.signal,
39
+ nothrow: true,
40
+ })
41
+
42
+ expect(out.code).not.toBe(0)
43
+ expect(Date.now() - started).toBeLessThan(1000)
44
+ }, 3000)
45
+
46
+ test("kills after timeout when process ignores terminate signal", async () => {
47
+ if (process.platform === "win32") return
48
+
49
+ const abort = new AbortController()
50
+ const started = Date.now()
51
+ setTimeout(() => abort.abort(), 25)
52
+
53
+ const out = await Process.run(node('process.on("SIGTERM", () => {}); setInterval(() => {}, 1000)'), {
54
+ abort: abort.signal,
55
+ nothrow: true,
56
+ timeout: 25,
57
+ })
58
+
59
+ expect(out.code).not.toBe(0)
60
+ expect(Date.now() - started).toBeLessThan(1000)
61
+ }, 3000)
62
+
63
+ test("uses cwd when spawning commands", async () => {
64
+ await using tmp = await tmpdir()
65
+ const out = await Process.run(node("process.stdout.write(process.cwd())"), {
66
+ cwd: tmp.path,
67
+ })
68
+ expect(out.stdout.toString()).toBe(tmp.path)
69
+ })
70
+
71
+ test("merges environment overrides", async () => {
72
+ const out = await Process.run(node('process.stdout.write(process.env.EPOCHCLI_TEST ?? "")'), {
73
+ env: {
74
+ EPOCHCLI_TEST: "set",
75
+ },
76
+ })
77
+ expect(out.stdout.toString()).toBe("set")
78
+ })
79
+
80
+ test("uses shell in run on Windows", async () => {
81
+ if (process.platform !== "win32") return
82
+
83
+ const out = await Process.run(["set", "EPOCHCLI_TEST_SHELL"], {
84
+ shell: true,
85
+ env: {
86
+ EPOCHCLI_TEST_SHELL: "ok",
87
+ },
88
+ })
89
+
90
+ expect(out.code).toBe(0)
91
+ expect(out.stdout.toString()).toContain("EPOCHCLI_TEST_SHELL=ok")
92
+ })
93
+
94
+ test("runs cmd scripts with spaces on Windows without shell", async () => {
95
+ if (process.platform !== "win32") return
96
+
97
+ await using tmp = await tmpdir()
98
+ const dir = path.join(tmp.path, "with space")
99
+ const file = path.join(dir, "echo cmd.cmd")
100
+
101
+ await fs.mkdir(dir, { recursive: true })
102
+ await Bun.write(file, "@echo off\r\nif %~1==--stdio exit /b 0\r\nexit /b 7\r\n")
103
+
104
+ const proc = Process.spawn([file, "--stdio"], {
105
+ stdin: "pipe",
106
+ stdout: "pipe",
107
+ stderr: "pipe",
108
+ })
109
+
110
+ expect(await proc.exited).toBe(0)
111
+ })
112
+
113
+ test("rejects missing commands without leaking unhandled errors", async () => {
114
+ await using tmp = await tmpdir()
115
+ const cmd = path.join(tmp.path, "missing" + (process.platform === "win32" ? ".cmd" : ""))
116
+ const err = await Process.spawn([cmd], {
117
+ stdin: "pipe",
118
+ stdout: "pipe",
119
+ stderr: "pipe",
120
+ }).exited.catch((err) => err)
121
+
122
+ expect(err).toBeInstanceOf(Error)
123
+ if (!(err instanceof Error)) throw err
124
+ expect(err).toMatchObject({
125
+ code: "ENOENT",
126
+ })
127
+ })
128
+ })
@@ -0,0 +1,104 @@
1
+ import { describe, it, expect, afterAll, beforeAll } from "bun:test"
2
+ import * as fs from "fs/promises"
3
+ import * as path from "path"
4
+ import { SessionAnalyzer } from "../../src/util/session-analyzer"
5
+ import { SessionTelemetry } from "../../src/util/session-telemetry"
6
+
7
+ describe("Telemetry Integration - End-to-End State Extraction", () => {
8
+ const tmpDir = path.join(process.cwd(), "test-telemetry-tmp")
9
+
10
+ beforeAll(async () => {
11
+ await fs.mkdir(tmpDir, { recursive: true })
12
+ // Initialize SessionTelemetry to point to our temp dir
13
+ process.env.XDG_DATA_HOME = tmpDir
14
+ })
15
+
16
+ afterAll(async () => {
17
+ await fs.rm(tmpDir, { recursive: true, force: true })
18
+ })
19
+
20
+ it("successfully extracts completed tools from structured telemetry", async () => {
21
+ // 1. Mock session ID and history
22
+ const sessionID = "test-session-123"
23
+ const chatHistory = [
24
+ {
25
+ info: { role: "assistant" },
26
+ parts: [
27
+ {
28
+ type: "tool",
29
+ tool: "bash",
30
+ state: { status: "completed", input: { command: "spec sc_init --name eventbus" } },
31
+ },
32
+ ],
33
+ },
34
+ ] as any
35
+
36
+ // 2. Emit telemetry via SessionTelemetry
37
+ await SessionTelemetry.init({ dev: true })
38
+ SessionTelemetry.emitModelEvent({
39
+ mainEpochId: sessionID,
40
+ event: "START_GENERATE",
41
+ providerId: "local-main",
42
+ phase: "Phase 1",
43
+ activeAgent: "plan",
44
+ } as any)
45
+
46
+ SessionTelemetry.emitToolEvent({
47
+ sessionID,
48
+ callID: "call_init",
49
+ event: "TOOL_START",
50
+ tool: "bash",
51
+ input: { command: "spec sc_init --name eventbus" },
52
+ } as any)
53
+
54
+ SessionTelemetry.emitToolEvent({
55
+ sessionID,
56
+ callID: "call_init",
57
+ event: "TOOL_END",
58
+ tool: "bash",
59
+ status: "completed",
60
+ output: "Successfully ran: spec sc_init --name eventbus",
61
+ } as any)
62
+
63
+ SessionTelemetry.emitModelEvent({
64
+ mainEpochId: sessionID,
65
+ event: "END_GENERATE",
66
+ metrics: { promptTokens: 100 },
67
+ } as any)
68
+
69
+ await SessionTelemetry.flush()
70
+
71
+ // 3. Analyze via SessionAnalyzer
72
+ const analysis = await SessionAnalyzer.analyze(sessionID, chatHistory, process.cwd())
73
+
74
+ // 4. Verify results
75
+ // Updated expectation: The timeline uses semantically mapped tool names and Turn indexing
76
+ expect(analysis.actionTimeline).toContain(
77
+ 'Turn 1: Tool \'sc_init\' executed (input: {"command":"spec sc_init --name eventbus"}) -> Result: completed',
78
+ )
79
+ })
80
+
81
+ it("handles failed tools correctly", async () => {
82
+ const sessionID = "test-session-456"
83
+ const chatHistory = [] as any
84
+
85
+ await SessionTelemetry.init({ dev: true })
86
+ SessionTelemetry.emitModelEvent({
87
+ mainEpochId: sessionID,
88
+ event: "START_GENERATE",
89
+ } as any)
90
+
91
+ SessionTelemetry.emitToolEvent({
92
+ sessionID,
93
+ event: "TOOL_END",
94
+ tool: "read",
95
+ status: "failed",
96
+ error: "File not found",
97
+ })
98
+
99
+ await SessionTelemetry.flush()
100
+
101
+ const analysis = await SessionAnalyzer.analyze(sessionID, chatHistory, process.cwd())
102
+ expect(analysis.telemetry.mcpxFailures).toContain("read: File not found")
103
+ })
104
+ })
@@ -0,0 +1,21 @@
1
+ import { describe, expect, test } from "bun:test"
2
+ import { withTimeout } from "../../src/util/timeout"
3
+
4
+ describe("util.timeout", () => {
5
+ test("should resolve when promise completes before timeout", async () => {
6
+ const fastPromise = new Promise<string>((resolve) => {
7
+ setTimeout(() => resolve("fast"), 10)
8
+ })
9
+
10
+ const result = await withTimeout(fastPromise, 100)
11
+ expect(result).toBe("fast")
12
+ })
13
+
14
+ test("should reject when promise exceeds timeout", async () => {
15
+ const slowPromise = new Promise<string>((resolve) => {
16
+ setTimeout(() => resolve("slow"), 200)
17
+ })
18
+
19
+ await expect(withTimeout(slowPromise, 50)).rejects.toThrow("Operation timed out after 50ms")
20
+ })
21
+ })
@@ -0,0 +1,100 @@
1
+ import { describe, expect, test } from "bun:test"
2
+ import fs from "fs/promises"
3
+ import path from "path"
4
+ import { which } from "../../src/util/which"
5
+ import { tmpdir } from "../fixture/fixture"
6
+
7
+ async function cmd(dir: string, name: string, exec = true) {
8
+ const ext = process.platform === "win32" ? ".cmd" : ""
9
+ const file = path.join(dir, name + ext)
10
+ const body = process.platform === "win32" ? "@echo off\r\n" : "#!/bin/sh\n"
11
+ await fs.writeFile(file, body)
12
+ if (process.platform !== "win32") {
13
+ await fs.chmod(file, exec ? 0o755 : 0o644)
14
+ }
15
+ return file
16
+ }
17
+
18
+ function env(PATH: string): NodeJS.ProcessEnv {
19
+ return {
20
+ PATH,
21
+ PATHEXT: process.env["PATHEXT"],
22
+ }
23
+ }
24
+
25
+ function envPath(Path: string): NodeJS.ProcessEnv {
26
+ return {
27
+ Path,
28
+ PathExt: process.env["PathExt"] ?? process.env["PATHEXT"],
29
+ }
30
+ }
31
+
32
+ function same(a: string | null, b: string) {
33
+ if (process.platform === "win32") {
34
+ expect(a?.toLowerCase()).toBe(b.toLowerCase())
35
+ return
36
+ }
37
+
38
+ expect(a).toBe(b)
39
+ }
40
+
41
+ describe("util.which", () => {
42
+ test("returns null when command is missing", () => {
43
+ expect(which("epochcli-missing-command-for-test")).toBeNull()
44
+ })
45
+
46
+ test("finds a command from PATH override", async () => {
47
+ await using tmp = await tmpdir()
48
+ const bin = path.join(tmp.path, "bin")
49
+ await fs.mkdir(bin)
50
+ const file = await cmd(bin, "tool")
51
+
52
+ same(which("tool", env(bin)), file)
53
+ })
54
+
55
+ test("uses first PATH match", async () => {
56
+ await using tmp = await tmpdir()
57
+ const a = path.join(tmp.path, "a")
58
+ const b = path.join(tmp.path, "b")
59
+ await fs.mkdir(a)
60
+ await fs.mkdir(b)
61
+ const first = await cmd(a, "dupe")
62
+ await cmd(b, "dupe")
63
+
64
+ same(which("dupe", env([a, b].join(path.delimiter))), first)
65
+ })
66
+
67
+ test("returns null for non-executable file on unix", async () => {
68
+ if (process.platform === "win32") return
69
+
70
+ await using tmp = await tmpdir()
71
+ const bin = path.join(tmp.path, "bin")
72
+ await fs.mkdir(bin)
73
+ await cmd(bin, "noexec", false)
74
+
75
+ expect(which("noexec", env(bin))).toBeNull()
76
+ })
77
+
78
+ test("uses PATHEXT on windows", async () => {
79
+ if (process.platform !== "win32") return
80
+
81
+ await using tmp = await tmpdir()
82
+ const bin = path.join(tmp.path, "bin")
83
+ await fs.mkdir(bin)
84
+ const file = path.join(bin, "pathext.CMD")
85
+ await fs.writeFile(file, "@echo off\r\n")
86
+
87
+ expect(which("pathext", { PATH: bin, PATHEXT: ".CMD" })).toBe(file)
88
+ })
89
+
90
+ test("uses Windows Path casing fallback", async () => {
91
+ if (process.platform !== "win32") return
92
+
93
+ await using tmp = await tmpdir()
94
+ const bin = path.join(tmp.path, "bin")
95
+ await fs.mkdir(bin)
96
+ const file = await cmd(bin, "mixed")
97
+
98
+ same(which("mixed", envPath(bin)), file)
99
+ })
100
+ })
@@ -0,0 +1,90 @@
1
+ import { test, expect } from "bun:test"
2
+ import { Wildcard } from "../../src/util/wildcard"
3
+
4
+ test("match handles glob tokens", () => {
5
+ expect(Wildcard.match("file1.txt", "file?.txt")).toBe(true)
6
+ expect(Wildcard.match("file12.txt", "file?.txt")).toBe(false)
7
+ expect(Wildcard.match("foo+bar", "foo+bar")).toBe(true)
8
+ })
9
+
10
+ test("match with trailing space+wildcard matches command with or without args", () => {
11
+ // "ls *" should match "ls" (no args) and "ls -la" (with args)
12
+ expect(Wildcard.match("ls", "ls *")).toBe(true)
13
+ expect(Wildcard.match("ls -la", "ls *")).toBe(true)
14
+ expect(Wildcard.match("ls foo bar", "ls *")).toBe(true)
15
+
16
+ // "ls*" (no space) should NOT match "ls" alone — wait, it should because .* matches empty
17
+ // but it WILL match "lstmeval" which is the dangerous case users should avoid
18
+ expect(Wildcard.match("ls", "ls*")).toBe(true)
19
+ expect(Wildcard.match("lstmeval", "ls*")).toBe(true)
20
+
21
+ // "ls *" (with space) should NOT match "lstmeval"
22
+ expect(Wildcard.match("lstmeval", "ls *")).toBe(false)
23
+
24
+ // multi-word commands
25
+ expect(Wildcard.match("git status", "git *")).toBe(true)
26
+ expect(Wildcard.match("git", "git *")).toBe(true)
27
+ expect(Wildcard.match("git commit -m foo", "git *")).toBe(true)
28
+ })
29
+
30
+ test("all picks the most specific pattern", () => {
31
+ const rules = {
32
+ "*": "deny",
33
+ "git *": "ask",
34
+ "git status": "allow",
35
+ }
36
+ expect(Wildcard.all("git status", rules)).toBe("allow")
37
+ expect(Wildcard.all("git log", rules)).toBe("ask")
38
+ expect(Wildcard.all("echo hi", rules)).toBe("deny")
39
+ })
40
+
41
+ test("allStructured matches command sequences", () => {
42
+ const rules = {
43
+ "git *": "ask",
44
+ "git status*": "allow",
45
+ }
46
+ expect(Wildcard.allStructured({ head: "git", tail: ["status", "--short"] }, rules)).toBe("allow")
47
+ expect(Wildcard.allStructured({ head: "npm", tail: ["run", "build", "--watch"] }, { "npm run *": "allow" })).toBe(
48
+ "allow",
49
+ )
50
+ expect(Wildcard.allStructured({ head: "ls", tail: ["-la"] }, rules)).toBeUndefined()
51
+ })
52
+
53
+ test("allStructured prioritizes flag-specific patterns", () => {
54
+ const rules = {
55
+ "find *": "allow",
56
+ "find * -delete*": "ask",
57
+ "sort*": "allow",
58
+ "sort -o *": "ask",
59
+ }
60
+ expect(Wildcard.allStructured({ head: "find", tail: ["src", "-delete"] }, rules)).toBe("ask")
61
+ expect(Wildcard.allStructured({ head: "find", tail: ["src", "-print"] }, rules)).toBe("allow")
62
+ expect(Wildcard.allStructured({ head: "sort", tail: ["-o", "out.txt"] }, rules)).toBe("ask")
63
+ expect(Wildcard.allStructured({ head: "sort", tail: ["--reverse"] }, rules)).toBe("allow")
64
+ })
65
+
66
+ test("allStructured handles sed flags", () => {
67
+ const rules = {
68
+ "sed * -i*": "ask",
69
+ "sed -n*": "allow",
70
+ }
71
+ expect(Wildcard.allStructured({ head: "sed", tail: ["-i", "file"] }, rules)).toBe("ask")
72
+ expect(Wildcard.allStructured({ head: "sed", tail: ["-i.bak", "file"] }, rules)).toBe("ask")
73
+ expect(Wildcard.allStructured({ head: "sed", tail: ["-n", "1p", "file"] }, rules)).toBe("allow")
74
+ expect(Wildcard.allStructured({ head: "sed", tail: ["-i", "-n", "/./p", "myfile.txt"] }, rules)).toBe("ask")
75
+ })
76
+
77
+ test("match normalizes slashes for cross-platform globbing", () => {
78
+ expect(Wildcard.match("C:\\Windows\\System32\\*", "C:/Windows/System32/*")).toBe(true)
79
+ expect(Wildcard.match("C:/Windows/System32/drivers", "C:\\Windows\\System32\\*")).toBe(true)
80
+ })
81
+
82
+ test("match handles case-insensitivity on Windows", () => {
83
+ if (process.platform === "win32") {
84
+ expect(Wildcard.match("C:\\windows\\system32\\hosts", "C:/Windows/System32/*")).toBe(true)
85
+ expect(Wildcard.match("c:/windows/system32/hosts", "C:\\Windows\\System32\\*")).toBe(true)
86
+ } else {
87
+ // Unix paths are case-sensitive
88
+ expect(Wildcard.match("/users/test/file", "/Users/test/*")).toBe(false)
89
+ }
90
+ })
package/test-regex.js ADDED
@@ -0,0 +1,50 @@
1
+ const content = `expert_dev_guidance:
2
+
3
+ - rule:
4
+ id: TS-001
5
+ Trigger: Whenever iterating over an array structure or collection of data.
6
+ Behaviour: Never use for...in on arrays. Always use for...of to iterate over values safely, or utilize standard functional methods like .map() and .filter().
7
+ Example:
8
+ Correct:
9
+ <|">
10
+ for (const item of items) {
11
+ process(item);
12
+ }
13
+ <|">
14
+ Incorrect:
15
+ <|">
16
+ for (const i in items) {
17
+ process(items[i]);
18
+ }
19
+ <|">
20
+
21
+
22
+ - rule:
23
+ id: TS-002
24
+ Trigger: Writing functions that handle external I/O, file system parsing, or non-deterministic API calls.
25
+ Behaviour: Do not use throw new Error(). Implement a Discriminated Union Result type returning { success: true, data: T } or { success: false, error: Error } to enforce compiler error checking.
26
+ Example:
27
+ Correct:
28
+ <|">
29
+ type Result<T> = { ok: true, val: T } | { ok: false, err: Error };
30
+ return parseSafe(data);
31
+ <|">
32
+ Incorrect:
33
+ <|">
34
+ try {
35
+ return parse(data);
36
+ } catch (e) {
37
+ throw new Error(e);
38
+ }
39
+ <|">`
40
+
41
+ const devRulesRegex =
42
+ /- rule:\s*id:\s*([^\n]+)\s*Trigger:\s*([^\n]+)\s*Behaviour:\s*([^\n]+)\s*Example:([\s\S]*?)(?=- rule:|$)/g
43
+ let match
44
+ while ((match = devRulesRegex.exec(content)) !== null) {
45
+ console.log("ID:", match[1].trim())
46
+ console.log("Trigger:", match[2].trim())
47
+ console.log("Behaviour:", match[3].trim())
48
+ console.log("Example:", match[4].trim())
49
+ console.log("---")
50
+ }