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:
parent
eb700d35e1
commit
87d3207baf
@ -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
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),
|
||||||
|
Loading…
Reference in New Issue
Block a user