koro-ai 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/AGENTS.md +69 -0
- package/BUN_SHELL_MIGRATION_PLAN.md +136 -0
- package/Dockerfile +18 -0
- package/README.md +15 -0
- package/bin/koro +16 -0
- package/bin/opencode +179 -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/package.json +163 -0
- package/parsers-config.ts +290 -0
- package/script/build-node.ts +56 -0
- package/script/build.ts +276 -0
- package/script/check-migrations.ts +16 -0
- package/script/postinstall.mjs +131 -0
- package/script/publish.ts +181 -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 +294 -0
- package/specs/tui-plugins.md +436 -0
- package/src/account/account.sql.ts +39 -0
- package/src/account/index.ts +424 -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 +1763 -0
- package/src/acp/session.ts +116 -0
- package/src/acp/types.ts +24 -0
- package/src/agent/agent.ts +476 -0
- package/src/agent/generate.txt +75 -0
- package/src/agent/prompt/compaction.txt +15 -0
- package/src/agent/prompt/explore.txt +18 -0
- package/src/agent/prompt/summary.txt +11 -0
- package/src/agent/prompt/title.txt +44 -0
- package/src/auth/index.ts +109 -0
- package/src/bus/bus-event.ts +40 -0
- package/src/bus/global.ts +10 -0
- package/src/bus/index.ts +185 -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 +1646 -0
- package/src/cli/cmd/import.ts +207 -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 +676 -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 +919 -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-import-share.tsx +118 -0
- package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
- package/src/cli/cmd/tui/component/dialog-model.tsx +179 -0
- package/src/cli/cmd/tui/component/dialog-provider.tsx +329 -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 +108 -0
- package/src/cli/cmd/tui/component/prompt/index.tsx +1310 -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/task-panel.tsx +44 -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/token-bar.tsx +60 -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 +412 -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 +128 -0
- package/src/cli/cmd/tui/context/sync.tsx +504 -0
- package/src/cli/cmd/tui/context/theme/aura.json +69 -0
- package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
- package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
- package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
- package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
- package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
- package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
- package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
- package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
- package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
- package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
- package/src/cli/cmd/tui/context/theme/github.json +233 -0
- package/src/cli/cmd/tui/context/theme/gruvbox.json +242 -0
- package/src/cli/cmd/tui/context/theme/gtr.json +245 -0
- package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
- package/src/cli/cmd/tui/context/theme/koro.json +241 -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/opencode.json +245 -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 +1240 -0
- package/src/cli/cmd/tui/context/tui-config.tsx +9 -0
- package/src/cli/cmd/tui/event.ts +49 -0
- package/src/cli/cmd/tui/feature-plugins/home/footer.tsx +93 -0
- package/src/cli/cmd/tui/feature-plugins/home/tips-view.tsx +152 -0
- package/src/cli/cmd/tui/feature-plugins/home/tips.tsx +50 -0
- package/src/cli/cmd/tui/feature-plugins/sidebar/context.tsx +63 -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 +430 -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 +1033 -0
- package/src/cli/cmd/tui/plugin/slots.tsx +60 -0
- package/src/cli/cmd/tui/routes/home.tsx +84 -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 +93 -0
- package/src/cli/cmd/tui/routes/session/index.tsx +2270 -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 +74 -0
- package/src/cli/cmd/tui/routes/session/subagent-footer.tsx +131 -0
- package/src/cli/cmd/tui/thread.ts +232 -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 +130 -0
- package/src/cli/cmd/tui/ui/dialog-select.tsx +409 -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/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 +175 -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 +14 -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 +195 -0
- package/src/command/template/initialize.txt +66 -0
- package/src/command/template/review.txt +101 -0
- package/src/config/config.ts +1591 -0
- package/src/config/markdown.ts +99 -0
- package/src/config/paths.ts +181 -0
- package/src/config/tui-migrate.ts +155 -0
- package/src/config/tui-schema.ts +36 -0
- package/src/config/tui.ts +171 -0
- package/src/control-plane/adaptors/index.ts +20 -0
- package/src/control-plane/adaptors/worktree.ts +38 -0
- package/src/control-plane/schema.ts +17 -0
- package/src/control-plane/sse.ts +66 -0
- package/src/control-plane/types.ts +21 -0
- package/src/control-plane/workspace.sql.ts +17 -0
- package/src/control-plane/workspace.ts +154 -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 +171 -0
- package/src/filesystem/index.ts +226 -0
- package/src/flag/flag.ts +155 -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 +161 -0
- package/src/id/id.ts +85 -0
- package/src/ide/index.ts +74 -0
- package/src/index.ts +240 -0
- package/src/installation/index.ts +355 -0
- package/src/installation/meta.ts +7 -0
- package/src/lsp/client.ts +252 -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 +1958 -0
- package/src/mcp/auth.ts +173 -0
- package/src/mcp/index.ts +921 -0
- package/src/mcp/oauth-callback.ts +215 -0
- package/src/mcp/oauth-provider.ts +185 -0
- package/src/memory/index.ts +117 -0
- package/src/node.ts +1 -0
- package/src/npm/index.ts +180 -0
- package/src/orchestrator/agent-registry.ts +38 -0
- package/src/orchestrator/conflict.ts +25 -0
- package/src/orchestrator/context-manager.ts +22 -0
- package/src/orchestrator/index.ts +9 -0
- package/src/orchestrator/scheduler.ts +30 -0
- package/src/orchestrator/state-tracker.ts +71 -0
- package/src/orchestrator/task-manager.ts +69 -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 +325 -0
- package/src/permission/schema.ts +17 -0
- package/src/plugin/codex.ts +596 -0
- package/src/plugin/github-copilot/copilot.ts +353 -0
- package/src/plugin/github-copilot/models.ts +144 -0
- package/src/plugin/index.ts +281 -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 +307 -0
- package/src/project/bootstrap.ts +31 -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 +197 -0
- package/src/provider/models-snapshot.ts +60410 -0
- package/src/provider/models.ts +162 -0
- package/src/provider/provider.ts +1677 -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 +815 -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 +1046 -0
- package/src/pty/index.ts +401 -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 +314 -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/router.ts +99 -0
- package/src/server/routes/config.ts +92 -0
- package/src/server/routes/event.ts +83 -0
- package/src/server/routes/experimental.ts +271 -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 +211 -0
- package/src/server/routes/question.ts +99 -0
- package/src/server/routes/session.ts +1031 -0
- package/src/server/routes/tui.ts +379 -0
- package/src/server/routes/workspace.ts +94 -0
- package/src/server/server.ts +312 -0
- package/src/session/compaction.ts +428 -0
- package/src/session/index.ts +887 -0
- package/src/session/instruction.ts +258 -0
- package/src/session/llm.ts +370 -0
- package/src/session/message-v2.ts +1031 -0
- package/src/session/message.ts +191 -0
- package/src/session/overflow.ts +22 -0
- package/src/session/processor.ts +523 -0
- package/src/session/projectors.ts +135 -0
- package/src/session/prompt/anthropic.txt +105 -0
- package/src/session/prompt/beast.txt +147 -0
- package/src/session/prompt/build-switch.txt +5 -0
- package/src/session/prompt/codex.txt +79 -0
- package/src/session/prompt/copilot-gpt-5.txt +143 -0
- package/src/session/prompt/default.txt +105 -0
- package/src/session/prompt/gemini.txt +155 -0
- package/src/session/prompt/gpt.txt +107 -0
- package/src/session/prompt/kimi.txt +114 -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/trinity.txt +97 -0
- package/src/session/prompt.ts +1908 -0
- package/src/session/retry.ts +106 -0
- package/src/session/revert.ts +176 -0
- package/src/session/schema.ts +38 -0
- package/src/session/session.sql.ts +103 -0
- package/src/session/status.ts +102 -0
- package/src/session/summary.ts +177 -0
- package/src/session/system.ts +76 -0
- package/src/session/todo.ts +95 -0
- package/src/share/share-next.ts +369 -0
- package/src/share/share.sql.ts +13 -0
- package/src/shell/shell.ts +110 -0
- package/src/skill/discovery.ts +116 -0
- package/src/skill/index.ts +277 -0
- package/src/snapshot/index.ts +571 -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 +425 -0
- package/src/storage/schema.sql.ts +10 -0
- package/src/storage/schema.ts +5 -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/token/index.ts +77 -0
- package/src/tool/apply_patch.ts +281 -0
- package/src/tool/apply_patch.txt +33 -0
- package/src/tool/bash.ts +496 -0
- package/src/tool/bash.txt +117 -0
- package/src/tool/batch.ts +183 -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 +667 -0
- package/src/tool/edit.txt +10 -0
- package/src/tool/external-directory.ts +37 -0
- package/src/tool/glob.ts +78 -0
- package/src/tool/glob.txt +6 -0
- package/src/tool/grep.ts +156 -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 +97 -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 +131 -0
- package/src/tool/question.ts +46 -0
- package/src/tool/question.txt +10 -0
- package/src/tool/read.ts +296 -0
- package/src/tool/read.txt +14 -0
- package/src/tool/registry.ts +248 -0
- package/src/tool/schema.ts +17 -0
- package/src/tool/skill.ts +105 -0
- package/src/tool/task.ts +166 -0
- package/src/tool/task.txt +60 -0
- package/src/tool/todo.ts +48 -0
- package/src/tool/todowrite.txt +167 -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 +13 -0
- package/src/tool/websearch.ts +150 -0
- package/src/tool/websearch.txt +14 -0
- package/src/tool/write.ts +84 -0
- package/src/tool/write.txt +8 -0
- package/src/util/abort.ts +35 -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.ts +182 -0
- package/src/util/network.ts +9 -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/signal.ts +12 -0
- package/src/util/timeout.ts +14 -0
- package/src/util/token.ts +7 -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 +717 -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/import.test.ts +54 -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 +2348 -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 +752 -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 +81 -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 +337 -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 +151 -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 +795 -0
- package/test/lsp/client.test.ts +95 -0
- package/test/lsp/index.test.ts +55 -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/memory/abort-leak.test.ts +137 -0
- package/test/patch/patch.test.ts +348 -0
- package/test/permission/arity.test.ts +33 -0
- package/test/permission/next.test.ts +1148 -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/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/gitlab-duo.test.ts +412 -0
- package/test/provider/provider.test.ts +2284 -0
- package/test/provider/transform.test.ts +2839 -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 +1212 -0
- package/test/session/instruction.test.ts +286 -0
- package/test/session/llm.test.ts +1098 -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 +747 -0
- package/test/session/prompt-effect.test.ts +1241 -0
- package/test/session/prompt.test.ts +518 -0
- package/test/session/retry.test.ts +232 -0
- package/test/session/revert-compact.test.ts +621 -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/share/share-next.test.ts +333 -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 +1312 -0
- package/test/storage/db.test.ts +14 -0
- package/test/storage/json-migration.test.ts +849 -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 +546 -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 +353 -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/module.test.ts +59 -0
- package/test/util/process.test.ts +128 -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/tsconfig.json +23 -0
|
@@ -0,0 +1,174 @@
|
|
|
1
|
+
import { Config } from "@/config/config"
|
|
2
|
+
import { Installation } from "@/installation"
|
|
3
|
+
import {
|
|
4
|
+
checkPluginCompatibility,
|
|
5
|
+
createPluginEntry,
|
|
6
|
+
isDeprecatedPlugin,
|
|
7
|
+
pluginSource,
|
|
8
|
+
resolvePluginTarget,
|
|
9
|
+
type PluginKind,
|
|
10
|
+
type PluginPackage,
|
|
11
|
+
type PluginSource,
|
|
12
|
+
} from "./shared"
|
|
13
|
+
|
|
14
|
+
export namespace PluginLoader {
|
|
15
|
+
export type Plan = {
|
|
16
|
+
spec: string
|
|
17
|
+
options: Config.PluginOptions | undefined
|
|
18
|
+
deprecated: boolean
|
|
19
|
+
}
|
|
20
|
+
export type Resolved = Plan & {
|
|
21
|
+
source: PluginSource
|
|
22
|
+
target: string
|
|
23
|
+
entry: string
|
|
24
|
+
pkg?: PluginPackage
|
|
25
|
+
}
|
|
26
|
+
export type Missing = Plan & {
|
|
27
|
+
source: PluginSource
|
|
28
|
+
target: string
|
|
29
|
+
pkg?: PluginPackage
|
|
30
|
+
message: string
|
|
31
|
+
}
|
|
32
|
+
export type Loaded = Resolved & {
|
|
33
|
+
mod: Record<string, unknown>
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
type Candidate = { origin: Config.PluginOrigin; plan: Plan }
|
|
37
|
+
type Report = {
|
|
38
|
+
start?: (candidate: Candidate, retry: boolean) => void
|
|
39
|
+
missing?: (candidate: Candidate, retry: boolean, message: string, resolved: Missing) => void
|
|
40
|
+
error?: (
|
|
41
|
+
candidate: Candidate,
|
|
42
|
+
retry: boolean,
|
|
43
|
+
stage: "install" | "entry" | "compatibility" | "load",
|
|
44
|
+
error: unknown,
|
|
45
|
+
resolved?: Resolved,
|
|
46
|
+
) => void
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
function plan(item: Config.PluginSpec): Plan {
|
|
50
|
+
const spec = Config.pluginSpecifier(item)
|
|
51
|
+
return { spec, options: Config.pluginOptions(item), deprecated: isDeprecatedPlugin(spec) }
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
export async function resolve(
|
|
55
|
+
plan: Plan,
|
|
56
|
+
kind: PluginKind,
|
|
57
|
+
): Promise<
|
|
58
|
+
| { ok: true; value: Resolved }
|
|
59
|
+
| { ok: false; stage: "missing"; value: Missing }
|
|
60
|
+
| { ok: false; stage: "install" | "entry" | "compatibility"; error: unknown }
|
|
61
|
+
> {
|
|
62
|
+
let target = ""
|
|
63
|
+
try {
|
|
64
|
+
target = await resolvePluginTarget(plan.spec)
|
|
65
|
+
} catch (error) {
|
|
66
|
+
return { ok: false, stage: "install", error }
|
|
67
|
+
}
|
|
68
|
+
if (!target) return { ok: false, stage: "install", error: new Error(`Plugin ${plan.spec} target is empty`) }
|
|
69
|
+
|
|
70
|
+
let base
|
|
71
|
+
try {
|
|
72
|
+
base = await createPluginEntry(plan.spec, target, kind)
|
|
73
|
+
} catch (error) {
|
|
74
|
+
return { ok: false, stage: "entry", error }
|
|
75
|
+
}
|
|
76
|
+
if (!base.entry)
|
|
77
|
+
return {
|
|
78
|
+
ok: false,
|
|
79
|
+
stage: "missing",
|
|
80
|
+
value: {
|
|
81
|
+
...plan,
|
|
82
|
+
source: base.source,
|
|
83
|
+
target: base.target,
|
|
84
|
+
pkg: base.pkg,
|
|
85
|
+
message: `Plugin ${plan.spec} does not expose a ${kind} entrypoint`,
|
|
86
|
+
},
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
if (base.source === "npm") {
|
|
90
|
+
try {
|
|
91
|
+
await checkPluginCompatibility(base.target, Installation.VERSION, base.pkg)
|
|
92
|
+
} catch (error) {
|
|
93
|
+
return { ok: false, stage: "compatibility", error }
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
return { ok: true, value: { ...plan, source: base.source, target: base.target, entry: base.entry, pkg: base.pkg } }
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
export async function load(row: Resolved): Promise<{ ok: true; value: Loaded } | { ok: false; error: unknown }> {
|
|
100
|
+
let mod
|
|
101
|
+
try {
|
|
102
|
+
mod = await import(row.entry)
|
|
103
|
+
} catch (error) {
|
|
104
|
+
return { ok: false, error }
|
|
105
|
+
}
|
|
106
|
+
if (!mod) return { ok: false, error: new Error(`Plugin ${row.spec} module is empty`) }
|
|
107
|
+
return { ok: true, value: { ...row, mod } }
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
async function attempt<R>(
|
|
111
|
+
candidate: Candidate,
|
|
112
|
+
kind: PluginKind,
|
|
113
|
+
retry: boolean,
|
|
114
|
+
finish: ((load: Loaded, origin: Config.PluginOrigin, retry: boolean) => Promise<R | undefined>) | undefined,
|
|
115
|
+
missing: ((value: Missing, origin: Config.PluginOrigin, retry: boolean) => Promise<R | undefined>) | undefined,
|
|
116
|
+
report: Report | undefined,
|
|
117
|
+
): Promise<R | undefined> {
|
|
118
|
+
const plan = candidate.plan
|
|
119
|
+
if (plan.deprecated) return
|
|
120
|
+
report?.start?.(candidate, retry)
|
|
121
|
+
const resolved = await resolve(plan, kind)
|
|
122
|
+
if (!resolved.ok) {
|
|
123
|
+
if (resolved.stage === "missing") {
|
|
124
|
+
if (missing) {
|
|
125
|
+
const value = await missing(resolved.value, candidate.origin, retry)
|
|
126
|
+
if (value !== undefined) return value
|
|
127
|
+
}
|
|
128
|
+
report?.missing?.(candidate, retry, resolved.value.message, resolved.value)
|
|
129
|
+
return
|
|
130
|
+
}
|
|
131
|
+
report?.error?.(candidate, retry, resolved.stage, resolved.error)
|
|
132
|
+
return
|
|
133
|
+
}
|
|
134
|
+
const loaded = await load(resolved.value)
|
|
135
|
+
if (!loaded.ok) {
|
|
136
|
+
report?.error?.(candidate, retry, "load", loaded.error, resolved.value)
|
|
137
|
+
return
|
|
138
|
+
}
|
|
139
|
+
if (!finish) return loaded.value as R
|
|
140
|
+
return finish(loaded.value, candidate.origin, retry)
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
type Input<R> = {
|
|
144
|
+
items: Config.PluginOrigin[]
|
|
145
|
+
kind: PluginKind
|
|
146
|
+
wait?: () => Promise<void>
|
|
147
|
+
finish?: (load: Loaded, origin: Config.PluginOrigin, retry: boolean) => Promise<R | undefined>
|
|
148
|
+
missing?: (value: Missing, origin: Config.PluginOrigin, retry: boolean) => Promise<R | undefined>
|
|
149
|
+
report?: Report
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
export async function loadExternal<R = Loaded>(input: Input<R>): Promise<R[]> {
|
|
153
|
+
const candidates = input.items.map((origin) => ({ origin, plan: plan(origin.spec) }))
|
|
154
|
+
const list: Array<Promise<R | undefined>> = []
|
|
155
|
+
for (const candidate of candidates) {
|
|
156
|
+
list.push(attempt(candidate, input.kind, false, input.finish, input.missing, input.report))
|
|
157
|
+
}
|
|
158
|
+
const out = await Promise.all(list)
|
|
159
|
+
if (input.wait) {
|
|
160
|
+
let deps: Promise<void> | undefined
|
|
161
|
+
for (let i = 0; i < candidates.length; i++) {
|
|
162
|
+
if (out[i] !== undefined) continue
|
|
163
|
+
const candidate = candidates[i]
|
|
164
|
+
if (!candidate || pluginSource(candidate.plan.spec) !== "file") continue
|
|
165
|
+
deps ??= input.wait()
|
|
166
|
+
await deps
|
|
167
|
+
out[i] = await attempt(candidate, input.kind, true, input.finish, input.missing, input.report)
|
|
168
|
+
}
|
|
169
|
+
}
|
|
170
|
+
const ready: R[] = []
|
|
171
|
+
for (const item of out) if (item !== undefined) ready.push(item)
|
|
172
|
+
return ready
|
|
173
|
+
}
|
|
174
|
+
}
|
|
@@ -0,0 +1,188 @@
|
|
|
1
|
+
import path from "path"
|
|
2
|
+
import { fileURLToPath } from "url"
|
|
3
|
+
|
|
4
|
+
import { Flag } from "@/flag/flag"
|
|
5
|
+
import { Global } from "@/global"
|
|
6
|
+
import { Filesystem } from "@/util/filesystem"
|
|
7
|
+
import { Flock } from "@/util/flock"
|
|
8
|
+
|
|
9
|
+
import { parsePluginSpecifier, pluginSource } from "./shared"
|
|
10
|
+
|
|
11
|
+
export namespace PluginMeta {
|
|
12
|
+
type Source = "file" | "npm"
|
|
13
|
+
|
|
14
|
+
export type Theme = {
|
|
15
|
+
src: string
|
|
16
|
+
dest: string
|
|
17
|
+
mtime?: number
|
|
18
|
+
size?: number
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
export type Entry = {
|
|
22
|
+
id: string
|
|
23
|
+
source: Source
|
|
24
|
+
spec: string
|
|
25
|
+
target: string
|
|
26
|
+
requested?: string
|
|
27
|
+
version?: string
|
|
28
|
+
modified?: number
|
|
29
|
+
first_time: number
|
|
30
|
+
last_time: number
|
|
31
|
+
time_changed: number
|
|
32
|
+
load_count: number
|
|
33
|
+
fingerprint: string
|
|
34
|
+
themes?: Record<string, Theme>
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
export type State = "first" | "updated" | "same"
|
|
38
|
+
|
|
39
|
+
export type Touch = {
|
|
40
|
+
spec: string
|
|
41
|
+
target: string
|
|
42
|
+
id: string
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
type Store = Record<string, Entry>
|
|
46
|
+
type Core = Omit<Entry, "first_time" | "last_time" | "time_changed" | "load_count" | "fingerprint" | "themes">
|
|
47
|
+
type Row = Touch & { core: Core }
|
|
48
|
+
|
|
49
|
+
function storePath() {
|
|
50
|
+
return Flag.OPENCODE_PLUGIN_META_FILE ?? path.join(Global.Path.state, "plugin-meta.json")
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
function lock(file: string) {
|
|
54
|
+
return `plugin-meta:${file}`
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
function fileTarget(spec: string, target: string) {
|
|
58
|
+
if (spec.startsWith("file://")) return fileURLToPath(spec)
|
|
59
|
+
if (target.startsWith("file://")) return fileURLToPath(target)
|
|
60
|
+
return
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
async function modifiedAt(file: string) {
|
|
64
|
+
const stat = await Filesystem.statAsync(file)
|
|
65
|
+
if (!stat) return
|
|
66
|
+
const mtime = stat.mtimeMs
|
|
67
|
+
return Math.floor(typeof mtime === "bigint" ? Number(mtime) : mtime)
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
function resolvedTarget(target: string) {
|
|
71
|
+
if (target.startsWith("file://")) return fileURLToPath(target)
|
|
72
|
+
return target
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
async function npmVersion(target: string) {
|
|
76
|
+
const resolved = resolvedTarget(target)
|
|
77
|
+
const stat = await Filesystem.statAsync(resolved)
|
|
78
|
+
const dir = stat?.isDirectory() ? resolved : path.dirname(resolved)
|
|
79
|
+
return Filesystem.readJson<{ version?: string }>(path.join(dir, "package.json"))
|
|
80
|
+
.then((item) => item.version)
|
|
81
|
+
.catch(() => undefined)
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
async function entryCore(item: Touch): Promise<Core> {
|
|
85
|
+
const spec = item.spec
|
|
86
|
+
const target = item.target
|
|
87
|
+
const source = pluginSource(spec)
|
|
88
|
+
if (source === "file") {
|
|
89
|
+
const file = fileTarget(spec, target)
|
|
90
|
+
return {
|
|
91
|
+
id: item.id,
|
|
92
|
+
source,
|
|
93
|
+
spec,
|
|
94
|
+
target,
|
|
95
|
+
modified: file ? await modifiedAt(file) : undefined,
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
return {
|
|
100
|
+
id: item.id,
|
|
101
|
+
source,
|
|
102
|
+
spec,
|
|
103
|
+
target,
|
|
104
|
+
requested: parsePluginSpecifier(spec).version,
|
|
105
|
+
version: await npmVersion(target),
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
function fingerprint(value: Core) {
|
|
110
|
+
if (value.source === "file") return [value.target, value.modified ?? ""].join("|")
|
|
111
|
+
return [value.target, value.requested ?? "", value.version ?? ""].join("|")
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
async function read(file: string): Promise<Store> {
|
|
115
|
+
return Filesystem.readJson<Store>(file).catch(() => ({}) as Store)
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
async function row(item: Touch): Promise<Row> {
|
|
119
|
+
return {
|
|
120
|
+
...item,
|
|
121
|
+
core: await entryCore(item),
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
function next(prev: Entry | undefined, core: Core, now: number): { state: State; entry: Entry } {
|
|
126
|
+
const entry: Entry = {
|
|
127
|
+
...core,
|
|
128
|
+
first_time: prev?.first_time ?? now,
|
|
129
|
+
last_time: now,
|
|
130
|
+
time_changed: prev?.time_changed ?? now,
|
|
131
|
+
load_count: (prev?.load_count ?? 0) + 1,
|
|
132
|
+
fingerprint: fingerprint(core),
|
|
133
|
+
themes: prev?.themes,
|
|
134
|
+
}
|
|
135
|
+
const state: State = !prev ? "first" : prev.fingerprint === entry.fingerprint ? "same" : "updated"
|
|
136
|
+
if (state === "updated") entry.time_changed = now
|
|
137
|
+
return {
|
|
138
|
+
state,
|
|
139
|
+
entry,
|
|
140
|
+
}
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
export async function touchMany(items: Touch[]): Promise<Array<{ state: State; entry: Entry }>> {
|
|
144
|
+
if (!items.length) return []
|
|
145
|
+
const file = storePath()
|
|
146
|
+
const rows = await Promise.all(items.map((item) => row(item)))
|
|
147
|
+
|
|
148
|
+
return Flock.withLock(lock(file), async () => {
|
|
149
|
+
const store = await read(file)
|
|
150
|
+
const now = Date.now()
|
|
151
|
+
const out: Array<{ state: State; entry: Entry }> = []
|
|
152
|
+
for (const item of rows) {
|
|
153
|
+
const hit = next(store[item.id], item.core, now)
|
|
154
|
+
store[item.id] = hit.entry
|
|
155
|
+
out.push(hit)
|
|
156
|
+
}
|
|
157
|
+
await Filesystem.writeJson(file, store)
|
|
158
|
+
return out
|
|
159
|
+
})
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
export async function touch(spec: string, target: string, id: string): Promise<{ state: State; entry: Entry }> {
|
|
163
|
+
return touchMany([{ spec, target, id }]).then((item) => {
|
|
164
|
+
const hit = item[0]
|
|
165
|
+
if (hit) return hit
|
|
166
|
+
throw new Error("Failed to touch plugin metadata.")
|
|
167
|
+
})
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
export async function setTheme(id: string, name: string, theme: Theme): Promise<void> {
|
|
171
|
+
const file = storePath()
|
|
172
|
+
await Flock.withLock(lock(file), async () => {
|
|
173
|
+
const store = await read(file)
|
|
174
|
+
const entry = store[id]
|
|
175
|
+
if (!entry) return
|
|
176
|
+
entry.themes = {
|
|
177
|
+
...(entry.themes ?? {}),
|
|
178
|
+
[name]: theme,
|
|
179
|
+
}
|
|
180
|
+
await Filesystem.writeJson(file, store)
|
|
181
|
+
})
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
export async function list(): Promise<Store> {
|
|
185
|
+
const file = storePath()
|
|
186
|
+
return Flock.withLock(lock(file), async () => read(file))
|
|
187
|
+
}
|
|
188
|
+
}
|
|
@@ -0,0 +1,307 @@
|
|
|
1
|
+
import path from "path"
|
|
2
|
+
import { fileURLToPath, pathToFileURL } from "url"
|
|
3
|
+
import semver from "semver"
|
|
4
|
+
import { Npm } from "@/npm"
|
|
5
|
+
import { Filesystem } from "@/util/filesystem"
|
|
6
|
+
import { isRecord } from "@/util/record"
|
|
7
|
+
|
|
8
|
+
// Old npm package names for plugins that are now built-in
|
|
9
|
+
export const DEPRECATED_PLUGIN_PACKAGES = ["opencode-openai-codex-auth", "opencode-copilot-auth"]
|
|
10
|
+
|
|
11
|
+
export function isDeprecatedPlugin(spec: string) {
|
|
12
|
+
return DEPRECATED_PLUGIN_PACKAGES.some((pkg) => spec.includes(pkg))
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
export function parsePluginSpecifier(spec: string) {
|
|
16
|
+
const lastAt = spec.lastIndexOf("@")
|
|
17
|
+
const pkg = lastAt > 0 ? spec.substring(0, lastAt) : spec
|
|
18
|
+
const version = lastAt > 0 ? spec.substring(lastAt + 1) : "latest"
|
|
19
|
+
return { pkg, version }
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
export type PluginSource = "file" | "npm"
|
|
23
|
+
export type PluginKind = "server" | "tui"
|
|
24
|
+
type PluginMode = "strict" | "detect"
|
|
25
|
+
|
|
26
|
+
export type PluginPackage = {
|
|
27
|
+
dir: string
|
|
28
|
+
pkg: string
|
|
29
|
+
json: Record<string, unknown>
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
export type PluginEntry = {
|
|
33
|
+
spec: string
|
|
34
|
+
source: PluginSource
|
|
35
|
+
target: string
|
|
36
|
+
pkg?: PluginPackage
|
|
37
|
+
entry?: string
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
const INDEX_FILES = ["index.ts", "index.tsx", "index.js", "index.mjs", "index.cjs"]
|
|
41
|
+
|
|
42
|
+
export function pluginSource(spec: string): PluginSource {
|
|
43
|
+
if (isPathPluginSpec(spec)) return "file"
|
|
44
|
+
return "npm"
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
function resolveExportPath(raw: string, dir: string) {
|
|
48
|
+
if (raw.startsWith("file://")) return fileURLToPath(raw)
|
|
49
|
+
if (path.isAbsolute(raw)) return raw
|
|
50
|
+
return path.resolve(dir, raw)
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
function isAbsolutePath(raw: string) {
|
|
54
|
+
return path.isAbsolute(raw) || /^[A-Za-z]:[\\/]/.test(raw)
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
function extractExportValue(value: unknown): string | undefined {
|
|
58
|
+
if (typeof value === "string") return value
|
|
59
|
+
if (!isRecord(value)) return undefined
|
|
60
|
+
for (const key of ["import", "default"]) {
|
|
61
|
+
const nested = value[key]
|
|
62
|
+
if (typeof nested === "string") return nested
|
|
63
|
+
}
|
|
64
|
+
return undefined
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
function packageMain(pkg: PluginPackage) {
|
|
68
|
+
const value = pkg.json.main
|
|
69
|
+
if (typeof value !== "string") return
|
|
70
|
+
const next = value.trim()
|
|
71
|
+
if (!next) return
|
|
72
|
+
return next
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
function resolvePackageFile(spec: string, raw: string, kind: string, pkg: PluginPackage) {
|
|
76
|
+
const resolved = resolveExportPath(raw, pkg.dir)
|
|
77
|
+
const root = Filesystem.resolve(pkg.dir)
|
|
78
|
+
const next = Filesystem.resolve(resolved)
|
|
79
|
+
if (!Filesystem.contains(root, next)) {
|
|
80
|
+
throw new Error(`Plugin ${spec} resolved ${kind} entry outside plugin directory`)
|
|
81
|
+
}
|
|
82
|
+
return next
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
function resolvePackagePath(spec: string, raw: string, kind: PluginKind, pkg: PluginPackage) {
|
|
86
|
+
return pathToFileURL(resolvePackageFile(spec, raw, kind, pkg)).href
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
function resolvePackageEntrypoint(spec: string, kind: PluginKind, pkg: PluginPackage) {
|
|
90
|
+
const exports = pkg.json.exports
|
|
91
|
+
if (isRecord(exports)) {
|
|
92
|
+
const raw = extractExportValue(exports[`./${kind}`])
|
|
93
|
+
if (raw) return resolvePackagePath(spec, raw, kind, pkg)
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
if (kind !== "server") return
|
|
97
|
+
const main = packageMain(pkg)
|
|
98
|
+
if (!main) return
|
|
99
|
+
return resolvePackagePath(spec, main, kind, pkg)
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
function targetPath(target: string) {
|
|
103
|
+
if (target.startsWith("file://")) return fileURLToPath(target)
|
|
104
|
+
if (path.isAbsolute(target)) return target
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
async function resolveDirectoryIndex(dir: string) {
|
|
108
|
+
for (const name of INDEX_FILES) {
|
|
109
|
+
const file = path.join(dir, name)
|
|
110
|
+
if (await Filesystem.exists(file)) return file
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
async function resolveTargetDirectory(target: string) {
|
|
115
|
+
const file = targetPath(target)
|
|
116
|
+
if (!file) return
|
|
117
|
+
const stat = await Filesystem.statAsync(file)
|
|
118
|
+
if (!stat?.isDirectory()) return
|
|
119
|
+
return file
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
async function resolvePluginEntrypoint(spec: string, target: string, kind: PluginKind, pkg?: PluginPackage) {
|
|
123
|
+
const source = pluginSource(spec)
|
|
124
|
+
const hit =
|
|
125
|
+
pkg ?? (source === "npm" ? await readPluginPackage(target) : await readPluginPackage(target).catch(() => undefined))
|
|
126
|
+
if (!hit) return target
|
|
127
|
+
|
|
128
|
+
const entry = resolvePackageEntrypoint(spec, kind, hit)
|
|
129
|
+
if (entry) return entry
|
|
130
|
+
|
|
131
|
+
const dir = await resolveTargetDirectory(target)
|
|
132
|
+
|
|
133
|
+
if (kind === "tui") {
|
|
134
|
+
if (source === "file" && dir) {
|
|
135
|
+
const index = await resolveDirectoryIndex(dir)
|
|
136
|
+
if (index) return pathToFileURL(index).href
|
|
137
|
+
}
|
|
138
|
+
|
|
139
|
+
if (source === "npm") return
|
|
140
|
+
if (dir) return
|
|
141
|
+
|
|
142
|
+
return target
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
if (dir && isRecord(hit.json.exports)) {
|
|
146
|
+
if (source === "file") {
|
|
147
|
+
const index = await resolveDirectoryIndex(dir)
|
|
148
|
+
if (index) return pathToFileURL(index).href
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
return
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
return target
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
export function isPathPluginSpec(spec: string) {
|
|
158
|
+
return spec.startsWith("file://") || spec.startsWith(".") || isAbsolutePath(spec)
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
export async function resolvePathPluginTarget(spec: string) {
|
|
162
|
+
const raw = spec.startsWith("file://") ? fileURLToPath(spec) : spec
|
|
163
|
+
const file = path.isAbsolute(raw) || /^[A-Za-z]:[\\/]/.test(raw) ? raw : path.resolve(raw)
|
|
164
|
+
const stat = await Filesystem.statAsync(file)
|
|
165
|
+
if (!stat?.isDirectory()) {
|
|
166
|
+
if (spec.startsWith("file://")) return spec
|
|
167
|
+
return pathToFileURL(file).href
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
if (await Filesystem.exists(path.join(file, "package.json"))) {
|
|
171
|
+
return pathToFileURL(file).href
|
|
172
|
+
}
|
|
173
|
+
|
|
174
|
+
const index = await resolveDirectoryIndex(file)
|
|
175
|
+
if (index) return pathToFileURL(index).href
|
|
176
|
+
|
|
177
|
+
throw new Error(`Plugin directory ${file} is missing package.json or index file`)
|
|
178
|
+
}
|
|
179
|
+
|
|
180
|
+
export async function checkPluginCompatibility(target: string, opencodeVersion: string, pkg?: PluginPackage) {
|
|
181
|
+
if (!semver.valid(opencodeVersion) || semver.major(opencodeVersion) === 0) return
|
|
182
|
+
const hit = pkg ?? (await readPluginPackage(target).catch(() => undefined))
|
|
183
|
+
if (!hit) return
|
|
184
|
+
const engines = hit.json.engines
|
|
185
|
+
if (!isRecord(engines)) return
|
|
186
|
+
const range = engines.opencode
|
|
187
|
+
if (typeof range !== "string") return
|
|
188
|
+
if (!semver.satisfies(opencodeVersion, range)) {
|
|
189
|
+
throw new Error(`Plugin requires opencode ${range} but running ${opencodeVersion}`)
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
export async function resolvePluginTarget(spec: string, parsed = parsePluginSpecifier(spec)) {
|
|
194
|
+
if (isPathPluginSpec(spec)) return resolvePathPluginTarget(spec)
|
|
195
|
+
const result = await Npm.add(parsed.pkg + "@" + parsed.version)
|
|
196
|
+
return result.directory
|
|
197
|
+
}
|
|
198
|
+
|
|
199
|
+
export async function readPluginPackage(target: string): Promise<PluginPackage> {
|
|
200
|
+
const file = target.startsWith("file://") ? fileURLToPath(target) : target
|
|
201
|
+
const stat = await Filesystem.statAsync(file)
|
|
202
|
+
const dir = stat?.isDirectory() ? file : path.dirname(file)
|
|
203
|
+
const pkg = path.join(dir, "package.json")
|
|
204
|
+
const json = await Filesystem.readJson<Record<string, unknown>>(pkg)
|
|
205
|
+
return { dir, pkg, json }
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
export async function createPluginEntry(spec: string, target: string, kind: PluginKind): Promise<PluginEntry> {
|
|
209
|
+
const source = pluginSource(spec)
|
|
210
|
+
const pkg =
|
|
211
|
+
source === "npm" ? await readPluginPackage(target) : await readPluginPackage(target).catch(() => undefined)
|
|
212
|
+
const entry = await resolvePluginEntrypoint(spec, target, kind, pkg)
|
|
213
|
+
return {
|
|
214
|
+
spec,
|
|
215
|
+
source,
|
|
216
|
+
target,
|
|
217
|
+
pkg,
|
|
218
|
+
entry,
|
|
219
|
+
}
|
|
220
|
+
}
|
|
221
|
+
|
|
222
|
+
export function readPackageThemes(spec: string, pkg: PluginPackage) {
|
|
223
|
+
const field = pkg.json["oc-themes"]
|
|
224
|
+
if (field === undefined) return []
|
|
225
|
+
if (!Array.isArray(field)) {
|
|
226
|
+
throw new TypeError(`Plugin ${spec} has invalid oc-themes field`)
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
const list = field.map((item) => {
|
|
230
|
+
if (typeof item !== "string") {
|
|
231
|
+
throw new TypeError(`Plugin ${spec} has invalid oc-themes entry`)
|
|
232
|
+
}
|
|
233
|
+
|
|
234
|
+
const raw = item.trim()
|
|
235
|
+
if (!raw) {
|
|
236
|
+
throw new TypeError(`Plugin ${spec} has empty oc-themes entry`)
|
|
237
|
+
}
|
|
238
|
+
if (raw.startsWith("file://") || isAbsolutePath(raw)) {
|
|
239
|
+
throw new TypeError(`Plugin ${spec} oc-themes entry must be relative: ${item}`)
|
|
240
|
+
}
|
|
241
|
+
|
|
242
|
+
return resolvePackageFile(spec, raw, "oc-themes", pkg)
|
|
243
|
+
})
|
|
244
|
+
|
|
245
|
+
return Array.from(new Set(list))
|
|
246
|
+
}
|
|
247
|
+
|
|
248
|
+
export function readPluginId(id: unknown, spec: string) {
|
|
249
|
+
if (id === undefined) return
|
|
250
|
+
if (typeof id !== "string") throw new TypeError(`Plugin ${spec} has invalid id type ${typeof id}`)
|
|
251
|
+
const value = id.trim()
|
|
252
|
+
if (!value) throw new TypeError(`Plugin ${spec} has an empty id`)
|
|
253
|
+
return value
|
|
254
|
+
}
|
|
255
|
+
|
|
256
|
+
export function readV1Plugin(
|
|
257
|
+
mod: Record<string, unknown>,
|
|
258
|
+
spec: string,
|
|
259
|
+
kind: PluginKind,
|
|
260
|
+
mode: PluginMode = "strict",
|
|
261
|
+
) {
|
|
262
|
+
const value = mod.default
|
|
263
|
+
if (!isRecord(value)) {
|
|
264
|
+
if (mode === "detect") return
|
|
265
|
+
throw new TypeError(`Plugin ${spec} must default export an object with ${kind}()`)
|
|
266
|
+
}
|
|
267
|
+
if (mode === "detect" && !("id" in value) && !("server" in value) && !("tui" in value)) return
|
|
268
|
+
|
|
269
|
+
const server = "server" in value ? value.server : undefined
|
|
270
|
+
const tui = "tui" in value ? value.tui : undefined
|
|
271
|
+
if (server !== undefined && typeof server !== "function") {
|
|
272
|
+
throw new TypeError(`Plugin ${spec} has invalid server export`)
|
|
273
|
+
}
|
|
274
|
+
if (tui !== undefined && typeof tui !== "function") {
|
|
275
|
+
throw new TypeError(`Plugin ${spec} has invalid tui export`)
|
|
276
|
+
}
|
|
277
|
+
if (server !== undefined && tui !== undefined) {
|
|
278
|
+
throw new TypeError(`Plugin ${spec} must default export either server() or tui(), not both`)
|
|
279
|
+
}
|
|
280
|
+
if (kind === "server" && server === undefined) {
|
|
281
|
+
throw new TypeError(`Plugin ${spec} must default export an object with server()`)
|
|
282
|
+
}
|
|
283
|
+
if (kind === "tui" && tui === undefined) {
|
|
284
|
+
throw new TypeError(`Plugin ${spec} must default export an object with tui()`)
|
|
285
|
+
}
|
|
286
|
+
|
|
287
|
+
return value
|
|
288
|
+
}
|
|
289
|
+
|
|
290
|
+
export async function resolvePluginId(
|
|
291
|
+
source: PluginSource,
|
|
292
|
+
spec: string,
|
|
293
|
+
target: string,
|
|
294
|
+
id: string | undefined,
|
|
295
|
+
pkg?: PluginPackage,
|
|
296
|
+
) {
|
|
297
|
+
if (source === "file") {
|
|
298
|
+
if (id) return id
|
|
299
|
+
throw new TypeError(`Path plugin ${spec} must export id`)
|
|
300
|
+
}
|
|
301
|
+
if (id) return id
|
|
302
|
+
const hit = pkg ?? (await readPluginPackage(target))
|
|
303
|
+
if (typeof hit.json.name !== "string" || !hit.json.name.trim()) {
|
|
304
|
+
throw new TypeError(`Plugin package ${hit.pkg} is missing name`)
|
|
305
|
+
}
|
|
306
|
+
return hit.json.name.trim()
|
|
307
|
+
}
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
import { Plugin } from "../plugin"
|
|
2
|
+
import { Format } from "../format"
|
|
3
|
+
import { LSP } from "../lsp"
|
|
4
|
+
import { File } from "../file"
|
|
5
|
+
import { FileWatcher } from "../file/watcher"
|
|
6
|
+
import { Snapshot } from "../snapshot"
|
|
7
|
+
import { Project } from "./project"
|
|
8
|
+
import { Vcs } from "./vcs"
|
|
9
|
+
import { Bus } from "../bus"
|
|
10
|
+
import { Command } from "../command"
|
|
11
|
+
import { Instance } from "./instance"
|
|
12
|
+
import { Log } from "@/util/log"
|
|
13
|
+
import { ShareNext } from "@/share/share-next"
|
|
14
|
+
|
|
15
|
+
export async function InstanceBootstrap() {
|
|
16
|
+
Log.Default.info("bootstrapping", { directory: Instance.directory })
|
|
17
|
+
await Plugin.init()
|
|
18
|
+
ShareNext.init()
|
|
19
|
+
Format.init()
|
|
20
|
+
await LSP.init()
|
|
21
|
+
File.init()
|
|
22
|
+
FileWatcher.init()
|
|
23
|
+
Vcs.init()
|
|
24
|
+
Snapshot.init()
|
|
25
|
+
|
|
26
|
+
Bus.subscribe(Command.Event.Executed, async (payload) => {
|
|
27
|
+
if (payload.properties.name === Command.Default.INIT) {
|
|
28
|
+
Project.setInitialized(Instance.project.id)
|
|
29
|
+
}
|
|
30
|
+
})
|
|
31
|
+
}
|