bonecode 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/ARCHITECTURE.md +183 -0
- package/README.md +71 -0
- package/bin/bonecode +62 -0
- package/bone/migrations/rag_vectors.sql +258 -0
- package/bone/output/agent/.dockerignore +7 -0
- package/bone/output/agent/.env.example +36 -0
- package/bone/output/agent/.github/workflows/ci.yaml +58 -0
- package/bone/output/agent/AgentDomain.bone.map +350 -0
- package/bone/output/agent/AgentDomain.postman_collection.json +958 -0
- package/bone/output/agent/Dockerfile +22 -0
- package/bone/output/agent/README.md +47 -0
- package/bone/output/agent/admin/index.html +740 -0
- package/bone/output/agent/docker-compose.yaml +22 -0
- package/bone/output/agent/k8s/deployment.yaml +75 -0
- package/bone/output/agent/migrations/agent.sql +36 -0
- package/bone/output/agent/migrations/agent_instance.sql +36 -0
- package/bone/output/agent/migrations/audit_log.sql +18 -0
- package/bone/output/agent/migrations/build_step.sql +34 -0
- package/bone/output/agent/migrations/event_outbox.sql +31 -0
- package/bone/output/agent/migrations/plan.sql +30 -0
- package/bone/output/agent/migrations/task.sql +30 -0
- package/bone/output/agent/migrations/tool_call.sql +33 -0
- package/bone/output/agent/openapi.yaml +1116 -0
- package/bone/output/agent/package.json +36 -0
- package/bone/output/agent/schema.graphql +233 -0
- package/bone/output/agent/sdk/client.ts +231 -0
- package/bone/output/agent/src/algorithms.ts +2 -0
- package/bone/output/agent/src/audit.ts +44 -0
- package/bone/output/agent/src/auth.ts +57 -0
- package/bone/output/agent/src/cron.ts +12 -0
- package/bone/output/agent/src/db.ts +32 -0
- package/bone/output/agent/src/debug.ts +66 -0
- package/bone/output/agent/src/events.ts +243 -0
- package/bone/output/agent/src/extensions.ts +54 -0
- package/bone/output/agent/src/failure_rules.ts +323 -0
- package/bone/output/agent/src/flows.ts +168 -0
- package/bone/output/agent/src/health.ts +43 -0
- package/bone/output/agent/src/index.ts +100 -0
- package/bone/output/agent/src/logger.ts +66 -0
- package/bone/output/agent/src/metrics.ts +75 -0
- package/bone/output/agent/src/migrate.ts +352 -0
- package/bone/output/agent/src/migration_diff.ts +108 -0
- package/bone/output/agent/src/notify.ts +125 -0
- package/bone/output/agent/src/routes/agent_instance.ts +234 -0
- package/bone/output/agent/src/routes/build_step.ts +105 -0
- package/bone/output/agent/src/routes/plan.ts +91 -0
- package/bone/output/agent/src/routes/task.ts +105 -0
- package/bone/output/agent/src/routes/tool_call.ts +166 -0
- package/bone/output/agent/src/schemas.ts +384 -0
- package/bone/output/agent/src/state_machines/agent_instance.ts +24 -0
- package/bone/output/agent/src/state_machines/build_step.ts +22 -0
- package/bone/output/agent/src/state_machines/plan.ts +22 -0
- package/bone/output/agent/src/state_machines/task.ts +22 -0
- package/bone/output/agent/src/state_machines/tool_call.ts +22 -0
- package/bone/output/agent/src/tests.ts +362 -0
- package/bone/output/agent/src/websocket.ts +201 -0
- package/bone/output/agent/tsconfig.json +25 -0
- package/bone/output/rag/.dockerignore +7 -0
- package/bone/output/rag/.env.example +36 -0
- package/bone/output/rag/.github/workflows/ci.yaml +58 -0
- package/bone/output/rag/Dockerfile +22 -0
- package/bone/output/rag/RAGDomain.bone.map +287 -0
- package/bone/output/rag/RAGDomain.postman_collection.json +923 -0
- package/bone/output/rag/README.md +47 -0
- package/bone/output/rag/admin/index.html +818 -0
- package/bone/output/rag/docker-compose.yaml +22 -0
- package/bone/output/rag/k8s/deployment.yaml +75 -0
- package/bone/output/rag/migrations/audit_log.sql +18 -0
- package/bone/output/rag/migrations/code_chunk.sql +34 -0
- package/bone/output/rag/migrations/code_file.sql +33 -0
- package/bone/output/rag/migrations/event_outbox.sql +31 -0
- package/bone/output/rag/migrations/indexing_job.sql +33 -0
- package/bone/output/rag/migrations/knowledge_base.sql +35 -0
- package/bone/output/rag/migrations/memory_entry.sql +34 -0
- package/bone/output/rag/openapi.yaml +1097 -0
- package/bone/output/rag/package.json +36 -0
- package/bone/output/rag/schema.graphql +245 -0
- package/bone/output/rag/sdk/client.ts +234 -0
- package/bone/output/rag/src/algorithms.ts +2 -0
- package/bone/output/rag/src/audit.ts +37 -0
- package/bone/output/rag/src/auth.ts +57 -0
- package/bone/output/rag/src/cron.ts +12 -0
- package/bone/output/rag/src/db.ts +32 -0
- package/bone/output/rag/src/debug.ts +66 -0
- package/bone/output/rag/src/events.ts +243 -0
- package/bone/output/rag/src/extensions.ts +350 -0
- package/bone/output/rag/src/failure_rules.ts +315 -0
- package/bone/output/rag/src/flows.ts +239 -0
- package/bone/output/rag/src/health.ts +43 -0
- package/bone/output/rag/src/index.ts +95 -0
- package/bone/output/rag/src/logger.ts +66 -0
- package/bone/output/rag/src/metrics.ts +75 -0
- package/bone/output/rag/src/migrate.ts +364 -0
- package/bone/output/rag/src/migration_diff.ts +108 -0
- package/bone/output/rag/src/notify.ts +99 -0
- package/bone/output/rag/src/routes/code_chunk.ts +75 -0
- package/bone/output/rag/src/routes/code_file.ts +101 -0
- package/bone/output/rag/src/routes/indexing_job.ts +87 -0
- package/bone/output/rag/src/routes/knowledge_base.ts +230 -0
- package/bone/output/rag/src/routes/memory_entry.ts +87 -0
- package/bone/output/rag/src/schemas.ts +394 -0
- package/bone/output/rag/src/state_machines/code_file.ts +23 -0
- package/bone/output/rag/src/state_machines/indexing_job.ts +22 -0
- package/bone/output/rag/src/state_machines/knowledge_base.ts +23 -0
- package/bone/output/rag/src/state_machines/memory_entry.ts +20 -0
- package/bone/output/rag/src/tests.ts +340 -0
- package/bone/output/rag/tsconfig.json +25 -0
- package/bone/output/session/.dockerignore +7 -0
- package/bone/output/session/.env.example +36 -0
- package/bone/output/session/.github/workflows/ci.yaml +58 -0
- package/bone/output/session/Dockerfile +22 -0
- package/bone/output/session/README.md +47 -0
- package/bone/output/session/SessionDomain.bone.map +350 -0
- package/bone/output/session/SessionDomain.postman_collection.json +958 -0
- package/bone/output/session/admin/index.html +667 -0
- package/bone/output/session/docker-compose.yaml +22 -0
- package/bone/output/session/k8s/deployment.yaml +75 -0
- package/bone/output/session/migrations/audit_log.sql +18 -0
- package/bone/output/session/migrations/event_outbox.sql +31 -0
- package/bone/output/session/migrations/message.sql +31 -0
- package/bone/output/session/migrations/part.sql +28 -0
- package/bone/output/session/migrations/permission.sql +28 -0
- package/bone/output/session/migrations/project.sql +28 -0
- package/bone/output/session/migrations/session.sql +38 -0
- package/bone/output/session/openapi.yaml +1101 -0
- package/bone/output/session/package.json +36 -0
- package/bone/output/session/schema.graphql +222 -0
- package/bone/output/session/sdk/client.ts +225 -0
- package/bone/output/session/src/algorithms.ts +2 -0
- package/bone/output/session/src/audit.ts +44 -0
- package/bone/output/session/src/auth.ts +57 -0
- package/bone/output/session/src/cron.ts +12 -0
- package/bone/output/session/src/db.ts +32 -0
- package/bone/output/session/src/debug.ts +66 -0
- package/bone/output/session/src/events.ts +270 -0
- package/bone/output/session/src/extensions.ts +215 -0
- package/bone/output/session/src/failure_rules.ts +284 -0
- package/bone/output/session/src/flows.ts +168 -0
- package/bone/output/session/src/health.ts +43 -0
- package/bone/output/session/src/index.ts +100 -0
- package/bone/output/session/src/logger.ts +66 -0
- package/bone/output/session/src/metrics.ts +75 -0
- package/bone/output/session/src/migrate.ts +332 -0
- package/bone/output/session/src/migration_diff.ts +108 -0
- package/bone/output/session/src/notify.ts +112 -0
- package/bone/output/session/src/routes/message.ts +93 -0
- package/bone/output/session/src/routes/part.ts +79 -0
- package/bone/output/session/src/routes/permission.ts +79 -0
- package/bone/output/session/src/routes/project.ts +79 -0
- package/bone/output/session/src/routes/session.ts +294 -0
- package/bone/output/session/src/schemas.ts +357 -0
- package/bone/output/session/src/state_machines/session.ts +23 -0
- package/bone/output/session/src/tests.ts +326 -0
- package/bone/output/session/src/websocket.ts +201 -0
- package/bone/output/session/tsconfig.json +25 -0
- package/bone/output/workspace/.dockerignore +7 -0
- package/bone/output/workspace/.env.example +36 -0
- package/bone/output/workspace/.github/workflows/ci.yaml +58 -0
- package/bone/output/workspace/Dockerfile +22 -0
- package/bone/output/workspace/README.md +45 -0
- package/bone/output/workspace/WorkspaceDomain.bone.map +189 -0
- package/bone/output/workspace/WorkspaceDomain.postman_collection.json +621 -0
- package/bone/output/workspace/admin/index.html +485 -0
- package/bone/output/workspace/docker-compose.yaml +22 -0
- package/bone/output/workspace/k8s/deployment.yaml +75 -0
- package/bone/output/workspace/migrations/audit_log.sql +18 -0
- package/bone/output/workspace/migrations/codebase.sql +34 -0
- package/bone/output/workspace/migrations/event_outbox.sql +31 -0
- package/bone/output/workspace/migrations/snapshot.sql +32 -0
- package/bone/output/workspace/migrations/workspace.sql +33 -0
- package/bone/output/workspace/openapi.yaml +721 -0
- package/bone/output/workspace/package.json +36 -0
- package/bone/output/workspace/schema.graphql +153 -0
- package/bone/output/workspace/sdk/client.ts +155 -0
- package/bone/output/workspace/src/algorithms.ts +2 -0
- package/bone/output/workspace/src/audit.ts +37 -0
- package/bone/output/workspace/src/auth.ts +57 -0
- package/bone/output/workspace/src/cron.ts +12 -0
- package/bone/output/workspace/src/db.ts +32 -0
- package/bone/output/workspace/src/debug.ts +66 -0
- package/bone/output/workspace/src/events.ts +243 -0
- package/bone/output/workspace/src/extensions.ts +44 -0
- package/bone/output/workspace/src/failure_rules.ts +153 -0
- package/bone/output/workspace/src/health.ts +43 -0
- package/bone/output/workspace/src/index.ts +89 -0
- package/bone/output/workspace/src/logger.ts +66 -0
- package/bone/output/workspace/src/metrics.ts +75 -0
- package/bone/output/workspace/src/migrate.ts +220 -0
- package/bone/output/workspace/src/migration_diff.ts +108 -0
- package/bone/output/workspace/src/notify.ts +73 -0
- package/bone/output/workspace/src/routes/codebase.ts +87 -0
- package/bone/output/workspace/src/routes/snapshot.ts +127 -0
- package/bone/output/workspace/src/routes/workspace.ts +190 -0
- package/bone/output/workspace/src/schemas.ts +231 -0
- package/bone/output/workspace/src/state_machines/codebase.ts +21 -0
- package/bone/output/workspace/src/state_machines/snapshot.ts +20 -0
- package/bone/output/workspace/src/state_machines/workspace.ts +21 -0
- package/bone/output/workspace/src/tests.ts +249 -0
- package/bone/output/workspace/tsconfig.json +25 -0
- package/compat/opencode_adapter.ts +410 -0
- package/package.json +69 -0
- package/scripts/check_benchmark_session.js +34 -0
- package/scripts/check_finish_event.js +24 -0
- package/scripts/check_parts.js +15 -0
- package/scripts/compile.js +79 -0
- package/scripts/copy_opencode.ps1 +53 -0
- package/scripts/create_functions.sql +129 -0
- package/scripts/migrate.js +85 -0
- package/scripts/migrate_from_opencode.ts +218 -0
- package/scripts/test_agent_loop.js +101 -0
- package/scripts/test_api.ps1 +116 -0
- package/scripts/test_context_builder.js +136 -0
- package/scripts/test_context_builder.ts +97 -0
- package/scripts/test_rag.js +189 -0
- package/scripts/test_stream_events.js +36 -0
- package/scripts/test_websocket_and_saga.js +216 -0
- package/src/cli.ts +475 -0
- package/src/config.ts +162 -0
- package/src/context_builder.ts +598 -0
- package/src/engine/account/account.sql.ts +39 -0
- package/src/engine/account/account.ts +456 -0
- package/src/engine/account/repo.ts +166 -0
- package/src/engine/account/schema.ts +99 -0
- package/src/engine/account/url.ts +8 -0
- package/src/engine/acp/README.md +174 -0
- package/src/engine/acp/agent.ts +1968 -0
- package/src/engine/acp/runtime.ts +22 -0
- package/src/engine/acp/session.ts +122 -0
- package/src/engine/acp/types.ts +24 -0
- package/src/engine/agent/agent.ts +463 -0
- package/src/engine/agent/generate.txt +75 -0
- package/src/engine/agent/prompt/compaction.txt +9 -0
- package/src/engine/agent/prompt/explore.txt +18 -0
- package/src/engine/agent/prompt/scout.txt +36 -0
- package/src/engine/agent/prompt/summary.txt +11 -0
- package/src/engine/agent/prompt/title.txt +44 -0
- package/src/engine/agent/subagent-permissions.ts +34 -0
- package/src/engine/auth/index.ts +96 -0
- package/src/engine/background/background/job.ts +200 -0
- package/src/engine/background/job.ts +200 -0
- package/src/engine/bus/bus-event.ts +45 -0
- package/src/engine/bus/global.ts +22 -0
- package/src/engine/bus/index.ts +203 -0
- package/src/engine/command/command/index.ts +181 -0
- package/src/engine/command/command/template/initialize.txt +66 -0
- package/src/engine/command/command/template/review.txt +101 -0
- package/src/engine/command/index.ts +181 -0
- package/src/engine/command/template/initialize.txt +66 -0
- package/src/engine/command/template/review.txt +101 -0
- package/src/engine/config/agent.ts +172 -0
- package/src/engine/config/attachment.ts +25 -0
- package/src/engine/config/command.ts +62 -0
- package/src/engine/config/config.ts +833 -0
- package/src/engine/config/console-state.ts +14 -0
- package/src/engine/config/entry-name.ts +16 -0
- package/src/engine/config/error.ts +23 -0
- package/src/engine/config/formatter.ts +13 -0
- package/src/engine/config/layout.ts +6 -0
- package/src/engine/config/lsp.ts +43 -0
- package/src/engine/config/managed.ts +71 -0
- package/src/engine/config/markdown.ts +96 -0
- package/src/engine/config/mcp.ts +56 -0
- package/src/engine/config/model-id.ts +5 -0
- package/src/engine/config/parse.ts +79 -0
- package/src/engine/config/paths.ts +45 -0
- package/src/engine/config/permission.ts +58 -0
- package/src/engine/config/plugin.ts +84 -0
- package/src/engine/config/provider.ts +111 -0
- package/src/engine/config/reference.ts +23 -0
- package/src/engine/config/server.ts +19 -0
- package/src/engine/config/skills.ts +14 -0
- package/src/engine/config/variable.ts +90 -0
- package/src/engine/control-plane/adapters/index.ts +41 -0
- package/src/engine/control-plane/adapters/worktree.ts +96 -0
- package/src/engine/control-plane/dev/README.md +19 -0
- package/src/engine/control-plane/dev/debug-workspace-plugin.ts +73 -0
- package/src/engine/control-plane/schema.ts +14 -0
- package/src/engine/control-plane/types.ts +59 -0
- package/src/engine/control-plane/util.ts +39 -0
- package/src/engine/control-plane/workspace-adapter-runtime.ts +51 -0
- package/src/engine/control-plane/workspace-context.ts +26 -0
- package/src/engine/control-plane/workspace.sql.ts +20 -0
- package/src/engine/control-plane/workspace.ts +1072 -0
- package/src/engine/data-migration.ts +161 -0
- package/src/engine/effect/app-runtime.ts +143 -0
- package/src/engine/effect/bootstrap-runtime.ts +29 -0
- package/src/engine/effect/bridge.ts +84 -0
- package/src/engine/effect/config-service.ts +67 -0
- package/src/engine/effect/instance-ref.ts +11 -0
- package/src/engine/effect/instance-registry.ts +12 -0
- package/src/engine/effect/instance-state.ts +72 -0
- package/src/engine/effect/promise.ts +17 -0
- package/src/engine/effect/run-service.ts +47 -0
- package/src/engine/effect/runner.ts +217 -0
- package/src/engine/effect/runtime-flags.ts +74 -0
- package/src/engine/effect/service-use.ts +38 -0
- package/src/engine/env/index.ts +37 -0
- package/src/engine/event-v2-bridge.ts +89 -0
- package/src/engine/file/file/ignore.ts +81 -0
- package/src/engine/file/file/index.ts +651 -0
- package/src/engine/file/file/protected.ts +59 -0
- package/src/engine/file/file/ripgrep.ts +481 -0
- package/src/engine/file/file/watcher.ts +167 -0
- package/src/engine/file/ignore.ts +81 -0
- package/src/engine/file/index.ts +651 -0
- package/src/engine/file/protected.ts +59 -0
- package/src/engine/file/ripgrep.ts +481 -0
- package/src/engine/file/watcher.ts +167 -0
- package/src/engine/format/format/formatter.ts +404 -0
- package/src/engine/format/format/index.ts +209 -0
- package/src/engine/format/formatter.ts +404 -0
- package/src/engine/format/index.ts +209 -0
- package/src/engine/git/git/index.ts +347 -0
- package/src/engine/git/index.ts +347 -0
- package/src/engine/id/id.ts +80 -0
- package/src/engine/ide/index.ts +70 -0
- package/src/engine/image/image/image.ts +176 -0
- package/src/engine/image/image.ts +176 -0
- package/src/engine/index.ts +251 -0
- package/src/engine/installation/index.ts +327 -0
- package/src/engine/lsp/client.ts +707 -0
- package/src/engine/lsp/diagnostic.ts +29 -0
- package/src/engine/lsp/language.ts +121 -0
- package/src/engine/lsp/launch.ts +21 -0
- package/src/engine/lsp/lsp/client.ts +707 -0
- package/src/engine/lsp/lsp/diagnostic.ts +29 -0
- package/src/engine/lsp/lsp/language.ts +121 -0
- package/src/engine/lsp/lsp/launch.ts +21 -0
- package/src/engine/lsp/lsp/lsp.ts +507 -0
- package/src/engine/lsp/lsp/server.ts +2064 -0
- package/src/engine/lsp/lsp.ts +507 -0
- package/src/engine/lsp/server.ts +2064 -0
- package/src/engine/mcp/auth.ts +146 -0
- package/src/engine/mcp/index.ts +958 -0
- package/src/engine/mcp/mcp/auth.ts +146 -0
- package/src/engine/mcp/mcp/index.ts +958 -0
- package/src/engine/mcp/mcp/oauth-callback.ts +232 -0
- package/src/engine/mcp/mcp/oauth-provider.ts +214 -0
- package/src/engine/mcp/oauth-callback.ts +232 -0
- package/src/engine/mcp/oauth-provider.ts +214 -0
- package/src/engine/node.ts +6 -0
- package/src/engine/patch/index.ts +689 -0
- package/src/engine/patch/patch/index.ts +689 -0
- package/src/engine/permission/arity.ts +163 -0
- package/src/engine/permission/evaluate.ts +15 -0
- package/src/engine/permission/index.ts +306 -0
- package/src/engine/permission/permission/arity.ts +163 -0
- package/src/engine/permission/permission/evaluate.ts +15 -0
- package/src/engine/permission/permission/index.ts +306 -0
- package/src/engine/permission/permission/schema.ts +13 -0
- package/src/engine/permission/schema.ts +13 -0
- package/src/engine/plugin/azure.ts +26 -0
- package/src/engine/plugin/cloudflare.ts +76 -0
- package/src/engine/plugin/codex.ts +622 -0
- package/src/engine/plugin/digitalocean.ts +411 -0
- package/src/engine/plugin/github-copilot/copilot.ts +394 -0
- package/src/engine/plugin/github-copilot/models.ts +196 -0
- package/src/engine/plugin/index.ts +295 -0
- package/src/engine/plugin/install.ts +439 -0
- package/src/engine/plugin/loader.ts +216 -0
- package/src/engine/plugin/meta.ts +188 -0
- package/src/engine/plugin/shared.ts +323 -0
- package/src/engine/project/bootstrap-service.ts +9 -0
- package/src/engine/project/bootstrap.ts +75 -0
- package/src/engine/project/instance-context.ts +24 -0
- package/src/engine/project/instance-layer.ts +11 -0
- package/src/engine/project/instance-runtime.ts +16 -0
- package/src/engine/project/instance-store.ts +193 -0
- package/src/engine/project/project.sql.ts +17 -0
- package/src/engine/project/project.ts +537 -0
- package/src/engine/project/schema.ts +13 -0
- package/src/engine/project/vcs.ts +405 -0
- package/src/engine/provider/auth.ts +225 -0
- package/src/engine/provider/error.ts +204 -0
- package/src/engine/provider/model-status.ts +8 -0
- package/src/engine/provider/provider.ts +1843 -0
- package/src/engine/provider/schema.ts +30 -0
- package/src/engine/provider/sdk/copilot/AGENTS.md +1 -0
- package/src/engine/provider/transform.ts +1376 -0
- package/src/engine/pty/index.ts +365 -0
- package/src/engine/pty/input.ts +24 -0
- package/src/engine/pty/pty/index.ts +365 -0
- package/src/engine/pty/pty/input.ts +24 -0
- package/src/engine/pty/pty/pty.bun.ts +26 -0
- package/src/engine/pty/pty/pty.node.ts +27 -0
- package/src/engine/pty/pty/pty.ts +25 -0
- package/src/engine/pty/pty/schema.ts +14 -0
- package/src/engine/pty/pty/ticket.ts +68 -0
- package/src/engine/pty/pty.bun.ts +26 -0
- package/src/engine/pty/pty.node.ts +27 -0
- package/src/engine/pty/pty.ts +25 -0
- package/src/engine/pty/schema.ts +14 -0
- package/src/engine/pty/ticket.ts +68 -0
- package/src/engine/question/index.ts +213 -0
- package/src/engine/question/question/index.ts +213 -0
- package/src/engine/question/question/schema.ts +10 -0
- package/src/engine/question/schema.ts +10 -0
- package/src/engine/reference/reference/reference.ts +241 -0
- package/src/engine/reference/reference/repository-cache.ts +147 -0
- package/src/engine/reference/reference.ts +241 -0
- package/src/engine/reference/repository-cache.ts +147 -0
- package/src/engine/session/compaction.ts +651 -0
- package/src/engine/session/compaction_logic.ts +120 -0
- package/src/engine/session/instruction.ts +238 -0
- package/src/engine/session/instruction_loader.ts +54 -0
- package/src/engine/session/llm.ts +459 -0
- package/src/engine/session/message-error.ts +14 -0
- package/src/engine/session/message-v2.ts +1202 -0
- package/src/engine/session/message.ts +146 -0
- package/src/engine/session/overflow.ts +32 -0
- package/src/engine/session/overflow_check.ts +46 -0
- package/src/engine/session/processor.ts +823 -0
- package/src/engine/session/prompt/anthropic.txt +105 -0
- package/src/engine/session/prompt/beast.txt +147 -0
- package/src/engine/session/prompt/build-switch.txt +5 -0
- package/src/engine/session/prompt/codex.txt +79 -0
- package/src/engine/session/prompt/copilot-gpt-5.txt +143 -0
- package/src/engine/session/prompt/default.txt +105 -0
- package/src/engine/session/prompt/gemini.txt +155 -0
- package/src/engine/session/prompt/gpt.txt +107 -0
- package/src/engine/session/prompt/kimi.txt +95 -0
- package/src/engine/session/prompt/max-steps.txt +16 -0
- package/src/engine/session/prompt/plan-reminder-anthropic.txt +67 -0
- package/src/engine/session/prompt/plan.txt +26 -0
- package/src/engine/session/prompt/trinity.txt +97 -0
- package/src/engine/session/prompt.ts +671 -0
- package/src/engine/session/provider_transform.ts +187 -0
- package/src/engine/session/retry.ts +200 -0
- package/src/engine/session/retry_logic.ts +65 -0
- package/src/engine/session/revert.ts +162 -0
- package/src/engine/session/run-state.ts +153 -0
- package/src/engine/session/schema.ts +26 -0
- package/src/engine/session/session.sql.ts +137 -0
- package/src/engine/session/session.ts +1011 -0
- package/src/engine/session/status.ts +94 -0
- package/src/engine/session/summary.ts +164 -0
- package/src/engine/session/system.ts +84 -0
- package/src/engine/session/system_prompt.ts +65 -0
- package/src/engine/session/todo.ts +81 -0
- package/src/engine/session/tool_registry.ts +162 -0
- package/src/engine/share/session.ts +61 -0
- package/src/engine/share/share-next.ts +376 -0
- package/src/engine/share/share.sql.ts +13 -0
- package/src/engine/shell/shell/shell.ts +215 -0
- package/src/engine/shell/shell.ts +215 -0
- package/src/engine/skill/discovery.ts +116 -0
- package/src/engine/skill/index.ts +336 -0
- package/src/engine/skill/prompt/customize-opencode.md +377 -0
- package/src/engine/skill/skill/discovery.ts +116 -0
- package/src/engine/skill/skill/index.ts +336 -0
- package/src/engine/skill/skill/prompt/customize-opencode.md +377 -0
- package/src/engine/snapshot/index.ts +762 -0
- package/src/engine/snapshot/snapshot/index.ts +762 -0
- package/src/engine/sync/README.md +179 -0
- package/src/engine/sync/event.sql.ts +17 -0
- package/src/engine/sync/index.ts +410 -0
- package/src/engine/sync/schema.ts +11 -0
- package/src/engine/temporary.ts +33 -0
- package/src/engine/tool/apply_patch.ts +313 -0
- package/src/engine/tool/apply_patch.txt +33 -0
- package/src/engine/tool/edit.ts +711 -0
- package/src/engine/tool/edit.txt +10 -0
- package/src/engine/tool/external-directory.ts +49 -0
- package/src/engine/tool/glob.ts +103 -0
- package/src/engine/tool/glob.txt +6 -0
- package/src/engine/tool/grep.ts +156 -0
- package/src/engine/tool/grep.txt +8 -0
- package/src/engine/tool/invalid.ts +21 -0
- package/src/engine/tool/json-schema.ts +164 -0
- package/src/engine/tool/lsp.ts +113 -0
- package/src/engine/tool/lsp.txt +24 -0
- package/src/engine/tool/mcp-websearch.ts +96 -0
- package/src/engine/tool/plan-enter.txt +14 -0
- package/src/engine/tool/plan-exit.txt +13 -0
- package/src/engine/tool/plan.ts +78 -0
- package/src/engine/tool/question.ts +44 -0
- package/src/engine/tool/question.txt +10 -0
- package/src/engine/tool/read.ts +337 -0
- package/src/engine/tool/read.txt +14 -0
- package/src/engine/tool/registry.ts +472 -0
- package/src/engine/tool/repo_clone.ts +80 -0
- package/src/engine/tool/repo_clone.txt +5 -0
- package/src/engine/tool/repo_overview.ts +279 -0
- package/src/engine/tool/repo_overview.txt +4 -0
- package/src/engine/tool/schema.ts +14 -0
- package/src/engine/tool/shell/id.ts +19 -0
- package/src/engine/tool/shell/prompt.ts +295 -0
- package/src/engine/tool/shell/shell.txt +77 -0
- package/src/engine/tool/shell.ts +647 -0
- package/src/engine/tool/skill.ts +75 -0
- package/src/engine/tool/skill.txt +5 -0
- package/src/engine/tool/task.ts +337 -0
- package/src/engine/tool/task.txt +58 -0
- package/src/engine/tool/task_status.ts +179 -0
- package/src/engine/tool/task_status.txt +13 -0
- package/src/engine/tool/todo.ts +57 -0
- package/src/engine/tool/todowrite.txt +167 -0
- package/src/engine/tool/tool/apply_patch.ts +313 -0
- package/src/engine/tool/tool/apply_patch.txt +33 -0
- package/src/engine/tool/tool/edit.ts +711 -0
- package/src/engine/tool/tool/edit.txt +10 -0
- package/src/engine/tool/tool/external-directory.ts +49 -0
- package/src/engine/tool/tool/glob.ts +103 -0
- package/src/engine/tool/tool/glob.txt +6 -0
- package/src/engine/tool/tool/grep.ts +156 -0
- package/src/engine/tool/tool/grep.txt +8 -0
- package/src/engine/tool/tool/invalid.ts +21 -0
- package/src/engine/tool/tool/json-schema.ts +164 -0
- package/src/engine/tool/tool/lsp.ts +113 -0
- package/src/engine/tool/tool/lsp.txt +24 -0
- package/src/engine/tool/tool/mcp-websearch.ts +96 -0
- package/src/engine/tool/tool/plan-enter.txt +14 -0
- package/src/engine/tool/tool/plan-exit.txt +13 -0
- package/src/engine/tool/tool/plan.ts +78 -0
- package/src/engine/tool/tool/question.ts +44 -0
- package/src/engine/tool/tool/question.txt +10 -0
- package/src/engine/tool/tool/read.ts +337 -0
- package/src/engine/tool/tool/read.txt +14 -0
- package/src/engine/tool/tool/registry.ts +472 -0
- package/src/engine/tool/tool/repo_clone.ts +80 -0
- package/src/engine/tool/tool/repo_clone.txt +5 -0
- package/src/engine/tool/tool/repo_overview.ts +279 -0
- package/src/engine/tool/tool/repo_overview.txt +4 -0
- package/src/engine/tool/tool/schema.ts +14 -0
- package/src/engine/tool/tool/shell/id.ts +19 -0
- package/src/engine/tool/tool/shell/prompt.ts +295 -0
- package/src/engine/tool/tool/shell/shell.txt +77 -0
- package/src/engine/tool/tool/shell.ts +647 -0
- package/src/engine/tool/tool/skill.ts +75 -0
- package/src/engine/tool/tool/skill.txt +5 -0
- package/src/engine/tool/tool/task.ts +337 -0
- package/src/engine/tool/tool/task.txt +58 -0
- package/src/engine/tool/tool/task_status.ts +179 -0
- package/src/engine/tool/tool/task_status.txt +13 -0
- package/src/engine/tool/tool/todo.ts +57 -0
- package/src/engine/tool/tool/todowrite.txt +167 -0
- package/src/engine/tool/tool/tool.ts +164 -0
- package/src/engine/tool/tool/truncate.ts +160 -0
- package/src/engine/tool/tool/truncation-dir.ts +4 -0
- package/src/engine/tool/tool/webfetch.ts +192 -0
- package/src/engine/tool/tool/webfetch.txt +13 -0
- package/src/engine/tool/tool/websearch.ts +143 -0
- package/src/engine/tool/tool/websearch.txt +14 -0
- package/src/engine/tool/tool/write.ts +104 -0
- package/src/engine/tool/tool/write.txt +8 -0
- package/src/engine/tool/tool.ts +164 -0
- package/src/engine/tool/truncate.ts +160 -0
- package/src/engine/tool/truncation-dir.ts +4 -0
- package/src/engine/tool/webfetch.ts +192 -0
- package/src/engine/tool/webfetch.txt +13 -0
- package/src/engine/tool/websearch.ts +143 -0
- package/src/engine/tool/websearch.txt +14 -0
- package/src/engine/tool/write.ts +104 -0
- package/src/engine/tool/write.txt +8 -0
- package/src/engine/util/archive.ts +17 -0
- package/src/engine/util/bom.ts +31 -0
- package/src/engine/util/data-url.ts +9 -0
- package/src/engine/util/defer.ts +10 -0
- package/src/engine/util/effect-http-client.ts +11 -0
- package/src/engine/util/error.ts +88 -0
- package/src/engine/util/filesystem.ts +252 -0
- package/src/engine/util/format.ts +20 -0
- package/src/engine/util/iife.ts +3 -0
- package/src/engine/util/lazy.ts +20 -0
- package/src/engine/util/local-context.ts +25 -0
- package/src/engine/util/locale.ts +86 -0
- package/src/engine/util/media.ts +26 -0
- package/src/engine/util/process.ts +176 -0
- package/src/engine/util/queue.ts +32 -0
- package/src/engine/util/record.ts +3 -0
- package/src/engine/util/repository.ts +158 -0
- package/src/engine/util/rpc.ts +66 -0
- package/src/engine/util/signal.ts +12 -0
- package/src/engine/util/timeout.ts +13 -0
- package/src/engine/util/token.ts +7 -0
- package/src/engine/util/util/archive.ts +17 -0
- package/src/engine/util/util/bom.ts +31 -0
- package/src/engine/util/util/data-url.ts +9 -0
- package/src/engine/util/util/defer.ts +10 -0
- package/src/engine/util/util/effect-http-client.ts +11 -0
- package/src/engine/util/util/error.ts +88 -0
- package/src/engine/util/util/filesystem.ts +252 -0
- package/src/engine/util/util/format.ts +20 -0
- package/src/engine/util/util/iife.ts +3 -0
- package/src/engine/util/util/lazy.ts +20 -0
- package/src/engine/util/util/local-context.ts +25 -0
- package/src/engine/util/util/locale.ts +86 -0
- package/src/engine/util/util/media.ts +26 -0
- package/src/engine/util/util/process.ts +176 -0
- package/src/engine/util/util/queue.ts +32 -0
- package/src/engine/util/util/record.ts +3 -0
- package/src/engine/util/util/repository.ts +158 -0
- package/src/engine/util/util/rpc.ts +66 -0
- package/src/engine/util/util/signal.ts +12 -0
- package/src/engine/util/util/timeout.ts +13 -0
- package/src/engine/util/util/token.ts +7 -0
- package/src/engine/util/util/which.ts +14 -0
- package/src/engine/util/util/wildcard.ts +59 -0
- package/src/engine/util/which.ts +14 -0
- package/src/engine/util/wildcard.ts +59 -0
- package/src/engine/worktree/index.ts +621 -0
- package/src/rag_worker.ts +519 -0
- package/src/server.ts +201 -0
- package/src/tui.ts +637 -0
- package/tsconfig.json +24 -0
|
@@ -0,0 +1,188 @@
|
|
|
1
|
+
import path from "path"
|
|
2
|
+
import { fileURLToPath } from "url"
|
|
3
|
+
|
|
4
|
+
import { Flag } from "@opencode-ai/core/flag/flag"
|
|
5
|
+
import { Global } from "@opencode-ai/core/global"
|
|
6
|
+
import { Filesystem } from "@/util/filesystem"
|
|
7
|
+
import { Flock } from "@opencode-ai/core/util/flock"
|
|
8
|
+
|
|
9
|
+
import { parsePluginSpecifier, pluginSource } from "./shared"
|
|
10
|
+
|
|
11
|
+
type Source = "file" | "npm"
|
|
12
|
+
|
|
13
|
+
export type Theme = {
|
|
14
|
+
src: string
|
|
15
|
+
dest: string
|
|
16
|
+
mtime?: number
|
|
17
|
+
size?: number
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
export type Entry = {
|
|
21
|
+
id: string
|
|
22
|
+
source: Source
|
|
23
|
+
spec: string
|
|
24
|
+
target: string
|
|
25
|
+
requested?: string
|
|
26
|
+
version?: string
|
|
27
|
+
modified?: number
|
|
28
|
+
first_time: number
|
|
29
|
+
last_time: number
|
|
30
|
+
time_changed: number
|
|
31
|
+
load_count: number
|
|
32
|
+
fingerprint: string
|
|
33
|
+
themes?: Record<string, Theme>
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
export type State = "first" | "updated" | "same"
|
|
37
|
+
|
|
38
|
+
export type Touch = {
|
|
39
|
+
spec: string
|
|
40
|
+
target: string
|
|
41
|
+
id: string
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
type Store = Record<string, Entry>
|
|
45
|
+
type Core = Omit<Entry, "first_time" | "last_time" | "time_changed" | "load_count" | "fingerprint" | "themes">
|
|
46
|
+
type Row = Touch & { core: Core }
|
|
47
|
+
|
|
48
|
+
function storePath() {
|
|
49
|
+
return Flag.OPENCODE_PLUGIN_META_FILE ?? path.join(Global.Path.state, "plugin-meta.json")
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
function lock(file: string) {
|
|
53
|
+
return `plugin-meta:${file}`
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
function fileTarget(spec: string, target: string) {
|
|
57
|
+
if (spec.startsWith("file://")) return fileURLToPath(spec)
|
|
58
|
+
if (target.startsWith("file://")) return fileURLToPath(target)
|
|
59
|
+
return
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
async function modifiedAt(file: string) {
|
|
63
|
+
const stat = await Filesystem.statAsync(file)
|
|
64
|
+
if (!stat) return
|
|
65
|
+
const mtime = stat.mtimeMs
|
|
66
|
+
return Math.floor(typeof mtime === "bigint" ? Number(mtime) : mtime)
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
function resolvedTarget(target: string) {
|
|
70
|
+
if (target.startsWith("file://")) return fileURLToPath(target)
|
|
71
|
+
return target
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
async function npmVersion(target: string) {
|
|
75
|
+
const resolved = resolvedTarget(target)
|
|
76
|
+
const stat = await Filesystem.statAsync(resolved)
|
|
77
|
+
const dir = stat?.isDirectory() ? resolved : path.dirname(resolved)
|
|
78
|
+
return Filesystem.readJson<{ version?: string }>(path.join(dir, "package.json"))
|
|
79
|
+
.then((item) => item.version)
|
|
80
|
+
.catch(() => undefined)
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
async function entryCore(item: Touch): Promise<Core> {
|
|
84
|
+
const spec = item.spec
|
|
85
|
+
const target = item.target
|
|
86
|
+
const source = pluginSource(spec)
|
|
87
|
+
if (source === "file") {
|
|
88
|
+
const file = fileTarget(spec, target)
|
|
89
|
+
return {
|
|
90
|
+
id: item.id,
|
|
91
|
+
source,
|
|
92
|
+
spec,
|
|
93
|
+
target,
|
|
94
|
+
modified: file ? await modifiedAt(file) : undefined,
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
return {
|
|
99
|
+
id: item.id,
|
|
100
|
+
source,
|
|
101
|
+
spec,
|
|
102
|
+
target,
|
|
103
|
+
requested: parsePluginSpecifier(spec).version,
|
|
104
|
+
version: await npmVersion(target),
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
function fingerprint(value: Core) {
|
|
109
|
+
if (value.source === "file") return [value.target, value.modified ?? ""].join("|")
|
|
110
|
+
return [value.target, value.requested ?? "", value.version ?? ""].join("|")
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
async function read(file: string): Promise<Store> {
|
|
114
|
+
return Filesystem.readJson<Store>(file).catch(() => ({}) as Store)
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
async function row(item: Touch): Promise<Row> {
|
|
118
|
+
return {
|
|
119
|
+
...item,
|
|
120
|
+
core: await entryCore(item),
|
|
121
|
+
}
|
|
122
|
+
}
|
|
123
|
+
|
|
124
|
+
function next(prev: Entry | undefined, core: Core, now: number): { state: State; entry: Entry } {
|
|
125
|
+
const entry: Entry = {
|
|
126
|
+
...core,
|
|
127
|
+
first_time: prev?.first_time ?? now,
|
|
128
|
+
last_time: now,
|
|
129
|
+
time_changed: prev?.time_changed ?? now,
|
|
130
|
+
load_count: (prev?.load_count ?? 0) + 1,
|
|
131
|
+
fingerprint: fingerprint(core),
|
|
132
|
+
themes: prev?.themes,
|
|
133
|
+
}
|
|
134
|
+
const state: State = !prev ? "first" : prev.fingerprint === entry.fingerprint ? "same" : "updated"
|
|
135
|
+
if (state === "updated") entry.time_changed = now
|
|
136
|
+
return {
|
|
137
|
+
state,
|
|
138
|
+
entry,
|
|
139
|
+
}
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
export async function touchMany(items: Touch[]): Promise<Array<{ state: State; entry: Entry }>> {
|
|
143
|
+
if (!items.length) return []
|
|
144
|
+
const file = storePath()
|
|
145
|
+
const rows = await Promise.all(items.map((item) => row(item)))
|
|
146
|
+
|
|
147
|
+
return Flock.withLock(lock(file), async () => {
|
|
148
|
+
const store = await read(file)
|
|
149
|
+
const now = Date.now()
|
|
150
|
+
const out: Array<{ state: State; entry: Entry }> = []
|
|
151
|
+
for (const item of rows) {
|
|
152
|
+
const hit = next(store[item.id], item.core, now)
|
|
153
|
+
store[item.id] = hit.entry
|
|
154
|
+
out.push(hit)
|
|
155
|
+
}
|
|
156
|
+
await Filesystem.writeJson(file, store)
|
|
157
|
+
return out
|
|
158
|
+
})
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
export async function touch(spec: string, target: string, id: string): Promise<{ state: State; entry: Entry }> {
|
|
162
|
+
return touchMany([{ spec, target, id }]).then((item) => {
|
|
163
|
+
const hit = item[0]
|
|
164
|
+
if (hit) return hit
|
|
165
|
+
throw new Error("Failed to touch plugin metadata.")
|
|
166
|
+
})
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
export async function setTheme(id: string, name: string, theme: Theme): Promise<void> {
|
|
170
|
+
const file = storePath()
|
|
171
|
+
await Flock.withLock(lock(file), async () => {
|
|
172
|
+
const store = await read(file)
|
|
173
|
+
const entry = store[id]
|
|
174
|
+
if (!entry) return
|
|
175
|
+
entry.themes = {
|
|
176
|
+
...entry.themes,
|
|
177
|
+
[name]: theme,
|
|
178
|
+
}
|
|
179
|
+
await Filesystem.writeJson(file, store)
|
|
180
|
+
})
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
export async function list(): Promise<Store> {
|
|
184
|
+
const file = storePath()
|
|
185
|
+
return Flock.withLock(lock(file), async () => read(file))
|
|
186
|
+
}
|
|
187
|
+
|
|
188
|
+
export * as PluginMeta from "./meta"
|
|
@@ -0,0 +1,323 @@
|
|
|
1
|
+
import path from "path"
|
|
2
|
+
import { fileURLToPath, pathToFileURL } from "url"
|
|
3
|
+
import npa from "npm-package-arg"
|
|
4
|
+
import semver from "semver"
|
|
5
|
+
import { Filesystem } from "@/util/filesystem"
|
|
6
|
+
import { isRecord } from "@/util/record"
|
|
7
|
+
import { Npm } from "@opencode-ai/core/npm"
|
|
8
|
+
|
|
9
|
+
// Old npm package names for plugins that are now built-in
|
|
10
|
+
export const DEPRECATED_PLUGIN_PACKAGES = ["opencode-openai-codex-auth", "opencode-copilot-auth"]
|
|
11
|
+
|
|
12
|
+
export function isDeprecatedPlugin(spec: string) {
|
|
13
|
+
return DEPRECATED_PLUGIN_PACKAGES.some((pkg) => spec.includes(pkg))
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
function parse(spec: string) {
|
|
17
|
+
try {
|
|
18
|
+
return npa(spec)
|
|
19
|
+
} catch {}
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
export function parsePluginSpecifier(spec: string) {
|
|
23
|
+
const hit = parse(spec)
|
|
24
|
+
if (hit?.type === "alias" && !hit.name) {
|
|
25
|
+
const sub = (hit as npa.AliasResult).subSpec
|
|
26
|
+
if (sub?.name) {
|
|
27
|
+
const version = !sub.rawSpec || sub.rawSpec === "*" ? "latest" : sub.rawSpec
|
|
28
|
+
return { pkg: sub.name, version }
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
if (!hit?.name) return { pkg: spec, version: "" }
|
|
32
|
+
if (hit.raw === hit.name) return { pkg: hit.name, version: "latest" }
|
|
33
|
+
return { pkg: hit.name, version: hit.rawSpec }
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
export type PluginSource = "file" | "npm"
|
|
37
|
+
export type PluginKind = "server" | "tui"
|
|
38
|
+
type PluginMode = "strict" | "detect"
|
|
39
|
+
|
|
40
|
+
export type PluginPackage = {
|
|
41
|
+
dir: string
|
|
42
|
+
pkg: string
|
|
43
|
+
json: Record<string, unknown>
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
export type PluginEntry = {
|
|
47
|
+
spec: string
|
|
48
|
+
source: PluginSource
|
|
49
|
+
target: string
|
|
50
|
+
pkg?: PluginPackage
|
|
51
|
+
entry?: string
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
const INDEX_FILES = ["index.ts", "index.tsx", "index.js", "index.mjs", "index.cjs"]
|
|
55
|
+
|
|
56
|
+
export function pluginSource(spec: string): PluginSource {
|
|
57
|
+
if (isPathPluginSpec(spec)) return "file"
|
|
58
|
+
return "npm"
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
function resolveExportPath(raw: string, dir: string) {
|
|
62
|
+
if (raw.startsWith("file://")) return fileURLToPath(raw)
|
|
63
|
+
if (path.isAbsolute(raw)) return raw
|
|
64
|
+
return path.resolve(dir, raw)
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
function isAbsolutePath(raw: string) {
|
|
68
|
+
return path.isAbsolute(raw) || /^[A-Za-z]:[\\/]/.test(raw)
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
function extractExportValue(value: unknown): string | undefined {
|
|
72
|
+
if (typeof value === "string") return value
|
|
73
|
+
if (!isRecord(value)) return undefined
|
|
74
|
+
for (const key of ["import", "default"]) {
|
|
75
|
+
const nested = value[key]
|
|
76
|
+
if (typeof nested === "string") return nested
|
|
77
|
+
}
|
|
78
|
+
return undefined
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
function packageMain(pkg: PluginPackage) {
|
|
82
|
+
const value = pkg.json.main
|
|
83
|
+
if (typeof value !== "string") return
|
|
84
|
+
const next = value.trim()
|
|
85
|
+
if (!next) return
|
|
86
|
+
return next
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
function resolvePackageFile(spec: string, raw: string, kind: string, pkg: PluginPackage) {
|
|
90
|
+
const resolved = resolveExportPath(raw, pkg.dir)
|
|
91
|
+
const root = Filesystem.resolve(pkg.dir)
|
|
92
|
+
const next = Filesystem.resolve(resolved)
|
|
93
|
+
if (!Filesystem.contains(root, next)) {
|
|
94
|
+
throw new Error(`Plugin ${spec} resolved ${kind} entry outside plugin directory`)
|
|
95
|
+
}
|
|
96
|
+
return next
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
function resolvePackagePath(spec: string, raw: string, kind: PluginKind, pkg: PluginPackage) {
|
|
100
|
+
return pathToFileURL(resolvePackageFile(spec, raw, kind, pkg)).href
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
function resolvePackageEntrypoint(spec: string, kind: PluginKind, pkg: PluginPackage) {
|
|
104
|
+
const exports = pkg.json.exports
|
|
105
|
+
if (isRecord(exports)) {
|
|
106
|
+
const raw = extractExportValue(exports[`./${kind}`])
|
|
107
|
+
if (raw) return resolvePackagePath(spec, raw, kind, pkg)
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
if (kind !== "server") return
|
|
111
|
+
const main = packageMain(pkg)
|
|
112
|
+
if (!main) return
|
|
113
|
+
return resolvePackagePath(spec, main, kind, pkg)
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
function targetPath(target: string) {
|
|
117
|
+
if (target.startsWith("file://")) return fileURLToPath(target)
|
|
118
|
+
if (path.isAbsolute(target)) return target
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
async function resolveDirectoryIndex(dir: string) {
|
|
122
|
+
for (const name of INDEX_FILES) {
|
|
123
|
+
const file = path.join(dir, name)
|
|
124
|
+
if (await Filesystem.exists(file)) return file
|
|
125
|
+
}
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
async function resolveTargetDirectory(target: string) {
|
|
129
|
+
const file = targetPath(target)
|
|
130
|
+
if (!file) return
|
|
131
|
+
const stat = await Filesystem.statAsync(file)
|
|
132
|
+
if (!stat?.isDirectory()) return
|
|
133
|
+
return file
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
async function resolvePluginEntrypoint(spec: string, target: string, kind: PluginKind, pkg?: PluginPackage) {
|
|
137
|
+
const source = pluginSource(spec)
|
|
138
|
+
const hit =
|
|
139
|
+
pkg ?? (source === "npm" ? await readPluginPackage(target) : await readPluginPackage(target).catch(() => undefined))
|
|
140
|
+
if (!hit) return target
|
|
141
|
+
|
|
142
|
+
const entry = resolvePackageEntrypoint(spec, kind, hit)
|
|
143
|
+
if (entry) return entry
|
|
144
|
+
|
|
145
|
+
const dir = await resolveTargetDirectory(target)
|
|
146
|
+
|
|
147
|
+
if (kind === "tui") {
|
|
148
|
+
if (source === "file" && dir) {
|
|
149
|
+
const index = await resolveDirectoryIndex(dir)
|
|
150
|
+
if (index) return pathToFileURL(index).href
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
if (source === "npm") return
|
|
154
|
+
if (dir) return
|
|
155
|
+
|
|
156
|
+
return target
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
if (dir && isRecord(hit.json.exports)) {
|
|
160
|
+
if (source === "file") {
|
|
161
|
+
const index = await resolveDirectoryIndex(dir)
|
|
162
|
+
if (index) return pathToFileURL(index).href
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
return
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
return target
|
|
169
|
+
}
|
|
170
|
+
|
|
171
|
+
export function isPathPluginSpec(spec: string) {
|
|
172
|
+
return spec.startsWith("file://") || spec.startsWith(".") || isAbsolutePath(spec)
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
export async function resolvePathPluginTarget(spec: string) {
|
|
176
|
+
const raw = spec.startsWith("file://") ? fileURLToPath(spec) : spec
|
|
177
|
+
const file = path.isAbsolute(raw) || /^[A-Za-z]:[\\/]/.test(raw) ? raw : path.resolve(raw)
|
|
178
|
+
const stat = await Filesystem.statAsync(file)
|
|
179
|
+
if (!stat?.isDirectory()) {
|
|
180
|
+
if (spec.startsWith("file://")) return spec
|
|
181
|
+
return pathToFileURL(file).href
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
if (await Filesystem.exists(path.join(file, "package.json"))) {
|
|
185
|
+
return pathToFileURL(file).href
|
|
186
|
+
}
|
|
187
|
+
|
|
188
|
+
const index = await resolveDirectoryIndex(file)
|
|
189
|
+
if (index) return pathToFileURL(index).href
|
|
190
|
+
|
|
191
|
+
throw new Error(`Plugin directory ${file} is missing package.json or index file`)
|
|
192
|
+
}
|
|
193
|
+
|
|
194
|
+
export async function checkPluginCompatibility(target: string, opencodeVersion: string, pkg?: PluginPackage) {
|
|
195
|
+
if (!semver.valid(opencodeVersion) || semver.major(opencodeVersion) === 0) return
|
|
196
|
+
const hit = pkg ?? (await readPluginPackage(target).catch(() => undefined))
|
|
197
|
+
if (!hit) return
|
|
198
|
+
const engines = hit.json.engines
|
|
199
|
+
if (!isRecord(engines)) return
|
|
200
|
+
const range = engines.opencode
|
|
201
|
+
if (typeof range !== "string") return
|
|
202
|
+
if (!semver.satisfies(opencodeVersion, range)) {
|
|
203
|
+
throw new Error(`Plugin requires opencode ${range} but running ${opencodeVersion}`)
|
|
204
|
+
}
|
|
205
|
+
}
|
|
206
|
+
|
|
207
|
+
export async function resolvePluginTarget(spec: string) {
|
|
208
|
+
if (isPathPluginSpec(spec)) return resolvePathPluginTarget(spec)
|
|
209
|
+
const hit = parse(spec)
|
|
210
|
+
const pkg = hit?.name && hit.raw === hit.name ? `${hit.name}@latest` : spec
|
|
211
|
+
const result = await Npm.add(pkg)
|
|
212
|
+
return result.directory
|
|
213
|
+
}
|
|
214
|
+
|
|
215
|
+
export async function readPluginPackage(target: string): Promise<PluginPackage> {
|
|
216
|
+
const file = target.startsWith("file://") ? fileURLToPath(target) : target
|
|
217
|
+
const stat = await Filesystem.statAsync(file)
|
|
218
|
+
const dir = stat?.isDirectory() ? file : path.dirname(file)
|
|
219
|
+
const pkg = path.join(dir, "package.json")
|
|
220
|
+
const json = await Filesystem.readJson<Record<string, unknown>>(pkg)
|
|
221
|
+
return { dir, pkg, json }
|
|
222
|
+
}
|
|
223
|
+
|
|
224
|
+
export async function createPluginEntry(spec: string, target: string, kind: PluginKind): Promise<PluginEntry> {
|
|
225
|
+
const source = pluginSource(spec)
|
|
226
|
+
const pkg =
|
|
227
|
+
source === "npm" ? await readPluginPackage(target) : await readPluginPackage(target).catch(() => undefined)
|
|
228
|
+
const entry = await resolvePluginEntrypoint(spec, target, kind, pkg)
|
|
229
|
+
return {
|
|
230
|
+
spec,
|
|
231
|
+
source,
|
|
232
|
+
target,
|
|
233
|
+
pkg,
|
|
234
|
+
entry,
|
|
235
|
+
}
|
|
236
|
+
}
|
|
237
|
+
|
|
238
|
+
export function readPackageThemes(spec: string, pkg: PluginPackage) {
|
|
239
|
+
const field = pkg.json["oc-themes"]
|
|
240
|
+
if (field === undefined) return []
|
|
241
|
+
if (!Array.isArray(field)) {
|
|
242
|
+
throw new TypeError(`Plugin ${spec} has invalid oc-themes field`)
|
|
243
|
+
}
|
|
244
|
+
|
|
245
|
+
const list = field.map((item) => {
|
|
246
|
+
if (typeof item !== "string") {
|
|
247
|
+
throw new TypeError(`Plugin ${spec} has invalid oc-themes entry`)
|
|
248
|
+
}
|
|
249
|
+
|
|
250
|
+
const raw = item.trim()
|
|
251
|
+
if (!raw) {
|
|
252
|
+
throw new TypeError(`Plugin ${spec} has empty oc-themes entry`)
|
|
253
|
+
}
|
|
254
|
+
if (raw.startsWith("file://") || isAbsolutePath(raw)) {
|
|
255
|
+
throw new TypeError(`Plugin ${spec} oc-themes entry must be relative: ${item}`)
|
|
256
|
+
}
|
|
257
|
+
|
|
258
|
+
return resolvePackageFile(spec, raw, "oc-themes", pkg)
|
|
259
|
+
})
|
|
260
|
+
|
|
261
|
+
return Array.from(new Set(list))
|
|
262
|
+
}
|
|
263
|
+
|
|
264
|
+
export function readPluginId(id: unknown, spec: string) {
|
|
265
|
+
if (id === undefined) return
|
|
266
|
+
if (typeof id !== "string") throw new TypeError(`Plugin ${spec} has invalid id type ${typeof id}`)
|
|
267
|
+
const value = id.trim()
|
|
268
|
+
if (!value) throw new TypeError(`Plugin ${spec} has an empty id`)
|
|
269
|
+
return value
|
|
270
|
+
}
|
|
271
|
+
|
|
272
|
+
export function readV1Plugin(
|
|
273
|
+
mod: Record<string, unknown>,
|
|
274
|
+
spec: string,
|
|
275
|
+
kind: PluginKind,
|
|
276
|
+
mode: PluginMode = "strict",
|
|
277
|
+
) {
|
|
278
|
+
const value = mod.default
|
|
279
|
+
if (!isRecord(value)) {
|
|
280
|
+
if (mode === "detect") return
|
|
281
|
+
throw new TypeError(`Plugin ${spec} must default export an object with ${kind}()`)
|
|
282
|
+
}
|
|
283
|
+
if (mode === "detect" && !("id" in value) && !("server" in value) && !("tui" in value)) return
|
|
284
|
+
|
|
285
|
+
const server = "server" in value ? value.server : undefined
|
|
286
|
+
const tui = "tui" in value ? value.tui : undefined
|
|
287
|
+
if (server !== undefined && typeof server !== "function") {
|
|
288
|
+
throw new TypeError(`Plugin ${spec} has invalid server export`)
|
|
289
|
+
}
|
|
290
|
+
if (tui !== undefined && typeof tui !== "function") {
|
|
291
|
+
throw new TypeError(`Plugin ${spec} has invalid tui export`)
|
|
292
|
+
}
|
|
293
|
+
if (server !== undefined && tui !== undefined) {
|
|
294
|
+
throw new TypeError(`Plugin ${spec} must default export either server() or tui(), not both`)
|
|
295
|
+
}
|
|
296
|
+
if (kind === "server" && server === undefined) {
|
|
297
|
+
throw new TypeError(`Plugin ${spec} must default export an object with server()`)
|
|
298
|
+
}
|
|
299
|
+
if (kind === "tui" && tui === undefined) {
|
|
300
|
+
throw new TypeError(`Plugin ${spec} must default export an object with tui()`)
|
|
301
|
+
}
|
|
302
|
+
|
|
303
|
+
return value
|
|
304
|
+
}
|
|
305
|
+
|
|
306
|
+
export async function resolvePluginId(
|
|
307
|
+
source: PluginSource,
|
|
308
|
+
spec: string,
|
|
309
|
+
target: string,
|
|
310
|
+
id: string | undefined,
|
|
311
|
+
pkg?: PluginPackage,
|
|
312
|
+
) {
|
|
313
|
+
if (source === "file") {
|
|
314
|
+
if (id) return id
|
|
315
|
+
throw new TypeError(`Path plugin ${spec} must export id`)
|
|
316
|
+
}
|
|
317
|
+
if (id) return id
|
|
318
|
+
const hit = pkg ?? (await readPluginPackage(target))
|
|
319
|
+
if (typeof hit.json.name !== "string" || !hit.json.name.trim()) {
|
|
320
|
+
throw new TypeError(`Plugin package ${hit.pkg} is missing name`)
|
|
321
|
+
}
|
|
322
|
+
return hit.json.name.trim()
|
|
323
|
+
}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
import { Context, Effect } from "effect"
|
|
2
|
+
|
|
3
|
+
export interface Interface {
|
|
4
|
+
readonly run: Effect.Effect<void>
|
|
5
|
+
}
|
|
6
|
+
|
|
7
|
+
export class Service extends Context.Service<Service, Interface>()("@opencode/InstanceBootstrap") {}
|
|
8
|
+
|
|
9
|
+
export * as InstanceBootstrap from "./bootstrap-service"
|
|
@@ -0,0 +1,75 @@
|
|
|
1
|
+
import { Plugin } from "../plugin"
|
|
2
|
+
import { Format } from "../format"
|
|
3
|
+
import { LSP } from "@/lsp/lsp"
|
|
4
|
+
import { File } from "../file"
|
|
5
|
+
import { Snapshot } from "../snapshot"
|
|
6
|
+
import * as Project from "./project"
|
|
7
|
+
import * as Vcs from "./vcs"
|
|
8
|
+
import { Bus } from "../bus"
|
|
9
|
+
import { InstanceState } from "@/effect/instance-state"
|
|
10
|
+
import { FileWatcher } from "@/file/watcher"
|
|
11
|
+
import { ShareNext } from "@/share/share-next"
|
|
12
|
+
import { Effect, Layer } from "effect"
|
|
13
|
+
import { Config } from "@/config/config"
|
|
14
|
+
import { Service } from "./bootstrap-service"
|
|
15
|
+
import { Reference } from "@/reference/reference"
|
|
16
|
+
|
|
17
|
+
export { Service } from "./bootstrap-service"
|
|
18
|
+
export type { Interface } from "./bootstrap-service"
|
|
19
|
+
|
|
20
|
+
export const layer = Layer.effect(
|
|
21
|
+
Service,
|
|
22
|
+
Effect.gen(function* () {
|
|
23
|
+
// Yield each bootstrap dep at layer init so `run` itself has R = never.
|
|
24
|
+
// InstanceStore imports only the lightweight tag from bootstrap-service.ts,
|
|
25
|
+
// so it can depend on bootstrap without importing this implementation graph.
|
|
26
|
+
const config = yield* Config.Service
|
|
27
|
+
const file = yield* File.Service
|
|
28
|
+
const fileWatcher = yield* FileWatcher.Service
|
|
29
|
+
const format = yield* Format.Service
|
|
30
|
+
const lsp = yield* LSP.Service
|
|
31
|
+
const plugin = yield* Plugin.Service
|
|
32
|
+
const project = yield* Project.Service
|
|
33
|
+
const reference = yield* Reference.Service
|
|
34
|
+
const shareNext = yield* ShareNext.Service
|
|
35
|
+
const snapshot = yield* Snapshot.Service
|
|
36
|
+
const vcs = yield* Vcs.Service
|
|
37
|
+
|
|
38
|
+
const run = Effect.gen(function* () {
|
|
39
|
+
const ctx = yield* InstanceState.context
|
|
40
|
+
yield* Effect.logInfo("bootstrapping").pipe(Effect.annotateLogs("directory", ctx.directory))
|
|
41
|
+
// everything depends on config so eager load it for nice traces
|
|
42
|
+
yield* config.get()
|
|
43
|
+
// Plugin can mutate config so it has to be initialized before anything else.
|
|
44
|
+
yield* plugin.init()
|
|
45
|
+
// Each service self-manages its own slow work via Effect.forkScoped against
|
|
46
|
+
// its per-instance state scope. We just await materialization here.
|
|
47
|
+
yield* Effect.forEach(
|
|
48
|
+
[reference, lsp, shareNext, format, file, fileWatcher, vcs, snapshot, project],
|
|
49
|
+
(s) => s.init().pipe(Effect.catchCause((cause) => Effect.logWarning("init failed", { cause }))),
|
|
50
|
+
{ concurrency: "unbounded", discard: true },
|
|
51
|
+
).pipe(Effect.withSpan("InstanceBootstrap.init"))
|
|
52
|
+
}).pipe(Effect.withSpan("InstanceBootstrap"))
|
|
53
|
+
|
|
54
|
+
return Service.of({ run })
|
|
55
|
+
}),
|
|
56
|
+
)
|
|
57
|
+
|
|
58
|
+
export const defaultLayer: Layer.Layer<Service> = layer.pipe(
|
|
59
|
+
Layer.provide([
|
|
60
|
+
Bus.layer,
|
|
61
|
+
Config.defaultLayer,
|
|
62
|
+
File.defaultLayer,
|
|
63
|
+
FileWatcher.defaultLayer,
|
|
64
|
+
Format.defaultLayer,
|
|
65
|
+
LSP.defaultLayer,
|
|
66
|
+
Plugin.defaultLayer,
|
|
67
|
+
Project.defaultLayer,
|
|
68
|
+
Reference.defaultLayer,
|
|
69
|
+
ShareNext.defaultLayer,
|
|
70
|
+
Snapshot.defaultLayer,
|
|
71
|
+
Vcs.defaultLayer,
|
|
72
|
+
]),
|
|
73
|
+
)
|
|
74
|
+
|
|
75
|
+
export * as InstanceBootstrap from "./bootstrap"
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import { LocalContext } from "@/util/local-context"
|
|
2
|
+
import { AppFileSystem } from "@opencode-ai/core/filesystem"
|
|
3
|
+
import type * as Project from "./project"
|
|
4
|
+
|
|
5
|
+
export interface InstanceContext {
|
|
6
|
+
directory: string
|
|
7
|
+
worktree: string
|
|
8
|
+
project: Project.Info
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
export const context = LocalContext.create<InstanceContext>("instance")
|
|
12
|
+
|
|
13
|
+
/**
|
|
14
|
+
* Check if a path is within the project boundary.
|
|
15
|
+
* Returns true if path is inside ctx.directory OR ctx.worktree.
|
|
16
|
+
* Paths within the worktree but outside the working directory should not trigger external_directory permission.
|
|
17
|
+
*/
|
|
18
|
+
export function containsPath(filepath: string, ctx: InstanceContext): boolean {
|
|
19
|
+
if (AppFileSystem.contains(ctx.directory, filepath)) return true
|
|
20
|
+
// Non-git projects set worktree to "/" which would match ANY absolute path.
|
|
21
|
+
// Skip worktree check in this case to preserve external_directory permissions.
|
|
22
|
+
if (ctx.worktree === "/") return false
|
|
23
|
+
return AppFileSystem.contains(ctx.worktree, filepath)
|
|
24
|
+
}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import { Effect, Layer } from "effect"
|
|
2
|
+
import { InstanceStore } from "./instance-store"
|
|
3
|
+
|
|
4
|
+
export const layer = Layer.unwrap(
|
|
5
|
+
Effect.promise(async () => {
|
|
6
|
+
const { InstanceBootstrap } = await import("./bootstrap")
|
|
7
|
+
return InstanceStore.defaultLayer.pipe(Layer.provide(InstanceBootstrap.defaultLayer))
|
|
8
|
+
}),
|
|
9
|
+
)
|
|
10
|
+
|
|
11
|
+
export * as InstanceLayer from "./instance-layer"
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
import { AppRuntime } from "@/effect/app-runtime"
|
|
2
|
+
import { type InstanceContext } from "./instance-context"
|
|
3
|
+
import { InstanceStore, type LoadInput } from "./instance-store"
|
|
4
|
+
|
|
5
|
+
// Bridge for Promise/ALS callers that cannot yet yield InstanceStore.Service.
|
|
6
|
+
// Delete this module once those callers are migrated to Effect boundaries that
|
|
7
|
+
// provide InstanceStore directly.
|
|
8
|
+
|
|
9
|
+
export const load = (input: LoadInput) => AppRuntime.runPromise(InstanceStore.Service.use((store) => store.load(input)))
|
|
10
|
+
export const disposeInstance = (ctx: InstanceContext) =>
|
|
11
|
+
AppRuntime.runPromise(InstanceStore.Service.use((store) => store.dispose(ctx)))
|
|
12
|
+
export const disposeAllInstances = () => AppRuntime.runPromise(InstanceStore.Service.use((store) => store.disposeAll()))
|
|
13
|
+
export const reloadInstance = (input: LoadInput) =>
|
|
14
|
+
AppRuntime.runPromise(InstanceStore.Service.use((store) => store.reload(input)))
|
|
15
|
+
|
|
16
|
+
export * as InstanceRuntime from "./instance-runtime"
|