From 002bde2233182425d5f6d7f3d2456db482d2bb1f Mon Sep 17 00:00:00 2001 From: Sascha Grunert Date: Fri, 29 Nov 2019 14:09:46 +0100 Subject: [PATCH 1/7] Add suggestions support The new option `app.Suggest` enables command and flag suggestions via the jaro-winkler distance algorithm. Flags are scoped to their appropriate commands whereas command suggestions are scoped to the current command level. Signed-off-by: Sascha Grunert --- app.go | 12 +++++ command.go | 6 +++ docs/v2/manual.md | 8 +++ go.mod | 1 + go.sum | 2 + help.go | 21 ++++++-- parse.go | 18 +++++-- suggestions.go | 75 +++++++++++++++++++++++++++ suggestions_test.go | 122 ++++++++++++++++++++++++++++++++++++++++++++ 9 files changed, 257 insertions(+), 8 deletions(-) create mode 100644 suggestions.go create mode 100644 suggestions_test.go diff --git a/app.go b/app.go index dd8f1de..f3d24d0 100644 --- a/app.go +++ b/app.go @@ -88,6 +88,8 @@ type App struct { // single-character bool arguments into one // i.e. foobar -o -v -> foobar -ov UseShortOptionHandling bool + // Enable suggestions for commands and flags + Suggest bool didSetup bool } @@ -250,6 +252,11 @@ func (a *App) RunContext(ctx context.Context, arguments []string) (err error) { return err } _, _ = fmt.Fprintf(a.Writer, "%s %s\n\n", "Incorrect Usage.", err.Error()) + if a.Suggest { + if suggestion, err := a.suggestFlagFromError(err, ""); err == nil { + fmt.Fprintf(a.Writer, suggestion) + } + } _ = ShowAppHelp(context) return err } @@ -381,6 +388,11 @@ func (a *App) RunAsSubcommand(ctx *Context) (err error) { return err } _, _ = fmt.Fprintf(a.Writer, "%s %s\n\n", "Incorrect Usage.", err.Error()) + if a.Suggest { + if suggestion, err := a.suggestFlagFromError(err, context.Command.Name); err == nil { + fmt.Fprintf(a.Writer, suggestion) + } + } _ = ShowSubcommandHelp(context) return err } diff --git a/command.go b/command.go index db6c802..5300ee8 100644 --- a/command.go +++ b/command.go @@ -116,6 +116,11 @@ func (c *Command) Run(ctx *Context) (err error) { } _, _ = fmt.Fprintln(context.App.Writer, "Incorrect Usage:", err.Error()) _, _ = fmt.Fprintln(context.App.Writer) + if ctx.App.Suggest { + if suggestion, err := ctx.App.suggestFlagFromError(err, c.Name); err == nil { + fmt.Fprintf(context.App.Writer, suggestion) + } + } _ = ShowCommandHelp(context, c.Name) return err } @@ -244,6 +249,7 @@ func (c *Command) startApp(ctx *Context) error { app.ErrWriter = ctx.App.ErrWriter app.ExitErrHandler = ctx.App.ExitErrHandler app.UseShortOptionHandling = ctx.App.UseShortOptionHandling + app.Suggest = ctx.App.Suggest app.categories = newCommandCategories() for _, command := range c.Subcommands { diff --git a/docs/v2/manual.md b/docs/v2/manual.md index ce61075..29c2638 100644 --- a/docs/v2/manual.md +++ b/docs/v2/manual.md @@ -34,6 +34,7 @@ cli v2 manual * [Version Flag](#version-flag) + [Customization](#customization-2) * [Timestamp Flag](#timestamp-flag) + * [Suggestions](#suggestions) * [Full API Example](#full-api-example) @@ -1426,6 +1427,13 @@ In this example the flag could be used like this : Side note: quotes may be necessary around the date depending on your layout (if you have spaces for instance) +### Suggestions + +To enable flag and command suggestions, set `app.Suggest = true`. If the suggest +feature is enabled, then the help output of the corresponding command will +provide an appropriate suggestion for the provided flag or subcommand if +available. + ### Full API Example **Notice**: This is a contrived (functioning) example meant strictly for API diff --git a/go.mod b/go.mod index c38d41c..1c280a6 100644 --- a/go.mod +++ b/go.mod @@ -4,6 +4,7 @@ go 1.11 require ( github.com/BurntSushi/toml v0.3.1 + github.com/antzucaro/matchr v0.0.0-20180616170659-cbc221335f3c github.com/cpuguy83/go-md2man/v2 v2.0.0-20190314233015-f79a8a8ca69d gopkg.in/yaml.v2 v2.2.2 ) diff --git a/go.sum b/go.sum index ef121ff..6165e94 100644 --- a/go.sum +++ b/go.sum @@ -1,5 +1,7 @@ github.com/BurntSushi/toml v0.3.1 h1:WXkYYl6Yr3qBf1K79EBnL4mak0OimBfB0XUf9Vl28OQ= github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= +github.com/antzucaro/matchr v0.0.0-20180616170659-cbc221335f3c h1:CucViv7orgFBMkehuFFdkCVF5ERovbkRRyhvaYaHu/k= +github.com/antzucaro/matchr v0.0.0-20180616170659-cbc221335f3c/go.mod h1:bV/CkX4+ANGDaBwbHkt9kK287al/i9BsB18PRBvyqYo= github.com/cpuguy83/go-md2man/v2 v2.0.0-20190314233015-f79a8a8ca69d h1:U+s90UTSYgptZMwQh2aRr3LuazLJIa+Pg3Kc1ylSYVY= github.com/cpuguy83/go-md2man/v2 v2.0.0-20190314233015-f79a8a8ca69d/go.mod h1:maD7wRr/U5Z6m/iR4s+kqSMx2CaBsrgA7czyZG/E6dU= github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= diff --git a/help.go b/help.go index c1e974a..9162661 100644 --- a/help.go +++ b/help.go @@ -10,9 +10,14 @@ import ( "unicode/utf8" ) +const ( + helpName = "help" + helpAlias = "h" +) + var helpCommand = &Command{ - Name: "help", - Aliases: []string{"h"}, + Name: helpName, + Aliases: []string{helpAlias}, Usage: "Shows a list of commands or help for one command", ArgsUsage: "[command]", Action: func(c *Context) error { @@ -27,8 +32,8 @@ var helpCommand = &Command{ } var helpSubcommand = &Command{ - Name: "help", - Aliases: []string{"h"}, + Name: helpName, + Aliases: []string{helpAlias}, Usage: "Shows a list of commands or help for one command", ArgsUsage: "[command]", Action: func(c *Context) error { @@ -207,7 +212,13 @@ func ShowCommandHelp(ctx *Context, command string) error { } if ctx.App.CommandNotFound == nil { - return Exit(fmt.Sprintf("No help topic for '%v'", command), 3) + errMsg := fmt.Sprintf("No help topic for '%v'", command) + if ctx.App.Suggest { + if suggestion := suggestCommand(ctx.App.Commands, command); suggestion != "" { + errMsg += ". " + suggestion + } + } + return Exit(errMsg, 3) } ctx.App.CommandNotFound(ctx, command) diff --git a/parse.go b/parse.go index 7df1729..a2db306 100644 --- a/parse.go +++ b/parse.go @@ -26,9 +26,8 @@ func parseIter(set *flag.FlagSet, ip iterativeParser, args []string, shellComple return err } - errStr := err.Error() - trimmed := strings.TrimPrefix(errStr, "flag provided but not defined: -") - if errStr == trimmed { + trimmed, trimErr := flagFromError(err) + if trimErr != nil { return err } @@ -67,6 +66,19 @@ func parseIter(set *flag.FlagSet, ip iterativeParser, args []string, shellComple } } +const providedButNotDefinedErrMsg = "flag provided but not defined: -" + +// flagFromError tries to parse a provided flag from an error message. If the +// parsing fials, it returns the input error and an empty string +func flagFromError(err error) (string, error) { + errStr := err.Error() + trimmed := strings.TrimPrefix(errStr, providedButNotDefinedErrMsg) + if errStr == trimmed { + return "", err + } + return trimmed, nil +} + func splitShortOptions(set *flag.FlagSet, arg string) []string { shortFlagsExist := func(s string) bool { for _, c := range s[1:] { diff --git a/suggestions.go b/suggestions.go new file mode 100644 index 0000000..476af4d --- /dev/null +++ b/suggestions.go @@ -0,0 +1,75 @@ +package cli + +import ( + "fmt" + + "github.com/antzucaro/matchr" +) + +const didYouMeanTemplate = "Did you mean '%s'?" + +func (a *App) suggestFlagFromError(err error, command string) (string, error) { + flag, parseErr := flagFromError(err) + if parseErr != nil { + return "", err + } + + flags := a.Flags + if command != "" { + cmd := a.Command(command) + if cmd == nil { + return "", err + } + flags = cmd.Flags + } + + suggestion := a.suggestFlag(flags, flag) + if len(suggestion) == 0 { + return "", err + } + + return fmt.Sprintf(didYouMeanTemplate+"\n\n", suggestion), nil +} + +func (a *App) suggestFlag(flags []Flag, provided string) (suggestion string) { + distance := 0.0 + + for _, flag := range flags { + flagNames := flag.Names() + if !a.HideHelp { + flagNames = append(flagNames, HelpFlag.Names()...) + } + for _, name := range flagNames { + newDistance := matchr.JaroWinkler(name, provided, true) + if newDistance > distance { + distance = newDistance + suggestion = name + } + } + } + + if len(suggestion) == 1 { + suggestion = "-" + suggestion + } else if len(suggestion) > 1 { + suggestion = "--" + suggestion + } + + return suggestion +} + +// suggestCommand takes a list of commands and a provided string to suggest a +// command name +func suggestCommand(commands []*Command, provided string) (suggestion string) { + distance := 0.0 + for _, command := range commands { + for _, name := range append(command.Names(), helpName, helpAlias) { + newDistance := matchr.JaroWinkler(name, provided, true) + if newDistance > distance { + distance = newDistance + suggestion = name + } + } + } + + return fmt.Sprintf(didYouMeanTemplate, suggestion) +} diff --git a/suggestions_test.go b/suggestions_test.go new file mode 100644 index 0000000..3b31250 --- /dev/null +++ b/suggestions_test.go @@ -0,0 +1,122 @@ +package cli + +import ( + "errors" + "fmt" + "testing" +) + +func TestSuggestFlag(t *testing.T) { + // Given + app := testApp() + + for _, testCase := range []struct { + provided, expected string + }{ + {"", ""}, + {"a", "--another-flag"}, + {"hlp", "--help"}, + {"k", ""}, + {"s", "-s"}, + } { + // When + res := app.suggestFlag(app.Flags, testCase.provided) + + // Then + expect(t, res, testCase.expected) + } +} + +func TestSuggestFlagHideHelp(t *testing.T) { + // Given + app := testApp() + app.HideHelp = true + + // When + res := app.suggestFlag(app.Flags, "hlp") + + // Then + expect(t, res, "--fl") +} + +func TestSuggestFlagFromError(t *testing.T) { + // Given + app := testApp() + + for _, testCase := range []struct { + command, provided, expected string + }{ + {"", "hel", "--help"}, + {"", "soccer", "--socket"}, + {"config", "anot", "--another-flag"}, + } { + // When + res, _ := app.suggestFlagFromError( + errors.New(providedButNotDefinedErrMsg+testCase.provided), + testCase.command, + ) + + // Then + expect(t, res, fmt.Sprintf(didYouMeanTemplate+"\n\n", testCase.expected)) + } +} + +func TestSuggestFlagFromErrorWrongError(t *testing.T) { + // Given + app := testApp() + + // When + _, err := app.suggestFlagFromError(errors.New("invalid"), "") + + // Then + expect(t, true, err != nil) +} + +func TestSuggestFlagFromErrorWrongCommand(t *testing.T) { + // Given + app := testApp() + + // When + _, err := app.suggestFlagFromError( + errors.New(providedButNotDefinedErrMsg+"flag"), + "invalid", + ) + + // Then + expect(t, true, err != nil) +} + +func TestSuggestFlagFromErrorNoSuggestion(t *testing.T) { + // Given + app := testApp() + + // When + _, err := app.suggestFlagFromError( + errors.New(providedButNotDefinedErrMsg+""), + "", + ) + + // Then + expect(t, true, err != nil) +} + +func TestSuggestCommand(t *testing.T) { + // Given + app := testApp() + + for _, testCase := range []struct { + provided, expected string + }{ + {"", ""}, + {"conf", "config"}, + {"i", "i"}, + {"information", "info"}, + {"not-existing", "info"}, + } { + // When + res := suggestCommand(app.Commands, testCase.provided) + + // Then + expect(t, res, fmt.Sprintf(didYouMeanTemplate, testCase.expected)) + } +} From 63b1a7deee832fd3b2d5e235fe22b0699c5c9163 Mon Sep 17 00:00:00 2001 From: Dan Buch Date: Sat, 7 May 2022 14:43:15 -0400 Subject: [PATCH 2/7] A few follow-up conflict resolutions --- app.go | 2 +- command.go | 2 +- godoc-current.txt | 2 ++ testdata/godoc-v2.x.txt | 2 ++ 4 files changed, 6 insertions(+), 2 deletions(-) diff --git a/app.go b/app.go index 5edfa46..463437a 100644 --- a/app.go +++ b/app.go @@ -391,7 +391,7 @@ func (a *App) RunAsSubcommand(ctx *Context) (err error) { } _, _ = fmt.Fprintf(a.Writer, "%s %s\n\n", "Incorrect Usage.", err.Error()) if a.Suggest { - if suggestion, err := a.suggestFlagFromError(err, context.Command.Name); err == nil { + if suggestion, err := a.suggestFlagFromError(err, cCtx.Command.Name); err == nil { fmt.Fprintf(a.Writer, suggestion) } } diff --git a/command.go b/command.go index 16ac7c3..3b9b837 100644 --- a/command.go +++ b/command.go @@ -121,7 +121,7 @@ func (c *Command) Run(ctx *Context) (err error) { _, _ = fmt.Fprintln(cCtx.App.Writer) if ctx.App.Suggest { if suggestion, err := ctx.App.suggestFlagFromError(err, c.Name); err == nil { - fmt.Fprintf(context.App.Writer, suggestion) + fmt.Fprintf(cCtx.App.Writer, suggestion) } } _ = ShowCommandHelp(cCtx, c.Name) diff --git a/godoc-current.txt b/godoc-current.txt index 1bddc54..90700eb 100644 --- a/godoc-current.txt +++ b/godoc-current.txt @@ -305,6 +305,8 @@ type App struct { // single-character bool arguments into one // i.e. foobar -o -v -> foobar -ov UseShortOptionHandling bool + // Enable suggestions for commands and flags + Suggest bool // Has unexported fields. } diff --git a/testdata/godoc-v2.x.txt b/testdata/godoc-v2.x.txt index 1bddc54..90700eb 100644 --- a/testdata/godoc-v2.x.txt +++ b/testdata/godoc-v2.x.txt @@ -305,6 +305,8 @@ type App struct { // single-character bool arguments into one // i.e. foobar -o -v -> foobar -ov UseShortOptionHandling bool + // Enable suggestions for commands and flags + Suggest bool // Has unexported fields. } From 974f2d410de5e80cdf8b51850ec15934332620d5 Mon Sep 17 00:00:00 2001 From: Dan Buch Date: Sat, 7 May 2022 15:21:00 -0400 Subject: [PATCH 3/7] Guard suggestion capability (+ dependency) with build tag --- .github/workflows/cli.yml | 10 ++++++++-- Makefile | 4 ++-- README.md | 12 +++++++++++- docs.go | 4 ++-- docs_test.go | 4 ++-- go.mod | 2 +- go.sum | 10 ++++++---- internal/build/build.go | 2 +- suggestions.go | 3 +++ suggestions_stubs.go | 12 ++++++++++++ suggestions_test.go | 3 +++ 11 files changed, 51 insertions(+), 15 deletions(-) create mode 100644 suggestions_stubs.go diff --git a/.github/workflows/cli.yml b/.github/workflows/cli.yml index 84f9cae..cad0389 100644 --- a/.github/workflows/cli.yml +++ b/.github/workflows/cli.yml @@ -37,9 +37,15 @@ jobs: - name: vet run: go run internal/build/build.go vet - - name: test with tags + - name: test with urfave_cli_core tag + run: go run internal/build/build.go -tags urfave_cli_core test + + - name: test with urfave_cli_no_docs tag run: go run internal/build/build.go -tags urfave_cli_no_docs test + - name: test with urfave_cli_no_suggest tag + run: go run internal/build/build.go -tags urfave_cli_no_suggest test + - name: test run: go run internal/build/build.go test @@ -47,7 +53,7 @@ jobs: run: go run internal/build/build.go check-binary-size - name: check-binary-size with tags (informational only) - run: go run internal/build/build.go -tags urfave_cli_no_docs check-binary-size || true + run: go run internal/build/build.go -tags urfave_cli_core check-binary-size - name: Upload coverage to Codecov if: success() && matrix.go == '1.18.x' && matrix.os == 'ubuntu-latest' diff --git a/Makefile b/Makefile index 52e9204..9ca3c35 100644 --- a/Makefile +++ b/Makefile @@ -17,11 +17,11 @@ all: generate vet tag-test test check-binary-size tag-check-binary-size gfmrun t .PHONY: tag-test tag-test: - go run internal/build/build.go -tags urfave_cli_no_docs test + go run internal/build/build.go -tags urfave_cli_core test .PHONY: tag-check-binary-size tag-check-binary-size: - go run internal/build/build.go -tags urfave_cli_no_docs check-binary-size + go run internal/build/build.go -tags urfave_cli_core check-binary-size .PHONY: gfmrun gfmrun: diff --git a/README.md b/README.md index 6e4d698..35694d2 100644 --- a/README.md +++ b/README.md @@ -59,11 +59,21 @@ import ( You can use the following build tags: +#### `urfave_cli_core` + +When set, applies all `urfave_cli_no.+` build tags to minimize resulting binary +size. + #### `urfave_cli_no_docs` When set, this removes `ToMarkdown` and `ToMan` methods, so your application won't be able to call those. This reduces the resulting binary size by about -300-400 KB (measured using Go 1.18.1 on Linux/amd64), due to less dependencies. +300-400 KB (measured using Go 1.18.1 on Linux/amd64), due to fewer dependencies. + +#### `urfave_cli_no_suggest` + +When set, the capability enabled by setting `App.Suggest` will be a no-op. This +reduces the resulting binary size due to fewer dependencies. ### GOPATH diff --git a/docs.go b/docs.go index 8b1c9c8..4aba31b 100644 --- a/docs.go +++ b/docs.go @@ -1,5 +1,5 @@ -//go:build !urfave_cli_no_docs -// +build !urfave_cli_no_docs +//go:build !urfave_cli_no_docs && !urfave_cli_core +// +build !urfave_cli_no_docs,!urfave_cli_core package cli diff --git a/docs_test.go b/docs_test.go index 12d5d3c..bc3703f 100644 --- a/docs_test.go +++ b/docs_test.go @@ -1,5 +1,5 @@ -//go:build !urfave_cli_no_docs -// +build !urfave_cli_no_docs +//go:build !urfave_cli_no_docs && !urfave_cli_core +// +build !urfave_cli_no_docs,!urfave_cli_core package cli diff --git a/go.mod b/go.mod index 7042c4e..f9111b0 100644 --- a/go.mod +++ b/go.mod @@ -4,7 +4,7 @@ go 1.18 require ( github.com/BurntSushi/toml v1.1.0 - github.com/antzucaro/matchr v0.0.0-20180616170659-cbc221335f3c + github.com/antzucaro/matchr v0.0.0-20210222213004-b04723ef80f0 github.com/cpuguy83/go-md2man/v2 v2.0.1 golang.org/x/text v0.3.7 gopkg.in/yaml.v2 v2.4.0 diff --git a/go.sum b/go.sum index 3cb2b0b..afac06b 100644 --- a/go.sum +++ b/go.sum @@ -2,14 +2,16 @@ github.com/BurntSushi/toml v1.1.0 h1:ksErzDEI1khOiGPgpwuI7x2ebx/uXQNw7xJpn9Eq1+I github.com/BurntSushi/toml v1.1.0/go.mod h1:CxXYINrC8qIiEnFrOxCa7Jy5BFHlXnUU2pbicEuybxQ= github.com/antzucaro/matchr v0.0.0-20180616170659-cbc221335f3c h1:CucViv7orgFBMkehuFFdkCVF5ERovbkRRyhvaYaHu/k= github.com/antzucaro/matchr v0.0.0-20180616170659-cbc221335f3c/go.mod h1:bV/CkX4+ANGDaBwbHkt9kK287al/i9BsB18PRBvyqYo= -github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= -github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= -github.com/shurcooL/sanitized_anchor_name v1.0.0 h1:PdmoCO6wvbs+7yrJyMORt4/BmY5IYyJwS/kOiWx8mHo= -github.com/shurcooL/sanitized_anchor_name v1.0.0/go.mod h1:1NzhyTcUVG4SuEtjjoZeVRXNmyL/1OwPU0+IJeTBvfc= +github.com/antzucaro/matchr v0.0.0-20210222213004-b04723ef80f0 h1:R/qAiUxFT3mNgQaNqJe0IVznjKRNm23ohAIh9lgtlzc= +github.com/antzucaro/matchr v0.0.0-20210222213004-b04723ef80f0/go.mod h1:v3ZDlfVAL1OrkKHbGSFFK60k0/7hruHPDq2XMs9Gu6U= github.com/cpuguy83/go-md2man/v2 v2.0.1 h1:r/myEWzV9lfsM1tFLgDyu0atFtJ1fXn261LKYj/3DxU= github.com/cpuguy83/go-md2man/v2 v2.0.1/go.mod h1:tgQtvFlXSQOSOSIRvRPT7W67SCa46tRHOmNcaadrF8o= +github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= +github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= github.com/russross/blackfriday/v2 v2.1.0 h1:JIOH55/0cWyOuilr9/qlrm0BSXldqnqwMsf35Ld67mk= github.com/russross/blackfriday/v2 v2.1.0/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM= +github.com/shurcooL/sanitized_anchor_name v1.0.0 h1:PdmoCO6wvbs+7yrJyMORt4/BmY5IYyJwS/kOiWx8mHo= +github.com/shurcooL/sanitized_anchor_name v1.0.0/go.mod h1:1NzhyTcUVG4SuEtjjoZeVRXNmyL/1OwPU0+IJeTBvfc= golang.org/x/text v0.3.7 h1:olpwvP2KacW1ZWvsR7uQhoyTYvKAupfQrRGBFM352Gk= golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e h1:FDhOuMEY4JVRztM/gsbk+IKUQ8kj74bxZrgw87eMMVc= diff --git a/internal/build/build.go b/internal/build/build.go index 929ab0f..e74acfb 100644 --- a/internal/build/build.go +++ b/internal/build/build.go @@ -242,7 +242,7 @@ func checkBinarySizeActionFunc(c *cli.Context) (err error) { tags := c.String("tags") - if strings.Contains(tags, "urfave_cli_no_docs") { + if strings.Contains(tags, "urfave_cli_core") || strings.Contains(tags, "urfave_cli_no_docs") { desiredMinBinarySize = 1.39 } diff --git a/suggestions.go b/suggestions.go index 476af4d..301c598 100644 --- a/suggestions.go +++ b/suggestions.go @@ -1,3 +1,6 @@ +//go:build !urfave_cli_no_suggest && !urfave_cli_core +// +build !urfave_cli_no_suggest,!urfave_cli_core + package cli import ( diff --git a/suggestions_stubs.go b/suggestions_stubs.go new file mode 100644 index 0000000..48643bb --- /dev/null +++ b/suggestions_stubs.go @@ -0,0 +1,12 @@ +//go:build urfave_cli_no_suggest || urfave_cli_core +// +build urfave_cli_no_suggest urfave_cli_core + +package cli + +func (a *App) suggestFlagFromError(err error, _ string) (string, error) { + return "", err +} + +func suggestCommand([]*Command, string) string { + return "" +} diff --git a/suggestions_test.go b/suggestions_test.go index 3b31250..f87e702 100644 --- a/suggestions_test.go +++ b/suggestions_test.go @@ -1,3 +1,6 @@ +//go:build !urfave_cli_no_suggest && !urfave_cli_core +// +build !urfave_cli_no_suggest,!urfave_cli_core + package cli import ( From 9da2c564f8b1563222e3e60a1761de1d0a3c43c4 Mon Sep 17 00:00:00 2001 From: Dan Buch Date: Sat, 7 May 2022 20:57:44 -0400 Subject: [PATCH 4/7] Add example app with `Suggest` support --- suggestions_test.go | 33 +++++++++++++++++++++++++++++++++ 1 file changed, 33 insertions(+) diff --git a/suggestions_test.go b/suggestions_test.go index f87e702..55ef29e 100644 --- a/suggestions_test.go +++ b/suggestions_test.go @@ -123,3 +123,36 @@ func TestSuggestCommand(t *testing.T) { expect(t, res, fmt.Sprintf(didYouMeanTemplate, testCase.expected)) } } + +func ExampleApp_Suggest() { + app := &App{ + Name: "greet", + Suggest: true, + Flags: []Flag{ + &StringFlag{Name: "name", Value: "squirrel", Usage: "a name to say"}, + }, + Action: func(c *Context) error { + fmt.Printf("Hello %v\n", c.String("name")) + return nil + }, + } + + app.Run([]string{"greet", "--nema", "chipmunk"}) + // Output: + // Incorrect Usage. flag provided but not defined: -nema + // + // Did you mean '--name'? + // + // NAME: + // greet - A new cli application + // + // USAGE: + // greet [global options] command [command options] [arguments...] + // + // COMMANDS: + // help, h Shows a list of commands or help for one command + // + // GLOBAL OPTIONS: + // --name value a name to say (default: "squirrel") + // --help, -h show help (default: false) +} From 34eed95a8ee897b19e8abe77ddffcc7f7e39bdfb Mon Sep 17 00:00:00 2001 From: Dan Buch Date: Sat, 7 May 2022 22:06:39 -0400 Subject: [PATCH 5/7] Reduce amount of text compared in suggest example app --- suggestions_test.go | 12 +++++------- 1 file changed, 5 insertions(+), 7 deletions(-) diff --git a/suggestions_test.go b/suggestions_test.go index 55ef29e..359c793 100644 --- a/suggestions_test.go +++ b/suggestions_test.go @@ -126,8 +126,10 @@ func TestSuggestCommand(t *testing.T) { func ExampleApp_Suggest() { app := &App{ - Name: "greet", - Suggest: true, + Name: "greet", + Suggest: true, + HideHelp: true, + HideHelpCommand: true, Flags: []Flag{ &StringFlag{Name: "name", Value: "squirrel", Usage: "a name to say"}, }, @@ -147,12 +149,8 @@ func ExampleApp_Suggest() { // greet - A new cli application // // USAGE: - // greet [global options] command [command options] [arguments...] - // - // COMMANDS: - // help, h Shows a list of commands or help for one command + // greet [global options] [arguments...] // // GLOBAL OPTIONS: // --name value a name to say (default: "squirrel") - // --help, -h show help (default: false) } From 9bd6349ed21eb7338d28f2c58c3a17eef0895741 Mon Sep 17 00:00:00 2001 From: Dan Buch Date: Sat, 7 May 2022 23:06:43 -0400 Subject: [PATCH 6/7] Add example app for suggestion in command and further reduce example output diff potential --- suggestions_test.go | 63 +++++++++++++++++++++++++++++++++++---------- 1 file changed, 49 insertions(+), 14 deletions(-) diff --git a/suggestions_test.go b/suggestions_test.go index 359c793..fa7b0c9 100644 --- a/suggestions_test.go +++ b/suggestions_test.go @@ -126,15 +126,16 @@ func TestSuggestCommand(t *testing.T) { func ExampleApp_Suggest() { app := &App{ - Name: "greet", - Suggest: true, - HideHelp: true, - HideHelpCommand: true, + Name: "greet", + Suggest: true, + HideHelp: true, + HideHelpCommand: true, + CustomAppHelpTemplate: "(this space intentionally left blank)\n", Flags: []Flag{ &StringFlag{Name: "name", Value: "squirrel", Usage: "a name to say"}, }, - Action: func(c *Context) error { - fmt.Printf("Hello %v\n", c.String("name")) + Action: func(cCtx *Context) error { + fmt.Printf("Hello %v\n", cCtx.String("name")) return nil }, } @@ -145,12 +146,46 @@ func ExampleApp_Suggest() { // // Did you mean '--name'? // - // NAME: - // greet - A new cli application - // - // USAGE: - // greet [global options] [arguments...] - // - // GLOBAL OPTIONS: - // --name value a name to say (default: "squirrel") + // (this space intentionally left blank) +} + +func ExampleApp_Suggest_command() { + app := &App{ + Name: "greet", + Suggest: true, + HideHelp: true, + HideHelpCommand: true, + CustomAppHelpTemplate: "(this space intentionally left blank)\n", + Flags: []Flag{ + &StringFlag{Name: "name", Value: "squirrel", Usage: "a name to say"}, + }, + Action: func(cCtx *Context) error { + fmt.Printf("Hello %v\n", cCtx.String("name")) + return nil + }, + Commands: []*Command{ + { + Name: "neighbors", + CustomHelpTemplate: "(this space intentionally left blank)\n", + Flags: []Flag{ + &BoolFlag{Name: "smiling"}, + }, + Action: func(cCtx *Context) error { + if cCtx.Bool("smiling") { + fmt.Println("😀") + } + fmt.Println("Hello, neighbors") + return nil + }, + }, + }, + } + + app.Run([]string{"greet", "neighbors", "--sliming"}) + // Output: + // Incorrect Usage: flag provided but not defined: -sliming + // + // Did you mean '--smiling'? + // + // (this space intentionally left blank) } From f3cf7640c7b846daeb4bedfb4b1e70334ffe8039 Mon Sep 17 00:00:00 2001 From: Dan Buch Date: Tue, 17 May 2022 08:49:15 -0400 Subject: [PATCH 7/7] Backing out build tag for suggestions per recommendation from @kolyshkin :bow: --- .github/workflows/cli.yml | 8 +------- Makefile | 4 ++-- README.md | 10 ---------- docs.go | 4 ++-- docs_test.go | 4 ++-- internal/build/build.go | 2 +- suggestions.go | 3 --- suggestions_stubs.go | 12 ------------ suggestions_test.go | 3 --- 9 files changed, 8 insertions(+), 42 deletions(-) delete mode 100644 suggestions_stubs.go diff --git a/.github/workflows/cli.yml b/.github/workflows/cli.yml index caab03d..a7c395a 100644 --- a/.github/workflows/cli.yml +++ b/.github/workflows/cli.yml @@ -37,15 +37,9 @@ jobs: - name: vet run: go run internal/build/build.go vet - - name: test with urfave_cli_core tag - run: go run internal/build/build.go -tags urfave_cli_core test - - name: test with urfave_cli_no_docs tag run: go run internal/build/build.go -tags urfave_cli_no_docs test - - name: test with urfave_cli_no_suggest tag - run: go run internal/build/build.go -tags urfave_cli_no_suggest test - - name: test run: go run internal/build/build.go test @@ -53,7 +47,7 @@ jobs: run: go run internal/build/build.go check-binary-size - name: check-binary-size with tags (informational only) - run: go run internal/build/build.go -tags urfave_cli_core check-binary-size + run: go run internal/build/build.go -tags urfave_cli_no_docs check-binary-size - name: Upload coverage to Codecov if: success() && matrix.go == '1.18.x' && matrix.os == 'ubuntu-latest' diff --git a/Makefile b/Makefile index 6ab812a..ba783cb 100644 --- a/Makefile +++ b/Makefile @@ -17,11 +17,11 @@ all: generate vet tag-test test check-binary-size tag-check-binary-size gfmrun t .PHONY: tag-test tag-test: - go run internal/build/build.go -tags urfave_cli_core test + go run internal/build/build.go -tags urfave_cli_no_docs test .PHONY: tag-check-binary-size tag-check-binary-size: - go run internal/build/build.go -tags urfave_cli_core check-binary-size + go run internal/build/build.go -tags urfave_cli_no_docs check-binary-size .PHONY: gfmrun gfmrun: diff --git a/README.md b/README.md index 35694d2..beb5963 100644 --- a/README.md +++ b/README.md @@ -59,22 +59,12 @@ import ( You can use the following build tags: -#### `urfave_cli_core` - -When set, applies all `urfave_cli_no.+` build tags to minimize resulting binary -size. - #### `urfave_cli_no_docs` When set, this removes `ToMarkdown` and `ToMan` methods, so your application won't be able to call those. This reduces the resulting binary size by about 300-400 KB (measured using Go 1.18.1 on Linux/amd64), due to fewer dependencies. -#### `urfave_cli_no_suggest` - -When set, the capability enabled by setting `App.Suggest` will be a no-op. This -reduces the resulting binary size due to fewer dependencies. - ### GOPATH Make sure your `PATH` includes the `$GOPATH/bin` directory so your commands can diff --git a/docs.go b/docs.go index 4aba31b..8b1c9c8 100644 --- a/docs.go +++ b/docs.go @@ -1,5 +1,5 @@ -//go:build !urfave_cli_no_docs && !urfave_cli_core -// +build !urfave_cli_no_docs,!urfave_cli_core +//go:build !urfave_cli_no_docs +// +build !urfave_cli_no_docs package cli diff --git a/docs_test.go b/docs_test.go index bc3703f..12d5d3c 100644 --- a/docs_test.go +++ b/docs_test.go @@ -1,5 +1,5 @@ -//go:build !urfave_cli_no_docs && !urfave_cli_core -// +build !urfave_cli_no_docs,!urfave_cli_core +//go:build !urfave_cli_no_docs +// +build !urfave_cli_no_docs package cli diff --git a/internal/build/build.go b/internal/build/build.go index e74acfb..929ab0f 100644 --- a/internal/build/build.go +++ b/internal/build/build.go @@ -242,7 +242,7 @@ func checkBinarySizeActionFunc(c *cli.Context) (err error) { tags := c.String("tags") - if strings.Contains(tags, "urfave_cli_core") || strings.Contains(tags, "urfave_cli_no_docs") { + if strings.Contains(tags, "urfave_cli_no_docs") { desiredMinBinarySize = 1.39 } diff --git a/suggestions.go b/suggestions.go index 301c598..476af4d 100644 --- a/suggestions.go +++ b/suggestions.go @@ -1,6 +1,3 @@ -//go:build !urfave_cli_no_suggest && !urfave_cli_core -// +build !urfave_cli_no_suggest,!urfave_cli_core - package cli import ( diff --git a/suggestions_stubs.go b/suggestions_stubs.go deleted file mode 100644 index 48643bb..0000000 --- a/suggestions_stubs.go +++ /dev/null @@ -1,12 +0,0 @@ -//go:build urfave_cli_no_suggest || urfave_cli_core -// +build urfave_cli_no_suggest urfave_cli_core - -package cli - -func (a *App) suggestFlagFromError(err error, _ string) (string, error) { - return "", err -} - -func suggestCommand([]*Command, string) string { - return "" -} diff --git a/suggestions_test.go b/suggestions_test.go index fa7b0c9..4ebe9c0 100644 --- a/suggestions_test.go +++ b/suggestions_test.go @@ -1,6 +1,3 @@ -//go:build !urfave_cli_no_suggest && !urfave_cli_core -// +build !urfave_cli_no_suggest,!urfave_cli_core - package cli import (