package runtime import ( "encoding/json" "os" "sync" "github.com/FreifunkBremen/yanic/jsontime" yanic "github.com/FreifunkBremen/yanic/runtime" "github.com/genofire/golang-lib/log" "github.com/FreifunkBremen/freifunkmanager/ssh" ) type Nodes struct { List map[string]*Node `json:"nodes"` Current map[string]*Node `json:"-"` ssh *ssh.Manager statePath string iface string notifyFunc []func(*Node, bool) sync.Mutex } func NewNodes(path string, iface string, mgmt *ssh.Manager) *Nodes { nodes := &Nodes{ List: make(map[string]*Node), Current: make(map[string]*Node), ssh: mgmt, statePath: path, iface: iface, } nodes.load() return nodes } func (nodes *Nodes) LearnNode(n *yanic.Node) { node := NewNode(n) if node == nil { return } node.Lastseen = jsontime.Now() logger := log.Log.WithField("method", "LearnNode").WithField("node_id", node.NodeID) nodes.Lock() defer nodes.Unlock() if lNode := nodes.List[node.NodeID]; lNode != nil { lNode.Lastseen = jsontime.Now() lNode.Stats = node.Stats } else { nodes.List[node.NodeID] = node nodes.notify(node, true) } if _, ok := nodes.Current[node.NodeID]; ok { nodes.Current[node.NodeID] = node nodes.notify(node, false) return } // session := nodes.ssh.ConnectTo(node.Address) result, err := nodes.ssh.RunOn(node.GetAddress(nodes.iface), "uptime") if err != nil { logger.Error("init ssh command not run", err) return } uptime := ssh.SSHResultToString(result) logger.Infof("new node with uptime: %s", uptime) nodes.Current[node.NodeID] = node if lNode := nodes.List[node.NodeID]; lNode != nil { lNode.Address = node.Address go lNode.SSHUpdate(nodes.ssh, nodes.iface, node) } nodes.notify(node, false) } func (nodes *Nodes) AddNotify(f func(*Node, bool)) { nodes.notifyFunc = append(nodes.notifyFunc, f) } func (nodes *Nodes) notify(node *Node, system bool) { for _, f := range nodes.notifyFunc { f(node, system) } } func (nodes *Nodes) UpdateNode(node *Node) { if node == nil { log.Log.Warn("no new node to update") return } nodes.Lock() defer nodes.Unlock() if n, ok := nodes.List[node.NodeID]; ok { go node.SSHUpdate(nodes.ssh, nodes.iface, n) } nodes.List[node.NodeID] = node nodes.notify(node, true) } func (nodes *Nodes) Updater() { nodes.Lock() defer nodes.Unlock() for nodeid, node := range nodes.List { if n, ok := nodes.Current[nodeid]; ok { go n.SSHUpdate(nodes.ssh, nodes.iface, node) } } log.Log.Debug("updater per ssh runs") } func (nodes *Nodes) load() { if f, err := os.Open(nodes.statePath); err == nil { // transform data to legacy meshviewer if err = json.NewDecoder(f).Decode(nodes); err == nil { log.Log.Infof("loaded %d nodes", len(nodes.List)) } else { log.Log.Error("failed to unmarshal nodes:", err) } } else { log.Log.Error("failed to load cached nodes:", err) } } func (nodes *Nodes) Saver() { nodes.Lock() yanic.SaveJSON(nodes, nodes.statePath) nodes.Unlock() log.Log.Debug("saved state file") }