120 lines
3.1 KiB
Go
120 lines
3.1 KiB
Go
package client
|
|
|
|
import (
|
|
"bufio"
|
|
"fmt"
|
|
"os"
|
|
"strings"
|
|
|
|
"github.com/bgentry/speakeasy"
|
|
isatty "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) (seed string, err error) {
|
|
if inputIsTty() {
|
|
fmt.Println(prompt)
|
|
}
|
|
seed, err = readLineFromBuf(buf)
|
|
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) {
|
|
// 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
|
|
}
|
|
}
|
|
}
|
|
|
|
// 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
|
|
}
|