mirror of
https://github.com/kubernetes/sample-controller.git
synced 2025-01-21 17:32:50 +08:00
196 lines
4.6 KiB
Go
196 lines
4.6 KiB
Go
|
/*
|
||
|
Copyright 2016 The Kubernetes Authors.
|
||
|
|
||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||
|
you may not use this file except in compliance with the License.
|
||
|
You may obtain a copy of the License at
|
||
|
|
||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||
|
|
||
|
Unless required by applicable law or agreed to in writing, software
|
||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
|
See the License for the specific language governing permissions and
|
||
|
limitations under the License.
|
||
|
*/
|
||
|
|
||
|
package workqueue
|
||
|
|
||
|
import (
|
||
|
"sync"
|
||
|
"time"
|
||
|
)
|
||
|
|
||
|
// This file provides abstractions for setting the provider (e.g., prometheus)
|
||
|
// of metrics.
|
||
|
|
||
|
type queueMetrics interface {
|
||
|
add(item t)
|
||
|
get(item t)
|
||
|
done(item t)
|
||
|
}
|
||
|
|
||
|
// GaugeMetric represents a single numerical value that can arbitrarily go up
|
||
|
// and down.
|
||
|
type GaugeMetric interface {
|
||
|
Inc()
|
||
|
Dec()
|
||
|
}
|
||
|
|
||
|
// CounterMetric represents a single numerical value that only ever
|
||
|
// goes up.
|
||
|
type CounterMetric interface {
|
||
|
Inc()
|
||
|
}
|
||
|
|
||
|
// SummaryMetric captures individual observations.
|
||
|
type SummaryMetric interface {
|
||
|
Observe(float64)
|
||
|
}
|
||
|
|
||
|
type noopMetric struct{}
|
||
|
|
||
|
func (noopMetric) Inc() {}
|
||
|
func (noopMetric) Dec() {}
|
||
|
func (noopMetric) Observe(float64) {}
|
||
|
|
||
|
type defaultQueueMetrics struct {
|
||
|
// current depth of a workqueue
|
||
|
depth GaugeMetric
|
||
|
// total number of adds handled by a workqueue
|
||
|
adds CounterMetric
|
||
|
// how long an item stays in a workqueue
|
||
|
latency SummaryMetric
|
||
|
// how long processing an item from a workqueue takes
|
||
|
workDuration SummaryMetric
|
||
|
addTimes map[t]time.Time
|
||
|
processingStartTimes map[t]time.Time
|
||
|
}
|
||
|
|
||
|
func (m *defaultQueueMetrics) add(item t) {
|
||
|
if m == nil {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
m.adds.Inc()
|
||
|
m.depth.Inc()
|
||
|
if _, exists := m.addTimes[item]; !exists {
|
||
|
m.addTimes[item] = time.Now()
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (m *defaultQueueMetrics) get(item t) {
|
||
|
if m == nil {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
m.depth.Dec()
|
||
|
m.processingStartTimes[item] = time.Now()
|
||
|
if startTime, exists := m.addTimes[item]; exists {
|
||
|
m.latency.Observe(sinceInMicroseconds(startTime))
|
||
|
delete(m.addTimes, item)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (m *defaultQueueMetrics) done(item t) {
|
||
|
if m == nil {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
if startTime, exists := m.processingStartTimes[item]; exists {
|
||
|
m.workDuration.Observe(sinceInMicroseconds(startTime))
|
||
|
delete(m.processingStartTimes, item)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Gets the time since the specified start in microseconds.
|
||
|
func sinceInMicroseconds(start time.Time) float64 {
|
||
|
return float64(time.Since(start).Nanoseconds() / time.Microsecond.Nanoseconds())
|
||
|
}
|
||
|
|
||
|
type retryMetrics interface {
|
||
|
retry()
|
||
|
}
|
||
|
|
||
|
type defaultRetryMetrics struct {
|
||
|
retries CounterMetric
|
||
|
}
|
||
|
|
||
|
func (m *defaultRetryMetrics) retry() {
|
||
|
if m == nil {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
m.retries.Inc()
|
||
|
}
|
||
|
|
||
|
// MetricsProvider generates various metrics used by the queue.
|
||
|
type MetricsProvider interface {
|
||
|
NewDepthMetric(name string) GaugeMetric
|
||
|
NewAddsMetric(name string) CounterMetric
|
||
|
NewLatencyMetric(name string) SummaryMetric
|
||
|
NewWorkDurationMetric(name string) SummaryMetric
|
||
|
NewRetriesMetric(name string) CounterMetric
|
||
|
}
|
||
|
|
||
|
type noopMetricsProvider struct{}
|
||
|
|
||
|
func (_ noopMetricsProvider) NewDepthMetric(name string) GaugeMetric {
|
||
|
return noopMetric{}
|
||
|
}
|
||
|
|
||
|
func (_ noopMetricsProvider) NewAddsMetric(name string) CounterMetric {
|
||
|
return noopMetric{}
|
||
|
}
|
||
|
|
||
|
func (_ noopMetricsProvider) NewLatencyMetric(name string) SummaryMetric {
|
||
|
return noopMetric{}
|
||
|
}
|
||
|
|
||
|
func (_ noopMetricsProvider) NewWorkDurationMetric(name string) SummaryMetric {
|
||
|
return noopMetric{}
|
||
|
}
|
||
|
|
||
|
func (_ noopMetricsProvider) NewRetriesMetric(name string) CounterMetric {
|
||
|
return noopMetric{}
|
||
|
}
|
||
|
|
||
|
var metricsFactory = struct {
|
||
|
metricsProvider MetricsProvider
|
||
|
setProviders sync.Once
|
||
|
}{
|
||
|
metricsProvider: noopMetricsProvider{},
|
||
|
}
|
||
|
|
||
|
func newQueueMetrics(name string) queueMetrics {
|
||
|
var ret *defaultQueueMetrics
|
||
|
if len(name) == 0 {
|
||
|
return ret
|
||
|
}
|
||
|
return &defaultQueueMetrics{
|
||
|
depth: metricsFactory.metricsProvider.NewDepthMetric(name),
|
||
|
adds: metricsFactory.metricsProvider.NewAddsMetric(name),
|
||
|
latency: metricsFactory.metricsProvider.NewLatencyMetric(name),
|
||
|
workDuration: metricsFactory.metricsProvider.NewWorkDurationMetric(name),
|
||
|
addTimes: map[t]time.Time{},
|
||
|
processingStartTimes: map[t]time.Time{},
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func newRetryMetrics(name string) retryMetrics {
|
||
|
var ret *defaultRetryMetrics
|
||
|
if len(name) == 0 {
|
||
|
return ret
|
||
|
}
|
||
|
return &defaultRetryMetrics{
|
||
|
retries: metricsFactory.metricsProvider.NewRetriesMetric(name),
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// SetProvider sets the metrics provider of the metricsFactory.
|
||
|
func SetProvider(metricsProvider MetricsProvider) {
|
||
|
metricsFactory.setProviders.Do(func() {
|
||
|
metricsFactory.metricsProvider = metricsProvider
|
||
|
})
|
||
|
}
|