bonecode 1.0.0 → 1.1.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/LICENSE +21 -0
- package/README.md +64 -50
- package/bone/output/agent/.dockerignore +7 -7
- package/bone/output/agent/.env.example +36 -36
- package/bone/output/agent/.github/workflows/ci.yaml +58 -58
- package/bone/output/agent/AgentDomain.bone.map +349 -349
- package/bone/output/agent/AgentDomain.postman_collection.json +957 -957
- package/bone/output/agent/Dockerfile +22 -22
- package/bone/output/agent/README.md +47 -47
- package/bone/output/agent/admin/index.html +739 -739
- package/bone/output/agent/docker-compose.yaml +22 -22
- package/bone/output/agent/k8s/deployment.yaml +75 -75
- package/bone/output/agent/migrations/agent.sql +36 -36
- package/bone/output/agent/migrations/agent_instance.sql +36 -36
- package/bone/output/agent/migrations/audit_log.sql +18 -18
- package/bone/output/agent/migrations/build_step.sql +34 -34
- package/bone/output/agent/migrations/event_outbox.sql +31 -31
- package/bone/output/agent/migrations/plan.sql +30 -30
- package/bone/output/agent/migrations/task.sql +30 -30
- package/bone/output/agent/migrations/tool_call.sql +33 -33
- package/bone/output/agent/openapi.yaml +1116 -1116
- package/bone/output/agent/package.json +35 -35
- package/bone/output/agent/schema.graphql +233 -233
- package/bone/output/agent/sdk/client.ts +231 -231
- package/bone/output/agent/src/algorithms.ts +2 -2
- package/bone/output/agent/src/audit.ts +44 -44
- package/bone/output/agent/src/auth.ts +57 -57
- package/bone/output/agent/src/cron.ts +12 -12
- package/bone/output/agent/src/db.ts +31 -31
- package/bone/output/agent/src/debug.ts +66 -66
- package/bone/output/agent/src/events.ts +243 -243
- package/bone/output/agent/src/extensions.ts +54 -54
- package/bone/output/agent/src/failure_rules.ts +322 -322
- package/bone/output/agent/src/flows.ts +168 -168
- package/bone/output/agent/src/health.ts +43 -43
- package/bone/output/agent/src/index.ts +99 -99
- package/bone/output/agent/src/logger.ts +69 -66
- package/bone/output/agent/src/metrics.ts +75 -75
- package/bone/output/agent/src/migrate.ts +351 -351
- package/bone/output/agent/src/migration_diff.ts +108 -108
- package/bone/output/agent/src/notify.ts +125 -125
- package/bone/output/agent/src/routes/plan.ts +91 -91
- package/bone/output/agent/src/routes/task.ts +105 -105
- package/bone/output/agent/src/routes/tool_call.ts +166 -166
- package/bone/output/agent/src/schemas.ts +384 -384
- package/bone/output/agent/src/state_machines/agent_instance.ts +24 -24
- package/bone/output/agent/src/state_machines/build_step.ts +22 -22
- package/bone/output/agent/src/state_machines/plan.ts +22 -22
- package/bone/output/agent/src/state_machines/task.ts +22 -22
- package/bone/output/agent/src/state_machines/tool_call.ts +22 -22
- package/bone/output/agent/src/tests.ts +361 -361
- package/bone/output/agent/src/websocket.ts +200 -200
- package/bone/output/agent/tsconfig.json +24 -24
- package/bone/output/rag/.dockerignore +7 -7
- package/bone/output/rag/.env.example +36 -36
- package/bone/output/rag/.github/workflows/ci.yaml +58 -58
- package/bone/output/rag/Dockerfile +22 -22
- package/bone/output/rag/RAGDomain.bone.map +286 -286
- package/bone/output/rag/RAGDomain.postman_collection.json +922 -922
- package/bone/output/rag/README.md +47 -47
- package/bone/output/rag/admin/index.html +817 -817
- package/bone/output/rag/docker-compose.yaml +22 -22
- package/bone/output/rag/k8s/deployment.yaml +75 -75
- package/bone/output/rag/migrations/audit_log.sql +18 -18
- package/bone/output/rag/migrations/code_chunk.sql +34 -34
- package/bone/output/rag/migrations/code_file.sql +33 -33
- package/bone/output/rag/migrations/event_outbox.sql +31 -31
- package/bone/output/rag/migrations/indexing_job.sql +33 -33
- package/bone/output/rag/migrations/knowledge_base.sql +35 -35
- package/bone/output/rag/migrations/memory_entry.sql +34 -34
- package/bone/output/rag/openapi.yaml +1097 -1097
- package/bone/output/rag/package.json +35 -35
- package/bone/output/rag/schema.graphql +245 -245
- package/bone/output/rag/sdk/client.ts +234 -234
- package/bone/output/rag/src/algorithms.ts +2 -2
- package/bone/output/rag/src/audit.ts +37 -37
- package/bone/output/rag/src/auth.ts +57 -57
- package/bone/output/rag/src/cron.ts +12 -12
- package/bone/output/rag/src/db.ts +31 -31
- package/bone/output/rag/src/debug.ts +66 -66
- package/bone/output/rag/src/events.ts +243 -243
- package/bone/output/rag/src/extensions.ts +350 -350
- package/bone/output/rag/src/failure_rules.ts +314 -314
- package/bone/output/rag/src/flows.ts +239 -239
- package/bone/output/rag/src/health.ts +43 -43
- package/bone/output/rag/src/index.ts +94 -94
- package/bone/output/rag/src/logger.ts +69 -66
- package/bone/output/rag/src/metrics.ts +75 -75
- package/bone/output/rag/src/migrate.ts +363 -363
- package/bone/output/rag/src/migration_diff.ts +108 -108
- package/bone/output/rag/src/notify.ts +99 -99
- package/bone/output/rag/src/routes/code_chunk.ts +75 -75
- package/bone/output/rag/src/routes/code_file.ts +101 -101
- package/bone/output/rag/src/routes/indexing_job.ts +87 -87
- package/bone/output/rag/src/routes/knowledge_base.ts +230 -230
- package/bone/output/rag/src/routes/memory_entry.ts +87 -87
- package/bone/output/rag/src/schemas.ts +394 -394
- package/bone/output/rag/src/state_machines/code_file.ts +23 -23
- package/bone/output/rag/src/state_machines/indexing_job.ts +22 -22
- package/bone/output/rag/src/state_machines/knowledge_base.ts +23 -23
- package/bone/output/rag/src/state_machines/memory_entry.ts +20 -20
- package/bone/output/rag/src/tests.ts +339 -339
- package/bone/output/rag/tsconfig.json +24 -24
- package/bone/output/session/.dockerignore +7 -7
- package/bone/output/session/.env.example +36 -36
- package/bone/output/session/.github/workflows/ci.yaml +58 -58
- package/bone/output/session/Dockerfile +22 -22
- package/bone/output/session/README.md +47 -47
- package/bone/output/session/SessionDomain.bone.map +349 -349
- package/bone/output/session/SessionDomain.postman_collection.json +957 -957
- package/bone/output/session/admin/index.html +666 -666
- package/bone/output/session/docker-compose.yaml +22 -22
- package/bone/output/session/k8s/deployment.yaml +75 -75
- package/bone/output/session/migrations/audit_log.sql +18 -18
- package/bone/output/session/migrations/event_outbox.sql +31 -31
- package/bone/output/session/migrations/message.sql +31 -31
- package/bone/output/session/migrations/part.sql +28 -28
- package/bone/output/session/migrations/permission.sql +28 -28
- package/bone/output/session/migrations/project.sql +28 -28
- package/bone/output/session/migrations/session.sql +38 -38
- package/bone/output/session/openapi.yaml +1101 -1101
- package/bone/output/session/package.json +35 -35
- package/bone/output/session/schema.graphql +222 -222
- package/bone/output/session/sdk/client.ts +225 -225
- package/bone/output/session/src/algorithms.ts +2 -2
- package/bone/output/session/src/audit.ts +44 -44
- package/bone/output/session/src/auth.ts +57 -57
- package/bone/output/session/src/cron.ts +12 -12
- package/bone/output/session/src/db.ts +31 -31
- package/bone/output/session/src/debug.ts +66 -66
- package/bone/output/session/src/events.ts +270 -270
- package/bone/output/session/src/extensions.ts +215 -215
- package/bone/output/session/src/failure_rules.ts +283 -283
- package/bone/output/session/src/flows.ts +168 -168
- package/bone/output/session/src/health.ts +43 -43
- package/bone/output/session/src/index.ts +99 -99
- package/bone/output/session/src/logger.ts +67 -66
- package/bone/output/session/src/metrics.ts +75 -75
- package/bone/output/session/src/migrate.ts +331 -331
- package/bone/output/session/src/migration_diff.ts +108 -108
- package/bone/output/session/src/notify.ts +112 -112
- package/bone/output/session/src/routes/message.ts +93 -93
- package/bone/output/session/src/routes/part.ts +79 -79
- package/bone/output/session/src/routes/permission.ts +79 -79
- package/bone/output/session/src/routes/project.ts +79 -79
- package/bone/output/session/src/routes/session.ts +294 -294
- package/bone/output/session/src/schemas.ts +357 -357
- package/bone/output/session/src/state_machines/session.ts +23 -23
- package/bone/output/session/src/tests.ts +325 -325
- package/bone/output/session/src/websocket.ts +223 -200
- package/bone/output/session/tsconfig.json +24 -24
- package/bone/output/workspace/.dockerignore +7 -7
- package/bone/output/workspace/.env.example +36 -36
- package/bone/output/workspace/.github/workflows/ci.yaml +58 -58
- package/bone/output/workspace/Dockerfile +22 -22
- package/bone/output/workspace/README.md +45 -45
- package/bone/output/workspace/WorkspaceDomain.bone.map +188 -188
- package/bone/output/workspace/WorkspaceDomain.postman_collection.json +620 -620
- package/bone/output/workspace/admin/index.html +484 -484
- package/bone/output/workspace/docker-compose.yaml +22 -22
- package/bone/output/workspace/k8s/deployment.yaml +75 -75
- package/bone/output/workspace/migrations/audit_log.sql +18 -18
- package/bone/output/workspace/migrations/codebase.sql +34 -34
- package/bone/output/workspace/migrations/event_outbox.sql +31 -31
- package/bone/output/workspace/migrations/snapshot.sql +32 -32
- package/bone/output/workspace/migrations/workspace.sql +33 -33
- package/bone/output/workspace/openapi.yaml +721 -721
- package/bone/output/workspace/package.json +35 -35
- package/bone/output/workspace/schema.graphql +153 -153
- package/bone/output/workspace/sdk/client.ts +155 -155
- package/bone/output/workspace/src/algorithms.ts +2 -2
- package/bone/output/workspace/src/audit.ts +37 -37
- package/bone/output/workspace/src/auth.ts +57 -57
- package/bone/output/workspace/src/cron.ts +12 -12
- package/bone/output/workspace/src/db.ts +31 -31
- package/bone/output/workspace/src/debug.ts +66 -66
- package/bone/output/workspace/src/events.ts +243 -243
- package/bone/output/workspace/src/extensions.ts +44 -44
- package/bone/output/workspace/src/failure_rules.ts +152 -152
- package/bone/output/workspace/src/health.ts +43 -43
- package/bone/output/workspace/src/index.ts +88 -88
- package/bone/output/workspace/src/logger.ts +69 -66
- package/bone/output/workspace/src/metrics.ts +75 -75
- package/bone/output/workspace/src/migrate.ts +219 -219
- package/bone/output/workspace/src/migration_diff.ts +108 -108
- package/bone/output/workspace/src/notify.ts +73 -73
- package/bone/output/workspace/src/routes/codebase.ts +87 -87
- package/bone/output/workspace/src/routes/snapshot.ts +127 -127
- package/bone/output/workspace/src/routes/workspace.ts +190 -190
- package/bone/output/workspace/src/schemas.ts +231 -231
- package/bone/output/workspace/src/state_machines/codebase.ts +21 -21
- package/bone/output/workspace/src/state_machines/snapshot.ts +20 -20
- package/bone/output/workspace/src/state_machines/workspace.ts +21 -21
- package/bone/output/workspace/src/tests.ts +248 -248
- package/bone/output/workspace/tsconfig.json +24 -24
- package/compat/opencode_adapter.ts +94 -17
- package/package.json +15 -2
- package/src/cli.ts +66 -107
- package/src/db_adapter.ts +354 -0
- package/src/engine/account/account.sql.ts +39 -39
- package/src/engine/account/account.ts +456 -456
- package/src/engine/account/repo.ts +166 -166
- package/src/engine/account/schema.ts +99 -99
- package/src/engine/account/url.ts +8 -8
- package/src/engine/acp/README.md +174 -174
- package/src/engine/acp/agent.ts +1968 -1968
- package/src/engine/acp/runtime.ts +22 -22
- package/src/engine/acp/session.ts +122 -122
- package/src/engine/acp/types.ts +24 -24
- package/src/engine/agent/agent.ts +463 -463
- package/src/engine/agent/generate.txt +75 -75
- package/src/engine/agent/prompt/compaction.txt +9 -9
- package/src/engine/agent/prompt/explore.txt +18 -18
- package/src/engine/agent/prompt/scout.txt +36 -36
- package/src/engine/agent/prompt/summary.txt +11 -11
- package/src/engine/agent/prompt/title.txt +44 -44
- package/src/engine/agent/subagent-permissions.ts +34 -34
- package/src/engine/auth/index.ts +96 -96
- package/src/engine/background/background/job.ts +200 -200
- package/src/engine/background/job.ts +200 -200
- package/src/engine/bus/bus-event.ts +45 -45
- package/src/engine/bus/global.ts +22 -22
- package/src/engine/bus/index.ts +203 -203
- package/src/engine/command/command/index.ts +181 -181
- package/src/engine/command/command/template/initialize.txt +66 -66
- package/src/engine/command/command/template/review.txt +101 -101
- package/src/engine/command/index.ts +181 -181
- package/src/engine/command/template/initialize.txt +66 -66
- package/src/engine/command/template/review.txt +101 -101
- package/src/engine/config/agent.ts +172 -172
- package/src/engine/config/attachment.ts +25 -25
- package/src/engine/config/command.ts +62 -62
- package/src/engine/config/config.ts +833 -833
- package/src/engine/config/console-state.ts +14 -14
- package/src/engine/config/entry-name.ts +16 -16
- package/src/engine/config/error.ts +23 -23
- package/src/engine/config/formatter.ts +13 -13
- package/src/engine/config/layout.ts +6 -6
- package/src/engine/config/lsp.ts +43 -43
- package/src/engine/config/managed.ts +71 -71
- package/src/engine/config/markdown.ts +96 -96
- package/src/engine/config/mcp.ts +56 -56
- package/src/engine/config/model-id.ts +5 -5
- package/src/engine/config/parse.ts +79 -79
- package/src/engine/config/paths.ts +45 -45
- package/src/engine/config/permission.ts +58 -58
- package/src/engine/config/plugin.ts +84 -84
- package/src/engine/config/provider.ts +111 -111
- package/src/engine/config/reference.ts +23 -23
- package/src/engine/config/server.ts +19 -19
- package/src/engine/config/skills.ts +14 -14
- package/src/engine/config/variable.ts +90 -90
- package/src/engine/control-plane/adapters/index.ts +41 -41
- package/src/engine/control-plane/adapters/worktree.ts +96 -96
- package/src/engine/control-plane/dev/README.md +19 -19
- package/src/engine/control-plane/dev/debug-workspace-plugin.ts +73 -73
- package/src/engine/control-plane/schema.ts +14 -14
- package/src/engine/control-plane/types.ts +59 -59
- package/src/engine/control-plane/util.ts +39 -39
- package/src/engine/control-plane/workspace-adapter-runtime.ts +51 -51
- package/src/engine/control-plane/workspace-context.ts +26 -26
- package/src/engine/control-plane/workspace.sql.ts +20 -20
- package/src/engine/control-plane/workspace.ts +1072 -1072
- package/src/engine/data-migration.ts +161 -161
- package/src/engine/effect/app-runtime.ts +143 -143
- package/src/engine/effect/bootstrap-runtime.ts +29 -29
- package/src/engine/effect/bridge.ts +84 -84
- package/src/engine/effect/config-service.ts +67 -67
- package/src/engine/effect/instance-ref.ts +11 -11
- package/src/engine/effect/instance-registry.ts +12 -12
- package/src/engine/effect/instance-state.ts +72 -72
- package/src/engine/effect/promise.ts +17 -17
- package/src/engine/effect/run-service.ts +47 -47
- package/src/engine/effect/runner.ts +217 -217
- package/src/engine/effect/runtime-flags.ts +74 -74
- package/src/engine/effect/service-use.ts +38 -38
- package/src/engine/env/index.ts +37 -37
- package/src/engine/event-v2-bridge.ts +89 -89
- package/src/engine/file/file/ignore.ts +81 -81
- package/src/engine/file/file/index.ts +651 -651
- package/src/engine/file/file/protected.ts +59 -59
- package/src/engine/file/file/ripgrep.ts +481 -481
- package/src/engine/file/file/watcher.ts +167 -167
- package/src/engine/file/ignore.ts +81 -81
- package/src/engine/file/index.ts +651 -651
- package/src/engine/file/protected.ts +59 -59
- package/src/engine/file/ripgrep.ts +481 -481
- package/src/engine/file/watcher.ts +167 -167
- package/src/engine/format/format/formatter.ts +404 -404
- package/src/engine/format/format/index.ts +209 -209
- package/src/engine/format/formatter.ts +404 -404
- package/src/engine/format/index.ts +209 -209
- package/src/engine/git/git/index.ts +347 -347
- package/src/engine/git/index.ts +347 -347
- package/src/engine/id/id.ts +80 -80
- package/src/engine/ide/index.ts +70 -70
- package/src/engine/image/image/image.ts +176 -176
- package/src/engine/image/image.ts +176 -176
- package/src/engine/index.ts +251 -251
- package/src/engine/installation/index.ts +327 -327
- package/src/engine/lsp/client.ts +707 -707
- package/src/engine/lsp/diagnostic.ts +29 -29
- package/src/engine/lsp/language.ts +121 -121
- package/src/engine/lsp/launch.ts +21 -21
- package/src/engine/lsp/lsp/client.ts +707 -707
- package/src/engine/lsp/lsp/diagnostic.ts +29 -29
- package/src/engine/lsp/lsp/language.ts +121 -121
- package/src/engine/lsp/lsp/launch.ts +21 -21
- package/src/engine/lsp/lsp/lsp.ts +507 -507
- package/src/engine/lsp/lsp/server.ts +2064 -2064
- package/src/engine/lsp/lsp.ts +507 -507
- package/src/engine/lsp/server.ts +2064 -2064
- package/src/engine/mcp/auth.ts +146 -146
- package/src/engine/mcp/index.ts +958 -958
- package/src/engine/mcp/mcp/auth.ts +146 -146
- package/src/engine/mcp/mcp/index.ts +958 -958
- package/src/engine/mcp/mcp/oauth-callback.ts +232 -232
- package/src/engine/mcp/mcp/oauth-provider.ts +214 -214
- package/src/engine/mcp/oauth-callback.ts +232 -232
- package/src/engine/mcp/oauth-provider.ts +214 -214
- package/src/engine/node.ts +6 -6
- package/src/engine/patch/index.ts +689 -689
- package/src/engine/patch/patch/index.ts +689 -689
- package/src/engine/permission/arity.ts +163 -163
- package/src/engine/permission/evaluate.ts +15 -15
- package/src/engine/permission/index.ts +306 -306
- package/src/engine/permission/permission/arity.ts +163 -163
- package/src/engine/permission/permission/evaluate.ts +15 -15
- package/src/engine/permission/permission/index.ts +306 -306
- package/src/engine/permission/permission/schema.ts +13 -13
- package/src/engine/permission/schema.ts +13 -13
- package/src/engine/plugin/azure.ts +26 -26
- package/src/engine/plugin/cloudflare.ts +76 -76
- package/src/engine/plugin/codex.ts +622 -622
- package/src/engine/plugin/digitalocean.ts +411 -411
- package/src/engine/plugin/github-copilot/copilot.ts +394 -394
- package/src/engine/plugin/github-copilot/models.ts +196 -196
- package/src/engine/plugin/index.ts +295 -295
- package/src/engine/plugin/install.ts +439 -439
- package/src/engine/plugin/loader.ts +216 -216
- package/src/engine/plugin/meta.ts +188 -188
- package/src/engine/plugin/shared.ts +323 -323
- package/src/engine/project/bootstrap-service.ts +9 -9
- package/src/engine/project/bootstrap.ts +75 -75
- package/src/engine/project/instance-context.ts +24 -24
- package/src/engine/project/instance-layer.ts +11 -11
- package/src/engine/project/instance-runtime.ts +16 -16
- package/src/engine/project/instance-store.ts +193 -193
- package/src/engine/project/project.sql.ts +17 -17
- package/src/engine/project/project.ts +537 -537
- package/src/engine/project/schema.ts +13 -13
- package/src/engine/project/vcs.ts +405 -405
- package/src/engine/provider/auth.ts +225 -225
- package/src/engine/provider/error.ts +204 -204
- package/src/engine/provider/model-status.ts +8 -8
- package/src/engine/provider/provider.ts +1843 -1843
- package/src/engine/provider/schema.ts +30 -30
- package/src/engine/provider/transform.ts +1376 -1376
- package/src/engine/pty/index.ts +365 -365
- package/src/engine/pty/input.ts +24 -24
- package/src/engine/pty/pty/index.ts +365 -365
- package/src/engine/pty/pty/input.ts +24 -24
- package/src/engine/pty/pty/pty.bun.ts +26 -26
- package/src/engine/pty/pty/pty.node.ts +27 -27
- package/src/engine/pty/pty/pty.ts +25 -25
- package/src/engine/pty/pty/schema.ts +14 -14
- package/src/engine/pty/pty/ticket.ts +68 -68
- package/src/engine/pty/pty.bun.ts +26 -26
- package/src/engine/pty/pty.node.ts +27 -27
- package/src/engine/pty/pty.ts +25 -25
- package/src/engine/pty/schema.ts +14 -14
- package/src/engine/pty/ticket.ts +68 -68
- package/src/engine/question/index.ts +213 -213
- package/src/engine/question/question/index.ts +213 -213
- package/src/engine/question/question/schema.ts +10 -10
- package/src/engine/question/schema.ts +10 -10
- package/src/engine/reference/reference/reference.ts +241 -241
- package/src/engine/reference/reference/repository-cache.ts +147 -147
- package/src/engine/reference/reference.ts +241 -241
- package/src/engine/reference/repository-cache.ts +147 -147
- package/src/engine/session/compaction.ts +651 -651
- package/src/engine/session/instruction.ts +238 -238
- package/src/engine/session/llm.ts +459 -459
- package/src/engine/session/message-error.ts +14 -14
- package/src/engine/session/message-v2.ts +1202 -1202
- package/src/engine/session/message.ts +146 -146
- package/src/engine/session/overflow.ts +32 -32
- package/src/engine/session/processor.ts +823 -823
- package/src/engine/session/prompt/anthropic.txt +105 -105
- package/src/engine/session/prompt/beast.txt +147 -147
- package/src/engine/session/prompt/build-switch.txt +5 -5
- package/src/engine/session/prompt/codex.txt +79 -79
- package/src/engine/session/prompt/copilot-gpt-5.txt +143 -143
- package/src/engine/session/prompt/default.txt +105 -105
- package/src/engine/session/prompt/gemini.txt +155 -155
- package/src/engine/session/prompt/gpt.txt +107 -107
- package/src/engine/session/prompt/kimi.txt +95 -95
- package/src/engine/session/prompt/max-steps.txt +15 -15
- package/src/engine/session/prompt/plan-reminder-anthropic.txt +67 -67
- package/src/engine/session/prompt/plan.txt +26 -26
- package/src/engine/session/prompt/trinity.txt +97 -97
- package/src/engine/session/prompt.ts +66 -9
- package/src/engine/session/retry.ts +200 -200
- package/src/engine/session/revert.ts +162 -162
- package/src/engine/session/run-state.ts +153 -153
- package/src/engine/session/schema.ts +26 -26
- package/src/engine/session/session.sql.ts +137 -137
- package/src/engine/session/session.ts +1011 -1011
- package/src/engine/session/status.ts +94 -94
- package/src/engine/session/summary.ts +164 -164
- package/src/engine/session/system.ts +84 -84
- package/src/engine/session/todo.ts +81 -81
- package/src/engine/share/session.ts +61 -61
- package/src/engine/share/share-next.ts +376 -376
- package/src/engine/share/share.sql.ts +13 -13
- package/src/engine/shell/shell/shell.ts +215 -215
- package/src/engine/shell/shell.ts +215 -215
- package/src/engine/skill/discovery.ts +116 -116
- package/src/engine/skill/index.ts +336 -336
- package/src/engine/skill/prompt/customize-opencode.md +377 -377
- package/src/engine/skill/skill/discovery.ts +116 -116
- package/src/engine/skill/skill/index.ts +336 -336
- package/src/engine/skill/skill/prompt/customize-opencode.md +377 -377
- package/src/engine/snapshot/index.ts +762 -762
- package/src/engine/snapshot/snapshot/index.ts +762 -762
- package/src/engine/sync/README.md +179 -179
- package/src/engine/sync/event.sql.ts +17 -17
- package/src/engine/sync/index.ts +410 -410
- package/src/engine/sync/schema.ts +11 -11
- package/src/engine/temporary.ts +33 -33
- package/src/engine/tool/apply_patch.ts +313 -313
- package/src/engine/tool/apply_patch.txt +33 -33
- package/src/engine/tool/edit.ts +711 -711
- package/src/engine/tool/edit.txt +10 -10
- package/src/engine/tool/external-directory.ts +49 -49
- package/src/engine/tool/glob.ts +103 -103
- package/src/engine/tool/glob.txt +6 -6
- package/src/engine/tool/grep.ts +156 -156
- package/src/engine/tool/grep.txt +8 -8
- package/src/engine/tool/invalid.ts +21 -21
- package/src/engine/tool/json-schema.ts +164 -164
- package/src/engine/tool/lsp.ts +113 -113
- package/src/engine/tool/lsp.txt +24 -24
- package/src/engine/tool/mcp-websearch.ts +96 -96
- package/src/engine/tool/plan-enter.txt +14 -14
- package/src/engine/tool/plan-exit.txt +13 -13
- package/src/engine/tool/plan.ts +78 -78
- package/src/engine/tool/question.ts +44 -44
- package/src/engine/tool/question.txt +10 -10
- package/src/engine/tool/read.ts +337 -337
- package/src/engine/tool/read.txt +14 -14
- package/src/engine/tool/registry.ts +472 -472
- package/src/engine/tool/repo_clone.ts +80 -80
- package/src/engine/tool/repo_clone.txt +5 -5
- package/src/engine/tool/repo_overview.ts +279 -279
- package/src/engine/tool/repo_overview.txt +4 -4
- package/src/engine/tool/schema.ts +14 -14
- package/src/engine/tool/shell/id.ts +19 -19
- package/src/engine/tool/shell/prompt.ts +295 -295
- package/src/engine/tool/shell/shell.txt +77 -77
- package/src/engine/tool/shell.ts +647 -647
- package/src/engine/tool/skill.ts +75 -75
- package/src/engine/tool/skill.txt +5 -5
- package/src/engine/tool/task.ts +337 -337
- package/src/engine/tool/task.txt +58 -58
- package/src/engine/tool/task_status.ts +179 -179
- package/src/engine/tool/task_status.txt +13 -13
- package/src/engine/tool/todo.ts +57 -57
- package/src/engine/tool/todowrite.txt +167 -167
- package/src/engine/tool/tool/apply_patch.ts +313 -313
- package/src/engine/tool/tool/apply_patch.txt +33 -33
- package/src/engine/tool/tool/edit.ts +711 -711
- package/src/engine/tool/tool/edit.txt +10 -10
- package/src/engine/tool/tool/external-directory.ts +49 -49
- package/src/engine/tool/tool/glob.ts +103 -103
- package/src/engine/tool/tool/glob.txt +6 -6
- package/src/engine/tool/tool/grep.ts +156 -156
- package/src/engine/tool/tool/grep.txt +8 -8
- package/src/engine/tool/tool/invalid.ts +21 -21
- package/src/engine/tool/tool/json-schema.ts +164 -164
- package/src/engine/tool/tool/lsp.ts +113 -113
- package/src/engine/tool/tool/lsp.txt +24 -24
- package/src/engine/tool/tool/mcp-websearch.ts +96 -96
- package/src/engine/tool/tool/plan-enter.txt +14 -14
- package/src/engine/tool/tool/plan-exit.txt +13 -13
- package/src/engine/tool/tool/plan.ts +78 -78
- package/src/engine/tool/tool/question.ts +44 -44
- package/src/engine/tool/tool/question.txt +10 -10
- package/src/engine/tool/tool/read.ts +337 -337
- package/src/engine/tool/tool/read.txt +14 -14
- package/src/engine/tool/tool/registry.ts +472 -472
- package/src/engine/tool/tool/repo_clone.ts +80 -80
- package/src/engine/tool/tool/repo_clone.txt +5 -5
- package/src/engine/tool/tool/repo_overview.ts +279 -279
- package/src/engine/tool/tool/repo_overview.txt +4 -4
- package/src/engine/tool/tool/schema.ts +14 -14
- package/src/engine/tool/tool/shell/id.ts +19 -19
- package/src/engine/tool/tool/shell/prompt.ts +295 -295
- package/src/engine/tool/tool/shell/shell.txt +77 -77
- package/src/engine/tool/tool/shell.ts +647 -647
- package/src/engine/tool/tool/skill.ts +75 -75
- package/src/engine/tool/tool/skill.txt +5 -5
- package/src/engine/tool/tool/task.ts +337 -337
- package/src/engine/tool/tool/task.txt +58 -58
- package/src/engine/tool/tool/task_status.ts +179 -179
- package/src/engine/tool/tool/task_status.txt +13 -13
- package/src/engine/tool/tool/todo.ts +57 -57
- package/src/engine/tool/tool/todowrite.txt +167 -167
- package/src/engine/tool/tool/tool.ts +164 -164
- package/src/engine/tool/tool/truncate.ts +160 -160
- package/src/engine/tool/tool/truncation-dir.ts +4 -4
- package/src/engine/tool/tool/webfetch.ts +192 -192
- package/src/engine/tool/tool/webfetch.txt +13 -13
- package/src/engine/tool/tool/websearch.ts +143 -143
- package/src/engine/tool/tool/websearch.txt +14 -14
- package/src/engine/tool/tool/write.ts +104 -104
- package/src/engine/tool/tool/write.txt +8 -8
- package/src/engine/tool/tool.ts +164 -164
- package/src/engine/tool/truncate.ts +160 -160
- package/src/engine/tool/truncation-dir.ts +4 -4
- package/src/engine/tool/webfetch.ts +192 -192
- package/src/engine/tool/webfetch.txt +13 -13
- package/src/engine/tool/websearch.ts +143 -143
- package/src/engine/tool/websearch.txt +14 -14
- package/src/engine/tool/write.ts +104 -104
- package/src/engine/tool/write.txt +8 -8
- package/src/engine/util/archive.ts +17 -17
- package/src/engine/util/bom.ts +31 -31
- package/src/engine/util/data-url.ts +9 -9
- package/src/engine/util/defer.ts +10 -10
- package/src/engine/util/effect-http-client.ts +11 -11
- package/src/engine/util/error.ts +88 -88
- package/src/engine/util/filesystem.ts +252 -252
- package/src/engine/util/format.ts +20 -20
- package/src/engine/util/iife.ts +3 -3
- package/src/engine/util/lazy.ts +20 -20
- package/src/engine/util/local-context.ts +25 -25
- package/src/engine/util/locale.ts +86 -86
- package/src/engine/util/media.ts +26 -26
- package/src/engine/util/process.ts +176 -176
- package/src/engine/util/queue.ts +32 -32
- package/src/engine/util/record.ts +3 -3
- package/src/engine/util/repository.ts +158 -158
- package/src/engine/util/rpc.ts +66 -66
- package/src/engine/util/signal.ts +12 -12
- package/src/engine/util/timeout.ts +13 -13
- package/src/engine/util/token.ts +7 -7
- package/src/engine/util/util/archive.ts +17 -17
- package/src/engine/util/util/bom.ts +31 -31
- package/src/engine/util/util/data-url.ts +9 -9
- package/src/engine/util/util/defer.ts +10 -10
- package/src/engine/util/util/effect-http-client.ts +11 -11
- package/src/engine/util/util/error.ts +88 -88
- package/src/engine/util/util/filesystem.ts +252 -252
- package/src/engine/util/util/format.ts +20 -20
- package/src/engine/util/util/iife.ts +3 -3
- package/src/engine/util/util/lazy.ts +20 -20
- package/src/engine/util/util/local-context.ts +25 -25
- package/src/engine/util/util/locale.ts +86 -86
- package/src/engine/util/util/media.ts +26 -26
- package/src/engine/util/util/process.ts +176 -176
- package/src/engine/util/util/queue.ts +32 -32
- package/src/engine/util/util/record.ts +3 -3
- package/src/engine/util/util/repository.ts +158 -158
- package/src/engine/util/util/rpc.ts +66 -66
- package/src/engine/util/util/signal.ts +12 -12
- package/src/engine/util/util/timeout.ts +13 -13
- package/src/engine/util/util/token.ts +7 -7
- package/src/engine/util/util/which.ts +14 -14
- package/src/engine/util/util/wildcard.ts +59 -59
- package/src/engine/util/which.ts +14 -14
- package/src/engine/util/wildcard.ts +59 -59
- package/src/engine/worktree/index.ts +621 -621
- package/src/server.ts +121 -158
- package/src/tui.ts +485 -502
|
@@ -1,621 +1,621 @@
|
|
|
1
|
-
import { Global } from "@opencode-ai/core/global"
|
|
2
|
-
import { InstanceLayer } from "@/project/instance-layer"
|
|
3
|
-
import { InstanceStore } from "@/project/instance-store"
|
|
4
|
-
import { Project } from "@/project/project"
|
|
5
|
-
import { Database } from "@/storage/db"
|
|
6
|
-
import { eq } from "drizzle-orm"
|
|
7
|
-
import { ProjectTable } from "../project/project.sql"
|
|
8
|
-
import type { ProjectID } from "../project/schema"
|
|
9
|
-
import * as Log from "@opencode-ai/core/util/log"
|
|
10
|
-
import { Slug } from "@opencode-ai/core/util/slug"
|
|
11
|
-
import { errorMessage } from "../util/error"
|
|
12
|
-
import { BusEvent } from "@/bus/bus-event"
|
|
13
|
-
import { GlobalBus } from "@/bus/global"
|
|
14
|
-
import { Git } from "@/git"
|
|
15
|
-
import { Effect, Layer, Path, Schema, Scope, Context } from "effect"
|
|
16
|
-
import { ChildProcess } from "effect/unstable/process"
|
|
17
|
-
import { NodePath } from "@effect/platform-node"
|
|
18
|
-
import { AppFileSystem } from "@opencode-ai/core/filesystem"
|
|
19
|
-
import { BootstrapRuntime } from "@/effect/bootstrap-runtime"
|
|
20
|
-
import { AppProcess } from "@opencode-ai/core/process"
|
|
21
|
-
import { InstanceState } from "@/effect/instance-state"
|
|
22
|
-
|
|
23
|
-
const log = Log.create({ service: "worktree" })
|
|
24
|
-
|
|
25
|
-
export const Event = {
|
|
26
|
-
Ready: BusEvent.define(
|
|
27
|
-
"worktree.ready",
|
|
28
|
-
Schema.Struct({
|
|
29
|
-
name: Schema.String,
|
|
30
|
-
branch: Schema.optional(Schema.String),
|
|
31
|
-
}),
|
|
32
|
-
),
|
|
33
|
-
Failed: BusEvent.define(
|
|
34
|
-
"worktree.failed",
|
|
35
|
-
Schema.Struct({
|
|
36
|
-
message: Schema.String,
|
|
37
|
-
}),
|
|
38
|
-
),
|
|
39
|
-
}
|
|
40
|
-
|
|
41
|
-
export const Info = Schema.Struct({
|
|
42
|
-
name: Schema.String,
|
|
43
|
-
branch: Schema.optional(Schema.String),
|
|
44
|
-
directory: Schema.String,
|
|
45
|
-
}).annotate({ identifier: "Worktree" })
|
|
46
|
-
export type Info = Schema.Schema.Type<typeof Info>
|
|
47
|
-
|
|
48
|
-
export const CreateInput = Schema.Struct({
|
|
49
|
-
name: Schema.optional(Schema.String),
|
|
50
|
-
startCommand: Schema.optional(
|
|
51
|
-
Schema.String.annotate({ description: "Additional startup script to run after the project's start command" }),
|
|
52
|
-
),
|
|
53
|
-
}).annotate({ identifier: "WorktreeCreateInput" })
|
|
54
|
-
export type CreateInput = Schema.Schema.Type<typeof CreateInput>
|
|
55
|
-
|
|
56
|
-
export const RemoveInput = Schema.Struct({
|
|
57
|
-
directory: Schema.String,
|
|
58
|
-
}).annotate({ identifier: "WorktreeRemoveInput" })
|
|
59
|
-
export type RemoveInput = Schema.Schema.Type<typeof RemoveInput>
|
|
60
|
-
|
|
61
|
-
export const ResetInput = Schema.Struct({
|
|
62
|
-
directory: Schema.String,
|
|
63
|
-
}).annotate({ identifier: "WorktreeResetInput" })
|
|
64
|
-
export type ResetInput = Schema.Schema.Type<typeof ResetInput>
|
|
65
|
-
|
|
66
|
-
export class NotGitError extends Schema.TaggedErrorClass<NotGitError>()("WorktreeNotGitError", {
|
|
67
|
-
message: Schema.String,
|
|
68
|
-
}) {}
|
|
69
|
-
|
|
70
|
-
export class NameGenerationFailedError extends Schema.TaggedErrorClass<NameGenerationFailedError>()(
|
|
71
|
-
"WorktreeNameGenerationFailedError",
|
|
72
|
-
{
|
|
73
|
-
message: Schema.String,
|
|
74
|
-
},
|
|
75
|
-
) {}
|
|
76
|
-
|
|
77
|
-
export class CreateFailedError extends Schema.TaggedErrorClass<CreateFailedError>()("WorktreeCreateFailedError", {
|
|
78
|
-
message: Schema.String,
|
|
79
|
-
}) {}
|
|
80
|
-
|
|
81
|
-
export class StartCommandFailedError extends Schema.TaggedErrorClass<StartCommandFailedError>()(
|
|
82
|
-
"WorktreeStartCommandFailedError",
|
|
83
|
-
{
|
|
84
|
-
message: Schema.String,
|
|
85
|
-
},
|
|
86
|
-
) {}
|
|
87
|
-
|
|
88
|
-
export class RemoveFailedError extends Schema.TaggedErrorClass<RemoveFailedError>()("WorktreeRemoveFailedError", {
|
|
89
|
-
message: Schema.String,
|
|
90
|
-
}) {}
|
|
91
|
-
|
|
92
|
-
export class ResetFailedError extends Schema.TaggedErrorClass<ResetFailedError>()("WorktreeResetFailedError", {
|
|
93
|
-
message: Schema.String,
|
|
94
|
-
}) {}
|
|
95
|
-
|
|
96
|
-
export class ListFailedError extends Schema.TaggedErrorClass<ListFailedError>()("WorktreeListFailedError", {
|
|
97
|
-
message: Schema.String,
|
|
98
|
-
}) {}
|
|
99
|
-
|
|
100
|
-
export type Error =
|
|
101
|
-
| NotGitError
|
|
102
|
-
| NameGenerationFailedError
|
|
103
|
-
| CreateFailedError
|
|
104
|
-
| StartCommandFailedError
|
|
105
|
-
| RemoveFailedError
|
|
106
|
-
| ResetFailedError
|
|
107
|
-
| ListFailedError
|
|
108
|
-
|
|
109
|
-
function slugify(input: string) {
|
|
110
|
-
return input
|
|
111
|
-
.trim()
|
|
112
|
-
.toLowerCase()
|
|
113
|
-
.replace(/[^a-z0-9]+/g, "-")
|
|
114
|
-
.replace(/^-+/, "")
|
|
115
|
-
.replace(/-+$/, "")
|
|
116
|
-
}
|
|
117
|
-
|
|
118
|
-
function failedRemoves(...chunks: string[]) {
|
|
119
|
-
return chunks.filter(Boolean).flatMap((chunk) =>
|
|
120
|
-
chunk
|
|
121
|
-
.split("\n")
|
|
122
|
-
.map((line) => line.trim())
|
|
123
|
-
.flatMap((line) => {
|
|
124
|
-
const match = line.match(/^warning:\s+failed to remove\s+(.+):\s+/i)
|
|
125
|
-
if (!match) return []
|
|
126
|
-
const value = match[1]?.trim().replace(/^['"]|['"]$/g, "")
|
|
127
|
-
if (!value) return []
|
|
128
|
-
return [value]
|
|
129
|
-
}),
|
|
130
|
-
)
|
|
131
|
-
}
|
|
132
|
-
|
|
133
|
-
// ---------------------------------------------------------------------------
|
|
134
|
-
// Effect service
|
|
135
|
-
// ---------------------------------------------------------------------------
|
|
136
|
-
|
|
137
|
-
export interface Interface {
|
|
138
|
-
readonly makeWorktreeInfo: (options?: { name?: string; detached?: boolean }) => Effect.Effect<Info, Error>
|
|
139
|
-
readonly createFromInfo: (info: Info, startCommand?: string) => Effect.Effect<void, Error>
|
|
140
|
-
readonly create: (input?: CreateInput) => Effect.Effect<Info, Error>
|
|
141
|
-
readonly list: () => Effect.Effect<(Omit<Info, "branch"> & { branch?: string })[], Error>
|
|
142
|
-
readonly remove: (input: RemoveInput) => Effect.Effect<boolean, Error>
|
|
143
|
-
readonly reset: (input: ResetInput) => Effect.Effect<boolean, Error>
|
|
144
|
-
}
|
|
145
|
-
|
|
146
|
-
export class Service extends Context.Service<Service, Interface>()("@opencode/Worktree") {}
|
|
147
|
-
|
|
148
|
-
type GitResult = { code: number; text: string; stderr: string }
|
|
149
|
-
|
|
150
|
-
export const layer: Layer.Layer<
|
|
151
|
-
Service,
|
|
152
|
-
never,
|
|
153
|
-
AppFileSystem.Service | Path.Path | AppProcess.Service | Git.Service | Project.Service | InstanceStore.Service
|
|
154
|
-
> = Layer.effect(
|
|
155
|
-
Service,
|
|
156
|
-
Effect.gen(function* () {
|
|
157
|
-
const scope = yield* Scope.Scope
|
|
158
|
-
const fs = yield* AppFileSystem.Service
|
|
159
|
-
const pathSvc = yield* Path.Path
|
|
160
|
-
const appProcess = yield* AppProcess.Service
|
|
161
|
-
const gitSvc = yield* Git.Service
|
|
162
|
-
const project = yield* Project.Service
|
|
163
|
-
const store = yield* InstanceStore.Service
|
|
164
|
-
|
|
165
|
-
const git = Effect.fnUntraced(
|
|
166
|
-
function* (args: string[], opts?: { cwd?: string }) {
|
|
167
|
-
const result = yield* appProcess.run(
|
|
168
|
-
ChildProcess.make("git", args, { cwd: opts?.cwd, extendEnv: true, stdin: "ignore" }),
|
|
169
|
-
)
|
|
170
|
-
return {
|
|
171
|
-
code: result.exitCode,
|
|
172
|
-
text: result.stdout.toString("utf8"),
|
|
173
|
-
stderr: result.stderr.toString("utf8"),
|
|
174
|
-
} satisfies GitResult
|
|
175
|
-
},
|
|
176
|
-
Effect.catch((e) =>
|
|
177
|
-
Effect.succeed({
|
|
178
|
-
code: 1,
|
|
179
|
-
text: "",
|
|
180
|
-
stderr: e instanceof Error ? e.message : String(e),
|
|
181
|
-
} satisfies GitResult),
|
|
182
|
-
),
|
|
183
|
-
)
|
|
184
|
-
|
|
185
|
-
const MAX_NAME_ATTEMPTS = 26
|
|
186
|
-
const candidate = Effect.fn("Worktree.candidate")(function* (input: {
|
|
187
|
-
root: string
|
|
188
|
-
name?: string
|
|
189
|
-
detached?: boolean
|
|
190
|
-
}) {
|
|
191
|
-
const ctx = yield* InstanceState.context
|
|
192
|
-
for (const attempt of Array.from({ length: MAX_NAME_ATTEMPTS }, (_, i) => i)) {
|
|
193
|
-
const name = input.name ? (attempt === 0 ? input.name : `${input.name}-${Slug.create()}`) : Slug.create()
|
|
194
|
-
const branch = input.detached ? undefined : `opencode/${name}`
|
|
195
|
-
const directory = pathSvc.join(input.root, name)
|
|
196
|
-
|
|
197
|
-
if (yield* fs.exists(directory).pipe(Effect.orDie)) continue
|
|
198
|
-
|
|
199
|
-
if (branch) {
|
|
200
|
-
const ref = `refs/heads/${branch}`
|
|
201
|
-
const branchCheck = yield* git(["show-ref", "--verify", "--quiet", ref], { cwd: ctx.worktree })
|
|
202
|
-
if (branchCheck.code === 0) continue
|
|
203
|
-
}
|
|
204
|
-
|
|
205
|
-
return { name, directory, ...(branch ? { branch } : {}) }
|
|
206
|
-
}
|
|
207
|
-
return yield* new NameGenerationFailedError({ message: "Failed to generate a unique worktree name" })
|
|
208
|
-
})
|
|
209
|
-
|
|
210
|
-
const makeWorktreeInfo = Effect.fn("Worktree.makeWorktreeInfo")(function* (input?: {
|
|
211
|
-
name?: string
|
|
212
|
-
detached?: boolean
|
|
213
|
-
}) {
|
|
214
|
-
const ctx = yield* InstanceState.context
|
|
215
|
-
if (ctx.project.vcs !== "git") {
|
|
216
|
-
return yield* new NotGitError({ message: "Worktrees are only supported for git projects" })
|
|
217
|
-
}
|
|
218
|
-
|
|
219
|
-
const root = pathSvc.join(Global.Path.data, "worktree", ctx.project.id)
|
|
220
|
-
yield* fs.makeDirectory(root, { recursive: true }).pipe(Effect.orDie)
|
|
221
|
-
|
|
222
|
-
return yield* candidate({ root, name: input?.name ? slugify(input.name) : "", detached: input?.detached })
|
|
223
|
-
})
|
|
224
|
-
|
|
225
|
-
const setup = Effect.fnUntraced(function* (info: Info) {
|
|
226
|
-
const ctx = yield* InstanceState.context
|
|
227
|
-
const created = yield* git(
|
|
228
|
-
info.branch
|
|
229
|
-
? ["worktree", "add", "--no-checkout", "-b", info.branch, info.directory]
|
|
230
|
-
: ["worktree", "add", "--no-checkout", "--detach", info.directory, "HEAD"],
|
|
231
|
-
{ cwd: ctx.worktree },
|
|
232
|
-
)
|
|
233
|
-
if (created.code !== 0) {
|
|
234
|
-
return yield* new CreateFailedError({
|
|
235
|
-
message: created.stderr || created.text || "Failed to create git worktree",
|
|
236
|
-
})
|
|
237
|
-
}
|
|
238
|
-
|
|
239
|
-
yield* project.addSandbox(ctx.project.id, info.directory).pipe(Effect.catch(() => Effect.void))
|
|
240
|
-
})
|
|
241
|
-
|
|
242
|
-
const boot = Effect.fnUntraced(function* (info: Info, startCommand?: string) {
|
|
243
|
-
const ctx = yield* InstanceState.context
|
|
244
|
-
const workspaceID = yield* InstanceState.workspaceID
|
|
245
|
-
const projectID = ctx.project.id
|
|
246
|
-
const extra = startCommand?.trim()
|
|
247
|
-
|
|
248
|
-
const populated = yield* git(["reset", "--hard"], { cwd: info.directory })
|
|
249
|
-
if (populated.code !== 0) {
|
|
250
|
-
const message = populated.stderr || populated.text || "Failed to populate worktree"
|
|
251
|
-
log.error("worktree checkout failed", { directory: info.directory, message })
|
|
252
|
-
GlobalBus.emit("event", {
|
|
253
|
-
directory: info.directory,
|
|
254
|
-
project: ctx.project.id,
|
|
255
|
-
workspace: workspaceID,
|
|
256
|
-
payload: { type: Event.Failed.type, properties: { message } },
|
|
257
|
-
})
|
|
258
|
-
return
|
|
259
|
-
}
|
|
260
|
-
|
|
261
|
-
const booted = yield* store.load({ directory: info.directory }).pipe(
|
|
262
|
-
Effect.as(true),
|
|
263
|
-
Effect.catch((error) =>
|
|
264
|
-
Effect.sync(() => {
|
|
265
|
-
const message = errorMessage(error)
|
|
266
|
-
log.error("worktree bootstrap failed", { directory: info.directory, message })
|
|
267
|
-
GlobalBus.emit("event", {
|
|
268
|
-
directory: info.directory,
|
|
269
|
-
project: ctx.project.id,
|
|
270
|
-
workspace: workspaceID,
|
|
271
|
-
payload: { type: Event.Failed.type, properties: { message } },
|
|
272
|
-
})
|
|
273
|
-
return false
|
|
274
|
-
}),
|
|
275
|
-
),
|
|
276
|
-
)
|
|
277
|
-
if (!booted) return
|
|
278
|
-
|
|
279
|
-
GlobalBus.emit("event", {
|
|
280
|
-
directory: info.directory,
|
|
281
|
-
project: ctx.project.id,
|
|
282
|
-
workspace: workspaceID,
|
|
283
|
-
payload: {
|
|
284
|
-
type: Event.Ready.type,
|
|
285
|
-
properties: { name: info.name, ...(info.branch ? { branch: info.branch } : {}) },
|
|
286
|
-
},
|
|
287
|
-
})
|
|
288
|
-
|
|
289
|
-
yield* runStartScripts(info.directory, { projectID, extra })
|
|
290
|
-
})
|
|
291
|
-
|
|
292
|
-
const createFromInfo = Effect.fn("Worktree.createFromInfo")(function* (info: Info, startCommand?: string) {
|
|
293
|
-
yield* setup(info)
|
|
294
|
-
yield* boot(info, startCommand).pipe(
|
|
295
|
-
Effect.catchCause((cause) => Effect.sync(() => log.error("worktree bootstrap failed", { cause }))),
|
|
296
|
-
Effect.forkIn(scope),
|
|
297
|
-
)
|
|
298
|
-
})
|
|
299
|
-
|
|
300
|
-
const create = Effect.fn("Worktree.create")(function* (input?: CreateInput) {
|
|
301
|
-
const info = yield* makeWorktreeInfo({ name: input?.name })
|
|
302
|
-
yield* createFromInfo(info, input?.startCommand)
|
|
303
|
-
return info
|
|
304
|
-
})
|
|
305
|
-
|
|
306
|
-
const canonical = Effect.fnUntraced(function* (input: string) {
|
|
307
|
-
const abs = pathSvc.resolve(input)
|
|
308
|
-
const real = yield* fs.realPath(abs).pipe(Effect.catch(() => Effect.succeed(abs)))
|
|
309
|
-
const normalized = pathSvc.normalize(real)
|
|
310
|
-
return process.platform === "win32" ? normalized.toLowerCase() : normalized
|
|
311
|
-
})
|
|
312
|
-
|
|
313
|
-
function parseWorktreeList(text: string) {
|
|
314
|
-
return text
|
|
315
|
-
.split("\n")
|
|
316
|
-
.map((line) => line.trim())
|
|
317
|
-
.reduce<{ path?: string; branch?: string }[]>((acc, line) => {
|
|
318
|
-
if (!line) return acc
|
|
319
|
-
if (line.startsWith("worktree ")) {
|
|
320
|
-
acc.push({ path: line.slice("worktree ".length).trim() })
|
|
321
|
-
return acc
|
|
322
|
-
}
|
|
323
|
-
const current = acc[acc.length - 1]
|
|
324
|
-
if (!current) return acc
|
|
325
|
-
if (line.startsWith("branch ")) {
|
|
326
|
-
current.branch = line.slice("branch ".length).trim()
|
|
327
|
-
}
|
|
328
|
-
return acc
|
|
329
|
-
}, [])
|
|
330
|
-
}
|
|
331
|
-
|
|
332
|
-
const locateWorktree = Effect.fnUntraced(function* (
|
|
333
|
-
entries: { path?: string; branch?: string }[],
|
|
334
|
-
directory: string,
|
|
335
|
-
) {
|
|
336
|
-
for (const item of entries) {
|
|
337
|
-
if (!item.path) continue
|
|
338
|
-
const key = yield* canonical(item.path)
|
|
339
|
-
if (key === directory) return item
|
|
340
|
-
}
|
|
341
|
-
return undefined
|
|
342
|
-
})
|
|
343
|
-
|
|
344
|
-
const list = Effect.fn("Worktree.list")(function* () {
|
|
345
|
-
const ctx = yield* InstanceState.context
|
|
346
|
-
if (ctx.project.vcs !== "git") {
|
|
347
|
-
return []
|
|
348
|
-
}
|
|
349
|
-
|
|
350
|
-
const result = yield* git(["worktree", "list", "--porcelain"], { cwd: ctx.worktree })
|
|
351
|
-
if (result.code !== 0) {
|
|
352
|
-
return yield* new ListFailedError({ message: result.stderr || result.text || "Failed to read git worktrees" })
|
|
353
|
-
}
|
|
354
|
-
|
|
355
|
-
const primary = yield* canonical(ctx.worktree)
|
|
356
|
-
const primaryName = pathSvc.basename(primary).toLowerCase()
|
|
357
|
-
return yield* Effect.forEach(parseWorktreeList(result.text), (entry) =>
|
|
358
|
-
Effect.gen(function* () {
|
|
359
|
-
if (!entry.path) return undefined
|
|
360
|
-
const directory = yield* canonical(entry.path)
|
|
361
|
-
if (directory === primary) return undefined
|
|
362
|
-
const name = pathSvc.basename(directory).toLowerCase()
|
|
363
|
-
return {
|
|
364
|
-
name: name === primaryName ? pathSvc.basename(pathSvc.dirname(directory)) : name,
|
|
365
|
-
directory,
|
|
366
|
-
...(entry.branch ? { branch: entry.branch.replace(/^refs\/heads\//, "") } : {}),
|
|
367
|
-
}
|
|
368
|
-
}),
|
|
369
|
-
).pipe(Effect.map((items) => items.filter((item) => item !== undefined)))
|
|
370
|
-
})
|
|
371
|
-
|
|
372
|
-
function stopFsmonitor(target: string) {
|
|
373
|
-
return fs.exists(target).pipe(
|
|
374
|
-
Effect.orDie,
|
|
375
|
-
Effect.flatMap((exists) => (exists ? git(["fsmonitor--daemon", "stop"], { cwd: target }) : Effect.void)),
|
|
376
|
-
)
|
|
377
|
-
}
|
|
378
|
-
|
|
379
|
-
function cleanDirectory(target: string) {
|
|
380
|
-
return Effect.tryPromise({
|
|
381
|
-
try: () =>
|
|
382
|
-
import("fs/promises").then((fsp) =>
|
|
383
|
-
fsp.rm(target, { recursive: true, force: true, maxRetries: 5, retryDelay: 100 }),
|
|
384
|
-
),
|
|
385
|
-
catch: (error) =>
|
|
386
|
-
new RemoveFailedError({ message: errorMessage(error) || "Failed to remove git worktree directory" }),
|
|
387
|
-
})
|
|
388
|
-
}
|
|
389
|
-
|
|
390
|
-
const remove = Effect.fn("Worktree.remove")(function* (input: RemoveInput) {
|
|
391
|
-
const ctx = yield* InstanceState.context
|
|
392
|
-
if (ctx.project.vcs !== "git") {
|
|
393
|
-
return yield* new NotGitError({ message: "Worktrees are only supported for git projects" })
|
|
394
|
-
}
|
|
395
|
-
|
|
396
|
-
const directory = yield* canonical(input.directory)
|
|
397
|
-
|
|
398
|
-
const list = yield* git(["worktree", "list", "--porcelain"], { cwd: ctx.worktree })
|
|
399
|
-
if (list.code !== 0) {
|
|
400
|
-
return yield* new RemoveFailedError({ message: list.stderr || list.text || "Failed to read git worktrees" })
|
|
401
|
-
}
|
|
402
|
-
|
|
403
|
-
const entries = parseWorktreeList(list.text)
|
|
404
|
-
const entry = yield* locateWorktree(entries, directory)
|
|
405
|
-
|
|
406
|
-
if (!entry?.path) {
|
|
407
|
-
const directoryExists = yield* fs.exists(directory).pipe(Effect.orDie)
|
|
408
|
-
if (directoryExists) {
|
|
409
|
-
yield* stopFsmonitor(directory)
|
|
410
|
-
yield* cleanDirectory(directory)
|
|
411
|
-
}
|
|
412
|
-
return true
|
|
413
|
-
}
|
|
414
|
-
|
|
415
|
-
yield* stopFsmonitor(entry.path)
|
|
416
|
-
const removed = yield* git(["worktree", "remove", "--force", entry.path], { cwd: ctx.worktree })
|
|
417
|
-
if (removed.code !== 0) {
|
|
418
|
-
const next = yield* git(["worktree", "list", "--porcelain"], { cwd: ctx.worktree })
|
|
419
|
-
if (next.code !== 0) {
|
|
420
|
-
return yield* new RemoveFailedError({
|
|
421
|
-
message: removed.stderr || removed.text || next.stderr || next.text || "Failed to remove git worktree",
|
|
422
|
-
})
|
|
423
|
-
}
|
|
424
|
-
|
|
425
|
-
const stale = yield* locateWorktree(parseWorktreeList(next.text), directory)
|
|
426
|
-
if (stale?.path) {
|
|
427
|
-
return yield* new RemoveFailedError({
|
|
428
|
-
message: removed.stderr || removed.text || "Failed to remove git worktree",
|
|
429
|
-
})
|
|
430
|
-
}
|
|
431
|
-
}
|
|
432
|
-
|
|
433
|
-
yield* cleanDirectory(entry.path)
|
|
434
|
-
|
|
435
|
-
const branch = entry.branch?.replace(/^refs\/heads\//, "")
|
|
436
|
-
if (branch) {
|
|
437
|
-
const deleted = yield* git(["branch", "-D", branch], { cwd: ctx.worktree })
|
|
438
|
-
if (deleted.code !== 0) {
|
|
439
|
-
return yield* new RemoveFailedError({
|
|
440
|
-
message: deleted.stderr || deleted.text || "Failed to delete worktree branch",
|
|
441
|
-
})
|
|
442
|
-
}
|
|
443
|
-
}
|
|
444
|
-
|
|
445
|
-
return true
|
|
446
|
-
})
|
|
447
|
-
|
|
448
|
-
const gitExpect = Effect.fnUntraced(function* (
|
|
449
|
-
args: string[],
|
|
450
|
-
opts: { cwd: string },
|
|
451
|
-
error: (r: GitResult) => Error,
|
|
452
|
-
) {
|
|
453
|
-
const result = yield* git(args, opts)
|
|
454
|
-
if (result.code !== 0) return yield* error(result)
|
|
455
|
-
return result
|
|
456
|
-
})
|
|
457
|
-
|
|
458
|
-
const runStartCommand = Effect.fnUntraced(
|
|
459
|
-
function* (directory: string, cmd: string) {
|
|
460
|
-
const [shell, args] = process.platform === "win32" ? ["cmd", ["/c", cmd]] : ["bash", ["-lc", cmd]]
|
|
461
|
-
const result = yield* appProcess.run(
|
|
462
|
-
ChildProcess.make(shell, args as string[], { cwd: directory, extendEnv: true, stdin: "ignore" }),
|
|
463
|
-
)
|
|
464
|
-
return { code: result.exitCode, stderr: result.stderr.toString("utf8") }
|
|
465
|
-
},
|
|
466
|
-
Effect.catch(() => Effect.succeed({ code: 1, stderr: "" })),
|
|
467
|
-
)
|
|
468
|
-
|
|
469
|
-
const runStartScript = Effect.fnUntraced(function* (directory: string, cmd: string, kind: string) {
|
|
470
|
-
const text = cmd.trim()
|
|
471
|
-
if (!text) return true
|
|
472
|
-
const result = yield* runStartCommand(directory, text)
|
|
473
|
-
if (result.code === 0) return true
|
|
474
|
-
log.error("worktree start command failed", { kind, directory, message: result.stderr })
|
|
475
|
-
return false
|
|
476
|
-
})
|
|
477
|
-
|
|
478
|
-
const runStartScripts = Effect.fnUntraced(function* (
|
|
479
|
-
directory: string,
|
|
480
|
-
input: { projectID: ProjectID; extra?: string },
|
|
481
|
-
) {
|
|
482
|
-
const row = yield* Effect.sync(() =>
|
|
483
|
-
Database.use((db) => db.select().from(ProjectTable).where(eq(ProjectTable.id, input.projectID)).get()),
|
|
484
|
-
)
|
|
485
|
-
const project = row ? Project.fromRow(row) : undefined
|
|
486
|
-
const startup = project?.commands?.start?.trim() ?? ""
|
|
487
|
-
const ok = yield* runStartScript(directory, startup, "project")
|
|
488
|
-
if (!ok) return false
|
|
489
|
-
yield* runStartScript(directory, input.extra ?? "", "worktree")
|
|
490
|
-
return true
|
|
491
|
-
})
|
|
492
|
-
|
|
493
|
-
const prune = Effect.fnUntraced(function* (root: string, entries: string[]) {
|
|
494
|
-
const base = yield* canonical(root)
|
|
495
|
-
yield* Effect.forEach(
|
|
496
|
-
entries,
|
|
497
|
-
(entry) =>
|
|
498
|
-
Effect.gen(function* () {
|
|
499
|
-
const target = yield* canonical(pathSvc.resolve(root, entry))
|
|
500
|
-
if (target === base) return
|
|
501
|
-
if (!target.startsWith(`${base}${pathSvc.sep}`)) return
|
|
502
|
-
yield* fs.remove(target, { recursive: true }).pipe(Effect.ignore)
|
|
503
|
-
}),
|
|
504
|
-
{ concurrency: "unbounded" },
|
|
505
|
-
)
|
|
506
|
-
})
|
|
507
|
-
|
|
508
|
-
const sweep = Effect.fnUntraced(function* (root: string) {
|
|
509
|
-
const first = yield* git(["clean", "-ffdx"], { cwd: root })
|
|
510
|
-
if (first.code === 0) return first
|
|
511
|
-
|
|
512
|
-
const entries = failedRemoves(first.stderr, first.text)
|
|
513
|
-
if (!entries.length) return first
|
|
514
|
-
|
|
515
|
-
yield* prune(root, entries)
|
|
516
|
-
return yield* git(["clean", "-ffdx"], { cwd: root })
|
|
517
|
-
})
|
|
518
|
-
|
|
519
|
-
const reset = Effect.fn("Worktree.reset")(function* (input: ResetInput) {
|
|
520
|
-
const ctx = yield* InstanceState.context
|
|
521
|
-
if (ctx.project.vcs !== "git") {
|
|
522
|
-
return yield* new NotGitError({ message: "Worktrees are only supported for git projects" })
|
|
523
|
-
}
|
|
524
|
-
|
|
525
|
-
const directory = yield* canonical(input.directory)
|
|
526
|
-
const primary = yield* canonical(ctx.worktree)
|
|
527
|
-
if (directory === primary) {
|
|
528
|
-
return yield* new ResetFailedError({ message: "Cannot reset the primary workspace" })
|
|
529
|
-
}
|
|
530
|
-
|
|
531
|
-
const list = yield* git(["worktree", "list", "--porcelain"], { cwd: ctx.worktree })
|
|
532
|
-
if (list.code !== 0) {
|
|
533
|
-
return yield* new ResetFailedError({ message: list.stderr || list.text || "Failed to read git worktrees" })
|
|
534
|
-
}
|
|
535
|
-
|
|
536
|
-
const entry = yield* locateWorktree(parseWorktreeList(list.text), directory)
|
|
537
|
-
if (!entry?.path) {
|
|
538
|
-
return yield* new ResetFailedError({ message: "Worktree not found" })
|
|
539
|
-
}
|
|
540
|
-
|
|
541
|
-
const worktreePath = entry.path
|
|
542
|
-
|
|
543
|
-
const base = yield* gitSvc.defaultBranch(ctx.worktree)
|
|
544
|
-
if (!base) {
|
|
545
|
-
return yield* new ResetFailedError({ message: "Default branch not found" })
|
|
546
|
-
}
|
|
547
|
-
|
|
548
|
-
const sep = base.ref.indexOf("/")
|
|
549
|
-
if (base.ref !== base.name && sep > 0) {
|
|
550
|
-
const remote = base.ref.slice(0, sep)
|
|
551
|
-
const branch = base.ref.slice(sep + 1)
|
|
552
|
-
yield* gitExpect(
|
|
553
|
-
["fetch", remote, branch],
|
|
554
|
-
{ cwd: ctx.worktree },
|
|
555
|
-
(r) => new ResetFailedError({ message: r.stderr || r.text || `Failed to fetch ${base.ref}` }),
|
|
556
|
-
)
|
|
557
|
-
}
|
|
558
|
-
|
|
559
|
-
yield* gitExpect(
|
|
560
|
-
["reset", "--hard", base.ref],
|
|
561
|
-
{ cwd: worktreePath },
|
|
562
|
-
(r) => new ResetFailedError({ message: r.stderr || r.text || "Failed to reset worktree to target" }),
|
|
563
|
-
)
|
|
564
|
-
|
|
565
|
-
const cleanResult = yield* sweep(worktreePath)
|
|
566
|
-
if (cleanResult.code !== 0) {
|
|
567
|
-
return yield* new ResetFailedError({
|
|
568
|
-
message: cleanResult.stderr || cleanResult.text || "Failed to clean worktree",
|
|
569
|
-
})
|
|
570
|
-
}
|
|
571
|
-
|
|
572
|
-
yield* gitExpect(
|
|
573
|
-
["submodule", "update", "--init", "--recursive", "--force"],
|
|
574
|
-
{ cwd: worktreePath },
|
|
575
|
-
(r) => new ResetFailedError({ message: r.stderr || r.text || "Failed to update submodules" }),
|
|
576
|
-
)
|
|
577
|
-
|
|
578
|
-
yield* gitExpect(
|
|
579
|
-
["submodule", "foreach", "--recursive", "git", "reset", "--hard"],
|
|
580
|
-
{ cwd: worktreePath },
|
|
581
|
-
(r) => new ResetFailedError({ message: r.stderr || r.text || "Failed to reset submodules" }),
|
|
582
|
-
)
|
|
583
|
-
|
|
584
|
-
yield* gitExpect(
|
|
585
|
-
["submodule", "foreach", "--recursive", "git", "clean", "-fdx"],
|
|
586
|
-
{ cwd: worktreePath },
|
|
587
|
-
(r) => new ResetFailedError({ message: r.stderr || r.text || "Failed to clean submodules" }),
|
|
588
|
-
)
|
|
589
|
-
|
|
590
|
-
const status = yield* git(["-c", "core.fsmonitor=false", "status", "--porcelain=v1"], { cwd: worktreePath })
|
|
591
|
-
if (status.code !== 0) {
|
|
592
|
-
return yield* new ResetFailedError({ message: status.stderr || status.text || "Failed to read git status" })
|
|
593
|
-
}
|
|
594
|
-
|
|
595
|
-
if (status.text.trim()) {
|
|
596
|
-
return yield* new ResetFailedError({ message: `Worktree reset left local changes:\n${status.text.trim()}` })
|
|
597
|
-
}
|
|
598
|
-
|
|
599
|
-
yield* runStartScripts(worktreePath, { projectID: ctx.project.id }).pipe(
|
|
600
|
-
Effect.catchCause((cause) => Effect.sync(() => log.error("worktree start task failed", { cause }))),
|
|
601
|
-
Effect.forkIn(scope),
|
|
602
|
-
)
|
|
603
|
-
|
|
604
|
-
return true
|
|
605
|
-
})
|
|
606
|
-
|
|
607
|
-
return Service.of({ makeWorktreeInfo, createFromInfo, create, list, remove, reset })
|
|
608
|
-
}),
|
|
609
|
-
)
|
|
610
|
-
|
|
611
|
-
export const appLayer = layer.pipe(
|
|
612
|
-
Layer.provide(Git.defaultLayer),
|
|
613
|
-
Layer.provide(AppProcess.defaultLayer),
|
|
614
|
-
Layer.provide(Project.defaultLayer),
|
|
615
|
-
Layer.provide(AppFileSystem.defaultLayer),
|
|
616
|
-
Layer.provide(NodePath.layer),
|
|
617
|
-
)
|
|
618
|
-
|
|
619
|
-
export const defaultLayer = appLayer.pipe(Layer.provide(InstanceLayer.layer))
|
|
620
|
-
|
|
621
|
-
export * as Worktree from "."
|
|
1
|
+
import { Global } from "@opencode-ai/core/global"
|
|
2
|
+
import { InstanceLayer } from "@/project/instance-layer"
|
|
3
|
+
import { InstanceStore } from "@/project/instance-store"
|
|
4
|
+
import { Project } from "@/project/project"
|
|
5
|
+
import { Database } from "@/storage/db"
|
|
6
|
+
import { eq } from "drizzle-orm"
|
|
7
|
+
import { ProjectTable } from "../project/project.sql"
|
|
8
|
+
import type { ProjectID } from "../project/schema"
|
|
9
|
+
import * as Log from "@opencode-ai/core/util/log"
|
|
10
|
+
import { Slug } from "@opencode-ai/core/util/slug"
|
|
11
|
+
import { errorMessage } from "../util/error"
|
|
12
|
+
import { BusEvent } from "@/bus/bus-event"
|
|
13
|
+
import { GlobalBus } from "@/bus/global"
|
|
14
|
+
import { Git } from "@/git"
|
|
15
|
+
import { Effect, Layer, Path, Schema, Scope, Context } from "effect"
|
|
16
|
+
import { ChildProcess } from "effect/unstable/process"
|
|
17
|
+
import { NodePath } from "@effect/platform-node"
|
|
18
|
+
import { AppFileSystem } from "@opencode-ai/core/filesystem"
|
|
19
|
+
import { BootstrapRuntime } from "@/effect/bootstrap-runtime"
|
|
20
|
+
import { AppProcess } from "@opencode-ai/core/process"
|
|
21
|
+
import { InstanceState } from "@/effect/instance-state"
|
|
22
|
+
|
|
23
|
+
const log = Log.create({ service: "worktree" })
|
|
24
|
+
|
|
25
|
+
export const Event = {
|
|
26
|
+
Ready: BusEvent.define(
|
|
27
|
+
"worktree.ready",
|
|
28
|
+
Schema.Struct({
|
|
29
|
+
name: Schema.String,
|
|
30
|
+
branch: Schema.optional(Schema.String),
|
|
31
|
+
}),
|
|
32
|
+
),
|
|
33
|
+
Failed: BusEvent.define(
|
|
34
|
+
"worktree.failed",
|
|
35
|
+
Schema.Struct({
|
|
36
|
+
message: Schema.String,
|
|
37
|
+
}),
|
|
38
|
+
),
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
export const Info = Schema.Struct({
|
|
42
|
+
name: Schema.String,
|
|
43
|
+
branch: Schema.optional(Schema.String),
|
|
44
|
+
directory: Schema.String,
|
|
45
|
+
}).annotate({ identifier: "Worktree" })
|
|
46
|
+
export type Info = Schema.Schema.Type<typeof Info>
|
|
47
|
+
|
|
48
|
+
export const CreateInput = Schema.Struct({
|
|
49
|
+
name: Schema.optional(Schema.String),
|
|
50
|
+
startCommand: Schema.optional(
|
|
51
|
+
Schema.String.annotate({ description: "Additional startup script to run after the project's start command" }),
|
|
52
|
+
),
|
|
53
|
+
}).annotate({ identifier: "WorktreeCreateInput" })
|
|
54
|
+
export type CreateInput = Schema.Schema.Type<typeof CreateInput>
|
|
55
|
+
|
|
56
|
+
export const RemoveInput = Schema.Struct({
|
|
57
|
+
directory: Schema.String,
|
|
58
|
+
}).annotate({ identifier: "WorktreeRemoveInput" })
|
|
59
|
+
export type RemoveInput = Schema.Schema.Type<typeof RemoveInput>
|
|
60
|
+
|
|
61
|
+
export const ResetInput = Schema.Struct({
|
|
62
|
+
directory: Schema.String,
|
|
63
|
+
}).annotate({ identifier: "WorktreeResetInput" })
|
|
64
|
+
export type ResetInput = Schema.Schema.Type<typeof ResetInput>
|
|
65
|
+
|
|
66
|
+
export class NotGitError extends Schema.TaggedErrorClass<NotGitError>()("WorktreeNotGitError", {
|
|
67
|
+
message: Schema.String,
|
|
68
|
+
}) {}
|
|
69
|
+
|
|
70
|
+
export class NameGenerationFailedError extends Schema.TaggedErrorClass<NameGenerationFailedError>()(
|
|
71
|
+
"WorktreeNameGenerationFailedError",
|
|
72
|
+
{
|
|
73
|
+
message: Schema.String,
|
|
74
|
+
},
|
|
75
|
+
) {}
|
|
76
|
+
|
|
77
|
+
export class CreateFailedError extends Schema.TaggedErrorClass<CreateFailedError>()("WorktreeCreateFailedError", {
|
|
78
|
+
message: Schema.String,
|
|
79
|
+
}) {}
|
|
80
|
+
|
|
81
|
+
export class StartCommandFailedError extends Schema.TaggedErrorClass<StartCommandFailedError>()(
|
|
82
|
+
"WorktreeStartCommandFailedError",
|
|
83
|
+
{
|
|
84
|
+
message: Schema.String,
|
|
85
|
+
},
|
|
86
|
+
) {}
|
|
87
|
+
|
|
88
|
+
export class RemoveFailedError extends Schema.TaggedErrorClass<RemoveFailedError>()("WorktreeRemoveFailedError", {
|
|
89
|
+
message: Schema.String,
|
|
90
|
+
}) {}
|
|
91
|
+
|
|
92
|
+
export class ResetFailedError extends Schema.TaggedErrorClass<ResetFailedError>()("WorktreeResetFailedError", {
|
|
93
|
+
message: Schema.String,
|
|
94
|
+
}) {}
|
|
95
|
+
|
|
96
|
+
export class ListFailedError extends Schema.TaggedErrorClass<ListFailedError>()("WorktreeListFailedError", {
|
|
97
|
+
message: Schema.String,
|
|
98
|
+
}) {}
|
|
99
|
+
|
|
100
|
+
export type Error =
|
|
101
|
+
| NotGitError
|
|
102
|
+
| NameGenerationFailedError
|
|
103
|
+
| CreateFailedError
|
|
104
|
+
| StartCommandFailedError
|
|
105
|
+
| RemoveFailedError
|
|
106
|
+
| ResetFailedError
|
|
107
|
+
| ListFailedError
|
|
108
|
+
|
|
109
|
+
function slugify(input: string) {
|
|
110
|
+
return input
|
|
111
|
+
.trim()
|
|
112
|
+
.toLowerCase()
|
|
113
|
+
.replace(/[^a-z0-9]+/g, "-")
|
|
114
|
+
.replace(/^-+/, "")
|
|
115
|
+
.replace(/-+$/, "")
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
function failedRemoves(...chunks: string[]) {
|
|
119
|
+
return chunks.filter(Boolean).flatMap((chunk) =>
|
|
120
|
+
chunk
|
|
121
|
+
.split("\n")
|
|
122
|
+
.map((line) => line.trim())
|
|
123
|
+
.flatMap((line) => {
|
|
124
|
+
const match = line.match(/^warning:\s+failed to remove\s+(.+):\s+/i)
|
|
125
|
+
if (!match) return []
|
|
126
|
+
const value = match[1]?.trim().replace(/^['"]|['"]$/g, "")
|
|
127
|
+
if (!value) return []
|
|
128
|
+
return [value]
|
|
129
|
+
}),
|
|
130
|
+
)
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
// ---------------------------------------------------------------------------
|
|
134
|
+
// Effect service
|
|
135
|
+
// ---------------------------------------------------------------------------
|
|
136
|
+
|
|
137
|
+
export interface Interface {
|
|
138
|
+
readonly makeWorktreeInfo: (options?: { name?: string; detached?: boolean }) => Effect.Effect<Info, Error>
|
|
139
|
+
readonly createFromInfo: (info: Info, startCommand?: string) => Effect.Effect<void, Error>
|
|
140
|
+
readonly create: (input?: CreateInput) => Effect.Effect<Info, Error>
|
|
141
|
+
readonly list: () => Effect.Effect<(Omit<Info, "branch"> & { branch?: string })[], Error>
|
|
142
|
+
readonly remove: (input: RemoveInput) => Effect.Effect<boolean, Error>
|
|
143
|
+
readonly reset: (input: ResetInput) => Effect.Effect<boolean, Error>
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
export class Service extends Context.Service<Service, Interface>()("@opencode/Worktree") {}
|
|
147
|
+
|
|
148
|
+
type GitResult = { code: number; text: string; stderr: string }
|
|
149
|
+
|
|
150
|
+
export const layer: Layer.Layer<
|
|
151
|
+
Service,
|
|
152
|
+
never,
|
|
153
|
+
AppFileSystem.Service | Path.Path | AppProcess.Service | Git.Service | Project.Service | InstanceStore.Service
|
|
154
|
+
> = Layer.effect(
|
|
155
|
+
Service,
|
|
156
|
+
Effect.gen(function* () {
|
|
157
|
+
const scope = yield* Scope.Scope
|
|
158
|
+
const fs = yield* AppFileSystem.Service
|
|
159
|
+
const pathSvc = yield* Path.Path
|
|
160
|
+
const appProcess = yield* AppProcess.Service
|
|
161
|
+
const gitSvc = yield* Git.Service
|
|
162
|
+
const project = yield* Project.Service
|
|
163
|
+
const store = yield* InstanceStore.Service
|
|
164
|
+
|
|
165
|
+
const git = Effect.fnUntraced(
|
|
166
|
+
function* (args: string[], opts?: { cwd?: string }) {
|
|
167
|
+
const result = yield* appProcess.run(
|
|
168
|
+
ChildProcess.make("git", args, { cwd: opts?.cwd, extendEnv: true, stdin: "ignore" }),
|
|
169
|
+
)
|
|
170
|
+
return {
|
|
171
|
+
code: result.exitCode,
|
|
172
|
+
text: result.stdout.toString("utf8"),
|
|
173
|
+
stderr: result.stderr.toString("utf8"),
|
|
174
|
+
} satisfies GitResult
|
|
175
|
+
},
|
|
176
|
+
Effect.catch((e) =>
|
|
177
|
+
Effect.succeed({
|
|
178
|
+
code: 1,
|
|
179
|
+
text: "",
|
|
180
|
+
stderr: e instanceof Error ? e.message : String(e),
|
|
181
|
+
} satisfies GitResult),
|
|
182
|
+
),
|
|
183
|
+
)
|
|
184
|
+
|
|
185
|
+
const MAX_NAME_ATTEMPTS = 26
|
|
186
|
+
const candidate = Effect.fn("Worktree.candidate")(function* (input: {
|
|
187
|
+
root: string
|
|
188
|
+
name?: string
|
|
189
|
+
detached?: boolean
|
|
190
|
+
}) {
|
|
191
|
+
const ctx = yield* InstanceState.context
|
|
192
|
+
for (const attempt of Array.from({ length: MAX_NAME_ATTEMPTS }, (_, i) => i)) {
|
|
193
|
+
const name = input.name ? (attempt === 0 ? input.name : `${input.name}-${Slug.create()}`) : Slug.create()
|
|
194
|
+
const branch = input.detached ? undefined : `opencode/${name}`
|
|
195
|
+
const directory = pathSvc.join(input.root, name)
|
|
196
|
+
|
|
197
|
+
if (yield* fs.exists(directory).pipe(Effect.orDie)) continue
|
|
198
|
+
|
|
199
|
+
if (branch) {
|
|
200
|
+
const ref = `refs/heads/${branch}`
|
|
201
|
+
const branchCheck = yield* git(["show-ref", "--verify", "--quiet", ref], { cwd: ctx.worktree })
|
|
202
|
+
if (branchCheck.code === 0) continue
|
|
203
|
+
}
|
|
204
|
+
|
|
205
|
+
return { name, directory, ...(branch ? { branch } : {}) }
|
|
206
|
+
}
|
|
207
|
+
return yield* new NameGenerationFailedError({ message: "Failed to generate a unique worktree name" })
|
|
208
|
+
})
|
|
209
|
+
|
|
210
|
+
const makeWorktreeInfo = Effect.fn("Worktree.makeWorktreeInfo")(function* (input?: {
|
|
211
|
+
name?: string
|
|
212
|
+
detached?: boolean
|
|
213
|
+
}) {
|
|
214
|
+
const ctx = yield* InstanceState.context
|
|
215
|
+
if (ctx.project.vcs !== "git") {
|
|
216
|
+
return yield* new NotGitError({ message: "Worktrees are only supported for git projects" })
|
|
217
|
+
}
|
|
218
|
+
|
|
219
|
+
const root = pathSvc.join(Global.Path.data, "worktree", ctx.project.id)
|
|
220
|
+
yield* fs.makeDirectory(root, { recursive: true }).pipe(Effect.orDie)
|
|
221
|
+
|
|
222
|
+
return yield* candidate({ root, name: input?.name ? slugify(input.name) : "", detached: input?.detached })
|
|
223
|
+
})
|
|
224
|
+
|
|
225
|
+
const setup = Effect.fnUntraced(function* (info: Info) {
|
|
226
|
+
const ctx = yield* InstanceState.context
|
|
227
|
+
const created = yield* git(
|
|
228
|
+
info.branch
|
|
229
|
+
? ["worktree", "add", "--no-checkout", "-b", info.branch, info.directory]
|
|
230
|
+
: ["worktree", "add", "--no-checkout", "--detach", info.directory, "HEAD"],
|
|
231
|
+
{ cwd: ctx.worktree },
|
|
232
|
+
)
|
|
233
|
+
if (created.code !== 0) {
|
|
234
|
+
return yield* new CreateFailedError({
|
|
235
|
+
message: created.stderr || created.text || "Failed to create git worktree",
|
|
236
|
+
})
|
|
237
|
+
}
|
|
238
|
+
|
|
239
|
+
yield* project.addSandbox(ctx.project.id, info.directory).pipe(Effect.catch(() => Effect.void))
|
|
240
|
+
})
|
|
241
|
+
|
|
242
|
+
const boot = Effect.fnUntraced(function* (info: Info, startCommand?: string) {
|
|
243
|
+
const ctx = yield* InstanceState.context
|
|
244
|
+
const workspaceID = yield* InstanceState.workspaceID
|
|
245
|
+
const projectID = ctx.project.id
|
|
246
|
+
const extra = startCommand?.trim()
|
|
247
|
+
|
|
248
|
+
const populated = yield* git(["reset", "--hard"], { cwd: info.directory })
|
|
249
|
+
if (populated.code !== 0) {
|
|
250
|
+
const message = populated.stderr || populated.text || "Failed to populate worktree"
|
|
251
|
+
log.error("worktree checkout failed", { directory: info.directory, message })
|
|
252
|
+
GlobalBus.emit("event", {
|
|
253
|
+
directory: info.directory,
|
|
254
|
+
project: ctx.project.id,
|
|
255
|
+
workspace: workspaceID,
|
|
256
|
+
payload: { type: Event.Failed.type, properties: { message } },
|
|
257
|
+
})
|
|
258
|
+
return
|
|
259
|
+
}
|
|
260
|
+
|
|
261
|
+
const booted = yield* store.load({ directory: info.directory }).pipe(
|
|
262
|
+
Effect.as(true),
|
|
263
|
+
Effect.catch((error) =>
|
|
264
|
+
Effect.sync(() => {
|
|
265
|
+
const message = errorMessage(error)
|
|
266
|
+
log.error("worktree bootstrap failed", { directory: info.directory, message })
|
|
267
|
+
GlobalBus.emit("event", {
|
|
268
|
+
directory: info.directory,
|
|
269
|
+
project: ctx.project.id,
|
|
270
|
+
workspace: workspaceID,
|
|
271
|
+
payload: { type: Event.Failed.type, properties: { message } },
|
|
272
|
+
})
|
|
273
|
+
return false
|
|
274
|
+
}),
|
|
275
|
+
),
|
|
276
|
+
)
|
|
277
|
+
if (!booted) return
|
|
278
|
+
|
|
279
|
+
GlobalBus.emit("event", {
|
|
280
|
+
directory: info.directory,
|
|
281
|
+
project: ctx.project.id,
|
|
282
|
+
workspace: workspaceID,
|
|
283
|
+
payload: {
|
|
284
|
+
type: Event.Ready.type,
|
|
285
|
+
properties: { name: info.name, ...(info.branch ? { branch: info.branch } : {}) },
|
|
286
|
+
},
|
|
287
|
+
})
|
|
288
|
+
|
|
289
|
+
yield* runStartScripts(info.directory, { projectID, extra })
|
|
290
|
+
})
|
|
291
|
+
|
|
292
|
+
const createFromInfo = Effect.fn("Worktree.createFromInfo")(function* (info: Info, startCommand?: string) {
|
|
293
|
+
yield* setup(info)
|
|
294
|
+
yield* boot(info, startCommand).pipe(
|
|
295
|
+
Effect.catchCause((cause) => Effect.sync(() => log.error("worktree bootstrap failed", { cause }))),
|
|
296
|
+
Effect.forkIn(scope),
|
|
297
|
+
)
|
|
298
|
+
})
|
|
299
|
+
|
|
300
|
+
const create = Effect.fn("Worktree.create")(function* (input?: CreateInput) {
|
|
301
|
+
const info = yield* makeWorktreeInfo({ name: input?.name })
|
|
302
|
+
yield* createFromInfo(info, input?.startCommand)
|
|
303
|
+
return info
|
|
304
|
+
})
|
|
305
|
+
|
|
306
|
+
const canonical = Effect.fnUntraced(function* (input: string) {
|
|
307
|
+
const abs = pathSvc.resolve(input)
|
|
308
|
+
const real = yield* fs.realPath(abs).pipe(Effect.catch(() => Effect.succeed(abs)))
|
|
309
|
+
const normalized = pathSvc.normalize(real)
|
|
310
|
+
return process.platform === "win32" ? normalized.toLowerCase() : normalized
|
|
311
|
+
})
|
|
312
|
+
|
|
313
|
+
function parseWorktreeList(text: string) {
|
|
314
|
+
return text
|
|
315
|
+
.split("\n")
|
|
316
|
+
.map((line) => line.trim())
|
|
317
|
+
.reduce<{ path?: string; branch?: string }[]>((acc, line) => {
|
|
318
|
+
if (!line) return acc
|
|
319
|
+
if (line.startsWith("worktree ")) {
|
|
320
|
+
acc.push({ path: line.slice("worktree ".length).trim() })
|
|
321
|
+
return acc
|
|
322
|
+
}
|
|
323
|
+
const current = acc[acc.length - 1]
|
|
324
|
+
if (!current) return acc
|
|
325
|
+
if (line.startsWith("branch ")) {
|
|
326
|
+
current.branch = line.slice("branch ".length).trim()
|
|
327
|
+
}
|
|
328
|
+
return acc
|
|
329
|
+
}, [])
|
|
330
|
+
}
|
|
331
|
+
|
|
332
|
+
const locateWorktree = Effect.fnUntraced(function* (
|
|
333
|
+
entries: { path?: string; branch?: string }[],
|
|
334
|
+
directory: string,
|
|
335
|
+
) {
|
|
336
|
+
for (const item of entries) {
|
|
337
|
+
if (!item.path) continue
|
|
338
|
+
const key = yield* canonical(item.path)
|
|
339
|
+
if (key === directory) return item
|
|
340
|
+
}
|
|
341
|
+
return undefined
|
|
342
|
+
})
|
|
343
|
+
|
|
344
|
+
const list = Effect.fn("Worktree.list")(function* () {
|
|
345
|
+
const ctx = yield* InstanceState.context
|
|
346
|
+
if (ctx.project.vcs !== "git") {
|
|
347
|
+
return []
|
|
348
|
+
}
|
|
349
|
+
|
|
350
|
+
const result = yield* git(["worktree", "list", "--porcelain"], { cwd: ctx.worktree })
|
|
351
|
+
if (result.code !== 0) {
|
|
352
|
+
return yield* new ListFailedError({ message: result.stderr || result.text || "Failed to read git worktrees" })
|
|
353
|
+
}
|
|
354
|
+
|
|
355
|
+
const primary = yield* canonical(ctx.worktree)
|
|
356
|
+
const primaryName = pathSvc.basename(primary).toLowerCase()
|
|
357
|
+
return yield* Effect.forEach(parseWorktreeList(result.text), (entry) =>
|
|
358
|
+
Effect.gen(function* () {
|
|
359
|
+
if (!entry.path) return undefined
|
|
360
|
+
const directory = yield* canonical(entry.path)
|
|
361
|
+
if (directory === primary) return undefined
|
|
362
|
+
const name = pathSvc.basename(directory).toLowerCase()
|
|
363
|
+
return {
|
|
364
|
+
name: name === primaryName ? pathSvc.basename(pathSvc.dirname(directory)) : name,
|
|
365
|
+
directory,
|
|
366
|
+
...(entry.branch ? { branch: entry.branch.replace(/^refs\/heads\//, "") } : {}),
|
|
367
|
+
}
|
|
368
|
+
}),
|
|
369
|
+
).pipe(Effect.map((items) => items.filter((item) => item !== undefined)))
|
|
370
|
+
})
|
|
371
|
+
|
|
372
|
+
function stopFsmonitor(target: string) {
|
|
373
|
+
return fs.exists(target).pipe(
|
|
374
|
+
Effect.orDie,
|
|
375
|
+
Effect.flatMap((exists) => (exists ? git(["fsmonitor--daemon", "stop"], { cwd: target }) : Effect.void)),
|
|
376
|
+
)
|
|
377
|
+
}
|
|
378
|
+
|
|
379
|
+
function cleanDirectory(target: string) {
|
|
380
|
+
return Effect.tryPromise({
|
|
381
|
+
try: () =>
|
|
382
|
+
import("fs/promises").then((fsp) =>
|
|
383
|
+
fsp.rm(target, { recursive: true, force: true, maxRetries: 5, retryDelay: 100 }),
|
|
384
|
+
),
|
|
385
|
+
catch: (error) =>
|
|
386
|
+
new RemoveFailedError({ message: errorMessage(error) || "Failed to remove git worktree directory" }),
|
|
387
|
+
})
|
|
388
|
+
}
|
|
389
|
+
|
|
390
|
+
const remove = Effect.fn("Worktree.remove")(function* (input: RemoveInput) {
|
|
391
|
+
const ctx = yield* InstanceState.context
|
|
392
|
+
if (ctx.project.vcs !== "git") {
|
|
393
|
+
return yield* new NotGitError({ message: "Worktrees are only supported for git projects" })
|
|
394
|
+
}
|
|
395
|
+
|
|
396
|
+
const directory = yield* canonical(input.directory)
|
|
397
|
+
|
|
398
|
+
const list = yield* git(["worktree", "list", "--porcelain"], { cwd: ctx.worktree })
|
|
399
|
+
if (list.code !== 0) {
|
|
400
|
+
return yield* new RemoveFailedError({ message: list.stderr || list.text || "Failed to read git worktrees" })
|
|
401
|
+
}
|
|
402
|
+
|
|
403
|
+
const entries = parseWorktreeList(list.text)
|
|
404
|
+
const entry = yield* locateWorktree(entries, directory)
|
|
405
|
+
|
|
406
|
+
if (!entry?.path) {
|
|
407
|
+
const directoryExists = yield* fs.exists(directory).pipe(Effect.orDie)
|
|
408
|
+
if (directoryExists) {
|
|
409
|
+
yield* stopFsmonitor(directory)
|
|
410
|
+
yield* cleanDirectory(directory)
|
|
411
|
+
}
|
|
412
|
+
return true
|
|
413
|
+
}
|
|
414
|
+
|
|
415
|
+
yield* stopFsmonitor(entry.path)
|
|
416
|
+
const removed = yield* git(["worktree", "remove", "--force", entry.path], { cwd: ctx.worktree })
|
|
417
|
+
if (removed.code !== 0) {
|
|
418
|
+
const next = yield* git(["worktree", "list", "--porcelain"], { cwd: ctx.worktree })
|
|
419
|
+
if (next.code !== 0) {
|
|
420
|
+
return yield* new RemoveFailedError({
|
|
421
|
+
message: removed.stderr || removed.text || next.stderr || next.text || "Failed to remove git worktree",
|
|
422
|
+
})
|
|
423
|
+
}
|
|
424
|
+
|
|
425
|
+
const stale = yield* locateWorktree(parseWorktreeList(next.text), directory)
|
|
426
|
+
if (stale?.path) {
|
|
427
|
+
return yield* new RemoveFailedError({
|
|
428
|
+
message: removed.stderr || removed.text || "Failed to remove git worktree",
|
|
429
|
+
})
|
|
430
|
+
}
|
|
431
|
+
}
|
|
432
|
+
|
|
433
|
+
yield* cleanDirectory(entry.path)
|
|
434
|
+
|
|
435
|
+
const branch = entry.branch?.replace(/^refs\/heads\//, "")
|
|
436
|
+
if (branch) {
|
|
437
|
+
const deleted = yield* git(["branch", "-D", branch], { cwd: ctx.worktree })
|
|
438
|
+
if (deleted.code !== 0) {
|
|
439
|
+
return yield* new RemoveFailedError({
|
|
440
|
+
message: deleted.stderr || deleted.text || "Failed to delete worktree branch",
|
|
441
|
+
})
|
|
442
|
+
}
|
|
443
|
+
}
|
|
444
|
+
|
|
445
|
+
return true
|
|
446
|
+
})
|
|
447
|
+
|
|
448
|
+
const gitExpect = Effect.fnUntraced(function* (
|
|
449
|
+
args: string[],
|
|
450
|
+
opts: { cwd: string },
|
|
451
|
+
error: (r: GitResult) => Error,
|
|
452
|
+
) {
|
|
453
|
+
const result = yield* git(args, opts)
|
|
454
|
+
if (result.code !== 0) return yield* error(result)
|
|
455
|
+
return result
|
|
456
|
+
})
|
|
457
|
+
|
|
458
|
+
const runStartCommand = Effect.fnUntraced(
|
|
459
|
+
function* (directory: string, cmd: string) {
|
|
460
|
+
const [shell, args] = process.platform === "win32" ? ["cmd", ["/c", cmd]] : ["bash", ["-lc", cmd]]
|
|
461
|
+
const result = yield* appProcess.run(
|
|
462
|
+
ChildProcess.make(shell, args as string[], { cwd: directory, extendEnv: true, stdin: "ignore" }),
|
|
463
|
+
)
|
|
464
|
+
return { code: result.exitCode, stderr: result.stderr.toString("utf8") }
|
|
465
|
+
},
|
|
466
|
+
Effect.catch(() => Effect.succeed({ code: 1, stderr: "" })),
|
|
467
|
+
)
|
|
468
|
+
|
|
469
|
+
const runStartScript = Effect.fnUntraced(function* (directory: string, cmd: string, kind: string) {
|
|
470
|
+
const text = cmd.trim()
|
|
471
|
+
if (!text) return true
|
|
472
|
+
const result = yield* runStartCommand(directory, text)
|
|
473
|
+
if (result.code === 0) return true
|
|
474
|
+
log.error("worktree start command failed", { kind, directory, message: result.stderr })
|
|
475
|
+
return false
|
|
476
|
+
})
|
|
477
|
+
|
|
478
|
+
const runStartScripts = Effect.fnUntraced(function* (
|
|
479
|
+
directory: string,
|
|
480
|
+
input: { projectID: ProjectID; extra?: string },
|
|
481
|
+
) {
|
|
482
|
+
const row = yield* Effect.sync(() =>
|
|
483
|
+
Database.use((db) => db.select().from(ProjectTable).where(eq(ProjectTable.id, input.projectID)).get()),
|
|
484
|
+
)
|
|
485
|
+
const project = row ? Project.fromRow(row) : undefined
|
|
486
|
+
const startup = project?.commands?.start?.trim() ?? ""
|
|
487
|
+
const ok = yield* runStartScript(directory, startup, "project")
|
|
488
|
+
if (!ok) return false
|
|
489
|
+
yield* runStartScript(directory, input.extra ?? "", "worktree")
|
|
490
|
+
return true
|
|
491
|
+
})
|
|
492
|
+
|
|
493
|
+
const prune = Effect.fnUntraced(function* (root: string, entries: string[]) {
|
|
494
|
+
const base = yield* canonical(root)
|
|
495
|
+
yield* Effect.forEach(
|
|
496
|
+
entries,
|
|
497
|
+
(entry) =>
|
|
498
|
+
Effect.gen(function* () {
|
|
499
|
+
const target = yield* canonical(pathSvc.resolve(root, entry))
|
|
500
|
+
if (target === base) return
|
|
501
|
+
if (!target.startsWith(`${base}${pathSvc.sep}`)) return
|
|
502
|
+
yield* fs.remove(target, { recursive: true }).pipe(Effect.ignore)
|
|
503
|
+
}),
|
|
504
|
+
{ concurrency: "unbounded" },
|
|
505
|
+
)
|
|
506
|
+
})
|
|
507
|
+
|
|
508
|
+
const sweep = Effect.fnUntraced(function* (root: string) {
|
|
509
|
+
const first = yield* git(["clean", "-ffdx"], { cwd: root })
|
|
510
|
+
if (first.code === 0) return first
|
|
511
|
+
|
|
512
|
+
const entries = failedRemoves(first.stderr, first.text)
|
|
513
|
+
if (!entries.length) return first
|
|
514
|
+
|
|
515
|
+
yield* prune(root, entries)
|
|
516
|
+
return yield* git(["clean", "-ffdx"], { cwd: root })
|
|
517
|
+
})
|
|
518
|
+
|
|
519
|
+
const reset = Effect.fn("Worktree.reset")(function* (input: ResetInput) {
|
|
520
|
+
const ctx = yield* InstanceState.context
|
|
521
|
+
if (ctx.project.vcs !== "git") {
|
|
522
|
+
return yield* new NotGitError({ message: "Worktrees are only supported for git projects" })
|
|
523
|
+
}
|
|
524
|
+
|
|
525
|
+
const directory = yield* canonical(input.directory)
|
|
526
|
+
const primary = yield* canonical(ctx.worktree)
|
|
527
|
+
if (directory === primary) {
|
|
528
|
+
return yield* new ResetFailedError({ message: "Cannot reset the primary workspace" })
|
|
529
|
+
}
|
|
530
|
+
|
|
531
|
+
const list = yield* git(["worktree", "list", "--porcelain"], { cwd: ctx.worktree })
|
|
532
|
+
if (list.code !== 0) {
|
|
533
|
+
return yield* new ResetFailedError({ message: list.stderr || list.text || "Failed to read git worktrees" })
|
|
534
|
+
}
|
|
535
|
+
|
|
536
|
+
const entry = yield* locateWorktree(parseWorktreeList(list.text), directory)
|
|
537
|
+
if (!entry?.path) {
|
|
538
|
+
return yield* new ResetFailedError({ message: "Worktree not found" })
|
|
539
|
+
}
|
|
540
|
+
|
|
541
|
+
const worktreePath = entry.path
|
|
542
|
+
|
|
543
|
+
const base = yield* gitSvc.defaultBranch(ctx.worktree)
|
|
544
|
+
if (!base) {
|
|
545
|
+
return yield* new ResetFailedError({ message: "Default branch not found" })
|
|
546
|
+
}
|
|
547
|
+
|
|
548
|
+
const sep = base.ref.indexOf("/")
|
|
549
|
+
if (base.ref !== base.name && sep > 0) {
|
|
550
|
+
const remote = base.ref.slice(0, sep)
|
|
551
|
+
const branch = base.ref.slice(sep + 1)
|
|
552
|
+
yield* gitExpect(
|
|
553
|
+
["fetch", remote, branch],
|
|
554
|
+
{ cwd: ctx.worktree },
|
|
555
|
+
(r) => new ResetFailedError({ message: r.stderr || r.text || `Failed to fetch ${base.ref}` }),
|
|
556
|
+
)
|
|
557
|
+
}
|
|
558
|
+
|
|
559
|
+
yield* gitExpect(
|
|
560
|
+
["reset", "--hard", base.ref],
|
|
561
|
+
{ cwd: worktreePath },
|
|
562
|
+
(r) => new ResetFailedError({ message: r.stderr || r.text || "Failed to reset worktree to target" }),
|
|
563
|
+
)
|
|
564
|
+
|
|
565
|
+
const cleanResult = yield* sweep(worktreePath)
|
|
566
|
+
if (cleanResult.code !== 0) {
|
|
567
|
+
return yield* new ResetFailedError({
|
|
568
|
+
message: cleanResult.stderr || cleanResult.text || "Failed to clean worktree",
|
|
569
|
+
})
|
|
570
|
+
}
|
|
571
|
+
|
|
572
|
+
yield* gitExpect(
|
|
573
|
+
["submodule", "update", "--init", "--recursive", "--force"],
|
|
574
|
+
{ cwd: worktreePath },
|
|
575
|
+
(r) => new ResetFailedError({ message: r.stderr || r.text || "Failed to update submodules" }),
|
|
576
|
+
)
|
|
577
|
+
|
|
578
|
+
yield* gitExpect(
|
|
579
|
+
["submodule", "foreach", "--recursive", "git", "reset", "--hard"],
|
|
580
|
+
{ cwd: worktreePath },
|
|
581
|
+
(r) => new ResetFailedError({ message: r.stderr || r.text || "Failed to reset submodules" }),
|
|
582
|
+
)
|
|
583
|
+
|
|
584
|
+
yield* gitExpect(
|
|
585
|
+
["submodule", "foreach", "--recursive", "git", "clean", "-fdx"],
|
|
586
|
+
{ cwd: worktreePath },
|
|
587
|
+
(r) => new ResetFailedError({ message: r.stderr || r.text || "Failed to clean submodules" }),
|
|
588
|
+
)
|
|
589
|
+
|
|
590
|
+
const status = yield* git(["-c", "core.fsmonitor=false", "status", "--porcelain=v1"], { cwd: worktreePath })
|
|
591
|
+
if (status.code !== 0) {
|
|
592
|
+
return yield* new ResetFailedError({ message: status.stderr || status.text || "Failed to read git status" })
|
|
593
|
+
}
|
|
594
|
+
|
|
595
|
+
if (status.text.trim()) {
|
|
596
|
+
return yield* new ResetFailedError({ message: `Worktree reset left local changes:\n${status.text.trim()}` })
|
|
597
|
+
}
|
|
598
|
+
|
|
599
|
+
yield* runStartScripts(worktreePath, { projectID: ctx.project.id }).pipe(
|
|
600
|
+
Effect.catchCause((cause) => Effect.sync(() => log.error("worktree start task failed", { cause }))),
|
|
601
|
+
Effect.forkIn(scope),
|
|
602
|
+
)
|
|
603
|
+
|
|
604
|
+
return true
|
|
605
|
+
})
|
|
606
|
+
|
|
607
|
+
return Service.of({ makeWorktreeInfo, createFromInfo, create, list, remove, reset })
|
|
608
|
+
}),
|
|
609
|
+
)
|
|
610
|
+
|
|
611
|
+
export const appLayer = layer.pipe(
|
|
612
|
+
Layer.provide(Git.defaultLayer),
|
|
613
|
+
Layer.provide(AppProcess.defaultLayer),
|
|
614
|
+
Layer.provide(Project.defaultLayer),
|
|
615
|
+
Layer.provide(AppFileSystem.defaultLayer),
|
|
616
|
+
Layer.provide(NodePath.layer),
|
|
617
|
+
)
|
|
618
|
+
|
|
619
|
+
export const defaultLayer = appLayer.pipe(Layer.provide(InstanceLayer.layer))
|
|
620
|
+
|
|
621
|
+
export * as Worktree from "."
|