123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256 |
- 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
- }
|