2023-06-21 03:29:52 -06:00
|
|
|
package notifier
|
|
|
|
|
|
|
|
import (
|
|
|
|
"sync"
|
|
|
|
|
2023-06-29 04:20:22 -06:00
|
|
|
"github.com/juanfont/headscale/hscontrol/types"
|
2023-06-21 03:29:52 -06:00
|
|
|
"github.com/juanfont/headscale/hscontrol/util"
|
2023-07-24 00:58:51 -06:00
|
|
|
"github.com/rs/zerolog/log"
|
2023-11-19 14:37:04 -07:00
|
|
|
"tailscale.com/types/key"
|
2023-06-21 03:29:52 -06:00
|
|
|
)
|
|
|
|
|
|
|
|
type Notifier struct {
|
2023-09-11 05:08:44 -06:00
|
|
|
l sync.RWMutex
|
2023-06-29 04:20:22 -06:00
|
|
|
nodes map[string]chan<- types.StateUpdate
|
2023-06-21 03:29:52 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
func NewNotifier() *Notifier {
|
|
|
|
return &Notifier{}
|
|
|
|
}
|
|
|
|
|
2023-11-19 14:37:04 -07:00
|
|
|
func (n *Notifier) AddNode(machineKey key.MachinePublic, c chan<- types.StateUpdate) {
|
|
|
|
log.Trace().Caller().Str("key", machineKey.ShortString()).Msg("acquiring lock to add node")
|
|
|
|
defer log.Trace().Caller().Str("key", machineKey.ShortString()).Msg("releasing lock to add node")
|
2023-09-11 05:08:44 -06:00
|
|
|
|
2023-06-21 03:29:52 -06:00
|
|
|
n.l.Lock()
|
|
|
|
defer n.l.Unlock()
|
|
|
|
|
|
|
|
if n.nodes == nil {
|
2023-06-29 04:20:22 -06:00
|
|
|
n.nodes = make(map[string]chan<- types.StateUpdate)
|
2023-06-21 03:29:52 -06:00
|
|
|
}
|
|
|
|
|
2023-11-19 14:37:04 -07:00
|
|
|
n.nodes[machineKey.String()] = c
|
2023-07-24 00:58:51 -06:00
|
|
|
|
|
|
|
log.Trace().
|
2023-11-19 14:37:04 -07:00
|
|
|
Str("machine_key", machineKey.ShortString()).
|
2023-07-24 00:58:51 -06:00
|
|
|
Int("open_chans", len(n.nodes)).
|
|
|
|
Msg("Added new channel")
|
2023-06-21 03:29:52 -06:00
|
|
|
}
|
|
|
|
|
2023-11-19 14:37:04 -07:00
|
|
|
func (n *Notifier) RemoveNode(machineKey key.MachinePublic) {
|
|
|
|
log.Trace().Caller().Str("key", machineKey.ShortString()).Msg("acquiring lock to remove node")
|
|
|
|
defer log.Trace().Caller().Str("key", machineKey.ShortString()).Msg("releasing lock to remove node")
|
2023-09-11 05:08:44 -06:00
|
|
|
|
2023-06-21 03:29:52 -06:00
|
|
|
n.l.Lock()
|
|
|
|
defer n.l.Unlock()
|
|
|
|
|
|
|
|
if n.nodes == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2023-11-19 14:37:04 -07:00
|
|
|
delete(n.nodes, machineKey.String())
|
2023-07-24 00:58:51 -06:00
|
|
|
|
|
|
|
log.Trace().
|
2023-11-19 14:37:04 -07:00
|
|
|
Str("machine_key", machineKey.ShortString()).
|
2023-07-24 00:58:51 -06:00
|
|
|
Int("open_chans", len(n.nodes)).
|
|
|
|
Msg("Removed channel")
|
2023-06-21 03:29:52 -06:00
|
|
|
}
|
|
|
|
|
2023-06-29 04:20:22 -06:00
|
|
|
func (n *Notifier) NotifyAll(update types.StateUpdate) {
|
|
|
|
n.NotifyWithIgnore(update)
|
2023-06-21 03:29:52 -06:00
|
|
|
}
|
|
|
|
|
2023-06-29 04:20:22 -06:00
|
|
|
func (n *Notifier) NotifyWithIgnore(update types.StateUpdate, ignore ...string) {
|
2023-09-11 05:08:44 -06:00
|
|
|
log.Trace().Caller().Interface("type", update.Type).Msg("acquiring lock to notify")
|
|
|
|
defer log.Trace().
|
|
|
|
Caller().
|
|
|
|
Interface("type", update.Type).
|
|
|
|
Msg("releasing lock, finished notifing")
|
|
|
|
|
|
|
|
n.l.RLock()
|
|
|
|
defer n.l.RUnlock()
|
2023-06-21 03:29:52 -06:00
|
|
|
|
|
|
|
for key, c := range n.nodes {
|
|
|
|
if util.IsStringInSlice(ignore, key) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2023-09-11 05:08:44 -06:00
|
|
|
log.Trace().Caller().Str("machine", key).Strs("ignoring", ignore).Msg("sending update")
|
2023-06-29 04:20:22 -06:00
|
|
|
c <- update
|
2023-06-21 03:29:52 -06:00
|
|
|
}
|
|
|
|
}
|