From edd83381803119836ad678f0335c2692e4cfab32 Mon Sep 17 00:00:00 2001 From: Daniel Nelson Date: Tue, 19 May 2020 11:53:10 -0700 Subject: [PATCH] Close HTTP2 connections on timeout in influxdb outputs (#7517) --- internal/http.go | 11 +++++++++++ internal/http_go1.11.go | 15 --------------- internal/http_go1.12.go | 9 --------- plugins/outputs/influxdb/http.go | 8 +++++--- plugins/outputs/influxdb_v2/http.go | 7 ++++--- 5 files changed, 20 insertions(+), 30 deletions(-) delete mode 100644 internal/http_go1.11.go delete mode 100644 internal/http_go1.12.go diff --git a/internal/http.go b/internal/http.go index a44506719e4fc..04b8a9368906e 100644 --- a/internal/http.go +++ b/internal/http.go @@ -4,6 +4,7 @@ import ( "crypto/subtle" "net" "net/http" + "net/url" ) type BasicAuthErrorFunc func(rw http.ResponseWriter) @@ -95,3 +96,13 @@ func (h *ipRangeHandler) ServeHTTP(rw http.ResponseWriter, req *http.Request) { h.onError(rw, http.StatusForbidden) } + +func OnClientError(client *http.Client, err error) { + // Close connection after a timeout error. If this is a HTTP2 + // connection this ensures that next interval a new connection will be + // used and name lookup will be performed. + // https://github.com/golang/go/issues/36026 + if err, ok := err.(*url.Error); ok && err.Timeout() { + client.CloseIdleConnections() + } +} diff --git a/internal/http_go1.11.go b/internal/http_go1.11.go deleted file mode 100644 index d1a1ae31adb91..0000000000000 --- a/internal/http_go1.11.go +++ /dev/null @@ -1,15 +0,0 @@ -// +build !go1.12 - -package internal - -import "net/http" - -func CloseIdleConnections(c *http.Client) { - type closeIdler interface { - CloseIdleConnections() - } - - if tr, ok := c.Transport.(closeIdler); ok { - tr.CloseIdleConnections() - } -} diff --git a/internal/http_go1.12.go b/internal/http_go1.12.go deleted file mode 100644 index d5b1a847f12a7..0000000000000 --- a/internal/http_go1.12.go +++ /dev/null @@ -1,9 +0,0 @@ -// +build go1.12 - -package internal - -import "net/http" - -func CloseIdleConnections(c *http.Client) { - c.CloseIdleConnections() -} diff --git a/plugins/outputs/influxdb/http.go b/plugins/outputs/influxdb/http.go index 92498f0228d73..19ae6f31f45c6 100644 --- a/plugins/outputs/influxdb/http.go +++ b/plugins/outputs/influxdb/http.go @@ -209,6 +209,7 @@ func (c *httpClient) CreateDatabase(ctx context.Context, database string) error resp, err := c.client.Do(req.WithContext(ctx)) if err != nil { + internal.OnClientError(c.client, err) return err } defer resp.Body.Close() @@ -311,7 +312,7 @@ func (c *httpClient) Write(ctx context.Context, metrics []telegraf.Metric) error } func (c *httpClient) writeBatch(ctx context.Context, db, rp string, metrics []telegraf.Metric) error { - url, err := makeWriteURL(c.config.URL, db, rp, c.config.Consistency) + loc, err := makeWriteURL(c.config.URL, db, rp, c.config.Consistency) if err != nil { return err } @@ -322,13 +323,14 @@ func (c *httpClient) writeBatch(ctx context.Context, db, rp string, metrics []te } defer reader.Close() - req, err := c.makeWriteRequest(url, reader) + req, err := c.makeWriteRequest(loc, reader) if err != nil { return err } resp, err := c.client.Do(req.WithContext(ctx)) if err != nil { + internal.OnClientError(c.client, err) return err } defer resp.Body.Close() @@ -505,5 +507,5 @@ func makeQueryURL(loc *url.URL) (string, error) { } func (c *httpClient) Close() { - internal.CloseIdleConnections(c.client) + c.client.CloseIdleConnections() } diff --git a/plugins/outputs/influxdb_v2/http.go b/plugins/outputs/influxdb_v2/http.go index 3034207dd6c6a..2a32c5f4c60ea 100644 --- a/plugins/outputs/influxdb_v2/http.go +++ b/plugins/outputs/influxdb_v2/http.go @@ -210,7 +210,7 @@ func (c *httpClient) Write(ctx context.Context, metrics []telegraf.Metric) error } func (c *httpClient) writeBatch(ctx context.Context, bucket string, metrics []telegraf.Metric) error { - url, err := makeWriteURL(*c.url, c.Organization, bucket) + loc, err := makeWriteURL(*c.url, c.Organization, bucket) if err != nil { return err } @@ -221,13 +221,14 @@ func (c *httpClient) writeBatch(ctx context.Context, bucket string, metrics []te } defer reader.Close() - req, err := c.makeWriteRequest(url, reader) + req, err := c.makeWriteRequest(loc, reader) if err != nil { return err } resp, err := c.client.Do(req.WithContext(ctx)) if err != nil { + internal.OnClientError(c.client, err) return err } defer resp.Body.Close() @@ -347,5 +348,5 @@ func makeWriteURL(loc url.URL, org, bucket string) (string, error) { } func (c *httpClient) Close() { - internal.CloseIdleConnections(c.client) + c.client.CloseIdleConnections() }