Some checks failed
Build SimApp / build (amd64) (push) Waiting to run
Build SimApp / build (arm64) (push) Waiting to run
CodeQL / Analyze (push) Waiting to run
Build & Push / build (push) Waiting to run
Run Gosec / Gosec (push) Waiting to run
Lint / golangci-lint (push) Waiting to run
Checks dependencies and mocks generation / Check go mod tidy (push) Waiting to run
Checks dependencies and mocks generation / Check up to date mocks (push) Waiting to run
System Tests / setup (push) Waiting to run
System Tests / test-system (push) Blocked by required conditions
System Tests / test-system-legacy (push) Blocked by required conditions
Tests / Code Coverage / split-test-files (push) Waiting to run
Tests / Code Coverage / tests (00) (push) Blocked by required conditions
Tests / Code Coverage / tests (01) (push) Blocked by required conditions
Tests / Code Coverage / tests (02) (push) Blocked by required conditions
Tests / Code Coverage / tests (03) (push) Blocked by required conditions
Tests / Code Coverage / test-integration (push) Waiting to run
Tests / Code Coverage / test-e2e (push) Waiting to run
Tests / Code Coverage / repo-analysis (push) Blocked by required conditions
Tests / Code Coverage / test-sim-nondeterminism (push) Waiting to run
Tests / Code Coverage / test-clientv2 (push) Waiting to run
Tests / Code Coverage / test-core (push) Waiting to run
Tests / Code Coverage / test-depinject (push) Waiting to run
Tests / Code Coverage / test-errors (push) Waiting to run
Tests / Code Coverage / test-math (push) Waiting to run
Tests / Code Coverage / test-schema (push) Waiting to run
Tests / Code Coverage / test-collections (push) Waiting to run
Tests / Code Coverage / test-cosmovisor (push) Waiting to run
Tests / Code Coverage / test-confix (push) Waiting to run
Tests / Code Coverage / test-store (push) Waiting to run
Tests / Code Coverage / test-log (push) Waiting to run
Tests / Code Coverage / test-x-tx (push) Waiting to run
Tests / Code Coverage / test-x-nft (push) Waiting to run
Tests / Code Coverage / test-x-circuit (push) Waiting to run
Tests / Code Coverage / test-x-feegrant (push) Waiting to run
Tests / Code Coverage / test-x-evidence (push) Waiting to run
Tests / Code Coverage / test-x-upgrade (push) Waiting to run
Tests / Code Coverage / test-tools-benchmark (push) Waiting to run
Build & Push SDK Proto Builder / build (push) Has been cancelled
110 lines
2.7 KiB
Go
110 lines
2.7 KiB
Go
package input
|
|
|
|
import (
|
|
"bufio"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"os"
|
|
"strings"
|
|
|
|
"github.com/bgentry/speakeasy"
|
|
isatty "github.com/mattn/go-isatty"
|
|
)
|
|
|
|
// MinPassLength is the minimum acceptable password length
|
|
const MinPassLength = 8
|
|
|
|
// 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, fmt.Errorf("password must be at least %d characters", MinPassLength)
|
|
}
|
|
|
|
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 returns false and a nil error.
|
|
func GetConfirmation(prompt string, r *bufio.Reader, w io.Writer) (bool, error) {
|
|
if inputIsTty() {
|
|
_, _ = fmt.Fprintf(w, "%s [y/N]: ", prompt)
|
|
}
|
|
|
|
response, err := readLineFromBuf(r)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
response = strings.TrimSpace(response)
|
|
if len(response) == 0 {
|
|
return false, nil
|
|
}
|
|
|
|
response = strings.ToLower(response)
|
|
if response[0] == 'y' {
|
|
return true, nil
|
|
}
|
|
|
|
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 != "" {
|
|
fmt.Fprintf(os.Stderr, "> %s\n", 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 reader.
|
|
// 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')
|
|
|
|
switch {
|
|
case errors.Is(err, io.EOF):
|
|
// If by any chance the error is EOF, but we were actually able to read
|
|
// something from the reader then don't return the EOF error.
|
|
// If we didn't read anything from the reader and got the EOF error, then
|
|
// it's safe to return EOF back to the caller.
|
|
if len(pass) > 0 {
|
|
// exit the switch statement
|
|
break
|
|
}
|
|
return "", err
|
|
|
|
case err != nil:
|
|
return "", err
|
|
}
|
|
|
|
return strings.TrimSpace(pass), nil
|
|
}
|