我在golang中为http处理程序编写单元测试。在查看代码覆盖率报告时,我遇到了以下问题:从请求中读取请求正文时,ioutil.ReadAll
可能会返回我需要处理的错误。然而,当我为我的处理程序编写单元测试时,我不知道如何以一种会触发这种错误的方式向我的处理程序发送请求(内容的过早结束似乎不会产生这样的错误但会产生错误解构身体)。这就是我想要做的事情:
package demo
import (
"bytes"
"io/ioutil"
"net/http"
"net/http/httptest"
"testing"
)
func HandlePostRequest(w http.ResponseWriter, r *http.Request) {
body, bytesErr := ioutil.ReadAll(r.Body)
if bytesErr != nil {
// intricate logic goes here, how can i test it?
http.Error(w, http.StatusText(http.StatusBadRequest), http.StatusBadRequest)
return
}
defer r.Body.Close()
// continue...
}
func TestHandlePostRequest(t *testing.T) {
ts := httptest.NewServer(http.HandlerFunc(HandlePostRequest))
data, _ := ioutil.ReadFile("testdata/fixture.json")
res, err := http.Post(ts.URL, "application/json", bytes.NewReader(data))
// continue...
}
如何编写HandlePostRequest
的测试用例,其中也包含bytesErr
不是nil
的情况?
答案 0 :(得分:16)
您可以创建并使用由您伪造的http.Request
,在阅读其正文时故意返回错误。你不一定需要一个全新的请求,一个有缺陷的身体就足够了(这是一个io.ReadCloser
)。
使用httptest.NewRequest()
函数实现最简单,您可以将io.Reader
值传递给Go Playground值(包装为io.ReadCloser
)作为请求体。
这是一个示例io.Reader
,它在尝试从中读取时故意返回错误:
type errReader int
func (errReader) Read(p []byte) (n int, err error) {
return 0, errors.New("test error")
}
将涵盖您的错误案例的示例:
func HandlePostRequest(w http.ResponseWriter, r *http.Request) {
defer r.Body.Close()
body, err := ioutil.ReadAll(r.Body)
if err != nil {
fmt.Printf("Error reading the body: %v\n", err)
return
}
fmt.Printf("No error, body: %s\n", body)
}
func main() {
testRequest := httptest.NewRequest(http.MethodPost, "/something", errReader(0))
HandlePostRequest(nil, testRequest)
}
输出(在{{3}}上尝试):
Error reading the body: test error