真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

go語(yǔ)言單元測(cè)試框架,Go語(yǔ)言框架

深入剖析:一套在 Go 中傳遞、返回、暴露錯(cuò)誤,便于回查的解決方案

作者:andruzhang,騰訊 IEG 后臺(tái)開發(fā)工程師

讓客戶滿意是我們工作的目標(biāo),不斷超越客戶的期望值來(lái)自于我們對(duì)這個(gè)行業(yè)的熱愛。我們立志把好的技術(shù)通過有效、簡(jiǎn)單的方式提供給客戶,將通過不懈努力成為客戶在信息化領(lǐng)域值得信任、有價(jià)值的長(zhǎng)期合作伙伴,公司提供的服務(wù)項(xiàng)目有:主機(jī)域名、網(wǎng)頁(yè)空間、營(yíng)銷軟件、網(wǎng)站建設(shè)、鄒平網(wǎng)站維護(hù)、網(wǎng)站推廣。

在后臺(tái)開發(fā)中,針對(duì)錯(cuò)誤處理,有三個(gè)維度的問題需要解決:

一個(gè)面向過程的函數(shù),在不同的處理過程中需要 handle 不同的錯(cuò)誤信息;一個(gè)面向?qū)ο蟮暮瘮?shù),針對(duì)一個(gè)操作所返回的不同類型的錯(cuò)誤,有可能需要進(jìn)行不同的處理。此外,在遇到錯(cuò)誤時(shí),也可以使用斷言的方式,快速中止函數(shù)流程,大大提高代碼的可讀性。

在許多高級(jí)語(yǔ)言中都提供了 try ... catch 的語(yǔ)法,函數(shù)內(nèi)部可以通過這種方案,實(shí)現(xiàn)一個(gè)統(tǒng)一的錯(cuò)誤處理邏輯。而即便是 C 這種 “中級(jí)語(yǔ)言” 雖然沒有,但是程序員也可以使用宏定義的方式,來(lái)實(shí)現(xiàn)某種程度上的錯(cuò)誤斷言。

但是,對(duì)于 Go 的情況就比較尷尬了。

我們先來(lái)看斷言,我們的目的是,僅使用一行代碼就能夠檢查錯(cuò)誤并終止當(dāng)前函數(shù)。由于沒有 throw,沒有宏,如果要實(shí)現(xiàn)一行斷言,有兩種方法。

第一種是把 if 的錯(cuò)誤判斷寫在一行內(nèi),比如:

第二種方法是借用 panic 函數(shù),結(jié)合 recover 來(lái)實(shí)現(xiàn):

這兩種方法都值得商榷。

首先,將 if 寫在同一行內(nèi)的問題有:

至于第二種方法,我們要分情況看;

不過使用 panic 來(lái)斷言的方案,雖然在業(yè)務(wù)邏輯中基本上不用,但在測(cè)試場(chǎng)景下則是非常常見的。測(cè)試嘛,用牛刀有何不可?稍微大一點(diǎn)的系統(tǒng)開銷也沒啥問題。對(duì)于 Go 來(lái)說(shuō),非常熱門的單元測(cè)試框架 goconvey 就是使用 panic 機(jī)制來(lái)實(shí)現(xiàn)單元測(cè)試中的斷言,用的人都說(shuō)好。

綜上,在 Go 中,對(duì)于業(yè)務(wù)代碼,筆者不建議采用斷言,遇到錯(cuò)誤的時(shí)候建議還是老老實(shí)實(shí)采用這種格式:

而在單測(cè)代碼中,則完全可以大大方方地采用類似于 goconvey 之類基于 panic 機(jī)制的斷言。

眾所周知 Go 是沒有 try ... catch 的,而且從官方的態(tài)度來(lái)看,短時(shí)間內(nèi)也沒有考慮的計(jì)劃。但程序員有這個(gè)需求呀。筆者采用的方法,是將需要返回的 err 變量在函數(shù)內(nèi)部全局化,然后結(jié)合 defer 統(tǒng)一處理:

