如何使用httptest在Go中测试HTTP调用


76

我有以下代码:

package main

import (
    "encoding/json"
    "fmt"
    "io/ioutil"
    "log"
    "net/http"
    "time"
)

type twitterResult struct {
    Results []struct {
        Text     string `json:"text"`
        Ids      string `json:"id_str"`
        Name     string `json:"from_user_name"`
        Username string `json:"from_user"`
        UserId   string `json:"from_user_id_str"`
    }
}

var (
  twitterUrl = "http://search.twitter.com/search.json?q=%23UCL"
  pauseDuration = 5 * time.Second
)

func retrieveTweets(c chan<- *twitterResult) {
    for {
        resp, err := http.Get(twitterUrl)
        if err != nil {
            log.Fatal(err)
        }

        defer resp.Body.Close()
        body, err := ioutil.ReadAll(resp.Body)
        r := new(twitterResult) //or &twitterResult{} which returns *twitterResult
        err = json.Unmarshal(body, &r)
        if err != nil {
            log.Fatal(err)
        }
        c <- r
        time.Sleep(pauseDuration)
    }

}

func displayTweets(c chan *twitterResult) {
    tweets := <-c
    for _, v := range tweets.Results {
        fmt.Printf("%v:%v\n", v.Username, v.Text)
    }

}

func main() {
    c := make(chan *twitterResult)
    go retrieveTweets(c)
    for {
        displayTweets(c)
    }

}

我想为此编写一些测试,但是我不确定如何使用httptest包http://golang.org/pkg/net/http/httptest/会得到一些提示

我想出了这一点(从测试中毫不客气地复制了Go OAuth https://code.google.com/p/goauth2/source/browse/oauth/oauth_test.go):

var request = struct {
    path, query       string // request
    contenttype, body string // response
}{
    path:        "/search.json?",
    query:       "q=%23Kenya",
    contenttype: "application/json",
    body:        twitterResponse,
}

var (
    twitterResponse = `{ 'results': [{'text':'hello','id_str':'34455w4','from_user_name':'bob','from_user_id_str':'345424'}]}`
)

func TestRetrieveTweets(t *testing.T) {
    handler := func(w http.ResponseWriter, r *http.Request) {

        w.Header().Set("Content-Type", request.contenttype)
        io.WriteString(w, request.body)
    }

    server := httptest.NewServer(http.HandlerFunc(handler))
    defer server.Close()

    resp, err := http.Get(server.URL)
    if err != nil {
        t.Fatalf("Get: %v", err)
    }
    checkBody(t, resp, twitterResponse)
}

func checkBody(t *testing.T, r *http.Response, body string) {
    b, err := ioutil.ReadAll(r.Body)
    if err != nil {
        t.Error("reading reponse body: %v, want %q", err, body)
    }
    if g, w := string(b), body; g != w {
        t.Errorf("request body mismatch: got %q, want %q", g, w)
    }
}

Answers:


91

httptest执行两种类型的测试:响应和服务器

响应测试:

func TestHeader3D(t *testing.T) {
    resp := httptest.NewRecorder()

    uri := "/3D/header/?"
    path := "/home/test"
    unlno := "997225821"

    param := make(url.Values)
    param["param1"] = []string{path}
    param["param2"] = []string{unlno}

    req, err := http.NewRequest("GET", uri+param.Encode(), nil)
    if err != nil {
            t.Fatal(err)
    }

    http.DefaultServeMux.ServeHTTP(resp, req)
    if p, err := ioutil.ReadAll(resp.Body); err != nil {
            t.Fail()
    } else {
            if strings.Contains(string(p), "Error") {
                    t.Errorf("header response shouldn't return error: %s", p)
            } else if !strings.Contains(string(p), `expected result`) {
                    t.Errorf("header response doen't match:\n%s", p)
            }
    }
}

服务器测试(这是您需要使用的):

func TestIt(t *testing.T){
    ts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
        w.Header().Set("Content-Type", "application/json")
        fmt.Fprintln(w, `{"fake twitter json string"}`)
    }))
    defer ts.Close()

    twitterUrl = ts.URL
    c := make(chan *twitterResult)
    go retrieveTweets(c)

    tweet := <-c
    if tweet != expected1 {
        t.Fail()
    }
    tweet = <-c
    if tweet != expected2 {
        t.Fail()
    }
}

顺便说一句,您不需要传入r的指针,因为它已经是一个指针。

err = json.Unmarshal(body, r)

编辑:对于我的记录器测试,我可以这样使用我的http处理程序:

handler(resp, req)

但是我的原始代码没有使用默认的多路复用器(而是从Gorilla / mux中获取),并且我对多路复用器进行了一些包装,例如插入服务器日志记录,并添加了请求上下文(Gorilla / context),因此我必须从多路复用器开始并调用ServeHTTP


12
请添加完整的代码,例如导入所有软件包。新手在这里;)谢谢!
马萨尔娟

8
您可以使用goimports
030,

9

最初,此代码段位于GitHub Gist上,但是在尝试将该概念应用于我的一个项目时,我意识到我必须对主要代码进行重大修改,因此我决定通过使用docker和curl的集成测试来测试这些调用。


8

如果要测试程序,通常最好在编写测试时就考虑到这一点。例如,如果您将retrieveTweets函数的内部循环提取为如下所示:

func downloadTweets(tweetsUrl string) (*twitterResult, error)

您可以使用通过该httptest软件包设置的测试服务器的URL调用它,而不必担心睡眠或重复请求。


我可以模拟服务器吗?我认为弄湿一个测试服务器会否定测试的目的,如果我可以模拟出一个服务器,并且期望该响应与测试很吻合
jwesonga 2013年

4
httptest软件包是为测试设置小型HTTP服务器的基础结构。您可以以通常的方式实现请求处理程序,然后在该服务器而不是Twitter上运行代码。
James Henstridge

@JamesHenstridge谢谢你,很好的回答!您评论的内容比net / http / httptest的官方文档中的概述声明要有用得多
Rob Kielty
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.