package config import ( "flag" "os" "path/filepath" "sync/atomic" "github.com/sirupsen/logrus" "github.com/spf13/viper" "github.com/ivuorinen/gibidify/gibidiutils" ) // LoadConfig reads configuration from a YAML file. // It looks for config in the following order: // 1. $XDG_CONFIG_HOME/gibidify/config.yaml // 2. $HOME/.config/gibidify/config.yaml // 3. The current directory as fallback. // // Note: LoadConfig relies on isRunningTest() which requires the testing package // to have registered its flags (e.g., via flag.Parse() or during test initialization). // If called too early (e.g., from init() or before TestMain), test detection may not work reliably. // For explicit control, use SetRunningInTest() before calling LoadConfig. func LoadConfig() { viper.SetConfigName("config") viper.SetConfigType("yaml") if xdgConfig := os.Getenv("XDG_CONFIG_HOME"); xdgConfig != "" { // Validate XDG_CONFIG_HOME for path traversal attempts if err := gibidiutils.ValidateConfigPath(xdgConfig); err != nil { logrus.Warnf("Invalid XDG_CONFIG_HOME path, using default config: %v", err) } else { configPath := filepath.Join(xdgConfig, "gibidify") viper.AddConfigPath(configPath) } } else if home, err := os.UserHomeDir(); err == nil { viper.AddConfigPath(filepath.Join(home, ".config", "gibidify")) } // Only add current directory if no config file named gibidify.yaml exists // to avoid conflicts with the project's output file if _, err := os.Stat("gibidify.yaml"); os.IsNotExist(err) { viper.AddConfigPath(".") } if err := viper.ReadInConfig(); err != nil { // Suppress this info-level log when running tests. // Prefer an explicit test flag (SetRunningInTest) but fall back to runtime detection. if runningInTest.Load() || isRunningTest() { // Keep a debug-level record so tests that enable debug can still see it. logrus.Debugf("Config file not found (tests): %v", err) } else { logrus.Infof("Config file not found, using default values: %v", err) } setDefaultConfig() } else { logrus.Infof("Using config file: %s", viper.ConfigFileUsed()) // Validate configuration after loading if err := ValidateConfig(); err != nil { logrus.Warnf("Configuration validation failed: %v", err) logrus.Info("Falling back to default configuration") // Reset viper and set defaults when validation fails viper.Reset() setDefaultConfig() } } } // setDefaultConfig sets default configuration values. func setDefaultConfig() { viper.SetDefault("fileSizeLimit", DefaultFileSizeLimit) // Default ignored directories. viper.SetDefault("ignoreDirectories", []string{ "vendor", "node_modules", ".git", "dist", "build", "target", "bower_components", "cache", "tmp", }) // FileTypeRegistry defaults viper.SetDefault("fileTypes.enabled", true) viper.SetDefault("fileTypes.customImageExtensions", []string{}) viper.SetDefault("fileTypes.customBinaryExtensions", []string{}) viper.SetDefault("fileTypes.customLanguages", map[string]string{}) viper.SetDefault("fileTypes.disabledImageExtensions", []string{}) viper.SetDefault("fileTypes.disabledBinaryExtensions", []string{}) viper.SetDefault("fileTypes.disabledLanguageExtensions", []string{}) // Back-pressure and memory management defaults viper.SetDefault("backpressure.enabled", true) viper.SetDefault("backpressure.maxPendingFiles", 1000) // Max files in file channel buffer viper.SetDefault("backpressure.maxPendingWrites", 100) // Max writes in write channel buffer viper.SetDefault("backpressure.maxMemoryUsage", 104857600) // 100MB max memory usage viper.SetDefault("backpressure.memoryCheckInterval", 1000) // Check memory every 1000 files // Resource limit defaults viper.SetDefault("resourceLimits.enabled", true) viper.SetDefault("resourceLimits.maxFiles", DefaultMaxFiles) viper.SetDefault("resourceLimits.maxTotalSize", DefaultMaxTotalSize) viper.SetDefault("resourceLimits.fileProcessingTimeoutSec", DefaultFileProcessingTimeoutSec) viper.SetDefault("resourceLimits.overallTimeoutSec", DefaultOverallTimeoutSec) viper.SetDefault("resourceLimits.maxConcurrentReads", DefaultMaxConcurrentReads) viper.SetDefault("resourceLimits.rateLimitFilesPerSec", DefaultRateLimitFilesPerSec) viper.SetDefault("resourceLimits.hardMemoryLimitMB", DefaultHardMemoryLimitMB) viper.SetDefault("resourceLimits.enableGracefulDegradation", true) viper.SetDefault("resourceLimits.enableResourceMonitoring", true) } var runningInTest atomic.Bool // SetRunningInTest allows tests to explicitly indicate they are running under `go test`. // Call this from TestMain in tests to suppress noisy info logs while still allowing // debug-level output for tests that enable it. func SetRunningInTest(b bool) { runningInTest.Store(b) } // isRunningTest attempts to detect if the binary is running under `go test`. // Prefer checking for standard test flags registered by the testing package. // This is reliable when `go test` initializes the flag set. // // IMPORTANT: This function relies on flag.Lookup which returns nil if the testing // package hasn't registered test flags yet. Callers must invoke this after flag // parsing (or test flag registration) has occurred. If invoked too early (e.g., // from init() or early in TestMain before flags are parsed), detection will fail. // For explicit control, use SetRunningInTest() instead. func isRunningTest() bool { // Look for the well-known test flags created by the testing package. // If any are present in the flag registry, we're running under `go test`. if flag.Lookup("test.v") != nil || flag.Lookup("test.run") != nil || flag.Lookup("test.bench") != nil { return true } return false }