這種方案要特別注意變量作用域問題.比如前面的 if err = DoSomething(); err != nil { 行,如果我們將 err = ... 改為 err := ...,那么這一行中的 err 變量和函數(shù)最前面定義的 (err error) 不是同一個(gè)變量,因此即便在此處發(fā)生了錯(cuò)誤,但是在 defer 函數(shù)中無(wú)法捕獲到 err 變量了。

在 try ... catch 方面,筆者其實(shí)沒有特別好的方法來(lái)模擬,即便是上面的方法也有一個(gè)很讓人頭疼的問題:defer 寫法導(dǎo)致錯(cuò)誤處理前置,而正常邏輯后置了,從可讀性的角度來(lái)說(shuō)非常不友好。因此也希望讀者能夠指教。同時(shí)還是希望 Go 官方能夠繼續(xù)迭代,支持這種語(yǔ)法。

這一點(diǎn)在 Go 里面,一開始看起來(lái)還是比較統(tǒng)一的,這就是 Go 最開始就定義的 error 類型,以系統(tǒng)標(biāo)準(zhǔn)的方式,統(tǒng)一了進(jìn)程內(nèi)函數(shù)級(jí)的錯(cuò)誤返回模式。調(diào)用方使用 if err != nil 的統(tǒng)一模式,來(lái)判斷一個(gè)調(diào)用是不是成功了。

但是隨著 Go 的逐步推廣,由于 error 接口的高自由度,程序員們對(duì)于 “如何判斷該錯(cuò)誤是什么錯(cuò)誤” 的時(shí)候,出現(xiàn)了分歧。

在 Go 1.13 之前,對(duì)于 error 類型的傳遞,有三種常見的模式:

這個(gè)流派很簡(jiǎn)單,就是將各種錯(cuò)誤信息直接定義為一個(gè)類枚舉值的模式,比如:

當(dāng)遇到相應(yīng)的錯(cuò)誤信息時(shí),直接返回對(duì)應(yīng)的 error 類枚舉值就行了。對(duì)于調(diào)用方也非常方便,可以采用 switch - case 來(lái)判斷錯(cuò)誤類型:

個(gè)人覺得這種設(shè)計(jì)模式本質(zhì)上還是 C error code 模式。

這種流派則是充分使用了 “error 是一個(gè) interface” 的特性,重新自定義一個(gè) error 類型。一方面是用不同的類型來(lái)表示不同的錯(cuò)誤分類,另一方面則能夠?qū)崿F(xiàn)對(duì)于同一錯(cuò)誤類型,能夠給調(diào)用方提供更佳詳盡的信息。舉個(gè)例子,我們可以定義多個(gè)不同的錯(cuò)誤類型如下:

對(duì)于調(diào)用方,則通過以下代碼來(lái)判斷不同的錯(cuò)誤:

這種模式,一方面可以透?jìng)鞯讓渝e(cuò)誤,另一方面又可以添加自定義的信息。但對(duì)于調(diào)用方而言,災(zāi)難在于如果要判斷某一個(gè)錯(cuò)誤的具體類型,只能用 strings.Contains() 來(lái)實(shí)現(xiàn),而錯(cuò)誤的具體描述文字是不可靠的,同一類型的信息可能會(huì)有不同的表達(dá);而在 fmt.Errorf 的過程中,各個(gè)業(yè)務(wù)添加的額外信息也可能會(huì)有不同的文字,這帶來(lái)了極大的不可靠性,提高了模塊之間的耦合度。

在 go 1.13 版本發(fā)布之后,針對(duì) fmt.Errorf 增加了 wraping 功能,并在 errors 包中添加了 Is() 和 As() 函數(shù)。關(guān)于這個(gè)模式的原理和使用已經(jīng)有很多文章了,本文就不再贅述。

這個(gè)功能,合并并改造了前文的所謂 “== 流派” 和 “fmt.Errorf” 流派,統(tǒng)一使用 errors.Is() 函數(shù);此外,也算是官方對(duì)類型斷言流派的認(rèn)可(專門用 As() 函數(shù)來(lái)支持)。

在實(shí)際應(yīng)用中,函數(shù)/模塊透?jìng)麇e(cuò)誤時(shí),應(yīng)該采用 Go 的 error wrapping 模式,也就是 fmt.Errorf() 配合 %w 使用,業(yè)務(wù)方可以放心地添加自己的錯(cuò)誤信息,只要調(diào)用方統(tǒng)一采用 errors.Is() 和 errors.As() 即可。

服務(wù)/系統(tǒng)層面的錯(cuò)誤信息返回,大部分協(xié)議都可以看成是 code - message 模式或者是其變體:

