225 lines
6.1 KiB
Go
225 lines
6.1 KiB
Go
package chanfitness
|
|
|
|
import (
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/btcsuite/btcd/wire"
|
|
"github.com/lightningnetwork/lnd/clock"
|
|
"github.com/lightningnetwork/lnd/routing/route"
|
|
)
|
|
|
|
type eventType int
|
|
|
|
const (
|
|
peerOnlineEvent eventType = iota
|
|
peerOfflineEvent
|
|
)
|
|
|
|
// String provides string representations of channel events.
|
|
func (e eventType) String() string {
|
|
switch e {
|
|
case peerOnlineEvent:
|
|
return "peer_online"
|
|
|
|
case peerOfflineEvent:
|
|
return "peer_offline"
|
|
}
|
|
|
|
return "unknown"
|
|
}
|
|
|
|
// channelEvent is a a timestamped event which is observed on a per channel
|
|
// basis.
|
|
type channelEvent struct {
|
|
timestamp time.Time
|
|
eventType eventType
|
|
}
|
|
|
|
// chanEventLog stores all events that have occurred over a channel's lifetime.
|
|
type chanEventLog struct {
|
|
// channelPoint is the outpoint for the channel's funding transaction.
|
|
channelPoint wire.OutPoint
|
|
|
|
// peer is the compressed public key of the peer being monitored.
|
|
peer route.Vertex
|
|
|
|
// events is a log of timestamped events observed for the channel.
|
|
events []*channelEvent
|
|
|
|
// clock allows creation of deterministic unit tests.
|
|
clock clock.Clock
|
|
|
|
// openedAt tracks the first time this channel was seen. This is not
|
|
// necessarily the time that it confirmed on chain because channel
|
|
// events are not persisted at present.
|
|
openedAt time.Time
|
|
|
|
// closedAt is the time that the channel was closed. If the channel has
|
|
// not been closed yet, it is zero.
|
|
closedAt time.Time
|
|
}
|
|
|
|
// newEventLog creates an event log for a channel with the openedAt time set.
|
|
func newEventLog(channelPoint wire.OutPoint, peer route.Vertex,
|
|
clock clock.Clock) *chanEventLog {
|
|
|
|
eventlog := &chanEventLog{
|
|
channelPoint: channelPoint,
|
|
peer: peer,
|
|
clock: clock,
|
|
openedAt: clock.Now(),
|
|
}
|
|
|
|
return eventlog
|
|
}
|
|
|
|
// close sets the closing time for an event log.
|
|
func (e *chanEventLog) close() {
|
|
e.closedAt = e.clock.Now()
|
|
}
|
|
|
|
// add appends an event with the given type and current time to the event log.
|
|
// The open time for the eventLog will be set to the event's timestamp if it is
|
|
// not set yet.
|
|
func (e *chanEventLog) add(eventType eventType) {
|
|
// If the channel is already closed, return early without adding an
|
|
// event.
|
|
if !e.closedAt.IsZero() {
|
|
return
|
|
}
|
|
|
|
// Add the event to the eventLog with the current timestamp.
|
|
event := &channelEvent{
|
|
timestamp: e.clock.Now(),
|
|
eventType: eventType,
|
|
}
|
|
e.events = append(e.events, event)
|
|
|
|
log.Debugf("Channel %v recording event: %v", e.channelPoint, eventType)
|
|
}
|
|
|
|
// onlinePeriod represents a period of time over which a peer was online.
|
|
type onlinePeriod struct {
|
|
start, end time.Time
|
|
}
|
|
|
|
// getOnlinePeriods returns a list of all the periods that the event log has
|
|
// recorded the remote peer as being online. In the unexpected case where there
|
|
// are no events, the function returns early. Online periods are defined as a
|
|
// peer online event which is terminated by a peer offline event. This function
|
|
// expects the event log provided to be ordered by ascending timestamp.
|
|
func (e *chanEventLog) getOnlinePeriods() []*onlinePeriod {
|
|
// Return early if there are no events, there are no online periods.
|
|
if len(e.events) == 0 {
|
|
return nil
|
|
}
|
|
|
|
var (
|
|
lastOnline time.Time
|
|
offline bool
|
|
onlinePeriods []*onlinePeriod
|
|
)
|
|
|
|
// Loop through all events to build a list of periods that the peer was
|
|
// online. Online periods are added when they are terminated with a peer
|
|
// offline event. If the log ends on an online event, the period between
|
|
// the online event and the present is not tracked. The type of the most
|
|
// recent event is tracked using the offline bool so that we can add a
|
|
// final online period if necessary.
|
|
for _, event := range e.events {
|
|
|
|
switch event.eventType {
|
|
case peerOnlineEvent:
|
|
lastOnline = event.timestamp
|
|
offline = false
|
|
|
|
case peerOfflineEvent:
|
|
offline = true
|
|
|
|
// Do not add to uptime if there is no previous online
|
|
// timestamp, the event log has started with an offline
|
|
// event
|
|
if lastOnline.IsZero() {
|
|
continue
|
|
}
|
|
|
|
// The eventLog has recorded an offline event, having
|
|
// previously been online so we add an online period to
|
|
// set of online periods.
|
|
onlinePeriods = append(onlinePeriods, &onlinePeriod{
|
|
start: lastOnline,
|
|
end: event.timestamp,
|
|
})
|
|
}
|
|
}
|
|
|
|
// If the last event was an peer offline event, we do not need to
|
|
// calculate a final online period and can return online periods as is.
|
|
if offline {
|
|
return onlinePeriods
|
|
}
|
|
|
|
// The log ended on an online event, so we need to add a final online
|
|
// event. If the channel is closed, this period is until channel
|
|
// closure. It it is still open, we calculate it until the present.
|
|
endTime := e.closedAt
|
|
if endTime.IsZero() {
|
|
endTime = e.clock.Now()
|
|
}
|
|
|
|
// Add the final online period to the set and return.
|
|
return append(onlinePeriods, &onlinePeriod{
|
|
start: lastOnline,
|
|
end: endTime,
|
|
})
|
|
}
|
|
|
|
// uptime calculates the total uptime we have recorded for a channel over the
|
|
// inclusive range specified. An error is returned if the end of the range is
|
|
// before the start or a zero end time is returned.
|
|
func (e *chanEventLog) uptime(start, end time.Time) (time.Duration, error) {
|
|
// Error if we are provided with an invalid range to calculate uptime
|
|
// for.
|
|
if end.Before(start) {
|
|
return 0, fmt.Errorf("end time: %v before start time: %v",
|
|
end, start)
|
|
}
|
|
if end.IsZero() {
|
|
return 0, fmt.Errorf("zero end time")
|
|
}
|
|
|
|
var uptime time.Duration
|
|
|
|
for _, p := range e.getOnlinePeriods() {
|
|
// The online period ends before the range we're looking at, so
|
|
// we can skip over it.
|
|
if p.end.Before(start) {
|
|
continue
|
|
}
|
|
// The online period starts after the range we're looking at, so
|
|
// can stop calculating uptime.
|
|
if p.start.After(end) {
|
|
break
|
|
}
|
|
|
|
// If the online period starts before our range, shift the start
|
|
// time up so that we only calculate uptime from the start of
|
|
// our range.
|
|
if p.start.Before(start) {
|
|
p.start = start
|
|
}
|
|
|
|
// If the online period ends before our range, shift the end
|
|
// time forward so that we only calculate uptime until the end
|
|
// of the range.
|
|
if p.end.After(end) {
|
|
p.end = end
|
|
}
|
|
|
|
uptime += p.end.Sub(p.start)
|
|
}
|
|
|
|
return uptime, nil
|
|
}
|