155 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			155 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			Go
		
	
	
	
| // Copyright 2017 The Prometheus 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 remote
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"errors"
 | |
| 	"net/http"
 | |
| 	"net/http/httptest"
 | |
| 	"net/url"
 | |
| 	"strings"
 | |
| 	"testing"
 | |
| 	"time"
 | |
| 
 | |
| 	config_util "github.com/prometheus/common/config"
 | |
| 	"github.com/prometheus/common/model"
 | |
| 	"github.com/stretchr/testify/require"
 | |
| )
 | |
| 
 | |
| var longErrMessage = strings.Repeat("error message", maxErrMsgLen)
 | |
| 
 | |
| func TestStoreHTTPErrorHandling(t *testing.T) {
 | |
| 	tests := []struct {
 | |
| 		code int
 | |
| 		err  error
 | |
| 	}{
 | |
| 		{
 | |
| 			code: 200,
 | |
| 			err:  nil,
 | |
| 		},
 | |
| 		{
 | |
| 			code: 300,
 | |
| 			err:  errors.New("server returned HTTP status 300 Multiple Choices: " + longErrMessage[:maxErrMsgLen]),
 | |
| 		},
 | |
| 		{
 | |
| 			code: 404,
 | |
| 			err:  errors.New("server returned HTTP status 404 Not Found: " + longErrMessage[:maxErrMsgLen]),
 | |
| 		},
 | |
| 		{
 | |
| 			code: 500,
 | |
| 			err:  RecoverableError{errors.New("server returned HTTP status 500 Internal Server Error: " + longErrMessage[:maxErrMsgLen]), defaultBackoff},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	for _, test := range tests {
 | |
| 		server := httptest.NewServer(
 | |
| 			http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
 | |
| 				http.Error(w, longErrMessage, test.code)
 | |
| 			}),
 | |
| 		)
 | |
| 
 | |
| 		serverURL, err := url.Parse(server.URL)
 | |
| 		require.NoError(t, err)
 | |
| 
 | |
| 		conf := &ClientConfig{
 | |
| 			URL:     &config_util.URL{URL: serverURL},
 | |
| 			Timeout: model.Duration(time.Second),
 | |
| 		}
 | |
| 
 | |
| 		hash, err := toHash(conf)
 | |
| 		require.NoError(t, err)
 | |
| 		c, err := NewWriteClient(hash, conf)
 | |
| 		require.NoError(t, err)
 | |
| 
 | |
| 		err = c.Store(context.Background(), []byte{})
 | |
| 		if test.err != nil {
 | |
| 			require.EqualError(t, err, test.err.Error())
 | |
| 		} else {
 | |
| 			require.NoError(t, err)
 | |
| 		}
 | |
| 
 | |
| 		server.Close()
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestClientRetryAfter(t *testing.T) {
 | |
| 	server := httptest.NewServer(
 | |
| 		http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
 | |
| 			http.Error(w, longErrMessage, 429)
 | |
| 		}),
 | |
| 	)
 | |
| 	defer server.Close()
 | |
| 
 | |
| 	getClient := func(conf *ClientConfig) WriteClient {
 | |
| 		hash, err := toHash(conf)
 | |
| 		require.NoError(t, err)
 | |
| 		c, err := NewWriteClient(hash, conf)
 | |
| 		require.NoError(t, err)
 | |
| 		return c
 | |
| 	}
 | |
| 
 | |
| 	serverURL, err := url.Parse(server.URL)
 | |
| 	require.NoError(t, err)
 | |
| 
 | |
| 	conf := &ClientConfig{
 | |
| 		URL:              &config_util.URL{URL: serverURL},
 | |
| 		Timeout:          model.Duration(time.Second),
 | |
| 		RetryOnRateLimit: false,
 | |
| 	}
 | |
| 
 | |
| 	c := getClient(conf)
 | |
| 	err = c.Store(context.Background(), []byte{})
 | |
| 	_, ok := err.(RecoverableError)
 | |
| 	require.False(t, ok, "Recoverable error not expected.")
 | |
| 
 | |
| 	conf = &ClientConfig{
 | |
| 		URL:              &config_util.URL{URL: serverURL},
 | |
| 		Timeout:          model.Duration(time.Second),
 | |
| 		RetryOnRateLimit: true,
 | |
| 	}
 | |
| 
 | |
| 	c = getClient(conf)
 | |
| 	err = c.Store(context.Background(), []byte{})
 | |
| 	_, ok = err.(RecoverableError)
 | |
| 	require.True(t, ok, "Recoverable error was expected.")
 | |
| }
 | |
| 
 | |
| func TestRetryAfterDuration(t *testing.T) {
 | |
| 	tc := []struct {
 | |
| 		name     string
 | |
| 		tInput   string
 | |
| 		expected model.Duration
 | |
| 	}{
 | |
| 		{
 | |
| 			name:     "seconds",
 | |
| 			tInput:   "120",
 | |
| 			expected: model.Duration(time.Second * 120),
 | |
| 		},
 | |
| 		{
 | |
| 			name:     "date-time default",
 | |
| 			tInput:   time.RFC1123, // Expected layout is http.TimeFormat, hence an error.
 | |
| 			expected: defaultBackoff,
 | |
| 		},
 | |
| 		{
 | |
| 			name:     "retry-after not provided",
 | |
| 			tInput:   "", // Expected layout is http.TimeFormat, hence an error.
 | |
| 			expected: defaultBackoff,
 | |
| 		},
 | |
| 	}
 | |
| 	for _, c := range tc {
 | |
| 		require.Equal(t, c.expected, retryAfterDuration(c.tInput), c.name)
 | |
| 	}
 | |
| }
 |