innocode 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (434) hide show
  1. package/AGENTS.md +27 -0
  2. package/Dockerfile +18 -0
  3. package/README.md +15 -0
  4. package/bin/innocode +84 -0
  5. package/bin/opencode +84 -0
  6. package/bunfig.toml +5 -0
  7. package/package.json +126 -0
  8. package/parsers-config.ts +253 -0
  9. package/script/build.ts +198 -0
  10. package/script/postinstall.mjs +125 -0
  11. package/script/publish.ts +186 -0
  12. package/script/schema.ts +47 -0
  13. package/script/seed-e2e.ts +50 -0
  14. package/src/acp/README.md +164 -0
  15. package/src/acp/agent.ts +1676 -0
  16. package/src/acp/session.ts +117 -0
  17. package/src/acp/types.ts +23 -0
  18. package/src/agent/agent.ts +338 -0
  19. package/src/agent/generate.txt +75 -0
  20. package/src/agent/prompt/compaction.txt +14 -0
  21. package/src/agent/prompt/explore.txt +18 -0
  22. package/src/agent/prompt/summary.txt +11 -0
  23. package/src/agent/prompt/title.txt +44 -0
  24. package/src/auth/index.ts +70 -0
  25. package/src/bun/index.ts +137 -0
  26. package/src/bun/registry.ts +48 -0
  27. package/src/bus/bus-event.ts +43 -0
  28. package/src/bus/global.ts +10 -0
  29. package/src/bus/index.ts +105 -0
  30. package/src/cli/bootstrap.ts +17 -0
  31. package/src/cli/cmd/acp.ts +70 -0
  32. package/src/cli/cmd/agent.ts +257 -0
  33. package/src/cli/cmd/auth.ts +400 -0
  34. package/src/cli/cmd/cmd.ts +7 -0
  35. package/src/cli/cmd/debug/agent.ts +167 -0
  36. package/src/cli/cmd/debug/config.ts +16 -0
  37. package/src/cli/cmd/debug/file.ts +97 -0
  38. package/src/cli/cmd/debug/index.ts +48 -0
  39. package/src/cli/cmd/debug/lsp.ts +52 -0
  40. package/src/cli/cmd/debug/ripgrep.ts +87 -0
  41. package/src/cli/cmd/debug/scrap.ts +16 -0
  42. package/src/cli/cmd/debug/skill.ts +16 -0
  43. package/src/cli/cmd/debug/snapshot.ts +52 -0
  44. package/src/cli/cmd/export.ts +88 -0
  45. package/src/cli/cmd/generate.ts +38 -0
  46. package/src/cli/cmd/github.ts +1540 -0
  47. package/src/cli/cmd/import.ts +147 -0
  48. package/src/cli/cmd/mcp.ts +765 -0
  49. package/src/cli/cmd/models.ts +77 -0
  50. package/src/cli/cmd/pr.ts +113 -0
  51. package/src/cli/cmd/run.ts +598 -0
  52. package/src/cli/cmd/serve.ts +20 -0
  53. package/src/cli/cmd/session.ts +135 -0
  54. package/src/cli/cmd/stats.ts +426 -0
  55. package/src/cli/cmd/tui/app.tsx +812 -0
  56. package/src/cli/cmd/tui/attach.ts +60 -0
  57. package/src/cli/cmd/tui/component/border.tsx +21 -0
  58. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  59. package/src/cli/cmd/tui/component/dialog-command.tsx +148 -0
  60. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  61. package/src/cli/cmd/tui/component/dialog-model.tsx +165 -0
  62. package/src/cli/cmd/tui/component/dialog-provider.tsx +243 -0
  63. package/src/cli/cmd/tui/component/dialog-session-list.tsx +108 -0
  64. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  65. package/src/cli/cmd/tui/component/dialog-skill.tsx +36 -0
  66. package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
  67. package/src/cli/cmd/tui/component/dialog-status.tsx +167 -0
  68. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  69. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  70. package/src/cli/cmd/tui/component/logo.tsx +85 -0
  71. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +666 -0
  72. package/src/cli/cmd/tui/component/prompt/frecency.tsx +89 -0
  73. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  74. package/src/cli/cmd/tui/component/prompt/index.tsx +1153 -0
  75. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  76. package/src/cli/cmd/tui/component/spinner.tsx +24 -0
  77. package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
  78. package/src/cli/cmd/tui/component/tips.tsx +153 -0
  79. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  80. package/src/cli/cmd/tui/context/args.tsx +15 -0
  81. package/src/cli/cmd/tui/context/directory.ts +13 -0
  82. package/src/cli/cmd/tui/context/exit.tsx +54 -0
  83. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  84. package/src/cli/cmd/tui/context/keybind.tsx +100 -0
  85. package/src/cli/cmd/tui/context/kv.tsx +52 -0
  86. package/src/cli/cmd/tui/context/local.tsx +409 -0
  87. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  88. package/src/cli/cmd/tui/context/route.tsx +46 -0
  89. package/src/cli/cmd/tui/context/sdk.tsx +101 -0
  90. package/src/cli/cmd/tui/context/sync.tsx +470 -0
  91. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  92. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  93. package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
  94. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  95. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  96. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  97. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  98. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  99. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  100. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  101. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  102. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  103. package/src/cli/cmd/tui/context/theme/gruvbox.json +242 -0
  104. package/src/cli/cmd/tui/context/theme/innocode.json +245 -0
  105. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  106. package/src/cli/cmd/tui/context/theme/lucent-orng.json +237 -0
  107. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  108. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  109. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  110. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  111. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  112. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  113. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  114. package/src/cli/cmd/tui/context/theme/orng.json +249 -0
  115. package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
  116. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  117. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  118. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  119. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  120. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  121. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  122. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  123. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  124. package/src/cli/cmd/tui/context/theme.tsx +1154 -0
  125. package/src/cli/cmd/tui/event.ts +48 -0
  126. package/src/cli/cmd/tui/routes/home.tsx +145 -0
  127. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  128. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  129. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  130. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  131. package/src/cli/cmd/tui/routes/session/footer.tsx +91 -0
  132. package/src/cli/cmd/tui/routes/session/header.tsx +135 -0
  133. package/src/cli/cmd/tui/routes/session/index.tsx +2139 -0
  134. package/src/cli/cmd/tui/routes/session/permission.tsx +508 -0
  135. package/src/cli/cmd/tui/routes/session/question.tsx +466 -0
  136. package/src/cli/cmd/tui/routes/session/sidebar.tsx +313 -0
  137. package/src/cli/cmd/tui/thread.ts +188 -0
  138. package/src/cli/cmd/tui/ui/dialog-alert.tsx +59 -0
  139. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +85 -0
  140. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +207 -0
  141. package/src/cli/cmd/tui/ui/dialog-help.tsx +40 -0
  142. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +80 -0
  143. package/src/cli/cmd/tui/ui/dialog-select.tsx +401 -0
  144. package/src/cli/cmd/tui/ui/dialog.tsx +167 -0
  145. package/src/cli/cmd/tui/ui/link.tsx +28 -0
  146. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  147. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  148. package/src/cli/cmd/tui/util/clipboard.ts +159 -0
  149. package/src/cli/cmd/tui/util/editor.ts +32 -0
  150. package/src/cli/cmd/tui/util/signal.ts +7 -0
  151. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  152. package/src/cli/cmd/tui/util/transcript.ts +98 -0
  153. package/src/cli/cmd/tui/win32.ts +129 -0
  154. package/src/cli/cmd/tui/worker.ts +152 -0
  155. package/src/cli/cmd/uninstall.ts +363 -0
  156. package/src/cli/cmd/upgrade.ts +73 -0
  157. package/src/cli/cmd/web.ts +81 -0
  158. package/src/cli/error.ts +57 -0
  159. package/src/cli/logo.ts +6 -0
  160. package/src/cli/network.ts +60 -0
  161. package/src/cli/ui.ts +113 -0
  162. package/src/cli/upgrade.ts +25 -0
  163. package/src/command/index.ts +150 -0
  164. package/src/command/template/initialize.txt +10 -0
  165. package/src/command/template/review.txt +101 -0
  166. package/src/config/config.ts +1517 -0
  167. package/src/config/markdown.ts +98 -0
  168. package/src/env/index.ts +28 -0
  169. package/src/file/ignore.ts +83 -0
  170. package/src/file/index.ts +583 -0
  171. package/src/file/ripgrep.ts +375 -0
  172. package/src/file/time.ts +69 -0
  173. package/src/file/watcher.ts +127 -0
  174. package/src/flag/flag.ts +148 -0
  175. package/src/format/formatter.ts +366 -0
  176. package/src/format/index.ts +137 -0
  177. package/src/global/index.ts +80 -0
  178. package/src/id/id.ts +83 -0
  179. package/src/ide/index.ts +76 -0
  180. package/src/index.ts +160 -0
  181. package/src/installation/index.ts +268 -0
  182. package/src/lsp/client.ts +252 -0
  183. package/src/lsp/index.ts +485 -0
  184. package/src/lsp/language.ts +119 -0
  185. package/src/lsp/server.ts +2046 -0
  186. package/src/mcp/auth.ts +132 -0
  187. package/src/mcp/index.ts +937 -0
  188. package/src/mcp/oauth-callback.ts +200 -0
  189. package/src/mcp/oauth-provider.ts +154 -0
  190. package/src/patch/index.ts +680 -0
  191. package/src/permission/arity.ts +163 -0
  192. package/src/permission/index.ts +210 -0
  193. package/src/permission/next.ts +280 -0
  194. package/src/plugin/codex.ts +624 -0
  195. package/src/plugin/copilot.ts +327 -0
  196. package/src/plugin/index.ts +138 -0
  197. package/src/project/bootstrap.ts +35 -0
  198. package/src/project/instance.ts +114 -0
  199. package/src/project/project.ts +371 -0
  200. package/src/project/state.ts +70 -0
  201. package/src/project/vcs.ts +76 -0
  202. package/src/provider/auth.ts +147 -0
  203. package/src/provider/error.ts +189 -0
  204. package/src/provider/models.ts +133 -0
  205. package/src/provider/provider.ts +1370 -0
  206. package/src/provider/sdk/copilot/README.md +5 -0
  207. package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +164 -0
  208. package/src/provider/sdk/copilot/chat/get-response-metadata.ts +15 -0
  209. package/src/provider/sdk/copilot/chat/map-openai-compatible-finish-reason.ts +17 -0
  210. package/src/provider/sdk/copilot/chat/openai-compatible-api-types.ts +64 -0
  211. package/src/provider/sdk/copilot/chat/openai-compatible-chat-language-model.ts +780 -0
  212. package/src/provider/sdk/copilot/chat/openai-compatible-chat-options.ts +28 -0
  213. package/src/provider/sdk/copilot/chat/openai-compatible-metadata-extractor.ts +44 -0
  214. package/src/provider/sdk/copilot/chat/openai-compatible-prepare-tools.ts +87 -0
  215. package/src/provider/sdk/copilot/copilot-provider.ts +100 -0
  216. package/src/provider/sdk/copilot/index.ts +2 -0
  217. package/src/provider/sdk/copilot/openai-compatible-error.ts +27 -0
  218. package/src/provider/sdk/copilot/responses/convert-to-openai-responses-input.ts +303 -0
  219. package/src/provider/sdk/copilot/responses/map-openai-responses-finish-reason.ts +22 -0
  220. package/src/provider/sdk/copilot/responses/openai-config.ts +18 -0
  221. package/src/provider/sdk/copilot/responses/openai-error.ts +22 -0
  222. package/src/provider/sdk/copilot/responses/openai-responses-api-types.ts +207 -0
  223. package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +1732 -0
  224. package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +177 -0
  225. package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +1 -0
  226. package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +88 -0
  227. package/src/provider/sdk/copilot/responses/tool/file-search.ts +128 -0
  228. package/src/provider/sdk/copilot/responses/tool/image-generation.ts +115 -0
  229. package/src/provider/sdk/copilot/responses/tool/local-shell.ts +65 -0
  230. package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +104 -0
  231. package/src/provider/sdk/copilot/responses/tool/web-search.ts +103 -0
  232. package/src/provider/transform.ts +806 -0
  233. package/src/pty/index.ts +286 -0
  234. package/src/question/index.ts +171 -0
  235. package/src/scheduler/index.ts +61 -0
  236. package/src/server/error.ts +36 -0
  237. package/src/server/event.ts +7 -0
  238. package/src/server/mdns.ts +60 -0
  239. package/src/server/routes/config.ts +92 -0
  240. package/src/server/routes/experimental.ts +208 -0
  241. package/src/server/routes/file.ts +197 -0
  242. package/src/server/routes/global.ts +183 -0
  243. package/src/server/routes/mcp.ts +225 -0
  244. package/src/server/routes/permission.ts +68 -0
  245. package/src/server/routes/project.ts +82 -0
  246. package/src/server/routes/provider.ts +179 -0
  247. package/src/server/routes/pty.ts +176 -0
  248. package/src/server/routes/question.ts +98 -0
  249. package/src/server/routes/session.ts +939 -0
  250. package/src/server/routes/tui.ts +379 -0
  251. package/src/server/server.ts +621 -0
  252. package/src/session/compaction.ts +261 -0
  253. package/src/session/index.ts +543 -0
  254. package/src/session/instruction.ts +197 -0
  255. package/src/session/llm.ts +283 -0
  256. package/src/session/message-v2.ts +841 -0
  257. package/src/session/message.ts +189 -0
  258. package/src/session/processor.ts +410 -0
  259. package/src/session/prompt/anthropic-20250930.txt +166 -0
  260. package/src/session/prompt/anthropic.txt +105 -0
  261. package/src/session/prompt/beast.txt +147 -0
  262. package/src/session/prompt/build-switch.txt +5 -0
  263. package/src/session/prompt/codex_header.txt +79 -0
  264. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  265. package/src/session/prompt/gemini.txt +155 -0
  266. package/src/session/prompt/max-steps.txt +16 -0
  267. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  268. package/src/session/prompt/plan.txt +26 -0
  269. package/src/session/prompt/qwen.txt +109 -0
  270. package/src/session/prompt/trinity.txt +97 -0
  271. package/src/session/prompt.ts +1964 -0
  272. package/src/session/retry.ts +101 -0
  273. package/src/session/revert.ts +121 -0
  274. package/src/session/status.ts +76 -0
  275. package/src/session/summary.ts +203 -0
  276. package/src/session/system.ts +54 -0
  277. package/src/session/todo.ts +37 -0
  278. package/src/share/share-next.ts +200 -0
  279. package/src/share/share.ts +92 -0
  280. package/src/shell/shell.ts +67 -0
  281. package/src/skill/discovery.ts +97 -0
  282. package/src/skill/index.ts +1 -0
  283. package/src/skill/skill.ts +188 -0
  284. package/src/snapshot/index.ts +255 -0
  285. package/src/storage/storage.ts +227 -0
  286. package/src/tool/apply_patch.ts +281 -0
  287. package/src/tool/apply_patch.txt +33 -0
  288. package/src/tool/bash.ts +269 -0
  289. package/src/tool/bash.txt +115 -0
  290. package/src/tool/batch.ts +175 -0
  291. package/src/tool/batch.txt +24 -0
  292. package/src/tool/codesearch.ts +132 -0
  293. package/src/tool/codesearch.txt +12 -0
  294. package/src/tool/edit.ts +655 -0
  295. package/src/tool/edit.txt +10 -0
  296. package/src/tool/external-directory.ts +32 -0
  297. package/src/tool/glob.ts +80 -0
  298. package/src/tool/glob.txt +6 -0
  299. package/src/tool/grep.ts +150 -0
  300. package/src/tool/grep.txt +8 -0
  301. package/src/tool/invalid.ts +17 -0
  302. package/src/tool/ls.ts +121 -0
  303. package/src/tool/ls.txt +1 -0
  304. package/src/tool/lsp.ts +96 -0
  305. package/src/tool/lsp.txt +19 -0
  306. package/src/tool/multiedit.ts +46 -0
  307. package/src/tool/multiedit.txt +41 -0
  308. package/src/tool/plan-enter.txt +14 -0
  309. package/src/tool/plan-exit.txt +13 -0
  310. package/src/tool/plan.ts +130 -0
  311. package/src/tool/question.ts +33 -0
  312. package/src/tool/question.txt +10 -0
  313. package/src/tool/read.ts +261 -0
  314. package/src/tool/read.txt +14 -0
  315. package/src/tool/registry.ts +160 -0
  316. package/src/tool/skill.ts +123 -0
  317. package/src/tool/task.ts +165 -0
  318. package/src/tool/task.txt +60 -0
  319. package/src/tool/todo.ts +53 -0
  320. package/src/tool/todoread.txt +14 -0
  321. package/src/tool/todowrite.txt +167 -0
  322. package/src/tool/tool.ts +89 -0
  323. package/src/tool/truncation.ts +106 -0
  324. package/src/tool/webfetch.ts +186 -0
  325. package/src/tool/webfetch.txt +13 -0
  326. package/src/tool/websearch.ts +150 -0
  327. package/src/tool/websearch.txt +14 -0
  328. package/src/tool/write.ts +85 -0
  329. package/src/tool/write.txt +8 -0
  330. package/src/util/abort.ts +35 -0
  331. package/src/util/archive.ts +16 -0
  332. package/src/util/color.ts +19 -0
  333. package/src/util/context.ts +25 -0
  334. package/src/util/defer.ts +12 -0
  335. package/src/util/eventloop.ts +20 -0
  336. package/src/util/filesystem.ts +93 -0
  337. package/src/util/fn.ts +11 -0
  338. package/src/util/format.ts +20 -0
  339. package/src/util/iife.ts +3 -0
  340. package/src/util/keybind.ts +103 -0
  341. package/src/util/lazy.ts +18 -0
  342. package/src/util/locale.ts +81 -0
  343. package/src/util/lock.ts +98 -0
  344. package/src/util/log.ts +180 -0
  345. package/src/util/proxied.ts +3 -0
  346. package/src/util/queue.ts +32 -0
  347. package/src/util/rpc.ts +66 -0
  348. package/src/util/scrap.ts +10 -0
  349. package/src/util/signal.ts +12 -0
  350. package/src/util/timeout.ts +14 -0
  351. package/src/util/token.ts +7 -0
  352. package/src/util/wildcard.ts +56 -0
  353. package/src/worktree/index.ts +612 -0
  354. package/sst-env.d.ts +9 -0
  355. package/test/acp/agent-interface.test.ts +51 -0
  356. package/test/acp/event-subscription.test.ts +436 -0
  357. package/test/agent/agent.test.ts +675 -0
  358. package/test/bun.test.ts +53 -0
  359. package/test/cli/github-action.test.ts +161 -0
  360. package/test/cli/github-remote.test.ts +80 -0
  361. package/test/cli/import.test.ts +38 -0
  362. package/test/cli/tui/transcript.test.ts +322 -0
  363. package/test/config/agent-color.test.ts +71 -0
  364. package/test/config/config.test.ts +1802 -0
  365. package/test/config/fixtures/empty-frontmatter.md +4 -0
  366. package/test/config/fixtures/frontmatter.md +28 -0
  367. package/test/config/fixtures/markdown-header.md +11 -0
  368. package/test/config/fixtures/no-frontmatter.md +1 -0
  369. package/test/config/fixtures/weird-model-id.md +13 -0
  370. package/test/config/markdown.test.ts +228 -0
  371. package/test/file/ignore.test.ts +10 -0
  372. package/test/file/path-traversal.test.ts +198 -0
  373. package/test/file/ripgrep.test.ts +39 -0
  374. package/test/fixture/fixture.ts +45 -0
  375. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  376. package/test/ide/ide.test.ts +82 -0
  377. package/test/keybind.test.ts +421 -0
  378. package/test/lsp/client.test.ts +95 -0
  379. package/test/mcp/headers.test.ts +153 -0
  380. package/test/mcp/oauth-browser.test.ts +249 -0
  381. package/test/memory/abort-leak.test.ts +136 -0
  382. package/test/patch/patch.test.ts +348 -0
  383. package/test/permission/arity.test.ts +33 -0
  384. package/test/permission/next.test.ts +690 -0
  385. package/test/permission-task.test.ts +319 -0
  386. package/test/plugin/auth-override.test.ts +44 -0
  387. package/test/plugin/codex.test.ts +123 -0
  388. package/test/preload.ts +63 -0
  389. package/test/project/project.test.ts +120 -0
  390. package/test/provider/amazon-bedrock.test.ts +445 -0
  391. package/test/provider/copilot/convert-to-copilot-messages.test.ts +523 -0
  392. package/test/provider/copilot/copilot-chat-model.test.ts +592 -0
  393. package/test/provider/gitlab-duo.test.ts +262 -0
  394. package/test/provider/provider.test.ts +2129 -0
  395. package/test/provider/transform.test.ts +1928 -0
  396. package/test/question/question.test.ts +300 -0
  397. package/test/scheduler.test.ts +73 -0
  398. package/test/server/session-list.test.ts +39 -0
  399. package/test/server/session-select.test.ts +78 -0
  400. package/test/session/compaction.test.ts +423 -0
  401. package/test/session/instruction.test.ts +170 -0
  402. package/test/session/llm.test.ts +667 -0
  403. package/test/session/message-v2.test.ts +924 -0
  404. package/test/session/prompt-missing-file.test.ts +53 -0
  405. package/test/session/prompt-special-chars.test.ts +56 -0
  406. package/test/session/prompt-variant.test.ts +68 -0
  407. package/test/session/retry.test.ts +188 -0
  408. package/test/session/revert-compact.test.ts +285 -0
  409. package/test/session/session.test.ts +71 -0
  410. package/test/session/structured-output-integration.test.ts +233 -0
  411. package/test/session/structured-output.test.ts +385 -0
  412. package/test/skill/discovery.test.ts +60 -0
  413. package/test/skill/skill.test.ts +388 -0
  414. package/test/snapshot/snapshot.test.ts +1040 -0
  415. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  416. package/test/tool/apply_patch.test.ts +559 -0
  417. package/test/tool/bash.test.ts +399 -0
  418. package/test/tool/external-directory.test.ts +127 -0
  419. package/test/tool/fixtures/large-image.png +0 -0
  420. package/test/tool/fixtures/models-api.json +38413 -0
  421. package/test/tool/grep.test.ts +110 -0
  422. package/test/tool/question.test.ts +107 -0
  423. package/test/tool/read.test.ts +421 -0
  424. package/test/tool/registry.test.ts +122 -0
  425. package/test/tool/skill.test.ts +112 -0
  426. package/test/tool/truncation.test.ts +159 -0
  427. package/test/util/filesystem.test.ts +39 -0
  428. package/test/util/format.test.ts +59 -0
  429. package/test/util/iife.test.ts +36 -0
  430. package/test/util/lazy.test.ts +50 -0
  431. package/test/util/lock.test.ts +72 -0
  432. package/test/util/timeout.test.ts +21 -0
  433. package/test/util/wildcard.test.ts +75 -0
  434. package/tsconfig.json +16 -0
