268 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			268 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package config
 | ||
| 
 | ||
| import (
 | ||
| 	"fmt"
 | ||
| 	"os"
 | ||
| 	"path/filepath"
 | ||
| 	"strings"
 | ||
| 	"time"
 | ||
| 
 | ||
| 	"github.com/spf13/viper"
 | ||
| )
 | ||
| 
 | ||
| // LoadConfig 加载应用程序配置
 | ||
| func LoadConfig() (*Config, error) {
 | ||
| 	// 1️⃣ 获取环境变量决定配置文件
 | ||
| 	env := getEnvironment()
 | ||
| 	fmt.Printf("🔧 当前运行环境: %s\n", env)
 | ||
| 
 | ||
| 	// 2️⃣ 加载基础配置文件
 | ||
| 	baseConfig := viper.New()
 | ||
| 	baseConfig.SetConfigName("config")
 | ||
| 	baseConfig.SetConfigType("yaml")
 | ||
| 	baseConfig.AddConfigPath(".")
 | ||
| 	baseConfig.AddConfigPath("./configs")
 | ||
| 	baseConfig.AddConfigPath("$HOME/.tyapi")
 | ||
| 
 | ||
| 	// 读取基础配置文件
 | ||
| 	if err := baseConfig.ReadInConfig(); err != nil {
 | ||
| 		if _, ok := err.(viper.ConfigFileNotFoundError); !ok {
 | ||
| 			return nil, fmt.Errorf("读取基础配置文件失败: %w", err)
 | ||
| 		}
 | ||
| 		return nil, fmt.Errorf("未找到 config.yaml 文件,请确保配置文件存在")
 | ||
| 	}
 | ||
| 	fmt.Printf("✅ 已加载配置文件: %s\n", baseConfig.ConfigFileUsed())
 | ||
| 
 | ||
| 	// 3️⃣ 加载环境特定配置文件
 | ||
| 	envConfigFile := findEnvConfigFile(env)
 | ||
| 	if envConfigFile != "" {
 | ||
| 		// 创建一个新的viper实例来读取环境配置
 | ||
| 		envConfig := viper.New()
 | ||
| 		envConfig.SetConfigFile(envConfigFile)
 | ||
| 
 | ||
| 		if err := envConfig.ReadInConfig(); err != nil {
 | ||
| 			fmt.Printf("⚠️  环境配置文件加载警告: %v\n", err)
 | ||
| 		} else {
 | ||
| 			fmt.Printf("✅ 已加载环境配置: %s\n", envConfigFile)
 | ||
| 
 | ||
| 			// 将环境配置合并到基础配置中
 | ||
| 			if err := mergeConfigs(baseConfig, envConfig.AllSettings()); err != nil {
 | ||
| 				return nil, fmt.Errorf("合并配置失败: %w", err)
 | ||
| 			}
 | ||
| 		}
 | ||
| 	} else {
 | ||
| 		fmt.Printf("ℹ️  未找到环境配置文件 configs/env.%s.yaml,将使用基础配置\n", env)
 | ||
| 	}
 | ||
| 
 | ||
| 	// 4️⃣ 手动处理环境变量覆盖,避免空值覆盖配置文件
 | ||
| 	// overrideWithEnvVars(baseConfig)
 | ||
| 
 | ||
| 	// 5️⃣ 解析配置到结构体
 | ||
| 	var config Config
 | ||
| 	if err := baseConfig.Unmarshal(&config); err != nil {
 | ||
| 		return nil, fmt.Errorf("解析配置失败: %w", err)
 | ||
| 	}
 | ||
| 
 | ||
| 	// 6️⃣ 验证配置
 | ||
| 	if err := validateConfig(&config); err != nil {
 | ||
| 		return nil, fmt.Errorf("配置验证失败: %w", err)
 | ||
| 	}
 | ||
| 
 | ||
| 	// 7️⃣ 输出配置摘要
 | ||
| 	printConfigSummary(&config, env)
 | ||
| 
 | ||
| 	return &config, nil
 | ||
| }
 | ||
| 
 | ||
| // mergeConfigs 递归合并配置
 | ||
