| 
									
										
										
										
											2017-09-14 09:36:06 +01:00
										 |  |  | // Package metrics implement a handler and plugin that provides Prometheus metrics.
 | 
					
						
							| 
									
										
										
										
											2016-03-18 20:57:35 +00:00
										 |  |  | package metrics
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import (
 | 
					
						
							| 
									
										
										
										
											2016-04-06 09:21:46 +01:00
										 |  |  | 	"log"
 | 
					
						
							| 
									
										
										
										
											2016-04-29 07:28:35 +01:00
										 |  |  | 	"net"
 | 
					
						
							| 
									
										
										
										
											2016-03-18 20:57:35 +00:00
										 |  |  | 	"net/http"
 | 
					
						
							|  |  |  | 	"sync"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-14 09:36:06 +01:00
										 |  |  | 	"github.com/coredns/coredns/plugin"
 | 
					
						
							|  |  |  | 	"github.com/coredns/coredns/plugin/metrics/vars"
 | 
					
						
							| 
									
										
										
										
											2016-04-21 21:46:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-18 20:57:35 +00:00
										 |  |  | 	"github.com/prometheus/client_golang/prometheus"
 | 
					
						
							|  |  |  | )
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-21 19:34:40 +00:00
										 |  |  | func init() {
 | 
					
						
							|  |  |  | 	prometheus.MustRegister(vars.RequestCount)
 | 
					
						
							|  |  |  | 	prometheus.MustRegister(vars.RequestDuration)
 | 
					
						
							|  |  |  | 	prometheus.MustRegister(vars.RequestSize)
 | 
					
						
							|  |  |  | 	prometheus.MustRegister(vars.RequestDo)
 | 
					
						
							|  |  |  | 	prometheus.MustRegister(vars.RequestType)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	prometheus.MustRegister(vars.ResponseSize)
 | 
					
						
							|  |  |  | 	prometheus.MustRegister(vars.ResponseRcode)
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-18 20:57:35 +00:00
										 |  |  | // Metrics holds the prometheus configuration. The metrics' path is fixed to be /metrics
 | 
					
						
							|  |  |  | type Metrics struct {
 | 
					
						
							| 
									
										
										
										
											2017-09-14 09:36:06 +01:00
										 |  |  | 	Next plugin.Handler
 | 
					
						
							| 
									
										
										
										
											2016-10-26 10:01:52 +01:00
										 |  |  | 	Addr string
 | 
					
						
							|  |  |  | 	ln   net.Listener
 | 
					
						
							|  |  |  | 	mux  *http.ServeMux
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	zoneNames []string
 | 
					
						
							|  |  |  | 	zoneMap   map[string]bool
 | 
					
						
							|  |  |  | 	zoneMu    sync.RWMutex
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // AddZone adds zone z to m.
 | 
					
						
							|  |  |  | func (m *Metrics) AddZone(z string) {
 | 
					
						
							|  |  |  | 	m.zoneMu.Lock()
 | 
					
						
							|  |  |  | 	m.zoneMap[z] = true
 | 
					
						
							|  |  |  | 	m.zoneNames = keys(m.zoneMap)
 | 
					
						
							|  |  |  | 	m.zoneMu.Unlock()
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // RemoveZone remove zone z from m.
 | 
					
						
							|  |  |  | func (m *Metrics) RemoveZone(z string) {
 | 
					
						
							|  |  |  | 	m.zoneMu.Lock()
 | 
					
						
							|  |  |  | 	delete(m.zoneMap, z)
 | 
					
						
							|  |  |  | 	m.zoneNames = keys(m.zoneMap)
 | 
					
						
							|  |  |  | 	m.zoneMu.Unlock()
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // ZoneNames returns the zones of m.
 | 
					
						
							|  |  |  | func (m *Metrics) ZoneNames() []string {
 | 
					
						
							|  |  |  | 	m.zoneMu.RLock()
 | 
					
						
							|  |  |  | 	s := m.zoneNames
 | 
					
						
							|  |  |  | 	m.zoneMu.RUnlock()
 | 
					
						
							|  |  |  | 	return s
 | 
					
						
							| 
									
										
										
										
											2016-03-18 20:57:35 +00:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-23 09:14:12 +01:00
										 |  |  | // OnStartup sets up the metrics on startup.
 | 
					
						
							|  |  |  | func (m *Metrics) OnStartup() error {
 | 
					
						
							| 
									
										
										
										
											2017-02-21 19:34:40 +00:00
										 |  |  | 	ln, err := net.Listen("tcp", m.Addr)
 | 
					
						
							|  |  |  | 	if err != nil {
 | 
					
						
							|  |  |  | 		log.Printf("[ERROR] Failed to start metrics handler: %s", err)
 | 
					
						
							|  |  |  | 		return err
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							| 
									
										
										
										
											2016-06-25 18:12:13 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-21 19:34:40 +00:00
										 |  |  | 	m.ln = ln
 | 
					
						
							|  |  |  | 	ListenAddr = m.ln.Addr().String()
 | 
					
						
							| 
									
										
										
										
											2016-03-18 20:57:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-21 19:34:40 +00:00
										 |  |  | 	m.mux = http.NewServeMux()
 | 
					
						
							|  |  |  | 	m.mux.Handle("/metrics", prometheus.Handler())
 | 
					
						
							| 
									
										
										
										
											2016-04-29 07:28:35 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-21 19:34:40 +00:00
										 |  |  | 	go func() {
 | 
					
						
							|  |  |  | 		http.Serve(m.ln, m.mux)
 | 
					
						
							|  |  |  | 	}()
 | 
					
						
							| 
									
										
										
										
											2016-03-18 20:57:35 +00:00
										 |  |  | 	return nil
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-21 19:34:40 +00:00
										 |  |  | // OnShutdown tears down the metrics on shutdown and restart.
 | 
					
						
							| 
									
										
										
										
											2016-09-23 09:14:12 +01:00
										 |  |  | func (m *Metrics) OnShutdown() error {
 | 
					
						
							| 
									
										
										
										
											2016-04-29 07:28:35 +01:00
										 |  |  | 	if m.ln != nil {
 | 
					
						
							|  |  |  | 		return m.ln.Close()
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 	return nil
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-26 10:01:52 +01:00
										 |  |  | func keys(m map[string]bool) []string {
 | 
					
						
							|  |  |  | 	sx := []string{}
 | 
					
						
							|  |  |  | 	for k := range m {
 | 
					
						
							|  |  |  | 		sx = append(sx, k)
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 	return sx
 | 
					
						
							| 
									
										
										
										
											2016-03-18 20:57:35 +00:00
										 |  |  | }
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:57:10 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | // ListenAddr is assigned the address of the prometheus listener. Its use is mainly in tests where
 | 
					
						
							|  |  |  | // we listen on "localhost:0" and need to retrieve the actual address.
 | 
					
						
							|  |  |  | var ListenAddr string
 |