From 84127098b565a654e73b8a41e091c4bfc1e31e58 Mon Sep 17 00:00:00 2001 From: Ononiwu Maureen <59079323+Chinwendu20@users.noreply.github.com> Date: Mon, 13 May 2024 21:11:18 +0100 Subject: [PATCH] wallet: select utxos should not contain duplicates Signed-off-by: Ononiwu Maureen <59079323+Chinwendu20@users.noreply.github.com> --- go.mod | 2 + go.sum | 4 ++ wallet/createtx.go | 6 ++ wallet/createtx_test.go | 133 ++++++++++++++++++++++++++++++---------- 4 files changed, 111 insertions(+), 34 deletions(-) diff --git a/go.mod b/go.mod index 881ceef144..d59c7aa6ba 100644 --- a/go.mod +++ b/go.mod @@ -21,6 +21,7 @@ require ( github.com/lightninglabs/gozmq v0.0.0-20191113021534-d20a764486bf github.com/lightninglabs/neutrino v0.16.0 github.com/lightninglabs/neutrino/cache v1.1.2 + github.com/lightningnetwork/lnd/fn v1.2.5 github.com/lightningnetwork/lnd/ticker v1.0.0 github.com/lightningnetwork/lnd/tlv v1.0.2 github.com/stretchr/testify v1.9.0 @@ -45,6 +46,7 @@ require ( github.com/rogpeppe/go-internal v1.12.0 // indirect github.com/stretchr/objx v0.5.2 // indirect go.etcd.io/bbolt v1.3.11 // indirect + golang.org/x/exp v0.0.0-20231226003508-02704c960a9b // indirect golang.org/x/sys v0.19.0 // indirect golang.org/x/text v0.14.0 // indirect google.golang.org/genproto/googleapis/rpc v0.0.0-20231030173426-d783a09b4405 // indirect diff --git a/go.sum b/go.sum index 23d0b1abc6..3320f59a81 100644 --- a/go.sum +++ b/go.sum @@ -103,6 +103,8 @@ github.com/lightninglabs/neutrino/cache v1.1.2 h1:C9DY/DAPaPxbFC+xNNEI/z1SJY9GS3 github.com/lightninglabs/neutrino/cache v1.1.2/go.mod h1:XJNcgdOw1LQnanGjw8Vj44CvguYA25IMKjWFZczwZuo= github.com/lightningnetwork/lnd/clock v1.0.1 h1:QQod8+m3KgqHdvVMV+2DRNNZS1GRFir8mHZYA+Z2hFo= github.com/lightningnetwork/lnd/clock v1.0.1/go.mod h1:KnQudQ6w0IAMZi1SgvecLZQZ43ra2vpDNj7H/aasemg= +github.com/lightningnetwork/lnd/fn v1.2.5 h1:pGMz0BDUxrhvOtShD4FIysdVy+ulfFAnFvTKjZO5Pp8= +github.com/lightningnetwork/lnd/fn v1.2.5/go.mod h1:SyFohpVrARPKH3XVAJZlXdVe+IwMYc4OMAvrDY32kw0= github.com/lightningnetwork/lnd/queue v1.0.1 h1:jzJKcTy3Nj5lQrooJ3aaw9Lau3I0IwvQR5sqtjdv2R0= github.com/lightningnetwork/lnd/queue v1.0.1/go.mod h1:vaQwexir73flPW43Mrm7JOgJHmcEFBWWSl9HlyASoms= github.com/lightningnetwork/lnd/ticker v1.0.0 h1:S1b60TEGoTtCe2A0yeB+ecoj/kkS4qpwh6l+AkQEZwU= @@ -138,6 +140,8 @@ golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACk golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= golang.org/x/crypto v0.22.0 h1:g1v0xeRhjcugydODzvb3mEM9SQ0HGp9s/nh3COQ/C30= golang.org/x/crypto v0.22.0/go.mod h1:vr6Su+7cTlO45qkww3VDJlzDn0ctJvRgYbC2NvXHt+M= +golang.org/x/exp v0.0.0-20231226003508-02704c960a9b h1:kLiC65FbiHWFAOu+lxwNPujcsl8VYyTYYEZnsOO1WK4= +golang.org/x/exp v0.0.0-20231226003508-02704c960a9b/go.mod h1:iRJReGqOEeBhDZGkGbynYwcHlctCvnjTYIamk7uXpHI= golang.org/x/net v0.0.0-20180719180050-a680a1efc54d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= diff --git a/wallet/createtx.go b/wallet/createtx.go index 3fe133e16f..a7a8aecc20 100644 --- a/wallet/createtx.go +++ b/wallet/createtx.go @@ -20,6 +20,7 @@ import ( "github.com/btcsuite/btcwallet/wallet/txsizes" "github.com/btcsuite/btcwallet/walletdb" "github.com/btcsuite/btcwallet/wtxmgr" + "github.com/lightningnetwork/lnd/fn" ) func makeInputSource(eligible []Coin) txauthor.InputSource { @@ -191,6 +192,11 @@ func (w *Wallet) txToOutputs(outputs []*wire.TxOut, var inputSource txauthor.InputSource if len(selectedUtxos) > 0 { + dedupUtxos := fn.NewSet(selectedUtxos...) + if len(dedupUtxos) != len(selectedUtxos) { + return errors.New("selected UTXOs contain " + + "duplicate values") + } eligibleByOutpoint := make( map[wire.OutPoint]wtxmgr.Credit, ) diff --git a/wallet/createtx_test.go b/wallet/createtx_test.go index c0badf35d7..417a3a296c 100644 --- a/wallet/createtx_test.go +++ b/wallet/createtx_test.go @@ -441,50 +441,115 @@ func TestSelectUtxosTxoToOutpoint(t *testing.T) { } addUtxo(t, w, incomingTx) - // We expect 4 unspent utxos. + // We expect 4 unspent UTXOs. unspent, err := w.ListUnspent(0, 80, "") - require.NoError(t, err, "unexpected error while calling "+ - "list unspent") - - require.Len(t, unspent, 4, "expected 4 unspent "+ - "utxos") + require.NoError(t, err) + require.Len(t, unspent, 4, "expected 4 unspent UTXOs") - selectUtxos := []wire.OutPoint{ + tCases := []struct { + name string + selectUTXOs []wire.OutPoint + errString string + }{ + { + name: "Duplicate utxo values", + selectUTXOs: []wire.OutPoint{ + { + Hash: incomingTx.TxHash(), + Index: 1, + }, + { + Hash: incomingTx.TxHash(), + Index: 1, + }, + }, + errString: "selected UTXOs contain duplicate values", + }, + { + name: "all selected UTXOs not eligible for spending", + selectUTXOs: []wire.OutPoint{ + { + Hash: chainhash.Hash([32]byte{1}), + Index: 1, + }, + { + Hash: chainhash.Hash([32]byte{3}), + Index: 1, + }, + }, + errString: "selected outpoint not eligible for " + + "spending", + }, { - Hash: incomingTx.TxHash(), - Index: 1, + name: "some select UTXOs not eligible for spending", + selectUTXOs: []wire.OutPoint{ + { + Hash: chainhash.Hash([32]byte{1}), + Index: 1, + }, + { + Hash: incomingTx.TxHash(), + Index: 1, + }, + }, + errString: "selected outpoint not eligible for " + + "spending", }, { - Hash: incomingTx.TxHash(), - Index: 2, + name: "select utxo, no duplicates and all eligible " + + "for spending", + selectUTXOs: []wire.OutPoint{ + { + Hash: incomingTx.TxHash(), + Index: 1, + }, + { + Hash: incomingTx.TxHash(), + Index: 2, + }, + }, }, } - // Test by sending 200_000. - targetTxOut := &wire.TxOut{ - Value: 200_000, - PkScript: p2trScript, - } - tx1, err := w.txToOutputs( - []*wire.TxOut{targetTxOut}, nil, nil, 0, 1, 1000, - CoinSelectionLargest, true, selectUtxos, alwaysAllowUtxo, - ) - require.NoError(t, err) + for _, tc := range tCases { + t.Run(tc.name, func(t *testing.T) { + // Test by sending 200_000. + targetTxOut := &wire.TxOut{ + Value: 200_000, + PkScript: p2trScript, + } + tx1, err := w.txToOutputs( + []*wire.TxOut{targetTxOut}, nil, nil, 0, 1, + 1000, CoinSelectionLargest, true, + tc.selectUTXOs, alwaysAllowUtxo, + ) + if tc.errString != "" { + require.ErrorContains(t, err, tc.errString) + require.Nil(t, tx1) - // We expect all and only our select utxos to be input in this - // transaction. - require.Len(t, tx1.Tx.TxIn, len(selectUtxos)) + return + } - lookupSelectUtxos := make(map[wire.OutPoint]struct{}) - for _, utxo := range selectUtxos { - lookupSelectUtxos[utxo] = struct{}{} - } + require.NoError(t, err) + require.NotNil(t, tx1) - for _, tx := range tx1.Tx.TxIn { - _, ok := lookupSelectUtxos[tx.PreviousOutPoint] - require.True(t, ok, "unexpected outpoint in txin") - } + // We expect all and only our select UTXOs to be input + // in this transaction. + require.Len(t, tx1.Tx.TxIn, len(tc.selectUTXOs)) - // Expect two outputs, change and the actual payment to the address. - require.Len(t, tx1.Tx.TxOut, 2) + lookupSelectUtxos := make(map[wire.OutPoint]struct{}) + for _, utxo := range tc.selectUTXOs { + lookupSelectUtxos[utxo] = struct{}{} + } + + for _, tx := range tx1.Tx.TxIn { + _, ok := lookupSelectUtxos[tx.PreviousOutPoint] + require.True(t, ok) + } + + // Expect two outputs, change and the actual payment to + // the address. + require.Len(t, tx1.Tx.TxOut, 2) + }) + } }