lncli: add commands "create" and "unlock"

lncli create:
This command is used to set up a wallet encryption password for
use with lnd at first time use. It will ask fot the user to
confirm the chosen password, then do a call to the lnd RPC method
CreateWallet with the chosen password.

lncli unlock:
This command is used to unlock the wallet of a running lnd instance.
It calls the RPC method UnlockWallet with the provided password.

Both methods makes use of the terminal.ReadPassword method, to
securely read a password from user input without making it
replayable in the terminal.
This commit is contained in:
Johan T. Halseth 2017-10-12 11:42:06 +02:00 committed by Olaoluwa Osuntokun
parent 2c2d18ee10
commit 00ee3afab8
2 changed files with 82 additions and 0 deletions

@ -20,6 +20,7 @@ import (
"github.com/roasbeef/btcd/chaincfg/chainhash" "github.com/roasbeef/btcd/chaincfg/chainhash"
"github.com/roasbeef/btcutil" "github.com/roasbeef/btcutil"
"github.com/urfave/cli" "github.com/urfave/cli"
"golang.org/x/crypto/ssh/terminal"
"golang.org/x/net/context" "golang.org/x/net/context"
) )
@ -600,6 +601,75 @@ func listPeers(ctx *cli.Context) error {
return nil return nil
} }
var createCommand = cli.Command{
Name: "create",
Usage: "used to set the wallet password at lnd startup",
Action: create,
}
func create(ctx *cli.Context) error {
ctxb := context.Background()
client, cleanUp := getWalletUnlockerClient(ctx)
defer cleanUp()
fmt.Printf("Input wallet password: ")
pw1, err := terminal.ReadPassword(0)
if err != nil {
return err
}
fmt.Println()
fmt.Printf("Confirm wallet password: ")
pw2, err := terminal.ReadPassword(0)
if err != nil {
return err
}
fmt.Println()
if !bytes.Equal(pw1, pw2) {
return fmt.Errorf("passwords don't match")
}
req := &lnrpc.CreateWalletRequest{
Password: pw1,
}
_, err = client.CreateWallet(ctxb, req)
if err != nil {
return err
}
return nil
}
var unlockCommand = cli.Command{
Name: "unlock",
Usage: "unlock encrypted wallet at lnd startup",
Action: unlock,
}
func unlock(ctx *cli.Context) error {
ctxb := context.Background()
client, cleanUp := getWalletUnlockerClient(ctx)
defer cleanUp()
fmt.Printf("Input wallet password: ")
pw, err := terminal.ReadPassword(0)
if err != nil {
return err
}
fmt.Println()
req := &lnrpc.UnlockWalletRequest{
Password: pw,
}
_, err = client.UnlockWallet(ctxb, req)
if err != nil {
return err
}
return nil
}
var walletBalanceCommand = cli.Command{ var walletBalanceCommand = cli.Command{
Name: "walletbalance", Name: "walletbalance",
Usage: "compute and display the wallet's current balance", Usage: "compute and display the wallet's current balance",

@ -34,6 +34,16 @@ func fatal(err error) {
os.Exit(1) os.Exit(1)
} }
func getWalletUnlockerClient(ctx *cli.Context) (lnrpc.WalletUnlockerClient, func()) {
conn := getClientConn(ctx)
cleanUp := func() {
conn.Close()
}
return lnrpc.NewWalletUnlockerClient(conn), cleanUp
}
func getClient(ctx *cli.Context) (lnrpc.LightningClient, func()) { func getClient(ctx *cli.Context) (lnrpc.LightningClient, func()) {
conn := getClientConn(ctx) conn := getClientConn(ctx)
@ -146,6 +156,8 @@ func main() {
}, },
} }
app.Commands = []cli.Command{ app.Commands = []cli.Command{
createCommand,
unlockCommand,
newAddressCommand, newAddressCommand,
sendManyCommand, sendManyCommand,
sendCoinsCommand, sendCoinsCommand,