Merge pull request #2879 from joostjager/outgoing-go-to-chain
htlcswitch: revert forwarding policy block delta requirements
This commit is contained in:
commit
caa0e2f0b8
@ -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,
|
||||||
)
|
)
|
||||||
|
25
lnd_test.go
25
lnd_test.go
@ -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
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)
|
||||||
|
17
server.go
17
server.go
@ -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)
|
||||||
|
Loading…
Reference in New Issue
Block a user