go-common/app/interface/main/mcn/tool/datacenter/http_client.go

257 lines
6.2 KiB
Go
Raw Normal View History

2019-04-22 10:49:16 +00:00
package datacenter
import (
"bytes"
"context"
"crypto/md5"
"encoding/json"
"fmt"
"io"
"net/http"
"net/url"
"sort"
"strings"
"time"
"go-common/library/log"
pkgerr "github.com/pkg/errors"
"go-common/library/stat"
"strconv"
)
/*
访问数据平台的http client处理了签名接口监控等
*/
//ClientConfig client config
type ClientConfig struct {
Key string
Secret string
Dial time.Duration
Timeout time.Duration
KeepAlive time.Duration
}
//New new client
func New(c *ClientConfig) *HttpClient {
return &HttpClient{
client: &http.Client{},
conf: c,
}
}
//HttpClient http client
type HttpClient struct {
client *http.Client
conf *ClientConfig
Debug bool
}
//Response response
type Response struct {
Code int `json:"code"`
Msg string `json:"msg"`
Result interface{} `json:"result"`
}
const (
keyAppKey = "appKey"
keyAppID = "apiId"
keyTimeStamp = "timestamp"
keySign = "sign"
keySignMethod = "signMethod"
keyVersion = "version"
//TimeStampFormat time format in second
TimeStampFormat = "2006-01-02 15:04:05"
)
var (
clientStats = stat.HTTPClient
)
// Get issues a GET to the specified URL.
func (client *HttpClient) Get(c context.Context, uri string, params url.Values, res interface{}) (err error) {
req, err := client.NewRequest(http.MethodGet, uri, params)
if err != nil {
return
}
return client.Do(c, req, res)
}
// NewRequest new http request with method, uri, ip, values and headers.
// TODO(zhoujiahui): param realIP should be removed later.
func (client *HttpClient) NewRequest(method, uri string, params url.Values) (req *http.Request, err error) {
signStr, err := client.sign(params)
if err != nil {
err = pkgerr.Wrapf(err, "uri:%s,params:%v", uri, params)
return
}
params.Add(keySign, signStr)
enc := params.Encode()
ru := uri
if enc != "" {
ru = uri + "?" + enc
}
if method == http.MethodGet {
req, err = http.NewRequest(http.MethodGet, ru, nil)
} else {
req, err = http.NewRequest(http.MethodPost, uri, strings.NewReader(enc))
}
if err != nil {
err = pkgerr.Wrapf(err, "method:%s,uri:%s", method, ru)
return
}
const (
_contentType = "Content-Type"
_urlencoded = "application/x-www-form-urlencoded"
)
if method == http.MethodPost {
req.Header.Set(_contentType, _urlencoded)
}
return
}
// Do sends an HTTP request and returns an HTTP json response.
func (client *HttpClient) Do(c context.Context, req *http.Request, res interface{}, v ...string) (err error) {
var bs []byte
if bs, err = client.Raw(c, req, v...); err != nil {
return
}
if res != nil {
if err = json.Unmarshal(bs, res); err != nil {
err = pkgerr.Wrapf(err, "host:%s, url:%s, response:%s", req.URL.Host, realURL(req), string(bs))
}
}
return
}
//Raw get from url
func (client *HttpClient) Raw(c context.Context, req *http.Request, v ...string) (bs []byte, err error) {
var resp *http.Response
var uri = fmt.Sprintf("%s://%s%s", req.URL.Scheme, req.Host, req.URL.Path)
var now = time.Now()
var code string
defer func() {
clientStats.Timing(uri, int64(time.Since(now)/time.Millisecond))
if code != "" {
clientStats.Incr(uri, code)
}
}()
req = req.WithContext(c)
if resp, err = client.client.Do(req); err != nil {
err = pkgerr.Wrapf(err, "host:%s, url:%s", req.URL.Host, realURL(req))
code = "failed"
return
}
defer resp.Body.Close()
if resp.StatusCode >= http.StatusBadRequest {
err = pkgerr.Errorf("incorrect http status:%d host:%s, url:%s", resp.StatusCode, req.URL.Host, realURL(req))
code = strconv.Itoa(resp.StatusCode)
return
}
if bs, err = readAll(resp.Body, 16*1024); err != nil {
err = pkgerr.Wrapf(err, "host:%s, url:%s", req.URL.Host, realURL(req))
return
}
if client.Debug {
log.Info("reqeust: host:%s, url:%s, response body:%s", req.URL.Host, realURL(req), string(bs))
}
return
}
// sign calc appkey and appsecret sign.
// see http://info.bilibili.co/pages/viewpage.action?pageId=5410881#id-%E6%95%B0%E6%8D%AE%E7%9B%98%EF%BC%8D%E5%AE%89%E5%85%A8%E8%AE%A4%E8%AF%81-%E4%BA%8C%E7%AD%BE%E5%90%8D%E7%AE%97%E6%B3%95
func (client *HttpClient) sign(params url.Values) (sign string, err error) {
key := client.conf.Key
secret := client.conf.Secret
if params == nil {
params = url.Values{}
}
params.Set(keyAppKey, key)
params.Set(keyVersion, "1.0")
if params.Get(keyTimeStamp) == "" {
params.Set(keyTimeStamp, time.Now().Format(TimeStampFormat))
}
params.Set(keySignMethod, "md5")
var needSignParams = url.Values{}
needSignParams.Add(keyAppKey, key)
needSignParams.Add(keyTimeStamp, params.Get(keyTimeStamp))
needSignParams.Add(keyVersion, params.Get(keyVersion))
//tmp := params.Encode()
var valueMap = map[string][]string(needSignParams)
var buf bytes.Buffer
// 开头与结尾加secret
buf.Write([]byte(secret))
keys := make([]string, 0, len(valueMap))
for k := range valueMap {
keys = append(keys, k)
}
sort.Strings(keys)
for _, k := range keys {
vs := valueMap[k]
prefix := k
buf.WriteString(prefix)
for _, v := range vs {
buf.WriteString(v)
break
}
}
buf.Write([]byte(secret))
var md5 = md5.New()
md5.Write(buf.Bytes())
sign = fmt.Sprintf("%X", md5.Sum(nil))
return
}
// readAll reads from r until an error or EOF and returns the data it read
// from the internal buffer allocated with a specified capacity.
func readAll(r io.Reader, capacity int64) (b []byte, err error) {
buf := bytes.NewBuffer(make([]byte, 0, capacity))
// If the buffer overflows, we will get bytes.ErrTooLarge.
// Return that as an error. Any other panic remains.
defer func() {
e := recover()
if e == nil {
return
}
if panicErr, ok := e.(error); ok && panicErr == bytes.ErrTooLarge {
err = panicErr
} else {
panic(e)
}
}()
_, err = buf.ReadFrom(r)
return buf.Bytes(), err
}
// realUrl return url with http://host/params.
func realURL(req *http.Request) string {
if req.Method == http.MethodGet {
return req.URL.String()
} else if req.Method == http.MethodPost {
ru := req.URL.Path
if req.Body != nil {
rd, ok := req.Body.(io.Reader)
if ok {
buf := bytes.NewBuffer([]byte{})
buf.ReadFrom(rd)
ru = ru + "?" + buf.String()
}
}
return ru
}
return req.URL.Path
}
// SetTransport set client transport
func (client *HttpClient) SetTransport(t http.RoundTripper) {
client.client.Transport = t
}