go-arg/v2/parse_test.go

1462 lines
32 KiB
Go

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)
require.NotNil(t, args.Ptr)
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 TestSliceWithEqualsSign(t *testing.T) {
var args struct {
Strings []string
}
_, err := parse(&args, "--strings=test")
require.NoError(t, err)
assert.Equal(t, []string{"test"}, 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 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) {
// check that default values are not automatically applied
// in ProcessCommandLine or ProcessEnvironment
var args struct {
Foo string `default:"hello"`
}
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, "<foo@example.com>", 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 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())
}