metrics.go 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186
  1. package controllers
  2. import (
  3. "encoding/json"
  4. "net/http"
  5. proLogic "github.com/gravitl/netmaker/pro/logic"
  6. "golang.org/x/exp/slog"
  7. "github.com/gorilla/mux"
  8. "github.com/gravitl/netmaker/database"
  9. "github.com/gravitl/netmaker/logger"
  10. "github.com/gravitl/netmaker/logic"
  11. "github.com/gravitl/netmaker/models"
  12. )
  13. // MetricHandlers - How we handle Pro Metrics
  14. func MetricHandlers(r *mux.Router) {
  15. r.HandleFunc("/api/metrics/{network}/{nodeid}", logic.SecurityCheck(true, http.HandlerFunc(getNodeMetrics))).Methods(http.MethodGet)
  16. r.HandleFunc("/api/metrics/{network}", logic.SecurityCheck(true, http.HandlerFunc(getNetworkNodesMetrics))).Methods(http.MethodGet)
  17. r.HandleFunc("/api/metrics", logic.SecurityCheck(true, http.HandlerFunc(getAllMetrics))).Methods(http.MethodGet)
  18. r.HandleFunc("/api/metrics-ext/{network}", logic.SecurityCheck(true, http.HandlerFunc(getNetworkExtMetrics))).Methods(http.MethodGet)
  19. r.HandleFunc("/api/v1/graph/{network}", logic.SecurityCheck(true, http.HandlerFunc(graph))).Methods(http.MethodGet)
  20. }
  21. // get the metrics of a given node
  22. func getNodeMetrics(w http.ResponseWriter, r *http.Request) {
  23. // set header.
  24. w.Header().Set("Content-Type", "application/json")
  25. var params = mux.Vars(r)
  26. nodeID := params["nodeid"]
  27. logger.Log(1, r.Header.Get("user"), "requested fetching metrics for node", nodeID, "on network", params["network"])
  28. metrics, err := logic.GetMetrics(nodeID)
  29. if err != nil {
  30. logger.Log(1, r.Header.Get("user"), "failed to fetch metrics of node", nodeID, err.Error())
  31. logic.ReturnErrorResponse(w, r, logic.FormatError(err, "internal"))
  32. return
  33. }
  34. logger.Log(1, r.Header.Get("user"), "fetched metrics for node", params["nodeid"])
  35. w.WriteHeader(http.StatusOK)
  36. json.NewEncoder(w).Encode(metrics)
  37. }
  38. // get the metrics of all nodes in given network
  39. func getNetworkNodesMetrics(w http.ResponseWriter, r *http.Request) {
  40. // set header.
  41. w.Header().Set("Content-Type", "application/json")
  42. var params = mux.Vars(r)
  43. network := params["network"]
  44. logger.Log(1, r.Header.Get("user"), "requested fetching network node metrics on network", network)
  45. networkNodes, err := logic.GetNetworkNodes(network)
  46. if err != nil {
  47. logger.Log(1, r.Header.Get("user"), "failed to fetch metrics of all nodes in network", network, err.Error())
  48. logic.ReturnErrorResponse(w, r, logic.FormatError(err, "internal"))
  49. return
  50. }
  51. networkMetrics := models.NetworkMetrics{}
  52. networkMetrics.Nodes = make(models.MetricsMap)
  53. for i := range networkNodes {
  54. id := networkNodes[i].ID
  55. metrics, err := proLogic.GetMetrics(id.String())
  56. if err != nil {
  57. logger.Log(1, r.Header.Get("user"), "failed to append metrics of node", id.String(), "during network metrics fetch", err.Error())
  58. continue
  59. }
  60. networkMetrics.Nodes[id.String()] = *metrics
  61. }
  62. logger.Log(1, r.Header.Get("user"), "fetched metrics for network", network)
  63. w.WriteHeader(http.StatusOK)
  64. json.NewEncoder(w).Encode(networkMetrics)
  65. }
  66. // get the metrics for ext clients on a given network
  67. func getNetworkExtMetrics(w http.ResponseWriter, r *http.Request) {
  68. // set header.
  69. w.Header().Set("Content-Type", "application/json")
  70. var params = mux.Vars(r)
  71. network := params["network"]
  72. logger.Log(1, r.Header.Get("user"), "requested fetching external client metrics on network", network)
  73. ingresses, err := proLogic.GetNetworkIngresses(network) // grab all the ingress gateways
  74. if err != nil {
  75. logger.Log(1, r.Header.Get("user"), "failed to fetch metrics of ext clients in network", network, err.Error())
  76. logic.ReturnErrorResponse(w, r, logic.FormatError(err, "internal"))
  77. return
  78. }
  79. clients, err := logic.GetNetworkExtClients(network) // grab all the network ext clients
  80. if err != nil {
  81. if database.IsEmptyRecord(err) {
  82. var metrics struct{}
  83. w.WriteHeader(http.StatusOK)
  84. json.NewEncoder(w).Encode(metrics)
  85. return
  86. }
  87. logger.Log(1, r.Header.Get("user"), "failed to fetch metrics of ext clients in network", network, err.Error())
  88. logic.ReturnErrorResponse(w, r, logic.FormatError(err, "internal"))
  89. return
  90. }
  91. networkMetrics := models.Metrics{}
  92. networkMetrics.Connectivity = make(map[string]models.Metric)
  93. for i := range ingresses {
  94. id := ingresses[i].ID
  95. ingressMetrics, err := proLogic.GetMetrics(id.String())
  96. if err != nil {
  97. logger.Log(1, r.Header.Get("user"), "failed to append external client metrics from ingress node", id.String(), err.Error())
  98. continue
  99. }
  100. if ingressMetrics.Connectivity == nil {
  101. continue
  102. }
  103. for j := range clients {
  104. if clients[j].Network != network {
  105. continue
  106. }
  107. // if metrics for that client have been reported, append them
  108. if _, ok := ingressMetrics.Connectivity[clients[j].ClientID]; ok {
  109. networkMetrics.Connectivity[clients[j].ClientID] = ingressMetrics.Connectivity[clients[j].ClientID]
  110. }
  111. }
  112. }
  113. slog.Debug("sending collected client metrics", "metrics", networkMetrics.Connectivity)
  114. logger.Log(1, r.Header.Get("user"), "fetched ext client metrics for network", network)
  115. w.WriteHeader(http.StatusOK)
  116. json.NewEncoder(w).Encode(networkMetrics.Connectivity)
  117. }
  118. // get Metrics of all nodes on server, lots of data
  119. func getAllMetrics(w http.ResponseWriter, r *http.Request) {
  120. w.Header().Set("Content-Type", "application/json")
  121. logger.Log(1, r.Header.Get("user"), "requested fetching all metrics")
  122. allNodes, err := logic.GetAllNodes()
  123. if err != nil {
  124. logger.Log(1, r.Header.Get("user"), "failed to fetch metrics of all nodes on server", err.Error())
  125. logic.ReturnErrorResponse(w, r, logic.FormatError(err, "internal"))
  126. return
  127. }
  128. networkMetrics := models.NetworkMetrics{}
  129. networkMetrics.Nodes = make(models.MetricsMap)
  130. for i := range allNodes {
  131. id := allNodes[i].ID
  132. metrics, err := proLogic.GetMetrics(id.String())
  133. if err != nil {
  134. logger.Log(1, r.Header.Get("user"), "failed to append metrics of node", id.String(), "during all nodes metrics fetch", err.Error())
  135. continue
  136. }
  137. networkMetrics.Nodes[id.String()] = *metrics
  138. }
  139. logger.Log(1, r.Header.Get("user"), "fetched metrics for all nodes on server")
  140. w.WriteHeader(http.StatusOK)
  141. json.NewEncoder(w).Encode(networkMetrics)
  142. }
  143. func graph(w http.ResponseWriter, r *http.Request) {
  144. w.Header().Set("Content-Type", "application/json")
  145. var params = mux.Vars(r)
  146. network := params["network"]
  147. networkNodes, err := logic.GetNetworkNodes(network)
  148. if err != nil {
  149. logger.Log(1, r.Header.Get("user"), "failed to get network nodes", err.Error())
  150. return
  151. }
  152. networkNodes = logic.AddStaticNodestoList(networkNodes)
  153. // return all the nodes in JSON/API format
  154. apiNodes := logic.GetAllNodesAPIWithLocation(networkNodes[:])
  155. logic.SortApiNodes(apiNodes[:])
  156. w.WriteHeader(http.StatusOK)
  157. json.NewEncoder(w).Encode(apiNodes)
  158. }