rpcserver: check for compatible network in SendCoins

lnd_test: adding address validation for send coins

The commit adds a test that checks that when a user calls sendcoins, the
receiving address is validated according to the current network. If the
address is not compatible with the current network, it will return an
error to the user.

rpcserver: adding a check for compatible network in SendCoins

This commit adds a check in SendCoins that checks whether the receiving
address is compatible with the current network.

Fixes #2677.
This commit is contained in:
ccdle12 2019-02-27 19:16:34 +00:00 committed by Olaoluwa Osuntokun
parent 341f5e4329
commit abfbdf6aec
2 changed files with 46 additions and 13 deletions

@ -13168,12 +13168,40 @@ func testSweepAllCoins(net *lntest.NetworkHarness, t *harnessTest) {
// With the two coins above mined, we'll now instruct ainz to sweep all // With the two coins above mined, we'll now instruct ainz to sweep all
// the coins to an external address not under its control. // the coins to an external address not under its control.
// We will first attempt to send the coins to addresses that are not
// compatible with the current network. This is to test that the wallet
// will prevent any onchain transactions to addresses that are not on the
// same network as the user.
// Send coins to a testnet3 address.
ctxt, _ = context.WithTimeout(ctxb, defaultTimeout)
sweepReq := &lnrpc.SendCoinsRequest{
Addr: "tb1qfc8fusa98jx8uvnhzavxccqlzvg749tvjw82tg",
SendAll: true,
}
_, err = ainz.SendCoins(ctxt, sweepReq)
if err == nil {
t.Fatalf("expected SendCoins to different network to fail")
}
// Send coins to a mainnet address.
ctxt, _ = context.WithTimeout(ctxb, defaultTimeout)
sweepReq = &lnrpc.SendCoinsRequest{
Addr: "1MPaXKp5HhsLNjVSqaL7fChE3TVyrTMRT3",
SendAll: true,
}
_, err = ainz.SendCoins(ctxt, sweepReq)
if err == nil {
t.Fatalf("expected SendCoins to different network to fail")
}
// Send coins to a compatible address.
minerAddr, err := net.Miner.NewAddress() minerAddr, err := net.Miner.NewAddress()
if err != nil { if err != nil {
t.Fatalf("unable to create new miner addr: %v", err) t.Fatalf("unable to create new miner addr: %v", err)
} }
sweepReq := &lnrpc.SendCoinsRequest{ sweepReq = &lnrpc.SendCoinsRequest{
Addr: minerAddr.String(), Addr: minerAddr.String(),
SendAll: true, SendAll: true,
} }
@ -13183,7 +13211,7 @@ func testSweepAllCoins(net *lntest.NetworkHarness, t *harnessTest) {
t.Fatalf("unable to sweep coins: %v", err) t.Fatalf("unable to sweep coins: %v", err)
} }
// We'll mine a block wish should include the sweep transaction we // We'll mine a block which should include the sweep transaction we
// generated above. // generated above.
block := mineBlocks(t, net, 1, 1)[0] block := mineBlocks(t, net, 1, 1)[0]

@ -880,6 +880,20 @@ func (r *rpcServer) SendCoins(ctx context.Context,
in.Addr, btcutil.Amount(in.Amount), int64(feePerKw), in.Addr, btcutil.Amount(in.Amount), int64(feePerKw),
in.SendAll) in.SendAll)
// Decode the address receiving the coins, we need to check whether the
// address is valid for this network.
targetAddr, err := btcutil.DecodeAddress(in.Addr, activeNetParams.Params)
if err != nil {
return nil, err
}
// Make the check on the decoded address according to the active network.
if !targetAddr.IsForNet(activeNetParams.Params) {
return nil, fmt.Errorf("address: %v is not valid for this "+
"network: %v", targetAddr.String(),
activeNetParams.Params.Name)
}
var txid *chainhash.Hash var txid *chainhash.Hash
wallet := r.server.cc.wallet wallet := r.server.cc.wallet
@ -896,15 +910,6 @@ func (r *rpcServer) SendCoins(ctx context.Context,
"active") "active")
} }
// Additionally, we'll need to convert the sweep address passed
// into a useable struct, and also query for the latest block
// height so we can properly construct the transaction.
sweepAddr, err := btcutil.DecodeAddress(
in.Addr, activeNetParams.Params,
)
if err != nil {
return nil, err
}
_, bestHeight, err := r.server.cc.chainIO.GetBestBlock() _, bestHeight, err := r.server.cc.chainIO.GetBestBlock()
if err != nil { if err != nil {
return nil, err return nil, err
@ -915,7 +920,7 @@ func (r *rpcServer) SendCoins(ctx context.Context,
// single transaction. This will be generated in a concurrent // single transaction. This will be generated in a concurrent
// safe manner, so no need to worry about locking. // safe manner, so no need to worry about locking.
sweepTxPkg, err := sweep.CraftSweepAllTx( sweepTxPkg, err := sweep.CraftSweepAllTx(
feePerKw, uint32(bestHeight), sweepAddr, wallet, feePerKw, uint32(bestHeight), targetAddr, wallet,
wallet.WalletController, wallet.WalletController, wallet.WalletController, wallet.WalletController,
r.server.cc.feeEstimator, r.server.cc.signer, r.server.cc.feeEstimator, r.server.cc.signer,
) )
@ -945,7 +950,7 @@ func (r *rpcServer) SendCoins(ctx context.Context,
// coin selection synchronization method to ensure that no coin // coin selection synchronization method to ensure that no coin
// selection (funding, sweep alls, other sends) can proceed // selection (funding, sweep alls, other sends) can proceed
// while we instruct the wallet to send this transaction. // while we instruct the wallet to send this transaction.
paymentMap := map[string]int64{in.Addr: in.Amount} paymentMap := map[string]int64{targetAddr.String(): in.Amount}
err := wallet.WithCoinSelectLock(func() error { err := wallet.WithCoinSelectLock(func() error {
newTXID, err := r.sendCoinsOnChain(paymentMap, feePerKw) newTXID, err := r.sendCoinsOnChain(paymentMap, feePerKw)
if err != nil { if err != nil {