Skip to content

Commit

Permalink
fix: retry on download errors
Browse files Browse the repository at this point in the history
to avoid failing because of issues like connection reset
  • Loading branch information
carlossg committed Jun 30, 2024
1 parent 347cd5e commit 7b9163b
Show file tree
Hide file tree
Showing 2 changed files with 67 additions and 13 deletions.
19 changes: 19 additions & 0 deletions pkg/registry/http.go
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ import (
"fmt"
"io"
"log"
"net"
"net/http"
"os"
"time"
Expand Down Expand Up @@ -74,6 +75,24 @@ func (r SchemaRegistry) DownloadSchema(resourceKind, resourceAPIVersion, k8sVers
}

resp, err := r.c.Get(url)
// retry on transient errors, ie. connection reset by peer
if err != nil {
if opErr, ok := err.(*net.OpError); ok {
if r.debug {
log.Printf("failed downloading schema at %s due to network error, retrying: %s", url, opErr)
}
time.Sleep(1 * time.Second)
resp, err = r.c.Get(url)
}
}
// retry on server errors
if resp != nil && resp.StatusCode >= 500 {
if r.debug {
log.Printf("failed downloading schema at %s due to server error, retrying: %d", url, resp.StatusCode)
}
time.Sleep(1 * time.Second)
resp, err = r.c.Get(url)
}
if err != nil {
msg := fmt.Sprintf("failed downloading schema at %s: %s", url, err)
if r.debug {
Expand Down
61 changes: 48 additions & 13 deletions pkg/registry/http_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -2,24 +2,29 @@ package registry

import (
"bytes"
"errors"
"fmt"
"io"
"net"
"net/http"
"strings"
"testing"
)

type mockHTTPGetter struct {
httpGet func(string) (*http.Response, error)
callNumber int
httpGet func(mockHTTPGetter, string) (*http.Response, error)
}

func newMockHTTPGetter(f func(string) (*http.Response, error)) *mockHTTPGetter {
func newMockHTTPGetter(f func(mockHTTPGetter, string) (*http.Response, error)) *mockHTTPGetter {
return &mockHTTPGetter{
httpGet: f,
callNumber: 0,
httpGet: f,
}
}
func (m mockHTTPGetter) Get(url string) (resp *http.Response, err error) {
return m.httpGet(url)
func (m *mockHTTPGetter) Get(url string) (resp *http.Response, err error) {
m.callNumber = m.callNumber + 1
return m.httpGet(*m, url)
}

func TestDownloadSchema(t *testing.T) {
Expand All @@ -33,21 +38,28 @@ func TestDownloadSchema(t *testing.T) {
expectErr error
}{
{
"error when downloading",
newMockHTTPGetter(func(url string) (resp *http.Response, err error) {
return nil, fmt.Errorf("failed downloading from registry")
"retry connection reset by peer",
newMockHTTPGetter(func(c mockHTTPGetter, url string) (resp *http.Response, err error) {
if c.callNumber == 1 {
return nil, &net.OpError{Err: errors.New("connection reset by peer")}
} else {
return &http.Response{
StatusCode: http.StatusOK,
Body: io.NopCloser(strings.NewReader("http response mock body")),
}, nil
}
}),
"http://kubernetesjson.dev",
true,
"Deployment",
"v1",
"1.18.0",
[]byte("http response mock body"),
nil,
fmt.Errorf("failed downloading schema at http://kubernetesjson.dev: failed downloading from registry"),
},
{
"getting 404",
newMockHTTPGetter(func(url string) (resp *http.Response, err error) {
newMockHTTPGetter(func(c mockHTTPGetter, url string) (resp *http.Response, err error) {
return &http.Response{
StatusCode: http.StatusNotFound,
Body: io.NopCloser(strings.NewReader("http response mock body")),
Expand All @@ -62,8 +74,8 @@ func TestDownloadSchema(t *testing.T) {
fmt.Errorf("could not find schema at http://kubernetesjson.dev"),
},
{
"getting 503",
newMockHTTPGetter(func(url string) (resp *http.Response, err error) {
"getting 500",
newMockHTTPGetter(func(c mockHTTPGetter, url string) (resp *http.Response, err error) {
return &http.Response{
StatusCode: http.StatusServiceUnavailable,
Body: io.NopCloser(strings.NewReader("http response mock body")),
Expand All @@ -77,9 +89,32 @@ func TestDownloadSchema(t *testing.T) {
nil,
fmt.Errorf("error while downloading schema at http://kubernetesjson.dev - received HTTP status 503"),
},
{
"retry 503",
newMockHTTPGetter(func(c mockHTTPGetter, url string) (resp *http.Response, err error) {
if c.callNumber == 1 {
return &http.Response{
StatusCode: http.StatusServiceUnavailable,
Body: io.NopCloser(strings.NewReader("503 http response mock body")),
}, nil
} else {
return &http.Response{
StatusCode: http.StatusOK,
Body: io.NopCloser(strings.NewReader("http response mock body")),
}, nil
}
}),
"http://kubernetesjson.dev",
true,
"Deployment",
"v1",
"1.18.0",
[]byte("http response mock body"),
nil,
},
{
"200",
newMockHTTPGetter(func(url string) (resp *http.Response, err error) {
newMockHTTPGetter(func(c mockHTTPGetter, url string) (resp *http.Response, err error) {
return &http.Response{
StatusCode: http.StatusOK,
Body: io.NopCloser(strings.NewReader("http response mock body")),
Expand Down

0 comments on commit 7b9163b

Please sign in to comment.