go-arg/usage_test.go

539 lines
13 KiB
Go
Raw Normal View History

2015-11-04 11:47:58 -06:00
package arg
import (
"bytes"
"errors"
"fmt"
2015-11-04 11:47:58 -06:00
"os"
"strings"
2015-11-04 11:47:58 -06:00
"testing"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
)
type NameDotName struct {
Head, Tail string
}
func (n *NameDotName) UnmarshalText(b []byte) error {
s := string(b)
pos := strings.Index(s, ".")
if pos == -1 {
return fmt.Errorf("missing period in %s", s)
}
n.Head = s[:pos]
n.Tail = s[pos+1:]
return nil
}
func (n *NameDotName) MarshalText() (text []byte, err error) {
text = []byte(fmt.Sprintf("%s.%s", n.Head, n.Tail))
return
}
2015-11-04 11:47:58 -06:00
func TestWriteUsage(t *testing.T) {
2021-04-19 23:03:43 -05:00
expectedUsage := "Usage: example [--name NAME] [--value VALUE] [--verbose] [--dataset DATASET] [--optimize OPTIMIZE] [--ids IDS] [--values VALUES] [--workers WORKERS] [--testenv TESTENV] [--file FILE] INPUT [OUTPUT [OUTPUT ...]]"
2015-11-04 11:47:58 -06:00
2021-04-19 23:03:43 -05:00
expectedHelp := `
Usage: example [--name NAME] [--value VALUE] [--verbose] [--dataset DATASET] [--optimize OPTIMIZE] [--ids IDS] [--values VALUES] [--workers WORKERS] [--testenv TESTENV] [--file FILE] INPUT [OUTPUT [OUTPUT ...]]
2015-11-04 11:47:58 -06:00
2017-03-08 13:44:01 -06:00
Positional arguments:
INPUT
OUTPUT list of outputs
2015-11-04 11:47:58 -06:00
2017-03-08 13:44:01 -06:00
Options:
--name NAME name to use [default: Foo Bar]
--value VALUE secret value [default: 42]
2015-11-04 11:47:58 -06:00
--verbose, -v verbosity level
--dataset DATASET dataset to use
--optimize OPTIMIZE, -O OPTIMIZE
optimization level
--ids IDS Ids
--values VALUES Values [default: [3.14 42 256]]
2016-01-18 12:42:04 -06:00
--workers WORKERS, -w WORKERS
number of workers to start [default: 10, env: WORKERS]
--testenv TESTENV, -a TESTENV [env: TEST_ENV]
--file FILE, -f FILE File with mandatory extension [default: scratch.txt]
--help, -h display this help and exit
2015-11-04 11:47:58 -06:00
`
2021-04-19 23:03:43 -05:00
2015-11-04 11:47:58 -06:00
var args struct {
Input string `arg:"positional"`
Output []string `arg:"positional" help:"list of outputs"`
Name string `help:"name to use"`
Value int `help:"secret value"`
Verbose bool `arg:"-v" help:"verbosity level"`
Dataset string `help:"dataset to use"`
Optimize int `arg:"-O" help:"optimization level"`
Ids []int64 `help:"Ids"`
Values []float64 `help:"Values"`
Workers int `arg:"-w,env:WORKERS" help:"number of workers to start" default:"10"`
TestEnv string `arg:"-a,env:TEST_ENV"`
File *NameDotName `arg:"-f" help:"File with mandatory extension"`
2015-11-04 11:47:58 -06:00
}
args.Name = "Foo Bar"
args.Value = 42
args.Values = []float64{3.14, 42, 256}
args.File = &NameDotName{"scratch", "txt"}
p, err := NewParser(Config{Program: "example"}, &args)
2015-11-04 11:47:58 -06:00
require.NoError(t, err)
os.Args[0] = "example"
var help bytes.Buffer
p.WriteHelp(&help)
2021-04-19 23:03:43 -05:00
assert.Equal(t, expectedHelp[1:], help.String())
var usage bytes.Buffer
p.WriteUsage(&usage)
assert.Equal(t, expectedUsage, strings.TrimSpace(usage.String()))
2015-11-04 11:47:58 -06:00
}
type MyEnum int
func (n *MyEnum) UnmarshalText(b []byte) error {
return nil
}
func (n *MyEnum) MarshalText() ([]byte, error) {
return nil, errors.New("There was a problem")
}
2019-10-20 01:23:32 -05:00
func TestUsageWithDefaults(t *testing.T) {
2021-04-19 23:03:43 -05:00
expectedUsage := "Usage: example [--label LABEL] [--content CONTENT]"
expectedHelp := `
Usage: example [--label LABEL] [--content CONTENT]
Options:
2019-10-20 01:23:32 -05:00
--label LABEL [default: cat]
--content CONTENT [default: dog]
--help, -h display this help and exit
`
var args struct {
2019-10-20 01:23:32 -05:00
Label string
Content string `default:"dog"`
}
2019-10-20 01:23:32 -05:00
args.Label = "cat"
p, err := NewParser(Config{Program: "example"}, &args)
require.NoError(t, err)
2019-10-20 01:23:32 -05:00
args.Label = "should_ignore_this"
var help bytes.Buffer
p.WriteHelp(&help)
2021-04-19 23:03:43 -05:00
assert.Equal(t, expectedHelp[1:], help.String())
var usage bytes.Buffer
p.WriteUsage(&usage)
assert.Equal(t, expectedUsage, strings.TrimSpace(usage.String()))
}
2019-10-20 01:23:32 -05:00
func TestUsageCannotMarshalToString(t *testing.T) {
var args struct {
Name *MyEnum
}
v := MyEnum(42)
args.Name = &v
_, err := NewParser(Config{Program: "example"}, &args)
2019-10-20 01:23:32 -05:00
assert.EqualError(t, err, `args.Name: error marshaling default value to string: There was a problem`)
}
func TestUsageLongPositionalWithHelp_legacyForm(t *testing.T) {
2021-04-19 23:03:43 -05:00
expectedUsage := "Usage: example VERYLONGPOSITIONALWITHHELP"
expectedHelp := `
Usage: example VERYLONGPOSITIONALWITHHELP
2017-03-08 13:44:01 -06:00
Positional arguments:
VERYLONGPOSITIONALWITHHELP
2017-10-02 08:36:23 -05:00
this positional argument is very long but cannot include commas
2017-03-08 13:44:01 -06:00
Options:
--help, -h display this help and exit
`
var args struct {
2017-10-02 08:36:23 -05:00
VeryLongPositionalWithHelp string `arg:"positional,help:this positional argument is very long but cannot include commas"`
}
2021-04-19 23:03:43 -05:00
p, err := NewParser(Config{Program: "example"}, &args)
require.NoError(t, err)
var help bytes.Buffer
p.WriteHelp(&help)
2021-04-19 23:03:43 -05:00
assert.Equal(t, expectedHelp[1:], help.String())
var usage bytes.Buffer
p.WriteUsage(&usage)
assert.Equal(t, expectedUsage, strings.TrimSpace(usage.String()))
}
func TestUsageLongPositionalWithHelp_newForm(t *testing.T) {
2021-04-19 23:03:43 -05:00
expectedUsage := "Usage: example VERYLONGPOSITIONALWITHHELP"
expectedHelp := `
Usage: example VERYLONGPOSITIONALWITHHELP
Positional arguments:
VERYLONGPOSITIONALWITHHELP
2017-10-02 08:36:23 -05:00
this positional argument is very long, and includes: commas, colons etc
Options:
--help, -h display this help and exit
`
var args struct {
2017-10-02 08:36:23 -05:00
VeryLongPositionalWithHelp string `arg:"positional" help:"this positional argument is very long, and includes: commas, colons etc"`
}
2021-04-19 23:03:43 -05:00
p, err := NewParser(Config{Program: "example"}, &args)
require.NoError(t, err)
var help bytes.Buffer
p.WriteHelp(&help)
2021-04-19 23:03:43 -05:00
assert.Equal(t, expectedHelp[1:], help.String())
var usage bytes.Buffer
p.WriteUsage(&usage)
assert.Equal(t, expectedUsage, strings.TrimSpace(usage.String()))
}
func TestUsageWithProgramName(t *testing.T) {
2021-04-19 23:03:43 -05:00
expectedUsage := "Usage: myprogram"
expectedHelp := `
Usage: myprogram
2017-03-08 13:44:01 -06:00
Options:
--help, -h display this help and exit
`
config := Config{
Program: "myprogram",
}
p, err := NewParser(config, &struct{}{})
require.NoError(t, err)
os.Args[0] = "example"
2021-04-19 23:03:43 -05:00
var help bytes.Buffer
p.WriteHelp(&help)
2021-04-19 23:03:43 -05:00
assert.Equal(t, expectedHelp[1:], help.String())
var usage bytes.Buffer
p.WriteUsage(&usage)
assert.Equal(t, expectedUsage, strings.TrimSpace(usage.String()))
}
2016-09-08 23:18:19 -05:00
type versioned struct{}
// Version returns the version for this program
func (versioned) Version() string {
return "example 3.2.1"
}
func TestUsageWithVersion(t *testing.T) {
2021-04-19 23:03:43 -05:00
expectedUsage := "example 3.2.1\nUsage: example"
expectedHelp := `
example 3.2.1
2017-03-08 13:44:01 -06:00
Usage: example
2016-09-08 23:18:19 -05:00
2017-03-08 13:44:01 -06:00
Options:
2016-09-08 23:18:19 -05:00
--help, -h display this help and exit
--version display version and exit
`
os.Args[0] = "example"
p, err := NewParser(Config{}, &versioned{})
require.NoError(t, err)
var help bytes.Buffer
p.WriteHelp(&help)
2021-04-19 23:03:43 -05:00
assert.Equal(t, expectedHelp[1:], help.String())
var usage bytes.Buffer
p.WriteUsage(&usage)
assert.Equal(t, expectedUsage, strings.TrimSpace(usage.String()))
2016-09-08 23:18:19 -05:00
}
2017-01-23 19:41:12 -06:00
type described struct{}
// Described returns the description for this program
func (described) Description() string {
return "this program does this and that"
}
func TestUsageWithDescription(t *testing.T) {
2021-04-19 23:03:43 -05:00
expectedUsage := "Usage: example"
expectedHelp := `
this program does this and that
2017-03-08 13:44:01 -06:00
Usage: example
2017-01-23 19:41:12 -06:00
2017-03-08 13:44:01 -06:00
Options:
2017-01-23 19:41:12 -06:00
--help, -h display this help and exit
`
os.Args[0] = "example"
p, err := NewParser(Config{}, &described{})
require.NoError(t, err)
var help bytes.Buffer
p.WriteHelp(&help)
2021-04-19 23:03:43 -05:00
assert.Equal(t, expectedHelp[1:], help.String())
var usage bytes.Buffer
p.WriteUsage(&usage)
assert.Equal(t, expectedUsage, strings.TrimSpace(usage.String()))
2017-01-23 19:41:12 -06:00
}
2017-03-30 13:47:59 -05:00
func TestRequiredMultiplePositionals(t *testing.T) {
2021-04-19 23:03:43 -05:00
expectedUsage := "Usage: example REQUIREDMULTIPLE [REQUIREDMULTIPLE ...]"
expectedHelp := `
Usage: example REQUIREDMULTIPLE [REQUIREDMULTIPLE ...]
2017-03-30 13:47:59 -05:00
Positional arguments:
REQUIREDMULTIPLE required multiple positional
Options:
--help, -h display this help and exit
`
var args struct {
RequiredMultiple []string `arg:"positional,required" help:"required multiple positional"`
2017-03-30 13:47:59 -05:00
}
p, err := NewParser(Config{}, &args)
require.NoError(t, err)
var help bytes.Buffer
p.WriteHelp(&help)
2021-04-19 23:03:43 -05:00
assert.Equal(t, expectedHelp[1:], help.String())
var usage bytes.Buffer
p.WriteUsage(&usage)
assert.Equal(t, expectedUsage, strings.TrimSpace(usage.String()))
}
2020-04-03 10:52:00 -05:00
func TestUsageWithNestedSubcommands(t *testing.T) {
2021-04-19 23:03:43 -05:00
expectedUsage := "Usage: example child nested [--enable] OUTPUT"
expectedHelp := `
Usage: example child nested [--enable] OUTPUT
Positional arguments:
OUTPUT
Options:
--enable
Global options:
--values VALUES Values
--verbose, -v verbosity level
--help, -h display this help and exit
`
var args struct {
Verbose bool `arg:"-v" help:"verbosity level"`
Child *struct {
Values []float64 `help:"Values"`
Nested *struct {
Enable bool
Output string `arg:"positional,required"`
} `arg:"subcommand:nested"`
} `arg:"subcommand:child"`
}
2017-03-30 13:47:59 -05:00
os.Args[0] = "example"
p, err := NewParser(Config{}, &args)
require.NoError(t, err)
2020-01-23 23:36:24 -06:00
_ = p.Parse([]string{"child", "nested", "value"})
2017-03-30 13:47:59 -05:00
var help bytes.Buffer
p.WriteHelp(&help)
2021-04-19 23:03:43 -05:00
assert.Equal(t, expectedHelp[1:], help.String())
var help2 bytes.Buffer
p.WriteHelpForSubcommand(&help2, "child", "nested")
assert.Equal(t, expectedHelp[1:], help2.String())
2021-04-19 23:03:43 -05:00
var usage bytes.Buffer
p.WriteUsage(&usage)
assert.Equal(t, expectedUsage, strings.TrimSpace(usage.String()))
var usage2 bytes.Buffer
p.WriteUsageForSubcommand(&usage2, "child", "nested")
assert.Equal(t, expectedUsage, strings.TrimSpace(usage2.String()))
}
func TestNonexistentSubcommand(t *testing.T) {
var args struct {
sub *struct{} `arg:"subcommand"`
}
p, err := NewParser(Config{}, &args)
require.NoError(t, err)
var b bytes.Buffer
err = p.WriteUsageForSubcommand(&b, "does_not_exist")
assert.Error(t, err)
err = p.WriteHelpForSubcommand(&b, "does_not_exist")
assert.Error(t, err)
err = p.FailSubcommand("something went wrong", "does_not_exist")
assert.Error(t, err)
err = p.WriteUsageForSubcommand(&b, "sub", "does_not_exist")
assert.Error(t, err)
err = p.WriteHelpForSubcommand(&b, "sub", "does_not_exist")
assert.Error(t, err)
err = p.FailSubcommand("something went wrong", "sub", "does_not_exist")
assert.Error(t, err)
2017-03-30 13:47:59 -05:00
}
2020-12-19 17:54:03 -06:00
func TestUsageWithoutLongNames(t *testing.T) {
2021-04-19 23:03:43 -05:00
expectedUsage := "Usage: example [-a PLACEHOLDER] -b SHORTONLY2"
expectedHelp := `
Usage: example [-a PLACEHOLDER] -b SHORTONLY2
2020-12-19 17:54:03 -06:00
Options:
-a PLACEHOLDER some help [default: some val]
-b SHORTONLY2 some help2
--help, -h display this help and exit
`
var args struct {
ShortOnly string `arg:"-a,--" help:"some help" default:"some val" placeholder:"PLACEHOLDER"`
ShortOnly2 string `arg:"-b,--,required" help:"some help2"`
}
p, err := NewParser(Config{Program: "example"}, &args)
assert.NoError(t, err)
2021-04-19 23:03:43 -05:00
2020-12-19 17:54:03 -06:00
var help bytes.Buffer
p.WriteHelp(&help)
2021-04-19 23:03:43 -05:00
assert.Equal(t, expectedHelp[1:], help.String())
var usage bytes.Buffer
p.WriteUsage(&usage)
assert.Equal(t, expectedUsage, strings.TrimSpace(usage.String()))
2020-12-19 17:54:03 -06:00
}
2020-12-19 18:51:33 -06:00
func TestUsageWithShortFirst(t *testing.T) {
2021-04-19 23:03:43 -05:00
expectedUsage := "Usage: example [-c CAT] [--dog DOG]"
expectedHelp := `
Usage: example [-c CAT] [--dog DOG]
Options:
-c CAT
--dog DOG
--help, -h display this help and exit
`
var args struct {
Dog string
Cat string `arg:"-c,--"`
}
p, err := NewParser(Config{Program: "example"}, &args)
assert.NoError(t, err)
2021-04-19 23:03:43 -05:00
var help bytes.Buffer
p.WriteHelp(&help)
2021-04-19 23:03:43 -05:00
assert.Equal(t, expectedHelp[1:], help.String())
var usage bytes.Buffer
p.WriteUsage(&usage)
assert.Equal(t, expectedUsage, strings.TrimSpace(usage.String()))
}
2020-12-19 18:51:33 -06:00
func TestUsageWithEnvOptions(t *testing.T) {
2021-04-19 23:03:43 -05:00
expectedUsage := "Usage: example [-s SHORT]"
expectedHelp := `
Usage: example [-s SHORT]
2020-12-19 18:51:33 -06:00
Options:
-s SHORT [env: SHORT]
--help, -h display this help and exit
`
var args struct {
Short string `arg:"--,-s,env"`
EnvOnly string `arg:"--,env"`
EnvOnlyOverriden string `arg:"--,env:CUSTOM"`
}
p, err := NewParser(Config{Program: "example"}, &args)
assert.NoError(t, err)
2021-04-19 23:03:43 -05:00
2020-12-19 18:51:33 -06:00
var help bytes.Buffer
p.WriteHelp(&help)
2021-04-19 23:03:43 -05:00
assert.Equal(t, expectedHelp[1:], help.String())
var usage bytes.Buffer
p.WriteUsage(&usage)
assert.Equal(t, expectedUsage, strings.TrimSpace(usage.String()))
}
func TestFail(t *testing.T) {
originalStderr := stderr
originalExit := osExit
defer func() {
stderr = originalStderr
osExit = originalExit
}()
var b bytes.Buffer
stderr = &b
var exitCode int
osExit = func(code int) { exitCode = code }
expectedStdout := `
Usage: example [--foo FOO]
error: something went wrong
`
var args struct {
Foo int
}
p, err := NewParser(Config{Program: "example"}, &args)
require.NoError(t, err)
p.Fail("something went wrong")
assert.Equal(t, expectedStdout[1:], b.String())
assert.Equal(t, -1, exitCode)
2020-12-19 18:51:33 -06:00
}
func TestFailSubcommand(t *testing.T) {
originalStderr := stderr
originalExit := osExit
defer func() {
stderr = originalStderr
osExit = originalExit
}()
var b bytes.Buffer
stderr = &b
var exitCode int
osExit = func(code int) { exitCode = code }
expectedStdout := `
Usage: example sub
error: something went wrong
`
var args struct {
Sub *struct{} `arg:"subcommand"`
}
p, err := NewParser(Config{Program: "example"}, &args)
require.NoError(t, err)
err = p.FailSubcommand("something went wrong", "sub")
require.NoError(t, err)
assert.Equal(t, expectedStdout[1:], b.String())
assert.Equal(t, -1, exitCode)
}