Tests for CLI parsing

This commit is contained in:
Kovid Goyal 2022-09-23 20:50:35 +05:30
parent 04022ed363
commit c4ab964d09
No known key found for this signature in database
GPG Key ID: 06BC317B515ACE7C
2 changed files with 150 additions and 33 deletions

View File

@ -69,6 +69,12 @@ type Option struct {
seen_option string
}
func (self *Option) reset() {
self.values_from_cmdline = self.values_from_cmdline[:0]
self.parsed_values_from_cmdline = self.parsed_values_from_cmdline[:0]
self.seen_option = ""
}
func (self *Option) needs_argument() bool {
return self.OptionType != BoolOption && self.OptionType != CountOption
}
@ -129,7 +135,7 @@ func (self *Option) parse_value(val string) (any, error) {
return nil, &ParseError{Option: self, Message: fmt.Sprintf(
":yellow:`%s` is not a valid number for :bold:`%s`. Only integers in decimal, hexadecimal, binary or octal notation are accepted.", val, self.seen_option)}
}
return pval, nil
return int(pval), nil
case FloatOption:
pval, err := strconv.ParseFloat(val, 64)
if err != nil {
@ -286,7 +292,6 @@ type Command struct { // {{{
Run func(cmd *Command, args []string) (int, error)
option_map map[string]*Option
exe_name string
}
func (self *Command) Clone(parent *Command) *Command {
@ -355,40 +360,45 @@ func (self *Command) Validate() error {
}
}
seen_flags := make(map[string]bool)
seen_dests := make(map[string]bool)
for _, g := range self.OptionGroups {
for _, o := range g.Options {
if seen_dests[o.Name] {
return &ParseError{Message: fmt.Sprintf("The option :yellow:`%s` occurs twice inside %s", o.Name, self.Name)}
}
seen_dests[o.Name] = true
for _, a := range o.Aliases {
q := a.String()
if seen_flags[q] {
return &ParseError{Message: fmt.Sprintf("The option :yellow:`%s` occurs twice inside %s", q, self.Name)}
}
seen_flags[q] = true
}
self.option_map = make(map[string]*Option, 128)
validate_options := func(opt *Option) error {
if self.option_map[opt.Name] != nil {
return &ParseError{Message: fmt.Sprintf("The option :yellow:`%s` occurs twice inside %s", opt.Name, self.Name)}
}
for _, a := range opt.Aliases {
q := a.String()
if seen_flags[q] {
return &ParseError{Message: fmt.Sprintf("The option :yellow:`%s` occurs twice inside %s", q, self.Name)}
}
seen_flags[q] = true
}
self.option_map[opt.Name] = opt
return nil
}
if !seen_dests["Help"] {
err := self.VisitAllOptions(validate_options)
if err != nil {
return err
}
if self.option_map["Help"] == nil {
if seen_flags["-h"] || seen_flags["--help"] {
return &ParseError{Message: fmt.Sprintf("The --help or -h flags are assigned to an option other than Help in %s", self.Name)}
}
self.Add(OptionSpec{Name: "--help -h", Type: "bool-set", Help: "Show help for this command"})
self.option_map["Help"] = self.Add(OptionSpec{Name: "--help -h", Type: "bool-set", Help: "Show help for this command"})
}
if self.Parent == nil && !seen_dests["Version"] {
if self.Parent == nil && self.option_map["Version"] == nil {
if seen_flags["--version"] {
return &ParseError{Message: fmt.Sprintf("The --version flag is assigned to an option other than Version in %s", self.Name)}
}
self.Add(OptionSpec{Name: "--version", Type: "bool-set", Help: "Show version"})
self.option_map["Version"] = self.Add(OptionSpec{Name: "--version", Type: "bool-set", Help: "Show version"})
}
return nil
}
func (self *Command) Root(args []string) *Command {
func (self *Command) Root() *Command {
p := self
for p.Parent != nil {
p = p.Parent
@ -402,8 +412,6 @@ func (self *Command) CommandStringForUsage() string {
for p != nil {
if p.Name != "" {
names = append(names, p.Name)
} else if p.exe_name != "" {
names = append(names, p.Name)
}
p = p.Parent
}
@ -411,8 +419,7 @@ func (self *Command) CommandStringForUsage() string {
}
func (self *Command) ParseArgs(args []string) (*Command, error) {
if self.Parent != nil {
return nil, &ParseError{Message: "ParseArgs() must be called on the Root command"}
for ; self.Parent != nil; self = self.Parent {
}
err := self.Validate()
if err != nil {
@ -425,18 +432,26 @@ func (self *Command) ParseArgs(args []string) (*Command, error) {
return nil, &ParseError{Message: "At least one arg must be supplied"}
}
ctx := Context{SeenCommands: make([]*Command, 0, 4)}
self.exe_name = args[0]
err = self.parse_args(&ctx, args[1:])
if err != nil {
return nil, err
}
self.option_map = make(map[string]*Option, 128)
for _, g := range self.OptionGroups {
for _, o := range g.Options {
self.option_map[o.Name] = o
return ctx.SeenCommands[len(ctx.SeenCommands)-1], nil
}
func (self *Command) ResetAfterParseArgs() {
for _, g := range self.SubCommandGroups {
for _, sc := range g.SubCommands {
sc.ResetAfterParseArgs()
}
}
return ctx.SeenCommands[len(ctx.SeenCommands)-1], nil
for _, g := range self.OptionGroups {
for _, o := range g.Options {
o.reset()
}
}
self.Args = make([]string, 0, 8)
}
func (self *Command) HasSubCommands() bool {
@ -457,6 +472,31 @@ func (self *Command) HasVisibleSubCommands() bool {
return false
}
func (self *Command) VisitAllOptions(callback func(*Option) error) error {
depth := 0
iter_opts := func(cmd *Command) error {
for _, g := range cmd.OptionGroups {
for _, o := range g.Options {
if o.Depth >= depth {
err := callback(o)
if err != nil {
return err
}
}
}
}
return nil
}
for p := self; p != nil; p = p.Parent {
err := iter_opts(p)
if err != nil {
return err
}
depth++
}
return nil
}
func (self *Command) GetVisibleOptions() ([]string, map[string][]*Option) {
group_titles := make([]string, 0, len(self.OptionGroups))
gmap := make(map[string][]*Option)
@ -634,13 +674,13 @@ func (self *Command) Exec(args ...string) {
os.Exit(1)
}
help_opt := cmd.option_map["Help"]
version_opt := cmd.option_map["Version"]
version_opt := root.option_map["Version"]
exit_code := 0
if help_opt != nil && help_opt.parsed_value().(bool) {
cmd.ShowHelp()
os.Exit(exit_code)
} else if version_opt != nil && version_opt.parsed_value().(bool) {
cmd.ShowVersion()
root.ShowVersion()
os.Exit(exit_code)
} else if cmd.Run != nil {
exit_code, err = cmd.Run(cmd, cmd.Args)

77
tools/cli/types_test.go Normal file
View File

@ -0,0 +1,77 @@
// License: GPLv3 Copyright: 2022, Kovid Goyal, <kovid at kovidgoyal.net>
package cli
import (
"fmt"
"reflect"
"strings"
"testing"
"github.com/google/shlex"
)
var _ = fmt.Print
type options struct {
SimpleString string
Choices string
FromParent int
SetMe bool
Int int
Float float64
}
func TestCLIParsing(t *testing.T) {
rt := func(expected_cmd *Command, cmdline string, expected_options any, expected_args ...string) {
cp, err := shlex.Split(cmdline)
if err != nil {
t.Fatal(err)
}
cmd, err := expected_cmd.ParseArgs(cp)
if err != nil {
t.Fatal(err)
}
actual_options := reflect.New(reflect.TypeOf(expected_options).Elem()).Interface()
err = cmd.GetOptionValues(actual_options)
if err != nil {
t.Fatal(err)
}
if !reflect.DeepEqual(expected_options, actual_options) {
t.Fatalf("Option values incorrect (expected != actual):\nCommand line: %s\n%#v != %#v", cmdline, expected_options, actual_options)
}
if expected_args == nil {
expected_args = []string{}
}
if !reflect.DeepEqual(expected_args, cmd.Args) {
t.Fatalf("Argument values incorrect (expected != actual):\nCommand line: %s\n%#v != %#v", cmdline, expected_args, cmd.Args)
}
cmd.Root().ResetAfterParseArgs()
}
root := NewRootCommand()
root.Add(OptionSpec{Name: "--from-parent -p", Type: "count", Depth: 1})
child1 := root.AddSubCommand("", "child1")
child1.Add(OptionSpec{Name: "--choices", Choices: "a b c"})
child1.Add(OptionSpec{Name: "--simple-string -s"})
child1.Add(OptionSpec{Name: "--set-me", Type: "bool-set"})
child1.Add(OptionSpec{Name: "--int", Type: "int"})
child1.Add(OptionSpec{Name: "--float", Type: "float"})
rt(
child1, "test --from-parent child1 -ps ss --choices b --from-parent one two",
&options{SimpleString: "ss", Choices: "b", FromParent: 3},
"one", "two",
)
rt(child1, "test child1", &options{Choices: "a"})
rt(child1, "test child1 --set-me --simple-string=foo one", &options{Choices: "a", SimpleString: "foo", SetMe: true}, "one")
rt(child1, "test child1 --set-me --simple-string= one", &options{Choices: "a", SetMe: true}, "one")
rt(child1, "test child1 --int -3 --simple-string -s --float=3.3", &options{Choices: "a", SimpleString: "-s", Int: -3, Float: 3.3})
_, err := child1.ParseArgs(strings.Split("test child1 --choices x", " "))
if err == nil {
t.Fatalf("Invalid choice not caught")
}
root.ResetAfterParseArgs()
}