easc-cli 1.1.28
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 +108 -0
- package/bunfig.toml +7 -0
- package/package.json +132 -0
- package/parsers-config.ts +253 -0
- package/script/build.ts +172 -0
- package/script/deploy.ts +64 -0
- package/script/postinstall.mjs +125 -0
- package/script/publish-registries.ts +187 -0
- package/script/publish.ts +70 -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 +1285 -0
- package/src/acp/session.ts +105 -0
- package/src/acp/types.ts +22 -0
- package/src/agent/agent.ts +332 -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 +43 -0
- package/src/auth/eliseart.ts +76 -0
- package/src/auth/index.ts +73 -0
- package/src/bun/index.ts +134 -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/account.ts +81 -0
- package/src/cli/cmd/acp.ts +69 -0
- package/src/cli/cmd/agent.ts +257 -0
- package/src/cli/cmd/auth.ts +427 -0
- package/src/cli/cmd/cmd.ts +7 -0
- package/src/cli/cmd/debug/agent.ts +166 -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 +1548 -0
- package/src/cli/cmd/import.ts +98 -0
- package/src/cli/cmd/mcp.ts +827 -0
- package/src/cli/cmd/models.ts +77 -0
- package/src/cli/cmd/pr.ts +112 -0
- package/src/cli/cmd/run.ts +407 -0
- package/src/cli/cmd/serve.ts +20 -0
- package/src/cli/cmd/session.ts +135 -0
- package/src/cli/cmd/stats.ts +402 -0
- package/src/cli/cmd/tui/app.tsx +774 -0
- package/src/cli/cmd/tui/attach.ts +31 -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 +256 -0
- package/src/cli/cmd/tui/component/dialog-session-list.tsx +114 -0
- package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
- package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
- package/src/cli/cmd/tui/component/dialog-status.tsx +164 -0
- package/src/cli/cmd/tui/component/dialog-supabase.tsx +102 -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 +88 -0
- package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +653 -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 +1182 -0
- package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
- package/src/cli/cmd/tui/component/spinner.tsx +16 -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 +14 -0
- package/src/cli/cmd/tui/context/directory.ts +13 -0
- package/src/cli/cmd/tui/context/exit.tsx +23 -0
- package/src/cli/cmd/tui/context/helper.tsx +25 -0
- package/src/cli/cmd/tui/context/keybind.tsx +101 -0
- package/src/cli/cmd/tui/context/kv.tsx +52 -0
- package/src/cli/cmd/tui/context/local.tsx +402 -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 +94 -0
- package/src/cli/cmd/tui/context/sync.tsx +445 -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 +95 -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/dialog-tool.tsx +63 -0
- package/src/cli/cmd/tui/routes/session/footer.tsx +129 -0
- package/src/cli/cmd/tui/routes/session/header.tsx +136 -0
- package/src/cli/cmd/tui/routes/session/index.tsx +2132 -0
- package/src/cli/cmd/tui/routes/session/permission.tsx +495 -0
- package/src/cli/cmd/tui/routes/session/question.tsx +435 -0
- package/src/cli/cmd/tui/routes/session/sidebar.tsx +313 -0
- package/src/cli/cmd/tui/thread.ts +165 -0
- package/src/cli/cmd/tui/ui/dialog-alert.tsx +57 -0
- package/src/cli/cmd/tui/ui/dialog-confirm.tsx +83 -0
- package/src/cli/cmd/tui/ui/dialog-export-options.tsx +204 -0
- package/src/cli/cmd/tui/ui/dialog-help.tsx +38 -0
- package/src/cli/cmd/tui/ui/dialog-prompt.tsx +77 -0
- package/src/cli/cmd/tui/ui/dialog-select.tsx +376 -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 +160 -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/network.ts +53 -0
- package/src/cli/ui.ts +84 -0
- package/src/cli/upgrade.ts +25 -0
- package/src/command/index.ts +131 -0
- package/src/command/template/initialize.txt +10 -0
- package/src/command/template/review.txt +99 -0
- package/src/config/config.ts +1361 -0
- package/src/config/markdown.ts +93 -0
- package/src/env/index.ts +26 -0
- package/src/file/ignore.ts +83 -0
- package/src/file/index.ts +411 -0
- package/src/file/ripgrep.ts +407 -0
- package/src/file/time.ts +64 -0
- package/src/file/watcher.ts +127 -0
- package/src/flag/flag.ts +54 -0
- package/src/format/formatter.ts +342 -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 +162 -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 +135 -0
- package/src/mcp/index.ts +931 -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 +269 -0
- package/src/plugin/codex.ts +493 -0
- package/src/plugin/copilot.ts +269 -0
- package/src/plugin/index.ts +135 -0
- package/src/project/bootstrap.ts +35 -0
- package/src/project/instance.ts +91 -0
- package/src/project/project.ts +339 -0
- package/src/project/state.ts +66 -0
- package/src/project/vcs.ts +76 -0
- package/src/provider/auth.ts +147 -0
- package/src/provider/models-macro.ts +11 -0
- package/src/provider/models.ts +112 -0
- package/src/provider/provider.ts +1391 -0
- package/src/provider/sdk/openai-compatible/src/README.md +5 -0
- package/src/provider/sdk/openai-compatible/src/index.ts +2 -0
- package/src/provider/sdk/openai-compatible/src/openai-compatible-provider.ts +100 -0
- package/src/provider/sdk/openai-compatible/src/responses/convert-to-openai-responses-input.ts +303 -0
- package/src/provider/sdk/openai-compatible/src/responses/map-openai-responses-finish-reason.ts +22 -0
- package/src/provider/sdk/openai-compatible/src/responses/openai-config.ts +18 -0
- package/src/provider/sdk/openai-compatible/src/responses/openai-error.ts +22 -0
- package/src/provider/sdk/openai-compatible/src/responses/openai-responses-api-types.ts +207 -0
- package/src/provider/sdk/openai-compatible/src/responses/openai-responses-language-model.ts +1732 -0
- package/src/provider/sdk/openai-compatible/src/responses/openai-responses-prepare-tools.ts +177 -0
- package/src/provider/sdk/openai-compatible/src/responses/openai-responses-settings.ts +1 -0
- package/src/provider/sdk/openai-compatible/src/responses/tool/code-interpreter.ts +88 -0
- package/src/provider/sdk/openai-compatible/src/responses/tool/file-search.ts +128 -0
- package/src/provider/sdk/openai-compatible/src/responses/tool/image-generation.ts +115 -0
- package/src/provider/sdk/openai-compatible/src/responses/tool/local-shell.ts +65 -0
- package/src/provider/sdk/openai-compatible/src/responses/tool/web-search-preview.ts +104 -0
- package/src/provider/sdk/openai-compatible/src/responses/tool/web-search.ts +103 -0
- package/src/provider/transform.ts +733 -0
- package/src/pty/index.ts +232 -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 +59 -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 +135 -0
- package/src/server/routes/mcp.ts +361 -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 +935 -0
- package/src/server/routes/tui.ts +379 -0
- package/src/server/server.ts +573 -0
- package/src/session/compaction.ts +225 -0
- package/src/session/index.ts +488 -0
- package/src/session/llm.ts +279 -0
- package/src/session/message-v2.ts +702 -0
- package/src/session/message.ts +189 -0
- package/src/session/processor.ts +406 -0
- package/src/session/prompt/anthropic-20250930.txt +166 -0
- package/src/session/prompt/anthropic.txt +105 -0
- package/src/session/prompt/anthropic_spoof.txt +1 -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.ts +1820 -0
- package/src/session/retry.ts +90 -0
- package/src/session/revert.ts +108 -0
- package/src/session/status.ts +76 -0
- package/src/session/summary.ts +150 -0
- package/src/session/system.ts +152 -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/index.ts +1 -0
- package/src/skill/skill.ts +136 -0
- package/src/snapshot/index.ts +236 -0
- package/src/storage/storage.ts +227 -0
- package/src/tool/apply_patch.ts +269 -0
- package/src/tool/apply_patch.txt +33 -0
- package/src/tool/bash.ts +259 -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/codesearch.ts +132 -0
- package/src/tool/codesearch.txt +12 -0
- package/src/tool/edit.ts +645 -0
- package/src/tool/edit.txt +10 -0
- package/src/tool/external-directory.ts +32 -0
- package/src/tool/glob.ts +77 -0
- package/src/tool/glob.txt +6 -0
- package/src/tool/grep.ts +154 -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 +202 -0
- package/src/tool/read.txt +12 -0
- package/src/tool/registry.ts +163 -0
- package/src/tool/skill.ts +75 -0
- package/src/tool/task.ts +188 -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 +88 -0
- package/src/tool/truncation.ts +106 -0
- package/src/tool/webfetch.ts +182 -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 +80 -0
- package/src/tool/write.txt +8 -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/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 +424 -0
- package/sst-env.d.ts +9 -0
- package/test/acp/event-subscription.test.ts +436 -0
- package/test/agent/agent.test.ts +638 -0
- package/test/bun.test.ts +53 -0
- package/test/cli/github-action.test.ts +129 -0
- package/test/cli/github-remote.test.ts +80 -0
- package/test/cli/tui/transcript.test.ts +297 -0
- package/test/config/agent-color.test.ts +66 -0
- package/test/config/config.test.ts +1414 -0
- package/test/config/fixtures/empty-frontmatter.md +4 -0
- package/test/config/fixtures/frontmatter.md +28 -0
- package/test/config/fixtures/no-frontmatter.md +1 -0
- package/test/config/markdown.test.ts +192 -0
- package/test/file/ignore.test.ts +10 -0
- package/test/file/path-traversal.test.ts +198 -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 +261 -0
- package/test/patch/patch.test.ts +348 -0
- package/test/permission/arity.test.ts +33 -0
- package/test/permission/next.test.ts +652 -0
- package/test/permission-task.test.ts +319 -0
- package/test/plugin/codex.test.ts +123 -0
- package/test/preload.ts +65 -0
- package/test/project/project.test.ts +120 -0
- package/test/provider/amazon-bedrock.test.ts +268 -0
- package/test/provider/gitlab-duo.test.ts +286 -0
- package/test/provider/provider.test.ts +2149 -0
- package/test/provider/transform.test.ts +1596 -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/llm.test.ts +90 -0
- package/test/session/message-v2.test.ts +662 -0
- package/test/session/retry.test.ts +131 -0
- package/test/session/revert-compact.test.ts +285 -0
- package/test/session/session.test.ts +71 -0
- package/test/skill/skill.test.ts +185 -0
- package/test/snapshot/snapshot.test.ts +939 -0
- package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
- package/test/tool/apply_patch.test.ts +499 -0
- package/test/tool/bash.test.ts +320 -0
- package/test/tool/external-directory.test.ts +126 -0
- package/test/tool/fixtures/large-image.png +0 -0
- package/test/tool/fixtures/models-api.json +33453 -0
- package/test/tool/grep.test.ts +109 -0
- package/test/tool/question.test.ts +105 -0
- package/test/tool/read.test.ts +332 -0
- package/test/tool/registry.test.ts +76 -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,424 @@
|
|
|
1
|
+
import { $ } from "bun"
|
|
2
|
+
import fs from "fs/promises"
|
|
3
|
+
import path from "path"
|
|
4
|
+
import z from "zod"
|
|
5
|
+
import { NamedError } from "@eliseart.ai/util/error"
|
|
6
|
+
import { Global } from "../global"
|
|
7
|
+
import { Instance } from "../project/instance"
|
|
8
|
+
import { Project } from "../project/project"
|
|
9
|
+
import { fn } from "../util/fn"
|
|
10
|
+
import { Config } from "@/config/config"
|
|
11
|
+
|
|
12
|
+
export namespace Worktree {
|
|
13
|
+
export const Info = z
|
|
14
|
+
.object({
|
|
15
|
+
name: z.string(),
|
|
16
|
+
branch: z.string(),
|
|
17
|
+
directory: z.string(),
|
|
18
|
+
})
|
|
19
|
+
.meta({
|
|
20
|
+
ref: "Worktree",
|
|
21
|
+
})
|
|
22
|
+
|
|
23
|
+
export type Info = z.infer<typeof Info>
|
|
24
|
+
|
|
25
|
+
export const CreateInput = z
|
|
26
|
+
.object({
|
|
27
|
+
name: z.string().optional(),
|
|
28
|
+
startCommand: z.string().optional(),
|
|
29
|
+
})
|
|
30
|
+
.meta({
|
|
31
|
+
ref: "WorktreeCreateInput",
|
|
32
|
+
})
|
|
33
|
+
|
|
34
|
+
export type CreateInput = z.infer<typeof CreateInput>
|
|
35
|
+
|
|
36
|
+
export const RemoveInput = z
|
|
37
|
+
.object({
|
|
38
|
+
directory: z.string(),
|
|
39
|
+
})
|
|
40
|
+
.meta({
|
|
41
|
+
ref: "WorktreeRemoveInput",
|
|
42
|
+
})
|
|
43
|
+
|
|
44
|
+
export type RemoveInput = z.infer<typeof RemoveInput>
|
|
45
|
+
|
|
46
|
+
export const ResetInput = z
|
|
47
|
+
.object({
|
|
48
|
+
directory: z.string(),
|
|
49
|
+
})
|
|
50
|
+
.meta({
|
|
51
|
+
ref: "WorktreeResetInput",
|
|
52
|
+
})
|
|
53
|
+
|
|
54
|
+
export type ResetInput = z.infer<typeof ResetInput>
|
|
55
|
+
|
|
56
|
+
export const NotGitError = NamedError.create(
|
|
57
|
+
"WorktreeNotGitError",
|
|
58
|
+
z.object({
|
|
59
|
+
message: z.string(),
|
|
60
|
+
}),
|
|
61
|
+
)
|
|
62
|
+
|
|
63
|
+
export const NameGenerationFailedError = NamedError.create(
|
|
64
|
+
"WorktreeNameGenerationFailedError",
|
|
65
|
+
z.object({
|
|
66
|
+
message: z.string(),
|
|
67
|
+
}),
|
|
68
|
+
)
|
|
69
|
+
|
|
70
|
+
export const CreateFailedError = NamedError.create(
|
|
71
|
+
"WorktreeCreateFailedError",
|
|
72
|
+
z.object({
|
|
73
|
+
message: z.string(),
|
|
74
|
+
}),
|
|
75
|
+
)
|
|
76
|
+
|
|
77
|
+
export const StartCommandFailedError = NamedError.create(
|
|
78
|
+
"WorktreeStartCommandFailedError",
|
|
79
|
+
z.object({
|
|
80
|
+
message: z.string(),
|
|
81
|
+
}),
|
|
82
|
+
)
|
|
83
|
+
|
|
84
|
+
export const RemoveFailedError = NamedError.create(
|
|
85
|
+
"WorktreeRemoveFailedError",
|
|
86
|
+
z.object({
|
|
87
|
+
message: z.string(),
|
|
88
|
+
}),
|
|
89
|
+
)
|
|
90
|
+
|
|
91
|
+
export const ResetFailedError = NamedError.create(
|
|
92
|
+
"WorktreeResetFailedError",
|
|
93
|
+
z.object({
|
|
94
|
+
message: z.string(),
|
|
95
|
+
}),
|
|
96
|
+
)
|
|
97
|
+
|
|
98
|
+
const ADJECTIVES = [
|
|
99
|
+
"brave",
|
|
100
|
+
"calm",
|
|
101
|
+
"clever",
|
|
102
|
+
"cosmic",
|
|
103
|
+
"crisp",
|
|
104
|
+
"curious",
|
|
105
|
+
"eager",
|
|
106
|
+
"gentle",
|
|
107
|
+
"glowing",
|
|
108
|
+
"happy",
|
|
109
|
+
"hidden",
|
|
110
|
+
"jolly",
|
|
111
|
+
"kind",
|
|
112
|
+
"lucky",
|
|
113
|
+
"mighty",
|
|
114
|
+
"misty",
|
|
115
|
+
"neon",
|
|
116
|
+
"nimble",
|
|
117
|
+
"playful",
|
|
118
|
+
"proud",
|
|
119
|
+
"quick",
|
|
120
|
+
"quiet",
|
|
121
|
+
"shiny",
|
|
122
|
+
"silent",
|
|
123
|
+
"stellar",
|
|
124
|
+
"sunny",
|
|
125
|
+
"swift",
|
|
126
|
+
"tidy",
|
|
127
|
+
"witty",
|
|
128
|
+
] as const
|
|
129
|
+
|
|
130
|
+
const NOUNS = [
|
|
131
|
+
"cabin",
|
|
132
|
+
"cactus",
|
|
133
|
+
"canyon",
|
|
134
|
+
"circuit",
|
|
135
|
+
"comet",
|
|
136
|
+
"eagle",
|
|
137
|
+
"engine",
|
|
138
|
+
"falcon",
|
|
139
|
+
"forest",
|
|
140
|
+
"garden",
|
|
141
|
+
"harbor",
|
|
142
|
+
"island",
|
|
143
|
+
"knight",
|
|
144
|
+
"lagoon",
|
|
145
|
+
"meadow",
|
|
146
|
+
"moon",
|
|
147
|
+
"mountain",
|
|
148
|
+
"nebula",
|
|
149
|
+
"orchid",
|
|
150
|
+
"otter",
|
|
151
|
+
"panda",
|
|
152
|
+
"pixel",
|
|
153
|
+
"planet",
|
|
154
|
+
"river",
|
|
155
|
+
"rocket",
|
|
156
|
+
"sailor",
|
|
157
|
+
"squid",
|
|
158
|
+
"star",
|
|
159
|
+
"tiger",
|
|
160
|
+
"wizard",
|
|
161
|
+
"wolf",
|
|
162
|
+
] as const
|
|
163
|
+
|
|
164
|
+
function pick<const T extends readonly string[]>(list: T) {
|
|
165
|
+
return list[Math.floor(Math.random() * list.length)]
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
function slug(input: string) {
|
|
169
|
+
return input
|
|
170
|
+
.trim()
|
|
171
|
+
.toLowerCase()
|
|
172
|
+
.replace(/[^a-z0-9]+/g, "-")
|
|
173
|
+
.replace(/^-+/, "")
|
|
174
|
+
.replace(/-+$/, "")
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
function randomName() {
|
|
178
|
+
return `${pick(ADJECTIVES)}-${pick(NOUNS)}`
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
async function exists(target: string) {
|
|
182
|
+
return fs
|
|
183
|
+
.stat(target)
|
|
184
|
+
.then(() => true)
|
|
185
|
+
.catch(() => false)
|
|
186
|
+
}
|
|
187
|
+
|
|
188
|
+
function outputText(input: Uint8Array | undefined) {
|
|
189
|
+
if (!input?.length) return ""
|
|
190
|
+
return new TextDecoder().decode(input).trim()
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
function errorText(result: { stdout?: Uint8Array; stderr?: Uint8Array }) {
|
|
194
|
+
return [outputText(result.stderr), outputText(result.stdout)].filter(Boolean).join("\n")
|
|
195
|
+
}
|
|
196
|
+
|
|
197
|
+
async function candidate(root: string, base?: string) {
|
|
198
|
+
for (const attempt of Array.from({ length: 26 }, (_, i) => i)) {
|
|
199
|
+
const name = base ? (attempt === 0 ? base : `${base}-${randomName()}`) : randomName()
|
|
200
|
+
const branch = `opencode/${name}`
|
|
201
|
+
const directory = path.join(root, name)
|
|
202
|
+
|
|
203
|
+
if (await exists(directory)) continue
|
|
204
|
+
|
|
205
|
+
const ref = `refs/heads/${branch}`
|
|
206
|
+
const branchCheck = await $`git show-ref --verify --quiet ${ref}`.quiet().nothrow().cwd(Instance.worktree)
|
|
207
|
+
if (branchCheck.exitCode === 0) continue
|
|
208
|
+
|
|
209
|
+
return Info.parse({ name, branch, directory })
|
|
210
|
+
}
|
|
211
|
+
|
|
212
|
+
throw new NameGenerationFailedError({ message: "Failed to generate a unique worktree name" })
|
|
213
|
+
}
|
|
214
|
+
|
|
215
|
+
async function runStartCommand(directory: string, cmd: string) {
|
|
216
|
+
if (process.platform === "win32") {
|
|
217
|
+
return $`cmd /c ${cmd}`.nothrow().cwd(directory)
|
|
218
|
+
}
|
|
219
|
+
return $`bash -lc ${cmd}`.nothrow().cwd(directory)
|
|
220
|
+
}
|
|
221
|
+
|
|
222
|
+
export const create = fn(CreateInput.optional(), async (input) => {
|
|
223
|
+
if (Instance.project.vcs !== "git") {
|
|
224
|
+
throw new NotGitError({ message: "Worktrees are only supported for git projects" })
|
|
225
|
+
}
|
|
226
|
+
|
|
227
|
+
const root = path.join(Global.Path.data, "worktree", Instance.project.id)
|
|
228
|
+
await fs.mkdir(root, { recursive: true })
|
|
229
|
+
|
|
230
|
+
const base = input?.name ? slug(input.name) : ""
|
|
231
|
+
const info = await candidate(root, base || undefined)
|
|
232
|
+
|
|
233
|
+
const created = await $`git worktree add -b ${info.branch} ${info.directory}`
|
|
234
|
+
.quiet()
|
|
235
|
+
.nothrow()
|
|
236
|
+
.cwd(Instance.worktree)
|
|
237
|
+
if (created.exitCode !== 0) {
|
|
238
|
+
throw new CreateFailedError({ message: errorText(created) || "Failed to create git worktree" })
|
|
239
|
+
}
|
|
240
|
+
|
|
241
|
+
const cmd = input?.startCommand?.trim()
|
|
242
|
+
if (!cmd) return info
|
|
243
|
+
|
|
244
|
+
const ran = await runStartCommand(info.directory, cmd)
|
|
245
|
+
if (ran.exitCode !== 0) {
|
|
246
|
+
throw new StartCommandFailedError({ message: errorText(ran) || "Worktree start command failed" })
|
|
247
|
+
}
|
|
248
|
+
|
|
249
|
+
return info
|
|
250
|
+
})
|
|
251
|
+
|
|
252
|
+
export const remove = fn(RemoveInput, async (input) => {
|
|
253
|
+
if (Instance.project.vcs !== "git") {
|
|
254
|
+
throw new NotGitError({ message: "Worktrees are only supported for git projects" })
|
|
255
|
+
}
|
|
256
|
+
|
|
257
|
+
const directory = path.resolve(input.directory)
|
|
258
|
+
const list = await $`git worktree list --porcelain`.quiet().nothrow().cwd(Instance.worktree)
|
|
259
|
+
if (list.exitCode !== 0) {
|
|
260
|
+
throw new RemoveFailedError({ message: errorText(list) || "Failed to read git worktrees" })
|
|
261
|
+
}
|
|
262
|
+
|
|
263
|
+
const lines = outputText(list.stdout)
|
|
264
|
+
.split("\n")
|
|
265
|
+
.map((line) => line.trim())
|
|
266
|
+
const entries = lines.reduce<{ path?: string; branch?: string }[]>((acc, line) => {
|
|
267
|
+
if (!line) return acc
|
|
268
|
+
if (line.startsWith("worktree ")) {
|
|
269
|
+
acc.push({ path: line.slice("worktree ".length).trim() })
|
|
270
|
+
return acc
|
|
271
|
+
}
|
|
272
|
+
const current = acc[acc.length - 1]
|
|
273
|
+
if (!current) return acc
|
|
274
|
+
if (line.startsWith("branch ")) {
|
|
275
|
+
current.branch = line.slice("branch ".length).trim()
|
|
276
|
+
}
|
|
277
|
+
return acc
|
|
278
|
+
}, [])
|
|
279
|
+
|
|
280
|
+
const entry = entries.find((item) => item.path && path.resolve(item.path) === directory)
|
|
281
|
+
if (!entry?.path) {
|
|
282
|
+
throw new RemoveFailedError({ message: "Worktree not found" })
|
|
283
|
+
}
|
|
284
|
+
|
|
285
|
+
const removed = await $`git worktree remove --force ${entry.path}`.quiet().nothrow().cwd(Instance.worktree)
|
|
286
|
+
if (removed.exitCode !== 0) {
|
|
287
|
+
throw new RemoveFailedError({ message: errorText(removed) || "Failed to remove git worktree" })
|
|
288
|
+
}
|
|
289
|
+
|
|
290
|
+
const branch = entry.branch?.replace(/^refs\/heads\//, "")
|
|
291
|
+
if (branch) {
|
|
292
|
+
const deleted = await $`git branch -D ${branch}`.quiet().nothrow().cwd(Instance.worktree)
|
|
293
|
+
if (deleted.exitCode !== 0) {
|
|
294
|
+
throw new RemoveFailedError({ message: errorText(deleted) || "Failed to delete worktree branch" })
|
|
295
|
+
}
|
|
296
|
+
}
|
|
297
|
+
|
|
298
|
+
return true
|
|
299
|
+
})
|
|
300
|
+
|
|
301
|
+
export const reset = fn(ResetInput, async (input) => {
|
|
302
|
+
if (Instance.project.vcs !== "git") {
|
|
303
|
+
throw new NotGitError({ message: "Worktrees are only supported for git projects" })
|
|
304
|
+
}
|
|
305
|
+
|
|
306
|
+
const directory = path.resolve(input.directory)
|
|
307
|
+
if (directory === path.resolve(Instance.worktree)) {
|
|
308
|
+
throw new ResetFailedError({ message: "Cannot reset the primary workspace" })
|
|
309
|
+
}
|
|
310
|
+
|
|
311
|
+
const list = await $`git worktree list --porcelain`.quiet().nothrow().cwd(Instance.worktree)
|
|
312
|
+
if (list.exitCode !== 0) {
|
|
313
|
+
throw new ResetFailedError({ message: errorText(list) || "Failed to read git worktrees" })
|
|
314
|
+
}
|
|
315
|
+
|
|
316
|
+
const lines = outputText(list.stdout)
|
|
317
|
+
.split("\n")
|
|
318
|
+
.map((line) => line.trim())
|
|
319
|
+
const entries = lines.reduce<{ path?: string; branch?: string }[]>((acc, line) => {
|
|
320
|
+
if (!line) return acc
|
|
321
|
+
if (line.startsWith("worktree ")) {
|
|
322
|
+
acc.push({ path: line.slice("worktree ".length).trim() })
|
|
323
|
+
return acc
|
|
324
|
+
}
|
|
325
|
+
const current = acc[acc.length - 1]
|
|
326
|
+
if (!current) return acc
|
|
327
|
+
if (line.startsWith("branch ")) {
|
|
328
|
+
current.branch = line.slice("branch ".length).trim()
|
|
329
|
+
}
|
|
330
|
+
return acc
|
|
331
|
+
}, [])
|
|
332
|
+
|
|
333
|
+
const entry = entries.find((item) => item.path && path.resolve(item.path) === directory)
|
|
334
|
+
if (!entry?.path) {
|
|
335
|
+
throw new ResetFailedError({ message: "Worktree not found" })
|
|
336
|
+
}
|
|
337
|
+
|
|
338
|
+
const remoteList = await $`git remote`.quiet().nothrow().cwd(Instance.worktree)
|
|
339
|
+
if (remoteList.exitCode !== 0) {
|
|
340
|
+
throw new ResetFailedError({ message: errorText(remoteList) || "Failed to list git remotes" })
|
|
341
|
+
}
|
|
342
|
+
|
|
343
|
+
const remotes = outputText(remoteList.stdout)
|
|
344
|
+
.split("\n")
|
|
345
|
+
.map((line) => line.trim())
|
|
346
|
+
.filter(Boolean)
|
|
347
|
+
|
|
348
|
+
const remote = remotes.includes("origin")
|
|
349
|
+
? "origin"
|
|
350
|
+
: remotes.length === 1
|
|
351
|
+
? remotes[0]
|
|
352
|
+
: remotes.includes("upstream")
|
|
353
|
+
? "upstream"
|
|
354
|
+
: ""
|
|
355
|
+
|
|
356
|
+
const remoteHead = remote
|
|
357
|
+
? await $`git symbolic-ref refs/remotes/${remote}/HEAD`.quiet().nothrow().cwd(Instance.worktree)
|
|
358
|
+
: { exitCode: 1, stdout: undefined, stderr: undefined }
|
|
359
|
+
|
|
360
|
+
const remoteRef = remoteHead.exitCode === 0 ? outputText(remoteHead.stdout) : ""
|
|
361
|
+
const remoteTarget = remoteRef ? remoteRef.replace(/^refs\/remotes\//, "") : ""
|
|
362
|
+
const remoteBranch = remote && remoteTarget.startsWith(`${remote}/`) ? remoteTarget.slice(`${remote}/`.length) : ""
|
|
363
|
+
|
|
364
|
+
const mainCheck = await $`git show-ref --verify --quiet refs/heads/main`.quiet().nothrow().cwd(Instance.worktree)
|
|
365
|
+
const masterCheck = await $`git show-ref --verify --quiet refs/heads/master`
|
|
366
|
+
.quiet()
|
|
367
|
+
.nothrow()
|
|
368
|
+
.cwd(Instance.worktree)
|
|
369
|
+
const localBranch = mainCheck.exitCode === 0 ? "main" : masterCheck.exitCode === 0 ? "master" : ""
|
|
370
|
+
|
|
371
|
+
const target = remoteBranch ? `${remote}/${remoteBranch}` : localBranch
|
|
372
|
+
if (!target) {
|
|
373
|
+
throw new ResetFailedError({ message: "Default branch not found" })
|
|
374
|
+
}
|
|
375
|
+
|
|
376
|
+
if (remoteBranch) {
|
|
377
|
+
const fetch = await $`git fetch ${remote} ${remoteBranch}`.quiet().nothrow().cwd(Instance.worktree)
|
|
378
|
+
if (fetch.exitCode !== 0) {
|
|
379
|
+
throw new ResetFailedError({ message: errorText(fetch) || `Failed to fetch ${target}` })
|
|
380
|
+
}
|
|
381
|
+
}
|
|
382
|
+
|
|
383
|
+
if (!entry.path) {
|
|
384
|
+
throw new ResetFailedError({ message: "Worktree path not found" })
|
|
385
|
+
}
|
|
386
|
+
|
|
387
|
+
const worktreePath = entry.path
|
|
388
|
+
|
|
389
|
+
const resetToTarget = await $`git reset --hard ${target}`.quiet().nothrow().cwd(worktreePath)
|
|
390
|
+
if (resetToTarget.exitCode !== 0) {
|
|
391
|
+
throw new ResetFailedError({ message: errorText(resetToTarget) || "Failed to reset worktree to target" })
|
|
392
|
+
}
|
|
393
|
+
|
|
394
|
+
const clean = await $`git clean -fdx`.quiet().nothrow().cwd(worktreePath)
|
|
395
|
+
if (clean.exitCode !== 0) {
|
|
396
|
+
throw new ResetFailedError({ message: errorText(clean) || "Failed to clean worktree" })
|
|
397
|
+
}
|
|
398
|
+
|
|
399
|
+
const update = await $`git submodule update --init --recursive --force`.quiet().nothrow().cwd(worktreePath)
|
|
400
|
+
if (update.exitCode !== 0) {
|
|
401
|
+
throw new ResetFailedError({ message: errorText(update) || "Failed to update submodules" })
|
|
402
|
+
}
|
|
403
|
+
|
|
404
|
+
const subReset = await $`git submodule foreach --recursive git reset --hard`.quiet().nothrow().cwd(worktreePath)
|
|
405
|
+
if (subReset.exitCode !== 0) {
|
|
406
|
+
throw new ResetFailedError({ message: errorText(subReset) || "Failed to reset submodules" })
|
|
407
|
+
}
|
|
408
|
+
|
|
409
|
+
const subClean = await $`git submodule foreach --recursive git clean -fdx`.quiet().nothrow().cwd(worktreePath)
|
|
410
|
+
if (subClean.exitCode !== 0) {
|
|
411
|
+
throw new ResetFailedError({ message: errorText(subClean) || "Failed to clean submodules" })
|
|
412
|
+
}
|
|
413
|
+
|
|
414
|
+
const status = await $`git status --porcelain=v1`.quiet().nothrow().cwd(worktreePath)
|
|
415
|
+
if (status.exitCode !== 0) {
|
|
416
|
+
throw new ResetFailedError({ message: errorText(status) || "Failed to read git status" })
|
|
417
|
+
}
|
|
418
|
+
|
|
419
|
+
const dirty = outputText(status.stdout)
|
|
420
|
+
if (!dirty) return true
|
|
421
|
+
|
|
422
|
+
throw new ResetFailedError({ message: `Worktree reset left local changes:\n${dirty}` })
|
|
423
|
+
})
|
|
424
|
+
}
|