@epoch-ai/cli 2.2.5 → 2.2.7
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/.artifacts/unit/junit.xml +2823 -0
- package/.project-map/backups/20260530_223453/.project-map.json +90101 -0
- package/.project-map/backups/20260530_223507/.project-map.json +90101 -0
- package/.project-map/backups/20260530_223512/.project-map.json +90101 -0
- package/.project-map/backups/20260530_223512/map.toon +666 -0
- package/.project-map/backups/20260530_223516/.project-map.json +90101 -0
- package/.project-map/backups/20260530_223516/map.toon +666 -0
- package/.project-map/backups/20260530_223520/.project-map.json +90101 -0
- package/.project-map/backups/20260530_223520/map.toon +666 -0
- package/AGENTS.md +47 -0
- package/BUN_SHELL_MIGRATION_PLAN.md +136 -0
- package/Dockerfile +18 -0
- package/README.md +15 -0
- package/bunfig.toml +7 -0
- package/drizzle.config.ts +10 -0
- package/git +0 -0
- package/migration/20260127222353_familiar_lady_ursula/migration.sql +90 -0
- package/migration/20260127222353_familiar_lady_ursula/snapshot.json +796 -0
- package/migration/20260211171708_add_project_commands/migration.sql +1 -0
- package/migration/20260211171708_add_project_commands/snapshot.json +806 -0
- package/migration/20260213144116_wakeful_the_professor/migration.sql +11 -0
- package/migration/20260213144116_wakeful_the_professor/snapshot.json +897 -0
- package/migration/20260225215848_workspace/migration.sql +7 -0
- package/migration/20260225215848_workspace/snapshot.json +959 -0
- package/migration/20260227213759_add_session_workspace_id/migration.sql +2 -0
- package/migration/20260227213759_add_session_workspace_id/snapshot.json +983 -0
- package/migration/20260228203230_blue_harpoon/migration.sql +17 -0
- package/migration/20260228203230_blue_harpoon/snapshot.json +1102 -0
- package/migration/20260303231226_add_workspace_fields/migration.sql +5 -0
- package/migration/20260303231226_add_workspace_fields/snapshot.json +1013 -0
- package/migration/20260309230000_move_org_to_state/migration.sql +3 -0
- package/migration/20260309230000_move_org_to_state/snapshot.json +1156 -0
- package/migration/20260312043431_session_message_cursor/migration.sql +4 -0
- package/migration/20260312043431_session_message_cursor/snapshot.json +1168 -0
- package/migration/20260323234822_events/migration.sql +13 -0
- package/migration/20260323234822_events/snapshot.json +1271 -0
- package/migration/20260418092949_add_yolo_to_session/migration.sql +2 -0
- package/migration/20260418092949_add_yolo_to_session/snapshot.json +1199 -0
- package/migration/20260419120000_add_intervention_to_session/migration.sql +2 -0
- package/package.json +179 -18
- package/parsers-config.ts +290 -0
- package/script/build-node.ts +71 -0
- package/script/build.ts +255 -0
- package/script/check-migrations.ts +16 -0
- package/script/fix-node-pty.ts +28 -0
- package/script/publish.ts +182 -0
- package/script/schema.ts +63 -0
- package/script/seed-e2e.ts +60 -0
- package/script/upgrade-opentui.ts +64 -0
- package/specs/effect-migration.md +310 -0
- package/specs/tui-plugins.md +436 -0
- package/specs/v2.md +14 -0
- package/src/account/account.sql.ts +39 -0
- package/src/account/index.ts +465 -0
- package/src/account/repo.ts +163 -0
- package/src/account/schema.ts +91 -0
- package/src/acp/README.md +174 -0
- package/src/acp/agent.ts +1847 -0
- package/src/acp/session.ts +116 -0
- package/src/acp/types.ts +24 -0
- package/src/agent/agent.ts +445 -0
- package/src/agent/generate.txt +75 -0
- package/src/agent/prompt/compaction.txt +15 -0
- package/src/agent/prompt/explore.txt +9 -0
- package/src/agent/prompt/summary.txt +11 -0
- package/src/agent/prompt/title.txt +44 -0
- package/src/auth/index.ts +110 -0
- package/src/bus/bus-event.ts +40 -0
- package/src/bus/global.ts +10 -0
- package/src/bus/index.ts +232 -0
- package/src/cli/bootstrap.ts +17 -0
- package/src/cli/cmd/account.ts +257 -0
- package/src/cli/cmd/acp.ts +70 -0
- package/src/cli/cmd/agent.ts +245 -0
- package/src/cli/cmd/cmd.ts +7 -0
- package/src/cli/cmd/db.ts +119 -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 +53 -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 +89 -0
- package/src/cli/cmd/generate.ts +38 -0
- package/src/cli/cmd/github.ts +1639 -0
- package/src/cli/cmd/import.ts +169 -0
- package/src/cli/cmd/mcp.ts +754 -0
- package/src/cli/cmd/models.ts +78 -0
- package/src/cli/cmd/plug.ts +233 -0
- package/src/cli/cmd/pr.ts +127 -0
- package/src/cli/cmd/providers.ts +478 -0
- package/src/cli/cmd/run.ts +681 -0
- package/src/cli/cmd/serve.ts +24 -0
- package/src/cli/cmd/session.ts +159 -0
- package/src/cli/cmd/stats.ts +410 -0
- package/src/cli/cmd/tui/app.tsx +945 -0
- package/src/cli/cmd/tui/attach.ts +88 -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 +171 -0
- package/src/cli/cmd/tui/component/dialog-console-org.tsx +103 -0
- package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
- package/src/cli/cmd/tui/component/dialog-model.tsx +190 -0
- package/src/cli/cmd/tui/component/dialog-provider.tsx +364 -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 +168 -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/dialog-variant.tsx +39 -0
- package/src/cli/cmd/tui/component/dialog-workspace-list.tsx +320 -0
- package/src/cli/cmd/tui/component/error-component.tsx +92 -0
- package/src/cli/cmd/tui/component/logo.tsx +85 -0
- package/src/cli/cmd/tui/component/plugin-route-missing.tsx +14 -0
- package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +672 -0
- package/src/cli/cmd/tui/component/prompt/frecency.tsx +90 -0
- package/src/cli/cmd/tui/component/prompt/history.tsx +109 -0
- package/src/cli/cmd/tui/component/prompt/index.tsx +1348 -0
- package/src/cli/cmd/tui/component/prompt/part.ts +16 -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/startup-loading.tsx +63 -0
- package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
- package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
- package/src/cli/cmd/tui/component/workspace/dialog-session-list.tsx +151 -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 +60 -0
- package/src/cli/cmd/tui/context/helper.tsx +25 -0
- package/src/cli/cmd/tui/context/keybind.tsx +105 -0
- package/src/cli/cmd/tui/context/kv.tsx +52 -0
- package/src/cli/cmd/tui/context/local.tsx +456 -0
- package/src/cli/cmd/tui/context/plugin-keybinds.ts +41 -0
- package/src/cli/cmd/tui/context/prompt.tsx +18 -0
- package/src/cli/cmd/tui/context/route.tsx +52 -0
- package/src/cli/cmd/tui/context/sdk.tsx +115 -0
- package/src/cli/cmd/tui/context/sync.tsx +516 -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/epochcli.json +245 -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 +1236 -0
- package/src/cli/cmd/tui/context/tui-config.tsx +9 -0
- package/src/cli/cmd/tui/event.ts +48 -0
- package/src/cli/cmd/tui/feature-plugins/home/footer.tsx +93 -0
- package/src/cli/cmd/tui/feature-plugins/home/tips-view.tsx +145 -0
- package/src/cli/cmd/tui/feature-plugins/home/tips.tsx +50 -0
- package/src/cli/cmd/tui/feature-plugins/sidebar/context.tsx +82 -0
- package/src/cli/cmd/tui/feature-plugins/sidebar/files.tsx +62 -0
- package/src/cli/cmd/tui/feature-plugins/sidebar/footer.tsx +93 -0
- package/src/cli/cmd/tui/feature-plugins/sidebar/lsp.tsx +66 -0
- package/src/cli/cmd/tui/feature-plugins/sidebar/mcp.tsx +96 -0
- package/src/cli/cmd/tui/feature-plugins/sidebar/todo.tsx +48 -0
- package/src/cli/cmd/tui/feature-plugins/system/plugins.tsx +270 -0
- package/src/cli/cmd/tui/plugin/api.tsx +397 -0
- package/src/cli/cmd/tui/plugin/index.ts +3 -0
- package/src/cli/cmd/tui/plugin/internal.ts +27 -0
- package/src/cli/cmd/tui/plugin/runtime.ts +1031 -0
- package/src/cli/cmd/tui/plugin/slots.tsx +60 -0
- package/src/cli/cmd/tui/routes/home.tsx +94 -0
- package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +65 -0
- package/src/cli/cmd/tui/routes/session/dialog-message.tsx +110 -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/index.tsx +2161 -0
- package/src/cli/cmd/tui/routes/session/permission.tsx +691 -0
- package/src/cli/cmd/tui/routes/session/question.tsx +468 -0
- package/src/cli/cmd/tui/routes/session/sidebar.tsx +70 -0
- package/src/cli/cmd/tui/routes/session/subagent-footer.tsx +131 -0
- package/src/cli/cmd/tui/thread.ts +241 -0
- package/src/cli/cmd/tui/ui/dialog-alert.tsx +59 -0
- package/src/cli/cmd/tui/ui/dialog-confirm.tsx +89 -0
- package/src/cli/cmd/tui/ui/dialog-export-options.tsx +211 -0
- package/src/cli/cmd/tui/ui/dialog-help.tsx +40 -0
- package/src/cli/cmd/tui/ui/dialog-prompt.tsx +115 -0
- package/src/cli/cmd/tui/ui/dialog-select.tsx +417 -0
- package/src/cli/cmd/tui/ui/dialog.tsx +192 -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 +192 -0
- package/src/cli/cmd/tui/util/editor.ts +37 -0
- package/src/cli/cmd/tui/util/model.ts +23 -0
- package/src/cli/cmd/tui/util/provider-origin.ts +20 -0
- package/src/cli/cmd/tui/util/scroll.ts +23 -0
- package/src/cli/cmd/tui/util/selection.ts +25 -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 +112 -0
- package/src/cli/cmd/tui/win32.ts +129 -0
- package/src/cli/cmd/tui/worker.ts +195 -0
- package/src/cli/cmd/uninstall.ts +353 -0
- package/src/cli/cmd/upgrade.ts +73 -0
- package/src/cli/cmd/web.ts +81 -0
- package/src/cli/effect/prompt.ts +25 -0
- package/src/cli/error.ts +46 -0
- package/src/cli/heap.ts +59 -0
- package/src/cli/logo.ts +6 -0
- package/src/cli/network.ts +60 -0
- package/src/cli/ui.ts +133 -0
- package/src/cli/upgrade.ts +31 -0
- package/src/command/index.ts +197 -0
- package/src/command/template/initialize.txt +66 -0
- package/src/command/template/review.txt +101 -0
- package/src/config/config.ts +1610 -0
- package/src/config/console-state.ts +15 -0
- package/src/config/markdown.ts +99 -0
- package/src/config/paths.ts +167 -0
- package/src/config/tui-migrate.ts +155 -0
- package/src/config/tui-schema.ts +37 -0
- package/src/config/tui.ts +179 -0
- package/src/config/validator.ts +52 -0
- package/src/control-plane/adaptors/index.ts +20 -0
- package/src/control-plane/adaptors/worktree.ts +42 -0
- package/src/control-plane/schema.ts +17 -0
- package/src/control-plane/sse.ts +66 -0
- package/src/control-plane/types.ts +32 -0
- package/src/control-plane/workspace.sql.ts +17 -0
- package/src/control-plane/workspace.ts +168 -0
- package/src/effect/cross-spawn-spawner.ts +502 -0
- package/src/effect/instance-ref.ts +6 -0
- package/src/effect/instance-registry.ts +12 -0
- package/src/effect/instance-state.ts +82 -0
- package/src/effect/run-service.ts +33 -0
- package/src/effect/runner.ts +216 -0
- package/src/env/index.ts +28 -0
- package/src/file/ignore.ts +82 -0
- package/src/file/index.ts +686 -0
- package/src/file/protected.ts +59 -0
- package/src/file/ripgrep.ts +376 -0
- package/src/file/time.ts +133 -0
- package/src/file/watcher.ts +172 -0
- package/src/filesystem/index.ts +236 -0
- package/src/flag/flag.ts +157 -0
- package/src/format/formatter.ts +413 -0
- package/src/format/index.ts +203 -0
- package/src/git/index.ts +303 -0
- package/src/global/index.ts +54 -0
- package/src/id/id.ts +85 -0
- package/src/ide/index.ts +74 -0
- package/src/index.ts +253 -0
- package/src/installation/index.ts +355 -0
- package/src/installation/meta.ts +7 -0
- package/src/lsp/client.ts +256 -0
- package/src/lsp/index.ts +558 -0
- package/src/lsp/language.ts +120 -0
- package/src/lsp/launch.ts +21 -0
- package/src/lsp/server.ts +1968 -0
- package/src/mcp/auth.ts +173 -0
- package/src/mcp/index.ts +1250 -0
- package/src/mcp/oauth-callback.ts +216 -0
- package/src/mcp/oauth-provider.ts +185 -0
- package/src/mcp/schema-loader.ts +82 -0
- package/src/node.ts +1 -0
- package/src/npm/index.ts +188 -0
- package/src/patch/index.ts +680 -0
- package/src/permission/arity.ts +163 -0
- package/src/permission/evaluate.ts +15 -0
- package/src/permission/index.ts +323 -0
- package/src/permission/schema.ts +17 -0
- package/src/plugin/cloudflare.ts +67 -0
- package/src/plugin/codex.ts +608 -0
- package/src/plugin/github-copilot/copilot.ts +361 -0
- package/src/plugin/github-copilot/models.ts +144 -0
- package/src/plugin/index.ts +288 -0
- package/src/plugin/install.ts +439 -0
- package/src/plugin/loader.ts +174 -0
- package/src/plugin/meta.ts +188 -0
- package/src/plugin/shared.ts +323 -0
- package/src/project/bootstrap.ts +35 -0
- package/src/project/init-files.ts +328 -0
- package/src/project/instance.ts +175 -0
- package/src/project/project.sql.ts +16 -0
- package/src/project/project.ts +519 -0
- package/src/project/schema.ts +16 -0
- package/src/project/state.ts +70 -0
- package/src/project/vcs.ts +240 -0
- package/src/provider/auth.ts +253 -0
- package/src/provider/error.ts +297 -0
- package/src/provider/models.ts +162 -0
- package/src/provider/provider.ts +1776 -0
- package/src/provider/schema.ts +38 -0
- package/src/provider/sdk/copilot/README.md +5 -0
- package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +170 -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 +19 -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 +814 -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 +83 -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 +335 -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 +214 -0
- package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +1769 -0
- package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +173 -0
- package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +1 -0
- package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +87 -0
- package/src/provider/sdk/copilot/responses/tool/file-search.ts +127 -0
- package/src/provider/sdk/copilot/responses/tool/image-generation.ts +114 -0
- package/src/provider/sdk/copilot/responses/tool/local-shell.ts +64 -0
- package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +103 -0
- package/src/provider/sdk/copilot/responses/tool/web-search.ts +102 -0
- package/src/provider/transform.ts +1124 -0
- package/src/pty/index.ts +397 -0
- package/src/pty/pty.bun.ts +26 -0
- package/src/pty/pty.node.ts +27 -0
- package/src/pty/pty.ts +25 -0
- package/src/pty/schema.ts +17 -0
- package/src/question/index.ts +224 -0
- package/src/question/schema.ts +17 -0
- package/src/server/error.ts +36 -0
- package/src/server/event.ts +7 -0
- package/src/server/instance.ts +315 -0
- package/src/server/mdns.ts +60 -0
- package/src/server/middleware.ts +33 -0
- package/src/server/projectors.ts +28 -0
- package/src/server/proxy.ts +130 -0
- package/src/server/router.ts +105 -0
- package/src/server/routes/config.ts +92 -0
- package/src/server/routes/event.ts +83 -0
- package/src/server/routes/experimental.ts +374 -0
- package/src/server/routes/file.ts +197 -0
- package/src/server/routes/global.ts +312 -0
- package/src/server/routes/mcp.ts +225 -0
- package/src/server/routes/permission.ts +69 -0
- package/src/server/routes/project.ts +118 -0
- package/src/server/routes/provider.ts +171 -0
- package/src/server/routes/pty.ts +210 -0
- package/src/server/routes/question.ts +99 -0
- package/src/server/routes/session.ts +984 -0
- package/src/server/routes/tui.ts +378 -0
- package/src/server/routes/workspace.ts +94 -0
- package/src/server/server.ts +353 -0
- package/src/session/compaction.ts +86 -0
- package/src/session/index.ts +904 -0
- package/src/session/instruction.ts +261 -0
- package/src/session/llm/monitor.ts +87 -0
- package/src/session/llm.ts +1676 -0
- package/src/session/message-v2.ts +1082 -0
- package/src/session/message.ts +191 -0
- package/src/session/overflow.ts +34 -0
- package/src/session/processor.ts +635 -0
- package/src/session/projectors.ts +136 -0
- package/src/session/prompt/build-switch.txt +5 -0
- package/src/session/prompt/builder.ts +135 -0
- package/src/session/prompt/default.txt +11 -0
- package/src/session/prompt/engine.ts +1072 -0
- package/src/session/prompt/gemma4.txt +1 -0
- package/src/session/prompt/max-steps.txt +16 -0
- package/src/session/prompt/orchestrator.ts +426 -0
- package/src/session/prompt/plan.txt +28 -0
- package/src/session/prompt/qwen.txt +19 -0
- package/src/session/prompt/resolver.ts +670 -0
- package/src/session/prompt/router.ts +197 -0
- package/src/session/prompt/state.ts +96 -0
- package/src/session/prompt/types.ts +115 -0
- package/src/session/prompt/utils.ts +15 -0
- package/src/session/prompt.ts +362 -0
- package/src/session/retry.ts +106 -0
- package/src/session/revert.ts +176 -0
- package/src/session/sanitizer.ts +125 -0
- package/src/session/schema.ts +38 -0
- package/src/session/session.sql.ts +106 -0
- package/src/session/status.ts +102 -0
- package/src/session/summary.ts +183 -0
- package/src/session/system.ts +79 -0
- package/src/session/todo.ts +166 -0
- package/src/session/worker.ts +382 -0
- package/src/shell/shell.ts +110 -0
- package/src/skill/discovery.ts +116 -0
- package/src/skill/index.ts +287 -0
- package/src/snapshot/index.ts +726 -0
- package/src/sql.d.ts +4 -0
- package/src/storage/db.bun.ts +8 -0
- package/src/storage/db.node.ts +8 -0
- package/src/storage/db.ts +174 -0
- package/src/storage/json-migration.ts +387 -0
- package/src/storage/schema.sql.ts +10 -0
- package/src/storage/schema.ts +4 -0
- package/src/storage/storage.ts +353 -0
- package/src/sync/README.md +179 -0
- package/src/sync/event.sql.ts +16 -0
- package/src/sync/index.ts +263 -0
- package/src/sync/schema.ts +14 -0
- package/src/tool/apply_patch.ts +281 -0
- package/src/tool/apply_patch.txt +1 -0
- package/src/tool/arbitration.txt +5 -0
- package/src/tool/bash.ts +494 -0
- package/src/tool/bash.txt +2 -0
- package/src/tool/batch.ts +183 -0
- package/src/tool/batch.txt +1 -0
- package/src/tool/codesearch.ts +132 -0
- package/src/tool/codesearch.txt +1 -0
- package/src/tool/edit.ts +734 -0
- package/src/tool/edit.txt +1 -0
- package/src/tool/external-directory.ts +46 -0
- package/src/tool/glob.ts +73 -0
- package/src/tool/glob.txt +2 -0
- package/src/tool/grep.ts +156 -0
- package/src/tool/grep.txt +2 -0
- package/src/tool/invalid.ts +20 -0
- package/src/tool/ls.ts +121 -0
- package/src/tool/ls.txt +1 -0
- package/src/tool/lsp.ts +97 -0
- package/src/tool/lsp.txt +1 -0
- package/src/tool/multiedit.ts +46 -0
- package/src/tool/multiedit.txt +1 -0
- package/src/tool/plan-enter.txt +14 -0
- package/src/tool/plan-exit.txt +13 -0
- package/src/tool/plan.ts +131 -0
- package/src/tool/question.ts +46 -0
- package/src/tool/question.txt +10 -0
- package/src/tool/read.ts +332 -0
- package/src/tool/read.txt +1 -0
- package/src/tool/registry.ts +288 -0
- package/src/tool/revert.ts +37 -0
- package/src/tool/schema.ts +17 -0
- package/src/tool/skill.ts +105 -0
- package/src/tool/task.ts +150 -0
- package/src/tool/task.txt +3 -0
- package/src/tool/task_complete.ts +21 -0
- package/src/tool/tool.ts +112 -0
- package/src/tool/truncate.ts +144 -0
- package/src/tool/truncation-dir.ts +4 -0
- package/src/tool/webfetch.ts +206 -0
- package/src/tool/webfetch.txt +1 -0
- package/src/tool/websearch.ts +150 -0
- package/src/tool/websearch.txt +1 -0
- package/src/tool/write.ts +101 -0
- package/src/tool/write.txt +1 -0
- package/src/util/abort.ts +35 -0
- package/src/util/ai-sdk.ts +59 -0
- package/src/util/archive.ts +17 -0
- package/src/util/color.ts +19 -0
- package/src/util/context.ts +25 -0
- package/src/util/data-url.ts +9 -0
- package/src/util/defer.ts +12 -0
- package/src/util/effect-http-client.ts +11 -0
- package/src/util/effect-zod.ts +98 -0
- package/src/util/error.ts +77 -0
- package/src/util/filesystem.ts +245 -0
- package/src/util/flock.ts +333 -0
- package/src/util/fn.ts +21 -0
- package/src/util/format.ts +20 -0
- package/src/util/glob.ts +34 -0
- package/src/util/hash.ts +7 -0
- package/src/util/iife.ts +3 -0
- package/src/util/keybind.ts +103 -0
- package/src/util/lazy.ts +23 -0
- package/src/util/locale.ts +81 -0
- package/src/util/lock.ts +98 -0
- package/src/util/log-parser.ts +114 -0
- package/src/util/log.ts +250 -0
- package/src/util/network.ts +23 -0
- package/src/util/process.ts +176 -0
- package/src/util/queue.ts +32 -0
- package/src/util/record.ts +3 -0
- package/src/util/rpc.ts +66 -0
- package/src/util/schema.ts +53 -0
- package/src/util/scrap.ts +10 -0
- package/src/util/session-analyzer.ts +331 -0
- package/src/util/session-telemetry.ts +91 -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/tokenizer.ts +50 -0
- package/src/util/toon.ts +45 -0
- package/src/util/update-schema.ts +13 -0
- package/src/util/which.ts +14 -0
- package/src/util/wildcard.ts +59 -0
- package/src/worktree/index.ts +612 -0
- package/sst-env.d.ts +10 -0
- package/test/AGENTS.md +81 -0
- package/test/account/repo.test.ts +326 -0
- package/test/account/service.test.ts +393 -0
- package/test/acp/agent-interface.test.ts +51 -0
- package/test/acp/event-subscription.test.ts +685 -0
- package/test/agent/agent.test.ts +716 -0
- package/test/auth/auth.test.ts +58 -0
- package/test/bus/bus-effect.test.ts +164 -0
- package/test/bus/bus-integration.test.ts +87 -0
- package/test/bus/bus.test.ts +219 -0
- package/test/cli/account.test.ts +26 -0
- package/test/cli/cmd/tui/prompt-part.test.ts +47 -0
- package/test/cli/github-action.test.ts +198 -0
- package/test/cli/github-remote.test.ts +80 -0
- package/test/cli/plugin-auth-picker.test.ts +120 -0
- package/test/cli/tui/keybind-plugin.test.ts +90 -0
- package/test/cli/tui/plugin-add.test.ts +107 -0
- package/test/cli/tui/plugin-install.test.ts +89 -0
- package/test/cli/tui/plugin-lifecycle.test.ts +225 -0
- package/test/cli/tui/plugin-loader-entrypoint.test.ts +492 -0
- package/test/cli/tui/plugin-loader-pure.test.ts +72 -0
- package/test/cli/tui/plugin-loader.test.ts +752 -0
- package/test/cli/tui/plugin-toggle.test.ts +159 -0
- package/test/cli/tui/slot-replace.test.tsx +47 -0
- package/test/cli/tui/theme-store.test.ts +51 -0
- package/test/cli/tui/thread.test.ts +128 -0
- package/test/cli/tui/transcript.test.ts +426 -0
- package/test/config/agent-color.test.ts +71 -0
- package/test/config/config.test.ts +2337 -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/config/tui.test.ts +800 -0
- package/test/control-plane/sse.test.ts +56 -0
- package/test/effect/cross-spawn-spawner.test.ts +412 -0
- package/test/effect/instance-state.test.ts +482 -0
- package/test/effect/run-service.test.ts +46 -0
- package/test/effect/runner.test.ts +523 -0
- package/test/fake/provider.ts +82 -0
- package/test/file/fsmonitor.test.ts +62 -0
- package/test/file/ignore.test.ts +10 -0
- package/test/file/index.test.ts +946 -0
- package/test/file/path-traversal.test.ts +198 -0
- package/test/file/ripgrep.test.ts +54 -0
- package/test/file/time.test.ts +445 -0
- package/test/file/watcher.test.ts +247 -0
- package/test/filesystem/filesystem.test.ts +319 -0
- package/test/fixture/db.ts +11 -0
- package/test/fixture/fixture.test.ts +26 -0
- package/test/fixture/fixture.ts +172 -0
- package/test/fixture/flock-worker.ts +72 -0
- package/test/fixture/lsp/fake-lsp-server.js +77 -0
- package/test/fixture/plug-worker.ts +93 -0
- package/test/fixture/plugin-meta-worker.ts +26 -0
- package/test/fixture/skills/agents-sdk/SKILL.md +152 -0
- package/test/fixture/skills/agents-sdk/references/callable.md +92 -0
- package/test/fixture/skills/cloudflare/SKILL.md +211 -0
- package/test/fixture/skills/index.json +6 -0
- package/test/fixture/tui-plugin.ts +328 -0
- package/test/fixture/tui-runtime.ts +27 -0
- package/test/format/format.test.ts +171 -0
- package/test/git/git.test.ts +128 -0
- package/test/ide/ide.test.ts +82 -0
- package/test/installation/installation.test.ts +152 -0
- package/test/keybind.test.ts +421 -0
- package/test/lib/effect.ts +53 -0
- package/test/lib/filesystem.ts +10 -0
- package/test/lib/llm-server.ts +794 -0
- package/test/lsp/client.test.ts +95 -0
- package/test/lsp/index.test.ts +133 -0
- package/test/lsp/launch.test.ts +22 -0
- package/test/lsp/lifecycle.test.ts +147 -0
- package/test/mcp/headers.test.ts +153 -0
- package/test/mcp/lifecycle.test.ts +750 -0
- package/test/mcp/oauth-auto-connect.test.ts +199 -0
- package/test/mcp/oauth-browser.test.ts +249 -0
- package/test/mcp/sc-approve-validator.test.ts +431 -0
- package/test/memory/abort-leak.test.ts +137 -0
- package/test/npm.test.ts +18 -0
- package/test/patch/patch.test.ts +348 -0
- package/test/permission/arity.test.ts +33 -0
- package/test/permission/next.test.ts +1123 -0
- package/test/permission-task.test.ts +323 -0
- package/test/plugin/auth-override.test.ts +74 -0
- package/test/plugin/codex.test.ts +123 -0
- package/test/plugin/github-copilot-models.test.ts +117 -0
- package/test/plugin/install-concurrency.test.ts +140 -0
- package/test/plugin/install.test.ts +570 -0
- package/test/plugin/loader-shared.test.ts +1136 -0
- package/test/plugin/meta.test.ts +137 -0
- package/test/plugin/shared.test.ts +88 -0
- package/test/plugin/trigger.test.ts +111 -0
- package/test/preload.ts +90 -0
- package/test/project/migrate-global.test.ts +140 -0
- package/test/project/project.test.ts +459 -0
- package/test/project/state.test.ts +115 -0
- package/test/project/vcs.test.ts +228 -0
- package/test/project/worktree-remove.test.ts +96 -0
- package/test/project/worktree.test.ts +173 -0
- package/test/provider/amazon-bedrock.test.ts +447 -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/error.test.ts +49 -0
- package/test/provider/gitlab-duo.test.ts +412 -0
- package/test/provider/provider.test.ts +2494 -0
- package/test/provider/transform.test.ts +2944 -0
- package/test/pty/pty-output-isolation.test.ts +141 -0
- package/test/pty/pty-session.test.ts +92 -0
- package/test/pty/pty-shell.test.ts +59 -0
- package/test/question/question.test.ts +453 -0
- package/test/server/global-session-list.test.ts +89 -0
- package/test/server/project-init-git.test.ts +121 -0
- package/test/server/session-actions.test.ts +83 -0
- package/test/server/session-list.test.ts +98 -0
- package/test/server/session-messages.test.ts +159 -0
- package/test/server/session-select.test.ts +84 -0
- package/test/session/compaction.test.ts +683 -0
- package/test/session/continuity-handover.test.ts +620 -0
- package/test/session/deterministic-handover.test.ts +328 -0
- package/test/session/doom-protection.test.ts +247 -0
- package/test/session/hard-reset.test.ts +179 -0
- package/test/session/instruction.test.ts +286 -0
- package/test/session/llm/monitor.test.ts +53 -0
- package/test/session/llm-sanitizer.test.ts +90 -0
- package/test/session/llm-zones-e2e.test.ts +61 -0
- package/test/session/llm.test.ts +1308 -0
- package/test/session/mcpx-normalization.test.ts +86 -0
- package/test/session/mcpx-syntax-recovery.test.ts +28 -0
- package/test/session/message-v2.test.ts +957 -0
- package/test/session/messages-pagination.test.ts +885 -0
- package/test/session/processor-effect.test.ts +805 -0
- package/test/session/prompt/builder.test.ts +71 -0
- package/test/session/prompt/engine-loop.test.ts +80 -0
- package/test/session/prompt/orchestrator.test.ts +108 -0
- package/test/session/prompt/resolver.test.ts +211 -0
- package/test/session/prompt/router.test.ts +84 -0
- package/test/session/prompt/state.test.ts +57 -0
- package/test/session/prompt-effect.test.ts +1241 -0
- package/test/session/prompt.test.ts +522 -0
- package/test/session/refactor-system-zones.test.ts +241 -0
- package/test/session/retry.test.ts +232 -0
- package/test/session/revert-compact.test.ts +621 -0
- package/test/session/sanitizer.test.ts +61 -0
- package/test/session/session.test.ts +142 -0
- package/test/session/snapshot-tool-race.test.ts +242 -0
- package/test/session/structured-output-integration.test.ts +233 -0
- package/test/session/structured-output.test.ts +391 -0
- package/test/session/system.test.ts +59 -0
- package/test/session/telemetry.test.ts +35 -0
- package/test/shell/shell.test.ts +73 -0
- package/test/skill/discovery.test.ts +116 -0
- package/test/skill/skill.test.ts +392 -0
- package/test/snapshot/snapshot.test.ts +1404 -0
- package/test/storage/db.test.ts +14 -0
- package/test/storage/json-migration.test.ts +791 -0
- package/test/storage/storage.test.ts +295 -0
- package/test/sync/index.test.ts +191 -0
- package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
- package/test/tool/apply_patch.test.ts +567 -0
- package/test/tool/bash.test.ts +1099 -0
- package/test/tool/edit.test.ts +681 -0
- package/test/tool/external-directory.test.ts +198 -0
- package/test/tool/fixtures/large-image.png +0 -0
- package/test/tool/fixtures/models-api.json +65179 -0
- package/test/tool/grep.test.ts +111 -0
- package/test/tool/question.test.ts +126 -0
- package/test/tool/read.test.ts +468 -0
- package/test/tool/registry.test.ts +126 -0
- package/test/tool/skill.test.ts +167 -0
- package/test/tool/task.test.ts +49 -0
- package/test/tool/tool-define.test.ts +101 -0
- package/test/tool/truncation.test.ts +161 -0
- package/test/tool/webfetch.test.ts +101 -0
- package/test/tool/write.test.ts +354 -0
- package/test/util/data-url.test.ts +14 -0
- package/test/util/effect-zod.test.ts +61 -0
- package/test/util/error.test.ts +38 -0
- package/test/util/filesystem.test.ts +656 -0
- package/test/util/flock.test.ts +383 -0
- package/test/util/format.test.ts +59 -0
- package/test/util/glob.test.ts +164 -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/log-parser.test.ts +61 -0
- package/test/util/module.test.ts +59 -0
- package/test/util/process.test.ts +128 -0
- package/test/util/telemetry-integration.test.ts +104 -0
- package/test/util/timeout.test.ts +21 -0
- package/test/util/which.test.ts +100 -0
- package/test/util/wildcard.test.ts +90 -0
- package/test-regex.js +50 -0
- package/tsconfig.json +23 -0
- package/LICENSE +0 -21
- /package/{postinstall.mjs → script/postinstall.mjs} +0 -0
|
@@ -0,0 +1,216 @@
|
|
|
1
|
+
import { createConnection } from "net"
|
|
2
|
+
import { createServer } from "http"
|
|
3
|
+
import { Log } from "../util/log"
|
|
4
|
+
import { OAUTH_CALLBACK_PORT, OAUTH_CALLBACK_PATH } from "./oauth-provider"
|
|
5
|
+
|
|
6
|
+
const log = Log.create({ service: "mcp.oauth-callback" })
|
|
7
|
+
|
|
8
|
+
const HTML_SUCCESS = `<!DOCTYPE html>
|
|
9
|
+
<html>
|
|
10
|
+
<head>
|
|
11
|
+
<title>Epoch CLI - Authorization Successful</title>
|
|
12
|
+
<style>
|
|
13
|
+
body { font-family: system-ui, -apple-system, sans-serif; display: flex; justify-content: center; align-items: center; height: 100vh; margin: 0; background: #1a1a2e; color: #eee; }
|
|
14
|
+
.container { text-align: center; padding: 2rem; }
|
|
15
|
+
h1 { color: #4ade80; margin-bottom: 1rem; }
|
|
16
|
+
p { color: #aaa; }
|
|
17
|
+
</style>
|
|
18
|
+
</head>
|
|
19
|
+
<body>
|
|
20
|
+
<div class="container">
|
|
21
|
+
<h1>Authorization Successful</h1>
|
|
22
|
+
<p>You can close this window and return to Epoch CLI.</p>
|
|
23
|
+
</div>
|
|
24
|
+
<script>setTimeout(() => window.close(), 2000);</script>
|
|
25
|
+
</body>
|
|
26
|
+
</html>`
|
|
27
|
+
|
|
28
|
+
const HTML_ERROR = (error: string) => `<!DOCTYPE html>
|
|
29
|
+
<html>
|
|
30
|
+
<head>
|
|
31
|
+
<title>Epoch CLI - Authorization Failed</title>
|
|
32
|
+
<style>
|
|
33
|
+
body { font-family: system-ui, -apple-system, sans-serif; display: flex; justify-content: center; align-items: center; height: 100vh; margin: 0; background: #1a1a2e; color: #eee; }
|
|
34
|
+
.container { text-align: center; padding: 2rem; }
|
|
35
|
+
h1 { color: #f87171; margin-bottom: 1rem; }
|
|
36
|
+
p { color: #aaa; }
|
|
37
|
+
.error { color: #fca5a5; font-family: monospace; margin-top: 1rem; padding: 1rem; background: rgba(248,113,113,0.1); border-radius: 0.5rem; }
|
|
38
|
+
</style>
|
|
39
|
+
</head>
|
|
40
|
+
<body>
|
|
41
|
+
<div class="container">
|
|
42
|
+
<h1>Authorization Failed</h1>
|
|
43
|
+
<p>An error occurred during authorization.</p>
|
|
44
|
+
<div class="error">${error}</div>
|
|
45
|
+
</div>
|
|
46
|
+
</body>
|
|
47
|
+
</html>`
|
|
48
|
+
|
|
49
|
+
interface PendingAuth {
|
|
50
|
+
resolve: (code: string) => void
|
|
51
|
+
reject: (error: Error) => void
|
|
52
|
+
timeout: ReturnType<typeof setTimeout>
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
export namespace McpOAuthCallback {
|
|
56
|
+
let server: ReturnType<typeof createServer> | undefined
|
|
57
|
+
const pendingAuths = new Map<string, PendingAuth>()
|
|
58
|
+
// Reverse index: mcpName → oauthState, so cancelPending(mcpName) can
|
|
59
|
+
// find the right entry in pendingAuths (which is keyed by oauthState).
|
|
60
|
+
const mcpNameToState = new Map<string, string>()
|
|
61
|
+
|
|
62
|
+
const CALLBACK_TIMEOUT_MS = 5 * 60 * 1000 // 5 minutes
|
|
63
|
+
|
|
64
|
+
function cleanupStateIndex(oauthState: string) {
|
|
65
|
+
for (const [name, state] of mcpNameToState) {
|
|
66
|
+
if (state === oauthState) {
|
|
67
|
+
mcpNameToState.delete(name)
|
|
68
|
+
break
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
function handleRequest(req: import("http").IncomingMessage, res: import("http").ServerResponse) {
|
|
74
|
+
const url = new URL(req.url || "/", `http://localhost:${OAUTH_CALLBACK_PORT}`)
|
|
75
|
+
|
|
76
|
+
if (url.pathname !== OAUTH_CALLBACK_PATH) {
|
|
77
|
+
res.writeHead(404)
|
|
78
|
+
res.end("Not found")
|
|
79
|
+
return
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
const code = url.searchParams.get("code")
|
|
83
|
+
const state = url.searchParams.get("state")
|
|
84
|
+
const error = url.searchParams.get("error")
|
|
85
|
+
const errorDescription = url.searchParams.get("error_description")
|
|
86
|
+
|
|
87
|
+
log.info("received oauth callback", { hasCode: !!code, state, error })
|
|
88
|
+
|
|
89
|
+
// Enforce state parameter presence
|
|
90
|
+
if (!state) {
|
|
91
|
+
const errorMsg = "Missing required state parameter - potential CSRF attack"
|
|
92
|
+
log.error("oauth callback missing state parameter", { url: url.toString() })
|
|
93
|
+
res.writeHead(400, { "Content-Type": "text/html" })
|
|
94
|
+
res.end(HTML_ERROR(errorMsg))
|
|
95
|
+
return
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
if (error) {
|
|
99
|
+
const errorMsg = errorDescription || error
|
|
100
|
+
if (pendingAuths.has(state)) {
|
|
101
|
+
const pending = pendingAuths.get(state)!
|
|
102
|
+
clearTimeout(pending.timeout)
|
|
103
|
+
pendingAuths.delete(state)
|
|
104
|
+
cleanupStateIndex(state)
|
|
105
|
+
pending.reject(new Error(errorMsg))
|
|
106
|
+
}
|
|
107
|
+
res.writeHead(200, { "Content-Type": "text/html" })
|
|
108
|
+
res.end(HTML_ERROR(errorMsg))
|
|
109
|
+
return
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
if (!code) {
|
|
113
|
+
res.writeHead(400, { "Content-Type": "text/html" })
|
|
114
|
+
res.end(HTML_ERROR("No authorization code provided"))
|
|
115
|
+
return
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
// Validate state parameter
|
|
119
|
+
if (!pendingAuths.has(state)) {
|
|
120
|
+
const errorMsg = "Invalid or expired state parameter - potential CSRF attack"
|
|
121
|
+
log.error("oauth callback with invalid state", { state, pendingStates: Array.from(pendingAuths.keys()) })
|
|
122
|
+
res.writeHead(400, { "Content-Type": "text/html" })
|
|
123
|
+
res.end(HTML_ERROR(errorMsg))
|
|
124
|
+
return
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
const pending = pendingAuths.get(state)!
|
|
128
|
+
|
|
129
|
+
clearTimeout(pending.timeout)
|
|
130
|
+
pendingAuths.delete(state)
|
|
131
|
+
cleanupStateIndex(state)
|
|
132
|
+
pending.resolve(code)
|
|
133
|
+
|
|
134
|
+
res.writeHead(200, { "Content-Type": "text/html" })
|
|
135
|
+
res.end(HTML_SUCCESS)
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
export async function ensureRunning(): Promise<void> {
|
|
139
|
+
if (server) return
|
|
140
|
+
|
|
141
|
+
const running = await isPortInUse()
|
|
142
|
+
if (running) {
|
|
143
|
+
log.info("oauth callback server already running on another instance", { port: OAUTH_CALLBACK_PORT })
|
|
144
|
+
return
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
server = createServer(handleRequest)
|
|
148
|
+
await new Promise<void>((resolve, reject) => {
|
|
149
|
+
server!.listen(OAUTH_CALLBACK_PORT, () => {
|
|
150
|
+
log.info("oauth callback server started", { port: OAUTH_CALLBACK_PORT })
|
|
151
|
+
resolve()
|
|
152
|
+
})
|
|
153
|
+
server!.on("error", reject)
|
|
154
|
+
})
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
export function waitForCallback(oauthState: string, mcpName?: string): Promise<string> {
|
|
158
|
+
if (mcpName) mcpNameToState.set(mcpName, oauthState)
|
|
159
|
+
return new Promise((resolve, reject) => {
|
|
160
|
+
const timeout = setTimeout(() => {
|
|
161
|
+
if (pendingAuths.has(oauthState)) {
|
|
162
|
+
pendingAuths.delete(oauthState)
|
|
163
|
+
if (mcpName) mcpNameToState.delete(mcpName)
|
|
164
|
+
reject(new Error("OAuth callback timeout - authorization took too long"))
|
|
165
|
+
}
|
|
166
|
+
}, CALLBACK_TIMEOUT_MS)
|
|
167
|
+
|
|
168
|
+
pendingAuths.set(oauthState, { resolve, reject, timeout })
|
|
169
|
+
})
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
export function cancelPending(mcpName: string): void {
|
|
173
|
+
// Look up the oauthState for this mcpName via the reverse index
|
|
174
|
+
const oauthState = mcpNameToState.get(mcpName)
|
|
175
|
+
const key = oauthState ?? mcpName
|
|
176
|
+
const pending = pendingAuths.get(key)
|
|
177
|
+
if (pending) {
|
|
178
|
+
clearTimeout(pending.timeout)
|
|
179
|
+
pendingAuths.delete(key)
|
|
180
|
+
mcpNameToState.delete(mcpName)
|
|
181
|
+
pending.reject(new Error("Authorization cancelled"))
|
|
182
|
+
}
|
|
183
|
+
}
|
|
184
|
+
|
|
185
|
+
export async function isPortInUse(): Promise<boolean> {
|
|
186
|
+
return new Promise((resolve) => {
|
|
187
|
+
const socket = createConnection(OAUTH_CALLBACK_PORT, "127.0.0.1")
|
|
188
|
+
socket.on("connect", () => {
|
|
189
|
+
socket.destroy()
|
|
190
|
+
resolve(true)
|
|
191
|
+
})
|
|
192
|
+
socket.on("error", () => {
|
|
193
|
+
resolve(false)
|
|
194
|
+
})
|
|
195
|
+
})
|
|
196
|
+
}
|
|
197
|
+
|
|
198
|
+
export async function stop(): Promise<void> {
|
|
199
|
+
if (server) {
|
|
200
|
+
await new Promise<void>((resolve) => server!.close(() => resolve()))
|
|
201
|
+
server = undefined
|
|
202
|
+
log.info("oauth callback server stopped")
|
|
203
|
+
}
|
|
204
|
+
|
|
205
|
+
for (const [name, pending] of pendingAuths) {
|
|
206
|
+
clearTimeout(pending.timeout)
|
|
207
|
+
pending.reject(new Error("OAuth callback server stopped"))
|
|
208
|
+
}
|
|
209
|
+
pendingAuths.clear()
|
|
210
|
+
mcpNameToState.clear()
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
export function isRunning(): boolean {
|
|
214
|
+
return server !== undefined
|
|
215
|
+
}
|
|
216
|
+
}
|
|
@@ -0,0 +1,185 @@
|
|
|
1
|
+
import type { OAuthClientProvider } from "@modelcontextprotocol/sdk/client/auth.js"
|
|
2
|
+
import type {
|
|
3
|
+
OAuthClientMetadata,
|
|
4
|
+
OAuthTokens,
|
|
5
|
+
OAuthClientInformation,
|
|
6
|
+
OAuthClientInformationFull,
|
|
7
|
+
} from "@modelcontextprotocol/sdk/shared/auth.js"
|
|
8
|
+
import { McpAuth } from "./auth"
|
|
9
|
+
import { Log } from "../util/log"
|
|
10
|
+
|
|
11
|
+
const log = Log.create({ service: "mcp.oauth" })
|
|
12
|
+
|
|
13
|
+
const OAUTH_CALLBACK_PORT = 19876
|
|
14
|
+
const OAUTH_CALLBACK_PATH = "/mcp/oauth/callback"
|
|
15
|
+
|
|
16
|
+
export interface McpOAuthConfig {
|
|
17
|
+
clientId?: string
|
|
18
|
+
clientSecret?: string
|
|
19
|
+
scope?: string
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
export interface McpOAuthCallbacks {
|
|
23
|
+
onRedirect: (url: URL) => void | Promise<void>
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
export class McpOAuthProvider implements OAuthClientProvider {
|
|
27
|
+
constructor(
|
|
28
|
+
private mcpName: string,
|
|
29
|
+
private serverUrl: string,
|
|
30
|
+
private config: McpOAuthConfig,
|
|
31
|
+
private callbacks: McpOAuthCallbacks,
|
|
32
|
+
) {}
|
|
33
|
+
|
|
34
|
+
get redirectUrl(): string {
|
|
35
|
+
return `http://127.0.0.1:${OAUTH_CALLBACK_PORT}${OAUTH_CALLBACK_PATH}`
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
get clientMetadata(): OAuthClientMetadata {
|
|
39
|
+
return {
|
|
40
|
+
redirect_uris: [this.redirectUrl],
|
|
41
|
+
client_name: "Epoch CLI",
|
|
42
|
+
client_uri: "https://epochcli.ai",
|
|
43
|
+
grant_types: ["authorization_code", "refresh_token"],
|
|
44
|
+
response_types: ["code"],
|
|
45
|
+
token_endpoint_auth_method: this.config.clientSecret ? "client_secret_post" : "none",
|
|
46
|
+
}
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
async clientInformation(): Promise<OAuthClientInformation | undefined> {
|
|
50
|
+
// Check config first (pre-registered client)
|
|
51
|
+
if (this.config.clientId) {
|
|
52
|
+
return {
|
|
53
|
+
client_id: this.config.clientId,
|
|
54
|
+
client_secret: this.config.clientSecret,
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
// Check stored client info (from dynamic registration)
|
|
59
|
+
// Use getForUrl to validate credentials are for the current server URL
|
|
60
|
+
const entry = await McpAuth.getForUrl(this.mcpName, this.serverUrl)
|
|
61
|
+
if (entry?.clientInfo) {
|
|
62
|
+
// Check if client secret has expired
|
|
63
|
+
if (entry.clientInfo.clientSecretExpiresAt && entry.clientInfo.clientSecretExpiresAt < Date.now() / 1000) {
|
|
64
|
+
log.info("client secret expired, need to re-register", { mcpName: this.mcpName })
|
|
65
|
+
return undefined
|
|
66
|
+
}
|
|
67
|
+
return {
|
|
68
|
+
client_id: entry.clientInfo.clientId,
|
|
69
|
+
client_secret: entry.clientInfo.clientSecret,
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
// No client info or URL changed - will trigger dynamic registration
|
|
74
|
+
return undefined
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
async saveClientInformation(info: OAuthClientInformationFull): Promise<void> {
|
|
78
|
+
await McpAuth.updateClientInfo(
|
|
79
|
+
this.mcpName,
|
|
80
|
+
{
|
|
81
|
+
clientId: info.client_id,
|
|
82
|
+
clientSecret: info.client_secret,
|
|
83
|
+
clientIdIssuedAt: info.client_id_issued_at,
|
|
84
|
+
clientSecretExpiresAt: info.client_secret_expires_at,
|
|
85
|
+
},
|
|
86
|
+
this.serverUrl,
|
|
87
|
+
)
|
|
88
|
+
log.info("saved dynamically registered client", {
|
|
89
|
+
mcpName: this.mcpName,
|
|
90
|
+
clientId: info.client_id,
|
|
91
|
+
})
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
async tokens(): Promise<OAuthTokens | undefined> {
|
|
95
|
+
// Use getForUrl to validate tokens are for the current server URL
|
|
96
|
+
const entry = await McpAuth.getForUrl(this.mcpName, this.serverUrl)
|
|
97
|
+
if (!entry?.tokens) return undefined
|
|
98
|
+
|
|
99
|
+
return {
|
|
100
|
+
access_token: entry.tokens.accessToken,
|
|
101
|
+
token_type: "Bearer",
|
|
102
|
+
refresh_token: entry.tokens.refreshToken,
|
|
103
|
+
expires_in: entry.tokens.expiresAt
|
|
104
|
+
? Math.max(0, Math.floor(entry.tokens.expiresAt - Date.now() / 1000))
|
|
105
|
+
: undefined,
|
|
106
|
+
scope: entry.tokens.scope,
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
async saveTokens(tokens: OAuthTokens): Promise<void> {
|
|
111
|
+
await McpAuth.updateTokens(
|
|
112
|
+
this.mcpName,
|
|
113
|
+
{
|
|
114
|
+
accessToken: tokens.access_token,
|
|
115
|
+
refreshToken: tokens.refresh_token,
|
|
116
|
+
expiresAt: tokens.expires_in ? Date.now() / 1000 + tokens.expires_in : undefined,
|
|
117
|
+
scope: tokens.scope,
|
|
118
|
+
},
|
|
119
|
+
this.serverUrl,
|
|
120
|
+
)
|
|
121
|
+
log.info("saved oauth tokens", { mcpName: this.mcpName })
|
|
122
|
+
}
|
|
123
|
+
|
|
124
|
+
async redirectToAuthorization(authorizationUrl: URL): Promise<void> {
|
|
125
|
+
log.info("redirecting to authorization", { mcpName: this.mcpName, url: authorizationUrl.toString() })
|
|
126
|
+
await this.callbacks.onRedirect(authorizationUrl)
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
async saveCodeVerifier(codeVerifier: string): Promise<void> {
|
|
130
|
+
await McpAuth.updateCodeVerifier(this.mcpName, codeVerifier)
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
async codeVerifier(): Promise<string> {
|
|
134
|
+
const entry = await McpAuth.get(this.mcpName)
|
|
135
|
+
if (!entry?.codeVerifier) {
|
|
136
|
+
throw new Error(`No code verifier saved for MCP server: ${this.mcpName}`)
|
|
137
|
+
}
|
|
138
|
+
return entry.codeVerifier
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
async saveState(state: string): Promise<void> {
|
|
142
|
+
await McpAuth.updateOAuthState(this.mcpName, state)
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
async state(): Promise<string> {
|
|
146
|
+
const entry = await McpAuth.get(this.mcpName)
|
|
147
|
+
if (entry?.oauthState) {
|
|
148
|
+
return entry.oauthState
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
// Generate a new state if none exists — the SDK calls state() as a
|
|
152
|
+
// generator, not just a reader, so we need to produce a value even when
|
|
153
|
+
// startAuth() hasn't pre-saved one (e.g. during automatic auth on first
|
|
154
|
+
// connect).
|
|
155
|
+
const newState = Array.from(crypto.getRandomValues(new Uint8Array(32)))
|
|
156
|
+
.map((b) => b.toString(16).padStart(2, "0"))
|
|
157
|
+
.join("")
|
|
158
|
+
await McpAuth.updateOAuthState(this.mcpName, newState)
|
|
159
|
+
return newState
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
async invalidateCredentials(type: "all" | "client" | "tokens"): Promise<void> {
|
|
163
|
+
log.info("invalidating credentials", { mcpName: this.mcpName, type })
|
|
164
|
+
const entry = await McpAuth.get(this.mcpName)
|
|
165
|
+
if (!entry) {
|
|
166
|
+
return
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
switch (type) {
|
|
170
|
+
case "all":
|
|
171
|
+
await McpAuth.remove(this.mcpName)
|
|
172
|
+
break
|
|
173
|
+
case "client":
|
|
174
|
+
delete entry.clientInfo
|
|
175
|
+
await McpAuth.set(this.mcpName, entry)
|
|
176
|
+
break
|
|
177
|
+
case "tokens":
|
|
178
|
+
delete entry.tokens
|
|
179
|
+
await McpAuth.set(this.mcpName, entry)
|
|
180
|
+
break
|
|
181
|
+
}
|
|
182
|
+
}
|
|
183
|
+
}
|
|
184
|
+
|
|
185
|
+
export { OAUTH_CALLBACK_PORT, OAUTH_CALLBACK_PATH }
|
|
@@ -0,0 +1,82 @@
|
|
|
1
|
+
import { MCP } from "./index"
|
|
2
|
+
import { Log } from "@/util/log"
|
|
3
|
+
import { Process } from "@/util/process"
|
|
4
|
+
import { Instance } from "@/project/instance"
|
|
5
|
+
|
|
6
|
+
const log = Log.create({ service: "mcp-schema-loader" })
|
|
7
|
+
|
|
8
|
+
export namespace SchemaContextLoader {
|
|
9
|
+
// Simple in-memory cache for tool schemas
|
|
10
|
+
const schemaCache = new Map<string, { schema: any; timestamp: number }>()
|
|
11
|
+
const CACHE_TTL = 60_000 // 1 minute
|
|
12
|
+
|
|
13
|
+
/**
|
|
14
|
+
* Retrieves the input schema for a specific mcpx tool via subprocess.
|
|
15
|
+
*/
|
|
16
|
+
export async function getMcpxToolSchema(
|
|
17
|
+
server: string,
|
|
18
|
+
tool: string,
|
|
19
|
+
binaryPath: string = "mcpx-rust",
|
|
20
|
+
): Promise<any | undefined> {
|
|
21
|
+
const cacheKey = `mcpx:${server}:${tool}`
|
|
22
|
+
const cached = schemaCache.get(cacheKey)
|
|
23
|
+
if (cached && Date.now() - cached.timestamp < CACHE_TTL) {
|
|
24
|
+
return cached.schema
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
try {
|
|
28
|
+
const res = await Process.run([binaryPath, server, tool, "--help", "--json"], {
|
|
29
|
+
cwd: Instance.directory,
|
|
30
|
+
nothrow: true,
|
|
31
|
+
})
|
|
32
|
+
|
|
33
|
+
if (res.code === 0) {
|
|
34
|
+
const data = JSON.parse(res.stdout.toString())
|
|
35
|
+
if (data && data.input_schema) {
|
|
36
|
+
schemaCache.set(cacheKey, { schema: data.input_schema, timestamp: Date.now() })
|
|
37
|
+
return data.input_schema
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
} catch (e) {
|
|
41
|
+
log.debug("Failed to fetch mcpx schema", { server, tool, error: String(e) })
|
|
42
|
+
}
|
|
43
|
+
return undefined
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
/**
|
|
47
|
+
* Retrieves the input schema for a specific tool.
|
|
48
|
+
*/
|
|
49
|
+
export async function getToolSchema(toolName: string): Promise<any | undefined> {
|
|
50
|
+
// Check cache first
|
|
51
|
+
const cached = schemaCache.get(toolName)
|
|
52
|
+
if (cached && Date.now() - cached.timestamp < CACHE_TTL) {
|
|
53
|
+
return cached.schema
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
try {
|
|
57
|
+
const mcpClientsRecord = await MCP.clients()
|
|
58
|
+
const mcpClients = Object.values(mcpClientsRecord) as any[]
|
|
59
|
+
|
|
60
|
+
for (const entry of mcpClients) {
|
|
61
|
+
try {
|
|
62
|
+
// toolName in MCP is often prefixed by server name, but epochcli might use raw names
|
|
63
|
+
// or handle the mapping. MCP.clients() returns a record where keys are server names.
|
|
64
|
+
// We need to find which server has this tool.
|
|
65
|
+
const toolsRes = await entry.client.listTools()
|
|
66
|
+
const tool = toolsRes.tools.find((t: any) => t.name === toolName)
|
|
67
|
+
if (tool) {
|
|
68
|
+
// Update cache
|
|
69
|
+
schemaCache.set(toolName, { schema: tool.inputSchema, timestamp: Date.now() })
|
|
70
|
+
return tool.inputSchema
|
|
71
|
+
}
|
|
72
|
+
} catch (e) {
|
|
73
|
+
// Continue to next client if one fails
|
|
74
|
+
log.debug("Failed to list tools for client", { clientId: entry.id, error: String(e) })
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
} catch (e) {
|
|
78
|
+
log.warn("Failed to load MCP clients for schema loading", { error: String(e) })
|
|
79
|
+
}
|
|
80
|
+
return undefined
|
|
81
|
+
}
|
|
82
|
+
}
|
package/src/node.ts
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export { Server } from "./server/server"
|
package/src/npm/index.ts
ADDED
|
@@ -0,0 +1,188 @@
|
|
|
1
|
+
import semver from "semver"
|
|
2
|
+
import z from "zod"
|
|
3
|
+
import { NamedError } from "@epoch-ai/util/error"
|
|
4
|
+
import { Global } from "../global"
|
|
5
|
+
import { Log } from "../util/log"
|
|
6
|
+
import path from "path"
|
|
7
|
+
import { readdir, rm } from "fs/promises"
|
|
8
|
+
import { Filesystem } from "@/util/filesystem"
|
|
9
|
+
import { Flock } from "@/util/flock"
|
|
10
|
+
import { Arborist } from "@npmcli/arborist"
|
|
11
|
+
|
|
12
|
+
export namespace Npm {
|
|
13
|
+
const log = Log.create({ service: "npm" })
|
|
14
|
+
const illegal = process.platform === "win32" ? new Set(["<", ">", ":", '"', "|", "?", "*"]) : undefined
|
|
15
|
+
|
|
16
|
+
export const InstallFailedError = NamedError.create(
|
|
17
|
+
"NpmInstallFailedError",
|
|
18
|
+
z.object({
|
|
19
|
+
pkg: z.string(),
|
|
20
|
+
}),
|
|
21
|
+
)
|
|
22
|
+
|
|
23
|
+
export function sanitize(pkg: string) {
|
|
24
|
+
if (!illegal) return pkg
|
|
25
|
+
return Array.from(pkg, (char) => (illegal.has(char) || char.charCodeAt(0) < 32 ? "_" : char)).join("")
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
function directory(pkg: string) {
|
|
29
|
+
return path.join(Global.Path.cache, "packages", sanitize(pkg))
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
function resolveEntryPoint(name: string, dir: string) {
|
|
33
|
+
let entrypoint: string | undefined
|
|
34
|
+
try {
|
|
35
|
+
entrypoint = typeof Bun !== "undefined" ? import.meta.resolve(name, dir) : import.meta.resolve(dir)
|
|
36
|
+
} catch {}
|
|
37
|
+
const result = {
|
|
38
|
+
directory: dir,
|
|
39
|
+
entrypoint,
|
|
40
|
+
}
|
|
41
|
+
return result
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
export async function outdated(pkg: string, cachedVersion: string): Promise<boolean> {
|
|
45
|
+
const response = await fetch(`https://registry.npmjs.org/${pkg}`)
|
|
46
|
+
if (!response.ok) {
|
|
47
|
+
log.warn("Failed to resolve latest version, using cached", { pkg, cachedVersion })
|
|
48
|
+
return false
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
const data = (await response.json()) as { "dist-tags"?: { latest?: string } }
|
|
52
|
+
const latestVersion = data?.["dist-tags"]?.latest
|
|
53
|
+
if (!latestVersion) {
|
|
54
|
+
log.warn("No latest version found, using cached", { pkg, cachedVersion })
|
|
55
|
+
return false
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
const range = /[\s^~*xX<>|=]/.test(cachedVersion)
|
|
59
|
+
if (range) return !semver.satisfies(latestVersion, cachedVersion)
|
|
60
|
+
|
|
61
|
+
return semver.lt(cachedVersion, latestVersion)
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
export async function add(pkg: string) {
|
|
65
|
+
const dir = directory(pkg)
|
|
66
|
+
await using _ = await Flock.acquire(`npm-install:${Filesystem.resolve(dir)}`)
|
|
67
|
+
log.info("installing package", {
|
|
68
|
+
pkg,
|
|
69
|
+
})
|
|
70
|
+
|
|
71
|
+
const arborist = new Arborist({
|
|
72
|
+
path: dir,
|
|
73
|
+
binLinks: true,
|
|
74
|
+
progress: false,
|
|
75
|
+
savePrefix: "",
|
|
76
|
+
ignoreScripts: true,
|
|
77
|
+
})
|
|
78
|
+
const tree = await arborist.loadVirtual().catch(() => {})
|
|
79
|
+
if (tree) {
|
|
80
|
+
const first = tree.edgesOut.values().next().value?.to
|
|
81
|
+
if (first) {
|
|
82
|
+
return resolveEntryPoint(first.name, first.path)
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
const result = await arborist
|
|
87
|
+
.reify({
|
|
88
|
+
add: [pkg],
|
|
89
|
+
save: true,
|
|
90
|
+
saveType: "prod",
|
|
91
|
+
})
|
|
92
|
+
.catch((cause) => {
|
|
93
|
+
throw new InstallFailedError(
|
|
94
|
+
{ pkg },
|
|
95
|
+
{
|
|
96
|
+
cause,
|
|
97
|
+
},
|
|
98
|
+
)
|
|
99
|
+
})
|
|
100
|
+
|
|
101
|
+
const first = result.edgesOut.values().next().value?.to
|
|
102
|
+
if (!first) throw new InstallFailedError({ pkg })
|
|
103
|
+
return resolveEntryPoint(first.name, first.path)
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
export async function install(dir: string) {
|
|
107
|
+
await using _ = await Flock.acquire(`npm-install:${dir}`)
|
|
108
|
+
log.info("checking dependencies", { dir })
|
|
109
|
+
|
|
110
|
+
const reify = async () => {
|
|
111
|
+
const arb = new Arborist({
|
|
112
|
+
path: dir,
|
|
113
|
+
binLinks: true,
|
|
114
|
+
progress: false,
|
|
115
|
+
savePrefix: "",
|
|
116
|
+
ignoreScripts: true,
|
|
117
|
+
})
|
|
118
|
+
await arb.reify().catch(() => {})
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
if (!(await Filesystem.exists(path.join(dir, "node_modules")))) {
|
|
122
|
+
log.info("node_modules missing, reifying")
|
|
123
|
+
await reify()
|
|
124
|
+
return
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
const pkg = await Filesystem.readJson(path.join(dir, "package.json")).catch(() => ({}))
|
|
128
|
+
const lock = await Filesystem.readJson(path.join(dir, "package-lock.json")).catch(() => ({}))
|
|
129
|
+
|
|
130
|
+
const declared = new Set([
|
|
131
|
+
...Object.keys(pkg.dependencies || {}),
|
|
132
|
+
...Object.keys(pkg.devDependencies || {}),
|
|
133
|
+
...Object.keys(pkg.peerDependencies || {}),
|
|
134
|
+
...Object.keys(pkg.optionalDependencies || {}),
|
|
135
|
+
])
|
|
136
|
+
|
|
137
|
+
const root = lock.packages?.[""] || {}
|
|
138
|
+
const locked = new Set([
|
|
139
|
+
...Object.keys(root.dependencies || {}),
|
|
140
|
+
...Object.keys(root.devDependencies || {}),
|
|
141
|
+
...Object.keys(root.peerDependencies || {}),
|
|
142
|
+
...Object.keys(root.optionalDependencies || {}),
|
|
143
|
+
])
|
|
144
|
+
|
|
145
|
+
for (const name of declared) {
|
|
146
|
+
if (!locked.has(name)) {
|
|
147
|
+
log.info("dependency not in lock file, reifying", { name })
|
|
148
|
+
await reify()
|
|
149
|
+
return
|
|
150
|
+
}
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
log.info("dependencies in sync")
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
export async function which(pkg: string) {
|
|
157
|
+
const dir = directory(pkg)
|
|
158
|
+
const binDir = path.join(dir, "node_modules", ".bin")
|
|
159
|
+
|
|
160
|
+
const pick = async () => {
|
|
161
|
+
const files = await readdir(binDir).catch(() => [])
|
|
162
|
+
if (files.length === 0) return undefined
|
|
163
|
+
if (files.length === 1) return files[0]
|
|
164
|
+
// Multiple binaries — resolve from package.json bin field like npx does
|
|
165
|
+
const pkgJson = await Filesystem.readJson<{ bin?: string | Record<string, string> }>(
|
|
166
|
+
path.join(dir, "node_modules", pkg, "package.json"),
|
|
167
|
+
).catch(() => undefined)
|
|
168
|
+
if (pkgJson?.bin) {
|
|
169
|
+
const unscoped = pkg.startsWith("@") ? pkg.split("/")[1] : pkg
|
|
170
|
+
const bin = pkgJson.bin
|
|
171
|
+
if (typeof bin === "string") return unscoped
|
|
172
|
+
const keys = Object.keys(bin)
|
|
173
|
+
if (keys.length === 1) return keys[0]
|
|
174
|
+
return bin[unscoped] ? unscoped : keys[0]
|
|
175
|
+
}
|
|
176
|
+
return files[0]
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
const bin = await pick()
|
|
180
|
+
if (bin) return path.join(binDir, bin)
|
|
181
|
+
|
|
182
|
+
await rm(path.join(dir, "package-lock.json"), { force: true })
|
|
183
|
+
await add(pkg)
|
|
184
|
+
const resolved = await pick()
|
|
185
|
+
if (!resolved) return
|
|
186
|
+
return path.join(binDir, resolved)
|
|
187
|
+
}
|
|
188
|
+
}
|