Skip to content

Commit

Permalink
make metrics recording async so that it will not block ip requests
Browse files Browse the repository at this point in the history
Signed-off-by: Evan Baker <rbtr@users.noreply.github.com>
  • Loading branch information
rbtr committed Nov 5, 2024
1 parent aa2cd0d commit 1d6c4b4
Showing 1 changed file with 40 additions and 7 deletions.
47 changes: 40 additions & 7 deletions cns/restserver/metrics.go
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@ package restserver
import (
"maps"
"net/http"
"sync"
"time"

"github.com/Azure/azure-container-networking/cns"
Expand Down Expand Up @@ -124,7 +125,6 @@ func init() {
// Every http response is 200 so we really want cns response code.
// Hard tto do with middleware unless we derserialize the responses but making it an explit header works around it.
// if that doesn't work we could have a separate countervec just for response codes.

func NewHandlerFuncWithHistogram(handler http.HandlerFunc, histogram *prometheus.HistogramVec) http.HandlerFunc {
return func(w http.ResponseWriter, req *http.Request) {
start := time.Now()
Expand Down Expand Up @@ -157,14 +157,26 @@ type ipState struct {
releasingIPs int64
}

// publishIPStateMetrics logs and publishes the IP Config state metrics to Prometheus.
func (service *HTTPRestService) publishIPStateMetrics() {
// copy state
service.RLock()
defer service.RUnlock()
type asyncMetricsRecorder struct {
podIPConfigSrc func() map[string]cns.IPConfigurationStatus
sig chan struct{}
once sync.Once
}

// singleton recorder
var recorder asyncMetricsRecorder

// run starts the asyncMetricsRecorder and listens for signals to record the metrics.
func (a *asyncMetricsRecorder) run() {
for range a.sig {
a.record()
}
}

// record records the IP Config state metrics to Prometheus.
func (a *asyncMetricsRecorder) record() {
var state ipState
for ipConfig := range maps.Values(service.PodIPConfigState) {
for ipConfig := range maps.Values(a.podIPConfigSrc()) {
state.allocatedIPs++
if ipConfig.GetState() == types.Assigned {
state.assignedIPs++
Expand Down Expand Up @@ -195,3 +207,24 @@ func (service *HTTPRestService) publishIPStateMetrics() {
pendingProgrammingIPCount.WithLabelValues(labels...).Set(float64(state.programmingIPs))
pendingReleaseIPCount.WithLabelValues(labels...).Set(float64(state.releasingIPs))
}

// publishIPStateMetrics logs and publishes the IP Config state metrics to Prometheus.
func (service *HTTPRestService) publishIPStateMetrics() {
recorder.once.Do(func() {
recorder.podIPConfigSrc = service.PodIPConfigStates
recorder.sig = make(chan struct{})
go recorder.run()
})
select {
case recorder.sig <- struct{}{}: // signal the recorder to record the metrics
default: // drop the signal if the recorder already has an event queued
}
}

// PodIPConfigStates returns a clone of the IP Config State map.
func (service *HTTPRestService) PodIPConfigStates() map[string]cns.IPConfigurationStatus {
// copy state
service.RLock()
defer service.RUnlock()
return maps.Clone(service.PodIPConfigState)
}

0 comments on commit 1d6c4b4

Please sign in to comment.