Golang的REST API的单元测试

用不同的语言、不同的方法和设计模式构建RESTful api一直是一种趋势,就像学习过程中遇到的困难一样。这是因为代码中有很多抽象、启动项目的痛苦以及更多的原因。在此基础上,为实现的服务编写测试用例也是一件麻烦事。

Go为您提供了以非常简单,优雅和简洁的方式编写REST API的特权。除此之外,Go中的单元测试也非常简单,只需一个命令即可运行测试用例。

由于本文只是在Go中编写单元测试用例,我假设您将知道如何在Go中编写REST实现。

为了更深入地了解这个简单的赋值,您可以查看此示例

我已经附加了postman集合以便于导入,sql转储以及附加的自述文件,以便您可以开始使用简单的赋值。

所以让我们开始 -

如果我们考虑一个编写测试用例的例子会更好。假设我们想要一个Online Address Book创建地址簿的应用程序,其中包含以下字段 -

package main

type entry struct {
    ID           int    `json:"id,omitempty"`
    FirstName    string `json:"first_name,omitempty"`
    LastName     string `json:"last_name,omitempty"`
    EmailAddress string `json:"email_address,omitempty"`
    PhoneNumber  string `json:"phone_number,omitempty"`
}

Entry modal

我们将假设我们有端点GetEntriesGetEntryByIDCreateEntryUpdateEntryDeleteEntry

GetEntries -> "/entries" -> Method **GET**

GetEntryByID -> "/entry?id=1234" -> Method **GET**

CreateEntry -> "/entry" -> Method **POST**

UpdateEntry -> "/entry" -> Method **PUT**

DeleteEntry -> "/entry" -> Method **DELETE**


