htlcswitch: properly verify OutgoingCTLV+Timeout when final hop in link
This commit fixes an existing bug in the way we perform validation of the timelock information as the final hop in the route. Previously, we would assert that the outgoing time lock in the per-hop payload would exactly match our time lock delta. Instead, we should be asserting two things: 1. That the time lock in the payload is >= the expected time lock 2. That timeout on the HTLC is exactly equal to the payload
This commit is contained in:
parent
246164e290
commit
bf071c1985
@ -865,6 +865,7 @@ func (l *channelLink) Bandwidth() lnwire.MilliSatoshi {
|
|||||||
// NOTE: Should be used inside main goroutine only, otherwise the result might
|
// NOTE: Should be used inside main goroutine only, otherwise the result might
|
||||||
// not be accurate.
|
// not be accurate.
|
||||||
func (l *channelLink) getBandwidth() lnwire.MilliSatoshi {
|
func (l *channelLink) getBandwidth() lnwire.MilliSatoshi {
|
||||||
|
// TODO(roasbeef): factor in reserve, just grab mutex
|
||||||
return l.channel.LocalAvailableBalance() - l.overflowQueue.pendingAmount()
|
return l.channel.LocalAvailableBalance() - l.overflowQueue.pendingAmount()
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1026,15 +1027,19 @@ func (l *channelLink) processLockedInHtlcs(
|
|||||||
// to produce initial obfuscation of the onion
|
// to produce initial obfuscation of the onion
|
||||||
// failureCode.
|
// failureCode.
|
||||||
onionReader := bytes.NewReader(onionBlob[:])
|
onionReader := bytes.NewReader(onionBlob[:])
|
||||||
obfuscator, failureCode := l.cfg.DecodeOnionObfuscator(onionReader)
|
obfuscator, failureCode := l.cfg.DecodeOnionObfuscator(
|
||||||
|
onionReader,
|
||||||
|
)
|
||||||
if failureCode != lnwire.CodeNone {
|
if failureCode != lnwire.CodeNone {
|
||||||
// If we unable to process the onion blob than
|
// If we're unable to process the onion blob
|
||||||
// we should send the malformed htlc error to
|
// than we should send the malformed htlc error
|
||||||
// payment sender.
|
// to payment sender.
|
||||||
l.sendMalformedHTLCError(pd.RHash, failureCode, onionBlob[:])
|
l.sendMalformedHTLCError(pd.RHash, failureCode,
|
||||||
|
onionBlob[:])
|
||||||
needUpdate = true
|
needUpdate = true
|
||||||
|
|
||||||
log.Error("unable to decode onion obfuscator")
|
log.Errorf("unable to decode onion "+
|
||||||
|
"obfuscator: %v", failureCode)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1054,13 +1059,15 @@ func (l *channelLink) processLockedInHtlcs(
|
|||||||
onionReader, pd.RHash[:],
|
onionReader, pd.RHash[:],
|
||||||
)
|
)
|
||||||
if failureCode != lnwire.CodeNone {
|
if failureCode != lnwire.CodeNone {
|
||||||
// If we unable to process the onion blob than
|
// If we're unable to process the onion blob
|
||||||
// we should send the malformed htlc error to
|
// than we should send the malformed htlc error
|
||||||
// payment sender.
|
// to payment sender.
|
||||||
l.sendMalformedHTLCError(pd.RHash, failureCode, onionBlob[:])
|
l.sendMalformedHTLCError(pd.RHash, failureCode,
|
||||||
|
onionBlob[:])
|
||||||
needUpdate = true
|
needUpdate = true
|
||||||
|
|
||||||
log.Error("unable to decode onion hop iterator")
|
log.Errorf("unable to decode onion hop "+
|
||||||
|
"iterator: %v", failureCode)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1122,19 +1129,35 @@ func (l *channelLink) processLockedInHtlcs(
|
|||||||
|
|
||||||
// We'll also ensure that our time-lock value
|
// We'll also ensure that our time-lock value
|
||||||
// has been computed correctly.
|
// has been computed correctly.
|
||||||
if !l.cfg.DebugHTLC &&
|
expectedHeight := heightNow + l.cfg.FwrdingPolicy.TimeLockDelta
|
||||||
fwdInfo.OutgoingCTLV != l.cfg.FwrdingPolicy.TimeLockDelta {
|
if !l.cfg.DebugHTLC {
|
||||||
|
switch {
|
||||||
|
case fwdInfo.OutgoingCTLV < expectedHeight:
|
||||||
log.Errorf("Onion payload of incoming "+
|
log.Errorf("Onion payload of incoming "+
|
||||||
"htlc(%x) has incorrect time-lock: "+
|
"htlc(%x) has incorrect time-lock: "+
|
||||||
"expected %v, got %v",
|
"expected %v, got %v",
|
||||||
pd.RHash[:], l.cfg.FwrdingPolicy.TimeLockDelta,
|
pd.RHash[:], expectedHeight,
|
||||||
fwdInfo.OutgoingCTLV)
|
fwdInfo.OutgoingCTLV)
|
||||||
|
|
||||||
failure := lnwire.NewFinalIncorrectCltvExpiry(fwdInfo.OutgoingCTLV)
|
failure := lnwire.NewFinalIncorrectCltvExpiry(
|
||||||
|
fwdInfo.OutgoingCTLV,
|
||||||
|
)
|
||||||
l.sendHTLCError(pd.RHash, failure, obfuscator)
|
l.sendHTLCError(pd.RHash, failure, obfuscator)
|
||||||
needUpdate = true
|
needUpdate = true
|
||||||
continue
|
continue
|
||||||
|
case pd.Timeout != fwdInfo.OutgoingCTLV:
|
||||||
|
log.Errorf("HTLC(%x) has incorrect "+
|
||||||
|
"time-lock: expected %v, got %v",
|
||||||
|
pd.RHash[:], pd.Timeout,
|
||||||
|
fwdInfo.OutgoingCTLV)
|
||||||
|
|
||||||
|
failure := lnwire.NewFinalIncorrectCltvExpiry(
|
||||||
|
fwdInfo.OutgoingCTLV,
|
||||||
|
)
|
||||||
|
l.sendHTLCError(pd.RHash, failure, obfuscator)
|
||||||
|
needUpdate = true
|
||||||
|
continue
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// If we're not currently in debug mode, and
|
// If we're not currently in debug mode, and
|
||||||
@ -1307,6 +1330,8 @@ func (l *channelLink) processLockedInHtlcs(
|
|||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO(roasbeef): also add max timeout value
|
||||||
|
|
||||||
// With all our forwarding constraints met,
|
// With all our forwarding constraints met,
|
||||||
// we'll create the outgoing HTLC using the
|
// we'll create the outgoing HTLC using the
|
||||||
// parameters as specified in the forwarding
|
// parameters as specified in the forwarding
|
||||||
|
Loading…
Reference in New Issue
Block a user