Przejdź do głównej treści
eLearner.app
Moduł 10 · Lekcja 5 z 550/50 w kursie~18 min
Lekcje modułu (5/5)

Mini-projekt: serwer HTTP

Pakiet stdlib net/http wystarczy, aby zbudować gotowy do produkcji serwer HTTP: routing, procedury obsługi, oprogramowanie pośrednie, płynne zamykanie. Nie wymaga frameworka. W przypadku typowych interfejsów API REST jest to podstawa.

Witaj, świecie

Go
package main

import (
    "fmt"
    "net/http"
)

func main() {
    http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
        fmt.Fprintf(w, "hello %s", r.URL.Path[1:])
    })
    if err := http.ListenAndServe(":8080", nil); err != nil {
        panic(err)
    }
}

http.ListenAndServe blokuje i zwraca tylko w przypadku błędu. nil jako drugi argument = „użyj DefaultServeMux”.

Podpis procedury obsługi

Wszystkie programy obsługi mają ten sam podpis:

Go
func(w http.ResponseWriter, r *http.Request)
  • w http.ResponseWriter — dane wyjściowe: nagłówki, kod statusu, treść.
  • r *http.Request — dane wejściowe: metoda, adres URL, nagłówki, treść, kontekst.

Typowy wzór:

Go
func ping(w http.ResponseWriter, r *http.Request) {
    if r.Method != http.MethodGet {
        http.Error(w, "method not allowed", http.StatusMethodNotAllowed)
        return
    }
    w.Header().Set("Content-Type", "application/json")
    w.WriteHeader(http.StatusOK) // optional, defaults to 200
    fmt.Fprintln(w, `{"ok":true}`)
}

Routing za pomocą ServeMux

*http.ServeMux jest routerem podstawowym. Od Go 1.22 obsługuje wzorce z parametrami metod i ścieżek:

Go
mux := http.NewServeMux()
mux.HandleFunc("GET /users/{id}", func(w http.ResponseWriter, r *http.Request) {
    id := r.PathValue("id")
    fmt.Fprintln(w, "user", id)
})
mux.HandleFunc("POST /users", createUser)

http.ListenAndServe(":8080", mux)

W przypadku bardziej wyrafinowanych routerów (grupy, zintegrowane oprogramowanie pośrednie, wyrażenie regularne) dostępne są chi, gorilla/mux, gin, echo. Stdlib obejmuje 90% przypadków.

Oprogramowanie pośrednie jako dekorator

Oprogramowanie pośrednie to funkcja func(http.Handler) http.Handler:

Go
func logMW(next http.Handler) http.Handler {
    return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
        start := time.Now()
        next.ServeHTTP(w, r)
        log.Printf("%s %s %s", r.Method, r.URL.Path, time.Since(start))
    })
}

http.ListenAndServe(":8080", logMW(mux))

Aby je połączyć: logMW(authMW(rateLimitMW(mux))). Każde oprogramowanie pośrednie zdobi następne.

http.Server dla dokładnej kontroli

http.ListenAndServe(addr, h) to skrót. Aby uzyskać przekroczenia limitu czasu, TLS i łagodne zamknięcie, musisz utworzyć instancję http.Server:

Go
srv := &http.Server{
    Addr:              ":8080",
    Handler:           mux,
    ReadHeaderTimeout: 5 * time.Second,
    ReadTimeout:       30 * time.Second,
    WriteTimeout:      30 * time.Second,
    IdleTimeout:       2 * time.Minute,
}
log.Fatal(srv.ListenAndServe())

Płynne zamykanie z kontekstem

Gdy proces otrzyma SIGINT/SIGTERM, chcesz zakończyć żądania w locie przed zamknięciem:

Go
ctx, stop := signal.NotifyContext(context.Background(),
    os.Interrupt, syscall.SIGTERM)
defer stop()

srv := &http.Server{Addr: ":8080", Handler: mux}
go func() {
    if err := srv.ListenAndServe(); err != nil && err != http.ErrServerClosed {
        log.Fatal(err)
    }
}()

<-ctx.Done() // wait for the signal

shutdownCtx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
defer cancel()
if err := srv.Shutdown(shutdownCtx); err != nil {
    log.Println("shutdown:", err)
}

srv.Shutdown przestaje przyjmować nowe połączenia i czeka na zakończenie aktywnych (do upływu terminu).

Czytanie treści JSON

Go
type CreateUserReq struct {
    Name string `json:"name"`
}

func createUser(w http.ResponseWriter, r *http.Request) {
    var req CreateUserReq
    if err := json.NewDecoder(r.Body).Decode(&req); err != nil {
        http.Error(w, err.Error(), http.StatusBadRequest)
        return
    }
    // ... use req.Name
    w.WriteHeader(http.StatusCreated)
}

r.Body to io.ReadCloser: serwer zamyka go automatycznie, ale json.NewDecoder nie czyta poza pierwszym obiektem JSON (przydatne do sprawdzania poprawności danych wejściowych).

Ćwiczenia

Ćwiczenie#go.m10.l5.e1
Próby: 0Ładowanie...

Zarejestruj się w obsłudze hello sulla Route '/' za pomocą http.HandleFunc.

Ładowanie edytora...

Rozwiązanie dostępne po 3 próbach

Ćwiczenie#go.m10.l5.e2
Próby: 0Ładowanie...

Twój serwer su :8080 z http.ListenAndServe i oznacza błędy, które nie są zerowe.

Ładowanie edytora...

Rozwiązanie dostępne po 3 próbach

Quiz#go.m10.l5.e3
Gotowe

Jaka jest firma idiomatyczna obsługa protokołu HTTP w Go?

Go
func H(???) { ... }
Opcje odpowiedzi