diff --git a/deploy/addons/ingress/ingress-dp.yaml.tmpl b/deploy/addons/ingress/ingress-dp.yaml.tmpl index 8ee21d80c1..46806633ac 100644 --- a/deploy/addons/ingress/ingress-dp.yaml.tmpl +++ b/deploy/addons/ingress/ingress-dp.yaml.tmpl @@ -12,51 +12,6 @@ # See the License for the specific language governing permissions and # limitations under the License. -apiVersion: extensions/v1beta1 -kind: Deployment -metadata: - name: default-http-backend - namespace: kube-system - labels: - app.kubernetes.io/name: default-http-backend - app.kubernetes.io/part-of: kube-system - addonmanager.kubernetes.io/mode: Reconcile -spec: - replicas: 1 - selector: - matchLabels: - app.kubernetes.io/name: default-http-backend - addonmanager.kubernetes.io/mode: Reconcile - template: - metadata: - labels: - app.kubernetes.io/name: default-http-backend - addonmanager.kubernetes.io/mode: Reconcile - spec: - terminationGracePeriodSeconds: 60 - containers: - - name: default-http-backend - # Any image is permissible as long as: - # 1. It serves a 404 page at / - # 2. It serves 200 on a /healthz endpoint - image: {{default "gcr.io/google_containers" .ImageRepository}}/defaultbackend{{.ExoticArch}}:1.4 - imagePullPolicy: IfNotPresent - livenessProbe: - httpGet: - path: /healthz - port: 8080 - scheme: HTTP - initialDelaySeconds: 30 - timeoutSeconds: 5 - ports: - - containerPort: 8080 - resources: - limits: - cpu: 20m - memory: 30Mi - requests: - cpu: 20m - memory: 30Mi --- apiVersion: extensions/v1beta1 kind: Deployment @@ -121,7 +76,6 @@ spec: hostPort: 18080 args: - /nginx-ingress-controller - - --default-backend-service=$(POD_NAMESPACE)/default-http-backend - --configmap=$(POD_NAMESPACE)/nginx-load-balancer-conf - --tcp-services-configmap=$(POD_NAMESPACE)/tcp-services - --udp-services-configmap=$(POD_NAMESPACE)/udp-services diff --git a/deploy/addons/ingress/ingress-svc.yaml.tmpl b/deploy/addons/ingress/ingress-svc.yaml.tmpl deleted file mode 100644 index f480bbfed4..0000000000 --- a/deploy/addons/ingress/ingress-svc.yaml.tmpl +++ /dev/null @@ -1,33 +0,0 @@ -# Copyright 2016 The Kubernetes Authors All rights reserved. -# -# 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. - -apiVersion: v1 -kind: Service -metadata: - name: default-http-backend - namespace: kube-system - labels: - app.kubernetes.io/name: default-http-backend - app.kubernetes.io/part-of: kube-system - kubernetes.io/minikube-addons: ingress - kubernetes.io/minikube-addons-endpoint: ingress - addonmanager.kubernetes.io/mode: Reconcile -spec: - type: NodePort - ports: - - port: 80 - targetPort: 8080 - nodePort: 30001 - selector: - app.kubernetes.io/name: default-http-backend \ No newline at end of file diff --git a/pkg/minikube/assets/addons.go b/pkg/minikube/assets/addons.go index 75c462bec0..969d5e30d9 100644 --- a/pkg/minikube/assets/addons.go +++ b/pkg/minikube/assets/addons.go @@ -220,12 +220,6 @@ var Addons = map[string]*Addon{ "ingress-dp.yaml", "0640", true), - MustBinAsset( - "deploy/addons/ingress/ingress-svc.yaml.tmpl", - constants.AddonsPath, - "ingress-svc.yaml", - "0640", - false), }, false, "ingress"), "metrics-server": NewAddon([]*BinAsset{ MustBinAsset( diff --git a/test/integration/fn_addons.go b/test/integration/fn_addons.go index af55a61e53..a0227127e1 100644 --- a/test/integration/fn_addons.go +++ b/test/integration/fn_addons.go @@ -130,10 +130,6 @@ func testIngressController(t *testing.T) { t.Fatalf("Failed waiting for ingress-controller to be up: %v", err) } - if err := util.WaitForIngressDefaultBackendRunning(t, p); err != nil { - t.Fatalf("Failed waiting for default-http-backend to be up: %v", err) - } - ingressPath := filepath.Join(*testdataDir, "nginx-ing.yaml") if _, err := kr.RunCommand([]string{"create", "-f", ingressPath}); err != nil { t.Fatalf("Failed creating nginx ingress resource: %v", err) diff --git a/test/integration/util/util.go b/test/integration/util/util.go new file mode 100644 index 0000000000..34d5731624 --- /dev/null +++ b/test/integration/util/util.go @@ -0,0 +1,445 @@ +/* +Copyright 2016 The Kubernetes Authors All rights reserved. + +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 util + +import ( + "bufio" + "bytes" + "context" + "encoding/json" + "fmt" + "math/rand" + "os/exec" + "path/filepath" + "regexp" + "strings" + "sync" + "testing" + "time" + + "github.com/pkg/errors" + "k8s.io/apimachinery/pkg/labels" + "k8s.io/minikube/pkg/minikube/assets" + commonutil "k8s.io/minikube/pkg/util" +) + +const kubectlBinary = "kubectl" + +// MinikubeRunner runs a command +type MinikubeRunner struct { + T *testing.T + BinaryPath string + GlobalArgs string + StartArgs string + MountArgs string + Runtime string +} + +// Logf writes logs to stdout if -v is set. +func Logf(str string, args ...interface{}) { + if !testing.Verbose() { + return + } + fmt.Printf(" %s | ", time.Now().Format("15:04:05")) + fmt.Println(fmt.Sprintf(str, args...)) +} + +// Run executes a command +func (m *MinikubeRunner) Run(cmd string) error { + _, err := m.SSH(cmd) + return err +} + +// Copy copies a file +func (m *MinikubeRunner) Copy(f assets.CopyableFile) error { + path, _ := filepath.Abs(m.BinaryPath) + cmd := exec.Command("/bin/bash", "-c", path, "ssh", "--", fmt.Sprintf("cat >> %s", filepath.Join(f.GetTargetDir(), f.GetTargetName()))) + Logf("Running: %s", cmd.Args) + return cmd.Run() +} + +// CombinedOutput executes a command, returning the combined stdout and stderr +func (m *MinikubeRunner) CombinedOutput(cmd string) (string, error) { + return m.SSH(cmd) +} + +// Remove removes a file +func (m *MinikubeRunner) Remove(f assets.CopyableFile) error { + _, err := m.SSH(fmt.Sprintf("rm -rf %s", filepath.Join(f.GetTargetDir(), f.GetTargetName()))) + return err +} + +// teeRun runs a command, streaming stdout, stderr to console +func (m *MinikubeRunner) teeRun(cmd *exec.Cmd) (string, string, error) { + errPipe, err := cmd.StderrPipe() + if err != nil { + return "", "", err + } + outPipe, err := cmd.StdoutPipe() + if err != nil { + return "", "", err + } + + if err := cmd.Start(); err != nil { + return "", "", err + } + var outB bytes.Buffer + var errB bytes.Buffer + var wg sync.WaitGroup + wg.Add(2) + go func() { + if err := commonutil.TeePrefix(commonutil.ErrPrefix, errPipe, &errB, Logf); err != nil { + m.T.Logf("tee: %v", err) + } + wg.Done() + }() + go func() { + if err := commonutil.TeePrefix(commonutil.OutPrefix, outPipe, &outB, Logf); err != nil { + m.T.Logf("tee: %v", err) + } + wg.Done() + }() + err = cmd.Wait() + wg.Wait() + return outB.String(), errB.String(), err +} + +// RunCommand executes a command, optionally checking for error +func (m *MinikubeRunner) RunCommand(command string, checkError bool) string { + commandArr := strings.Split(command, " ") + path, _ := filepath.Abs(m.BinaryPath) + cmd := exec.Command(path, commandArr...) + Logf("Run: %s", cmd.Args) + stdout, stderr, err := m.teeRun(cmd) + if checkError && err != nil { + if exitError, ok := err.(*exec.ExitError); ok { + m.T.Fatalf("Error running command: %s %s. Output: %s", command, exitError.Stderr, stdout) + } else { + m.T.Fatalf("Error running command: %s %v. Output: %s", command, err, stderr) + } + } + return stdout +} + +// RunWithContext calls the minikube command with a context, useful for timeouts. +func (m *MinikubeRunner) RunWithContext(ctx context.Context, command string) (string, string, error) { + commandArr := strings.Split(command, " ") + path, _ := filepath.Abs(m.BinaryPath) + cmd := exec.CommandContext(ctx, path, commandArr...) + Logf("Run: %s", cmd.Args) + return m.teeRun(cmd) +} + +// RunDaemon executes a command, returning the stdout +func (m *MinikubeRunner) RunDaemon(command string) (*exec.Cmd, *bufio.Reader) { + commandArr := strings.Split(command, " ") + path, _ := filepath.Abs(m.BinaryPath) + cmd := exec.Command(path, commandArr...) + stdoutPipe, err := cmd.StdoutPipe() + if err != nil { + m.T.Fatalf("stdout pipe failed: %s %v", command, err) + } + stderrPipe, err := cmd.StderrPipe() + if err != nil { + m.T.Fatalf("stderr pipe failed: %s %v", command, err) + } + + var errB bytes.Buffer + go func() { + if err := commonutil.TeePrefix(commonutil.ErrPrefix, stderrPipe, &errB, Logf); err != nil { + m.T.Logf("tee: %v", err) + } + }() + + err = cmd.Start() + if err != nil { + m.T.Fatalf("Error running command: %s %v", command, err) + } + return cmd, bufio.NewReader(stdoutPipe) + +} + +// RunDaemon2 executes a command, returning the stdout and stderr +func (m *MinikubeRunner) RunDaemon2(command string) (*exec.Cmd, *bufio.Reader, *bufio.Reader) { + commandArr := strings.Split(command, " ") + path, _ := filepath.Abs(m.BinaryPath) + cmd := exec.Command(path, commandArr...) + stdoutPipe, err := cmd.StdoutPipe() + if err != nil { + m.T.Fatalf("stdout pipe failed: %s %v", command, err) + } + stderrPipe, err := cmd.StderrPipe() + if err != nil { + m.T.Fatalf("stderr pipe failed: %s %v", command, err) + } + + err = cmd.Start() + if err != nil { + m.T.Fatalf("Error running command: %s %v", command, err) + } + return cmd, bufio.NewReader(stdoutPipe), bufio.NewReader(stderrPipe) +} + +// SSH returns the output of running a command using SSH +func (m *MinikubeRunner) SSH(command string) (string, error) { + path, _ := filepath.Abs(m.BinaryPath) + cmd := exec.Command(path, "ssh", command) + Logf("SSH: %s", command) + + stdout, err := cmd.CombinedOutput() + Logf("Output: %s", stdout) + if err, ok := err.(*exec.ExitError); ok { + return string(stdout), err + } + return string(stdout), nil +} + +// Start starts the cluster +func (m *MinikubeRunner) Start(opts ...string) { + cmd := fmt.Sprintf("start %s %s %s --alsologtostderr --v=2", m.StartArgs, m.GlobalArgs, strings.Join(opts, " ")) + m.RunCommand(cmd, true) +} + +// EnsureRunning makes sure the container runtime is running +func (m *MinikubeRunner) EnsureRunning() { + if m.GetStatus() != "Running" { + m.Start() + } + m.CheckStatus("Running") +} + +// ParseEnvCmdOutput parses the output of `env` (assumes bash) +func (m *MinikubeRunner) ParseEnvCmdOutput(out string) map[string]string { + env := map[string]string{} + re := regexp.MustCompile(`(\w+?) ?= ?"?(.+?)"?\n`) + for _, m := range re.FindAllStringSubmatch(out, -1) { + env[m[1]] = m[2] + } + return env +} + +// GetStatus returns the status of a service +func (m *MinikubeRunner) GetStatus() string { + return m.RunCommand(fmt.Sprintf("status --format={{.Host}} %s", m.GlobalArgs), false) +} + +// GetLogs returns the logs of a service +func (m *MinikubeRunner) GetLogs() string { + return m.RunCommand(fmt.Sprintf("logs %s", m.GlobalArgs), true) +} + +// CheckStatus makes sure the service has the desired status, or cause fatal error +func (m *MinikubeRunner) CheckStatus(desired string) { + if err := m.CheckStatusNoFail(desired); err != nil { + m.T.Fatalf("%v", err) + } +} + +// CheckStatusNoFail makes sure the service has the desired status, returning error +func (m *MinikubeRunner) CheckStatusNoFail(desired string) error { + s := m.GetStatus() + if s != desired { + return fmt.Errorf("got state: %q, expected %q", s, desired) + } + return nil +} + +// KubectlRunner runs a command using kubectl +type KubectlRunner struct { + T *testing.T + BinaryPath string +} + +// NewKubectlRunner creates a new KubectlRunner +func NewKubectlRunner(t *testing.T) *KubectlRunner { + p, err := exec.LookPath(kubectlBinary) + if err != nil { + t.Fatalf("Couldn't find kubectl on path.") + } + return &KubectlRunner{BinaryPath: p, T: t} +} + +// RunCommandParseOutput runs a command and parses the JSON output +func (k *KubectlRunner) RunCommandParseOutput(args []string, outputObj interface{}) error { + args = append(args, "-o=json") + output, err := k.RunCommand(args) + if err != nil { + return err + } + d := json.NewDecoder(bytes.NewReader(output)) + if err := d.Decode(outputObj); err != nil { + return err + } + return nil +} + +// RunCommand runs a command, returning stdout +func (k *KubectlRunner) RunCommand(args []string) (stdout []byte, err error) { + inner := func() error { + cmd := exec.Command(k.BinaryPath, args...) + stdout, err = cmd.CombinedOutput() + if err != nil { + retriable := &commonutil.RetriableError{Err: fmt.Errorf("error running command %s: %v. Stdout: \n %s", args, err, stdout)} + k.T.Log(retriable) + return retriable + } + return nil + } + + err = commonutil.RetryAfter(3, inner, 2*time.Second) + return stdout, err +} + +// CreateRandomNamespace creates a random namespace +func (k *KubectlRunner) CreateRandomNamespace() string { + const strLen = 20 + name := genRandString(strLen) + if _, err := k.RunCommand([]string{"create", "namespace", name}); err != nil { + k.T.Fatalf("Error creating namespace: %v", err) + } + return name +} + +func genRandString(strLen int) string { + const chars = "abcdefghijklmnopqrstuvwxyz0123456789" + rand.Seed(time.Now().UTC().UnixNano()) + result := make([]byte, strLen) + for i := 0; i < strLen; i++ { + result[i] = chars[rand.Intn(len(chars))] + } + return string(result) +} + +// DeleteNamespace deletes the namespace +func (k *KubectlRunner) DeleteNamespace(namespace string) error { + _, err := k.RunCommand([]string{"delete", "namespace", namespace}) + return err +} + +// WaitForBusyboxRunning waits until busybox pod to be running +func WaitForBusyboxRunning(t *testing.T, namespace string) error { + client, err := commonutil.GetClient() + if err != nil { + return errors.Wrap(err, "getting kubernetes client") + } + selector := labels.SelectorFromSet(labels.Set(map[string]string{"integration-test": "busybox"})) + return commonutil.WaitForPodsWithLabelRunning(client, namespace, selector) +} + +// WaitForIngressControllerRunning waits until ingress controller pod to be running +func WaitForIngressControllerRunning(t *testing.T) error { + client, err := commonutil.GetClient() + if err != nil { + return errors.Wrap(err, "getting kubernetes client") + } + + if err := commonutil.WaitForDeploymentToStabilize(client, "kube-system", "nginx-ingress-controller", time.Minute*10); err != nil { + return errors.Wrap(err, "waiting for ingress-controller deployment to stabilize") + } + + selector := labels.SelectorFromSet(labels.Set(map[string]string{"app.kubernetes.io/name": "nginx-ingress-controller"})) + if err := commonutil.WaitForPodsWithLabelRunning(client, "kube-system", selector); err != nil { + return errors.Wrap(err, "waiting for ingress-controller pods") + } + + return nil +} + +// WaitForGvisorControllerRunning waits for the gvisor controller pod to be running +func WaitForGvisorControllerRunning(t *testing.T) error { + client, err := commonutil.GetClient() + if err != nil { + return errors.Wrap(err, "getting kubernetes client") + } + + selector := labels.SelectorFromSet(labels.Set(map[string]string{"kubernetes.io/minikube-addons": "gvisor"})) + if err := commonutil.WaitForPodsWithLabelRunning(client, "kube-system", selector); err != nil { + return errors.Wrap(err, "waiting for gvisor controller pod to stabilize") + } + return nil +} + +// WaitForGvisorControllerDeleted waits for the gvisor controller pod to be deleted +func WaitForGvisorControllerDeleted() error { + client, err := commonutil.GetClient() + if err != nil { + return errors.Wrap(err, "getting kubernetes client") + } + + selector := labels.SelectorFromSet(labels.Set(map[string]string{"kubernetes.io/minikube-addons": "gvisor"})) + if err := commonutil.WaitForPodDelete(client, "kube-system", selector); err != nil { + return errors.Wrap(err, "waiting for gvisor controller pod deletion") + } + return nil +} + +// WaitForUntrustedNginxRunning waits for the untrusted nginx pod to start running +func WaitForUntrustedNginxRunning() error { + client, err := commonutil.GetClient() + if err != nil { + return errors.Wrap(err, "getting kubernetes client") + } + + selector := labels.SelectorFromSet(labels.Set(map[string]string{"run": "nginx"})) + if err := commonutil.WaitForPodsWithLabelRunning(client, "default", selector); err != nil { + return errors.Wrap(err, "waiting for nginx pods") + } + return nil +} + +// WaitForFailedCreatePodSandBoxEvent waits for a FailedCreatePodSandBox event to appear +func WaitForFailedCreatePodSandBoxEvent() error { + client, err := commonutil.GetClient() + if err != nil { + return errors.Wrap(err, "getting kubernetes client") + } + if err := commonutil.WaitForEvent(client, "default", "FailedCreatePodSandBox"); err != nil { + return errors.Wrap(err, "waiting for FailedCreatePodSandBox event") + } + return nil +} + +// WaitForNginxRunning waits for nginx service to be up +func WaitForNginxRunning(t *testing.T) error { + client, err := commonutil.GetClient() + + if err != nil { + return errors.Wrap(err, "getting kubernetes client") + } + + selector := labels.SelectorFromSet(labels.Set(map[string]string{"run": "nginx"})) + if err := commonutil.WaitForPodsWithLabelRunning(client, "default", selector); err != nil { + return errors.Wrap(err, "waiting for nginx pods") + } + + if err := commonutil.WaitForService(client, "default", "nginx", true, time.Millisecond*500, time.Minute*10); err != nil { + t.Errorf("Error waiting for nginx service to be up") + } + return nil +} + +// Retry tries the callback for a number of attempts, with a delay between attempts +func Retry(t *testing.T, callback func() error, d time.Duration, attempts int) (err error) { + for i := 0; i < attempts; i++ { + err = callback() + if err == nil { + return nil + } + time.Sleep(d) + } + return err +}