From fdfc10aba1d8f838e791392d79c8171d8b21c258 Mon Sep 17 00:00:00 2001 From: Olaoluwa Osuntokun Date: Wed, 24 Jan 2024 16:58:06 -0800 Subject: [PATCH] Merge pull request #900 from guggero/export-coin-selection wallet: export coin selection strategy code for re-use --- chain/bitcoind_client.go | 2 +- go.mod | 1 - go.sum | 1 - wallet/createtx.go | 158 ++++++++++++++++++++++++++------------- wallet/createtx_test.go | 4 +- wallet/psbt.go | 110 ++++++++++++++++----------- wallet/wallet.go | 24 +++++- 7 files changed, 195 insertions(+), 105 deletions(-) diff --git a/chain/bitcoind_client.go b/chain/bitcoind_client.go index bc401ccd52..fbb95e2f1c 100644 --- a/chain/bitcoind_client.go +++ b/chain/bitcoind_client.go @@ -191,7 +191,7 @@ func (c *BitcoindClient) GetRawTransactionVerbose( return c.chainConn.client.GetRawTransactionVerbose(hash) } -// GetRawTransaction returns a `btcutil.Tx` from the tx hash. +// GetRawTransaction returns a `ltcutil.Tx` from the tx hash. func (c *BitcoindClient) GetRawTransaction( hash *chainhash.Hash) (*ltcutil.Tx, error) { diff --git a/go.mod b/go.mod index 79a16c1528..091f954197 100644 --- a/go.mod +++ b/go.mod @@ -23,7 +23,6 @@ require ( go.etcd.io/bbolt v1.3.5 golang.org/x/crypto v0.7.0 golang.org/x/net v0.8.0 - golang.org/x/sync v0.0.0-20210220032951-036812b2e83c golang.org/x/term v0.6.0 google.golang.org/grpc v1.38.0 ) diff --git a/go.sum b/go.sum index 86dd53a510..6ad6db4944 100644 --- a/go.sum +++ b/go.sum @@ -982,7 +982,6 @@ golang.org/x/sync v0.0.0-20200317015054-43a5402ce75a/go.mod h1:RxMgew5VJxzue5/jJ golang.org/x/sync v0.0.0-20200625203802-6e8e738ad208/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20201207232520-09787c993a3a/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= -golang.org/x/sync v0.0.0-20210220032951-036812b2e83c h1:5KslGYwFpkhGh+Q16bwMP3cOontH8FOep7tGV86Y7SQ= golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sys v0.0.0-20171026204733-164713f0dfce/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20180823144017-11551d06cbcc/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= diff --git a/wallet/createtx.go b/wallet/createtx.go index 95e4f685b5..289e634e0b 100644 --- a/wallet/createtx.go +++ b/wallet/createtx.go @@ -6,6 +6,7 @@ package wallet import ( + "errors" "fmt" "math/rand" "sort" @@ -21,16 +22,8 @@ import ( "github.com/ltcsuite/ltcd/wire" ) -// byAmount defines the methods needed to satisify sort.Interface to -// sort credits by their output amount. -type byAmount []wtxmgr.Credit - -func (s byAmount) Len() int { return len(s) } -func (s byAmount) Less(i, j int) bool { return s[i].Amount < s[j].Amount } -func (s byAmount) Swap(i, j int) { s[i], s[j] = s[j], s[i] } - -func makeInputSource(eligible []wtxmgr.Credit) txauthor.InputSource { - // Current inputs and their total value. These are closed over by the +func makeInputSource(eligible []Coin) txauthor.InputSource { + // Current inputs and their total value. These are closed over by the // returned input source and reused across multiple calls. currentTotal := ltcutil.Amount(0) currentInputs := make([]*wire.TxIn, 0, len(eligible)) @@ -41,15 +34,25 @@ func makeInputSource(eligible []wtxmgr.Credit) txauthor.InputSource { []ltcutil.Amount, [][]byte, error) { for currentTotal < target && len(eligible) != 0 { - nextCredit := &eligible[0] + nextCredit := eligible[0] + prevOut := nextCredit.TxOut + outpoint := nextCredit.OutPoint eligible = eligible[1:] - nextInput := wire.NewTxIn(&nextCredit.OutPoint, nil, nil) - currentTotal += nextCredit.Amount + + nextInput := wire.NewTxIn(&outpoint, nil, nil) + currentTotal += ltcutil.Amount(prevOut.Value) currentInputs = append(currentInputs, nextInput) - currentScripts = append(currentScripts, nextCredit.PkScript) - currentInputValues = append(currentInputValues, nextCredit.Amount) + currentScripts = append( + currentScripts, prevOut.PkScript, + ) + currentInputValues = append( + currentInputValues, + ltcutil.Amount(prevOut.Value), + ) } - return currentTotal, currentInputs, currentInputValues, currentScripts, nil + + return currentTotal, currentInputs, currentInputValues, + currentScripts, nil } } @@ -123,6 +126,11 @@ func (w *Wallet) txToOutputs(outputs []*wire.TxOut, return nil, err } + // Fall back to default coin selection strategy if none is supplied. + if coinSelectionStrategy == nil { + coinSelectionStrategy = CoinSelectionLargest + } + var tx *txauthor.AuthoredTx err = walletdb.Update(w.db, func(dbtx walletdb.ReadWriteTx) error { addrmgrNs, changeSource, err := w.addrMgrWithChangeSource( @@ -139,40 +147,27 @@ func (w *Wallet) txToOutputs(outputs []*wire.TxOut, return err } - var inputSource txauthor.InputSource - - switch coinSelectionStrategy { - // Pick largest outputs first. - case CoinSelectionLargest: - sort.Sort(sort.Reverse(byAmount(eligible))) - inputSource = makeInputSource(eligible) - - // Select coins at random. This prevents the creation of ever - // smaller utxos over time that may never become economical to - // spend. - case CoinSelectionRandom: - // Skip inputs that do not raise the total transaction - // output value at the requested fee rate. - var positivelyYielding []wtxmgr.Credit - for _, output := range eligible { - output := output - - if !inputYieldsPositively(&output, feeSatPerKb) { - continue - } - - positivelyYielding = append( - positivelyYielding, output, - ) + // Wrap our coins in a type that implements the SelectableCoin + // interface, so we can arrange them according to the selected + // coin selection strategy. + wrappedEligible := make([]Coin, len(eligible)) + for i := range eligible { + wrappedEligible[i] = Coin{ + TxOut: wire.TxOut{ + Value: int64(eligible[i].Amount), + PkScript: eligible[i].PkScript, + }, + OutPoint: eligible[i].OutPoint, } - - rand.Shuffle(len(positivelyYielding), func(i, j int) { - positivelyYielding[i], positivelyYielding[j] = - positivelyYielding[j], positivelyYielding[i] - }) - - inputSource = makeInputSource(positivelyYielding) } + arrangedCoins, err := coinSelectionStrategy.ArrangeCoins( + wrappedEligible, feeSatPerKb, + ) + if err != nil { + return err + } + + inputSource := makeInputSource(arrangedCoins) tx, err = txauthor.NewUnsignedTransaction( outputs, feeSatPerKb, inputSource, changeSource, @@ -261,7 +256,7 @@ func (w *Wallet) txToOutputs(outputs []*wire.TxOut, return nil }) - if err != nil && err != walletdb.ErrDryRunRollBack { + if err != nil && !errors.Is(err, walletdb.ErrDryRunRollBack) { return nil, err } @@ -290,7 +285,7 @@ func (w *Wallet) findEligibleOutputs(dbtx walletdb.ReadTx, output := &unspent[i] // Only include this output if it meets the required number of - // confirmations. Coinbase transactions must have have reached + // confirmations. Coinbase transactions must have reached // maturity before their outputs may be spent. if !confirmed(minconf, output.Height, bs.Height) { continue @@ -337,11 +332,13 @@ func (w *Wallet) findEligibleOutputs(dbtx walletdb.ReadTx, // best-case added virtual size. For edge cases this function can return true // while the input is yielding slightly negative as part of the final // transaction. -func inputYieldsPositively(credit *wtxmgr.Credit, feeRatePerKb ltcutil.Amount) bool { +func inputYieldsPositively(credit *wire.TxOut, + feeRatePerKb ltcutil.Amount) bool { + inputSize := txsizes.GetMinInputVirtualSize(credit.PkScript) inputFee := feeRatePerKb * ltcutil.Amount(inputSize) / 1000 - return inputFee < credit.Amount + return inputFee < ltcutil.Amount(credit.Value) } // addrMgrWithChangeSource returns the address manager bucket and a change @@ -386,6 +383,9 @@ func (w *Wallet) addrMgrWithChangeSource(dbtx walletdb.ReadWriteTx, scriptSize = txsizes.P2WPKHPkScriptSize case waddrmgr.TaprootPubKey: scriptSize = txsizes.P2TRPkScriptSize + default: + return nil, nil, fmt.Errorf("unsupported address type: %v", + addrType) } newChangeScript := func() ([]byte, error) { @@ -440,3 +440,57 @@ func validateMsgTx(tx *wire.MsgTx, prevScripts [][]byte, inputValues []ltcutil.A } return nil } + +// sortByAmount is a generic sortable type for sorting coins by their amount. +type sortByAmount []Coin + +func (s sortByAmount) Len() int { return len(s) } +func (s sortByAmount) Less(i, j int) bool { + return s[i].Value < s[j].Value +} +func (s sortByAmount) Swap(i, j int) { s[i], s[j] = s[j], s[i] } + +// LargestFirstCoinSelector is an implementation of the CoinSelectionStrategy +// that always selects the largest coins first. +type LargestFirstCoinSelector struct{} + +// ArrangeCoins takes a list of coins and arranges them according to the +// specified coin selection strategy and fee rate. +func (*LargestFirstCoinSelector) ArrangeCoins(eligible []Coin, + _ ltcutil.Amount) ([]Coin, error) { + + sort.Sort(sort.Reverse(sortByAmount(eligible))) + + return eligible, nil +} + +// RandomCoinSelector is an implementation of the CoinSelectionStrategy that +// selects coins at random. This prevents the creation of ever smaller UTXOs +// over time that may never become economical to spend. +type RandomCoinSelector struct{} + +// ArrangeCoins takes a list of coins and arranges them according to the +// specified coin selection strategy and fee rate. +func (*RandomCoinSelector) ArrangeCoins(eligible []Coin, + feeSatPerKb ltcutil.Amount) ([]Coin, error) { + + // Skip inputs that do not raise the total transaction output + // value at the requested fee rate. + positivelyYielding := make([]Coin, 0, len(eligible)) + for _, output := range eligible { + output := output + + if !inputYieldsPositively(&output.TxOut, feeSatPerKb) { + continue + } + + positivelyYielding = append(positivelyYielding, output) + } + + rand.Shuffle(len(positivelyYielding), func(i, j int) { + positivelyYielding[i], positivelyYielding[j] = + positivelyYielding[j], positivelyYielding[i] + }) + + return positivelyYielding, nil +} diff --git a/wallet/createtx_test.go b/wallet/createtx_test.go index 4ac80170c5..963f1e6695 100644 --- a/wallet/createtx_test.go +++ b/wallet/createtx_test.go @@ -216,8 +216,8 @@ func TestInputYield(t *testing.T) { pkScript, err := txscript.PayToAddrScript(addr) require.NoError(t, err) - credit := &wtxmgr.Credit{ - Amount: 1000, + credit := &wire.TxOut{ + Value: 1000, PkScript: pkScript, } diff --git a/wallet/psbt.go b/wallet/psbt.go index e039ecc07d..dde4186bac 100644 --- a/wallet/psbt.go +++ b/wallet/psbt.go @@ -6,6 +6,7 @@ package wallet import ( "bytes" + "errors" "fmt" "github.com/dcrlabs/ltcwallet/waddrmgr" @@ -82,48 +83,6 @@ func (w *Wallet) FundPsbt(packet *psbt.Packet, keyScope *waddrmgr.KeyScope, amt += output.Value } - // addInputInfo is a helper function that fetches the UTXO information - // of an input and attaches it to the PSBT packet. - addInputInfo := func(inputs []*wire.TxIn) error { - packet.Inputs = make([]psbt.PInput, len(inputs)) - for idx, in := range inputs { - tx, utxo, derivationPath, _, err := w.FetchInputInfo( - &in.PreviousOutPoint, - ) - if err != nil { - return fmt.Errorf("error fetching UTXO: %v", - err) - } - - addr, witnessProgram, _, err := w.ScriptForOutput(utxo) - if err != nil { - return fmt.Errorf("error fetching UTXO "+ - "script: %v", err) - } - - // We don't want to include the witness or any script - // on the unsigned TX just yet. - packet.UnsignedTx.TxIn[idx].Witness = wire.TxWitness{} - packet.UnsignedTx.TxIn[idx].SignatureScript = nil - - switch { - case txscript.IsPayToTaproot(utxo.PkScript): - addInputInfoSegWitV1( - &packet.Inputs[idx], utxo, - derivationPath, - ) - - default: - addInputInfoSegWitV0( - &packet.Inputs[idx], tx, utxo, - derivationPath, addr, witnessProgram, - ) - } - } - - return nil - } - var tx *txauthor.AuthoredTx switch { // We need to do coin selection. @@ -146,7 +105,16 @@ func (w *Wallet) FundPsbt(packet *psbt.Packet, keyScope *waddrmgr.KeyScope, // include the witness as the resulting PSBT isn't expected not // should be signed yet. packet.UnsignedTx.TxIn = tx.Tx.TxIn - err = addInputInfo(tx.Tx.TxIn) + packet.Inputs = make([]psbt.PInput, len(packet.UnsignedTx.TxIn)) + + for idx := range packet.UnsignedTx.TxIn { + // We don't want to include the witness or any script + // on the unsigned TX just yet. + packet.UnsignedTx.TxIn[idx].Witness = wire.TxWitness{} + packet.UnsignedTx.TxIn[idx].SignatureScript = nil + } + + err := w.DecorateInputs(packet, true) if err != nil { return 0, err } @@ -155,7 +123,16 @@ func (w *Wallet) FundPsbt(packet *psbt.Packet, keyScope *waddrmgr.KeyScope, // a change output if necessary. default: // Make sure all inputs provided are actually ours. - err = addInputInfo(txIn) + packet.Inputs = make([]psbt.PInput, len(packet.UnsignedTx.TxIn)) + + for idx := range packet.UnsignedTx.TxIn { + // We don't want to include the witness or any script + // on the unsigned TX just yet. + packet.UnsignedTx.TxIn[idx].Witness = wire.TxWitness{} + packet.UnsignedTx.TxIn[idx].SignatureScript = nil + } + + err := w.DecorateInputs(packet, true) if err != nil { return 0, err } @@ -265,6 +242,51 @@ func (w *Wallet) FundPsbt(packet *psbt.Packet, keyScope *waddrmgr.KeyScope, return changeIndex, nil } +// DecorateInputs fetches the UTXO information of all inputs it can identify and +// adds the required information to the package's inputs. The failOnUnknown +// boolean controls whether the method should return an error if it cannot +// identify an input or if it should just skip it. +func (w *Wallet) DecorateInputs(packet *psbt.Packet, failOnUnknown bool) error { + for idx := range packet.Inputs { + txIn := packet.UnsignedTx.TxIn[idx] + + tx, utxo, derivationPath, _, err := w.FetchInputInfo( + &txIn.PreviousOutPoint, + ) + + switch { + // If the error just means it's not an input our wallet controls + // and the user doesn't care about that, then we can just skip + // this input and continue. + case errors.Is(err, ErrNotMine) && !failOnUnknown: + continue + + case err != nil: + return fmt.Errorf("error fetching UTXO: %v", err) + } + + addr, witnessProgram, _, err := w.ScriptForOutput(utxo) + if err != nil { + return fmt.Errorf("error fetching UTXO script: %v", err) + } + + switch { + case txscript.IsPayToTaproot(utxo.PkScript): + addInputInfoSegWitV1( + &packet.Inputs[idx], utxo, derivationPath, + ) + + default: + addInputInfoSegWitV0( + &packet.Inputs[idx], tx, utxo, derivationPath, + addr, witnessProgram, + ) + } + } + + return nil +} + // addInputInfoSegWitV0 adds the UTXO and BIP32 derivation info for a SegWit v0 // PSBT input (p2wkh, np2wkh) from the given wallet information. func addInputInfoSegWitV0(in *psbt.PInput, prevTx *wire.MsgTx, utxo *wire.TxOut, diff --git a/wallet/wallet.go b/wallet/wallet.go index c4fdc05b3c..61ce3d8d88 100644 --- a/wallet/wallet.go +++ b/wallet/wallet.go @@ -89,17 +89,33 @@ var ( wtxmgrNamespaceKey = []byte("wtxmgr") ) -type CoinSelectionStrategy int +// Coin represents a spendable UTXO which is available for coin selection. +type Coin struct { + wire.TxOut -const ( + wire.OutPoint +} + +// CoinSelectionStrategy is an interface that represents a coin selection +// strategy. A coin selection strategy is responsible for ordering, shuffling or +// filtering a list of coins before they are passed to the coin selection +// algorithm. +type CoinSelectionStrategy interface { + // ArrangeCoins takes a list of coins and arranges them according to the + // specified coin selection strategy and fee rate. + ArrangeCoins(eligible []Coin, feeSatPerKb ltcutil.Amount) ([]Coin, + error) +} + +var ( // CoinSelectionLargest always picks the largest available utxo to add // to the transaction next. - CoinSelectionLargest CoinSelectionStrategy = iota + CoinSelectionLargest CoinSelectionStrategy = &LargestFirstCoinSelector{} // CoinSelectionRandom randomly selects the next utxo to add to the // transaction. This strategy prevents the creation of ever smaller // utxos over time. - CoinSelectionRandom + CoinSelectionRandom CoinSelectionStrategy = &RandomCoinSelector{} ) // Wallet is a structure containing all the components for a