traefik/middlewares/retry/retry_test.go

264 lines
8.1 KiB
Go
Raw Normal View History

2018-11-14 10:18:03 +01:00
package retry
2017-04-18 08:22:06 +02:00
import (
2018-11-14 10:18:03 +01:00
"context"
2017-04-18 08:22:06 +02:00
"net/http"
"net/http/httptest"
2018-08-29 11:58:03 +02:00
"strings"
2017-04-18 08:22:06 +02:00
"testing"
2018-11-14 10:18:03 +01:00
"github.com/containous/traefik/config"
"github.com/containous/traefik/middlewares/emptybackendhandler"
"github.com/containous/traefik/testhelpers"
2018-08-29 11:58:03 +02:00
"github.com/gorilla/websocket"
2018-08-06 20:00:03 +02:00
"github.com/stretchr/testify/assert"
2018-08-29 11:58:03 +02:00
"github.com/stretchr/testify/require"
"github.com/vulcand/oxy/forward"
"github.com/vulcand/oxy/roundrobin"
2017-04-18 08:22:06 +02:00
)
func TestRetry(t *testing.T) {
testCases := []struct {
2018-08-29 11:58:03 +02:00
desc string
2018-11-14 10:18:03 +01:00
config config.Retry
2018-08-29 11:58:03 +02:00
wantRetryAttempts int
wantResponseStatus int
amountFaultyEndpoints int
2017-04-18 08:22:06 +02:00
}{
{
desc: "no retry on success",
2018-11-14 10:18:03 +01:00
config: config.Retry{Attempts: 1},
wantRetryAttempts: 0,
wantResponseStatus: http.StatusOK,
amountFaultyEndpoints: 0,
},
{
desc: "no retry when max request attempts is one",
2018-11-14 10:18:03 +01:00
config: config.Retry{Attempts: 1},
wantRetryAttempts: 0,
wantResponseStatus: http.StatusInternalServerError,
amountFaultyEndpoints: 1,
},
{
desc: "one retry when one server is faulty",
2018-11-14 10:18:03 +01:00
config: config.Retry{Attempts: 2},
wantRetryAttempts: 1,
wantResponseStatus: http.StatusOK,
amountFaultyEndpoints: 1,
},
{
desc: "two retries when two servers are faulty",
2018-11-14 10:18:03 +01:00
config: config.Retry{Attempts: 3},
wantRetryAttempts: 2,
wantResponseStatus: http.StatusOK,
amountFaultyEndpoints: 2,
},
{
desc: "max attempts exhausted delivers the 5xx response",
2018-11-14 10:18:03 +01:00
config: config.Retry{Attempts: 3},
wantRetryAttempts: 2,
wantResponseStatus: http.StatusInternalServerError,
amountFaultyEndpoints: 3,
2017-04-18 08:22:06 +02:00
},
}
backendServer := httptest.NewServer(http.HandlerFunc(func(rw http.ResponseWriter, req *http.Request) {
rw.WriteHeader(http.StatusOK)
2018-11-14 10:18:03 +01:00
_, err := rw.Write([]byte("OK"))
if err != nil {
http.Error(rw, err.Error(), http.StatusInternalServerError)
}
}))
forwarder, err := forward.New()
2018-11-14 10:18:03 +01:00
require.NoError(t, err)
2018-08-29 11:58:03 +02:00
for _, test := range testCases {
test := test
t.Run(test.desc, func(t *testing.T) {
2017-04-18 08:22:06 +02:00
t.Parallel()
loadBalancer, err := roundrobin.New(forwarder)
2018-11-14 10:18:03 +01:00
require.NoError(t, err)
basePort := 33444
2018-08-29 11:58:03 +02:00
for i := 0; i < test.amountFaultyEndpoints; i++ {
// 192.0.2.0 is a non-routable IP for testing purposes.
// See: https://stackoverflow.com/questions/528538/non-routable-ip-address/18436928#18436928
// We only use the port specification here because the URL is used as identifier
// in the load balancer and using the exact same URL would not add a new server.
2018-08-06 20:00:03 +02:00
err = loadBalancer.UpsertServer(testhelpers.MustParseURL("http://192.0.2.0:" + string(basePort+i)))
2018-11-14 10:18:03 +01:00
require.NoError(t, err)
}
// add the functioning server to the end of the load balancer list
2018-08-06 20:00:03 +02:00
err = loadBalancer.UpsertServer(testhelpers.MustParseURL(backendServer.URL))
2018-11-14 10:18:03 +01:00
require.NoError(t, err)
retryListener := &countingRetryListener{}
2018-11-14 10:18:03 +01:00
retry, err := New(context.Background(), loadBalancer, test.config, retryListener, "traefikTest")
require.NoError(t, err)
2017-04-18 08:22:06 +02:00
recorder := httptest.NewRecorder()
req := httptest.NewRequest(http.MethodGet, "http://localhost:3000/ok", nil)
retry.ServeHTTP(recorder, req)
2017-04-18 08:22:06 +02:00
2018-08-29 11:58:03 +02:00
assert.Equal(t, test.wantResponseStatus, recorder.Code)
assert.Equal(t, test.wantRetryAttempts, retryListener.timesCalled)
})
}
}
2018-11-14 10:18:03 +01:00
func TestRetryEmptyServerList(t *testing.T) {
forwarder, err := forward.New()
require.NoError(t, err)
loadBalancer, err := roundrobin.New(forwarder)
require.NoError(t, err)
// The EmptyBackend middleware ensures that there is a 503
// response status set when there is no backend server in the pool.
next := emptybackendhandler.New(loadBalancer)
retryListener := &countingRetryListener{}
retry, err := New(context.Background(), next, config.Retry{Attempts: 3}, retryListener, "traefikTest")
require.NoError(t, err)
recorder := httptest.NewRecorder()
req := httptest.NewRequest(http.MethodGet, "http://localhost:3000/ok", nil)
retry.ServeHTTP(recorder, req)
assert.Equal(t, http.StatusServiceUnavailable, recorder.Code)
assert.Equal(t, 0, retryListener.timesCalled)
}
func TestRetryListeners(t *testing.T) {
req := httptest.NewRequest(http.MethodGet, "/", nil)
retryListeners := Listeners{&countingRetryListener{}, &countingRetryListener{}}
retryListeners.Retried(req, 1)
retryListeners.Retried(req, 1)
for _, retryListener := range retryListeners {
listener := retryListener.(*countingRetryListener)
if listener.timesCalled != 2 {
t.Errorf("retry listener was called %d time(s), want %d time(s)", listener.timesCalled, 2)
}
}
}
// countingRetryListener is a Listener implementation to count the times the Retried fn is called.
type countingRetryListener struct {
timesCalled int
}
func (l *countingRetryListener) Retried(req *http.Request, attempt int) {
l.timesCalled++
}
func TestRetryWithFlush(t *testing.T) {
next := http.HandlerFunc(func(rw http.ResponseWriter, req *http.Request) {
rw.WriteHeader(200)
_, err := rw.Write([]byte("FULL "))
if err != nil {
http.Error(rw, err.Error(), http.StatusInternalServerError)
}
rw.(http.Flusher).Flush()
_, err = rw.Write([]byte("DATA"))
if err != nil {
http.Error(rw, err.Error(), http.StatusInternalServerError)
}
})
retry, err := New(context.Background(), next, config.Retry{Attempts: 1}, &countingRetryListener{}, "traefikTest")
require.NoError(t, err)
responseRecorder := httptest.NewRecorder()
retry.ServeHTTP(responseRecorder, &http.Request{})
assert.Equal(t, "FULL DATA", responseRecorder.Body.String())
}
2018-08-29 11:58:03 +02:00
func TestRetryWebsocket(t *testing.T) {
testCases := []struct {
desc string
maxRequestAttempts int
expectedRetryAttempts int
expectedResponseStatus int
expectedError bool
amountFaultyEndpoints int
}{
{
desc: "Switching ok after 2 retries",
maxRequestAttempts: 3,
expectedRetryAttempts: 2,
amountFaultyEndpoints: 2,
expectedResponseStatus: http.StatusSwitchingProtocols,
},
{
desc: "Switching failed",
maxRequestAttempts: 2,
expectedRetryAttempts: 1,
amountFaultyEndpoints: 2,
expectedResponseStatus: http.StatusBadGateway,
expectedError: true,
},
}
forwarder, err := forward.New()
if err != nil {
t.Fatalf("Error creating forwarder: %s", err)
}
backendServer := httptest.NewServer(http.HandlerFunc(func(rw http.ResponseWriter, req *http.Request) {
upgrader := websocket.Upgrader{}
2018-11-14 10:18:03 +01:00
_, err := upgrader.Upgrade(rw, req, nil)
if err != nil {
http.Error(rw, err.Error(), http.StatusInternalServerError)
}
2018-08-29 11:58:03 +02:00
}))
for _, test := range testCases {
test := test
t.Run(test.desc, func(t *testing.T) {
t.Parallel()
loadBalancer, err := roundrobin.New(forwarder)
if err != nil {
t.Fatalf("Error creating load balancer: %s", err)
2017-04-18 08:22:06 +02:00
}
2018-08-29 11:58:03 +02:00
basePort := 33444
for i := 0; i < test.amountFaultyEndpoints; i++ {
// 192.0.2.0 is a non-routable IP for testing purposes.
// See: https://stackoverflow.com/questions/528538/non-routable-ip-address/18436928#18436928
// We only use the port specification here because the URL is used as identifier
// in the load balancer and using the exact same URL would not add a new server.
2018-11-14 10:18:03 +01:00
_ = loadBalancer.UpsertServer(testhelpers.MustParseURL("http://192.0.2.0:" + string(basePort+i)))
2017-04-18 08:22:06 +02:00
}
2018-08-29 11:58:03 +02:00
// add the functioning server to the end of the load balancer list
loadBalancer.UpsertServer(testhelpers.MustParseURL(backendServer.URL))
retryListener := &countingRetryListener{}
2018-11-14 10:18:03 +01:00
retryH, err := New(context.Background(), loadBalancer, config.Retry{Attempts: test.maxRequestAttempts}, retryListener, "traefikTest")
require.NoError(t, err)
2018-08-29 11:58:03 +02:00
2018-11-14 10:18:03 +01:00
retryServer := httptest.NewServer(retryH)
2018-08-29 11:58:03 +02:00
url := strings.Replace(retryServer.URL, "http", "ws", 1)
_, response, err := websocket.DefaultDialer.Dial(url, nil)
if !test.expectedError {
require.NoError(t, err)
2017-04-18 08:22:06 +02:00
}
2018-08-29 11:58:03 +02:00
assert.Equal(t, test.expectedResponseStatus, response.StatusCode)
assert.Equal(t, test.expectedRetryAttempts, retryListener.timesCalled)
2017-04-18 08:22:06 +02:00
})
}
}