lnwallet: add CheckReservedValue
This commit is contained in:
parent
4a4e0c73f7
commit
3cc31ae841
@ -5,6 +5,7 @@ import (
|
|||||||
"crypto/sha256"
|
"crypto/sha256"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"math"
|
||||||
"net"
|
"net"
|
||||||
"sync"
|
"sync"
|
||||||
"sync/atomic"
|
"sync/atomic"
|
||||||
@ -32,6 +33,12 @@ const (
|
|||||||
// The size of the buffered queue of requests to the wallet from the
|
// The size of the buffered queue of requests to the wallet from the
|
||||||
// outside word.
|
// outside word.
|
||||||
msgBufferSize = 100
|
msgBufferSize = 100
|
||||||
|
|
||||||
|
// anchorChanReservedValue is the amount we'll keep around in the
|
||||||
|
// wallet in case we have to fee bump anchor channels on force close.
|
||||||
|
// TODO(halseth): update constant to target a specific commit size at
|
||||||
|
// set fee rate.
|
||||||
|
anchorChanReservedValue = btcutil.Amount(10_000)
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
@ -39,6 +46,12 @@ var (
|
|||||||
// contribution handling process if the process should be paused for
|
// contribution handling process if the process should be paused for
|
||||||
// the construction of a PSBT outside of lnd's wallet.
|
// the construction of a PSBT outside of lnd's wallet.
|
||||||
ErrPsbtFundingRequired = errors.New("PSBT funding required")
|
ErrPsbtFundingRequired = errors.New("PSBT funding required")
|
||||||
|
|
||||||
|
// ErrReservedValueInvalidated is returned if we try to publish a
|
||||||
|
// transaction that would take the walletbalance below what we require
|
||||||
|
// to keep around to fee bump our open anchor channels.
|
||||||
|
ErrReservedValueInvalidated = errors.New("reserved wallet balance " +
|
||||||
|
"invalidated")
|
||||||
)
|
)
|
||||||
|
|
||||||
// PsbtFundingRequired is a type that implements the error interface and
|
// PsbtFundingRequired is a type that implements the error interface and
|
||||||
@ -757,6 +770,87 @@ func (l *LightningWallet) handleFundingReserveRequest(req *InitFundingReserveMsg
|
|||||||
req.err <- nil
|
req.err <- nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// CheckReservedValue checks whether publishing a transaction with the given
|
||||||
|
// inputs and outputs would violate the value we reserve in the wallet for
|
||||||
|
// bumping the fee of anchor channels. The numAnchorChans argument should be
|
||||||
|
// set the the number of open anchor channels controlled by the wallet after
|
||||||
|
// the transaction has been published.
|
||||||
|
//
|
||||||
|
// If the reserved value is violated, the returned error will be
|
||||||
|
// ErrReservedValueInvalidated. The method will also return the current
|
||||||
|
// reserved value, both in case of success and in case of
|
||||||
|
// ErrReservedValueInvalidated.
|
||||||
|
//
|
||||||
|
// NOTE: This method should only be run with the CoinSelectLock held.
|
||||||
|
func (l *LightningWallet) CheckReservedValue(in []wire.OutPoint,
|
||||||
|
out []*wire.TxOut, numAnchorChans int) (btcutil.Amount, error) {
|
||||||
|
|
||||||
|
// Get all unspent coins in the wallet.
|
||||||
|
witnessOutputs, err := l.ListUnspentWitness(0, math.MaxInt32)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
ourInput := make(map[wire.OutPoint]struct{})
|
||||||
|
for _, op := range in {
|
||||||
|
ourInput[op] = struct{}{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// When crafting a transaction with inputs from the wallet, these coins
|
||||||
|
// will usually be locked in the process, and not be returned when
|
||||||
|
// listing unspents. In this case they have already been deducted from
|
||||||
|
// the wallet balance. In case they haven't been properly locked, we
|
||||||
|
// check whether they are still listed among our unspents and deduct
|
||||||
|
// them.
|
||||||
|
var walletBalance btcutil.Amount
|
||||||
|
for _, in := range witnessOutputs {
|
||||||
|
// Spending an unlocked wallet UTXO, don't add it to the
|
||||||
|
// balance.
|
||||||
|
if _, ok := ourInput[in.OutPoint]; ok {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
walletBalance += in.Value
|
||||||
|
}
|
||||||
|
|
||||||
|
// Now we go through the outputs of the transaction, if any of the
|
||||||
|
// outputs are paying into the wallet (likely a change output), we add
|
||||||
|
// it to our final balance.
|
||||||
|
for _, txOut := range out {
|
||||||
|
_, addrs, _, err := txscript.ExtractPkScriptAddrs(
|
||||||
|
txOut.PkScript, &l.Cfg.NetParams,
|
||||||
|
)
|
||||||
|
if err != nil {
|
||||||
|
// Non-standard outputs can safely be skipped because
|
||||||
|
// they're not supported by the wallet.
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, addr := range addrs {
|
||||||
|
if !l.IsOurAddress(addr) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
walletBalance += btcutil.Amount(txOut.Value)
|
||||||
|
|
||||||
|
// We break since we don't want to double count the output.
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// We reserve a given amount for each anchor channel.
|
||||||
|
reserved := btcutil.Amount(numAnchorChans) * anchorChanReservedValue
|
||||||
|
|
||||||
|
if walletBalance < reserved {
|
||||||
|
walletLog.Debugf("Reserved value=%v above final "+
|
||||||
|
"walletbalance=%v with %d anchor channels open",
|
||||||
|
reserved, walletBalance, numAnchorChans)
|
||||||
|
return reserved, ErrReservedValueInvalidated
|
||||||
|
}
|
||||||
|
|
||||||
|
return reserved, nil
|
||||||
|
}
|
||||||
|
|
||||||
// initOurContribution initializes the given ChannelReservation with our coins
|
// initOurContribution initializes the given ChannelReservation with our coins
|
||||||
// and change reserved for the channel, and derives the keys to use for this
|
// and change reserved for the channel, and derives the keys to use for this
|
||||||
// channel.
|
// channel.
|
||||||
|
Loading…
Reference in New Issue
Block a user