詳解Golang語言HTTP客戶端實(shí)踐
最近在學(xué)習(xí)Golang語言,中間遇到一個(gè)前輩指點(diǎn),有一個(gè)學(xué)習(xí)原則:Learning By Doing。跟我之前學(xué)習(xí)Java的經(jīng)驗(yàn)高度契合。在前一段時(shí)間學(xué)習(xí)洼坑中掙扎了好幾天,差點(diǎn)就忘記這個(gè)重要的成功經(jīng)驗(yàn)。
那么那什么來做練習(xí)呢?當(dāng)然結(jié)合當(dāng)下的工作啦,所以我列了一個(gè)路線給自己,那就是從接口測試開始學(xué)起來,從功能測試到性能測試,然后掌握基本Server開發(fā)技能。
首先,得先把HTTP接口測試常用的幾個(gè)功能實(shí)現(xiàn)了,主要是獲取HTTPrequest對象,發(fā)送請求解析響應(yīng)以及HttpClient的基本配置。
這里實(shí)現(xiàn)比較簡單和粗淺,讓我想起FunTester測試框架一開始的時(shí)候,也是從封裝HttpClient.jar提供的API開始的,感慨萬千。
這里我從了Golang SDK自帶的net/http包提供的HTTP相關(guān)API,雖然本身提供了http.PostForm()、http.Post()以及http.Get()封裝好的方法,但是在處理HTTPrequest的header和cookie處理缺乏靈活性,所以我重新將net/http封裝的API進(jìn)行第二次封裝。其中存在幾處遺漏,比如HTTPstatus的判斷以及header中Content-Type自動處理,這個(gè)以后再豐富,暫時(shí)的目標(biāo)就是能用。
HTTP客戶端封裝
package task import ( "bytes" "encoding/json" "fmt" "io/ioutil" "net/http" "net/url" "strings" "time" ) var Client http.Client = clients() // Res 模擬響應(yīng)結(jié)構(gòu) // @Description: type Res struct { Have string `json:"Have"` } // Get 獲取GET請求 // @Description: // @param uri // @param args // @return *http.Request func Get(uri string, args map[string]interface{}) *http.Request { uri = uri + "?" +ToValues(args) request, _ := http.NewRequest("get", uri, nil) return request } // PostForm POST接口form表單 // @Description: // @param path // @param args // @return *http.Request func PostForm(path string, args map[string]interface{}) *http.Request { request, _ := http.NewRequest("post", path, strings.NewReader(ToValues(args))) return request } // PostJson POST請求,JSON參數(shù) // @Description: // @param path // @param args // @return *http.Request func PostJson(path string, args map[string]interface{}) *http.Request { marshal, _ := json.Marshal(args) request, _ := http.NewRequest("post", path, bytes.NewReader(marshal)) return request } // ToValues 將map解析成HTTP參數(shù),用于GET和POST form表單 // @Description: // @param args // @return string func ToValues(args map[string]interface{}) string { if args != nil && len(args) > 0 { params := url.Values{} for k, v := range args { params.Set(k, fmt.Sprintf("%v", v)) } return params.Encode() } return "" } // Response 獲取響應(yīng)詳情,默認(rèn)[]byte格式 // @Description: // @param request // @return []byte func Response(request *http.Request) []byte { res, err := Client.Do(request) if err != nil { return nil } body, _ := ioutil.ReadAll(res.Body) // 讀取響應(yīng) body, 返回為 []byte defer res.Body.Close() return body } // clients 初始化請求客戶端 // @Description: // @return http.Client func clients() http.Client { return http.Client{ Timeout: time.Duration(5) * time.Second, //超時(shí)時(shí)間 Transport: &http.Transport{ MaxIdleConnsPerHost: 5, //單個(gè)路由最大空閑連接數(shù) MaxConnsPerHost: 100, //單個(gè)路由最大連接數(shù) IdleConnTimeout: 90 * time.Second, TLSHandshakeTimeout: 10 * time.Second, ExpectContinueTimeout: 1 * time.Second, }, } } // ParseRes 解析響應(yīng) // @Description: // @receiver r // @param res func (r *Res) ParseRes(res []byte) { json.Unmarshal(res, r) } // ParseRes 解析響應(yīng),將[]byte轉(zhuǎn)成傳入對象 // @Description: // @param res // @param r // func ParseRes(res []byte, r interface{}) { json.Unmarshal(res, r) }
測試腳本
package main import ( "fmt" "funtester/src/task" "io" "log" "net/http" "os" "time" ) const ( a = iota b c d e ) func init() { os.Mkdir("./log/", 0777) os.Mkdir("./long/", 0777) file := "./log/" + string(time.Now().Format("20060102")) + ".log" openFile, _ := os.OpenFile(file, os.O_RDWR|os.O_CREATE|os.O_APPEND, 0666) writer := io.MultiWriter(os.Stdout, openFile) log.SetOutput(writer) log.SetFlags(log.LstdFlags | log.Lshortfile | log.Ldate) } func main() { url := "http://localhost:12345/test" args := map[string]interface{}{ "name": "FunTester", "fun": "fdsafj", } cookie := &http.Cookie{ Name: "token", Value: "fsjej09u0934jtej", } get := task.Get(url, args) get.Header.Add("user_agent", "FunTester") get.AddCookie(cookie) response := task.Response(get) fmt.Println(string(response)) form := task.PostForm(url, args) bytes := task.Response(form) fmt.Println(string(bytes)) json := task.PostJson(url, args) res := task.Response(json) fmt.Println(string(res)) }
控制臺輸出
GOROOT=/usr/local/go #gosetup
GOPATH=/Users/oker/go #gosetup
/usr/local/go/bin/go build -o /private/var/folders/7b/0djfgf7j7p9ch_hgm9wx9n6w0000gn/T/GoLand/___go_build_funtester_src_m funtester/src/m #gosetup
/private/var/folders/7b/0djfgf7j7p9ch_hgm9wx9n6w0000gn/T/GoLand/___go_build_funtester_src_m
get請求
post請求form表單
post請求json表單Process finished with the exit code 0
測試服務(wù)
依舊采用了moco_FunTester測試框架實(shí)現(xiàn)。
package com.mocofun.moco.main import com.funtester.utils.ArgsUtil import com.mocofun.moco.MocoServer class Share extends MocoServer { static void main(String[] args) { def util = new ArgsUtil(args) // def server = getServerNoLog(util.getIntOrdefault(0,12345)) def server = getServer(util.getIntOrdefault(0, 12345)) server.get(urlStartsWith("/test")).response("get請求") server.post(both(urlStartsWith("/test"), existForm("fun"))).response("post請求form表單") server.post(both(urlStartsWith("/test"), existParams("fun"))).response("post請求json表單") server.get(urlStartsWith("/qps")).response(qps(textRes("恭喜到達(dá)QPS!"), 1)) // server.response(delay(jsonRes(getJson("Have=Fun ~ Tester !")), 1000)) server.response("Have Fun ~ Tester !") def run = run(server) waitForKey("fan") run.stop() } }
到此這篇關(guān)于詳解Golang語言HTTP客戶端實(shí)踐的文章就介紹到這了,更多相關(guān)Golang HTTP客戶端內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
詳解golang中?work與?module?的區(qū)別與聯(lián)系
Go?模塊通常由一個(gè)項(xiàng)目或庫組成,并包含一組隨后一起發(fā)布的?Go?包,Go?模塊通過允許用戶將項(xiàng)目代碼放在他們選擇的目錄中并為每個(gè)模塊指定依賴項(xiàng)的版本,解決了原始系統(tǒng)的許多問題,本文將給大家介紹一下golang中?work與?module?的區(qū)別與聯(lián)系,需要的朋友可以參考下2023-09-09通過Golang實(shí)現(xiàn)linux命令ls命令(命令行工具構(gòu)建)
這篇文章主要為大家詳細(xì)介紹了如何通過Golang實(shí)現(xiàn)一個(gè)linux命令ls命令(命令行工具構(gòu)建),文中的示例代碼講解詳細(xì),具有一定的學(xué)習(xí)價(jià)值,感興趣的可以了解一下2023-01-01Go操作各大消息隊(duì)列教程(RabbitMQ、Kafka)
消息隊(duì)列是一種異步的服務(wù)間通信方式,適用于無服務(wù)器和微服務(wù)架構(gòu),本文主要介紹了Go操作各大消息隊(duì)列教程(RabbitMQ、Kafka),需要的朋友可以了解一下2024-02-02golang切片擴(kuò)容規(guī)則實(shí)現(xiàn)
這篇文章主要介紹了golang切片擴(kuò)容規(guī)則實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-03-03golang協(xié)程關(guān)閉踩坑實(shí)戰(zhàn)記錄
協(xié)程(coroutine)是Go語言中的輕量級線程實(shí)現(xiàn),下面這篇文章主要給大家介紹了關(guān)于golang協(xié)程關(guān)閉踩坑的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-03-03