Skip to content

Commit

Permalink
lndmon: cache closedChannels response
Browse files Browse the repository at this point in the history
  • Loading branch information
djkazic committed Jan 26, 2024
1 parent f7d3fdf commit 33fe8a3
Show file tree
Hide file tree
Showing 2 changed files with 66 additions and 14 deletions.
61 changes: 50 additions & 11 deletions collectors/channels_collector.go
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,8 @@ import (
"context"
"fmt"
"strconv"
"sync"
"time"

"github.com/btcsuite/btcd/btcutil"
"github.com/lightninglabs/lndclient"
Expand Down Expand Up @@ -51,17 +53,25 @@ type ChannelsCollector struct {
// errChan is a channel that we send any errors that we encounter into.
// This channel should be buffered so that it does not block sends.
errChan chan<- error

// quit is a channel that we use to signal for graceful shutdown
quit chan struct{}

// cache is a map storing results from a ticker to reduce grpc server
// load on lnd
closedChannelsCache []lndclient.ClosedChannel
cacheMutex sync.RWMutex
}

// NewChannelsCollector returns a new instance of the ChannelsCollector for the
// target lnd client.
func NewChannelsCollector(lnd lndclient.LightningClient, errChan chan<- error,
func NewChannelsCollector(lnd lndclient.LightningClient, errChan chan<- error, quit chan struct{},

Check failure on line 68 in collectors/channels_collector.go

View workflow job for this annotation

GitHub Actions / lint and build code

unused-parameter: parameter 'quit' seems to be unused, consider removing or renaming it as _ (revive)
cfg *MonitoringConfig) *ChannelsCollector {

// Our set of labels, status should either be active or inactive. The
// initiator is "true" if we are the initiator, and "false" otherwise.
labels := []string{"chan_id", "status", "initiator", "peer"}
return &ChannelsCollector{
collector := &ChannelsCollector{
channelBalanceDesc: prometheus.NewDesc(
"lnd_channels_open_balance_sat",
"total balance of channels in satoshis",
Expand Down Expand Up @@ -174,10 +184,42 @@ func NewChannelsCollector(lnd lndclient.LightningClient, errChan chan<- error,
[]string{"amount"}, nil,
),

lnd: lnd,
primaryNode: cfg.PrimaryNode,
errChan: errChan,
lnd: lnd,
primaryNode: cfg.PrimaryNode,
closedChannelsCache: nil,
errChan: errChan,
}

// Perform an initial refresh for the cache
collector.refreshClosedChannelsCache()

// Start a ticker to update the cache once per 10m
go func() {
ticker := time.NewTicker(10 * time.Minute)
defer ticker.Stop()
for {
select {
case <-ticker.C:
collector.refreshClosedChannelsCache()

case <-collector.quit:
return
}
}
}()

return collector
}

func (c *ChannelsCollector) refreshClosedChannelsCache() {
data, err := c.lnd.ClosedChannels(context.Background())
if err != nil {
c.errChan <- err
return
}
c.cacheMutex.Lock()
c.closedChannelsCache = data
c.cacheMutex.Unlock()
}

// Describe sends the super-set of all possible descriptors of metrics
Expand Down Expand Up @@ -452,12 +494,9 @@ func (c *ChannelsCollector) Collect(ch chan<- prometheus.Metric) {
)

// Get the list of closed channels.
closedChannelsResp, err := c.lnd.ClosedChannels(context.Background())
if err != nil {
c.errChan <- fmt.Errorf("ChannelsCollector ClosedChannels "+
"failed with: %v", err)
return
}
c.cacheMutex.RLock()
closedChannelsResp := c.closedChannelsCache
c.cacheMutex.RUnlock()
closeCounts := make(map[string]int)
for _, channel := range closedChannelsResp {
typeString, ok := closeTypeLabelMap[channel.CloseType]
Expand Down
19 changes: 16 additions & 3 deletions collectors/prometheus.go
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,9 @@ import (
"log"
"net/http"
"os"
"os/signal"
"path/filepath"
"syscall"

"github.com/btcsuite/btcd/btcutil"
"github.com/lightninglabs/lndclient"
Expand Down Expand Up @@ -94,11 +96,22 @@ func NewPrometheusExporter(cfg *PrometheusConfig, lnd *lndclient.LndServices,

htlcMonitor := newHtlcMonitor(lnd.Router, errChan)

// Setup signalling so SIGTERM || SIGINT will stop the cache goroutine
// from ChannelsCollector
quit := make(chan struct{})
sigs := make(chan os.Signal, 1)
signal.Notify(sigs, syscall.SIGINT, syscall.SIGTERM)
chanCollector := NewChannelsCollector(
lnd.Client, errChan, quit, monitoringCfg,
)
go func() {
<-sigs
close(chanCollector.quit)
}()

collectors := []prometheus.Collector{
NewChainCollector(lnd.Client, errChan),
NewChannelsCollector(
lnd.Client, errChan, monitoringCfg,
),
chanCollector,
NewWalletCollector(lnd, errChan),
NewPeerCollector(lnd.Client, errChan),
NewInfoCollector(lnd.Client, errChan),
Expand Down

0 comments on commit 33fe8a3

Please sign in to comment.