Contents

Go Http Client

req

https://godoc.org/github.com/serialt/req?status.svg

Go语言人性化HTTP请求库

特性

  • 轻量级
  • 简单
  • 容易操作JSON和XML
  • 容易调试和日志记录
  • 容易上传和下载文件
  • 容易管理Cookie
  • 容易设置代理
  • 容易设置超时
  • 容易自定义HTTP客户端

安装

go get github.com/serialt/req

概要

req 基于标准库 net/http 实现了一个友好的API.

ReqResp 是两个最重要的结构体, 你可以把 Req 看作客户端, 把Resp 看作存放请求及其响应的容器,它们都提供许多简洁方便的API,让你可以很轻松做很多很多事情。

func (r *Req) Post(url string, v ...interface{}) (*Resp, error)

大多情况下,发起请求只有url是必选参数,其它都可选,比如请求头、请求参数、文件或请求体等。

包中含一个默认的 Req 对象, 它所有的公有方法都被req包对应的公有方法包装了,所以大多数情况下,你直接可以把req包看作一个Req对象来使用。

// 创建Req对象来发起请求
r := req.New()
r.Get(url)

// 直接使用req包发起请求
req.Get(url)

你可以使用 req.New() 方法来创建 *Req 作为一个单独的客户端

例子

基础用法
设置请求头
设置请求参数
设置请求体
调试
输出格式
ToJSON & ToXML
获取 *http.Response
上传
下载
Cookie
设置超时
设置代理
自定义 http.Client

基础用法

header := req.Header{
	"Accept":        "application/json",
	"Authorization": "Basic YWRtaW46YWRtaW4=",
}
param := req.Param{
	"name": "imroc",
	"cmd":  "add",
}
// 只有url必选,其它参数都是可选
r, err = req.Post("http://foo.bar/api", header, param)
if err != nil {
	log.Fatal(err)
}
r.ToJSON(&foo)       // 响应体转成对象
log.Printf("%+v", r) // 打印详细信息

设置请求头

使用 req.Header (它实际上是一个 map[string]string)

authHeader := req.Header{
	"Accept":        "application/json",
	"Authorization": "Basic YWRtaW46YWRtaW4=",
}
req.Get("https://www.baidu.com", authHeader, req.Header{"User-Agent": "V1.1"})

使用 http.Header

header := make(http.Header)
header.Set("Accept", "application/json")
req.Get("https://www.baidu.com", header)

你可以使用 struct 来设置请求头,用 HeaderFromStruct 这个函数来解析你的 struct

type HeaderStruct struct {
	UserAgent     string `json:"User-Agent"`
	Authorization string `json:"Authorization"`
}

func main(){
	h := HeaderStruct{
		"V1.0.0",
		"roc",
	}

	authHeader := req.HeaderFromStruct(h) 
	req.Get("https://www.baidu.com", authHeader, req.Header{"User-Agent": "V1.1"})
}

注:请给你的 struct 加上 json tag.

设置请求参数

Use req.Param (它实际上是一个 map[string]interface{})

param := req.Param{
	"id":  "imroc",
	"pwd": "roc",
}
req.Get("http://foo.bar/api", param) // http://foo.bar/api?id=imroc&pwd=roc
req.Post(url, param)                  // 请求体 => id=imroc&pwd=roc

使用 req.QueryParam 强制将请求参数拼在url后面 (它实际上也是一个 map[string]interface{})

req.Post("http://foo.bar/api", req.Param{"name": "roc", "age": "22"}, req.QueryParam{"access_token": "fedledGF9Hg9ehTU"})
/*
POST /api?access_token=fedledGF9Hg9ehTU HTTP/1.1
Host: foo.bar
User-Agent: Go-http-client/1.1
Content-Length: 15
Content-Type: application/x-www-form-urlencoded;charset=UTF-8
Accept-Encoding: gzip

age=22&name=roc
*/

设置请求体

Put string, []byte and io.Reader as body directly.

req.Post(url, "id=roc&cmd=query")

将对象作为JSON或XML请求体(自动添加 Content-Type 请求头)

req.Post(url, req.BodyJSON(&foo))
req.Post(url, req.BodyXML(&bar))

调试

将全局变量 req.Debug 设置为true,将会把所有请求的详细信息打印在标准输出。

req.Debug = true
req.Post("http://localhost/test" "hi")

post.png

输出格式

您可以使用指定类型的输出格式在日志文件中记录请求和响应的信息。例如,在开发阶段使用%+v格式,可以让你观察请求和响应的细节信息。 在生产阶段使用%v%-v输出格式,只记录所需要的信息。

