package arg import ( "bytes" "net" "net/mail" "net/url" "os" "strings" "testing" "time" "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" ) func parse(dest interface{}, cmdline string, env ...string) (*Parser, error) { p, err := NewParser(dest) if err != nil { return nil, err } // split the command line tokens := []string{"program"} // first token is the program name if len(cmdline) > 0 { tokens = append(tokens, strings.Split(cmdline, " ")...) } // execute the parser return p, p.Parse(tokens, env) } func TestString(t *testing.T) { var args struct { Foo string Ptr *string } _, err := parse(&args, "--foo bar --ptr baz") require.NoError(t, err) assert.Equal(t, "bar", args.Foo) assert.Equal(t, "baz", *args.Ptr) } func TestBool(t *testing.T) { var args struct { A bool B bool C *bool D *bool } _, err := parse(&args, "--a --c") require.NoError(t, err) assert.True(t, args.A) assert.False(t, args.B) assert.True(t, *args.C) assert.Nil(t, args.D) } func TestInt(t *testing.T) { var args struct { Foo int Ptr *int } _, err := parse(&args, "--foo 7 --ptr 8") require.NoError(t, err) assert.EqualValues(t, 7, args.Foo) assert.EqualValues(t, 8, *args.Ptr) } func TestHexOctBin(t *testing.T) { var args struct { Hex int Oct int Bin int Underscored int } _, err := parse(&args, "--hex 0xA --oct 0o10 --bin 0b101 --underscored 123_456") require.NoError(t, err) assert.EqualValues(t, 10, args.Hex) assert.EqualValues(t, 8, args.Oct) assert.EqualValues(t, 5, args.Bin) assert.EqualValues(t, 123456, args.Underscored) } func TestNegativeInt(t *testing.T) { var args struct { Foo int } _, err := parse(&args, "-foo=-100") require.NoError(t, err) assert.EqualValues(t, args.Foo, -100) } func TestNumericOptionName(t *testing.T) { var args struct { N int `arg:"--100"` } _, err := parse(&args, "-100 6") require.NoError(t, err) assert.EqualValues(t, args.N, 6) } func TestUint(t *testing.T) { var args struct { Foo uint Ptr *uint } _, err := parse(&args, "--foo 7 --ptr 8") require.NoError(t, err) assert.EqualValues(t, 7, args.Foo) assert.EqualValues(t, 8, *args.Ptr) } func TestFloat(t *testing.T) { var args struct { Foo float32 Ptr *float32 } _, err := parse(&args, "--foo 3.4 --ptr 3.5") require.NoError(t, err) assert.EqualValues(t, 3.4, args.Foo) assert.EqualValues(t, 3.5, *args.Ptr) } func TestDuration(t *testing.T) { var args struct { Foo time.Duration Ptr *time.Duration } _, err := parse(&args, "--foo 3ms --ptr 4ms") require.NoError(t, err) assert.Equal(t, 3*time.Millisecond, args.Foo) assert.Equal(t, 4*time.Millisecond, *args.Ptr) } func TestInvalidDuration(t *testing.T) { var args struct { Foo time.Duration } _, err := parse(&args, "--foo xxx") require.Error(t, err) } func TestIntPtr(t *testing.T) { var args struct { Foo *int } _, err := parse(&args, "--foo 123") require.NoError(t, err) require.NotNil(t, args.Foo) assert.Equal(t, 123, *args.Foo) } func TestIntPtrNotPresent(t *testing.T) { var args struct { Foo *int } _, err := parse(&args, "") require.NoError(t, err) assert.Nil(t, args.Foo) } func TestMixed(t *testing.T) { var args struct { Foo string `arg:"-f"` Bar int Baz uint `arg:"positional"` Ham bool Spam float32 } args.Bar = 3 _, err := parse(&args, "123 -spam=1.2 -ham -f xyz") require.NoError(t, err) assert.Equal(t, "xyz", args.Foo) assert.Equal(t, 3, args.Bar) assert.Equal(t, uint(123), args.Baz) assert.Equal(t, true, args.Ham) assert.EqualValues(t, 1.2, args.Spam) } func TestRequired(t *testing.T) { var args struct { Foo string `arg:"required"` } _, err := parse(&args, "--foo=abc") require.NoError(t, err) } func TestMissingRequired(t *testing.T) { var args struct { Foo string `arg:"required"` } _, err := parse(&args, "") require.Error(t, err, "--foo is required") } func TestMissingRequiredWithEnv(t *testing.T) { var args struct { Foo string `arg:"required,env:FOO"` } _, err := parse(&args, "") require.Error(t, err, "--foo is required (or environment variable FOO)") } func TestShortFlag(t *testing.T) { var args struct { Foo string `arg:"-f"` } _, err := parse(&args, "-f a") require.NoError(t, err) assert.Equal(t, "a", args.Foo) _, err = parse(&args, "-foo b") require.NoError(t, err) assert.Equal(t, "b", args.Foo) _, err = parse(&args, "--foo c") require.NoError(t, err) assert.Equal(t, "c", args.Foo) } func TestInvalidShortFlag(t *testing.T) { var args struct { Foo string `arg:"-foo"` } _, err := parse(&args, "") assert.Error(t, err) } func TestLongFlag(t *testing.T) { var args struct { Foo string `arg:"--abc"` } _, err := parse(&args, "-abc xyz") require.NoError(t, err) assert.Equal(t, "xyz", args.Foo) _, err = parse(&args, "--abc xyz") require.NoError(t, err) assert.Equal(t, "xyz", args.Foo) } func TestSlice(t *testing.T) { var args struct { Strings []string } _, err := parse(&args, "--strings a b c") require.NoError(t, err) assert.Equal(t, []string{"a", "b", "c"}, args.Strings) } func TestSliceOfBools(t *testing.T) { var args struct { B []bool } _, err := parse(&args, "--b true false true") require.NoError(t, err) assert.Equal(t, []bool{true, false, true}, args.B) } func TestMap(t *testing.T) { var args struct { Values map[string]int } _, err := parse(&args, "--values a=1 b=2 c=3") require.NoError(t, err) assert.Len(t, args.Values, 3) assert.Equal(t, 1, args.Values["a"]) assert.Equal(t, 2, args.Values["b"]) assert.Equal(t, 3, args.Values["c"]) } func TestMapPositional(t *testing.T) { var args struct { Values map[string]int `arg:"positional"` } _, err := parse(&args, "a=1 b=2 c=3") require.NoError(t, err) assert.Len(t, args.Values, 3) assert.Equal(t, 1, args.Values["a"]) assert.Equal(t, 2, args.Values["b"]) assert.Equal(t, 3, args.Values["c"]) } func TestMapWithSeparate(t *testing.T) { var args struct { Values map[string]int `arg:"separate"` } _, err := parse(&args, "--values a=1 --values b=2 --values c=3") require.NoError(t, err) assert.Len(t, args.Values, 3) assert.Equal(t, 1, args.Values["a"]) assert.Equal(t, 2, args.Values["b"]) assert.Equal(t, 3, args.Values["c"]) } func TestPlaceholder(t *testing.T) { var args struct { Input string `arg:"positional" placeholder:"SRC"` Output []string `arg:"positional" placeholder:"DST"` Optimize int `arg:"-O" placeholder:"LEVEL"` MaxJobs int `arg:"-j" placeholder:"N"` } _, err := parse(&args, "-O 5 --maxjobs 2 src dest1 dest2") assert.NoError(t, err) } func TestNoLongName(t *testing.T) { var args struct { ShortOnly string `arg:"-s,--"` EnvOnly string `arg:"--,env"` } _, err := parse(&args, "-s TestVal2", "ENVONLY=TestVal") assert.NoError(t, err) assert.Equal(t, "TestVal", args.EnvOnly) assert.Equal(t, "TestVal2", args.ShortOnly) } func TestCaseSensitive(t *testing.T) { var args struct { Lower bool `arg:"-v"` Upper bool `arg:"-V"` } _, err := parse(&args, "-v") require.NoError(t, err) assert.True(t, args.Lower) assert.False(t, args.Upper) } func TestCaseSensitive2(t *testing.T) { var args struct { Lower bool `arg:"-v"` Upper bool `arg:"-V"` } _, err := parse(&args, "-V") require.NoError(t, err) assert.False(t, args.Lower) assert.True(t, args.Upper) } func TestPositional(t *testing.T) { var args struct { Input string `arg:"positional"` Output string `arg:"positional"` } _, err := parse(&args, "foo") require.NoError(t, err) assert.Equal(t, "foo", args.Input) assert.Equal(t, "", args.Output) } func TestPositionalPointer(t *testing.T) { var args struct { Input string `arg:"positional"` Output []*string `arg:"positional"` } _, err := parse(&args, "foo bar baz") require.NoError(t, err) assert.Equal(t, "foo", args.Input) bar := "bar" baz := "baz" assert.Equal(t, []*string{&bar, &baz}, args.Output) } func TestRequiredPositional(t *testing.T) { var args struct { Input string `arg:"positional"` Output string `arg:"positional,required"` } _, err := parse(&args, "foo") assert.Error(t, err) } func TestRequiredPositionalMultiple(t *testing.T) { var args struct { Input string `arg:"positional"` Multiple []string `arg:"positional,required"` } _, err := parse(&args, "foo") assert.Error(t, err) } func TestTooManyPositional(t *testing.T) { var args struct { Input string `arg:"positional"` Output string `arg:"positional"` } _, err := parse(&args, "foo bar baz") assert.Error(t, err) } func TestMultiple(t *testing.T) { var args struct { Foo []int Bar []string } _, err := parse(&args, "--foo 1 2 3 --bar x y z") require.NoError(t, err) assert.Equal(t, []int{1, 2, 3}, args.Foo) assert.Equal(t, []string{"x", "y", "z"}, args.Bar) } func TestMultiplePositionals(t *testing.T) { var args struct { Input string `arg:"positional"` Multiple []string `arg:"positional,required"` } _, err := parse(&args, "foo a b c") assert.NoError(t, err) assert.Equal(t, "foo", args.Input) assert.Equal(t, []string{"a", "b", "c"}, args.Multiple) } func TestMultipleWithEq(t *testing.T) { var args struct { Foo []int Bar []string } _, err := parse(&args, "--foo 1 2 3 --bar=x") require.NoError(t, err) assert.Equal(t, []int{1, 2, 3}, args.Foo) assert.Equal(t, []string{"x"}, args.Bar) } func TestMultipleWithDefault(t *testing.T) { var args struct { Foo []int Bar []string } args.Foo = []int{42} args.Bar = []string{"foo"} _, err := parse(&args, "--foo 1 2 3 --bar x y z") require.NoError(t, err) assert.Equal(t, []int{1, 2, 3}, args.Foo) assert.Equal(t, []string{"x", "y", "z"}, args.Bar) } func TestExemptField(t *testing.T) { var args struct { Foo string Bar interface{} `arg:"-"` } _, err := parse(&args, "--foo xyz") require.NoError(t, err) assert.Equal(t, "xyz", args.Foo) } func TestUnknownField(t *testing.T) { var args struct { Foo string } _, err := parse(&args, "--bar xyz") assert.Error(t, err) } func TestNonsenseKey(t *testing.T) { var args struct { X []string `arg:"positional, nonsense"` } _, err := parse(&args, "x") assert.Error(t, err) } func TestMissingValueAtEnd(t *testing.T) { var args struct { Foo string } _, err := parse(&args, "--foo") assert.Error(t, err) } func TestMissingValueInMiddle(t *testing.T) { var args struct { Foo string Bar string } _, err := parse(&args, "--foo --bar=abc") assert.Error(t, err) } func TestNegativeValue(t *testing.T) { var args struct { Foo int } _, err := parse(&args, "--foo=-123") require.NoError(t, err) assert.Equal(t, -123, args.Foo) } func TestInvalidInt(t *testing.T) { var args struct { Foo int } _, err := parse(&args, "--foo=xyz") assert.Error(t, err) } func TestInvalidUint(t *testing.T) { var args struct { Foo uint } _, err := parse(&args, "--foo=xyz") assert.Error(t, err) } func TestInvalidFloat(t *testing.T) { var args struct { Foo float64 } _, err := parse(&args, "--foo xyz") require.Error(t, err) } func TestInvalidBool(t *testing.T) { var args struct { Foo bool } _, err := parse(&args, "--foo=xyz") require.Error(t, err) } func TestInvalidIntSlice(t *testing.T) { var args struct { Foo []int } _, err := parse(&args, "--foo 1 2 xyz") require.Error(t, err) } func TestInvalidPositional(t *testing.T) { var args struct { Foo int `arg:"positional"` } _, err := parse(&args, "xyz") require.Error(t, err) } func TestInvalidPositionalSlice(t *testing.T) { var args struct { Foo []int `arg:"positional"` } _, err := parse(&args, "1 2 xyz") require.Error(t, err) } func TestNoMoreOptions(t *testing.T) { var args struct { Foo string Bar []string `arg:"positional"` } _, err := parse(&args, "abc -- --foo xyz") require.NoError(t, err) assert.Equal(t, "", args.Foo) assert.Equal(t, []string{"abc", "--foo", "xyz"}, args.Bar) } func TestNoMoreOptionsBeforeHelp(t *testing.T) { var args struct { Foo int } _, err := parse(&args, "not_an_integer -- --help") assert.NotEqual(t, ErrHelp, err) } func TestHelpFlag(t *testing.T) { var args struct { Foo string Bar interface{} `arg:"-"` } _, err := parse(&args, "--help") assert.Equal(t, ErrHelp, err) } func TestPanicOnNonPointer(t *testing.T) { var args struct{} assert.Panics(t, func() { _, _ = parse(args, "") }) } func TestErrorOnNonStruct(t *testing.T) { var args string _, err := parse(&args, "") assert.Error(t, err) } func TestUnsupportedType(t *testing.T) { var args struct { Foo interface{} } _, err := parse(&args, "--foo") assert.Error(t, err) } func TestUnsupportedSliceElement(t *testing.T) { var args struct { Foo []interface{} } _, err := parse(&args, "--foo 3") assert.Error(t, err) } func TestUnsupportedSliceElementMissingValue(t *testing.T) { var args struct { Foo []interface{} } _, err := parse(&args, "--foo") assert.Error(t, err) } func TestUnknownTag(t *testing.T) { var args struct { Foo string `arg:"this_is_not_valid"` } _, err := parse(&args, "--foo xyz") assert.Error(t, err) } func TestParse(t *testing.T) { var args struct { Foo string } _, err := parse(&args, "--foo bar") require.NoError(t, err) assert.Equal(t, "bar", args.Foo) } func TestParseError(t *testing.T) { var args struct { Foo string `arg:"this_is_not_valid"` } _, err := NewParser(&args) assert.Error(t, err) } func TestMustParse(t *testing.T) { var args struct { Foo string } os.Args = []string{"example", "--foo", "bar"} parser := MustParse(&args) assert.Equal(t, "bar", args.Foo) assert.NotNil(t, parser) } func TestEnvironmentVariable(t *testing.T) { var args struct { Foo string `arg:"env"` } _, err := parse(&args, "", "FOO=bar") require.NoError(t, err) assert.Equal(t, "bar", args.Foo) } func TestEnvironmentVariableNotPresent(t *testing.T) { var args struct { NotPresent string `arg:"env"` } _, err := parse(&args, "", "") require.NoError(t, err) assert.Equal(t, "", args.NotPresent) } func TestEnvironmentVariableOverrideName(t *testing.T) { var args struct { Foo string `arg:"env:BAZ"` } _, err := parse(&args, "", "BAZ=bar") require.NoError(t, err) assert.Equal(t, "bar", args.Foo) } func TestCommandLineSupercedesEnv(t *testing.T) { var args struct { Foo string `arg:"env"` } _, err := parse(&args, "--foo zzz", "FOO=bar") require.NoError(t, err) assert.Equal(t, "zzz", args.Foo) } func TestEnvironmentVariableError(t *testing.T) { var args struct { Foo int `arg:"env"` } _, err := parse(&args, "", "FOO=bar") assert.Error(t, err) } func TestEnvironmentVariableRequired(t *testing.T) { var args struct { Foo string `arg:"env,required"` } _, err := parse(&args, "", "FOO=bar") require.NoError(t, err) assert.Equal(t, "bar", args.Foo) } func TestEnvironmentVariableSliceArgumentString(t *testing.T) { var args struct { Foo []string `arg:"env"` } _, err := parse(&args, "", `FOO=bar,"baz, qux"`) require.NoError(t, err) assert.Equal(t, []string{"bar", "baz, qux"}, args.Foo) } func TestEnvironmentVariableSliceEmpty(t *testing.T) { var args struct { Foo []string `arg:"env"` } _, err := parse(&args, "", `FOO=`) require.NoError(t, err) assert.Len(t, args.Foo, 0) } func TestEnvironmentVariableSliceArgumentInteger(t *testing.T) { var args struct { Foo []int `arg:"env"` } _, err := parse(&args, "", `FOO=1,99`) require.NoError(t, err) assert.Equal(t, []int{1, 99}, args.Foo) } func TestEnvironmentVariableSliceArgumentFloat(t *testing.T) { var args struct { Foo []float32 `arg:"env"` } _, err := parse(&args, "", `FOO=1.1,99.9`) require.NoError(t, err) assert.Equal(t, []float32{1.1, 99.9}, args.Foo) } func TestEnvironmentVariableSliceArgumentBool(t *testing.T) { var args struct { Foo []bool `arg:"env"` } _, err := parse(&args, "", `FOO=true,false,0,1`) require.NoError(t, err) assert.Equal(t, []bool{true, false, false, true}, args.Foo) } func TestEnvironmentVariableSliceArgumentWrongCsv(t *testing.T) { var args struct { Foo []int `arg:"env"` } _, err := parse(&args, "", `FOO=1,99\"`) assert.Error(t, err) } func TestEnvironmentVariableSliceArgumentWrongType(t *testing.T) { var args struct { Foo []bool `arg:"env"` } _, err := parse(&args, "", `FOO=one,two`) assert.Error(t, err) } func TestEnvironmentVariableMap(t *testing.T) { var args struct { Foo map[int]string `arg:"env"` } _, err := parse(&args, "", `FOO=1=one,99=ninetynine`) require.NoError(t, err) assert.Len(t, args.Foo, 2) assert.Equal(t, "one", args.Foo[1]) assert.Equal(t, "ninetynine", args.Foo[99]) } func TestEnvironmentVariableEmptyMap(t *testing.T) { var args struct { Foo map[int]string `arg:"env"` } _, err := parse(&args, "", `FOO=`) require.NoError(t, err) assert.Len(t, args.Foo, 0) } func TestEnvironmentVariableIgnored(t *testing.T) { var args struct { Foo string `arg:"env"` } // the library should never read env vars direct from os os.Setenv("FOO", "123") _, err := parse(&args, "") require.NoError(t, err) assert.Equal(t, "", args.Foo) } func TestDefaultValuesIgnored(t *testing.T) { var args struct { Foo string `default:"bad"` } // just checking that default values are not automatically applied // in ProcessCommandLine or ProcessEnvironment p, err := NewParser(&args) require.NoError(t, err) err = p.ProcessCommandLine(nil) assert.NoError(t, err) err = p.ProcessEnvironment(nil) assert.NoError(t, err) assert.Equal(t, "", args.Foo) } func TestEnvironmentVariableInSubcommand(t *testing.T) { var args struct { Sub *struct { Foo string `arg:"env:FOO"` } `arg:"subcommand"` } _, err := parse(&args, "sub", "FOO=abc") require.NoError(t, err) require.NotNil(t, args.Sub) assert.Equal(t, "abc", args.Sub.Foo) } func TestEnvironmentVariableInSubcommandEmpty(t *testing.T) { var args struct { Sub *struct { Foo string `arg:"env:FOO"` } `arg:"subcommand"` } _, err := parse(&args, "sub") require.NoError(t, err) require.NotNil(t, args.Sub) assert.Equal(t, "", args.Sub.Foo) } type textUnmarshaler struct { val int } func (f *textUnmarshaler) UnmarshalText(b []byte) error { f.val = len(b) return nil } func TestTextUnmarshaler(t *testing.T) { // fields that implement TextUnmarshaler should be parsed using that interface var args struct { Foo textUnmarshaler } _, err := parse(&args, "--foo abc") require.NoError(t, err) assert.Equal(t, 3, args.Foo.val) } func TestPtrToTextUnmarshaler(t *testing.T) { // fields that implement TextUnmarshaler should be parsed using that interface var args struct { Foo *textUnmarshaler } _, err := parse(&args, "--foo abc") require.NoError(t, err) assert.Equal(t, 3, args.Foo.val) } func TestRepeatedTextUnmarshaler(t *testing.T) { // fields that implement TextUnmarshaler should be parsed using that interface var args struct { Foo []textUnmarshaler } _, err := parse(&args, "--foo abc d ef") require.NoError(t, err) require.Len(t, args.Foo, 3) assert.Equal(t, 3, args.Foo[0].val) assert.Equal(t, 1, args.Foo[1].val) assert.Equal(t, 2, args.Foo[2].val) } func TestRepeatedPtrToTextUnmarshaler(t *testing.T) { // fields that implement TextUnmarshaler should be parsed using that interface var args struct { Foo []*textUnmarshaler } _, err := parse(&args, "--foo abc d ef") require.NoError(t, err) require.Len(t, args.Foo, 3) assert.Equal(t, 3, args.Foo[0].val) assert.Equal(t, 1, args.Foo[1].val) assert.Equal(t, 2, args.Foo[2].val) } func TestPositionalTextUnmarshaler(t *testing.T) { // fields that implement TextUnmarshaler should be parsed using that interface var args struct { Foo []textUnmarshaler `arg:"positional"` } _, err := parse(&args, "abc d ef") require.NoError(t, err) require.Len(t, args.Foo, 3) assert.Equal(t, 3, args.Foo[0].val) assert.Equal(t, 1, args.Foo[1].val) assert.Equal(t, 2, args.Foo[2].val) } func TestPositionalPtrToTextUnmarshaler(t *testing.T) { // fields that implement TextUnmarshaler should be parsed using that interface var args struct { Foo []*textUnmarshaler `arg:"positional"` } _, err := parse(&args, "abc d ef") require.NoError(t, err) require.Len(t, args.Foo, 3) assert.Equal(t, 3, args.Foo[0].val) assert.Equal(t, 1, args.Foo[1].val) assert.Equal(t, 2, args.Foo[2].val) } type boolUnmarshaler bool func (p *boolUnmarshaler) UnmarshalText(b []byte) error { *p = len(b)%2 == 0 return nil } func TestBoolUnmarhsaler(t *testing.T) { // test that a bool type that implements TextUnmarshaler is // handled as a TextUnmarshaler not as a bool var args struct { Foo *boolUnmarshaler } _, err := parse(&args, "--foo ab") require.NoError(t, err) assert.EqualValues(t, true, *args.Foo) } type sliceUnmarshaler []int func (p *sliceUnmarshaler) UnmarshalText(b []byte) error { *p = sliceUnmarshaler{len(b)} return nil } func TestSliceUnmarhsaler(t *testing.T) { // test that a slice type that implements TextUnmarshaler is // handled as a TextUnmarshaler not as a slice var args struct { Foo *sliceUnmarshaler Bar string `arg:"positional"` } _, err := parse(&args, "--foo abcde xyz") require.NoError(t, err) require.Len(t, *args.Foo, 1) assert.EqualValues(t, 5, (*args.Foo)[0]) assert.Equal(t, "xyz", args.Bar) } func TestIP(t *testing.T) { var args struct { Host net.IP } _, err := parse(&args, "--host 192.168.0.1") require.NoError(t, err) assert.Equal(t, "192.168.0.1", args.Host.String()) } func TestPtrToIP(t *testing.T) { var args struct { Host *net.IP } _, err := parse(&args, "--host 192.168.0.1") require.NoError(t, err) assert.Equal(t, "192.168.0.1", args.Host.String()) } func TestURL(t *testing.T) { var args struct { URL url.URL } _, err := parse(&args, "--url https://example.com/get?item=xyz") require.NoError(t, err) assert.Equal(t, "https://example.com/get?item=xyz", args.URL.String()) } func TestPtrToURL(t *testing.T) { var args struct { URL *url.URL } _, err := parse(&args, "--url http://example.com/#xyz") require.NoError(t, err) assert.Equal(t, "http://example.com/#xyz", args.URL.String()) } func TestIPSlice(t *testing.T) { var args struct { Host []net.IP } _, err := parse(&args, "--host 192.168.0.1 127.0.0.1") require.NoError(t, err) require.Len(t, args.Host, 2) assert.Equal(t, "192.168.0.1", args.Host[0].String()) assert.Equal(t, "127.0.0.1", args.Host[1].String()) } func TestInvalidIPAddress(t *testing.T) { var args struct { Host net.IP } _, err := parse(&args, "--host xxx") assert.Error(t, err) } func TestMAC(t *testing.T) { var args struct { Host net.HardwareAddr } _, err := parse(&args, "--host 0123.4567.89ab") require.NoError(t, err) assert.Equal(t, "01:23:45:67:89:ab", args.Host.String()) } func TestInvalidMac(t *testing.T) { var args struct { Host net.HardwareAddr } _, err := parse(&args, "--host xxx") assert.Error(t, err) } func TestMailAddr(t *testing.T) { var args struct { Recipient mail.Address } _, err := parse(&args, "--recipient foo@example.com") require.NoError(t, err) assert.Equal(t, "", args.Recipient.String()) } func TestInvalidMailAddr(t *testing.T) { var args struct { Recipient mail.Address } _, err := parse(&args, "--recipient xxx") assert.Error(t, err) } type A struct { X string } type B struct { Y int } func TestEmbedded(t *testing.T) { var args struct { A B Z bool } _, err := parse(&args, "--x=hello --y=321 --z") require.NoError(t, err) assert.Equal(t, "hello", args.X) assert.Equal(t, 321, args.Y) assert.Equal(t, true, args.Z) } func TestEmbeddedPtr(t *testing.T) { // embedded pointer fields are not supported so this should return an error var args struct { *A } _, err := parse(&args, "--x=hello") require.Error(t, err) } func TestEmbeddedPtrIgnored(t *testing.T) { // embedded pointer fields are not normally supported but here // we explicitly exclude it so the non-nil embedded structs // should work as expected var args struct { *A `arg:"-"` B } _, err := parse(&args, "--y=321") require.NoError(t, err) assert.Equal(t, 321, args.Y) } func TestEmbeddedWithDuplicateField(t *testing.T) { // see https://github.com/alexflint/go-arg/issues/100 type T struct { A string `arg:"--cat"` } type U struct { A string `arg:"--dog"` } var args struct { T U } _, err := parse(&args, "--cat=cat --dog=dog") require.NoError(t, err) assert.Equal(t, "cat", args.T.A) assert.Equal(t, "dog", args.U.A) } func TestEmbeddedWithDuplicateField2(t *testing.T) { // see https://github.com/alexflint/go-arg/issues/100 type T struct { A string } type U struct { A string } var args struct { T U } _, err := parse(&args, "--a=xyz") require.NoError(t, err) assert.Equal(t, "xyz", args.T.A) assert.Equal(t, "", args.U.A) } func TestUnexportedEmbedded(t *testing.T) { type embeddedArgs struct { Foo string } var args struct { embeddedArgs } _, err := parse(&args, "--foo bar") require.NoError(t, err) assert.Equal(t, "bar", args.Foo) } func TestIgnoredEmbedded(t *testing.T) { type embeddedArgs struct { Foo string } var args struct { embeddedArgs `arg:"-"` } _, err := parse(&args, "--foo bar") require.Error(t, err) } func TestEmptyArgs(t *testing.T) { origArgs := os.Args // test what happens if somehow os.Args is empty os.Args = nil var args struct { Foo string } err := Parse(&args) require.NoError(t, err) // put the original arguments back os.Args = origArgs } func TestTooManyHyphens(t *testing.T) { var args struct { TooManyHyphens string `arg:"---x"` } _, err := parse(&args, "--foo -") assert.Error(t, err) } func TestHyphenAsOption(t *testing.T) { var args struct { Foo string } _, err := parse(&args, "--foo -") require.NoError(t, err) assert.Equal(t, "-", args.Foo) } func TestHyphenAsPositional(t *testing.T) { var args struct { Foo string `arg:"positional"` } _, err := parse(&args, "-") require.NoError(t, err) assert.Equal(t, "-", args.Foo) } func TestHyphenInMultiOption(t *testing.T) { var args struct { Foo []string Bar int } _, err := parse(&args, "--foo --- x - y --bar 3") require.NoError(t, err) assert.Equal(t, []string{"---", "x", "-", "y"}, args.Foo) assert.Equal(t, 3, args.Bar) } func TestHyphenInMultiPositional(t *testing.T) { var args struct { Foo []string `arg:"positional"` } _, err := parse(&args, "--- x - y") require.NoError(t, err) assert.Equal(t, []string{"---", "x", "-", "y"}, args.Foo) } func TestSeparate(t *testing.T) { for _, val := range []string{"-f one", "-f=one", "--foo one", "--foo=one"} { var args struct { Foo []string `arg:"--foo,-f,separate"` } _, err := parse(&args, val) require.NoError(t, err) assert.Equal(t, []string{"one"}, args.Foo) } } func TestSeparateWithDefault(t *testing.T) { args := struct { Foo []string `arg:"--foo,-f,separate"` }{ Foo: []string{"default"}, } _, err := parse(&args, "-f one -f=two") require.NoError(t, err) assert.Equal(t, []string{"default", "one", "two"}, args.Foo) } func TestSeparateWithPositional(t *testing.T) { var args struct { Foo []string `arg:"--foo,-f,separate"` Bar string `arg:"positional"` Moo string `arg:"positional"` } _, err := parse(&args, "zzz --foo one -f=two --foo=three -f four aaa") require.NoError(t, err) assert.Equal(t, []string{"one", "two", "three", "four"}, args.Foo) assert.Equal(t, "zzz", args.Bar) assert.Equal(t, "aaa", args.Moo) } func TestSeparatePositionalInterweaved(t *testing.T) { var args struct { Foo []string `arg:"--foo,-f,separate"` Bar []string `arg:"--bar,-b,separate"` Pre string `arg:"positional"` Post []string `arg:"positional"` } _, err := parse(&args, "zzz -f foo1 -b=bar1 --foo=foo2 -b bar2 post1 -b bar3 post2 post3") require.NoError(t, err) assert.Equal(t, []string{"foo1", "foo2"}, args.Foo) assert.Equal(t, []string{"bar1", "bar2", "bar3"}, args.Bar) assert.Equal(t, "zzz", args.Pre) assert.Equal(t, []string{"post1", "post2", "post3"}, args.Post) } func TestSpacesAllowedInTags(t *testing.T) { var args struct { Foo []string `arg:"--foo, -f, separate, required, help:quite nice really"` } _, err := parse(&args, "--foo one -f=two --foo=three -f four") require.NoError(t, err) assert.Equal(t, []string{"one", "two", "three", "four"}, args.Foo) } func TestReuseParser(t *testing.T) { var args struct { Foo string `arg:"required"` } p, err := NewParser(&args) require.NoError(t, err) err = p.Parse([]string{"program", "--foo=abc"}, nil) require.NoError(t, err) assert.Equal(t, args.Foo, "abc") err = p.Parse([]string{}, nil) assert.Error(t, err) } func TestVersion(t *testing.T) { var args struct{} _, err := parse(&args, "--version") assert.Equal(t, ErrVersion, err) } func TestMultipleTerminates(t *testing.T) { var args struct { X []string Y string `arg:"positional"` } _, err := parse(&args, "--x a b -- c") require.NoError(t, err) assert.Equal(t, []string{"a", "b"}, args.X) assert.Equal(t, "c", args.Y) } func TestDefaultOptionValues(t *testing.T) { var args struct { A int `default:"123"` B *int `default:"123"` C string `default:"abc"` D *string `default:"abc"` E float64 `default:"1.23"` F *float64 `default:"1.23"` G bool `default:"true"` H *bool `default:"true"` } _, err := parse(&args, "--c=xyz --e=4.56") require.NoError(t, err) assert.Equal(t, 123, args.A) assert.Equal(t, 123, *args.B) assert.Equal(t, "xyz", args.C) assert.Equal(t, "abc", *args.D) assert.Equal(t, 4.56, args.E) assert.Equal(t, 1.23, *args.F) assert.True(t, args.G) assert.True(t, args.G) } func TestDefaultUnparseable(t *testing.T) { var args struct { A int `default:"x"` } _, err := parse(&args, "") assert.EqualError(t, err, `error processing default value for --a: strconv.ParseInt: parsing "x": invalid syntax`) } func TestDefaultPositionalValues(t *testing.T) { var args struct { A int `arg:"positional" default:"123"` B *int `arg:"positional" default:"123"` C string `arg:"positional" default:"abc"` D *string `arg:"positional" default:"abc"` E float64 `arg:"positional" default:"1.23"` F *float64 `arg:"positional" default:"1.23"` G bool `arg:"positional" default:"true"` H *bool `arg:"positional" default:"true"` } _, err := parse(&args, "456 789") require.NoError(t, err) assert.Equal(t, 456, args.A) assert.Equal(t, 789, *args.B) assert.Equal(t, "abc", args.C) assert.Equal(t, "abc", *args.D) assert.Equal(t, 1.23, args.E) assert.Equal(t, 1.23, *args.F) assert.True(t, args.G) assert.True(t, args.G) } func TestDefaultValuesNotAllowedWithRequired(t *testing.T) { var args struct { A int `arg:"required" default:"123"` // required not allowed with default! } _, err := parse(&args, "") assert.EqualError(t, err, ".A: 'required' cannot be used when a default value is specified") } func TestDefaultValuesNotAllowedWithSlice(t *testing.T) { var args struct { A []int `default:"123"` // required not allowed with default! } _, err := parse(&args, "") assert.EqualError(t, err, ".A: default values are not supported for slice or map fields") } func TestUnexportedFieldsSkipped(t *testing.T) { var args struct { unexported struct{} } _, err := NewParser(&args) require.NoError(t, err) } func TestMustParseInvalidParser(t *testing.T) { originalExit := osExit originalStdout := stdout defer func() { osExit = originalExit stdout = originalStdout }() var exitCode int osExit = func(code int) { exitCode = code } stdout = &bytes.Buffer{} var args struct { CannotParse struct{} } parser := MustParse(&args) assert.Nil(t, parser) assert.Equal(t, -1, exitCode) } func TestMustParsePrintsHelp(t *testing.T) { originalExit := osExit originalStdout := stdout originalArgs := os.Args defer func() { osExit = originalExit stdout = originalStdout os.Args = originalArgs }() var exitCode *int osExit = func(code int) { exitCode = &code } os.Args = []string{"someprogram", "--help"} stdout = &bytes.Buffer{} var args struct{} parser := MustParse(&args) assert.NotNil(t, parser) require.NotNil(t, exitCode) assert.Equal(t, 0, *exitCode) } func TestMustParsePrintsVersion(t *testing.T) { originalExit := osExit originalStdout := stdout originalArgs := os.Args defer func() { osExit = originalExit stdout = originalStdout os.Args = originalArgs }() var exitCode *int osExit = func(code int) { exitCode = &code } os.Args = []string{"someprogram", "--version"} var b bytes.Buffer stdout = &b var args versioned parser := MustParse(&args) require.NotNil(t, parser) require.NotNil(t, exitCode) assert.Equal(t, 0, *exitCode) assert.Equal(t, "example 3.2.1\n", b.String()) }