feat: Add gt config command for managing agent settings

Implements GitHub issue #127 - allow custom agent configuration
through a CLI interface instead of command-line aliases.

The gt config command provides:
- gt config agent list [--json]    List all agents
- gt config agent get <name>       Show agent configuration
- gt config agent set <name> <cmd> Set custom agent command
- gt config agent remove <name>    Remove custom agent
- gt config default-agent [name]   Get/set default agent

Users can now define custom agents (e.g., claude-glm) and
override built-in presets (claude, gemini, codex) through
town settings instead of shell aliases.

Changes:
- Add SaveTownSettings() to internal/config/loader.go
- Add internal/cmd/config.go with full config command implementation
- Add comprehensive unit tests for both SaveTownSettings and
  all config subcommands (17 test cases covering success and
  error scenarios)

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
Darko Luketic
2026-01-05 06:58:46 +01:00
parent f0c94db99e
commit 5787a16067
4 changed files with 1206 additions and 0 deletions

View File

@@ -970,3 +970,133 @@ func TestLoadRuntimeConfigFallsBackToDefaults(t *testing.T) {
t.Errorf("Command = %q, want %q (default)", rc.Command, "claude")
}
}
func TestSaveTownSettings(t *testing.T) {
t.Run("saves valid town settings", func(t *testing.T) {
tmpDir := t.TempDir()
settingsPath := filepath.Join(tmpDir, "settings", "config.json")
settings := &TownSettings{
Type: "town-settings",
Version: CurrentTownSettingsVersion,
DefaultAgent: "gemini",
Agents: map[string]*RuntimeConfig{
"my-agent": {
Command: "my-agent",
Args: []string{"--arg1", "--arg2"},
},
},
}
err := SaveTownSettings(settingsPath, settings)
if err != nil {
t.Fatalf("SaveTownSettings failed: %v", err)
}
// Verify file exists
data, err := os.ReadFile(settingsPath)
if err != nil {
t.Fatalf("reading settings file: %v", err)
}
// Verify it contains expected content
content := string(data)
if !strings.Contains(content, `"type": "town-settings"`) {
t.Errorf("missing type field")
}
if !strings.Contains(content, `"default_agent": "gemini"`) {
t.Errorf("missing default_agent field")
}
if !strings.Contains(content, `"my-agent"`) {
t.Errorf("missing custom agent")
}
})
t.Run("creates parent directories", func(t *testing.T) {
tmpDir := t.TempDir()
settingsPath := filepath.Join(tmpDir, "deeply", "nested", "settings", "config.json")
settings := NewTownSettings()
err := SaveTownSettings(settingsPath, settings)
if err != nil {
t.Fatalf("SaveTownSettings failed: %v", err)
}
// Verify file exists
if _, err := os.Stat(settingsPath); err != nil {
t.Errorf("settings file not created: %v", err)
}
})
t.Run("rejects invalid type", func(t *testing.T) {
tmpDir := t.TempDir()
settingsPath := filepath.Join(tmpDir, "config.json")
settings := &TownSettings{
Type: "invalid-type",
Version: CurrentTownSettingsVersion,
}
err := SaveTownSettings(settingsPath, settings)
if err == nil {
t.Error("expected error for invalid type")
}
})
t.Run("rejects unsupported version", func(t *testing.T) {
tmpDir := t.TempDir()
settingsPath := filepath.Join(tmpDir, "config.json")
settings := &TownSettings{
Type: "town-settings",
Version: CurrentTownSettingsVersion + 100,
}
err := SaveTownSettings(settingsPath, settings)
if err == nil {
t.Error("expected error for unsupported version")
}
})
t.Run("roundtrip save and load", func(t *testing.T) {
tmpDir := t.TempDir()
settingsPath := filepath.Join(tmpDir, "config.json")
original := &TownSettings{
Type: "town-settings",
Version: CurrentTownSettingsVersion,
DefaultAgent: "codex",
Agents: map[string]*RuntimeConfig{
"custom-1": {
Command: "custom-agent",
Args: []string{"--flag"},
},
},
}
err := SaveTownSettings(settingsPath, original)
if err != nil {
t.Fatalf("SaveTownSettings failed: %v", err)
}
loaded, err := LoadOrCreateTownSettings(settingsPath)
if err != nil {
t.Fatalf("LoadOrCreateTownSettings failed: %v", err)
}
if loaded.Type != original.Type {
t.Errorf("Type = %q, want %q", loaded.Type, original.Type)
}
if loaded.Version != original.Version {
t.Errorf("Version = %d, want %d", loaded.Version, original.Version)
}
if loaded.DefaultAgent != original.DefaultAgent {
t.Errorf("DefaultAgent = %q, want %q", loaded.DefaultAgent, original.DefaultAgent)
}
if len(loaded.Agents) != len(original.Agents) {
t.Errorf("Agents count = %d, want %d", len(loaded.Agents), len(original.Agents))
}
})
}