mirror of
https://github.com/cexll/myclaude.git
synced 2026-02-05 02:30:26 +08:00
修复 PR #53 中发现的问题,实现完整的多后端功能: **多后端功能完整性** - Claude/Gemini 后端支持 workdir (-C) 和 resume (--session-id) 参数 - 并行模式支持全局 --backend 参数和任务级 backend 配置 - 后端参数映射统一,支持 new/resume 两种模式 **安全控制** - Claude 后端默认启用 --dangerously-skip-permissions 以支持自动化 - 通过 CODEAGENT_SKIP_PERMISSIONS 环境变量控制权限检查 - 不同后端行为区分:Claude 默认跳过,Codex/Gemini 默认启用 **并发控制** - 新增 CODEAGENT_MAX_PARALLEL_WORKERS 环境变量限制并发数 - 实现 fail-fast context 取消机制 - Worker pool 防止资源耗尽,支持并发监控日志 **向后兼容** - 版本号统一管理,提供 codex-wrapper 兼容脚本 - 所有默认行为保持不变 - 支持渐进式迁移 **测试覆盖** - 总体覆盖率 93.4%(超过 90% 要求) - 新增后端参数、并行模式、并发控制测试用例 - 核心模块覆盖率:backend.go 100%, config.go 97.8%, executor.go 96.4% **文档更新** - 更新 skills/codeagent/SKILL.md 反映多后端和安全控制 - 添加 CHANGELOG.md 记录重要变更 - 更新 README 版本说明和安装脚本 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
123 lines
4.0 KiB
Go
123 lines
4.0 KiB
Go
package main
|
|
|
|
import (
|
|
"reflect"
|
|
"testing"
|
|
)
|
|
|
|
func TestClaudeBuildArgs_ModesAndPermissions(t *testing.T) {
|
|
backend := ClaudeBackend{}
|
|
|
|
t.Run("new mode uses workdir without skip by default", func(t *testing.T) {
|
|
cfg := &Config{Mode: "new", WorkDir: "/repo"}
|
|
got := backend.BuildArgs(cfg, "todo")
|
|
want := []string{"-p", "-C", "/repo", "--output-format", "stream-json", "--verbose", "todo"}
|
|
if !reflect.DeepEqual(got, want) {
|
|
t.Fatalf("got %v, want %v", got, want)
|
|
}
|
|
})
|
|
|
|
t.Run("new mode opt-in skip permissions with default workdir", func(t *testing.T) {
|
|
cfg := &Config{Mode: "new", SkipPermissions: true}
|
|
got := backend.BuildArgs(cfg, "-")
|
|
want := []string{"-p", "--dangerously-skip-permissions", "-C", defaultWorkdir, "--output-format", "stream-json", "--verbose", "-"}
|
|
if !reflect.DeepEqual(got, want) {
|
|
t.Fatalf("got %v, want %v", got, want)
|
|
}
|
|
})
|
|
|
|
t.Run("resume mode uses session id and omits workdir", func(t *testing.T) {
|
|
cfg := &Config{Mode: "resume", SessionID: "sid-123", WorkDir: "/ignored"}
|
|
got := backend.BuildArgs(cfg, "resume-task")
|
|
want := []string{"-p", "--session-id", "sid-123", "--output-format", "stream-json", "--verbose", "resume-task"}
|
|
if !reflect.DeepEqual(got, want) {
|
|
t.Fatalf("got %v, want %v", got, want)
|
|
}
|
|
})
|
|
|
|
t.Run("resume mode without session still returns base flags", func(t *testing.T) {
|
|
cfg := &Config{Mode: "resume", WorkDir: "/ignored"}
|
|
got := backend.BuildArgs(cfg, "follow-up")
|
|
want := []string{"-p", "--output-format", "stream-json", "--verbose", "follow-up"}
|
|
if !reflect.DeepEqual(got, want) {
|
|
t.Fatalf("got %v, want %v", got, want)
|
|
}
|
|
})
|
|
|
|
t.Run("nil config returns nil", func(t *testing.T) {
|
|
if backend.BuildArgs(nil, "ignored") != nil {
|
|
t.Fatalf("nil config should return nil args")
|
|
}
|
|
})
|
|
}
|
|
|
|
func TestClaudeBuildArgs_GeminiAndCodexModes(t *testing.T) {
|
|
t.Run("gemini new mode defaults workdir", func(t *testing.T) {
|
|
backend := GeminiBackend{}
|
|
cfg := &Config{Mode: "new", WorkDir: "/workspace"}
|
|
got := backend.BuildArgs(cfg, "task")
|
|
want := []string{"-o", "stream-json", "-y", "-C", "/workspace", "-p", "task"}
|
|
if !reflect.DeepEqual(got, want) {
|
|
t.Fatalf("got %v, want %v", got, want)
|
|
}
|
|
})
|
|
|
|
t.Run("gemini resume mode uses session id", func(t *testing.T) {
|
|
backend := GeminiBackend{}
|
|
cfg := &Config{Mode: "resume", SessionID: "sid-999"}
|
|
got := backend.BuildArgs(cfg, "resume")
|
|
want := []string{"-o", "stream-json", "-y", "--session-id", "sid-999", "-p", "resume"}
|
|
if !reflect.DeepEqual(got, want) {
|
|
t.Fatalf("got %v, want %v", got, want)
|
|
}
|
|
})
|
|
|
|
t.Run("gemini resume mode without session omits identifier", func(t *testing.T) {
|
|
backend := GeminiBackend{}
|
|
cfg := &Config{Mode: "resume"}
|
|
got := backend.BuildArgs(cfg, "resume")
|
|
want := []string{"-o", "stream-json", "-y", "-p", "resume"}
|
|
if !reflect.DeepEqual(got, want) {
|
|
t.Fatalf("got %v, want %v", got, want)
|
|
}
|
|
})
|
|
|
|
t.Run("gemini nil config returns nil", func(t *testing.T) {
|
|
backend := GeminiBackend{}
|
|
if backend.BuildArgs(nil, "ignored") != nil {
|
|
t.Fatalf("nil config should return nil args")
|
|
}
|
|
})
|
|
|
|
t.Run("codex build args passthrough remains intact", func(t *testing.T) {
|
|
backend := CodexBackend{}
|
|
cfg := &Config{Mode: "new", WorkDir: "/tmp"}
|
|
got := backend.BuildArgs(cfg, "task")
|
|
want := []string{"e", "--skip-git-repo-check", "-C", "/tmp", "--json", "task"}
|
|
if !reflect.DeepEqual(got, want) {
|
|
t.Fatalf("got %v, want %v", got, want)
|
|
}
|
|
})
|
|
}
|
|
|
|
func TestClaudeBuildArgs_BackendMetadata(t *testing.T) {
|
|
tests := []struct {
|
|
backend Backend
|
|
name string
|
|
command string
|
|
}{
|
|
{backend: CodexBackend{}, name: "codex", command: "codex"},
|
|
{backend: ClaudeBackend{}, name: "claude", command: "claude"},
|
|
{backend: GeminiBackend{}, name: "gemini", command: "gemini"},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
if got := tt.backend.Name(); got != tt.name {
|
|
t.Fatalf("Name() = %s, want %s", got, tt.name)
|
|
}
|
|
if got := tt.backend.Command(); got != tt.command {
|
|
t.Fatalf("Command() = %s, want %s", got, tt.command)
|
|
}
|
|
}
|
|
}
|