Merge pull request #2879 from joostjager/outgoing-go-to-chain

htlcswitch: revert forwarding policy block delta requirements
This commit is contained in:
Conner Fromknecht 2019-04-05 15:45:30 -07:00 committed by GitHub
commit caa0e2f0b8
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
11 changed files with 154 additions and 91 deletions

@ -67,7 +67,11 @@ const (
defaultTorV2PrivateKeyFilename = "v2_onion_private_key" defaultTorV2PrivateKeyFilename = "v2_onion_private_key"
defaultTorV3PrivateKeyFilename = "v3_onion_private_key" defaultTorV3PrivateKeyFilename = "v3_onion_private_key"
defaultBroadcastDelta = 10 defaultIncomingBroadcastDelta = 20
defaultFinalCltvRejectDelta = 2
defaultOutgoingBroadcastDelta = 10
defaultOutgoingCltvRejectDelta = 0
// minTimeLockDelta is the minimum timelock we require for incoming // minTimeLockDelta is the minimum timelock we require for incoming
// HTLCs on our channels. // HTLCs on our channels.

@ -53,13 +53,20 @@ type ChainArbitratorConfig struct {
// ChainHash is the chain that this arbitrator is to operate within. // ChainHash is the chain that this arbitrator is to operate within.
ChainHash chainhash.Hash ChainHash chainhash.Hash
// BroadcastDelta is the delta that we'll use to decide when to // IncomingBroadcastDelta is the delta that we'll use to decide when to
// broadcast our commitment transaction. This value should be set // broadcast our commitment transaction if we have incoming htlcs. This
// based on our current fee estimation of the commitment transaction. // value should be set based on our current fee estimation of the
// We use this to determine when we should broadcast instead of the // commitment transaction. We use this to determine when we should
// just the HTLC timeout, as we want to ensure that the commitment // broadcast instead of the just the HTLC timeout, as we want to ensure
// transaction is already confirmed, by the time the HTLC expires. // that the commitment transaction is already confirmed, by the time the
BroadcastDelta uint32 // HTLC expires. Otherwise we may end up not settling the htlc on-chain
// because the other party managed to time it out.
IncomingBroadcastDelta uint32
// OutgoingBroadcastDelta is the delta that we'll use to decide when to
// broadcast our commitment transaction if there are active outgoing
// htlcs. This value can be lower than the incoming broadcast delta.
OutgoingBroadcastDelta uint32
// NewSweepAddr is a function that returns a new address under control // NewSweepAddr is a function that returns a new address under control
// by the wallet. We'll use this to sweep any no-delay outputs as a // by the wallet. We'll use this to sweep any no-delay outputs as a

@ -1106,7 +1106,8 @@ func (c *ChannelArbitrator) checkChainActions(height uint32,
// We'll need to go on-chain for an outgoing HTLC if it was // We'll need to go on-chain for an outgoing HTLC if it was
// never resolved downstream, and it's "close" to timing out. // never resolved downstream, and it's "close" to timing out.
haveChainActions = haveChainActions || c.shouldGoOnChain( haveChainActions = haveChainActions || c.shouldGoOnChain(
htlc.RefundTimeout, c.cfg.BroadcastDelta, height, htlc.RefundTimeout, c.cfg.OutgoingBroadcastDelta,
height,
) )
} }
for _, htlc := range c.activeHTLCs.incomingHTLCs { for _, htlc := range c.activeHTLCs.incomingHTLCs {
@ -1124,7 +1125,8 @@ func (c *ChannelArbitrator) checkChainActions(height uint32,
continue continue
} }
haveChainActions = haveChainActions || c.shouldGoOnChain( haveChainActions = haveChainActions || c.shouldGoOnChain(
htlc.RefundTimeout, c.cfg.BroadcastDelta, height, htlc.RefundTimeout, c.cfg.IncomingBroadcastDelta,
height,
) )
} }
@ -1162,7 +1164,8 @@ func (c *ChannelArbitrator) checkChainActions(height uint32,
// until the HTLC times out to see if we can also redeem it // until the HTLC times out to see if we can also redeem it
// on-chain. // on-chain.
case !c.shouldGoOnChain( case !c.shouldGoOnChain(
htlc.RefundTimeout, c.cfg.BroadcastDelta, height, htlc.RefundTimeout, c.cfg.OutgoingBroadcastDelta,
height,
): ):
// TODO(roasbeef): also need to be able to query // TODO(roasbeef): also need to be able to query
// circuit map to see if HTLC hasn't been fully // circuit map to see if HTLC hasn't been fully

@ -165,7 +165,8 @@ func createTestChannelArbitrator(log ArbitratorLog) (*ChannelArbitrator,
DeliverResolutionMsg: func(...ResolutionMsg) error { DeliverResolutionMsg: func(...ResolutionMsg) error {
return nil return nil
}, },
BroadcastDelta: 5, OutgoingBroadcastDelta: 5,
IncomingBroadcastDelta: 5,
Notifier: &mockNotifier{ Notifier: &mockNotifier{
epochChan: make(chan *chainntnfs.BlockEpoch), epochChan: make(chan *chainntnfs.BlockEpoch),
spendChan: make(chan *chainntnfs.SpendDetail), spendChan: make(chan *chainntnfs.SpendDetail),

@ -235,17 +235,18 @@ type ChannelLinkConfig struct {
MinFeeUpdateTimeout time.Duration MinFeeUpdateTimeout time.Duration
MaxFeeUpdateTimeout time.Duration MaxFeeUpdateTimeout time.Duration
// ExpiryGraceDelta is the minimum difference between the current block // FinalCltvRejectDelta defines the number of blocks before the expiry
// height and the CLTV we require on 1) an outgoing HTLC in order to // of the htlc where we no longer settle it as an exit hop and instead
// forward as an intermediary hop, or 2) an incoming HTLC to reveal the // cancel it back. Normally this value should be lower than the cltv
// preimage as the final hop. We'll reject any HTLC's who's timeout minus // expiry of any invoice we create and the code effectuating this should
// this value is less than or equal to the current block height. We require // not be hit.
// this in order to ensure that we have sufficient time to claim or FinalCltvRejectDelta uint32
// timeout an HTLC on chain.
// // OutgoingCltvRejectDelta defines the number of blocks before expiry of
// This MUST be greater than the maximum BroadcastDelta of the // an htlc where we don't offer an htlc anymore. This should be at least
// ChannelArbitrator for the outbound channel. // the outgoing broadcast delta, because in any case we don't want to
ExpiryGraceDelta uint32 // risk offering an htlc that triggers channel closure.
OutgoingCltvRejectDelta uint32
} }
// channelLink is the service which drives a channel's commitment update // channelLink is the service which drives a channel's commitment update
@ -2154,9 +2155,9 @@ func (l *channelLink) HtlcSatifiesPolicy(payHash [32]byte,
} }
// We want to avoid offering an HTLC which will expire in the near // We want to avoid offering an HTLC which will expire in the near
// future, so we'll reject an HTLC if the outgoing expiration time is too // future, so we'll reject an HTLC if the outgoing expiration time is
// close to the current height. // too close to the current height.
if outgoingTimeout-l.cfg.ExpiryGraceDelta <= heightNow { if outgoingTimeout <= heightNow+l.cfg.OutgoingCltvRejectDelta {
l.errorf("htlc(%x) has an expiry that's too soon: "+ l.errorf("htlc(%x) has an expiry that's too soon: "+
"outgoing_expiry=%v, best_height=%v", payHash[:], "outgoing_expiry=%v, best_height=%v", payHash[:],
outgoingTimeout, heightNow) outgoingTimeout, heightNow)
@ -2174,7 +2175,8 @@ func (l *channelLink) HtlcSatifiesPolicy(payHash [32]byte,
return failure return failure
} }
if outgoingTimeout-heightNow > maxCltvExpiry { // Check absolute max delta.
if outgoingTimeout > maxCltvExpiry+heightNow {
l.errorf("outgoing htlc(%x) has a time lock too far in the "+ l.errorf("outgoing htlc(%x) has a time lock too far in the "+
"future: got %v, but maximum is %v", payHash[:], "future: got %v, but maximum is %v", payHash[:],
outgoingTimeout-heightNow, maxCltvExpiry) outgoingTimeout-heightNow, maxCltvExpiry)
@ -2187,7 +2189,7 @@ func (l *channelLink) HtlcSatifiesPolicy(payHash [32]byte,
// delta should equal the outgoing time lock. Otherwise, whether the // delta should equal the outgoing time lock. Otherwise, whether the
// sender messed up, or an intermediate node tampered with the HTLC. // sender messed up, or an intermediate node tampered with the HTLC.
timeDelta := policy.TimeLockDelta timeDelta := policy.TimeLockDelta
if incomingTimeout-timeDelta < outgoingTimeout { if incomingTimeout < outgoingTimeout+timeDelta {
l.errorf("Incoming htlc(%x) has incorrect time-lock value: "+ l.errorf("Incoming htlc(%x) has incorrect time-lock value: "+
"expected at least %v block delta, got %v block delta", "expected at least %v block delta, got %v block delta",
payHash[:], timeDelta, incomingTimeout-outgoingTimeout) payHash[:], timeDelta, incomingTimeout-outgoingTimeout)
@ -2679,7 +2681,7 @@ func (l *channelLink) processExitHop(pd *lnwallet.PaymentDescriptor,
// First, we'll check the expiry of the HTLC itself against, the current // First, we'll check the expiry of the HTLC itself against, the current
// block height. If the timeout is too soon, then we'll reject the HTLC. // block height. If the timeout is too soon, then we'll reject the HTLC.
if pd.Timeout-l.cfg.ExpiryGraceDelta <= heightNow { if pd.Timeout <= heightNow+l.cfg.FinalCltvRejectDelta {
log.Errorf("htlc(%x) has an expiry that's too soon: expiry=%v"+ log.Errorf("htlc(%x) has an expiry that's too soon: expiry=%v"+
", best_height=%v", pd.RHash[:], pd.Timeout, heightNow) ", best_height=%v", pd.RHash[:], pd.Timeout, heightNow)

@ -179,7 +179,8 @@ func createInterceptorFunc(prefix, receiver string, messages []expectedMessage,
func TestChannelLinkSingleHopPayment(t *testing.T) { func TestChannelLinkSingleHopPayment(t *testing.T) {
t.Parallel() t.Parallel()
channels, cleanUp, _, err := createClusterChannels( // Setup a alice-bob network.
aliceChannel, bobChannel, cleanUp, err := createTwoClusterChannels(
btcutil.SatoshiPerBitcoin*3, btcutil.SatoshiPerBitcoin*3,
btcutil.SatoshiPerBitcoin*5) btcutil.SatoshiPerBitcoin*5)
if err != nil { if err != nil {
@ -187,15 +188,14 @@ func TestChannelLinkSingleHopPayment(t *testing.T) {
} }
defer cleanUp() defer cleanUp()
n := newThreeHopNetwork(t, channels.aliceToBob, channels.bobToAlice, n := newTwoHopNetwork(t, aliceChannel, bobChannel, testStartingHeight)
channels.bobToCarol, channels.carolToBob, testStartingHeight)
if err := n.start(); err != nil { if err := n.start(); err != nil {
t.Fatal(err) t.Fatal(err)
} }
defer n.stop() defer n.stop()
aliceBandwidthBefore := n.aliceChannelLink.Bandwidth() aliceBandwidthBefore := n.aliceChannelLink.Bandwidth()
bobBandwidthBefore := n.firstBobChannelLink.Bandwidth() bobBandwidthBefore := n.bobChannelLink.Bandwidth()
debug := false debug := false
if debug { if debug {
@ -205,12 +205,12 @@ func TestChannelLinkSingleHopPayment(t *testing.T) {
// Log message that bob receives. // Log message that bob receives.
n.bobServer.intersect(createLogFunc("bob", n.bobServer.intersect(createLogFunc("bob",
n.firstBobChannelLink.ChanID())) n.bobChannelLink.ChanID()))
} }
amount := lnwire.NewMSatFromSatoshis(btcutil.SatoshiPerBitcoin) amount := lnwire.NewMSatFromSatoshis(btcutil.SatoshiPerBitcoin)
htlcAmt, totalTimelock, hops := generateHops(amount, testStartingHeight, htlcAmt, totalTimelock, hops := generateHops(amount, testStartingHeight,
n.firstBobChannelLink) n.bobChannelLink)
// Wait for: // Wait for:
// * HTLC add request to be sent to bob. // * HTLC add request to be sent to bob.
@ -219,7 +219,7 @@ func TestChannelLinkSingleHopPayment(t *testing.T) {
// * alice<->bob commitment state to be updated. // * alice<->bob commitment state to be updated.
// * user notification to be sent. // * user notification to be sent.
receiver := n.bobServer receiver := n.bobServer
firstHop := n.firstBobChannelLink.ShortChanID() firstHop := n.bobChannelLink.ShortChanID()
rhash, err := makePayment( rhash, err := makePayment(
n.aliceServer, receiver, firstHop, hops, amount, htlcAmt, n.aliceServer, receiver, firstHop, hops, amount, htlcAmt,
totalTimelock, totalTimelock,
@ -248,10 +248,10 @@ func TestChannelLinkSingleHopPayment(t *testing.T) {
"amount") "amount")
} }
if bobBandwidthBefore+amount != n.firstBobChannelLink.Bandwidth() { if bobBandwidthBefore+amount != n.bobChannelLink.Bandwidth() {
t.Fatalf("bob bandwidth isn't match: expected %v, got %v", t.Fatalf("bob bandwidth isn't match: expected %v, got %v",
bobBandwidthBefore+amount, bobBandwidthBefore+amount,
n.firstBobChannelLink.Bandwidth()) n.bobChannelLink.Bandwidth())
} }
} }
@ -394,7 +394,29 @@ func TestChannelLinkBidirectionalOneHopPayments(t *testing.T) {
// hops. In this test we send the payment from Carol to Alice over Bob peer. // hops. In this test we send the payment from Carol to Alice over Bob peer.
// (Carol -> Bob -> Alice) and checking that HTLC was settled properly and // (Carol -> Bob -> Alice) and checking that HTLC was settled properly and
// balances were changed in two channels. // balances were changed in two channels.
//
// The test is executed with two different OutgoingCltvRejectDelta values for
// bob. In addition to a normal positive value, we also test the zero case
// because this is currently the configured value in lnd
// (defaultOutgoingCltvRejectDelta).
func TestChannelLinkMultiHopPayment(t *testing.T) { func TestChannelLinkMultiHopPayment(t *testing.T) {
t.Run(
"bobOutgoingCltvRejectDelta 3",
func(t *testing.T) {
testChannelLinkMultiHopPayment(t, 3)
},
)
t.Run(
"bobOutgoingCltvRejectDelta 0",
func(t *testing.T) {
testChannelLinkMultiHopPayment(t, 0)
},
)
}
func testChannelLinkMultiHopPayment(t *testing.T,
bobOutgoingCltvRejectDelta uint32) {
t.Parallel() t.Parallel()
channels, cleanUp, _, err := createClusterChannels( channels, cleanUp, _, err := createClusterChannels(
@ -407,6 +429,13 @@ func TestChannelLinkMultiHopPayment(t *testing.T) {
n := newThreeHopNetwork(t, channels.aliceToBob, channels.bobToAlice, n := newThreeHopNetwork(t, channels.aliceToBob, channels.bobToAlice,
channels.bobToCarol, channels.carolToBob, testStartingHeight) channels.bobToCarol, channels.carolToBob, testStartingHeight)
n.firstBobChannelLink.cfg.OutgoingCltvRejectDelta =
bobOutgoingCltvRejectDelta
n.secondBobChannelLink.cfg.OutgoingCltvRejectDelta =
bobOutgoingCltvRejectDelta
if err := n.start(); err != nil { if err := n.start(); err != nil {
t.Fatal(err) t.Fatal(err)
} }

@ -728,6 +728,8 @@ func newDB() (*channeldb.DB, func(), error) {
return cdb, cleanUp, nil return cdb, cleanUp, nil
} }
const testInvoiceCltvExpiry = 4
type mockInvoiceRegistry struct { type mockInvoiceRegistry struct {
settleChan chan lntypes.Hash settleChan chan lntypes.Hash
@ -743,7 +745,7 @@ func newMockRegistry(minDelta uint32) *mockInvoiceRegistry {
} }
decodeExpiry := func(invoice string) (uint32, error) { decodeExpiry := func(invoice string) (uint32, error) {
return 3, nil return testInvoiceCltvExpiry, nil
} }
registry := invoices.NewRegistry(cdb, decodeExpiry) registry := invoices.NewRegistry(cdb, decodeExpiry)

@ -527,6 +527,12 @@ func generatePaymentWithPreimage(invoiceAmt, htlcAmt lnwire.MilliSatoshi,
preimage, rhash [32]byte) (*channeldb.Invoice, *lnwire.UpdateAddHTLC, preimage, rhash [32]byte) (*channeldb.Invoice, *lnwire.UpdateAddHTLC,
error) { error) {
// Create the db invoice. Normally the payment requests needs to be set,
// because it is decoded in InvoiceRegistry to obtain the cltv expiry.
// But because the mock registry used in tests is mocking the decode
// step and always returning the value of testInvoiceCltvExpiry, we
// don't need to bother here with creating and signing a payment
// request.
invoice := &channeldb.Invoice{ invoice := &channeldb.Invoice{
CreationDate: time.Now(), CreationDate: time.Now(),
Terms: channeldb.ContractTerm{ Terms: channeldb.ContractTerm{
@ -602,8 +608,6 @@ type threeHopNetwork struct {
func generateHops(payAmt lnwire.MilliSatoshi, startingHeight uint32, func generateHops(payAmt lnwire.MilliSatoshi, startingHeight uint32,
path ...*channelLink) (lnwire.MilliSatoshi, uint32, []ForwardingInfo) { path ...*channelLink) (lnwire.MilliSatoshi, uint32, []ForwardingInfo) {
lastHop := path[len(path)-1]
totalTimelock := startingHeight totalTimelock := startingHeight
runningAmt := payAmt runningAmt := payAmt
@ -620,7 +624,7 @@ func generateHops(payAmt lnwire.MilliSatoshi, startingHeight uint32,
// If this is the last, hop, then the time lock will be their // If this is the last, hop, then the time lock will be their
// specified delta policy plus our starting height. // specified delta policy plus our starting height.
if i == len(path)-1 { if i == len(path)-1 {
totalTimelock += lastHop.cfg.FwrdingPolicy.TimeLockDelta totalTimelock += testInvoiceCltvExpiry
timeLock = totalTimelock timeLock = totalTimelock
} else { } else {
// Otherwise, the outgoing time lock should be the // Otherwise, the outgoing time lock should be the
@ -1019,7 +1023,6 @@ func (h *hopNetwork) createChannelLink(server, peer *mockServer,
fwdPkgTimeout = 15 * time.Second fwdPkgTimeout = 15 * time.Second
minFeeUpdateTimeout = 30 * time.Minute minFeeUpdateTimeout = 30 * time.Minute
maxFeeUpdateTimeout = 40 * time.Minute maxFeeUpdateTimeout = 40 * time.Minute
expiryGraceDelta = 3
) )
link := NewChannelLink( link := NewChannelLink(
@ -1041,15 +1044,16 @@ func (h *hopNetwork) createChannelLink(server, peer *mockServer,
UpdateContractSignals: func(*contractcourt.ContractSignals) error { UpdateContractSignals: func(*contractcourt.ContractSignals) error {
return nil return nil
}, },
ChainEvents: &contractcourt.ChainEventSubscription{}, ChainEvents: &contractcourt.ChainEventSubscription{},
SyncStates: true, SyncStates: true,
BatchSize: 10, BatchSize: 10,
BatchTicker: ticker.NewForce(batchTimeout), BatchTicker: ticker.NewForce(batchTimeout),
FwdPkgGCTicker: ticker.NewForce(fwdPkgTimeout), FwdPkgGCTicker: ticker.NewForce(fwdPkgTimeout),
MinFeeUpdateTimeout: minFeeUpdateTimeout, MinFeeUpdateTimeout: minFeeUpdateTimeout,
MaxFeeUpdateTimeout: maxFeeUpdateTimeout, MaxFeeUpdateTimeout: maxFeeUpdateTimeout,
OnChannelFailure: func(lnwire.ChannelID, lnwire.ShortChannelID, LinkFailureError) {}, OnChannelFailure: func(lnwire.ChannelID, lnwire.ShortChannelID, LinkFailureError) {},
ExpiryGraceDelta: expiryGraceDelta, FinalCltvRejectDelta: 3,
OutgoingCltvRejectDelta: 3,
}, },
channel, channel,
) )

@ -9420,7 +9420,7 @@ func testMultiHopHtlcLocalTimeout(net *lntest.NetworkHarness, t *harnessTest) {
// We'll now mine enough blocks to trigger Bob's broadcast of his // We'll now mine enough blocks to trigger Bob's broadcast of his
// commitment transaction due to the fact that the HTLC is about to // commitment transaction due to the fact that the HTLC is about to
// timeout. // timeout.
numBlocks := uint32(finalCltvDelta - defaultBroadcastDelta) numBlocks := uint32(finalCltvDelta - defaultOutgoingBroadcastDelta)
if _, err := net.Miner.Node.Generate(numBlocks); err != nil { if _, err := net.Miner.Node.Generate(numBlocks); err != nil {
t.Fatalf("unable to generate blocks: %v", err) t.Fatalf("unable to generate blocks: %v", err)
} }
@ -9469,7 +9469,10 @@ func testMultiHopHtlcLocalTimeout(net *lntest.NetworkHarness, t *harnessTest) {
// We'll now mine the remaining blocks to cause the HTLC itself to // We'll now mine the remaining blocks to cause the HTLC itself to
// timeout. // timeout.
if _, err := net.Miner.Node.Generate(defaultBroadcastDelta - defaultCSV); err != nil { _, err = net.Miner.Node.Generate(
defaultOutgoingBroadcastDelta - defaultCSV,
)
if err != nil {
t.Fatalf("unable to generate blocks: %v", err) t.Fatalf("unable to generate blocks: %v", err)
} }
@ -9600,7 +9603,7 @@ func testMultiHopReceiverChainClaim(net *lntest.NetworkHarness, t *harnessTest)
const invoiceAmt = 100000 const invoiceAmt = 100000
invoiceReq := &lnrpc.Invoice{ invoiceReq := &lnrpc.Invoice{
Value: invoiceAmt, Value: invoiceAmt,
CltvExpiry: 20, CltvExpiry: 40,
} }
ctxt, _ := context.WithTimeout(ctxb, defaultTimeout) ctxt, _ := context.WithTimeout(ctxb, defaultTimeout)
carolInvoice, err := carol.AddInvoice(ctxt, invoiceReq) carolInvoice, err := carol.AddInvoice(ctxt, invoiceReq)
@ -9644,7 +9647,9 @@ func testMultiHopReceiverChainClaim(net *lntest.NetworkHarness, t *harnessTest)
// Now we'll mine enough blocks to prompt carol to actually go to the // Now we'll mine enough blocks to prompt carol to actually go to the
// chain in order to sweep her HTLC since the value is high enough. // chain in order to sweep her HTLC since the value is high enough.
// TODO(roasbeef): modify once go to chain policy changes // TODO(roasbeef): modify once go to chain policy changes
numBlocks := uint32(invoiceReq.CltvExpiry - defaultBroadcastDelta) numBlocks := uint32(
invoiceReq.CltvExpiry - defaultIncomingBroadcastDelta,
)
if _, err := net.Miner.Node.Generate(numBlocks); err != nil { if _, err := net.Miner.Node.Generate(numBlocks); err != nil {
t.Fatalf("unable to generate blocks") t.Fatalf("unable to generate blocks")
} }
@ -10324,7 +10329,7 @@ func testMultiHopHtlcLocalChainClaim(net *lntest.NetworkHarness, t *harnessTest)
// With the network active, we'll now add a new invoice at Carol's end. // With the network active, we'll now add a new invoice at Carol's end.
invoiceReq := &lnrpc.Invoice{ invoiceReq := &lnrpc.Invoice{
Value: 100000, Value: 100000,
CltvExpiry: 20, CltvExpiry: 40,
} }
ctxt, _ := context.WithTimeout(ctxb, defaultTimeout) ctxt, _ := context.WithTimeout(ctxb, defaultTimeout)
carolInvoice, err := carol.AddInvoice(ctxt, invoiceReq) carolInvoice, err := carol.AddInvoice(ctxt, invoiceReq)
@ -10380,7 +10385,9 @@ func testMultiHopHtlcLocalChainClaim(net *lntest.NetworkHarness, t *harnessTest)
// We'll now mine enough blocks so Carol decides that she needs to go // We'll now mine enough blocks so Carol decides that she needs to go
// on-chain to claim the HTLC as Bob has been inactive. // on-chain to claim the HTLC as Bob has been inactive.
numBlocks := uint32(20 - defaultBroadcastDelta) numBlocks := uint32(invoiceReq.CltvExpiry -
defaultIncomingBroadcastDelta)
if _, err := net.Miner.Node.Generate(numBlocks); err != nil { if _, err := net.Miner.Node.Generate(numBlocks); err != nil {
t.Fatalf("unable to generate blocks") t.Fatalf("unable to generate blocks")
} }
@ -10661,7 +10668,7 @@ func testMultiHopHtlcRemoteChainClaim(net *lntest.NetworkHarness, t *harnessTest
const invoiceAmt = 100000 const invoiceAmt = 100000
invoiceReq := &lnrpc.Invoice{ invoiceReq := &lnrpc.Invoice{
Value: invoiceAmt, Value: invoiceAmt,
CltvExpiry: 20, CltvExpiry: 40,
} }
ctxt, _ := context.WithTimeout(ctxb, defaultTimeout) ctxt, _ := context.WithTimeout(ctxb, defaultTimeout)
carolInvoice, err := carol.AddInvoice(ctxt, invoiceReq) carolInvoice, err := carol.AddInvoice(ctxt, invoiceReq)
@ -10731,7 +10738,9 @@ func testMultiHopHtlcRemoteChainClaim(net *lntest.NetworkHarness, t *harnessTest
// We'll now mine enough blocks so Carol decides that she needs to go // We'll now mine enough blocks so Carol decides that she needs to go
// on-chain to claim the HTLC as Bob has been inactive. // on-chain to claim the HTLC as Bob has been inactive.
numBlocks := uint32(20-defaultBroadcastDelta) - defaultCSV numBlocks := uint32(invoiceReq.CltvExpiry-
defaultIncomingBroadcastDelta) - defaultCSV
if _, err := net.Miner.Node.Generate(numBlocks); err != nil { if _, err := net.Miner.Node.Generate(numBlocks); err != nil {
t.Fatalf("unable to generate blocks") t.Fatalf("unable to generate blocks")
} }

47
peer.go

@ -58,13 +58,6 @@ const (
// messages to be sent across the wire, requested by objects outside // messages to be sent across the wire, requested by objects outside
// this struct. // this struct.
outgoingQueueLen = 50 outgoingQueueLen = 50
// extraExpiryGraceDelta is the additional number of blocks required by
// the ExpiryGraceDelta of the forwarding policy beyond the maximum
// broadcast delta. This is the minimum number of blocks between the
// expiry on an accepted or offered HTLC and the block height at which
// we will go to chain.
extraExpiryGraceDelta = 3
) )
// outgoingMsg packages an lnwire.Message to be sent out on the wire, along with // outgoingMsg packages an lnwire.Message to be sent out on the wire, along with
@ -206,11 +199,13 @@ type peer struct {
// remote node. // remote node.
localFeatures *lnwire.RawFeatureVector localFeatures *lnwire.RawFeatureVector
// expiryGraceDelta is the block time allowance for HTLCs offered and // finalCltvRejectDelta defines the number of blocks before the expiry
// received on channels with this peer. The parameter is used to // of the htlc where we no longer settle it as an exit hop.
// configure links with the peer. See ExpiryGraceDelta on finalCltvRejectDelta uint32
// ChannelLinkConfig for more information.
expiryGraceDelta uint32 // outgoingCltvRejectDelta defines the number of blocks before expiry of
// an htlc where we don't offer an htlc anymore.
outgoingCltvRejectDelta uint32
// remoteLocalFeatures is the local feature vector received from the // remoteLocalFeatures is the local feature vector received from the
// peer during the connection handshake. // peer during the connection handshake.
@ -249,7 +244,8 @@ var _ lnpeer.Peer = (*peer)(nil)
func newPeer(conn net.Conn, connReq *connmgr.ConnReq, server *server, func newPeer(conn net.Conn, connReq *connmgr.ConnReq, server *server,
addr *lnwire.NetAddress, inbound bool, addr *lnwire.NetAddress, inbound bool,
localFeatures *lnwire.RawFeatureVector, localFeatures *lnwire.RawFeatureVector,
chanActiveTimeout time.Duration, expiryGraceDelta uint32) ( chanActiveTimeout time.Duration,
finalCltvRejectDelta, outgoingCltvRejectDelta uint32) (
*peer, error) { *peer, error) {
nodePub := addr.IdentityKey nodePub := addr.IdentityKey
@ -263,8 +259,10 @@ func newPeer(conn net.Conn, connReq *connmgr.ConnReq, server *server,
server: server, server: server,
localFeatures: localFeatures, localFeatures: localFeatures,
expiryGraceDelta: expiryGraceDelta,
finalCltvRejectDelta: finalCltvRejectDelta,
outgoingCltvRejectDelta: outgoingCltvRejectDelta,
sendQueue: make(chan outgoingMsg), sendQueue: make(chan outgoingMsg),
outgoingQueue: make(chan outgoingMsg), outgoingQueue: make(chan outgoingMsg),
@ -596,15 +594,16 @@ func (p *peer) addLink(chanPoint *wire.OutPoint,
*chanPoint, signals, *chanPoint, signals,
) )
}, },
OnChannelFailure: onChannelFailure, OnChannelFailure: onChannelFailure,
SyncStates: syncStates, SyncStates: syncStates,
BatchTicker: ticker.New(50 * time.Millisecond), BatchTicker: ticker.New(50 * time.Millisecond),
FwdPkgGCTicker: ticker.New(time.Minute), FwdPkgGCTicker: ticker.New(time.Minute),
BatchSize: 10, BatchSize: 10,
UnsafeReplay: cfg.UnsafeReplay, UnsafeReplay: cfg.UnsafeReplay,
MinFeeUpdateTimeout: htlcswitch.DefaultMinLinkFeeUpdateTimeout, MinFeeUpdateTimeout: htlcswitch.DefaultMinLinkFeeUpdateTimeout,
MaxFeeUpdateTimeout: htlcswitch.DefaultMaxLinkFeeUpdateTimeout, MaxFeeUpdateTimeout: htlcswitch.DefaultMaxLinkFeeUpdateTimeout,
ExpiryGraceDelta: p.expiryGraceDelta, FinalCltvRejectDelta: p.finalCltvRejectDelta,
OutgoingCltvRejectDelta: p.outgoingCltvRejectDelta,
} }
link := htlcswitch.NewChannelLink(linkCfg, lnChan) link := htlcswitch.NewChannelLink(linkCfg, lnChan)

@ -728,8 +728,9 @@ func newServer(listenAddrs []net.Addr, chanDB *channeldb.DB, cc *chainControl,
contractBreaches := make(chan *ContractBreachEvent, 1) contractBreaches := make(chan *ContractBreachEvent, 1)
s.chainArb = contractcourt.NewChainArbitrator(contractcourt.ChainArbitratorConfig{ s.chainArb = contractcourt.NewChainArbitrator(contractcourt.ChainArbitratorConfig{
ChainHash: *activeNetParams.GenesisHash, ChainHash: *activeNetParams.GenesisHash,
BroadcastDelta: defaultBroadcastDelta, IncomingBroadcastDelta: defaultIncomingBroadcastDelta,
OutgoingBroadcastDelta: defaultOutgoingBroadcastDelta,
NewSweepAddr: func() ([]byte, error) { NewSweepAddr: func() ([]byte, error) {
return newSweepPkScript(cc.wallet) return newSweepPkScript(cc.wallet)
}, },
@ -2475,14 +2476,16 @@ func (s *server) peerConnected(conn net.Conn, connReq *connmgr.ConnReq,
localFeatures.Set(lnwire.GossipQueriesOptional) localFeatures.Set(lnwire.GossipQueriesOptional)
// Now that we've established a connection, create a peer, and it to the // Now that we've established a connection, create a peer, and it to the
// set of currently active peers. Configure the peer with a expiry grace // set of currently active peers. Configure the peer with the incoming
// delta greater than the broadcast delta, to prevent links from // and outgoing broadcast deltas to prevent htlcs from being accepted or
// accepting htlcs that may trigger channel arbitrator force close the // offered that would trigger channel closure. In case of outgoing
// channel immediately. // htlcs, an extra block is added to prevent the channel from being
// closed when the htlc is outstanding and a new block comes in.
p, err := newPeer( p, err := newPeer(
conn, connReq, s, peerAddr, inbound, localFeatures, conn, connReq, s, peerAddr, inbound, localFeatures,
cfg.ChanEnableTimeout, cfg.ChanEnableTimeout,
defaultBroadcastDelta+extraExpiryGraceDelta, defaultFinalCltvRejectDelta,
defaultOutgoingCltvRejectDelta,
) )
if err != nil { if err != nil {
srvrLog.Errorf("unable to create peer %v", err) srvrLog.Errorf("unable to create peer %v", err)