routing/payment_lifecycle: extract attempt sending logic

Define shardHandler which is a struct holding what is needed to send
attempts along given routes. The reason we define the logic on this
struct instead of the paymentLifecycle is that we later will make
SendToRoute calls not go through the payment lifecycle, but only using
this struct.

The launch shard is responsible for registering the attempt with the
control tower, failing it if the launch fails. Note that it is NOT
responsible for marking the _payment_ failed in case a terminal error is
encountered. This is important since we will later reuse this method for
SendToRoute, where whether to fail the payment cannot be decided on the
shard level.
This commit is contained in:
Johan T. Halseth 2020-04-01 00:13:24 +02:00
parent bcca1ab821
commit 9712dd1a7f
No known key found for this signature in database
GPG Key ID: 15BAADA29DA20D26
2 changed files with 99 additions and 47 deletions

@ -37,11 +37,15 @@ type paymentLifecycle struct {
paySession PaymentSession paySession PaymentSession
timeoutChan <-chan time.Time timeoutChan <-chan time.Time
currentHeight int32 currentHeight int32
lastError error
} }
// resumePayment resumes the paymentLifecycle from the current state. // resumePayment resumes the paymentLifecycle from the current state.
func (p *paymentLifecycle) resumePayment() ([32]byte, *route.Route, error) { func (p *paymentLifecycle) resumePayment() ([32]byte, *route.Route, error) {
shardHandler := &shardHandler{
router: p.router,
paymentHash: p.paymentHash,
}
// We'll continue until either our payment succeeds, or we encounter a // We'll continue until either our payment succeeds, or we encounter a
// critical error during path finding. // critical error during path finding.
for { for {
@ -144,19 +148,20 @@ func (p *paymentLifecycle) resumePayment() ([32]byte, *route.Route, error) {
// If there was an error already recorded for this // If there was an error already recorded for this
// payment, we'll return that. // payment, we'll return that.
if p.lastError != nil { if shardHandler.lastError != nil {
return [32]byte{}, nil, errNoRoute{lastError: p.lastError} return [32]byte{}, nil, errNoRoute{
lastError: shardHandler.lastError,
}
} }
// Terminal state, return. // Terminal state, return.
return [32]byte{}, nil, err return [32]byte{}, nil, err
} }
// Using the route received from the payment session, // With the route in hand, launch a new shard.
// create a new shard to send. var outcome *launchOutcome
firstHop, htlcAdd, a, err := p.createNewPaymentAttempt( attempt, outcome, err = shardHandler.launchShard(rt)
rt,
)
// With SendToRoute, it can happen that the route exceeds protocol // With SendToRoute, it can happen that the route exceeds protocol
// constraints. Mark the payment as failed with an internal error. // constraints. Mark the payment as failed with an internal error.
if err == route.ErrMaxRouteHopsExceeded || if err == route.ErrMaxRouteHopsExceeded ||
@ -178,43 +183,21 @@ func (p *paymentLifecycle) resumePayment() ([32]byte, *route.Route, error) {
return [32]byte{}, nil, err return [32]byte{}, nil, err
} }
attempt = a // We ew encountered a non-critical error when
// launching the shard, handle it
// Before sending this HTLC to the switch, we checkpoint the if outcome.err != nil {
// fresh paymentID and route to the DB. This lets us know on
// startup the ID of the payment that we attempted to send,
// such that we can query the Switch for its whereabouts. The
// route is needed to handle the result when it eventually
// comes back.
err = p.router.cfg.Control.RegisterAttempt(p.paymentHash, attempt)
if err != nil {
return [32]byte{}, nil, err
}
// Now that the attempt is created and checkpointed to
// the DB, we send it.
sendErr := p.sendPaymentAttempt(
attempt, firstHop, htlcAdd,
)
if sendErr != nil {
// TODO(joostjager): Distinguish unexpected
// internal errors from real send errors.
err = p.failAttempt(attempt, sendErr)
if err != nil {
return [32]byte{}, nil, err
}
// We must inspect the error to know whether it // We must inspect the error to know whether it
// was critical or not, to decide whether we // was critical or not, to decide whether we
// should continue trying. // should continue trying.
err := p.handleSendError(attempt, sendErr) err = shardHandler.handleSendError(
attempt, outcome.err,
)
if err != nil { if err != nil {
return [32]byte{}, nil, err return [32]byte{}, nil, err
} }
// Error was handled successfully, reset the // Error was handled successfully, continue to
// attempt to indicate we want to make a new // make a new attempt.
// attempt.
continue continue
} }
} }
@ -250,7 +233,7 @@ func (p *paymentLifecycle) resumePayment() ([32]byte, *route.Route, error) {
"the Switch, retrying.", attempt.AttemptID, "the Switch, retrying.", attempt.AttemptID,
p.paymentHash) p.paymentHash)
err = p.failAttempt(attempt, err) err = shardHandler.failAttempt(attempt, err)
if err != nil { if err != nil {
return [32]byte{}, nil, err return [32]byte{}, nil, err
} }
@ -290,7 +273,7 @@ func (p *paymentLifecycle) resumePayment() ([32]byte, *route.Route, error) {
log.Errorf("Attempt to send payment %x failed: %v", log.Errorf("Attempt to send payment %x failed: %v",
p.paymentHash, result.Error) p.paymentHash, result.Error)
err = p.failAttempt(attempt, result.Error) err = shardHandler.failAttempt(attempt, result.Error)
if err != nil { if err != nil {
return [32]byte{}, nil, err return [32]byte{}, nil, err
} }
@ -298,7 +281,10 @@ func (p *paymentLifecycle) resumePayment() ([32]byte, *route.Route, error) {
// We must inspect the error to know whether it was // We must inspect the error to know whether it was
// critical or not, to decide whether we should // critical or not, to decide whether we should
// continue trying. // continue trying.
if err := p.handleSendError(attempt, result.Error); err != nil { err := shardHandler.handleSendError(
attempt, result.Error,
)
if err != nil {
return [32]byte{}, nil, err return [32]byte{}, nil, err
} }
@ -337,6 +323,74 @@ func (p *paymentLifecycle) resumePayment() ([32]byte, *route.Route, error) {
} }
} }
// shardHandler holds what is necessary to send and collect the result of
// shards.
type shardHandler struct {
paymentHash lntypes.Hash
router *ChannelRouter
lastError error
}
// launchOutcome is a type returned from launchShard that indicates whether the
// shard was successfully send onto the network.
type launchOutcome struct {
// err is non-nil if a non-critical error was encountered when trying
// to send the shard, and we successfully updated the control tower to
// reflect this error. This can be errors like not enough local
// balance for the given route etc.
err error
}
// launchShard creates and sends an HTLC attempt along the given route,
// registering it with the control tower before sending it. It returns the
// HTLCAttemptInfo that was created for the shard, along with a launchOutcome.
// The launchOutcome is used to indicate whether the attempt was successfully
// sent. If the launchOutcome wraps a non-nil error, it means that the attempt
// was not sent onto the network, so no result will be available in the future
// for it.
func (p *shardHandler) launchShard(rt *route.Route) (*channeldb.HTLCAttemptInfo,
*launchOutcome, error) {
// Using the route received from the payment session, create a new
// shard to send.
firstHop, htlcAdd, attempt, err := p.createNewPaymentAttempt(
rt,
)
if err != nil {
return nil, nil, err
}
// Before sending this HTLC to the switch, we checkpoint the fresh
// paymentID and route to the DB. This lets us know on startup the ID
// of the payment that we attempted to send, such that we can query the
// Switch for its whereabouts. The route is needed to handle the result
// when it eventually comes back.
err = p.router.cfg.Control.RegisterAttempt(p.paymentHash, attempt)
if err != nil {
return nil, nil, err
}
// Now that the attempt is created and checkpointed to the DB, we send
// it.
sendErr := p.sendPaymentAttempt(attempt, firstHop, htlcAdd)
if sendErr != nil {
// TODO(joostjager): Distinguish unexpected internal errors
// from real send errors.
err := p.failAttempt(attempt, sendErr)
if err != nil {
return nil, nil, err
}
// Return a launchOutcome indicating the shard failed.
return attempt, &launchOutcome{
err: sendErr,
}, nil
}
return attempt, &launchOutcome{}, nil
}
// errorToPaymentFailure takes a path finding error and converts it into a // errorToPaymentFailure takes a path finding error and converts it into a
// payment-level failure. // payment-level failure.
func errorToPaymentFailure(err error) channeldb.FailureReason { func errorToPaymentFailure(err error) channeldb.FailureReason {
@ -357,7 +411,7 @@ func errorToPaymentFailure(err error) channeldb.FailureReason {
} }
// createNewPaymentAttempt creates a new payment attempt from the given route. // createNewPaymentAttempt creates a new payment attempt from the given route.
func (p *paymentLifecycle) createNewPaymentAttempt(rt *route.Route) ( func (p *shardHandler) createNewPaymentAttempt(rt *route.Route) (
lnwire.ShortChannelID, *lnwire.UpdateAddHTLC, lnwire.ShortChannelID, *lnwire.UpdateAddHTLC,
*channeldb.HTLCAttemptInfo, error) { *channeldb.HTLCAttemptInfo, error) {
@ -414,7 +468,7 @@ func (p *paymentLifecycle) createNewPaymentAttempt(rt *route.Route) (
} }
// sendPaymentAttempt attempts to send the current attempt to the switch. // sendPaymentAttempt attempts to send the current attempt to the switch.
func (p *paymentLifecycle) sendPaymentAttempt( func (p *shardHandler) sendPaymentAttempt(
attempt *channeldb.HTLCAttemptInfo, firstHop lnwire.ShortChannelID, attempt *channeldb.HTLCAttemptInfo, firstHop lnwire.ShortChannelID,
htlcAdd *lnwire.UpdateAddHTLC) error { htlcAdd *lnwire.UpdateAddHTLC) error {
@ -447,7 +501,7 @@ func (p *paymentLifecycle) sendPaymentAttempt(
// handleSendError inspects the given error from the Switch and determines // handleSendError inspects the given error from the Switch and determines
// whether we should make another payment attempt. // whether we should make another payment attempt.
func (p *paymentLifecycle) handleSendError(attempt *channeldb.HTLCAttemptInfo, func (p *shardHandler) handleSendError(attempt *channeldb.HTLCAttemptInfo,
sendErr error) error { sendErr error) error {
reason := p.router.processSendError( reason := p.router.processSendError(
@ -457,7 +511,6 @@ func (p *paymentLifecycle) handleSendError(attempt *channeldb.HTLCAttemptInfo,
// Save the forwarding error so it can be returned if // Save the forwarding error so it can be returned if
// this turns out to be the last attempt. // this turns out to be the last attempt.
p.lastError = sendErr p.lastError = sendErr
return nil return nil
} }
@ -480,7 +533,7 @@ func (p *paymentLifecycle) handleSendError(attempt *channeldb.HTLCAttemptInfo,
} }
// failAttempt calls control tower to fail the current payment attempt. // failAttempt calls control tower to fail the current payment attempt.
func (p *paymentLifecycle) failAttempt(attempt *channeldb.HTLCAttemptInfo, func (p *shardHandler) failAttempt(attempt *channeldb.HTLCAttemptInfo,
sendError error) error { sendError error) error {
failInfo := marshallError( failInfo := marshallError(

@ -1827,7 +1827,6 @@ func (r *ChannelRouter) sendPayment(
paymentHash: paymentHash, paymentHash: paymentHash,
paySession: paySession, paySession: paySession,
currentHeight: currentHeight, currentHeight: currentHeight,
lastError: nil,
} }
// If a timeout is specified, create a timeout channel. If no timeout is // If a timeout is specified, create a timeout channel. If no timeout is