%+v%+s

详细输出

r, _ := req.Post(url, header, param)
log.Printf("%+v", r) // 输出格式和Debug开启时的格式一样

%v%s

简单输出(默认格式)

r, _ := req.Get(url, param)
log.Printf("%v\n", r) // GET http://foo.bar/api?name=roc&cmd=add {"code":"0","msg":"success"}
log.Prinln(r)         // 和上面一样

%-v%-s

简单输出并保持所有内容在一行内(请求体或响应体可能包含多行,这种格式会将所有换行、回车替换成" ", 这在会让你在查日志的时候非常有用)

Flag

你可以调用 SetFlags 控制输出内容,决定哪些部分能够被输出。

const (
	LreqHead  = 1 << iota // 输出请求首部(包含请求行和请求头)
	LreqBody              // 输出请求体
	LrespHead             // 输出响应首部(包含响应行和响应头)
	LrespBody             // 输出响应体
	Lcost                 // 输出请求所消耗掉时长
	LstdFlags = LreqHead | LreqBody | LrespHead | LrespBody
)
req.SetFlags(req.LreqHead | req.LreqBody | req.LrespHead)

监控请求耗时

req.SetFlags(req.LstdFlags | req.Lcost) // 输出格式显示请求耗时
r,_ := req.Get(url)
log.Println(r) // http://foo.bar/api 3.260802ms {"code":0 "msg":"success"}
if r.Cost() > 3 * time.Second { // 检查耗时
	log.Println("WARN: slow request:", r)
}

ToJSON & ToXML

r, _ := req.Get(url)
r.ToJSON(&foo)
r, _ = req.Post(url, req.BodyXML(&bar))
r.ToXML(&baz)

获取 *http.Response

// func (r *Req) Response() *http.Response
r, _ := req.Get(url)
resp := r.Response()
fmt.Println(resp.StatusCode)

上传

使用 req.File 匹配文件

req.Post(url, req.File("imroc.png"), req.File("/Users/roc/Pictures/*.png"))

使用 req.FileUpload 细粒度控制上传

file, _ := os.Open("imroc.png")
req.Post(url, req.FileUpload{
	File:      file,
	FieldName: "file",       // FieldName 是表单字段名
	FileName:  "avatar.png", // Filename 是要上传的文件的名称,我们使用它来猜测mimetype,并将其上传到服务器上
})

使用req.UploadProgress监听上传进度

progress := func(current, total int64) {
	fmt.Println(float32(current)/float32(total)*100, "%")
}
req.Post(url, req.File("/Users/roc/Pictures/*.png"), req.UploadProgress(progress))
fmt.Println("upload complete")

下载

r, _ := req.Get(url)
r.ToFile("imroc.png")

使用req.DownloadProgress监听下载进度

progress := func(current, total int64) {
	fmt.Println(float32(current)/float32(total)*100, "%")
}
r, _ := req.Get(url, req.DownloadProgress(progress))
r.ToFile("hello.mp4")
fmt.Println("download complete")

Cookie

默认情况下,底层的 *http.Client 会自动管理你的cookie(如果服务器给你发了cookie,之后的请求它会自动带上cookie请求头给服务器), 你可以调用这个方法取消自动管理:

req.EnableCookie(false)

你还可以在发送请求的时候自己传入 *http.Cookie

cookie := new(http.Cookie)
// ......
req.Get(url, cookie)

设置超时

req.SetTimeout(50 * time.Second)

设置代理

默认情况下,如果系统环境变量有 http_proxyhttps_proxy ,req会讲对应的地址作为对应协议的代理,你也可以自定义设置代理,或者将其置为nil,即取消代理。

req.SetProxy(func(r *http.Request) (*url.URL, error) {
	if strings.Contains(r.URL.Hostname(), "google") {
		return url.Parse("http://my.vpn.com:23456")
	}
	return nil, nil
})

设置简单代理(将所有请求都转发到指定代理url地址上)

req.SetProxyUrl("http://my.proxy.com:23456")

自定义HTTP客户端

使用 SetClient 改变底层的 *http.Client

req.SetClient(client)

给某个请求制定特定的 *http.Client

client := &http.Client{Timeout: 30 * time.Second}
req.Get(url, client)

改变底层 *http.Client 的某些属性

req.Client().Jar, _ = cookiejar.New(nil)
trans, _ := req.Client().Transport.(*http.Transport)
trans.MaxIdleConns = 20
trans.TLSHandshakeTimeout = 20 * time.Second
trans.DisableKeepAlives = true
trans.TLSClientConfig = &tls.Config{InsecureSkipVerify: true}