3b55b50eac
Change-Id: I2cf58c1c41bfffa800949dc441a488c16a448375
182 lines
4.4 KiB
Go
182 lines
4.4 KiB
Go
// Copyright (C) 2019 Storj Labs, Inc.
|
|
// See LICENSE for copying information
|
|
|
|
package wizard
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"net"
|
|
"os"
|
|
"strings"
|
|
|
|
"github.com/spf13/cobra"
|
|
"github.com/zeebo/errs"
|
|
"golang.org/x/crypto/ssh/terminal"
|
|
)
|
|
|
|
// applyDefaultHostAndPortToAddr applies the default host and/or port if either is missing in the specified address.
|
|
func applyDefaultHostAndPortToAddr(address, defaultAddress string) (string, error) {
|
|
defaultHost, defaultPort, err := net.SplitHostPort(defaultAddress)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
addressParts := strings.Split(address, ":")
|
|
numberOfParts := len(addressParts)
|
|
|
|
if numberOfParts > 1 && len(addressParts[0]) > 0 && len(addressParts[1]) > 0 {
|
|
// address is host:port so skip applying any defaults.
|
|
return address, nil
|
|
}
|
|
|
|
// We are missing a host:port part. Figure out which part we are missing.
|
|
indexOfPortSeparator := strings.Index(address, ":")
|
|
lengthOfFirstPart := len(addressParts[0])
|
|
|
|
if indexOfPortSeparator < 0 {
|
|
if lengthOfFirstPart == 0 {
|
|
// address is blank.
|
|
return defaultAddress, nil
|
|
}
|
|
// address is host
|
|
return net.JoinHostPort(addressParts[0], defaultPort), nil
|
|
}
|
|
|
|
if indexOfPortSeparator == 0 {
|
|
// address is :1234
|
|
return net.JoinHostPort(defaultHost, addressParts[1]), nil
|
|
}
|
|
|
|
// address is host:
|
|
return net.JoinHostPort(addressParts[0], defaultPort), nil
|
|
}
|
|
|
|
// PromptForAccessName handles user input for access name to be used with wizards
|
|
func PromptForAccessName() (string, error) {
|
|
_, err := fmt.Printf("Choose an access name [\"default\"]: ")
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
var accessName string
|
|
n, err := fmt.Scanln(&accessName)
|
|
if err != nil && n != 0 {
|
|
return "", err
|
|
}
|
|
|
|
if accessName == "" {
|
|
return "default", nil
|
|
}
|
|
return accessName, nil
|
|
}
|
|
|
|
// PromptForSatellite handles user input for a satellite address to be used with wizards
|
|
func PromptForSatellite(cmd *cobra.Command) (string, error) {
|
|
satellites := []string{"us-central-1.tardigrade.io", "europe-west-1.tardigrade.io", "asia-east-1.tardigrade.io"}
|
|
|
|
_, err := fmt.Print("Select your satellite:\n")
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
for iterator, value := range satellites {
|
|
_, err := fmt.Printf("\t[%d] %s\n", iterator+1, value)
|
|
if err != nil {
|
|
return "", nil
|
|
}
|
|
}
|
|
|
|
_, err = fmt.Print("Please enter numeric choice or enter satellite address manually [1]: ")
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
var satelliteAddress string
|
|
n, err := fmt.Scanln(&satelliteAddress)
|
|
if err != nil {
|
|
if n == 0 {
|
|
// fmt.Scanln cannot handle empty input
|
|
satelliteAddress = satellites[0]
|
|
} else {
|
|
return "", err
|
|
}
|
|
}
|
|
|
|
// TODO add better validation
|
|
if satelliteAddress == "" {
|
|
return "", errs.New("satellite address cannot be empty")
|
|
} else if len(satelliteAddress) == 1 {
|
|
switch satelliteAddress {
|
|
case "1":
|
|
satelliteAddress = satellites[0]
|
|
case "2":
|
|
satelliteAddress = satellites[1]
|
|
case "3":
|
|
satelliteAddress = satellites[2]
|
|
default:
|
|
return "", errs.New("Satellite address cannot be one character")
|
|
}
|
|
}
|
|
|
|
return applyDefaultHostAndPortToAddr(satelliteAddress, cmd.Flags().Lookup("satellite-addr").Value.String())
|
|
}
|
|
|
|
// PromptForAPIKey handles user input for an API key to be used with wizards
|
|
func PromptForAPIKey() (string, error) {
|
|
_, err := fmt.Print("Enter your API key: ")
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
var apiKey string
|
|
n, err := fmt.Scanln(&apiKey)
|
|
if err != nil && n != 0 {
|
|
return "", err
|
|
}
|
|
|
|
if apiKey == "" {
|
|
return "", errs.New("API key cannot be empty")
|
|
}
|
|
|
|
return apiKey, nil
|
|
}
|
|
|
|
// PromptForEncryptionPassphrase handles user input for an encryption passphrase to be used with wizards
|
|
func PromptForEncryptionPassphrase() (string, error) {
|
|
_, err := fmt.Print("Enter your encryption passphrase: ")
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
encKey, err := terminal.ReadPassword(int(os.Stdin.Fd()))
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
_, err = fmt.Println()
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
if len(encKey) == 0 {
|
|
return "", errs.New("Encryption passphrase cannot be empty")
|
|
}
|
|
|
|
_, err = fmt.Print("Enter your encryption passphrase again: ")
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
repeatedEncKey, err := terminal.ReadPassword(int(os.Stdin.Fd()))
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
_, err = fmt.Println()
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
if !bytes.Equal(encKey, repeatedEncKey) {
|
|
return "", errs.New("encryption passphrase does not match")
|
|
}
|
|
|
|
return string(encKey), nil
|
|
}
|