package client import ( "bufio" "fmt" "os" "strings" "github.com/bgentry/speakeasy" "github.com/mattn/go-isatty" "github.com/pkg/errors" ) // MinPassLength is the minimum acceptable password length const MinPassLength = 8 // BufferStdin is used to allow reading prompts for stdin // multiple times, when we read from non-tty func BufferStdin() *bufio.Reader { return bufio.NewReader(os.Stdin) } // GetPassword will prompt for a password one-time (to sign a tx) // It enforces the password length func GetPassword(prompt string, buf *bufio.Reader) (pass string, err error) { if inputIsTty() { pass, err = speakeasy.FAsk(os.Stderr, prompt) } else { pass, err = readLineFromBuf(buf) } if err != nil { return "", err } if len(pass) < MinPassLength { // Return the given password to the upstream client so it can handle a // non-STDIN failure gracefully. return pass, errors.Errorf("password must be at least %d characters", MinPassLength) } return pass, nil } // GetSeed will request a seed phrase from stdin and trims off // leading/trailing spaces func GetSeed(prompt string, buf *bufio.Reader) (string, error) { return GetString(prompt, buf) } // GetCheckPassword will prompt for a password twice to verify they // match (for creating a new password). // It enforces the password length. Only parses password once if // input is piped in. func GetCheckPassword(prompt, prompt2 string, buf *bufio.Reader) (string, error) { // simple read on no-tty if !inputIsTty() { return GetPassword(prompt, buf) } // TODO: own function??? pass, err := GetPassword(prompt, buf) if err != nil { return "", err } pass2, err := GetPassword(prompt2, buf) if err != nil { return "", err } if pass != pass2 { return "", errors.New("passphrases don't match") } return pass, nil } // GetConfirmation will request user give the confirmation from stdin. // "y", "Y", "yes", "YES", and "Yes" all count as confirmations. // If the input is not recognized, it will ask again. func GetConfirmation(prompt string, buf *bufio.Reader) (bool, error) { for { if inputIsTty() { fmt.Print(fmt.Sprintf("%s [y/n]:", prompt)) } response, err := readLineFromBuf(buf) if err != nil { return false, err } response = strings.ToLower(strings.TrimSpace(response)) if response == "y" || response == "yes" { return true, nil } else if response == "n" || response == "no" { return false, nil } } } // GetString simply returns the trimmed string output of a given reader. func GetString(prompt string, buf *bufio.Reader) (string, error) { if inputIsTty() && prompt != "" { PrintPrefixed(prompt) } out, err := readLineFromBuf(buf) if err != nil { return "", err } return strings.TrimSpace(out), nil } // inputIsTty returns true iff we have an interactive prompt, // where we can disable echo and request to repeat the password. // If false, we can optimize for piped input from another command func inputIsTty() bool { return isatty.IsTerminal(os.Stdin.Fd()) || isatty.IsCygwinTerminal(os.Stdin.Fd()) } // readLineFromBuf reads one line from stdin. // Subsequent calls reuse the same buffer, so we don't lose // any input when reading a password twice (to verify) func readLineFromBuf(buf *bufio.Reader) (string, error) { pass, err := buf.ReadString('\n') if err != nil { return "", err } return strings.TrimSpace(pass), nil } // PrintPrefixed prints a string with > prefixed for use in prompts. func PrintPrefixed(msg string) { msg = fmt.Sprintf("> %s\n", msg) fmt.Fprint(os.Stderr, msg) }