cli: use Writer from app instead of Stdout

It is useful in tests.
This commit is contained in:
Evgenii Stratonikov 2020-08-28 12:11:19 +03:00
parent bfe3b3d05d
commit 911be78cc7
7 changed files with 96 additions and 93 deletions

View file

@ -357,9 +357,9 @@ func startServer(ctx *cli.Context) error {
go serv.Start(errChan) go serv.Start(errChan)
go rpcServer.Start(errChan) go rpcServer.Start(errChan)
fmt.Println(logo()) fmt.Fprintln(ctx.App.Writer, logo())
fmt.Println(serv.UserAgent) fmt.Fprintln(ctx.App.Writer, serv.UserAgent)
fmt.Println() fmt.Fprintln(ctx.App.Writer)
var shutdownErr error var shutdownErr error
Main: Main:

View file

@ -366,7 +366,7 @@ func initSmartContract(ctx *cli.Context) error {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
fmt.Printf("Successfully initialized smart contract [%s]\n", contractName) fmt.Fprintf(ctx.App.Writer, "Successfully initialized smart contract [%s]\n", contractName)
return nil return nil
} }
@ -405,7 +405,7 @@ func contractCompile(ctx *cli.Context) error {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
if ctx.Bool("verbose") { if ctx.Bool("verbose") {
fmt.Println(hex.EncodeToString(result)) fmt.Fprintln(ctx.App.Writer, hex.EncodeToString(result))
} }
return nil return nil
@ -495,14 +495,14 @@ func invokeInternal(ctx *cli.Context, signAndPush bool) error {
if err != nil { if err != nil {
return cli.NewExitError(fmt.Errorf("failed to push invocation tx: %w", err), 1) return cli.NewExitError(fmt.Errorf("failed to push invocation tx: %w", err), 1)
} }
fmt.Printf("Sent invocation transaction %s\n", txHash.StringLE()) fmt.Fprintf(ctx.App.Writer, "Sent invocation transaction %s\n", txHash.StringLE())
} else { } else {
b, err := json.MarshalIndent(resp, "", " ") b, err := json.MarshalIndent(resp, "", " ")
if err != nil { if err != nil {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
fmt.Println(string(b)) fmt.Fprintln(ctx.App.Writer, string(b))
} }
return nil return nil
@ -599,7 +599,7 @@ func testInvokeScript(ctx *cli.Context) error {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
fmt.Println(string(b)) fmt.Fprintln(ctx.App.Writer, string(b))
return nil return nil
} }
@ -680,9 +680,9 @@ func getAccFromContext(ctx *cli.Context) (*wallet.Account, error) {
return nil, cli.NewExitError(fmt.Errorf("wallet contains no account for '%s'", address.Uint160ToString(addr)), 1) return nil, cli.NewExitError(fmt.Errorf("wallet contains no account for '%s'", address.Uint160ToString(addr)), 1)
} }
fmt.Printf("Enter account %s password > ", address.Uint160ToString(addr)) fmt.Fprintf(ctx.App.Writer, "Enter account %s password > ", address.Uint160ToString(addr))
rawPass, err := terminal.ReadPassword(syscall.Stdin) rawPass, err := terminal.ReadPassword(syscall.Stdin)
fmt.Println() fmt.Fprintln(ctx.App.Writer)
if err != nil { if err != nil {
return nil, cli.NewExitError(err, 1) return nil, cli.NewExitError(err, 1)
} }
@ -751,7 +751,7 @@ func contractDeploy(ctx *cli.Context) error {
if err != nil { if err != nil {
return cli.NewExitError(fmt.Errorf("failed to push invocation tx: %w", err), 1) return cli.NewExitError(fmt.Errorf("failed to push invocation tx: %w", err), 1)
} }
fmt.Printf("Sent deployment transaction %s for contract %s\n", txHash.StringLE(), nefFile.Header.ScriptHash.StringLE()) fmt.Fprintf(ctx.App.Writer, "Sent deployment transaction %s for contract %s\n", txHash.StringLE(), nefFile.Header.ScriptHash.StringLE())
return nil return nil
} }

