http_utils.go 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310
  1. package utils
  2. import (
  3. "compress/flate"
  4. "compress/gzip"
  5. "compress/zlib"
  6. "context"
  7. "fmt"
  8. "golang.org/x/net/proxy"
  9. "io"
  10. "io/ioutil"
  11. "net"
  12. "net/http"
  13. "net/url"
  14. "strings"
  15. "time"
  16. )
  17. // Error Code接口
  18. type IErrorCode interface {
  19. // 转换为int
  20. ToInt() int
  21. // 方便与int比较
  22. Equals(errCode int) bool
  23. // 获取错误信息
  24. ErrMsg(args ...interface{}) string
  25. }
  26. type Res struct {
  27. Head ResHead `json:"head"`
  28. Data interface{} `json:"data,omitempty"`
  29. }
  30. type ResHead struct {
  31. ErrCode int `json:"errcode"`
  32. ErrMsg string `json:"errmsg,omitempty"`
  33. Detail string `json:"-"`
  34. HttpStatus int `json:"-"`
  35. }
  36. func E2(errDetail interface{}, errCode IErrorCode, args ...interface{}) *Res {
  37. errMsg := errCode.ErrMsg(args...)
  38. return E(errCode.ToInt(), errMsg, errDetail)
  39. }
  40. func E(status int, errMsg string, errDetail interface{}) *Res {
  41. var msg, detail string
  42. if nil != errDetail {
  43. detail = strings.TrimSpace(fmt.Sprintf("%s", errDetail))
  44. }
  45. msg = strings.TrimSpace(errMsg)
  46. return &Res{
  47. Head: ResHead{
  48. ErrCode: status,
  49. ErrMsg: msg,
  50. Detail: detail,
  51. HttpStatus: http.StatusInternalServerError,
  52. },
  53. }
  54. }
  55. func R(data interface{}) *Res {
  56. return &Res{
  57. Head: ResHead{
  58. ErrCode: 0,
  59. ErrMsg: "",
  60. HttpStatus: http.StatusOK,
  61. },
  62. Data: data,
  63. }
  64. }
  65. // http请求相关默认配置
  66. const (
  67. DefaultHttpDialTimeout = 20 * time.Second
  68. DefaultHttpKeepAlive = 120 * time.Second
  69. DefaultHttpMaxIdleConns = 1000
  70. DefaultHttpMaxIdleConnsPerHost = 1000
  71. DefaultHttpIdleConnTimeout = 90 * time.Second
  72. DefaultHttpTimeout = 30 * time.Second
  73. )
  74. // http请求配置
  75. type RequestPromise struct {
  76. headers http.Header
  77. encoding Charset
  78. timeout time.Duration
  79. proxy func(*http.Request) (*url.URL, error)
  80. dialContext func(ctx context.Context, network, addr string) (net.Conn, error)
  81. client *http.Client
  82. }
  83. // 返回一个http请求配置对象,默认带上压缩头
  84. func NewRequest() *RequestPromise {
  85. return (&RequestPromise{}).
  86. SetHeader("Accept-Encoding", "gzip, deflate, zlib")
  87. }
  88. // 返回一个http请求配置对象,默认带上压缩头和Content-Type = application/json; charset=utf-8
  89. func JSONRequest() *RequestPromise {
  90. return (&RequestPromise{}).
  91. SetHeader("Accept-Encoding", "gzip, deflate, zlib").
  92. SetHeader("Content-Type", "application/json; charset=utf-8")
  93. }
  94. // 返回一个http请求配置对象,默认带上压缩头和Content-Type = application/xml; charset=utf-8
  95. func XMLRequest() *RequestPromise {
  96. return (&RequestPromise{}).
  97. SetHeader("Accept-Encoding", "gzip, deflate, zlib").
  98. SetHeader("Content-Type", "application/xml; charset=utf-8")
  99. }
  100. // 返回一个采用了连接池和Keepalive配置的http client,可以配合RequestPromise的SetClient函数使用
  101. // 默认不使用它,而是每次请求新建连接
  102. func NewPoolingHttpClient() *http.Client {
  103. return &http.Client{
  104. Transport: &http.Transport{
  105. DialContext: (&net.Dialer{
  106. Timeout: DefaultHttpDialTimeout,
  107. KeepAlive: DefaultHttpKeepAlive,
  108. }).DialContext,
  109. MaxIdleConns: DefaultHttpMaxIdleConns,
  110. MaxIdleConnsPerHost: DefaultHttpMaxIdleConnsPerHost,
  111. IdleConnTimeout: DefaultHttpIdleConnTimeout,
  112. },
  113. Timeout: DefaultHttpTimeout, // 此处设置小于等于零的值,意为不超时
  114. }
  115. }
  116. // 设置http header
  117. func (r *RequestPromise) SetHeader(key string, value string) *RequestPromise {
  118. if len(strings.TrimSpace(key)) == 0 {
  119. return r
  120. }
  121. key = strings.TrimSpace(key)
  122. if nil == r.headers {
  123. r.headers = make(http.Header)
  124. }
  125. r.headers.Set(key, value)
  126. return r
  127. }
  128. // 设置http响应的编码,默认utf8
  129. func (r *RequestPromise) SetEncoding(encoding Charset) *RequestPromise {
  130. if encoding == UTF8 {
  131. return r
  132. }
  133. r.encoding = encoding
  134. return r
  135. }
  136. // 设置超时时间,从连接到接收到响应的总时间
  137. // 如果此处不设置则采用http client中设置的超时时间,默认http client超时时间30秒
  138. // 如果此处设置不等于零的值,则覆盖http client中设置的超时时间
  139. // 如果此处设置小于零的值,意为不超时
  140. func (r *RequestPromise) SetTimeout(timeout time.Duration) *RequestPromise {
  141. if timeout == 0 {
  142. return r
  143. }
  144. r.timeout = timeout
  145. return r
  146. }
  147. // 设置http或https代理,默认无代理
  148. func (r *RequestPromise) SetHttpProxy(proxyUri string) *RequestPromise {
  149. if len(strings.TrimSpace(proxyUri)) == 0 {
  150. return r
  151. }
  152. proxyUri = strings.TrimSpace(proxyUri)
  153. uri, err := (&url.URL{}).Parse(proxyUri)
  154. if nil != err {
  155. return r
  156. }
  157. r.proxy = http.ProxyURL(uri)
  158. return r
  159. }
  160. // 设置socket5代理,默认无代理
  161. func (r *RequestPromise) SetSocket5Proxy(proxyUri string) *RequestPromise {
  162. if len(strings.TrimSpace(proxyUri)) == 0 {
  163. return r
  164. }
  165. proxyUri = strings.TrimSpace(proxyUri)
  166. dialer, err := proxy.SOCKS5("tcp", proxyUri, nil, proxy.Direct)
  167. if nil != err {
  168. return r
  169. }
  170. r.dialContext = func(_ context.Context, network string, address string) (net.Conn, error) {
  171. return dialer.Dial(network, address)
  172. }
  173. return r
  174. }
  175. // 设置事先实例化好的http client,默认每次请求会新建一个http client
  176. func (r *RequestPromise) SetClient(client *http.Client) *RequestPromise {
  177. if nil == client {
  178. return r
  179. }
  180. r.client = client
  181. return r
  182. }
  183. // 发起请求并返回响应内容
  184. // FORM方式提交数据请设置Content-Type=application/x-www-form-urlencoded请求头,且io.Reader传url.Values.Encode得到的字符串的reader
  185. func (r *RequestPromise) Call(httpMethod string, targetUri string, data io.Reader) ([]byte, error) {
  186. targetUri = strings.TrimSpace(targetUri)
  187. if len(targetUri) == 0 {
  188. return nil, nil
  189. }
  190. // http request handle
  191. if len(strings.TrimSpace(httpMethod)) == 0 {
  192. httpMethod = http.MethodGet
  193. } else {
  194. httpMethod = strings.ToUpper(strings.TrimSpace(httpMethod))
  195. }
  196. req, err := http.NewRequest(httpMethod, targetUri, data)
  197. if err != nil {
  198. return nil, err
  199. }
  200. if nil != r.headers {
  201. req.Header = r.headers
  202. }
  203. r.initClient()
  204. // send http request & get http response
  205. resp, err := r.client.Do(req)
  206. if err != nil {
  207. return nil, err
  208. }
  209. return r.readResponseBody(resp)
  210. }
  211. func (r *RequestPromise) initClient() {
  212. // http client handle
  213. if nil == r.client { // create new http client instance
  214. r.client = &http.Client{Timeout: DefaultHttpTimeout} // default timeout
  215. }
  216. if r.timeout != 0 {
  217. r.client.Timeout = r.timeout
  218. }
  219. if r.client.Timeout < 0 {
  220. r.client.Timeout = 0
  221. }
  222. if nil != r.proxy || nil != r.dialContext {
  223. if nil == r.client.Transport {
  224. r.client.Transport = &http.Transport{}
  225. }
  226. transport := (r.client.Transport).(*http.Transport)
  227. if nil != r.proxy {
  228. transport.Proxy = r.proxy
  229. }
  230. if nil != r.dialContext {
  231. transport.DialContext = r.dialContext
  232. }
  233. }
  234. }
  235. func (r *RequestPromise) readResponseBody(resp *http.Response) ([]byte, error) {
  236. defer func(body io.ReadCloser) {
  237. _ = body.Close()
  238. }(resp.Body)
  239. var reader io.ReadCloser
  240. switch resp.Header.Get("Content-Encoding") {
  241. case "gzip":
  242. reader, _ = gzip.NewReader(resp.Body)
  243. defer func(reader io.Closer) {
  244. _ = reader.Close()
  245. }(reader)
  246. case "deflate":
  247. reader = flate.NewReader(resp.Body)
  248. defer func(reader io.Closer) {
  249. _ = reader.Close()
  250. }(reader)
  251. case "zlib":
  252. reader, _ = zlib.NewReader(resp.Body)
  253. defer func(reader io.Closer) {
  254. _ = reader.Close()
  255. }(reader)
  256. default:
  257. reader = resp.Body
  258. }
  259. body, err := ioutil.ReadAll(reader)
  260. if err != nil {
  261. return nil, err
  262. }
  263. if r.encoding != "" {
  264. body = ConvertToEncodingBytes(body, r.encoding)
  265. }
  266. return body, nil
  267. }
  268. // 获取客户端IP地址
  269. func GetRemoteIP(req *http.Request) string {
  270. remoteAddr := req.RemoteAddr
  271. if ip := req.Header.Get("Remote_addr"); ip != "" {
  272. remoteAddr = ip
  273. } else {
  274. remoteAddr, _, _ = net.SplitHostPort(remoteAddr)
  275. }
  276. if remoteAddr == "::1" {
  277. remoteAddr = "127.0.0.1"
  278. }
  279. return remoteAddr
  280. }