http_utils.go 8.3 KB

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