Refactor flags mechanisms
I think It's more obvious now to understand the inheritance of flags. Fix #403 Fix #404 Performance improvements: benchmark old ns/op new ns/op delta BenchmarkInheritedFlags-4 6536 5595 -14.40% BenchmarkLocalFlags-4 3193 3235 +1.32% benchmark old allocs new allocs delta BenchmarkInheritedFlags-4 49 39 -20.41% BenchmarkLocalFlags-4 23 21 -8.70% benchmark old bytes new bytes delta BenchmarkInheritedFlags-4 2040 1000 -50.98% BenchmarkLocalFlags-4 1008 544 -46.03%
This commit is contained in:
parent
a3cd8ab85a
commit
6202b5942b
@ -3,7 +3,6 @@ package cobra
|
|||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"fmt"
|
"fmt"
|
||||||
"os"
|
|
||||||
"reflect"
|
"reflect"
|
||||||
"runtime"
|
"runtime"
|
||||||
"strings"
|
"strings"
|
||||||
@ -13,9 +12,6 @@ import (
|
|||||||
"github.com/spf13/pflag"
|
"github.com/spf13/pflag"
|
||||||
)
|
)
|
||||||
|
|
||||||
var _ = fmt.Println
|
|
||||||
var _ = os.Stderr
|
|
||||||
|
|
||||||
var tp, te, tt, t1, tr []string
|
var tp, te, tt, t1, tr []string
|
||||||
var rootPersPre, echoPre, echoPersPre, timesPersPre []string
|
var rootPersPre, echoPre, echoPersPre, timesPersPre []string
|
||||||
var flagb1, flagb2, flagb3, flagbr, flagbp bool
|
var flagb1, flagb2, flagb3, flagbr, flagbp bool
|
||||||
@ -166,20 +162,22 @@ func flagInit() {
|
|||||||
cmdRootWithRun.ResetFlags()
|
cmdRootWithRun.ResetFlags()
|
||||||
cmdSubNoRun.ResetFlags()
|
cmdSubNoRun.ResetFlags()
|
||||||
cmdCustomFlags.ResetFlags()
|
cmdCustomFlags.ResetFlags()
|
||||||
cmdRootNoRun.PersistentFlags().StringVarP(&flags2a, "strtwo", "t", "two", strtwoParentHelp)
|
|
||||||
cmdEcho.Flags().IntVarP(&flagi1, "intone", "i", 123, "help message for flag intone")
|
|
||||||
cmdTimes.Flags().IntVarP(&flagi2, "inttwo", "j", 234, "help message for flag inttwo")
|
|
||||||
cmdPrint.Flags().IntVarP(&flagi3, "intthree", "i", 345, "help message for flag intthree")
|
|
||||||
cmdCustomFlags.Flags().IntVar(&flagi4, "intfour", 456, "help message for flag intfour")
|
|
||||||
cmdEcho.PersistentFlags().StringVarP(&flags1, "strone", "s", "one", "help message for flag strone")
|
|
||||||
cmdEcho.PersistentFlags().BoolVarP(&flagbp, "persistentbool", "p", false, "help message for flag persistentbool")
|
|
||||||
cmdTimes.PersistentFlags().StringVarP(&flags2b, "strtwo", "t", "2", strtwoChildHelp)
|
|
||||||
cmdPrint.PersistentFlags().StringVarP(&flags3, "strthree", "s", "three", "help message for flag strthree")
|
|
||||||
cmdEcho.Flags().BoolVarP(&flagb1, "boolone", "b", true, "help message for flag boolone")
|
|
||||||
cmdTimes.Flags().BoolVarP(&flagb2, "booltwo", "c", false, "help message for flag booltwo")
|
|
||||||
cmdPrint.Flags().BoolVarP(&flagb3, "boolthree", "b", true, "help message for flag boolthree")
|
|
||||||
cmdVersion1.ResetFlags()
|
cmdVersion1.ResetFlags()
|
||||||
cmdVersion2.ResetFlags()
|
cmdVersion2.ResetFlags()
|
||||||
|
|
||||||
|
cmdRootNoRun.PersistentFlags().StringVarP(&flags2a, "strtwo", "t", "two", strtwoParentHelp)
|
||||||
|
cmdCustomFlags.Flags().IntVar(&flagi4, "intfour", 456, "help message for flag intfour")
|
||||||
|
cmdEcho.Flags().BoolVarP(&flagb1, "boolone", "b", true, "help message for flag boolone")
|
||||||
|
cmdEcho.Flags().IntVarP(&flagi1, "intone", "i", 123, "help message for flag intone")
|
||||||
|
cmdEcho.PersistentFlags().BoolVarP(&flagbp, "persistentbool", "p", false, "help message for flag persistentbool")
|
||||||
|
cmdEcho.PersistentFlags().StringVarP(&flags1, "strone", "s", "one", "help message for flag strone")
|
||||||
|
cmdPrint.Flags().IntVarP(&flagi3, "intthree", "i", 345, "help message for flag intthree")
|
||||||
|
cmdTimes.Flags().BoolVarP(&flagb2, "booltwo", "c", false, "help message for flag booltwo")
|
||||||
|
cmdTimes.Flags().IntVarP(&flagi2, "inttwo", "j", 234, "help message for flag inttwo")
|
||||||
|
cmdTimes.Flags().StringVarP(&flags2b, "strtwo", "t", "2", strtwoChildHelp)
|
||||||
|
cmdTimes.PersistentFlags().StringVarP(&flags2b, "strtwo", "t", "2", strtwoChildHelp)
|
||||||
|
cmdPrint.Flags().BoolVarP(&flagb3, "boolthree", "b", true, "help message for flag boolthree")
|
||||||
|
cmdPrint.PersistentFlags().StringVarP(&flags3, "strthree", "s", "three", "help message for flag strthree")
|
||||||
}
|
}
|
||||||
|
|
||||||
func commandInit() {
|
func commandInit() {
|
||||||
@ -858,7 +856,6 @@ func TestFlagAccess(t *testing.T) {
|
|||||||
}
|
}
|
||||||
if inherited.Lookup("strtwo") != nil {
|
if inherited.Lookup("strtwo") != nil {
|
||||||
t.Errorf("InheritedFlags shouldn not contain overwritten flag strtwo")
|
t.Errorf("InheritedFlags shouldn not contain overwritten flag strtwo")
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1150,14 +1147,6 @@ func TestGlobalNormFuncPropagation(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestFlagOnPflagCommandLine(t *testing.T) {
|
|
||||||
flagName := "flagOnCommandLine"
|
|
||||||
pflag.CommandLine.String(flagName, "", "about my flag")
|
|
||||||
r := fullSetupTest("--help")
|
|
||||||
|
|
||||||
checkResultContains(t, r, flagName)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestAddTemplateFunctions(t *testing.T) {
|
func TestAddTemplateFunctions(t *testing.T) {
|
||||||
AddTemplateFunc("t", func() bool { return true })
|
AddTemplateFunc("t", func() bool { return true })
|
||||||
AddTemplateFuncs(template.FuncMap{
|
AddTemplateFuncs(template.FuncMap{
|
||||||
@ -1185,3 +1174,23 @@ func TestUsageIsNotPrintedTwice(t *testing.T) {
|
|||||||
t.Error("Usage output is not printed exactly once")
|
t.Error("Usage output is not printed exactly once")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func BenchmarkInheritedFlags(b *testing.B) {
|
||||||
|
initialize()
|
||||||
|
cmdEcho.AddCommand(cmdTimes)
|
||||||
|
b.ResetTimer()
|
||||||
|
|
||||||
|
for i := 0; i < b.N; i++ {
|
||||||
|
cmdTimes.InheritedFlags()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func BenchmarkLocalFlags(b *testing.B) {
|
||||||
|
initialize()
|
||||||
|
cmdEcho.AddCommand(cmdTimes)
|
||||||
|
b.ResetTimer()
|
||||||
|
|
||||||
|
for i := 0; i < b.N; i++ {
|
||||||
|
cmdTimes.LocalFlags()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
144
command.go
144
command.go
@ -66,6 +66,10 @@ type Command struct {
|
|||||||
pflags *flag.FlagSet
|
pflags *flag.FlagSet
|
||||||
// Flags that are declared specifically by this command (not inherited).
|
// Flags that are declared specifically by this command (not inherited).
|
||||||
lflags *flag.FlagSet
|
lflags *flag.FlagSet
|
||||||
|
// Inherited flags.
|
||||||
|
iflags *flag.FlagSet
|
||||||
|
// All persistent flags of cmd's parents.
|
||||||
|
parentsPflags *flag.FlagSet
|
||||||
// SilenceErrors is an option to quiet errors down stream
|
// SilenceErrors is an option to quiet errors down stream
|
||||||
SilenceErrors bool
|
SilenceErrors bool
|
||||||
// Silence Usage is an option to silence usage when an error occurs.
|
// Silence Usage is an option to silence usage when an error occurs.
|
||||||
@ -544,32 +548,18 @@ func (c *Command) SuggestionsFor(typedName string) []string {
|
|||||||
|
|
||||||
// VisitParents visits all parents of the command and invokes fn on each parent.
|
// VisitParents visits all parents of the command and invokes fn on each parent.
|
||||||
func (c *Command) VisitParents(fn func(*Command)) {
|
func (c *Command) VisitParents(fn func(*Command)) {
|
||||||
var traverse func(*Command) *Command
|
if c.HasParent() {
|
||||||
|
fn(c.Parent())
|
||||||
traverse = func(x *Command) *Command {
|
c.Parent().VisitParents(fn)
|
||||||
if x != c {
|
|
||||||
fn(x)
|
|
||||||
}
|
|
||||||
if x.HasParent() {
|
|
||||||
return traverse(x.parent)
|
|
||||||
}
|
|
||||||
return x
|
|
||||||
}
|
}
|
||||||
traverse(c)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Root finds root command.
|
// Root finds root command.
|
||||||
func (c *Command) Root() *Command {
|
func (c *Command) Root() *Command {
|
||||||
var findRoot func(*Command) *Command
|
if c.HasParent() {
|
||||||
|
return c.Parent().Root()
|
||||||
findRoot = func(x *Command) *Command {
|
|
||||||
if x.HasParent() {
|
|
||||||
return findRoot(x.parent)
|
|
||||||
}
|
|
||||||
return x
|
|
||||||
}
|
}
|
||||||
|
return c
|
||||||
return findRoot(c)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// ArgsLenAtDash will return the length of f.Args at the moment when a -- was
|
// ArgsLenAtDash will return the length of f.Args at the moment when a -- was
|
||||||
@ -913,12 +903,8 @@ func (c *Command) DebugFlags() {
|
|||||||
}
|
}
|
||||||
if x.HasFlags() {
|
if x.HasFlags() {
|
||||||
x.flags.VisitAll(func(f *flag.Flag) {
|
x.flags.VisitAll(func(f *flag.Flag) {
|
||||||
if x.HasPersistentFlags() {
|
if x.HasPersistentFlags() && x.persistentFlag(f.Name) != nil {
|
||||||
if x.persistentFlag(f.Name) == nil {
|
c.Println(" -"+f.Shorthand+",", "--"+f.Name, "["+f.DefValue+"]", "", f.Value, " [LP]")
|
||||||
c.Println(" -"+f.Shorthand+",", "--"+f.Name, "["+f.DefValue+"]", "", f.Value, " [L]")
|
|
||||||
} else {
|
|
||||||
c.Println(" -"+f.Shorthand+",", "--"+f.Name, "["+f.DefValue+"]", "", f.Value, " [LP]")
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
c.Println(" -"+f.Shorthand+",", "--"+f.Name, "["+f.DefValue+"]", "", f.Value, " [L]")
|
c.Println(" -"+f.Shorthand+",", "--"+f.Name, "["+f.DefValue+"]", "", f.Value, " [L]")
|
||||||
}
|
}
|
||||||
@ -1077,6 +1063,7 @@ func (c *Command) Flags() *flag.FlagSet {
|
|||||||
c.flags = flag.NewFlagSet(c.Name(), flag.ContinueOnError)
|
c.flags = flag.NewFlagSet(c.Name(), flag.ContinueOnError)
|
||||||
c.flags.SetOutput(c.OutOrStderr())
|
c.flags.SetOutput(c.OutOrStderr())
|
||||||
}
|
}
|
||||||
|
|
||||||
return c.flags
|
return c.flags
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1096,48 +1083,44 @@ func (c *Command) LocalNonPersistentFlags() *flag.FlagSet {
|
|||||||
// LocalFlags returns the local FlagSet specifically set in the current command.
|
// LocalFlags returns the local FlagSet specifically set in the current command.
|
||||||
func (c *Command) LocalFlags() *flag.FlagSet {
|
func (c *Command) LocalFlags() *flag.FlagSet {
|
||||||
c.mergePersistentFlags()
|
c.mergePersistentFlags()
|
||||||
|
c.updateParentsPersistentFlags()
|
||||||
|
|
||||||
local := flag.NewFlagSet(c.Name(), flag.ContinueOnError)
|
if c.lflags == nil {
|
||||||
c.lflags.VisitAll(func(f *flag.Flag) {
|
c.lflags = flag.NewFlagSet(c.Name(), flag.ContinueOnError)
|
||||||
local.AddFlag(f)
|
c.lflags.SetOutput(c.OutOrStderr())
|
||||||
})
|
|
||||||
if !c.HasParent() {
|
|
||||||
flag.CommandLine.VisitAll(func(f *flag.Flag) {
|
|
||||||
if local.Lookup(f.Name) == nil {
|
|
||||||
local.AddFlag(f)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
return local
|
|
||||||
|
flags := c.Flags()
|
||||||
|
c.lflags.SortFlags = flags.SortFlags
|
||||||
|
|
||||||
|
addToLocal := func(f *flag.Flag) {
|
||||||
|
if c.lflags.Lookup(f.Name) == nil && c.parentsPflags.Lookup(f.Name) == nil {
|
||||||
|
c.lflags.AddFlag(f)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
c.flags.VisitAll(addToLocal)
|
||||||
|
c.PersistentFlags().VisitAll(addToLocal)
|
||||||
|
return c.lflags
|
||||||
}
|
}
|
||||||
|
|
||||||
// InheritedFlags returns all flags which were inherited from parents commands.
|
// InheritedFlags returns all flags which were inherited from parents commands.
|
||||||
func (c *Command) InheritedFlags() *flag.FlagSet {
|
func (c *Command) InheritedFlags() *flag.FlagSet {
|
||||||
c.mergePersistentFlags()
|
c.mergePersistentFlags()
|
||||||
|
c.updateParentsPersistentFlags()
|
||||||
|
|
||||||
|
if c.iflags == nil {
|
||||||
|
c.iflags = flag.NewFlagSet(c.Name(), flag.ContinueOnError)
|
||||||
|
}
|
||||||
|
|
||||||
inherited := flag.NewFlagSet(c.Name(), flag.ContinueOnError)
|
|
||||||
local := c.LocalFlags()
|
local := c.LocalFlags()
|
||||||
|
|
||||||
var rmerge func(x *Command)
|
c.parentsPflags.VisitAll(func(f *flag.Flag) {
|
||||||
|
if c.iflags.Lookup(f.Name) == nil && local.Lookup(f.Name) == nil {
|
||||||
rmerge = func(x *Command) {
|
c.iflags.AddFlag(f)
|
||||||
if x.HasPersistentFlags() {
|
|
||||||
x.PersistentFlags().VisitAll(func(f *flag.Flag) {
|
|
||||||
if inherited.Lookup(f.Name) == nil && local.Lookup(f.Name) == nil {
|
|
||||||
inherited.AddFlag(f)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
if x.HasParent() {
|
})
|
||||||
rmerge(x.parent)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if c.HasParent() {
|
return c.iflags
|
||||||
rmerge(c.parent)
|
|
||||||
}
|
|
||||||
|
|
||||||
return inherited
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// NonInheritedFlags returns all flags which were not inherited from parent commands.
|
// NonInheritedFlags returns all flags which were not inherited from parent commands.
|
||||||
@ -1243,38 +1226,35 @@ func (c *Command) Parent() *Command {
|
|||||||
return c.parent
|
return c.parent
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Command) mergePersistentFlags() {
|
func (c *Command) updateParentsPersistentFlags() {
|
||||||
var rmerge func(x *Command)
|
if c.parentsPflags == nil {
|
||||||
|
c.parentsPflags = flag.NewFlagSet(c.Name(), flag.ContinueOnError)
|
||||||
// Save the set of local flags
|
c.parentsPflags.SetOutput(c.OutOrStderr())
|
||||||
if c.lflags == nil {
|
|
||||||
c.lflags = flag.NewFlagSet(c.Name(), flag.ContinueOnError)
|
|
||||||
c.lflags.SetOutput(c.OutOrStderr())
|
|
||||||
addtolocal := func(f *flag.Flag) {
|
|
||||||
c.lflags.AddFlag(f)
|
|
||||||
}
|
|
||||||
c.Flags().VisitAll(addtolocal)
|
|
||||||
c.PersistentFlags().VisitAll(addtolocal)
|
|
||||||
}
|
}
|
||||||
rmerge = func(x *Command) {
|
|
||||||
if !x.HasParent() {
|
c.VisitParents(func(x *Command) {
|
||||||
flag.CommandLine.VisitAll(func(f *flag.Flag) {
|
|
||||||
if x.PersistentFlags().Lookup(f.Name) == nil {
|
|
||||||
x.PersistentFlags().AddFlag(f)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
if x.HasPersistentFlags() {
|
if x.HasPersistentFlags() {
|
||||||
x.PersistentFlags().VisitAll(func(f *flag.Flag) {
|
x.PersistentFlags().VisitAll(func(f *flag.Flag) {
|
||||||
if c.Flags().Lookup(f.Name) == nil {
|
if c.parentsPflags.Lookup(f.Name) == nil {
|
||||||
c.Flags().AddFlag(f)
|
c.parentsPflags.AddFlag(f)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
if x.HasParent() {
|
})
|
||||||
rmerge(x.parent)
|
}
|
||||||
|
|
||||||
|
func (c *Command) mergePersistentFlags() {
|
||||||
|
flags := c.Flags()
|
||||||
|
|
||||||
|
merge := func(x *Command) {
|
||||||
|
if x.HasPersistentFlags() {
|
||||||
|
x.PersistentFlags().VisitAll(func(f *flag.Flag) {
|
||||||
|
if flags.Lookup(f.Name) == nil {
|
||||||
|
flags.AddFlag(f)
|
||||||
|
}
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
merge(c)
|
||||||
rmerge(c)
|
c.VisitParents(merge)
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user