詳解Golang語(yǔ)言HTTP客戶端實(shí)踐
最近在學(xué)習(xí)Golang語(yǔ)言,中間遇到一個(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)。
那么那什么來(lái)做練習(xí)呢?當(dāng)然結(jié)合當(dāng)下的工作啦,所以我列了一個(gè)路線給自己,那就是從接口測(cè)試開始學(xué)起來(lái),從功能測(cè)試到性能測(cè)試,然后掌握基本Server開發(fā)技能。
首先,得先把HTTP接口測(cè)試常用的幾個(gè)功能實(shí)現(xiàn)了,主要是獲取HTTPrequest對(duì)象,發(fā)送請(qǐng)求解析響應(yīng)以及HttpClient的基本配置。
這里實(shí)現(xiàn)比較簡(jiǎn)單和粗淺,讓我想起FunTester測(cè)試框架一開始的時(shí)候,也是從封裝HttpClient.jar提供的API開始的,感慨萬(wàn)千。
這里我從了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自動(dòng)處理,這個(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請(qǐng)求
// @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請(qǐng)求,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 初始化請(qǐng)求客戶端
// @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)成傳入對(duì)象
// @Description:
// @param res
// @param r
//
func ParseRes(res []byte, r interface{}) {
json.Unmarshal(res, r)
}
測(cè)試腳本
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))
}
控制臺(tái)輸出
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請(qǐng)求
post請(qǐng)求form表單
post請(qǐng)求json表單Process finished with the exit code 0
測(cè)試服務(wù)
依舊采用了moco_FunTester測(cè)試框架實(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請(qǐng)求")
server.post(both(urlStartsWith("/test"), existForm("fun"))).response("post請(qǐng)求form表單")
server.post(both(urlStartsWith("/test"), existParams("fun"))).response("post請(qǐng)求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語(yǔ)言HTTP客戶端實(shí)踐的文章就介紹到這了,更多相關(guān)Golang HTTP客戶端內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- golang構(gòu)建HTTP服務(wù)的實(shí)現(xiàn)步驟
- 淺談golang fasthttp踩坑經(jīng)驗(yàn)
- Golang請(qǐng)求fasthttp實(shí)踐
- golang中g(shù)in框架接入jwt使用token驗(yàn)證身份
- Golang Gin框架實(shí)現(xiàn)文件下載功能的示例代碼
- golang gin框架獲取參數(shù)的操作
- golang gin 框架 異步同步 goroutine 并發(fā)操作
- golang 網(wǎng)絡(luò)框架之gin的使用方法
- Golang詳細(xì)講解常用Http庫(kù)及Gin框架的應(yīng)用
相關(guān)文章
解決vscode中g(shù)olang插件依賴安裝失敗問題
這篇文章主要介紹了解決vscode中g(shù)olang插件依賴安裝失敗問題,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2019-08-08
Go 自動(dòng)升級(jí)依賴版本的幾種方法實(shí)現(xiàn)
Go語(yǔ)言使用GoModules進(jìn)行依賴管理,可以通過多種方式自動(dòng)升級(jí)依賴,本文就來(lái)介紹Go 自動(dòng)升級(jí)依賴版本的7種方法,感興趣的可以了解一下2025-03-03
一文帶你使用golang手?jǐn)]一個(gè)websocket中間件
這篇文章主要為大家詳細(xì)介紹了如何使用golang手?jǐn)]一個(gè)websocket中間件,文中的示例代碼講解詳細(xì),具有一定的借鑒價(jià)值,感興趣的小伙伴可以參考一下2023-12-12
Go語(yǔ)言面向?qū)ο笾械亩鄳B(tài)你學(xué)會(huì)了嗎
面向?qū)ο笾械亩鄳B(tài)(Polymorphism)是指一個(gè)對(duì)象可以具有多種不同的形態(tài)或表現(xiàn)方式,本文將通過一些簡(jiǎn)單的示例為大家講解一下多態(tài)的實(shí)現(xiàn),需要的可以參考下2023-07-07
Go語(yǔ)言中http.ResponseWriter接口
http.ResponseWriter是Go語(yǔ)言中用來(lái)設(shè)置HTTP響應(yīng)的接口,本文主要介紹了Go語(yǔ)言中http.ResponseWriter接口,具有一定的參考價(jià)值,感興趣的可以了解一下2024-08-08
go帶緩沖chan實(shí)現(xiàn)消息隊(duì)列功能
本文主要介紹了go帶緩沖chan實(shí)現(xiàn)消息隊(duì)列功能,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-02-02
go mod 安裝依賴 unkown revision問題的解決方案
這篇文章主要介紹了go mod 安裝依賴 unkown revision問題的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來(lái)看看吧2021-05-05

