lnwallet: switch fee estimation to return sat/kw fee rates
In this commit, we modify our FeeEstimator interface to return an estimated fee rate in sat/kw. Recently, due to low fees on the network, users have been experiencing failures broadcasting transactions due to not meeting specific fee requirements. This was happening more often than not, as the estimated fee returned by backend nodes (bitcoind and btcd) only takes into account vbytes, rather than weight. The fees returned are also expressed in sat/kb, so we must take care that we do not lose precision while converting to sat/kw. In the event that this happens, a fee floor of 253 sat/kw has been added. This fee rate originates from bitcoind rounding up the conversion from weight to vbytes.
This commit is contained in:
parent
30b156c706
commit
b3d5d7ab9c
@ -8,38 +8,49 @@ import (
|
|||||||
"github.com/btcsuite/btcutil"
|
"github.com/btcsuite/btcutil"
|
||||||
)
|
)
|
||||||
|
|
||||||
// SatPerVByte represents a fee rate in satoshis per vbyte.
|
const (
|
||||||
type SatPerVByte btcutil.Amount
|
// FeePerKwFloor is the lowest fee rate in sat/kw that we should use for
|
||||||
|
// determining transaction fees.
|
||||||
|
FeePerKwFloor SatPerKWeight = 253
|
||||||
|
)
|
||||||
|
|
||||||
// FeeForVSize calculates the fee resulting from this fee rate and
|
// SatPerKVByte represents a fee rate in sat/kb.
|
||||||
// the given vsize in vbytes.
|
type SatPerKVByte btcutil.Amount
|
||||||
func (s SatPerVByte) FeeForVSize(vbytes int64) btcutil.Amount {
|
|
||||||
return btcutil.Amount(s) * btcutil.Amount(vbytes)
|
// FeeForVSize calculates the fee resulting from this fee rate and the given
|
||||||
|
// vsize in vbytes.
|
||||||
|
func (s SatPerKVByte) FeeForVSize(vbytes int64) btcutil.Amount {
|
||||||
|
return btcutil.Amount(s) * btcutil.Amount(vbytes) / 1000
|
||||||
}
|
}
|
||||||
|
|
||||||
// FeePerKWeight converts the fee rate into SatPerKWeight.
|
// FeePerKWeight converts the current fee rate from sat/kb to sat/kw.
|
||||||
func (s SatPerVByte) FeePerKWeight() SatPerKWeight {
|
func (s SatPerKVByte) FeePerKWeight() SatPerKWeight {
|
||||||
return SatPerKWeight(s * 1000 / blockchain.WitnessScaleFactor)
|
return SatPerKWeight(s / blockchain.WitnessScaleFactor)
|
||||||
}
|
}
|
||||||
|
|
||||||
// SatPerKWeight represents a fee rate in satoshis per kilo weight unit.
|
// SatPerKWeight represents a fee rate in sat/kw.
|
||||||
type SatPerKWeight btcutil.Amount
|
type SatPerKWeight btcutil.Amount
|
||||||
|
|
||||||
// FeeForWeight calculates the fee resulting from this fee rate and the
|
// FeeForWeight calculates the fee resulting from this fee rate and the given
|
||||||
// given weight in weight units (wu).
|
// weight in weight units (wu).
|
||||||
func (s SatPerKWeight) FeeForWeight(wu int64) btcutil.Amount {
|
func (s SatPerKWeight) FeeForWeight(wu int64) btcutil.Amount {
|
||||||
// The resulting fee is rounded down, as specified in BOLT#03.
|
// The resulting fee is rounded down, as specified in BOLT#03.
|
||||||
return btcutil.Amount(s) * btcutil.Amount(wu) / 1000
|
return btcutil.Amount(s) * btcutil.Amount(wu) / 1000
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// FeePerKVByte converts the current fee rate from sat/kw to sat/kb.
|
||||||
|
func (s SatPerKWeight) FeePerKVByte() SatPerKVByte {
|
||||||
|
return SatPerKVByte(s * blockchain.WitnessScaleFactor)
|
||||||
|
}
|
||||||
|
|
||||||
// FeeEstimator provides the ability to estimate on-chain transaction fees for
|
// FeeEstimator provides the ability to estimate on-chain transaction fees for
|
||||||
// various combinations of transaction sizes and desired confirmation time
|
// various combinations of transaction sizes and desired confirmation time
|
||||||
// (measured by number of blocks).
|
// (measured by number of blocks).
|
||||||
type FeeEstimator interface {
|
type FeeEstimator interface {
|
||||||
// EstimateFeePerVSize takes in a target for the number of blocks until
|
// EstimateFeePerKW takes in a target for the number of blocks until an
|
||||||
// an initial confirmation and returns the estimated fee expressed in
|
// initial confirmation and returns the estimated fee expressed in
|
||||||
// satoshis/vbyte.
|
// sat/kw.
|
||||||
EstimateFeePerVSize(numBlocks uint32) (SatPerVByte, error)
|
EstimateFeePerKW(numBlocks uint32) (SatPerKWeight, error)
|
||||||
|
|
||||||
// Start signals the FeeEstimator to start any processes or goroutines
|
// Start signals the FeeEstimator to start any processes or goroutines
|
||||||
// it needs to perform its duty.
|
// it needs to perform its duty.
|
||||||
@ -54,16 +65,16 @@ type FeeEstimator interface {
|
|||||||
// requests. It is designed to be replaced by a proper fee calculation
|
// requests. It is designed to be replaced by a proper fee calculation
|
||||||
// implementation.
|
// implementation.
|
||||||
type StaticFeeEstimator struct {
|
type StaticFeeEstimator struct {
|
||||||
// FeeRate is the static fee rate in satoshis-per-vbyte that will be
|
// FeePerKW is the static fee rate in satoshis-per-vbyte that will be
|
||||||
// returned by this fee estimator.
|
// returned by this fee estimator.
|
||||||
FeeRate SatPerVByte
|
FeePerKW SatPerKWeight
|
||||||
}
|
}
|
||||||
|
|
||||||
// EstimateFeePerVSize will return a static value for fee calculations.
|
// EstimateFeePerKW will return a static value for fee calculations.
|
||||||
//
|
//
|
||||||
// NOTE: This method is part of the FeeEstimator interface.
|
// NOTE: This method is part of the FeeEstimator interface.
|
||||||
func (e StaticFeeEstimator) EstimateFeePerVSize(numBlocks uint32) (SatPerVByte, error) {
|
func (e StaticFeeEstimator) EstimateFeePerKW(numBlocks uint32) (SatPerKWeight, error) {
|
||||||
return e.FeeRate, nil
|
return e.FeePerKW, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Start signals the FeeEstimator to start any processes or goroutines
|
// Start signals the FeeEstimator to start any processes or goroutines
|
||||||
@ -90,16 +101,16 @@ var _ FeeEstimator = (*StaticFeeEstimator)(nil)
|
|||||||
// by the RPC interface of an active btcd node. This implementation will proxy
|
// by the RPC interface of an active btcd node. This implementation will proxy
|
||||||
// any fee estimation requests to btcd's RPC interface.
|
// any fee estimation requests to btcd's RPC interface.
|
||||||
type BtcdFeeEstimator struct {
|
type BtcdFeeEstimator struct {
|
||||||
// fallBackFeeRate is the fall back fee rate in satoshis per vbyte that
|
// fallbackFeePerKW is the fall back fee rate in sat/kw that is returned
|
||||||
// is returned if the fee estimator does not yet have enough data to
|
// if the fee estimator does not yet have enough data to actually
|
||||||
// actually produce fee estimates.
|
// produce fee estimates.
|
||||||
fallBackFeeRate SatPerVByte
|
fallbackFeePerKW SatPerKWeight
|
||||||
|
|
||||||
// minFeeRate is the minimum relay fee, in sat/vbyte, of the backend
|
// minFeePerKW is the minimum fee, in sat/kw, that we should enforce.
|
||||||
// node. This will be used as the default fee rate of a transaction when
|
// This will be used as the default fee rate for a transaction when the
|
||||||
// the estimated fee rate is too low to allow the transaction to
|
// estimated fee rate is too low to allow the transaction to propagate
|
||||||
// propagate through the network.
|
// through the network.
|
||||||
minFeeRate SatPerVByte
|
minFeePerKW SatPerKWeight
|
||||||
|
|
||||||
btcdConn *rpcclient.Client
|
btcdConn *rpcclient.Client
|
||||||
}
|
}
|
||||||
@ -110,7 +121,7 @@ type BtcdFeeEstimator struct {
|
|||||||
// the occasion that the estimator has insufficient data, or returns zero for a
|
// the occasion that the estimator has insufficient data, or returns zero for a
|
||||||
// fee estimate.
|
// fee estimate.
|
||||||
func NewBtcdFeeEstimator(rpcConfig rpcclient.ConnConfig,
|
func NewBtcdFeeEstimator(rpcConfig rpcclient.ConnConfig,
|
||||||
fallBackFeeRate SatPerVByte) (*BtcdFeeEstimator, error) {
|
fallBackFeeRate SatPerKWeight) (*BtcdFeeEstimator, error) {
|
||||||
|
|
||||||
rpcConfig.DisableConnectOnNew = true
|
rpcConfig.DisableConnectOnNew = true
|
||||||
rpcConfig.DisableAutoReconnect = false
|
rpcConfig.DisableAutoReconnect = false
|
||||||
@ -120,7 +131,7 @@ func NewBtcdFeeEstimator(rpcConfig rpcclient.ConnConfig,
|
|||||||
}
|
}
|
||||||
|
|
||||||
return &BtcdFeeEstimator{
|
return &BtcdFeeEstimator{
|
||||||
fallBackFeeRate: fallBackFeeRate,
|
fallbackFeePerKW: fallBackFeeRate,
|
||||||
btcdConn: chainConn,
|
btcdConn: chainConn,
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
@ -146,9 +157,20 @@ func (b *BtcdFeeEstimator) Start() error {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// The fee rate is expressed in sat/KB, so we'll manually convert it to
|
// The fee rate is expressed in sat/kb, so we'll manually convert it to
|
||||||
// our desired sat/vbyte rate.
|
// our desired sat/kw rate.
|
||||||
b.minFeeRate = SatPerVByte(relayFee / 1000)
|
minRelayFeePerKw := SatPerKVByte(relayFee).FeePerKWeight()
|
||||||
|
|
||||||
|
// By default, we'll use the backend node's minimum relay fee as the
|
||||||
|
// minimum fee rate we'll propose for transacations. However, if this
|
||||||
|
// happens to be lower than our fee floor, we'll enforce that instead.
|
||||||
|
b.minFeePerKW = minRelayFeePerKw
|
||||||
|
if b.minFeePerKW < FeePerKwFloor {
|
||||||
|
b.minFeePerKW = FeePerKwFloor
|
||||||
|
}
|
||||||
|
|
||||||
|
walletLog.Debugf("Using minimum fee rate of %v sat/kw",
|
||||||
|
int64(b.minFeePerKW))
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
@ -163,13 +185,12 @@ func (b *BtcdFeeEstimator) Stop() error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// EstimateFeePerVSize takes in a target for the number of blocks until an
|
// EstimateFeePerKW takes in a target for the number of blocks until an initial
|
||||||
// initial confirmation and returns the estimated fee expressed in
|
// confirmation and returns the estimated fee expressed in sat/kw.
|
||||||
// satoshis/vbyte.
|
|
||||||
//
|
//
|
||||||
// NOTE: This method is part of the FeeEstimator interface.
|
// NOTE: This method is part of the FeeEstimator interface.
|
||||||
func (b *BtcdFeeEstimator) EstimateFeePerVSize(numBlocks uint32) (SatPerVByte, error) {
|
func (b *BtcdFeeEstimator) EstimateFeePerKW(numBlocks uint32) (SatPerKWeight, error) {
|
||||||
feeEstimate, err := b.fetchEstimatePerVSize(numBlocks)
|
feeEstimate, err := b.fetchEstimate(numBlocks)
|
||||||
switch {
|
switch {
|
||||||
// If the estimator doesn't have enough data, or returns an error, then
|
// If the estimator doesn't have enough data, or returns an error, then
|
||||||
// to return a proper value, then we'll return the default fall back
|
// to return a proper value, then we'll return the default fall back
|
||||||
@ -179,16 +200,15 @@ func (b *BtcdFeeEstimator) EstimateFeePerVSize(numBlocks uint32) (SatPerVByte, e
|
|||||||
fallthrough
|
fallthrough
|
||||||
|
|
||||||
case feeEstimate == 0:
|
case feeEstimate == 0:
|
||||||
return b.fallBackFeeRate, nil
|
return b.fallbackFeePerKW, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
return feeEstimate, nil
|
return feeEstimate, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// fetchEstimate returns a fee estimate for a transaction to be confirmed in
|
// fetchEstimate returns a fee estimate for a transaction to be confirmed in
|
||||||
// confTarget blocks. The estimate is returned in sat/vbyte.
|
// confTarget blocks. The estimate is returned in sat/kw.
|
||||||
func (b *BtcdFeeEstimator) fetchEstimatePerVSize(
|
func (b *BtcdFeeEstimator) fetchEstimate(confTarget uint32) (SatPerKWeight, error) {
|
||||||
confTarget uint32) (SatPerVByte, error) {
|
|
||||||
// First, we'll fetch the estimate for our confirmation target.
|
// First, we'll fetch the estimate for our confirmation target.
|
||||||
btcPerKB, err := b.btcdConn.EstimateFee(int64(confTarget))
|
btcPerKB, err := b.btcdConn.EstimateFee(int64(confTarget))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -202,23 +222,21 @@ func (b *BtcdFeeEstimator) fetchEstimatePerVSize(
|
|||||||
return 0, err
|
return 0, err
|
||||||
}
|
}
|
||||||
|
|
||||||
// The value returned is expressed in fees per KB, while we want
|
// Since we use fee rates in sat/kw internally, we'll convert the
|
||||||
// fee-per-byte, so we'll divide by 1000 to map to satoshis-per-byte
|
// estimated fee rate from its sat/kb representation to sat/kw.
|
||||||
// before returning the estimate.
|
satPerKw := SatPerKVByte(satPerKB).FeePerKWeight()
|
||||||
satPerByte := SatPerVByte(satPerKB / 1000)
|
|
||||||
|
|
||||||
// Before proceeding, we'll make sure that this fee rate respects the
|
// Finally, we'll enforce our fee floor.
|
||||||
// minimum relay fee set on the backend node.
|
if satPerKw < b.minFeePerKW {
|
||||||
if satPerByte < b.minFeeRate {
|
walletLog.Debugf("Estimated fee rate of %v sat/kw is too low, "+
|
||||||
walletLog.Debugf("Using backend node's minimum relay fee rate "+
|
"using fee floor of %v sat/kw instead", b.minFeePerKW)
|
||||||
"of %v sat/vbyte", b.minFeeRate)
|
satPerKw = b.minFeePerKW
|
||||||
satPerByte = b.minFeeRate
|
|
||||||
}
|
}
|
||||||
|
|
||||||
walletLog.Debugf("Returning %v sat/vbyte for conf target of %v",
|
walletLog.Debugf("Returning %v sat/kw for conf target of %v",
|
||||||
int64(satPerByte), confTarget)
|
int64(satPerKw), confTarget)
|
||||||
|
|
||||||
return satPerByte, nil
|
return satPerKw, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// A compile-time assertion to ensure that BtcdFeeEstimator implements the
|
// A compile-time assertion to ensure that BtcdFeeEstimator implements the
|
||||||
@ -229,16 +247,16 @@ var _ FeeEstimator = (*BtcdFeeEstimator)(nil)
|
|||||||
// backed by the RPC interface of an active bitcoind node. This implementation
|
// backed by the RPC interface of an active bitcoind node. This implementation
|
||||||
// will proxy any fee estimation requests to bitcoind's RPC interface.
|
// will proxy any fee estimation requests to bitcoind's RPC interface.
|
||||||
type BitcoindFeeEstimator struct {
|
type BitcoindFeeEstimator struct {
|
||||||
// fallBackFeeRate is the fall back fee rate in satoshis per vbyte that
|
// fallbackFeePerKW is the fallback fee rate in sat/kw that is returned
|
||||||
// is returned if the fee estimator does not yet have enough data to
|
// if the fee estimator does not yet have enough data to actually
|
||||||
// actually produce fee estimates.
|
// produce fee estimates.
|
||||||
fallBackFeeRate SatPerVByte
|
fallbackFeePerKW SatPerKWeight
|
||||||
|
|
||||||
// minFeeRate is the minimum relay fee, in sat/vbyte, of the backend
|
// minFeePerKW is the minimum fee, in sat/kw, that we should enforce.
|
||||||
// node. This will be used as the default fee rate of a transaction when
|
// This will be used as the default fee rate for a transaction when the
|
||||||
// the estimated fee rate is too low to allow the transaction to
|
// estimated fee rate is too low to allow the transaction to propagate
|
||||||
// propagate through the network.
|
// through the network.
|
||||||
minFeeRate SatPerVByte
|
minFeePerKW SatPerKWeight
|
||||||
|
|
||||||
bitcoindConn *rpcclient.Client
|
bitcoindConn *rpcclient.Client
|
||||||
}
|
}
|
||||||
@ -249,7 +267,7 @@ type BitcoindFeeEstimator struct {
|
|||||||
// is used in the occasion that the estimator has insufficient data, or returns
|
// is used in the occasion that the estimator has insufficient data, or returns
|
||||||
// zero for a fee estimate.
|
// zero for a fee estimate.
|
||||||
func NewBitcoindFeeEstimator(rpcConfig rpcclient.ConnConfig,
|
func NewBitcoindFeeEstimator(rpcConfig rpcclient.ConnConfig,
|
||||||
fallBackFeeRate SatPerVByte) (*BitcoindFeeEstimator, error) {
|
fallBackFeeRate SatPerKWeight) (*BitcoindFeeEstimator, error) {
|
||||||
|
|
||||||
rpcConfig.DisableConnectOnNew = true
|
rpcConfig.DisableConnectOnNew = true
|
||||||
rpcConfig.DisableAutoReconnect = false
|
rpcConfig.DisableAutoReconnect = false
|
||||||
@ -261,7 +279,7 @@ func NewBitcoindFeeEstimator(rpcConfig rpcclient.ConnConfig,
|
|||||||
}
|
}
|
||||||
|
|
||||||
return &BitcoindFeeEstimator{
|
return &BitcoindFeeEstimator{
|
||||||
fallBackFeeRate: fallBackFeeRate,
|
fallbackFeePerKW: fallBackFeeRate,
|
||||||
bitcoindConn: chainConn,
|
bitcoindConn: chainConn,
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
@ -293,9 +311,20 @@ func (b *BitcoindFeeEstimator) Start() error {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// The fee rate is expressed in sat/KB, so we'll manually convert it to
|
// The fee rate is expressed in sat/kb, so we'll manually convert it to
|
||||||
// our desired sat/vbyte rate.
|
// our desired sat/kw rate.
|
||||||
b.minFeeRate = SatPerVByte(relayFee / 1000)
|
minRelayFeePerKw := SatPerKVByte(relayFee).FeePerKWeight()
|
||||||
|
|
||||||
|
// By default, we'll use the backend node's minimum relay fee as the
|
||||||
|
// minimum fee rate we'll propose for transacations. However, if this
|
||||||
|
// happens to be lower than our fee floor, we'll enforce that instead.
|
||||||
|
b.minFeePerKW = minRelayFeePerKw
|
||||||
|
if b.minFeePerKW < FeePerKwFloor {
|
||||||
|
b.minFeePerKW = FeePerKwFloor
|
||||||
|
}
|
||||||
|
|
||||||
|
walletLog.Debugf("Using minimum fee rate of %v sat/kw",
|
||||||
|
int64(b.minFeePerKW))
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
@ -308,13 +337,12 @@ func (b *BitcoindFeeEstimator) Stop() error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// EstimateFeePerVSize takes in a target for the number of blocks until an
|
// EstimateFeePerKW takes in a target for the number of blocks until an initial
|
||||||
// initial confirmation and returns the estimated fee expressed in
|
// confirmation and returns the estimated fee expressed in sat/kw.
|
||||||
// satoshis/vbyte.
|
|
||||||
//
|
//
|
||||||
// NOTE: This method is part of the FeeEstimator interface.
|
// NOTE: This method is part of the FeeEstimator interface.
|
||||||
func (b *BitcoindFeeEstimator) EstimateFeePerVSize(numBlocks uint32) (SatPerVByte, error) {
|
func (b *BitcoindFeeEstimator) EstimateFeePerKW(numBlocks uint32) (SatPerKWeight, error) {
|
||||||
feeEstimate, err := b.fetchEstimatePerVSize(numBlocks)
|
feeEstimate, err := b.fetchEstimate(numBlocks)
|
||||||
switch {
|
switch {
|
||||||
// If the estimator doesn't have enough data, or returns an error, then
|
// If the estimator doesn't have enough data, or returns an error, then
|
||||||
// to return a proper value, then we'll return the default fall back
|
// to return a proper value, then we'll return the default fall back
|
||||||
@ -324,16 +352,15 @@ func (b *BitcoindFeeEstimator) EstimateFeePerVSize(numBlocks uint32) (SatPerVByt
|
|||||||
fallthrough
|
fallthrough
|
||||||
|
|
||||||
case feeEstimate == 0:
|
case feeEstimate == 0:
|
||||||
return b.fallBackFeeRate, nil
|
return b.fallbackFeePerKW, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
return feeEstimate, nil
|
return feeEstimate, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// fetchEstimatePerVSize returns a fee estimate for a transaction to be confirmed in
|
// fetchEstimate returns a fee estimate for a transaction to be confirmed in
|
||||||
// confTarget blocks. The estimate is returned in sat/vbyte.
|
// confTarget blocks. The estimate is returned in sat/kw.
|
||||||
func (b *BitcoindFeeEstimator) fetchEstimatePerVSize(
|
func (b *BitcoindFeeEstimator) fetchEstimate(confTarget uint32) (SatPerKWeight, error) {
|
||||||
confTarget uint32) (SatPerVByte, error) {
|
|
||||||
// First, we'll send an "estimatesmartfee" command as a raw request,
|
// First, we'll send an "estimatesmartfee" command as a raw request,
|
||||||
// since it isn't supported by btcd but is available in bitcoind.
|
// since it isn't supported by btcd but is available in bitcoind.
|
||||||
target, err := json.Marshal(uint64(confTarget))
|
target, err := json.Marshal(uint64(confTarget))
|
||||||
@ -341,45 +368,44 @@ func (b *BitcoindFeeEstimator) fetchEstimatePerVSize(
|
|||||||
return 0, err
|
return 0, err
|
||||||
}
|
}
|
||||||
// TODO: Allow selection of economical/conservative modifiers.
|
// TODO: Allow selection of economical/conservative modifiers.
|
||||||
resp, err := b.bitcoindConn.RawRequest("estimatesmartfee",
|
resp, err := b.bitcoindConn.RawRequest(
|
||||||
[]json.RawMessage{target})
|
"estimatesmartfee", []json.RawMessage{target},
|
||||||
|
)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return 0, err
|
return 0, err
|
||||||
}
|
}
|
||||||
|
|
||||||
// Next, we'll parse the response to get the BTC per KB.
|
// Next, we'll parse the response to get the BTC per KB.
|
||||||
feeEstimate := struct {
|
feeEstimate := struct {
|
||||||
Feerate float64 `json:"feerate"`
|
FeeRate float64 `json:"feerate"`
|
||||||
}{}
|
}{}
|
||||||
err = json.Unmarshal(resp, &feeEstimate)
|
err = json.Unmarshal(resp, &feeEstimate)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return 0, err
|
return 0, err
|
||||||
}
|
}
|
||||||
|
|
||||||
// Next, we'll convert the returned value to satoshis, as it's
|
// Next, we'll convert the returned value to satoshis, as it's currently
|
||||||
// currently returned in BTC.
|
// returned in BTC.
|
||||||
satPerKB, err := btcutil.NewAmount(feeEstimate.Feerate)
|
satPerKB, err := btcutil.NewAmount(feeEstimate.FeeRate)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return 0, err
|
return 0, err
|
||||||
}
|
}
|
||||||
|
|
||||||
// The value returned is expressed in fees per KB, while we want
|
// Since we use fee rates in sat/kw internally, we'll convert the
|
||||||
// fee-per-byte, so we'll divide by 1000 to map to satoshis-per-byte
|
// estimated fee rate from its sat/kb representation to sat/kw.
|
||||||
// before returning the estimate.
|
satPerKw := SatPerKVByte(satPerKB).FeePerKWeight()
|
||||||
satPerByte := SatPerVByte(satPerKB / 1000)
|
|
||||||
|
|
||||||
// Before proceeding, we'll make sure that this fee rate respects the
|
// Finally, we'll enforce our fee floor.
|
||||||
// minimum relay fee set on the backend node.
|
if satPerKw < b.minFeePerKW {
|
||||||
if satPerByte < b.minFeeRate {
|
walletLog.Debugf("Estimated fee rate of %v sat/kw is too low, "+
|
||||||
walletLog.Debugf("Using backend node's minimum relay fee rate "+
|
"using fee floor of %v sat/kw instead", b.minFeePerKW)
|
||||||
"of %v sat/vbyte", b.minFeeRate)
|
satPerKw = b.minFeePerKW
|
||||||
satPerByte = b.minFeeRate
|
|
||||||
}
|
}
|
||||||
|
|
||||||
walletLog.Debugf("Returning %v sat/vbyte for conf target of %v",
|
walletLog.Debugf("Returning %v sat/kw for conf target of %v",
|
||||||
int64(satPerByte), confTarget)
|
int64(satPerKw), confTarget)
|
||||||
|
|
||||||
return satPerByte, nil
|
return satPerKw, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// A compile-time assertion to ensure that BitcoindFeeEstimator implements the
|
// A compile-time assertion to ensure that BitcoindFeeEstimator implements the
|
||||||
|
@ -13,57 +13,56 @@ import (
|
|||||||
func TestFeeRateTypes(t *testing.T) {
|
func TestFeeRateTypes(t *testing.T) {
|
||||||
t.Parallel()
|
t.Parallel()
|
||||||
|
|
||||||
// Let our fee rate be 100 sat/vbyte.
|
// We'll be calculating the transaction fees for the given measurements
|
||||||
feePerVSize := lnwallet.SatPerVByte(100)
|
// using different fee rates and expecting them to match.
|
||||||
|
const vsize = 300
|
||||||
|
const weight = vsize * 4
|
||||||
|
|
||||||
// It is also equivalent to 25000 sat/kw.
|
// Test the conversion from sat/kw to sat/kb.
|
||||||
feePerKw := feePerVSize.FeePerKWeight()
|
for feePerKw := lnwallet.SatPerKWeight(250); feePerKw < 10000; feePerKw += 50 {
|
||||||
if feePerKw != 25000 {
|
feePerKB := feePerKw.FeePerKVByte()
|
||||||
t.Fatalf("expected %d sat/kw, got %d sat/kw", 25000,
|
if feePerKB != lnwallet.SatPerKVByte(feePerKw*4) {
|
||||||
feePerKw)
|
t.Fatalf("expected %d sat/kb, got %d sat/kb when "+
|
||||||
|
"converting from %d sat/kw", feePerKw*4,
|
||||||
|
feePerKB, feePerKw)
|
||||||
}
|
}
|
||||||
|
|
||||||
const txVSize = 300
|
// The resulting transaction fee should be the same when using
|
||||||
|
// both rates.
|
||||||
// We'll now run through a set of values for the fee per vsize type,
|
expectedFee := btcutil.Amount(feePerKw * weight / 1000)
|
||||||
// making sure the conversion to sat/kw and fee calculation is done
|
fee1 := feePerKw.FeeForWeight(weight)
|
||||||
// correctly.
|
if fee1 != expectedFee {
|
||||||
for f := lnwallet.SatPerVByte(0); f <= 40; f++ {
|
t.Fatalf("expected fee of %d sats, got %d sats",
|
||||||
fPerKw := f.FeePerKWeight()
|
expectedFee, fee1)
|
||||||
|
|
||||||
// The kw is always 250*vsize.
|
|
||||||
if fPerKw != lnwallet.SatPerKWeight(f*250) {
|
|
||||||
t.Fatalf("expected %d sat/kw, got %d sat/kw, when "+
|
|
||||||
"converting %d sat/vbyte", f*250, fPerKw, f)
|
|
||||||
}
|
}
|
||||||
|
fee2 := feePerKB.FeeForVSize(vsize)
|
||||||
// The tx fee should simply be f*txvsize.
|
if fee2 != expectedFee {
|
||||||
fee := f.FeeForVSize(txVSize)
|
t.Fatalf("expected fee of %d sats, got %d sats",
|
||||||
if fee != btcutil.Amount(f*txVSize) {
|
expectedFee, fee2)
|
||||||
t.Fatalf("expected tx fee to be %d sat, was %d sat",
|
|
||||||
f*txVSize, fee)
|
|
||||||
}
|
|
||||||
|
|
||||||
// The weight is 4*vsize. Fee calculation from the fee/kw
|
|
||||||
// should result in the same fee.
|
|
||||||
fee2 := fPerKw.FeeForWeight(txVSize * 4)
|
|
||||||
if fee != fee2 {
|
|
||||||
t.Fatalf("fee calculated from vsize (%d) not equal "+
|
|
||||||
"fee calculated from weight (%d)", fee, fee2)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Do the same for fee per kw.
|
// Test the conversion from sat/kb to sat/kw.
|
||||||
for f := lnwallet.SatPerKWeight(0); f < 1500; f++ {
|
for feePerKB := lnwallet.SatPerKVByte(1000); feePerKB < 40000; feePerKB += 1000 {
|
||||||
weight := int64(txVSize * 4)
|
feePerKw := feePerKB.FeePerKWeight()
|
||||||
|
if feePerKw != lnwallet.SatPerKWeight(feePerKB/4) {
|
||||||
|
t.Fatalf("expected %d sat/kw, got %d sat/kw when "+
|
||||||
|
"converting from %d sat/kb", feePerKB/4,
|
||||||
|
feePerKw, feePerKB)
|
||||||
|
}
|
||||||
|
|
||||||
// The expected fee is weight*f / 1000, since the fee is
|
// The resulting transaction fee should be the same when using
|
||||||
// denominated per 1000 wu.
|
// both rates.
|
||||||
expFee := btcutil.Amount(weight) * btcutil.Amount(f) / 1000
|
expectedFee := btcutil.Amount(feePerKB * vsize / 1000)
|
||||||
fee := f.FeeForWeight(weight)
|
fee1 := feePerKB.FeeForVSize(vsize)
|
||||||
if fee != expFee {
|
if fee1 != expectedFee {
|
||||||
t.Fatalf("expected fee to be %d sat, was %d",
|
t.Fatalf("expected fee of %d sats, got %d sats",
|
||||||
fee, expFee)
|
expectedFee, fee1)
|
||||||
|
}
|
||||||
|
fee2 := feePerKw.FeeForWeight(weight)
|
||||||
|
if fee2 != expectedFee {
|
||||||
|
t.Fatalf("expected fee of %d sats, got %d sats",
|
||||||
|
expectedFee, fee2)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -73,22 +72,22 @@ func TestFeeRateTypes(t *testing.T) {
|
|||||||
func TestStaticFeeEstimator(t *testing.T) {
|
func TestStaticFeeEstimator(t *testing.T) {
|
||||||
t.Parallel()
|
t.Parallel()
|
||||||
|
|
||||||
const feePerVSize = 100
|
const feePerKw = lnwallet.FeePerKwFloor
|
||||||
|
|
||||||
feeEstimator := &lnwallet.StaticFeeEstimator{
|
feeEstimator := &lnwallet.StaticFeeEstimator{
|
||||||
FeeRate: feePerVSize,
|
FeePerKW: feePerKw,
|
||||||
}
|
}
|
||||||
if err := feeEstimator.Start(); err != nil {
|
if err := feeEstimator.Start(); err != nil {
|
||||||
t.Fatalf("unable to start fee estimator: %v", err)
|
t.Fatalf("unable to start fee estimator: %v", err)
|
||||||
}
|
}
|
||||||
defer feeEstimator.Stop()
|
defer feeEstimator.Stop()
|
||||||
|
|
||||||
feeRate, err := feeEstimator.EstimateFeePerVSize(6)
|
feeRate, err := feeEstimator.EstimateFeePerKW(6)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("unable to get fee rate: %v", err)
|
t.Fatalf("unable to get fee rate: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if feeRate != feePerVSize {
|
if feeRate != feePerKw {
|
||||||
t.Fatalf("expected fee rate %v, got %v", feePerVSize, feeRate)
|
t.Fatalf("expected fee rate %v, got %v", feePerKw, feeRate)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user