656297e1fe
Reviewed-on: https://go-review.googlesource.com/c/gofrontend/+/194698 From-SVN: r275691
194 lines
5.3 KiB
Go
194 lines
5.3 KiB
Go
// Copyright 2012 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 (
|
|
"context"
|
|
"fmt"
|
|
"io"
|
|
"io/ioutil"
|
|
"log"
|
|
"net/http"
|
|
"os"
|
|
"os/signal"
|
|
)
|
|
|
|
func ExampleHijacker() {
|
|
http.HandleFunc("/hijack", func(w http.ResponseWriter, r *http.Request) {
|
|
hj, ok := w.(http.Hijacker)
|
|
if !ok {
|
|
http.Error(w, "webserver doesn't support hijacking", http.StatusInternalServerError)
|
|
return
|
|
}
|
|
conn, bufrw, err := hj.Hijack()
|
|
if err != nil {
|
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
|
return
|
|
}
|
|
// Don't forget to close the connection:
|
|
defer conn.Close()
|
|
bufrw.WriteString("Now we're speaking raw TCP. Say hi: ")
|
|
bufrw.Flush()
|
|
s, err := bufrw.ReadString('\n')
|
|
if err != nil {
|
|
log.Printf("error reading string: %v", err)
|
|
return
|
|
}
|
|
fmt.Fprintf(bufrw, "You said: %q\nBye.\n", s)
|
|
bufrw.Flush()
|
|
})
|
|
}
|
|
|
|
func ExampleGet() {
|
|
res, err := http.Get("http://www.google.com/robots.txt")
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
robots, err := ioutil.ReadAll(res.Body)
|
|
res.Body.Close()
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
fmt.Printf("%s", robots)
|
|
}
|
|
|
|
func ExampleFileServer() {
|
|
// Simple static webserver:
|
|
log.Fatal(http.ListenAndServe(":8080", http.FileServer(http.Dir("/usr/share/doc"))))
|
|
}
|
|
|
|
func ExampleFileServer_stripPrefix() {
|
|
// To serve a directory on disk (/tmp) under an alternate URL
|
|
// path (/tmpfiles/), use StripPrefix to modify the request
|
|
// URL's path before the FileServer sees it:
|
|
http.Handle("/tmpfiles/", http.StripPrefix("/tmpfiles/", http.FileServer(http.Dir("/tmp"))))
|
|
}
|
|
|
|
func ExampleStripPrefix() {
|
|
// To serve a directory on disk (/tmp) under an alternate URL
|
|
// path (/tmpfiles/), use StripPrefix to modify the request
|
|
// URL's path before the FileServer sees it:
|
|
http.Handle("/tmpfiles/", http.StripPrefix("/tmpfiles/", http.FileServer(http.Dir("/tmp"))))
|
|
}
|
|
|
|
type apiHandler struct{}
|
|
|
|
func (apiHandler) ServeHTTP(http.ResponseWriter, *http.Request) {}
|
|
|
|
func ExampleServeMux_Handle() {
|
|
mux := http.NewServeMux()
|
|
mux.Handle("/api/", apiHandler{})
|
|
mux.HandleFunc("/", func(w http.ResponseWriter, req *http.Request) {
|
|
// The "/" pattern matches everything, so we need to check
|
|
// that we're at the root here.
|
|
if req.URL.Path != "/" {
|
|
http.NotFound(w, req)
|
|
return
|
|
}
|
|
fmt.Fprintf(w, "Welcome to the home page!")
|
|
})
|
|
}
|
|
|
|
// HTTP Trailers are a set of key/value pairs like headers that come
|
|
// after the HTTP response, instead of before.
|
|
func ExampleResponseWriter_trailers() {
|
|
mux := http.NewServeMux()
|
|
mux.HandleFunc("/sendstrailers", func(w http.ResponseWriter, req *http.Request) {
|
|
// Before any call to WriteHeader or Write, declare
|
|
// the trailers you will set during the HTTP
|
|
// response. These three headers are actually sent in
|
|
// the trailer.
|
|
w.Header().Set("Trailer", "AtEnd1, AtEnd2")
|
|
w.Header().Add("Trailer", "AtEnd3")
|
|
|
|
w.Header().Set("Content-Type", "text/plain; charset=utf-8") // normal header
|
|
w.WriteHeader(http.StatusOK)
|
|
|
|
w.Header().Set("AtEnd1", "value 1")
|
|
io.WriteString(w, "This HTTP response has both headers before this text and trailers at the end.\n")
|
|
w.Header().Set("AtEnd2", "value 2")
|
|
w.Header().Set("AtEnd3", "value 3") // These will appear as trailers.
|
|
})
|
|
}
|
|
|
|
func ExampleServer_Shutdown() {
|
|
var srv http.Server
|
|
|
|
idleConnsClosed := make(chan struct{})
|
|
go func() {
|
|
sigint := make(chan os.Signal, 1)
|
|
signal.Notify(sigint, os.Interrupt)
|
|
<-sigint
|
|
|
|
// We received an interrupt signal, shut down.
|
|
if err := srv.Shutdown(context.Background()); err != nil {
|
|
// Error from closing listeners, or context timeout:
|
|
log.Printf("HTTP server Shutdown: %v", err)
|
|
}
|
|
close(idleConnsClosed)
|
|
}()
|
|
|
|
if err := srv.ListenAndServe(); err != http.ErrServerClosed {
|
|
// Error starting or closing listener:
|
|
log.Fatalf("HTTP server ListenAndServe: %v", err)
|
|
}
|
|
|
|
<-idleConnsClosed
|
|
}
|
|
|
|
func ExampleListenAndServeTLS() {
|
|
http.HandleFunc("/", func(w http.ResponseWriter, req *http.Request) {
|
|
io.WriteString(w, "Hello, TLS!\n")
|
|
})
|
|
|
|
// One can use generate_cert.go in crypto/tls to generate cert.pem and key.pem.
|
|
log.Printf("About to listen on 8443. Go to https://127.0.0.1:8443/")
|
|
err := http.ListenAndServeTLS(":8443", "cert.pem", "key.pem", nil)
|
|
log.Fatal(err)
|
|
}
|
|
|
|
func ExampleListenAndServe() {
|
|
// Hello world, the web server
|
|
|
|
helloHandler := func(w http.ResponseWriter, req *http.Request) {
|
|
io.WriteString(w, "Hello, world!\n")
|
|
}
|
|
|
|
http.HandleFunc("/hello", helloHandler)
|
|
log.Fatal(http.ListenAndServe(":8080", nil))
|
|
}
|
|
|
|
func ExampleHandleFunc() {
|
|
h1 := func(w http.ResponseWriter, _ *http.Request) {
|
|
io.WriteString(w, "Hello from a HandleFunc #1!\n")
|
|
}
|
|
h2 := func(w http.ResponseWriter, _ *http.Request) {
|
|
io.WriteString(w, "Hello from a HandleFunc #2!\n")
|
|
}
|
|
|
|
http.HandleFunc("/", h1)
|
|
http.HandleFunc("/endpoint", h2)
|
|
|
|
log.Fatal(http.ListenAndServe(":8080", nil))
|
|
}
|
|
|
|
func newPeopleHandler() http.Handler {
|
|
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
fmt.Fprintln(w, "This is the people handler.")
|
|
})
|
|
}
|
|
|
|
func ExampleNotFoundHandler() {
|
|
mux := http.NewServeMux()
|
|
|
|
// Create sample handler to returns 404
|
|
mux.Handle("/resources", http.NotFoundHandler())
|
|
|
|
// Create sample handler that returns 200
|
|
mux.Handle("/resources/people/", newPeopleHandler())
|
|
|
|
log.Fatal(http.ListenAndServe(":8080", mux))
|
|
}
|