這種模式的特點(diǎn)是:code 是給程序代碼使用的,代碼判斷這是一個(gè)什么類型的錯(cuò)誤,進(jìn)入相應(yīng)的分支處理;而 message 是給人看的,程序可以以某種形式拋出或者記錄這個(gè)錯(cuò)誤信息,供用戶查看。

在這一層面有什么問題呢?code for computer,message for user,好像挺好的。

但有時(shí)候,我們可能會(huì)收到用戶/客戶反饋一個(gè)問題:“XXX 報(bào)錯(cuò)了,幫忙看看什么問題?”。用戶看不懂我們的錯(cuò)誤提示嗎?

在筆者的經(jīng)驗(yàn)中,我們?cè)谑褂?code - message 機(jī)制的時(shí)候,特別是業(yè)務(wù)初期,難以避免的是前后端的設(shè)計(jì)文案沒能完整地覆蓋所有的錯(cuò)誤用例,或者是錯(cuò)誤極其罕見。因此當(dāng)出現(xiàn)錯(cuò)誤時(shí),提示曖昧不清(甚至是直接提示錯(cuò)誤信息),導(dǎo)致用戶從錯(cuò)誤信息中找到解決方案

在這種情況下,盡量覆蓋所有錯(cuò)誤路徑肯定是最完美的方法。不過在做到這一點(diǎn)之前,碼農(nóng)們往往有下面的解決方案:

既要隱藏信息,又要暴露信息,我可以摔盤子嗎……

這里,筆者從日益普及的短信驗(yàn)證碼有了個(gè)靈感——人的短期記憶對(duì) 4 個(gè)字符還是比較強(qiáng)的,因此我們可以考慮把錯(cuò)誤代碼縮短到 4 個(gè)字符——不區(qū)分大小寫,因?yàn)槿绻嗽谟洃洉r(shí)還要記錄大小寫的話,難度會(huì)增加不少。

怎么用 4 個(gè)字符表示盡量多的數(shù)據(jù)呢?數(shù)字+字母總共有 36 個(gè)字符,理論上使用 4 位 36 進(jìn)制可以表示 36x36x36x36 = 1679616 個(gè)值。因此我們只要找到一個(gè)針對(duì)錯(cuò)誤信息字符串的哈希算法,把輸出值限制在 1679616 范圍內(nèi)就行了。

這里我采用的是 MD5 作為例子。MD5 的輸出是 128 位,理論上我可以取 MD5 的輸出,模 1679616 就可以得到一個(gè)簡(jiǎn)易的結(jié)果。實(shí)際上為了減少除法運(yùn)算,我采用的是取高 20 位(0xFFFFF)的簡(jiǎn)易方式(20 位二進(jìn)制的最大值為 1048575),然后將這個(gè)數(shù)字轉(zhuǎn)成 36 進(jìn)制的字符串輸出。

當(dāng)出現(xiàn)異常錯(cuò)誤時(shí),我們可以將 message 的提示信息如下展示:“未知錯(cuò)誤,錯(cuò)誤代碼 30EV,如需協(xié)助,請(qǐng)聯(lián)系 XXX”。順帶一提,30EV 是 "Access denied for user 'db_user'@'127.0.0.1'" 的計(jì)算結(jié)果,這樣一來(lái),我就對(duì)調(diào)用方隱藏了敏感信息。

至于后臺(tái)側(cè),還是需要實(shí)實(shí)在在地將這個(gè)哈希值和具體的錯(cuò)誤信息記錄在日志或者其他支持搜索的渠道里。當(dāng)用戶提供該代碼時(shí),可以快速定位。

這種方案的優(yōu)點(diǎn)很明顯:

簡(jiǎn)易的錯(cuò)誤碼生成代碼如下:

當(dāng)然這種方案也有局限性,筆者能想到的是需要注意以下兩點(diǎn):

此外,筆者需要再?gòu)?qiáng)調(diào)的是:在開發(fā)中,針對(duì)各種不同的、正式的錯(cuò)誤用例依然需要完整覆蓋,盡可能通過已有的 code - message 機(jī)制將足夠清晰的信息告知主調(diào)方。這種 hashcode 的錯(cuò)誤代碼生成方法,僅適用于錯(cuò)誤用例遺漏、或者是快速迭代過程中,用于發(fā)現(xiàn)和調(diào)試遺漏的錯(cuò)誤用例的臨時(shí)方案。

