eve 0.7.4 → 0.8.1

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 (337) hide show
  1. package/CHANGELOG.md +45 -0
  2. package/README.md +11 -11
  3. package/dist/src/chunks/{use-eve-agent-DErQj5hs.js → use-eve-agent-BSXZSn-R.js} +6 -1
  4. package/dist/src/chunks/{use-eve-agent-DoR8C4i6.js → use-eve-agent-Dlut2Qzt.js} +6 -1
  5. package/dist/src/cli/commands/init.js +1 -1
  6. package/dist/src/cli/dev/tui/blocks.d.ts +1 -1
  7. package/dist/src/cli/dev/tui/blocks.js +4 -3
  8. package/dist/src/cli/dev/tui/command-typeahead.d.ts +13 -3
  9. package/dist/src/cli/dev/tui/command-typeahead.js +1 -1
  10. package/dist/src/cli/dev/tui/line-editor.d.ts +6 -0
  11. package/dist/src/cli/dev/tui/line-editor.js +1 -1
  12. package/dist/src/cli/dev/tui/prompt-command-handler.d.ts +2 -0
  13. package/dist/src/cli/dev/tui/prompt-command-handler.js +1 -1
  14. package/dist/src/cli/dev/tui/runner.d.ts +15 -0
  15. package/dist/src/cli/dev/tui/runner.js +1 -1
  16. package/dist/src/cli/dev/tui/setup-commands.d.ts +11 -7
  17. package/dist/src/cli/dev/tui/setup-commands.js +2 -2
  18. package/dist/src/cli/dev/tui/setup-flow.d.ts +72 -10
  19. package/dist/src/cli/dev/tui/setup-issues.js +1 -1
  20. package/dist/src/cli/dev/tui/setup-panel.d.ts +91 -39
  21. package/dist/src/cli/dev/tui/setup-panel.js +3 -1
  22. package/dist/src/cli/dev/tui/setup-selection-input.d.ts +48 -0
  23. package/dist/src/cli/dev/tui/setup-selection-input.js +1 -0
  24. package/dist/src/cli/dev/tui/status-line.d.ts +13 -7
  25. package/dist/src/cli/dev/tui/status-line.js +1 -1
  26. package/dist/src/cli/dev/tui/stream-format.d.ts +2 -0
  27. package/dist/src/cli/dev/tui/stream-format.js +2 -2
  28. package/dist/src/cli/dev/tui/terminal-renderer.d.ts +5 -2
  29. package/dist/src/cli/dev/tui/terminal-renderer.js +6 -5
  30. package/dist/src/cli/dev/tui/theme.d.ts +3 -1
  31. package/dist/src/cli/dev/tui/theme.js +1 -1
  32. package/dist/src/cli/dev/tui/tui-prompter.js +1 -1
  33. package/dist/src/cli/dev/tui/tui.d.ts +6 -0
  34. package/dist/src/cli/dev/tui/tui.js +1 -1
  35. package/dist/src/cli/dev/tui/vercel-status.d.ts +4 -4
  36. package/dist/src/cli/dev/tui/vercel-status.js +1 -1
  37. package/dist/src/cli/run.d.ts +4 -3
  38. package/dist/src/cli/run.js +2 -2
  39. package/dist/src/client/client.js +1 -1
  40. package/dist/src/client/session-utils.d.ts +1 -0
  41. package/dist/src/client/session-utils.js +1 -1
  42. package/dist/src/client/session.d.ts +1 -0
  43. package/dist/src/client/session.js +1 -1
  44. package/dist/src/client/types.d.ts +19 -0
  45. package/dist/src/compiled/.vendor-stamp.json +1 -2
  46. package/dist/src/compiler/compile-from-memory.js +1 -1
  47. package/dist/src/compiler/manifest.d.ts +28 -7
  48. package/dist/src/compiler/manifest.js +1 -1
  49. package/dist/src/compiler/normalize-agent-config.js +1 -1
  50. package/dist/src/compiler/normalize-sandbox.js +1 -1
  51. package/dist/src/evals/cli/eval-client.js +1 -1
  52. package/dist/src/execution/sandbox/bash-tool.js +1 -1
  53. package/dist/src/execution/sandbox/bindings/docker-base-setup.d.ts +9 -0
  54. package/dist/src/execution/sandbox/bindings/docker-base-setup.js +2 -0
  55. package/dist/src/execution/sandbox/bindings/docker-cli.d.ts +77 -0
  56. package/dist/src/execution/sandbox/bindings/docker-cli.js +2 -0
  57. package/dist/src/execution/sandbox/bindings/docker-container.d.ts +16 -0
  58. package/dist/src/execution/sandbox/bindings/docker-container.js +1 -0
  59. package/dist/src/execution/sandbox/bindings/docker-network.d.ts +8 -0
  60. package/dist/src/execution/sandbox/bindings/docker-network.js +1 -0
  61. package/dist/src/execution/sandbox/bindings/docker-options.d.ts +21 -0
  62. package/dist/src/execution/sandbox/bindings/docker-options.js +1 -0
  63. package/dist/src/execution/sandbox/bindings/docker-session.d.ts +7 -0
  64. package/dist/src/execution/sandbox/bindings/docker-session.js +2 -0
  65. package/dist/src/execution/sandbox/bindings/docker-templates.d.ts +35 -0
  66. package/dist/src/execution/sandbox/bindings/docker-templates.js +1 -0
  67. package/dist/src/execution/sandbox/bindings/docker-utils.d.ts +2 -0
  68. package/dist/src/execution/sandbox/bindings/docker-utils.js +1 -0
  69. package/dist/src/execution/sandbox/bindings/docker.d.ts +33 -0
  70. package/dist/src/execution/sandbox/bindings/docker.js +1 -0
  71. package/dist/src/execution/sandbox/bindings/just-bash-runtime.d.ts +30 -0
  72. package/dist/src/execution/sandbox/bindings/just-bash-runtime.js +1 -0
  73. package/dist/src/execution/sandbox/bindings/just-bash.d.ts +33 -0
  74. package/dist/src/execution/sandbox/bindings/just-bash.js +1 -0
  75. package/dist/src/execution/sandbox/bindings/local-backend-utils.d.ts +23 -0
  76. package/dist/src/execution/sandbox/bindings/local-backend-utils.js +1 -0
  77. package/dist/src/execution/sandbox/bindings/local-template-prune.d.ts +16 -0
  78. package/dist/src/execution/sandbox/bindings/local-template-prune.js +1 -0
  79. package/dist/src/execution/sandbox/bindings/local.d.ts +16 -25
  80. package/dist/src/execution/sandbox/bindings/local.js +1 -1
  81. package/dist/src/execution/sandbox/bindings/microsandbox-lifecycle.d.ts +16 -0
  82. package/dist/src/execution/sandbox/bindings/microsandbox-lifecycle.js +1 -0
  83. package/dist/src/execution/sandbox/bindings/microsandbox-metadata.d.ts +21 -0
  84. package/dist/src/execution/sandbox/bindings/microsandbox-metadata.js +1 -0
  85. package/dist/src/execution/sandbox/bindings/microsandbox-network.d.ts +18 -0
  86. package/dist/src/execution/sandbox/bindings/microsandbox-network.js +1 -0
  87. package/dist/src/execution/sandbox/bindings/microsandbox-options.d.ts +34 -0
  88. package/dist/src/execution/sandbox/bindings/microsandbox-options.js +1 -0
  89. package/dist/src/execution/sandbox/bindings/microsandbox-platform.d.ts +22 -0
  90. package/dist/src/execution/sandbox/bindings/microsandbox-platform.js +178 -0
  91. package/dist/src/execution/sandbox/bindings/microsandbox-process.d.ts +3 -0
  92. package/dist/src/execution/sandbox/bindings/microsandbox-process.js +1 -0
  93. package/dist/src/execution/sandbox/bindings/microsandbox-runtime.d.ts +75 -0
  94. package/dist/src/execution/sandbox/bindings/microsandbox-runtime.js +1 -0
  95. package/dist/src/execution/sandbox/bindings/microsandbox-templates.d.ts +13 -0
  96. package/dist/src/execution/sandbox/bindings/microsandbox-templates.js +1 -0
  97. package/dist/src/execution/sandbox/bindings/microsandbox.d.ts +22 -0
  98. package/dist/src/execution/sandbox/bindings/microsandbox.js +1 -0
  99. package/dist/src/execution/sandbox/bindings/vercel-create-api.d.ts +20 -0
  100. package/dist/src/execution/sandbox/bindings/vercel-create-api.js +1 -0
  101. package/dist/src/execution/sandbox/bindings/vercel.d.ts +7 -8
  102. package/dist/src/execution/sandbox/bindings/vercel.js +1 -1
  103. package/dist/src/execution/sandbox/development-cleanup.d.ts +4 -0
  104. package/dist/src/execution/sandbox/development-cleanup.js +1 -0
  105. package/dist/src/execution/sandbox/development-prewarm.d.ts +15 -0
  106. package/dist/src/execution/sandbox/development-prewarm.js +1 -0
  107. package/dist/src/execution/sandbox/development-run.d.ts +8 -0
  108. package/dist/src/execution/sandbox/development-run.js +1 -0
  109. package/dist/src/execution/sandbox/ensure.js +1 -1
  110. package/dist/src/execution/sandbox/lazy-backend.d.ts +1 -1
  111. package/dist/src/execution/sandbox/logging-session.d.ts +5 -0
  112. package/dist/src/execution/sandbox/logging-session.js +1 -0
  113. package/dist/src/execution/sandbox/prewarm.js +1 -1
  114. package/dist/src/execution/sandbox/template-prewarm-lock.d.ts +8 -0
  115. package/dist/src/execution/sandbox/template-prewarm-lock.js +1 -0
  116. package/dist/src/internal/application/optional-package-install.d.ts +40 -0
  117. package/dist/src/internal/application/optional-package-install.js +1 -0
  118. package/dist/src/internal/application/package.js +1 -1
  119. package/dist/src/internal/authored-definition/sandbox.d.ts +1 -1
  120. package/dist/src/internal/authored-definition/sandbox.js +1 -1
  121. package/dist/src/internal/authored-module-loader.js +2 -2
  122. package/dist/src/internal/classify-model-routing.d.ts +24 -0
  123. package/dist/src/internal/classify-model-routing.js +1 -0
  124. package/dist/src/internal/nitro/host/channel-routes.js +2 -2
  125. package/dist/src/internal/nitro/host/compiled-sandbox-backend-prune-plugin.d.ts +5 -2
  126. package/dist/src/internal/nitro/host/compiled-sandbox-backend-prune-plugin.js +1 -1
  127. package/dist/src/internal/nitro/host/create-application-nitro.d.ts +10 -0
  128. package/dist/src/internal/nitro/host/create-application-nitro.js +1 -1
  129. package/dist/src/internal/nitro/host/dev-authored-source-watcher.js +1 -1
  130. package/dist/src/internal/nitro/host/dev-live-virtual-modules.d.ts +15 -0
  131. package/dist/src/internal/nitro/host/dev-live-virtual-modules.js +1 -0
  132. package/dist/src/internal/nitro/host/optional-engine-dependency-plugin.d.ts +28 -0
  133. package/dist/src/internal/nitro/host/optional-engine-dependency-plugin.js +1 -0
  134. package/dist/src/internal/nitro/host/start-development-server.js +1 -1
  135. package/dist/src/internal/nitro/routes/agent-info/build-agent-info-response-from-manifest.d.ts +2 -0
  136. package/dist/src/internal/nitro/routes/agent-info/build-agent-info-response-from-manifest.js +1 -1
  137. package/dist/src/internal/nitro/routes/agent-info/build-agent-info-response.d.ts +4 -0
  138. package/dist/src/internal/nitro/routes/info.js +1 -1
  139. package/dist/src/internal/nitro/routes/runtime-artifacts.js +1 -1
  140. package/dist/src/internal/resolve-model-endpoint-status.d.ts +23 -0
  141. package/dist/src/internal/resolve-model-endpoint-status.js +1 -0
  142. package/dist/src/internal/workflow-bundle/builder-support.js +2 -2
  143. package/dist/src/internal/workflow-bundle/vercel-workflow-output.d.ts +1 -1
  144. package/dist/src/internal/workflow-bundle/vercel-workflow-output.js +1 -1
  145. package/dist/src/public/definitions/sandbox-backend.js +1 -1
  146. package/dist/src/public/sandbox/backends/default.d.ts +37 -10
  147. package/dist/src/public/sandbox/backends/default.js +1 -1
  148. package/dist/src/public/sandbox/backends/docker.d.ts +14 -0
  149. package/dist/src/public/sandbox/backends/docker.js +1 -0
  150. package/dist/src/public/sandbox/backends/just-bash.d.ts +17 -0
  151. package/dist/src/public/sandbox/backends/just-bash.js +1 -0
  152. package/dist/src/public/sandbox/backends/microsandbox.d.ts +17 -0
  153. package/dist/src/public/sandbox/backends/microsandbox.js +1 -0
  154. package/dist/src/public/sandbox/backends/vercel.d.ts +5 -2
  155. package/dist/src/public/sandbox/backends/vercel.js +1 -1
  156. package/dist/src/public/sandbox/docker-sandbox.d.ts +45 -0
  157. package/dist/src/public/sandbox/index.d.ts +8 -4
  158. package/dist/src/public/sandbox/index.js +1 -1
  159. package/dist/src/public/sandbox/just-bash-sandbox.d.ts +17 -0
  160. package/dist/src/public/sandbox/just-bash-sandbox.js +1 -0
  161. package/dist/src/public/sandbox/microsandbox-sandbox.d.ts +54 -0
  162. package/dist/src/public/sandbox/microsandbox-sandbox.js +1 -0
  163. package/dist/src/public/sandbox/vercel-sandbox.d.ts +1 -1
  164. package/dist/src/runtime/compiled-artifacts-source.d.ts +11 -0
  165. package/dist/src/runtime/compiled-artifacts-source.js +1 -1
  166. package/dist/src/runtime/resolve-sandbox.js +1 -1
  167. package/dist/src/runtime/sandbox/keys.js +1 -1
  168. package/dist/src/runtime/sandbox/registry.d.ts +2 -2
  169. package/dist/src/runtime/types.d.ts +1 -1
  170. package/dist/src/runtime/workspace/spec.js +1 -1
  171. package/dist/src/setup/boxes/add-channels.d.ts +3 -2
  172. package/dist/src/setup/boxes/add-channels.js +2 -2
  173. package/dist/src/setup/boxes/apply-ai-gateway-credential.js +1 -1
  174. package/dist/src/setup/boxes/deploy-project.js +1 -1
  175. package/dist/src/setup/boxes/link-project.js +1 -1
  176. package/dist/src/setup/boxes/resolve-provisioning.d.ts +6 -0
  177. package/dist/src/setup/boxes/resolve-provisioning.js +1 -1
  178. package/dist/src/setup/boxes/select-model.d.ts +2 -2
  179. package/dist/src/setup/boxes/select-model.js +1 -1
  180. package/dist/src/setup/cli/channel-setup-prompter.d.ts +28 -0
  181. package/dist/src/setup/cli/channel-setup-prompter.js +1 -1
  182. package/dist/src/setup/cli/index.d.ts +2 -2
  183. package/dist/src/setup/cli/index.js +1 -1
  184. package/dist/src/setup/cli/option-row.d.ts +89 -0
  185. package/dist/src/setup/cli/option-row.js +1 -0
  186. package/dist/src/setup/cli/prompt-ui.d.ts +15 -29
  187. package/dist/src/setup/cli/prompt-ui.js +2 -2
  188. package/dist/src/setup/cli/select-component.d.ts +4 -3
  189. package/dist/src/setup/cli/select-component.js +1 -1
  190. package/dist/src/setup/cli/select-state.d.ts +3 -3
  191. package/dist/src/setup/cli/select-state.js +1 -1
  192. package/dist/src/setup/flows/channels.d.ts +29 -14
  193. package/dist/src/setup/flows/channels.js +1 -1
  194. package/dist/src/setup/flows/deploy.d.ts +1 -0
  195. package/dist/src/setup/flows/deploy.js +1 -1
  196. package/dist/src/setup/flows/link.d.ts +1 -0
  197. package/dist/src/setup/flows/link.js +1 -1
  198. package/dist/src/setup/flows/model.d.ts +30 -6
  199. package/dist/src/setup/flows/model.js +1 -1
  200. package/dist/src/setup/flows/vercel.d.ts +4 -1
  201. package/dist/src/setup/flows/vercel.js +2 -2
  202. package/dist/src/setup/index.js +1 -1
  203. package/dist/src/setup/primitives/open-url.d.ts +14 -0
  204. package/dist/src/setup/primitives/open-url.js +1 -0
  205. package/dist/src/setup/primitives/pm/pnpm.d.ts +9 -0
  206. package/dist/src/setup/primitives/pm/pnpm.js +2 -2
  207. package/dist/src/setup/primitives/pm/run.d.ts +12 -4
  208. package/dist/src/setup/primitives/pm/run.js +1 -1
  209. package/dist/src/setup/primitives/pm/types.d.ts +3 -0
  210. package/dist/src/setup/primitives/process-abort.d.ts +7 -0
  211. package/dist/src/setup/primitives/process-abort.js +1 -0
  212. package/dist/src/setup/primitives/run-vercel.d.ts +2 -0
  213. package/dist/src/setup/primitives/run-vercel.js +1 -1
  214. package/dist/src/setup/project-resolution.d.ts +6 -3
  215. package/dist/src/setup/project-resolution.js +1 -1
  216. package/dist/src/setup/prompter.d.ts +63 -6
  217. package/dist/src/setup/prompter.js +1 -1
  218. package/dist/src/setup/run-vercel-link.d.ts +1 -1
  219. package/dist/src/setup/run-vercel-link.js +1 -1
  220. package/dist/src/setup/runner.d.ts +2 -0
  221. package/dist/src/setup/runner.js +1 -1
  222. package/dist/src/setup/scaffold/channels-catalog.d.ts +2 -0
  223. package/dist/src/setup/scaffold/channels-catalog.js +1 -1
  224. package/dist/src/setup/scaffold/create/add-to-project.d.ts +1 -0
  225. package/dist/src/setup/scaffold/create/add-to-project.js +1 -1
  226. package/dist/src/setup/scaffold/create/project.d.ts +2 -0
  227. package/dist/src/setup/scaffold/create/project.js +6 -3
  228. package/dist/src/setup/scaffold/update/channels.d.ts +2 -0
  229. package/dist/src/setup/scaffold/update/channels.js +3 -3
  230. package/dist/src/setup/slack-connect-lifecycle.d.ts +97 -0
  231. package/dist/src/setup/slack-connect-lifecycle.js +1 -0
  232. package/dist/src/setup/slack-connect.d.ts +51 -0
  233. package/dist/src/setup/slack-connect.js +1 -0
  234. package/dist/src/setup/slackbot.d.ts +60 -47
  235. package/dist/src/setup/slackbot.js +1 -1
  236. package/dist/src/setup/state.d.ts +2 -1
  237. package/dist/src/setup/state.js +1 -1
  238. package/dist/src/setup/step.d.ts +4 -0
  239. package/dist/src/setup/validate-gateway-key.d.ts +30 -0
  240. package/dist/src/setup/validate-gateway-key.js +1 -0
  241. package/dist/src/setup/vercel-project.d.ts +19 -12
  242. package/dist/src/setup/vercel-project.js +1 -1
  243. package/dist/src/shared/agent-definition.d.ts +26 -0
  244. package/dist/src/shared/model-endpoint-status.d.ts +27 -0
  245. package/dist/src/shared/model-endpoint-status.js +1 -0
  246. package/dist/src/shared/sandbox-backend.d.ts +8 -2
  247. package/dist/src/shared/sandbox-definition.d.ts +4 -3
  248. package/dist/src/shared/sandbox-network-policy.d.ts +4 -2
  249. package/dist/src/shared/sandbox-session.d.ts +3 -2
  250. package/dist/src/svelte/index.js +1 -1
  251. package/dist/src/svelte/use-eve-agent.js +1 -1
  252. package/dist/src/vue/index.js +1 -1
  253. package/dist/src/vue/use-eve-agent.js +1 -1
  254. package/{dist/docs/public → docs}/getting-started.mdx +14 -4
  255. package/{dist/docs/public → docs}/guides/deployment.md +3 -3
  256. package/{dist/docs/public → docs}/guides/dev-tui.md +2 -2
  257. package/{dist/docs/public → docs}/reference/cli.md +2 -1
  258. package/{dist/docs/public → docs}/sandbox.mdx +39 -15
  259. package/package.json +11 -1
  260. package/dist/src/compiled/just-bash/LICENSE +0 -201
  261. package/dist/src/compiled/just-bash/index.d.ts +0 -139
  262. package/dist/src/compiled/just-bash/index.js +0 -2200
  263. package/dist/src/compiled/just-bash/network/types.d.ts +0 -155
  264. package/dist/src/public/sandbox/backends/local.d.ts +0 -16
  265. package/dist/src/public/sandbox/backends/local.js +0 -1
  266. package/dist/src/public/sandbox/local-sandbox.d.ts +0 -7
  267. /package/dist/src/public/sandbox/{local-sandbox.js → docker-sandbox.js} +0 -0
  268. /package/{dist/docs/public → docs}/README.md +0 -0
  269. /package/{dist/docs/public → docs}/agent-config.md +0 -0
  270. /package/{dist/docs/public → docs}/channels/custom.mdx +0 -0
  271. /package/{dist/docs/public → docs}/channels/discord.mdx +0 -0
  272. /package/{dist/docs/public → docs}/channels/eve.mdx +0 -0
  273. /package/{dist/docs/public → docs}/channels/github.mdx +0 -0
  274. /package/{dist/docs/public → docs}/channels/linear.mdx +0 -0
  275. /package/{dist/docs/public → docs}/channels/meta.json +0 -0
  276. /package/{dist/docs/public → docs}/channels/overview.mdx +0 -0
  277. /package/{dist/docs/public → docs}/channels/slack.mdx +0 -0
  278. /package/{dist/docs/public → docs}/channels/teams.mdx +0 -0
  279. /package/{dist/docs/public → docs}/channels/telegram.mdx +0 -0
  280. /package/{dist/docs/public → docs}/channels/twilio.mdx +0 -0
  281. /package/{dist/docs/public → docs}/concepts/context-control.md +0 -0
  282. /package/{dist/docs/public → docs}/concepts/default-harness.md +0 -0
  283. /package/{dist/docs/public → docs}/concepts/execution-model-and-durability.md +0 -0
  284. /package/{dist/docs/public → docs}/concepts/meta.json +0 -0
  285. /package/{dist/docs/public → docs}/concepts/security-model.md +0 -0
  286. /package/{dist/docs/public → docs}/concepts/sessions-runs-and-streaming.md +0 -0
  287. /package/{dist/docs/public → docs}/connections.mdx +0 -0
  288. /package/{dist/docs/public → docs}/evals/assertions.mdx +0 -0
  289. /package/{dist/docs/public → docs}/evals/cases.mdx +0 -0
  290. /package/{dist/docs/public → docs}/evals/judge.mdx +0 -0
  291. /package/{dist/docs/public → docs}/evals/meta.json +0 -0
  292. /package/{dist/docs/public → docs}/evals/overview.mdx +0 -0
  293. /package/{dist/docs/public → docs}/evals/reporters.mdx +0 -0
  294. /package/{dist/docs/public → docs}/evals/running.mdx +0 -0
  295. /package/{dist/docs/public → docs}/evals/targets.mdx +0 -0
  296. /package/{dist/docs/public → docs}/guides/auth-and-route-protection.md +0 -0
  297. /package/{dist/docs/public → docs}/guides/client/continuations.mdx +0 -0
  298. /package/{dist/docs/public → docs}/guides/client/messages.mdx +0 -0
  299. /package/{dist/docs/public → docs}/guides/client/meta.json +0 -0
  300. /package/{dist/docs/public → docs}/guides/client/output-schema.mdx +0 -0
  301. /package/{dist/docs/public → docs}/guides/client/overview.mdx +0 -0
  302. /package/{dist/docs/public → docs}/guides/client/streaming.mdx +0 -0
  303. /package/{dist/docs/public → docs}/guides/dynamic-capabilities.md +0 -0
  304. /package/{dist/docs/public → docs}/guides/dynamic-workflows.md +0 -0
  305. /package/{dist/docs/public → docs}/guides/frontend/meta.json +0 -0
  306. /package/{dist/docs/public → docs}/guides/frontend/nextjs.mdx +0 -0
  307. /package/{dist/docs/public → docs}/guides/frontend/nuxt.mdx +0 -0
  308. /package/{dist/docs/public → docs}/guides/frontend/overview.mdx +0 -0
  309. /package/{dist/docs/public → docs}/guides/frontend/sveltekit.mdx +0 -0
  310. /package/{dist/docs/public → docs}/guides/frontend/use-eve-agent-svelte.mdx +0 -0
  311. /package/{dist/docs/public → docs}/guides/frontend/use-eve-agent-vue.mdx +0 -0
  312. /package/{dist/docs/public → docs}/guides/hooks.md +0 -0
  313. /package/{dist/docs/public → docs}/guides/instrumentation.md +0 -0
  314. /package/{dist/docs/public → docs}/guides/meta.json +0 -0
  315. /package/{dist/docs/public → docs}/guides/remote-agents.md +0 -0
  316. /package/{dist/docs/public → docs}/guides/session-context.md +0 -0
  317. /package/{dist/docs/public → docs}/guides/state.md +0 -0
  318. /package/{dist/docs/public → docs}/instructions.mdx +0 -0
  319. /package/{dist/docs/public → docs}/introduction.md +0 -0
  320. /package/{dist/docs/public → docs}/meta.json +0 -0
  321. /package/{dist/docs/public → docs}/reference/meta.json +0 -0
  322. /package/{dist/docs/public → docs}/reference/project-layout.md +0 -0
  323. /package/{dist/docs/public → docs}/reference/typescript-api.md +0 -0
  324. /package/{dist/docs/public → docs}/schedules.mdx +0 -0
  325. /package/{dist/docs/public → docs}/skills.mdx +0 -0
  326. /package/{dist/docs/public → docs}/subagents.mdx +0 -0
  327. /package/{dist/docs/public → docs}/tools.mdx +0 -0
  328. /package/{dist/docs/public → docs}/tutorial/connect-a-warehouse.mdx +0 -0
  329. /package/{dist/docs/public → docs}/tutorial/first-agent.mdx +0 -0
  330. /package/{dist/docs/public → docs}/tutorial/guard-the-spend.mdx +0 -0
  331. /package/{dist/docs/public → docs}/tutorial/how-it-runs.mdx +0 -0
  332. /package/{dist/docs/public → docs}/tutorial/meta.json +0 -0
  333. /package/{dist/docs/public → docs}/tutorial/query-sample-data.mdx +0 -0
  334. /package/{dist/docs/public → docs}/tutorial/remember-definitions.mdx +0 -0
  335. /package/{dist/docs/public → docs}/tutorial/run-analysis.mdx +0 -0
  336. /package/{dist/docs/public → docs}/tutorial/ship-it.mdx +0 -0
  337. /package/{dist/docs/public → docs}/tutorial/team-playbooks.mdx +0 -0
