From df59d4420f1e7b5c3ffbb0f94e667535ccc846c6 Mon Sep 17 00:00:00 2001 From: "Alexander A. Klimov" Date: Tue, 12 Mar 2024 16:31:39 +0100 Subject: [PATCH] Icinga2Client#Do(): call parent method, re-try on HTTP 503 300 times --- utils/icinga2_client.go | 34 ++++++++++++++++++++++++++++++++++ 1 file changed, 34 insertions(+) diff --git a/utils/icinga2_client.go b/utils/icinga2_client.go index eb3cbde..ebe2134 100644 --- a/utils/icinga2_client.go +++ b/utils/icinga2_client.go @@ -9,6 +9,7 @@ import ( "io" "net/http" "testing" + "time" ) type Icinga2Client struct { @@ -35,6 +36,39 @@ func NewIcinga2Client(address string, username string, password string) *Icinga2 } } +// getNoBody successfully returns http.NoBody. +func getNoBody() (io.ReadCloser, error) { + return http.NoBody, nil +} + +func (c *Icinga2Client) Do(req *http.Request) (*http.Response, error) { + if req.Body == nil && req.GetBody == nil { + req.Body = http.NoBody + req.GetBody = getNoBody + } + + for attempt := 1; ; attempt++ { + response, err := c.Client.Do(req) + if err == nil && response.StatusCode == http.StatusServiceUnavailable && req.GetBody != nil && attempt < 300 { + if body, err := req.GetBody(); err == nil { + _ = response.Body.Close() + req.Body = body + ctx := req.Context() + + select { + case <-time.After(time.Second): + case <-ctx.Done(): + return nil, ctx.Err() + } + + continue + } + } + + return response, err + } +} + func (c *Icinga2Client) addJsonHeaders(req *http.Request) { req.Header.Add("Accept", "application/json") if req.Body != nil {