mirror of
				https://github.com/k3s-io/kubernetes.git
				synced 2025-11-04 07:49:35 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			233 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			233 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
/*
 | 
						|
Copyright 2014 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 e2e
 | 
						|
 | 
						|
import (
 | 
						|
	"fmt"
 | 
						|
	"math"
 | 
						|
	"net/http"
 | 
						|
	"strings"
 | 
						|
	"sync"
 | 
						|
	"time"
 | 
						|
 | 
						|
	"github.com/GoogleCloudPlatform/kubernetes/pkg/api"
 | 
						|
	"github.com/GoogleCloudPlatform/kubernetes/pkg/api/testapi"
 | 
						|
	"github.com/GoogleCloudPlatform/kubernetes/pkg/client"
 | 
						|
	"github.com/GoogleCloudPlatform/kubernetes/pkg/fields"
 | 
						|
	"github.com/GoogleCloudPlatform/kubernetes/pkg/labels"
 | 
						|
	"github.com/GoogleCloudPlatform/kubernetes/pkg/util"
 | 
						|
 | 
						|
	. "github.com/onsi/ginkgo"
 | 
						|
	. "github.com/onsi/gomega"
 | 
						|
)
 | 
						|
 | 
						|
var _ = Describe("Proxy", func() {
 | 
						|
	version := testapi.Version()
 | 
						|
	Context("version "+version, func() { proxyContext(version) })
 | 
						|
})
 | 
						|
 | 
						|
const (
 | 
						|
	// Try all the proxy tests this many times (to catch even rare flakes).
 | 
						|
	proxyAttempts = 20
 | 
						|
	// Only print this many characters of the response (to keep the logs
 | 
						|
	// legible).
 | 
						|
	maxDisplayBodyLen = 100
 | 
						|
)
 | 
						|
 | 
						|
func proxyContext(version string) {
 | 
						|
	f := NewFramework("proxy")
 | 
						|
	prefix := "/api/" + version
 | 
						|
 | 
						|
	It("should proxy logs on node with explicit kubelet port", func() { nodeProxyTest(f, version, ":10250/logs/") })
 | 
						|
 | 
						|
	It("should proxy logs on node", func() { nodeProxyTest(f, version, "/logs/") })
 | 
						|
 | 
						|
	It("should proxy to cadvisor", func() { nodeProxyTest(f, version, ":4194/containers/") })
 | 
						|
 | 
						|
	It("should proxy through a service and a pod", func() {
 | 
						|
		labels := map[string]string{"proxy-service-target": "true"}
 | 
						|
		service, err := f.Client.Services(f.Namespace.Name).Create(&api.Service{
 | 
						|
			ObjectMeta: api.ObjectMeta{
 | 
						|
				GenerateName: "proxy-service-",
 | 
						|
			},
 | 
						|
			Spec: api.ServiceSpec{
 | 
						|
				Selector: labels,
 | 
						|
				Ports: []api.ServicePort{
 | 
						|
					{
 | 
						|
						Name:       "portname1",
 | 
						|
						Port:       80,
 | 
						|
						TargetPort: util.NewIntOrStringFromString("dest1"),
 | 
						|
					},
 | 
						|
					{
 | 
						|
						Name:       "portname2",
 | 
						|
						Port:       81,
 | 
						|
						TargetPort: util.NewIntOrStringFromInt(162),
 | 
						|
					},
 | 
						|
				},
 | 
						|
			},
 | 
						|
		})
 | 
						|
		Expect(err).NotTo(HaveOccurred())
 | 
						|
		defer func(name string) {
 | 
						|
			err := f.Client.Services(f.Namespace.Name).Delete(name)
 | 
						|
			if err != nil {
 | 
						|
				Logf("Failed deleting service %v: %v", name, err)
 | 
						|
			}
 | 
						|
		}(service.Name)
 | 
						|
 | 
						|
		// Make an RC with a single pod.
 | 
						|
		pods := []*api.Pod{}
 | 
						|
		cfg := RCConfig{
 | 
						|
			Client:       f.Client,
 | 
						|
			Image:        "gcr.io/google_containers/porter:59ad46ed2c56ba50fa7f1dc176c07c37",
 | 
						|
			Name:         service.Name,
 | 
						|
			Namespace:    f.Namespace.Name,
 | 
						|
			Replicas:     1,
 | 
						|
			PollInterval: time.Second,
 | 
						|
			Env: map[string]string{
 | 
						|
				"SERVE_PORT_80":  `<a href="/rewriteme">test</a>`,
 | 
						|
				"SERVE_PORT_160": "foo",
 | 
						|
				"SERVE_PORT_162": "bar",
 | 
						|
			},
 | 
						|
			Ports: map[string]int{
 | 
						|
				"dest1": 160,
 | 
						|
				"dest2": 162,
 | 
						|
			},
 | 
						|
			Labels:      labels,
 | 
						|
			CreatedPods: &pods,
 | 
						|
		}
 | 
						|
		Expect(RunRC(cfg)).NotTo(HaveOccurred())
 | 
						|
		defer DeleteRC(f.Client, f.Namespace.Name, cfg.Name)
 | 
						|
 | 
						|
		Expect(f.WaitForAnEndpoint(service.Name)).NotTo(HaveOccurred())
 | 
						|
 | 
						|
		// Try proxying through the service and directly to through the pod.
 | 
						|
		svcPrefix := prefix + "/proxy/namespaces/" + f.Namespace.Name + "/services/" + service.Name
 | 
						|
		podPrefix := prefix + "/proxy/namespaces/" + f.Namespace.Name + "/pods/" + pods[0].Name
 | 
						|
		expectations := map[string]string{
 | 
						|
			svcPrefix + ":portname1/": "foo",
 | 
						|
			svcPrefix + ":portname2/": "bar",
 | 
						|
			podPrefix + ":80/":        `<a href="` + podPrefix + `:80/rewriteme">test</a>`,
 | 
						|
			podPrefix + ":160/":       "foo",
 | 
						|
			podPrefix + ":162/":       "bar",
 | 
						|
			// TODO: below entries don't work, but I believe we should make them work.
 | 
						|
			// svcPrefix + ":80": "foo",
 | 
						|
			// svcPrefix + ":81": "bar",
 | 
						|
			// podPrefix + ":dest1": "foo",
 | 
						|
			// podPrefix + ":dest2": "bar",
 | 
						|
		}
 | 
						|
 | 
						|
		wg := sync.WaitGroup{}
 | 
						|
		errors := []string{}
 | 
						|
		errLock := sync.Mutex{}
 | 
						|
		recordError := func(s string) {
 | 
						|
			errLock.Lock()
 | 
						|
			defer errLock.Unlock()
 | 
						|
			errors = append(errors, s)
 | 
						|
		}
 | 
						|
		for i := 0; i < proxyAttempts; i++ {
 | 
						|
			for path, val := range expectations {
 | 
						|
				wg.Add(1)
 | 
						|
				go func(i int, path, val string) {
 | 
						|
					defer wg.Done()
 | 
						|
					body, status, d, err := doProxy(f, path)
 | 
						|
					if err != nil {
 | 
						|
						recordError(fmt.Sprintf("%v: path %v gave error: %v", i, path, err))
 | 
						|
						return
 | 
						|
					}
 | 
						|
					if status != http.StatusOK {
 | 
						|
						recordError(fmt.Sprintf("%v: path %v gave status: %v", i, path, status))
 | 
						|
					}
 | 
						|
					if e, a := val, string(body); e != a {
 | 
						|
						recordError(fmt.Sprintf("%v: path %v: wanted %v, got %v", i, path, e, a))
 | 
						|
					}
 | 
						|
					if d > 15*time.Second {
 | 
						|
						recordError(fmt.Sprintf("%v: path %v took %v > 15s", i, path, d))
 | 
						|
					}
 | 
						|
				}(i, path, val)
 | 
						|
				time.Sleep(150 * time.Millisecond)
 | 
						|
			}
 | 
						|
		}
 | 
						|
		wg.Wait()
 | 
						|
 | 
						|
		if len(errors) != 0 {
 | 
						|
			Fail(strings.Join(errors, "\n"))
 | 
						|
		}
 | 
						|
	})
 | 
						|
}
 | 
						|
 | 
						|
func doProxy(f *Framework, path string) (body []byte, statusCode int, d time.Duration, err error) {
 | 
						|
	// About all of the proxy accesses in this file:
 | 
						|
	// * AbsPath is used because it preserves the trailing '/'.
 | 
						|
	// * Do().Raw() is used (instead of DoRaw()) because it will turn an
 | 
						|
	//   error from apiserver proxy into an actual error, and there is no
 | 
						|
	//   chance of the things we are talking to being confused for an error
 | 
						|
	//   that apiserver would have emitted.
 | 
						|
	start := time.Now()
 | 
						|
	body, err = f.Client.Get().AbsPath(path).Do().StatusCode(&statusCode).Raw()
 | 
						|
	d = time.Since(start)
 | 
						|
	if len(body) > 0 {
 | 
						|
		Logf("%v: %s (%v; %v)", path, truncate(body, maxDisplayBodyLen), statusCode, d)
 | 
						|
	}
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func truncate(b []byte, maxLen int) []byte {
 | 
						|
	if len(b) <= maxLen-3 {
 | 
						|
		return b
 | 
						|
	}
 | 
						|
	b2 := append([]byte(nil), b[:maxLen-3]...)
 | 
						|
	b2 = append(b2, '.', '.', '.')
 | 
						|
	return b2
 | 
						|
}
 | 
						|
 | 
						|
func pickNode(c *client.Client) (string, error) {
 | 
						|
	nodes, err := c.Nodes().List(labels.Everything(), fields.Everything())
 | 
						|
	if err != nil {
 | 
						|
		return "", err
 | 
						|
	}
 | 
						|
	if len(nodes.Items) == 0 {
 | 
						|
		return "", fmt.Errorf("no nodes exist, can't test node proxy")
 | 
						|
	}
 | 
						|
	return nodes.Items[0].Name, nil
 | 
						|
}
 | 
						|
 | 
						|
func nodeProxyTest(f *Framework, version, nodeDest string) {
 | 
						|
	prefix := "/api/" + version
 | 
						|
	node, err := pickNode(f.Client)
 | 
						|
	Expect(err).NotTo(HaveOccurred())
 | 
						|
	// TODO: Change it to test whether all requests succeeded when requests
 | 
						|
	// not reaching Kubelet issue is debugged.
 | 
						|
	serviceUnavailableErrors := 0
 | 
						|
	for i := 0; i < proxyAttempts; i++ {
 | 
						|
		_, status, d, err := doProxy(f, prefix+"/proxy/nodes/"+node+nodeDest)
 | 
						|
		if status == http.StatusServiceUnavailable {
 | 
						|
			Logf("Failed proxying node logs due to service unavailable: %v", err)
 | 
						|
			time.Sleep(time.Second)
 | 
						|
			serviceUnavailableErrors++
 | 
						|
		} else {
 | 
						|
			Expect(err).NotTo(HaveOccurred())
 | 
						|
			Expect(status).To(Equal(http.StatusOK))
 | 
						|
			Expect(d).To(BeNumerically("<", 15*time.Second))
 | 
						|
		}
 | 
						|
	}
 | 
						|
	if serviceUnavailableErrors > 0 {
 | 
						|
		Logf("error: %d requests to proxy node logs failed", serviceUnavailableErrors)
 | 
						|
	}
 | 
						|
	maxFailures := int(math.Floor(0.1 * float64(proxyAttempts)))
 | 
						|
	Expect(serviceUnavailableErrors).To(BeNumerically("<", maxFailures))
 | 
						|
}
 |