mirror of
https://github.com/lightningnetwork/lnd.git
synced 2025-01-19 05:45:21 +01:00
htlcswitch: add ClearTextError interface
This commit adds a ClearTextError interface which is implemented by non-opaque errors that we know the underlying wire failure message for. This interface is implemented by ForwardingErrors, because we can fully decrypt the onion blob to obtain the underlying failure reason. This interface will also be implemented by errors which originate at our node in following commits, because we know the failure reason when we fail the htlc. The lnwire interface is un-embedded in the ForwardingError struct in favour of implementing this interface. This change is made to protect against accidental passing of a ForwardingError to the wire, where the embedded FailureMessage interface will present as wire failure but will not serialize properly.
This commit is contained in:
parent
6f0a342f92
commit
102f9b003f
@ -9,6 +9,22 @@ import (
|
|||||||
"github.com/lightningnetwork/lnd/lnwire"
|
"github.com/lightningnetwork/lnd/lnwire"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// ClearTextError is an interface which is implemented by errors that occur
|
||||||
|
// when we know the underlying wire failure message. These errors are the
|
||||||
|
// opposite to opaque errors which are onion-encrypted blobs only understandable
|
||||||
|
// to the initiating node. ClearTextErrors are used when we fail a htlc at our
|
||||||
|
// node, or one of our initiated payments failed and we can decrypt the onion
|
||||||
|
// encrypted error fully.
|
||||||
|
type ClearTextError interface {
|
||||||
|
error
|
||||||
|
|
||||||
|
// WireMessage extracts a valid wire failure message from an internal
|
||||||
|
// error which may contain additional metadata (which should not be
|
||||||
|
// exposed to the network). This value may be nil in the case where
|
||||||
|
// an unknown wire error is returned by one of our peers.
|
||||||
|
WireMessage() lnwire.FailureMessage
|
||||||
|
}
|
||||||
|
|
||||||
// ForwardingError wraps an lnwire.FailureMessage in a struct that also
|
// ForwardingError wraps an lnwire.FailureMessage in a struct that also
|
||||||
// includes the source of the error.
|
// includes the source of the error.
|
||||||
type ForwardingError struct {
|
type ForwardingError struct {
|
||||||
@ -22,7 +38,20 @@ type ForwardingError struct {
|
|||||||
// order to provide context specific error details.
|
// order to provide context specific error details.
|
||||||
ExtraMsg string
|
ExtraMsg string
|
||||||
|
|
||||||
lnwire.FailureMessage
|
// msg is the wire message associated with the error. This value may
|
||||||
|
// be nil in the case where we fail to decode failure message sent by
|
||||||
|
// a peer.
|
||||||
|
msg lnwire.FailureMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// WireMessage extracts a valid wire failure message from an internal
|
||||||
|
// error which may contain additional metadata (which should not be
|
||||||
|
// exposed to the network). This value may be nil in the case where
|
||||||
|
// an unknown wire error is returned by one of our peers.
|
||||||
|
//
|
||||||
|
// Note this is part of the ClearTextError interface.
|
||||||
|
func (f *ForwardingError) WireMessage() lnwire.FailureMessage {
|
||||||
|
return f.msg
|
||||||
}
|
}
|
||||||
|
|
||||||
// Error implements the built-in error interface. We use this method to allow
|
// Error implements the built-in error interface. We use this method to allow
|
||||||
@ -30,13 +59,11 @@ type ForwardingError struct {
|
|||||||
// returned.
|
// returned.
|
||||||
func (f *ForwardingError) Error() string {
|
func (f *ForwardingError) Error() string {
|
||||||
if f.ExtraMsg == "" {
|
if f.ExtraMsg == "" {
|
||||||
return fmt.Sprintf(
|
return fmt.Sprintf("%v@%v", f.msg, f.FailureSourceIdx)
|
||||||
"%v@%v", f.FailureMessage, f.FailureSourceIdx,
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return fmt.Sprintf(
|
return fmt.Sprintf(
|
||||||
"%v@%v: %v", f.FailureMessage, f.FailureSourceIdx, f.ExtraMsg,
|
"%v@%v: %v", f.msg, f.FailureSourceIdx, f.ExtraMsg,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -47,7 +74,7 @@ func NewForwardingError(failure lnwire.FailureMessage, index int,
|
|||||||
|
|
||||||
return &ForwardingError{
|
return &ForwardingError{
|
||||||
FailureSourceIdx: index,
|
FailureSourceIdx: index,
|
||||||
FailureMessage: failure,
|
msg: failure,
|
||||||
ExtraMsg: extraMsg,
|
ExtraMsg: extraMsg,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -579,7 +579,7 @@ func TestExitNodeTimelockPayloadMismatch(t *testing.T) {
|
|||||||
t.Fatalf("expected a ForwardingError, instead got: %T", err)
|
t.Fatalf("expected a ForwardingError, instead got: %T", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
switch ferr.FailureMessage.(type) {
|
switch ferr.WireMessage().(type) {
|
||||||
case *lnwire.FailFinalIncorrectCltvExpiry:
|
case *lnwire.FailFinalIncorrectCltvExpiry:
|
||||||
default:
|
default:
|
||||||
t.Fatalf("incorrect error, expected incorrect cltv expiry, "+
|
t.Fatalf("incorrect error, expected incorrect cltv expiry, "+
|
||||||
@ -679,7 +679,7 @@ func TestLinkForwardTimelockPolicyMismatch(t *testing.T) {
|
|||||||
t.Fatalf("expected a ForwardingError, instead got: %T", err)
|
t.Fatalf("expected a ForwardingError, instead got: %T", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
switch ferr.FailureMessage.(type) {
|
switch ferr.WireMessage().(type) {
|
||||||
case *lnwire.FailIncorrectCltvExpiry:
|
case *lnwire.FailIncorrectCltvExpiry:
|
||||||
default:
|
default:
|
||||||
t.Fatalf("incorrect error, expected incorrect cltv expiry, "+
|
t.Fatalf("incorrect error, expected incorrect cltv expiry, "+
|
||||||
@ -737,7 +737,7 @@ func TestLinkForwardFeePolicyMismatch(t *testing.T) {
|
|||||||
t.Fatalf("expected a ForwardingError, instead got: %T", err)
|
t.Fatalf("expected a ForwardingError, instead got: %T", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
switch ferr.FailureMessage.(type) {
|
switch ferr.WireMessage().(type) {
|
||||||
case *lnwire.FailFeeInsufficient:
|
case *lnwire.FailFeeInsufficient:
|
||||||
default:
|
default:
|
||||||
t.Fatalf("incorrect error, expected fee insufficient, "+
|
t.Fatalf("incorrect error, expected fee insufficient, "+
|
||||||
@ -795,7 +795,7 @@ func TestLinkForwardMinHTLCPolicyMismatch(t *testing.T) {
|
|||||||
t.Fatalf("expected a ForwardingError, instead got: %T", err)
|
t.Fatalf("expected a ForwardingError, instead got: %T", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
switch ferr.FailureMessage.(type) {
|
switch ferr.WireMessage().(type) {
|
||||||
case *lnwire.FailAmountBelowMinimum:
|
case *lnwire.FailAmountBelowMinimum:
|
||||||
default:
|
default:
|
||||||
t.Fatalf("incorrect error, expected amount below minimum, "+
|
t.Fatalf("incorrect error, expected amount below minimum, "+
|
||||||
@ -862,7 +862,7 @@ func TestLinkForwardMaxHTLCPolicyMismatch(t *testing.T) {
|
|||||||
t.Fatalf("expected a ForwardingError, instead got: %T", err)
|
t.Fatalf("expected a ForwardingError, instead got: %T", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
switch ferr.FailureMessage.(type) {
|
switch ferr.WireMessage().(type) {
|
||||||
case *lnwire.FailTemporaryChannelFailure:
|
case *lnwire.FailTemporaryChannelFailure:
|
||||||
default:
|
default:
|
||||||
t.Fatalf("incorrect error, expected temporary channel failure, "+
|
t.Fatalf("incorrect error, expected temporary channel failure, "+
|
||||||
@ -968,7 +968,7 @@ func TestUpdateForwardingPolicy(t *testing.T) {
|
|||||||
if !ok {
|
if !ok {
|
||||||
t.Fatalf("expected a ForwardingError, instead got (%T): %v", err, err)
|
t.Fatalf("expected a ForwardingError, instead got (%T): %v", err, err)
|
||||||
}
|
}
|
||||||
switch ferr.FailureMessage.(type) {
|
switch ferr.WireMessage().(type) {
|
||||||
case *lnwire.FailFeeInsufficient:
|
case *lnwire.FailFeeInsufficient:
|
||||||
default:
|
default:
|
||||||
t.Fatalf("expected FailFeeInsufficient instead got: %v", err)
|
t.Fatalf("expected FailFeeInsufficient instead got: %v", err)
|
||||||
@ -1008,7 +1008,7 @@ func TestUpdateForwardingPolicy(t *testing.T) {
|
|||||||
t.Fatalf("expected a ForwardingError, instead got (%T): %v",
|
t.Fatalf("expected a ForwardingError, instead got (%T): %v",
|
||||||
err, err)
|
err, err)
|
||||||
}
|
}
|
||||||
switch ferr.FailureMessage.(type) {
|
switch ferr.WireMessage().(type) {
|
||||||
case *lnwire.FailTemporaryChannelFailure:
|
case *lnwire.FailTemporaryChannelFailure:
|
||||||
default:
|
default:
|
||||||
t.Fatalf("expected TemporaryChannelFailure, instead got: %v",
|
t.Fatalf("expected TemporaryChannelFailure, instead got: %v",
|
||||||
@ -1253,9 +1253,9 @@ func TestChannelLinkMultiHopUnknownNextHop(t *testing.T) {
|
|||||||
if !ok {
|
if !ok {
|
||||||
t.Fatalf("expected ForwardingError")
|
t.Fatalf("expected ForwardingError")
|
||||||
}
|
}
|
||||||
if _, ok = fErr.FailureMessage.(*lnwire.FailUnknownNextPeer); !ok {
|
if _, ok = fErr.WireMessage().(*lnwire.FailUnknownNextPeer); !ok {
|
||||||
t.Fatalf("wrong error has been received: %T",
|
t.Fatalf("wrong error has been received: %T",
|
||||||
fErr.FailureMessage)
|
fErr.WireMessage())
|
||||||
}
|
}
|
||||||
|
|
||||||
// Wait for Alice to receive the revocation.
|
// Wait for Alice to receive the revocation.
|
||||||
@ -1369,7 +1369,7 @@ func TestChannelLinkMultiHopDecodeError(t *testing.T) {
|
|||||||
t.Fatalf("expected a ForwardingError, instead got: %T", err)
|
t.Fatalf("expected a ForwardingError, instead got: %T", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
switch ferr.FailureMessage.(type) {
|
switch ferr.WireMessage().(type) {
|
||||||
case *lnwire.FailInvalidOnionVersion:
|
case *lnwire.FailInvalidOnionVersion:
|
||||||
default:
|
default:
|
||||||
t.Fatalf("wrong error have been received: %v", err)
|
t.Fatalf("wrong error have been received: %v", err)
|
||||||
@ -1462,7 +1462,7 @@ func TestChannelLinkExpiryTooSoonExitNode(t *testing.T) {
|
|||||||
err, err)
|
err, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
switch ferr.FailureMessage.(type) {
|
switch ferr.WireMessage().(type) {
|
||||||
case *lnwire.FailIncorrectDetails:
|
case *lnwire.FailIncorrectDetails:
|
||||||
default:
|
default:
|
||||||
t.Fatalf("expected incorrect_or_unknown_payment_details, "+
|
t.Fatalf("expected incorrect_or_unknown_payment_details, "+
|
||||||
@ -1524,7 +1524,7 @@ func TestChannelLinkExpiryTooSoonMidNode(t *testing.T) {
|
|||||||
t.Fatalf("expected a ForwardingError, instead got: %T: %v", err, err)
|
t.Fatalf("expected a ForwardingError, instead got: %T: %v", err, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
switch ferr.FailureMessage.(type) {
|
switch ferr.WireMessage().(type) {
|
||||||
case *lnwire.FailExpiryTooSoon:
|
case *lnwire.FailExpiryTooSoon:
|
||||||
default:
|
default:
|
||||||
t.Fatalf("incorrect error, expected final time lock too "+
|
t.Fatalf("incorrect error, expected final time lock too "+
|
||||||
@ -5636,7 +5636,7 @@ func TestChannelLinkCanceledInvoice(t *testing.T) {
|
|||||||
if !ok {
|
if !ok {
|
||||||
t.Fatalf("expected ForwardingError, but got %v", err)
|
t.Fatalf("expected ForwardingError, but got %v", err)
|
||||||
}
|
}
|
||||||
_, ok = fErr.FailureMessage.(*lnwire.FailIncorrectDetails)
|
_, ok = fErr.WireMessage().(*lnwire.FailIncorrectDetails)
|
||||||
if !ok {
|
if !ok {
|
||||||
t.Fatalf("expected unknown payment hash, but got %v", err)
|
t.Fatalf("expected unknown payment hash, but got %v", err)
|
||||||
}
|
}
|
||||||
@ -6221,8 +6221,8 @@ func assertFailureCode(t *testing.T, err error, code lnwire.FailCode) {
|
|||||||
t.Fatalf("expected ForwardingError but got %T", err)
|
t.Fatalf("expected ForwardingError but got %T", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if fErr.FailureMessage.Code() != code {
|
if fErr.WireMessage().Code() != code {
|
||||||
t.Fatalf("expected %v but got %v",
|
t.Fatalf("expected %v but got %v",
|
||||||
code, fErr.FailureMessage.Code())
|
code, fErr.WireMessage().Code())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2179,10 +2179,10 @@ func TestUpdateFailMalformedHTLCErrorConversion(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fwdingErr := err.(*ForwardingError)
|
fwdingErr := err.(*ForwardingError)
|
||||||
failureMsg := fwdingErr.FailureMessage
|
failureMsg := fwdingErr.WireMessage()
|
||||||
if _, ok := failureMsg.(*lnwire.FailInvalidOnionKey); !ok {
|
if _, ok := failureMsg.(*lnwire.FailInvalidOnionKey); !ok {
|
||||||
t.Fatalf("expected onion failure instead got: %v",
|
t.Fatalf("expected onion failure instead got: %v",
|
||||||
fwdingErr.FailureMessage)
|
fwdingErr.WireMessage())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2448,7 +2448,7 @@ func TestInvalidFailure(t *testing.T) {
|
|||||||
if fErr.FailureSourceIdx != 2 {
|
if fErr.FailureSourceIdx != 2 {
|
||||||
t.Fatal("unexpected error source index")
|
t.Fatal("unexpected error source index")
|
||||||
}
|
}
|
||||||
if fErr.FailureMessage != nil {
|
if fErr.WireMessage() != nil {
|
||||||
t.Fatal("expected empty failure message")
|
t.Fatal("expected empty failure message")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -331,7 +331,7 @@ func marshallError(sendError error) (*Failure, error) {
|
|||||||
return nil, sendError
|
return nil, sendError
|
||||||
}
|
}
|
||||||
|
|
||||||
switch onionErr := fErr.FailureMessage.(type) {
|
switch onionErr := fErr.WireMessage().(type) {
|
||||||
|
|
||||||
case *lnwire.FailIncorrectDetails:
|
case *lnwire.FailIncorrectDetails:
|
||||||
response.Code = Failure_INCORRECT_OR_UNKNOWN_PAYMENT_DETAILS
|
response.Code = Failure_INCORRECT_OR_UNKNOWN_PAYMENT_DETAILS
|
||||||
|
@ -1917,7 +1917,7 @@ func (r *ChannelRouter) processSendError(paymentID uint64, rt *route.Route,
|
|||||||
return &internalErrorReason
|
return &internalErrorReason
|
||||||
}
|
}
|
||||||
|
|
||||||
failureMessage := fErr.FailureMessage
|
failureMessage := fErr.WireMessage()
|
||||||
failureSourceIdx := fErr.FailureSourceIdx
|
failureSourceIdx := fErr.FailureSourceIdx
|
||||||
|
|
||||||
// Apply channel update if the error contains one. For unknown
|
// Apply channel update if the error contains one. For unknown
|
||||||
|
@ -3320,7 +3320,7 @@ func TestSendToRouteStructuredError(t *testing.T) {
|
|||||||
t.Fatalf("expected forwarding error")
|
t.Fatalf("expected forwarding error")
|
||||||
}
|
}
|
||||||
|
|
||||||
if _, ok := fErr.FailureMessage.(*lnwire.FailFeeInsufficient); !ok {
|
if _, ok := fErr.WireMessage().(*lnwire.FailFeeInsufficient); !ok {
|
||||||
t.Fatalf("expected fee insufficient error")
|
t.Fatalf("expected fee insufficient error")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user