测试 HTTP handler

在这节我们将学习如果用 Go 测试 HTTP handlers,这是一个用 Go 测试的特别例子。我们将已 www.go 的代码为基础,修改需要调整的地方。

新版本的 www.go 命名为 testWWW.go,并分为三部分展示。testWWW.go 的第一部分代码如下:

package main

import (
    "fmt"
    "net/http"
    "os"
)

func CheckStatusOK(w http.ResponseWriter, r *http.Request){
    w.WriteHeader(http.StatusOK)
    fmt.Fprintf(w, `Fine!`)
}

testWWW.go 的第二部分代码如下:

func StatusNotFound(w http.ResponseWriter, r *http.Request) {
    fmt.Fprintf(w, "Serving: %s\n", r.URL.Path)
    fmt.Printf("Served: %s\n", r.Host)
}

func MyHandler(w http.ResponseWriter, r *http.Request) {
    fmt.Fprintf(w, "Serving: %s\n", r.URL.Path)
    fmt.Printf("Served: %s\n", r.Host)
}

testWWW.go 余下代码如下:

func main() {
    PORT := ":8001"
    arguments := os.Args
    if len(arguments) == 1 {
        fmt.Println("Using default port number:", PORT)
    } else {
        PORT = ":" + arguments[1]
    }

    http.HandleFunc("/CheckStatusOK", CheckStatusOK)
    http.HandleFunc("/StatusNotFound", StatusNotFound)
    http.HandleFunc("/", MyHandler)

    err := http.ListenAndServe(PORT, nil)
    if err != nil {
        fmt.Println(err)
        return
    }
}

我们现在需要开始测试 testWWW.go 了,为此我们应该创建一个 testWWW_test.go 文件。这个文件的内容由四部分展示。

testWWW_test.go 的第一部分包容如下代码:

package main

import (
    "fmt"
    "net/http"
    "net/http/httptest"
    "testing"
)

注意,为了用 Go 测试 web 应用,您需要引入 net/http/httptest 标准包。

testWWW_tesst.go 的第二部分如下:

func TestCheckStatusOK(t *testing.T) {
    req, err := http.NewRequest("GET", "/CheckStatusOK", nil)
    if err != nil {
        fmt.Println(err)
        return
    }

    rr := httptest.NewRecorder()
    handler := http.HandlerFunc(CheckStatusOK)
    handler.ServeHTTP(rr, req)

httptest.NewRecorder() 函数返回一个 httptest.ResponseRecorder 对象,它被用于记录 HTTP 响应。

testWWW_test.go 的第三部分如下:

    status := rr.Code
    if status != http.StatusOK {
        t.Errorf("handler returned %v", status)
    }

    expect := `Fine!`
    if rr.Body.String() != expect {
        t.Errorf("handler returned %v", rr.Body.String())
    }
}

您首先检查返回码是所期望的,然后再验证返回消息体也是您期望的。

testWWW_test.go 的余下代码如下:

func TestStatusNotFound(t *testing.T) {
    req, err := http.NewRequest("GET", "/StatusNotFound", nil)
    if err != nil {
        fmt.Println(err)
        return
    }

    rr := httptest.NewRecorder()
    handler := http.HandlerFunc(StatusNotFound)
    handler.ServeHTTP(rr, req)

    status := rr.Code
    if status != http.StatusNotFound {
        t.Errorf("handler returned %v", status)
    }
}

这个测试函数验证 main 包中的 StatusNotFound() 函数是否如期运行。

执行 testWWW_test.go 的这俩个测试函数将产生如下输出:

$ go test testWWW.go testWWW_test.go -v
=== RUN     TestCheckStatusOK
--- PASS:   TestCheckStatusOK (0.00s)
=== RUN     TestStatusNotFound
--- PASS:   TestStatusNotFound (0.00s)
PASS
ok      command-line-arguments  (cached)

results matching ""

    No results matching ""