go 基于gin的單元測(cè)試

1.方法一:本地啟動(dòng)服務(wù),用瀏覽器或者postman測(cè)試,但是項(xiàng)目有改動(dòng)再次測(cè)試不是很方便

2.方法二:使用httptest結(jié)合testing來(lái)實(shí)現(xiàn)針對(duì)handlers接口函數(shù)的單元測(cè)試 這里直接使用一個(gè)開源的單元測(cè)試包,已經(jīng)封裝好了

github項(xiàng)目地址:

要測(cè)試接口的處理函數(shù)

type User struct {

Username string form:"username" json:"username" binding:"required"

Password string form:"password" json:"password" binding:"required"

Age int form:"age" json:"age" binding:"required"

}

func LoginHandler(c *gin.Context) {

req := User{}

if err := c.Bind(req); err != nil {

log.Printf("err:%v", err)

c.JSON(http.StatusOK, gin.H{

"errno": "1",

"errmsg": "parameters not match",

})

return

}

// judge the password and username

if req.UserName != "Valiben" || req.Password != "123456" {

c.JSON(http.StatusOK, gin.H{

"errno": "2",

"errmsg": "password or username is wrong",

})

return

}

c.JSON(http.StatusOK, gin.H{

"errno": "0",

"errmsg": "login success",

})

}

單元測(cè)試:

func init() {

//初始化路由

router := gin.New()

router.POST("/login", LoginHandler)

myLog := log.New(os.Stdout, "", log.Lshortfile|log.Ltime)

utilTest.SetRouter(router)

utilTest.SetLog(myLog)

}

type OrdinaryResponse struct {

Errno string json:"errno"

Errmsg string json:"errmsg"

Data UserActivityRespone json:"data"

}

func TestLoginHandler(t *testing.T) {

resp := OrdinaryResponse{}

err := utils.TestHandlerUnMarshalResp(utils.POST, "/login", utils.Form, user, resp)

if err != nil {

t.Errorf("TestLoginHandler: %v\n", err)

return

}

if resp.Errno != "0" {

t.Errorf("TestLoginHandler: response is not expected\n")

return

}

t.Log(resp.Data)

}

文件上傳測(cè)試:

func TestSaveFileHandler(t *testing.T) {

param := make(map[string]interface{})

param["file_name"] = "test1.txt"

param["upload_name"] = "Valiben"

resp := OrdinaryResponse{}

err := utils.TestFileHandlerUnMarshalResp(utils.POST, "/upload", (param["file_name"]).(string),

"file", param, resp)

if err != nil {

t.Errorf("TestSaveFileHandler: %v\n", err)

return

}

if resp.Errno != "0" {

t.Errorf("TestSaveFileHandler: response is not expected\n")

return

}

}

轉(zhuǎn)自:

go原生接口單元測(cè)試:

golang性能測(cè)試框架k6源碼分析

k6是新興的性能測(cè)試框架,比肩jmeter,另外測(cè)試腳本使用js,更加適合自動(dòng)化的架構(gòu)。

k6啟動(dòng)的框架是使用golang的cli標(biāo)準(zhǔn)框架cobra,入口函數(shù)

進(jìn)入cobra框架后,我們直接查看getRunCmd,這個(gè)是命令run的入口,主要工作都是從這里開始。

重點(diǎn)關(guān)注初始化Runner,這個(gè)是通過js腳本,使用goja庫(kù)解析后,生成的實(shí)際執(zhí)行單元。

進(jìn)入js目錄,查看Runner的結(jié)構(gòu),runner.go

Runner有一些配置屬性,另外還有方法,方法用lib.Runner的接口進(jìn)行規(guī)范。

Runner有一個(gè)NewVU方法,里面定義了連接參數(shù),實(shí)現(xiàn)api測(cè)試

返回主函數(shù),在初始化完成Runner后,啟動(dòng)調(diào)度器,以及做結(jié)果收集

最終封裝成一個(gè)engine

啟動(dòng)測(cè)試

go中如何寫單元測(cè)試

通過實(shí)際項(xiàng)目中的單測(cè)編寫,總結(jié)如下幾點(diǎn),幫助我更好的完成單元測(cè)試


網(wǎng)頁(yè)名稱:go語(yǔ)言單元測(cè)試框架,Go語(yǔ)言框架
本文路徑:http://weahome.cn/article/hoecdc.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部