2020-03-16 17:03:05 +00:00
|
|
|
// Copyright 2013 The Prometheus 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 event
|
|
|
|
|
|
|
|
import (
|
|
|
|
"sync"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/prometheus/client_golang/prometheus"
|
|
|
|
"github.com/prometheus/statsd_exporter/pkg/clock"
|
|
|
|
"github.com/prometheus/statsd_exporter/pkg/mapper"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Event interface {
|
|
|
|
MetricName() string
|
|
|
|
Value() float64
|
|
|
|
Labels() map[string]string
|
|
|
|
MetricType() mapper.MetricType
|
|
|
|
}
|
|
|
|
|
|
|
|
type CounterEvent struct {
|
|
|
|
CMetricName string
|
|
|
|
CValue float64
|
|
|
|
CLabels map[string]string
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *CounterEvent) MetricName() string { return c.CMetricName }
|
|
|
|
func (c *CounterEvent) Value() float64 { return c.CValue }
|
|
|
|
func (c *CounterEvent) Labels() map[string]string { return c.CLabels }
|
|
|
|
func (c *CounterEvent) MetricType() mapper.MetricType { return mapper.MetricTypeCounter }
|
|
|
|
|
|
|
|
type GaugeEvent struct {
|
|
|
|
GMetricName string
|
|
|
|
GValue float64
|
|
|
|
GRelative bool
|
|
|
|
GLabels map[string]string
|
|
|
|
}
|
|
|
|
|
|
|
|
func (g *GaugeEvent) MetricName() string { return g.GMetricName }
|
|
|
|
func (g *GaugeEvent) Value() float64 { return g.GValue }
|
|
|
|
func (c *GaugeEvent) Labels() map[string]string { return c.GLabels }
|
|
|
|
func (c *GaugeEvent) MetricType() mapper.MetricType { return mapper.MetricTypeGauge }
|
|
|
|
|
|
|
|
type TimerEvent struct {
|
|
|
|
TMetricName string
|
|
|
|
TValue float64
|
|
|
|
TLabels map[string]string
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *TimerEvent) MetricName() string { return t.TMetricName }
|
|
|
|
func (t *TimerEvent) Value() float64 { return t.TValue }
|
|
|
|
func (c *TimerEvent) Labels() map[string]string { return c.TLabels }
|
|
|
|
func (c *TimerEvent) MetricType() mapper.MetricType { return mapper.MetricTypeTimer }
|
|
|
|
|
|
|
|
type Events []Event
|
|
|
|
|
|
|
|
type EventQueue struct {
|
|
|
|
C chan Events
|
|
|
|
q Events
|
|
|
|
m sync.Mutex
|
|
|
|
flushTicker *time.Ticker
|
2020-04-08 18:59:19 +00:00
|
|
|
flushThreshold int
|
|
|
|
flushInterval time.Duration
|
|
|
|
eventsFlushed prometheus.Counter
|
2020-03-16 17:03:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type EventHandler interface {
|
2020-04-08 18:59:19 +00:00
|
|
|
Queue(event Events)
|
2020-03-16 17:03:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func NewEventQueue(c chan Events, flushThreshold int, flushInterval time.Duration, eventsFlushed prometheus.Counter) *EventQueue {
|
|
|
|
ticker := clock.NewTicker(flushInterval)
|
|
|
|
eq := &EventQueue{
|
|
|
|
C: c,
|
|
|
|
flushThreshold: flushThreshold,
|
2020-05-14 07:02:20 +00:00
|
|
|
flushInterval: flushInterval,
|
2020-03-16 17:03:05 +00:00
|
|
|
flushTicker: ticker,
|
|
|
|
q: make([]Event, 0, flushThreshold),
|
2020-04-08 18:59:19 +00:00
|
|
|
eventsFlushed: eventsFlushed,
|
2020-03-16 17:03:05 +00:00
|
|
|
}
|
|
|
|
go func() {
|
|
|
|
for {
|
|
|
|
<-ticker.C
|
2020-04-08 18:59:19 +00:00
|
|
|
eq.Flush()
|
2020-03-16 17:03:05 +00:00
|
|
|
}
|
|
|
|
}()
|
|
|
|
return eq
|
|
|
|
}
|
|
|
|
|
2020-04-08 18:59:19 +00:00
|
|
|
func (eq *EventQueue) Queue(events Events) {
|
2020-03-16 17:03:05 +00:00
|
|
|
eq.m.Lock()
|
|
|
|
defer eq.m.Unlock()
|
|
|
|
|
|
|
|
for _, e := range events {
|
|
|
|
eq.q = append(eq.q, e)
|
|
|
|
if len(eq.q) >= eq.flushThreshold {
|
2020-04-08 18:59:19 +00:00
|
|
|
eq.FlushUnlocked()
|
2020-03-16 17:03:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-08 18:59:19 +00:00
|
|
|
func (eq *EventQueue) Flush() {
|
2020-03-16 17:03:05 +00:00
|
|
|
eq.m.Lock()
|
|
|
|
defer eq.m.Unlock()
|
2020-04-08 18:59:19 +00:00
|
|
|
eq.FlushUnlocked()
|
2020-03-16 17:03:05 +00:00
|
|
|
}
|
|
|
|
|
2020-04-08 18:59:19 +00:00
|
|
|
func (eq *EventQueue) FlushUnlocked() {
|
2020-03-16 17:03:05 +00:00
|
|
|
eq.C <- eq.q
|
|
|
|
eq.q = make([]Event, 0, cap(eq.q))
|
2020-04-08 18:59:19 +00:00
|
|
|
eq.eventsFlushed.Inc()
|
2020-03-16 17:03:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (eq *EventQueue) Len() int {
|
|
|
|
eq.m.Lock()
|
|
|
|
defer eq.m.Unlock()
|
|
|
|
|
|
|
|
return len(eq.q)
|
|
|
|
}
|
|
|
|
|
|
|
|
type UnbufferedEventHandler struct {
|
|
|
|
C chan Events
|
|
|
|
}
|
|
|
|
|
2020-04-08 18:59:19 +00:00
|
|
|
func (ueh *UnbufferedEventHandler) Queue(events Events) {
|
2020-03-16 17:03:05 +00:00
|
|
|
ueh.C <- events
|
|
|
|
}
|