mirror of
https://github.com/traefik/traefik
synced 2026-02-03 10:00:33 +00:00
311 lines
11 KiB
Go
311 lines
11 KiB
Go
package server
|
|
|
|
import (
|
|
"net/http"
|
|
"net/http/httptest"
|
|
"net/url"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
"github.com/traefik/traefik/v3/pkg/config/dynamic"
|
|
"github.com/traefik/traefik/v3/pkg/config/runtime"
|
|
"github.com/traefik/traefik/v3/pkg/config/static"
|
|
"github.com/traefik/traefik/v3/pkg/server/middleware"
|
|
"github.com/traefik/traefik/v3/pkg/server/service"
|
|
"github.com/traefik/traefik/v3/pkg/tcp"
|
|
th "github.com/traefik/traefik/v3/pkg/testhelpers"
|
|
"github.com/traefik/traefik/v3/pkg/tls"
|
|
)
|
|
|
|
func TestReuseService(t *testing.T) {
|
|
testServer := httptest.NewServer(http.HandlerFunc(func(rw http.ResponseWriter, req *http.Request) {
|
|
rw.WriteHeader(http.StatusOK)
|
|
}))
|
|
defer testServer.Close()
|
|
|
|
staticConfig := static.Configuration{
|
|
EntryPoints: map[string]*static.EntryPoint{
|
|
"web": {},
|
|
},
|
|
}
|
|
|
|
dynamicConfigs := th.BuildConfiguration(
|
|
th.WithRouters(
|
|
th.WithRouter("foo",
|
|
th.WithEntryPoints("web"),
|
|
th.WithServiceName("bar"),
|
|
th.WithRule("Path(`/ok`)")),
|
|
th.WithRouter("foo2",
|
|
th.WithEntryPoints("web"),
|
|
th.WithRule("Path(`/unauthorized`)"),
|
|
th.WithServiceName("bar"),
|
|
th.WithRouterMiddlewares("basicauth")),
|
|
),
|
|
th.WithMiddlewares(th.WithMiddleware("basicauth",
|
|
th.WithBasicAuth(&dynamic.BasicAuth{Users: []string{"foo:bar"}}),
|
|
)),
|
|
th.WithServices(
|
|
th.WithService("bar", th.WithServiceServersLoadBalancer(th.WithServers(th.WithServer(testServer.URL)))),
|
|
),
|
|
)
|
|
|
|
transportManager := service.NewTransportManager(nil)
|
|
transportManager.Update(map[string]*dynamic.ServersTransport{"default@internal": {}})
|
|
|
|
managerFactory := service.NewManagerFactory(staticConfig, nil, nil, transportManager, proxyBuilderMock{}, nil)
|
|
tlsManager := tls.NewManager(nil)
|
|
|
|
dialerManager := tcp.NewDialerManager(nil)
|
|
dialerManager.Update(map[string]*dynamic.TCPServersTransport{"default@internal": {}})
|
|
factory, err := NewRouterFactory(staticConfig, managerFactory, tlsManager, nil, nil, dialerManager)
|
|
require.NoError(t, err)
|
|
|
|
entryPointsHandlers, _ := factory.CreateRouters(runtime.NewConfig(dynamic.Configuration{HTTP: dynamicConfigs}))
|
|
|
|
// Test that the /ok path returns a status 200.
|
|
responseRecorderOk := &httptest.ResponseRecorder{}
|
|
requestOk := httptest.NewRequest(http.MethodGet, testServer.URL+"/ok", nil)
|
|
entryPointsHandlers["web"].GetHTTPHandler().ServeHTTP(responseRecorderOk, requestOk)
|
|
|
|
assert.Equal(t, http.StatusOK, responseRecorderOk.Result().StatusCode, "status code")
|
|
|
|
// Test that the /unauthorized path returns a 401 because of
|
|
// the basic authentication defined on the frontend.
|
|
responseRecorderUnauthorized := &httptest.ResponseRecorder{}
|
|
requestUnauthorized := httptest.NewRequest(http.MethodGet, testServer.URL+"/unauthorized", nil)
|
|
entryPointsHandlers["web"].GetHTTPHandler().ServeHTTP(responseRecorderUnauthorized, requestUnauthorized)
|
|
|
|
assert.Equal(t, http.StatusUnauthorized, responseRecorderUnauthorized.Result().StatusCode, "status code")
|
|
}
|
|
|
|
func TestServerResponseEmptyBackend(t *testing.T) {
|
|
const requestPath = "/path"
|
|
const routeRule = "Path(`" + requestPath + "`)"
|
|
|
|
testCases := []struct {
|
|
desc string
|
|
config func(testServerURL string) *dynamic.HTTPConfiguration
|
|
expectedStatusCode int
|
|
}{
|
|
{
|
|
desc: "Ok",
|
|
config: func(testServerURL string) *dynamic.HTTPConfiguration {
|
|
return th.BuildConfiguration(
|
|
th.WithRouters(th.WithRouter("foo",
|
|
th.WithEntryPoints("web"),
|
|
th.WithServiceName("bar"),
|
|
th.WithRule(routeRule)),
|
|
),
|
|
th.WithServices(
|
|
th.WithService("bar", th.WithServiceServersLoadBalancer(th.WithServers(th.WithServer(testServerURL)))),
|
|
),
|
|
)
|
|
},
|
|
expectedStatusCode: http.StatusOK,
|
|
},
|
|
{
|
|
desc: "No Frontend",
|
|
config: func(testServerURL string) *dynamic.HTTPConfiguration {
|
|
return th.BuildConfiguration()
|
|
},
|
|
expectedStatusCode: http.StatusNotFound,
|
|
},
|
|
{
|
|
desc: "Empty Backend LB",
|
|
config: func(testServerURL string) *dynamic.HTTPConfiguration {
|
|
return th.BuildConfiguration(
|
|
th.WithRouters(th.WithRouter("foo",
|
|
th.WithEntryPoints("web"),
|
|
th.WithServiceName("bar"),
|
|
th.WithRule(routeRule)),
|
|
),
|
|
th.WithServices(
|
|
th.WithService("bar", th.WithServiceServersLoadBalancer()),
|
|
),
|
|
)
|
|
},
|
|
expectedStatusCode: http.StatusServiceUnavailable,
|
|
},
|
|
{
|
|
desc: "Empty Backend LB Sticky",
|
|
config: func(testServerURL string) *dynamic.HTTPConfiguration {
|
|
return th.BuildConfiguration(
|
|
th.WithRouters(th.WithRouter("foo",
|
|
th.WithEntryPoints("web"),
|
|
th.WithServiceName("bar"),
|
|
th.WithRule(routeRule)),
|
|
),
|
|
th.WithServices(
|
|
th.WithService("bar", th.WithServiceServersLoadBalancer(th.WithSticky("test"))),
|
|
),
|
|
)
|
|
},
|
|
expectedStatusCode: http.StatusServiceUnavailable,
|
|
},
|
|
{
|
|
desc: "Empty Backend LB Sticky",
|
|
config: func(testServerURL string) *dynamic.HTTPConfiguration {
|
|
return th.BuildConfiguration(
|
|
th.WithRouters(th.WithRouter("foo",
|
|
th.WithEntryPoints("web"),
|
|
th.WithServiceName("bar"),
|
|
th.WithRule(routeRule)),
|
|
),
|
|
th.WithServices(
|
|
th.WithService("bar", th.WithServiceServersLoadBalancer(th.WithSticky("test"))),
|
|
),
|
|
)
|
|
},
|
|
expectedStatusCode: http.StatusServiceUnavailable,
|
|
},
|
|
}
|
|
|
|
for _, test := range testCases {
|
|
t.Run(test.desc, func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
testServer := httptest.NewServer(http.HandlerFunc(func(rw http.ResponseWriter, req *http.Request) {
|
|
rw.WriteHeader(http.StatusOK)
|
|
}))
|
|
defer testServer.Close()
|
|
|
|
staticConfig := static.Configuration{
|
|
EntryPoints: map[string]*static.EntryPoint{
|
|
"web": {},
|
|
},
|
|
}
|
|
|
|
transportManager := service.NewTransportManager(nil)
|
|
transportManager.Update(map[string]*dynamic.ServersTransport{"default@internal": {}})
|
|
|
|
managerFactory := service.NewManagerFactory(staticConfig, nil, nil, transportManager, proxyBuilderMock{}, nil)
|
|
tlsManager := tls.NewManager(nil)
|
|
|
|
dialerManager := tcp.NewDialerManager(nil)
|
|
dialerManager.Update(map[string]*dynamic.TCPServersTransport{"default@internal": {}})
|
|
observabiltyMgr := middleware.NewObservabilityMgr(staticConfig, nil, nil, nil, nil, nil)
|
|
factory, err := NewRouterFactory(staticConfig, managerFactory, tlsManager, observabiltyMgr, nil, dialerManager)
|
|
require.NoError(t, err)
|
|
|
|
entryPointsHandlers, _ := factory.CreateRouters(runtime.NewConfig(dynamic.Configuration{HTTP: test.config(testServer.URL)}))
|
|
|
|
responseRecorder := &httptest.ResponseRecorder{}
|
|
request := httptest.NewRequest(http.MethodGet, testServer.URL+requestPath, nil)
|
|
|
|
entryPointsHandlers["web"].GetHTTPHandler().ServeHTTP(responseRecorder, request)
|
|
|
|
assert.Equal(t, test.expectedStatusCode, responseRecorder.Result().StatusCode, "status code")
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestInternalServices(t *testing.T) {
|
|
testServer := httptest.NewServer(http.HandlerFunc(func(rw http.ResponseWriter, req *http.Request) {
|
|
rw.WriteHeader(http.StatusOK)
|
|
}))
|
|
defer testServer.Close()
|
|
|
|
staticConfig := static.Configuration{
|
|
API: &static.API{},
|
|
EntryPoints: map[string]*static.EntryPoint{
|
|
"web": {},
|
|
},
|
|
}
|
|
|
|
dynamicConfigs := th.BuildConfiguration(
|
|
th.WithRouters(
|
|
th.WithRouter("foo",
|
|
th.WithEntryPoints("web"),
|
|
th.WithServiceName("api@internal"),
|
|
th.WithRule("PathPrefix(`/api`)")),
|
|
),
|
|
)
|
|
|
|
transportManager := service.NewTransportManager(nil)
|
|
transportManager.Update(map[string]*dynamic.ServersTransport{"default@internal": {}})
|
|
|
|
managerFactory := service.NewManagerFactory(staticConfig, nil, nil, transportManager, nil, nil)
|
|
tlsManager := tls.NewManager(nil)
|
|
|
|
dialerManager := tcp.NewDialerManager(nil)
|
|
dialerManager.Update(map[string]*dynamic.TCPServersTransport{"default@internal": {}})
|
|
factory, err := NewRouterFactory(staticConfig, managerFactory, tlsManager, nil, nil, dialerManager)
|
|
require.NoError(t, err)
|
|
|
|
entryPointsHandlers, _ := factory.CreateRouters(runtime.NewConfig(dynamic.Configuration{HTTP: dynamicConfigs}))
|
|
|
|
// Test that the /ok path returns a status 200.
|
|
responseRecorderOk := &httptest.ResponseRecorder{}
|
|
requestOk := httptest.NewRequest(http.MethodGet, testServer.URL+"/api/rawdata", nil)
|
|
entryPointsHandlers["web"].GetHTTPHandler().ServeHTTP(responseRecorderOk, requestOk)
|
|
|
|
assert.Equal(t, http.StatusOK, responseRecorderOk.Result().StatusCode, "status code")
|
|
}
|
|
|
|
func TestRecursionService(t *testing.T) {
|
|
testServer := httptest.NewServer(http.HandlerFunc(func(rw http.ResponseWriter, req *http.Request) {
|
|
rw.WriteHeader(http.StatusOK)
|
|
}))
|
|
defer testServer.Close()
|
|
|
|
staticConfig := static.Configuration{
|
|
EntryPoints: map[string]*static.EntryPoint{
|
|
"web": {},
|
|
},
|
|
}
|
|
|
|
dynamicConfigs := th.BuildConfiguration(
|
|
th.WithRouters(
|
|
th.WithRouter("foo@provider1",
|
|
th.WithEntryPoints("web"),
|
|
th.WithServiceName("bar"),
|
|
th.WithRule("Path(`/ok`)")),
|
|
),
|
|
th.WithMiddlewares(th.WithMiddleware("customerror",
|
|
th.WithErrorPage(&dynamic.ErrorPage{Service: "bar"}),
|
|
)),
|
|
th.WithServices(
|
|
th.WithService("bar@provider1", th.WithServiceWRR(th.WithWRRServices(th.WithWRRService("foo")))),
|
|
th.WithService("foo@provider1", th.WithServiceWRR(th.WithWRRServices(th.WithWRRService("bar")))),
|
|
),
|
|
)
|
|
|
|
transportManager := service.NewTransportManager(nil)
|
|
transportManager.Update(map[string]*dynamic.ServersTransport{"default@internal": {}})
|
|
|
|
managerFactory := service.NewManagerFactory(staticConfig, nil, nil, transportManager, nil, nil)
|
|
tlsManager := tls.NewManager(nil)
|
|
|
|
dialerManager := tcp.NewDialerManager(nil)
|
|
dialerManager.Update(map[string]*dynamic.TCPServersTransport{"default@internal": {}})
|
|
factory, err := NewRouterFactory(staticConfig, managerFactory, tlsManager, nil, nil, dialerManager)
|
|
require.NoError(t, err)
|
|
|
|
rtConf := runtime.NewConfig(dynamic.Configuration{HTTP: dynamicConfigs})
|
|
entryPointsHandlers, _ := factory.CreateRouters(rtConf)
|
|
|
|
// Test that the /ok path returns a status 404.
|
|
responseRecorderOk := &httptest.ResponseRecorder{}
|
|
requestOk := httptest.NewRequest(http.MethodGet, testServer.URL+"/ok", nil)
|
|
entryPointsHandlers["web"].GetHTTPHandler().ServeHTTP(responseRecorderOk, requestOk)
|
|
|
|
assert.Equal(t, http.StatusNotFound, responseRecorderOk.Result().StatusCode, "status code")
|
|
|
|
require.NotNil(t, rtConf.Routers["foo@provider1"])
|
|
assert.Contains(t, rtConf.Routers["foo@provider1"].Err, "building HTTP service \"foo\": building HTTP service \"bar\": could not instantiate service bar@provider1: recursion detected in service:bar@provider1->service:foo@provider1->service:bar@provider1")
|
|
require.NotNil(t, rtConf.Services["bar@provider1"])
|
|
assert.Contains(t, rtConf.Services["bar@provider1"].Err, "could not instantiate service bar@provider1: recursion detected in service:bar@provider1->service:foo@provider1->service:bar@provider1")
|
|
}
|
|
|
|
type proxyBuilderMock struct{}
|
|
|
|
func (p proxyBuilderMock) Build(_ string, _ *url.URL, _, _ bool, _ time.Duration) (http.Handler, error) {
|
|
return http.HandlerFunc(func(responseWriter http.ResponseWriter, req *http.Request) {}), nil
|
|
}
|
|
|
|
func (p proxyBuilderMock) Update(_ map[string]*dynamic.ServersTransport) {
|
|
panic("implement me")
|
|
}
|