blob: c9d32bf9493135ea1870605164c120ddb10917ca [file] [log] [blame]
// Copyright 2016 Google LLC
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//go:build go1.13
// +build go1.13
package metadata
import (
"io"
"io/ioutil"
"net/http"
"strings"
"testing"
)
func TestRetry(t *testing.T) {
tests := []struct {
name string
timesToFail int
failCode int
failErr error
response string
expectError bool
}{
{
name: "no retries",
response: "test",
},
{
name: "retry 500 once",
response: "test",
failCode: 500,
timesToFail: 1,
},
{
name: "retry io.ErrUnexpectedEOF once",
response: "test",
failErr: io.ErrUnexpectedEOF,
timesToFail: 1,
},
{
name: "retry io.ErrUnexpectedEOF permanent",
failErr: io.ErrUnexpectedEOF,
timesToFail: maxRetryAttempts + 1,
expectError: true,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
ft := &failingTransport{
timesToFail: tt.timesToFail,
failCode: tt.failCode,
failErr: tt.failErr,
response: tt.response,
}
c := NewClient(&http.Client{Transport: ft})
s, err := c.Get("")
if tt.expectError && err == nil {
t.Fatalf("did not receive expected error")
} else if !tt.expectError && err != nil {
t.Fatalf("unexpected error: %v", err)
}
expectedCount := ft.failedAttempts + 1
if tt.expectError {
expectedCount = ft.failedAttempts
} else if s != tt.response {
// Responses are only meaningful if err == nil
t.Fatalf("c.Get() = %q, want %q", s, tt.response)
}
if ft.called != expectedCount {
t.Fatalf("failed %d times, want %d", ft.called, expectedCount)
}
})
}
}
type failingTransport struct {
timesToFail int
failCode int
failErr error
response string
failedAttempts int
called int
}
func (r *failingTransport) RoundTrip(req *http.Request) (*http.Response, error) {
r.called++
if r.failedAttempts < r.timesToFail {
r.failedAttempts++
if r.failErr != nil {
return nil, r.failErr
}
return &http.Response{StatusCode: r.failCode}, nil
}
return &http.Response{StatusCode: http.StatusOK, Body: ioutil.NopCloser(strings.NewReader(r.response))}, nil
}