Merge pull request #4666 from halseth/autopilot-mis-totalfunds
[autopilot] Use local channel balance in allocation calculation
This commit is contained in:
commit
46225e2921
@ -65,11 +65,11 @@ type Config struct {
|
|||||||
// channelState is a type that represents the set of active channels of the
|
// channelState is a type that represents the set of active channels of the
|
||||||
// backing LN node that the Agent should be aware of. This type contains a few
|
// backing LN node that the Agent should be aware of. This type contains a few
|
||||||
// helper utility methods.
|
// helper utility methods.
|
||||||
type channelState map[lnwire.ShortChannelID]Channel
|
type channelState map[lnwire.ShortChannelID]LocalChannel
|
||||||
|
|
||||||
// Channels returns a slice of all the active channels.
|
// Channels returns a slice of all the active channels.
|
||||||
func (c channelState) Channels() []Channel {
|
func (c channelState) Channels() []LocalChannel {
|
||||||
chans := make([]Channel, 0, len(c))
|
chans := make([]LocalChannel, 0, len(c))
|
||||||
for _, channel := range c {
|
for _, channel := range c {
|
||||||
chans = append(chans, channel)
|
chans = append(chans, channel)
|
||||||
}
|
}
|
||||||
@ -163,7 +163,7 @@ type Agent struct {
|
|||||||
// initiated, but haven't yet been confirmed as being fully opened.
|
// initiated, but haven't yet been confirmed as being fully opened.
|
||||||
// This state is required as otherwise, we may go over our allotted
|
// This state is required as otherwise, we may go over our allotted
|
||||||
// channel limit, or open multiple channels to the same node.
|
// channel limit, or open multiple channels to the same node.
|
||||||
pendingOpens map[NodeID]Channel
|
pendingOpens map[NodeID]LocalChannel
|
||||||
pendingMtx sync.Mutex
|
pendingMtx sync.Mutex
|
||||||
|
|
||||||
quit chan struct{}
|
quit chan struct{}
|
||||||
@ -174,10 +174,10 @@ type Agent struct {
|
|||||||
// configuration and initial channel state. The initial channel state slice
|
// configuration and initial channel state. The initial channel state slice
|
||||||
// should be populated with the set of Channels that are currently opened by
|
// should be populated with the set of Channels that are currently opened by
|
||||||
// the backing Lightning Node.
|
// the backing Lightning Node.
|
||||||
func New(cfg Config, initialState []Channel) (*Agent, error) {
|
func New(cfg Config, initialState []LocalChannel) (*Agent, error) {
|
||||||
a := &Agent{
|
a := &Agent{
|
||||||
cfg: cfg,
|
cfg: cfg,
|
||||||
chanState: make(map[lnwire.ShortChannelID]Channel),
|
chanState: make(map[lnwire.ShortChannelID]LocalChannel),
|
||||||
quit: make(chan struct{}),
|
quit: make(chan struct{}),
|
||||||
stateUpdates: make(chan interface{}),
|
stateUpdates: make(chan interface{}),
|
||||||
balanceUpdates: make(chan *balanceUpdate, 1),
|
balanceUpdates: make(chan *balanceUpdate, 1),
|
||||||
@ -187,7 +187,7 @@ func New(cfg Config, initialState []Channel) (*Agent, error) {
|
|||||||
pendingOpenUpdates: make(chan *chanPendingOpenUpdate, 1),
|
pendingOpenUpdates: make(chan *chanPendingOpenUpdate, 1),
|
||||||
failedNodes: make(map[NodeID]struct{}),
|
failedNodes: make(map[NodeID]struct{}),
|
||||||
pendingConns: make(map[NodeID]struct{}),
|
pendingConns: make(map[NodeID]struct{}),
|
||||||
pendingOpens: make(map[NodeID]Channel),
|
pendingOpens: make(map[NodeID]LocalChannel),
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, c := range initialState {
|
for _, c := range initialState {
|
||||||
@ -249,7 +249,7 @@ type nodeUpdates struct{}
|
|||||||
// channel has been opened, either by the Agent itself (within the main
|
// channel has been opened, either by the Agent itself (within the main
|
||||||
// controller loop), or by an external user to the system.
|
// controller loop), or by an external user to the system.
|
||||||
type chanOpenUpdate struct {
|
type chanOpenUpdate struct {
|
||||||
newChan Channel
|
newChan LocalChannel
|
||||||
}
|
}
|
||||||
|
|
||||||
// chanPendingOpenUpdate is a type of external state update that indicates a new
|
// chanPendingOpenUpdate is a type of external state update that indicates a new
|
||||||
@ -294,7 +294,7 @@ func (a *Agent) OnNodeUpdates() {
|
|||||||
|
|
||||||
// OnChannelOpen is a callback that should be executed each time a new channel
|
// OnChannelOpen is a callback that should be executed each time a new channel
|
||||||
// is manually opened by the user or any system outside the autopilot agent.
|
// is manually opened by the user or any system outside the autopilot agent.
|
||||||
func (a *Agent) OnChannelOpen(c Channel) {
|
func (a *Agent) OnChannelOpen(c LocalChannel) {
|
||||||
a.wg.Add(1)
|
a.wg.Add(1)
|
||||||
go func() {
|
go func() {
|
||||||
defer a.wg.Done()
|
defer a.wg.Done()
|
||||||
@ -356,7 +356,7 @@ func (a *Agent) OnHeuristicUpdate(h AttachmentHeuristic) {
|
|||||||
// channels open to, with the other sets of nodes that should be removed from
|
// channels open to, with the other sets of nodes that should be removed from
|
||||||
// consideration during heuristic selection. This ensures that the Agent doesn't
|
// consideration during heuristic selection. This ensures that the Agent doesn't
|
||||||
// attempt to open any "duplicate" channels to the same node.
|
// attempt to open any "duplicate" channels to the same node.
|
||||||
func mergeNodeMaps(c map[NodeID]Channel,
|
func mergeNodeMaps(c map[NodeID]LocalChannel,
|
||||||
skips ...map[NodeID]struct{}) map[NodeID]struct{} {
|
skips ...map[NodeID]struct{}) map[NodeID]struct{} {
|
||||||
|
|
||||||
numNodes := len(c)
|
numNodes := len(c)
|
||||||
@ -380,11 +380,11 @@ func mergeNodeMaps(c map[NodeID]Channel,
|
|||||||
// mergeChanState merges the Agent's set of active channels, with the set of
|
// mergeChanState merges the Agent's set of active channels, with the set of
|
||||||
// channels awaiting confirmation. This ensures that the agent doesn't go over
|
// channels awaiting confirmation. This ensures that the agent doesn't go over
|
||||||
// the prescribed channel limit or fund allocation limit.
|
// the prescribed channel limit or fund allocation limit.
|
||||||
func mergeChanState(pendingChans map[NodeID]Channel,
|
func mergeChanState(pendingChans map[NodeID]LocalChannel,
|
||||||
activeChans channelState) []Channel {
|
activeChans channelState) []LocalChannel {
|
||||||
|
|
||||||
numChans := len(pendingChans) + len(activeChans)
|
numChans := len(pendingChans) + len(activeChans)
|
||||||
totalChans := make([]Channel, 0, numChans)
|
totalChans := make([]LocalChannel, 0, numChans)
|
||||||
|
|
||||||
totalChans = append(totalChans, activeChans.Channels()...)
|
totalChans = append(totalChans, activeChans.Channels()...)
|
||||||
|
|
||||||
@ -549,7 +549,7 @@ func (a *Agent) controller() {
|
|||||||
// openChans queries the agent's heuristic for a set of channel candidates, and
|
// openChans queries the agent's heuristic for a set of channel candidates, and
|
||||||
// attempts to open channels to them.
|
// attempts to open channels to them.
|
||||||
func (a *Agent) openChans(availableFunds btcutil.Amount, numChans uint32,
|
func (a *Agent) openChans(availableFunds btcutil.Amount, numChans uint32,
|
||||||
totalChans []Channel) error {
|
totalChans []LocalChannel) error {
|
||||||
|
|
||||||
// As channel size we'll use the maximum channel size available.
|
// As channel size we'll use the maximum channel size available.
|
||||||
chanSize := a.cfg.Constraints.MaxChanSize()
|
chanSize := a.cfg.Constraints.MaxChanSize()
|
||||||
@ -828,8 +828,8 @@ func (a *Agent) executeDirective(directive AttachmentDirective) {
|
|||||||
// opens. We do this here to ensure we don't stall on selecting new
|
// opens. We do this here to ensure we don't stall on selecting new
|
||||||
// peers if the connection attempt happens to take too long.
|
// peers if the connection attempt happens to take too long.
|
||||||
delete(a.pendingConns, nodeID)
|
delete(a.pendingConns, nodeID)
|
||||||
a.pendingOpens[nodeID] = Channel{
|
a.pendingOpens[nodeID] = LocalChannel{
|
||||||
Capacity: directive.ChanAmt,
|
Balance: directive.ChanAmt,
|
||||||
Node: nodeID,
|
Node: nodeID,
|
||||||
}
|
}
|
||||||
a.pendingMtx.Unlock()
|
a.pendingMtx.Unlock()
|
||||||
|
@ -13,7 +13,7 @@ type AgentConstraints interface {
|
|||||||
// the first return value will represent the amount of additional funds
|
// the first return value will represent the amount of additional funds
|
||||||
// available towards creating channels. The second return value is the
|
// available towards creating channels. The second return value is the
|
||||||
// exact *number* of additional channels available.
|
// exact *number* of additional channels available.
|
||||||
ChannelBudget(chans []Channel, balance btcutil.Amount) (
|
ChannelBudget(chans []LocalChannel, balance btcutil.Amount) (
|
||||||
btcutil.Amount, uint32)
|
btcutil.Amount, uint32)
|
||||||
|
|
||||||
// MaxPendingOpens returns the maximum number of pending channel
|
// MaxPendingOpens returns the maximum number of pending channel
|
||||||
@ -82,7 +82,7 @@ func NewConstraints(minChanSize, maxChanSize btcutil.Amount, chanLimit,
|
|||||||
// additional channels available.
|
// additional channels available.
|
||||||
//
|
//
|
||||||
// Note: part of the AgentConstraints interface.
|
// Note: part of the AgentConstraints interface.
|
||||||
func (h *agentConstraints) ChannelBudget(channels []Channel,
|
func (h *agentConstraints) ChannelBudget(channels []LocalChannel,
|
||||||
funds btcutil.Amount) (btcutil.Amount, uint32) {
|
funds btcutil.Amount) (btcutil.Amount, uint32) {
|
||||||
|
|
||||||
// If we're already over our maximum allowed number of channels, then
|
// If we're already over our maximum allowed number of channels, then
|
||||||
@ -100,7 +100,7 @@ func (h *agentConstraints) ChannelBudget(channels []Channel,
|
|||||||
// present within the set of active channels.
|
// present within the set of active channels.
|
||||||
var totalChanAllocation btcutil.Amount
|
var totalChanAllocation btcutil.Amount
|
||||||
for _, channel := range channels {
|
for _, channel := range channels {
|
||||||
totalChanAllocation += channel.Capacity
|
totalChanAllocation += channel.Balance
|
||||||
}
|
}
|
||||||
|
|
||||||
// With this value known, we'll now compute the total amount of fund
|
// With this value known, we'll now compute the total amount of fund
|
||||||
|
@ -37,7 +37,7 @@ func TestConstraintsChannelBudget(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
testCases := []struct {
|
testCases := []struct {
|
||||||
channels []Channel
|
channels []LocalChannel
|
||||||
walletAmt btcutil.Amount
|
walletAmt btcutil.Amount
|
||||||
|
|
||||||
needMore bool
|
needMore bool
|
||||||
@ -47,18 +47,18 @@ func TestConstraintsChannelBudget(t *testing.T) {
|
|||||||
// Many available funds, but already have too many active open
|
// Many available funds, but already have too many active open
|
||||||
// channels.
|
// channels.
|
||||||
{
|
{
|
||||||
[]Channel{
|
[]LocalChannel{
|
||||||
{
|
{
|
||||||
ChanID: randChanID(),
|
ChanID: randChanID(),
|
||||||
Capacity: btcutil.Amount(prand.Int31()),
|
Balance: btcutil.Amount(prand.Int31()),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
ChanID: randChanID(),
|
ChanID: randChanID(),
|
||||||
Capacity: btcutil.Amount(prand.Int31()),
|
Balance: btcutil.Amount(prand.Int31()),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
ChanID: randChanID(),
|
ChanID: randChanID(),
|
||||||
Capacity: btcutil.Amount(prand.Int31()),
|
Balance: btcutil.Amount(prand.Int31()),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
btcutil.Amount(btcutil.SatoshiPerBitcoin * 10),
|
btcutil.Amount(btcutil.SatoshiPerBitcoin * 10),
|
||||||
@ -70,14 +70,14 @@ func TestConstraintsChannelBudget(t *testing.T) {
|
|||||||
// Ratio of funds in channels and total funds meets the
|
// Ratio of funds in channels and total funds meets the
|
||||||
// threshold.
|
// threshold.
|
||||||
{
|
{
|
||||||
[]Channel{
|
[]LocalChannel{
|
||||||
{
|
{
|
||||||
ChanID: randChanID(),
|
ChanID: randChanID(),
|
||||||
Capacity: btcutil.Amount(btcutil.SatoshiPerBitcoin),
|
Balance: btcutil.Amount(btcutil.SatoshiPerBitcoin),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
ChanID: randChanID(),
|
ChanID: randChanID(),
|
||||||
Capacity: btcutil.Amount(btcutil.SatoshiPerBitcoin),
|
Balance: btcutil.Amount(btcutil.SatoshiPerBitcoin),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
btcutil.Amount(btcutil.SatoshiPerBitcoin * 2),
|
btcutil.Amount(btcutil.SatoshiPerBitcoin * 2),
|
||||||
@ -93,10 +93,10 @@ func TestConstraintsChannelBudget(t *testing.T) {
|
|||||||
// recommended. We should also request 2 more channels as the
|
// recommended. We should also request 2 more channels as the
|
||||||
// limit is 3.
|
// limit is 3.
|
||||||
{
|
{
|
||||||
[]Channel{
|
[]LocalChannel{
|
||||||
{
|
{
|
||||||
ChanID: randChanID(),
|
ChanID: randChanID(),
|
||||||
Capacity: btcutil.Amount(btcutil.SatoshiPerBitcoin),
|
Balance: btcutil.Amount(btcutil.SatoshiPerBitcoin),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
btcutil.Amount(btcutil.SatoshiPerBitcoin * 9),
|
btcutil.Amount(btcutil.SatoshiPerBitcoin * 9),
|
||||||
@ -113,14 +113,14 @@ func TestConstraintsChannelBudget(t *testing.T) {
|
|||||||
// to be committed. We should only request a single additional
|
// to be committed. We should only request a single additional
|
||||||
// channel as the limit is 3.
|
// channel as the limit is 3.
|
||||||
{
|
{
|
||||||
[]Channel{
|
[]LocalChannel{
|
||||||
{
|
{
|
||||||
ChanID: randChanID(),
|
ChanID: randChanID(),
|
||||||
Capacity: btcutil.Amount(btcutil.SatoshiPerBitcoin),
|
Balance: btcutil.Amount(btcutil.SatoshiPerBitcoin),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
ChanID: randChanID(),
|
ChanID: randChanID(),
|
||||||
Capacity: btcutil.Amount(btcutil.SatoshiPerBitcoin * 3),
|
Balance: btcutil.Amount(btcutil.SatoshiPerBitcoin * 3),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
btcutil.Amount(btcutil.SatoshiPerBitcoin * 10),
|
btcutil.Amount(btcutil.SatoshiPerBitcoin * 10),
|
||||||
@ -132,14 +132,14 @@ func TestConstraintsChannelBudget(t *testing.T) {
|
|||||||
// Ratio of funds in channels and total funds is above the
|
// Ratio of funds in channels and total funds is above the
|
||||||
// threshold.
|
// threshold.
|
||||||
{
|
{
|
||||||
[]Channel{
|
[]LocalChannel{
|
||||||
{
|
{
|
||||||
ChanID: randChanID(),
|
ChanID: randChanID(),
|
||||||
Capacity: btcutil.Amount(btcutil.SatoshiPerBitcoin),
|
Balance: btcutil.Amount(btcutil.SatoshiPerBitcoin),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
ChanID: randChanID(),
|
ChanID: randChanID(),
|
||||||
Capacity: btcutil.Amount(btcutil.SatoshiPerBitcoin),
|
Balance: btcutil.Amount(btcutil.SatoshiPerBitcoin),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
btcutil.Amount(btcutil.SatoshiPerBitcoin),
|
btcutil.Amount(btcutil.SatoshiPerBitcoin),
|
||||||
|
@ -19,7 +19,7 @@ type moreChansResp struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
type moreChanArg struct {
|
type moreChanArg struct {
|
||||||
chans []Channel
|
chans []LocalChannel
|
||||||
balance btcutil.Amount
|
balance btcutil.Amount
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -29,7 +29,7 @@ type mockConstraints struct {
|
|||||||
quit chan struct{}
|
quit chan struct{}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *mockConstraints) ChannelBudget(chans []Channel,
|
func (m *mockConstraints) ChannelBudget(chans []LocalChannel,
|
||||||
balance btcutil.Amount) (btcutil.Amount, uint32) {
|
balance btcutil.Amount) (btcutil.Amount, uint32) {
|
||||||
|
|
||||||
if m.moreChanArgs != nil {
|
if m.moreChanArgs != nil {
|
||||||
@ -76,7 +76,7 @@ type mockHeuristic struct {
|
|||||||
type directiveArg struct {
|
type directiveArg struct {
|
||||||
graph ChannelGraph
|
graph ChannelGraph
|
||||||
amt btcutil.Amount
|
amt btcutil.Amount
|
||||||
chans []Channel
|
chans []LocalChannel
|
||||||
nodes map[NodeID]struct{}
|
nodes map[NodeID]struct{}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -84,7 +84,7 @@ func (m *mockHeuristic) Name() string {
|
|||||||
return "mock"
|
return "mock"
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *mockHeuristic) NodeScores(g ChannelGraph, chans []Channel,
|
func (m *mockHeuristic) NodeScores(g ChannelGraph, chans []LocalChannel,
|
||||||
chanSize btcutil.Amount, nodes map[NodeID]struct{}) (
|
chanSize btcutil.Amount, nodes map[NodeID]struct{}) (
|
||||||
map[NodeID]*NodeScore, error) {
|
map[NodeID]*NodeScore, error) {
|
||||||
|
|
||||||
@ -139,14 +139,6 @@ func (m *mockChanController) OpenChannel(target *btcec.PublicKey,
|
|||||||
func (m *mockChanController) CloseChannel(chanPoint *wire.OutPoint) error {
|
func (m *mockChanController) CloseChannel(chanPoint *wire.OutPoint) error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
func (m *mockChanController) SpliceIn(chanPoint *wire.OutPoint,
|
|
||||||
amt btcutil.Amount) (*Channel, error) {
|
|
||||||
return nil, nil
|
|
||||||
}
|
|
||||||
func (m *mockChanController) SpliceOut(chanPoint *wire.OutPoint,
|
|
||||||
amt btcutil.Amount) (*Channel, error) {
|
|
||||||
return nil, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
var _ ChannelController = (*mockChanController)(nil)
|
var _ ChannelController = (*mockChanController)(nil)
|
||||||
|
|
||||||
@ -162,7 +154,7 @@ type testContext struct {
|
|||||||
sync.Mutex
|
sync.Mutex
|
||||||
}
|
}
|
||||||
|
|
||||||
func setup(t *testing.T, initialChans []Channel) (*testContext, func()) {
|
func setup(t *testing.T, initialChans []LocalChannel) (*testContext, func()) {
|
||||||
t.Helper()
|
t.Helper()
|
||||||
|
|
||||||
// First, we'll create all the dependencies that we'll need in order to
|
// First, we'll create all the dependencies that we'll need in order to
|
||||||
@ -299,9 +291,9 @@ func TestAgentChannelOpenSignal(t *testing.T) {
|
|||||||
|
|
||||||
// Next we'll signal a new channel being opened by the backing LN node,
|
// Next we'll signal a new channel being opened by the backing LN node,
|
||||||
// with a capacity of 1 BTC.
|
// with a capacity of 1 BTC.
|
||||||
newChan := Channel{
|
newChan := LocalChannel{
|
||||||
ChanID: randChanID(),
|
ChanID: randChanID(),
|
||||||
Capacity: btcutil.SatoshiPerBitcoin,
|
Balance: btcutil.SatoshiPerBitcoin,
|
||||||
}
|
}
|
||||||
testCtx.agent.OnChannelOpen(newChan)
|
testCtx.agent.OnChannelOpen(newChan)
|
||||||
|
|
||||||
@ -391,14 +383,6 @@ func (m *mockFailingChanController) OpenChannel(target *btcec.PublicKey,
|
|||||||
func (m *mockFailingChanController) CloseChannel(chanPoint *wire.OutPoint) error {
|
func (m *mockFailingChanController) CloseChannel(chanPoint *wire.OutPoint) error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
func (m *mockFailingChanController) SpliceIn(chanPoint *wire.OutPoint,
|
|
||||||
amt btcutil.Amount) (*Channel, error) {
|
|
||||||
return nil, nil
|
|
||||||
}
|
|
||||||
func (m *mockFailingChanController) SpliceOut(chanPoint *wire.OutPoint,
|
|
||||||
amt btcutil.Amount) (*Channel, error) {
|
|
||||||
return nil, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
var _ ChannelController = (*mockFailingChanController)(nil)
|
var _ ChannelController = (*mockFailingChanController)(nil)
|
||||||
|
|
||||||
@ -448,14 +432,14 @@ func TestAgentChannelFailureSignal(t *testing.T) {
|
|||||||
func TestAgentChannelCloseSignal(t *testing.T) {
|
func TestAgentChannelCloseSignal(t *testing.T) {
|
||||||
t.Parallel()
|
t.Parallel()
|
||||||
// We'll start the agent with two channels already being active.
|
// We'll start the agent with two channels already being active.
|
||||||
initialChans := []Channel{
|
initialChans := []LocalChannel{
|
||||||
{
|
{
|
||||||
ChanID: randChanID(),
|
ChanID: randChanID(),
|
||||||
Capacity: btcutil.SatoshiPerBitcoin,
|
Balance: btcutil.SatoshiPerBitcoin,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
ChanID: randChanID(),
|
ChanID: randChanID(),
|
||||||
Capacity: btcutil.SatoshiPerBitcoin * 2,
|
Balance: btcutil.SatoshiPerBitcoin * 2,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -746,9 +730,9 @@ func TestAgentPendingChannelState(t *testing.T) {
|
|||||||
t.Fatalf("should include pending chan in current "+
|
t.Fatalf("should include pending chan in current "+
|
||||||
"state, instead have %v chans", len(req.chans))
|
"state, instead have %v chans", len(req.chans))
|
||||||
}
|
}
|
||||||
if req.chans[0].Capacity != chanAmt {
|
if req.chans[0].Balance != chanAmt {
|
||||||
t.Fatalf("wrong chan capacity: expected %v, got %v",
|
t.Fatalf("wrong chan balance: expected %v, got %v",
|
||||||
req.chans[0].Capacity, chanAmt)
|
req.chans[0].Balance, chanAmt)
|
||||||
}
|
}
|
||||||
if req.chans[0].Node != nodeID {
|
if req.chans[0].Node != nodeID {
|
||||||
t.Fatalf("wrong node ID: expected %x, got %x",
|
t.Fatalf("wrong node ID: expected %x, got %x",
|
||||||
|
@ -70,7 +70,7 @@ func (c *WeightedCombAttachment) Name() string {
|
|||||||
// is the maximum possible improvement in connectivity.
|
// is the maximum possible improvement in connectivity.
|
||||||
//
|
//
|
||||||
// NOTE: This is a part of the AttachmentHeuristic interface.
|
// NOTE: This is a part of the AttachmentHeuristic interface.
|
||||||
func (c *WeightedCombAttachment) NodeScores(g ChannelGraph, chans []Channel,
|
func (c *WeightedCombAttachment) NodeScores(g ChannelGraph, chans []LocalChannel,
|
||||||
chanSize btcutil.Amount, nodes map[NodeID]struct{}) (
|
chanSize btcutil.Amount, nodes map[NodeID]struct{}) (
|
||||||
map[NodeID]*NodeScore, error) {
|
map[NodeID]*NodeScore, error) {
|
||||||
|
|
||||||
|
@ -80,7 +80,7 @@ func (s *ExternalScoreAttachment) SetNodeScores(targetHeuristic string,
|
|||||||
// not known will get a score of 0.
|
// not known will get a score of 0.
|
||||||
//
|
//
|
||||||
// NOTE: This is a part of the AttachmentHeuristic interface.
|
// NOTE: This is a part of the AttachmentHeuristic interface.
|
||||||
func (s *ExternalScoreAttachment) NodeScores(g ChannelGraph, chans []Channel,
|
func (s *ExternalScoreAttachment) NodeScores(g ChannelGraph, chans []LocalChannel,
|
||||||
chanSize btcutil.Amount, nodes map[NodeID]struct{}) (
|
chanSize btcutil.Amount, nodes map[NodeID]struct{}) (
|
||||||
map[NodeID]*NodeScore, error) {
|
map[NodeID]*NodeScore, error) {
|
||||||
|
|
||||||
|
@ -99,12 +99,8 @@ func (d dbNode) ForEachChannel(cb func(ChannelEdge) error) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
edge := ChannelEdge{
|
edge := ChannelEdge{
|
||||||
Channel: Channel{
|
|
||||||
ChanID: lnwire.NewShortChanIDFromInt(ep.ChannelID),
|
ChanID: lnwire.NewShortChanIDFromInt(ep.ChannelID),
|
||||||
Capacity: ei.Capacity,
|
Capacity: ei.Capacity,
|
||||||
FundedAmt: ei.Capacity,
|
|
||||||
Node: NodeID(ep.Node.PubKeyBytes),
|
|
||||||
},
|
|
||||||
Peer: dbNode{
|
Peer: dbNode{
|
||||||
tx: tx,
|
tx: tx,
|
||||||
node: ep.Node,
|
node: ep.Node,
|
||||||
@ -265,19 +261,15 @@ func (d *databaseChannelGraph) addRandChannel(node1, node2 *btcec.PublicKey,
|
|||||||
}
|
}
|
||||||
|
|
||||||
return &ChannelEdge{
|
return &ChannelEdge{
|
||||||
Channel: Channel{
|
|
||||||
ChanID: chanID,
|
ChanID: chanID,
|
||||||
Capacity: capacity,
|
Capacity: capacity,
|
||||||
},
|
|
||||||
Peer: dbNode{
|
Peer: dbNode{
|
||||||
node: vertex1,
|
node: vertex1,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
&ChannelEdge{
|
&ChannelEdge{
|
||||||
Channel: Channel{
|
|
||||||
ChanID: chanID,
|
ChanID: chanID,
|
||||||
Capacity: capacity,
|
Capacity: capacity,
|
||||||
},
|
|
||||||
Peer: dbNode{
|
Peer: dbNode{
|
||||||
node: vertex2,
|
node: vertex2,
|
||||||
},
|
},
|
||||||
@ -425,19 +417,16 @@ func (m *memChannelGraph) addRandChannel(node1, node2 *btcec.PublicKey,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
channel := Channel{
|
edge1 := ChannelEdge{
|
||||||
ChanID: randChanID(),
|
ChanID: randChanID(),
|
||||||
Capacity: capacity,
|
Capacity: capacity,
|
||||||
}
|
|
||||||
|
|
||||||
edge1 := ChannelEdge{
|
|
||||||
Channel: channel,
|
|
||||||
Peer: vertex2,
|
Peer: vertex2,
|
||||||
}
|
}
|
||||||
vertex1.chans = append(vertex1.chans, edge1)
|
vertex1.chans = append(vertex1.chans, edge1)
|
||||||
|
|
||||||
edge2 := ChannelEdge{
|
edge2 := ChannelEdge{
|
||||||
Channel: channel,
|
ChanID: randChanID(),
|
||||||
|
Capacity: capacity,
|
||||||
Peer: vertex1,
|
Peer: vertex1,
|
||||||
}
|
}
|
||||||
vertex2.chans = append(vertex2.chans, edge2)
|
vertex2.chans = append(vertex2.chans, edge2)
|
||||||
|
@ -36,22 +36,16 @@ type Node interface {
|
|||||||
ForEachChannel(func(ChannelEdge) error) error
|
ForEachChannel(func(ChannelEdge) error) error
|
||||||
}
|
}
|
||||||
|
|
||||||
// Channel is a simple struct which contains relevant details of a particular
|
// LocalChannel is a simple struct which contains relevant details of a
|
||||||
// channel within the channel graph. The fields in this struct may be used a
|
// particular channel the local node has. The fields in this struct may be used
|
||||||
// signals for various AttachmentHeuristic implementations.
|
// a signals for various AttachmentHeuristic implementations.
|
||||||
type Channel struct {
|
type LocalChannel struct {
|
||||||
// ChanID is the short channel ID for this channel as defined within
|
// ChanID is the short channel ID for this channel as defined within
|
||||||
// BOLT-0007.
|
// BOLT-0007.
|
||||||
ChanID lnwire.ShortChannelID
|
ChanID lnwire.ShortChannelID
|
||||||
|
|
||||||
// Capacity is the capacity of the channel expressed in satoshis.
|
// Balance is the local balance of the channel expressed in satoshis.
|
||||||
Capacity btcutil.Amount
|
Balance btcutil.Amount
|
||||||
|
|
||||||
// FundedAmt is the amount the local node funded into the target
|
|
||||||
// channel.
|
|
||||||
//
|
|
||||||
// TODO(roasbeef): need this?
|
|
||||||
FundedAmt btcutil.Amount
|
|
||||||
|
|
||||||
// Node is the peer that this channel has been established with.
|
// Node is the peer that this channel has been established with.
|
||||||
Node NodeID
|
Node NodeID
|
||||||
@ -65,8 +59,12 @@ type Channel struct {
|
|||||||
// edge within the graph. The existence of this reference to the connected node
|
// edge within the graph. The existence of this reference to the connected node
|
||||||
// will allow callers to traverse the graph in an object-oriented manner.
|
// will allow callers to traverse the graph in an object-oriented manner.
|
||||||
type ChannelEdge struct {
|
type ChannelEdge struct {
|
||||||
// Channel contains the attributes of this channel.
|
// ChanID is the short channel ID for this channel as defined within
|
||||||
Channel
|
// BOLT-0007.
|
||||||
|
ChanID lnwire.ShortChannelID
|
||||||
|
|
||||||
|
// Capacity is the capacity of the channel expressed in satoshis.
|
||||||
|
Capacity btcutil.Amount
|
||||||
|
|
||||||
// Peer is the peer that this channel creates an edge to in the channel
|
// Peer is the peer that this channel creates an edge to in the channel
|
||||||
// graph.
|
// graph.
|
||||||
@ -142,7 +140,7 @@ type AttachmentHeuristic interface {
|
|||||||
//
|
//
|
||||||
// NOTE: A NodeID not found in the returned map is implicitly given a
|
// NOTE: A NodeID not found in the returned map is implicitly given a
|
||||||
// score of 0.
|
// score of 0.
|
||||||
NodeScores(g ChannelGraph, chans []Channel,
|
NodeScores(g ChannelGraph, chans []LocalChannel,
|
||||||
chanSize btcutil.Amount, nodes map[NodeID]struct{}) (
|
chanSize btcutil.Amount, nodes map[NodeID]struct{}) (
|
||||||
map[NodeID]*NodeScore, error)
|
map[NodeID]*NodeScore, error)
|
||||||
}
|
}
|
||||||
@ -217,14 +215,4 @@ type ChannelController interface {
|
|||||||
//
|
//
|
||||||
// TODO(roasbeef): add force option?
|
// TODO(roasbeef): add force option?
|
||||||
CloseChannel(chanPoint *wire.OutPoint) error
|
CloseChannel(chanPoint *wire.OutPoint) error
|
||||||
|
|
||||||
// SpliceIn attempts to add additional funds to the target channel via
|
|
||||||
// a splice in mechanism. The new channel with an updated capacity
|
|
||||||
// should be returned.
|
|
||||||
SpliceIn(chanPoint *wire.OutPoint, amt btcutil.Amount) (*Channel, error)
|
|
||||||
|
|
||||||
// SpliceOut attempts to remove funds from an existing channels using a
|
|
||||||
// splice out mechanism. The removed funds from the channel should be
|
|
||||||
// returned to an output under the control of the backing wallet.
|
|
||||||
SpliceOut(chanPoint *wire.OutPoint, amt btcutil.Amount) (*Channel, error)
|
|
||||||
}
|
}
|
||||||
|
@ -5,6 +5,7 @@ import (
|
|||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
"github.com/btcsuite/btcd/btcec"
|
"github.com/btcsuite/btcd/btcec"
|
||||||
|
"github.com/btcsuite/btcd/wire"
|
||||||
"github.com/lightningnetwork/lnd/lnwallet"
|
"github.com/lightningnetwork/lnd/lnwallet"
|
||||||
"github.com/lightningnetwork/lnd/lnwire"
|
"github.com/lightningnetwork/lnd/lnwire"
|
||||||
"github.com/lightningnetwork/lnd/routing"
|
"github.com/lightningnetwork/lnd/routing"
|
||||||
@ -23,7 +24,11 @@ type ManagerCfg struct {
|
|||||||
|
|
||||||
// ChannelState is a function closure that returns the current set of
|
// ChannelState is a function closure that returns the current set of
|
||||||
// channels managed by this node.
|
// channels managed by this node.
|
||||||
ChannelState func() ([]Channel, error)
|
ChannelState func() ([]LocalChannel, error)
|
||||||
|
|
||||||
|
// ChannelInfo is a function closure that returns the channel managed
|
||||||
|
// by the node given by the passed channel point.
|
||||||
|
ChannelInfo func(wire.OutPoint) (*LocalChannel, error)
|
||||||
|
|
||||||
// SubscribeTransactions is used to get a subscription for transactions
|
// SubscribeTransactions is used to get a subscription for transactions
|
||||||
// relevant to this node's wallet.
|
// relevant to this node's wallet.
|
||||||
@ -194,18 +199,16 @@ func (m *Manager) StartAgent() error {
|
|||||||
// opened, then we'll convert it to the
|
// opened, then we'll convert it to the
|
||||||
// autopilot.Channel format, and notify
|
// autopilot.Channel format, and notify
|
||||||
// the pilot of the new channel.
|
// the pilot of the new channel.
|
||||||
chanNode := NewNodeID(
|
cp := edgeUpdate.ChanPoint
|
||||||
edgeUpdate.ConnectingNode,
|
edge, err := m.cfg.ChannelInfo(cp)
|
||||||
)
|
if err != nil {
|
||||||
chanID := lnwire.NewShortChanIDFromInt(
|
log.Errorf("Unable to fetch "+
|
||||||
edgeUpdate.ChanID,
|
"channel info for %v: "+
|
||||||
)
|
"%v", cp, err)
|
||||||
edge := Channel{
|
continue
|
||||||
ChanID: chanID,
|
|
||||||
Capacity: edgeUpdate.Capacity,
|
|
||||||
Node: chanNode,
|
|
||||||
}
|
}
|
||||||
pilot.OnChannelOpen(edge)
|
|
||||||
|
pilot.OnChannelOpen(*edge)
|
||||||
}
|
}
|
||||||
|
|
||||||
// For each closed channel, we'll obtain
|
// For each closed channel, we'll obtain
|
||||||
@ -292,7 +295,7 @@ func (m *Manager) queryHeuristics(nodes map[NodeID]struct{}, localState bool) (
|
|||||||
// If we want to take the local state into action when querying the
|
// If we want to take the local state into action when querying the
|
||||||
// heuristics, we fetch it. If not we'll just pass an emply slice to
|
// heuristics, we fetch it. If not we'll just pass an emply slice to
|
||||||
// the heuristic.
|
// the heuristic.
|
||||||
var totalChans []Channel
|
var totalChans []LocalChannel
|
||||||
var err error
|
var err error
|
||||||
if localState {
|
if localState {
|
||||||
// Fetch the current set of channels.
|
// Fetch the current set of channels.
|
||||||
|
@ -78,7 +78,7 @@ func (p *PrefAttachment) Name() string {
|
|||||||
// given to nodes already having high connectivity in the graph.
|
// given to nodes already having high connectivity in the graph.
|
||||||
//
|
//
|
||||||
// NOTE: This is a part of the AttachmentHeuristic interface.
|
// NOTE: This is a part of the AttachmentHeuristic interface.
|
||||||
func (p *PrefAttachment) NodeScores(g ChannelGraph, chans []Channel,
|
func (p *PrefAttachment) NodeScores(g ChannelGraph, chans []LocalChannel,
|
||||||
chanSize btcutil.Amount, nodes map[NodeID]struct{}) (
|
chanSize btcutil.Amount, nodes map[NodeID]struct{}) (
|
||||||
map[NodeID]*NodeScore, error) {
|
map[NodeID]*NodeScore, error) {
|
||||||
|
|
||||||
|
@ -422,10 +422,10 @@ func TestPrefAttachmentSelectSkipNodes(t *testing.T) {
|
|||||||
|
|
||||||
// We'll simulate a channel update by adding the nodes
|
// We'll simulate a channel update by adding the nodes
|
||||||
// to our set of channels.
|
// to our set of channels.
|
||||||
var chans []Channel
|
var chans []LocalChannel
|
||||||
for _, candidate := range scores {
|
for _, candidate := range scores {
|
||||||
chans = append(chans,
|
chans = append(chans,
|
||||||
Channel{
|
LocalChannel{
|
||||||
Node: candidate.NodeID,
|
Node: candidate.NodeID,
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
@ -50,7 +50,7 @@ func (g *TopCentrality) Name() string {
|
|||||||
// As our current implementation of betweenness centrality is non-incremental,
|
// As our current implementation of betweenness centrality is non-incremental,
|
||||||
// NodeScores will recalculate the centrality values on every call, which is
|
// NodeScores will recalculate the centrality values on every call, which is
|
||||||
// slow for large graphs.
|
// slow for large graphs.
|
||||||
func (g *TopCentrality) NodeScores(graph ChannelGraph, chans []Channel,
|
func (g *TopCentrality) NodeScores(graph ChannelGraph, chans []LocalChannel,
|
||||||
chanSize btcutil.Amount, nodes map[NodeID]struct{}) (
|
chanSize btcutil.Amount, nodes map[NodeID]struct{}) (
|
||||||
map[NodeID]*NodeScore, error) {
|
map[NodeID]*NodeScore, error) {
|
||||||
|
|
||||||
|
@ -16,9 +16,9 @@ func testTopCentrality(t *testing.T, graph testGraph,
|
|||||||
|
|
||||||
topCentrality := NewTopCentrality()
|
topCentrality := NewTopCentrality()
|
||||||
|
|
||||||
var channels []Channel
|
var channels []LocalChannel
|
||||||
for _, ch := range channelsWith {
|
for _, ch := range channelsWith {
|
||||||
channels = append(channels, Channel{
|
channels = append(channels, LocalChannel{
|
||||||
Node: NewNodeID(graphNodes[ch]),
|
Node: NewNodeID(graphNodes[ch]),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
37
pilot.go
37
pilot.go
@ -124,14 +124,6 @@ func (c *chanController) OpenChannel(target *btcec.PublicKey,
|
|||||||
func (c *chanController) CloseChannel(chanPoint *wire.OutPoint) error {
|
func (c *chanController) CloseChannel(chanPoint *wire.OutPoint) error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
func (c *chanController) SpliceIn(chanPoint *wire.OutPoint,
|
|
||||||
amt btcutil.Amount) (*autopilot.Channel, error) {
|
|
||||||
return nil, nil
|
|
||||||
}
|
|
||||||
func (c *chanController) SpliceOut(chanPoint *wire.OutPoint,
|
|
||||||
amt btcutil.Amount) (*autopilot.Channel, error) {
|
|
||||||
return nil, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// A compile time assertion to ensure chanController meets the
|
// A compile time assertion to ensure chanController meets the
|
||||||
// autopilot.ChannelController interface.
|
// autopilot.ChannelController interface.
|
||||||
@ -257,7 +249,7 @@ func initAutoPilot(svr *server, cfg *lncfg.AutoPilot,
|
|||||||
return &autopilot.ManagerCfg{
|
return &autopilot.ManagerCfg{
|
||||||
Self: self,
|
Self: self,
|
||||||
PilotCfg: &pilotCfg,
|
PilotCfg: &pilotCfg,
|
||||||
ChannelState: func() ([]autopilot.Channel, error) {
|
ChannelState: func() ([]autopilot.LocalChannel, error) {
|
||||||
// We'll fetch the current state of open
|
// We'll fetch the current state of open
|
||||||
// channels from the database to use as initial
|
// channels from the database to use as initial
|
||||||
// state for the auto-pilot agent.
|
// state for the auto-pilot agent.
|
||||||
@ -265,19 +257,38 @@ func initAutoPilot(svr *server, cfg *lncfg.AutoPilot,
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
chanState := make([]autopilot.Channel,
|
chanState := make([]autopilot.LocalChannel,
|
||||||
len(activeChannels))
|
len(activeChannels))
|
||||||
for i, channel := range activeChannels {
|
for i, channel := range activeChannels {
|
||||||
chanState[i] = autopilot.Channel{
|
localCommit := channel.LocalCommitment
|
||||||
|
balance := localCommit.LocalBalance.ToSatoshis()
|
||||||
|
|
||||||
|
chanState[i] = autopilot.LocalChannel{
|
||||||
ChanID: channel.ShortChanID(),
|
ChanID: channel.ShortChanID(),
|
||||||
Capacity: channel.Capacity,
|
Balance: balance,
|
||||||
Node: autopilot.NewNodeID(
|
Node: autopilot.NewNodeID(
|
||||||
channel.IdentityPub),
|
channel.IdentityPub,
|
||||||
|
),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return chanState, nil
|
return chanState, nil
|
||||||
},
|
},
|
||||||
|
ChannelInfo: func(chanPoint wire.OutPoint) (
|
||||||
|
*autopilot.LocalChannel, error) {
|
||||||
|
|
||||||
|
channel, err := svr.remoteChanDB.FetchChannel(chanPoint)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
localCommit := channel.LocalCommitment
|
||||||
|
return &autopilot.LocalChannel{
|
||||||
|
ChanID: channel.ShortChanID(),
|
||||||
|
Balance: localCommit.LocalBalance.ToSatoshis(),
|
||||||
|
Node: autopilot.NewNodeID(channel.IdentityPub),
|
||||||
|
}, nil
|
||||||
|
},
|
||||||
SubscribeTransactions: svr.cc.wallet.SubscribeTransactions,
|
SubscribeTransactions: svr.cc.wallet.SubscribeTransactions,
|
||||||
SubscribeTopology: svr.chanRouter.SubscribeTopology,
|
SubscribeTopology: svr.chanRouter.SubscribeTopology,
|
||||||
}, nil
|
}, nil
|
||||||
|
Loading…
Reference in New Issue
Block a user