lnwallet/channel: account for HTLC fee when reporting available balance

This commit is contained in:
Johan T. Halseth 2020-02-19 12:27:42 +01:00
parent 5e89d5b6c2
commit 9ff79ae595
No known key found for this signature in database
GPG Key ID: 15BAADA29DA20D26
3 changed files with 166 additions and 18 deletions

@ -1982,8 +1982,11 @@ func TestChannelLinkBandwidthConsistency(t *testing.T) {
) )
// The starting bandwidth of the channel should be exactly the amount // The starting bandwidth of the channel should be exactly the amount
// that we created the channel between her and Bob. // that we created the channel between her and Bob, minus the
expectedBandwidth := lnwire.NewMSatFromSatoshis(chanAmt - defaultCommitFee) // commitment fee and fee for adding an additional HTLC.
expectedBandwidth := lnwire.NewMSatFromSatoshis(
chanAmt-defaultCommitFee,
) - htlcFee
assertLinkBandwidth(t, aliceLink, expectedBandwidth) assertLinkBandwidth(t, aliceLink, expectedBandwidth)
// Next, we'll create an HTLC worth 1 BTC, and send it into the link as // Next, we'll create an HTLC worth 1 BTC, and send it into the link as
@ -2656,8 +2659,10 @@ func TestChannelLinkTrimCircuitsPending(t *testing.T) {
// The starting bandwidth of the channel should be exactly the amount // The starting bandwidth of the channel should be exactly the amount
// that we created the channel between her and Bob, minus the commitment // that we created the channel between her and Bob, minus the commitment
// fee. // fee and fee of adding an HTLC.
expectedBandwidth := lnwire.NewMSatFromSatoshis(chanAmt - defaultCommitFee) expectedBandwidth := lnwire.NewMSatFromSatoshis(
chanAmt-defaultCommitFee,
) - htlcFee
assertLinkBandwidth(t, alice.link, expectedBandwidth) assertLinkBandwidth(t, alice.link, expectedBandwidth)
// Capture Alice's starting bandwidth to perform later, relative // Capture Alice's starting bandwidth to perform later, relative
@ -2935,8 +2940,10 @@ func TestChannelLinkTrimCircuitsNoCommit(t *testing.T) {
// The starting bandwidth of the channel should be exactly the amount // The starting bandwidth of the channel should be exactly the amount
// that we created the channel between her and Bob, minus the commitment // that we created the channel between her and Bob, minus the commitment
// fee. // fee and fee for adding an additional HTLC.
expectedBandwidth := lnwire.NewMSatFromSatoshis(chanAmt - defaultCommitFee) expectedBandwidth := lnwire.NewMSatFromSatoshis(
chanAmt-defaultCommitFee,
) - htlcFee
assertLinkBandwidth(t, alice.link, expectedBandwidth) assertLinkBandwidth(t, alice.link, expectedBandwidth)
// Capture Alice's starting bandwidth to perform later, relative // Capture Alice's starting bandwidth to perform later, relative
@ -3191,9 +3198,9 @@ func TestChannelLinkBandwidthChanReserve(t *testing.T) {
// The starting bandwidth of the channel should be exactly the amount // The starting bandwidth of the channel should be exactly the amount
// that we created the channel between her and Bob, minus the channel // that we created the channel between her and Bob, minus the channel
// reserve. // reserve, commitment fee and fee for adding an additional HTLC.
expectedBandwidth := lnwire.NewMSatFromSatoshis( expectedBandwidth := lnwire.NewMSatFromSatoshis(
chanAmt - defaultCommitFee - chanReserve) chanAmt-defaultCommitFee-chanReserve) - htlcFee
assertLinkBandwidth(t, aliceLink, expectedBandwidth) assertLinkBandwidth(t, aliceLink, expectedBandwidth)
// Next, we'll create an HTLC worth 3 BTC, and send it into the link as // Next, we'll create an HTLC worth 3 BTC, and send it into the link as

@ -6068,20 +6068,30 @@ func (lc *LightningChannel) availableCommitmentBalance(view *htlcView) (
ourBalance = 0 ourBalance = 0
} }
// Given the commitment weight, find the commitment fee in case of no // Calculate the commitment fee in the case where we would add another
// added HTLC output. // HTLC to the commitment, as only the balance remaining after this fee
// has been paid is actually available for sending.
feePerKw := filteredView.feePerKw feePerKw := filteredView.feePerKw
baseCommitFee := lnwire.NewMSatFromSatoshis( htlcCommitFee := lnwire.NewMSatFromSatoshis(
feePerKw.FeeForWeight(commitWeight), feePerKw.FeeForWeight(commitWeight + input.HTLCWeight),
) )
// If we are the channel initiator, we must to subtract the commitment // If we are the channel initiator, we must to subtract this commitment
// fee from our available balance. // fee from our available balance in order to ensure we can afford both
// the value of the HTLC and the additional commitment fee from adding
// the HTLC.
if lc.channelState.IsInitiator { if lc.channelState.IsInitiator {
if ourBalance < baseCommitFee { // There is an edge case where our non-zero balance is lower
// than the htlcCommitFee, where we could still be sending dust
// HTLCs, but we return 0 in this case. This is to avoid
// lowering our balance even further, as this takes us into a
// bad state wehere neither we nor our channel counterparty can
// add HTLCs.
if ourBalance < htlcCommitFee {
return 0, commitWeight return 0, commitWeight
} }
ourBalance -= baseCommitFee
ourBalance -= htlcCommitFee
} }
return ourBalance, commitWeight return ourBalance, commitWeight

@ -4602,6 +4602,12 @@ func TestChanAvailableBandwidth(t *testing.T) {
aliceReserve := lnwire.NewMSatFromSatoshis( aliceReserve := lnwire.NewMSatFromSatoshis(
aliceChannel.channelState.LocalChanCfg.ChanReserve, aliceChannel.channelState.LocalChanCfg.ChanReserve,
) )
feeRate := chainfee.SatPerKWeight(
aliceChannel.channelState.LocalCommitment.FeePerKw,
)
htlcFee := lnwire.NewMSatFromSatoshis(
feeRate.FeeForWeight(input.HTLCWeight),
)
assertBandwidthEstimateCorrect := func(aliceInitiate bool) { assertBandwidthEstimateCorrect := func(aliceInitiate bool) {
// With the HTLC's added, we'll now query the AvailableBalance // With the HTLC's added, we'll now query the AvailableBalance
@ -4631,8 +4637,9 @@ func TestChanAvailableBandwidth(t *testing.T) {
aliceBalance := aliceChannel.channelState.LocalCommitment.LocalBalance aliceBalance := aliceChannel.channelState.LocalCommitment.LocalBalance
// The balance we have available for new HTLCs should be the // The balance we have available for new HTLCs should be the
// current local commitment balance, minus the channel reserve. // current local commitment balance, minus the channel reserve
expBalance := aliceBalance - aliceReserve // and the fee for adding an HTLC.
expBalance := aliceBalance - aliceReserve - htlcFee
if expBalance != aliceAvailableBalance { if expBalance != aliceAvailableBalance {
_, _, line, _ := runtime.Caller(1) _, _, line, _ := runtime.Caller(1)
t.Fatalf("line: %v, incorrect balance: expected %v, "+ t.Fatalf("line: %v, incorrect balance: expected %v, "+
@ -4714,6 +4721,130 @@ func TestChanAvailableBandwidth(t *testing.T) {
// TODO(roasbeef): additional tests from diff starting conditions // TODO(roasbeef): additional tests from diff starting conditions
} }
// TestChanAvailableBalanceNearHtlcFee checks that we get the expected reported
// balance when it is close to the htlc fee.
func TestChanAvailableBalanceNearHtlcFee(t *testing.T) {
t.Parallel()
// Create a test channel which will be used for the duration of this
// unittest. The channel will be funded evenly with Alice having 5 BTC,
// and Bob having 5 BTC.
aliceChannel, bobChannel, cleanUp, err := CreateTestChannels(true)
if err != nil {
t.Fatalf("unable to create test channels: %v", err)
}
defer cleanUp()
// Alice starts with half the channel capacity.
aliceBalance := lnwire.NewMSatFromSatoshis(5 * btcutil.SatoshiPerBitcoin)
aliceReserve := lnwire.NewMSatFromSatoshis(
aliceChannel.channelState.LocalChanCfg.ChanReserve,
)
aliceDustlimit := lnwire.NewMSatFromSatoshis(
aliceChannel.channelState.LocalChanCfg.DustLimit,
)
feeRate := chainfee.SatPerKWeight(
aliceChannel.channelState.LocalCommitment.FeePerKw,
)
htlcFee := lnwire.NewMSatFromSatoshis(
feeRate.FeeForWeight(input.HTLCWeight),
)
commitFee := lnwire.NewMSatFromSatoshis(
aliceChannel.channelState.LocalCommitment.CommitFee,
)
htlcTimeoutFee := lnwire.NewMSatFromSatoshis(
htlcTimeoutFee(feeRate),
)
// Helper method to check the current reported balance.
checkBalance := func(t *testing.T, expBalanceAlice lnwire.MilliSatoshi) {
t.Helper()
balance := aliceChannel.AvailableBalance()
if balance != expBalanceAlice {
t.Fatalf("Expected balance %v, got %v", expBalanceAlice,
balance)
}
}
// Helper method to send an HTLC from Alice to Bob, decreasing Alice's
// balance.
htlcIndex := uint64(0)
sendHtlc := func(htlcAmt lnwire.MilliSatoshi) {
t.Helper()
htlc, preImage := createHTLC(int(htlcIndex), htlcAmt)
if _, err := aliceChannel.AddHTLC(htlc, nil); err != nil {
t.Fatalf("unable to add htlc: %v", err)
}
if _, err := bobChannel.ReceiveHTLC(htlc); err != nil {
t.Fatalf("unable to recv htlc: %v", err)
}
if err := ForceStateTransition(aliceChannel, bobChannel); err != nil {
t.Fatalf("unable to complete alice's state "+
"transition: %v", err)
}
err = bobChannel.SettleHTLC(preImage, htlcIndex, nil, nil, nil)
if err != nil {
t.Fatalf("unable to settle htlc: %v", err)
}
err = aliceChannel.ReceiveHTLCSettle(preImage, htlcIndex)
if err != nil {
t.Fatalf("unable to settle htlc: %v", err)
}
if err := ForceStateTransition(aliceChannel, bobChannel); err != nil {
t.Fatalf("unable to complete alice's state "+
"transition: %v", err)
}
htlcIndex++
aliceBalance -= htlcAmt
}
// Balance should start out equal to half the channel capacity minus
// the commitment fee Alice must pay and the channel reserve. In
// addition the HTLC fee will be subtracted fromt the balance to
// reflect that this value must be reserved for any payment above the
// dust limit.
expAliceBalance := aliceBalance - commitFee - aliceReserve - htlcFee
checkBalance(t, expAliceBalance)
// Find the minumim size of a non-dust HTLC.
aliceNonDustHtlc := aliceDustlimit + htlcTimeoutFee
// Send a HTLC leaving the remaining balance just enough to have
// nonDustHtlc left after paying the commit fee and htlc fee.
htlcAmt := aliceBalance - (commitFee + aliceReserve + htlcFee + aliceNonDustHtlc)
sendHtlc(htlcAmt)
// Now the real balance left will be
// nonDustHtlc+commitfee+aliceReserve+htlcfee. The available balance
// reported will just be nonDustHtlc, since the rest of the balance is
// reserved.
expAliceBalance = aliceNonDustHtlc
checkBalance(t, expAliceBalance)
// Send an HTLC using all but one msat of the reported balance.
htlcAmt = aliceNonDustHtlc - 1
sendHtlc(htlcAmt)
// 1 msat should be left.
expAliceBalance = 1
checkBalance(t, expAliceBalance)
// Sendng the last msat.
htlcAmt = 1
sendHtlc(htlcAmt)
// No balance left.
expAliceBalance = 0
checkBalance(t, expAliceBalance)
}
// TestSignCommitmentFailNotLockedIn tests that a channel will not attempt to // TestSignCommitmentFailNotLockedIn tests that a channel will not attempt to
// create a new state if it doesn't yet know of the next revocation point for // create a new state if it doesn't yet know of the next revocation point for
// the remote party. // the remote party.