http.go 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282
  1. package util
  2. import (
  3. "bytes"
  4. "context"
  5. "crypto/tls"
  6. "encoding/json"
  7. "encoding/pem"
  8. "encoding/xml"
  9. "fmt"
  10. "io"
  11. "log"
  12. "mime/multipart"
  13. "net/http"
  14. "os"
  15. "golang.org/x/crypto/pkcs12"
  16. )
  17. // HTTPGet get 请求
  18. func HTTPGet(uri string) ([]byte, error) {
  19. return HTTPGetContext(context.Background(), uri)
  20. }
  21. // HTTPGetContext get 请求
  22. func HTTPGetContext(ctx context.Context, uri string) ([]byte, error) {
  23. request, err := http.NewRequestWithContext(ctx, http.MethodGet, uri, nil)
  24. if err != nil {
  25. return nil, err
  26. }
  27. response, err := http.DefaultClient.Do(request)
  28. if err != nil {
  29. return nil, err
  30. }
  31. defer response.Body.Close()
  32. if response.StatusCode != http.StatusOK {
  33. return nil, fmt.Errorf("http get error : uri=%v , statusCode=%v", uri, response.StatusCode)
  34. }
  35. return io.ReadAll(response.Body)
  36. }
  37. // HTTPPost post 请求
  38. func HTTPPost(uri string, data string) ([]byte, error) {
  39. return HTTPPostContext(context.Background(), uri, []byte(data), nil)
  40. }
  41. // HTTPPostContext post 请求
  42. func HTTPPostContext(ctx context.Context, uri string, data []byte, header map[string]string) ([]byte, error) {
  43. body := bytes.NewBuffer(data)
  44. request, err := http.NewRequestWithContext(ctx, http.MethodPost, uri, body)
  45. if err != nil {
  46. return nil, err
  47. }
  48. for key, value := range header {
  49. request.Header.Set(key, value)
  50. }
  51. response, err := http.DefaultClient.Do(request)
  52. if err != nil {
  53. return nil, err
  54. }
  55. defer response.Body.Close()
  56. if response.StatusCode != http.StatusOK {
  57. return nil, fmt.Errorf("http post error : uri=%v , statusCode=%v", uri, response.StatusCode)
  58. }
  59. return io.ReadAll(response.Body)
  60. }
  61. // PostJSONContext post json 数据请求
  62. func PostJSONContext(ctx context.Context, uri string, obj interface{}) ([]byte, error) {
  63. jsonBuf := new(bytes.Buffer)
  64. enc := json.NewEncoder(jsonBuf)
  65. enc.SetEscapeHTML(false)
  66. err := enc.Encode(obj)
  67. if err != nil {
  68. return nil, err
  69. }
  70. req, err := http.NewRequestWithContext(ctx, "POST", uri, jsonBuf)
  71. if err != nil {
  72. return nil, err
  73. }
  74. req.Header.Set("Content-Type", "application/json;charset=utf-8")
  75. response, err := http.DefaultClient.Do(req)
  76. if err != nil {
  77. return nil, err
  78. }
  79. defer response.Body.Close()
  80. if response.StatusCode != http.StatusOK {
  81. return nil, fmt.Errorf("http get error : uri=%v , statusCode=%v", uri, response.StatusCode)
  82. }
  83. return io.ReadAll(response.Body)
  84. }
  85. // PostJSON post json 数据请求
  86. func PostJSON(uri string, obj interface{}) ([]byte, error) {
  87. return PostJSONContext(context.Background(), uri, obj)
  88. }
  89. // PostJSONWithRespContentType post json 数据请求,且返回数据类型
  90. func PostJSONWithRespContentType(uri string, obj interface{}) ([]byte, string, error) {
  91. jsonBuf := new(bytes.Buffer)
  92. enc := json.NewEncoder(jsonBuf)
  93. enc.SetEscapeHTML(false)
  94. err := enc.Encode(obj)
  95. if err != nil {
  96. return nil, "", err
  97. }
  98. response, err := http.Post(uri, "application/json;charset=utf-8", jsonBuf)
  99. if err != nil {
  100. return nil, "", err
  101. }
  102. defer response.Body.Close()
  103. if response.StatusCode != http.StatusOK {
  104. return nil, "", fmt.Errorf("http get error : uri=%v , statusCode=%v", uri, response.StatusCode)
  105. }
  106. responseData, err := io.ReadAll(response.Body)
  107. contentType := response.Header.Get("Content-Type")
  108. return responseData, contentType, err
  109. }
  110. // PostFile 上传文件
  111. func PostFile(fieldName, filename, uri string) ([]byte, error) {
  112. fields := []MultipartFormField{
  113. {
  114. IsFile: true,
  115. Fieldname: fieldName,
  116. Filename: filename,
  117. },
  118. }
  119. return PostMultipartForm(fields, uri)
  120. }
  121. // MultipartFormField 保存文件或其他字段信息
  122. type MultipartFormField struct {
  123. IsFile bool
  124. Fieldname string
  125. Value []byte
  126. Filename string
  127. }
  128. // PostMultipartForm 上传文件或其他多个字段
  129. func PostMultipartForm(fields []MultipartFormField, uri string) (respBody []byte, err error) {
  130. bodyBuf := &bytes.Buffer{}
  131. bodyWriter := multipart.NewWriter(bodyBuf)
  132. for _, field := range fields {
  133. if field.IsFile {
  134. fileWriter, e := bodyWriter.CreateFormFile(field.Fieldname, field.Filename)
  135. if e != nil {
  136. err = fmt.Errorf("error writing to buffer , err=%v", e)
  137. return
  138. }
  139. fh, e := os.Open(field.Filename)
  140. if e != nil {
  141. err = fmt.Errorf("error opening file , err=%v", e)
  142. return
  143. }
  144. defer fh.Close()
  145. if _, err = io.Copy(fileWriter, fh); err != nil {
  146. return
  147. }
  148. } else {
  149. partWriter, e := bodyWriter.CreateFormField(field.Fieldname)
  150. if e != nil {
  151. err = e
  152. return
  153. }
  154. valueReader := bytes.NewReader(field.Value)
  155. if _, err = io.Copy(partWriter, valueReader); err != nil {
  156. return
  157. }
  158. }
  159. }
  160. contentType := bodyWriter.FormDataContentType()
  161. bodyWriter.Close()
  162. resp, e := http.Post(uri, contentType, bodyBuf)
  163. if e != nil {
  164. err = e
  165. return
  166. }
  167. defer resp.Body.Close()
  168. if resp.StatusCode != http.StatusOK {
  169. return nil, err
  170. }
  171. respBody, err = io.ReadAll(resp.Body)
  172. return
  173. }
  174. // PostXML perform a HTTP/POST request with XML body
  175. func PostXML(uri string, obj interface{}) ([]byte, error) {
  176. xmlData, err := xml.Marshal(obj)
  177. if err != nil {
  178. return nil, err
  179. }
  180. body := bytes.NewBuffer(xmlData)
  181. response, err := http.Post(uri, "application/xml;charset=utf-8", body)
  182. if err != nil {
  183. return nil, err
  184. }
  185. defer response.Body.Close()
  186. if response.StatusCode != http.StatusOK {
  187. return nil, fmt.Errorf("http code error : uri=%v , statusCode=%v", uri, response.StatusCode)
  188. }
  189. return io.ReadAll(response.Body)
  190. }
  191. // httpWithTLS CA 证书
  192. func httpWithTLS(rootCa, key string) (*http.Client, error) {
  193. var client *http.Client
  194. certData, err := os.ReadFile(rootCa)
  195. if err != nil {
  196. return nil, fmt.Errorf("unable to find cert path=%s, error=%v", rootCa, err)
  197. }
  198. cert := pkcs12ToPem(certData, key)
  199. config := &tls.Config{
  200. Certificates: []tls.Certificate{cert},
  201. }
  202. tr := &http.Transport{
  203. TLSClientConfig: config,
  204. DisableCompression: true,
  205. }
  206. client = &http.Client{Transport: tr}
  207. return client, nil
  208. }
  209. // pkcs12ToPem 将 Pkcs12 转成 Pem
  210. func pkcs12ToPem(p12 []byte, password string) tls.Certificate {
  211. blocks, err := pkcs12.ToPEM(p12, password)
  212. defer func() {
  213. if x := recover(); x != nil {
  214. log.Print(x)
  215. }
  216. }()
  217. if err != nil {
  218. panic(err)
  219. }
  220. var pemData []byte
  221. for _, b := range blocks {
  222. pemData = append(pemData, pem.EncodeToMemory(b)...)
  223. }
  224. cert, err := tls.X509KeyPair(pemData, pemData)
  225. if err != nil {
  226. panic(err)
  227. }
  228. return cert
  229. }
  230. // PostXMLWithTLS perform a HTTP/POST request with XML body and TLS
  231. func PostXMLWithTLS(uri string, obj interface{}, ca, key string) ([]byte, error) {
  232. xmlData, err := xml.Marshal(obj)
  233. if err != nil {
  234. return nil, err
  235. }
  236. body := bytes.NewBuffer(xmlData)
  237. client, err := httpWithTLS(ca, key)
  238. if err != nil {
  239. return nil, err
  240. }
  241. response, err := client.Post(uri, "application/xml;charset=utf-8", body)
  242. if err != nil {
  243. return nil, err
  244. }
  245. defer response.Body.Close()
  246. if response.StatusCode != http.StatusOK {
  247. return nil, fmt.Errorf("http code error : uri=%v , statusCode=%v", uri, response.StatusCode)
  248. }
  249. return io.ReadAll(response.Body)
  250. }