164 lines
4.4 KiB
Go
164 lines
4.4 KiB
Go
// Copyright 2014 The Go Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style
|
|
// license that can be found in the LICENSE file.
|
|
|
|
// Tests of internal functions and things with no better homes.
|
|
|
|
package http
|
|
|
|
import (
|
|
"bytes"
|
|
"internal/testenv"
|
|
"net/url"
|
|
"os/exec"
|
|
"reflect"
|
|
"testing"
|
|
"time"
|
|
)
|
|
|
|
func init() {
|
|
shutdownPollInterval = 5 * time.Millisecond
|
|
}
|
|
|
|
func TestForeachHeaderElement(t *testing.T) {
|
|
tests := []struct {
|
|
in string
|
|
want []string
|
|
}{
|
|
{"Foo", []string{"Foo"}},
|
|
{" Foo", []string{"Foo"}},
|
|
{"Foo ", []string{"Foo"}},
|
|
{" Foo ", []string{"Foo"}},
|
|
|
|
{"foo", []string{"foo"}},
|
|
{"anY-cAsE", []string{"anY-cAsE"}},
|
|
|
|
{"", nil},
|
|
{",,,, , ,, ,,, ,", nil},
|
|
|
|
{" Foo,Bar, Baz,lower,,Quux ", []string{"Foo", "Bar", "Baz", "lower", "Quux"}},
|
|
}
|
|
for _, tt := range tests {
|
|
var got []string
|
|
foreachHeaderElement(tt.in, func(v string) {
|
|
got = append(got, v)
|
|
})
|
|
if !reflect.DeepEqual(got, tt.want) {
|
|
t.Errorf("foreachHeaderElement(%q) = %q; want %q", tt.in, got, tt.want)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestCleanHost(t *testing.T) {
|
|
tests := []struct {
|
|
in, want string
|
|
}{
|
|
{"www.google.com", "www.google.com"},
|
|
{"www.google.com foo", "www.google.com"},
|
|
{"www.google.com/foo", "www.google.com"},
|
|
{" first character is a space", ""},
|
|
{"[1::6]:8080", "[1::6]:8080"},
|
|
|
|
// Punycode:
|
|
{"гофер.рф/foo", "xn--c1ae0ajs.xn--p1ai"},
|
|
{"bücher.de", "xn--bcher-kva.de"},
|
|
{"bücher.de:8080", "xn--bcher-kva.de:8080"},
|
|
// Verify we convert to lowercase before punycode:
|
|
{"BÜCHER.de", "xn--bcher-kva.de"},
|
|
{"BÜCHER.de:8080", "xn--bcher-kva.de:8080"},
|
|
// Verify we normalize to NFC before punycode:
|
|
{"gophér.nfc", "xn--gophr-esa.nfc"}, // NFC input; no work needed
|
|
{"goph\u0065\u0301r.nfd", "xn--gophr-esa.nfd"}, // NFD input
|
|
}
|
|
for _, tt := range tests {
|
|
got := cleanHost(tt.in)
|
|
if tt.want != got {
|
|
t.Errorf("cleanHost(%q) = %q, want %q", tt.in, got, tt.want)
|
|
}
|
|
}
|
|
}
|
|
|
|
// Test that cmd/go doesn't link in the HTTP server.
|
|
//
|
|
// This catches accidental dependencies between the HTTP transport and
|
|
// server code.
|
|
func TestCmdGoNoHTTPServer(t *testing.T) {
|
|
t.Parallel()
|
|
goBin := testenv.GoToolPath(t)
|
|
out, err := exec.Command(goBin, "tool", "nm", goBin).CombinedOutput()
|
|
if err != nil {
|
|
t.Fatalf("go tool nm: %v: %s", err, out)
|
|
}
|
|
wantSym := map[string]bool{
|
|
// Verify these exist: (sanity checking this test)
|
|
"net/http.(*Client).Get": true,
|
|
"net/http.(*Transport).RoundTrip": true,
|
|
|
|
// Verify these don't exist:
|
|
"net/http.http2Server": false,
|
|
"net/http.(*Server).Serve": false,
|
|
"net/http.(*ServeMux).ServeHTTP": false,
|
|
"net/http.DefaultServeMux": false,
|
|
}
|
|
for sym, want := range wantSym {
|
|
got := bytes.Contains(out, []byte(sym))
|
|
if !want && got {
|
|
t.Errorf("cmd/go unexpectedly links in HTTP server code; found symbol %q in cmd/go", sym)
|
|
}
|
|
if want && !got {
|
|
t.Errorf("expected to find symbol %q in cmd/go; not found", sym)
|
|
}
|
|
}
|
|
}
|
|
|
|
// Tests that the nethttpomithttp2 build tag doesn't rot too much,
|
|
// even if there's not a regular builder on it.
|
|
func TestOmitHTTP2(t *testing.T) {
|
|
if testing.Short() {
|
|
t.Skip("skipping in short mode")
|
|
}
|
|
t.Parallel()
|
|
goTool := testenv.GoToolPath(t)
|
|
out, err := exec.Command(goTool, "test", "-short", "-tags=nethttpomithttp2", "net/http").CombinedOutput()
|
|
if err != nil {
|
|
t.Fatalf("go test -short failed: %v, %s", err, out)
|
|
}
|
|
}
|
|
|
|
// Tests that the nethttpomithttp2 build tag at least type checks
|
|
// in short mode.
|
|
// The TestOmitHTTP2 test above actually runs tests (in long mode).
|
|
func TestOmitHTTP2Vet(t *testing.T) {
|
|
t.Parallel()
|
|
goTool := testenv.GoToolPath(t)
|
|
out, err := exec.Command(goTool, "vet", "-tags=nethttpomithttp2", "net/http").CombinedOutput()
|
|
if err != nil {
|
|
t.Fatalf("go vet failed: %v, %s", err, out)
|
|
}
|
|
}
|
|
|
|
var valuesCount int
|
|
|
|
func BenchmarkCopyValues(b *testing.B) {
|
|
b.ReportAllocs()
|
|
src := url.Values{
|
|
"a": {"1", "2", "3", "4", "5"},
|
|
"b": {"2", "2", "3", "4", "5"},
|
|
"c": {"3", "2", "3", "4", "5"},
|
|
"d": {"4", "2", "3", "4", "5"},
|
|
"e": {"1", "1", "2", "3", "4", "5", "6", "7", "abcdef", "l", "a", "b", "c", "d", "z"},
|
|
"j": {"1", "2"},
|
|
"m": nil,
|
|
}
|
|
for i := 0; i < b.N; i++ {
|
|
dst := url.Values{"a": {"b"}, "b": {"2"}, "c": {"3"}, "d": {"4"}, "j": nil, "m": {"x"}}
|
|
copyValues(dst, src)
|
|
if valuesCount = len(dst["a"]); valuesCount != 6 {
|
|
b.Fatalf(`%d items in dst["a"] but expected 6`, valuesCount)
|
|
}
|
|
}
|
|
if valuesCount == 0 {
|
|
b.Fatal("Benchmark wasn't run")
|
|
}
|
|
}
|