github.com/reiver/go@v0.0.0-20150109200633-1d0c7792f172/src/net/http/httputil/reverseproxy_test.go (about)

     1  // Copyright 2011 The Go Authors. All rights reserved.
     2  // Use of this source code is governed by a BSD-style
     3  // license that can be found in the LICENSE file.
     4  
     5  // Reverse proxy tests.
     6  
     7  package httputil
     8  
     9  import (
    10  	"io/ioutil"
    11  	"log"
    12  	"net/http"
    13  	"net/http/httptest"
    14  	"net/url"
    15  	"strings"
    16  	"testing"
    17  	"time"
    18  )
    19  
    20  const fakeHopHeader = "X-Fake-Hop-Header-For-Test"
    21  
    22  func init() {
    23  	hopHeaders = append(hopHeaders, fakeHopHeader)
    24  }
    25  
    26  func TestReverseProxy(t *testing.T) {
    27  	const backendResponse = "I am the backend"
    28  	const backendStatus = 404
    29  	backend := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
    30  		if len(r.TransferEncoding) > 0 {
    31  			t.Errorf("backend got unexpected TransferEncoding: %v", r.TransferEncoding)
    32  		}
    33  		if r.Header.Get("X-Forwarded-For") == "" {
    34  			t.Errorf("didn't get X-Forwarded-For header")
    35  		}
    36  		if c := r.Header.Get("Connection"); c != "" {
    37  			t.Errorf("handler got Connection header value %q", c)
    38  		}
    39  		if c := r.Header.Get("Upgrade"); c != "" {
    40  			t.Errorf("handler got Upgrade header value %q", c)
    41  		}
    42  		if g, e := r.Host, "some-name"; g != e {
    43  			t.Errorf("backend got Host header %q, want %q", g, e)
    44  		}
    45  		w.Header().Set("X-Foo", "bar")
    46  		w.Header().Set("Upgrade", "foo")
    47  		w.Header().Set(fakeHopHeader, "foo")
    48  		w.Header().Add("X-Multi-Value", "foo")
    49  		w.Header().Add("X-Multi-Value", "bar")
    50  		http.SetCookie(w, &http.Cookie{Name: "flavor", Value: "chocolateChip"})
    51  		w.WriteHeader(backendStatus)
    52  		w.Write([]byte(backendResponse))
    53  	}))
    54  	defer backend.Close()
    55  	backendURL, err := url.Parse(backend.URL)
    56  	if err != nil {
    57  		t.Fatal(err)
    58  	}
    59  	proxyHandler := NewSingleHostReverseProxy(backendURL)
    60  	frontend := httptest.NewServer(proxyHandler)
    61  	defer frontend.Close()
    62  
    63  	getReq, _ := http.NewRequest("GET", frontend.URL, nil)
    64  	getReq.Host = "some-name"
    65  	getReq.Header.Set("Connection", "close")
    66  	getReq.Header.Set("Upgrade", "foo")
    67  	getReq.Close = true
    68  	res, err := http.DefaultClient.Do(getReq)
    69  	if err != nil {
    70  		t.Fatalf("Get: %v", err)
    71  	}
    72  	if g, e := res.StatusCode, backendStatus; g != e {
    73  		t.Errorf("got res.StatusCode %d; expected %d", g, e)
    74  	}
    75  	if g, e := res.Header.Get("X-Foo"), "bar"; g != e {
    76  		t.Errorf("got X-Foo %q; expected %q", g, e)
    77  	}
    78  	if c := res.Header.Get(fakeHopHeader); c != "" {
    79  		t.Errorf("got %s header value %q", fakeHopHeader, c)
    80  	}
    81  	if g, e := len(res.Header["X-Multi-Value"]), 2; g != e {
    82  		t.Errorf("got %d X-Multi-Value header values; expected %d", g, e)
    83  	}
    84  	if g, e := len(res.Header["Set-Cookie"]), 1; g != e {
    85  		t.Fatalf("got %d SetCookies, want %d", g, e)
    86  	}
    87  	if cookie := res.Cookies()[0]; cookie.Name != "flavor" {
    88  		t.Errorf("unexpected cookie %q", cookie.Name)
    89  	}
    90  	bodyBytes, _ := ioutil.ReadAll(res.Body)
    91  	if g, e := string(bodyBytes), backendResponse; g != e {
    92  		t.Errorf("got body %q; expected %q", g, e)
    93  	}
    94  }
    95  
    96  func TestXForwardedFor(t *testing.T) {
    97  	const prevForwardedFor = "client ip"
    98  	const backendResponse = "I am the backend"
    99  	const backendStatus = 404
   100  	backend := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
   101  		if r.Header.Get("X-Forwarded-For") == "" {
   102  			t.Errorf("didn't get X-Forwarded-For header")
   103  		}
   104  		if !strings.Contains(r.Header.Get("X-Forwarded-For"), prevForwardedFor) {
   105  			t.Errorf("X-Forwarded-For didn't contain prior data")
   106  		}
   107  		w.WriteHeader(backendStatus)
   108  		w.Write([]byte(backendResponse))
   109  	}))
   110  	defer backend.Close()
   111  	backendURL, err := url.Parse(backend.URL)
   112  	if err != nil {
   113  		t.Fatal(err)
   114  	}
   115  	proxyHandler := NewSingleHostReverseProxy(backendURL)
   116  	frontend := httptest.NewServer(proxyHandler)
   117  	defer frontend.Close()
   118  
   119  	getReq, _ := http.NewRequest("GET", frontend.URL, nil)
   120  	getReq.Host = "some-name"
   121  	getReq.Header.Set("Connection", "close")
   122  	getReq.Header.Set("X-Forwarded-For", prevForwardedFor)
   123  	getReq.Close = true
   124  	res, err := http.DefaultClient.Do(getReq)
   125  	if err != nil {
   126  		t.Fatalf("Get: %v", err)
   127  	}
   128  	if g, e := res.StatusCode, backendStatus; g != e {
   129  		t.Errorf("got res.StatusCode %d; expected %d", g, e)
   130  	}
   131  	bodyBytes, _ := ioutil.ReadAll(res.Body)
   132  	if g, e := string(bodyBytes), backendResponse; g != e {
   133  		t.Errorf("got body %q; expected %q", g, e)
   134  	}
   135  }
   136  
   137  var proxyQueryTests = []struct {
   138  	baseSuffix string // suffix to add to backend URL
   139  	reqSuffix  string // suffix to add to frontend's request URL
   140  	want       string // what backend should see for final request URL (without ?)
   141  }{
   142  	{"", "", ""},
   143  	{"?sta=tic", "?us=er", "sta=tic&us=er"},
   144  	{"", "?us=er", "us=er"},
   145  	{"?sta=tic", "", "sta=tic"},
   146  }
   147  
   148  func TestReverseProxyQuery(t *testing.T) {
   149  	backend := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
   150  		w.Header().Set("X-Got-Query", r.URL.RawQuery)
   151  		w.Write([]byte("hi"))
   152  	}))
   153  	defer backend.Close()
   154  
   155  	for i, tt := range proxyQueryTests {
   156  		backendURL, err := url.Parse(backend.URL + tt.baseSuffix)
   157  		if err != nil {
   158  			t.Fatal(err)
   159  		}
   160  		frontend := httptest.NewServer(NewSingleHostReverseProxy(backendURL))
   161  		req, _ := http.NewRequest("GET", frontend.URL+tt.reqSuffix, nil)
   162  		req.Close = true
   163  		res, err := http.DefaultClient.Do(req)
   164  		if err != nil {
   165  			t.Fatalf("%d. Get: %v", i, err)
   166  		}
   167  		if g, e := res.Header.Get("X-Got-Query"), tt.want; g != e {
   168  			t.Errorf("%d. got query %q; expected %q", i, g, e)
   169  		}
   170  		res.Body.Close()
   171  		frontend.Close()
   172  	}
   173  }
   174  
   175  func TestReverseProxyFlushInterval(t *testing.T) {
   176  	const expected = "hi"
   177  	backend := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
   178  		w.Write([]byte(expected))
   179  	}))
   180  	defer backend.Close()
   181  
   182  	backendURL, err := url.Parse(backend.URL)
   183  	if err != nil {
   184  		t.Fatal(err)
   185  	}
   186  
   187  	proxyHandler := NewSingleHostReverseProxy(backendURL)
   188  	proxyHandler.FlushInterval = time.Microsecond
   189  
   190  	done := make(chan bool)
   191  	onExitFlushLoop = func() { done <- true }
   192  	defer func() { onExitFlushLoop = nil }()
   193  
   194  	frontend := httptest.NewServer(proxyHandler)
   195  	defer frontend.Close()
   196  
   197  	req, _ := http.NewRequest("GET", frontend.URL, nil)
   198  	req.Close = true
   199  	res, err := http.DefaultClient.Do(req)
   200  	if err != nil {
   201  		t.Fatalf("Get: %v", err)
   202  	}
   203  	defer res.Body.Close()
   204  	if bodyBytes, _ := ioutil.ReadAll(res.Body); string(bodyBytes) != expected {
   205  		t.Errorf("got body %q; expected %q", bodyBytes, expected)
   206  	}
   207  
   208  	select {
   209  	case <-done:
   210  		// OK
   211  	case <-time.After(5 * time.Second):
   212  		t.Error("maxLatencyWriter flushLoop() never exited")
   213  	}
   214  }
   215  
   216  func TestReverseProxyCancellation(t *testing.T) {
   217  	const backendResponse = "I am the backend"
   218  
   219  	reqInFlight := make(chan struct{})
   220  	backend := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
   221  		close(reqInFlight)
   222  
   223  		select {
   224  		case <-time.After(10 * time.Second):
   225  			// Note: this should only happen in broken implementations, and the
   226  			// closenotify case should be instantaneous.
   227  			t.Log("Failed to close backend connection")
   228  			t.Fail()
   229  		case <-w.(http.CloseNotifier).CloseNotify():
   230  		}
   231  
   232  		w.WriteHeader(http.StatusOK)
   233  		w.Write([]byte(backendResponse))
   234  	}))
   235  
   236  	defer backend.Close()
   237  
   238  	backend.Config.ErrorLog = log.New(ioutil.Discard, "", 0)
   239  
   240  	backendURL, err := url.Parse(backend.URL)
   241  	if err != nil {
   242  		t.Fatal(err)
   243  	}
   244  
   245  	proxyHandler := NewSingleHostReverseProxy(backendURL)
   246  
   247  	// Discards errors of the form:
   248  	// http: proxy error: read tcp 127.0.0.1:44643: use of closed network connection
   249  	proxyHandler.ErrorLog = log.New(ioutil.Discard, "", 0)
   250  
   251  	frontend := httptest.NewServer(proxyHandler)
   252  	defer frontend.Close()
   253  
   254  	getReq, _ := http.NewRequest("GET", frontend.URL, nil)
   255  	go func() {
   256  		<-reqInFlight
   257  		http.DefaultTransport.(*http.Transport).CancelRequest(getReq)
   258  	}()
   259  	res, err := http.DefaultClient.Do(getReq)
   260  	if res != nil {
   261  		t.Fatal("Non-nil response")
   262  	}
   263  	if err == nil {
   264  		// This should be an error like:
   265  		// Get http://127.0.0.1:58079: read tcp 127.0.0.1:58079:
   266  		//    use of closed network connection
   267  		t.Fatal("DefaultClient.Do() returned nil error")
   268  	}
   269  }