package tendrils import ( "encoding/json" "log" "net/http" "sort" "github.com/fvbommel/sortorder" ) type StatusResponse struct { Nodes []*Node `json:"nodes"` Links []*Link `json:"links"` MulticastGroups []*MulticastGroupMembers `json:"multicast_groups"` ArtNetNodes []*ArtNetNode `json:"artnet_nodes"` DanteFlows []*DanteFlow `json:"dante_flows"` } func (t *Tendrils) startHTTPServer() { if t.HTTPPort == "" { return } mux := http.NewServeMux() mux.HandleFunc("/api/status", t.handleAPIStatus) mux.Handle("/", http.FileServer(http.Dir("static"))) log.Printf("[http] listening on %s", t.HTTPPort) go func() { if err := http.ListenAndServe(t.HTTPPort, mux); err != nil { log.Printf("[ERROR] http server failed: %v", err) } }() } func (t *Tendrils) handleAPIStatus(w http.ResponseWriter, r *http.Request) { status := t.GetStatus() w.Header().Set("Content-Type", "application/json") json.NewEncoder(w).Encode(status) } func (t *Tendrils) GetStatus() *StatusResponse { return &StatusResponse{ Nodes: t.getNodes(), Links: t.getLinks(), MulticastGroups: t.getMulticastGroups(), ArtNetNodes: t.getArtNetNodes(), DanteFlows: t.getDanteFlows(), } } func (t *Tendrils) getNodes() []*Node { t.nodes.mu.RLock() defer t.nodes.mu.RUnlock() nodes := make([]*Node, 0, len(t.nodes.nodes)) for _, node := range t.nodes.nodes { nodes = append(nodes, node) } sort.Slice(nodes, func(i, j int) bool { return sortorder.NaturalLess(nodes[i].DisplayName(), nodes[j].DisplayName()) }) return nodes } func (t *Tendrils) getLinks() []*Link { t.nodes.mu.RLock() defer t.nodes.mu.RUnlock() links := t.nodes.getDirectLinks() sort.Slice(links, func(i, j int) bool { if links[i].NodeA.DisplayName() != links[j].NodeA.DisplayName() { return sortorder.NaturalLess(links[i].NodeA.DisplayName(), links[j].NodeA.DisplayName()) } if links[i].InterfaceA != links[j].InterfaceA { return sortorder.NaturalLess(links[i].InterfaceA, links[j].InterfaceA) } if links[i].NodeB.DisplayName() != links[j].NodeB.DisplayName() { return sortorder.NaturalLess(links[i].NodeB.DisplayName(), links[j].NodeB.DisplayName()) } return sortorder.NaturalLess(links[i].InterfaceB, links[j].InterfaceB) }) return links } func (t *Tendrils) getMulticastGroups() []*MulticastGroupMembers { t.nodes.mu.Lock() t.nodes.expireMulticastMemberships() t.nodes.mu.Unlock() t.nodes.mu.RLock() defer t.nodes.mu.RUnlock() groups := make([]*MulticastGroupMembers, 0, len(t.nodes.multicastGroups)) for _, gm := range t.nodes.multicastGroups { groups = append(groups, gm) } sort.Slice(groups, func(i, j int) bool { return sortorder.NaturalLess(groups[i].Group.Name, groups[j].Group.Name) }) return groups } func (t *Tendrils) getArtNetNodes() []*ArtNetNode { t.artnet.Expire() t.artnet.mu.RLock() defer t.artnet.mu.RUnlock() nodes := make([]*ArtNetNode, 0, len(t.artnet.nodes)) for _, node := range t.artnet.nodes { nodes = append(nodes, node) } sort.Slice(nodes, func(i, j int) bool { return sortorder.NaturalLess(nodes[i].Node.DisplayName(), nodes[j].Node.DisplayName()) }) return nodes } func (t *Tendrils) getDanteFlows() []*DanteFlow { t.danteFlows.Expire() t.danteFlows.mu.RLock() defer t.danteFlows.mu.RUnlock() flows := make([]*DanteFlow, 0, len(t.danteFlows.flows)) for _, flow := range t.danteFlows.flows { flows = append(flows, flow) } sort.Slice(flows, func(i, j int) bool { return sortorder.NaturalLess(flows[i].Source.DisplayName(), flows[j].Source.DisplayName()) }) return flows }