channeldb+routing: move control tower interface to routing

This commit creates an empty shall for control tower in the routing
package. It is a preparation for adding event notification.
This commit is contained in:
Joost Jager 2019-05-29 08:57:04 +02:00
parent eb700d35e1
commit 87d3207baf
No known key found for this signature in database
GPG Key ID: A61B9D4C393C59C7
6 changed files with 109 additions and 45 deletions

@ -35,46 +35,14 @@ var (
ErrUnknownPaymentStatus = errors.New("unknown payment status") ErrUnknownPaymentStatus = errors.New("unknown payment status")
) )
// ControlTower tracks all outgoing payments made, whose primary purpose is to // PaymentControl implements persistence for payments and payment attempts.
// prevent duplicate payments to the same payment hash. In production, a type PaymentControl struct {
// persistent implementation is preferred so that tracking can survive across
// restarts. Payments are transitioned through various payment states, and the
// ControlTower interface provides access to driving the state transitions.
type ControlTower interface {
// InitPayment atomically moves the payment into the InFlight state.
// This method checks that no suceeded payment exist for this payment
// hash.
InitPayment(lntypes.Hash, *PaymentCreationInfo) error
// RegisterAttempt atomically records the provided PaymentAttemptInfo.
RegisterAttempt(lntypes.Hash, *PaymentAttemptInfo) error
// Success transitions a payment into the Succeeded state. After
// invoking this method, InitPayment should always return an error to
// prevent us from making duplicate payments to the same payment hash.
// The provided preimage is atomically saved to the DB for record
// keeping.
Success(lntypes.Hash, lntypes.Preimage) error
// Fail transitions a payment into the Failed state, and records the
// reason the payment failed. After invoking this method, InitPayment
// should return nil on its next call for this payment hash, allowing
// the switch to make a subsequent payment.
Fail(lntypes.Hash, FailureReason) error
// FetchInFlightPayments returns all payments with status InFlight.
FetchInFlightPayments() ([]*InFlightPayment, error)
}
// paymentControl is persistent implementation of ControlTower to restrict
// double payment sending.
type paymentControl struct {
db *DB db *DB
} }
// NewPaymentControl creates a new instance of the paymentControl. // NewPaymentControl creates a new instance of the PaymentControl.
func NewPaymentControl(db *DB) ControlTower { func NewPaymentControl(db *DB) *PaymentControl {
return &paymentControl{ return &PaymentControl{
db: db, db: db,
} }
} }
@ -83,7 +51,7 @@ func NewPaymentControl(db *DB) ControlTower {
// making sure it does not already exist as an in-flight payment. Then this // making sure it does not already exist as an in-flight payment. Then this
// method returns successfully, the payment is guranteeed to be in the InFlight // method returns successfully, the payment is guranteeed to be in the InFlight
// state. // state.
func (p *paymentControl) InitPayment(paymentHash lntypes.Hash, func (p *PaymentControl) InitPayment(paymentHash lntypes.Hash,
info *PaymentCreationInfo) error { info *PaymentCreationInfo) error {
var b bytes.Buffer var b bytes.Buffer
@ -173,7 +141,7 @@ func (p *paymentControl) InitPayment(paymentHash lntypes.Hash,
// RegisterAttempt atomically records the provided PaymentAttemptInfo to the // RegisterAttempt atomically records the provided PaymentAttemptInfo to the
// DB. // DB.
func (p *paymentControl) RegisterAttempt(paymentHash lntypes.Hash, func (p *PaymentControl) RegisterAttempt(paymentHash lntypes.Hash,
attempt *PaymentAttemptInfo) error { attempt *PaymentAttemptInfo) error {
// Serialize the information before opening the db transaction. // Serialize the information before opening the db transaction.
@ -218,7 +186,7 @@ func (p *paymentControl) RegisterAttempt(paymentHash lntypes.Hash,
// method, InitPayment should always return an error to prevent us from making // method, InitPayment should always return an error to prevent us from making
// duplicate payments to the same payment hash. The provided preimage is // duplicate payments to the same payment hash. The provided preimage is
// atomically saved to the DB for record keeping. // atomically saved to the DB for record keeping.
func (p *paymentControl) Success(paymentHash lntypes.Hash, func (p *PaymentControl) Success(paymentHash lntypes.Hash,
preimage lntypes.Preimage) error { preimage lntypes.Preimage) error {
var updateErr error var updateErr error
@ -257,7 +225,7 @@ func (p *paymentControl) Success(paymentHash lntypes.Hash,
// payment failed. After invoking this method, InitPayment should return nil on // payment failed. After invoking this method, InitPayment should return nil on
// its next call for this payment hash, allowing the switch to make a // its next call for this payment hash, allowing the switch to make a
// subsequent payment. // subsequent payment.
func (p *paymentControl) Fail(paymentHash lntypes.Hash, func (p *PaymentControl) Fail(paymentHash lntypes.Hash,
reason FailureReason) error { reason FailureReason) error {
var updateErr error var updateErr error
@ -402,7 +370,7 @@ type InFlightPayment struct {
} }
// FetchInFlightPayments returns all payments with status InFlight. // FetchInFlightPayments returns all payments with status InFlight.
func (p *paymentControl) FetchInFlightPayments() ([]*InFlightPayment, error) { func (p *PaymentControl) FetchInFlightPayments() ([]*InFlightPayment, error) {
var inFlights []*InFlightPayment var inFlights []*InFlightPayment
err := p.db.View(func(tx *bbolt.Tx) error { err := p.db.View(func(tx *bbolt.Tx) error {
payments := tx.Bucket(paymentsRootBucket) payments := tx.Bucket(paymentsRootBucket)

94
routing/control_tower.go Normal file

@ -0,0 +1,94 @@
package routing
import (
"github.com/lightningnetwork/lnd/channeldb"
"github.com/lightningnetwork/lnd/lntypes"
)
// ControlTower tracks all outgoing payments made, whose primary purpose is to
// prevent duplicate payments to the same payment hash. In production, a
// persistent implementation is preferred so that tracking can survive across
// restarts. Payments are transitioned through various payment states, and the
// ControlTower interface provides access to driving the state transitions.
type ControlTower interface {
// InitPayment atomically moves the payment into the InFlight state.
// This method checks that no suceeded payment exist for this payment
// hash.
InitPayment(lntypes.Hash, *channeldb.PaymentCreationInfo) error
// RegisterAttempt atomically records the provided PaymentAttemptInfo.
RegisterAttempt(lntypes.Hash, *channeldb.PaymentAttemptInfo) error
// Success transitions a payment into the Succeeded state. After
// invoking this method, InitPayment should always return an error to
// prevent us from making duplicate payments to the same payment hash.
// The provided preimage is atomically saved to the DB for record
// keeping.
Success(lntypes.Hash, lntypes.Preimage) error
// Fail transitions a payment into the Failed state, and records the
// reason the payment failed. After invoking this method, InitPayment
// should return nil on its next call for this payment hash, allowing
// the switch to make a subsequent payment.
Fail(lntypes.Hash, channeldb.FailureReason) error
// FetchInFlightPayments returns all payments with status InFlight.
FetchInFlightPayments() ([]*channeldb.InFlightPayment, error)
}
// controlTower is persistent implementation of ControlTower to restrict
// double payment sending.
type controlTower struct {
db *channeldb.PaymentControl
}
// NewControlTower creates a new instance of the controlTower.
func NewControlTower(db *channeldb.PaymentControl) ControlTower {
return &controlTower{
db: db,
}
}
// InitPayment checks or records the given PaymentCreationInfo with the DB,
// making sure it does not already exist as an in-flight payment. Then this
// method returns successfully, the payment is guranteeed to be in the InFlight
// state.
func (p *controlTower) InitPayment(paymentHash lntypes.Hash,
info *channeldb.PaymentCreationInfo) error {
return p.db.InitPayment(paymentHash, info)
}
// RegisterAttempt atomically records the provided PaymentAttemptInfo to the
// DB.
func (p *controlTower) RegisterAttempt(paymentHash lntypes.Hash,
attempt *channeldb.PaymentAttemptInfo) error {
return p.db.RegisterAttempt(paymentHash, attempt)
}
// Success transitions a payment into the Succeeded state. After invoking this
// method, InitPayment should always return an error to prevent us from making
// duplicate payments to the same payment hash. The provided preimage is
// atomically saved to the DB for record keeping.
func (p *controlTower) Success(paymentHash lntypes.Hash,
preimage lntypes.Preimage) error {
return p.db.Success(paymentHash, preimage)
}
// Fail transitions a payment into the Failed state, and records the reason the
// payment failed. After invoking this method, InitPayment should return nil on
// its next call for this payment hash, allowing the switch to make a
// subsequent payment.
func (p *controlTower) Fail(paymentHash lntypes.Hash,
reason channeldb.FailureReason) error {
return p.db.Fail(paymentHash, reason)
}
// FetchInFlightPayments returns all payments with status InFlight.
func (p *controlTower) FetchInFlightPayments() ([]*channeldb.InFlightPayment, error) {
return p.db.FetchInFlightPayments()
}

@ -182,7 +182,7 @@ type mockControlTower struct {
sync.Mutex sync.Mutex
} }
var _ channeldb.ControlTower = (*mockControlTower)(nil) var _ ControlTower = (*mockControlTower)(nil)
func makeMockControlTower() *mockControlTower { func makeMockControlTower() *mockControlTower {
return &mockControlTower{ return &mockControlTower{

@ -218,7 +218,7 @@ type Config struct {
// Control keeps track of the status of ongoing payments, ensuring we // Control keeps track of the status of ongoing payments, ensuring we
// can properly resume them across restarts. // can properly resume them across restarts.
Control channeldb.ControlTower Control ControlTower
// MissionControl is a shared memory of sorts that executions of // MissionControl is a shared memory of sorts that executions of
// payment path finding use in order to remember which vertexes/edges // payment path finding use in order to remember which vertexes/edges

@ -649,12 +649,14 @@ func newServer(listenAddrs []net.Addr, chanDB *channeldb.DB, cc *chainControl,
routerrpc.GetMissionControlConfig(cfg.SubRPCServers.RouterRPC), routerrpc.GetMissionControlConfig(cfg.SubRPCServers.RouterRPC),
) )
paymentControl := channeldb.NewPaymentControl(chanDB)
s.chanRouter, err = routing.New(routing.Config{ s.chanRouter, err = routing.New(routing.Config{
Graph: chanGraph, Graph: chanGraph,
Chain: cc.chainIO, Chain: cc.chainIO,
ChainView: cc.chainView, ChainView: cc.chainView,
Payer: s.htlcSwitch, Payer: s.htlcSwitch,
Control: channeldb.NewPaymentControl(chanDB), Control: routing.NewControlTower(paymentControl),
MissionControl: s.missionControl, MissionControl: s.missionControl,
ChannelPruneExpiry: routing.DefaultChannelPruneExpiry, ChannelPruneExpiry: routing.DefaultChannelPruneExpiry,
GraphPruneInterval: time.Duration(time.Hour), GraphPruneInterval: time.Duration(time.Hour),