@jingyi0605/codingns 0.2.0 → 0.3.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/README.md +44 -0
- package/bin/codingns.mjs +918 -55
- package/dist/public/assets/{TerminalPage-BlbQuWi1.js → TerminalPage-Dfw1QUqW.js} +19 -19
- package/dist/public/assets/index-DR2rPNi7.css +1 -0
- package/dist/public/assets/index-DTOruahn.js +114 -0
- package/dist/public/index.html +2 -2
- package/dist/server/config/env.d.ts +3 -0
- package/dist/server/config/env.js +35 -3
- package/dist/server/config/env.js.map +1 -1
- package/dist/server/modules/assistant-capability/assistant-capability-controller.d.ts +89 -0
- package/dist/server/modules/assistant-capability/assistant-capability-controller.js +138 -0
- package/dist/server/modules/assistant-capability/assistant-capability-controller.js.map +1 -0
- package/dist/server/modules/assistant-capability/assistant-capability-service.d.ts +115 -0
- package/dist/server/modules/assistant-capability/assistant-capability-service.js +241 -0
- package/dist/server/modules/assistant-capability/assistant-capability-service.js.map +1 -0
- package/dist/server/modules/butler/butler-control-session-service.d.ts +3 -1
- package/dist/server/modules/butler/butler-control-session-service.js +96 -33
- package/dist/server/modules/butler/butler-control-session-service.js.map +1 -1
- package/dist/server/modules/butler/butler-follow-up-scheduler.d.ts +9 -0
- package/dist/server/modules/butler/butler-follow-up-scheduler.js +47 -11
- package/dist/server/modules/butler/butler-follow-up-scheduler.js.map +1 -1
- package/dist/server/modules/butler/butler-follow-up-service.d.ts +7 -1
- package/dist/server/modules/butler/butler-follow-up-service.js +10 -0
- package/dist/server/modules/butler/butler-follow-up-service.js.map +1 -1
- package/dist/server/modules/butler/butler-session-service.d.ts +3 -1
- package/dist/server/modules/butler/butler-session-service.js +82 -16
- package/dist/server/modules/butler/butler-session-service.js.map +1 -1
- package/dist/server/modules/butler/butler-session-summary-service.d.ts +8 -1
- package/dist/server/modules/butler/butler-session-summary-service.js +34 -7
- package/dist/server/modules/butler/butler-session-summary-service.js.map +1 -1
- package/dist/server/modules/butler/context-aggregator.js +44 -13
- package/dist/server/modules/butler/context-aggregator.js.map +1 -1
- package/dist/server/modules/butler/patrol-scheduler.d.ts +9 -0
- package/dist/server/modules/butler/patrol-scheduler.js +63 -9
- package/dist/server/modules/butler/patrol-scheduler.js.map +1 -1
- package/dist/server/modules/butler/session-summary-scheduler.d.ts +9 -0
- package/dist/server/modules/butler/session-summary-scheduler.js +47 -11
- package/dist/server/modules/butler/session-summary-scheduler.js.map +1 -1
- package/dist/server/modules/debug-target/debug-runtime-reconciliation-scheduler.d.ts +38 -0
- package/dist/server/modules/debug-target/debug-runtime-reconciliation-scheduler.js +99 -0
- package/dist/server/modules/debug-target/debug-runtime-reconciliation-scheduler.js.map +1 -0
- package/dist/server/modules/debug-target/debug-target-controller.d.ts +70 -0
- package/dist/server/modules/debug-target/debug-target-controller.js +113 -0
- package/dist/server/modules/debug-target/debug-target-controller.js.map +1 -0
- package/dist/server/modules/debug-target/debug-target-service.d.ts +105 -0
- package/dist/server/modules/debug-target/debug-target-service.js +1644 -0
- package/dist/server/modules/debug-target/debug-target-service.js.map +1 -0
- package/dist/server/modules/debug-target/framework-compatibility-matrix.d.ts +4 -0
- package/dist/server/modules/debug-target/framework-compatibility-matrix.js +45 -0
- package/dist/server/modules/debug-target/framework-compatibility-matrix.js.map +1 -0
- package/dist/server/modules/debug-target/launch-adapter-registry.d.ts +25 -0
- package/dist/server/modules/debug-target/launch-adapter-registry.js +445 -0
- package/dist/server/modules/debug-target/launch-adapter-registry.js.map +1 -0
- package/dist/server/modules/file/file-constants.d.ts +1 -0
- package/dist/server/modules/file/file-constants.js +1 -0
- package/dist/server/modules/file/file-constants.js.map +1 -1
- package/dist/server/modules/file/file-content-service.d.ts +2 -1
- package/dist/server/modules/file/file-content-service.js +53 -0
- package/dist/server/modules/file/file-content-service.js.map +1 -1
- package/dist/server/modules/file/file-controller.js +12 -3
- package/dist/server/modules/file/file-controller.js.map +1 -1
- package/dist/server/modules/file/file-preview-link-service.js +6 -37
- package/dist/server/modules/file/file-preview-link-service.js.map +1 -1
- package/dist/server/modules/file/file-preview-service.d.ts +6 -12
- package/dist/server/modules/file/file-preview-service.js +114 -28
- package/dist/server/modules/file/file-preview-service.js.map +1 -1
- package/dist/server/modules/file/file-preview-types.d.ts +37 -0
- package/dist/server/modules/file/file-preview-types.js +84 -0
- package/dist/server/modules/file/file-preview-types.js.map +1 -0
- package/dist/server/modules/git/commit-orchestrator.d.ts +4 -1
- package/dist/server/modules/git/commit-orchestrator.js +18 -1
- package/dist/server/modules/git/commit-orchestrator.js.map +1 -1
- package/dist/server/modules/git/git-auth.d.ts +25 -0
- package/dist/server/modules/git/git-auth.js +88 -0
- package/dist/server/modules/git/git-auth.js.map +1 -0
- package/dist/server/modules/git/git-controller.d.ts +12 -0
- package/dist/server/modules/git/git-controller.js +18 -1
- package/dist/server/modules/git/git-controller.js.map +1 -1
- package/dist/server/modules/git/git-read-service.d.ts +3 -1
- package/dist/server/modules/git/git-read-service.js +119 -2
- package/dist/server/modules/git/git-read-service.js.map +1 -1
- package/dist/server/modules/git/git-remote-credential-service.d.ts +9 -0
- package/dist/server/modules/git/git-remote-credential-service.js +76 -0
- package/dist/server/modules/git/git-remote-credential-service.js.map +1 -0
- package/dist/server/modules/git/git-write-service.d.ts +5 -2
- package/dist/server/modules/git/git-write-service.js +33 -17
- package/dist/server/modules/git/git-write-service.js.map +1 -1
- package/dist/server/modules/git/types.d.ts +26 -0
- package/dist/server/modules/git/workspace-repo-guard.js +3 -2
- package/dist/server/modules/git/workspace-repo-guard.js.map +1 -1
- package/dist/server/modules/provider/codex-model-options.d.ts +3 -1
- package/dist/server/modules/provider/codex-model-options.js +4 -1
- package/dist/server/modules/provider/codex-model-options.js.map +1 -1
- package/dist/server/modules/provider/opencode-model-options.d.ts +3 -1
- package/dist/server/modules/provider/opencode-model-options.js +5 -1
- package/dist/server/modules/provider/opencode-model-options.js.map +1 -1
- package/dist/server/modules/provider/provider-discovery-helper-client.d.ts +24 -0
- package/dist/server/modules/provider/provider-discovery-helper-client.js +14 -0
- package/dist/server/modules/provider/provider-discovery-helper-client.js.map +1 -1
- package/dist/server/modules/provider/provider-discovery-helper-process.js +54 -0
- package/dist/server/modules/provider/provider-discovery-helper-process.js.map +1 -1
- package/dist/server/modules/sessions/session-activity-authority-service.js +13 -1
- package/dist/server/modules/sessions/session-activity-authority-service.js.map +1 -1
- package/dist/server/modules/sessions/session-activity-inspector.js +21 -5
- package/dist/server/modules/sessions/session-activity-inspector.js.map +1 -1
- package/dist/server/modules/sessions/session-controller.d.ts +5 -0
- package/dist/server/modules/sessions/session-controller.js +16 -0
- package/dist/server/modules/sessions/session-controller.js.map +1 -1
- package/dist/server/modules/sessions/session-history-service.d.ts +27 -7
- package/dist/server/modules/sessions/session-history-service.js +439 -81
- package/dist/server/modules/sessions/session-history-service.js.map +1 -1
- package/dist/server/modules/sessions/session-live-runtime-service.d.ts +2 -1
- package/dist/server/modules/sessions/session-live-runtime-service.js +12 -0
- package/dist/server/modules/sessions/session-live-runtime-service.js.map +1 -1
- package/dist/server/modules/skills/skill-controller.d.ts +23 -0
- package/dist/server/modules/skills/skill-controller.js +35 -0
- package/dist/server/modules/skills/skill-controller.js.map +1 -0
- package/dist/server/modules/skills/skill-manager-service.d.ts +86 -0
- package/dist/server/modules/skills/skill-manager-service.js +557 -0
- package/dist/server/modules/skills/skill-manager-service.js.map +1 -0
- package/dist/server/modules/skills/skill-reconciler.d.ts +21 -0
- package/dist/server/modules/skills/skill-reconciler.js +99 -0
- package/dist/server/modules/skills/skill-reconciler.js.map +1 -0
- package/dist/server/modules/skills/skill-sync-planner.d.ts +8 -0
- package/dist/server/modules/skills/skill-sync-planner.js +20 -0
- package/dist/server/modules/skills/skill-sync-planner.js.map +1 -0
- package/dist/server/modules/skills/skill-target-adapter.d.ts +34 -0
- package/dist/server/modules/skills/skill-target-adapter.js +65 -0
- package/dist/server/modules/skills/skill-target-adapter.js.map +1 -0
- package/dist/server/modules/tailscale/tailscale-controller.d.ts +15 -0
- package/dist/server/modules/tailscale/tailscale-controller.js +33 -0
- package/dist/server/modules/tailscale/tailscale-controller.js.map +1 -0
- package/dist/server/modules/tailscale/tailscale-helper-client.d.ts +41 -0
- package/dist/server/modules/tailscale/tailscale-helper-client.js +135 -0
- package/dist/server/modules/tailscale/tailscale-helper-client.js.map +1 -0
- package/dist/server/modules/tailscale/tailscale-helper-process.d.ts +1 -0
- package/dist/server/modules/tailscale/tailscale-helper-process.js +327 -0
- package/dist/server/modules/tailscale/tailscale-helper-process.js.map +1 -0
- package/dist/server/modules/tailscale/tailscale-manager.d.ts +41 -0
- package/dist/server/modules/tailscale/tailscale-manager.js +259 -0
- package/dist/server/modules/tailscale/tailscale-manager.js.map +1 -0
- package/dist/server/modules/tailscale/tailscale-service.d.ts +43 -0
- package/dist/server/modules/tailscale/tailscale-service.js +201 -0
- package/dist/server/modules/tailscale/tailscale-service.js.map +1 -0
- package/dist/server/modules/tasks/event-loop-monitor.d.ts +21 -0
- package/dist/server/modules/tasks/event-loop-monitor.js +64 -0
- package/dist/server/modules/tasks/event-loop-monitor.js.map +1 -0
- package/dist/server/modules/tasks/observability-controller.d.ts +30 -0
- package/dist/server/modules/tasks/observability-controller.js +44 -0
- package/dist/server/modules/tasks/observability-controller.js.map +1 -0
- package/dist/server/modules/tasks/observability-service.d.ts +32 -0
- package/dist/server/modules/tasks/observability-service.js +104 -0
- package/dist/server/modules/tasks/observability-service.js.map +1 -0
- package/dist/server/modules/tasks/scheduler-metrics.d.ts +41 -0
- package/dist/server/modules/tasks/scheduler-metrics.js +92 -0
- package/dist/server/modules/tasks/scheduler-metrics.js.map +1 -0
- package/dist/server/modules/tasks/task-activity-log.d.ts +39 -0
- package/dist/server/modules/tasks/task-activity-log.js +43 -0
- package/dist/server/modules/tasks/task-activity-log.js.map +1 -0
- package/dist/server/modules/tasks/task-helper-client.d.ts +11 -0
- package/dist/server/modules/tasks/task-helper-client.js +132 -0
- package/dist/server/modules/tasks/task-helper-client.js.map +1 -0
- package/dist/server/modules/tasks/task-helper-process-handlers.d.ts +16 -0
- package/dist/server/modules/tasks/task-helper-process-handlers.js +14 -0
- package/dist/server/modules/tasks/task-helper-process-handlers.js.map +1 -0
- package/dist/server/modules/tasks/task-helper-process.d.ts +1 -0
- package/dist/server/modules/tasks/task-helper-process.js +49 -0
- package/dist/server/modules/tasks/task-helper-process.js.map +1 -0
- package/dist/server/modules/tasks/task-lane-executors.d.ts +2 -0
- package/dist/server/modules/tasks/task-lane-executors.js +15 -0
- package/dist/server/modules/tasks/task-lane-executors.js.map +1 -0
- package/dist/server/modules/tasks/task-manager.d.ts +15 -0
- package/dist/server/modules/tasks/task-manager.js +36 -0
- package/dist/server/modules/tasks/task-manager.js.map +1 -0
- package/dist/server/modules/tasks/task-metrics.d.ts +9 -0
- package/dist/server/modules/tasks/task-metrics.js +81 -0
- package/dist/server/modules/tasks/task-metrics.js.map +1 -0
- package/dist/server/modules/tasks/task-registry.d.ts +7 -0
- package/dist/server/modules/tasks/task-registry.js +21 -0
- package/dist/server/modules/tasks/task-registry.js.map +1 -0
- package/dist/server/modules/tasks/task-scheduler.d.ts +31 -0
- package/dist/server/modules/tasks/task-scheduler.js +473 -0
- package/dist/server/modules/tasks/task-scheduler.js.map +1 -0
- package/dist/server/modules/tasks/task-types.d.ts +106 -0
- package/dist/server/modules/tasks/task-types.js +23 -0
- package/dist/server/modules/tasks/task-types.js.map +1 -0
- package/dist/server/modules/terminal/command-template-service.d.ts +4 -0
- package/dist/server/modules/terminal/command-template-service.js +5 -3
- package/dist/server/modules/terminal/command-template-service.js.map +1 -1
- package/dist/server/modules/terminal/runtime/terminal-log-spooler.d.ts +7 -3
- package/dist/server/modules/terminal/runtime/terminal-log-spooler.js +95 -15
- package/dist/server/modules/terminal/runtime/terminal-log-spooler.js.map +1 -1
- package/dist/server/modules/terminal/runtime/terminal-log-writer-client.d.ts +21 -0
- package/dist/server/modules/terminal/runtime/terminal-log-writer-client.js +144 -0
- package/dist/server/modules/terminal/runtime/terminal-log-writer-client.js.map +1 -0
- package/dist/server/modules/terminal/runtime/terminal-log-writer-process.d.ts +1 -0
- package/dist/server/modules/terminal/runtime/terminal-log-writer-process.js +187 -0
- package/dist/server/modules/terminal/runtime/terminal-log-writer-process.js.map +1 -0
- package/dist/server/modules/terminal/terminal-service.d.ts +12 -0
- package/dist/server/modules/terminal/terminal-service.js +34 -17
- package/dist/server/modules/terminal/terminal-service.js.map +1 -1
- package/dist/server/modules/workbench/workbench-service.d.ts +23 -2
- package/dist/server/modules/workbench/workbench-service.js +126 -15
- package/dist/server/modules/workbench/workbench-service.js.map +1 -1
- package/dist/server/modules/workbench/workspace-panel-snapshot-service.d.ts +5 -1
- package/dist/server/modules/workbench/workspace-panel-snapshot-service.js +88 -19
- package/dist/server/modules/workbench/workspace-panel-snapshot-service.js.map +1 -1
- package/dist/server/modules/workspace/workspace-code-composition.d.ts +2 -0
- package/dist/server/modules/workspace/workspace-code-composition.js +154 -0
- package/dist/server/modules/workspace/workspace-code-composition.js.map +1 -0
- package/dist/server/modules/workspace/workspace-controller.d.ts +14 -0
- package/dist/server/modules/workspace/workspace-controller.js +19 -0
- package/dist/server/modules/workspace/workspace-controller.js.map +1 -1
- package/dist/server/modules/workspace/workspace-service.d.ts +21 -14
- package/dist/server/modules/workspace/workspace-service.js +183 -234
- package/dist/server/modules/workspace/workspace-service.js.map +1 -1
- package/dist/server/modules/worktree/worktree-cleanup-service.d.ts +35 -0
- package/dist/server/modules/worktree/worktree-cleanup-service.js +210 -0
- package/dist/server/modules/worktree/worktree-cleanup-service.js.map +1 -0
- package/dist/server/modules/worktree/worktree-controller.d.ts +44 -0
- package/dist/server/modules/worktree/worktree-controller.js +40 -0
- package/dist/server/modules/worktree/worktree-controller.js.map +1 -0
- package/dist/server/modules/worktree/worktree-manager.d.ts +34 -0
- package/dist/server/modules/worktree/worktree-manager.js +292 -0
- package/dist/server/modules/worktree/worktree-manager.js.map +1 -0
- package/dist/server/modules/worktree/worktree-merge-service.d.ts +52 -0
- package/dist/server/modules/worktree/worktree-merge-service.js +293 -0
- package/dist/server/modules/worktree/worktree-merge-service.js.map +1 -0
- package/dist/server/modules/worktree/worktree-sync-service.d.ts +23 -0
- package/dist/server/modules/worktree/worktree-sync-service.js +166 -0
- package/dist/server/modules/worktree/worktree-sync-service.js.map +1 -0
- package/dist/server/routes/assistant.d.ts +3 -0
- package/dist/server/routes/assistant.js +15 -0
- package/dist/server/routes/assistant.js.map +1 -0
- package/dist/server/routes/debug-targets.d.ts +3 -0
- package/dist/server/routes/debug-targets.js +15 -0
- package/dist/server/routes/debug-targets.js.map +1 -0
- package/dist/server/routes/git.js +2 -0
- package/dist/server/routes/git.js.map +1 -1
- package/dist/server/routes/observability.d.ts +3 -0
- package/dist/server/routes/observability.js +7 -0
- package/dist/server/routes/observability.js.map +1 -0
- package/dist/server/routes/skills.d.ts +3 -0
- package/dist/server/routes/skills.js +7 -0
- package/dist/server/routes/skills.js.map +1 -0
- package/dist/server/routes/system.d.ts +3 -0
- package/dist/server/routes/system.js +9 -0
- package/dist/server/routes/system.js.map +1 -0
- package/dist/server/routes/workspaces.js +2 -0
- package/dist/server/routes/workspaces.js.map +1 -1
- package/dist/server/routes/worktrees.d.ts +3 -0
- package/dist/server/routes/worktrees.js +8 -0
- package/dist/server/routes/worktrees.js.map +1 -0
- package/dist/server/server/create-server.d.ts +46 -0
- package/dist/server/server/create-server.js +141 -12
- package/dist/server/server/create-server.js.map +1 -1
- package/dist/server/shared/utils/command-availability.d.ts +1 -0
- package/dist/server/shared/utils/command-availability.js +26 -3
- package/dist/server/shared/utils/command-availability.js.map +1 -1
- package/dist/server/shared/utils/secret-box.d.ts +2 -0
- package/dist/server/shared/utils/secret-box.js +29 -0
- package/dist/server/shared/utils/secret-box.js.map +1 -0
- package/dist/server/shared/utils/terminal-debug-log.js +5 -3
- package/dist/server/shared/utils/terminal-debug-log.js.map +1 -1
- package/dist/server/storage/repositories/ai-fallback-edit-repository.d.ts +11 -0
- package/dist/server/storage/repositories/ai-fallback-edit-repository.js +118 -0
- package/dist/server/storage/repositories/ai-fallback-edit-repository.js.map +1 -0
- package/dist/server/storage/repositories/debug-runtime-session-repository.d.ts +11 -0
- package/dist/server/storage/repositories/debug-runtime-session-repository.js +100 -0
- package/dist/server/storage/repositories/debug-runtime-session-repository.js.map +1 -0
- package/dist/server/storage/repositories/debug-service-repository.d.ts +9 -0
- package/dist/server/storage/repositories/debug-service-repository.js +99 -0
- package/dist/server/storage/repositories/debug-service-repository.js.map +1 -0
- package/dist/server/storage/repositories/debug-target-repository.d.ts +11 -0
- package/dist/server/storage/repositories/debug-target-repository.js +100 -0
- package/dist/server/storage/repositories/debug-target-repository.js.map +1 -0
- package/dist/server/storage/repositories/framework-analysis-result-repository.d.ts +9 -0
- package/dist/server/storage/repositories/framework-analysis-result-repository.js +98 -0
- package/dist/server/storage/repositories/framework-analysis-result-repository.js.map +1 -0
- package/dist/server/storage/repositories/git-remote-credential-repository.d.ts +9 -0
- package/dist/server/storage/repositories/git-remote-credential-repository.js +51 -0
- package/dist/server/storage/repositories/git-remote-credential-repository.js.map +1 -0
- package/dist/server/storage/repositories/instance-tailscale-repository.d.ts +10 -0
- package/dist/server/storage/repositories/instance-tailscale-repository.js +112 -0
- package/dist/server/storage/repositories/instance-tailscale-repository.js.map +1 -0
- package/dist/server/storage/repositories/managed-skill-repository.d.ts +11 -0
- package/dist/server/storage/repositories/managed-skill-repository.js +102 -0
- package/dist/server/storage/repositories/managed-skill-repository.js.map +1 -0
- package/dist/server/storage/repositories/port-lease-repository.d.ts +12 -0
- package/dist/server/storage/repositories/port-lease-repository.js +124 -0
- package/dist/server/storage/repositories/port-lease-repository.js.map +1 -0
- package/dist/server/storage/repositories/runtime-binding-repository.d.ts +10 -0
- package/dist/server/storage/repositories/runtime-binding-repository.js +89 -0
- package/dist/server/storage/repositories/runtime-binding-repository.js.map +1 -0
- package/dist/server/storage/repositories/skill-target-binding-repository.d.ts +10 -0
- package/dist/server/storage/repositories/skill-target-binding-repository.js +77 -0
- package/dist/server/storage/repositories/skill-target-binding-repository.js.map +1 -0
- package/dist/server/storage/repositories/terminal-command-template-repository.js +77 -4
- package/dist/server/storage/repositories/terminal-command-template-repository.js.map +1 -1
- package/dist/server/storage/repositories/terminal-instance-repository.js +89 -7
- package/dist/server/storage/repositories/terminal-instance-repository.js.map +1 -1
- package/dist/server/storage/repositories/workspace-navigation-state-repository.d.ts +9 -0
- package/dist/server/storage/repositories/workspace-navigation-state-repository.js +49 -0
- package/dist/server/storage/repositories/workspace-navigation-state-repository.js.map +1 -0
- package/dist/server/storage/repositories/workspace-repository.d.ts +7 -1
- package/dist/server/storage/repositories/workspace-repository.js +32 -8
- package/dist/server/storage/repositories/workspace-repository.js.map +1 -1
- package/dist/server/storage/repositories/workspace-worktree-repository.d.ts +13 -0
- package/dist/server/storage/repositories/workspace-worktree-repository.js +158 -0
- package/dist/server/storage/repositories/workspace-worktree-repository.js.map +1 -0
- package/dist/server/storage/sqlite/client.js +311 -0
- package/dist/server/storage/sqlite/client.js.map +1 -1
- package/dist/server/storage/sqlite/schema.sql +303 -0
- package/dist/server/types/domain.d.ts +303 -0
- package/dist/server/ws/workbench-ws-hub.js +33 -9
- package/dist/server/ws/workbench-ws-hub.js.map +1 -1
- package/dist/server/ws/ws-server.js +4 -4
- package/dist/server/ws/ws-server.js.map +1 -1
- package/node_modules/@codingns/session-sync-core/dist/providers/claude-code.js +18 -6
- package/node_modules/@codingns/session-sync-core/dist/providers/claude-code.js.map +1 -1
- package/node_modules/@codingns/session-sync-core/dist/providers/codex.js +14 -2
- package/node_modules/@codingns/session-sync-core/dist/providers/codex.js.map +1 -1
- package/node_modules/@codingns/session-sync-core/dist/providers/opencode.js +25 -1
- package/node_modules/@codingns/session-sync-core/dist/providers/opencode.js.map +1 -1
- package/node_modules/@codingns/session-sync-core/dist/types.d.ts +6 -0
- package/package.json +1 -1
- package/dist/public/assets/index-1VIm8lVL.css +0 -1
- package/dist/public/assets/index-Dti93O2S.js +0 -109
|
@@ -0,0 +1,292 @@
|
|
|
1
|
+
import fs from "node:fs";
|
|
2
|
+
import path from "node:path";
|
|
3
|
+
import { AppError } from "../../shared/errors/app-error.js";
|
|
4
|
+
import { nowIso } from "../../shared/utils/time.js";
|
|
5
|
+
const WORKTREE_CREATE_TIMEOUT_MS = 30_000;
|
|
6
|
+
export class WorktreeManager {
|
|
7
|
+
workspaceService;
|
|
8
|
+
workspaceWorktreeRepository;
|
|
9
|
+
gitReadService;
|
|
10
|
+
gitCommandRunner;
|
|
11
|
+
constructor(workspaceService, workspaceWorktreeRepository, gitReadService, gitCommandRunner) {
|
|
12
|
+
this.workspaceService = workspaceService;
|
|
13
|
+
this.workspaceWorktreeRepository = workspaceWorktreeRepository;
|
|
14
|
+
this.gitReadService = gitReadService;
|
|
15
|
+
this.gitCommandRunner = gitCommandRunner;
|
|
16
|
+
}
|
|
17
|
+
async create(input) {
|
|
18
|
+
const sourceWorkspaceId = input.sourceWorkspaceId.trim();
|
|
19
|
+
const branchName = normalizeBranchName(input.branchName);
|
|
20
|
+
if (!sourceWorkspaceId) {
|
|
21
|
+
throw new AppError({
|
|
22
|
+
statusCode: 400,
|
|
23
|
+
errorCode: "INVALID_INPUT",
|
|
24
|
+
detail: "来源工作区不能为空",
|
|
25
|
+
field: "sourceWorkspaceId"
|
|
26
|
+
});
|
|
27
|
+
}
|
|
28
|
+
const sourceWorkspace = this.workspaceService.getWorkspaceOrThrow(sourceWorkspaceId);
|
|
29
|
+
const sourceMeta = this.workspaceWorktreeRepository.findByWorkspaceId(sourceWorkspaceId);
|
|
30
|
+
const rootWorkspace = this.workspaceService.getWorkspaceOrThrow(sourceMeta?.rootWorkspaceId ?? sourceWorkspaceId);
|
|
31
|
+
const sourceStatus = await this.gitReadService.getStatus(sourceWorkspaceId);
|
|
32
|
+
if (sourceStatus.snapshot.isDirty) {
|
|
33
|
+
throw new AppError({
|
|
34
|
+
statusCode: 409,
|
|
35
|
+
errorCode: "WORKTREE_SOURCE_DIRTY",
|
|
36
|
+
detail: "来源工作区存在未提交改动,不能直接创建子工作树"
|
|
37
|
+
});
|
|
38
|
+
}
|
|
39
|
+
await this.ensureBranchNameValid(sourceStatus.snapshot.repoRoot, sourceWorkspaceId, branchName);
|
|
40
|
+
await this.ensureBranchDoesNotExist(sourceStatus.snapshot.repoRoot, sourceWorkspaceId, branchName);
|
|
41
|
+
const baseRef = (input.baseRef?.trim() || sourceStatus.snapshot.branch || "HEAD").trim();
|
|
42
|
+
const baseCommit = await this.resolveBaseCommit(sourceStatus.snapshot.repoRoot, sourceWorkspaceId, baseRef);
|
|
43
|
+
const targetPath = buildTargetPath(rootWorkspace.path, branchName);
|
|
44
|
+
ensureTargetPathSafe(rootWorkspace.path, targetPath);
|
|
45
|
+
if (fs.existsSync(targetPath)) {
|
|
46
|
+
throw new AppError({
|
|
47
|
+
statusCode: 409,
|
|
48
|
+
errorCode: "WORKTREE_PATH_CONFLICT",
|
|
49
|
+
detail: "目标工作树目录已经存在,不能直接覆盖"
|
|
50
|
+
});
|
|
51
|
+
}
|
|
52
|
+
const timestamp = nowIso();
|
|
53
|
+
const displayName = normalizeDisplayName(input.displayName, branchName);
|
|
54
|
+
let createdWorkspace = null;
|
|
55
|
+
let worktreeCreated = false;
|
|
56
|
+
try {
|
|
57
|
+
fs.mkdirSync(path.dirname(targetPath), { recursive: true });
|
|
58
|
+
await this.gitCommandRunner.run(sourceStatus.snapshot.repoRoot, ["worktree", "add", "-b", branchName, targetPath, baseRef], {
|
|
59
|
+
timeoutMs: WORKTREE_CREATE_TIMEOUT_MS,
|
|
60
|
+
workspaceId: sourceWorkspaceId,
|
|
61
|
+
operation: "worktree.create"
|
|
62
|
+
});
|
|
63
|
+
worktreeCreated = true;
|
|
64
|
+
createdWorkspace = this.workspaceService.importWorkspace(targetPath, displayName);
|
|
65
|
+
const headCommit = await this.resolveHeadCommit(targetPath, createdWorkspace.id);
|
|
66
|
+
const meta = this.workspaceWorktreeRepository.create({
|
|
67
|
+
workspaceId: createdWorkspace.id,
|
|
68
|
+
rootWorkspaceId: rootWorkspace.id,
|
|
69
|
+
parentWorkspaceId: sourceWorkspace.id,
|
|
70
|
+
sourceWorkspaceId: sourceWorkspace.id,
|
|
71
|
+
mergeTargetWorkspaceId: sourceWorkspace.id,
|
|
72
|
+
branchName,
|
|
73
|
+
baseRef,
|
|
74
|
+
baseCommit,
|
|
75
|
+
headCommit,
|
|
76
|
+
displayName,
|
|
77
|
+
depth: sourceMeta ? sourceMeta.depth + 1 : 1,
|
|
78
|
+
lifecycleStatus: "active",
|
|
79
|
+
mergedAt: null,
|
|
80
|
+
removedAt: null,
|
|
81
|
+
createdAt: timestamp,
|
|
82
|
+
updatedAt: timestamp
|
|
83
|
+
});
|
|
84
|
+
return {
|
|
85
|
+
workspace: createdWorkspace,
|
|
86
|
+
meta
|
|
87
|
+
};
|
|
88
|
+
}
|
|
89
|
+
catch (error) {
|
|
90
|
+
await this.rollbackCreate(sourceStatus.snapshot.repoRoot, sourceWorkspaceId, branchName, targetPath, createdWorkspace, worktreeCreated);
|
|
91
|
+
throw mapCreateWorktreeError(error);
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
getTree(rootWorkspaceId) {
|
|
95
|
+
const requestedWorkspaceId = rootWorkspaceId.trim();
|
|
96
|
+
if (!requestedWorkspaceId) {
|
|
97
|
+
throw new AppError({
|
|
98
|
+
statusCode: 400,
|
|
99
|
+
errorCode: "INVALID_INPUT",
|
|
100
|
+
detail: "根工作区不能为空",
|
|
101
|
+
field: "rootWorkspaceId"
|
|
102
|
+
});
|
|
103
|
+
}
|
|
104
|
+
const rootMeta = this.workspaceWorktreeRepository.findByWorkspaceId(requestedWorkspaceId);
|
|
105
|
+
const resolvedRootWorkspaceId = rootMeta?.rootWorkspaceId ?? requestedWorkspaceId;
|
|
106
|
+
this.workspaceService.getWorkspaceOrThrow(resolvedRootWorkspaceId);
|
|
107
|
+
const records = this.workspaceWorktreeRepository
|
|
108
|
+
.listByRootWorkspaceId(resolvedRootWorkspaceId)
|
|
109
|
+
.filter((record) => record.lifecycleStatus !== "removed");
|
|
110
|
+
const nodeByWorkspaceId = new Map();
|
|
111
|
+
const roots = [];
|
|
112
|
+
for (const record of records) {
|
|
113
|
+
nodeByWorkspaceId.set(record.workspaceId, {
|
|
114
|
+
workspace: this.workspaceService.getWorkspaceOrThrow(record.workspaceId),
|
|
115
|
+
meta: record,
|
|
116
|
+
children: []
|
|
117
|
+
});
|
|
118
|
+
}
|
|
119
|
+
for (const record of records) {
|
|
120
|
+
const currentNode = nodeByWorkspaceId.get(record.workspaceId);
|
|
121
|
+
if (!currentNode) {
|
|
122
|
+
continue;
|
|
123
|
+
}
|
|
124
|
+
if (record.parentWorkspaceId === resolvedRootWorkspaceId) {
|
|
125
|
+
roots.push(currentNode);
|
|
126
|
+
continue;
|
|
127
|
+
}
|
|
128
|
+
const parentNode = nodeByWorkspaceId.get(record.parentWorkspaceId);
|
|
129
|
+
if (parentNode) {
|
|
130
|
+
parentNode.children.push(currentNode);
|
|
131
|
+
continue;
|
|
132
|
+
}
|
|
133
|
+
roots.push(currentNode);
|
|
134
|
+
}
|
|
135
|
+
return roots;
|
|
136
|
+
}
|
|
137
|
+
async ensureBranchNameValid(repoRoot, workspaceId, branchName) {
|
|
138
|
+
const result = await this.gitCommandRunner.run(repoRoot, ["check-ref-format", "--branch", branchName], {
|
|
139
|
+
allowNonZeroExit: true,
|
|
140
|
+
workspaceId,
|
|
141
|
+
operation: "worktree.create.validateBranch"
|
|
142
|
+
});
|
|
143
|
+
if (result.exitCode === 0) {
|
|
144
|
+
return;
|
|
145
|
+
}
|
|
146
|
+
throw new AppError({
|
|
147
|
+
statusCode: 400,
|
|
148
|
+
errorCode: "INVALID_INPUT",
|
|
149
|
+
detail: "工作树分支名不合法",
|
|
150
|
+
field: "branchName"
|
|
151
|
+
});
|
|
152
|
+
}
|
|
153
|
+
async ensureBranchDoesNotExist(repoRoot, workspaceId, branchName) {
|
|
154
|
+
const result = await this.gitCommandRunner.run(repoRoot, ["show-ref", "--verify", "--quiet", `refs/heads/${branchName}`], {
|
|
155
|
+
allowNonZeroExit: true,
|
|
156
|
+
workspaceId,
|
|
157
|
+
operation: "worktree.create.checkBranchExists"
|
|
158
|
+
});
|
|
159
|
+
if (result.exitCode !== 0) {
|
|
160
|
+
return;
|
|
161
|
+
}
|
|
162
|
+
throw new AppError({
|
|
163
|
+
statusCode: 409,
|
|
164
|
+
errorCode: "WORKTREE_BRANCH_EXISTS",
|
|
165
|
+
detail: "目标分支已经存在,不能重复创建工作树"
|
|
166
|
+
});
|
|
167
|
+
}
|
|
168
|
+
async resolveBaseCommit(repoRoot, workspaceId, baseRef) {
|
|
169
|
+
const result = await this.gitCommandRunner.run(repoRoot, ["rev-parse", "--verify", baseRef], {
|
|
170
|
+
allowNonZeroExit: true,
|
|
171
|
+
workspaceId,
|
|
172
|
+
operation: "worktree.create.resolveBaseRef"
|
|
173
|
+
});
|
|
174
|
+
const baseCommit = result.stdout.trim();
|
|
175
|
+
if (result.exitCode === 0 && baseCommit) {
|
|
176
|
+
return baseCommit;
|
|
177
|
+
}
|
|
178
|
+
throw new AppError({
|
|
179
|
+
statusCode: 404,
|
|
180
|
+
errorCode: "WORKTREE_BASE_REF_NOT_FOUND",
|
|
181
|
+
detail: "指定的 baseRef 不存在,不能创建工作树",
|
|
182
|
+
field: "baseRef"
|
|
183
|
+
});
|
|
184
|
+
}
|
|
185
|
+
async resolveHeadCommit(targetPath, workspaceId) {
|
|
186
|
+
const result = await this.gitCommandRunner.run(targetPath, ["rev-parse", "HEAD"], {
|
|
187
|
+
workspaceId,
|
|
188
|
+
operation: "worktree.create.resolveHeadCommit"
|
|
189
|
+
});
|
|
190
|
+
return result.stdout.trim();
|
|
191
|
+
}
|
|
192
|
+
async rollbackCreate(repoRoot, workspaceId, branchName, targetPath, createdWorkspace, worktreeCreated) {
|
|
193
|
+
if (createdWorkspace) {
|
|
194
|
+
this.workspaceService.removeWorkspace(createdWorkspace.id);
|
|
195
|
+
}
|
|
196
|
+
if (worktreeCreated || fs.existsSync(targetPath)) {
|
|
197
|
+
try {
|
|
198
|
+
await this.gitCommandRunner.run(repoRoot, ["worktree", "remove", "--force", targetPath], {
|
|
199
|
+
allowNonZeroExit: true,
|
|
200
|
+
workspaceId,
|
|
201
|
+
operation: "worktree.create.rollbackRemove"
|
|
202
|
+
});
|
|
203
|
+
}
|
|
204
|
+
catch {
|
|
205
|
+
// 回滚阶段不再向外抛,优先保住原始错误。
|
|
206
|
+
}
|
|
207
|
+
if (fs.existsSync(targetPath)) {
|
|
208
|
+
fs.rmSync(targetPath, { recursive: true, force: true });
|
|
209
|
+
}
|
|
210
|
+
}
|
|
211
|
+
try {
|
|
212
|
+
await this.gitCommandRunner.run(repoRoot, ["branch", "-D", branchName], {
|
|
213
|
+
allowNonZeroExit: true,
|
|
214
|
+
workspaceId,
|
|
215
|
+
operation: "worktree.create.rollbackDeleteBranch"
|
|
216
|
+
});
|
|
217
|
+
}
|
|
218
|
+
catch {
|
|
219
|
+
// 分支删除失败只保留残留,不能覆盖原始错误。
|
|
220
|
+
}
|
|
221
|
+
}
|
|
222
|
+
}
|
|
223
|
+
function normalizeBranchName(branchName) {
|
|
224
|
+
const normalized = branchName.trim();
|
|
225
|
+
if (!normalized) {
|
|
226
|
+
throw new AppError({
|
|
227
|
+
statusCode: 400,
|
|
228
|
+
errorCode: "INVALID_INPUT",
|
|
229
|
+
detail: "工作树分支名不能为空",
|
|
230
|
+
field: "branchName"
|
|
231
|
+
});
|
|
232
|
+
}
|
|
233
|
+
return normalized;
|
|
234
|
+
}
|
|
235
|
+
function normalizeDisplayName(displayName, branchName) {
|
|
236
|
+
return displayName?.trim() || branchName;
|
|
237
|
+
}
|
|
238
|
+
function buildTargetPath(rootWorkspacePath, branchName) {
|
|
239
|
+
const rootPath = path.resolve(rootWorkspacePath);
|
|
240
|
+
const rootParentPath = path.dirname(rootPath);
|
|
241
|
+
const rootName = path.basename(rootPath);
|
|
242
|
+
return path.join(rootParentPath, `${rootName}.worktrees`, sanitizePathSegment(branchName));
|
|
243
|
+
}
|
|
244
|
+
function ensureTargetPathSafe(rootWorkspacePath, targetPath) {
|
|
245
|
+
const normalizedRootPath = path.resolve(rootWorkspacePath);
|
|
246
|
+
const normalizedTargetPath = path.resolve(targetPath);
|
|
247
|
+
const relative = path.relative(normalizedRootPath, normalizedTargetPath);
|
|
248
|
+
if (relative === "" || (!relative.startsWith("..") && !path.isAbsolute(relative))) {
|
|
249
|
+
throw new AppError({
|
|
250
|
+
statusCode: 400,
|
|
251
|
+
errorCode: "WORKTREE_PATH_INVALID",
|
|
252
|
+
detail: "工作树目录不能落在根工作区目录内部"
|
|
253
|
+
});
|
|
254
|
+
}
|
|
255
|
+
}
|
|
256
|
+
function sanitizePathSegment(input) {
|
|
257
|
+
const sanitized = input
|
|
258
|
+
.trim()
|
|
259
|
+
.replace(/[\\/]+/g, "-")
|
|
260
|
+
.replace(/[^a-zA-Z0-9._-]+/g, "-")
|
|
261
|
+
.replace(/-+/g, "-")
|
|
262
|
+
.replace(/^-|-$/g, "");
|
|
263
|
+
return sanitized || "worktree";
|
|
264
|
+
}
|
|
265
|
+
function mapCreateWorktreeError(error) {
|
|
266
|
+
if (!(error instanceof AppError)) {
|
|
267
|
+
return error instanceof Error ? error : new Error(String(error));
|
|
268
|
+
}
|
|
269
|
+
if (error.errorCode !== "GIT_COMMAND_FAILED") {
|
|
270
|
+
return error;
|
|
271
|
+
}
|
|
272
|
+
if (error.message.includes("already exists")) {
|
|
273
|
+
return new AppError({
|
|
274
|
+
statusCode: 409,
|
|
275
|
+
errorCode: "WORKTREE_BRANCH_EXISTS",
|
|
276
|
+
detail: "目标分支已经存在,不能重复创建工作树"
|
|
277
|
+
});
|
|
278
|
+
}
|
|
279
|
+
if (error.message.includes("is already checked out")) {
|
|
280
|
+
return new AppError({
|
|
281
|
+
statusCode: 409,
|
|
282
|
+
errorCode: "WORKTREE_BRANCH_IN_USE",
|
|
283
|
+
detail: "目标分支已经被其他工作树占用,不能再次创建"
|
|
284
|
+
});
|
|
285
|
+
}
|
|
286
|
+
return new AppError({
|
|
287
|
+
statusCode: 500,
|
|
288
|
+
errorCode: "WORKTREE_CREATE_FAILED",
|
|
289
|
+
detail: error.message
|
|
290
|
+
});
|
|
291
|
+
}
|
|
292
|
+
//# sourceMappingURL=worktree-manager.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"worktree-manager.js","sourceRoot":"","sources":["../../../../src/modules/worktree/worktree-manager.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,52 @@
|
|
|
1
|
+
import type { Workspace, WorkspaceWorktreeRecord } from "../../types/domain.js";
|
|
2
|
+
import type { WorkspaceWorktreeRepository } from "../../storage/repositories/workspace-worktree-repository.js";
|
|
3
|
+
import type { WorkspaceService } from "../workspace/workspace-service.js";
|
|
4
|
+
import type { GitReadService } from "../git/git-read-service.js";
|
|
5
|
+
import type { GitCommandRunner } from "../git/git-command-runner.js";
|
|
6
|
+
import type { WorktreeSyncService } from "./worktree-sync-service.js";
|
|
7
|
+
export type WorktreeMergeBlockerCode = "SOURCE_NOT_ACTIVE" | "SOURCE_DIRTY" | "TARGET_DIRTY" | "HAS_ACTIVE_CHILDREN" | "NO_COMMITS_TO_MERGE" | "HAS_CONFLICTS";
|
|
8
|
+
export interface WorktreeMergeBlocker {
|
|
9
|
+
code: WorktreeMergeBlockerCode;
|
|
10
|
+
detail: string;
|
|
11
|
+
}
|
|
12
|
+
export interface WorktreeMergePreviewResult {
|
|
13
|
+
workspaceId: string;
|
|
14
|
+
sourceWorkspace: Workspace;
|
|
15
|
+
targetWorkspace: Workspace;
|
|
16
|
+
meta: WorkspaceWorktreeRecord;
|
|
17
|
+
sourceBranchName: string;
|
|
18
|
+
targetBranchName: string;
|
|
19
|
+
sourceHeadCommit: string | null;
|
|
20
|
+
targetHeadCommit: string | null;
|
|
21
|
+
mergeBaseCommit: string | null;
|
|
22
|
+
ahead: number;
|
|
23
|
+
behind: number;
|
|
24
|
+
hasConflicts: boolean;
|
|
25
|
+
conflictPaths: string[];
|
|
26
|
+
alreadyMerged: boolean;
|
|
27
|
+
canMerge: boolean;
|
|
28
|
+
blockers: WorktreeMergeBlocker[];
|
|
29
|
+
}
|
|
30
|
+
export interface WorktreeMergeApplyResult {
|
|
31
|
+
preview: WorktreeMergePreviewResult;
|
|
32
|
+
applied: boolean;
|
|
33
|
+
mergeCommit: string | null;
|
|
34
|
+
meta: WorkspaceWorktreeRecord;
|
|
35
|
+
}
|
|
36
|
+
export declare class WorktreeMergeService {
|
|
37
|
+
private readonly workspaceService;
|
|
38
|
+
private readonly workspaceWorktreeRepository;
|
|
39
|
+
private readonly gitReadService;
|
|
40
|
+
private readonly gitCommandRunner;
|
|
41
|
+
private readonly worktreeSyncService;
|
|
42
|
+
constructor(workspaceService: WorkspaceService, workspaceWorktreeRepository: WorkspaceWorktreeRepository, gitReadService: GitReadService, gitCommandRunner: GitCommandRunner, worktreeSyncService: WorktreeSyncService);
|
|
43
|
+
preview(workspaceId: string): Promise<WorktreeMergePreviewResult>;
|
|
44
|
+
apply(workspaceId: string): Promise<WorktreeMergeApplyResult>;
|
|
45
|
+
private resolveActiveWorktreeMeta;
|
|
46
|
+
private normalizePreviewMeta;
|
|
47
|
+
private resolveCommit;
|
|
48
|
+
private resolveMergeBase;
|
|
49
|
+
private resolveAheadBehind;
|
|
50
|
+
private isAncestor;
|
|
51
|
+
private detectConflictPaths;
|
|
52
|
+
}
|
|
@@ -0,0 +1,293 @@
|
|
|
1
|
+
import { AppError } from "../../shared/errors/app-error.js";
|
|
2
|
+
import { nowIso } from "../../shared/utils/time.js";
|
|
3
|
+
const WORKTREE_MERGE_TIMEOUT_MS = 60_000;
|
|
4
|
+
export class WorktreeMergeService {
|
|
5
|
+
workspaceService;
|
|
6
|
+
workspaceWorktreeRepository;
|
|
7
|
+
gitReadService;
|
|
8
|
+
gitCommandRunner;
|
|
9
|
+
worktreeSyncService;
|
|
10
|
+
constructor(workspaceService, workspaceWorktreeRepository, gitReadService, gitCommandRunner, worktreeSyncService) {
|
|
11
|
+
this.workspaceService = workspaceService;
|
|
12
|
+
this.workspaceWorktreeRepository = workspaceWorktreeRepository;
|
|
13
|
+
this.gitReadService = gitReadService;
|
|
14
|
+
this.gitCommandRunner = gitCommandRunner;
|
|
15
|
+
this.worktreeSyncService = worktreeSyncService;
|
|
16
|
+
}
|
|
17
|
+
async preview(workspaceId) {
|
|
18
|
+
const meta = await this.resolveActiveWorktreeMeta(workspaceId);
|
|
19
|
+
const sourceWorkspace = this.workspaceService.getWorkspaceOrThrow(meta.workspaceId);
|
|
20
|
+
const targetWorkspace = this.workspaceService.getWorkspaceOrThrow(meta.parentWorkspaceId);
|
|
21
|
+
const [sourceStatus, targetStatus] = await Promise.all([
|
|
22
|
+
this.gitReadService.getStatus(sourceWorkspace.id),
|
|
23
|
+
this.gitReadService.getStatus(targetWorkspace.id)
|
|
24
|
+
]);
|
|
25
|
+
const sourceHeadCommit = await this.resolveCommit(sourceWorkspace.path, sourceWorkspace.id, "HEAD");
|
|
26
|
+
const targetHeadCommit = await this.resolveCommit(targetWorkspace.path, targetWorkspace.id, "HEAD");
|
|
27
|
+
const mergeBaseCommit = sourceHeadCommit && targetHeadCommit
|
|
28
|
+
? await this.resolveMergeBase(targetWorkspace.path, targetWorkspace.id, targetHeadCommit, sourceHeadCommit)
|
|
29
|
+
: null;
|
|
30
|
+
const { ahead, behind } = sourceHeadCommit && targetHeadCommit
|
|
31
|
+
? await this.resolveAheadBehind(targetWorkspace.path, targetWorkspace.id, targetHeadCommit, sourceHeadCommit)
|
|
32
|
+
: { ahead: 0, behind: 0 };
|
|
33
|
+
const alreadyMerged = Boolean(sourceHeadCommit)
|
|
34
|
+
&& Boolean(targetHeadCommit)
|
|
35
|
+
&& await this.isAncestor(targetWorkspace.path, targetWorkspace.id, sourceHeadCommit ?? "", targetHeadCommit ?? "");
|
|
36
|
+
const childRecords = this.workspaceWorktreeRepository
|
|
37
|
+
.listByParentWorkspaceId(meta.workspaceId)
|
|
38
|
+
.filter((record) => record.lifecycleStatus === "active" || record.lifecycleStatus === "removing");
|
|
39
|
+
const blockers = [];
|
|
40
|
+
const normalizedMeta = this.normalizePreviewMeta(meta, alreadyMerged);
|
|
41
|
+
if (!alreadyMerged && normalizedMeta.lifecycleStatus !== "active") {
|
|
42
|
+
blockers.push({
|
|
43
|
+
code: "SOURCE_NOT_ACTIVE",
|
|
44
|
+
detail: "当前子工作树不是活跃状态,不能继续合并"
|
|
45
|
+
});
|
|
46
|
+
}
|
|
47
|
+
if (sourceStatus.snapshot.isDirty) {
|
|
48
|
+
blockers.push({
|
|
49
|
+
code: "SOURCE_DIRTY",
|
|
50
|
+
detail: "当前子工作树存在未提交改动,先提交或清理后再合并"
|
|
51
|
+
});
|
|
52
|
+
}
|
|
53
|
+
if (targetStatus.snapshot.isDirty) {
|
|
54
|
+
blockers.push({
|
|
55
|
+
code: "TARGET_DIRTY",
|
|
56
|
+
detail: "直接父工作区存在未提交改动,不能接收合并"
|
|
57
|
+
});
|
|
58
|
+
}
|
|
59
|
+
if (childRecords.length > 0) {
|
|
60
|
+
blockers.push({
|
|
61
|
+
code: "HAS_ACTIVE_CHILDREN",
|
|
62
|
+
detail: "当前子工作树下面还有活跃子节点,必须先从叶子节点开始回收"
|
|
63
|
+
});
|
|
64
|
+
}
|
|
65
|
+
if (!alreadyMerged && ahead === 0) {
|
|
66
|
+
blockers.push({
|
|
67
|
+
code: "NO_COMMITS_TO_MERGE",
|
|
68
|
+
detail: "当前子工作树没有领先父工作区的提交"
|
|
69
|
+
});
|
|
70
|
+
}
|
|
71
|
+
const conflictPaths = !alreadyMerged
|
|
72
|
+
&& ahead > 0
|
|
73
|
+
&& blockers.every((item) => item.code !== "SOURCE_DIRTY" && item.code !== "TARGET_DIRTY")
|
|
74
|
+
? await this.detectConflictPaths(targetWorkspace.path, targetWorkspace.id, targetHeadCommit, sourceHeadCommit)
|
|
75
|
+
: [];
|
|
76
|
+
if (conflictPaths.length > 0) {
|
|
77
|
+
blockers.push({
|
|
78
|
+
code: "HAS_CONFLICTS",
|
|
79
|
+
detail: `检测到合并冲突:${conflictPaths.join("、")}`
|
|
80
|
+
});
|
|
81
|
+
}
|
|
82
|
+
return {
|
|
83
|
+
workspaceId: meta.workspaceId,
|
|
84
|
+
sourceWorkspace,
|
|
85
|
+
targetWorkspace,
|
|
86
|
+
meta: normalizedMeta,
|
|
87
|
+
sourceBranchName: sourceStatus.snapshot.branch,
|
|
88
|
+
targetBranchName: targetStatus.snapshot.branch,
|
|
89
|
+
sourceHeadCommit,
|
|
90
|
+
targetHeadCommit,
|
|
91
|
+
mergeBaseCommit,
|
|
92
|
+
ahead,
|
|
93
|
+
behind,
|
|
94
|
+
hasConflicts: conflictPaths.length > 0,
|
|
95
|
+
conflictPaths,
|
|
96
|
+
alreadyMerged,
|
|
97
|
+
canMerge: blockers.length === 0 && !alreadyMerged,
|
|
98
|
+
blockers
|
|
99
|
+
};
|
|
100
|
+
}
|
|
101
|
+
async apply(workspaceId) {
|
|
102
|
+
const preview = await this.preview(workspaceId);
|
|
103
|
+
const meta = preview.meta;
|
|
104
|
+
if (!preview.alreadyMerged && preview.blockers.length > 0) {
|
|
105
|
+
const blocker = preview.blockers[0];
|
|
106
|
+
throw new AppError({
|
|
107
|
+
statusCode: 409,
|
|
108
|
+
errorCode: `WORKTREE_MERGE_${blocker.code}`,
|
|
109
|
+
detail: blocker.detail
|
|
110
|
+
});
|
|
111
|
+
}
|
|
112
|
+
const targetWorkspace = preview.targetWorkspace;
|
|
113
|
+
const timestamp = nowIso();
|
|
114
|
+
let applied = false;
|
|
115
|
+
if (!preview.alreadyMerged) {
|
|
116
|
+
const mergeResult = await this.gitCommandRunner.run(targetWorkspace.path, ["merge", "--no-ff", "--no-edit", preview.meta.branchName], {
|
|
117
|
+
allowNonZeroExit: true,
|
|
118
|
+
timeoutMs: WORKTREE_MERGE_TIMEOUT_MS,
|
|
119
|
+
workspaceId: targetWorkspace.id,
|
|
120
|
+
operation: "worktree.merge.apply"
|
|
121
|
+
});
|
|
122
|
+
if (mergeResult.exitCode !== 0) {
|
|
123
|
+
await this.gitCommandRunner.run(targetWorkspace.path, ["merge", "--abort"], {
|
|
124
|
+
allowNonZeroExit: true,
|
|
125
|
+
workspaceId: targetWorkspace.id,
|
|
126
|
+
operation: "worktree.merge.abort"
|
|
127
|
+
});
|
|
128
|
+
throw new AppError({
|
|
129
|
+
statusCode: 409,
|
|
130
|
+
errorCode: "WORKTREE_MERGE_APPLY_FAILED",
|
|
131
|
+
detail: mergeResult.stderr.trim() || mergeResult.stdout.trim() || "合并执行失败"
|
|
132
|
+
});
|
|
133
|
+
}
|
|
134
|
+
applied = true;
|
|
135
|
+
}
|
|
136
|
+
const mergeCommit = await this.resolveCommit(targetWorkspace.path, targetWorkspace.id, "HEAD");
|
|
137
|
+
const nextMeta = this.workspaceWorktreeRepository.update({
|
|
138
|
+
...meta,
|
|
139
|
+
lifecycleStatus: "merged",
|
|
140
|
+
mergedAt: meta.mergedAt ?? timestamp,
|
|
141
|
+
updatedAt: timestamp
|
|
142
|
+
});
|
|
143
|
+
if (!nextMeta) {
|
|
144
|
+
throw new AppError({
|
|
145
|
+
statusCode: 500,
|
|
146
|
+
errorCode: "WORKTREE_META_UPDATE_FAILED",
|
|
147
|
+
detail: "工作树合并成功,但元数据更新失败"
|
|
148
|
+
});
|
|
149
|
+
}
|
|
150
|
+
const parentMeta = this.workspaceWorktreeRepository.findByWorkspaceId(targetWorkspace.id);
|
|
151
|
+
if (parentMeta) {
|
|
152
|
+
this.workspaceWorktreeRepository.update({
|
|
153
|
+
...parentMeta,
|
|
154
|
+
headCommit: mergeCommit,
|
|
155
|
+
updatedAt: timestamp
|
|
156
|
+
});
|
|
157
|
+
}
|
|
158
|
+
return {
|
|
159
|
+
preview: {
|
|
160
|
+
...preview,
|
|
161
|
+
meta: nextMeta,
|
|
162
|
+
alreadyMerged: true,
|
|
163
|
+
canMerge: false,
|
|
164
|
+
blockers: [],
|
|
165
|
+
targetHeadCommit: mergeCommit
|
|
166
|
+
},
|
|
167
|
+
applied,
|
|
168
|
+
mergeCommit,
|
|
169
|
+
meta: nextMeta
|
|
170
|
+
};
|
|
171
|
+
}
|
|
172
|
+
async resolveActiveWorktreeMeta(workspaceId) {
|
|
173
|
+
const normalizedWorkspaceId = workspaceId.trim();
|
|
174
|
+
if (!normalizedWorkspaceId) {
|
|
175
|
+
throw new AppError({
|
|
176
|
+
statusCode: 400,
|
|
177
|
+
errorCode: "INVALID_INPUT",
|
|
178
|
+
detail: "工作树 workspaceId 不能为空",
|
|
179
|
+
field: "workspaceId"
|
|
180
|
+
});
|
|
181
|
+
}
|
|
182
|
+
const meta = this.workspaceWorktreeRepository.findByWorkspaceId(normalizedWorkspaceId);
|
|
183
|
+
if (!meta) {
|
|
184
|
+
throw new AppError({
|
|
185
|
+
statusCode: 404,
|
|
186
|
+
errorCode: "WORKTREE_NOT_FOUND",
|
|
187
|
+
detail: "指定工作区不是子工作树"
|
|
188
|
+
});
|
|
189
|
+
}
|
|
190
|
+
await this.worktreeSyncService.syncRoot(meta.rootWorkspaceId);
|
|
191
|
+
const nextMeta = this.workspaceWorktreeRepository.findByWorkspaceId(normalizedWorkspaceId);
|
|
192
|
+
if (!nextMeta || nextMeta.lifecycleStatus === "removed") {
|
|
193
|
+
throw new AppError({
|
|
194
|
+
statusCode: 404,
|
|
195
|
+
errorCode: "WORKTREE_NOT_FOUND",
|
|
196
|
+
detail: "指定工作树已经不存在"
|
|
197
|
+
});
|
|
198
|
+
}
|
|
199
|
+
return nextMeta;
|
|
200
|
+
}
|
|
201
|
+
normalizePreviewMeta(meta, alreadyMerged) {
|
|
202
|
+
if (alreadyMerged) {
|
|
203
|
+
if (meta.lifecycleStatus === "merged" && meta.mergedAt) {
|
|
204
|
+
return meta;
|
|
205
|
+
}
|
|
206
|
+
return (this.workspaceWorktreeRepository.update({
|
|
207
|
+
...meta,
|
|
208
|
+
lifecycleStatus: "merged",
|
|
209
|
+
mergedAt: meta.mergedAt ?? nowIso(),
|
|
210
|
+
updatedAt: nowIso()
|
|
211
|
+
}) ?? meta);
|
|
212
|
+
}
|
|
213
|
+
if (meta.lifecycleStatus === "active") {
|
|
214
|
+
return meta;
|
|
215
|
+
}
|
|
216
|
+
return (this.workspaceWorktreeRepository.update({
|
|
217
|
+
...meta,
|
|
218
|
+
lifecycleStatus: "active",
|
|
219
|
+
mergedAt: null,
|
|
220
|
+
updatedAt: nowIso()
|
|
221
|
+
}) ?? meta);
|
|
222
|
+
}
|
|
223
|
+
async resolveCommit(cwd, workspaceId, ref) {
|
|
224
|
+
const result = await this.gitCommandRunner.run(cwd, ["rev-parse", "--verify", ref], {
|
|
225
|
+
allowNonZeroExit: true,
|
|
226
|
+
workspaceId,
|
|
227
|
+
operation: "worktree.merge.resolveCommit"
|
|
228
|
+
});
|
|
229
|
+
return result.exitCode === 0 ? result.stdout.trim() || null : null;
|
|
230
|
+
}
|
|
231
|
+
async resolveMergeBase(cwd, workspaceId, targetHeadCommit, sourceHeadCommit) {
|
|
232
|
+
const result = await this.gitCommandRunner.run(cwd, ["merge-base", targetHeadCommit, sourceHeadCommit], {
|
|
233
|
+
allowNonZeroExit: true,
|
|
234
|
+
workspaceId,
|
|
235
|
+
operation: "worktree.merge.previewBase"
|
|
236
|
+
});
|
|
237
|
+
return result.exitCode === 0 ? result.stdout.trim() || null : null;
|
|
238
|
+
}
|
|
239
|
+
async resolveAheadBehind(cwd, workspaceId, targetHeadCommit, sourceHeadCommit) {
|
|
240
|
+
const result = await this.gitCommandRunner.run(cwd, ["rev-list", "--left-right", "--count", `${targetHeadCommit}...${sourceHeadCommit}`], {
|
|
241
|
+
workspaceId,
|
|
242
|
+
operation: "worktree.merge.previewAheadBehind"
|
|
243
|
+
});
|
|
244
|
+
const [behindRaw, aheadRaw] = result.stdout.trim().split(/\s+/);
|
|
245
|
+
return {
|
|
246
|
+
ahead: Number(aheadRaw ?? "0") || 0,
|
|
247
|
+
behind: Number(behindRaw ?? "0") || 0
|
|
248
|
+
};
|
|
249
|
+
}
|
|
250
|
+
async isAncestor(cwd, workspaceId, ancestorCommit, descendantCommit) {
|
|
251
|
+
const result = await this.gitCommandRunner.run(cwd, ["merge-base", "--is-ancestor", ancestorCommit, descendantCommit], {
|
|
252
|
+
allowNonZeroExit: true,
|
|
253
|
+
workspaceId,
|
|
254
|
+
operation: "worktree.merge.previewAncestor"
|
|
255
|
+
});
|
|
256
|
+
return result.exitCode === 0;
|
|
257
|
+
}
|
|
258
|
+
async detectConflictPaths(cwd, workspaceId, targetHeadCommit, sourceHeadCommit) {
|
|
259
|
+
if (!targetHeadCommit || !sourceHeadCommit) {
|
|
260
|
+
return [];
|
|
261
|
+
}
|
|
262
|
+
const result = await this.gitCommandRunner.run(cwd, ["merge-tree", "--write-tree", targetHeadCommit, sourceHeadCommit], {
|
|
263
|
+
allowNonZeroExit: true,
|
|
264
|
+
workspaceId,
|
|
265
|
+
operation: "worktree.merge.previewConflicts"
|
|
266
|
+
});
|
|
267
|
+
if (result.exitCode === 0) {
|
|
268
|
+
return [];
|
|
269
|
+
}
|
|
270
|
+
return Array.from(parseMergeConflictPaths(result.stdout, result.stderr));
|
|
271
|
+
}
|
|
272
|
+
}
|
|
273
|
+
function parseMergeConflictPaths(stdout, stderr) {
|
|
274
|
+
const paths = new Set();
|
|
275
|
+
const content = `${stdout}\n${stderr}`;
|
|
276
|
+
for (const line of content.split(/\r?\n/)) {
|
|
277
|
+
const trimmed = line.trim();
|
|
278
|
+
if (!trimmed) {
|
|
279
|
+
continue;
|
|
280
|
+
}
|
|
281
|
+
const stageMatch = trimmed.match(/^\d{6}\s+[0-9a-f]{40}\s+\d\t(.+)$/i);
|
|
282
|
+
if (stageMatch?.[1]) {
|
|
283
|
+
paths.add(stageMatch[1]);
|
|
284
|
+
continue;
|
|
285
|
+
}
|
|
286
|
+
const conflictMatch = trimmed.match(/Merge conflict in (.+)$/i);
|
|
287
|
+
if (conflictMatch?.[1]) {
|
|
288
|
+
paths.add(conflictMatch[1]);
|
|
289
|
+
}
|
|
290
|
+
}
|
|
291
|
+
return paths;
|
|
292
|
+
}
|
|
293
|
+
//# sourceMappingURL=worktree-merge-service.js.map
|