2015-11-04 11:47:58 -06:00
|
|
|
package arg
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"os"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestWriteUsage(t *testing.T) {
|
2017-03-08 13:44:01 -06:00
|
|
|
expectedUsage := "Usage: example [--name NAME] [--value VALUE] [--verbose] [--dataset DATASET] [--optimize OPTIMIZE] [--ids IDS] [--values VALUES] [--workers WORKERS] INPUT [OUTPUT [OUTPUT ...]]\n"
|
2015-11-04 11:47:58 -06:00
|
|
|
|
2017-03-08 13:44:01 -06:00
|
|
|
expectedHelp := `Usage: example [--name NAME] [--value VALUE] [--verbose] [--dataset DATASET] [--optimize OPTIMIZE] [--ids IDS] [--values VALUES] [--workers WORKERS] 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
|
2016-03-06 14:07:01 -06:00
|
|
|
--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
|
2015-11-11 03:15:57 -06:00
|
|
|
--help, -h display this help and exit
|
2015-11-04 11:47:58 -06:00
|
|
|
`
|
|
|
|
var args struct {
|
2016-03-06 14:07:01 -06:00
|
|
|
Input string `arg:"positional"`
|
2017-10-02 08:18:41 -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"`
|
|
|
|
Workers int `arg:"-w,env:WORKERS" help:"number of workers to start"`
|
2015-11-04 11:47:58 -06:00
|
|
|
}
|
2015-11-19 07:34:09 -06:00
|
|
|
args.Name = "Foo Bar"
|
|
|
|
args.Value = 42
|
2016-03-06 14:07:01 -06:00
|
|
|
args.Values = []float64{3.14, 42, 256}
|
2016-01-18 12:31:01 -06:00
|
|
|
p, err := NewParser(Config{}, &args)
|
2015-11-04 11:47:58 -06:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
os.Args[0] = "example"
|
|
|
|
|
|
|
|
var usage bytes.Buffer
|
|
|
|
p.WriteUsage(&usage)
|
|
|
|
assert.Equal(t, expectedUsage, usage.String())
|
|
|
|
|
|
|
|
var help bytes.Buffer
|
|
|
|
p.WriteHelp(&help)
|
|
|
|
assert.Equal(t, expectedHelp, help.String())
|
|
|
|
}
|
2015-11-21 17:59:40 -06:00
|
|
|
|
2017-10-02 08:18:41 -05:00
|
|
|
func TestUsageLongPositionalWithHelp_legacyForm(t *testing.T) {
|
2017-03-08 13:44:01 -06:00
|
|
|
expectedHelp := `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
|
|
|
}
|
|
|
|
|
2016-01-18 12:31:01 -06:00
|
|
|
p, err := NewParser(Config{}, &args)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
os.Args[0] = "example"
|
|
|
|
var help bytes.Buffer
|
|
|
|
p.WriteHelp(&help)
|
|
|
|
assert.Equal(t, expectedHelp, help.String())
|
|
|
|
}
|
|
|
|
|
2017-10-02 08:18:41 -05:00
|
|
|
func TestUsageLongPositionalWithHelp_newForm(t *testing.T) {
|
|
|
|
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
|
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
|
|
|
}
|
|
|
|
|
|
|
|
p, err := NewParser(Config{}, &args)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
os.Args[0] = "example"
|
|
|
|
var help bytes.Buffer
|
|
|
|
p.WriteHelp(&help)
|
|
|
|
assert.Equal(t, expectedHelp, help.String())
|
|
|
|
}
|
|
|
|
|
2016-01-18 12:31:01 -06:00
|
|
|
func TestUsageWithProgramName(t *testing.T) {
|
2017-03-08 13:44:01 -06:00
|
|
|
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"
|
|
|
|
var help bytes.Buffer
|
|
|
|
p.WriteHelp(&help)
|
|
|
|
assert.Equal(t, expectedHelp, help.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) {
|
|
|
|
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)
|
|
|
|
actual := help.String()
|
|
|
|
t.Logf("Expected:\n%s", expectedHelp)
|
|
|
|
t.Logf("Actual:\n%s", actual)
|
|
|
|
if expectedHelp != actual {
|
|
|
|
t.Fail()
|
|
|
|
}
|
|
|
|
}
|
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) {
|
|
|
|
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)
|
|
|
|
actual := help.String()
|
|
|
|
t.Logf("Expected:\n%s", expectedHelp)
|
|
|
|
t.Logf("Actual:\n%s", actual)
|
|
|
|
if expectedHelp != actual {
|
|
|
|
t.Fail()
|
|
|
|
}
|
|
|
|
}
|
2017-03-30 13:47:59 -05:00
|
|
|
|
|
|
|
func TestRequiredMultiplePositionals(t *testing.T) {
|
|
|
|
expectedHelp := `Usage: example REQUIREDMULTIPLE [REQUIREDMULTIPLE ...]
|
|
|
|
|
|
|
|
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
|
|
|
}
|
|
|
|
|
|
|
|
p, err := NewParser(Config{}, &args)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
os.Args[0] = "example"
|
|
|
|
var help bytes.Buffer
|
|
|
|
p.WriteHelp(&help)
|
|
|
|
assert.Equal(t, expectedHelp, help.String())
|
|
|
|
}
|