2019-03-11 14:54:05 +01:00
|
|
|
package integration
|
|
|
|
|
|
|
|
import (
|
2019-06-11 15:12:04 +02:00
|
|
|
"bytes"
|
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
|
|
|
"flag"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
2024-01-09 17:00:07 +01:00
|
|
|
"net"
|
2019-03-11 14:54:05 +01:00
|
|
|
"net/http"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2019-06-11 15:12:04 +02:00
|
|
|
"regexp"
|
2024-01-09 17:00:07 +01:00
|
|
|
"strings"
|
|
|
|
"testing"
|
2019-03-11 14:54:05 +01:00
|
|
|
"time"
|
|
|
|
|
2019-06-11 15:12:04 +02:00
|
|
|
"github.com/pmezard/go-difflib/difflib"
|
2024-01-09 17:00:07 +01:00
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"github.com/stretchr/testify/suite"
|
2020-09-16 15:46:04 +02:00
|
|
|
"github.com/traefik/traefik/v2/integration/try"
|
|
|
|
"github.com/traefik/traefik/v2/pkg/api"
|
|
|
|
"github.com/traefik/traefik/v2/pkg/log"
|
2019-03-11 14:54:05 +01:00
|
|
|
)
|
|
|
|
|
2019-06-11 15:12:04 +02:00
|
|
|
var updateExpected = flag.Bool("update_expected", false, "Update expected files in testdata")
|
|
|
|
|
2020-05-11 12:06:07 +02:00
|
|
|
// K8sSuite tests suite.
|
2019-03-11 14:54:05 +01:00
|
|
|
type K8sSuite struct{ BaseSuite }
|
|
|
|
|
2024-01-09 17:00:07 +01:00
|
|
|
func TestK8sSuite(t *testing.T) {
|
|
|
|
suite.Run(t, new(K8sSuite))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *K8sSuite) SetupSuite() {
|
|
|
|
s.BaseSuite.SetupSuite()
|
|
|
|
|
|
|
|
s.createComposeProject("k8s")
|
|
|
|
s.composeUp()
|
2019-03-11 14:54:05 +01:00
|
|
|
|
2019-08-11 12:22:14 +02:00
|
|
|
abs, err := filepath.Abs("./fixtures/k8s/config.skip/kubeconfig.yaml")
|
2024-01-09 17:00:07 +01:00
|
|
|
require.NoError(s.T(), err)
|
2019-03-11 14:54:05 +01:00
|
|
|
|
2020-02-24 16:06:05 +01:00
|
|
|
err = try.Do(60*time.Second, func() error {
|
2019-03-11 14:54:05 +01:00
|
|
|
_, err := os.Stat(abs)
|
|
|
|
return err
|
2020-02-24 16:06:05 +01:00
|
|
|
})
|
2024-01-09 17:00:07 +01:00
|
|
|
require.NoError(s.T(), err)
|
|
|
|
|
|
|
|
data, err := os.ReadFile(abs)
|
|
|
|
require.NoError(s.T(), err)
|
|
|
|
|
|
|
|
content := strings.ReplaceAll(string(data), "https://server:6443", fmt.Sprintf("https://%s", net.JoinHostPort(s.getComposeServiceIP("server"), "6443")))
|
|
|
|
|
|
|
|
err = os.WriteFile(abs, []byte(content), 0o644)
|
|
|
|
require.NoError(s.T(), err)
|
2019-03-11 14:54:05 +01:00
|
|
|
|
|
|
|
err = os.Setenv("KUBECONFIG", abs)
|
2024-01-09 17:00:07 +01:00
|
|
|
require.NoError(s.T(), err)
|
2019-03-14 15:56:06 +01:00
|
|
|
}
|
|
|
|
|
2024-01-09 17:00:07 +01:00
|
|
|
func (s *K8sSuite) TearDownSuite() {
|
|
|
|
s.BaseSuite.TearDownSuite()
|
2019-03-14 15:56:06 +01:00
|
|
|
|
2019-08-11 12:22:14 +02:00
|
|
|
generatedFiles := []string{
|
|
|
|
"./fixtures/k8s/config.skip/kubeconfig.yaml",
|
|
|
|
"./fixtures/k8s/config.skip/k3s.log",
|
|
|
|
"./fixtures/k8s/coredns.yaml",
|
|
|
|
"./fixtures/k8s/rolebindings.yaml",
|
|
|
|
"./fixtures/k8s/traefik.yaml",
|
2020-12-10 14:58:04 +01:00
|
|
|
"./fixtures/k8s/ccm.yaml",
|
2019-03-14 15:56:06 +01:00
|
|
|
}
|
2019-08-11 12:22:14 +02:00
|
|
|
|
|
|
|
for _, filename := range generatedFiles {
|
2021-11-25 10:10:06 +00:00
|
|
|
if err := os.Remove(filename); err != nil {
|
2019-08-11 12:22:14 +02:00
|
|
|
log.WithoutContext().Warning(err)
|
|
|
|
}
|
2019-03-14 15:56:06 +01:00
|
|
|
}
|
|
|
|
}
|
2019-03-11 14:54:05 +01:00
|
|
|
|
2024-01-09 17:00:07 +01:00
|
|
|
func (s *K8sSuite) TestIngressConfiguration() {
|
|
|
|
s.traefikCmd(withConfigFile("fixtures/k8s_default.toml"))
|
2019-03-11 14:54:05 +01:00
|
|
|
|
2024-01-09 17:00:07 +01:00
|
|
|
s.testConfiguration("testdata/rawdata-ingress.json", "8080")
|
2019-03-14 15:56:06 +01:00
|
|
|
}
|
|
|
|
|
2024-01-09 17:00:07 +01:00
|
|
|
func (s *K8sSuite) TestIngressLabelSelector() {
|
|
|
|
s.traefikCmd(withConfigFile("fixtures/k8s_ingress_label_selector.toml"))
|
2020-11-20 00:18:04 +01:00
|
|
|
|
2024-01-09 17:00:07 +01:00
|
|
|
s.testConfiguration("testdata/rawdata-ingress-label-selector.json", "8080")
|
2020-11-20 00:18:04 +01:00
|
|
|
}
|
|
|
|
|
2024-01-09 17:00:07 +01:00
|
|
|
func (s *K8sSuite) TestCRDConfiguration() {
|
|
|
|
s.traefikCmd(withConfigFile("fixtures/k8s_crd.toml"))
|
2019-03-11 14:54:05 +01:00
|
|
|
|
2024-01-09 17:00:07 +01:00
|
|
|
s.testConfiguration("testdata/rawdata-crd.json", "8000")
|
2019-06-11 15:12:04 +02:00
|
|
|
}
|
|
|
|
|
2024-01-09 17:00:07 +01:00
|
|
|
func (s *K8sSuite) TestCRDLabelSelector() {
|
|
|
|
s.traefikCmd(withConfigFile("fixtures/k8s_crd_label_selector.toml"))
|
2020-11-20 00:18:04 +01:00
|
|
|
|
2024-01-09 17:00:07 +01:00
|
|
|
s.testConfiguration("testdata/rawdata-crd-label-selector.json", "8000")
|
2020-11-20 00:18:04 +01:00
|
|
|
}
|
|
|
|
|
2024-01-09 17:00:07 +01:00
|
|
|
func (s *K8sSuite) TestGatewayConfiguration() {
|
|
|
|
s.traefikCmd(withConfigFile("fixtures/k8s_gateway.toml"))
|
2020-12-15 16:40:05 +01:00
|
|
|
|
2024-01-09 17:00:07 +01:00
|
|
|
s.testConfiguration("testdata/rawdata-gateway.json", "8080")
|
2020-12-15 16:40:05 +01:00
|
|
|
}
|
|
|
|
|
2024-01-09 17:00:07 +01:00
|
|
|
func (s *K8sSuite) TestIngressclass() {
|
|
|
|
s.traefikCmd(withConfigFile("fixtures/k8s_ingressclass.toml"))
|
2021-03-02 21:34:03 +01:00
|
|
|
|
2024-01-09 17:00:07 +01:00
|
|
|
s.testConfiguration("testdata/rawdata-ingressclass.json", "8080")
|
2021-03-02 21:34:03 +01:00
|
|
|
}
|
|
|
|
|
2024-01-09 17:00:07 +01:00
|
|
|
func (s *K8sSuite) testConfiguration(path, apiPort string) {
|
2019-11-14 19:28:04 +01:00
|
|
|
err := try.GetRequest("http://127.0.0.1:"+apiPort+"/api/entrypoints", 20*time.Second, try.BodyContains(`"name":"web"`))
|
2024-01-09 17:00:07 +01:00
|
|
|
require.NoError(s.T(), err)
|
2019-08-11 12:22:14 +02:00
|
|
|
|
2019-06-11 15:12:04 +02:00
|
|
|
expectedJSON := filepath.FromSlash(path)
|
|
|
|
|
|
|
|
if *updateExpected {
|
|
|
|
fi, err := os.Create(expectedJSON)
|
2024-01-09 17:00:07 +01:00
|
|
|
require.NoError(s.T(), err)
|
2019-06-11 15:12:04 +02:00
|
|
|
err = fi.Close()
|
2024-01-09 17:00:07 +01:00
|
|
|
require.NoError(s.T(), err)
|
2019-06-11 15:12:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
var buf bytes.Buffer
|
2020-02-24 16:06:05 +01:00
|
|
|
err = try.GetRequest("http://127.0.0.1:"+apiPort+"/api/rawdata", 1*time.Minute, try.StatusCodeIs(http.StatusOK), matchesConfig(expectedJSON, &buf))
|
2019-03-11 14:54:05 +01:00
|
|
|
|
2019-06-11 15:12:04 +02:00
|
|
|
if !*updateExpected {
|
2024-01-09 17:00:07 +01:00
|
|
|
require.NoError(s.T(), err)
|
2019-06-11 15:12:04 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
2024-01-09 17:00:07 +01:00
|
|
|
log.WithoutContext().Infof("In file update mode, got expected error: %v", err)
|
2019-06-11 15:12:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
var rtRepr api.RunTimeRepresentation
|
|
|
|
err = json.Unmarshal(buf.Bytes(), &rtRepr)
|
2024-01-09 17:00:07 +01:00
|
|
|
require.NoError(s.T(), err)
|
2019-03-11 14:54:05 +01:00
|
|
|
|
2019-06-11 15:12:04 +02:00
|
|
|
newJSON, err := json.MarshalIndent(rtRepr, "", "\t")
|
2024-01-09 17:00:07 +01:00
|
|
|
require.NoError(s.T(), err)
|
2019-03-11 14:54:05 +01:00
|
|
|
|
2021-03-04 20:08:03 +01:00
|
|
|
err = os.WriteFile(expectedJSON, newJSON, 0o644)
|
2024-01-09 17:00:07 +01:00
|
|
|
require.NoError(s.T(), err)
|
|
|
|
|
|
|
|
s.T().Fatal("We do not want a passing test in file update mode")
|
2019-06-11 15:12:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func matchesConfig(wantConfig string, buf *bytes.Buffer) try.ResponseCondition {
|
|
|
|
return func(res *http.Response) error {
|
2021-03-04 20:08:03 +01:00
|
|
|
body, err := io.ReadAll(res.Body)
|
2019-06-11 15:12:04 +02:00
|
|
|
if err != nil {
|
2020-05-11 12:06:07 +02:00
|
|
|
return fmt.Errorf("failed to read response body: %w", err)
|
2019-06-11 15:12:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if err := res.Body.Close(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
var obtained api.RunTimeRepresentation
|
|
|
|
err = json.Unmarshal(body, &obtained)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if buf != nil {
|
|
|
|
buf.Reset()
|
|
|
|
if _, err := io.Copy(buf, bytes.NewReader(body)); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
got, err := json.MarshalIndent(obtained, "", "\t")
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2021-03-04 20:08:03 +01:00
|
|
|
expected, err := os.ReadFile(wantConfig)
|
2019-06-11 15:12:04 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// The pods IPs are dynamic, so we cannot predict them,
|
|
|
|
// which is why we have to ignore them in the comparison.
|
|
|
|
rxURL := regexp.MustCompile(`"(url|address)":\s+(".*")`)
|
|
|
|
sanitizedExpected := rxURL.ReplaceAll(expected, []byte(`"$1": "XXXX"`))
|
|
|
|
sanitizedGot := rxURL.ReplaceAll(got, []byte(`"$1": "XXXX"`))
|
|
|
|
|
|
|
|
rxServerStatus := regexp.MustCompile(`"http://.*?":\s+(".*")`)
|
|
|
|
sanitizedExpected = rxServerStatus.ReplaceAll(sanitizedExpected, []byte(`"http://XXXX": $1`))
|
|
|
|
sanitizedGot = rxServerStatus.ReplaceAll(sanitizedGot, []byte(`"http://XXXX": $1`))
|
|
|
|
|
|
|
|
if bytes.Equal(sanitizedExpected, sanitizedGot) {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
diff := difflib.UnifiedDiff{
|
|
|
|
FromFile: "Expected",
|
|
|
|
A: difflib.SplitLines(string(sanitizedExpected)),
|
|
|
|
ToFile: "Got",
|
|
|
|
B: difflib.SplitLines(string(sanitizedGot)),
|
|
|
|
Context: 3,
|
|
|
|
}
|
|
|
|
|
|
|
|
text, err := difflib.GetUnifiedDiffString(diff)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return errors.New(text)
|
|
|
|
}
|
2019-03-11 14:54:05 +01:00
|
|
|
}
|