@@ -0,0 +1,598 @@
1
+ import type { Argv } from "yargs"
2
+ import path from "path"
3
+ import { pathToFileURL } from "bun"
4
+ import { UI } from "../ui"
5
+ import { cmd } from "./cmd"
6
+ import { Flag } from "../../flag/flag"
7
+ import { bootstrap } from "../bootstrap"
8
+ import { EOL } from "os"
9
+ import { createOpencodeClient, type Message, type OpencodeClient, type ToolPart } from "@opencode-ai/sdk/v2"
10
+ import { Server } from "../../server/server"
11
+ import { Provider } from "../../provider/provider"
12
+ import { Agent } from "../../agent/agent"
13
+ import { PermissionNext } from "../../permission/next"
14
+ import { Tool } from "../../tool/tool"
15
+ import { GlobTool } from "../../tool/glob"
16
+ import { GrepTool } from "../../tool/grep"
17
+ import { ListTool } from "../../tool/ls"
18
+ import { ReadTool } from "../../tool/read"
19
+ import { WebFetchTool } from "../../tool/webfetch"
20
+ import { EditTool } from "../../tool/edit"
21
+ import { WriteTool } from "../../tool/write"
22
+ import { CodeSearchTool } from "../../tool/codesearch"
23
+ import { WebSearchTool } from "../../tool/websearch"
24
+ import { TaskTool } from "../../tool/task"
25
+ import { SkillTool } from "../../tool/skill"
26
+ import { BashTool } from "../../tool/bash"
27
+ import { TodoWriteTool } from "../../tool/todo"
28
+ import { Locale } from "../../util/locale"
29
+
30
+ type ToolProps<T extends Tool.Info> = {
31
+ input: Tool.InferParameters<T>
32
+ metadata: Tool.InferMetadata<T>
33
+ part: ToolPart
34
+ }
35
+
36
+ function props<T extends Tool.Info>(part: ToolPart): ToolProps<T> {
37
+ const state = part.state
38
+ return {
39
+ input: state.input as Tool.InferParameters<T>,
40
+ metadata: ("metadata" in state ? state.metadata : {}) as Tool.InferMetadata<T>,
41
+ part,
42
+ }
43
+ }
44
+
45
+ type Inline = {
46
+ icon: string
47
+ title: string
48
+ description?: string
49
+ }
50
+
51
+ function inline(info: Inline) {
52
+ const suffix = info.description ? UI.Style.TEXT_DIM + ` ${info.description}` + UI.Style.TEXT_NORMAL : ""
53
+ UI.println(UI.Style.TEXT_NORMAL + info.icon, UI.Style.TEXT_NORMAL + info.title + suffix)
54
+ }
55
+
56
+ function block(info: Inline, output?: string) {
57
+ UI.empty()
58
+ inline(info)
59
+ if (!output?.trim()) return
60
+ UI.println(output)
61
+ UI.empty()
62
+ }
63
+
64
+ function fallback(part: ToolPart) {
65
+ const state = part.state
66
+ const input = "input" in state ? state.input : undefined
67
+ const title =
68
+ ("title" in state && state.title ? state.title : undefined) ||
69
+ (input && typeof input === "object" && Object.keys(input).length > 0 ? JSON.stringify(input) : "Unknown")
70
+ inline({
71
+ icon: "⚙",
72
+ title: `${part.tool} ${title}`,
73
+ })
74
+ }
75
+
76
+ function glob(info: ToolProps<typeof GlobTool>) {
77
+ const root = info.input.path ?? ""
78
+ const title = `Glob "${info.input.pattern}"`
79
+ const suffix = root ? `in ${normalizePath(root)}` : ""
80
+ const num = info.metadata.count
81
+ const description =
82
+ num === undefined ? suffix : `${suffix}${suffix ? " · " : ""}${num} ${num === 1 ? "match" : "matches"}`
83
+ inline({
84
+ icon: "✱",
85
+ title,
86
+ ...(description && { description }),
87
+ })
88
+ }
89
+
90
+ function grep(info: ToolProps<typeof GrepTool>) {
91
+ const root = info.input.path ?? ""
92
+ const title = `Grep "${info.input.pattern}"`
93
+ const suffix = root ? `in ${normalizePath(root)}` : ""
94
+ const num = info.metadata.matches
95
+ const description =
96
+ num === undefined ? suffix : `${suffix}${suffix ? " · " : ""}${num} ${num === 1 ? "match" : "matches"}`
97
+ inline({
98
+ icon: "✱",
99
+ title,
100
+ ...(description && { description }),
101
+ })
102
+ }
103
+
104
+ function list(info: ToolProps<typeof ListTool>) {
105
+ const dir = info.input.path ? normalizePath(info.input.path) : ""
106
+ inline({
107
+ icon: "→",
108
+ title: dir ? `List ${dir}` : "List",
109
+ })
110
+ }
111
+
112
+ function read(info: ToolProps<typeof ReadTool>) {
113
+ const file = normalizePath(info.input.filePath)
114
+ const pairs = Object.entries(info.input).filter(([key, value]) => {
115
+ if (key === "filePath") return false
116
+ return typeof value === "string" || typeof value === "number" || typeof value === "boolean"
117
+ })
118
+ const description = pairs.length ? `[${pairs.map(([key, value]) => `${key}=${value}`).join(", ")}]` : undefined
119
+ inline({
120
+ icon: "→",
121
+ title: `Read ${file}`,
122
+ ...(description && { description }),
123
+ })
124
+ }
125
+
126
+ function write(info: ToolProps<typeof WriteTool>) {
127
+ block(
128
+ {
129
+ icon: "←",
130
+ title: `Write ${normalizePath(info.input.filePath)}`,
131
+ },
132
+ info.part.state.status === "completed" ? info.part.state.output : undefined,
133
+ )
134
+ }
135
+
136
+ function webfetch(info: ToolProps<typeof WebFetchTool>) {
137
+ inline({
138
+ icon: "%",
139
+ title: `WebFetch ${info.input.url}`,
140
+ })
141
+ }
142
+
143
+ function edit(info: ToolProps<typeof EditTool>) {
144
+ const title = normalizePath(info.input.filePath)
145
+ const diff = info.metadata.diff
146
+ block(
147
+ {
148
+ icon: "←",
149
+ title: `Edit ${title}`,
150
+ },
151
+ diff,
152
+ )
153
+ }
154
+
155
+ function codesearch(info: ToolProps<typeof CodeSearchTool>) {
156
+ inline({
157
+ icon: "◇",
158
+ title: `Exa Code Search "${info.input.query}"`,
159
+ })
160
+ }
161
+
162
+ function websearch(info: ToolProps<typeof WebSearchTool>) {
163
+ inline({
164
+ icon: "◈",
165
+ title: `Exa Web Search "${info.input.query}"`,
166
+ })
167
+ }
168
+
169
+ function task(info: ToolProps<typeof TaskTool>) {
170
+ const agent = Locale.titlecase(info.input.subagent_type)
171
+ const desc = info.input.description
172
+ const started = info.part.state.status === "running"
173
+ const name = desc ?? `${agent} Task`
174
+ inline({
175
+ icon: started ? "•" : "✓",
176
+ title: name,
177
+ description: desc ? `${agent} Agent` : undefined,
178
+ })
179
+ }
180
+
181
+ function skill(info: ToolProps<typeof SkillTool>) {
182
+ inline({
183
+ icon: "→",
184
+ title: `Skill "${info.input.name}"`,
185
+ })
186
+ }
187
+
188
+ function bash(info: ToolProps<typeof BashTool>) {
189
+ const output = info.part.state.status === "completed" ? info.part.state.output?.trim() : undefined
190
+ block(
191
+ {
192
+ icon: "$",
193
+ title: `${info.input.command}`,
194
+ },
195
+ output,
196
+ )
197
+ }
198
+
199
+ function todo(info: ToolProps<typeof TodoWriteTool>) {
200
+ block(
201
+ {
202
+ icon: "#",
203
+ title: "Todos",
204
+ },
205
+ info.input.todos.map((item) => `${item.status === "completed" ? "[x]" : "[ ]"} ${item.content}`).join("\n"),
206
+ )
207
+ }
208
+
209
+ function normalizePath(input?: string) {
210
+ if (!input) return ""
211
+ if (path.isAbsolute(input)) return path.relative(process.cwd(), input) || "."
212
+ return input
213
+ }
214
+
215
+ export const RunCommand = cmd({
216
+ command: "run [message..]",
217
+ describe: "run innocode with a message",
218
+ builder: (yargs: Argv) => {
219
+ return yargs
220
+ .positional("message", {
221
+ describe: "message to send",
222
+ type: "string",
223
+ array: true,
224
+ default: [],
225
+ })
226
+ .option("command", {
227
+ describe: "the command to run, use message for args",
228
+ type: "string",
229
+ })
230
+ .option("continue", {
231
+ alias: ["c"],
232
+ describe: "continue the last session",
233
+ type: "boolean",
234
+ })
235
+ .option("session", {
236
+ alias: ["s"],
237
+ describe: "session id to continue",
238
+ type: "string",
239
+ })
240
+ .option("fork", {
241
+ describe: "fork the session before continuing (requires --continue or --session)",
242
+ type: "boolean",
243
+ })
244
+ .option("share", {
245
+ type: "boolean",
246
+ describe: "share the session",
247
+ })
248
+ .option("model", {
249
+ type: "string",
250
+ alias: ["m"],
251
+ describe: "model to use in the format of provider/model",
252
+ })
253
+ .option("agent", {
254
+ type: "string",
255
+ describe: "agent to use",
256
+ })
257
+ .option("format", {
258
+ type: "string",
259
+ choices: ["default", "json"],
260
+ default: "default",
261
+ describe: "format: default (formatted) or json (raw JSON events)",
262
+ })
263
+ .option("file", {
264
+ alias: ["f"],
265
+ type: "string",
266
+ array: true,
267
+ describe: "file(s) to attach to message",
268
+ })
269
+ .option("title", {
270
+ type: "string",
271
+ describe: "title for the session (uses truncated prompt if no value provided)",
272
+ })
273
+ .option("attach", {
274
+ type: "string",
275
+ describe: "attach to a running innocode server (e.g., http://localhost:4096)",
276
+ })
277
+ .option("port", {
278
+ type: "number",
279
+ describe: "port for the local server (defaults to random port if no value provided)",
280
+ })
281
+ .option("variant", {
282
+ type: "string",
283
+ describe: "model variant (provider-specific reasoning effort, e.g., high, max, minimal)",
284
+ })
285
+ .option("thinking", {
286
+ type: "boolean",
287
+ describe: "show thinking blocks",
288
+ default: false,
289
+ })
290
+ },
291
+ handler: async (args) => {
292
+ let message = [...args.message, ...(args["--"] || [])]
293
+ .map((arg) => (arg.includes(" ") ? `"${arg.replace(/"/g, '\\"')}"` : arg))
294
+ .join(" ")
295
+
296
+ const files: { type: "file"; url: string; filename: string; mime: string }[] = []
297
+ if (args.file) {
298
+ const list = Array.isArray(args.file) ? args.file : [args.file]
299
+
300
+ for (const filePath of list) {
301
+ const resolvedPath = path.resolve(process.cwd(), filePath)
302
+ const file = Bun.file(resolvedPath)
303
+ const stats = await file.stat().catch(() => {})
304
+ if (!stats) {
305
+ UI.error(`File not found: ${filePath}`)
306
+ process.exit(1)
307
+ }
308
+ if (!(await file.exists())) {
309
+ UI.error(`File not found: ${filePath}`)
310
+ process.exit(1)
311
+ }
312
+
313
+ const stat = await file.stat()
314
+ const mime = stat.isDirectory() ? "application/x-directory" : "text/plain"
315
+
316
+ files.push({
317
+ type: "file",
318
+ url: pathToFileURL(resolvedPath).href,
319
+ filename: path.basename(resolvedPath),
320
+ mime,
321
+ })
322
+ }
323
+ }
324
+
325
+ if (!process.stdin.isTTY) message += "\n" + (await Bun.stdin.text())
326
+
327
+ if (message.trim().length === 0 && !args.command) {
328
+ UI.error("You must provide a message or a command")
329
+ process.exit(1)
330
+ }
331
+
332
+ if (args.fork && !args.continue && !args.session) {
333
+ UI.error("--fork requires --continue or --session")
334
+ process.exit(1)
335
+ }
336
+
337
+ const rules: PermissionNext.Ruleset = [
338
+ {
339
+ permission: "question",
340
+ action: "deny",
341
+ pattern: "*",
342
+ },
343
+ {
344
+ permission: "plan_enter",
345
+ action: "deny",
346
+ pattern: "*",
347
+ },
348
+ {
349
+ permission: "plan_exit",
350
+ action: "deny",
351
+ pattern: "*",
352
+ },
353
+ ]
354
+
355
+ function title() {
356
+ if (args.title === undefined) return
357
+ if (args.title !== "") return args.title
358
+ return message.slice(0, 50) + (message.length > 50 ? "..." : "")
359
+ }
360
+
361
+ async function session(sdk: OpencodeClient) {
362
+ const baseID = args.continue ? (await sdk.session.list()).data?.find((s) => !s.parentID)?.id : args.session
363
+
364
+ if (baseID && args.fork) {
365
+ const forked = await sdk.session.fork({ sessionID: baseID })
366
+ return forked.data?.id
367
+ }
368
+
369
+ if (baseID) return baseID
370
+
371
+ const name = title()
372
+ const result = await sdk.session.create({ title: name, permission: rules })
373
+ return result.data?.id
374
+ }
375
+
376
+ async function share(sdk: OpencodeClient, sessionID: string) {
377
+ const cfg = await sdk.config.get()
378
+ if (!cfg.data) return
379
+ if (cfg.data.share !== "auto" && !Flag.OPENCODE_AUTO_SHARE && !args.share) return
380
+ const res = await sdk.session.share({ sessionID }).catch((error) => {
381
+ if (error instanceof Error && error.message.includes("disabled")) {
382
+ UI.println(UI.Style.TEXT_DANGER_BOLD + "! " + error.message)
383
+ }
384
+ return { error }
385
+ })
386
+ if (!res.error && "data" in res && res.data?.share?.url) {
387
+ UI.println(UI.Style.TEXT_INFO_BOLD + "~ " + res.data.share.url)
388
+ }
389
+ }
390
+
391
+ async function execute(sdk: OpencodeClient) {
392
+ function tool(part: ToolPart) {
393
+ if (part.tool === "bash") return bash(props<typeof BashTool>(part))
394
+ if (part.tool === "glob") return glob(props<typeof GlobTool>(part))
395
+ if (part.tool === "grep") return grep(props<typeof GrepTool>(part))
396
+ if (part.tool === "list") return list(props<typeof ListTool>(part))
397
+ if (part.tool === "read") return read(props<typeof ReadTool>(part))
398
+ if (part.tool === "write") return write(props<typeof WriteTool>(part))
399
+ if (part.tool === "webfetch") return webfetch(props<typeof WebFetchTool>(part))
400
+ if (part.tool === "edit") return edit(props<typeof EditTool>(part))
401
+ if (part.tool === "codesearch") return codesearch(props<typeof CodeSearchTool>(part))
402
+ if (part.tool === "websearch") return websearch(props<typeof WebSearchTool>(part))
403
+ if (part.tool === "task") return task(props<typeof TaskTool>(part))
404
+ if (part.tool === "todowrite") return todo(props<typeof TodoWriteTool>(part))
405
+ if (part.tool === "skill") return skill(props<typeof SkillTool>(part))
406
+ return fallback(part)
407
+ }
408
+
409
+ function emit(type: string, data: Record<string, unknown>) {
410
+ if (args.format === "json") {
411
+ process.stdout.write(JSON.stringify({ type, timestamp: Date.now(), sessionID, ...data }) + EOL)
412
+ return true
413
+ }
414
+ return false
415
+ }
416
+
417
+ const events = await sdk.event.subscribe()
418
+ let error: string | undefined
419
+
420
+ async function loop() {
421
+ const toggles = new Map<string, boolean>()
422
+
423
+ for await (const event of events.stream) {
424
+ if (
425
+ event.type === "message.updated" &&
426
+ event.properties.info.role === "assistant" &&
427
+ args.format !== "json" &&
428
+ toggles.get("start") !== true
429
+ ) {
430
+ UI.empty()
431
+ UI.println(`> ${event.properties.info.agent} · ${event.properties.info.modelID}`)
432
+ UI.empty()
433
+ toggles.set("start", true)
434
+ }
435
+
436
+ if (event.type === "message.part.updated") {
437
+ const part = event.properties.part
438
+ if (part.sessionID !== sessionID) continue
439
+
440
+ if (part.type === "tool" && part.state.status === "completed") {
441
+ if (emit("tool_use", { part })) continue
442
+ tool(part)
443
+ }
444
+
445
+ if (
446
+ part.type === "tool" &&
447
+ part.tool === "task" &&
448
+ part.state.status === "running" &&
449
+ args.format !== "json"
450
+ ) {
451
+ if (toggles.get(part.id) === true) continue
452
+ task(props<typeof TaskTool>(part))
453
+ toggles.set(part.id, true)
454
+ }
455
+
456
+ if (part.type === "step-start") {
457
+ if (emit("step_start", { part })) continue
458
+ }
459
+
460
+ if (part.type === "step-finish") {
461
+ if (emit("step_finish", { part })) continue
462
+ }
463
+
464
+ if (part.type === "text" && part.time?.end) {
465
+ if (emit("text", { part })) continue
466
+ const text = part.text.trim()
467
+ if (!text) continue
468
+ if (!process.stdout.isTTY) {
469
+ process.stdout.write(text + EOL)
470
+ continue
471
+ }
472
+ UI.empty()
473
+ UI.println(text)
474
+ UI.empty()
475
+ }
476
+
477
+ if (part.type === "reasoning" && part.time?.end && args.thinking) {
478
+ if (emit("reasoning", { part })) continue
479
+ const text = part.text.trim()
480
+ if (!text) continue
481
+ const line = `Thinking: ${text}`
482
+ if (process.stdout.isTTY) {
483
+ UI.empty()
484
+ UI.println(`${UI.Style.TEXT_DIM}\u001b[3m${line}\u001b[0m${UI.Style.TEXT_NORMAL}`)
485
+ UI.empty()
486
+ continue
487
+ }
488
+ process.stdout.write(line + EOL)
489
+ }
490
+ }
491
+
492
+ if (event.type === "session.error") {
493
+ const props = event.properties
494
+ if (props.sessionID !== sessionID || !props.error) continue
495
+ let err = String(props.error.name)
496
+ if ("data" in props.error && props.error.data && "message" in props.error.data) {
497
+ err = String(props.error.data.message)
498
+ }
499
+ error = error ? error + EOL + err : err
500
+ if (emit("error", { error: props.error })) continue
501
+ UI.error(err)
502
+ }
503
+
504
+ if (
505
+ event.type === "session.status" &&
506
+ event.properties.sessionID === sessionID &&
507
+ event.properties.status.type === "idle"
508
+ ) {
509
+ break
510
+ }
511
+
512
+ if (event.type === "permission.asked") {
513
+ const permission = event.properties
514
+ if (permission.sessionID !== sessionID) continue
515
+ UI.println(
516
+ UI.Style.TEXT_WARNING_BOLD + "!",
517
+ UI.Style.TEXT_NORMAL +
518
+ `permission requested: ${permission.permission} (${permission.patterns.join(", ")}); auto-rejecting`,
519
+ )
520
+ await sdk.permission.reply({
521
+ requestID: permission.id,
522
+ reply: "reject",
523
+ })
524
+ }
525
+ }
526
+ }
527
+
528
+ // Validate agent if specified
529
+ const agent = await (async () => {
530
+ if (!args.agent) return undefined
531
+ const entry = await Agent.get(args.agent)
532
+ if (!entry) {
533
+ UI.println(
534
+ UI.Style.TEXT_WARNING_BOLD + "!",
535
+ UI.Style.TEXT_NORMAL,
536
+ `agent "${args.agent}" not found. Falling back to default agent`,
537
+ )
538
+ return undefined
539
+ }
540
+ if (entry.mode === "subagent") {
541
+ UI.println(
542
+ UI.Style.TEXT_WARNING_BOLD + "!",
543
+ UI.Style.TEXT_NORMAL,
544
+ `agent "${args.agent}" is a subagent, not a primary agent. Falling back to default agent`,
545
+ )
546
+ return undefined
547
+ }
548
+ return args.agent
549
+ })()
550
+
551
+ const sessionID = await session(sdk)
552
+ if (!sessionID) {
553
+ UI.error("Session not found")
554
+ process.exit(1)
555
+ }
556
+ await share(sdk, sessionID)
557
+
558
+ loop().catch((e) => {
559
+ console.error(e)
560
+ process.exit(1)
561
+ })
562
+
563
+ if (args.command) {
564
+ await sdk.session.command({
565
+ sessionID,
566
+ agent,
567
+ model: args.model,
568
+ command: args.command,
569
+ arguments: message,
570
+ variant: args.variant,
571
+ })
572
+ } else {
573
+ const model = args.model ? Provider.parseModel(args.model) : undefined
574
+ await sdk.session.prompt({
575
+ sessionID,
576
+ agent,
577
+ model,
578
+ variant: args.variant,
579
+ parts: [...files, { type: "text", text: message }],
580
+ })
581
+ }
582
+ }
583
+
584
+ if (args.attach) {
585
+ const sdk = createOpencodeClient({ baseUrl: args.attach })
586
+ return await execute(sdk)
587
+ }
588
+
589
+ await bootstrap(process.cwd(), async () => {
590
+ const fetchFn = (async (input: RequestInfo | URL, init?: RequestInit) => {
591
+ const request = new Request(input, init)
592
+ return Server.App().fetch(request)
593
+ }) as typeof globalThis.fetch
594
+ const sdk = createOpencodeClient({ baseUrl: "http://opencode.internal", fetch: fetchFn })
595
+ await execute(sdk)
596
+ })
597
+ },
598
+ })
@@ -0,0 +1,20 @@
1
+ import { Server } from "../../server/server"
2
+ import { cmd } from "./cmd"
3
+ import { withNetworkOptions, resolveNetworkOptions } from "../network"
4
+ import { Flag } from "../../flag/flag"
5
+
6
+ export const ServeCommand = cmd({
7
+ command: "serve",
8
+ builder: (yargs) => withNetworkOptions(yargs),
9
+ describe: "starts a headless innocode server",
10
+ handler: async (args) => {
11
+ if (!Flag.INNOCODE_SERVER_PASSWORD) {
12
+ console.log("Warning: INNOCODE_SERVER_PASSWORD is not set; server is unsecured.")
13
+ }
14
+ const opts = await resolveNetworkOptions(args)
15
+ const server = Server.listen(opts)
16
+ console.log(`innocode server listening on http://${server.hostname}:${server.port}`)
17
+ await new Promise(() => {})
18
+ await server.stop()
19
+ },
20
+ })