| 
									
										
										
										
											2016-09-25 08:39:20 +01:00
										 |  |  | // Package health implements an HTTP handler that responds to health checks.
 | 
					
						
							| 
									
										
										
										
											2016-04-06 09:21:46 +01:00
										 |  |  | package health
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import (
 | 
					
						
							|  |  |  | 	"io"
 | 
					
						
							| 
									
										
										
										
											2016-04-29 07:28:35 +01:00
										 |  |  | 	"net"
 | 
					
						
							| 
									
										
										
										
											2016-04-06 09:21:46 +01:00
										 |  |  | 	"net/http"
 | 
					
						
							|  |  |  | 	"sync"
 | 
					
						
							| 
									
										
										
										
											2018-01-18 10:40:09 +00:00
										 |  |  | 	"time"
 | 
					
						
							| 
									
										
										
										
											2018-04-19 07:41:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-22 21:40:33 +01:00
										 |  |  | 	clog "github.com/coredns/coredns/plugin/pkg/log"
 | 
					
						
							| 
									
										
										
										
											2016-04-06 09:21:46 +01:00
										 |  |  | )
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-22 21:40:33 +01:00
										 |  |  | var log = clog.NewWithPlugin("health")
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-18 10:40:09 +00:00
										 |  |  | // Health implements healthchecks by polling plugins.
 | 
					
						
							| 
									
										
										
										
											2016-09-23 09:14:12 +01:00
										 |  |  | type health struct {
 | 
					
						
							| 
									
										
										
										
											2018-01-18 10:40:09 +00:00
										 |  |  | 	Addr     string
 | 
					
						
							|  |  |  | 	lameduck time.Duration
 | 
					
						
							| 
									
										
										
										
											2016-08-19 17:14:17 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-21 17:43:02 +01:00
										 |  |  | 	ln      net.Listener
 | 
					
						
							|  |  |  | 	nlSetup bool
 | 
					
						
							|  |  |  | 	mux     *http.ServeMux
 | 
					
						
							| 
									
										
										
										
											2017-08-27 21:33:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-14 09:36:06 +01:00
										 |  |  | 	// A slice of Healthers that the health plugin will poll every second for their health status.
 | 
					
						
							| 
									
										
										
										
											2017-08-27 21:33:38 +01:00
										 |  |  | 	h []Healther
 | 
					
						
							|  |  |  | 	sync.RWMutex
 | 
					
						
							| 
									
										
										
										
											2017-09-14 09:36:06 +01:00
										 |  |  | 	ok bool // ok is the global boolean indicating an all healthy plugin stack
 | 
					
						
							| 
									
										
										
										
											2018-01-10 11:41:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-18 10:40:09 +00:00
										 |  |  | 	stop     chan bool
 | 
					
						
							|  |  |  | 	pollstop chan bool
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // newHealth returns a new initialized health.
 | 
					
						
							|  |  |  | func newHealth(addr string) *health {
 | 
					
						
							|  |  |  | 	return &health{Addr: addr, stop: make(chan bool), pollstop: make(chan bool)}
 | 
					
						
							| 
									
										
										
										
											2016-04-06 09:21:46 +01:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-10 11:41:22 +00:00
										 |  |  | func (h *health) OnStartup() error {
 | 
					
						
							| 
									
										
										
										
											2016-04-06 09:21:46 +01:00
										 |  |  | 	if h.Addr == "" {
 | 
					
						
							|  |  |  | 		h.Addr = defAddr
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							| 
									
										
										
										
											2016-04-29 07:28:35 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-02 21:40:14 -08:00
										 |  |  | 	ln, err := net.Listen("tcp", h.Addr)
 | 
					
						
							|  |  |  | 	if err != nil {
 | 
					
						
							|  |  |  | 		return err
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	h.ln = ln
 | 
					
						
							|  |  |  | 	h.mux = http.NewServeMux()
 | 
					
						
							| 
									
										
										
										
											2018-04-21 17:43:02 +01:00
										 |  |  | 	h.nlSetup = true
 | 
					
						
							| 
									
										
										
										
											2018-03-02 21:40:14 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	h.mux.HandleFunc(path, func(w http.ResponseWriter, r *http.Request) {
 | 
					
						
							|  |  |  | 		if h.Ok() {
 | 
					
						
							|  |  |  | 			w.WriteHeader(http.StatusOK)
 | 
					
						
							|  |  |  | 			io.WriteString(w, ok)
 | 
					
						
							| 
									
										
										
										
											2016-04-29 07:28:35 +01:00
										 |  |  | 			return
 | 
					
						
							|  |  |  | 		}
 | 
					
						
							| 
									
										
										
										
											2018-03-02 21:40:14 -08:00
										 |  |  | 		w.WriteHeader(http.StatusServiceUnavailable)
 | 
					
						
							| 
									
										
										
										
											2016-04-06 09:21:46 +01:00
										 |  |  | 	})
 | 
					
						
							| 
									
										
										
										
											2018-03-02 21:40:14 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	go func() { http.Serve(h.ln, h.mux) }()
 | 
					
						
							|  |  |  | 	go func() { h.overloaded() }()
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-06 09:21:46 +01:00
										 |  |  | 	return nil
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-21 17:43:02 +01:00
										 |  |  | func (h *health) OnRestart() error { return h.OnFinalShutdown() }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (h *health) OnFinalShutdown() error {
 | 
					
						
							|  |  |  | 	if !h.nlSetup {
 | 
					
						
							|  |  |  | 		return nil
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-18 10:40:09 +00:00
										 |  |  | 	// Stop polling plugins
 | 
					
						
							|  |  |  | 	h.pollstop <- true
 | 
					
						
							|  |  |  | 	// NACK health
 | 
					
						
							|  |  |  | 	h.SetOk(false)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if h.lameduck > 0 {
 | 
					
						
							| 
									
										
										
										
											2018-04-19 07:41:56 +01:00
										 |  |  | 		log.Infof("Going into lameduck mode for %s", h.lameduck)
 | 
					
						
							| 
									
										
										
										
											2018-01-18 10:40:09 +00:00
										 |  |  | 		time.Sleep(h.lameduck)
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-21 17:43:02 +01:00
										 |  |  | 	h.ln.Close()
 | 
					
						
							| 
									
										
										
										
											2018-01-10 11:41:22 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	h.stop <- true
 | 
					
						
							| 
									
										
										
										
											2018-04-21 17:43:02 +01:00
										 |  |  | 	h.nlSetup = false
 | 
					
						
							| 
									
										
										
										
											2016-04-29 07:28:35 +01:00
										 |  |  | 	return nil
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-06 09:21:46 +01:00
										 |  |  | const (
 | 
					
						
							|  |  |  | 	ok      = "OK"
 | 
					
						
							|  |  |  | 	defAddr = ":8080"
 | 
					
						
							| 
									
										
										
										
											2016-04-29 07:28:35 +01:00
										 |  |  | 	path    = "/health"
 | 
					
						
							| 
									
										
										
										
											2016-04-06 09:21:46 +01:00
										 |  |  | )
 |