用于Go中的单元测试的包

  1. [testing](https://golang.org/pkg/testing/) - 这是一个内置的Golang包,用于实现和运行单元/自动化测试用例。它旨在使用go test带有可选参数的命令来接受要测试的go文件。
  2. [net/http/httptest](https://golang.org/pkg/net/http/httptest/) - 这也是一个内置的Golang包,它提供了进行HTTP测试的权限。在我们的例子中,我们希望记录来自端点的响应并进行相应的检查。

NOTE:避免使用断言。


写单元测试

在Go中编写单元测试用例可以在同一个包中,也可以在不同的包中。Go testing包有两个标准来标识测试用例。

  • 文件名应以_test。结尾。例如 -endpoints_test.go
  • 测试用例函数应该以Test开头。例如 -
func TestGetEntries(t *testing.T) { 
....
}

编写REST API端点的单元测试用例

让我们逐个测试每个端点,看看如何测试上面指定的示例中的所有端点例子,即GetEntriesGetEntryByIDGetEntryByIDNotFoundCreateEntryEditEntryDeleteEntry

让我们从编写以下测试用例开始 -

GetEntries测试案例 -

func TestGetEntries(t *testing.T) {
    req, err := http.NewRequest("GET", "/entries", nil)
    if err != nil {
        t.Fatal(err)
    }
    rr := httptest.NewRecorder()
    handler := http.HandlerFunc(GetEntries)
    handler.ServeHTTP(rr, req)
    if status := rr.Code; status != http.StatusOK {
        t.Errorf("handler returned wrong status code: got %v want %v",
            status, http.StatusOK)
    }

    // Check the response body is what we expect.
    expected := `[{"id":1,"first_name":"Krish","last_name":"Bhanushali","email_address":"krishsb@g.com","phone_number":"0987654321"},{"id":2,"first_name":"xyz","last_name":"pqr","email_address":"xyz@pqr.com","phone_number":"1234567890"},{"id":6,"first_name":"FirstNameSample","last_name":"LastNameSample","email_address":"lr@gmail.com","phone_number":"1111111111"}]`
    if rr.Body.String() != expected {
        t.Errorf("handler returned unexpected body: got %v want %v",
            rr.Body.String(), expected)
    }
}

TestGetEntries()

注意:上面指定的Github存储库不包含每个测试用例的单独go文件。我在一个go文件中指定了所有测试用例。

让我们get_entries_test.go一行一行 -

  • 第1-号行。请在此处注明功能名称。它首先Test使testing包识别出测试用例,而用例中的下一个单词也以大写字母开头。该函数有一个参数,它是指向testingT变量的指针,表示它是一个测试用例。
  • 第2行 - 这会向/entries端点创建新请求。
  • 第6行 - 创建一个新的记录器来记录entries端点收到的响应。
  • 8号线 - 用记录仪和请求命中终点。
  • 第9行 - 检查响应是否为200 OK。
  • 第10行 - 将错误标记发送为测试失败。
  • 第15行 - 是端点的预期输出。
  • 第16行 - 检查响应是否等于预期。

GetEntryByID测试用例 -

func TestGetEntryByID(t *testing.T) {

    req, err := http.NewRequest("GET", "/entry", nil)
    if err != nil {
        t.Fatal(err)
    }
    q := req.URL.Query()
    q.Add("id", "1")
    req.URL.RawQuery = q.Encode()
    rr := httptest.NewRecorder()
    handler := http.HandlerFunc(GetEntryByID)
    handler.ServeHTTP(rr, req)
    if status := rr.Code; status != http.StatusOK {
        t.Errorf("handler returned wrong status code: got %v want %v",
            status, http.StatusOK)
    }

    // Check the response body is what we expect.
    expected := `{"id":1,"first_name":"Krish","last_name":"Bhanushali","email_address":"krishsb2405@gmail.com","phone_number":"0987654321"}`
    if rr.Body.String() != expected {
        t.Errorf("handler returned unexpected body: got %v want %v",
            rr.Body.String(), expected)
    }
}

TestGetEntryByID()

GetEntryByIDNotFound测试用例 -

func TestGetEntryByIDNotFound(t *testing.T) {
    req, err := http.NewRequest("GET", "/entry", nil)
    if err != nil {
        t.Fatal(err)
    }
    q := req.URL.Query()
    q.Add("id", "123")
    req.URL.RawQuery = q.Encode()
    rr := httptest.NewRecorder()
    handler := http.HandlerFunc(GetEntryByID)
    handler.ServeHTTP(rr, req)
    if status := rr.Code; status == http.StatusOK {
        t.Errorf("handler returned wrong status code: got %v want %v",
            status, http.StatusBadRequest)
    }
}

TestGetEntryByIDNotFound()

CreateEntry测试用例 -

func TestCreateEntry(t *testing.T) {

    var jsonStr = []byte(`{"id":4,"first_name":"xyz","last_name":"pqr","email_address":"xyz@pqr.com","phone_number":"1234567890"}`)

    req, err := http.NewRequest("POST", "/entry", bytes.NewBuffer(jsonStr))
    if err != nil {
        t.Fatal(err)
    }
    req.Header.Set("Content-Type", "application/json")
    rr := httptest.NewRecorder()
    handler := http.HandlerFunc(CreateEntry)
    handler.ServeHTTP(rr, req)
    if status := rr.Code; status != http.StatusOK {
        t.Errorf("handler returned wrong status code: got %v want %v",
            status, http.StatusOK)
    }
    expected := `{"id":4,"first_name":"xyz","last_name":"pqr","email_address":"xyz@pqr.com","phone_number":"1234567890"}`
    if rr.Body.String() != expected {
        t.Errorf("handler returned unexpected body: got %v want %v",
            rr.Body.String(), expected)
    }
}

TestCreateEntry()

让我们create_entry_test.go一行一行地进行 -

第1-8行与上面的描述相同,其中输入是jsonStr条目对象的JSON字符串,我们使用新方法创建新请求POST

第9行 - 设置标题为Content-Typetoapplication/json

第9-22行 - 同样,发出请求并将响应与预期进行比较的情况相同。如果发出的请求不是200 OK或实际响应不等于预期响应,则测试用例失败。

EditEntry测试案例 -

func TestEditEntry(t *testing.T) {

    var jsonStr = []byte(`{"id":4,"first_name":"xyz change","last_name":"pqr","email_address":"xyz@pqr.com","phone_number":"1234567890"}`)

    req, err := http.NewRequest("PUT", "/entry", bytes.NewBuffer(jsonStr))
    if err != nil {
        t.Fatal(err)
    }
    req.Header.Set("Content-Type", "application/json")
    rr := httptest.NewRecorder()
    handler := http.HandlerFunc(UpdateEntry)
    handler.ServeHTTP(rr, req)
    if status := rr.Code; status != http.StatusOK {
        t.Errorf("handler returned wrong status code: got %v want %v",
            status, http.StatusOK)
    }
    expected := `{"id":4,"first_name":"xyz change","last_name":"pqr","email_address":"xyz@pqr.com","phone_number":"1234567890"}`
    if rr.Body.String() != expected {
        t.Errorf("handler returned unexpected body: got %v want %v",
            rr.Body.String(), expected)
    }
}

TestEditEntry()

EditEntry测试用例是相同的CreateEntry测试情况下,除了所述请求的方法是PUTEditEntry

DeleteEntry测试用例 -

func TestDeleteEntry(t *testing.T) {
    req, err := http.NewRequest("DELETE", "/entry", nil)
    if err != nil {
        t.Fatal(err)
    }
    q := req.URL.Query()
    q.Add("id", "4")
    req.URL.RawQuery = q.Encode()
    rr := httptest.NewRecorder()
    handler := http.HandlerFunc(DeleteEntry)
    handler.ServeHTTP(rr, req)
    if status := rr.Code; status != http.StatusOK {
        t.Errorf("handler returned wrong status code: got %v want %v",
            status, http.StatusOK)
    }
    expected := `{"id":4,"first_name":"xyz change","last_name":"pqr","email_address":"xyz@pqr.com","phone_number":"1234567890"}`
    if rr.Body.String() != expected {
        t.Errorf("handler returned unexpected body: got %v want %v",
            rr.Body.String(), expected)
    }
}

TestDeleteEntry()

DeleteEntry测试用例再次是相同的GetEntryByID测试情况下,除了所述请求的方法是DELETEDeleteEntry


运行测试用例

让我们首先启动服务器,以便我们的测试用例通过go run api.go entry.go考虑指定的示例来到达端点。

要在测试套件中运行所有测试用例,您可以执行以下操作 -

go test -v


运行测试套件

要运行一个测试用例,您只需使用以下操作 -

go test -v -run <Test Function Name>

运行一个测试用例,如:TestGetEntries

因此,我们可以在此得出结论,我们现在知道如何在Go中为RESTful API编写单元测试用例。

希望这可以帮助。

转:https://codeburst.io/unit-testing-for-rest-apis-in-go-86c70dada52d

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 119,397评论 1 241
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 52,183评论 1 200
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 74,775评论 0 167
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 36,619评论 0 127
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 43,458评论 1 206
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 36,068评论 1 126
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 28,056评论 2 209
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 27,184评论 0 120
  • 想象着我的养父在大火中拼命挣扎,窒息,最后皮肤化为焦炭。我心中就已经是抑制不住地欢快,这就叫做以其人之道,还治其人...
    爱写小说的胖达阅读 26,124评论 5 173
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 30,241评论 0 178
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 27,450评论 1 170
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 28,689评论 1 178
  • 白月光回国,霸总把我这个替身辞退。还一脸阴沉的警告我。[不要出现在思思面前, 不然我有一百种方法让你生不如死。]我...
    爱写小说的胖达阅读 22,990评论 0 25
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 25,606评论 2 164
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 29,463评论 3 173
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 24,454评论 0 4
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 24,469评论 0 113
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 30,687评论 2 189
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 31,094评论 2 188

推荐阅读更多精彩内容