mirror of
				https://github.com/k3s-io/kubernetes.git
				synced 2025-10-25 01:20:18 +00:00 
			
		
		
		
	https://github.com/golang/go/issues/12262 . See #19254 for more details. This change should be reverted when we upgrade to Go 1.6.
		
			
				
	
	
		
			263 lines
		
	
	
		
			9.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			263 lines
		
	
	
		
			9.7 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 proxy
 | |
| 
 | |
| import (
 | |
| 	"fmt"
 | |
| 	"io/ioutil"
 | |
| 	"net/http"
 | |
| 	"net/http/httptest"
 | |
| 	"net/url"
 | |
| 	"strings"
 | |
| 	"testing"
 | |
| )
 | |
| 
 | |
| func parseURLOrDie(inURL string) *url.URL {
 | |
| 	parsed, err := url.Parse(inURL)
 | |
| 	if err != nil {
 | |
| 		panic(err)
 | |
| 	}
 | |
| 	return parsed
 | |
| }
 | |
| 
 | |
| func TestProxyTransport(t *testing.T) {
 | |
| 	testTransport := &Transport{
 | |
| 		Scheme:      "http",
 | |
| 		Host:        "foo.com",
 | |
| 		PathPrepend: "/proxy/minion/minion1:10250",
 | |
| 	}
 | |
| 	testTransport2 := &Transport{
 | |
| 		Scheme:      "https",
 | |
| 		Host:        "foo.com",
 | |
| 		PathPrepend: "/proxy/minion/minion1:8080",
 | |
| 	}
 | |
| 	emptyHostTransport := &Transport{
 | |
| 		Scheme:      "https",
 | |
| 		PathPrepend: "/proxy/minion/minion1:10250",
 | |
| 	}
 | |
| 	emptySchemeTransport := &Transport{
 | |
| 		Host:        "foo.com",
 | |
| 		PathPrepend: "/proxy/minion/minion1:10250",
 | |
| 	}
 | |
| 	type Item struct {
 | |
| 		input        string
 | |
| 		sourceURL    string
 | |
| 		transport    *Transport
 | |
| 		output       string
 | |
| 		contentType  string
 | |
| 		forwardedURI string
 | |
| 		redirect     string
 | |
| 		redirectWant string
 | |
| 	}
 | |
| 
 | |
| 	table := map[string]Item{
 | |
| 		"normal": {
 | |
| 			input:        `<pre><a href="kubelet.log">kubelet.log</a><a href="/google.log">google.log</a></pre>`,
 | |
| 			sourceURL:    "http://myminion.com/logs/log.log",
 | |
| 			transport:    testTransport,
 | |
| 			output:       `<pre><a href="kubelet.log">kubelet.log</a><a href="http://foo.com/proxy/minion/minion1:10250/google.log">google.log</a></pre>`,
 | |
| 			contentType:  "text/html",
 | |
| 			forwardedURI: "/proxy/minion/minion1:10250/logs/log.log",
 | |
| 		},
 | |
| 		"full document": {
 | |
| 			input:        `<html><header></header><body><pre><a href="kubelet.log">kubelet.log</a><a href="/google.log">google.log</a></pre></body></html>`,
 | |
| 			sourceURL:    "http://myminion.com/logs/log.log",
 | |
| 			transport:    testTransport,
 | |
| 			output:       `<html><header></header><body><pre><a href="kubelet.log">kubelet.log</a><a href="http://foo.com/proxy/minion/minion1:10250/google.log">google.log</a></pre></body></html>`,
 | |
| 			contentType:  "text/html",
 | |
| 			forwardedURI: "/proxy/minion/minion1:10250/logs/log.log",
 | |
| 		},
 | |
| 		"trailing slash": {
 | |
| 			input:        `<pre><a href="kubelet.log">kubelet.log</a><a href="/google.log/">google.log</a></pre>`,
 | |
| 			sourceURL:    "http://myminion.com/logs/log.log",
 | |
| 			transport:    testTransport,
 | |
| 			output:       `<pre><a href="kubelet.log">kubelet.log</a><a href="http://foo.com/proxy/minion/minion1:10250/google.log/">google.log</a></pre>`,
 | |
| 			contentType:  "text/html",
 | |
| 			forwardedURI: "/proxy/minion/minion1:10250/logs/log.log",
 | |
| 		},
 | |
| 		"content-type charset": {
 | |
| 			input:        `<pre><a href="kubelet.log">kubelet.log</a><a href="/google.log">google.log</a></pre>`,
 | |
| 			sourceURL:    "http://myminion.com/logs/log.log",
 | |
| 			transport:    testTransport,
 | |
| 			output:       `<pre><a href="kubelet.log">kubelet.log</a><a href="http://foo.com/proxy/minion/minion1:10250/google.log">google.log</a></pre>`,
 | |
| 			contentType:  "text/html; charset=utf-8",
 | |
| 			forwardedURI: "/proxy/minion/minion1:10250/logs/log.log",
 | |
| 		},
 | |
| 		"content-type passthrough": {
 | |
| 			input:        `<pre><a href="kubelet.log">kubelet.log</a><a href="/google.log">google.log</a></pre>`,
 | |
| 			sourceURL:    "http://myminion.com/logs/log.log",
 | |
| 			transport:    testTransport,
 | |
| 			output:       `<pre><a href="kubelet.log">kubelet.log</a><a href="/google.log">google.log</a></pre>`,
 | |
| 			contentType:  "text/plain",
 | |
| 			forwardedURI: "/proxy/minion/minion1:10250/logs/log.log",
 | |
| 		},
 | |
| 		"subdir": {
 | |
| 			input:        `<a href="kubelet.log">kubelet.log</a><a href="/google.log">google.log</a>`,
 | |
| 			sourceURL:    "http://myminion.com/whatever/apt/somelog.log",
 | |
| 			transport:    testTransport2,
 | |
| 			output:       `<a href="kubelet.log">kubelet.log</a><a href="https://foo.com/proxy/minion/minion1:8080/google.log">google.log</a>`,
 | |
| 			contentType:  "text/html",
 | |
| 			forwardedURI: "/proxy/minion/minion1:8080/whatever/apt/somelog.log",
 | |
| 		},
 | |
| 		"image": {
 | |
| 			input:        `<pre><img src="kubernetes.jpg"/><img src="/kubernetes_abs.jpg"/></pre>`,
 | |
| 			sourceURL:    "http://myminion.com/",
 | |
| 			transport:    testTransport,
 | |
| 			output:       `<pre><img src="kubernetes.jpg"/><img src="http://foo.com/proxy/minion/minion1:10250/kubernetes_abs.jpg"/></pre>`,
 | |
| 			contentType:  "text/html",
 | |
| 			forwardedURI: "/proxy/minion/minion1:10250/",
 | |
| 		},
 | |
| 		"abs": {
 | |
| 			input:        `<script src="http://google.com/kubernetes.js"/>`,
 | |
| 			sourceURL:    "http://myminion.com/any/path/",
 | |
| 			transport:    testTransport,
 | |
| 			output:       `<script src="http://google.com/kubernetes.js"/>`,
 | |
| 			contentType:  "text/html",
 | |
| 			forwardedURI: "/proxy/minion/minion1:10250/any/path/",
 | |
| 		},
 | |
| 		"abs but same host": {
 | |
| 			input:        `<script src="http://myminion.com/kubernetes.js"/>`,
 | |
| 			sourceURL:    "http://myminion.com/any/path/",
 | |
| 			transport:    testTransport,
 | |
| 			output:       `<script src="http://foo.com/proxy/minion/minion1:10250/kubernetes.js"/>`,
 | |
| 			contentType:  "text/html",
 | |
| 			forwardedURI: "/proxy/minion/minion1:10250/any/path/",
 | |
| 		},
 | |
| 		"redirect rel": {
 | |
| 			sourceURL:    "http://myminion.com/redirect",
 | |
| 			transport:    testTransport,
 | |
| 			redirect:     "/redirected/target/",
 | |
| 			redirectWant: "http://foo.com/proxy/minion/minion1:10250/redirected/target/",
 | |
| 			forwardedURI: "/proxy/minion/minion1:10250/redirect",
 | |
| 		},
 | |
| 		"redirect abs same host": {
 | |
| 			sourceURL:    "http://myminion.com/redirect",
 | |
| 			transport:    testTransport,
 | |
| 			redirect:     "http://myminion.com/redirected/target/",
 | |
| 			redirectWant: "http://foo.com/proxy/minion/minion1:10250/redirected/target/",
 | |
| 			forwardedURI: "/proxy/minion/minion1:10250/redirect",
 | |
| 		},
 | |
| 		"redirect abs other host": {
 | |
| 			sourceURL:    "http://myminion.com/redirect",
 | |
| 			transport:    testTransport,
 | |
| 			redirect:     "http://example.com/redirected/target/",
 | |
| 			redirectWant: "http://example.com/redirected/target/",
 | |
| 			forwardedURI: "/proxy/minion/minion1:10250/redirect",
 | |
| 		},
 | |
| 		"source contains the redirect already": {
 | |
| 			input:        `<pre><a href="kubelet.log">kubelet.log</a><a href="http://foo.com/proxy/minion/minion1:10250/google.log">google.log</a></pre>`,
 | |
| 			sourceURL:    "http://foo.com/logs/log.log",
 | |
| 			transport:    testTransport,
 | |
| 			output:       `<pre><a href="kubelet.log">kubelet.log</a><a href="http://foo.com/proxy/minion/minion1:10250/google.log">google.log</a></pre>`,
 | |
| 			contentType:  "text/html",
 | |
| 			forwardedURI: "/proxy/minion/minion1:10250/logs/log.log",
 | |
| 		},
 | |
| 		"no host": {
 | |
| 			input:        "<html></html>",
 | |
| 			sourceURL:    "http://myminion.com/logs/log.log",
 | |
| 			transport:    emptyHostTransport,
 | |
| 			output:       "<html></html>",
 | |
| 			contentType:  "text/html",
 | |
| 			forwardedURI: "/proxy/minion/minion1:10250/logs/log.log",
 | |
| 		},
 | |
| 		"no scheme": {
 | |
| 			input:        "<html></html>",
 | |
| 			sourceURL:    "http://myminion.com/logs/log.log",
 | |
| 			transport:    emptySchemeTransport,
 | |
| 			output:       "<html></html>",
 | |
| 			contentType:  "text/html",
 | |
| 			forwardedURI: "/proxy/minion/minion1:10250/logs/log.log",
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	testItem := func(name string, item *Item) {
 | |
| 		server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
 | |
| 			// Check request headers.
 | |
| 			if got, want := r.Header.Get("X-Forwarded-Uri"), item.forwardedURI; got != want {
 | |
| 				t.Errorf("%v: X-Forwarded-Uri = %q, want %q", name, got, want)
 | |
| 			}
 | |
| 			if len(item.transport.Host) == 0 {
 | |
| 				_, present := r.Header["X-Forwarded-Host"]
 | |
| 				if present {
 | |
| 					t.Errorf("%v: X-Forwarded-Host header should not be present", name)
 | |
| 				}
 | |
| 			} else {
 | |
| 				if got, want := r.Header.Get("X-Forwarded-Host"), item.transport.Host; got != want {
 | |
| 					t.Errorf("%v: X-Forwarded-Host = %q, want %q", name, got, want)
 | |
| 				}
 | |
| 			}
 | |
| 			if len(item.transport.Scheme) == 0 {
 | |
| 				_, present := r.Header["X-Forwarded-Proto"]
 | |
| 				if present {
 | |
| 					t.Errorf("%v: X-Forwarded-Proto header should not be present", name)
 | |
| 				}
 | |
| 			} else {
 | |
| 				if got, want := r.Header.Get("X-Forwarded-Proto"), item.transport.Scheme; got != want {
 | |
| 					t.Errorf("%v: X-Forwarded-Proto = %q, want %q", name, got, want)
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			// Send response.
 | |
| 			if item.redirect != "" {
 | |
| 				http.Redirect(w, r, item.redirect, http.StatusMovedPermanently)
 | |
| 				return
 | |
| 			}
 | |
| 			w.Header().Set("Content-Type", item.contentType)
 | |
| 			fmt.Fprint(w, item.input)
 | |
| 		}))
 | |
| 		// TODO: Uncomment when fix #19254
 | |
| 		// defer server.Close()
 | |
| 
 | |
| 		// Replace source URL with our test server address.
 | |
| 		sourceURL := parseURLOrDie(item.sourceURL)
 | |
| 		serverURL := parseURLOrDie(server.URL)
 | |
| 		item.input = strings.Replace(item.input, sourceURL.Host, serverURL.Host, -1)
 | |
| 		item.redirect = strings.Replace(item.redirect, sourceURL.Host, serverURL.Host, -1)
 | |
| 		sourceURL.Host = serverURL.Host
 | |
| 
 | |
| 		req, err := http.NewRequest("GET", sourceURL.String(), nil)
 | |
| 		if err != nil {
 | |
| 			t.Errorf("%v: Unexpected error: %v", name, err)
 | |
| 			return
 | |
| 		}
 | |
| 		resp, err := item.transport.RoundTrip(req)
 | |
| 		if err != nil {
 | |
| 			t.Errorf("%v: Unexpected error: %v", name, err)
 | |
| 			return
 | |
| 		}
 | |
| 		if item.redirect != "" {
 | |
| 			// Check that redirect URLs get rewritten properly.
 | |
| 			if got, want := resp.Header.Get("Location"), item.redirectWant; got != want {
 | |
| 				t.Errorf("%v: Location header = %q, want %q", name, got, want)
 | |
| 			}
 | |
| 			return
 | |
| 		}
 | |
| 		body, err := ioutil.ReadAll(resp.Body)
 | |
| 		if err != nil {
 | |
| 			t.Errorf("%v: Unexpected error: %v", name, err)
 | |
| 			return
 | |
| 		}
 | |
| 		if e, a := item.output, string(body); e != a {
 | |
| 			t.Errorf("%v: expected %v, but got %v", name, e, a)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	for name, item := range table {
 | |
| 		testItem(name, &item)
 | |
| 	}
 | |
| }
 |