Files
goclaw/internal/config/config_load.go
T
viettranx 84b1b07634 refactor(config): centralize hardcoded ~/.goclaw paths via config resolution
Replace all hardcoded ~/.goclaw path constructions with configurable
sources (cfg.ResolvedDataDir() for service dirs, cfg.Agents.Defaults.Workspace
for agent workspaces). This fixes data persistence issues in Docker
deployments where paths differ from local dev.

- Add DataDir field to Config with ResolvedDataDir() resolver
- Add ResolvedDataDirFromEnv() package-level helper for packages without Config
- Populate StoreConfig.SkillsStorageDir (was never set, caused hardcoded fallback)
- Agent workspaces now use subdirectory format (workspace/{key}) for volume compatibility
- Remove dead GOCLAW_SESSIONS_STORAGE env/config (sessions moved to PostgreSQL)
- Fix deploy-stg.sh trailing space after backslash + remove deprecated GOCLAW_MODE
- Add GOCLAW_SKILLS_DIR override in docker-compose for volume persistence
2026-03-15 21:20:46 +07:00

441 lines
13 KiB
Go

package config
import (
"crypto/sha256"
"encoding/json"
"fmt"
"log/slog"
"os"
"path/filepath"
"strconv"
"strings"
"github.com/titanous/json5"
)
// Default returns a Config with sensible defaults.
func Default() *Config {
return &Config{
DataDir: "~/.goclaw/data",
Agents: AgentsConfig{
Defaults: AgentDefaults{
Workspace: "~/.goclaw/workspace",
RestrictToWorkspace: true,
Provider: "anthropic",
Model: "claude-sonnet-4-5-20250929",
MaxTokens: 8192,
Temperature: 0.7,
MaxToolIterations: 20,
MaxToolCalls: 25,
ContextWindow: 200000,
Subagents: &SubagentsConfig{
MaxConcurrent: 20,
MaxSpawnDepth: 1,
},
},
},
Channels: ChannelsConfig{
Telegram: TelegramConfig{
ReactionLevel: "full",
},
},
Gateway: GatewayConfig{
Host: "0.0.0.0",
Port: 18790,
MaxMessageChars: 32000,
RateLimitRPM: 20,
},
Tools: ToolsConfig{
Web: WebToolsConfig{
DuckDuckGo: DuckDuckGoConfig{Enabled: true, MaxResults: 5},
},
Browser: BrowserToolConfig{
Enabled: true,
Headless: true,
},
ExecApproval: ExecApprovalCfg{
Security: "full",
Ask: "off",
},
RateLimitPerHour: 150,
},
Sessions: SessionsConfig{},
}
}
// Load reads config from a JSON file, then overlays env vars.
func Load(path string) (*Config, error) {
cfg := Default()
data, err := os.ReadFile(path)
if err != nil {
if os.IsNotExist(err) {
cfg.applyEnvOverrides()
cfg.applyContextPruningDefaults()
return cfg, nil
}
return nil, fmt.Errorf("read config: %w", err)
}
if err := json5.Unmarshal(data, cfg); err != nil {
return nil, fmt.Errorf("parse config: %w", err)
}
cfg.applyEnvOverrides()
cfg.applyContextPruningDefaults()
return cfg, nil
}
// applyEnvOverrides overlays env vars onto the config.
// Env vars take precedence over file values.
func (c *Config) applyEnvOverrides() {
envStr := func(key string, dst *string) {
if v := os.Getenv(key); v != "" {
*dst = v
}
}
envStr("GOCLAW_ANTHROPIC_API_KEY", &c.Providers.Anthropic.APIKey)
envStr("GOCLAW_ANTHROPIC_BASE_URL", &c.Providers.Anthropic.APIBase)
envStr("GOCLAW_OPENAI_API_KEY", &c.Providers.OpenAI.APIKey)
envStr("GOCLAW_OPENAI_BASE_URL", &c.Providers.OpenAI.APIBase)
envStr("GOCLAW_OPENROUTER_API_KEY", &c.Providers.OpenRouter.APIKey)
envStr("GOCLAW_GROQ_API_KEY", &c.Providers.Groq.APIKey)
envStr("GOCLAW_DEEPSEEK_API_KEY", &c.Providers.DeepSeek.APIKey)
envStr("GOCLAW_GEMINI_API_KEY", &c.Providers.Gemini.APIKey)
envStr("GOCLAW_MISTRAL_API_KEY", &c.Providers.Mistral.APIKey)
envStr("GOCLAW_XAI_API_KEY", &c.Providers.XAI.APIKey)
envStr("GOCLAW_MINIMAX_API_KEY", &c.Providers.MiniMax.APIKey)
envStr("GOCLAW_COHERE_API_KEY", &c.Providers.Cohere.APIKey)
envStr("GOCLAW_PERPLEXITY_API_KEY", &c.Providers.Perplexity.APIKey)
envStr("GOCLAW_DASHSCOPE_API_KEY", &c.Providers.DashScope.APIKey)
envStr("GOCLAW_BAILIAN_API_KEY", &c.Providers.Bailian.APIKey)
envStr("GOCLAW_ZAI_API_KEY", &c.Providers.Zai.APIKey)
envStr("GOCLAW_ZAI_CODING_API_KEY", &c.Providers.ZaiCoding.APIKey)
envStr("GOCLAW_OLLAMA_HOST", &c.Providers.Ollama.Host)
envStr("GOCLAW_OLLAMA_CLOUD_API_KEY", &c.Providers.OllamaCloud.APIKey)
envStr("GOCLAW_OLLAMA_CLOUD_API_BASE", &c.Providers.OllamaCloud.APIBase)
envStr("GOCLAW_GATEWAY_TOKEN", &c.Gateway.Token)
envStr("GOCLAW_TELEGRAM_TOKEN", &c.Channels.Telegram.Token)
envStr("GOCLAW_DISCORD_TOKEN", &c.Channels.Discord.Token)
envStr("GOCLAW_ZALO_TOKEN", &c.Channels.Zalo.Token)
envStr("GOCLAW_LARK_APP_ID", &c.Channels.Feishu.AppID)
envStr("GOCLAW_LARK_APP_SECRET", &c.Channels.Feishu.AppSecret)
envStr("GOCLAW_LARK_ENCRYPT_KEY", &c.Channels.Feishu.EncryptKey)
envStr("GOCLAW_LARK_VERIFICATION_TOKEN", &c.Channels.Feishu.VerificationToken)
envStr("GOCLAW_WHATSAPP_BRIDGE_URL", &c.Channels.WhatsApp.BridgeURL)
envStr("GOCLAW_SLACK_BOT_TOKEN", &c.Channels.Slack.BotToken)
envStr("GOCLAW_SLACK_APP_TOKEN", &c.Channels.Slack.AppToken)
envStr("GOCLAW_SLACK_USER_TOKEN", &c.Channels.Slack.UserToken)
// TTS secrets
envStr("GOCLAW_TTS_OPENAI_API_KEY", &c.Tts.OpenAI.APIKey)
envStr("GOCLAW_TTS_ELEVENLABS_API_KEY", &c.Tts.ElevenLabs.APIKey)
envStr("GOCLAW_TTS_MINIMAX_API_KEY", &c.Tts.MiniMax.APIKey)
envStr("GOCLAW_TTS_MINIMAX_GROUP_ID", &c.Tts.MiniMax.GroupID)
// Auto-enable channels if credentials are provided via env
if c.Channels.Telegram.Token != "" {
c.Channels.Telegram.Enabled = true
}
if c.Channels.Discord.Token != "" {
c.Channels.Discord.Enabled = true
}
if c.Channels.Zalo.Token != "" {
c.Channels.Zalo.Enabled = true
}
if c.Channels.Feishu.AppID != "" && c.Channels.Feishu.AppSecret != "" {
c.Channels.Feishu.Enabled = true
}
if c.Channels.WhatsApp.BridgeURL != "" {
c.Channels.WhatsApp.Enabled = true
}
if c.Channels.Slack.BotToken != "" && c.Channels.Slack.AppToken != "" {
c.Channels.Slack.Enabled = true
}
// Claude CLI provider
envStr("GOCLAW_CLAUDE_CLI_PATH", &c.Providers.ClaudeCLI.CLIPath)
envStr("GOCLAW_CLAUDE_CLI_MODEL", &c.Providers.ClaudeCLI.Model)
envStr("GOCLAW_CLAUDE_CLI_WORK_DIR", &c.Providers.ClaudeCLI.BaseWorkDir)
// Default provider/model: env is fallback only (applied when config has no value).
// The onboard wizard sets these in .env for initial bootstrap; once the user
// saves a provider/model via the Dashboard, the config-file value wins.
envFallback := func(key string, dst *string) {
if *dst == "" {
if v := os.Getenv(key); v != "" {
*dst = v
}
}
}
envFallback("GOCLAW_PROVIDER", &c.Agents.Defaults.Provider)
envFallback("GOCLAW_MODEL", &c.Agents.Defaults.Model)
// Data directory, workspace & sessions
envStr("GOCLAW_DATA_DIR", &c.DataDir)
envStr("GOCLAW_WORKSPACE", &c.Agents.Defaults.Workspace)
// Gateway host/port
envStr("GOCLAW_HOST", &c.Gateway.Host)
if v := os.Getenv("GOCLAW_PORT"); v != "" {
if port, err := strconv.Atoi(v); err == nil && port > 0 {
c.Gateway.Port = port
}
}
// Database
envStr("GOCLAW_POSTGRES_DSN", &c.Database.PostgresDSN)
envStr("GOCLAW_REDIS_DSN", &c.Database.RedisDSN)
// Deprecation warning for GOCLAW_MODE (removed — PostgreSQL is always active)
if v := os.Getenv("GOCLAW_MODE"); v != "" {
slog.Warn("GOCLAW_MODE is deprecated; managed mode is now the only mode", "value", v)
}
// Telemetry
envStr("GOCLAW_TELEMETRY_ENDPOINT", &c.Telemetry.Endpoint)
envStr("GOCLAW_TELEMETRY_PROTOCOL", &c.Telemetry.Protocol)
envStr("GOCLAW_TELEMETRY_SERVICE_NAME", &c.Telemetry.ServiceName)
if v := os.Getenv("GOCLAW_TELEMETRY_ENABLED"); v != "" {
c.Telemetry.Enabled = v == "true" || v == "1"
}
if v := os.Getenv("GOCLAW_TELEMETRY_INSECURE"); v != "" {
c.Telemetry.Insecure = v == "true" || v == "1"
}
// Owner IDs from env (comma-separated)
if v := os.Getenv("GOCLAW_OWNER_IDS"); v != "" {
c.Gateway.OwnerIDs = strings.Split(v, ",")
}
// Tailscale (tsnet)
envStr("GOCLAW_TSNET_HOSTNAME", &c.Tailscale.Hostname)
envStr("GOCLAW_TSNET_AUTH_KEY", &c.Tailscale.AuthKey)
envStr("GOCLAW_TSNET_DIR", &c.Tailscale.StateDir)
// Sandbox (for Docker-compose sandbox overlay)
ensureSandbox := func() {
if c.Agents.Defaults.Sandbox == nil {
c.Agents.Defaults.Sandbox = &SandboxConfig{}
}
}
if v := os.Getenv("GOCLAW_SANDBOX_MODE"); v != "" {
ensureSandbox()
c.Agents.Defaults.Sandbox.Mode = v
}
if v := os.Getenv("GOCLAW_SANDBOX_IMAGE"); v != "" {
ensureSandbox()
c.Agents.Defaults.Sandbox.Image = v
}
if v := os.Getenv("GOCLAW_SANDBOX_WORKSPACE_ACCESS"); v != "" {
ensureSandbox()
c.Agents.Defaults.Sandbox.WorkspaceAccess = v
}
if v := os.Getenv("GOCLAW_SANDBOX_SCOPE"); v != "" {
ensureSandbox()
c.Agents.Defaults.Sandbox.Scope = v
}
if v := os.Getenv("GOCLAW_SANDBOX_MEMORY_MB"); v != "" {
ensureSandbox()
if mb, err := strconv.Atoi(v); err == nil && mb > 0 {
c.Agents.Defaults.Sandbox.MemoryMB = mb
}
}
if v := os.Getenv("GOCLAW_SANDBOX_CPUS"); v != "" {
ensureSandbox()
if cpus, err := strconv.ParseFloat(v, 64); err == nil && cpus > 0 {
c.Agents.Defaults.Sandbox.CPUs = cpus
}
}
if v := os.Getenv("GOCLAW_SANDBOX_TIMEOUT_SEC"); v != "" {
ensureSandbox()
if sec, err := strconv.Atoi(v); err == nil && sec > 0 {
c.Agents.Defaults.Sandbox.TimeoutSec = sec
}
}
if v := os.Getenv("GOCLAW_SANDBOX_NETWORK"); v != "" {
ensureSandbox()
c.Agents.Defaults.Sandbox.NetworkEnabled = v == "true" || v == "1"
}
// Browser (for Docker-compose browser sidecar overlay)
envStr("GOCLAW_BROWSER_REMOTE_URL", &c.Tools.Browser.RemoteURL)
if c.Tools.Browser.RemoteURL != "" {
c.Tools.Browser.Enabled = true
}
}
// applyContextPruningDefaults auto-enables context pruning when the Anthropic
// provider is configured, matching TS applyContextPruningDefaults() in
// src/config/defaults.ts.
//
// Go port does not have OAuth vs API-key distinction — we always treat it as
// API-key mode.
func (c *Config) applyContextPruningDefaults() {
// Only apply when Anthropic is configured.
if c.Providers.Anthropic.APIKey == "" {
return
}
defaults := &c.Agents.Defaults
// Auto-enable context pruning if mode not explicitly set.
if defaults.ContextPruning == nil {
defaults.ContextPruning = &ContextPruningConfig{
Mode: "cache-ttl",
}
} else if defaults.ContextPruning.Mode == "" {
defaults.ContextPruning.Mode = "cache-ttl"
}
}
// Save writes the config to a JSON file.
func Save(path string, cfg *Config) error {
cfg.mu.RLock()
defer cfg.mu.RUnlock()
data, err := json.MarshalIndent(cfg, "", " ")
if err != nil {
return err
}
dir := filepath.Dir(path)
if err := os.MkdirAll(dir, 0755); err != nil {
return err
}
return os.WriteFile(path, data, 0600)
}
// Hash returns a SHA-256 hash of the config for optimistic concurrency.
func (c *Config) Hash() string {
c.mu.RLock()
defer c.mu.RUnlock()
data, _ := json.Marshal(c)
h := sha256.Sum256(data)
return fmt.Sprintf("%x", h[:8])
}
// ResolvedDataDir returns the expanded data directory path.
func (c *Config) ResolvedDataDir() string {
c.mu.RLock()
defer c.mu.RUnlock()
return ExpandHome(c.DataDir)
}
// ResolvedDataDirFromEnv returns the data dir from GOCLAW_DATA_DIR env or default.
// Use this in packages that don't have access to a Config instance.
func ResolvedDataDirFromEnv() string {
if v := os.Getenv("GOCLAW_DATA_DIR"); v != "" {
return ExpandHome(v)
}
return ExpandHome("~/.goclaw/data")
}
// WorkspacePath returns the expanded workspace path.
func (c *Config) WorkspacePath() string {
c.mu.RLock()
defer c.mu.RUnlock()
return ExpandHome(c.Agents.Defaults.Workspace)
}
// ResolveAgent returns the effective config for a given agent ID,
// merging defaults with per-agent overrides.
func (c *Config) ResolveAgent(agentID string) AgentDefaults {
c.mu.RLock()
defer c.mu.RUnlock()
d := c.Agents.Defaults
if spec, ok := c.Agents.List[agentID]; ok {
if spec.Provider != "" {
d.Provider = spec.Provider
}
if spec.Model != "" {
d.Model = spec.Model
}
if spec.MaxTokens > 0 {
d.MaxTokens = spec.MaxTokens
}
if spec.Temperature > 0 {
d.Temperature = spec.Temperature
}
if spec.MaxToolIterations > 0 {
d.MaxToolIterations = spec.MaxToolIterations
}
if spec.ContextWindow > 0 {
d.ContextWindow = spec.ContextWindow
}
if spec.MaxToolCalls > 0 {
d.MaxToolCalls = spec.MaxToolCalls
}
if spec.Workspace != "" {
d.Workspace = spec.Workspace
}
if spec.Sandbox != nil {
d.Sandbox = spec.Sandbox
}
if spec.AgentType != "" {
d.AgentType = spec.AgentType
}
}
return d
}
// ResolveDefaultAgentID returns the ID of the agent marked as default,
// or "default" if none is explicitly marked.
func (c *Config) ResolveDefaultAgentID() string {
c.mu.RLock()
defer c.mu.RUnlock()
for id, spec := range c.Agents.List {
if spec.Default {
return id
}
}
return DefaultAgentID
}
// ResolveDisplayName returns the display name for an agent.
// Falls back to "GoClaw" if not configured.
func (c *Config) ResolveDisplayName(agentID string) string {
c.mu.RLock()
defer c.mu.RUnlock()
if spec, ok := c.Agents.List[agentID]; ok && spec.DisplayName != "" {
return spec.DisplayName
}
return "GoClaw"
}
// ApplyEnvOverrides re-applies environment variable overrides onto the config.
// Call this after modifying config to restore runtime secrets from env vars.
func (c *Config) ApplyEnvOverrides() {
c.applyEnvOverrides()
c.applyContextPruningDefaults()
}
// ExpandHome replaces leading ~ with the user home directory.
func ExpandHome(path string) string {
if path == "" || path[0] != '~' {
return path
}
home, _ := os.UserHomeDir()
if len(path) > 1 && path[1] == '/' {
return home + path[1:]
}
return home
}
// ContractHome replaces the user home directory prefix with ~.
// Reverse of ExpandHome — used to store portable paths in the database.
func ContractHome(path string) string {
if path == "" {
return path
}
home, _ := os.UserHomeDir()
if home != "" && strings.HasPrefix(path, home) {
return "~" + path[len(home):]
}
return path
}