opencode-v2 1.1.53
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.
- package/AGENTS.md +27 -0
- package/Dockerfile +18 -0
- package/README.md +15 -0
- package/bin/opencode +84 -0
- package/bunfig.toml +5 -0
- package/package.json +126 -0
- package/parsers-config.ts +253 -0
- package/script/build.ts +193 -0
- package/script/postinstall.mjs +125 -0
- package/script/publish.ts +181 -0
- package/script/schema.ts +47 -0
- package/script/seed-e2e.ts +50 -0
- package/src/acp/README.md +164 -0
- package/src/acp/agent.ts +1676 -0
- package/src/acp/session.ts +117 -0
- package/src/acp/types.ts +23 -0
- package/src/agent/agent.ts +414 -0
- package/src/agent/generate.txt +75 -0
- package/src/agent/prompt/compaction.txt +12 -0
- package/src/agent/prompt/explore.txt +18 -0
- package/src/agent/prompt/summary.txt +11 -0
- package/src/agent/prompt/title.txt +44 -0
- package/src/auth/index.ts +70 -0
- package/src/bun/index.ts +137 -0
- package/src/bun/registry.ts +48 -0
- package/src/bus/bus-event.ts +43 -0
- package/src/bus/global.ts +10 -0
- package/src/bus/index.ts +105 -0
- package/src/cli/bootstrap.ts +17 -0
- package/src/cli/cmd/acp.ts +70 -0
- package/src/cli/cmd/agent.ts +257 -0
- package/src/cli/cmd/auth.ts +400 -0
- package/src/cli/cmd/cmd.ts +7 -0
- package/src/cli/cmd/debug/agent.ts +167 -0
- package/src/cli/cmd/debug/config.ts +16 -0
- package/src/cli/cmd/debug/file.ts +97 -0
- package/src/cli/cmd/debug/index.ts +48 -0
- package/src/cli/cmd/debug/lsp.ts +52 -0
- package/src/cli/cmd/debug/ripgrep.ts +87 -0
- package/src/cli/cmd/debug/scrap.ts +16 -0
- package/src/cli/cmd/debug/skill.ts +16 -0
- package/src/cli/cmd/debug/snapshot.ts +52 -0
- package/src/cli/cmd/export.ts +88 -0
- package/src/cli/cmd/generate.ts +38 -0
- package/src/cli/cmd/github.ts +1540 -0
- package/src/cli/cmd/import.ts +147 -0
- package/src/cli/cmd/mcp.ts +755 -0
- package/src/cli/cmd/models.ts +77 -0
- package/src/cli/cmd/pr.ts +112 -0
- package/src/cli/cmd/run.ts +617 -0
- package/src/cli/cmd/serve.ts +20 -0
- package/src/cli/cmd/session.ts +135 -0
- package/src/cli/cmd/stats.ts +426 -0
- package/src/cli/cmd/tui/app.tsx +801 -0
- package/src/cli/cmd/tui/attach.ts +52 -0
- package/src/cli/cmd/tui/component/border.tsx +21 -0
- package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
- package/src/cli/cmd/tui/component/dialog-command.tsx +148 -0
- package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
- package/src/cli/cmd/tui/component/dialog-model.tsx +234 -0
- package/src/cli/cmd/tui/component/dialog-provider.tsx +266 -0
- package/src/cli/cmd/tui/component/dialog-session-list.tsx +108 -0
- package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
- package/src/cli/cmd/tui/component/dialog-skill.tsx +36 -0
- package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
- package/src/cli/cmd/tui/component/dialog-status.tsx +177 -0
- package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
- package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
- package/src/cli/cmd/tui/component/logo.tsx +85 -0
- package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +666 -0
- package/src/cli/cmd/tui/component/prompt/frecency.tsx +89 -0
- package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
- package/src/cli/cmd/tui/component/prompt/index.tsx +1132 -0
- package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
- package/src/cli/cmd/tui/component/spinner.tsx +24 -0
- package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
- package/src/cli/cmd/tui/component/tips.tsx +153 -0
- package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
- package/src/cli/cmd/tui/context/args.tsx +15 -0
- package/src/cli/cmd/tui/context/directory.ts +13 -0
- package/src/cli/cmd/tui/context/exit.tsx +52 -0
- package/src/cli/cmd/tui/context/helper.tsx +25 -0
- package/src/cli/cmd/tui/context/keybind.tsx +100 -0
- package/src/cli/cmd/tui/context/kv.tsx +52 -0
- package/src/cli/cmd/tui/context/local.tsx +409 -0
- package/src/cli/cmd/tui/context/prompt.tsx +18 -0
- package/src/cli/cmd/tui/context/route.tsx +46 -0
- package/src/cli/cmd/tui/context/sdk.tsx +101 -0
- package/src/cli/cmd/tui/context/sync.tsx +470 -0
- package/src/cli/cmd/tui/context/theme/aura.json +69 -0
- package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
- package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
- package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
- package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
- package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
- package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
- package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
- package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
- package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
- package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
- package/src/cli/cmd/tui/context/theme/github.json +233 -0
- package/src/cli/cmd/tui/context/theme/gruvbox.json +242 -0
- package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
- package/src/cli/cmd/tui/context/theme/lucent-orng.json +237 -0
- package/src/cli/cmd/tui/context/theme/material.json +235 -0
- package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
- package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
- package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
- package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
- package/src/cli/cmd/tui/context/theme/nord.json +223 -0
- package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
- package/src/cli/cmd/tui/context/theme/orng.json +249 -0
- package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
- package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
- package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
- package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
- package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
- package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
- package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
- package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
- package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
- package/src/cli/cmd/tui/context/theme.tsx +1152 -0
- package/src/cli/cmd/tui/event.ts +48 -0
- package/src/cli/cmd/tui/routes/home.tsx +140 -0
- package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
- package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
- package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
- package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
- package/src/cli/cmd/tui/routes/session/footer.tsx +91 -0
- package/src/cli/cmd/tui/routes/session/header.tsx +142 -0
- package/src/cli/cmd/tui/routes/session/index.tsx +2126 -0
- package/src/cli/cmd/tui/routes/session/permission.tsx +508 -0
- package/src/cli/cmd/tui/routes/session/question.tsx +466 -0
- package/src/cli/cmd/tui/routes/session/sidebar.tsx +313 -0
- package/src/cli/cmd/tui/thread.ts +175 -0
- package/src/cli/cmd/tui/ui/dialog-alert.tsx +68 -0
- package/src/cli/cmd/tui/ui/dialog-confirm.tsx +93 -0
- package/src/cli/cmd/tui/ui/dialog-export-options.tsx +215 -0
- package/src/cli/cmd/tui/ui/dialog-help.tsx +49 -0
- package/src/cli/cmd/tui/ui/dialog-prompt.tsx +88 -0
- package/src/cli/cmd/tui/ui/dialog-select.tsx +399 -0
- package/src/cli/cmd/tui/ui/dialog.tsx +167 -0
- package/src/cli/cmd/tui/ui/link.tsx +28 -0
- package/src/cli/cmd/tui/ui/spinner.ts +368 -0
- package/src/cli/cmd/tui/ui/toast.tsx +100 -0
- package/src/cli/cmd/tui/util/clipboard.ts +159 -0
- package/src/cli/cmd/tui/util/editor.ts +32 -0
- package/src/cli/cmd/tui/util/signal.ts +7 -0
- package/src/cli/cmd/tui/util/terminal.ts +114 -0
- package/src/cli/cmd/tui/util/transcript.ts +98 -0
- package/src/cli/cmd/tui/worker.ts +152 -0
- package/src/cli/cmd/uninstall.ts +357 -0
- package/src/cli/cmd/upgrade.ts +73 -0
- package/src/cli/cmd/web.ts +81 -0
- package/src/cli/error.ts +57 -0
- package/src/cli/logo.ts +6 -0
- package/src/cli/network.ts +60 -0
- package/src/cli/ui.ts +113 -0
- package/src/cli/upgrade.ts +25 -0
- package/src/command/index.ts +150 -0
- package/src/command/template/initialize.txt +10 -0
- package/src/command/template/review.txt +99 -0
- package/src/config/config.ts +1477 -0
- package/src/config/markdown.ts +98 -0
- package/src/env/index.ts +28 -0
- package/src/file/ignore.ts +83 -0
- package/src/file/index.ts +583 -0
- package/src/file/ripgrep.ts +375 -0
- package/src/file/time.ts +69 -0
- package/src/file/watcher.ts +127 -0
- package/src/flag/flag.ts +97 -0
- package/src/format/formatter.ts +366 -0
- package/src/format/index.ts +137 -0
- package/src/global/index.ts +55 -0
- package/src/id/id.ts +83 -0
- package/src/ide/index.ts +76 -0
- package/src/index.ts +159 -0
- package/src/installation/index.ts +246 -0
- package/src/lsp/client.ts +252 -0
- package/src/lsp/index.ts +485 -0
- package/src/lsp/language.ts +119 -0
- package/src/lsp/server.ts +2046 -0
- package/src/mcp/auth.ts +132 -0
- package/src/mcp/index.ts +934 -0
- package/src/mcp/oauth-callback.ts +200 -0
- package/src/mcp/oauth-provider.ts +154 -0
- package/src/patch/index.ts +680 -0
- package/src/permission/arity.ts +163 -0
- package/src/permission/index.ts +210 -0
- package/src/permission/next.ts +280 -0
- package/src/plugin/codex.ts +624 -0
- package/src/plugin/copilot.ts +327 -0
- package/src/plugin/index.ts +138 -0
- package/src/project/bootstrap.ts +35 -0
- package/src/project/instance.ts +114 -0
- package/src/project/project.ts +371 -0
- package/src/project/state.ts +70 -0
- package/src/project/vcs.ts +76 -0
- package/src/provider/auth.ts +147 -0
- package/src/provider/models.ts +133 -0
- package/src/provider/provider.ts +1262 -0
- package/src/provider/sdk/copilot/README.md +5 -0
- package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +164 -0
- package/src/provider/sdk/copilot/chat/get-response-metadata.ts +15 -0
- package/src/provider/sdk/copilot/chat/map-openai-compatible-finish-reason.ts +17 -0
- package/src/provider/sdk/copilot/chat/openai-compatible-api-types.ts +64 -0
- package/src/provider/sdk/copilot/chat/openai-compatible-chat-language-model.ts +780 -0
- package/src/provider/sdk/copilot/chat/openai-compatible-chat-options.ts +28 -0
- package/src/provider/sdk/copilot/chat/openai-compatible-metadata-extractor.ts +44 -0
- package/src/provider/sdk/copilot/chat/openai-compatible-prepare-tools.ts +87 -0
- package/src/provider/sdk/copilot/copilot-provider.ts +100 -0
- package/src/provider/sdk/copilot/index.ts +2 -0
- package/src/provider/sdk/copilot/openai-compatible-error.ts +27 -0
- package/src/provider/sdk/copilot/responses/convert-to-openai-responses-input.ts +303 -0
- package/src/provider/sdk/copilot/responses/map-openai-responses-finish-reason.ts +22 -0
- package/src/provider/sdk/copilot/responses/openai-config.ts +18 -0
- package/src/provider/sdk/copilot/responses/openai-error.ts +22 -0
- package/src/provider/sdk/copilot/responses/openai-responses-api-types.ts +207 -0
- package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +1732 -0
- package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +177 -0
- package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +1 -0
- package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +88 -0
- package/src/provider/sdk/copilot/responses/tool/file-search.ts +128 -0
- package/src/provider/sdk/copilot/responses/tool/image-generation.ts +115 -0
- package/src/provider/sdk/copilot/responses/tool/local-shell.ts +65 -0
- package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +104 -0
- package/src/provider/sdk/copilot/responses/tool/web-search.ts +103 -0
- package/src/provider/transform.ts +828 -0
- package/src/pty/index.ts +250 -0
- package/src/question/index.ts +171 -0
- package/src/scheduler/index.ts +61 -0
- package/src/server/error.ts +36 -0
- package/src/server/event.ts +7 -0
- package/src/server/mdns.ts +60 -0
- package/src/server/routes/config.ts +92 -0
- package/src/server/routes/experimental.ts +208 -0
- package/src/server/routes/file.ts +197 -0
- package/src/server/routes/global.ts +183 -0
- package/src/server/routes/mcp.ts +225 -0
- package/src/server/routes/permission.ts +68 -0
- package/src/server/routes/project.ts +82 -0
- package/src/server/routes/provider.ts +165 -0
- package/src/server/routes/pty.ts +169 -0
- package/src/server/routes/question.ts +98 -0
- package/src/server/routes/session.ts +939 -0
- package/src/server/routes/tui.ts +379 -0
- package/src/server/server.ts +613 -0
- package/src/session/compaction.ts +226 -0
- package/src/session/index.ts +524 -0
- package/src/session/instruction.ts +197 -0
- package/src/session/llm.ts +289 -0
- package/src/session/message-v2.ts +802 -0
- package/src/session/message.ts +189 -0
- package/src/session/processor.ts +407 -0
- package/src/session/prompt/agent.txt +43 -0
- package/src/session/prompt/anthropic-20250930.txt +166 -0
- package/src/session/prompt/anthropic.txt +105 -0
- package/src/session/prompt/beast.txt +147 -0
- package/src/session/prompt/build-switch.txt +5 -0
- package/src/session/prompt/codex_header.txt +79 -0
- package/src/session/prompt/copilot-gpt-5.txt +143 -0
- package/src/session/prompt/gemini.txt +155 -0
- package/src/session/prompt/max-steps.txt +16 -0
- package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
- package/src/session/prompt/plan.txt +26 -0
- package/src/session/prompt/qwen.txt +109 -0
- package/src/session/prompt/research.txt +81 -0
- package/src/session/prompt/trinity.txt +97 -0
- package/src/session/prompt.ts +1952 -0
- package/src/session/retry.ts +97 -0
- package/src/session/revert.ts +121 -0
- package/src/session/status.ts +76 -0
- package/src/session/summary.ts +217 -0
- package/src/session/system.ts +54 -0
- package/src/session/todo.ts +37 -0
- package/src/share/share-next.ts +200 -0
- package/src/share/share.ts +92 -0
- package/src/shell/shell.ts +67 -0
- package/src/skill/discovery.ts +97 -0
- package/src/skill/index.ts +1 -0
- package/src/skill/skill.ts +188 -0
- package/src/snapshot/index.ts +255 -0
- package/src/storage/storage.ts +227 -0
- package/src/tool/agent-enter.txt +1 -0
- package/src/tool/agent-exit.txt +1 -0
- package/src/tool/agent.ts +237 -0
- package/src/tool/apply_patch.ts +281 -0
- package/src/tool/apply_patch.txt +33 -0
- package/src/tool/bash.ts +269 -0
- package/src/tool/bash.txt +115 -0
- package/src/tool/batch.ts +175 -0
- package/src/tool/batch.txt +24 -0
- package/src/tool/chat-enter.txt +15 -0
- package/src/tool/chat-exit.txt +7 -0
- package/src/tool/chat.ts +217 -0
- package/src/tool/codesearch.ts +132 -0
- package/src/tool/codesearch.txt +12 -0
- package/src/tool/edit.ts +655 -0
- package/src/tool/edit.txt +10 -0
- package/src/tool/external-directory.ts +32 -0
- package/src/tool/glob.ts +78 -0
- package/src/tool/glob.txt +6 -0
- package/src/tool/grep.ts +147 -0
- package/src/tool/grep.txt +8 -0
- package/src/tool/invalid.ts +17 -0
- package/src/tool/ls.ts +121 -0
- package/src/tool/ls.txt +1 -0
- package/src/tool/lsp.ts +96 -0
- package/src/tool/lsp.txt +19 -0
- package/src/tool/multiedit.ts +46 -0
- package/src/tool/multiedit.txt +41 -0
- package/src/tool/plan-enter.txt +14 -0
- package/src/tool/plan-exit.txt +13 -0
- package/src/tool/plan.ts +130 -0
- package/src/tool/question.ts +33 -0
- package/src/tool/question.txt +10 -0
- package/src/tool/read.ts +211 -0
- package/src/tool/read.txt +12 -0
- package/src/tool/registry.ts +167 -0
- package/src/tool/research-enter.txt +1 -0
- package/src/tool/research-exit.txt +1 -0
- package/src/tool/research.ts +134 -0
- package/src/tool/skill.ts +123 -0
- package/src/tool/task.ts +165 -0
- package/src/tool/task.txt +60 -0
- package/src/tool/todo.ts +53 -0
- package/src/tool/todoread.txt +14 -0
- package/src/tool/todowrite.txt +167 -0
- package/src/tool/tool.ts +89 -0
- package/src/tool/truncation.ts +106 -0
- package/src/tool/webfetch.ts +186 -0
- package/src/tool/webfetch.txt +13 -0
- package/src/tool/websearch.ts +150 -0
- package/src/tool/websearch.txt +14 -0
- package/src/tool/write.ts +85 -0
- package/src/tool/write.txt +8 -0
- package/src/util/abort.ts +35 -0
- package/src/util/archive.ts +16 -0
- package/src/util/color.ts +19 -0
- package/src/util/context.ts +25 -0
- package/src/util/defer.ts +12 -0
- package/src/util/eventloop.ts +20 -0
- package/src/util/filesystem.ts +93 -0
- package/src/util/fn.ts +11 -0
- package/src/util/format.ts +20 -0
- package/src/util/iife.ts +3 -0
- package/src/util/keybind.ts +103 -0
- package/src/util/lazy.ts +18 -0
- package/src/util/locale.ts +81 -0
- package/src/util/lock.ts +98 -0
- package/src/util/log.ts +180 -0
- package/src/util/proxied.ts +3 -0
- package/src/util/queue.ts +32 -0
- package/src/util/rpc.ts +66 -0
- package/src/util/scrap.ts +10 -0
- package/src/util/signal.ts +12 -0
- package/src/util/timeout.ts +14 -0
- package/src/util/token.ts +7 -0
- package/src/util/wildcard.ts +56 -0
- package/src/worktree/index.ts +574 -0
- package/sst-env.d.ts +9 -0
- package/test/acp/agent-interface.test.ts +51 -0
- package/test/acp/event-subscription.test.ts +436 -0
- package/test/agent/agent.test.ts +675 -0
- package/test/bun.test.ts +53 -0
- package/test/cli/github-action.test.ts +161 -0
- package/test/cli/github-remote.test.ts +80 -0
- package/test/cli/import.test.ts +38 -0
- package/test/cli/tui/transcript.test.ts +322 -0
- package/test/config/agent-color.test.ts +71 -0
- package/test/config/config.test.ts +1802 -0
- package/test/config/fixtures/empty-frontmatter.md +4 -0
- package/test/config/fixtures/frontmatter.md +28 -0
- package/test/config/fixtures/markdown-header.md +11 -0
- package/test/config/fixtures/no-frontmatter.md +1 -0
- package/test/config/fixtures/weird-model-id.md +13 -0
- package/test/config/markdown.test.ts +228 -0
- package/test/file/ignore.test.ts +10 -0
- package/test/file/path-traversal.test.ts +198 -0
- package/test/file/ripgrep.test.ts +39 -0
- package/test/fixture/fixture.ts +45 -0
- package/test/fixture/lsp/fake-lsp-server.js +77 -0
- package/test/ide/ide.test.ts +82 -0
- package/test/keybind.test.ts +421 -0
- package/test/lsp/client.test.ts +95 -0
- package/test/mcp/headers.test.ts +153 -0
- package/test/mcp/oauth-browser.test.ts +249 -0
- package/test/memory/abort-leak.test.ts +136 -0
- package/test/patch/patch.test.ts +348 -0
- package/test/permission/arity.test.ts +33 -0
- package/test/permission/next.test.ts +690 -0
- package/test/permission-task.test.ts +319 -0
- package/test/plugin/auth-override.test.ts +44 -0
- package/test/plugin/codex.test.ts +123 -0
- package/test/preload.ts +63 -0
- package/test/project/project.test.ts +120 -0
- package/test/provider/amazon-bedrock.test.ts +445 -0
- package/test/provider/copilot/convert-to-copilot-messages.test.ts +523 -0
- package/test/provider/copilot/copilot-chat-model.test.ts +592 -0
- package/test/provider/gitlab-duo.test.ts +262 -0
- package/test/provider/provider.test.ts +2129 -0
- package/test/provider/transform.test.ts +2022 -0
- package/test/question/question.test.ts +300 -0
- package/test/scheduler.test.ts +73 -0
- package/test/server/session-list.test.ts +39 -0
- package/test/server/session-select.test.ts +78 -0
- package/test/session/compaction.test.ts +293 -0
- package/test/session/instruction.test.ts +170 -0
- package/test/session/llm.test.ts +691 -0
- package/test/session/message-v2.test.ts +786 -0
- package/test/session/prompt-missing-file.test.ts +53 -0
- package/test/session/prompt-special-chars.test.ts +56 -0
- package/test/session/prompt-variant.test.ts +60 -0
- package/test/session/retry.test.ts +179 -0
- package/test/session/revert-compact.test.ts +285 -0
- package/test/session/session.test.ts +71 -0
- package/test/skill/discovery.test.ts +60 -0
- package/test/skill/skill.test.ts +388 -0
- package/test/snapshot/snapshot.test.ts +1040 -0
- package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
- package/test/tool/apply_patch.test.ts +559 -0
- package/test/tool/bash.test.ts +399 -0
- package/test/tool/external-directory.test.ts +127 -0
- package/test/tool/fixtures/large-image.png +0 -0
- package/test/tool/fixtures/models-api.json +38413 -0
- package/test/tool/grep.test.ts +110 -0
- package/test/tool/question.test.ts +107 -0
- package/test/tool/read.test.ts +358 -0
- package/test/tool/registry.test.ts +122 -0
- package/test/tool/skill.test.ts +112 -0
- package/test/tool/truncation.test.ts +159 -0
- package/test/util/filesystem.test.ts +39 -0
- package/test/util/format.test.ts +59 -0
- package/test/util/iife.test.ts +36 -0
- package/test/util/lazy.test.ts +50 -0
- package/test/util/lock.test.ts +72 -0
- package/test/util/timeout.test.ts +21 -0
- package/test/util/wildcard.test.ts +75 -0
- package/tsconfig.json +16 -0
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
import path from "path"
|
|
2
|
+
import { describe, expect, test } from "bun:test"
|
|
3
|
+
import { Instance } from "../../src/project/instance"
|
|
4
|
+
import { Session } from "../../src/session"
|
|
5
|
+
import { SessionPrompt } from "../../src/session/prompt"
|
|
6
|
+
import { tmpdir } from "../fixture/fixture"
|
|
7
|
+
|
|
8
|
+
describe("session.prompt missing file", () => {
|
|
9
|
+
test("does not fail the prompt when a file part is missing", async () => {
|
|
10
|
+
await using tmp = await tmpdir({
|
|
11
|
+
git: true,
|
|
12
|
+
config: {
|
|
13
|
+
agent: {
|
|
14
|
+
build: {
|
|
15
|
+
model: "openai/gpt-5.2",
|
|
16
|
+
},
|
|
17
|
+
},
|
|
18
|
+
},
|
|
19
|
+
})
|
|
20
|
+
|
|
21
|
+
await Instance.provide({
|
|
22
|
+
directory: tmp.path,
|
|
23
|
+
fn: async () => {
|
|
24
|
+
const session = await Session.create({})
|
|
25
|
+
|
|
26
|
+
const missing = path.join(tmp.path, "does-not-exist.ts")
|
|
27
|
+
const msg = await SessionPrompt.prompt({
|
|
28
|
+
sessionID: session.id,
|
|
29
|
+
agent: "build",
|
|
30
|
+
noReply: true,
|
|
31
|
+
parts: [
|
|
32
|
+
{ type: "text", text: "please review @does-not-exist.ts" },
|
|
33
|
+
{
|
|
34
|
+
type: "file",
|
|
35
|
+
mime: "text/plain",
|
|
36
|
+
url: `file://${missing}`,
|
|
37
|
+
filename: "does-not-exist.ts",
|
|
38
|
+
},
|
|
39
|
+
],
|
|
40
|
+
})
|
|
41
|
+
|
|
42
|
+
if (msg.info.role !== "user") throw new Error("expected user message")
|
|
43
|
+
|
|
44
|
+
const hasFailure = msg.parts.some(
|
|
45
|
+
(part) => part.type === "text" && part.synthetic && part.text.includes("Read tool failed to read"),
|
|
46
|
+
)
|
|
47
|
+
expect(hasFailure).toBe(true)
|
|
48
|
+
|
|
49
|
+
await Session.remove(session.id)
|
|
50
|
+
},
|
|
51
|
+
})
|
|
52
|
+
})
|
|
53
|
+
})
|
|
@@ -0,0 +1,56 @@
|
|
|
1
|
+
import path from "path"
|
|
2
|
+
import { describe, expect, test } from "bun:test"
|
|
3
|
+
import { fileURLToPath } from "url"
|
|
4
|
+
import { Instance } from "../../src/project/instance"
|
|
5
|
+
import { Log } from "../../src/util/log"
|
|
6
|
+
import { Session } from "../../src/session"
|
|
7
|
+
import { SessionPrompt } from "../../src/session/prompt"
|
|
8
|
+
import { MessageV2 } from "../../src/session/message-v2"
|
|
9
|
+
import { tmpdir } from "../fixture/fixture"
|
|
10
|
+
|
|
11
|
+
Log.init({ print: false })
|
|
12
|
+
|
|
13
|
+
describe("session.prompt special characters", () => {
|
|
14
|
+
test("handles filenames with # character", async () => {
|
|
15
|
+
await using tmp = await tmpdir({
|
|
16
|
+
git: true,
|
|
17
|
+
init: async (dir) => {
|
|
18
|
+
await Bun.write(path.join(dir, "file#name.txt"), "special content\n")
|
|
19
|
+
},
|
|
20
|
+
})
|
|
21
|
+
|
|
22
|
+
await Instance.provide({
|
|
23
|
+
directory: tmp.path,
|
|
24
|
+
fn: async () => {
|
|
25
|
+
const session = await Session.create({})
|
|
26
|
+
const template = "Read @file#name.txt"
|
|
27
|
+
const parts = await SessionPrompt.resolvePromptParts(template)
|
|
28
|
+
const fileParts = parts.filter((part) => part.type === "file")
|
|
29
|
+
|
|
30
|
+
expect(fileParts.length).toBe(1)
|
|
31
|
+
expect(fileParts[0].filename).toBe("file#name.txt")
|
|
32
|
+
|
|
33
|
+
// Verify the URL is properly encoded (# should be %23)
|
|
34
|
+
expect(fileParts[0].url).toContain("%23")
|
|
35
|
+
|
|
36
|
+
// Verify the URL can be correctly converted back to a file path
|
|
37
|
+
const decodedPath = fileURLToPath(fileParts[0].url)
|
|
38
|
+
expect(decodedPath).toBe(path.join(tmp.path, "file#name.txt"))
|
|
39
|
+
|
|
40
|
+
const message = await SessionPrompt.prompt({
|
|
41
|
+
sessionID: session.id,
|
|
42
|
+
parts,
|
|
43
|
+
noReply: true,
|
|
44
|
+
})
|
|
45
|
+
const stored = await MessageV2.get({ sessionID: session.id, messageID: message.info.id })
|
|
46
|
+
|
|
47
|
+
// Verify the file content was read correctly
|
|
48
|
+
const textParts = stored.parts.filter((part) => part.type === "text")
|
|
49
|
+
const hasContent = textParts.some((part) => part.text.includes("special content"))
|
|
50
|
+
expect(hasContent).toBe(true)
|
|
51
|
+
|
|
52
|
+
await Session.remove(session.id)
|
|
53
|
+
},
|
|
54
|
+
})
|
|
55
|
+
})
|
|
56
|
+
})
|
|
@@ -0,0 +1,60 @@
|
|
|
1
|
+
import { describe, expect, test } from "bun:test"
|
|
2
|
+
import { Instance } from "../../src/project/instance"
|
|
3
|
+
import { Session } from "../../src/session"
|
|
4
|
+
import { SessionPrompt } from "../../src/session/prompt"
|
|
5
|
+
import { tmpdir } from "../fixture/fixture"
|
|
6
|
+
|
|
7
|
+
describe("session.prompt agent variant", () => {
|
|
8
|
+
test("applies agent variant only when using agent model", async () => {
|
|
9
|
+
await using tmp = await tmpdir({
|
|
10
|
+
git: true,
|
|
11
|
+
config: {
|
|
12
|
+
agent: {
|
|
13
|
+
build: {
|
|
14
|
+
model: "openai/gpt-5.2",
|
|
15
|
+
variant: "xhigh",
|
|
16
|
+
},
|
|
17
|
+
},
|
|
18
|
+
},
|
|
19
|
+
})
|
|
20
|
+
|
|
21
|
+
await Instance.provide({
|
|
22
|
+
directory: tmp.path,
|
|
23
|
+
fn: async () => {
|
|
24
|
+
const session = await Session.create({})
|
|
25
|
+
|
|
26
|
+
const other = await SessionPrompt.prompt({
|
|
27
|
+
sessionID: session.id,
|
|
28
|
+
agent: "build",
|
|
29
|
+
model: { providerID: "opencode", modelID: "kimi-k2.5-free" },
|
|
30
|
+
noReply: true,
|
|
31
|
+
parts: [{ type: "text", text: "hello" }],
|
|
32
|
+
})
|
|
33
|
+
if (other.info.role !== "user") throw new Error("expected user message")
|
|
34
|
+
expect(other.info.variant).toBeUndefined()
|
|
35
|
+
|
|
36
|
+
const match = await SessionPrompt.prompt({
|
|
37
|
+
sessionID: session.id,
|
|
38
|
+
agent: "build",
|
|
39
|
+
noReply: true,
|
|
40
|
+
parts: [{ type: "text", text: "hello again" }],
|
|
41
|
+
})
|
|
42
|
+
if (match.info.role !== "user") throw new Error("expected user message")
|
|
43
|
+
expect(match.info.model).toEqual({ providerID: "openai", modelID: "gpt-5.2" })
|
|
44
|
+
expect(match.info.variant).toBe("xhigh")
|
|
45
|
+
|
|
46
|
+
const override = await SessionPrompt.prompt({
|
|
47
|
+
sessionID: session.id,
|
|
48
|
+
agent: "build",
|
|
49
|
+
noReply: true,
|
|
50
|
+
variant: "high",
|
|
51
|
+
parts: [{ type: "text", text: "hello third" }],
|
|
52
|
+
})
|
|
53
|
+
if (override.info.role !== "user") throw new Error("expected user message")
|
|
54
|
+
expect(override.info.variant).toBe("high")
|
|
55
|
+
|
|
56
|
+
await Session.remove(session.id)
|
|
57
|
+
},
|
|
58
|
+
})
|
|
59
|
+
})
|
|
60
|
+
})
|
|
@@ -0,0 +1,179 @@
|
|
|
1
|
+
import { describe, expect, test } from "bun:test"
|
|
2
|
+
import type { NamedError } from "@opencode-ai/util/error"
|
|
3
|
+
import { APICallError } from "ai"
|
|
4
|
+
import { SessionRetry } from "../../src/session/retry"
|
|
5
|
+
import { MessageV2 } from "../../src/session/message-v2"
|
|
6
|
+
|
|
7
|
+
function apiError(headers?: Record<string, string>): MessageV2.APIError {
|
|
8
|
+
return new MessageV2.APIError({
|
|
9
|
+
message: "boom",
|
|
10
|
+
isRetryable: true,
|
|
11
|
+
responseHeaders: headers,
|
|
12
|
+
}).toObject() as MessageV2.APIError
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
function wrap(message: unknown): ReturnType<NamedError["toObject"]> {
|
|
16
|
+
return { data: { message } } as ReturnType<NamedError["toObject"]>
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
describe("session.retry.delay", () => {
|
|
20
|
+
test("caps delay at 30 seconds when headers missing", () => {
|
|
21
|
+
const error = apiError()
|
|
22
|
+
const delays = Array.from({ length: 10 }, (_, index) => SessionRetry.delay(index + 1, error))
|
|
23
|
+
expect(delays).toStrictEqual([2000, 4000, 8000, 16000, 30000, 30000, 30000, 30000, 30000, 30000])
|
|
24
|
+
})
|
|
25
|
+
|
|
26
|
+
test("prefers retry-after-ms when shorter than exponential", () => {
|
|
27
|
+
const error = apiError({ "retry-after-ms": "1500" })
|
|
28
|
+
expect(SessionRetry.delay(4, error)).toBe(1500)
|
|
29
|
+
})
|
|
30
|
+
|
|
31
|
+
test("uses retry-after seconds when reasonable", () => {
|
|
32
|
+
const error = apiError({ "retry-after": "30" })
|
|
33
|
+
expect(SessionRetry.delay(3, error)).toBe(30000)
|
|
34
|
+
})
|
|
35
|
+
|
|
36
|
+
test("accepts http-date retry-after values", () => {
|
|
37
|
+
const date = new Date(Date.now() + 20000).toUTCString()
|
|
38
|
+
const error = apiError({ "retry-after": date })
|
|
39
|
+
const d = SessionRetry.delay(1, error)
|
|
40
|
+
expect(d).toBeGreaterThanOrEqual(19000)
|
|
41
|
+
expect(d).toBeLessThanOrEqual(20000)
|
|
42
|
+
})
|
|
43
|
+
|
|
44
|
+
test("ignores invalid retry hints", () => {
|
|
45
|
+
const error = apiError({ "retry-after": "not-a-number" })
|
|
46
|
+
expect(SessionRetry.delay(1, error)).toBe(2000)
|
|
47
|
+
})
|
|
48
|
+
|
|
49
|
+
test("ignores malformed date retry hints", () => {
|
|
50
|
+
const error = apiError({ "retry-after": "Invalid Date String" })
|
|
51
|
+
expect(SessionRetry.delay(1, error)).toBe(2000)
|
|
52
|
+
})
|
|
53
|
+
|
|
54
|
+
test("ignores past date retry hints", () => {
|
|
55
|
+
const pastDate = new Date(Date.now() - 5000).toUTCString()
|
|
56
|
+
const error = apiError({ "retry-after": pastDate })
|
|
57
|
+
expect(SessionRetry.delay(1, error)).toBe(2000)
|
|
58
|
+
})
|
|
59
|
+
|
|
60
|
+
test("uses retry-after values even when exceeding 10 minutes with headers", () => {
|
|
61
|
+
const error = apiError({ "retry-after": "50" })
|
|
62
|
+
expect(SessionRetry.delay(1, error)).toBe(50000)
|
|
63
|
+
|
|
64
|
+
const longError = apiError({ "retry-after-ms": "700000" })
|
|
65
|
+
expect(SessionRetry.delay(1, longError)).toBe(700000)
|
|
66
|
+
})
|
|
67
|
+
|
|
68
|
+
test("sleep caps delay to max 32-bit signed integer to avoid TimeoutOverflowWarning", async () => {
|
|
69
|
+
const controller = new AbortController()
|
|
70
|
+
|
|
71
|
+
const warnings: string[] = []
|
|
72
|
+
const originalWarn = process.emitWarning
|
|
73
|
+
process.emitWarning = (warning: string | Error) => {
|
|
74
|
+
warnings.push(typeof warning === "string" ? warning : warning.message)
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
const promise = SessionRetry.sleep(2_560_914_000, controller.signal)
|
|
78
|
+
controller.abort()
|
|
79
|
+
|
|
80
|
+
try {
|
|
81
|
+
await promise
|
|
82
|
+
} catch {}
|
|
83
|
+
|
|
84
|
+
process.emitWarning = originalWarn
|
|
85
|
+
expect(warnings.some((w) => w.includes("TimeoutOverflowWarning"))).toBe(false)
|
|
86
|
+
})
|
|
87
|
+
})
|
|
88
|
+
|
|
89
|
+
describe("session.retry.retryable", () => {
|
|
90
|
+
test("maps too_many_requests json messages", () => {
|
|
91
|
+
const error = wrap(JSON.stringify({ type: "error", error: { type: "too_many_requests" } }))
|
|
92
|
+
expect(SessionRetry.retryable(error)).toBe("Too Many Requests")
|
|
93
|
+
})
|
|
94
|
+
|
|
95
|
+
test("maps overloaded provider codes", () => {
|
|
96
|
+
const error = wrap(JSON.stringify({ code: "resource_exhausted" }))
|
|
97
|
+
expect(SessionRetry.retryable(error)).toBe("Provider is overloaded")
|
|
98
|
+
})
|
|
99
|
+
|
|
100
|
+
test("handles json messages without code", () => {
|
|
101
|
+
const error = wrap(JSON.stringify({ error: { message: "no_kv_space" } }))
|
|
102
|
+
expect(SessionRetry.retryable(error)).toBe(`{"error":{"message":"no_kv_space"}}`)
|
|
103
|
+
})
|
|
104
|
+
|
|
105
|
+
test("does not throw on numeric error codes", () => {
|
|
106
|
+
const error = wrap(JSON.stringify({ type: "error", error: { code: 123 } }))
|
|
107
|
+
const result = SessionRetry.retryable(error)
|
|
108
|
+
expect(result).toBeUndefined()
|
|
109
|
+
})
|
|
110
|
+
|
|
111
|
+
test("returns undefined for non-json message", () => {
|
|
112
|
+
const error = wrap("not-json")
|
|
113
|
+
expect(SessionRetry.retryable(error)).toBeUndefined()
|
|
114
|
+
})
|
|
115
|
+
})
|
|
116
|
+
|
|
117
|
+
describe("session.message-v2.fromError", () => {
|
|
118
|
+
test.concurrent(
|
|
119
|
+
"converts ECONNRESET socket errors to retryable APIError",
|
|
120
|
+
async () => {
|
|
121
|
+
using server = Bun.serve({
|
|
122
|
+
port: 0,
|
|
123
|
+
idleTimeout: 8,
|
|
124
|
+
async fetch(req) {
|
|
125
|
+
return new Response(
|
|
126
|
+
new ReadableStream({
|
|
127
|
+
async pull(controller) {
|
|
128
|
+
controller.enqueue("Hello,")
|
|
129
|
+
await Bun.sleep(10000)
|
|
130
|
+
controller.enqueue(" World!")
|
|
131
|
+
controller.close()
|
|
132
|
+
},
|
|
133
|
+
}),
|
|
134
|
+
{ headers: { "Content-Type": "text/plain" } },
|
|
135
|
+
)
|
|
136
|
+
},
|
|
137
|
+
})
|
|
138
|
+
|
|
139
|
+
const error = await fetch(new URL("/", server.url.origin))
|
|
140
|
+
.then((res) => res.text())
|
|
141
|
+
.catch((e) => e)
|
|
142
|
+
|
|
143
|
+
const result = MessageV2.fromError(error, { providerID: "test" })
|
|
144
|
+
|
|
145
|
+
expect(MessageV2.APIError.isInstance(result)).toBe(true)
|
|
146
|
+
expect((result as MessageV2.APIError).data.isRetryable).toBe(true)
|
|
147
|
+
expect((result as MessageV2.APIError).data.message).toBe("Connection reset by server")
|
|
148
|
+
expect((result as MessageV2.APIError).data.metadata?.code).toBe("ECONNRESET")
|
|
149
|
+
expect((result as MessageV2.APIError).data.metadata?.message).toInclude("socket connection")
|
|
150
|
+
},
|
|
151
|
+
15_000,
|
|
152
|
+
)
|
|
153
|
+
|
|
154
|
+
test("ECONNRESET socket error is retryable", () => {
|
|
155
|
+
const error = new MessageV2.APIError({
|
|
156
|
+
message: "Connection reset by server",
|
|
157
|
+
isRetryable: true,
|
|
158
|
+
metadata: { code: "ECONNRESET", message: "The socket connection was closed unexpectedly" },
|
|
159
|
+
}).toObject() as MessageV2.APIError
|
|
160
|
+
|
|
161
|
+
const retryable = SessionRetry.retryable(error)
|
|
162
|
+
expect(retryable).toBeDefined()
|
|
163
|
+
expect(retryable).toBe("Connection reset by server")
|
|
164
|
+
})
|
|
165
|
+
|
|
166
|
+
test("marks OpenAI 404 status codes as retryable", () => {
|
|
167
|
+
const error = new APICallError({
|
|
168
|
+
message: "boom",
|
|
169
|
+
url: "https://api.openai.com/v1/chat/completions",
|
|
170
|
+
requestBodyValues: {},
|
|
171
|
+
statusCode: 404,
|
|
172
|
+
responseHeaders: { "content-type": "application/json" },
|
|
173
|
+
responseBody: '{"error":"boom"}',
|
|
174
|
+
isRetryable: false,
|
|
175
|
+
})
|
|
176
|
+
const result = MessageV2.fromError(error, { providerID: "openai" }) as MessageV2.APIError
|
|
177
|
+
expect(result.data.isRetryable).toBe(true)
|
|
178
|
+
})
|
|
179
|
+
})
|
|
@@ -0,0 +1,285 @@
|
|
|
1
|
+
import { describe, expect, test, beforeEach, afterEach } from "bun:test"
|
|
2
|
+
import path from "path"
|
|
3
|
+
import { Session } from "../../src/session"
|
|
4
|
+
import { SessionRevert } from "../../src/session/revert"
|
|
5
|
+
import { SessionCompaction } from "../../src/session/compaction"
|
|
6
|
+
import { MessageV2 } from "../../src/session/message-v2"
|
|
7
|
+
import { Log } from "../../src/util/log"
|
|
8
|
+
import { Instance } from "../../src/project/instance"
|
|
9
|
+
import { Identifier } from "../../src/id/id"
|
|
10
|
+
import { tmpdir } from "../fixture/fixture"
|
|
11
|
+
|
|
12
|
+
const projectRoot = path.join(__dirname, "../..")
|
|
13
|
+
Log.init({ print: false })
|
|
14
|
+
|
|
15
|
+
describe("revert + compact workflow", () => {
|
|
16
|
+
test("should properly handle compact command after revert", async () => {
|
|
17
|
+
await using tmp = await tmpdir({ git: true })
|
|
18
|
+
await Instance.provide({
|
|
19
|
+
directory: tmp.path,
|
|
20
|
+
fn: async () => {
|
|
21
|
+
// Create a session
|
|
22
|
+
const session = await Session.create({})
|
|
23
|
+
const sessionID = session.id
|
|
24
|
+
|
|
25
|
+
// Create a user message
|
|
26
|
+
const userMsg1 = await Session.updateMessage({
|
|
27
|
+
id: Identifier.ascending("message"),
|
|
28
|
+
role: "user",
|
|
29
|
+
sessionID,
|
|
30
|
+
agent: "default",
|
|
31
|
+
model: {
|
|
32
|
+
providerID: "openai",
|
|
33
|
+
modelID: "gpt-4",
|
|
34
|
+
},
|
|
35
|
+
time: {
|
|
36
|
+
created: Date.now(),
|
|
37
|
+
},
|
|
38
|
+
})
|
|
39
|
+
|
|
40
|
+
// Add a text part to the user message
|
|
41
|
+
await Session.updatePart({
|
|
42
|
+
id: Identifier.ascending("part"),
|
|
43
|
+
messageID: userMsg1.id,
|
|
44
|
+
sessionID,
|
|
45
|
+
type: "text",
|
|
46
|
+
text: "Hello, please help me",
|
|
47
|
+
})
|
|
48
|
+
|
|
49
|
+
// Create an assistant response message
|
|
50
|
+
const assistantMsg1: MessageV2.Assistant = {
|
|
51
|
+
id: Identifier.ascending("message"),
|
|
52
|
+
role: "assistant",
|
|
53
|
+
sessionID,
|
|
54
|
+
mode: "default",
|
|
55
|
+
agent: "default",
|
|
56
|
+
path: {
|
|
57
|
+
cwd: tmp.path,
|
|
58
|
+
root: tmp.path,
|
|
59
|
+
},
|
|
60
|
+
cost: 0,
|
|
61
|
+
tokens: {
|
|
62
|
+
output: 0,
|
|
63
|
+
input: 0,
|
|
64
|
+
reasoning: 0,
|
|
65
|
+
cache: { read: 0, write: 0 },
|
|
66
|
+
},
|
|
67
|
+
modelID: "gpt-4",
|
|
68
|
+
providerID: "openai",
|
|
69
|
+
parentID: userMsg1.id,
|
|
70
|
+
time: {
|
|
71
|
+
created: Date.now(),
|
|
72
|
+
},
|
|
73
|
+
finish: "end_turn",
|
|
74
|
+
}
|
|
75
|
+
await Session.updateMessage(assistantMsg1)
|
|
76
|
+
|
|
77
|
+
// Add a text part to the assistant message
|
|
78
|
+
await Session.updatePart({
|
|
79
|
+
id: Identifier.ascending("part"),
|
|
80
|
+
messageID: assistantMsg1.id,
|
|
81
|
+
sessionID,
|
|
82
|
+
type: "text",
|
|
83
|
+
text: "Sure, I'll help you!",
|
|
84
|
+
})
|
|
85
|
+
|
|
86
|
+
// Create another user message
|
|
87
|
+
const userMsg2 = await Session.updateMessage({
|
|
88
|
+
id: Identifier.ascending("message"),
|
|
89
|
+
role: "user",
|
|
90
|
+
sessionID,
|
|
91
|
+
agent: "default",
|
|
92
|
+
model: {
|
|
93
|
+
providerID: "openai",
|
|
94
|
+
modelID: "gpt-4",
|
|
95
|
+
},
|
|
96
|
+
time: {
|
|
97
|
+
created: Date.now(),
|
|
98
|
+
},
|
|
99
|
+
})
|
|
100
|
+
|
|
101
|
+
await Session.updatePart({
|
|
102
|
+
id: Identifier.ascending("part"),
|
|
103
|
+
messageID: userMsg2.id,
|
|
104
|
+
sessionID,
|
|
105
|
+
type: "text",
|
|
106
|
+
text: "What's the capital of France?",
|
|
107
|
+
})
|
|
108
|
+
|
|
109
|
+
// Create another assistant response
|
|
110
|
+
const assistantMsg2: MessageV2.Assistant = {
|
|
111
|
+
id: Identifier.ascending("message"),
|
|
112
|
+
role: "assistant",
|
|
113
|
+
sessionID,
|
|
114
|
+
mode: "default",
|
|
115
|
+
agent: "default",
|
|
116
|
+
path: {
|
|
117
|
+
cwd: tmp.path,
|
|
118
|
+
root: tmp.path,
|
|
119
|
+
},
|
|
120
|
+
cost: 0,
|
|
121
|
+
tokens: {
|
|
122
|
+
output: 0,
|
|
123
|
+
input: 0,
|
|
124
|
+
reasoning: 0,
|
|
125
|
+
cache: { read: 0, write: 0 },
|
|
126
|
+
},
|
|
127
|
+
modelID: "gpt-4",
|
|
128
|
+
providerID: "openai",
|
|
129
|
+
parentID: userMsg2.id,
|
|
130
|
+
time: {
|
|
131
|
+
created: Date.now(),
|
|
132
|
+
},
|
|
133
|
+
finish: "end_turn",
|
|
134
|
+
}
|
|
135
|
+
await Session.updateMessage(assistantMsg2)
|
|
136
|
+
|
|
137
|
+
await Session.updatePart({
|
|
138
|
+
id: Identifier.ascending("part"),
|
|
139
|
+
messageID: assistantMsg2.id,
|
|
140
|
+
sessionID,
|
|
141
|
+
type: "text",
|
|
142
|
+
text: "The capital of France is Paris.",
|
|
143
|
+
})
|
|
144
|
+
|
|
145
|
+
// Verify messages before revert
|
|
146
|
+
let messages = await Session.messages({ sessionID })
|
|
147
|
+
expect(messages.length).toBe(4) // 2 user + 2 assistant messages
|
|
148
|
+
const messageIds = messages.map((m) => m.info.id)
|
|
149
|
+
expect(messageIds).toContain(userMsg1.id)
|
|
150
|
+
expect(messageIds).toContain(userMsg2.id)
|
|
151
|
+
expect(messageIds).toContain(assistantMsg1.id)
|
|
152
|
+
expect(messageIds).toContain(assistantMsg2.id)
|
|
153
|
+
|
|
154
|
+
// Revert the last user message (userMsg2)
|
|
155
|
+
await SessionRevert.revert({
|
|
156
|
+
sessionID,
|
|
157
|
+
messageID: userMsg2.id,
|
|
158
|
+
})
|
|
159
|
+
|
|
160
|
+
// Check that revert state is set
|
|
161
|
+
let sessionInfo = await Session.get(sessionID)
|
|
162
|
+
expect(sessionInfo.revert).toBeDefined()
|
|
163
|
+
const revertMessageID = sessionInfo.revert?.messageID
|
|
164
|
+
expect(revertMessageID).toBeDefined()
|
|
165
|
+
|
|
166
|
+
// Messages should still be in the list (not removed yet, just marked for revert)
|
|
167
|
+
messages = await Session.messages({ sessionID })
|
|
168
|
+
expect(messages.length).toBe(4)
|
|
169
|
+
|
|
170
|
+
// Now clean up the revert state (this is what the compact endpoint should do)
|
|
171
|
+
await SessionRevert.cleanup(sessionInfo)
|
|
172
|
+
|
|
173
|
+
// After cleanup, the reverted messages (those after the revert point) should be removed
|
|
174
|
+
messages = await Session.messages({ sessionID })
|
|
175
|
+
const remainingIds = messages.map((m) => m.info.id)
|
|
176
|
+
// The revert point is somewhere in the message chain, so we should have fewer messages
|
|
177
|
+
expect(messages.length).toBeLessThan(4)
|
|
178
|
+
// userMsg2 and assistantMsg2 should be removed (they come after the revert point)
|
|
179
|
+
expect(remainingIds).not.toContain(userMsg2.id)
|
|
180
|
+
expect(remainingIds).not.toContain(assistantMsg2.id)
|
|
181
|
+
|
|
182
|
+
// Revert state should be cleared
|
|
183
|
+
sessionInfo = await Session.get(sessionID)
|
|
184
|
+
expect(sessionInfo.revert).toBeUndefined()
|
|
185
|
+
|
|
186
|
+
// Clean up
|
|
187
|
+
await Session.remove(sessionID)
|
|
188
|
+
},
|
|
189
|
+
})
|
|
190
|
+
})
|
|
191
|
+
|
|
192
|
+
test("should properly clean up revert state before creating compaction message", async () => {
|
|
193
|
+
await using tmp = await tmpdir({ git: true })
|
|
194
|
+
await Instance.provide({
|
|
195
|
+
directory: tmp.path,
|
|
196
|
+
fn: async () => {
|
|
197
|
+
// Create a session
|
|
198
|
+
const session = await Session.create({})
|
|
199
|
+
const sessionID = session.id
|
|
200
|
+
|
|
201
|
+
// Create initial messages
|
|
202
|
+
const userMsg = await Session.updateMessage({
|
|
203
|
+
id: Identifier.ascending("message"),
|
|
204
|
+
role: "user",
|
|
205
|
+
sessionID,
|
|
206
|
+
agent: "default",
|
|
207
|
+
model: {
|
|
208
|
+
providerID: "openai",
|
|
209
|
+
modelID: "gpt-4",
|
|
210
|
+
},
|
|
211
|
+
time: {
|
|
212
|
+
created: Date.now(),
|
|
213
|
+
},
|
|
214
|
+
})
|
|
215
|
+
|
|
216
|
+
await Session.updatePart({
|
|
217
|
+
id: Identifier.ascending("part"),
|
|
218
|
+
messageID: userMsg.id,
|
|
219
|
+
sessionID,
|
|
220
|
+
type: "text",
|
|
221
|
+
text: "Hello",
|
|
222
|
+
})
|
|
223
|
+
|
|
224
|
+
const assistantMsg: MessageV2.Assistant = {
|
|
225
|
+
id: Identifier.ascending("message"),
|
|
226
|
+
role: "assistant",
|
|
227
|
+
sessionID,
|
|
228
|
+
mode: "default",
|
|
229
|
+
agent: "default",
|
|
230
|
+
path: {
|
|
231
|
+
cwd: tmp.path,
|
|
232
|
+
root: tmp.path,
|
|
233
|
+
},
|
|
234
|
+
cost: 0,
|
|
235
|
+
tokens: {
|
|
236
|
+
output: 0,
|
|
237
|
+
input: 0,
|
|
238
|
+
reasoning: 0,
|
|
239
|
+
cache: { read: 0, write: 0 },
|
|
240
|
+
},
|
|
241
|
+
modelID: "gpt-4",
|
|
242
|
+
providerID: "openai",
|
|
243
|
+
parentID: userMsg.id,
|
|
244
|
+
time: {
|
|
245
|
+
created: Date.now(),
|
|
246
|
+
},
|
|
247
|
+
finish: "end_turn",
|
|
248
|
+
}
|
|
249
|
+
await Session.updateMessage(assistantMsg)
|
|
250
|
+
|
|
251
|
+
await Session.updatePart({
|
|
252
|
+
id: Identifier.ascending("part"),
|
|
253
|
+
messageID: assistantMsg.id,
|
|
254
|
+
sessionID,
|
|
255
|
+
type: "text",
|
|
256
|
+
text: "Hi there!",
|
|
257
|
+
})
|
|
258
|
+
|
|
259
|
+
// Revert the user message
|
|
260
|
+
await SessionRevert.revert({
|
|
261
|
+
sessionID,
|
|
262
|
+
messageID: userMsg.id,
|
|
263
|
+
})
|
|
264
|
+
|
|
265
|
+
// Check that revert state is set
|
|
266
|
+
let sessionInfo = await Session.get(sessionID)
|
|
267
|
+
expect(sessionInfo.revert).toBeDefined()
|
|
268
|
+
|
|
269
|
+
// Simulate what the compact endpoint does: cleanup revert before creating compaction
|
|
270
|
+
await SessionRevert.cleanup(sessionInfo)
|
|
271
|
+
|
|
272
|
+
// Verify revert state is cleared
|
|
273
|
+
sessionInfo = await Session.get(sessionID)
|
|
274
|
+
expect(sessionInfo.revert).toBeUndefined()
|
|
275
|
+
|
|
276
|
+
// Verify messages are properly cleaned up
|
|
277
|
+
const messages = await Session.messages({ sessionID })
|
|
278
|
+
expect(messages.length).toBe(0) // All messages should be reverted
|
|
279
|
+
|
|
280
|
+
// Clean up
|
|
281
|
+
await Session.remove(sessionID)
|
|
282
|
+
},
|
|
283
|
+
})
|
|
284
|
+
})
|
|
285
|
+
})
|