@@ -1,3 +1,4 @@
1
+ import { type ChannelSetupAwaitChoice } from "#setup/cli/index.js";
1
2
  import { type RailSpinner } from "#setup/cli/index.js";
2
3
  /**
3
4
  * Clack constrains option values to readonly primitives. Our scaffold flow
@@ -9,14 +10,30 @@ export interface SelectOption<T extends PrompterValue> {
9
10
  value: T;
10
11
  label: string;
11
12
  hint?: string;
13
+ /** Short inline annotation shown dimmed only while the cursor is on this row. */
14
+ focusHint?: string;
12
15
  /**
13
16
  * Longer, display-only explanation shown dimmed alongside the option while it
14
17
  * is highlighted during navigation. Hidden once a choice is submitted.
15
18
  */
16
19
  description?: string;
20
+ /** Cursor-pointer/active-label accent; "warning" turns them yellow for an attention row. */
21
+ accent?: "warning";
17
22
  disabled?: boolean;
18
23
  /** Parenthetical shown after a disabled option's label explaining why. */
19
24
  disabledReason?: string;
25
+ /**
26
+ * "warning" renders the disabled reason in yellow with a dimmed (not struck)
27
+ * label: the row is unavailable here but actionable elsewhere (e.g. a channel
28
+ * that needs a Vercel account points at /model), unlike the default disabled
29
+ * styling, which marks a hard conflict.
30
+ */
31
+ disabledReasonTone?: "warning";
32
+ /**
33
+ * Completed work: renders with a check and remains cursor-addressable for
34
+ * contextual feedback, but cannot be selected or toggled.
35
+ */
36
+ completed?: boolean;
20
37
  /**
21
38
  * Marks a mandatory row that is always selected and cannot be toggled off; the
22
39
  * cursor skips it and it renders a dimmed check. Mutually exclusive with
@@ -39,7 +56,7 @@ export interface SelectOption<T extends PrompterValue> {
39
56
  * prints it to scrollback before the prompt.
40
57
  */
