2015-11-04 11:47:58 -06:00
|
|
|
package arg
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2019-04-04 11:10:24 -05:00
|
|
|
"errors"
|
|
|
|
"fmt"
|
2015-11-04 11:47:58 -06:00
|
|
|
"os"
|
2019-04-04 11:10:24 -05:00
|
|
|
"strings"
|
2015-11-04 11:47:58 -06:00
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
)
|
|
|
|
|
2018-04-12 23:46:24 -05:00
|
|
|
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:
|
2015-11-19 07:34:09 -06:00
|
|
|
--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
|
2015-12-04 08:59:13 -06:00
|
|
|
--ids IDS Ids
|
2022-06-09 10:21:29 -05:00
|
|
|
--values VALUES Values
|
2016-01-18 12:42:04 -06:00
|
|
|
--workers WORKERS, -w WORKERS
|
2020-06-03 03:05:20 -05:00
|
|
|
number of workers to start [default: 10, env: WORKERS]
|
|
|
|
--testenv TESTENV, -a TESTENV [env: TEST_ENV]
|
2018-04-12 23:46:24 -05:00
|
|
|
--file FILE, -f FILE File with mandatory extension [default: scratch.txt]
|
2015-11-11 03:15:57 -06:00
|
|
|
--help, -h display this help and exit
|
2023-06-03 05:47:47 -05:00
|
|
|
|
|
|
|
Environment variables:
|
|
|
|
API_KEY Required. Only via env-var for security reasons
|
|
|
|
TRACE Optional. Record low-level trace
|
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 {
|
2022-02-09 08:31:34 -06:00
|
|
|
Input string `arg:"positional,required"`
|
2019-04-04 11:10:24 -05:00
|
|
|
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"`
|
2020-06-03 03:05:20 -05:00
|
|
|
Workers int `arg:"-w,env:WORKERS" help:"number of workers to start" default:"10"`
|
|
|
|
TestEnv string `arg:"-a,env:TEST_ENV"`
|
2023-06-03 05:47:47 -05:00
|
|
|
ApiKey string `arg:"required,-,--,env:API_KEY" help:"Only via env-var for security reasons"`
|
|
|
|
Trace bool `arg:"-,--,env" help:"Record low-level trace"`
|
2018-04-12 23:46:24 -05:00
|
|
|
File *NameDotName `arg:"-f" help:"File with mandatory extension"`
|
2015-11-04 11:47:58 -06:00
|
|
|
}
|
2015-11-19 07:34:09 -06:00
|
|
|
args.Name = "Foo Bar"
|
|
|
|
args.Value = 42
|
2018-04-12 23:46:24 -05:00
|
|
|
args.File = &NameDotName{"scratch", "txt"}
|
2020-03-01 16:32:59 -06:00
|
|
|
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
|
|
|
}
|
2015-11-21 17:59:40 -06:00
|
|
|
|
2018-04-15 20:07:48 -05:00
|
|
|
type MyEnum int
|
|
|
|
|
|
|
|
func (n *MyEnum) UnmarshalText(b []byte) error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-04-04 11:10:24 -05:00
|
|
|
func (n *MyEnum) MarshalText() ([]byte, error) {
|
|
|
|
return nil, errors.New("There was a problem")
|
2018-04-15 20:07:48 -05:00
|
|
|
}
|
|
|
|
|
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]
|
2018-04-15 20:07:48 -05:00
|
|
|
|
|
|
|
Options:
|
2019-10-20 01:23:32 -05:00
|
|
|
--label LABEL [default: cat]
|
|
|
|
--content CONTENT [default: dog]
|
2018-04-15 20:07:48 -05:00
|
|
|
--help, -h display this help and exit
|
|
|
|
`
|
|
|
|
var args struct {
|
2019-10-20 01:23:32 -05:00
|
|
|
Label string
|
|
|
|
Content string `default:"dog"`
|
2018-04-15 20:07:48 -05:00
|
|
|
}
|
2019-10-20 01:23:32 -05:00
|
|
|
args.Label = "cat"
|
2020-03-01 16:32:59 -06:00
|
|
|
p, err := NewParser(Config{Program: "example"}, &args)
|
2018-04-15 20:07:48 -05:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
2019-10-20 01:23:32 -05:00
|
|
|
args.Label = "should_ignore_this"
|
|
|
|
|
2018-04-15 20:07:48 -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()))
|
2018-04-15 20:07:48 -05:00
|
|
|
}
|
|
|
|
|
2019-10-20 01:23:32 -05:00
|
|
|
func TestUsageCannotMarshalToString(t *testing.T) {
|
|
|
|
var args struct {
|
|
|
|
Name *MyEnum
|
|
|
|
}
|
|
|
|
v := MyEnum(42)
|
|
|
|
args.Name = &v
|
2020-03-01 16:32:59 -06:00
|
|
|
_, 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`)
|
|
|
|
}
|
|
|
|
|
2017-10-02 08:18:41 -05:00
|
|
|
func TestUsageLongPositionalWithHelp_legacyForm(t *testing.T) {
|
2022-02-09 08:31:34 -06:00
|
|
|
expectedUsage := "Usage: example [VERYLONGPOSITIONALWITHHELP]"
|
2021-04-19 23:03:43 -05:00
|
|
|
|
|
|
|
expectedHelp := `
|
2022-02-09 08:31:34 -06:00
|
|
|
Usage: example [VERYLONGPOSITIONALWITHHELP]
|
2015-11-21 17:59:40 -06:00
|
|
|
|
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
|
2015-11-21 17:59:40 -06:00
|
|
|
|
2017-03-08 13:44:01 -06:00
|
|
|
Options:
|
2015-11-21 17:59:40 -06:00
|
|
|
--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"`
|
2015-11-21 17:59:40 -06:00
|
|
|
}
|
|
|
|
|
2021-04-19 23:03:43 -05:00
|
|
|
p, err := NewParser(Config{Program: "example"}, &args)
|
2016-01-18 12:31:01 -06:00
|
|
|
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-01-18 12:31:01 -06:00
|
|
|
}
|
|
|
|
|
2017-10-02 08:18:41 -05:00
|
|
|
func TestUsageLongPositionalWithHelp_newForm(t *testing.T) {
|
2022-02-09 08:31:34 -06:00
|
|
|
expectedUsage := "Usage: example [VERYLONGPOSITIONALWITHHELP]"
|
2021-04-19 23:03:43 -05:00
|
|
|
|
|
|
|
expectedHelp := `
|
2022-02-09 08:31:34 -06:00
|
|
|
Usage: example [VERYLONGPOSITIONALWITHHELP]
|
2017-10-02 08:18:41 -05:00
|
|
|
|
|
|
|
Positional arguments:
|
|
|
|
VERYLONGPOSITIONALWITHHELP
|
2017-10-02 08:36:23 -05:00
|
|
|
this positional argument is very long, and includes: commas, colons etc
|
2017-10-02 08:18:41 -05: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, and includes: commas, colons etc"`
|
2017-10-02 08:18:41 -05:00
|
|
|
}
|
|
|
|
|
2021-04-19 23:03:43 -05:00
|
|
|
p, err := NewParser(Config{Program: "example"}, &args)
|
2017-10-02 08:18:41 -05:00
|
|
|
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-10-02 08:18:41 -05:00
|
|
|
}
|
|
|
|
|
2016-01-18 12:31:01 -06:00
|
|
|
func TestUsageWithProgramName(t *testing.T) {
|
2021-04-19 23:03:43 -05:00
|
|
|
expectedUsage := "Usage: myprogram"
|
|
|
|
|
|
|
|
expectedHelp := `
|
|
|
|
Usage: myprogram
|
2016-01-18 12:31:01 -06:00
|
|
|
|
2017-03-08 13:44:01 -06:00
|
|
|
Options:
|
2016-01-18 12:31:01 -06:00
|
|
|
--help, -h display this help and exit
|
|
|
|
`
|
|
|
|
config := Config{
|
|
|
|
Program: "myprogram",
|
|
|
|
}
|
|
|
|
p, err := NewParser(config, &struct{}{})
|
2015-11-21 17:59:40 -06:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
os.Args[0] = "example"
|
2021-04-19 23:03:43 -05:00
|
|
|
|
2015-11-21 17:59:40 -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()))
|
2015-11-21 17:59:40 -06:00
|
|
|
}
|
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
|
|
|
|
2024-04-02 11:05:00 -05:00
|
|
|
type userDefinedVersionFlag struct {
|
|
|
|
ShowVersion bool `arg:"--version" help:"this is a user-defined version flag"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// Version returns the version for this program
|
|
|
|
func (userDefinedVersionFlag) Version() string {
|
|
|
|
return "example 3.2.1"
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestUsageWithUserDefinedVersionFlag(t *testing.T) {
|
|
|
|
expectedUsage := "example 3.2.1\nUsage: example [--version]"
|
|
|
|
|
|
|
|
expectedHelp := `
|
|
|
|
example 3.2.1
|
|
|
|
Usage: example [--version]
|
|
|
|
|
|
|
|
Options:
|
|
|
|
--version this is a user-defined version flag
|
|
|
|
--help, -h display this help and exit
|
|
|
|
`
|
|
|
|
os.Args[0] = "example"
|
|
|
|
p, err := NewParser(Config{}, &userDefinedVersionFlag{})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
var help bytes.Buffer
|
|
|
|
p.WriteHelp(&help)
|
|
|
|
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
|
|
|
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
|
|
|
|
2022-09-17 05:39:31 -05:00
|
|
|
type epilogued struct{}
|
|
|
|
|
|
|
|
// Epilogued returns the epilogue for this program
|
|
|
|
func (epilogued) Epilogue() string {
|
|
|
|
return "For more information visit github.com/alexflint/go-arg"
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestUsageWithEpilogue(t *testing.T) {
|
|
|
|
expectedUsage := "Usage: example"
|
|
|
|
|
|
|
|
expectedHelp := `
|
|
|
|
Usage: example
|
|
|
|
|
|
|
|
Options:
|
|
|
|
--help, -h display this help and exit
|
|
|
|
|
|
|
|
For more information visit github.com/alexflint/go-arg
|
|
|
|
`
|
|
|
|
os.Args[0] = "example"
|
|
|
|
p, err := NewParser(Config{}, &epilogued{})
|
|
|
|
require.NoError(t, err)
|
2017-01-23 19:41:12 -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()))
|
2017-01-23 19:41:12 -06:00
|
|
|
}
|
2017-03-30 13:47:59 -05:00
|
|
|
|
2022-02-09 08:31:34 -06:00
|
|
|
func TestUsageForRequiredPositionals(t *testing.T) {
|
|
|
|
expectedUsage := "Usage: example REQUIRED1 REQUIRED2\n"
|
|
|
|
var args struct {
|
|
|
|
Required1 string `arg:"positional,required"`
|
|
|
|
Required2 string `arg:"positional,required"`
|
|
|
|
}
|
|
|
|
|
|
|
|
p, err := NewParser(Config{Program: "example"}, &args)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
var usage bytes.Buffer
|
|
|
|
p.WriteUsage(&usage)
|
|
|
|
assert.Equal(t, expectedUsage, usage.String())
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestUsageForMixedPositionals(t *testing.T) {
|
|
|
|
expectedUsage := "Usage: example REQUIRED1 REQUIRED2 [OPTIONAL1 [OPTIONAL2]]\n"
|
|
|
|
var args struct {
|
|
|
|
Required1 string `arg:"positional,required"`
|
|
|
|
Required2 string `arg:"positional,required"`
|
|
|
|
Optional1 string `arg:"positional"`
|
|
|
|
Optional2 string `arg:"positional"`
|
|
|
|
}
|
|
|
|
|
|
|
|
p, err := NewParser(Config{Program: "example"}, &args)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
var usage bytes.Buffer
|
|
|
|
p.WriteUsage(&usage)
|
|
|
|
assert.Equal(t, expectedUsage, usage.String())
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestUsageForRepeatedPositionals(t *testing.T) {
|
|
|
|
expectedUsage := "Usage: example REQUIRED1 REQUIRED2 REPEATED [REPEATED ...]\n"
|
|
|
|
var args struct {
|
|
|
|
Required1 string `arg:"positional,required"`
|
|
|
|
Required2 string `arg:"positional,required"`
|
|
|
|
Repeated []string `arg:"positional,required"`
|
|
|
|
}
|
|
|
|
|
|
|
|
p, err := NewParser(Config{Program: "example"}, &args)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
var usage bytes.Buffer
|
|
|
|
p.WriteUsage(&usage)
|
|
|
|
assert.Equal(t, expectedUsage, usage.String())
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestUsageForMixedAndRepeatedPositionals(t *testing.T) {
|
|
|
|
expectedUsage := "Usage: example REQUIRED1 REQUIRED2 [OPTIONAL1 [OPTIONAL2 [REPEATED [REPEATED ...]]]]\n"
|
|
|
|
var args struct {
|
|
|
|
Required1 string `arg:"positional,required"`
|
|
|
|
Required2 string `arg:"positional,required"`
|
|
|
|
Optional1 string `arg:"positional"`
|
|
|
|
Optional2 string `arg:"positional"`
|
|
|
|
Repeated []string `arg:"positional"`
|
|
|
|
}
|
|
|
|
|
|
|
|
p, err := NewParser(Config{Program: "example"}, &args)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
var usage bytes.Buffer
|
|
|
|
p.WriteUsage(&usage)
|
|
|
|
assert.Equal(t, expectedUsage, usage.String())
|
|
|
|
}
|
|
|
|
|
2017-03-30 13:47:59 -05:00
|
|
|
func TestRequiredMultiplePositionals(t *testing.T) {
|
2022-02-09 08:31:34 -06:00
|
|
|
expectedUsage := "Usage: example REQUIREDMULTIPLE [REQUIREDMULTIPLE ...]\n"
|
2021-04-19 23:03:43 -05:00
|
|
|
|
|
|
|
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 {
|
2017-10-02 08:18:41 -05:00
|
|
|
RequiredMultiple []string `arg:"positional,required" help:"required multiple positional"`
|
2017-03-30 13:47:59 -05:00
|
|
|
}
|
|
|
|
|
2022-02-09 08:31:34 -06:00
|
|
|
p, err := NewParser(Config{Program: "example"}, &args)
|
2017-03-30 13:47:59 -05:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
2020-01-23 23:09:21 -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)
|
2022-02-09 08:31:34 -06:00
|
|
|
assert.Equal(t, expectedUsage, usage.String())
|
2020-01-23 23:09:21 -06:00
|
|
|
}
|
|
|
|
|
2024-04-02 11:10:52 -05:00
|
|
|
func TestUsageWithSubcommands(t *testing.T) {
|
|
|
|
expectedUsage := "Usage: example child [--values VALUES]"
|
|
|
|
|
|
|
|
expectedHelp := `
|
|
|
|
Usage: example child [--values VALUES]
|
|
|
|
|
|
|
|
Options:
|
|
|
|
--values VALUES Values
|
|
|
|
|
|
|
|
Global options:
|
|
|
|
--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"`
|
|
|
|
} `arg:"subcommand:child"`
|
|
|
|
}
|
|
|
|
|
|
|
|
os.Args[0] = "example"
|
|
|
|
p, err := NewParser(Config{}, &args)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
_ = p.Parse([]string{"child"})
|
|
|
|
|
|
|
|
var help bytes.Buffer
|
|
|
|
p.WriteHelp(&help)
|
|
|
|
assert.Equal(t, expectedHelp[1:], help.String())
|
|
|
|
|
|
|
|
var help2 bytes.Buffer
|
|
|
|
p.WriteHelpForSubcommand(&help2, "child")
|
|
|
|
assert.Equal(t, expectedHelp[1:], help2.String())
|
|
|
|
|
|
|
|
var usage bytes.Buffer
|
|
|
|
p.WriteUsage(&usage)
|
|
|
|
assert.Equal(t, expectedUsage, strings.TrimSpace(usage.String()))
|
|
|
|
|
|
|
|
var usage2 bytes.Buffer
|
|
|
|
p.WriteUsageForSubcommand(&usage2, "child")
|
|
|
|
assert.Equal(t, expectedUsage, strings.TrimSpace(usage2.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
|
2020-01-23 23:09:21 -06:00
|
|
|
|
|
|
|
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"
|
2020-01-23 23:09:21 -06:00
|
|
|
p, err := NewParser(Config{}, &args)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
2020-01-23 23:36:24 -06:00
|
|
|
_ = p.Parse([]string{"child", "nested", "value"})
|
2020-01-23 23:09:21 -06:00
|
|
|
|
2023-10-08 19:09:05 -05:00
|
|
|
assert.Equal(t, []string{"child", "nested"}, p.SubcommandNames())
|
|
|
|
|
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())
|
|
|
|
|
2021-05-09 15:55:34 -05:00
|
|
|
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()))
|
2021-05-09 15:55:34 -05:00
|
|
|
|
|
|
|
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"`
|
|
|
|
}
|
2023-10-08 19:09:05 -05:00
|
|
|
p, err := NewParser(Config{Exit: func(int) {}}, &args)
|
2021-05-09 15:55:34 -05:00
|
|
|
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
|
|
|
|
2021-01-31 21:15:49 -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)
|
2022-10-29 13:47:13 -05:00
|
|
|
require.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
|
|
|
|
2024-03-31 09:30:12 -05:00
|
|
|
func TestUsageWithEmptyPlaceholder(t *testing.T) {
|
|
|
|
expectedUsage := "Usage: example [-a] [--b] [--c]"
|
|
|
|
|
|
|
|
expectedHelp := `
|
|
|
|
Usage: example [-a] [--b] [--c]
|
|
|
|
|
|
|
|
Options:
|
|
|
|
-a some help for a
|
|
|
|
--b some help for b
|
|
|
|
--c, -c some help for c
|
|
|
|
--help, -h display this help and exit
|
|
|
|
`
|
|
|
|
var args struct {
|
|
|
|
ShortOnly string `arg:"-a,--" placeholder:"" help:"some help for a"`
|
|
|
|
LongOnly string `arg:"--b" placeholder:"" help:"some help for b"`
|
|
|
|
Both string `arg:"-c,--c" placeholder:"" help:"some help for c"`
|
|
|
|
}
|
|
|
|
p, err := NewParser(Config{Program: "example"}, &args)
|
2022-10-29 13:47:13 -05:00
|
|
|
require.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
|
|
|
|
2021-01-31 21:15:49 -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]
|
2021-01-31 21:15:49 -06:00
|
|
|
|
|
|
|
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
|
|
|
|
2021-01-31 21:15:49 -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()))
|
2021-01-31 21:15:49 -06:00
|
|
|
}
|
|
|
|
|
2020-12-19 18:51:33 -06:00
|
|
|
func TestUsageWithEnvOptions(t *testing.T) {
|
2023-06-03 02:50:42 -05:00
|
|
|
expectedUsage := "Usage: example [-s SHORT]"
|
2021-04-19 23:03:43 -05:00
|
|
|
|
|
|
|
expectedHelp := `
|
2023-06-03 02:50:42 -05:00
|
|
|
Usage: example [-s SHORT]
|
2020-12-19 18:51:33 -06:00
|
|
|
|
|
|
|
Options:
|
|
|
|
-s SHORT [env: SHORT]
|
|
|
|
--help, -h display this help and exit
|
2022-05-21 10:44:32 -05:00
|
|
|
|
|
|
|
Environment variables:
|
2023-06-03 05:47:47 -05:00
|
|
|
ENVONLY Optional.
|
|
|
|
ENVONLY2 Optional.
|
|
|
|
CUSTOM Optional.
|
2020-12-19 18:51:33 -06:00
|
|
|
`
|
|
|
|
var args struct {
|
|
|
|
Short string `arg:"--,-s,env"`
|
|
|
|
EnvOnly string `arg:"--,env"`
|
2023-06-03 05:47:47 -05:00
|
|
|
EnvOnly2 string `arg:"--,-,env"`
|
2020-12-19 18:51:33 -06:00
|
|
|
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()))
|
|
|
|
}
|
|
|
|
|
2023-06-03 05:47:47 -05:00
|
|
|
func TestEnvOnlyArgs(t *testing.T) {
|
|
|
|
expectedUsage := "Usage: example [--arg ARG]"
|
|
|
|
|
|
|
|
expectedHelp := `
|
|
|
|
Usage: example [--arg ARG]
|
|
|
|
|
|
|
|
Options:
|
|
|
|
--arg ARG, -a ARG [env: MY_ARG]
|
|
|
|
--help, -h display this help and exit
|
|
|
|
|
|
|
|
Environment variables:
|
|
|
|
AUTH_KEY Required.
|
|
|
|
`
|
|
|
|
var args struct {
|
|
|
|
ArgParam string `arg:"-a,--arg,env:MY_ARG"`
|
|
|
|
AuthKey string `arg:"required,--,env:AUTH_KEY"`
|
|
|
|
}
|
|
|
|
p, err := NewParser(Config{Program: "example"}, &args)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
var help bytes.Buffer
|
|
|
|
p.WriteHelp(&help)
|
|
|
|
assert.Equal(t, expectedHelp[1:], help.String())
|
|
|
|
|
|
|
|
var usage bytes.Buffer
|
|
|
|
p.WriteUsage(&usage)
|
|
|
|
assert.Equal(t, expectedUsage, strings.TrimSpace(usage.String()))
|
|
|
|
}
|
|
|
|
|
2021-04-19 23:03:43 -05:00
|
|
|
func TestFail(t *testing.T) {
|
2023-02-08 08:49:03 -06:00
|
|
|
var stdout bytes.Buffer
|
2021-04-19 23:03:43 -05:00
|
|
|
var exitCode int
|
2023-02-08 08:49:03 -06:00
|
|
|
exit := func(code int) { exitCode = code }
|
2021-04-19 23:03:43 -05:00
|
|
|
|
|
|
|
expectedStdout := `
|
|
|
|
Usage: example [--foo FOO]
|
|
|
|
error: something went wrong
|
|
|
|
`
|
|
|
|
|
|
|
|
var args struct {
|
|
|
|
Foo int
|
|
|
|
}
|
2023-02-08 08:49:03 -06:00
|
|
|
p, err := NewParser(Config{Program: "example", Exit: exit, Out: &stdout}, &args)
|
2021-04-19 23:03:43 -05:00
|
|
|
require.NoError(t, err)
|
|
|
|
p.Fail("something went wrong")
|
|
|
|
|
2023-02-08 08:49:03 -06:00
|
|
|
assert.Equal(t, expectedStdout[1:], stdout.String())
|
2021-04-19 23:03:43 -05:00
|
|
|
assert.Equal(t, -1, exitCode)
|
2020-12-19 18:51:33 -06:00
|
|
|
}
|
2021-05-09 15:55:34 -05:00
|
|
|
|
|
|
|
func TestFailSubcommand(t *testing.T) {
|
2023-02-08 08:49:03 -06:00
|
|
|
var stdout bytes.Buffer
|
2021-05-09 15:55:34 -05:00
|
|
|
var exitCode int
|
2023-02-08 08:49:03 -06:00
|
|
|
exit := func(code int) { exitCode = code }
|
2021-05-09 15:55:34 -05:00
|
|
|
|
|
|
|
expectedStdout := `
|
|
|
|
Usage: example sub
|
|
|
|
error: something went wrong
|
|
|
|
`
|
|
|
|
|
|
|
|
var args struct {
|
|
|
|
Sub *struct{} `arg:"subcommand"`
|
|
|
|
}
|
2023-02-08 08:49:03 -06:00
|
|
|
p, err := NewParser(Config{Program: "example", Exit: exit, Out: &stdout}, &args)
|
2021-05-09 15:55:34 -05:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
err = p.FailSubcommand("something went wrong", "sub")
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
2023-02-08 08:49:03 -06:00
|
|
|
assert.Equal(t, expectedStdout[1:], stdout.String())
|
2021-05-09 15:55:34 -05:00
|
|
|
assert.Equal(t, -1, exitCode)
|
|
|
|
}
|
2022-10-29 14:08:48 -05:00
|
|
|
|
|
|
|
type lengthOf struct {
|
|
|
|
Length int
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *lengthOf) UnmarshalText(b []byte) error {
|
|
|
|
p.Length = len(b)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestHelpShowsDefaultValueFromOriginalTag(t *testing.T) {
|
|
|
|
// check that the usage text prints the original string from the default tag, not
|
|
|
|
// the serialization of the parsed value
|
|
|
|
|
|
|
|
expectedHelp := `
|
|
|
|
Usage: example [--test TEST]
|
|
|
|
|
|
|
|
Options:
|
|
|
|
--test TEST [default: some_default_value]
|
|
|
|
--help, -h display this help and exit
|
|
|
|
`
|
|
|
|
|
|
|
|
var args struct {
|
|
|
|
Test *lengthOf `default:"some_default_value"`
|
|
|
|
}
|
|
|
|
p, err := NewParser(Config{Program: "example"}, &args)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
var help bytes.Buffer
|
|
|
|
p.WriteHelp(&help)
|
|
|
|
assert.Equal(t, expectedHelp[1:], help.String())
|
|
|
|
}
|
2023-10-08 19:09:05 -05:00
|
|
|
|
|
|
|
func TestHelpShowsSubcommandAliases(t *testing.T) {
|
|
|
|
expectedHelp := `
|
|
|
|
Usage: example <command> [<args>]
|
|
|
|
|
|
|
|
Options:
|
|
|
|
--help, -h display this help and exit
|
|
|
|
|
|
|
|
Commands:
|
|
|
|
remove, rm, r remove something from somewhere
|
|
|
|
simple do something simple
|
|
|
|
halt, stop stop now
|
|
|
|
`
|
|
|
|
|
|
|
|
var args struct {
|
|
|
|
Remove *struct{} `arg:"subcommand:remove|rm|r" help:"remove something from somewhere"`
|
|
|
|
Simple *struct{} `arg:"subcommand" help:"do something simple"`
|
|
|
|
Stop *struct{} `arg:"subcommand:halt|stop" help:"stop now"`
|
|
|
|
}
|
|
|
|
p, err := NewParser(Config{Program: "example"}, &args)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
var help bytes.Buffer
|
|
|
|
p.WriteHelp(&help)
|
|
|
|
assert.Equal(t, expectedHelp[1:], help.String())
|
|
|
|
}
|