// Copyright 2013 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. package http_test import ( "bufio" "bytes" "crypto/tls" "crypto/x509" "fmt" "io" . "net/http" "net/http/httptest" "strings" "testing" ) func TestNextProtoUpgrade(t *testing.T) { setParallel(t) defer afterTest(t) ts := httptest.NewUnstartedServer(HandlerFunc(func(w ResponseWriter, r *Request) { fmt.Fprintf(w, "path=%s,proto=", r.URL.Path) if r.TLS != nil { w.Write([]byte(r.TLS.NegotiatedProtocol)) } if r.RemoteAddr == "" { t.Error("request with no RemoteAddr") } if r.Body == nil { t.Errorf("request with nil Body") } })) ts.TLS = &tls.Config{ NextProtos: []string{"unhandled-proto", "tls-0.9"}, } ts.Config.TLSNextProto = map[string]func(*Server, *tls.Conn, Handler){ "tls-0.9": handleTLSProtocol09, } ts.StartTLS() defer ts.Close() // Normal request, without NPN. { c := ts.Client() res, err := c.Get(ts.URL) if err != nil { t.Fatal(err) } body, err := io.ReadAll(res.Body) if err != nil { t.Fatal(err) } if want := "path=/,proto="; string(body) != want { t.Errorf("plain request = %q; want %q", body, want) } } // Request to an advertised but unhandled NPN protocol. // Server will hang up. { certPool := x509.NewCertPool() certPool.AddCert(ts.Certificate()) tr := &Transport{ TLSClientConfig: &tls.Config{ RootCAs: certPool, NextProtos: []string{"unhandled-proto"}, }, } defer tr.CloseIdleConnections() c := &Client{ Transport: tr, } res, err := c.Get(ts.URL) if err == nil { defer res.Body.Close() var buf bytes.Buffer res.Write(&buf) t.Errorf("expected error on unhandled-proto request; got: %s", buf.Bytes()) } } // Request using the "tls-0.9" protocol, which we register here. // It is HTTP/0.9 over TLS. { c := ts.Client() tlsConfig := c.Transport.(*Transport).TLSClientConfig tlsConfig.NextProtos = []string{"tls-0.9"} conn, err := tls.Dial("tcp", ts.Listener.Addr().String(), tlsConfig) if err != nil { t.Fatal(err) } conn.Write([]byte("GET /foo\n")) body, err := io.ReadAll(conn) if err != nil { t.Fatal(err) } if want := "path=/foo,proto=tls-0.9"; string(body) != want { t.Errorf("plain request = %q; want %q", body, want) } } } // handleTLSProtocol09 implements the HTTP/0.9 protocol over TLS, for the // TestNextProtoUpgrade test. func handleTLSProtocol09(srv *Server, conn *tls.Conn, h Handler) { br := bufio.NewReader(conn) line, err := br.ReadString('\n') if err != nil { return } line = strings.TrimSpace(line) path := strings.TrimPrefix(line, "GET ") if path == line { return } req, _ := NewRequest("GET", path, nil) req.Proto = "HTTP/0.9" req.ProtoMajor = 0 req.ProtoMinor = 9 rw := &http09Writer{conn, make(Header)} h.ServeHTTP(rw, req) } type http09Writer struct { io.Writer h Header } func (w http09Writer) Header() Header { return w.h } func (w http09Writer) WriteHeader(int) {} // no headers