41
58
  export interface SelectNotice {
42
- tone: "success" | "info" | "warning";
59
+ tone: "success" | "info" | "warning" | "error";
43
60
  text: string;
44
61
  }
45
62
  /** Options common to every {@link Prompter.select} call. */
@@ -60,12 +77,14 @@ export interface SelectCommonOptions<T extends PrompterValue> {
60
77
  */
61
78
  required?: boolean;
62
79
  /**
63
- * "stacked" renders each option's hint on its own line below the label with
64
- * a blank line between options for small action menus whose hints carry
65
- * current values. A TUI-panel affordance: the CLI prompter keeps its inline
66
- * hint rendering.
80
+ * How option hints are laid out in the dev TUI panel (the CLI prompter ignores
81
+ * it and keeps its default inline, unnumbered rendering). "stacked" renders
82
+ * each hint on its own line below the label with a blank line between options —
83
+ * for small action menus whose hints carry current values. "inline" keeps hints
84
+ * on the label row, suppresses numeric shortcuts, and separates the trailing
85
+ * completion action (e.g. the `/channels` task list).
67
86
  */
68
- layout?: "stacked";
87
+ hintLayout?: "stacked" | "inline";
69
88
  /** Outcome lines from earlier laps of a looping menu. */
70
89
  notices?: readonly SelectNotice[];
71
90
  }
@@ -81,6 +100,24 @@ export interface MultiSelectOptions<T extends PrompterValue> extends SelectCommo
81
100
  /** Pre-mark these values as selected. */
82
101
  initialValues?: T[];
83
102
  }
103
+ /** Result of a single-select whose one row can be edited inline. */
104
+ export type EditableSelectResult<T extends PrompterValue> = {
105
+ kind: "selected";
106
+ value: T;
107
+ } | {
108
+ kind: "edited";
109
+ value: T;
110
+ text: string;
111
+ };
112
+ /** Inline-edit behavior for one row in an otherwise ordinary single-select. */
113
+ export interface EditableSelectOptions<T extends PrompterValue> extends SingleSelectOptions<T> {
114
+ editable: {
115
+ value: T;
116
+ defaultValue: string;
117
+ formatHint: (value: string) => string;
118
+ validate?: (value: string) => string | undefined;
119
+ };
120
+ }
84
121
  /** Color intent for {@link Prompter.note}: red warning (default) or green success. */
85
122
  export type NoteTone = "warning" | "success";
86
123
  /** Input for {@link Prompter.acknowledge}: a heading plus optional body lines. */
@@ -94,6 +131,13 @@ export interface Prompter {
94
131
  placeholder?: string;
95
132
  defaultValue?: string;
96
133
  validate?: (value: string) => string | undefined;
134
+ /**
135
+ * Context lines shown with the question (e.g. why it is being re-asked).
136
+ * They live and die with the question: the TUI paints them inside the
137
+ * question panel so they vanish on submit, the CLI prints them to
138
+ * scrollback above the prompt.
139
+ */
140
+ notices?: readonly SelectNotice[];
97
141
  }): Promise<string>;
