2022-01-10 18:43:13 +01:00
|
|
|
package peer
|
|
|
|
|
2023-03-03 19:49:18 +01:00
|
|
|
import (
|
|
|
|
"errors"
|
2024-06-13 13:24:24 +02:00
|
|
|
"net/netip"
|
2024-09-08 12:06:14 +02:00
|
|
|
"slices"
|
2023-03-03 19:49:18 +01:00
|
|
|
"sync"
|
|
|
|
"time"
|
2024-01-22 12:20:24 +01:00
|
|
|
|
2024-06-13 13:24:24 +02:00
|
|
|
"golang.org/x/exp/maps"
|
2024-03-08 18:28:13 +01:00
|
|
|
"google.golang.org/grpc/codes"
|
|
|
|
gstatus "google.golang.org/grpc/status"
|
|
|
|
|
2024-10-02 18:24:22 +02:00
|
|
|
"github.com/netbirdio/netbird/client/iface/configurer"
|
2024-01-22 12:20:24 +01:00
|
|
|
"github.com/netbirdio/netbird/client/internal/relay"
|
2024-06-13 13:24:24 +02:00
|
|
|
"github.com/netbirdio/netbird/management/domain"
|
2024-09-08 12:06:14 +02:00
|
|
|
relayClient "github.com/netbirdio/netbird/relay/client"
|
2023-03-03 19:49:18 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
// State contains the latest state of a peer
|
|
|
|
type State struct {
|
2024-04-11 22:12:23 +02:00
|
|
|
Mux *sync.RWMutex
|
2024-01-22 12:20:24 +01:00
|
|
|
IP string
|
|
|
|
PubKey string
|
|
|
|
FQDN string
|
|
|
|
ConnStatus ConnStatus
|
|
|
|
ConnStatusUpdate time.Time
|
|
|
|
Relayed bool
|
|
|
|
LocalIceCandidateType string
|
|
|
|
RemoteIceCandidateType string
|
|
|
|
LocalIceCandidateEndpoint string
|
|
|
|
RemoteIceCandidateEndpoint string
|
2024-09-08 12:06:14 +02:00
|
|
|
RelayServerAddress string
|
2024-01-22 12:20:24 +01:00
|
|
|
LastWireguardHandshake time.Time
|
|
|
|
BytesTx int64
|
|
|
|
BytesRx int64
|
2024-03-20 11:18:34 +01:00
|
|
|
Latency time.Duration
|
2024-02-24 12:41:13 +01:00
|
|
|
RosenpassEnabled bool
|
2024-04-11 22:12:23 +02:00
|
|
|
routes map[string]struct{}
|
|
|
|
}
|
|
|
|
|
|
|
|
// AddRoute add a single route to routes map
|
|
|
|
func (s *State) AddRoute(network string) {
|
|
|
|
s.Mux.Lock()
|
2024-06-13 13:24:24 +02:00
|
|
|
defer s.Mux.Unlock()
|
2024-04-11 22:12:23 +02:00
|
|
|
if s.routes == nil {
|
|
|
|
s.routes = make(map[string]struct{})
|
|
|
|
}
|
|
|
|
s.routes[network] = struct{}{}
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetRoutes set state routes
|
|
|
|
func (s *State) SetRoutes(routes map[string]struct{}) {
|
|
|
|
s.Mux.Lock()
|
2024-06-13 13:24:24 +02:00
|
|
|
defer s.Mux.Unlock()
|
2024-04-11 22:12:23 +02:00
|
|
|
s.routes = routes
|
|
|
|
}
|
|
|
|
|
|
|
|
// DeleteRoute removes a route from the network amp
|
|
|
|
func (s *State) DeleteRoute(network string) {
|
|
|
|
s.Mux.Lock()
|
2024-06-13 13:24:24 +02:00
|
|
|
defer s.Mux.Unlock()
|
2024-04-11 22:12:23 +02:00
|
|
|
delete(s.routes, network)
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetRoutes return routes map
|
|
|
|
func (s *State) GetRoutes() map[string]struct{} {
|
|
|
|
s.Mux.RLock()
|
|
|
|
defer s.Mux.RUnlock()
|
2024-11-11 14:55:10 +01:00
|
|
|
return maps.Clone(s.routes)
|
2023-03-03 19:49:18 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// LocalPeerState contains the latest state of the local peer
|
|
|
|
type LocalPeerState struct {
|
|
|
|
IP string
|
|
|
|
PubKey string
|
|
|
|
KernelInterface bool
|
|
|
|
FQDN string
|
2024-03-12 19:06:16 +01:00
|
|
|
Routes map[string]struct{}
|
2023-03-03 19:49:18 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// SignalState contains the latest state of a signal connection
|
|
|
|
type SignalState struct {
|
|
|
|
URL string
|
|
|
|
Connected bool
|
2024-01-22 12:20:24 +01:00
|
|
|
Error error
|
2023-03-03 19:49:18 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// ManagementState contains the latest state of a management connection
|
|
|
|
type ManagementState struct {
|
|
|
|
URL string
|
|
|
|
Connected bool
|
2024-01-22 12:20:24 +01:00
|
|
|
Error error
|
2023-03-03 19:49:18 +01:00
|
|
|
}
|
2022-01-10 18:43:13 +01:00
|
|
|
|
2024-02-24 12:41:13 +01:00
|
|
|
// RosenpassState contains the latest state of the Rosenpass configuration
|
|
|
|
type RosenpassState struct {
|
|
|
|
Enabled bool
|
|
|
|
Permissive bool
|
|
|
|
}
|
|
|
|
|
2024-03-12 19:06:16 +01:00
|
|
|
// NSGroupState represents the status of a DNS server group, including associated domains,
|
|
|
|
// whether it's enabled, and the last error message encountered during probing.
|
|
|
|
type NSGroupState struct {
|
|
|
|
ID string
|
|
|
|
Servers []string
|
|
|
|
Domains []string
|
|
|
|
Enabled bool
|
|
|
|
Error error
|
|
|
|
}
|
|
|
|
|
2023-03-03 19:49:18 +01:00
|
|
|
// FullStatus contains the full state held by the Status instance
|
|
|
|
type FullStatus struct {
|
|
|
|
Peers []State
|
|
|
|
ManagementState ManagementState
|
|
|
|
SignalState SignalState
|
|
|
|
LocalPeerState LocalPeerState
|
2024-02-24 12:41:13 +01:00
|
|
|
RosenpassState RosenpassState
|
2024-01-22 12:20:24 +01:00
|
|
|
Relays []relay.ProbeResult
|
2024-03-12 19:06:16 +01:00
|
|
|
NSGroupStates []NSGroupState
|
2023-03-03 19:49:18 +01:00
|
|
|
}
|
|
|
|
|
2024-01-22 12:20:24 +01:00
|
|
|
// Status holds a state of peers, signal, management connections and relays
|
2023-03-03 19:49:18 +01:00
|
|
|
type Status struct {
|
2024-06-13 13:24:24 +02:00
|
|
|
mux sync.Mutex
|
|
|
|
peers map[string]State
|
|
|
|
changeNotify map[string]chan struct{}
|
|
|
|
signalState bool
|
|
|
|
signalError error
|
|
|
|
managementState bool
|
|
|
|
managementError error
|
|
|
|
relayStates []relay.ProbeResult
|
|
|
|
localPeer LocalPeerState
|
|
|
|
offlinePeers []State
|
|
|
|
mgmAddress string
|
|
|
|
signalAddress string
|
|
|
|
notifier *notifier
|
|
|
|
rosenpassEnabled bool
|
|
|
|
rosenpassPermissive bool
|
|
|
|
nsGroupStates []NSGroupState
|
|
|
|
resolvedDomainsStates map[domain.Domain][]netip.Prefix
|
2023-06-19 11:20:34 +02:00
|
|
|
|
|
|
|
// To reduce the number of notification invocation this bool will be true when need to call the notification
|
|
|
|
// Some Peer actions mostly used by in a batch when the network map has been synchronized. In these type of events
|
|
|
|
// set to true this variable and at the end of the processing we will reset it by the FinishPeerListModifications()
|
|
|
|
peerListChangedForNotification bool
|
2024-09-08 12:06:14 +02:00
|
|
|
|
|
|
|
relayMgr *relayClient.Manager
|
2023-03-03 19:49:18 +01:00
|
|
|
}
|
2022-01-10 18:43:13 +01:00
|
|
|
|
2023-03-03 19:49:18 +01:00
|
|
|
// NewRecorder returns a new Status instance
|
2023-03-16 17:22:36 +01:00
|
|
|
func NewRecorder(mgmAddress string) *Status {
|
2023-03-03 19:49:18 +01:00
|
|
|
return &Status{
|
2024-06-13 13:24:24 +02:00
|
|
|
peers: make(map[string]State),
|
|
|
|
changeNotify: make(map[string]chan struct{}),
|
|
|
|
offlinePeers: make([]State, 0),
|
|
|
|
notifier: newNotifier(),
|
|
|
|
mgmAddress: mgmAddress,
|
|
|
|
resolvedDomainsStates: make(map[domain.Domain][]netip.Prefix),
|
2022-01-10 18:43:13 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-09-08 12:06:14 +02:00
|
|
|
func (d *Status) SetRelayMgr(manager *relayClient.Manager) {
|
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
|
|
|
d.relayMgr = manager
|
|
|
|
}
|
|
|
|
|
2023-03-07 10:17:25 +01:00
|
|
|
// ReplaceOfflinePeers replaces
|
|
|
|
func (d *Status) ReplaceOfflinePeers(replacement []State) {
|
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
|
|
|
d.offlinePeers = make([]State, len(replacement))
|
|
|
|
copy(d.offlinePeers, replacement)
|
2023-06-19 11:20:34 +02:00
|
|
|
|
|
|
|
// todo we should set to true in case if the list changed only
|
|
|
|
d.peerListChangedForNotification = true
|
2023-03-07 10:17:25 +01:00
|
|
|
}
|
|
|
|
|
2023-03-03 19:49:18 +01:00
|
|
|
// AddPeer adds peer to Daemon status map
|
2023-06-19 11:20:34 +02:00
|
|
|
func (d *Status) AddPeer(peerPubKey string, fqdn string) error {
|
2023-03-03 19:49:18 +01:00
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
|
|
|
|
|
|
|
_, ok := d.peers[peerPubKey]
|
|
|
|
if ok {
|
|
|
|
return errors.New("peer already exist")
|
|
|
|
}
|
2023-06-19 11:20:34 +02:00
|
|
|
d.peers[peerPubKey] = State{
|
|
|
|
PubKey: peerPubKey,
|
|
|
|
ConnStatus: StatusDisconnected,
|
|
|
|
FQDN: fqdn,
|
2024-04-11 22:12:23 +02:00
|
|
|
Mux: new(sync.RWMutex),
|
2023-06-19 11:20:34 +02:00
|
|
|
}
|
|
|
|
d.peerListChangedForNotification = true
|
2023-03-03 19:49:18 +01:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetPeer adds peer to Daemon status map
|
|
|
|
func (d *Status) GetPeer(peerPubKey string) (State, error) {
|
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
|
|
|
|
|
|
|
state, ok := d.peers[peerPubKey]
|
|
|
|
if !ok {
|
2024-10-02 18:24:22 +02:00
|
|
|
return State{}, configurer.ErrPeerNotFound
|
2023-03-03 19:49:18 +01:00
|
|
|
}
|
|
|
|
return state, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// RemovePeer removes peer from Daemon status map
|
|
|
|
func (d *Status) RemovePeer(peerPubKey string) error {
|
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
|
|
|
|
|
|
|
_, ok := d.peers[peerPubKey]
|
2023-06-19 11:20:34 +02:00
|
|
|
if !ok {
|
|
|
|
return errors.New("no peer with to remove")
|
2023-03-03 19:49:18 +01:00
|
|
|
}
|
|
|
|
|
2023-06-19 11:20:34 +02:00
|
|
|
delete(d.peers, peerPubKey)
|
|
|
|
d.peerListChangedForNotification = true
|
|
|
|
return nil
|
2023-03-03 19:49:18 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// UpdatePeerState updates peer status
|
|
|
|
func (d *Status) UpdatePeerState(receivedState State) error {
|
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
|
|
|
|
|
|
|
peerState, ok := d.peers[receivedState.PubKey]
|
|
|
|
if !ok {
|
|
|
|
return errors.New("peer doesn't exist")
|
|
|
|
}
|
|
|
|
|
|
|
|
if receivedState.IP != "" {
|
|
|
|
peerState.IP = receivedState.IP
|
|
|
|
}
|
|
|
|
|
2024-09-08 12:06:14 +02:00
|
|
|
skipNotification := shouldSkipNotify(receivedState.ConnStatus, peerState)
|
2023-06-17 09:03:52 +02:00
|
|
|
|
2023-03-03 19:49:18 +01:00
|
|
|
if receivedState.ConnStatus != peerState.ConnStatus {
|
|
|
|
peerState.ConnStatus = receivedState.ConnStatus
|
|
|
|
peerState.ConnStatusUpdate = receivedState.ConnStatusUpdate
|
|
|
|
peerState.Relayed = receivedState.Relayed
|
|
|
|
peerState.LocalIceCandidateType = receivedState.LocalIceCandidateType
|
|
|
|
peerState.RemoteIceCandidateType = receivedState.RemoteIceCandidateType
|
2024-01-22 12:20:24 +01:00
|
|
|
peerState.LocalIceCandidateEndpoint = receivedState.LocalIceCandidateEndpoint
|
|
|
|
peerState.RemoteIceCandidateEndpoint = receivedState.RemoteIceCandidateEndpoint
|
2024-09-08 12:06:14 +02:00
|
|
|
peerState.RelayServerAddress = receivedState.RelayServerAddress
|
2024-02-24 12:41:13 +01:00
|
|
|
peerState.RosenpassEnabled = receivedState.RosenpassEnabled
|
2023-03-03 19:49:18 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
d.peers[receivedState.PubKey] = peerState
|
|
|
|
|
2023-06-17 09:03:52 +02:00
|
|
|
if skipNotification {
|
2023-06-01 16:00:44 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-11-11 10:53:57 +01:00
|
|
|
d.notifyPeerListChanged()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *Status) AddPeerStateRoute(peer string, route string) error {
|
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
|
|
|
|
|
|
|
peerState, ok := d.peers[peer]
|
|
|
|
if !ok {
|
|
|
|
return errors.New("peer doesn't exist")
|
2023-03-03 19:49:18 +01:00
|
|
|
}
|
|
|
|
|
2024-11-11 10:53:57 +01:00
|
|
|
peerState.AddRoute(route)
|
|
|
|
d.peers[peer] = peerState
|
|
|
|
|
|
|
|
// todo: consider to make sense of this notification or not
|
|
|
|
d.notifyPeerListChanged()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *Status) RemovePeerStateRoute(peer string, route string) error {
|
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
|
|
|
|
|
|
|
peerState, ok := d.peers[peer]
|
|
|
|
if !ok {
|
|
|
|
return errors.New("peer doesn't exist")
|
|
|
|
}
|
|
|
|
|
|
|
|
peerState.DeleteRoute(route)
|
|
|
|
d.peers[peer] = peerState
|
|
|
|
|
|
|
|
// todo: consider to make sense of this notification or not
|
2023-03-17 10:37:27 +01:00
|
|
|
d.notifyPeerListChanged()
|
2023-03-03 19:49:18 +01:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-09-08 12:06:14 +02:00
|
|
|
func (d *Status) UpdatePeerICEState(receivedState State) error {
|
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
|
|
|
|
|
|
|
peerState, ok := d.peers[receivedState.PubKey]
|
|
|
|
if !ok {
|
|
|
|
return errors.New("peer doesn't exist")
|
|
|
|
}
|
|
|
|
|
|
|
|
if receivedState.IP != "" {
|
|
|
|
peerState.IP = receivedState.IP
|
|
|
|
}
|
|
|
|
|
|
|
|
skipNotification := shouldSkipNotify(receivedState.ConnStatus, peerState)
|
|
|
|
|
|
|
|
peerState.ConnStatus = receivedState.ConnStatus
|
|
|
|
peerState.ConnStatusUpdate = receivedState.ConnStatusUpdate
|
|
|
|
peerState.Relayed = receivedState.Relayed
|
|
|
|
peerState.LocalIceCandidateType = receivedState.LocalIceCandidateType
|
|
|
|
peerState.RemoteIceCandidateType = receivedState.RemoteIceCandidateType
|
|
|
|
peerState.LocalIceCandidateEndpoint = receivedState.LocalIceCandidateEndpoint
|
|
|
|
peerState.RemoteIceCandidateEndpoint = receivedState.RemoteIceCandidateEndpoint
|
|
|
|
peerState.RosenpassEnabled = receivedState.RosenpassEnabled
|
|
|
|
|
|
|
|
d.peers[receivedState.PubKey] = peerState
|
|
|
|
|
|
|
|
if skipNotification {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-11-11 10:53:57 +01:00
|
|
|
d.notifyPeerStateChangeListeners(receivedState.PubKey)
|
2024-09-08 12:06:14 +02:00
|
|
|
d.notifyPeerListChanged()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *Status) UpdatePeerRelayedState(receivedState State) error {
|
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
|
|
|
|
|
|
|
peerState, ok := d.peers[receivedState.PubKey]
|
|
|
|
if !ok {
|
|
|
|
return errors.New("peer doesn't exist")
|
|
|
|
}
|
|
|
|
|
|
|
|
skipNotification := shouldSkipNotify(receivedState.ConnStatus, peerState)
|
|
|
|
|
|
|
|
peerState.ConnStatus = receivedState.ConnStatus
|
|
|
|
peerState.ConnStatusUpdate = receivedState.ConnStatusUpdate
|
|
|
|
peerState.Relayed = receivedState.Relayed
|
|
|
|
peerState.RelayServerAddress = receivedState.RelayServerAddress
|
|
|
|
peerState.RosenpassEnabled = receivedState.RosenpassEnabled
|
|
|
|
|
|
|
|
d.peers[receivedState.PubKey] = peerState
|
|
|
|
|
|
|
|
if skipNotification {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-11-11 10:53:57 +01:00
|
|
|
d.notifyPeerStateChangeListeners(receivedState.PubKey)
|
2024-09-08 12:06:14 +02:00
|
|
|
d.notifyPeerListChanged()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *Status) UpdatePeerRelayedStateToDisconnected(receivedState State) error {
|
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
|
|
|
|
|
|
|
peerState, ok := d.peers[receivedState.PubKey]
|
|
|
|
if !ok {
|
|
|
|
return errors.New("peer doesn't exist")
|
|
|
|
}
|
|
|
|
|
|
|
|
skipNotification := shouldSkipNotify(receivedState.ConnStatus, peerState)
|
|
|
|
|
|
|
|
peerState.ConnStatus = receivedState.ConnStatus
|
|
|
|
peerState.Relayed = receivedState.Relayed
|
|
|
|
peerState.ConnStatusUpdate = receivedState.ConnStatusUpdate
|
|
|
|
peerState.RelayServerAddress = ""
|
|
|
|
|
|
|
|
d.peers[receivedState.PubKey] = peerState
|
|
|
|
|
|
|
|
if skipNotification {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-11-11 10:53:57 +01:00
|
|
|
d.notifyPeerStateChangeListeners(receivedState.PubKey)
|
2024-09-08 12:06:14 +02:00
|
|
|
d.notifyPeerListChanged()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *Status) UpdatePeerICEStateToDisconnected(receivedState State) error {
|
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
|
|
|
|
|
|
|
peerState, ok := d.peers[receivedState.PubKey]
|
|
|
|
if !ok {
|
|
|
|
return errors.New("peer doesn't exist")
|
|
|
|
}
|
|
|
|
|
|
|
|
skipNotification := shouldSkipNotify(receivedState.ConnStatus, peerState)
|
|
|
|
|
|
|
|
peerState.ConnStatus = receivedState.ConnStatus
|
|
|
|
peerState.Relayed = receivedState.Relayed
|
|
|
|
peerState.ConnStatusUpdate = receivedState.ConnStatusUpdate
|
|
|
|
peerState.LocalIceCandidateType = receivedState.LocalIceCandidateType
|
|
|
|
peerState.RemoteIceCandidateType = receivedState.RemoteIceCandidateType
|
|
|
|
peerState.LocalIceCandidateEndpoint = receivedState.LocalIceCandidateEndpoint
|
|
|
|
peerState.RemoteIceCandidateEndpoint = receivedState.RemoteIceCandidateEndpoint
|
|
|
|
|
|
|
|
d.peers[receivedState.PubKey] = peerState
|
|
|
|
|
|
|
|
if skipNotification {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-11-11 10:53:57 +01:00
|
|
|
d.notifyPeerStateChangeListeners(receivedState.PubKey)
|
2024-09-08 12:06:14 +02:00
|
|
|
d.notifyPeerListChanged()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-02-24 12:41:13 +01:00
|
|
|
// UpdateWireGuardPeerState updates the WireGuard bits of the peer state
|
2024-10-02 18:24:22 +02:00
|
|
|
func (d *Status) UpdateWireGuardPeerState(pubKey string, wgStats configurer.WGStats) error {
|
2024-01-22 12:20:24 +01:00
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
|
|
|
|
|
|
|
peerState, ok := d.peers[pubKey]
|
|
|
|
if !ok {
|
|
|
|
return errors.New("peer doesn't exist")
|
|
|
|
}
|
|
|
|
|
|
|
|
peerState.LastWireguardHandshake = wgStats.LastHandshake
|
|
|
|
peerState.BytesRx = wgStats.RxBytes
|
|
|
|
peerState.BytesTx = wgStats.TxBytes
|
|
|
|
|
|
|
|
d.peers[pubKey] = peerState
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-09-08 12:06:14 +02:00
|
|
|
func shouldSkipNotify(receivedConnStatus ConnStatus, curr State) bool {
|
2023-06-11 21:51:33 +02:00
|
|
|
switch {
|
2024-09-08 12:06:14 +02:00
|
|
|
case receivedConnStatus == StatusConnecting:
|
2023-06-11 21:51:33 +02:00
|
|
|
return true
|
2024-09-08 12:06:14 +02:00
|
|
|
case receivedConnStatus == StatusDisconnected && curr.ConnStatus == StatusConnecting:
|
2023-06-11 21:51:33 +02:00
|
|
|
return true
|
2024-09-08 12:06:14 +02:00
|
|
|
case receivedConnStatus == StatusDisconnected && curr.ConnStatus == StatusDisconnected:
|
2023-06-11 21:51:33 +02:00
|
|
|
return curr.IP != ""
|
|
|
|
default:
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-03 19:49:18 +01:00
|
|
|
// UpdatePeerFQDN update peer's state fqdn only
|
|
|
|
func (d *Status) UpdatePeerFQDN(peerPubKey, fqdn string) error {
|
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
|
|
|
|
|
|
|
peerState, ok := d.peers[peerPubKey]
|
|
|
|
if !ok {
|
|
|
|
return errors.New("peer doesn't exist")
|
|
|
|
}
|
|
|
|
|
|
|
|
peerState.FQDN = fqdn
|
|
|
|
d.peers[peerPubKey] = peerState
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-06-19 11:20:34 +02:00
|
|
|
// FinishPeerListModifications this event invoke the notification
|
|
|
|
func (d *Status) FinishPeerListModifications() {
|
|
|
|
d.mux.Lock()
|
|
|
|
|
|
|
|
if !d.peerListChangedForNotification {
|
|
|
|
d.mux.Unlock()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
d.peerListChangedForNotification = false
|
|
|
|
d.mux.Unlock()
|
|
|
|
|
|
|
|
d.notifyPeerListChanged()
|
|
|
|
}
|
|
|
|
|
2023-03-03 19:49:18 +01:00
|
|
|
// GetPeerStateChangeNotifier returns a change notifier channel for a peer
|
|
|
|
func (d *Status) GetPeerStateChangeNotifier(peer string) <-chan struct{} {
|
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
2024-11-11 10:53:57 +01:00
|
|
|
|
2023-03-03 19:49:18 +01:00
|
|
|
ch, found := d.changeNotify[peer]
|
2024-11-11 10:53:57 +01:00
|
|
|
if found {
|
|
|
|
return ch
|
2023-03-03 19:49:18 +01:00
|
|
|
}
|
2024-11-11 10:53:57 +01:00
|
|
|
|
|
|
|
ch = make(chan struct{})
|
|
|
|
d.changeNotify[peer] = ch
|
2023-03-03 19:49:18 +01:00
|
|
|
return ch
|
|
|
|
}
|
|
|
|
|
2024-03-12 19:06:16 +01:00
|
|
|
// GetLocalPeerState returns the local peer state
|
|
|
|
func (d *Status) GetLocalPeerState() LocalPeerState {
|
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
|
|
|
return d.localPeer
|
|
|
|
}
|
|
|
|
|
2023-03-03 19:49:18 +01:00
|
|
|
// UpdateLocalPeerState updates local peer status
|
|
|
|
func (d *Status) UpdateLocalPeerState(localPeerState LocalPeerState) {
|
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
|
|
|
|
|
|
|
d.localPeer = localPeerState
|
2023-03-24 18:51:35 +01:00
|
|
|
d.notifyAddressChanged()
|
2023-03-03 19:49:18 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// CleanLocalPeerState cleans local peer status
|
|
|
|
func (d *Status) CleanLocalPeerState() {
|
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
|
|
|
|
|
|
|
d.localPeer = LocalPeerState{}
|
2023-03-24 18:51:35 +01:00
|
|
|
d.notifyAddressChanged()
|
2023-03-03 19:49:18 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// MarkManagementDisconnected sets ManagementState to disconnected
|
2024-01-22 12:20:24 +01:00
|
|
|
func (d *Status) MarkManagementDisconnected(err error) {
|
2023-03-03 19:49:18 +01:00
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
2023-03-17 10:37:27 +01:00
|
|
|
defer d.onConnectionChanged()
|
|
|
|
|
2023-03-16 17:22:36 +01:00
|
|
|
d.managementState = false
|
2024-01-22 12:20:24 +01:00
|
|
|
d.managementError = err
|
2023-03-03 19:49:18 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// MarkManagementConnected sets ManagementState to connected
|
2023-03-16 17:22:36 +01:00
|
|
|
func (d *Status) MarkManagementConnected() {
|
2023-03-03 19:49:18 +01:00
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
2023-03-17 10:37:27 +01:00
|
|
|
defer d.onConnectionChanged()
|
|
|
|
|
2023-03-24 18:51:35 +01:00
|
|
|
d.managementState = true
|
2024-01-22 12:20:24 +01:00
|
|
|
d.managementError = nil
|
2023-03-16 17:22:36 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// UpdateSignalAddress update the address of the signal server
|
|
|
|
func (d *Status) UpdateSignalAddress(signalURL string) {
|
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
|
|
|
d.signalAddress = signalURL
|
|
|
|
}
|
|
|
|
|
|
|
|
// UpdateManagementAddress update the address of the management server
|
|
|
|
func (d *Status) UpdateManagementAddress(mgmAddress string) {
|
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
|
|
|
d.mgmAddress = mgmAddress
|
2023-03-03 19:49:18 +01:00
|
|
|
}
|
|
|
|
|
2024-02-24 12:41:13 +01:00
|
|
|
// UpdateRosenpass update the Rosenpass configuration
|
|
|
|
func (d *Status) UpdateRosenpass(rosenpassEnabled, rosenpassPermissive bool) {
|
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
|
|
|
d.rosenpassPermissive = rosenpassPermissive
|
|
|
|
d.rosenpassEnabled = rosenpassEnabled
|
|
|
|
}
|
|
|
|
|
2023-03-03 19:49:18 +01:00
|
|
|
// MarkSignalDisconnected sets SignalState to disconnected
|
2024-01-22 12:20:24 +01:00
|
|
|
func (d *Status) MarkSignalDisconnected(err error) {
|
2023-03-03 19:49:18 +01:00
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
2023-03-17 10:37:27 +01:00
|
|
|
defer d.onConnectionChanged()
|
|
|
|
|
2023-03-24 18:51:35 +01:00
|
|
|
d.signalState = false
|
2024-01-22 12:20:24 +01:00
|
|
|
d.signalError = err
|
2023-03-03 19:49:18 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// MarkSignalConnected sets SignalState to connected
|
2023-03-16 17:22:36 +01:00
|
|
|
func (d *Status) MarkSignalConnected() {
|
2023-03-03 19:49:18 +01:00
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
2023-03-17 10:37:27 +01:00
|
|
|
defer d.onConnectionChanged()
|
|
|
|
|
2023-03-16 17:22:36 +01:00
|
|
|
d.signalState = true
|
2024-01-22 12:20:24 +01:00
|
|
|
d.signalError = nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *Status) UpdateRelayStates(relayResults []relay.ProbeResult) {
|
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
|
|
|
d.relayStates = relayResults
|
|
|
|
}
|
|
|
|
|
2024-03-12 19:06:16 +01:00
|
|
|
func (d *Status) UpdateDNSStates(dnsStates []NSGroupState) {
|
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
|
|
|
d.nsGroupStates = dnsStates
|
|
|
|
}
|
|
|
|
|
2024-06-13 13:24:24 +02:00
|
|
|
func (d *Status) UpdateResolvedDomainsStates(domain domain.Domain, prefixes []netip.Prefix) {
|
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
|
|
|
d.resolvedDomainsStates[domain] = prefixes
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *Status) DeleteResolvedDomainsStates(domain domain.Domain) {
|
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
|
|
|
delete(d.resolvedDomainsStates, domain)
|
|
|
|
}
|
|
|
|
|
2024-02-24 12:41:13 +01:00
|
|
|
func (d *Status) GetRosenpassState() RosenpassState {
|
2024-09-15 16:07:26 +02:00
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
2024-02-24 12:41:13 +01:00
|
|
|
return RosenpassState{
|
|
|
|
d.rosenpassEnabled,
|
|
|
|
d.rosenpassPermissive,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-01-22 12:20:24 +01:00
|
|
|
func (d *Status) GetManagementState() ManagementState {
|
2024-09-15 16:07:26 +02:00
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
2024-01-22 12:20:24 +01:00
|
|
|
return ManagementState{
|
|
|
|
d.mgmAddress,
|
|
|
|
d.managementState,
|
|
|
|
d.managementError,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-20 11:18:34 +01:00
|
|
|
func (d *Status) UpdateLatency(pubKey string, latency time.Duration) error {
|
|
|
|
if latency <= 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
|
|
|
peerState, ok := d.peers[pubKey]
|
|
|
|
if !ok {
|
|
|
|
return errors.New("peer doesn't exist")
|
|
|
|
}
|
|
|
|
peerState.Latency = latency
|
|
|
|
d.peers[pubKey] = peerState
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-03-08 18:28:13 +01:00
|
|
|
// IsLoginRequired determines if a peer's login has expired.
|
|
|
|
func (d *Status) IsLoginRequired() bool {
|
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
|
|
|
|
|
|
|
// if peer is connected to the management then login is not expired
|
|
|
|
if d.managementState {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
s, ok := gstatus.FromError(d.managementError)
|
|
|
|
if ok && (s.Code() == codes.InvalidArgument || s.Code() == codes.PermissionDenied) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2024-01-22 12:20:24 +01:00
|
|
|
func (d *Status) GetSignalState() SignalState {
|
2024-09-15 16:07:26 +02:00
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
2024-01-22 12:20:24 +01:00
|
|
|
return SignalState{
|
|
|
|
d.signalAddress,
|
|
|
|
d.signalState,
|
|
|
|
d.signalError,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-09-08 12:06:14 +02:00
|
|
|
// GetRelayStates returns the stun/turn/permanent relay states
|
2024-01-22 12:20:24 +01:00
|
|
|
func (d *Status) GetRelayStates() []relay.ProbeResult {
|
2024-09-15 16:07:26 +02:00
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
2024-09-08 12:06:14 +02:00
|
|
|
if d.relayMgr == nil {
|
|
|
|
return d.relayStates
|
|
|
|
}
|
|
|
|
|
|
|
|
// extend the list of stun, turn servers with relay address
|
|
|
|
relayStates := slices.Clone(d.relayStates)
|
|
|
|
|
|
|
|
var relayState relay.ProbeResult
|
|
|
|
|
|
|
|
// if the server connection is not established then we will use the general address
|
|
|
|
// in case of connection we will use the instance specific address
|
|
|
|
instanceAddr, err := d.relayMgr.RelayInstanceAddress()
|
|
|
|
if err != nil {
|
|
|
|
// TODO add their status
|
|
|
|
if errors.Is(err, relayClient.ErrRelayClientNotConnected) {
|
|
|
|
for _, r := range d.relayMgr.ServerURLs() {
|
|
|
|
relayStates = append(relayStates, relay.ProbeResult{
|
|
|
|
URI: r,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
return relayStates
|
|
|
|
}
|
|
|
|
relayState.Err = err
|
|
|
|
}
|
|
|
|
|
|
|
|
relayState.URI = instanceAddr
|
|
|
|
return append(relayStates, relayState)
|
2023-03-03 19:49:18 +01:00
|
|
|
}
|
|
|
|
|
2024-03-12 19:06:16 +01:00
|
|
|
func (d *Status) GetDNSStates() []NSGroupState {
|
2024-09-15 16:07:26 +02:00
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
2024-03-12 19:06:16 +01:00
|
|
|
return d.nsGroupStates
|
|
|
|
}
|
|
|
|
|
2024-06-13 13:24:24 +02:00
|
|
|
func (d *Status) GetResolvedDomainsStates() map[domain.Domain][]netip.Prefix {
|
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
|
|
|
return maps.Clone(d.resolvedDomainsStates)
|
|
|
|
}
|
|
|
|
|
2023-03-03 19:49:18 +01:00
|
|
|
// GetFullStatus gets full status
|
|
|
|
func (d *Status) GetFullStatus() FullStatus {
|
|
|
|
fullStatus := FullStatus{
|
2024-01-22 12:20:24 +01:00
|
|
|
ManagementState: d.GetManagementState(),
|
|
|
|
SignalState: d.GetSignalState(),
|
|
|
|
Relays: d.GetRelayStates(),
|
2024-02-24 12:41:13 +01:00
|
|
|
RosenpassState: d.GetRosenpassState(),
|
2024-03-12 19:06:16 +01:00
|
|
|
NSGroupStates: d.GetDNSStates(),
|
2023-03-03 19:49:18 +01:00
|
|
|
}
|
|
|
|
|
2024-09-15 16:07:26 +02:00
|
|
|
d.mux.Lock()
|
|
|
|
defer d.mux.Unlock()
|
|
|
|
|
|
|
|
fullStatus.LocalPeerState = d.localPeer
|
|
|
|
|
2023-03-03 19:49:18 +01:00
|
|
|
for _, status := range d.peers {
|
|
|
|
fullStatus.Peers = append(fullStatus.Peers, status)
|
|
|
|
}
|
|
|
|
|
2023-03-07 10:17:25 +01:00
|
|
|
fullStatus.Peers = append(fullStatus.Peers, d.offlinePeers...)
|
2023-03-03 19:49:18 +01:00
|
|
|
return fullStatus
|
|
|
|
}
|
2023-03-17 10:37:27 +01:00
|
|
|
|
|
|
|
// ClientStart will notify all listeners about the new service state
|
|
|
|
func (d *Status) ClientStart() {
|
|
|
|
d.notifier.clientStart()
|
|
|
|
}
|
|
|
|
|
|
|
|
// ClientStop will notify all listeners about the new service state
|
|
|
|
func (d *Status) ClientStop() {
|
|
|
|
d.notifier.clientStop()
|
|
|
|
}
|
|
|
|
|
2023-03-29 10:39:54 +02:00
|
|
|
// ClientTeardown will notify all listeners about the service is under teardown
|
|
|
|
func (d *Status) ClientTeardown() {
|
|
|
|
d.notifier.clientTearDown()
|
|
|
|
}
|
|
|
|
|
2023-04-03 16:59:13 +02:00
|
|
|
// SetConnectionListener set a listener to the notifier
|
|
|
|
func (d *Status) SetConnectionListener(listener Listener) {
|
|
|
|
d.notifier.setListener(listener)
|
2023-03-17 10:37:27 +01:00
|
|
|
}
|
|
|
|
|
2023-04-03 16:59:13 +02:00
|
|
|
// RemoveConnectionListener remove the listener from the notifier
|
|
|
|
func (d *Status) RemoveConnectionListener() {
|
|
|
|
d.notifier.removeListener()
|
2023-03-17 10:37:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (d *Status) onConnectionChanged() {
|
|
|
|
d.notifier.updateServerStates(d.managementState, d.signalState)
|
|
|
|
}
|
|
|
|
|
2024-11-11 10:53:57 +01:00
|
|
|
// notifyPeerStateChangeListeners notifies route manager about the change in peer state
|
|
|
|
func (d *Status) notifyPeerStateChangeListeners(peerID string) {
|
|
|
|
ch, found := d.changeNotify[peerID]
|
|
|
|
if !found {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
close(ch)
|
|
|
|
delete(d.changeNotify, peerID)
|
|
|
|
}
|
|
|
|
|
2023-03-17 10:37:27 +01:00
|
|
|
func (d *Status) notifyPeerListChanged() {
|
2023-08-04 14:14:08 +02:00
|
|
|
d.notifier.peerListChanged(d.numOfPeers())
|
2023-03-17 10:37:27 +01:00
|
|
|
}
|
2023-03-24 18:51:35 +01:00
|
|
|
|
|
|
|
func (d *Status) notifyAddressChanged() {
|
|
|
|
d.notifier.localAddressChanged(d.localPeer.FQDN, d.localPeer.IP)
|
|
|
|
}
|
2023-08-04 14:14:08 +02:00
|
|
|
|
|
|
|
func (d *Status) numOfPeers() int {
|
|
|
|
return len(d.peers) + len(d.offlinePeers)
|
|
|
|
}
|