| func mergeConfigs(baseConfig *viper.Viper, overrideSettings map[string]interface{}) error {
 | ||
| 	for key, val := range overrideSettings {
 | ||
| 		// 如果值是一个嵌套的map,则递归合并
 | ||
| 		if subMap, ok := val.(map[string]interface{}); ok {
 | ||
| 			// 创建子键路径
 | ||
| 			subKey := key
 | ||
| 
 | ||
| 			// 递归合并子配置
 | ||
| 			for subK, subV := range subMap {
 | ||
| 				fullKey := fmt.Sprintf("%s.%s", subKey, subK)
 | ||
| 				baseConfig.Set(fullKey, subV)
 | ||
| 			}
 | ||
| 		} else {
 | ||
| 			// 直接设置值
 | ||
| 			baseConfig.Set(key, val)
 | ||
| 		}
 | ||
| 	}
 | ||
| 	return nil
 | ||
| }
 | ||
| 
 | ||
| // findEnvConfigFile 查找环境特定的配置文件
 | ||
| func findEnvConfigFile(env string) string {
 | ||
| 	// 只查找 configs 目录下的环境配置文件
 | ||
| 	possiblePaths := []string{
 | ||
| 		fmt.Sprintf("configs/env.%s.yaml", env),
 | ||
| 		fmt.Sprintf("configs/env.%s.yml", env),
 | ||
| 	}
 | ||
| 
 | ||
| 	for _, path := range possiblePaths {
 | ||
| 		if _, err := os.Stat(path); err == nil {
 | ||
| 			absPath, _ := filepath.Abs(path)
 | ||
| 			return absPath
 | ||
| 		}
 | ||
| 	}
 | ||
| 
 | ||
| 	return ""
 | ||
| }
 | ||
| 
 | ||
| // getEnvironment 获取当前环境
 | ||
| func getEnvironment() string {
 | ||
| 	var env string
 | ||
| 	var source string
 | ||
| 
 | ||
| 	// 优先级:CONFIG_ENV > ENV > APP_ENV > 默认值
 | ||
| 	if env = os.Getenv("CONFIG_ENV"); env != "" {
 | ||
| 		source = "CONFIG_ENV"
 | ||
| 	} else if env = os.Getenv("ENV"); env != "" {
 | ||
| 		source = "ENV"
 | ||
| 	} else if env = os.Getenv("APP_ENV"); env != "" {
 | ||
| 		source = "APP_ENV"
 | ||
| 	} else {
 | ||
| 		env = "development"
 | ||
| 		source = "默认值"
 | ||
| 	}
 | ||
| 
 | ||
| 	fmt.Printf("🌍 环境检测: %s (来源: %s)\n", env, source)
 | ||
| 
 | ||
| 	// 验证环境值
 | ||
| 	validEnvs := []string{"development", "production", "testing"}
 | ||
| 	isValid := false
 | ||
| 	for _, validEnv := range validEnvs {
 | ||
| 		if env == validEnv {
 | ||
| 			isValid = true
 | ||
| 			break
 | ||
| 		}
 | ||
| 	}
 | ||
| 
 | ||
| 	if !isValid {
 | ||
| 		fmt.Printf("⚠️  警告: 未识别的环境 '%s',将使用默认环境 'development'\n", env)
 | ||
| 		return "development"
 | ||
| 	}
 | ||
| 
 | ||
| 	return env
 | ||
| }
 | ||
| 
 | ||
| // printConfigSummary 打印配置摘要
 | ||
| func printConfigSummary(config *Config, env string) {
 | ||
| 	fmt.Printf("\n🔧 配置摘要:\n")
 | ||
| 	fmt.Printf("   🌍 环境: %s\n", env)
 | ||
| 	fmt.Printf("   📄 基础配置: config.yaml\n")
 | ||
| 	fmt.Printf("   📁 环境配置: configs/env.%s.yaml\n", env)
 | ||
| 	fmt.Printf("   📱 应用名称: %s\n", config.App.Name)
 | ||
| 	fmt.Printf("   🔖 版本: %s\n", config.App.Version)
 | ||
| 	fmt.Printf("   🌐 服务端口: %s\n", config.Server.Port)
 | ||
| 	fmt.Printf("   🗄️ 数据库: %s@%s:%s/%s\n",
 | ||
| 		config.Database.User,
 | ||
| 		config.Database.Host,
 | ||
| 		config.Database.Port,
 | ||
| 		config.Database.Name)
 | ||
| 	fmt.Printf("   📊 追踪状态: %v (端点: %s)\n",
 | ||
| 		config.Monitoring.TracingEnabled,
 | ||
| 		config.Monitoring.TracingEndpoint)
 | ||
| 	fmt.Printf("   📈 采样率: %.1f%%\n", config.Monitoring.SampleRate*100)
 | ||
| 	fmt.Printf("\n")
 | ||
| }
 | ||
| 
 | ||
| // validateConfig 验证配置
 | ||