98
142
  password(opts: {
99
143
  message: string;
@@ -108,6 +152,13 @@ export interface Prompter {
108
152
  */
109
153
  select<T extends PrompterValue>(opts: SingleSelectOptions<T>): Promise<T>;
110
154
  select<T extends PrompterValue>(opts: MultiSelectOptions<T>): Promise<T[]>;
155
+ /**
156
+ * TUI enhancement for a select row whose secondary value becomes an inline
157
+ * editor while focused. Typing and editing keys update it directly. Optional
158
+ * so non-repainting prompt implementations can keep the ordinary
159
+ * select-then-text fallback.
160
+ */
161
+ selectEditable?<T extends PrompterValue>(opts: EditableSelectOptions<T>): Promise<EditableSelectResult<T>>;
111
162
  /**
112
163
  * Static instructions the user dismisses before the flow moves on. The TUI
113
164
  * renders them in the question slot of the flow panel (the text takes the
@@ -116,6 +167,12 @@ export interface Prompter {
116
167
  * lightweight test fakes can omit it — flows fall back to {@link note}.
117
168
  */
118
169
  acknowledge?(opts: AcknowledgeOptions): Promise<void>;
170
+ /**
171
+ * Presents actions while a setup operation continues in the background.
172
+ * The TUI implements it; plain and headless prompters omit it, so callers
173
+ * fall back to waiting without concurrent controls.
174
+ */
175
+ awaitChoice?: ChannelSetupAwaitChoice;
119
176
  /**
120
177
  * Rail-attached notice, no bullet — reads as a follow-up to the previous
121
178
  * step. Red by default (warnings, collisions); pass `tone: "success"` for a
@@ -1 +1 @@
1
- import{__toESM}from"../_virtual/_rolldown/runtime.js";import{require_picocolors}from"../node_modules/.pnpm/picocolors@1.1.1/node_modules/picocolors/picocolors.js";import{h,ht,q}from"../node_modules/.pnpm/@clack_core@1.3.1/node_modules/@clack/core/dist/index.js";import{WizardCancelledError}from"./step.js";import{initialQuitGuardState,quitHintNote,reduceQuitGuard}from"./quit-guard.js";import{cornerFor,createRailLog,formatPromptCancellation,formatPromptHeader,formatPromptOpener,formatPromptOutro,formatPromptSubmission,formatRailLine,railFor,runSelectComponent}from"#setup/cli/index.js";var import_picocolors=__toESM(require_picocolors(),1);function guardCancel(e){if(q(e))throw process.stdout.write(formatPromptCancellation(`Setup cancelled.`,import_picocolors.default)),new WizardCancelledError;return e}let escapeCancelDisabled=!1;function disableEscapeCancel(){escapeCancelDisabled||=(h.aliases.delete(`escape`),!0)}function attachQuitGuard(e){let t=initialQuitGuardState;return e.on(`key`,(n,r)=>{let{state:i,action:a}=reduceQuitGuard(t,r?.name===`escape`?{type:`escape`}:{type:`other-key`});t=i,a===`quit`&&(e.state=`cancel`)}),{note:()=>quitHintNote(t,import_picocolors.default)}}function toPromptState(e){return e}function railForState(e){return railFor(toPromptState(e),import_picocolors.default)}function cornerForState(e){return cornerFor(toPromptState(e),import_picocolors.default)}function header(e,t,n){return formatPromptHeader(toPromptState(e),t,{colors:import_picocolors.default,leadingRail:n===0?`white`:`green`})}const REDACTED_DISPLAY=`••••••••`;function textPrompt(e,t){let n,i=new ht({validate:t.validate?e=>t.validate?.(e??``):void 0,placeholder:t.placeholder,defaultValue:t.defaultValue,render(){let r=header(this.state,t.message,e),i=t.placeholder?import_picocolors.default.inverse(t.placeholder[0])+import_picocolors.default.dim(t.placeholder.slice(1)):import_picocolors.default.inverse(import_picocolors.default.hidden(`_`)),a=this.value?`•`.repeat(this.value.length):i,o=t.mask?a:this.value?this.userInputWithCursor:i;switch(this.state){case`error`:return`${r.trim()}\n${railForState(this.state)} ${o}\n${cornerForState(this.state)} ${import_picocolors.default.red(this.error)}\n`;case`submit`:{let n=t.mask?this.value?REDACTED_DISPLAY:t.placeholder||``:this.value||t.placeholder||``;return formatPromptSubmission(this.state,t.message,n,{colors:import_picocolors.default,leadingRail:e===0?`white`:`green`})}case`cancel`:return`${r}${railForState(this.state)} ${import_picocolors.default.strikethrough(import_picocolors.default.dim(t.mask&&this.value?REDACTED_DISPLAY:this.value??``))}${this.value?.trim()?`\n${railForState(this.state)}`:``}`;default:return`${r}${railForState(this.state)} ${o}\n${cornerForState(this.state)}${cornerNote(n?.note())}\n`}}});return n=attachQuitGuard(i),i.prompt()}function cornerNote(e){return e?` ${e}`:``}function createPrompter(){disableEscapeCancel();let e=0,t=createRailLog({colors:import_picocolors.default,output:process.stdout});return{async text(n){t.settle();let r=guardCancel(await textPrompt(e,n));return e+=1,r??``},async password(n){t.settle();let r=guardCancel(await textPrompt(e,{message:n.message,mask:!0,validate:n.validate}));return e+=1,r??``},async select(n){t.settle();for(let e of n.notices??[])e.tone===`success`?t.success(e.text):e.tone===`warning`?t.warning(e.text):t.message(e.text);let r=guardCancel(await runSelectComponent({message:n.message,options:n.options,multiple:n.multiple===!0,search:n.search??!1,required:n.required??!1,placeholder:n.placeholder,defaultValue:n.multiple===!0?void 0:n.initialValue,initialValues:n.multiple===!0?n.initialValues:void 0,leadingRail:e===0?`white`:`green`,attachGuard:e=>attachQuitGuard(e)}));return e+=1,r},async acknowledge(e){t.settle(),process.stdout.write(formatRailLine(import_picocolors.default.bold(e.message),import_picocolors.default,process.stdout));for(let t of e.lines??[])process.stdout.write(formatRailLine(import_picocolors.default.dim(t),import_picocolors.default,process.stdout))},note(e,n,r){t.settle();let i=r?.tone===`success`?import_picocolors.default.green:import_picocolors.default.red;n&&process.stdout.write(formatRailLine(i(import_picocolors.default.bold(n)),import_picocolors.default,process.stdout)),process.stdout.write(formatRailLine(i(e),import_picocolors.default,process.stdout))},intro(e,n=`Production-grade agent framework.`){t.settle(),process.stdout.write(formatPromptOpener(e,n,import_picocolors.default))},outro(e){t.settle(),process.stdout.write(formatPromptOutro(e,import_picocolors.default))},log:t}}export{createPrompter};
1
+ import{__toESM}from"../_virtual/_rolldown/runtime.js";import{require_picocolors}from"../node_modules/.pnpm/picocolors@1.1.1/node_modules/picocolors/picocolors.js";import{WizardCancelledError}from"./step.js";import{h,ht,q}from"../node_modules/.pnpm/@clack_core@1.3.1/node_modules/@clack/core/dist/index.js";import{initialQuitGuardState,quitHintNote,reduceQuitGuard}from"./quit-guard.js";import{cornerFor,createRailLog,formatPromptCancellation,formatPromptHeader,formatPromptOpener,formatPromptOutro,formatPromptSubmission,formatRailLine,railFor,runSelectComponent}from"#setup/cli/index.js";var import_picocolors=__toESM(require_picocolors(),1);function guardCancel(e){if(q(e))throw process.stdout.write(formatPromptCancellation(`Setup cancelled.`,import_picocolors.default)),new WizardCancelledError;return e}let escapeCancelDisabled=!1;function disableEscapeCancel(){escapeCancelDisabled||=(h.aliases.delete(`escape`),!0)}function attachQuitGuard(e){let t=initialQuitGuardState;return e.on(`key`,(n,r)=>{let{state:i,action:a}=reduceQuitGuard(t,r?.name===`escape`?{type:`escape`}:{type:`other-key`});t=i,a===`quit`&&(e.state=`cancel`)}),{note:()=>quitHintNote(t,import_picocolors.default)}}function toPromptState(e){return e}function railForState(e){return railFor(toPromptState(e),import_picocolors.default)}function cornerForState(e){return cornerFor(toPromptState(e),import_picocolors.default)}function header(e,t,n){return formatPromptHeader(toPromptState(e),t,{colors:import_picocolors.default,leadingRail:n===0?`white`:`green`})}const REDACTED_DISPLAY=`••••••••`;function textPrompt(e,t){let n,r=new ht({validate:t.validate?e=>t.validate?.(e??``):void 0,placeholder:t.placeholder,defaultValue:t.defaultValue,render(){let r=header(this.state,t.message,e),i=t.placeholder?import_picocolors.default.inverse(t.placeholder[0])+import_picocolors.default.dim(t.placeholder.slice(1)):import_picocolors.default.inverse(import_picocolors.default.hidden(`_`)),a=this.value?`•`.repeat(this.value.length):i,o=i;switch(t.mask?o=a:this.value&&(o=this.userInputWithCursor),this.state){case`error`:return`${r.trim()}\n${railForState(this.state)} ${o}\n${cornerForState(this.state)} ${import_picocolors.default.red(this.error)}\n`;case`submit`:{let n=this.value||t.placeholder||``;return t.mask&&(n=this.value?REDACTED_DISPLAY:t.placeholder??``),formatPromptSubmission(this.state,t.message,n,{colors:import_picocolors.default,leadingRail:e===0?`white`:`green`})}case`cancel`:return`${r}${railForState(this.state)} ${import_picocolors.default.strikethrough(import_picocolors.default.dim(t.mask&&this.value?REDACTED_DISPLAY:this.value??``))}${this.value?.trim()?`\n${railForState(this.state)}`:``}`;default:return`${r}${railForState(this.state)} ${o}\n${cornerForState(this.state)}${cornerNote(n?.note())}\n`}}});return n=attachQuitGuard(r),r.prompt()}function cornerNote(e){return e?` ${e}`:``}function createPrompter(){disableEscapeCancel();let e=0,t=createRailLog({colors:import_picocolors.default,output:process.stdout});function printNotices(e){for(let n of e??[])n.tone===`success`?t.success(n.text):n.tone===`warning`?t.warning(n.text):n.tone===`error`?t.error(n.text):t.message(n.text)}return{async text(n){t.settle(),printNotices(n.notices);let r=guardCancel(await textPrompt(e,n));return e+=1,r??``},async password(n){t.settle();let r=guardCancel(await textPrompt(e,{message:n.message,mask:!0,validate:n.validate}));return e+=1,r??``},async select(n){t.settle(),printNotices(n.notices);let r=guardCancel(await runSelectComponent({message:n.message,options:n.options,multiple:n.multiple===!0,search:n.search??!1,required:n.required??!1,placeholder:n.placeholder,defaultValue:n.multiple===!0?void 0:n.initialValue,initialValues:n.multiple===!0?n.initialValues:void 0,leadingRail:e===0?`white`:`green`,attachGuard:e=>attachQuitGuard(e)}));return e+=1,r},async acknowledge(e){t.settle(),process.stdout.write(formatRailLine(import_picocolors.default.bold(e.message),import_picocolors.default,process.stdout));for(let t of e.lines??[])process.stdout.write(formatRailLine(import_picocolors.default.dim(t),import_picocolors.default,process.stdout))},note(e,n,r){t.settle();let i=r?.tone===`success`?import_picocolors.default.green:import_picocolors.default.red;n&&process.stdout.write(formatRailLine(i(import_picocolors.default.bold(n)),import_picocolors.default,process.stdout)),process.stdout.write(formatRailLine(i(e),import_picocolors.default,process.stdout))},intro(e,n=`Production-grade agent framework.`){t.settle(),process.stdout.write(formatPromptOpener(e,n,import_picocolors.default))},outro(e){t.settle(),process.stdout.write(formatPromptOutro(e,import_picocolors.default))},log:t}}export{createPrompter};
@@ -6,5 +6,5 @@ type VercelOutputHandler = NonNullable<RunVercelOptions["onOutput"]>;
6
6
  * AI Gateway model calls. Safe to call repeatedly; Vercel CLI no-ops if
7
7
  * the env is already fresh.
8
8
  */
9
- export declare function runVercelEnvPull(projectRoot: string, onOutput?: VercelOutputHandler): Promise<boolean>;
9
+ export declare function runVercelEnvPull(projectRoot: string, onOutput?: VercelOutputHandler, signal?: AbortSignal): Promise<boolean>;
10
10
  export {};
@@ -1 +1 @@
1
- import{runVercel}from"#setup/primitives/index.js";async function runVercelEnvPull(e,t){return runVercel([`env`,`pull`,`--yes`],{cwd:e,onOutput:t})}export{runVercelEnvPull};
1
+ import{runVercel}from"#setup/primitives/index.js";async function runVercelEnvPull(e,t,n){return runVercel([`env`,`pull`,`--yes`],{cwd:e,onOutput:t,signal:n})}export{runVercelEnvPull};
@@ -32,6 +32,8 @@ export interface RunnerOptions<State> {
32
32
  * Defaults to the raw state.
33
33
  */
34
34
  snapshot?: (state: State) => State;
35
+ /** Cancels the active gather/perform operation and prevents later boxes from starting. */
36
+ signal?: AbortSignal;
35
37
  }
36
38
  /**
37
39
  * Linear interactive runner: walk the boxes in order, prompting for each. A box
@@ -1 +1 @@
1
- import{WizardCancelledError}from"./step.js";function isCancellation(t){return t instanceof WizardCancelledError}var RetryableSetupError=class extends Error{constructor(e,t){super(e,t),this.name=`RetryableSetupError`}};function isRetryable(e){return e instanceof RetryableSetupError}async function runInteractive(e,t,n,r){let i=t,view=()=>r?.snapshot?r.snapshot(i):i,a={};try{for(let t of e)if(t.shouldRun?.(view())!==!1)for(;;){let e=await t.gather({state:view(),initial:a[t.id]});a[t.id]=e;try{let r=await t.perform({state:view(),input:e,sink:n});i=t.apply(i,r);break}catch(e){if(!isRetryable(e))throw e;n.write(e instanceof Error?e.message:String(e))}}}catch(e){if(isCancellation(e))return{kind:`cancelled`};throw e}return{kind:`done`,state:i}}async function runHeadless(e,t,n,r){let i=t,view=()=>r?.snapshot?r.snapshot(i):i;for(let t of e){if(t.shouldRun?.(view())===!1)continue;let e=await t.gather({state:view()}),r=await t.perform({state:view(),input:e,sink:n});i=t.apply(i,r)}return i}export{RetryableSetupError,runHeadless,runInteractive};
1
+ import{WizardCancelledError}from"./step.js";function isCancellation(t,n){return t instanceof WizardCancelledError||n?.aborted===!0}var RetryableSetupError=class extends Error{constructor(e,t){super(e,t),this.name=`RetryableSetupError`}};function isRetryable(e){return e instanceof RetryableSetupError}async function runInteractive(e,t,n,r){let i=t,view=()=>r?.snapshot?r.snapshot(i):i,a={};try{for(let t of e)if(r?.signal?.throwIfAborted(),t.shouldRun?.(view())!==!1)for(;;){let e=await t.gather({state:view(),initial:a[t.id],signal:r?.signal});r?.signal?.throwIfAborted(),a[t.id]=e;try{let a=await t.perform({state:view(),input:e,sink:n,signal:r?.signal});r?.signal?.throwIfAborted(),i=t.apply(i,a);break}catch(e){if(!isRetryable(e))throw e;n.write(e instanceof Error?e.message:String(e))}}}catch(e){if(isCancellation(e,r?.signal))return{kind:`cancelled`};throw e}return{kind:`done`,state:i}}async function runHeadless(e,t,n,r){let i=t,view=()=>r?.snapshot?r.snapshot(i):i;for(let t of e){if(r?.signal?.throwIfAborted(),t.shouldRun?.(view())===!1)continue;let e=await t.gather({state:view(),signal:r?.signal});r?.signal?.throwIfAborted();let a=await t.perform({state:view(),input:e,sink:n,signal:r?.signal});r?.signal?.throwIfAborted(),i=t.apply(i,a)}return i}export{RetryableSetupError,runHeadless,runInteractive};
@@ -20,6 +20,8 @@ export interface ScaffoldableChannel {
20
20
  label: string;
21
21
  /** Optional picker hint. */
22
22
  hint?: string;
23
+ /** The add sub-flow provisions against the linked Vercel project. */
24
+ requiresVercelProject?: true;
23
25
  }
24
26
  /**
25
27
  * Channels the CLI can scaffold, in picker order. Derived from
@@ -1 +1 @@
1
- import{channelEntries}from"../../packages/eve-catalog/src/index.js";const CHANNEL_SCAFFOLDS=[{slug:`eve`,kind:`web`,label:`Web Chat`,hint:`Next.js app`},{slug:`slack`,kind:`slack`,label:`Slack`,hint:`creates slackbot and deploys to Vercel`}];function buildScaffoldableChannels(){let n=new Set(channelEntries().filter(e=>e.surfaces.scaffoldable).map(e=>e.slug)),r=[];for(let e of CHANNEL_SCAFFOLDS){if(!n.delete(e.slug))throw Error(`Channel overlay "${e.slug}" is not a scaffoldable channel in @vercel/eve-catalog.`);let t={slug:e.slug,kind:e.kind,label:e.label};e.hint!==void 0&&(t.hint=e.hint),r.push(t)}if(n.size>0){let e=[...n].sort().join(`, `);throw Error(`Scaffoldable catalog channels missing a scaffolder overlay: ${e}.`)}return r}const SCAFFOLDABLE_CHANNELS=buildScaffoldableChannels();export{SCAFFOLDABLE_CHANNELS};
1
+ import{channelEntries}from"../../packages/eve-catalog/src/index.js";const CHANNEL_SCAFFOLDS=[{slug:`eve`,kind:`web`,label:`Web Chat`,hint:`Next.js app`},{slug:`slack`,kind:`slack`,label:`Slack`,hint:`Creates slackbot and deploys to Vercel`,requiresVercelProject:!0}];function buildScaffoldableChannels(){let n=new Set(channelEntries().filter(e=>e.surfaces.scaffoldable).map(e=>e.slug)),r=[];for(let e of CHANNEL_SCAFFOLDS){if(!n.delete(e.slug))throw Error(`Channel overlay "${e.slug}" is not a scaffoldable channel in @vercel/eve-catalog.`);let t={slug:e.slug,kind:e.kind,label:e.label};e.hint!==void 0&&(t.hint=e.hint),e.requiresVercelProject!==void 0&&(t.requiresVercelProject=e.requiresVercelProject),r.push(t)}if(n.size>0){let e=[...n].sort().join(`, `);throw Error(`Scaffoldable catalog channels missing a scaffolder overlay: ${e}.`)}return r}const SCAFFOLDABLE_CHANNELS=buildScaffoldableChannels();export{SCAFFOLDABLE_CHANNELS};
@@ -9,6 +9,7 @@ export interface AddAgentToProjectOptions {
9
9
  packageManager?: PackageManagerKind;
10
10
  evePackageVersion?: string;
11
11
  aiPackageVersion?: string;
12
+ connectPackageVersion?: string;
12
13
  zodPackageVersion?: string;
13
14
  }
14
15
  export interface AddAgentToProjectResult {
@@ -1 +1 @@
1
- import{getPackageManagerStrategy}from"../../primitives/pm/index.js";import{pathExists,writeTextFile}from"../files.js";import{patchPackageJson}from"../update/package-json.js";import{resolveVersionToken}from"../version-tokens.js";import{agentTemplateFiles,formatEveDependencySpecifier}from"./project.js";import{join}from"node:path";import{readFile}from"node:fs/promises";const DEPENDENCY_FIELDS=[`dependencies`,`devDependencies`,`optionalDependencies`,`peerDependencies`];function isJsonObject(e){return typeof e==`object`&&!!e&&!Array.isArray(e)}function hasDeclaredDependency(e,t){if(!isJsonObject(e))return!1;for(let n of DEPENDENCY_FIELDS){let r=e[n];if(isJsonObject(r)&&typeof r[t]==`string`)return!0}return!1}async function addAgentToProject(i){let a=join(i.projectRoot,`package.json`);if(!await pathExists(a))throw Error(`Cannot add an Eve agent to "${i.projectRoot}" because it has no package.json. Run \`eve init <name>\` to create a new project instead.`);let o=agentTemplateFiles(i.model),s=[];for(let e of Object.keys(o))await pathExists(join(i.projectRoot,e))&&s.push(e);if(s.length===0&&await pathExists(join(i.projectRoot,`agent`))&&s.push(`agent/`),s.length>0)throw Error(`Cannot add an Eve agent to "${i.projectRoot}" because it already has: ${s.join(`, `)}. Move them aside first.`);let c=resolveVersionToken(`evePackageVersion`,i.evePackageVersion??`0.7.4`),l=resolveVersionToken(`aiPackageVersion`,i.aiPackageVersion??`7.0.0-canary.171`),u=resolveVersionToken(`zodPackageVersion`,i.zodPackageVersion??`4.4.3`),d=[];for(let[e,t]of Object.entries(o)){let r=join(i.projectRoot,e);await writeTextFile(r,t),d.push(r)}let f=JSON.parse(await readFile(a,`utf8`)),p={ai:l,eve:formatEveDependencySpecifier(c),zod:u},m={};for(let[e,t]of Object.entries(p))hasDeclaredDependency(f,e)||(m[e]=t);return Object.keys(m).length>0&&await patchPackageJson(a,{dependencies:m}),await getPackageManagerStrategy(i.packageManager??`pnpm`).applyProjectConfiguration(i.projectRoot),{filesWritten:d,dependenciesAdded:Object.keys(m).sort()}}export{addAgentToProject};
1
+ import{getPackageManagerStrategy}from"../../primitives/pm/index.js";import{pathExists,writeTextFile}from"../files.js";import{patchPackageJson}from"../update/package-json.js";import{resolveVersionToken}from"../version-tokens.js";import{agentTemplateFiles,formatEveDependencySpecifier}from"./project.js";import{join}from"node:path";import{readFile}from"node:fs/promises";const DEPENDENCY_FIELDS=[`dependencies`,`devDependencies`,`optionalDependencies`,`peerDependencies`];function isJsonObject(e){return typeof e==`object`&&!!e&&!Array.isArray(e)}function hasDeclaredDependency(e,t){if(!isJsonObject(e))return!1;for(let n of DEPENDENCY_FIELDS){let r=e[n];if(isJsonObject(r)&&typeof r[t]==`string`)return!0}return!1}async function addAgentToProject(i){let a=join(i.projectRoot,`package.json`);if(!await pathExists(a))throw Error(`Cannot add an Eve agent to "${i.projectRoot}" because it has no package.json. Run \`eve init <name>\` to create a new project instead.`);let o=agentTemplateFiles(i.model),s=[];for(let e of Object.keys(o))await pathExists(join(i.projectRoot,e))&&s.push(e);if(s.length===0&&await pathExists(join(i.projectRoot,`agent`))&&s.push(`agent/`),s.length>0)throw Error(`Cannot add an Eve agent to "${i.projectRoot}" because it already has: ${s.join(`, `)}. Move them aside first.`);let c=resolveVersionToken(`evePackageVersion`,i.evePackageVersion??`0.8.1`),l=resolveVersionToken(`aiPackageVersion`,i.aiPackageVersion??`7.0.0-canary.171`),u=resolveVersionToken(`connectPackageVersion`,i.connectPackageVersion??`0.2.2`),d=resolveVersionToken(`zodPackageVersion`,i.zodPackageVersion??`4.4.3`),f=[];for(let[e,t]of Object.entries(o)){let r=join(i.projectRoot,e);await writeTextFile(r,t),f.push(r)}let p=JSON.parse(await readFile(a,`utf8`)),m={"@vercel/connect":u,ai:l,eve:formatEveDependencySpecifier(c),zod:d},h={};for(let[e,t]of Object.entries(m))hasDeclaredDependency(p,e)||(h[e]=t);return Object.keys(h).length>0&&await patchPackageJson(a,{dependencies:h}),await getPackageManagerStrategy(i.packageManager??`pnpm`).applyProjectConfiguration(i.projectRoot),{filesWritten:f,dependenciesAdded:Object.keys(h).sort()}}export{addAgentToProject};
@@ -2,6 +2,7 @@ import type { PackageManagerKind } from "../../package-manager.js";
2
2
  export declare const CURRENT_DIRECTORY_PROJECT_NAME = ".";
3
3
  export declare const DEFAULT_EVE_PACKAGE_VERSION = "__EVE_PACKAGE_VERSION__";
4
4
  export declare const DEFAULT_AI_PACKAGE_VERSION = "__AI_SDK_VERSION__";
5
+ export declare const DEFAULT_CONNECT_PACKAGE_VERSION = "__VERCEL_CONNECT_VERSION__";
5
6
  export declare const DEFAULT_ZOD_PACKAGE_VERSION = "__ZOD_VERSION__";
6
7
  /**
7
8
  * Provider slug a gateway model id routes through: the segment before the
@@ -40,6 +41,7 @@ export interface ScaffoldBaseProjectOptions {
40
41
  onOverwriteFile?: (filePath: string) => void | Promise<void>;
41
42
  evePackageVersion?: string;
42
43
  aiPackageVersion?: string;
44
+ connectPackageVersion?: string;
43
45
  zodPackageVersion?: string;
44
46
  tsgoPackageVersion?: string;
45
47
  typesNodePackageVersion?: string;
@@ -1,4 +1,4 @@
1
- import{getPackageManagerStrategy}from"../../primitives/pm/index.js";import{pathExists,writeTextFile}from"../files.js";import{resolveVersionToken}from"../version-tokens.js";import{SUPPORTED_AUTHORED_MODULE_FILE_EXTENSIONS}from"../update/module-files.js";import{WEB_APP_TEMPLATE_FILES}from"./web-template.js";import{basename,join,resolve}from"node:path";import{mkdir,readdir,stat}from"node:fs/promises";const CURRENT_DIRECTORY_PROJECT_NAME=`.`,ALLOWED_CREATE_IN_PLACE_ENTRIES=new Set([`.DS_Store`,`.git`,`.gitkeep`,`.hg`]),DEFAULT_EVE_PACKAGE_VERSION=`0.7.4`,DEFAULT_AI_PACKAGE_VERSION=`7.0.0-canary.171`,DEFAULT_ZOD_PACKAGE_VERSION=`4.4.3`;function modelProviderSlug(e){let t=(e.split(`/`)[0]??``).replaceAll(/[^A-Za-z0-9._-]/gu,``);return t.length>0?t:`anthropic`}function byokProviderEnvVar(e){let t=modelProviderSlug(e).toUpperCase().replaceAll(/[^A-Z0-9]/gu,`_`);return`${/^[0-9]/.test(t)?`_`:``}${t}_API_KEY`}function agentTemplateFiles(e){return{"agent/agent.ts":BASE_AGENT_TEMPLATE.replaceAll(`__EVE_INIT_MODEL__`,e),"agent/channels/eve.ts":WEB_APP_TEMPLATE_FILES[`agent/channels/eve.ts`],"agent/instructions.md":AGENT_INSTRUCTIONS_TEMPLATE}}function renderTemplate(e,t){return e.replaceAll(`__EVE_INIT_APP_NAME__`,t.appName).replaceAll(`__EVE_INIT_MODEL__`,t.model).replaceAll(`__EVE_INIT_BYOK_PROVIDER__`,modelProviderSlug(t.model)).replaceAll(`__EVE_INIT_BYOK_ENV_VAR__`,byokProviderEnvVar(t.model)).replaceAll(`__EVE_INIT_PACKAGE_VERSION__`,formatEveDependencySpecifier(t.evePackageVersion)).replaceAll(`__EVE_INIT_AI_SDK_VERSION__`,t.aiPackageVersion).replaceAll(`__EVE_INIT_ZOD_VERSION__`,t.zodPackageVersion).replaceAll(`__EVE_INIT_TSGO_VERSION__`,t.tsgoPackageVersion).replaceAll(`__EVE_INIT_TYPES_NODE_VERSION__`,t.typesNodePackageVersion)}function formatEveDependencySpecifier(e){return/^\d+\.\d+\.\d+(?:[-+][0-9A-Za-z-.]+)?$/.test(e)?`^${e}`:e}const BASE_AGENT_TEMPLATE=`import { defineAgent } from "eve";
1
+ import{getPackageManagerStrategy}from"../../primitives/pm/index.js";import{pathExists,writeTextFile}from"../files.js";import{resolveVersionToken}from"../version-tokens.js";import{SUPPORTED_AUTHORED_MODULE_FILE_EXTENSIONS}from"../update/module-files.js";import{WEB_APP_TEMPLATE_FILES}from"./web-template.js";import{basename,join,resolve}from"node:path";import{mkdir,readdir,stat}from"node:fs/promises";const CURRENT_DIRECTORY_PROJECT_NAME=`.`,ALLOWED_CREATE_IN_PLACE_ENTRIES=new Set([`.DS_Store`,`.git`,`.gitkeep`,`.hg`]),DEFAULT_EVE_PACKAGE_VERSION=`0.8.1`,DEFAULT_AI_PACKAGE_VERSION=`7.0.0-canary.171`,DEFAULT_CONNECT_PACKAGE_VERSION=`0.2.2`,DEFAULT_ZOD_PACKAGE_VERSION=`4.4.3`;function modelProviderSlug(e){let t=(e.split(`/`)[0]??``).replaceAll(/[^A-Za-z0-9._-]/gu,``);return t.length>0?t:`anthropic`}function byokProviderEnvVar(e){let t=modelProviderSlug(e).toUpperCase().replaceAll(/[^A-Z0-9]/gu,`_`);return`${/^[0-9]/.test(t)?`_`:``}${t}_API_KEY`}function agentTemplateFiles(e){return{"agent/agent.ts":BASE_AGENT_TEMPLATE.replaceAll(`__EVE_INIT_MODEL__`,e),"agent/channels/eve.ts":WEB_APP_TEMPLATE_FILES[`agent/channels/eve.ts`],"agent/instructions.md":AGENT_INSTRUCTIONS_TEMPLATE}}function renderTemplate(e,t){return e.replaceAll(`__EVE_INIT_APP_NAME__`,t.appName).replaceAll(`__EVE_INIT_MODEL__`,t.model).replaceAll(`__EVE_INIT_BYOK_PROVIDER__`,modelProviderSlug(t.model)).replaceAll(`__EVE_INIT_BYOK_ENV_VAR__`,byokProviderEnvVar(t.model)).replaceAll(`__EVE_INIT_PACKAGE_VERSION__`,formatEveDependencySpecifier(t.evePackageVersion)).replaceAll(`__EVE_INIT_AI_SDK_VERSION__`,t.aiPackageVersion).replaceAll(`__EVE_INIT_CONNECT_VERSION__`,t.connectPackageVersion).replaceAll(`__EVE_INIT_ZOD_VERSION__`,t.zodPackageVersion).replaceAll(`__EVE_INIT_TSGO_VERSION__`,t.tsgoPackageVersion).replaceAll(`__EVE_INIT_TYPES_NODE_VERSION__`,t.typesNodePackageVersion)}function formatEveDependencySpecifier(e){return/^\d+\.\d+\.\d+(?:[-+][0-9A-Za-z-.]+)?$/.test(e)?`^${e}`:e}const BASE_AGENT_TEMPLATE=`import { defineAgent } from "eve";
2
2
 
3
3
  export default defineAgent({
4
4
  model: "__EVE_INIT_MODEL__",
@@ -18,6 +18,7 @@ export default defineAgent({
18
18
  "typecheck": "tsgo"
19
19
  },
20
20
  "dependencies": {
21
+ "@vercel/connect": "__EVE_INIT_CONNECT_VERSION__",
21
22
  "ai": "__EVE_INIT_AI_SDK_VERSION__",
22
23
  "eve": "__EVE_INIT_PACKAGE_VERSION__",
23
24
  "zod": "__EVE_INIT_ZOD_VERSION__"
@@ -54,14 +55,16 @@ dist
54
55
  .DS_Store
55
56
  *.tsbuildinfo
56
57
  `,".vercelignore":`node_modules
58
+ .env*
57
59
  .eve
60
+ .workflow-data
58
61
  .next
59
62
  .output
60
63
  .nitro
61
64
  dist
62
65
  `,"AGENTS.md":`# Eve Agent App
63
66
 
64
- This project uses the Eve framework. Before writing code, always read the relevant guide in \`node_modules/eve/dist/docs/public/\`.
67
+ This project uses the Eve framework. Before writing code, always read the relevant guide in \`node_modules/eve/docs/\`.
65
68
  `,"CLAUDE.md":`@AGENTS.md
66
69
  `};function templateFiles(t,n){return{"agent/agent.ts":t?`import { defineAgent } from "eve";
67
70
 
@@ -77,4 +80,4 @@ export default defineAgent({
77
80
  },
78
81
  },
79
82
  });
80
- `:BASE_AGENT_TEMPLATE,...SHARED_TEMPLATE_FILES,"package.json":packageJsonTemplate(t),...getPackageManagerStrategy(n).scaffoldFiles}}async function assertCanCreateInPlace(e,n){if(!await pathExists(e))return;let r=(await readdir(e)).filter(e=>!ALLOWED_CREATE_IN_PLACE_ENTRIES.has(e));if(r.length>0&&!n){let e=r.slice(0,5).join(`, `),t=r.length>5?`, and ${r.length-5} more`:``;throw Error(`Cannot create project in current directory because it is not empty. Found: ${e}${t}. Use an empty directory.`)}}async function scaffoldBaseProject(e){let i=resolve(e.targetDirectory??process.cwd(),e.projectName),a=e.projectName===`.`,s=e.overwriteExisting??!1,u=e.byokProvider??!1,d=e.packageManager??`pnpm`;if(a)await assertCanCreateInPlace(i,s);else if(await pathExists(i))throw Error(`Cannot create project because "${i}" already exists.`);let f=e.typesNodePackageVersion??`25.9.1`,p={appName:basename(i),model:e.model,evePackageVersion:resolveVersionToken(`evePackageVersion`,e.evePackageVersion??`0.7.4`),aiPackageVersion:resolveVersionToken(`aiPackageVersion`,e.aiPackageVersion??`7.0.0-canary.171`),zodPackageVersion:resolveVersionToken(`zodPackageVersion`,e.zodPackageVersion??`4.4.3`),tsgoPackageVersion:resolveVersionToken(`tsgoPackageVersion`,e.tsgoPackageVersion??`7.0.0-dev.20260523.1`),typesNodePackageVersion:u?resolveVersionToken(`typesNodePackageVersion`,f):f};await mkdir(i,{recursive:!0});for(let[r,o]of Object.entries(templateFiles(u,d))){let c=`${i}/${r}`,l=await pathExists(c);await writeTextFile(c,renderTemplate(o,p),{force:a&&s}),l&&await e.onOverwriteFile?.(c)}return i}async function isEveProject(e){for(let t of SUPPORTED_AUTHORED_MODULE_FILE_EXTENSIONS)try{return await stat(join(e,`agent`,`agent${t}`)),!0}catch{}return!1}export{CURRENT_DIRECTORY_PROJECT_NAME,DEFAULT_AI_PACKAGE_VERSION,DEFAULT_EVE_PACKAGE_VERSION,DEFAULT_ZOD_PACKAGE_VERSION,agentTemplateFiles,byokProviderEnvVar,formatEveDependencySpecifier,isEveProject,modelProviderSlug,scaffoldBaseProject};
83
+ `:BASE_AGENT_TEMPLATE,...SHARED_TEMPLATE_FILES,"package.json":packageJsonTemplate(t),...getPackageManagerStrategy(n).scaffoldFiles}}async function assertCanCreateInPlace(e,n){if(!await pathExists(e))return;let r=(await readdir(e)).filter(e=>!ALLOWED_CREATE_IN_PLACE_ENTRIES.has(e));if(r.length>0&&!n){let e=r.slice(0,5).join(`, `),t=r.length>5?`, and ${r.length-5} more`:``;throw Error(`Cannot create project in current directory because it is not empty. Found: ${e}${t}. Use an empty directory.`)}}async function scaffoldBaseProject(e){let i=resolve(e.targetDirectory??process.cwd(),e.projectName),a=e.projectName===`.`,s=e.overwriteExisting??!1,u=e.byokProvider??!1,d=e.packageManager??`pnpm`;if(a)await assertCanCreateInPlace(i,s);else if(await pathExists(i))throw Error(`Cannot create project because "${i}" already exists.`);let f=e.typesNodePackageVersion??`25.9.1`,p={appName:basename(i),model:e.model,evePackageVersion:resolveVersionToken(`evePackageVersion`,e.evePackageVersion??`0.8.1`),aiPackageVersion:resolveVersionToken(`aiPackageVersion`,e.aiPackageVersion??`7.0.0-canary.171`),connectPackageVersion:resolveVersionToken(`connectPackageVersion`,e.connectPackageVersion??`0.2.2`),zodPackageVersion:resolveVersionToken(`zodPackageVersion`,e.zodPackageVersion??`4.4.3`),tsgoPackageVersion:resolveVersionToken(`tsgoPackageVersion`,e.tsgoPackageVersion??`7.0.0-dev.20260523.1`),typesNodePackageVersion:u?resolveVersionToken(`typesNodePackageVersion`,f):f};await mkdir(i,{recursive:!0});for(let[r,o]of Object.entries(templateFiles(u,d))){let c=`${i}/${r}`,l=await pathExists(c);await writeTextFile(c,renderTemplate(o,p),{force:a&&s}),l&&await e.onOverwriteFile?.(c)}return i}async function isEveProject(e){for(let t of SUPPORTED_AUTHORED_MODULE_FILE_EXTENSIONS)try{return await stat(join(e,`agent`,`agent${t}`)),!0}catch{}return!1}export{CURRENT_DIRECTORY_PROJECT_NAME,DEFAULT_AI_PACKAGE_VERSION,DEFAULT_CONNECT_PACKAGE_VERSION,DEFAULT_EVE_PACKAGE_VERSION,DEFAULT_ZOD_PACKAGE_VERSION,agentTemplateFiles,byokProviderEnvVar,formatEveDependencySpecifier,isEveProject,modelProviderSlug,scaffoldBaseProject};
@@ -67,6 +67,8 @@ export interface EnsureChannelOptions {
67
67
  /** Manager that owns generated project configuration. Defaults to pnpm. */
68
68
  packageManager?: PackageManagerKind;
69
69
  force?: boolean;
70
+ /** Exact UID returned by Vercel Connect; takes precedence over the derived slug. */
71
+ slackConnectorUid?: string;
70
72
  slackConnectorSlug?: SlackConnectorSlug;
71
73
  connectPackageVersion?: string;
72
74
  webPackageVersions?: WebPackageVersions;
@@ -1,7 +1,7 @@
1
- import{getPackageManagerStrategy}from"../../primitives/pm/index.js";import{pathExists,writeTextFile}from"../files.js";import{patchPackageJson}from"./package-json.js";import{resolveVersionToken}from"../version-tokens.js";import{getSupportedModuleBaseName,matchesSupportedModuleBaseName}from"./module-files.js";import{WEB_APP_TEMPLATE_FILES,WEB_APP_TEMPLATE_PACKAGE_JSON}from"../create/web-template.js";import"../create/project.js";import{basename,join,resolve}from"node:path";import{readFile,readdir,writeFile}from"node:fs/promises";const SLACK_CHANNEL_DEFAULT_ROUTE=`/eve/v1/slack`,DEFAULT_SLACK_CONNECTOR_SLUG=`my-agent`,PACKAGE_DEPENDENCY_FIELDS=[`dependencies`,`devDependencies`,`peerDependencies`,`optionalDependencies`],WEB_NEXT_CONFIG_PATH=`next.config.ts`,WEB_VERCEL_JSON_PATH=`vercel.json`,WEB_VERCEL_JSON_SCHEMA=`https://openapi.vercel.sh/vercel.json`,WEB_COMPETING_NEXT_CONFIG_PATHS=[`next.config.js`,`next.config.mjs`,WEB_NEXT_CONFIG_PATH,`next.config.mts`].filter(e=>e!==WEB_NEXT_CONFIG_PATH),WEB_DEFAULT_VERCEL_SERVICES={web:{entrypoint:`.`,framework:`nextjs`,routePrefix:`/`},eve:{buildCommand:`eve build`,entrypoint:`.`,framework:`eve`,routePrefix:`/_eve_internal/eve`}};function toSlackConnectorSlug(e){return e}function isJsonObject(e){return typeof e==`object`&&!!e&&!Array.isArray(e)}async function readDependencyVersion(e,t){let n=JSON.parse(await readFile(e,`utf8`));if(!isJsonObject(n)||!isJsonObject(n.dependencies))return;let r=n.dependencies[t];return typeof r==`string`?r:void 0}function packageJsonHasDependency(e,t){for(let n of PACKAGE_DEPENDENCY_FIELDS){let r=e[n];if(isJsonObject(r)&&typeof r[t]==`string`)return!0}return!1}async function hasPackageDependency(e,n){if(!await pathExists(e))return!1;let r=JSON.parse(await readFile(e,`utf8`));return isJsonObject(r)&&packageJsonHasDependency(r,n)}async function isNextJsProject(e){return hasPackageDependency(join(e,`package.json`),`next`)}async function ensurePackageDependency(e,n,i){return!await pathExists(e)||await readDependencyVersion(e,n)===i?[]:(await patchPackageJson(e,{dependencies:{[n]:i}}),[{path:e,dependencies:[n],devDependencies:[],scripts:[]}])}function resolveWebPackageVersions(e){return{evePackageVersion:e?.evePackageVersion??`0.7.4`,aiPackageVersion:e?.aiPackageVersion??`7.0.0-canary.171`,nextPackageVersion:e?.nextPackageVersion??`16.2.6`,reactPackageVersion:e?.reactPackageVersion??`19.2.6`,reactDomPackageVersion:e?.reactDomPackageVersion??`19.2.6`,streamdownPackageVersion:e?.streamdownPackageVersion??`2.5.0`,zodPackageVersion:e?.zodPackageVersion??`4.4.3`,tsgoPackageVersion:e?.tsgoPackageVersion??`7.0.0-dev.20260523.1`,typesNodePackageVersion:e?.typesNodePackageVersion??`25.9.1`,typesReactPackageVersion:e?.typesReactPackageVersion??`19.2.15`,typesReactDomPackageVersion:e?.typesReactDomPackageVersion??`19.2.3`}}function formatEveDependencySpecifier(e){return/^\d+\.\d+\.\d+(?:[-+][0-9A-Za-z-.]+)?$/.test(e)?`^${e}`:e}async function patchWebPackageJson(e,n){if(!await pathExists(e))return[];let a={...WEB_APP_TEMPLATE_PACKAGE_JSON.dependencies,ai:resolveVersionToken(`aiPackageVersion`,n.aiPackageVersion),eve:formatEveDependencySpecifier(resolveVersionToken(`evePackageVersion`,n.evePackageVersion)),next:resolveVersionToken(`nextPackageVersion`,n.nextPackageVersion),react:resolveVersionToken(`reactPackageVersion`,n.reactPackageVersion),"react-dom":resolveVersionToken(`reactDomPackageVersion`,n.reactDomPackageVersion),streamdown:resolveVersionToken(`streamdownPackageVersion`,n.streamdownPackageVersion),zod:resolveVersionToken(`zodPackageVersion`,n.zodPackageVersion)},o={...WEB_APP_TEMPLATE_PACKAGE_JSON.devDependencies,"@types/node":resolveVersionToken(`typesNodePackageVersion`,n.typesNodePackageVersion),"@types/react":resolveVersionToken(`typesReactPackageVersion`,n.typesReactPackageVersion),"@types/react-dom":resolveVersionToken(`typesReactDomPackageVersion`,n.typesReactDomPackageVersion),"@typescript/native-preview":resolveVersionToken(`tsgoPackageVersion`,n.tsgoPackageVersion)},s=WEB_APP_TEMPLATE_PACKAGE_JSON.scripts;return await patchPackageJson(e,{dependencies:a,devDependencies:o,scripts:s}),[{path:e,dependencies:Object.keys(a),devDependencies:Object.keys(o),scripts:Object.keys(s)}]}function normalizeSlackConnectorSlug(e){return toSlackConnectorSlug((e.trim().replace(/^@/,``).split(`/`).at(-1)??``).toLowerCase().replace(/[^a-z0-9_-]+/g,`-`).replace(/^[^a-z0-9]+/,``).replace(/[^a-z0-9]+$/,``).slice(0,100).replace(/[^a-z0-9]+$/,``)||`my-agent`)}async function deriveSlackConnectorSlug(e,t){if(t!==void 0&&t.length>0&&t!==`.`)return normalizeSlackConnectorSlug(t);try{let t=await readFile(join(e,`package.json`),`utf8`),n=JSON.parse(t);if(typeof n.name==`string`&&n.name.length>0)return normalizeSlackConnectorSlug(n.name)}catch{}return normalizeSlackConnectorSlug(basename(resolve(e))||`my-agent`)}function buildSlackTemplate(e){return`import { connectSlackCredentials } from "@vercel/connect/eve";
1
+ import{getPackageManagerStrategy}from"../../primitives/pm/index.js";import{pathExists,writeTextFile}from"../files.js";import{patchPackageJson}from"./package-json.js";import{resolveVersionToken}from"../version-tokens.js";import{getSupportedModuleBaseName,matchesSupportedModuleBaseName}from"./module-files.js";import{WEB_APP_TEMPLATE_FILES,WEB_APP_TEMPLATE_PACKAGE_JSON}from"../create/web-template.js";import"../create/project.js";import{basename,join,resolve}from"node:path";import{readFile,readdir,writeFile}from"node:fs/promises";const SLACK_CHANNEL_DEFAULT_ROUTE=`/eve/v1/slack`,DEFAULT_SLACK_CONNECTOR_SLUG=`my-agent`,PACKAGE_DEPENDENCY_FIELDS=[`dependencies`,`devDependencies`,`peerDependencies`,`optionalDependencies`],WEB_NEXT_CONFIG_PATH=`next.config.ts`,WEB_VERCEL_JSON_PATH=`vercel.json`,WEB_VERCEL_JSON_SCHEMA=`https://openapi.vercel.sh/vercel.json`,WEB_COMPETING_NEXT_CONFIG_PATHS=[`next.config.js`,`next.config.mjs`,WEB_NEXT_CONFIG_PATH,`next.config.mts`].filter(e=>e!==WEB_NEXT_CONFIG_PATH),WEB_DEFAULT_VERCEL_SERVICES={web:{entrypoint:`.`,framework:`nextjs`,routePrefix:`/`},eve:{buildCommand:`eve build`,entrypoint:`.`,framework:`eve`,routePrefix:`/_eve_internal/eve`}};function toSlackConnectorSlug(e){return e}function isJsonObject(e){return typeof e==`object`&&!!e&&!Array.isArray(e)}async function readDependencyVersion(e,t){let n=JSON.parse(await readFile(e,`utf8`));if(!isJsonObject(n)||!isJsonObject(n.dependencies))return;let r=n.dependencies[t];return typeof r==`string`?r:void 0}function packageJsonHasDependency(e,t){for(let n of PACKAGE_DEPENDENCY_FIELDS){let r=e[n];if(isJsonObject(r)&&typeof r[t]==`string`)return!0}return!1}async function hasPackageDependency(e,n){if(!await pathExists(e))return!1;let r=JSON.parse(await readFile(e,`utf8`));return isJsonObject(r)&&packageJsonHasDependency(r,n)}async function isNextJsProject(e){return hasPackageDependency(join(e,`package.json`),`next`)}async function ensurePackageDependency(e,n,i){return!await pathExists(e)||await readDependencyVersion(e,n)===i?[]:(await patchPackageJson(e,{dependencies:{[n]:i}}),[{path:e,dependencies:[n],devDependencies:[],scripts:[]}])}function resolveWebPackageVersions(e){return{evePackageVersion:e?.evePackageVersion??`0.8.1`,aiPackageVersion:e?.aiPackageVersion??`7.0.0-canary.171`,nextPackageVersion:e?.nextPackageVersion??`16.2.6`,reactPackageVersion:e?.reactPackageVersion??`19.2.6`,reactDomPackageVersion:e?.reactDomPackageVersion??`19.2.6`,streamdownPackageVersion:e?.streamdownPackageVersion??`2.5.0`,zodPackageVersion:e?.zodPackageVersion??`4.4.3`,tsgoPackageVersion:e?.tsgoPackageVersion??`7.0.0-dev.20260523.1`,typesNodePackageVersion:e?.typesNodePackageVersion??`25.9.1`,typesReactPackageVersion:e?.typesReactPackageVersion??`19.2.15`,typesReactDomPackageVersion:e?.typesReactDomPackageVersion??`19.2.3`}}function formatEveDependencySpecifier(e){return/^\d+\.\d+\.\d+(?:[-+][0-9A-Za-z-.]+)?$/.test(e)?`^${e}`:e}async function patchWebPackageJson(e,n){if(!await pathExists(e))return[];let a={...WEB_APP_TEMPLATE_PACKAGE_JSON.dependencies,ai:resolveVersionToken(`aiPackageVersion`,n.aiPackageVersion),eve:formatEveDependencySpecifier(resolveVersionToken(`evePackageVersion`,n.evePackageVersion)),next:resolveVersionToken(`nextPackageVersion`,n.nextPackageVersion),react:resolveVersionToken(`reactPackageVersion`,n.reactPackageVersion),"react-dom":resolveVersionToken(`reactDomPackageVersion`,n.reactDomPackageVersion),streamdown:resolveVersionToken(`streamdownPackageVersion`,n.streamdownPackageVersion),zod:resolveVersionToken(`zodPackageVersion`,n.zodPackageVersion)},o={...WEB_APP_TEMPLATE_PACKAGE_JSON.devDependencies,"@types/node":resolveVersionToken(`typesNodePackageVersion`,n.typesNodePackageVersion),"@types/react":resolveVersionToken(`typesReactPackageVersion`,n.typesReactPackageVersion),"@types/react-dom":resolveVersionToken(`typesReactDomPackageVersion`,n.typesReactDomPackageVersion),"@typescript/native-preview":resolveVersionToken(`tsgoPackageVersion`,n.tsgoPackageVersion)},s=WEB_APP_TEMPLATE_PACKAGE_JSON.scripts;return await patchPackageJson(e,{dependencies:a,devDependencies:o,scripts:s}),[{path:e,dependencies:Object.keys(a),devDependencies:Object.keys(o),scripts:Object.keys(s)}]}function normalizeSlackConnectorSlug(e){return toSlackConnectorSlug((e.trim().replace(/^@/,``).split(`/`).at(-1)??``).toLowerCase().replace(/[^a-z0-9_-]+/g,`-`).replace(/^[^a-z0-9]+/,``).replace(/[^a-z0-9]+$/,``).slice(0,100).replace(/[^a-z0-9]+$/,``)||`my-agent`)}async function deriveSlackConnectorSlug(e,t){if(t!==void 0&&t.length>0&&t!==`.`)return normalizeSlackConnectorSlug(t);try{let t=await readFile(join(e,`package.json`),`utf8`),n=JSON.parse(t);if(typeof n.name==`string`&&n.name.length>0)return normalizeSlackConnectorSlug(n.name)}catch{}return normalizeSlackConnectorSlug(basename(resolve(e))||`my-agent`)}function buildSlackTemplate(e){if(!e.startsWith(`slack/`)||e.length===6)throw Error(`Invalid Slack connector UID "${e}".`);return`import { connectSlackCredentials } from "@vercel/connect/eve";
2
2
  import { slackChannel } from "eve/channels/slack";
3
3
 
4
4
  export default slackChannel({
5
- credentials: connectSlackCredentials("slack/${e}"),
5
+ credentials: connectSlackCredentials(${JSON.stringify(e)}),
6
6
  });
7
- `}function renderWebAppTemplate(e,t,n){let r=n?``:`, { configureVercelOutput: false }`;return e.replaceAll(`__EVE_INIT_APP_NAME__`,t).replaceAll(`__EVE_INIT_WITH_EVE_OPTIONS__`,r)}function withWebVercelServices(e){let t=JSON.parse(e);if(!isJsonObject(t))throw Error(`${WEB_VERCEL_JSON_PATH} must contain a JSON object.`);let n=t.experimentalServices;if(n!==void 0&&!isJsonObject(n))throw Error(`${WEB_VERCEL_JSON_PATH} experimentalServices must contain a JSON object.`);let r={...t,$schema:typeof t.$schema==`string`?t.$schema:WEB_VERCEL_JSON_SCHEMA,experimentalServices:{...n,web:n?.web??WEB_DEFAULT_VERCEL_SERVICES.web,eve:n?.eve??WEB_DEFAULT_VERCEL_SERVICES.eve}};return JSON.stringify(t)===JSON.stringify(r)?e:`${JSON.stringify(r,null,2)}\n`}async function ensureWebVercelServices(e){if(!await pathExists(e))return await writeTextFile(e,`${JSON.stringify({$schema:WEB_VERCEL_JSON_SCHEMA,experimentalServices:WEB_DEFAULT_VERCEL_SERVICES},null,2)}\n`,{force:!0}),`written`;let r=await readFile(e,`utf8`),i=withWebVercelServices(r);return i===r?`skipped`:(await writeFile(e,i,`utf8`),`written`)}async function findCompetingNextConfigFiles(e){let n=[];for(let r of WEB_COMPETING_NEXT_CONFIG_PATHS){let i=join(e,r);await pathExists(i)&&n.push(i)}return n}async function ensureChannel(e){switch(e.kind){case`slack`:return ensureSlackChannel({...e,kind:`slack`});case`web`:return ensureWebChannel({...e,kind:`web`})}}async function ensureWebChannel(r){let i=join(r.projectRoot,`package.json`),a=await pathExists(join(r.projectRoot,`app/page.tsx`));if(!r.force&&await isNextJsProject(r.projectRoot))return{kind:`web`,action:`skipped`,skipReason:`nextjs-project`,filesWritten:[],filesSkipped:[i],packageJsonUpdated:[]};let o=await patchWebPackageJson(i,resolveWebPackageVersions(r.webPackageVersions)),c=[],u=[],d=[],f=[],p=basename(resolve(r.projectRoot)),m=r.configureVercelServices??!0;if(m){let e=join(r.projectRoot,WEB_VERCEL_JSON_PATH);await ensureWebVercelServices(e)===`written`?c.push(e):f.push(e)}let h=await getPackageManagerStrategy(r.packageManager??`pnpm`).applyProjectConfiguration(r.projectRoot);c.push(...h.filesWritten),f.push(...h.filesSkipped);for(let[e,i]of Object.entries(WEB_APP_TEMPLATE_FILES)){let a=join(r.projectRoot,e);if(e===`agent/channels/eve.ts`&&!r.force&&await pathExists(a)){f.push(a);continue}let o=await pathExists(a);await writeTextFile(a,renderWebAppTemplate(i,p,m),{force:!0}),c.push(a),o&&u.push(a)}d.push(...await findCompetingNextConfigFiles(r.projectRoot));let g={kind:`web`,action:a?`overwritten`:`created`,filesWritten:c,filesSkipped:f,packageJsonUpdated:o};return u.length>0&&(g.filesOverwritten=u),d.length>0&&(g.competingNextConfigFiles=d),g}async function ensureSlackChannel(e){let r=join(e.projectRoot,`agent/channels/slack.ts`),a=await pathExists(r);if(!e.force&&a)return{kind:`slack`,action:`skipped`,filesWritten:[],filesSkipped:[r],packageJsonUpdated:[]};let o=resolveVersionToken(`connectPackageVersion`,e.connectPackageVersion??`0.2.2`),s=await ensurePackageDependency(join(e.projectRoot,`package.json`),`@vercel/connect`,o),c=e.slackConnectorSlug??await deriveSlackConnectorSlug(e.projectRoot);await writeTextFile(r,buildSlackTemplate(c),{force:e.force});let l={kind:`slack`,action:a?`overwritten`:`created`,filesWritten:[r],filesSkipped:[],packageJsonUpdated:s,slackConnectorSlug:c};return a&&(l.filesOverwritten=[r]),l}async function listAuthoredChannels(e){let t=join(e,`agent/channels`),n;try{n=await readdir(t,{withFileTypes:!0})}catch(e){if(e.code===`ENOENT`)return[];throw e}let r=[];for(let e of n){if(e.isFile()){let t=getSupportedModuleBaseName(e.name);t!==null&&r.push(t);continue}if(e.isDirectory())try{(await readdir(join(t,e.name))).some(e=>matchesSupportedModuleBaseName(e,`connection`))&&r.push(e.name)}catch{}}return r.sort()}export{DEFAULT_SLACK_CONNECTOR_SLUG,SLACK_CHANNEL_DEFAULT_ROUTE,deriveSlackConnectorSlug,ensureChannel,isNextJsProject,listAuthoredChannels,normalizeSlackConnectorSlug};
7
+ `}function renderWebAppTemplate(e,t,n){let r=n?``:`, { configureVercelOutput: false }`;return e.replaceAll(`__EVE_INIT_APP_NAME__`,t).replaceAll(`__EVE_INIT_WITH_EVE_OPTIONS__`,r)}function withWebVercelServices(e){let t=JSON.parse(e);if(!isJsonObject(t))throw Error(`${WEB_VERCEL_JSON_PATH} must contain a JSON object.`);let n=t.experimentalServices;if(n!==void 0&&!isJsonObject(n))throw Error(`${WEB_VERCEL_JSON_PATH} experimentalServices must contain a JSON object.`);let r={...t,$schema:typeof t.$schema==`string`?t.$schema:WEB_VERCEL_JSON_SCHEMA,experimentalServices:{...n,web:n?.web??WEB_DEFAULT_VERCEL_SERVICES.web,eve:n?.eve??WEB_DEFAULT_VERCEL_SERVICES.eve}};return JSON.stringify(t)===JSON.stringify(r)?e:`${JSON.stringify(r,null,2)}\n`}async function ensureWebVercelServices(e){if(!await pathExists(e))return await writeTextFile(e,`${JSON.stringify({$schema:WEB_VERCEL_JSON_SCHEMA,experimentalServices:WEB_DEFAULT_VERCEL_SERVICES},null,2)}\n`,{force:!0}),`written`;let r=await readFile(e,`utf8`),i=withWebVercelServices(r);return i===r?`skipped`:(await writeFile(e,i,`utf8`),`written`)}async function findCompetingNextConfigFiles(e){let n=[];for(let r of WEB_COMPETING_NEXT_CONFIG_PATHS){let i=join(e,r);await pathExists(i)&&n.push(i)}return n}async function ensureChannel(e){switch(e.kind){case`slack`:return ensureSlackChannel({...e,kind:`slack`});case`web`:return ensureWebChannel({...e,kind:`web`})}}async function ensureWebChannel(r){let i=join(r.projectRoot,`package.json`),a=await pathExists(join(r.projectRoot,`app/page.tsx`));if(!r.force&&await isNextJsProject(r.projectRoot))return{kind:`web`,action:`skipped`,skipReason:`nextjs-project`,filesWritten:[],filesSkipped:[i],packageJsonUpdated:[]};let o=await patchWebPackageJson(i,resolveWebPackageVersions(r.webPackageVersions)),c=[],u=[],d=[],f=[],p=basename(resolve(r.projectRoot)),m=r.configureVercelServices??!0;if(m){let e=join(r.projectRoot,WEB_VERCEL_JSON_PATH);await ensureWebVercelServices(e)===`written`?c.push(e):f.push(e)}let h=await getPackageManagerStrategy(r.packageManager??`pnpm`).applyProjectConfiguration(r.projectRoot);c.push(...h.filesWritten),f.push(...h.filesSkipped);for(let[e,i]of Object.entries(WEB_APP_TEMPLATE_FILES)){let a=join(r.projectRoot,e);if(e===`agent/channels/eve.ts`&&!r.force&&await pathExists(a)){f.push(a);continue}let o=await pathExists(a);await writeTextFile(a,renderWebAppTemplate(i,p,m),{force:!0}),c.push(a),o&&u.push(a)}d.push(...await findCompetingNextConfigFiles(r.projectRoot));let g={kind:`web`,action:a?`overwritten`:`created`,filesWritten:c,filesSkipped:f,packageJsonUpdated:o};return u.length>0&&(g.filesOverwritten=u),d.length>0&&(g.competingNextConfigFiles=d),g}async function ensureSlackChannel(e){let r=join(e.projectRoot,`agent/channels/slack.ts`),a=await pathExists(r);if(!e.force&&a)return{kind:`slack`,action:`skipped`,filesWritten:[],filesSkipped:[r],packageJsonUpdated:[]};let o=resolveVersionToken(`connectPackageVersion`,e.connectPackageVersion??`0.2.2`),s=await ensurePackageDependency(join(e.projectRoot,`package.json`),`@vercel/connect`,o),c=e.slackConnectorSlug??await deriveSlackConnectorSlug(e.projectRoot);await writeTextFile(r,buildSlackTemplate(e.slackConnectorUid??`slack/${c}`),{force:e.force});let l={kind:`slack`,action:a?`overwritten`:`created`,filesWritten:[r],filesSkipped:[],packageJsonUpdated:s,slackConnectorSlug:c};return a&&(l.filesOverwritten=[r]),l}async function listAuthoredChannels(e){let t=join(e,`agent/channels`),n;try{n=await readdir(t,{withFileTypes:!0})}catch(e){if(e.code===`ENOENT`)return[];throw e}let r=[];for(let e of n){if(e.isFile()){let t=getSupportedModuleBaseName(e.name);t!==null&&r.push(t);continue}if(e.isDirectory())try{(await readdir(join(t,e.name))).some(e=>matchesSupportedModuleBaseName(e,`connection`))&&r.push(e.name)}catch{}}return r.sort()}export{DEFAULT_SLACK_CONNECTOR_SLUG,SLACK_CHANNEL_DEFAULT_ROUTE,deriveSlackConnectorSlug,ensureChannel,isNextJsProject,listAuthoredChannels,normalizeSlackConnectorSlug};
@@ -0,0 +1,97 @@
1
+ import { createPromptCommandOutput, type ChannelSetupLog } from "#setup/cli/index.js";
2
+ import { captureVercel, runVercel } from "#setup/primitives/run-vercel.js";
3
+ import { type RawSlackConnector, type SlackConnectorRef } from "./slack-connect.js";
4
+ export declare const CONNECT_LOOKUP_TIMEOUT_MS = 60000;
5
+ export declare const CONNECT_MUTATION_TIMEOUT_MS: number;
6
+ /** Connect subprocess operations needed to inventory and remove Slack connectors. */
7
+ export interface SlackConnectLifecycleDeps {
8
+ captureVercel: typeof captureVercel;
9
+ runVercel: typeof runVercel;
10
+ }
11
+ export type SlackConnectorInventory = {
12
+ state: "ok";
13
+ body: unknown;
14
+ connectors: readonly RawSlackConnector[];
15
+ } | {
16
+ state: "failed";
17
+ message: string;
18
+ };
19
+ export type SlackConnectorCleanupResult = {
20
+ state: "clean";
21
+ } | {
22
+ state: "failed";
23
+ connectorUids: readonly string[];
24
+ };
25
+ type CommandOutput = ReturnType<typeof createPromptCommandOutput>;
26
+ /** Shared dependencies and output routing for one connector cleanup operation. */
27
+ export interface SlackConnectorCleanupContext {
28
+ log: ChannelSetupLog;
29
+ deps: SlackConnectLifecycleDeps;
30
+ projectRoot: string;
31
+ onOutput: CommandOutput;
32
+ }
33
+ /** Reads and validates the account-level Slack connector inventory. */
34
+ export declare function listSlackConnectors(deps: SlackConnectLifecycleDeps, projectRoot: string, onOutput: CommandOutput, signal?: AbortSignal): Promise<SlackConnectorInventory>;
35
+ /** Reads the linked Vercel project id from the on-disk `.vercel/project.json`. */
36
+ export declare function readProjectId(projectRoot: string): Promise<string | undefined>;
37
+ export type SlackConnectorLookup = {
38
+ state: "found";
39
+ connector: SlackConnectorRef;
40
+ connectorUids: ReadonlySet<string>;
41
+ } | {
42
+ state: "not-found";
43
+ connectorUids: ReadonlySet<string>;
44
+ } | {
45
+ state: "failed";
46
+ message: string;
47
+ };
48
+ /** Resolves the expected (or newest project-attached) Slack connector from the inventory. */
49
+ export declare function findSlackConnector(deps: SlackConnectLifecycleDeps, projectRoot: string, projectId: string | undefined, expectedUid: string | undefined, onOutput: CommandOutput, signal?: AbortSignal): Promise<SlackConnectorLookup>;
50
+ /**
51
+ * Outcome of pointing a connector's trigger destination at this project's Eve
52
+ * route. Detach and attach are reported separately because attach must not run
53
+ * while an old trigger destination may still exist. A `detach-failed` connector
54
+ * is left in a known-stale state the caller surfaces with manual recovery steps.
55
+ */
56
+ export type SlackConnectorAttachmentResult = {
57
+ state: "attached";
58
+ } | {
59
+ state: "detach-failed";
60
+ } | {
61
+ state: "attach-failed";
62
+ };
63
+ /**
64
+ * Replaces the connector's default trigger destination with the Eve route:
65
+ * detach the existing destination first, then attach this project. Either
66
+ * subprocess failing short-circuits so a half-configured connector is reported
67
+ * rather than silently left pointing at the wrong place.
68
+ */
69
+ export declare function attachSlackConnector(deps: SlackConnectLifecycleDeps, projectRoot: string, ref: SlackConnectorRef, onOutput: CommandOutput, signal?: AbortSignal): Promise<SlackConnectorAttachmentResult>;
70
+ export type InstallationLookup = {
71
+ state: "installed";
72
+ info: {
73
+ workspaceUrl: string;
74
+ workspaceName?: string;
75
+ };
76
+ } | {
77
+ state: "pending";
78
+ } | {
79
+ state: "failed";
80
+ message: string;
81
+ };
82
+ /** Fetches the Slack workspace installation for a connector, if one exists yet. */
83
+ export declare function fetchInstallationInfo(deps: SlackConnectLifecycleDeps, projectRoot: string, connectorId: string, onOutput: CommandOutput, timeoutMs: number, signal?: AbortSignal): Promise<InstallationLookup>;
84
+ /**
85
+ * Removes the exact connector returned by `connect create`. When no UID was
86
+ * returned, ownership cannot be proven: a concurrent or eventually-consistent
87
+ * connector looks the same as this attempt's. So cleanup fails closed. It
88
+ * removes nothing and instead surfaces the connectors that look like this
89
+ * attempt's, matching `expectedUid` and absent from the pre-create snapshot, so
90
+ * the caller can stop rather than risk removing a bystander's connector.
91
+ */
92
+ export declare function cleanupCreatedAttempt(context: SlackConnectorCleanupContext, input: {
93
+ expectedUid: string;
94
+ baselineConnectorUids: ReadonlySet<string>;
95
+ createdRef: SlackConnectorRef | undefined;
96
+ }): Promise<SlackConnectorCleanupResult>;
97
+ export {};
@@ -0,0 +1 @@
1
+ import{parseInstallation,parseSlackConnectors,pickSlackConnector}from"./slack-connect.js";import{join}from"node:path";import{SLACK_CHANNEL_DEFAULT_ROUTE}from"#setup/scaffold/index.js";import{readFile}from"node:fs/promises";import"#setup/cli/index.js";import"#setup/primitives/run-vercel.js";const CONNECT_LOOKUP_TIMEOUT_MS=6e4,CONNECT_MUTATION_TIMEOUT_MS=2*6e4;async function listSlackConnectors(e,n,r,i){let a=await e.captureVercel([`connect`,`list`,`-F`,`json`,`--all-projects`],{cwd:n,onOutput:r,timeoutMs:CONNECT_LOOKUP_TIMEOUT_MS,signal:i});if(!a.ok)return{state:`failed`,message:a.failure.message};try{let e=JSON.parse(a.stdout);if(typeof e!=`object`||!e)return{state:`failed`,message:`Vercel returned a malformed connector list.`};let n=e;return Array.isArray(n.connectors??n.clients)?{state:`ok`,body:e,connectors:parseSlackConnectors(e)}:{state:`failed`,message:`Vercel returned a malformed connector list.`}}catch{return{state:`failed`,message:`Vercel returned invalid JSON for the connector list.`}}}async function readProjectId(e){try{let t=await readFile(join(e,`.vercel`,`project.json`),`utf8`),n=JSON.parse(t);return typeof n.projectId==`string`?n.projectId:void 0}catch{return}}async function findSlackConnector(e,t,r,i,a,o){let s=await listSlackConnectors(e,t,a,o);if(s.state===`failed`)return s;let c=new Set(s.connectors.map(e=>e.uid)),l=pickSlackConnector(s.body,r,i);return l===void 0?{state:`not-found`,connectorUids:c}:{state:`found`,connector:l,connectorUids:c}}async function attachSlackConnector(e,t,n,r,a){return await e.runVercel([`connect`,`detach`,n.uid,`--yes`],{cwd:t,onOutput:r,nonInteractive:!0,timeoutMs:CONNECT_MUTATION_TIMEOUT_MS,signal:a})?await e.runVercel([`connect`,`attach`,n.uid,`--triggers`,`--trigger-path`,SLACK_CHANNEL_DEFAULT_ROUTE,`--yes`],{cwd:t,onOutput:r,nonInteractive:!0,timeoutMs:CONNECT_MUTATION_TIMEOUT_MS,signal:a})?{state:`attached`}:{state:`attach-failed`}:{state:`detach-failed`}}async function fetchInstallationInfo(t,n,r,i,a,o){let s=await t.captureVercel([`api`,`/v1/connex/clients/${r}/installations`],{cwd:n,onOutput:i,timeoutMs:a,signal:o});if(!s.ok)return{state:`failed`,message:s.failure.message};try{let t=JSON.parse(s.stdout);if(typeof t!=`object`||!t)return{state:`failed`,message:`Vercel returned a malformed installation response.`};let n=t.installations;if(!Array.isArray(n))return{state:`failed`,message:`Vercel returned a malformed installation response.`};if(n.length===0)return{state:`pending`};let r=parseInstallation(t);return r===void 0?{state:`failed`,message:`Vercel returned an invalid Slack installation.`}:{state:`installed`,info:r}}catch{return{state:`failed`,message:`Vercel returned invalid JSON for Slack installations.`}}}async function cleanupConnectorUid(e,t){let{log:n,deps:r,projectRoot:i,onOutput:a}=e;return await r.runVercel([`connect`,`remove`,t,`--disconnect-all`,`--yes`],{cwd:i,onOutput:a,timeoutMs:CONNECT_MUTATION_TIMEOUT_MS})?!0:(n.warning(`Could not remove the abandoned Slack connector. Run \`vercel connect remove ${t} --disconnect-all --yes\` to clean it up.`),!1)}async function cleanupConnectorUids(e,t){let n=[];for(let r of new Set(t))await cleanupConnectorUid(e,r)||n.push(r);return n.length===0?{state:`clean`}:{state:`failed`,connectorUids:n}}async function cleanupCreatedAttempt(e,t){if(t.createdRef)return cleanupConnectorUids(e,[t.createdRef.uid]);e.log.warning(`Vercel returned no connector UID for the abandoned Slack Connect request, so Eve cannot prove that request was cancelled. No connector was removed; do not retry until the browser request is no longer usable.`);let n=await listSlackConnectors(e.deps,e.projectRoot,e.onOutput);return n.state===`failed`?{state:`failed`,connectorUids:[]}:{state:`failed`,connectorUids:n.connectors.map(e=>e.uid).filter(e=>(e===t.expectedUid||e.startsWith(`${t.expectedUid}-`))&&!t.baselineConnectorUids.has(e))}}export{CONNECT_LOOKUP_TIMEOUT_MS,CONNECT_MUTATION_TIMEOUT_MS,attachSlackConnector,cleanupCreatedAttempt,fetchInstallationInfo,findSlackConnector,listSlackConnectors,readProjectId};
@@ -0,0 +1,51 @@
1
+ /**
2
+ * Pure parsers for Vercel Connect CLI/API payloads used by Slackbot
3
+ * provisioning: connector-list shapes, the `connect create` stdout, and the
4
+ * installations response. No subprocesses or I/O — just shape validation — so
5
+ * the provisioning orchestrator ({@link import("./slackbot.js")}) stays focused
6
+ * on flow and these stay trivially testable.
7
+ */
8
+ export interface VercelConnectListResponse {
9
+ /** `vercel connect list -F json` (current CLI). */
10
+ connectors?: unknown;
11
+ /** Older CLI builds emitted the same array under `clients`. */
12
+ clients?: unknown;
13
+ }
14
+ /** Identifiers returned by Vercel Connect for a Slack connector. */
15
+ export interface SlackConnectorRef {
16
+ uid: string;
17
+ id: string;
18
+ }
19
+ /**
20
+ * Reads the connector identifiers from `vercel connect create … -F json`
21
+ * stdout, the authoritative source for the just-created connector's UID.
22
+ * Returns `undefined` when stdout is empty or not the expected JSON.
23
+ */
24
+ export declare function parseCreatedSlackConnector(stdout: string): SlackConnectorRef | undefined;
25
+ /**
26
+ * Finds the expected connector, or the newest Slack connector already attached to this project.
27
+ */
28
+ export declare function pickSlackConnector(listJson: unknown, projectId: string | undefined, expectedUid: string | undefined): SlackConnectorRef | undefined;
29
+ export interface ConnectInstallationsResponse {
30
+ installations?: unknown;
31
+ }
32
+ /** Extracts Slack workspace information from a Connect installations response. */
33
+ export declare function parseInstallation(body: unknown): {
34
+ workspaceUrl: string;
35
+ workspaceName?: string;
36
+ } | undefined;
37
+ /**
38
+ * Turns a Slack `app_redirect` install URL into a deep link that opens the
39
+ * Messages tab, a DM compose with the bot, instead of the app's about page.
40
+ * Slack honors `tab=messages` only on `app_redirect` links, the ones carrying
41
+ * `app` and `team` ids; any other URL is returned unchanged.
42
+ * See https://docs.slack.dev/interactivity/deep-linking/.
43
+ */
44
+ export declare function slackMessageDeepLink(url: string): string;
45
+ /** A Slack connector plus the project ids it is attached to. */
46
+ export interface RawSlackConnector {
47
+ uid: string;
48
+ projectIds: readonly string[];
49
+ }
50
+ /** Parses Slack connectors (uid + attached project ids) from a connect-list response. */
51
+ export declare function parseSlackConnectors(listJson: unknown): RawSlackConnector[];
@@ -0,0 +1 @@
1
+ function parseCreatedSlackConnector(e){let t=e.trim();if(!t)return;let n;try{n=JSON.parse(t)}catch{return}if(typeof n!=`object`||!n)return;let{uid:r,id:i}=n;if(!(typeof r!=`string`||typeof i!=`string`))return{uid:r,id:i}}function pickSlackConnector(e,t,n){if(t===void 0||typeof e!=`object`||!e)return;let r=e,i=r.connectors??r.clients;if(!Array.isArray(i))return;let a;for(let e of i){if(e.type!==`slack`||typeof e.uid!=`string`||typeof e.id!=`string`||!Array.isArray(e.projects)||!e.projects.some(e=>typeof e==`object`&&!!e&&e.id===t))continue;let r={uid:e.uid,id:e.id};if(n!==void 0&&r.uid===n)return r;let i=typeof e.createdAt==`number`?e.createdAt:0;(!a||i>a.createdAt)&&(a={ref:r,createdAt:i})}return a?.ref}function parseInstallation(e){if(typeof e!=`object`||!e)return;let t=e.installations;if(!Array.isArray(t)||t.length===0)return;let n=t[0];if(typeof n.tenantUrl==`string`)return{workspaceUrl:n.tenantUrl,workspaceName:typeof n.tenantName==`string`?n.tenantName:void 0}}function slackMessageDeepLink(e){let t=URL.parse(e);return t===null||!t.pathname.endsWith(`/app_redirect`)||!t.searchParams.has(`app`)||!t.searchParams.has(`team`)?e:(t.searchParams.set(`tab`,`messages`),t.href)}function parseSlackConnectors(e){if(typeof e!=`object`||!e)return[];let t=e,n=t.connectors??t.clients;if(!Array.isArray(n))return[];let r=[];for(let e of n){if(e.type!==`slack`||typeof e.uid!=`string`)continue;let t=Array.isArray(e.projects)?e.projects.map(e=>typeof e==`object`&&e?e.id:void 0).filter(e=>typeof e==`string`):[];r.push({uid:e.uid,projectIds:t})}return r}export{parseCreatedSlackConnector,parseInstallation,parseSlackConnectors,pickSlackConnector,slackMessageDeepLink};