http_client.go 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
  1. package httpClient
  2. import (
  3. "context"
  4. "encoding/json"
  5. "errors"
  6. "eta/eta_mini_crm_ht/utils"
  7. "fmt"
  8. "github.com/mitchellh/mapstructure"
  9. "io"
  10. "net/http"
  11. "strings"
  12. "time"
  13. )
  14. type HttpClient struct {
  15. *http.Client
  16. maxRetries int
  17. retryDelayFunc RetryDelayFunc
  18. header map[string]string
  19. }
  20. func (hc *HttpClient) AddHeader(header map[string]string) {
  21. hc.header = header
  22. }
  23. func (hc *HttpClient) StructToMap(data interface{}) (dataMap map[string]string, err error) {
  24. if data == nil {
  25. utils.FileLog.Warn("请求data为空")
  26. return
  27. }
  28. err = mapstructure.Decode(data, &dataMap)
  29. if err != nil {
  30. utils.FileLog.Error("结构体转Map失败", err.Error())
  31. return
  32. }
  33. return
  34. }
  35. // NewClient 构造函数,其中 delayFunc 参数是可选的
  36. func NewClient(timeout time.Duration, maxRetries int, delayFunc ...RetryDelayFunc) *HttpClient {
  37. var df RetryDelayFunc
  38. if len(delayFunc) > 0 {
  39. df = delayFunc[0]
  40. } else {
  41. df = defaultRetryDelayFunc
  42. }
  43. return &HttpClient{
  44. Client: &http.Client{Timeout: timeout},
  45. maxRetries: maxRetries,
  46. retryDelayFunc: df,
  47. }
  48. }
  49. func DefaultClient() *HttpClient {
  50. return NewClient(time.Second*10, 3)
  51. }
  52. func defaultRetryDelayFunc(attempt int) time.Duration {
  53. delay := time.Duration(attempt) * time.Second
  54. if attempt > 0 {
  55. delay *= 2
  56. }
  57. return delay
  58. }
  59. type RetryDelayFunc func(attempt int) time.Duration
  60. func retryErr(err error) bool {
  61. return errors.Is(err, context.DeadlineExceeded) || errors.Is(err, context.Canceled)
  62. }
  63. // DoWithRetry 发送带有重试机制的HTTP请求,允许用户自定义重试延迟逻辑
  64. func (hc *HttpClient) DoWithRetry(ctx context.Context, req *http.Request) (resp *http.Response, err error) {
  65. attempt := 0
  66. for {
  67. resp, err = hc.Do(req.WithContext(ctx))
  68. if err != nil && retryErr(err) {
  69. if attempt >= hc.maxRetries {
  70. return nil, fmt.Errorf("请求失败: %w", err)
  71. }
  72. attempt++
  73. delay := hc.retryDelayFunc(attempt)
  74. time.Sleep(delay)
  75. continue
  76. }
  77. return
  78. }
  79. }
  80. func (hc *HttpClient) buildHeader(header *http.Header) {
  81. for k, v := range hc.header {
  82. header.Add(k, v)
  83. }
  84. }
  85. func (hc *HttpClient) Post(url string, data interface{}) (resp *http.Response, err error) {
  86. dataStr, err := json.Marshal(data)
  87. if err != nil {
  88. utils.FileLog.Error("请求data json序列化失败,err:" + err.Error())
  89. }
  90. body := io.NopCloser(strings.NewReader(string(dataStr)))
  91. req, err := http.NewRequest(http.MethodPost, url, body)
  92. hc.buildHeader(&req.Header)
  93. req.Header.Set("Content-Type", "application/json")
  94. if err != nil {
  95. utils.FileLog.Error("创建POST请求失败: %v", err)
  96. }
  97. resp, err = hc.DoWithRetry(req.Context(), req)
  98. if err == nil {
  99. code := resp.StatusCode
  100. if code != 200 {
  101. utils.FileLog.Error("请求错误应答,状态码:%d", code)
  102. errMsg := fmt.Sprintf("请求状态码异常,StatusCode:[%d]", code)
  103. respBody, respErr := io.ReadAll(resp.Body)
  104. if respErr != nil {
  105. utils.FileLog.Error("读取body失败,err:%v", err)
  106. err = errors.New(errMsg)
  107. return
  108. }
  109. utils.FileLog.Error("请求错误应答,body:%s", string(respBody))
  110. errMsg = fmt.Sprintf("%s,body:%s", errMsg, string(respBody))
  111. err = errors.New(errMsg)
  112. return
  113. }
  114. } else {
  115. utils.FileLog.Error("未知的应答错误,获取第三方授权信息失败")
  116. }
  117. return
  118. }
  119. func (hc *HttpClient) PostWithAuth(url string, data interface{}, token string) (resp *http.Response, err error) {
  120. dataStr, err := json.Marshal(data)
  121. if err != nil {
  122. utils.FileLog.Error("请求data json序列化失败,err:" + err.Error())
  123. }
  124. body := io.NopCloser(strings.NewReader(string(dataStr)))
  125. req, err := http.NewRequest(http.MethodPost, url, body)
  126. hc.buildHeader(&req.Header)
  127. req.Header.Set("Content-Type", "application/json")
  128. req.Header.Set("Authorization", token)
  129. if err != nil {
  130. utils.FileLog.Error("创建POST请求失败: %v", err)
  131. }
  132. resp, err = hc.DoWithRetry(req.Context(), req)
  133. code := resp.StatusCode
  134. if code != 200 {
  135. utils.FileLog.Error("请求错误应答,状态码:%d", code)
  136. errMsg := fmt.Sprintf("请求状态码异常,StatusCode:[%d]", code)
  137. respBody, respErr := io.ReadAll(resp.Body)
  138. if respErr != nil {
  139. utils.FileLog.Error("读取body失败,err:%v", err)
  140. err = errors.New(errMsg)
  141. return
  142. }
  143. utils.FileLog.Error("请求错误应答,body:%s", string(respBody))
  144. errMsg = fmt.Sprintf("%s,body:%s", errMsg, string(respBody))
  145. err = errors.New(errMsg)
  146. return
  147. }
  148. return
  149. }
  150. func (hc *HttpClient) Get(url string) (resp *http.Response, err error) {
  151. req, err := http.NewRequest(http.MethodGet, url, nil)
  152. hc.buildHeader(&req.Header)
  153. if err != nil {
  154. utils.FileLog.Error("创建请求失败: %v", err)
  155. }
  156. resp, err = hc.DoWithRetry(req.Context(), req)
  157. return
  158. }