如何在Golang中测试参数的传递?

时间:2015-11-15 18:25:06

标签: testing go command-line-arguments

package main

import (
    "flag"
    "fmt"
)

func main() {
    passArguments()
}

func passArguments() string {
    username := flag.String("user", "root", "Username for this server")
    flag.Parse()
    fmt.Printf("Your username is %q.", *username)

    usernameToString := *username
    return usernameToString
}

将参数传递给已编译的代码:

./args -user=bla

结果:

Your username is "bla"

显示已传递的用户名。

目标:为了防止代码需要每次手动构建和运行以测试代码,目的是编写一个能够测试参数传递的测试。 / p>

尝试

运行以下测试:

package main

import (
    "os"
    "testing"
)

func TestArgs(t *testing.T) {
    expected := "bla"
    os.Args = []string{"-user=bla"}

    actual := passArguments()

    if actual != expected {
        t.Errorf("Test failed, expected: '%s', got:  '%s'", expected, actual)
    }
}

结果:

Your username is "root".Your username is "root".--- FAIL: TestArgs (0.00s)
    args_test.go:15: Test failed, expected: 'bla', got:  'root'
FAIL
coverage: 87.5% of statements
FAIL    tool    0.008s

问题

看起来os.Args = []string{"-user=bla无法将此参数传递给函数,因为结果为root而不是bla

1 个答案:

答案 0 :(得分:18)

根据我的评论,os.Args中的第一个值是(路径)可执行文件本身,因此os.Args = []string{"cmd", "-user=bla"}应该可以解决您的问题。您可以从标准包中查看flag test,他们正在做类似的事情。

此外,由于os.Args是"全局变量",因此在测试之前保持状态并在之后恢复它可能是个好主意。与链接测试类似:

oldArgs := os.Args
defer func() { os.Args = oldArgs }()

这可能对其他测试很有用,例如,在调用go test时检查传递的真实参数。