为了账号安全,请及时绑定邮箱和手机立即绑定

我应该如何测试处理设置大量环境配置/操作系统参数的函数?

我应该如何测试处理设置大量环境配置/操作系统参数的函数?

Go
冉冉说 2022-08-24 19:03:59
我已经编写了一个Go应用程序,所有软件包都具有完整的测试覆盖率。我正在编写我的包 - 它将处理函数中应用程序的所有初始设置 - 此函数当前读取14个环境变量,然后在应用程序中设置相关变量。代码的简单概述如下:mainmain()func main() {    myStruct1 := privatePackage.myStructType{}    myStruct2 := publicPackage.otherStructType{}    if config1 := os.Getenv("CONFIG_FOO"); config1 != "" {        myStruct1.attribute1 = config1    }    // ....    if config14 := os.Getenv("CONFIG_BAR"); config14 != "" {        myStruct2.attribute5 = config14    }}当我测试单元 env 变量/OS 参数时,我通常只直接在测试函数中设置 env 变量 - 所以像这样:func TestMyArgument(t *testing.T) {    os.Setenv("CONFIG_BAZ", "apple")    //Invoke function that depends on CONFIG_BAZ    //Assert that expected outcome occurred}我几乎总是使用表驱动的测试,所以上面的代码片段是一个简化的例子。问题在于,我的函数接收了 14 个(并且还在增长)env 变量,虽然一些 env 变量本质上是枚举(因此有少量有效选项 - 例如,有少量数据库驱动程序可供选择),但其他 env 变量具有几乎无限的潜在值。那么,如何有效地覆盖潜在配置的所有(或足够多的)排列呢?main()编辑:部署此应用程序后,它将进入K8s集群。其中一些变量是从安全存储中提取的机密。使用 JSON 文件是不可行的,因为某些值需要轻松加密/更改。此外,使用JSON文件需要我存储此文件并在数百/数千个正在运行的 Pod 之间共享它 - 然后此存储将充当故障点。为了澄清,这个问题不是关于env vars VS配置文件;这个问题是关于当有大量可配置变量时进行测试的最佳方法 - 每个变量都有大量的潜在值 - 导致数千个可能的配置排列。在这种情况下,如何保证足够的测试覆盖率?
查看完整描述

2 回答

?
天涯尽头无女友

TA贡献1831条经验 获得超9个赞

@Steven Penny是对的:使用json


使用refrebra可以使代码更简单:

package main


import (

    "encoding/json"

    "fmt"

    "os"

    "reflect"

    "strconv"

)


type MyStructType struct {

    Attribute1 string `json:"CONFIG_FOO"`

    Attribute2 string `json:"CONFIG_BAZ"`

    Attribute3 int `json:"CONFIG_BAR"`

}


func NewMyStructTypeFormEnv() *MyStructType {

    myStructType := MyStructType{}

    ReflectMyStructType(&myStructType)

    fmt.Println("myStructType is now", myStructType)

    return &myStructType

}


func NewMyStructTypeFormJson() *MyStructType {

    myStructType := MyStructType{}

    f, e := os.Open("file.json")

    if e != nil {

        panic(e)

    }

    defer f.Close()

    json.NewDecoder(f).Decode(&myStructType)

    fmt.Println("myStructType is now", myStructType)

    return &myStructType

}


func ReflectMyStructType(ptr interface{}){

    v := reflect.ValueOf(ptr).Elem()

    fmt.Printf("%v\n", v.Type())

    for i := 0; i < v.NumField(); i++ {

        env_str := v.Type().Field(i).Tag.Get("json")

        if(env_str == ""){continue}

        if config := os.Getenv(env_str); config != "" {

            if v.Field(i).Kind() == reflect.String{

                v.Field(i).SetString(config)

            }else if v.Field(i).Kind() == reflect.Int{

                iConfig,_ := strconv.Atoi(config)

                v.Field(i).SetInt(int64(iConfig))

            }


        }

    }

}


func main() {

    NewMyStructTypeFormJson()

    

    os.Setenv("CONFIG_FOO", "apple")

    os.Setenv("CONFIG_BAZ", "apple")

    os.Setenv("CONFIG_BAR", "1")

    NewMyStructTypeFormEnv()

}


查看完整回答
反对 回复 2022-08-24
?
弑天下

TA贡献1818条经验 获得超8个赞

除了一两个之外,我不认为使用环境变量是正确的方法,除非需要它(调用)。相反,最好从配置文件中读取。下面是一个 JSON 示例:os/exec


{

   "CONFIG_BAR": "east",

   "CONFIG_BAZ": "south",

   "CONFIG_FOO": "north"

}

package main


import (

   "encoding/json"

   "fmt"

   "os"

)


func main() {

   f, e := os.Open("file.json")

   if e != nil {

      panic(e)

   }

   defer f.Close()

   var s struct { CONFIG_BAR, CONFIG_BAZ, CONFIG_FOO string }

   json.NewDecoder(f).Decode(&s)

   // {CONFIG_BAR:east CONFIG_BAZ:south CONFIG_FOO:north}

   fmt.Printf("%+v\n", s)

}

TOML也是一个不错的选择。


查看完整回答
反对 回复 2022-08-24
  • 2 回答
  • 0 关注
  • 133 浏览
慕课专栏
更多

添加回答

举报

0/150
提交
取消
微信客服

购课补贴
联系客服咨询优惠详情

帮助反馈 APP下载

慕课网APP
您的移动学习伙伴

公众号

扫描二维码
关注慕课网微信公众号