| 
									
										
										
										
											2022-04-05 11:15:32 +00:00
										 |  |  | package web | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"net/http" | 
					
						
							|  |  |  | 	"time" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-06 10:46:02 +00:00
										 |  |  | 	"github.com/go-chi/render" | 
					
						
							| 
									
										
										
										
											2022-04-05 11:15:32 +00:00
										 |  |  | 	log "scm.yoorie.de/go-lib/gelf" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* | 
					
						
							|  |  |  | healthChecker Class | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | type healthChecker struct { | 
					
						
							|  |  |  | 	message     string | 
					
						
							|  |  |  | 	healthy     bool | 
					
						
							|  |  |  | 	lastChecked time.Time | 
					
						
							| 
									
										
										
										
											2022-04-06 10:46:02 +00:00
										 |  |  | 	startTime   time.Time | 
					
						
							| 
									
										
										
										
											2022-04-05 11:15:32 +00:00
										 |  |  | 	period      int | 
					
						
							|  |  |  | 	ticker      *time.Ticker | 
					
						
							|  |  |  | 	done        chan bool | 
					
						
							|  |  |  | 	//This is the healtchcheck you will have to provide.
 | 
					
						
							|  |  |  | 	checkFunc func() (bool, string) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-06 10:46:02 +00:00
										 |  |  | type HealthData struct { | 
					
						
							|  |  |  | 	Message     string    `json:"message,omitempty"` | 
					
						
							|  |  |  | 	LastChecked time.Time `json:"lastChecked"` | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-05 11:15:32 +00:00
										 |  |  | func newHealthChecker(checkFunction func() (bool, string)) *healthChecker { | 
					
						
							|  |  |  | 	hc := &healthChecker{} | 
					
						
							|  |  |  | 	hc.checkFunc = checkFunction | 
					
						
							| 
									
										
										
										
											2022-04-06 10:46:02 +00:00
										 |  |  | 	hc.startTime = time.Now() | 
					
						
							| 
									
										
										
										
											2022-04-05 11:15:32 +00:00
										 |  |  | 	return hc | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (h *healthChecker) start(checkPeriodInSeconds int) { | 
					
						
							|  |  |  | 	h.period = checkPeriodInSeconds | 
					
						
							|  |  |  | 	h.message = "service starting" | 
					
						
							|  |  |  | 	h.healthy = false | 
					
						
							|  |  |  | 	h.doCheck() | 
					
						
							|  |  |  | 	h.ticker = time.NewTicker(time.Second * time.Duration(h.period)) | 
					
						
							|  |  |  | 	h.done = make(chan bool) | 
					
						
							|  |  |  | 	go func() { | 
					
						
							|  |  |  | 		for { | 
					
						
							|  |  |  | 			select { | 
					
						
							|  |  |  | 			case <-h.done: | 
					
						
							|  |  |  | 				return | 
					
						
							|  |  |  | 			case <-h.ticker.C: | 
					
						
							|  |  |  | 				h.doCheck() | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (h *healthChecker) stop() { | 
					
						
							|  |  |  | 	h.ticker.Stop() | 
					
						
							|  |  |  | 	h.done <- true | 
					
						
							|  |  |  | 	log.Debug("Health checker stopped.") | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-07 07:17:18 +00:00
										 |  |  | // internal function to process the health check
 | 
					
						
							| 
									
										
										
										
											2022-04-05 11:15:32 +00:00
										 |  |  | func (h *healthChecker) doCheck() { | 
					
						
							|  |  |  | 	var msg string | 
					
						
							|  |  |  | 	h.healthy, msg = h.checkFunc() | 
					
						
							|  |  |  | 	if !h.healthy { | 
					
						
							|  |  |  | 		h.message = msg | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		h.message = "" | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	h.lastChecked = time.Now() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* | 
					
						
							|  |  |  | GetHealthyEndpoint is this service healthy | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2022-04-06 10:46:02 +00:00
										 |  |  | func (h *healthChecker) healthyEndpoint(responseWriter http.ResponseWriter, request *http.Request) { | 
					
						
							| 
									
										
										
										
											2022-04-05 11:15:32 +00:00
										 |  |  | 	t := time.Now() | 
					
						
							| 
									
										
										
										
											2022-04-06 10:46:02 +00:00
										 |  |  | 	status := http.StatusOK | 
					
						
							|  |  |  | 	healtData := &HealthData{ | 
					
						
							|  |  |  | 		Message:     "service up and running", | 
					
						
							|  |  |  | 		LastChecked: h.lastChecked, | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-04-05 11:15:32 +00:00
										 |  |  | 	if t.Sub(h.lastChecked) > (time.Second * time.Duration(2*h.period)) { | 
					
						
							|  |  |  | 		h.healthy = false | 
					
						
							|  |  |  | 		h.message = "Healthcheck not running" | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-04-06 10:46:02 +00:00
										 |  |  | 	if !h.healthy { | 
					
						
							|  |  |  | 		status = http.StatusServiceUnavailable | 
					
						
							|  |  |  | 		healtData.Message = fmt.Sprintf("service is unavailable: %s", h.message) | 
					
						
							| 
									
										
										
										
											2022-04-05 11:15:32 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-04-06 10:46:02 +00:00
										 |  |  | 	responseWriter.WriteHeader(status) | 
					
						
							|  |  |  | 	render.JSON(responseWriter, request, healtData) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-05 11:15:32 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* | 
					
						
							|  |  |  | GetReadinessEndpoint is this service ready for taking requests | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2022-04-06 10:46:02 +00:00
										 |  |  | func (h *healthChecker) readinessEndpoint(responseWriter http.ResponseWriter, request *http.Request) { | 
					
						
							|  |  |  | 	render.JSON(responseWriter, request, &HealthData{ | 
					
						
							|  |  |  | 		Message:     "service is ready", | 
					
						
							|  |  |  | 		LastChecked: h.startTime, | 
					
						
							|  |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2022-04-05 11:15:32 +00:00
										 |  |  | } |