vendor: update github.com/mitchellh/cli

This pulls in the patch that fixes the nil panic and the
data race in MockUi.
This commit is contained in:
Frank Schroeder 2017-05-23 19:35:23 +02:00
parent 53fd15070d
commit 3caadb37ac
No known key found for this signature in database
GPG Key ID: 4D65C6EAEC87DECD
5 changed files with 96 additions and 29 deletions

View File

@ -3,8 +3,11 @@
cli is a library for implementing powerful command-line interfaces in Go. cli is a library for implementing powerful command-line interfaces in Go.
cli is the library that powers the CLI for cli is the library that powers the CLI for
[Packer](https://github.com/mitchellh/packer), [Packer](https://github.com/mitchellh/packer),
[Serf](https://github.com/hashicorp/serf), and [Serf](https://github.com/hashicorp/serf),
[Consul](https://github.com/hashicorp/consul). [Consul](https://github.com/hashicorp/consul),
[Vault](https://github.com/hashicorp/vault),
[Terraform](https://github.com/hashicorp/terraform), and
[Nomad](https://github.com/hashicorp/nomad).
## Features ## Features

View File

@ -4,6 +4,7 @@ import (
"fmt" "fmt"
"io" "io"
"os" "os"
"regexp"
"sort" "sort"
"strings" "strings"
"sync" "sync"
@ -24,7 +25,7 @@ import (
// //
// * We use longest prefix matching to find a matching subcommand. This // * We use longest prefix matching to find a matching subcommand. This
// means if you register "foo bar" and the user executes "cli foo qux", // means if you register "foo bar" and the user executes "cli foo qux",
// the "foo" commmand will be executed with the arg "qux". It is up to // the "foo" command will be executed with the arg "qux". It is up to
// you to handle these args. One option is to just return the special // you to handle these args. One option is to just return the special
// help return code `RunResultHelp` to display help and exit. // help return code `RunResultHelp` to display help and exit.
// //
@ -119,7 +120,13 @@ func (c *CLI) Run() (int, error) {
// Just show the version and exit if instructed. // Just show the version and exit if instructed.
if c.IsVersion() && c.Version != "" { if c.IsVersion() && c.Version != "" {
c.HelpWriter.Write([]byte(c.Version + "\n")) c.HelpWriter.Write([]byte(c.Version + "\n"))
return 1, nil return 0, nil
}
// Just print the help when only '-h' or '--help' is passed.
if c.IsHelp() && c.Subcommand() == "" {
c.HelpWriter.Write([]byte(c.HelpFunc(c.Commands) + "\n"))
return 0, nil
} }
// Attempt to get the factory function for creating the command // Attempt to get the factory function for creating the command
@ -132,13 +139,13 @@ func (c *CLI) Run() (int, error) {
command, err := raw.(CommandFactory)() command, err := raw.(CommandFactory)()
if err != nil { if err != nil {
return 0, err return 1, err
} }
// If we've been instructed to just print the help, then print it // If we've been instructed to just print the help, then print it
if c.IsHelp() { if c.IsHelp() {
c.commandHelp(command) c.commandHelp(command)
return 1, nil return 0, nil
} }
// If there is an invalid flag, then error // If there is an invalid flag, then error
@ -249,7 +256,7 @@ func (c *CLI) init() {
c.commandTree.Walk(walkFn) c.commandTree.Walk(walkFn)
// Insert any that we're missing // Insert any that we're missing
for k, _ := range toInsert { for k := range toInsert {
var f CommandFactory = func() (Command, error) { var f CommandFactory = func() (Command, error) {
return &MockCommand{ return &MockCommand{
HelpText: "This command is accessed by using one of the subcommands below.", HelpText: "This command is accessed by using one of the subcommands below.",
@ -387,16 +394,22 @@ func (c *CLI) helpCommands(prefix string) map[string]CommandFactory {
func (c *CLI) processArgs() { func (c *CLI) processArgs() {
for i, arg := range c.Args { for i, arg := range c.Args {
if arg == "--" {
break
}
// Check for help flags.
if arg == "-h" || arg == "-help" || arg == "--help" {
c.isHelp = true
continue
}
if c.subcommand == "" { if c.subcommand == "" {
// Check for version and help flags if not in a subcommand // Check for version flags if not in a subcommand.
if arg == "-v" || arg == "-version" || arg == "--version" { if arg == "-v" || arg == "-version" || arg == "--version" {
c.isVersion = true c.isVersion = true
continue continue
} }
if arg == "-h" || arg == "-help" || arg == "--help" {
c.isHelp = true
continue
}
if arg != "" && arg[0] == '-' { if arg != "" && arg[0] == '-' {
// Record the arg... // Record the arg...
@ -405,16 +418,24 @@ func (c *CLI) processArgs() {
} }
// If we didn't find a subcommand yet and this is the first non-flag // If we didn't find a subcommand yet and this is the first non-flag
// argument, then this is our subcommand. j // argument, then this is our subcommand.
if c.subcommand == "" && arg != "" && arg[0] != '-' { if c.subcommand == "" && arg != "" && arg[0] != '-' {
c.subcommand = arg c.subcommand = arg
if c.commandNested { if c.commandNested {
// Nested CLI, the subcommand is actually the entire // Nested CLI, the subcommand is actually the entire
// arg list up to a flag that is still a valid subcommand. // arg list up to a flag that is still a valid subcommand.
k, _, ok := c.commandTree.LongestPrefix(strings.Join(c.Args[i:], " ")) searchKey := strings.Join(c.Args[i:], " ")
k, _, ok := c.commandTree.LongestPrefix(searchKey)
if ok { if ok {
c.subcommand = k // k could be a prefix that doesn't contain the full
i += strings.Count(k, " ") // command such as "foo" instead of "foobar", so we
// need to verify that we have an entire key. To do that,
// we look for an ending in a space or an end of string.
reVerify := regexp.MustCompile(regexp.QuoteMeta(k) + `( |$)`)
if reVerify.MatchString(searchKey) {
c.subcommand = k
i += strings.Count(k, " ")
}
} }
} }
@ -439,7 +460,7 @@ const defaultHelpTemplate = `
{{.Help}}{{if gt (len .Subcommands) 0}} {{.Help}}{{if gt (len .Subcommands) 0}}
Subcommands: Subcommands:
{{ range $value := .Subcommands }} {{- range $value := .Subcommands }}
{{ $value.NameAligned }} {{ $value.Synopsis }}{{ end }} {{ $value.NameAligned }} {{ $value.Synopsis }}{{ end }}
{{ end }} {{- end }}
` `

View File

@ -18,7 +18,7 @@ func BasicHelpFunc(app string) HelpFunc {
return func(commands map[string]CommandFactory) string { return func(commands map[string]CommandFactory) string {
var buf bytes.Buffer var buf bytes.Buffer
buf.WriteString(fmt.Sprintf( buf.WriteString(fmt.Sprintf(
"usage: %s [--version] [--help] <command> [<args>]\n\n", "Usage: %s [--version] [--help] <command> [<args>]\n\n",
app)) app))
buf.WriteString("Available commands are:\n") buf.WriteString("Available commands are:\n")
@ -26,7 +26,7 @@ func BasicHelpFunc(app string) HelpFunc {
// key length so they can be aligned properly. // key length so they can be aligned properly.
keys := make([]string, 0, len(commands)) keys := make([]string, 0, len(commands))
maxKeyLen := 0 maxKeyLen := 0
for key, _ := range commands { for key := range commands {
if len(key) > maxKeyLen { if len(key) > maxKeyLen {
maxKeyLen = len(key) maxKeyLen = len(key)
} }

View File

@ -7,12 +7,25 @@ import (
"sync" "sync"
) )
// MockUi is a mock UI that is used for tests and is exported publicly for // NewMockUi returns a fully initialized MockUi instance
// use in external tests if needed as well. // which is safe for concurrent use.
func NewMockUi() *MockUi {
m := new(MockUi)
m.once.Do(m.init)
return m
}
// MockUi is a mock UI that is used for tests and is exported publicly
// for use in external tests if needed as well. Do not instantite this
// directly since the buffers will be initialized on the first write. If
// there is no write then you will get a nil panic. Please use the
// NewMockUi() constructor function instead. You can fix your code with
//
// sed -i -e 's/new(cli.MockUi)/cli.NewMockUi()/g' *_test.go
type MockUi struct { type MockUi struct {
InputReader io.Reader InputReader io.Reader
ErrorWriter *bytes.Buffer ErrorWriter *syncBuffer
OutputWriter *bytes.Buffer OutputWriter *syncBuffer
once sync.Once once sync.Once
} }
@ -59,6 +72,36 @@ func (u *MockUi) Warn(message string) {
} }
func (u *MockUi) init() { func (u *MockUi) init() {
u.ErrorWriter = new(bytes.Buffer) u.ErrorWriter = new(syncBuffer)
u.OutputWriter = new(bytes.Buffer) u.OutputWriter = new(syncBuffer)
}
type syncBuffer struct {
sync.RWMutex
b bytes.Buffer
}
func (b *syncBuffer) Write(data []byte) (int, error) {
b.Lock()
defer b.Unlock()
return b.b.Write(data)
}
func (b *syncBuffer) Read(data []byte) (int, error) {
b.RLock()
defer b.RUnlock()
return b.b.Read(data)
}
func (b *syncBuffer) Reset() {
b.Lock()
b.b.Reset()
b.Unlock()
}
func (b *syncBuffer) String() string {
b.RLock()
data := b.b.Bytes()
b.RUnlock()
return string(data)
} }

6
vendor/vendor.json vendored
View File

@ -706,10 +706,10 @@
"revisionTime": "2016-07-26T03:20:27Z" "revisionTime": "2016-07-26T03:20:27Z"
}, },
{ {
"checksumSHA1": "yF39M9MGatDbq2d2oqlLy44jsRc=", "checksumSHA1": "bUuI7AVR3IZPLlBaEKmw/ke7wqA=",
"path": "github.com/mitchellh/cli", "path": "github.com/mitchellh/cli",
"revision": "168daae10d6ff81b8b1201b0a4c9607d7e9b82e3", "revision": "b481eac70eea3ad671b7c360a013f89bb759b252",
"revisionTime": "2016-03-23T17:07:00Z" "revisionTime": "2017-05-23T17:27:49Z"
}, },
{ {
"checksumSHA1": "86nE93o1VIND0Doe8PuhCXnhUx0=", "checksumSHA1": "86nE93o1VIND0Doe8PuhCXnhUx0=",