mirror of https://github.com/helm/helm
Signed-off-by: Pavani Pogula <pogulapavani@gmail.com>pull/31107/head
parent
3c5d68d62e
commit
d4ed9210df
@ -0,0 +1,161 @@
|
||||
/*
|
||||
Copyright The Helm Authors.
|
||||
|
||||
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.
|
||||
*/
|
||||
|
||||
package kube
|
||||
|
||||
import (
|
||||
"encoding/json"
|
||||
"errors"
|
||||
"io"
|
||||
"net/http"
|
||||
"strings"
|
||||
"testing"
|
||||
|
||||
"github.com/stretchr/testify/assert"
|
||||
)
|
||||
|
||||
type fakeRoundTripper struct {
|
||||
resp *http.Response
|
||||
err error
|
||||
calls int
|
||||
}
|
||||
|
||||
func (f *fakeRoundTripper) RoundTrip(_ *http.Request) (*http.Response, error) {
|
||||
f.calls++
|
||||
return f.resp, f.err
|
||||
}
|
||||
|
||||
func newRespWithBody(statusCode int, contentType, body string) *http.Response {
|
||||
return &http.Response{
|
||||
StatusCode: statusCode,
|
||||
Header: http.Header{"Content-Type": []string{contentType}},
|
||||
Body: io.NopCloser(strings.NewReader(body)),
|
||||
}
|
||||
}
|
||||
|
||||
func TestRetryingRoundTripper_RoundTrip(t *testing.T) {
|
||||
marshalErr := func(code int, msg string) string {
|
||||
b, _ := json.Marshal(kubernetesError{
|
||||
Code: code,
|
||||
Message: msg,
|
||||
})
|
||||
return string(b)
|
||||
}
|
||||
|
||||
tests := []struct {
|
||||
name string
|
||||
resp *http.Response
|
||||
err error
|
||||
expectedCalls int
|
||||
expectedErr string
|
||||
expectedCode int
|
||||
}{
|
||||
{
|
||||
name: "no retry, status < 500 returns response",
|
||||
resp: newRespWithBody(200, "application/json", `{"message":"ok","code":200}`),
|
||||
err: nil,
|
||||
expectedCalls: 1,
|
||||
expectedCode: 200,
|
||||
},
|
||||
{
|
||||
name: "error from wrapped RoundTripper propagates",
|
||||
resp: nil,
|
||||
err: errors.New("wrapped error"),
|
||||
expectedCalls: 1,
|
||||
expectedErr: "wrapped error",
|
||||
},
|
||||
{
|
||||
name: "no retry, content-type not application/json",
|
||||
resp: newRespWithBody(500, "text/plain", "server error"),
|
||||
err: nil,
|
||||
expectedCalls: 1,
|
||||
expectedCode: 500,
|
||||
},
|
||||
{
|
||||
name: "error reading body returns error",
|
||||
resp: &http.Response{
|
||||
StatusCode: http.StatusInternalServerError,
|
||||
Header: http.Header{"Content-Type": []string{"application/json"}},
|
||||
Body: &errReader{},
|
||||
},
|
||||
err: nil,
|
||||
expectedCalls: 1,
|
||||
expectedErr: "read error",
|
||||
},
|
||||
{
|
||||
name: "error decoding JSON returns error",
|
||||
resp: newRespWithBody(500, "application/json", `invalid-json`),
|
||||
err: nil,
|
||||
expectedCalls: 1,
|
||||
expectedErr: "invalid character",
|
||||
},
|
||||
{
|
||||
name: "retry on etcdserver leader changed message",
|
||||
resp: newRespWithBody(500, "application/json", marshalErr(500, "some error etcdserver: leader changed")),
|
||||
err: nil,
|
||||
expectedCalls: 2,
|
||||
expectedCode: 500,
|
||||
},
|
||||
{
|
||||
name: "retry on raft proposal dropped message",
|
||||
resp: newRespWithBody(500, "application/json", marshalErr(500, "rpc error: code = Unknown desc = raft proposal dropped")),
|
||||
err: nil,
|
||||
expectedCalls: 2,
|
||||
expectedCode: 500,
|
||||
},
|
||||
{
|
||||
name: "no retry on other error message",
|
||||
resp: newRespWithBody(500, "application/json", marshalErr(500, "other server error")),
|
||||
err: nil,
|
||||
expectedCalls: 1,
|
||||
expectedCode: 500,
|
||||
},
|
||||
}
|
||||
|
||||
for _, tt := range tests {
|
||||
t.Run(tt.name, func(t *testing.T) {
|
||||
fakeRT := &fakeRoundTripper{
|
||||
resp: tt.resp,
|
||||
err: tt.err,
|
||||
}
|
||||
rt := RetryingRoundTripper{
|
||||
Wrapped: fakeRT,
|
||||
}
|
||||
req, _ := http.NewRequest(http.MethodGet, "http://example.com", nil)
|
||||
resp, err := rt.RoundTrip(req)
|
||||
|
||||
if tt.expectedErr != "" {
|
||||
assert.Error(t, err)
|
||||
assert.Contains(t, err.Error(), tt.expectedErr)
|
||||
return
|
||||
}
|
||||
assert.NoError(t, err)
|
||||
|
||||
assert.Equal(t, tt.expectedCode, resp.StatusCode)
|
||||
assert.Equal(t, tt.expectedCalls, fakeRT.calls)
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
type errReader struct{}
|
||||
|
||||
func (e *errReader) Read(_ []byte) (int, error) {
|
||||
return 0, errors.New("read error")
|
||||
}
|
||||
|
||||
func (e *errReader) Close() error {
|
||||
return nil
|
||||
}
|
Loading…
Reference in new issue