2019-08-05 10:16:30 +00:00
|
|
|
package cli
|
|
|
|
|
|
|
|
import (
|
|
|
|
"flag"
|
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
|
|
|
type iterativeParser interface {
|
|
|
|
newFlagSet() (*flag.FlagSet, error)
|
|
|
|
useShortOptionHandling() bool
|
|
|
|
}
|
|
|
|
|
|
|
|
// To enable short-option handling (e.g., "-it" vs "-i -t") we have to
|
2019-11-27 16:42:48 +00:00
|
|
|
// iteratively catch parsing errors. This way we achieve LR parsing without
|
2019-08-05 10:16:30 +00:00
|
|
|
// transforming any arguments. Otherwise, there is no way we can discriminate
|
|
|
|
// combined short options from common arguments that should be left untouched.
|
2019-11-27 16:45:44 +00:00
|
|
|
// Pass `shellComplete` to continue parsing options on failure during shell
|
|
|
|
// completion when, the user-supplied options may be incomplete.
|
|
|
|
func parseIter(set *flag.FlagSet, ip iterativeParser, args []string, shellComplete bool) error {
|
2019-08-05 10:16:30 +00:00
|
|
|
for {
|
2019-09-13 09:30:07 +00:00
|
|
|
err := set.Parse(args)
|
2019-08-05 10:16:30 +00:00
|
|
|
if !ip.useShortOptionHandling() || err == nil {
|
2019-11-27 16:45:44 +00:00
|
|
|
if shellComplete {
|
2019-11-28 04:37:23 +00:00
|
|
|
return nil
|
2019-11-27 16:42:48 +00:00
|
|
|
}
|
2019-09-13 09:30:07 +00:00
|
|
|
return err
|
2019-08-05 10:16:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
errStr := err.Error()
|
2019-10-12 02:32:21 +00:00
|
|
|
trimmed := strings.TrimPrefix(errStr, "flag provided but not defined: -")
|
2019-11-27 16:45:44 +00:00
|
|
|
if errStr == trimmed {
|
2019-09-13 09:30:07 +00:00
|
|
|
return err
|
2019-08-05 10:16:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// regenerate the initial args with the split short opts
|
2019-10-12 02:57:55 +00:00
|
|
|
argsWereSplit := false
|
2019-08-05 10:16:30 +00:00
|
|
|
for i, arg := range args {
|
2019-10-12 02:32:21 +00:00
|
|
|
// skip args that are not part of the error message
|
|
|
|
if name := strings.TrimLeft(arg, "-"); name != trimmed {
|
2019-08-05 10:16:30 +00:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2019-10-12 02:32:21 +00:00
|
|
|
// if we can't split, the error was accurate
|
|
|
|
shortOpts := splitShortOptions(set, arg)
|
2019-08-05 10:16:30 +00:00
|
|
|
if len(shortOpts) == 1 {
|
2019-09-13 09:30:07 +00:00
|
|
|
return err
|
2019-08-05 10:16:30 +00:00
|
|
|
}
|
|
|
|
|
2021-03-28 12:18:46 +00:00
|
|
|
// Start processing only from failed argument and not
|
|
|
|
// from beginning
|
|
|
|
args = append(shortOpts, args[i+1:]...)
|
2019-10-12 02:57:55 +00:00
|
|
|
argsWereSplit = true
|
2019-10-12 02:32:21 +00:00
|
|
|
break
|
2019-08-05 10:16:30 +00:00
|
|
|
}
|
2019-09-13 09:30:07 +00:00
|
|
|
|
2019-10-12 02:57:55 +00:00
|
|
|
// This should be an impossible to reach code path, but in case the arg
|
|
|
|
// splitting failed to happen, this will prevent infinite loops
|
|
|
|
if !argsWereSplit {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-09-13 09:30:07 +00:00
|
|
|
// Since custom parsing failed, replace the flag set before retrying
|
|
|
|
newSet, err := ip.newFlagSet()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
*set = *newSet
|
2019-08-05 10:16:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func splitShortOptions(set *flag.FlagSet, arg string) []string {
|
|
|
|
shortFlagsExist := func(s string) bool {
|
|
|
|
for _, c := range s[1:] {
|
|
|
|
if f := set.Lookup(string(c)); f == nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
if !isSplittable(arg) || !shortFlagsExist(arg) {
|
|
|
|
return []string{arg}
|
|
|
|
}
|
|
|
|
|
|
|
|
separated := make([]string, 0, len(arg)-1)
|
|
|
|
for _, flagChar := range arg[1:] {
|
|
|
|
separated = append(separated, "-"+string(flagChar))
|
|
|
|
}
|
|
|
|
|
|
|
|
return separated
|
|
|
|
}
|
|
|
|
|
|
|
|
func isSplittable(flagArg string) bool {
|
|
|
|
return strings.HasPrefix(flagArg, "-") && !strings.HasPrefix(flagArg, "--") && len(flagArg) > 2
|
|
|
|
}
|