cosmos-sdk/client/input.go

115 lines
3.0 KiB
Go
Raw Normal View History

2018-02-23 01:40:10 -08:00
package client
import (
"bufio"
"fmt"
"os"
"strings"
"github.com/bgentry/speakeasy"
2018-02-23 02:00:33 -08:00
isatty "github.com/mattn/go-isatty"
2018-02-23 01:40:10 -08:00
"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)
}
2018-02-23 01:40:10 -08:00
// 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) {
2018-02-23 01:40:10 -08:00
if inputIsTty() {
pass, err = speakeasy.Ask(prompt)
} else {
pass, err = readLineFromBuf(buf)
2018-02-23 01:40:10 -08:00
}
if err != nil {
return "", err
}
if len(pass) < MinPassLength {
return "", 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) (seed string, err error) {
2018-02-23 01:40:10 -08:00
if inputIsTty() {
fmt.Println(prompt)
}
seed, err = readLineFromBuf(buf)
2018-02-23 01:40:10 -08:00
seed = strings.TrimSpace(seed)
return
}
// 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) {
2018-02-23 01:40:10 -08:00
// simple read on no-tty
if !inputIsTty() {
return GetPassword(prompt, buf)
2018-02-23 01:40:10 -08:00
}
// TODO: own function???
pass, err := GetPassword(prompt, buf)
2018-02-23 01:40:10 -08:00
if err != nil {
return "", err
}
pass2, err := GetPassword(prompt2, buf)
2018-02-23 01:40:10 -08:00
if err != nil {
return "", err
}
if pass != pass2 {
return "", errors.New("Passphrases don't match")
}
return pass, nil
}
2018-03-24 23:52:51 -07:00
// 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
}
}
}
2018-02-26 04:44:27 -08:00
// 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
2018-02-23 01:40:10 -08:00
func inputIsTty() bool {
return isatty.IsTerminal(os.Stdin.Fd()) || isatty.IsCygwinTerminal(os.Stdin.Fd())
}
// readLineFromBuf reads one line from stdin.
2018-02-26 04:44:27 -08:00
// 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) {
2018-02-23 01:40:10 -08:00
pass, err := buf.ReadString('\n')
if err != nil {
return "", err
}
return strings.TrimSpace(pass), nil
}