http.go 5.8 KB

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