1

下面是我的 toml 格式的配置文件。

[[hosts]]
name = "host1"
username = "user1"
password = "password1"

[[hosts]]
name = "host2"
username = "user2"
password = "password2"

...这是我加载它的代码:

import (
    "fmt"
    "github.com/spf13/viper"
    "strings"
)

type Config struct {
    Hosts []Host
}

type Host struct {
    Name      string `mapstructure:"name"`
    Username  string `mapstructure:"username"`
    Password  string `mapstructure:"password"`
}

func main() {
    viper.AddConfigPath(".")
    viper.AddConfigPath("./config")
    viper.SetConfigName("app")

    if err := viper.ReadInConfig(); err != nil {
        return nil, fmt.Errorf("error reading config file, %v", err)
    }

    config := new(Config)
    if err := viper.Unmarshal(config); err != nil {
        return nil, fmt.Errorf("error parsing config file, %v", err)
    }

    var username, password string

    for i, h := range config.Hosts {
        if len(h.Name) == 0 {
            return nil, fmt.Errorf("name not defined for host %d", i)
        }

        if username = os.Getenv(strings.ToUpper(h.Name) + "_" + "USERNAME"); len(username) > 0 {
            config.Hosts[i].Username = username
        } else if len(h.Username) == 0 {
            return nil, fmt.Errorf("username not defined for %s", e.Name)
        }

        if password = os.Getenv(strings.ToUpper(e.Name) + "_" + "PASSWORD"); len(password) > 0 {
            config.Hosts[i].Password = password
        } else if len(h.Password) == 0 {
            return nil, fmt.Errorf("password not defined for %s", e.Name)
        }

        fmt.Printf("Hostname: %s", h.name)
        fmt.Printf("Username: %s", h.Username)
        fmt.Printf("Password: %s", h.Password)
    }
}

例如,我首先检查环境变量HOST1_USERNAME1, HOST1_PASSWORD1,HOST2_USERNAME2和是否HOST2_PASSWORD2存在...如果存在,则将配置项设置为它们的值,否则我尝试从配置文件中获取值。

我知道 viper 提供AutomaticEnv了做类似事情的方法......但它是否适用于像我这样的集合(AutomaticEnv应在环境变量绑定后调用)?

鉴于我上面的代码,是否可以使用 viper 和 remove 提供的机制os.GetEnv

谢谢。

更新

下面是我更新的代码。在我定义了环境变量HOST1_USERNAME并将HOST1_PASSWORD配置文件中的相应设置设置为空字符串。

这是我的新配置文件:

[host1]
username = ""
password = ""

[host2]
username = "user2"
password = "password2"

这是我的代码:

package config

import (
    "fmt"
    "github.com/spf13/viper"
    "strings"
    "sync"
)

type Config struct {
    Hosts []Host
}

type Host struct {
    Name     string
    Username string
    Password string
}

var config *Config

func (c *Config) Load() (*Config, error) {
    if config == nil {
        viper.AddConfigPath(".")
        viper.AddConfigPath("./config")
        viper.SetConfigName("myapp")
        viper.AutomaticEnv()
        viper.SetEnvKeyReplacer(strings.NewReplacer(".", "_"))

        if err := viper.ReadInConfig(); err != nil {
            return nil, fmt.Errorf("error reading config file, %v", err)
        }

        allSettings := viper.AllSettings()
        hosts := make([]Host, 0, len(allSettings))

        for key, value := range allSettings {
            val := value.(map[string]interface{})

            if val["username"] == nil {
                return nil, fmt.Errorf("username not defined for host %s", key)
            }

            if val["password"] == nil {
                return nil, fmt.Errorf("password not defined for host %s", key)
            }

            hosts = append(hosts, Host{
                Name:      key,
                Unsername: val["username"].(string),
                Password: val["password"].(string),
            })
        }

        config = &Config{hosts}
    }

    return config, nil
}

我现在工作(感谢 Chrono Kitsune),希望对您有所帮助,j3d

4

1 回答 1

1

来自viper.Viper

源的优先级如下:

  1. 覆盖
  2. 旗帜
  3. 环境。变量
  4. 配置文件
  5. 键/值存储
  6. 默认值

在确定环境变量的名称时可能会遇到问题。您本质上需要一种绑定hosts[0].Username到环境变量的方法HOST1_USERNAME。但是,目前在 viper 中没有办法做到这一点。实际上,viper.Get("hosts[0].username")returnsnil意味着数组索引显然不能与viper.BindEnv. 您还需要对尽可能多的主机使用此函数,这意味着如果您列出了 20 个主机,则需要调用viper.BindEnv40 或 60 次,具体取决于主机名称是否可以被环境覆盖多变的。要解决此限制,您需要将主机作为独立表而不是表数组动态使用:

[host1]
username = "user1"
password = "password1"

[host2]
username = "user2"
password = "password2"

然后您可以使用viper.SetEnvKeyReplacerwith astrings.Replacer来处理环境变量问题:

// host1.username => HOST1_USERNAME
// host2.password => HOST2_PASSWORD
// etc.
viper.SetEnvKeyReplacer(strings.NewReplacer(".", "_"))

请注意,在撰写本文时,解决顺序方面存在一些错误。此问题影响viper.Unmarshal并且viper.Get:环境变量应覆盖配置文件值,但当前仍使用配置文件值。奇怪的是,viper.AllSettings工作正常。如果没有,您将无法执行以下操作来使用上述格式的主机:

// Manually collect hosts for storing in config.
func collectHosts() []Host {
    hosts := make([]Host, 0, 10)
    for key, value := range viper.AllSettings() {
        // viper.GetStringMapString(key)
        // won't work properly with environment vars, etc. until
        //   https://github.com/spf13/viper/issues/309
        // is resolved.
        v := value.(map[string]interface{})
        hosts = append(hosts, Host{
            Name:     key,
            Username: v["username"].(string),
            Password: v["password"].(string),
        })
    }
    return hosts
}

总结一下:

  1. 值应该取自第一个提供的:覆盖、标志、环境变量、配置文件、键/值存储、默认值。不幸的是,这并不总是遵循的顺序(因为错误)。
  2. 您需要更改配置格式并使用字符串替换器来利用viper.AutomaticEnv.
于 2017-09-02T03:29:54.770 回答