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,10 @@
|
|
|
1
|
+
Performs exact string replacements in files.
|
|
2
|
+
|
|
3
|
+
Usage:
|
|
4
|
+
- You must use your `Read` tool at least once in the conversation before editing. This tool will error if you attempt an edit without reading the file.
|
|
5
|
+
- When editing text from Read tool output, ensure you preserve the exact indentation (tabs/spaces) as it appears AFTER the line number prefix. The line number prefix format is: line number + colon + space (e.g., `1: `). Everything after that space is the actual file content to match. Never include any part of the line number prefix in the oldString or newString.
|
|
6
|
+
- ALWAYS prefer editing existing files in the codebase. NEVER write new files unless explicitly required.
|
|
7
|
+
- Only use emojis if the user explicitly requests it. Avoid adding emojis to files unless asked.
|
|
8
|
+
- The edit will FAIL if `oldString` is not found in the file with an error "oldString not found in content".
|
|
9
|
+
- The edit will FAIL if `oldString` is found multiple times in the file with an error "Found multiple matches for oldString. Provide more surrounding lines in oldString to identify the correct match." Either provide a larger string with more surrounding context to make it unique or use `replaceAll` to change every instance of `oldString`.
|
|
10
|
+
- Use `replaceAll` for replacing and renaming strings across the file. This parameter is useful if you want to rename a variable for instance.
|
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
import path from "path"
|
|
2
|
+
import { Effect } from "effect"
|
|
3
|
+
import * as EffectLogger from "@opencode-ai/core/effect/logger"
|
|
4
|
+
import { InstanceState } from "@/effect/instance-state"
|
|
5
|
+
import type * as Tool from "./tool"
|
|
6
|
+
import { containsPath } from "../project/instance-context"
|
|
7
|
+
import { AppFileSystem } from "@opencode-ai/core/filesystem"
|
|
8
|
+
|
|
9
|
+
type Kind = "file" | "directory"
|
|
10
|
+
|
|
11
|
+
type Options = {
|
|
12
|
+
bypass?: boolean
|
|
13
|
+
kind?: Kind
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
export const assertExternalDirectoryEffect = Effect.fn("Tool.assertExternalDirectory")(function* (
|
|
17
|
+
ctx: Tool.Context,
|
|
18
|
+
target?: string,
|
|
19
|
+
options?: Options,
|
|
20
|
+
) {
|
|
21
|
+
if (!target) return
|
|
22
|
+
|
|
23
|
+
if (options?.bypass) return
|
|
24
|
+
|
|
25
|
+
const ins = yield* InstanceState.context
|
|
26
|
+
const full = process.platform === "win32" ? AppFileSystem.normalizePath(target) : target
|
|
27
|
+
if (containsPath(full, ins)) return
|
|
28
|
+
|
|
29
|
+
const kind = options?.kind ?? "file"
|
|
30
|
+
const dir = kind === "directory" ? full : path.dirname(full)
|
|
31
|
+
const glob =
|
|
32
|
+
process.platform === "win32"
|
|
33
|
+
? AppFileSystem.normalizePathPattern(path.join(dir, "*"))
|
|
34
|
+
: path.join(dir, "*").replaceAll("\\", "/")
|
|
35
|
+
|
|
36
|
+
yield* ctx.ask({
|
|
37
|
+
permission: "external_directory",
|
|
38
|
+
patterns: [glob],
|
|
39
|
+
always: [glob],
|
|
40
|
+
metadata: {
|
|
41
|
+
filepath: full,
|
|
42
|
+
parentDir: dir,
|
|
43
|
+
},
|
|
44
|
+
})
|
|
45
|
+
})
|
|
46
|
+
|
|
47
|
+
export async function assertExternalDirectory(ctx: Tool.Context, target?: string, options?: Options) {
|
|
48
|
+
return Effect.runPromise(assertExternalDirectoryEffect(ctx, target, options).pipe(Effect.provide(EffectLogger.layer)))
|
|
49
|
+
}
|
|
@@ -0,0 +1,103 @@
|
|
|
1
|
+
import path from "path"
|
|
2
|
+
import { Effect, Option, Schema } from "effect"
|
|
3
|
+
import * as Stream from "effect/Stream"
|
|
4
|
+
import { InstanceState } from "@/effect/instance-state"
|
|
5
|
+
import { AppFileSystem } from "@opencode-ai/core/filesystem"
|
|
6
|
+
import { Ripgrep } from "../file/ripgrep"
|
|
7
|
+
import { assertExternalDirectoryEffect } from "./external-directory"
|
|
8
|
+
import DESCRIPTION from "./glob.txt"
|
|
9
|
+
import * as Tool from "./tool"
|
|
10
|
+
import { Reference } from "@/reference/reference"
|
|
11
|
+
|
|
12
|
+
export const Parameters = Schema.Struct({
|
|
13
|
+
pattern: Schema.String.annotate({ description: "The glob pattern to match files against" }),
|
|
14
|
+
path: Schema.optional(Schema.String).annotate({
|
|
15
|
+
description: `The directory to search in. If not specified, the current working directory will be used. IMPORTANT: Omit this field to use the default directory. DO NOT enter "undefined" or "null" - simply omit it for the default behavior. Must be a valid directory path if provided.`,
|
|
16
|
+
}),
|
|
17
|
+
})
|
|
18
|
+
|
|
19
|
+
export const GlobTool = Tool.define(
|
|
20
|
+
"glob",
|
|
21
|
+
Effect.gen(function* () {
|
|
22
|
+
const rg = yield* Ripgrep.Service
|
|
23
|
+
const fs = yield* AppFileSystem.Service
|
|
24
|
+
const reference = yield* Reference.Service
|
|
25
|
+
|
|
26
|
+
return {
|
|
27
|
+
description: DESCRIPTION,
|
|
28
|
+
parameters: Parameters,
|
|
29
|
+
execute: (params: { pattern: string; path?: string }, ctx: Tool.Context) =>
|
|
30
|
+
Effect.gen(function* () {
|
|
31
|
+
const ins = yield* InstanceState.context
|
|
32
|
+
yield* ctx.ask({
|
|
33
|
+
permission: "glob",
|
|
34
|
+
patterns: [params.pattern],
|
|
35
|
+
always: ["*"],
|
|
36
|
+
metadata: {
|
|
37
|
+
pattern: params.pattern,
|
|
38
|
+
path: params.path,
|
|
39
|
+
},
|
|
40
|
+
})
|
|
41
|
+
|
|
42
|
+
let search = params.path ?? ins.directory
|
|
43
|
+
search = path.isAbsolute(search) ? search : path.resolve(ins.directory, search)
|
|
44
|
+
yield* reference.ensure(search)
|
|
45
|
+
const info = yield* fs.stat(search).pipe(Effect.catch(() => Effect.succeed(undefined)))
|
|
46
|
+
if (info?.type === "File") {
|
|
47
|
+
throw new Error(`glob path must be a directory: ${search}`)
|
|
48
|
+
}
|
|
49
|
+
yield* assertExternalDirectoryEffect(ctx, search, {
|
|
50
|
+
bypass: yield* reference.contains(search),
|
|
51
|
+
kind: "directory",
|
|
52
|
+
})
|
|
53
|
+
|
|
54
|
+
const limit = 100
|
|
55
|
+
let truncated = false
|
|
56
|
+
const files = yield* rg.files({ cwd: search, glob: [params.pattern], signal: ctx.abort }).pipe(
|
|
57
|
+
Stream.mapEffect((file) =>
|
|
58
|
+
Effect.gen(function* () {
|
|
59
|
+
const full = path.resolve(search, file)
|
|
60
|
+
const info = yield* fs.stat(full).pipe(Effect.catch(() => Effect.succeed(undefined)))
|
|
61
|
+
const mtime =
|
|
62
|
+
info?.mtime.pipe(
|
|
63
|
+
Option.map((date) => date.getTime()),
|
|
64
|
+
Option.getOrElse(() => 0),
|
|
65
|
+
) ?? 0
|
|
66
|
+
return { path: full, mtime }
|
|
67
|
+
}),
|
|
68
|
+
),
|
|
69
|
+
Stream.take(limit + 1),
|
|
70
|
+
Stream.runCollect,
|
|
71
|
+
Effect.map((chunk) => [...chunk]),
|
|
72
|
+
)
|
|
73
|
+
|
|
74
|
+
if (files.length > limit) {
|
|
75
|
+
truncated = true
|
|
76
|
+
files.length = limit
|
|
77
|
+
}
|
|
78
|
+
files.sort((a, b) => b.mtime - a.mtime)
|
|
79
|
+
|
|
80
|
+
const output = []
|
|
81
|
+
if (files.length === 0) output.push("No files found")
|
|
82
|
+
if (files.length > 0) {
|
|
83
|
+
output.push(...files.map((file) => file.path))
|
|
84
|
+
if (truncated) {
|
|
85
|
+
output.push("")
|
|
86
|
+
output.push(
|
|
87
|
+
`(Results are truncated: showing first ${limit} results. Consider using a more specific path or pattern.)`,
|
|
88
|
+
)
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
return {
|
|
93
|
+
title: path.relative(ins.worktree, search),
|
|
94
|
+
metadata: {
|
|
95
|
+
count: files.length,
|
|
96
|
+
truncated,
|
|
97
|
+
},
|
|
98
|
+
output: output.join("\n"),
|
|
99
|
+
}
|
|
100
|
+
}).pipe(Effect.orDie),
|
|
101
|
+
}
|
|
102
|
+
}),
|
|
103
|
+
)
|
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
- Fast file pattern matching tool that works with any codebase size
|
|
2
|
+
- Supports glob patterns like "**/*.js" or "src/**/*.ts"
|
|
3
|
+
- Returns matching file paths sorted by modification time
|
|
4
|
+
- Use this tool when you need to find files by name patterns
|
|
5
|
+
- When you are doing an open-ended search that may require multiple rounds of globbing and grepping, use the Task tool instead
|
|
6
|
+
- You have the capability to call multiple tools in a single response. It is always better to speculatively perform multiple searches as a batch that are potentially useful.
|
|
@@ -0,0 +1,156 @@
|
|
|
1
|
+
import path from "path"
|
|
2
|
+
import { Schema } from "effect"
|
|
3
|
+
import { Effect, Option } from "effect"
|
|
4
|
+
import { InstanceState } from "@/effect/instance-state"
|
|
5
|
+
import { AppFileSystem } from "@opencode-ai/core/filesystem"
|
|
6
|
+
import { Ripgrep } from "../file/ripgrep"
|
|
7
|
+
import { assertExternalDirectoryEffect } from "./external-directory"
|
|
8
|
+
import DESCRIPTION from "./grep.txt"
|
|
9
|
+
import * as Tool from "./tool"
|
|
10
|
+
import { Reference } from "@/reference/reference"
|
|
11
|
+
|
|
12
|
+
const MAX_LINE_LENGTH = 2000
|
|
13
|
+
|
|
14
|
+
export const Parameters = Schema.Struct({
|
|
15
|
+
pattern: Schema.String.annotate({ description: "The regex pattern to search for in file contents" }),
|
|
16
|
+
path: Schema.optional(Schema.String).annotate({
|
|
17
|
+
description: "The directory to search in. Defaults to the current working directory.",
|
|
18
|
+
}),
|
|
19
|
+
include: Schema.optional(Schema.String).annotate({
|
|
20
|
+
description: 'File pattern to include in the search (e.g. "*.js", "*.{ts,tsx}")',
|
|
21
|
+
}),
|
|
22
|
+
})
|
|
23
|
+
|
|
24
|
+
export const GrepTool = Tool.define(
|
|
25
|
+
"grep",
|
|
26
|
+
Effect.gen(function* () {
|
|
27
|
+
const fs = yield* AppFileSystem.Service
|
|
28
|
+
const rg = yield* Ripgrep.Service
|
|
29
|
+
const reference = yield* Reference.Service
|
|
30
|
+
|
|
31
|
+
return {
|
|
32
|
+
description: DESCRIPTION,
|
|
33
|
+
parameters: Parameters,
|
|
34
|
+
execute: (params: { pattern: string; path?: string; include?: string }, ctx: Tool.Context) =>
|
|
35
|
+
Effect.gen(function* () {
|
|
36
|
+
const empty = {
|
|
37
|
+
title: params.pattern,
|
|
38
|
+
metadata: { matches: 0, truncated: false },
|
|
39
|
+
output: "No files found",
|
|
40
|
+
}
|
|
41
|
+
if (!params.pattern) {
|
|
42
|
+
throw new Error("pattern is required")
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
yield* ctx.ask({
|
|
46
|
+
permission: "grep",
|
|
47
|
+
patterns: [params.pattern],
|
|
48
|
+
always: ["*"],
|
|
49
|
+
metadata: {
|
|
50
|
+
pattern: params.pattern,
|
|
51
|
+
path: params.path,
|
|
52
|
+
include: params.include,
|
|
53
|
+
},
|
|
54
|
+
})
|
|
55
|
+
|
|
56
|
+
const ins = yield* InstanceState.context
|
|
57
|
+
const requested = path.isAbsolute(params.path ?? ins.directory)
|
|
58
|
+
? (params.path ?? ins.directory)
|
|
59
|
+
: path.join(ins.directory, params.path ?? ".")
|
|
60
|
+
yield* reference.ensure(requested)
|
|
61
|
+
const requestedInfo = yield* fs.stat(requested).pipe(Effect.catch(() => Effect.succeed(undefined)))
|
|
62
|
+
yield* assertExternalDirectoryEffect(ctx, requested, {
|
|
63
|
+
bypass: yield* reference.contains(requested),
|
|
64
|
+
kind: requestedInfo?.type === "Directory" ? "directory" : "file",
|
|
65
|
+
})
|
|
66
|
+
|
|
67
|
+
const search = AppFileSystem.resolve(requested)
|
|
68
|
+
const info = yield* fs.stat(search).pipe(Effect.catch(() => Effect.succeed(undefined)))
|
|
69
|
+
const cwd = info?.type === "Directory" ? search : path.dirname(search)
|
|
70
|
+
const file = info?.type === "Directory" ? undefined : [path.relative(cwd, search)]
|
|
71
|
+
|
|
72
|
+
const result = yield* rg.search({
|
|
73
|
+
cwd,
|
|
74
|
+
pattern: params.pattern,
|
|
75
|
+
glob: params.include ? [params.include] : undefined,
|
|
76
|
+
file,
|
|
77
|
+
signal: ctx.abort,
|
|
78
|
+
})
|
|
79
|
+
if (result.items.length === 0) return empty
|
|
80
|
+
|
|
81
|
+
const rows = result.items.map((item) => ({
|
|
82
|
+
path: AppFileSystem.resolve(
|
|
83
|
+
path.isAbsolute(item.path.text) ? item.path.text : path.join(cwd, item.path.text),
|
|
84
|
+
),
|
|
85
|
+
line: item.line_number,
|
|
86
|
+
text: item.lines.text,
|
|
87
|
+
}))
|
|
88
|
+
const times = new Map(
|
|
89
|
+
(yield* Effect.forEach(
|
|
90
|
+
[...new Set(rows.map((row) => row.path))],
|
|
91
|
+
Effect.fnUntraced(function* (file) {
|
|
92
|
+
const info = yield* fs.stat(file).pipe(Effect.catch(() => Effect.succeed(undefined)))
|
|
93
|
+
if (!info || info.type === "Directory") return undefined
|
|
94
|
+
return [
|
|
95
|
+
file,
|
|
96
|
+
info.mtime.pipe(
|
|
97
|
+
Option.map((time) => time.getTime()),
|
|
98
|
+
Option.getOrElse(() => 0),
|
|
99
|
+
) ?? 0,
|
|
100
|
+
] as const
|
|
101
|
+
}),
|
|
102
|
+
{ concurrency: 16 },
|
|
103
|
+
)).filter((entry): entry is readonly [string, number] => Boolean(entry)),
|
|
104
|
+
)
|
|
105
|
+
const matches = rows.flatMap((row) => {
|
|
106
|
+
const mtime = times.get(row.path)
|
|
107
|
+
if (mtime === undefined) return []
|
|
108
|
+
return [{ ...row, mtime }]
|
|
109
|
+
})
|
|
110
|
+
|
|
111
|
+
matches.sort((a, b) => b.mtime - a.mtime)
|
|
112
|
+
|
|
113
|
+
const limit = 100
|
|
114
|
+
const truncated = matches.length > limit
|
|
115
|
+
const final = truncated ? matches.slice(0, limit) : matches
|
|
116
|
+
if (final.length === 0) return empty
|
|
117
|
+
|
|
118
|
+
const total = matches.length
|
|
119
|
+
const output = [`Found ${total} matches${truncated ? ` (showing first ${limit})` : ""}`]
|
|
120
|
+
|
|
121
|
+
let current = ""
|
|
122
|
+
for (const match of final) {
|
|
123
|
+
if (current !== match.path) {
|
|
124
|
+
if (current !== "") output.push("")
|
|
125
|
+
current = match.path
|
|
126
|
+
output.push(`${match.path}:`)
|
|
127
|
+
}
|
|
128
|
+
const text =
|
|
129
|
+
match.text.length > MAX_LINE_LENGTH ? match.text.substring(0, MAX_LINE_LENGTH) + "..." : match.text
|
|
130
|
+
output.push(` Line ${match.line}: ${text}`)
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
if (truncated) {
|
|
134
|
+
output.push("")
|
|
135
|
+
output.push(
|
|
136
|
+
`(Results truncated: showing ${limit} of ${total} matches (${total - limit} hidden). Consider using a more specific path or pattern.)`,
|
|
137
|
+
)
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
if (result.partial) {
|
|
141
|
+
output.push("")
|
|
142
|
+
output.push("(Some paths were inaccessible and skipped)")
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
return {
|
|
146
|
+
title: params.pattern,
|
|
147
|
+
metadata: {
|
|
148
|
+
matches: total,
|
|
149
|
+
truncated,
|
|
150
|
+
},
|
|
151
|
+
output: output.join("\n"),
|
|
152
|
+
}
|
|
153
|
+
}).pipe(Effect.orDie),
|
|
154
|
+
}
|
|
155
|
+
}),
|
|
156
|
+
)
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
- Fast content search tool that works with any codebase size
|
|
2
|
+
- Searches file contents using regular expressions
|
|
3
|
+
- Supports full regex syntax (eg. "log.*Error", "function\s+\w+", etc.)
|
|
4
|
+
- Filter files by pattern with the include parameter (eg. "*.js", "*.{ts,tsx}")
|
|
5
|
+
- Returns file paths and line numbers with at least one match sorted by modification time
|
|
6
|
+
- Use this tool when you need to find files containing specific patterns
|
|
7
|
+
- If you need to identify/count the number of matches within files, use the Bash tool with `rg` (ripgrep) directly. Do NOT use `grep`.
|
|
8
|
+
- When you are doing an open-ended search that may require multiple rounds of globbing and grepping, use the Task tool instead
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
import { Effect, Schema } from "effect"
|
|
2
|
+
import * as Tool from "./tool"
|
|
3
|
+
|
|
4
|
+
export const Parameters = Schema.Struct({
|
|
5
|
+
tool: Schema.String,
|
|
6
|
+
error: Schema.String,
|
|
7
|
+
})
|
|
8
|
+
|
|
9
|
+
export const InvalidTool = Tool.define(
|
|
10
|
+
"invalid",
|
|
11
|
+
Effect.succeed({
|
|
12
|
+
description: "Do not use",
|
|
13
|
+
parameters: Parameters,
|
|
14
|
+
execute: (params: { tool: string; error: string }) =>
|
|
15
|
+
Effect.succeed({
|
|
16
|
+
title: "Invalid Tool",
|
|
17
|
+
output: `The arguments provided to the tool are invalid: ${params.error}`,
|
|
18
|
+
metadata: {},
|
|
19
|
+
}),
|
|
20
|
+
}),
|
|
21
|
+
)
|
|
@@ -0,0 +1,164 @@
|
|
|
1
|
+
import type { JSONSchema7 } from "@ai-sdk/provider"
|
|
2
|
+
import { JsonSchema, Schema } from "effect"
|
|
3
|
+
import type * as Tool from "./tool"
|
|
4
|
+
|
|
5
|
+
type JsonObject = Record<string, unknown>
|
|
6
|
+
const cache = new WeakMap<Schema.Top, JSONSchema7>()
|
|
7
|
+
|
|
8
|
+
export function fromSchema(schema: Schema.Top): JSONSchema7 {
|
|
9
|
+
const cached = cache.get(schema)
|
|
10
|
+
if (cached) return cached
|
|
11
|
+
|
|
12
|
+
const document = Schema.toJsonSchemaDocument(schema, { additionalProperties: true })
|
|
13
|
+
const result = normalize({
|
|
14
|
+
$schema: JsonSchema.META_SCHEMA_URI_DRAFT_2020_12,
|
|
15
|
+
...document.schema,
|
|
16
|
+
...(Object.keys(document.definitions).length > 0 ? { $defs: document.definitions } : {}),
|
|
17
|
+
})
|
|
18
|
+
const inlined = dropDefinitionsIfResolved(inlineLocalReferences(result))
|
|
19
|
+
if (!isJsonSchema(inlined)) throw new Error("tool JSON Schema helper produced a non-schema value")
|
|
20
|
+
cache.set(schema, inlined)
|
|
21
|
+
return inlined
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
export function fromTool(tool: Tool.Def): JSONSchema7 {
|
|
25
|
+
return tool.jsonSchema ?? fromSchema(tool.parameters as Schema.Top)
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
function normalize(value: unknown, options: { stripNull?: boolean } = {}): unknown {
|
|
29
|
+
if (Array.isArray(value)) return value.map((item) => normalize(item))
|
|
30
|
+
if (!isRecord(value)) return value
|
|
31
|
+
|
|
32
|
+
const required = Array.isArray(value.required)
|
|
33
|
+
? new Set(value.required.filter((item) => typeof item === "string"))
|
|
34
|
+
: undefined
|
|
35
|
+
const schema = Object.fromEntries(
|
|
36
|
+
Object.entries(value).map(([key, item]) => [
|
|
37
|
+
key,
|
|
38
|
+
key === "properties" && isRecord(item)
|
|
39
|
+
? Object.fromEntries(
|
|
40
|
+
Object.entries(item).map(([name, property]) => [
|
|
41
|
+
name,
|
|
42
|
+
normalize(property, { stripNull: !required?.has(name) }),
|
|
43
|
+
]),
|
|
44
|
+
)
|
|
45
|
+
: normalize(item),
|
|
46
|
+
]),
|
|
47
|
+
)
|
|
48
|
+
|
|
49
|
+
if (schema.additionalProperties === true) delete schema.additionalProperties
|
|
50
|
+
|
|
51
|
+
if (options.stripNull && Array.isArray(schema.anyOf)) {
|
|
52
|
+
const withoutNull = schema.anyOf.filter((item) => !isRecord(item) || item.type !== "null")
|
|
53
|
+
if (withoutNull.length !== schema.anyOf.length) return normalize({ ...schema, anyOf: withoutNull })
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
if (Array.isArray(schema.anyOf)) {
|
|
57
|
+
const withoutNull = schema.anyOf
|
|
58
|
+
const number = withoutNull.find((item) => isRecord(item) && item.type === "number")
|
|
59
|
+
const nonFinite = withoutNull.filter(
|
|
60
|
+
(item) => isRecord(item) && Array.isArray(item.enum) && item.enum.every((entry) => isNonFiniteNumber(entry)),
|
|
61
|
+
)
|
|
62
|
+
if (number && nonFinite.length === withoutNull.length - 1) {
|
|
63
|
+
const { anyOf: _, ...rest } = schema
|
|
64
|
+
return normalize({ ...number, ...rest })
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
if (isEmptyStructUnion(withoutNull)) {
|
|
68
|
+
const { anyOf: _, ...rest } = schema
|
|
69
|
+
return normalize({ type: "object", properties: {}, ...rest })
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
if (withoutNull.length === 1 && isRecord(withoutNull[0])) {
|
|
73
|
+
const { anyOf: _, ...rest } = schema
|
|
74
|
+
return normalize({ ...withoutNull[0], ...rest })
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
if (Array.isArray(schema.allOf) && schema.allOf.every(isRecord) && canFlattenAllOf(schema.allOf, schema)) {
|
|
79
|
+
const { allOf, ...rest } = schema
|
|
80
|
+
return normalize({ ...Object.assign({}, ...allOf), ...rest })
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
if (schema.type === "integer" && schema.maximum === undefined) {
|
|
84
|
+
return { minimum: Number.MIN_SAFE_INTEGER, ...schema, maximum: Number.MAX_SAFE_INTEGER }
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
return schema
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
function isRecord(value: unknown): value is JsonObject {
|
|
91
|
+
return typeof value === "object" && value !== null && !Array.isArray(value)
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
function isJsonSchema(value: unknown): value is JSONSchema7 {
|
|
95
|
+
return typeof value === "boolean" || isRecord(value)
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
function isNonFiniteNumber(value: unknown) {
|
|
99
|
+
return value === "NaN" || value === "Infinity" || value === "-Infinity"
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
function isEmptyStructUnion(items: unknown[]) {
|
|
103
|
+
return (
|
|
104
|
+
items.length === 2 &&
|
|
105
|
+
items.some((item) => isRecord(item) && item.type === "object" && item.properties === undefined) &&
|
|
106
|
+
items.some((item) => isRecord(item) && item.type === "array" && item.items === undefined)
|
|
107
|
+
)
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
function canFlattenAllOf(allOf: JsonObject[], parent: JsonObject) {
|
|
111
|
+
const keys = new Set(Object.keys(parent).filter((key) => key !== "allOf"))
|
|
112
|
+
return allOf.every((item) =>
|
|
113
|
+
Object.keys(item).every((key) => {
|
|
114
|
+
if (keys.has(key)) return false
|
|
115
|
+
keys.add(key)
|
|
116
|
+
return true
|
|
117
|
+
}),
|
|
118
|
+
)
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
function inlineLocalReferences(value: unknown, definitions?: JsonObject, seen = new Set<string>()): unknown {
|
|
122
|
+
if (Array.isArray(value)) return value.map((item) => inlineLocalReferences(item, definitions, seen))
|
|
123
|
+
if (!isRecord(value)) return value
|
|
124
|
+
|
|
125
|
+
const localDefinitions = definitions ?? (isRecord(value.$defs) ? value.$defs : undefined)
|
|
126
|
+
if (typeof value.$ref === "string" && localDefinitions) {
|
|
127
|
+
const name = value.$ref.match(/^#\/\$defs\/(.+)$/)?.[1] ?? value.$ref.match(/^#\/definitions\/(.+)$/)?.[1]
|
|
128
|
+
if (name && !seen.has(name)) {
|
|
129
|
+
const target = localDefinitions[name]
|
|
130
|
+
if (target) {
|
|
131
|
+
const { $ref: _, ...rest } = value
|
|
132
|
+
return inlineLocalReferences(
|
|
133
|
+
{ ...(isRecord(target) ? target : {}), ...rest },
|
|
134
|
+
localDefinitions,
|
|
135
|
+
new Set(seen).add(name),
|
|
136
|
+
)
|
|
137
|
+
}
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
return Object.fromEntries(
|
|
142
|
+
Object.entries(value).map(([key, item]) => [key, inlineLocalReferences(item, localDefinitions, seen)]),
|
|
143
|
+
)
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
function dropDefinitionsIfResolved(value: unknown): unknown {
|
|
147
|
+
if (!isRecord(value) || hasLocalReference(value)) return value
|
|
148
|
+
const { $defs: _, definitions: __, ...rest } = value
|
|
149
|
+
return rest
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
function hasLocalReference(value: unknown): boolean {
|
|
153
|
+
if (Array.isArray(value)) return value.some(hasLocalReference)
|
|
154
|
+
if (!isRecord(value)) return false
|
|
155
|
+
if (
|
|
156
|
+
typeof value.$ref === "string" &&
|
|
157
|
+
(value.$ref.startsWith("#/$defs/") || value.$ref.startsWith("#/definitions/"))
|
|
158
|
+
) {
|
|
159
|
+
return true
|
|
160
|
+
}
|
|
161
|
+
return Object.values(value).some(hasLocalReference)
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
export * as ToolJsonSchema from "./json-schema"
|
|
@@ -0,0 +1,113 @@
|
|
|
1
|
+
import { Effect, Schema } from "effect"
|
|
2
|
+
import * as Tool from "./tool"
|
|
3
|
+
import path from "path"
|
|
4
|
+
import { LSP } from "@/lsp/lsp"
|
|
5
|
+
import DESCRIPTION from "./lsp.txt"
|
|
6
|
+
import { InstanceState } from "@/effect/instance-state"
|
|
7
|
+
import { pathToFileURL } from "url"
|
|
8
|
+
import { assertExternalDirectoryEffect } from "./external-directory"
|
|
9
|
+
import { AppFileSystem } from "@opencode-ai/core/filesystem"
|
|
10
|
+
|
|
11
|
+
const operations = [
|
|
12
|
+
"goToDefinition",
|
|
13
|
+
"findReferences",
|
|
14
|
+
"hover",
|
|
15
|
+
"documentSymbol",
|
|
16
|
+
"workspaceSymbol",
|
|
17
|
+
"goToImplementation",
|
|
18
|
+
"prepareCallHierarchy",
|
|
19
|
+
"incomingCalls",
|
|
20
|
+
"outgoingCalls",
|
|
21
|
+
] as const
|
|
22
|
+
|
|
23
|
+
export const Parameters = Schema.Struct({
|
|
24
|
+
operation: Schema.Literals(operations).annotate({ description: "The LSP operation to perform" }),
|
|
25
|
+
filePath: Schema.String.annotate({ description: "The absolute or relative path to the file" }),
|
|
26
|
+
line: Schema.Int.check(Schema.isGreaterThanOrEqualTo(1)).annotate({
|
|
27
|
+
description: "The line number (1-based, as shown in editors)",
|
|
28
|
+
}),
|
|
29
|
+
character: Schema.Int.check(Schema.isGreaterThanOrEqualTo(1)).annotate({
|
|
30
|
+
description: "The character offset (1-based, as shown in editors)",
|
|
31
|
+
}),
|
|
32
|
+
query: Schema.optional(Schema.String).annotate({
|
|
33
|
+
description: "Search query for workspaceSymbol. Empty string requests all symbols.",
|
|
34
|
+
}),
|
|
35
|
+
})
|
|
36
|
+
|
|
37
|
+
export const LspTool = Tool.define(
|
|
38
|
+
"lsp",
|
|
39
|
+
Effect.gen(function* () {
|
|
40
|
+
const lsp = yield* LSP.Service
|
|
41
|
+
const fs = yield* AppFileSystem.Service
|
|
42
|
+
return {
|
|
43
|
+
description: DESCRIPTION,
|
|
44
|
+
parameters: Parameters,
|
|
45
|
+
execute: (args: Schema.Schema.Type<typeof Parameters>, ctx: Tool.Context) =>
|
|
46
|
+
Effect.gen(function* () {
|
|
47
|
+
const instance = yield* InstanceState.context
|
|
48
|
+
const file = path.isAbsolute(args.filePath) ? args.filePath : path.join(instance.directory, args.filePath)
|
|
49
|
+
yield* assertExternalDirectoryEffect(ctx, file)
|
|
50
|
+
const meta =
|
|
51
|
+
args.operation === "workspaceSymbol"
|
|
52
|
+
? { operation: args.operation }
|
|
53
|
+
: args.operation === "documentSymbol"
|
|
54
|
+
? { operation: args.operation, filePath: file }
|
|
55
|
+
: { operation: args.operation, filePath: file, line: args.line, character: args.character }
|
|
56
|
+
yield* ctx.ask({
|
|
57
|
+
permission: "lsp",
|
|
58
|
+
patterns: ["*"],
|
|
59
|
+
always: ["*"],
|
|
60
|
+
metadata: meta,
|
|
61
|
+
})
|
|
62
|
+
|
|
63
|
+
const uri = pathToFileURL(file).href
|
|
64
|
+
const position = { file, line: args.line - 1, character: args.character - 1 }
|
|
65
|
+
const relPath = path.relative(instance.worktree, file)
|
|
66
|
+
const detail =
|
|
67
|
+
args.operation === "workspaceSymbol"
|
|
68
|
+
? ""
|
|
69
|
+
: args.operation === "documentSymbol"
|
|
70
|
+
? relPath
|
|
71
|
+
: `${relPath}:${args.line}:${args.character}`
|
|
72
|
+
const title = detail ? `${args.operation} ${detail}` : args.operation
|
|
73
|
+
|
|
74
|
+
const exists = yield* fs.existsSafe(file)
|
|
75
|
+
if (!exists) throw new Error(`File not found: ${file}`)
|
|
76
|
+
|
|
77
|
+
const available = yield* lsp.hasClients(file)
|
|
78
|
+
if (!available) throw new Error("No LSP server available for this file type.")
|
|
79
|
+
|
|
80
|
+
yield* lsp.touchFile(file, "document")
|
|
81
|
+
|
|
82
|
+
const result: unknown[] = yield* (() => {
|
|
83
|
+
switch (args.operation) {
|
|
84
|
+
case "goToDefinition":
|
|
85
|
+
return lsp.definition(position)
|
|
86
|
+
case "findReferences":
|
|
87
|
+
return lsp.references(position)
|
|
88
|
+
case "hover":
|
|
89
|
+
return lsp.hover(position)
|
|
90
|
+
case "documentSymbol":
|
|
91
|
+
return lsp.documentSymbol(uri)
|
|
92
|
+
case "workspaceSymbol":
|
|
93
|
+
return lsp.workspaceSymbol(args.query ?? "")
|
|
94
|
+
case "goToImplementation":
|
|
95
|
+
return lsp.implementation(position)
|
|
96
|
+
case "prepareCallHierarchy":
|
|
97
|
+
return lsp.prepareCallHierarchy(position)
|
|
98
|
+
case "incomingCalls":
|
|
99
|
+
return lsp.incomingCalls(position)
|
|
100
|
+
case "outgoingCalls":
|
|
101
|
+
return lsp.outgoingCalls(position)
|
|
102
|
+
}
|
|
103
|
+
})()
|
|
104
|
+
|
|
105
|
+
return {
|
|
106
|
+
title,
|
|
107
|
+
metadata: { result },
|
|
108
|
+
output: result.length === 0 ? `No results found for ${args.operation}` : JSON.stringify(result, null, 2),
|
|
109
|
+
}
|
|
110
|
+
}).pipe(Effect.orDie),
|
|
111
|
+
}
|
|
112
|
+
}),
|
|
113
|
+
)
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
Interact with Language Server Protocol (LSP) servers to get code intelligence features.
|
|
2
|
+
|
|
3
|
+
Supported operations:
|
|
4
|
+
- goToDefinition: Find where a symbol is defined
|
|
5
|
+
- findReferences: Find all references to a symbol
|
|
6
|
+
- hover: Get hover information (documentation, type info) for a symbol
|
|
7
|
+
- documentSymbol: Get all symbols (functions, classes, variables) in a document
|
|
8
|
+
- workspaceSymbol: List project-wide symbols matching a query string
|
|
9
|
+
- goToImplementation: Find implementations of an interface or abstract method
|
|
10
|
+
- prepareCallHierarchy: Get call hierarchy item at a position (functions/methods)
|
|
11
|
+
- incomingCalls: Find all functions/methods that call the function at a position
|
|
12
|
+
- outgoingCalls: Find all functions/methods called by the function at a position
|
|
13
|
+
|
|
14
|
+
All operations require:
|
|
15
|
+
- filePath: The file to operate on
|
|
16
|
+
- line: The line number (1-based, as shown in editors)
|
|
17
|
+
- character: The character offset (1-based, as shown in editors)
|
|
18
|
+
|
|
19
|
+
workspaceSymbol also accepts:
|
|
20
|
+
- query: A query string to filter symbols by. Empty string requests all symbols.
|
|
21
|
+
|
|
22
|
+
For workspaceSymbol, filePath is not sent in the LSP workspace/symbol request. It is used by opencode to select and start the matching LSP server.
|
|
23
|
+
|
|
24
|
+
Note: LSP servers must be configured for the file type. If no server is available, an error will be returned.
|