| func validateConfig(config *Config) error {
 | ||
| 	// 验证必要的配置项
 | ||
| 	if config.Database.Host == "" {
 | ||
| 		return fmt.Errorf("数据库主机地址不能为空")
 | ||
| 	}
 | ||
| 
 | ||
| 	if config.Database.User == "" {
 | ||
| 		return fmt.Errorf("数据库用户名不能为空")
 | ||
| 	}
 | ||
| 
 | ||
| 	if config.Database.Name == "" {
 | ||
| 		return fmt.Errorf("数据库名称不能为空")
 | ||
| 	}
 | ||
| 
 | ||
| 	if config.JWT.Secret == "" || config.JWT.Secret == "your-super-secret-jwt-key-change-this-in-production" {
 | ||
| 		if config.App.IsProduction() {
 | ||
| 			return fmt.Errorf("生产环境必须设置安全的JWT密钥")
 | ||
| 		}
 | ||
| 	}
 | ||
| 
 | ||
| 	// 验证超时配置
 | ||
| 	if config.Server.ReadTimeout <= 0 {
 | ||
| 		return fmt.Errorf("服务器读取超时时间必须大于0")
 | ||
| 	}
 | ||
| 
 | ||
| 	if config.Server.WriteTimeout <= 0 {
 | ||
| 		return fmt.Errorf("服务器写入超时时间必须大于0")
 | ||
| 	}
 | ||
| 
 | ||
| 	// 验证数据库连接池配置
 | ||
| 	if config.Database.MaxOpenConns <= 0 {
 | ||
| 		return fmt.Errorf("数据库最大连接数必须大于0")
 | ||
| 	}
 | ||
| 
 | ||
| 	if config.Database.MaxIdleConns <= 0 {
 | ||
| 		return fmt.Errorf("数据库最大空闲连接数必须大于0")
 | ||
| 	}
 | ||
| 
 | ||
| 	if config.Database.MaxIdleConns > config.Database.MaxOpenConns {
 | ||
| 		return fmt.Errorf("数据库最大空闲连接数不能大于最大连接数")
 | ||
| 	}
 | ||
| 
 | ||
| 	return nil
 | ||
| }
 | ||
| 
 | ||
| // GetEnv 获取环境变量,如果不存在则返回默认值
 | ||
| func GetEnv(key, defaultValue string) string {
 | ||
| 	if value := os.Getenv(key); value != "" {
 | ||
| 		return value
 | ||
| 	}
 | ||
| 	return defaultValue
 | ||
| }
 | ||
| 
 | ||
| // ParseDuration 解析时间字符串
 | ||
| func ParseDuration(s string) time.Duration {
 | ||
| 	d, err := time.ParseDuration(s)
 | ||
| 	if err != nil {
 | ||
| 		return 0
 | ||
| 	}
 | ||
| 	return d
 | ||
| }
 | ||
| 
 | ||
| // overrideWithEnvVars 手动处理环境变量覆盖,避免空值覆盖配置文件
 | ||
| func overrideWithEnvVars(config *viper.Viper) {
 | ||
| 	// 定义需要环境变量覆盖的敏感配置项
 | ||
| 	sensitiveConfigs := map[string]string{
 | ||
| 		"database.password":       "DATABASE_PASSWORD",
 | ||
| 		"jwt.secret":              "JWT_SECRET",
 | ||
| 		"redis.password":          "REDIS_PASSWORD",
 | ||
| 		"wechat_work.webhook_url": "WECHAT_WORK_WEBHOOK_URL",
 | ||
| 		"wechat_work.secret":      "WECHAT_WORK_SECRET",
 | ||
| 	}
 | ||
| 
 | ||
| 	// 只覆盖明确设置的环境变量
 | ||
| 	for configKey, envKey := range sensitiveConfigs {
 | ||
| 		if envValue := os.Getenv(envKey); envValue != "" {
 | ||
| 			config.Set(configKey, envValue)
 | ||
| 			fmt.Printf("🔐 已从环境变量覆盖配置: %s\n", configKey)
 | ||
| 		}
 | ||
| 	}
 | ||
| }
 | ||
| 
 | ||
| // SplitAndTrim 分割字符串并去除空格
 | ||
| func SplitAndTrim(s, sep string) []string {
 | ||
| 	parts := strings.Split(s, sep)
 | ||
| 	result := make([]string, 0, len(parts))
 | ||
| 	for _, part := range parts {
 | ||
| 		if trimmed := strings.TrimSpace(part); trimmed != "" {
 | ||
| 			result = append(result, trimmed)
 | ||
| 		}
 | ||
| 	}
 | ||
| 	return result
 | ||
| }
 |