View file

@ -33,6 +33,6 @@ func handleParse(ctx *cli.Context) error {
if err != nil { if err != nil {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
fmt.Print(res) fmt.Fprint(ctx.App.Writer, res)
return nil return nil
} }

View file

@ -3,6 +3,7 @@ package wallet
import ( import (
"encoding/json" "encoding/json"
"fmt" "fmt"
"io"
"io/ioutil" "io/ioutil"
"github.com/nspcc-dev/neo-go/cli/options" "github.com/nspcc-dev/neo-go/cli/options"
@ -50,7 +51,7 @@ func signMultisig(ctx *cli.Context) error {
if err != nil { if err != nil {
return cli.NewExitError(fmt.Errorf("invalid address: %w", err), 1) return cli.NewExitError(fmt.Errorf("invalid address: %w", err), 1)
} }
acc, err := getDecryptedAccount(wall, sh) acc, err := getDecryptedAccount(ctx, wall, sh)
if err != nil { if err != nil {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
@ -59,7 +60,7 @@ func signMultisig(ctx *cli.Context) error {
if !ok { if !ok {
return cli.NewExitError("verifiable item is not a transaction", 1) return cli.NewExitError("verifiable item is not a transaction", 1)
} }
printTxInfo(tx) printTxInfo(ctx.App.Writer, tx)
priv := acc.PrivateKey() priv := acc.PrivateKey()
sign := priv.Sign(tx.GetSignedPart()) sign := priv.Sign(tx.GetSignedPart())
@ -86,11 +87,11 @@ func signMultisig(ctx *cli.Context) error {
if err != nil { if err != nil {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
fmt.Println(res.StringLE()) fmt.Fprintln(ctx.App.Writer, res.StringLE())
return nil return nil
} }
fmt.Println(tx.Hash().StringLE()) fmt.Fprintln(ctx.App.Writer, tx.Hash().StringLE())
return nil return nil
} }
@ -116,6 +117,6 @@ func writeParameterContext(c *context.ParameterContext, filename string) error {
return nil return nil
} }
func printTxInfo(t *transaction.Transaction) { func printTxInfo(w io.Writer, t *transaction.Transaction) {
fmt.Printf("Hash: %s\n", t.Hash().StringLE()) fmt.Fprintf(w, "Hash: %s\n", t.Hash().StringLE())
} }

View file

@ -162,9 +162,9 @@ func getNEP5Balance(ctx *cli.Context) error {
var token *wallet.Token var token *wallet.Token
name := ctx.String("token") name := ctx.String("token")
if name != "" { if name != "" {
token, err = getMatchingToken(wall, name) token, err = getMatchingToken(ctx, wall, name)
if err != nil { if err != nil {
token, err = getMatchingTokenRPC(c, addrHash, name) token, err = getMatchingTokenRPC(ctx, c, addrHash, name)
if err != nil { if err != nil {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
@ -181,26 +181,26 @@ func getNEP5Balance(ctx *cli.Context) error {
if name != "" && !token.Hash.Equals(asset) { if name != "" && !token.Hash.Equals(asset) {
continue continue
} }
fmt.Printf("TokenHash: %s\n", asset.StringLE()) fmt.Fprintf(ctx.App.Writer, "TokenHash: %s\n", asset.StringLE())
fmt.Printf("\tAmount : %s\n", balances.Balances[i].Amount) fmt.Fprintf(ctx.App.Writer, "\tAmount : %s\n", balances.Balances[i].Amount)
fmt.Printf("\tUpdated: %d\n", balances.Balances[i].LastUpdated) fmt.Fprintf(ctx.App.Writer, "\tUpdated: %d\n", balances.Balances[i].LastUpdated)
} }
return nil return nil
} }
func getMatchingToken(w *wallet.Wallet, name string) (*wallet.Token, error) { func getMatchingToken(ctx *cli.Context, w *wallet.Wallet, name string) (*wallet.Token, error) {
switch strings.ToLower(name) { switch strings.ToLower(name) {
case "neo": case "neo":
return neoToken, nil return neoToken, nil
case "gas": case "gas":
return gasToken, nil return gasToken, nil
} }
return getMatchingTokenAux(func(i int) *wallet.Token { return getMatchingTokenAux(ctx, func(i int) *wallet.Token {
return w.Extra.Tokens[i] return w.Extra.Tokens[i]
}, len(w.Extra.Tokens), name) }, len(w.Extra.Tokens), name)
} }
func getMatchingTokenRPC(c *client.Client, addr util.Uint160, name string) (*wallet.Token, error) { func getMatchingTokenRPC(ctx *cli.Context, c *client.Client, addr util.Uint160, name string) (*wallet.Token, error) {
bs, err := c.GetNEP5Balances(addr) bs, err := c.GetNEP5Balances(addr)
if err != nil { if err != nil {
return nil, err return nil, err
@ -209,18 +209,18 @@ func getMatchingTokenRPC(c *client.Client, addr util.Uint160, name string) (*wal
t, _ := c.NEP5TokenInfo(bs.Balances[i].Asset) t, _ := c.NEP5TokenInfo(bs.Balances[i].Asset)
return t return t
} }
return getMatchingTokenAux(get, len(bs.Balances), name) return getMatchingTokenAux(ctx, get, len(bs.Balances), name)
} }
func getMatchingTokenAux(get func(i int) *wallet.Token, n int, name string) (*wallet.Token, error) { func getMatchingTokenAux(ctx *cli.Context, get func(i int) *wallet.Token, n int, name string) (*wallet.Token, error) {
var token *wallet.Token var token *wallet.Token
var count int var count int
for i := 0; i < n; i++ { for i := 0; i < n; i++ {
t := get(i) t := get(i)
if t != nil && (t.Name == name || t.Symbol == name || t.Address() == name || t.Hash.StringLE() == name) { if t != nil && (t.Name == name || t.Symbol == name || t.Address() == name || t.Hash.StringLE() == name) {
if count == 1 { if count == 1 {
printTokenInfo(token) printTokenInfo(ctx, token)
printTokenInfo(t) printTokenInfo(ctx, t)
return nil, errors.New("multiple matching tokens found") return nil, errors.New("multiple matching tokens found")
} }
count++ count++
@ -247,7 +247,7 @@ func importNEP5Token(ctx *cli.Context) error {
for _, t := range wall.Extra.Tokens { for _, t := range wall.Extra.Tokens {
if t.Hash.Equals(tokenHash) { if t.Hash.Equals(tokenHash) {
printTokenInfo(t) printTokenInfo(ctx, t)
return cli.NewExitError("token already exists", 1) return cli.NewExitError("token already exists", 1)
} }
} }
@ -269,16 +269,17 @@ func importNEP5Token(ctx *cli.Context) error {
if err := wall.Save(); err != nil { if err := wall.Save(); err != nil {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
printTokenInfo(tok) printTokenInfo(ctx, tok)
return nil return nil
} }
func printTokenInfo(tok *wallet.Token) { func printTokenInfo(ctx *cli.Context, tok *wallet.Token) {
fmt.Printf("Name:\t%s\n", tok.Name) w := ctx.App.Writer
fmt.Printf("Symbol:\t%s\n", tok.Symbol) fmt.Fprintf(w, "Name:\t%s\n", tok.Name)
fmt.Printf("Hash:\t%s\n", tok.Hash.StringLE()) fmt.Fprintf(w, "Symbol:\t%s\n", tok.Symbol)
fmt.Printf("Decimals: %d\n", tok.Decimals) fmt.Fprintf(w, "Hash:\t%s\n", tok.Hash.StringLE())
fmt.Printf("Address: %s\n", tok.Address()) fmt.Fprintf(w, "Decimals: %d\n", tok.Decimals)
fmt.Fprintf(w, "Address: %s\n", tok.Address())
} }
func printNEP5Info(ctx *cli.Context) error { func printNEP5Info(ctx *cli.Context) error {
@ -289,19 +290,19 @@ func printNEP5Info(ctx *cli.Context) error {
defer wall.Close() defer wall.Close()
if name := ctx.String("token"); name != "" { if name := ctx.String("token"); name != "" {
token, err := getMatchingToken(wall, name) token, err := getMatchingToken(ctx, wall, name)
if err != nil { if err != nil {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
printTokenInfo(token) printTokenInfo(ctx, token)
return nil return nil
} }
for i, t := range wall.Extra.Tokens { for i, t := range wall.Extra.Tokens {
if i > 0 { if i > 0 {
fmt.Println() fmt.Fprintln(ctx.App.Writer)
} }
printTokenInfo(t) printTokenInfo(ctx, t)
} }
return nil return nil
} }
@ -317,12 +318,12 @@ func removeNEP5Token(ctx *cli.Context) error {
if name == "" { if name == "" {
return cli.NewExitError("token must be specified", 1) return cli.NewExitError("token must be specified", 1)
} }
token, err := getMatchingToken(wall, name) token, err := getMatchingToken(ctx, wall, name)
if err != nil { if err != nil {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
if !ctx.Bool("force") { if !ctx.Bool("force") {
if ok := askForConsent(); !ok { if ok := askForConsent(ctx.App.Writer); !ok {
return nil return nil
} }
} }
@ -343,7 +344,7 @@ func multiTransferNEP5(ctx *cli.Context) error {
fromFlag := ctx.Generic("from").(*flags.Address) fromFlag := ctx.Generic("from").(*flags.Address)
from := fromFlag.Uint160() from := fromFlag.Uint160()
acc, err := getDecryptedAccount(wall, from) acc, err := getDecryptedAccount(ctx, wall, from)
if err != nil { if err != nil {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
@ -369,10 +370,10 @@ func multiTransferNEP5(ctx *cli.Context) error {
} }
token, ok := cache[ss[0]] token, ok := cache[ss[0]]
if !ok { if !ok {
token, err = getMatchingToken(wall, ss[0]) token, err = getMatchingToken(ctx, wall, ss[0])
if err != nil { if err != nil {
fmt.Println("Can't find matching token in the wallet. Querying RPC-node for balances.") fmt.Fprintln(ctx.App.ErrWriter, "Can't find matching token in the wallet. Querying RPC-node for balances.")
token, err = getMatchingTokenRPC(c, from, ctx.String("token")) token, err = getMatchingTokenRPC(ctx, c, from, ctx.String("token"))
if err != nil { if err != nil {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
@ -406,7 +407,7 @@ func transferNEP5(ctx *cli.Context) error {
fromFlag := ctx.Generic("from").(*flags.Address) fromFlag := ctx.Generic("from").(*flags.Address)
from := fromFlag.Uint160() from := fromFlag.Uint160()
acc, err := getDecryptedAccount(wall, from) acc, err := getDecryptedAccount(ctx, wall, from)
if err != nil { if err != nil {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
@ -421,10 +422,10 @@ func transferNEP5(ctx *cli.Context) error {
toFlag := ctx.Generic("to").(*flags.Address) toFlag := ctx.Generic("to").(*flags.Address)
to := toFlag.Uint160() to := toFlag.Uint160()
token, err := getMatchingToken(wall, ctx.String("token")) token, err := getMatchingToken(ctx, wall, ctx.String("token"))
if err != nil { if err != nil {
fmt.Println("Can't find matching token in the wallet. Querying RPC-node for balances.") fmt.Fprintln(ctx.App.ErrWriter, "Can't find matching token in the wallet. Querying RPC-node for balances.")
token, err = getMatchingTokenRPC(c, from, ctx.String("token")) token, err = getMatchingTokenRPC(ctx, c, from, ctx.String("token"))
if err != nil { if err != nil {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
@ -470,10 +471,10 @@ func signAndSendTransfer(ctx *cli.Context, c *client.Client, acc *wallet.Account
if err != nil { if err != nil {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
fmt.Println(res.StringLE()) fmt.Fprintln(ctx.App.Writer, res.StringLE())
return nil return nil
} }
fmt.Println(tx.Hash().StringLE()) fmt.Fprintln(ctx.App.Writer, tx.Hash().StringLE())
return nil return nil
} }

View file

@ -83,7 +83,7 @@ func handleCandidate(ctx *cli.Context, method string) error {
addrFlag := ctx.Generic("address").(*flags.Address) addrFlag := ctx.Generic("address").(*flags.Address)
addr := addrFlag.Uint160() addr := addrFlag.Uint160()
acc, err := getDecryptedAccount(wall, addr) acc, err := getDecryptedAccount(ctx, wall, addr)
if err != nil { if err != nil {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
@ -111,7 +111,7 @@ func handleCandidate(ctx *cli.Context, method string) error {
if err != nil { if err != nil {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
fmt.Println(res.StringLE()) fmt.Fprintln(ctx.App.Writer, res.StringLE())
return nil return nil
} }
@ -123,7 +123,7 @@ func handleVote(ctx *cli.Context) error {
addrFlag := ctx.Generic("address").(*flags.Address) addrFlag := ctx.Generic("address").(*flags.Address)
addr := addrFlag.Uint160() addr := addrFlag.Uint160()
acc, err := getDecryptedAccount(wall, addr) acc, err := getDecryptedAccount(ctx, wall, addr)
if err != nil { if err != nil {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
@ -168,17 +168,17 @@ func handleVote(ctx *cli.Context) error {
if err != nil { if err != nil {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
fmt.Println(res.StringLE()) fmt.Fprintln(ctx.App.Writer, res.StringLE())
return nil return nil
} }
func getDecryptedAccount(wall *wallet.Wallet, addr util.Uint160) (*wallet.Account, error) { func getDecryptedAccount(ctx *cli.Context, wall *wallet.Wallet, addr util.Uint160) (*wallet.Account, error) {
acc := wall.GetAccount(addr) acc := wall.GetAccount(addr)
if acc == nil { if acc == nil {
return nil, fmt.Errorf("can't find account for the address: %s", address.Uint160ToString(addr)) return nil, fmt.Errorf("can't find account for the address: %s", address.Uint160ToString(addr))
} }
if pass, err := readPassword("Password > "); err != nil { if pass, err := readPassword(ctx.App.Writer, "Password > "); err != nil {
return nil, err return nil, err
} else if err := acc.Decrypt(pass); err != nil { } else if err := acc.Decrypt(pass); err != nil {
return nil, err return nil, err

View file

@ -5,6 +5,7 @@ import (
"encoding/hex" "encoding/hex"
"errors" "errors"
"fmt" "fmt"
"io"
"os" "os"
"strings" "strings"
"syscall" "syscall"
@ -222,7 +223,7 @@ func claimGas(ctx *cli.Context) error {
return cli.NewExitError("address was not provided", 1) return cli.NewExitError("address was not provided", 1)
} }
scriptHash := addrFlag.Uint160() scriptHash := addrFlag.Uint160()
acc, err := getDecryptedAccount(wall, scriptHash) acc, err := getDecryptedAccount(ctx, wall, scriptHash)
if err != nil { if err != nil {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
@ -245,7 +246,7 @@ func claimGas(ctx *cli.Context) error {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
fmt.Println(hash.StringLE()) fmt.Fprintln(ctx.App.Writer, hash.StringLE())
return nil return nil
} }
@ -265,7 +266,7 @@ func convertWallet(ctx *cli.Context) error {
for _, acc := range wall.Accounts { for _, acc := range wall.Accounts {
address.Prefix = address.NEO2Prefix address.Prefix = address.NEO2Prefix
pass, err := readPassword(fmt.Sprintf("Enter passphrase for account %s (label '%s') > ", acc.Address, acc.Label)) pass, err := readPassword(ctx.App.Writer, fmt.Sprintf("Enter passphrase for account %s (label '%s') > ", acc.Address, acc.Label))
if err != nil { if err != nil {
return cli.NewExitError(err, -1) return cli.NewExitError(err, -1)
} else if err := acc.Decrypt(pass); err != nil { } else if err := acc.Decrypt(pass); err != nil {
@ -347,7 +348,7 @@ loop:
for _, wif := range wifs { for _, wif := range wifs {
if decrypt { if decrypt {
pass, err := readPassword("Enter password > ") pass, err := readPassword(ctx.App.Writer, "Enter password > ")
if err != nil { if err != nil {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
@ -360,7 +361,7 @@ loop:
wif = pk.WIF() wif = pk.WIF()
} }
fmt.Println(wif) fmt.Fprintln(ctx.App.Writer, wif)
} }
return nil return nil
@ -389,7 +390,7 @@ func importMultisig(ctx *cli.Context) error {
} }
} }
acc, err := newAccountFromWIF(ctx.String("wif")) acc, err := newAccountFromWIF(ctx.App.Writer, ctx.String("wif"))
if err != nil { if err != nil {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
@ -419,7 +420,7 @@ func importDeployed(ctx *cli.Context) error {
return cli.NewExitError(fmt.Errorf("invalid contract hash: %w", err), 1) return cli.NewExitError(fmt.Errorf("invalid contract hash: %w", err), 1)
} }
acc, err := newAccountFromWIF(ctx.String("wif")) acc, err := newAccountFromWIF(ctx.App.Writer, ctx.String("wif"))
if err != nil { if err != nil {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
@ -463,7 +464,7 @@ func importWallet(ctx *cli.Context) error {
} }
defer wall.Close() defer wall.Close()
acc, err := newAccountFromWIF(ctx.String("wif")) acc, err := newAccountFromWIF(ctx.App.Writer, ctx.String("wif"))
if err != nil { if err != nil {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
@ -502,8 +503,8 @@ func removeAccount(ctx *cli.Context) error {
} }
if !ctx.Bool("force") { if !ctx.Bool("force") {
fmt.Printf("Account %s will be removed. This action is irreversible.\n", addrArg) fmt.Fprintf(ctx.App.Writer, "Account %s will be removed. This action is irreversible.\n", addrArg)
if ok := askForConsent(); !ok { if ok := askForConsent(ctx.App.Writer); !ok {
return nil return nil
} }
} }
@ -516,8 +517,8 @@ func removeAccount(ctx *cli.Context) error {
return nil return nil
} }
func askForConsent() bool { func askForConsent(w io.Writer) bool {
fmt.Print("Are you sure? [y/N]: ") fmt.Fprintln(w, "Are you sure? [y/N]: ")
reader := bufio.NewReader(os.Stdin) reader := bufio.NewReader(os.Stdin)
response, err := reader.ReadString('\n') response, err := reader.ReadString('\n')
if err == nil { if err == nil {
@ -526,7 +527,7 @@ func askForConsent() bool {
return true return true
} }
} }
fmt.Println("Cancelled.") fmt.Fprintln(w, "Cancelled.")
return false return false
} }
@ -536,7 +537,7 @@ func dumpWallet(ctx *cli.Context) error {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
if ctx.Bool("decrypt") { if ctx.Bool("decrypt") {
pass, err := readPassword("Enter wallet password > ") pass, err := readPassword(ctx.App.Writer, "Enter wallet password > ")
if err != nil { if err != nil {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
@ -548,7 +549,7 @@ func dumpWallet(ctx *cli.Context) error {
} }
} }
} }
fmtPrintWallet(wall) fmtPrintWallet(ctx.App.Writer, wall)
return nil return nil
} }
@ -571,20 +572,20 @@ func createWallet(ctx *cli.Context) error {
} }
} }
fmtPrintWallet(wall) fmtPrintWallet(ctx.App.Writer, wall)
fmt.Printf("wallet successfully created, file location is %s\n", wall.Path()) fmt.Fprintf(ctx.App.Writer, "wallet successfully created, file location is %s\n", wall.Path())
return nil return nil
} }
func readAccountInfo() (string, string, error) { func readAccountInfo(w io.Writer) (string, string, error) {
buf := bufio.NewReader(os.Stdin) buf := bufio.NewReader(os.Stdin)
fmt.Print("Enter the name of the account > ") fmt.Fprint(w, "Enter the name of the account > ")
rawName, _ := buf.ReadBytes('\n') rawName, _ := buf.ReadBytes('\n')
phrase, err := readPassword("Enter passphrase > ") phrase, err := readPassword(w, "Enter passphrase > ")
if err != nil { if err != nil {
return "", "", err return "", "", err
} }
phraseCheck, err := readPassword("Confirm passphrase > ") phraseCheck, err := readPassword(w, "Confirm passphrase > ")
if err != nil { if err != nil {
return "", "", err return "", "", err
} }
@ -598,7 +599,7 @@ func readAccountInfo() (string, string, error) {
} }
func createAccount(ctx *cli.Context, wall *wallet.Wallet) error { func createAccount(ctx *cli.Context, wall *wallet.Wallet) error {
name, phrase, err := readAccountInfo() name, phrase, err := readAccountInfo(ctx.App.Writer)
if err != nil { if err != nil {
return err return err
} }
@ -612,11 +613,11 @@ func openWallet(path string) (*wallet.Wallet, error) {
return wallet.NewWalletFromFile(path) return wallet.NewWalletFromFile(path)
} }
func newAccountFromWIF(wif string) (*wallet.Account, error) { func newAccountFromWIF(w io.Writer, wif string) (*wallet.Account, error) {
// note: NEP2 strings always have length of 58 even though // note: NEP2 strings always have length of 58 even though
// base58 strings can have different lengths even if slice lengths are equal // base58 strings can have different lengths even if slice lengths are equal
if len(wif) == 58 { if len(wif) == 58 {
pass, err := readPassword("Enter password > ") pass, err := readPassword(w, "Enter password > ")
if err != nil { if err != nil {
return nil, err return nil, err
} }
@ -629,8 +630,8 @@ func newAccountFromWIF(wif string) (*wallet.Account, error) {
return nil, err return nil, err
} }
fmt.Println("Provided WIF was unencrypted. Wallet can contain only encrypted keys.") fmt.Fprintln(w, "Provided WIF was unencrypted. Wallet can contain only encrypted keys.")
name, pass, err := readAccountInfo() name, pass, err := readAccountInfo(w)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@ -654,19 +655,19 @@ func addAccountAndSave(w *wallet.Wallet, acc *wallet.Account) error {
return w.Save() return w.Save()
} }
func readPassword(prompt string) (string, error) { func readPassword(w io.Writer, prompt string) (string, error) {
fmt.Print(prompt) fmt.Fprint(w, prompt)
rawPass, err := terminal.ReadPassword(syscall.Stdin) rawPass, err := terminal.ReadPassword(syscall.Stdin)
fmt.Println() fmt.Fprintln(w)
if err != nil { if err != nil {
return "", err return "", err
} }
return strings.TrimRight(string(rawPass), "\n"), nil return strings.TrimRight(string(rawPass), "\n"), nil
} }
func fmtPrintWallet(wall *wallet.Wallet) { func fmtPrintWallet(w io.Writer, wall *wallet.Wallet) {
b, _ := wall.JSON() b, _ := wall.JSON()
fmt.Println("") fmt.Fprintln(w, "")
fmt.Println(string(b)) fmt.Fprintln(w, string(b))
fmt.Println("") fmt.Fprintln(w, "")
} }