lnwallet: properly handle HTLC settles in channel state-machine
We no longer track HTLC’s by their r-hash within the log into the index, as we may have multiple HTLC’s that can be redeemed by the same pre-image. Instead we now use a separate index which is keyed by a log-index. Additionally, the SettleHTLC method now also returns the index of the HTLC being settled which allows the remote party to quickly locate the HTLC within their log. This commit also introduces a few trace/debug log messages which will likely be pruned in the near future
This commit is contained in:
parent
06f062e678
commit
7dea354711
@ -7,6 +7,7 @@ import (
|
|||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
"github.com/btcsuite/fastsha256"
|
"github.com/btcsuite/fastsha256"
|
||||||
|
"github.com/davecgh/go-spew/spew"
|
||||||
"github.com/lightningnetwork/lnd/chainntfs"
|
"github.com/lightningnetwork/lnd/chainntfs"
|
||||||
"github.com/lightningnetwork/lnd/channeldb"
|
"github.com/lightningnetwork/lnd/channeldb"
|
||||||
"github.com/lightningnetwork/lnd/lnwire"
|
"github.com/lightningnetwork/lnd/lnwire"
|
||||||
@ -144,6 +145,7 @@ type PaymentDescriptor struct {
|
|||||||
// isForwarded denotes if an incoming HTLC has been forwarded to any
|
// isForwarded denotes if an incoming HTLC has been forwarded to any
|
||||||
// possible upstream peers in the route.
|
// possible upstream peers in the route.
|
||||||
isForwarded bool
|
isForwarded bool
|
||||||
|
settled bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// commitment represents a commitment to a new state within an active channel.
|
// commitment represents a commitment to a new state within an active channel.
|
||||||
@ -331,24 +333,15 @@ type LightningChannel struct {
|
|||||||
// commitment. The log is compacted once a revocation is received.
|
// commitment. The log is compacted once a revocation is received.
|
||||||
stateUpdateLog *list.List
|
stateUpdateLog *list.List
|
||||||
|
|
||||||
// entriesByHash is an index into the above log. This index is used to
|
// logIndex is an index into the above log. This index is used to
|
||||||
// remove Add state updates, once a timeout/settle is received.
|
// remove Add state updates, once a timeout/settle is received.
|
||||||
entriesByHash map[PaymentHash]*list.Element
|
logIndex map[uint32]*list.Element
|
||||||
|
|
||||||
// Payment's which we've requested.
|
|
||||||
// TODO(roasbeef): move into InvoiceRegistry
|
|
||||||
unfufilledPayments map[PaymentHash]*PaymentRequest
|
|
||||||
|
|
||||||
fundingTxIn *wire.TxIn
|
fundingTxIn *wire.TxIn
|
||||||
fundingP2WSH []byte
|
fundingP2WSH []byte
|
||||||
|
|
||||||
// TODO(roasbeef): Stores all previous R values + timeouts for each
|
|
||||||
// commitment update, plus some other meta-data...Or just use OP_RETURN
|
|
||||||
// to help out?
|
|
||||||
// currently going for: nSequence/nLockTime overloading
|
|
||||||
channelDB *channeldb.DB
|
channelDB *channeldb.DB
|
||||||
|
|
||||||
// TODO(roasbeef): create and embed 'Service' interface w/ below?
|
|
||||||
started int32
|
started int32
|
||||||
shutdown int32
|
shutdown int32
|
||||||
|
|
||||||
@ -374,8 +367,7 @@ func NewLightningChannel(wallet *LightningWallet, events chainntnfs.ChainNotifie
|
|||||||
channelState: state,
|
channelState: state,
|
||||||
revocationWindowEdge: state.NumUpdates,
|
revocationWindowEdge: state.NumUpdates,
|
||||||
stateUpdateLog: list.New(),
|
stateUpdateLog: list.New(),
|
||||||
entriesByHash: make(map[PaymentHash]*list.Element),
|
logIndex: make(map[uint32]*list.Element),
|
||||||
unfufilledPayments: make(map[PaymentHash]*PaymentRequest),
|
|
||||||
channelDB: chanDB,
|
channelDB: chanDB,
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -701,6 +693,13 @@ func (lc *LightningChannel) SignNextCommitment() ([]byte, uint32, error) {
|
|||||||
return nil, 0, err
|
return nil, 0, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
log.Tracef("ChannelPoint(%v): extending remote chain to height %v",
|
||||||
|
lc.channelState.ChanID, newCommitView.height)
|
||||||
|
log.Tracef("ChannelPoint(%v): remote chain: our_balance=%v, "+
|
||||||
|
"their_balance=%v, commit_tx: %v", lc.channelState.ChanID,
|
||||||
|
newCommitView.ourBalance, newCommitView.theirBalance,
|
||||||
|
spew.Sdump(newCommitView.txn))
|
||||||
|
|
||||||
// Sign their version of the new commitment transaction.
|
// Sign their version of the new commitment transaction.
|
||||||
hashCache := txscript.NewTxSigHashes(newCommitView.txn)
|
hashCache := txscript.NewTxSigHashes(newCommitView.txn)
|
||||||
sig, err := txscript.RawTxInWitnessSignature(newCommitView.txn,
|
sig, err := txscript.RawTxInWitnessSignature(newCommitView.txn,
|
||||||
@ -762,6 +761,13 @@ func (lc *LightningChannel) ReceiveNewCommitment(rawSig []byte,
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
log.Tracef("ChannelPoint(%v): extending local chain to height %v",
|
||||||
|
lc.channelState.ChanID, localCommitmentView.height)
|
||||||
|
log.Tracef("ChannelPoint(%v): local chain: our_balance=%v, "+
|
||||||
|
"their_balance=%v, commit_tx: %v", lc.channelState.ChanID,
|
||||||
|
localCommitmentView.ourBalance, localCommitmentView.theirBalance,
|
||||||
|
spew.Sdump(localCommitmentView.txn))
|
||||||
|
|
||||||
// Construct the sighash of the commitment transaction corresponding to
|
// Construct the sighash of the commitment transaction corresponding to
|
||||||
// this newly proposed state update.
|
// this newly proposed state update.
|
||||||
localCommitTx := localCommitmentView.txn
|
localCommitTx := localCommitmentView.txn
|
||||||
@ -820,6 +826,10 @@ func (lc *LightningChannel) RevokeCurrentCommitment() (*lnwire.CommitRevocation,
|
|||||||
revocationEdge[:])
|
revocationEdge[:])
|
||||||
revocationMsg.NextRevocationHash = fastsha256.Sum256(revocationEdge[:])
|
revocationMsg.NextRevocationHash = fastsha256.Sum256(revocationEdge[:])
|
||||||
|
|
||||||
|
log.Tracef("ChannelPoint(%v): revoking height=%v, now at height=%v, window_edge=%v",
|
||||||
|
lc.channelState.ChanID, lc.localCommitChain.tail().height,
|
||||||
|
lc.currentHeight+1, lc.revocationWindowEdge)
|
||||||
|
|
||||||
// Advance our tail, as we've revoked our previous state.
|
// Advance our tail, as we've revoked our previous state.
|
||||||
lc.localCommitChain.advanceTail()
|
lc.localCommitChain.advanceTail()
|
||||||
|
|
||||||
@ -833,6 +843,10 @@ func (lc *LightningChannel) RevokeCurrentCommitment() (*lnwire.CommitRevocation,
|
|||||||
lc.channelState.OurCommitSig = tail.sig
|
lc.channelState.OurCommitSig = tail.sig
|
||||||
lc.channelState.NumUpdates++
|
lc.channelState.NumUpdates++
|
||||||
|
|
||||||
|
log.Tracef("ChannelPoint(%v): state transition accepted: "+
|
||||||
|
"our_balance=%v, their_balance=%v", lc.channelState.ChanID,
|
||||||
|
tail.ourBalance, tail.theirBalance)
|
||||||
|
|
||||||
// TODO(roasbeef): use RecordChannelDelta once fin
|
// TODO(roasbeef): use RecordChannelDelta once fin
|
||||||
if err := lc.channelState.FullSync(); err != nil {
|
if err := lc.channelState.FullSync(); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
@ -920,6 +934,8 @@ func (lc *LightningChannel) ReceiveRevocation(revMsg *lnwire.CommitRevocation) (
|
|||||||
// Now that we've verified the revocation update the state of the HTLC
|
// Now that we've verified the revocation update the state of the HTLC
|
||||||
// log as we may be able to prune portions of it now, and update their
|
// log as we may be able to prune portions of it now, and update their
|
||||||
// balance.
|
// balance.
|
||||||
|
// TODO(roasbeef): move this out to another func?
|
||||||
|
// * .CompactLog()
|
||||||
var next *list.Element
|
var next *list.Element
|
||||||
var htlcsToForward []*PaymentDescriptor
|
var htlcsToForward []*PaymentDescriptor
|
||||||
for e := lc.stateUpdateLog.Front(); e != nil; e = next {
|
for e := lc.stateUpdateLog.Front(); e != nil; e = next {
|
||||||
@ -933,8 +949,6 @@ func (lc *LightningChannel) ReceiveRevocation(revMsg *lnwire.CommitRevocation) (
|
|||||||
if remoteChainTail >= htlc.removeCommitHeightRemote &&
|
if remoteChainTail >= htlc.removeCommitHeightRemote &&
|
||||||
localChainTail >= htlc.removeCommitHeightLocal {
|
localChainTail >= htlc.removeCommitHeightLocal {
|
||||||
parentLink := htlc.parent
|
parentLink := htlc.parent
|
||||||
addHTLC := parentLink.Value.(*PaymentDescriptor)
|
|
||||||
delete(lc.entriesByHash, addHTLC.RHash)
|
|
||||||
lc.stateUpdateLog.Remove(e)
|
lc.stateUpdateLog.Remove(e)
|
||||||
lc.stateUpdateLog.Remove(parentLink)
|
lc.stateUpdateLog.Remove(parentLink)
|
||||||
}
|
}
|
||||||
@ -1003,9 +1017,7 @@ func (lc *LightningChannel) AddHTLC(htlc *lnwire.HTLCAddRequest, incoming bool)
|
|||||||
}
|
}
|
||||||
|
|
||||||
pd.Index = index
|
pd.Index = index
|
||||||
pdLink := lc.stateUpdateLog.PushBack(pd)
|
lc.stateUpdateLog.PushBack(pd)
|
||||||
// TODO(roabeef): this should be by HTLC key instead
|
|
||||||
lc.entriesByHash[pd.RHash] = pdLink
|
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
@ -1015,22 +1027,34 @@ func (lc *LightningChannel) AddHTLC(htlc *lnwire.HTLCAddRequest, incoming bool)
|
|||||||
// be false, when receiving a settlement to a previously outgoing HTLC, then
|
// be false, when receiving a settlement to a previously outgoing HTLC, then
|
||||||
// the value of incoming should be true. If the settlement fails due to an
|
// the value of incoming should be true. If the settlement fails due to an
|
||||||
// invalid preimage, then an error is returned.
|
// invalid preimage, then an error is returned.
|
||||||
func (lc *LightningChannel) SettleHTLC(msg *lnwire.HTLCSettleRequest, incoming bool) error {
|
func (lc *LightningChannel) SettleHTLC(preimage [32]byte, incoming bool) (uint32, error) {
|
||||||
preImage := msg.RedemptionProofs[0]
|
var targetHTLC *list.Element
|
||||||
|
|
||||||
paymentHash := PaymentHash(fastsha256.Sum256(preImage[:]))
|
// TODO(roasbeef): optimize
|
||||||
htlc, ok := lc.entriesByHash[paymentHash]
|
paymentHash := fastsha256.Sum256(preimage[:])
|
||||||
if !ok {
|
for e := lc.stateUpdateLog.Back(); e != nil; e = e.Next() {
|
||||||
return fmt.Errorf("unknown payment hash")
|
htlc := e.Value.(*PaymentDescriptor)
|
||||||
|
if htlc.entryType != Add {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if bytes.Equal(htlc.RHash[:], paymentHash[:]) && !htlc.settled {
|
||||||
|
htlc.settled = true
|
||||||
|
targetHTLC = e
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if targetHTLC == nil {
|
||||||
|
return 0, fmt.Errorf("invalid payment hash")
|
||||||
}
|
}
|
||||||
|
|
||||||
parentPd := htlc.Value.(*PaymentDescriptor)
|
parentPd := targetHTLC.Value.(*PaymentDescriptor)
|
||||||
|
|
||||||
// TODO(roasbeef): maybe make the log entries an interface?
|
// TODO(roasbeef): maybe make the log entries an interface?
|
||||||
pd := &PaymentDescriptor{}
|
pd := &PaymentDescriptor{}
|
||||||
pd.IsIncoming = parentPd.IsIncoming
|
pd.IsIncoming = parentPd.IsIncoming
|
||||||
pd.Amount = parentPd.Amount
|
pd.Amount = parentPd.Amount
|
||||||
pd.parent = htlc
|
pd.parent = targetHTLC
|
||||||
pd.entryType = Settle
|
pd.entryType = Settle
|
||||||
|
|
||||||
var index uint32
|
var index uint32
|
||||||
@ -1045,7 +1069,7 @@ func (lc *LightningChannel) SettleHTLC(msg *lnwire.HTLCSettleRequest, incoming b
|
|||||||
pd.Index = index
|
pd.Index = index
|
||||||
lc.stateUpdateLog.PushBack(pd)
|
lc.stateUpdateLog.PushBack(pd)
|
||||||
|
|
||||||
return nil
|
return targetHTLC.Value.(*PaymentDescriptor).Index, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// TimeoutHTLC...
|
// TimeoutHTLC...
|
||||||
@ -1278,6 +1302,7 @@ func createCommitTx(fundingOutput *wire.TxIn, selfKey, theirKey *btcec.PublicKey
|
|||||||
commitTx := wire.NewMsgTx()
|
commitTx := wire.NewMsgTx()
|
||||||
commitTx.Version = 2
|
commitTx.Version = 2
|
||||||
commitTx.AddTxIn(fundingOutput)
|
commitTx.AddTxIn(fundingOutput)
|
||||||
|
// TODO(roasbeef): don't make 0 BTC output...
|
||||||
commitTx.AddTxOut(wire.NewTxOut(int64(amountToSelf), payToUsScriptHash))
|
commitTx.AddTxOut(wire.NewTxOut(int64(amountToSelf), payToUsScriptHash))
|
||||||
commitTx.AddTxOut(wire.NewTxOut(int64(amountToThem), theirWitnessKeyHash))
|
commitTx.AddTxOut(wire.NewTxOut(int64(amountToThem), theirWitnessKeyHash))
|
||||||
|
|
||||||
|
@ -329,13 +329,10 @@ func TestSimpleAddSettleWorkflow(t *testing.T) {
|
|||||||
// HTLC once he learns of the preimage.
|
// HTLC once he learns of the preimage.
|
||||||
var preimage [32]byte
|
var preimage [32]byte
|
||||||
copy(preimage[:], paymentPreimage)
|
copy(preimage[:], paymentPreimage)
|
||||||
wireSettleMsg := &lnwire.HTLCSettleRequest{
|
if _, err := bobChannel.SettleHTLC(preimage, false); err != nil {
|
||||||
RedemptionProofs: [][32]byte{preimage},
|
|
||||||
}
|
|
||||||
if err := bobChannel.SettleHTLC(wireSettleMsg, false); err != nil {
|
|
||||||
t.Fatalf("bob unable to settle inbound htlc: %v", err)
|
t.Fatalf("bob unable to settle inbound htlc: %v", err)
|
||||||
}
|
}
|
||||||
if err := aliceChannel.SettleHTLC(wireSettleMsg, true); err != nil {
|
if _, err := aliceChannel.SettleHTLC(preimage, true); err != nil {
|
||||||
t.Fatalf("alice unable to accept settle of outbound htlc: %v", err)
|
t.Fatalf("alice unable to accept settle of outbound htlc: %v", err)
|
||||||
}
|
}
|
||||||
bobSig2, aliceIndex2, err := bobChannel.SignNextCommitment()
|
bobSig2, aliceIndex2, err := bobChannel.SignNextCommitment()
|
||||||
|
@ -348,7 +348,8 @@ func senderHtlcSpendTimeout(commitScript []byte, outputAmt btcutil.Amount,
|
|||||||
// OP_ENDIF
|
// OP_ENDIF
|
||||||
// <sender key> OP_CHECKSIG
|
// <sender key> OP_CHECKSIG
|
||||||
// OP_ENDIF
|
// OP_ENDIF
|
||||||
// TODO(roasbeef): rename these to sender vs receiver?
|
// TODO(roasbeef): go back to revocation keys in the HTLC outputs?
|
||||||
|
// * also could combine pre-image with their key?
|
||||||
func receiverHTLCScript(absoluteTimeout, relativeTimeout uint32, senderKey,
|
func receiverHTLCScript(absoluteTimeout, relativeTimeout uint32, senderKey,
|
||||||
receiverKey *btcec.PublicKey, revokeHash, paymentHash []byte) ([]byte, error) {
|
receiverKey *btcec.PublicKey, revokeHash, paymentHash []byte) ([]byte, error) {
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user