rimuru-ai 1.19.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/.rimuru/AGENTS.md +30 -0
- package/.rimuru/agents/backend.md +27 -0
- package/.rimuru/agents/database.md +31 -0
- package/.rimuru/agents/devops.md +30 -0
- package/.rimuru/agents/document-prep.md +49 -0
- package/.rimuru/agents/erp-architect.md +41 -0
- package/.rimuru/agents/ethical-hacking.md +49 -0
- package/.rimuru/agents/frontend.md +31 -0
- package/.rimuru/agents/fullstack.md +24 -0
- package/.rimuru/agents/system-engineer.md +31 -0
- package/.rimuru/agents/veldora-agent-tool-dev.md +30 -0
- package/.rimuru/agents/veldora-backend-dev.md +32 -0
- package/.rimuru/agents/veldora-cicd.md +32 -0
- package/.rimuru/agents/veldora-database.md +32 -0
- package/.rimuru/agents/veldora-doc.md +87 -0
- package/.rimuru/agents/veldora-frontend-dev.md +32 -0
- package/.rimuru/agents/veldora-great-sage.md +33 -0
- package/.rimuru/agents/veldora-mcp-creator.md +30 -0
- package/.rimuru/agents/veldora-pro.md +224 -0
- package/.rimuru/agents/veldora-prompt-enhancer.md +27 -0
- package/.rimuru/agents/veldora-skill-creator.md +28 -0
- package/.rimuru/agents/veldora.md +225 -0
- package/.rimuru/agents/veldorapro-agent-tool-dev.md +29 -0
- package/.rimuru/agents/veldorapro-backend-dev.md +29 -0
- package/.rimuru/agents/veldorapro-cicd.md +29 -0
- package/.rimuru/agents/veldorapro-database.md +29 -0
- package/.rimuru/agents/veldorapro-frontend-dev.md +29 -0
- package/.rimuru/agents/veldorapro-great-sage.md +33 -0
- package/.rimuru/agents/veldorapro-mcp-creator.md +27 -0
- package/.rimuru/agents/veldorapro-prompt-enhancer.md +25 -0
- package/.rimuru/agents/veldorapro-skill-creator.md +27 -0
- package/.rimuru/command/ai-deps.md +24 -0
- package/.rimuru/command/changelog.md +49 -0
- package/.rimuru/command/commit.md +37 -0
- package/.rimuru/command/issues.md +23 -0
- package/.rimuru/command/learn.md +42 -0
- package/.rimuru/command/rmslop.md +15 -0
- package/.rimuru/command/spellcheck.md +5 -0
- package/.rimuru/command/translate.md +14 -0
- package/.rimuru/glossary/README.md +63 -0
- package/.rimuru/glossary/ar.md +28 -0
- package/.rimuru/glossary/br.md +34 -0
- package/.rimuru/glossary/bs.md +33 -0
- package/.rimuru/glossary/da.md +27 -0
- package/.rimuru/glossary/de.md +27 -0
- package/.rimuru/glossary/es.md +27 -0
- package/.rimuru/glossary/fr.md +27 -0
- package/.rimuru/glossary/ja.md +33 -0
- package/.rimuru/glossary/ko.md +27 -0
- package/.rimuru/glossary/no.md +38 -0
- package/.rimuru/glossary/pl.md +27 -0
- package/.rimuru/glossary/ru.md +27 -0
- package/.rimuru/glossary/th.md +34 -0
- package/.rimuru/glossary/tr.md +38 -0
- package/.rimuru/glossary/zh-cn.md +42 -0
- package/.rimuru/glossary/zh-tw.md +42 -0
- package/.rimuru/improver/changelog.md +250 -0
- package/.rimuru/improver/knowledge.md +172 -0
- package/.rimuru/improver/plugins.md +21 -0
- package/.rimuru/improver/skills.md +60 -0
- package/.rimuru/improver/token-audit.md +21 -0
- package/.rimuru/opencode.jsonc +140 -0
- package/.rimuru/plugins/smoke-theme.json +223 -0
- package/.rimuru/plugins/tui-smoke.tsx +1019 -0
- package/.rimuru/skills/effect/SKILL.md +38 -0
- package/.rimuru/themes/mytheme.json +223 -0
- package/.rimuru/tool/github-pr-search.ts +64 -0
- package/.rimuru/tool/github-triage.ts +60 -0
- package/README.md +31 -0
- package/package.json +167 -0
- package/src/account/account.ts +463 -0
- package/src/account/repo.ts +173 -0
- package/src/account/schema.ts +99 -0
- package/src/account/url.ts +8 -0
- package/src/acp/agent.ts +95 -0
- package/src/acp/config-option.ts +203 -0
- package/src/acp/content.ts +250 -0
- package/src/acp/directory.ts +210 -0
- package/src/acp/error.ts +90 -0
- package/src/acp/event.ts +342 -0
- package/src/acp/permission.ts +124 -0
- package/src/acp/profile.ts +42 -0
- package/src/acp/service.ts +1048 -0
- package/src/acp/session.ts +231 -0
- package/src/acp/tool.ts +367 -0
- package/src/acp/usage.ts +232 -0
- package/src/agent/agent.ts +459 -0
- package/src/agent/generate.txt +75 -0
- package/src/agent/prompt/compaction.txt +9 -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/agent/subagent-permissions.ts +27 -0
- package/src/audio.d.ts +14 -0
- package/src/auth/index.ts +99 -0
- package/src/background/job.ts +39 -0
- package/src/bus/global.ts +22 -0
- package/src/cli/bootstrap.ts +11 -0
- package/src/cli/cmd/account.ts +264 -0
- package/src/cli/cmd/acp.ts +73 -0
- package/src/cli/cmd/agent.ts +259 -0
- package/src/cli/cmd/attach.ts +97 -0
- package/src/cli/cmd/cmd.ts +7 -0
- package/src/cli/cmd/db.ts +62 -0
- package/src/cli/cmd/debug/agent.handler.ts +193 -0
- package/src/cli/cmd/debug/agent.ts +27 -0
- package/src/cli/cmd/debug/config.ts +14 -0
- package/src/cli/cmd/debug/file.ts +73 -0
- package/src/cli/cmd/debug/index.ts +87 -0
- package/src/cli/cmd/debug/lsp.ts +50 -0
- package/src/cli/cmd/debug/ripgrep.ts +79 -0
- package/src/cli/cmd/debug/scrap.ts +15 -0
- package/src/cli/cmd/debug/skill.ts +15 -0
- package/src/cli/cmd/debug/snapshot.ts +50 -0
- package/src/cli/cmd/debug/startup.ts +11 -0
- package/src/cli/cmd/debug/v2.ts +49 -0
- package/src/cli/cmd/export.ts +292 -0
- package/src/cli/cmd/generate.ts +54 -0
- package/src/cli/cmd/github.handler.ts +1593 -0
- package/src/cli/cmd/github.shared.ts +30 -0
- package/src/cli/cmd/github.ts +42 -0
- package/src/cli/cmd/import.ts +224 -0
- package/src/cli/cmd/mcp.ts +850 -0
- package/src/cli/cmd/models.ts +66 -0
- package/src/cli/cmd/plug.ts +230 -0
- package/src/cli/cmd/pr.ts +115 -0
- package/src/cli/cmd/prompt-display.ts +1 -0
- package/src/cli/cmd/providers.ts +534 -0
- package/src/cli/cmd/run/demo.ts +1274 -0
- package/src/cli/cmd/run/entry.body.ts +205 -0
- package/src/cli/cmd/run/footer.command.tsx +1064 -0
- package/src/cli/cmd/run/footer.menu.tsx +351 -0
- package/src/cli/cmd/run/footer.permission.tsx +472 -0
- package/src/cli/cmd/run/footer.prompt.tsx +1306 -0
- package/src/cli/cmd/run/footer.question.tsx +573 -0
- package/src/cli/cmd/run/footer.subagent.tsx +173 -0
- package/src/cli/cmd/run/footer.ts +1129 -0
- package/src/cli/cmd/run/footer.view.tsx +943 -0
- package/src/cli/cmd/run/footer.width.ts +27 -0
- package/src/cli/cmd/run/permission.shared.ts +256 -0
- package/src/cli/cmd/run/prompt.editor.ts +157 -0
- package/src/cli/cmd/run/prompt.shared.ts +153 -0
- package/src/cli/cmd/run/question.shared.ts +340 -0
- package/src/cli/cmd/run/runtime.boot.ts +202 -0
- package/src/cli/cmd/run/runtime.lifecycle.ts +406 -0
- package/src/cli/cmd/run/runtime.queue.ts +349 -0
- package/src/cli/cmd/run/runtime.shared.ts +17 -0
- package/src/cli/cmd/run/runtime.stdin.ts +37 -0
- package/src/cli/cmd/run/runtime.ts +814 -0
- package/src/cli/cmd/run/scrollback.shared.ts +92 -0
- package/src/cli/cmd/run/scrollback.surface.ts +431 -0
- package/src/cli/cmd/run/scrollback.writer.tsx +352 -0
- package/src/cli/cmd/run/session-data.ts +1113 -0
- package/src/cli/cmd/run/session-replay.ts +374 -0
- package/src/cli/cmd/run/session.shared.ts +196 -0
- package/src/cli/cmd/run/splash.ts +280 -0
- package/src/cli/cmd/run/stream.transport.ts +1462 -0
- package/src/cli/cmd/run/stream.ts +175 -0
- package/src/cli/cmd/run/subagent-data.ts +876 -0
- package/src/cli/cmd/run/theme.ts +690 -0
- package/src/cli/cmd/run/tool.ts +1489 -0
- package/src/cli/cmd/run/trace.ts +94 -0
- package/src/cli/cmd/run/turn-summary.ts +47 -0
- package/src/cli/cmd/run/types.ts +350 -0
- package/src/cli/cmd/run/variant.shared.ts +215 -0
- package/src/cli/cmd/run.ts +894 -0
- package/src/cli/cmd/serve.ts +24 -0
- package/src/cli/cmd/session.ts +147 -0
- package/src/cli/cmd/stats.ts +393 -0
- package/src/cli/cmd/tui.ts +224 -0
- package/src/cli/cmd/uninstall.ts +353 -0
- package/src/cli/cmd/upgrade.ts +74 -0
- package/src/cli/cmd/web.ts +84 -0
- package/src/cli/effect/prompt.ts +37 -0
- package/src/cli/effect-cmd.ts +96 -0
- package/src/cli/error.ts +130 -0
- package/src/cli/heap.ts +45 -0
- package/src/cli/logo.ts +1 -0
- package/src/cli/network.ts +64 -0
- package/src/cli/tui/layer.ts +7 -0
- package/src/cli/tui/validate-session.ts +29 -0
- package/src/cli/tui/worker.ts +71 -0
- package/src/cli/ui.ts +98 -0
- package/src/cli/upgrade.ts +53 -0
- package/src/command/index.ts +184 -0
- package/src/command/template/initialize.txt +66 -0
- package/src/command/template/review.txt +101 -0
- package/src/config/agent.ts +59 -0
- package/src/config/command.ts +39 -0
- package/src/config/config.ts +686 -0
- package/src/config/entry-name.ts +19 -0
- package/src/config/managed.ts +69 -0
- package/src/config/markdown.ts +36 -0
- package/src/config/parse.ts +79 -0
- package/src/config/paths.ts +45 -0
- package/src/config/plugin.ts +79 -0
- package/src/config/tui-cwd.ts +5 -0
- package/src/config/tui-host-attention.ts +21 -0
- package/src/config/tui-migrate.ts +132 -0
- package/src/config/tui.ts +274 -0
- package/src/config/variable.ts +91 -0
- package/src/control-plane/adapters/index.ts +41 -0
- package/src/control-plane/adapters/worktree.ts +96 -0
- package/src/control-plane/dev/README.md +19 -0
- package/src/control-plane/dev/debug-workspace-plugin.ts +73 -0
- package/src/control-plane/types.ts +59 -0
- package/src/control-plane/util.ts +39 -0
- package/src/control-plane/workspace-adapter-runtime.ts +51 -0
- package/src/control-plane/workspace-context.ts +26 -0
- package/src/control-plane/workspace.ts +989 -0
- package/src/effect/app-runtime.ts +132 -0
- package/src/effect/bootstrap-runtime.ts +23 -0
- package/src/effect/bridge.ts +84 -0
- package/src/effect/config-service.ts +67 -0
- package/src/effect/instance-ref.ts +11 -0
- package/src/effect/instance-registry.ts +12 -0
- package/src/effect/instance-state.ts +69 -0
- package/src/effect/promise.ts +17 -0
- package/src/effect/run-service.ts +47 -0
- package/src/effect/runner.ts +217 -0
- package/src/effect/runtime-flags.ts +79 -0
- package/src/env/index.ts +43 -0
- package/src/event-v2-bridge.ts +79 -0
- package/src/format/formatter.ts +404 -0
- package/src/format/index.ts +205 -0
- package/src/git/index.ts +350 -0
- package/src/id/id.ts +80 -0
- package/src/ide/index.ts +61 -0
- package/src/image/image.ts +174 -0
- package/src/index.ts +142 -0
- package/src/installation/index.ts +350 -0
- package/src/lsp/client.ts +650 -0
- package/src/lsp/diagnostic.ts +29 -0
- package/src/lsp/language.ts +121 -0
- package/src/lsp/launch.ts +21 -0
- package/src/lsp/lsp.ts +511 -0
- package/src/lsp/server.ts +1983 -0
- package/src/markdown.d.ts +4 -0
- package/src/mcp/auth.ts +174 -0
- package/src/mcp/catalog.ts +153 -0
- package/src/mcp/index.ts +946 -0
- package/src/mcp/oauth-callback.ts +233 -0
- package/src/mcp/oauth-provider.ts +206 -0
- package/src/node.ts +4 -0
- package/src/patch/index.ts +686 -0
- package/src/permission/arity.ts +163 -0
- package/src/permission/evaluate.ts +1 -0
- package/src/permission/index.ts +230 -0
- package/src/plugin/azure.ts +26 -0
- package/src/plugin/cloudflare.ts +76 -0
- package/src/plugin/digitalocean.ts +383 -0
- package/src/plugin/github-copilot/copilot.ts +413 -0
- package/src/plugin/github-copilot/models.ts +246 -0
- package/src/plugin/index.ts +316 -0
- package/src/plugin/install.ts +439 -0
- package/src/plugin/loader.ts +237 -0
- package/src/plugin/meta.ts +188 -0
- package/src/plugin/openai/README.md +31 -0
- package/src/plugin/openai/codex.ts +641 -0
- package/src/plugin/openai/ws-pool.ts +270 -0
- package/src/plugin/openai/ws.ts +381 -0
- package/src/plugin/pty-environment.ts +24 -0
- package/src/plugin/shared.ts +323 -0
- package/src/plugin/snowflake-cortex.ts +529 -0
- package/src/plugin/tui/internal.ts +10 -0
- package/src/plugin/tui/runtime.ts +1130 -0
- package/src/plugin/xai.ts +716 -0
- package/src/project/bootstrap-service.ts +9 -0
- package/src/project/bootstrap.ts +76 -0
- package/src/project/instance-context.ts +24 -0
- package/src/project/instance-layer.ts +11 -0
- package/src/project/instance-runtime.ts +16 -0
- package/src/project/instance-store.ts +209 -0
- package/src/project/project.ts +519 -0
- package/src/project/vcs.ts +431 -0
- package/src/provider/auth.ts +233 -0
- package/src/provider/error.ts +188 -0
- package/src/provider/model-status.ts +8 -0
- package/src/provider/provider.ts +1979 -0
- package/src/provider/transform.ts +1426 -0
- package/src/question/index.ts +229 -0
- package/src/question/schema.ts +10 -0
- package/src/server/auth.ts +48 -0
- package/src/server/event.ts +13 -0
- package/src/server/global-lifecycle.ts +28 -0
- package/src/server/init-projectors.ts +3 -0
- package/src/server/mdns.ts +47 -0
- package/src/server/projectors.ts +1 -0
- package/src/server/proxy-util.ts +48 -0
- package/src/server/routes/instance/httpapi/AGENTS.md +39 -0
- package/src/server/routes/instance/httpapi/api.ts +78 -0
- package/src/server/routes/instance/httpapi/errors.ts +193 -0
- package/src/server/routes/instance/httpapi/groups/config.ts +65 -0
- package/src/server/routes/instance/httpapi/groups/control-plane.ts +35 -0
- package/src/server/routes/instance/httpapi/groups/control.ts +76 -0
- package/src/server/routes/instance/httpapi/groups/event.ts +29 -0
- package/src/server/routes/instance/httpapi/groups/experimental.ts +260 -0
- package/src/server/routes/instance/httpapi/groups/file.ts +185 -0
- package/src/server/routes/instance/httpapi/groups/global.ts +138 -0
- package/src/server/routes/instance/httpapi/groups/instance.ts +206 -0
- package/src/server/routes/instance/httpapi/groups/mcp.ts +156 -0
- package/src/server/routes/instance/httpapi/groups/metadata.ts +18 -0
- package/src/server/routes/instance/httpapi/groups/permission.ts +61 -0
- package/src/server/routes/instance/httpapi/groups/project-copy.ts +32 -0
- package/src/server/routes/instance/httpapi/groups/project.ts +93 -0
- package/src/server/routes/instance/httpapi/groups/provider.ts +101 -0
- package/src/server/routes/instance/httpapi/groups/pty.ts +172 -0
- package/src/server/routes/instance/httpapi/groups/query.ts +12 -0
- package/src/server/routes/instance/httpapi/groups/question.ts +74 -0
- package/src/server/routes/instance/httpapi/groups/session.ts +462 -0
- package/src/server/routes/instance/httpapi/groups/sync.ts +113 -0
- package/src/server/routes/instance/httpapi/groups/tui.ts +208 -0
- package/src/server/routes/instance/httpapi/groups/workspace.ts +141 -0
- package/src/server/routes/instance/httpapi/handlers/config.ts +34 -0
- package/src/server/routes/instance/httpapi/handlers/control-plane.ts +37 -0
- package/src/server/routes/instance/httpapi/handlers/control.ts +43 -0
- package/src/server/routes/instance/httpapi/handlers/event.ts +99 -0
- package/src/server/routes/instance/httpapi/handlers/experimental.ts +187 -0
- package/src/server/routes/instance/httpapi/handlers/file.ts +139 -0
- package/src/server/routes/instance/httpapi/handlers/global.ts +156 -0
- package/src/server/routes/instance/httpapi/handlers/instance.ts +110 -0
- package/src/server/routes/instance/httpapi/handlers/mcp.ts +111 -0
- package/src/server/routes/instance/httpapi/handlers/permission.ts +41 -0
- package/src/server/routes/instance/httpapi/handlers/project-copy.ts +83 -0
- package/src/server/routes/instance/httpapi/handlers/project.ts +63 -0
- package/src/server/routes/instance/httpapi/handlers/provider.ts +113 -0
- package/src/server/routes/instance/httpapi/handlers/pty.ts +273 -0
- package/src/server/routes/instance/httpapi/handlers/question.ts +54 -0
- package/src/server/routes/instance/httpapi/handlers/session-errors.ts +21 -0
- package/src/server/routes/instance/httpapi/handlers/session.ts +440 -0
- package/src/server/routes/instance/httpapi/handlers/sync.ts +89 -0
- package/src/server/routes/instance/httpapi/handlers/tui.ts +131 -0
- package/src/server/routes/instance/httpapi/handlers/workspace.ts +102 -0
- package/src/server/routes/instance/httpapi/lifecycle.ts +54 -0
- package/src/server/routes/instance/httpapi/middleware/authorization.ts +150 -0
- package/src/server/routes/instance/httpapi/middleware/compression.ts +64 -0
- package/src/server/routes/instance/httpapi/middleware/cors-vary.ts +29 -0
- package/src/server/routes/instance/httpapi/middleware/error.ts +43 -0
- package/src/server/routes/instance/httpapi/middleware/fence.ts +25 -0
- package/src/server/routes/instance/httpapi/middleware/instance-context.ts +43 -0
- package/src/server/routes/instance/httpapi/middleware/proxy.ts +108 -0
- package/src/server/routes/instance/httpapi/middleware/schema-error.ts +41 -0
- package/src/server/routes/instance/httpapi/middleware/workspace-routing.ts +250 -0
- package/src/server/routes/instance/httpapi/public.ts +535 -0
- package/src/server/routes/instance/httpapi/server.ts +298 -0
- package/src/server/routes/instance/httpapi/websocket-tracker.ts +57 -0
- package/src/server/server.ts +225 -0
- package/src/server/shared/fence.ts +60 -0
- package/src/server/shared/pty-ticket.ts +15 -0
- package/src/server/shared/public-ui.ts +12 -0
- package/src/server/shared/tui-control.ts +28 -0
- package/src/server/shared/ui.ts +122 -0
- package/src/server/shared/workspace-routing.ts +38 -0
- package/src/server/tui-event.ts +53 -0
- package/src/session/compaction.ts +620 -0
- package/src/session/instruction.ts +241 -0
- package/src/session/llm/AGENTS.md +90 -0
- package/src/session/llm/ai-sdk.ts +288 -0
- package/src/session/llm/native-request.ts +196 -0
- package/src/session/llm/native-runtime.ts +196 -0
- package/src/session/llm/request.ts +218 -0
- package/src/session/llm.ts +415 -0
- package/src/session/message-error.ts +14 -0
- package/src/session/message-v2.ts +744 -0
- package/src/session/message.ts +148 -0
- package/src/session/overflow.ts +34 -0
- package/src/session/processor.ts +1084 -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 +95 -0
- package/src/session/prompt/gemini.txt +155 -0
- package/src/session/prompt/gpt.txt +107 -0
- package/src/session/prompt/kimi.txt +95 -0
- package/src/session/prompt/max-steps.txt +16 -0
- package/src/session/prompt/plan-mode.txt +70 -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 +1722 -0
- package/src/session/reminders.ts +92 -0
- package/src/session/retry.ts +201 -0
- package/src/session/revert.ts +160 -0
- package/src/session/run-state.ts +156 -0
- package/src/session/schema.ts +26 -0
- package/src/session/session.ts +1119 -0
- package/src/session/status.ts +97 -0
- package/src/session/summary.ts +165 -0
- package/src/session/system.ts +117 -0
- package/src/session/todo.ts +90 -0
- package/src/session/tools.ts +207 -0
- package/src/share/session.ts +61 -0
- package/src/share/share-next.ts +385 -0
- package/src/skill/discovery.ts +109 -0
- package/src/skill/index.ts +366 -0
- package/src/snapshot/index.ts +808 -0
- package/src/sql.d.ts +4 -0
- package/src/storage/schema.ts +5 -0
- package/src/storage/storage.ts +329 -0
- package/src/sync/README.md +179 -0
- package/src/sync/schema.ts +11 -0
- package/src/temporary.ts +31 -0
- package/src/tool/apply_patch.ts +313 -0
- package/src/tool/apply_patch.txt +33 -0
- package/src/tool/edit.ts +737 -0
- package/src/tool/edit.txt +10 -0
- package/src/tool/external-directory.ts +49 -0
- package/src/tool/glob.ts +76 -0
- package/src/tool/glob.txt +6 -0
- package/src/tool/grep.ts +112 -0
- package/src/tool/grep.txt +8 -0
- package/src/tool/invalid.ts +21 -0
- package/src/tool/json-schema.ts +164 -0
- package/src/tool/lsp.ts +113 -0
- package/src/tool/lsp.txt +24 -0
- package/src/tool/mcp-websearch.ts +96 -0
- package/src/tool/plan-enter.txt +14 -0
- package/src/tool/plan-exit.txt +13 -0
- package/src/tool/plan.ts +79 -0
- package/src/tool/question.ts +44 -0
- package/src/tool/question.txt +10 -0
- package/src/tool/read.ts +386 -0
- package/src/tool/read.txt +14 -0
- package/src/tool/registry.ts +440 -0
- package/src/tool/schema.ts +14 -0
- package/src/tool/shell/id.ts +19 -0
- package/src/tool/shell/prompt.ts +307 -0
- package/src/tool/shell/shell.txt +21 -0
- package/src/tool/shell.ts +657 -0
- package/src/tool/skill.ts +71 -0
- package/src/tool/skill.txt +5 -0
- package/src/tool/task.ts +346 -0
- package/src/tool/task.txt +19 -0
- package/src/tool/todo.ts +57 -0
- package/src/tool/todowrite.txt +44 -0
- package/src/tool/tool.ts +183 -0
- package/src/tool/truncate.ts +158 -0
- package/src/tool/truncation-dir.ts +4 -0
- package/src/tool/webfetch.ts +192 -0
- package/src/tool/webfetch.txt +13 -0
- package/src/tool/websearch.ts +143 -0
- package/src/tool/websearch.txt +14 -0
- package/src/tool/write.ts +104 -0
- package/src/tool/write.txt +8 -0
- package/src/util/archive.ts +17 -0
- package/src/util/bom.ts +27 -0
- package/src/util/data-url.ts +9 -0
- package/src/util/defer.ts +10 -0
- package/src/util/effect-http-client.ts +11 -0
- package/src/util/error.ts +1 -0
- package/src/util/filesystem.ts +251 -0
- package/src/util/html.ts +8 -0
- package/src/util/iife.ts +3 -0
- package/src/util/lazy.ts +20 -0
- package/src/util/local-context.ts +25 -0
- package/src/util/locale.ts +2 -0
- package/src/util/media.ts +26 -0
- package/src/util/process.ts +177 -0
- package/src/util/proxy-env.ts +72 -0
- package/src/util/queue.ts +32 -0
- package/src/util/record.ts +1 -0
- package/src/util/repository.ts +232 -0
- package/src/util/rpc.ts +66 -0
- package/src/util/signal.ts +12 -0
- package/src/util/timeout.ts +13 -0
- package/src/util/token.ts +1 -0
- package/src/util/wildcard.ts +59 -0
- package/src/worktree/index.ts +654 -0
|
@@ -0,0 +1,439 @@
|
|
|
1
|
+
import path from "path"
|
|
2
|
+
import {
|
|
3
|
+
type ParseError as JsoncParseError,
|
|
4
|
+
applyEdits,
|
|
5
|
+
modify,
|
|
6
|
+
parse as parseJsonc,
|
|
7
|
+
printParseErrorCode,
|
|
8
|
+
} from "jsonc-parser"
|
|
9
|
+
|
|
10
|
+
import * as ConfigPaths from "@/config/paths"
|
|
11
|
+
import { Global } from "@rimurucode-ai/core/global"
|
|
12
|
+
import { Filesystem } from "@/util/filesystem"
|
|
13
|
+
import { Flock } from "@rimurucode-ai/core/util/flock"
|
|
14
|
+
import { isRecord } from "@/util/record"
|
|
15
|
+
|
|
16
|
+
import { parsePluginSpecifier, readPackageThemes, readPluginPackage, resolvePluginTarget } from "./shared"
|
|
17
|
+
|
|
18
|
+
type Mode = "noop" | "add" | "replace"
|
|
19
|
+
type Kind = "server" | "tui"
|
|
20
|
+
|
|
21
|
+
export type Target = {
|
|
22
|
+
kind: Kind
|
|
23
|
+
opts?: Record<string, unknown>
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
export type InstallDeps = {
|
|
27
|
+
resolve: (spec: string) => Promise<string>
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
export type PatchDeps = {
|
|
31
|
+
readText: (file: string) => Promise<string>
|
|
32
|
+
write: (file: string, text: string) => Promise<void>
|
|
33
|
+
exists: (file: string) => Promise<boolean>
|
|
34
|
+
files: (dir: string, name: "rimuru-ai" | "tui") => string[]
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
export type PatchInput = {
|
|
38
|
+
spec: string
|
|
39
|
+
targets: Target[]
|
|
40
|
+
force?: boolean
|
|
41
|
+
global?: boolean
|
|
42
|
+
vcs?: string
|
|
43
|
+
worktree: string
|
|
44
|
+
directory: string
|
|
45
|
+
config?: string
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
type Ok<T> = {
|
|
49
|
+
ok: true
|
|
50
|
+
} & T
|
|
51
|
+
|
|
52
|
+
type Err<C extends string, T> = {
|
|
53
|
+
ok: false
|
|
54
|
+
code: C
|
|
55
|
+
} & T
|
|
56
|
+
|
|
57
|
+
export type InstallResult = Ok<{ target: string }> | Err<"install_failed", { error: unknown }>
|
|
58
|
+
|
|
59
|
+
export type ManifestResult =
|
|
60
|
+
| Ok<{ targets: Target[] }>
|
|
61
|
+
| Err<"manifest_read_failed", { file: string; error: unknown }>
|
|
62
|
+
| Err<"manifest_no_targets", { file: string }>
|
|
63
|
+
|
|
64
|
+
export type PatchItem = {
|
|
65
|
+
kind: Kind
|
|
66
|
+
mode: Mode
|
|
67
|
+
file: string
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
type PatchErr =
|
|
71
|
+
| Err<"invalid_json", { kind: Kind; file: string; line: number; col: number; parse: string }>
|
|
72
|
+
| Err<"patch_failed", { kind: Kind; error: unknown }>
|
|
73
|
+
|
|
74
|
+
type PatchOne = Ok<{ item: PatchItem }> | PatchErr
|
|
75
|
+
|
|
76
|
+
export type PatchResult = Ok<{ dir: string; items: PatchItem[] }> | (PatchErr & { dir: string })
|
|
77
|
+
|
|
78
|
+
const defaultInstallDeps: InstallDeps = {
|
|
79
|
+
resolve: (spec) => resolvePluginTarget(spec),
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
const defaultPatchDeps: PatchDeps = {
|
|
83
|
+
readText: (file) => Filesystem.readText(file),
|
|
84
|
+
write: async (file, text) => {
|
|
85
|
+
await Filesystem.write(file, text)
|
|
86
|
+
},
|
|
87
|
+
exists: (file) => Filesystem.exists(file),
|
|
88
|
+
files: (dir, name) => ConfigPaths.fileInDirectory(dir, name),
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
function pluginSpec(item: unknown) {
|
|
92
|
+
if (typeof item === "string") return item
|
|
93
|
+
if (!Array.isArray(item)) return
|
|
94
|
+
if (typeof item[0] !== "string") return
|
|
95
|
+
return item[0]
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
function pluginList(data: unknown) {
|
|
99
|
+
if (!data || typeof data !== "object" || Array.isArray(data)) return
|
|
100
|
+
const item = data as { plugin?: unknown }
|
|
101
|
+
if (!Array.isArray(item.plugin)) return
|
|
102
|
+
return item.plugin
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
function exportValue(value: unknown): string | undefined {
|
|
106
|
+
if (typeof value === "string") {
|
|
107
|
+
const next = value.trim()
|
|
108
|
+
if (next) return next
|
|
109
|
+
return
|
|
110
|
+
}
|
|
111
|
+
if (!isRecord(value)) return
|
|
112
|
+
for (const key of ["import", "default"]) {
|
|
113
|
+
const next = value[key]
|
|
114
|
+
if (typeof next !== "string") continue
|
|
115
|
+
const hit = next.trim()
|
|
116
|
+
if (!hit) continue
|
|
117
|
+
return hit
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
function exportOptions(value: unknown): Record<string, unknown> | undefined {
|
|
122
|
+
if (!isRecord(value)) return
|
|
123
|
+
const config = value.config
|
|
124
|
+
if (!isRecord(config)) return
|
|
125
|
+
return config
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
function exportTarget(pkg: Record<string, unknown>, kind: Kind) {
|
|
129
|
+
const exports = pkg.exports
|
|
130
|
+
if (!isRecord(exports)) return
|
|
131
|
+
const value = exports[`./${kind}`]
|
|
132
|
+
const entry = exportValue(value)
|
|
133
|
+
if (!entry) return
|
|
134
|
+
return {
|
|
135
|
+
opts: exportOptions(value),
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
|
|
139
|
+
function hasMainTarget(pkg: Record<string, unknown>) {
|
|
140
|
+
const main = pkg.main
|
|
141
|
+
if (typeof main !== "string") return false
|
|
142
|
+
return Boolean(main.trim())
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
function packageTargets(pkg: { json: Record<string, unknown>; dir: string; pkg: string }) {
|
|
146
|
+
const spec =
|
|
147
|
+
typeof pkg.json.name === "string" && pkg.json.name.trim().length > 0 ? pkg.json.name.trim() : path.basename(pkg.dir)
|
|
148
|
+
const targets: Target[] = []
|
|
149
|
+
const server = exportTarget(pkg.json, "server")
|
|
150
|
+
if (server) {
|
|
151
|
+
targets.push({ kind: "server", opts: server.opts })
|
|
152
|
+
} else if (hasMainTarget(pkg.json)) {
|
|
153
|
+
targets.push({ kind: "server" })
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
const tui = exportTarget(pkg.json, "tui")
|
|
157
|
+
if (tui) {
|
|
158
|
+
targets.push({ kind: "tui", opts: tui.opts })
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
if (!targets.some((item) => item.kind === "tui") && readPackageThemes(spec, pkg).length) {
|
|
162
|
+
targets.push({ kind: "tui" })
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
return targets
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
function patch(text: string, path: Array<string | number>, value: unknown, insert = false) {
|
|
169
|
+
return applyEdits(
|
|
170
|
+
text,
|
|
171
|
+
modify(text, path, value, {
|
|
172
|
+
formattingOptions: {
|
|
173
|
+
tabSize: 2,
|
|
174
|
+
insertSpaces: true,
|
|
175
|
+
},
|
|
176
|
+
isArrayInsertion: insert,
|
|
177
|
+
}),
|
|
178
|
+
)
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
function patchPluginList(
|
|
182
|
+
text: string,
|
|
183
|
+
list: unknown[] | undefined,
|
|
184
|
+
spec: string,
|
|
185
|
+
next: unknown,
|
|
186
|
+
force = false,
|
|
187
|
+
): { mode: Mode; text: string } {
|
|
188
|
+
const pkg = parsePluginSpecifier(spec).pkg
|
|
189
|
+
const rows = (list ?? []).map((item, i) => ({
|
|
190
|
+
item,
|
|
191
|
+
i,
|
|
192
|
+
spec: pluginSpec(item),
|
|
193
|
+
}))
|
|
194
|
+
const dup = rows.filter((item) => {
|
|
195
|
+
if (!item.spec) return false
|
|
196
|
+
if (item.spec === spec) return true
|
|
197
|
+
if (item.spec.startsWith("file://")) return false
|
|
198
|
+
return parsePluginSpecifier(item.spec).pkg === pkg
|
|
199
|
+
})
|
|
200
|
+
|
|
201
|
+
if (!dup.length) {
|
|
202
|
+
if (!list) {
|
|
203
|
+
return {
|
|
204
|
+
mode: "add",
|
|
205
|
+
text: patch(text, ["plugin"], [next]),
|
|
206
|
+
}
|
|
207
|
+
}
|
|
208
|
+
return {
|
|
209
|
+
mode: "add",
|
|
210
|
+
text: patch(text, ["plugin", list.length], next, true),
|
|
211
|
+
}
|
|
212
|
+
}
|
|
213
|
+
|
|
214
|
+
if (!force) {
|
|
215
|
+
return {
|
|
216
|
+
mode: "noop",
|
|
217
|
+
text,
|
|
218
|
+
}
|
|
219
|
+
}
|
|
220
|
+
|
|
221
|
+
const keep = dup[0]
|
|
222
|
+
if (!keep) {
|
|
223
|
+
return {
|
|
224
|
+
mode: "noop",
|
|
225
|
+
text,
|
|
226
|
+
}
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
if (dup.length === 1 && keep.spec === spec) {
|
|
230
|
+
return {
|
|
231
|
+
mode: "noop",
|
|
232
|
+
text,
|
|
233
|
+
}
|
|
234
|
+
}
|
|
235
|
+
|
|
236
|
+
let out = text
|
|
237
|
+
if (typeof keep.item === "string") {
|
|
238
|
+
out = patch(out, ["plugin", keep.i], next)
|
|
239
|
+
}
|
|
240
|
+
if (Array.isArray(keep.item) && typeof keep.item[0] === "string") {
|
|
241
|
+
out = patch(out, ["plugin", keep.i, 0], spec)
|
|
242
|
+
}
|
|
243
|
+
|
|
244
|
+
const del = dup
|
|
245
|
+
.map((item) => item.i)
|
|
246
|
+
.filter((i) => i !== keep.i)
|
|
247
|
+
.sort((a, b) => b - a)
|
|
248
|
+
|
|
249
|
+
for (const i of del) {
|
|
250
|
+
out = patch(out, ["plugin", i], undefined)
|
|
251
|
+
}
|
|
252
|
+
|
|
253
|
+
return {
|
|
254
|
+
mode: "replace",
|
|
255
|
+
text: out,
|
|
256
|
+
}
|
|
257
|
+
}
|
|
258
|
+
|
|
259
|
+
export async function installPlugin(spec: string, dep: InstallDeps = defaultInstallDeps): Promise<InstallResult> {
|
|
260
|
+
const target = await dep.resolve(spec).then(
|
|
261
|
+
(item) => ({
|
|
262
|
+
ok: true as const,
|
|
263
|
+
item,
|
|
264
|
+
}),
|
|
265
|
+
(error: unknown) => ({
|
|
266
|
+
ok: false as const,
|
|
267
|
+
error,
|
|
268
|
+
}),
|
|
269
|
+
)
|
|
270
|
+
if (!target.ok) {
|
|
271
|
+
return {
|
|
272
|
+
ok: false,
|
|
273
|
+
code: "install_failed",
|
|
274
|
+
error: target.error,
|
|
275
|
+
}
|
|
276
|
+
}
|
|
277
|
+
return {
|
|
278
|
+
ok: true,
|
|
279
|
+
target: target.item,
|
|
280
|
+
}
|
|
281
|
+
}
|
|
282
|
+
|
|
283
|
+
export async function readPluginManifest(target: string): Promise<ManifestResult> {
|
|
284
|
+
const pkg = await readPluginPackage(target).then(
|
|
285
|
+
(item) => ({
|
|
286
|
+
ok: true as const,
|
|
287
|
+
item,
|
|
288
|
+
}),
|
|
289
|
+
(error: unknown) => ({
|
|
290
|
+
ok: false as const,
|
|
291
|
+
error,
|
|
292
|
+
}),
|
|
293
|
+
)
|
|
294
|
+
if (!pkg.ok) {
|
|
295
|
+
return {
|
|
296
|
+
ok: false,
|
|
297
|
+
code: "manifest_read_failed",
|
|
298
|
+
file: target,
|
|
299
|
+
error: pkg.error,
|
|
300
|
+
}
|
|
301
|
+
}
|
|
302
|
+
|
|
303
|
+
const targets = await Promise.resolve()
|
|
304
|
+
.then(() => packageTargets(pkg.item))
|
|
305
|
+
.then(
|
|
306
|
+
(item) => ({ ok: true as const, item }),
|
|
307
|
+
(error: unknown) => ({ ok: false as const, error }),
|
|
308
|
+
)
|
|
309
|
+
|
|
310
|
+
if (!targets.ok) {
|
|
311
|
+
return {
|
|
312
|
+
ok: false,
|
|
313
|
+
code: "manifest_read_failed",
|
|
314
|
+
file: pkg.item.pkg,
|
|
315
|
+
error: targets.error,
|
|
316
|
+
}
|
|
317
|
+
}
|
|
318
|
+
|
|
319
|
+
if (!targets.item.length) {
|
|
320
|
+
return {
|
|
321
|
+
ok: false,
|
|
322
|
+
code: "manifest_no_targets",
|
|
323
|
+
file: pkg.item.pkg,
|
|
324
|
+
}
|
|
325
|
+
}
|
|
326
|
+
|
|
327
|
+
return {
|
|
328
|
+
ok: true,
|
|
329
|
+
targets: targets.item,
|
|
330
|
+
}
|
|
331
|
+
}
|
|
332
|
+
|
|
333
|
+
function patchDir(input: PatchInput) {
|
|
334
|
+
if (input.global) return input.config ?? Global.Path.config
|
|
335
|
+
const git = input.vcs === "git" && input.worktree !== "/"
|
|
336
|
+
const root = git ? input.worktree : input.directory
|
|
337
|
+
return path.join(root, ".rimuru")
|
|
338
|
+
}
|
|
339
|
+
|
|
340
|
+
function patchName(kind: Kind): "rimuru-ai" | "tui" {
|
|
341
|
+
if (kind === "server") return "rimuru-ai"
|
|
342
|
+
return "tui"
|
|
343
|
+
}
|
|
344
|
+
|
|
345
|
+
async function patchOne(dir: string, target: Target, spec: string, force: boolean, dep: PatchDeps): Promise<PatchOne> {
|
|
346
|
+
const name = patchName(target.kind)
|
|
347
|
+
await using _ = await Flock.acquire(`plug-config:${Filesystem.resolve(path.join(dir, name))}`)
|
|
348
|
+
|
|
349
|
+
const files = dep.files(dir, name)
|
|
350
|
+
let cfg = files[0]
|
|
351
|
+
for (const file of files) {
|
|
352
|
+
if (!(await dep.exists(file))) continue
|
|
353
|
+
cfg = file
|
|
354
|
+
break
|
|
355
|
+
}
|
|
356
|
+
|
|
357
|
+
const src = await dep.readText(cfg).catch((err: NodeJS.ErrnoException) => {
|
|
358
|
+
if (err.code === "ENOENT") return "{}"
|
|
359
|
+
return err
|
|
360
|
+
})
|
|
361
|
+
if (src instanceof Error) {
|
|
362
|
+
return {
|
|
363
|
+
ok: false,
|
|
364
|
+
code: "patch_failed",
|
|
365
|
+
kind: target.kind,
|
|
366
|
+
error: src,
|
|
367
|
+
}
|
|
368
|
+
}
|
|
369
|
+
const text = src.trim() ? src : "{}"
|
|
370
|
+
|
|
371
|
+
const errs: JsoncParseError[] = []
|
|
372
|
+
const data = parseJsonc(text, errs, { allowTrailingComma: true })
|
|
373
|
+
if (errs.length) {
|
|
374
|
+
const err = errs[0]
|
|
375
|
+
const lines = text.substring(0, err.offset).split("\n")
|
|
376
|
+
return {
|
|
377
|
+
ok: false,
|
|
378
|
+
code: "invalid_json",
|
|
379
|
+
kind: target.kind,
|
|
380
|
+
file: cfg,
|
|
381
|
+
line: lines.length,
|
|
382
|
+
col: lines[lines.length - 1].length + 1,
|
|
383
|
+
parse: printParseErrorCode(err.error),
|
|
384
|
+
}
|
|
385
|
+
}
|
|
386
|
+
|
|
387
|
+
const list = pluginList(data)
|
|
388
|
+
const item = target.opts ? ([spec, target.opts] as const) : spec
|
|
389
|
+
const out = patchPluginList(text, list, spec, item, force)
|
|
390
|
+
if (out.mode === "noop") {
|
|
391
|
+
return {
|
|
392
|
+
ok: true,
|
|
393
|
+
item: {
|
|
394
|
+
kind: target.kind,
|
|
395
|
+
mode: out.mode,
|
|
396
|
+
file: cfg,
|
|
397
|
+
},
|
|
398
|
+
}
|
|
399
|
+
}
|
|
400
|
+
|
|
401
|
+
const write = await dep.write(cfg, out.text).catch((error: unknown) => error)
|
|
402
|
+
if (write instanceof Error) {
|
|
403
|
+
return {
|
|
404
|
+
ok: false,
|
|
405
|
+
code: "patch_failed",
|
|
406
|
+
kind: target.kind,
|
|
407
|
+
error: write,
|
|
408
|
+
}
|
|
409
|
+
}
|
|
410
|
+
|
|
411
|
+
return {
|
|
412
|
+
ok: true,
|
|
413
|
+
item: {
|
|
414
|
+
kind: target.kind,
|
|
415
|
+
mode: out.mode,
|
|
416
|
+
file: cfg,
|
|
417
|
+
},
|
|
418
|
+
}
|
|
419
|
+
}
|
|
420
|
+
|
|
421
|
+
export async function patchPluginConfig(input: PatchInput, dep: PatchDeps = defaultPatchDeps): Promise<PatchResult> {
|
|
422
|
+
const dir = patchDir(input)
|
|
423
|
+
const items: PatchItem[] = []
|
|
424
|
+
for (const target of input.targets) {
|
|
425
|
+
const hit = await patchOne(dir, target, input.spec, Boolean(input.force), dep)
|
|
426
|
+
if (!hit.ok) {
|
|
427
|
+
return {
|
|
428
|
+
...hit,
|
|
429
|
+
dir,
|
|
430
|
+
}
|
|
431
|
+
}
|
|
432
|
+
items.push(hit.item)
|
|
433
|
+
}
|
|
434
|
+
return {
|
|
435
|
+
ok: true,
|
|
436
|
+
dir,
|
|
437
|
+
items,
|
|
438
|
+
}
|
|
439
|
+
}
|
|
@@ -0,0 +1,237 @@
|
|
|
1
|
+
import {
|
|
2
|
+
checkPluginCompatibility,
|
|
3
|
+
createPluginEntry,
|
|
4
|
+
isDeprecatedPlugin,
|
|
5
|
+
pluginSource,
|
|
6
|
+
resolvePluginTarget,
|
|
7
|
+
type PluginKind,
|
|
8
|
+
type PluginPackage,
|
|
9
|
+
type PluginSource,
|
|
10
|
+
} from "./shared"
|
|
11
|
+
import { ConfigPlugin } from "@/config/plugin"
|
|
12
|
+
import { ConfigPluginV1 } from "@rimurucode-ai/core/v1/config/plugin"
|
|
13
|
+
import { InstallationVersion } from "@rimurucode-ai/core/installation/version"
|
|
14
|
+
|
|
15
|
+
export namespace PluginLoader {
|
|
16
|
+
// A normalized plugin declaration derived from config before any filesystem or npm work happens.
|
|
17
|
+
export type Plan = {
|
|
18
|
+
spec: string
|
|
19
|
+
options: ConfigPluginV1.Options | undefined
|
|
20
|
+
deprecated: boolean
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
// A plugin that has been resolved to a concrete target and entrypoint on disk.
|
|
24
|
+
export type Resolved = Plan & {
|
|
25
|
+
source: PluginSource
|
|
26
|
+
target: string
|
|
27
|
+
entry: string
|
|
28
|
+
pkg?: PluginPackage
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
// A plugin target we could inspect, but which does not expose the requested kind of entrypoint.
|
|
32
|
+
export type Missing = Plan & {
|
|
33
|
+
source: PluginSource
|
|
34
|
+
target: string
|
|
35
|
+
pkg?: PluginPackage
|
|
36
|
+
message: string
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
// A resolved plugin whose module has been imported successfully.
|
|
40
|
+
export type Loaded = Resolved & {
|
|
41
|
+
mod: Record<string, unknown>
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
type Candidate = { origin: ConfigPlugin.Origin; plan: Plan }
|
|
45
|
+
type Report = {
|
|
46
|
+
// Called before each attempt so callers can log initial load attempts and retries uniformly.
|
|
47
|
+
start?: (candidate: Candidate, retry: boolean) => void
|
|
48
|
+
// Called when the package exists but does not provide the requested entrypoint.
|
|
49
|
+
missing?: (candidate: Candidate, retry: boolean, message: string, resolved: Missing) => void
|
|
50
|
+
// Called for operational failures such as install, compatibility, or dynamic import errors.
|
|
51
|
+
error?: (
|
|
52
|
+
candidate: Candidate,
|
|
53
|
+
retry: boolean,
|
|
54
|
+
stage: "install" | "entry" | "compatibility" | "load",
|
|
55
|
+
error: unknown,
|
|
56
|
+
resolved?: Resolved,
|
|
57
|
+
) => void
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
type AttemptResult<R> = {
|
|
61
|
+
value?: R
|
|
62
|
+
retry: boolean
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
function errorMessage(error: unknown) {
|
|
66
|
+
if (!error || typeof error !== "object") return ""
|
|
67
|
+
const message = "message" in error && typeof error.message === "string" ? error.message : ""
|
|
68
|
+
return message
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
function isRetryableResolveError(stage: "install" | "entry" | "compatibility", error: unknown) {
|
|
72
|
+
if (stage !== "install") return false
|
|
73
|
+
return errorMessage(error).includes("missing package.json or index file")
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
// Normalize a config item into the loader's internal representation.
|
|
77
|
+
function plan(item: ConfigPluginV1.Spec): Plan {
|
|
78
|
+
const spec = ConfigPlugin.pluginSpecifier(item)
|
|
79
|
+
return { spec, options: ConfigPlugin.pluginOptions(item), deprecated: isDeprecatedPlugin(spec) }
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
// Resolve a configured plugin into a concrete entrypoint that can later be imported.
|
|
83
|
+
//
|
|
84
|
+
// The stages here intentionally separate install/target resolution, entrypoint detection,
|
|
85
|
+
// and compatibility checks so callers can report the exact reason a plugin was skipped.
|
|
86
|
+
export async function resolve(
|
|
87
|
+
plan: Plan,
|
|
88
|
+
kind: PluginKind,
|
|
89
|
+
): Promise<
|
|
90
|
+
| { ok: true; value: Resolved }
|
|
91
|
+
| { ok: false; stage: "missing"; value: Missing }
|
|
92
|
+
| { ok: false; stage: "install" | "entry" | "compatibility"; error: unknown }
|
|
93
|
+
> {
|
|
94
|
+
// First make sure the plugin exists locally, installing npm plugins on demand.
|
|
95
|
+
let target = ""
|
|
96
|
+
try {
|
|
97
|
+
target = await resolvePluginTarget(plan.spec)
|
|
98
|
+
} catch (error) {
|
|
99
|
+
return { ok: false, stage: "install", error }
|
|
100
|
+
}
|
|
101
|
+
if (!target) return { ok: false, stage: "install", error: new Error(`Plugin ${plan.spec} target is empty`) }
|
|
102
|
+
|
|
103
|
+
// Then inspect the target for the requested server/tui entrypoint.
|
|
104
|
+
let base
|
|
105
|
+
try {
|
|
106
|
+
base = await createPluginEntry(plan.spec, target, kind)
|
|
107
|
+
} catch (error) {
|
|
108
|
+
return { ok: false, stage: "entry", error }
|
|
109
|
+
}
|
|
110
|
+
if (!base.entry)
|
|
111
|
+
return {
|
|
112
|
+
ok: false,
|
|
113
|
+
stage: "missing",
|
|
114
|
+
value: {
|
|
115
|
+
...plan,
|
|
116
|
+
source: base.source,
|
|
117
|
+
target: base.target,
|
|
118
|
+
pkg: base.pkg,
|
|
119
|
+
message: `Plugin ${plan.spec} does not expose a ${kind} entrypoint`,
|
|
120
|
+
},
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
// npm plugins can declare which rimuru-ai versions they support; file plugins are treated
|
|
124
|
+
// as local development code and skip this compatibility gate.
|
|
125
|
+
if (base.source === "npm") {
|
|
126
|
+
try {
|
|
127
|
+
await checkPluginCompatibility(base.target, InstallationVersion, base.pkg)
|
|
128
|
+
} catch (error) {
|
|
129
|
+
return { ok: false, stage: "compatibility", error }
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
return { ok: true, value: { ...plan, source: base.source, target: base.target, entry: base.entry, pkg: base.pkg } }
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
// Import the resolved module only after all earlier validation has succeeded.
|
|
136
|
+
export async function load(row: Resolved): Promise<{ ok: true; value: Loaded } | { ok: false; error: unknown }> {
|
|
137
|
+
let mod
|
|
138
|
+
try {
|
|
139
|
+
mod = await import(row.entry)
|
|
140
|
+
} catch (error) {
|
|
141
|
+
return { ok: false, error }
|
|
142
|
+
}
|
|
143
|
+
if (!mod) return { ok: false, error: new Error(`Plugin ${row.spec} module is empty`) }
|
|
144
|
+
return { ok: true, value: { ...row, mod } }
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
// Run one candidate through the full pipeline: resolve, optionally surface a missing entry,
|
|
148
|
+
// import the module, and finally let the caller transform the loaded plugin into any result type.
|
|
149
|
+
async function attempt<R>(
|
|
150
|
+
candidate: Candidate,
|
|
151
|
+
kind: PluginKind,
|
|
152
|
+
retry: boolean,
|
|
153
|
+
finish: ((load: Loaded, origin: ConfigPlugin.Origin, retry: boolean) => Promise<R | undefined>) | undefined,
|
|
154
|
+
missing: ((value: Missing, origin: ConfigPlugin.Origin, retry: boolean) => Promise<R | undefined>) | undefined,
|
|
155
|
+
report: Report | undefined,
|
|
156
|
+
): Promise<AttemptResult<R>> {
|
|
157
|
+
const plan = candidate.plan
|
|
158
|
+
const filePlugin = pluginSource(plan.spec) === "file"
|
|
159
|
+
|
|
160
|
+
// Deprecated plugin packages are silently ignored because they are now built in.
|
|
161
|
+
if (plan.deprecated) return { retry: false }
|
|
162
|
+
|
|
163
|
+
report?.start?.(candidate, retry)
|
|
164
|
+
|
|
165
|
+
const resolved = await resolve(plan, kind)
|
|
166
|
+
if (!resolved.ok) {
|
|
167
|
+
if (resolved.stage === "missing") {
|
|
168
|
+
// Missing entrypoints are handled separately so callers can still inspect package metadata,
|
|
169
|
+
// for example to load theme files from a tui plugin package that has no code entrypoint.
|
|
170
|
+
if (missing) {
|
|
171
|
+
const value = await missing(resolved.value, candidate.origin, retry)
|
|
172
|
+
if (value !== undefined) return { value, retry: false }
|
|
173
|
+
}
|
|
174
|
+
report?.missing?.(candidate, retry, resolved.value.message, resolved.value)
|
|
175
|
+
return { retry: false }
|
|
176
|
+
}
|
|
177
|
+
report?.error?.(candidate, retry, resolved.stage, resolved.error)
|
|
178
|
+
return { retry: filePlugin && isRetryableResolveError(resolved.stage, resolved.error) }
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
const loaded = await load(resolved.value)
|
|
182
|
+
if (!loaded.ok) {
|
|
183
|
+
report?.error?.(candidate, retry, "load", loaded.error, resolved.value)
|
|
184
|
+
return { retry: false }
|
|
185
|
+
}
|
|
186
|
+
|
|
187
|
+
// The default behavior is to return the successfully loaded plugin as-is, but callers can
|
|
188
|
+
// provide a finisher to adapt the result into a more specific runtime shape.
|
|
189
|
+
if (!finish) return { value: loaded.value as R, retry: false }
|
|
190
|
+
const value = await finish(loaded.value, candidate.origin, retry)
|
|
191
|
+
return { value, retry: false }
|
|
192
|
+
}
|
|
193
|
+
|
|
194
|
+
type Input<R> = {
|
|
195
|
+
items: ConfigPlugin.Origin[]
|
|
196
|
+
kind: PluginKind
|
|
197
|
+
wait?: () => Promise<void>
|
|
198
|
+
finish?: (load: Loaded, origin: ConfigPlugin.Origin, retry: boolean) => Promise<R | undefined>
|
|
199
|
+
missing?: (value: Missing, origin: ConfigPlugin.Origin, retry: boolean) => Promise<R | undefined>
|
|
200
|
+
report?: Report
|
|
201
|
+
}
|
|
202
|
+
|
|
203
|
+
// Resolve and load all configured plugins in parallel.
|
|
204
|
+
//
|
|
205
|
+
// If `wait` is provided, file-based plugins with retryable pre-import setup failures are retried
|
|
206
|
+
// once after the caller finishes preparing dependencies. Once dynamic import runs, failures are
|
|
207
|
+
// treated as permanent for this process because Bun caches failed module resolution.
|
|
208
|
+
export async function loadExternal<R = Loaded>(input: Input<R>): Promise<R[]> {
|
|
209
|
+
const candidates = input.items.map((origin) => ({ origin, plan: plan(origin.spec) }))
|
|
210
|
+
const list: Array<Promise<AttemptResult<R>>> = []
|
|
211
|
+
for (const candidate of candidates) {
|
|
212
|
+
list.push(attempt(candidate, input.kind, false, input.finish, input.missing, input.report))
|
|
213
|
+
}
|
|
214
|
+
const out = await Promise.all(list)
|
|
215
|
+
if (input.wait) {
|
|
216
|
+
let deps: Promise<void> | undefined
|
|
217
|
+
for (let i = 0; i < candidates.length; i++) {
|
|
218
|
+
const previous = out[i]
|
|
219
|
+
if (previous?.value !== undefined) continue
|
|
220
|
+
if (previous?.retry !== true) continue
|
|
221
|
+
|
|
222
|
+
// Only pre-import file plugin setup failures are retried. Bun caches failed dynamic imports,
|
|
223
|
+
// so dependency waiting cannot fix load/build/runtime/shape failures in this process.
|
|
224
|
+
const candidate = candidates[i]
|
|
225
|
+
if (!candidate || pluginSource(candidate.plan.spec) !== "file") continue
|
|
226
|
+
deps ??= input.wait()
|
|
227
|
+
await deps
|
|
228
|
+
out[i] = await attempt(candidate, input.kind, true, input.finish, input.missing, input.report)
|
|
229
|
+
}
|
|
230
|
+
}
|
|
231
|
+
|
|
232
|
+
// Drop skipped/failed entries while preserving the successful result order.
|
|
233
|
+
const ready: R[] = []
|
|
234
|
+
for (const item of out) if (item.value !== undefined) ready.push(item.value)
|
|
235
|
+
return ready
|
|
236
|
+
}
|
|
237
|
+
}
|