Skip to content

Commit

Permalink
fix: show flags that shadow parent persistent flag in child help
Browse files Browse the repository at this point in the history
This fixes a bug where a child flag that shadows (has the same
name as) a parent persistent flag would not be shown in the
child command's help output and the parent flag would be shown
instead under the global flags section.

This change makes the help output consistent with the
observed behavior during execution, where the child flag is
the one that is actually used.

Fixes spf13/cobra#1651

Merge spf13/cobra#1776
  • Loading branch information
brianpursley authored and hoshsadiq committed Aug 29, 2022
1 parent 805db67 commit 0d45bd8
Show file tree
Hide file tree
Showing 2 changed files with 36 additions and 6 deletions.
3 changes: 2 additions & 1 deletion command.go
Original file line number Diff line number Diff line change
Expand Up @@ -1679,7 +1679,8 @@ func (c *Command) LocalFlags() *zflag.FlagSet {
}

addToLocal := func(f *zflag.Flag) {
if c.lflags.Lookup(f.Name) == nil && c.parentsPflags.Lookup(f.Name) == nil {
// Add the flag if it is not a parent PFlag, or it shadows a parent PFlag
if c.lflags.Lookup(f.Name) == nil && f != c.parentsPflags.Lookup(f.Name) {
c.lflags.AddFlag(f)
}
}
Expand Down
39 changes: 34 additions & 5 deletions command_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -712,10 +712,7 @@ func TestEmptyInputs(t *testing.T) {
}
}

func TestOverwrittenFlag(t *testing.T) {
// TODO: This test fails, but should work.
t.Skip()

func TestChildFlagShadowsParentPersistentFlag(t *testing.T) {
parent := &zulu.Command{Use: "parent", RunE: emptyRun}
child := &zulu.Command{Use: "child", RunE: emptyRun}

Expand All @@ -737,7 +734,7 @@ func TestOverwrittenFlag(t *testing.T) {
}

if childInherited.Lookup("intf") != nil {
t.Errorf(`InheritedFlags should not contain overwritten flag "intf"`)
t.Errorf(`InheritedFlags should not contain shadowed flag "intf"`)
}
if childLocal.Lookup("intf") == nil {
t.Error(`LocalFlags expected to contain "intf", got "nil"`)
Expand Down Expand Up @@ -905,6 +902,38 @@ func TestHelpCommandExecutedOnChild(t *testing.T) {
checkStringContains(t, output, childCmd.Long)
}

func TestHelpCommandExecutedOnChildWithFlagThatShadowsParentFlag(t *testing.T) {
parent := &zulu.Command{Use: "parent", RunE: emptyRun}
child := &zulu.Command{Use: "child", RunE: emptyRun}
parent.AddCommand(child)

parent.PersistentFlags().Bool("foo", false, "parent foo usage")
parent.PersistentFlags().Bool("bar", false, "parent bar usage")
child.Flags().Bool("foo", false, "child foo usage") // This shadows parent's foo flag
child.Flags().Bool("baz", false, "child baz usage")

got, err := executeCommand(parent, "help", "child")
if err != nil {
t.Errorf("Unexpected error: %v", err)
}

expected := `Usage:
parent child [flags]
Flags:
--baz child baz usage
--foo child foo usage
-h, --help help for child
Global Flags:
--bar parent bar usage
`

if got != expected {
t.Errorf("Help text mismatch.\nExpected:\n%s\n\nGot:\n%s\n", expected, got)
}
}

func TestSetHelpCommand(t *testing.T) {
c := &zulu.Command{Use: "c", RunE: emptyRun}
c.AddCommand(&zulu.Command{Use: "empty", RunE: emptyRun})
Expand Down

0 comments on commit 0d45bd8

Please sign in to comment.