hz_data_api.go 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142
  1. package services
  2. import (
  3. "bytes"
  4. "crypto/md5"
  5. "encoding/json"
  6. "errors"
  7. "eta/eta_task/utils"
  8. "fmt"
  9. "github.com/shopspring/decimal"
  10. "io/ioutil"
  11. "net/http"
  12. "reflect"
  13. "sort"
  14. "strings"
  15. "time"
  16. )
  17. const (
  18. HZ_DATA_API = "https://hzdataapi.hzinsights.com/hzdataapi/"
  19. )
  20. func HttpPost(method string, postDataMap map[string]interface{}) (result string, err error) {
  21. bytesData, err := handlePostData(postDataMap)
  22. if err != nil {
  23. return
  24. }
  25. client := &http.Client{}
  26. req, err := http.NewRequest("POST", HZ_DATA_API+method, bytes.NewReader(bytesData))
  27. if err != nil {
  28. return "", err
  29. }
  30. req.Header.Set("content-type", "application/json")
  31. resp, err := client.Do(req)
  32. if err != nil {
  33. return "", err
  34. }
  35. body, err := ioutil.ReadAll(resp.Body)
  36. if err != nil {
  37. return "", err
  38. }
  39. result = string(body)
  40. return
  41. }
  42. // handlePostData 处理post请求数据
  43. func handlePostData(postDataMap map[string]interface{}) (bytesData []byte, err error) {
  44. postDataMap["nonce_str"] = utils.GetRandString(16) // 随机字符串
  45. postDataMap["timestamp"] = time.Now().Unix() //当前格林威治时间,int64类型
  46. postDataMap["appid"] = utils.APPID //当前格林威治时间,int64类型
  47. // 待签名数据
  48. signData := convertParamInterface(postDataMap)
  49. sign, err := getSignData(signData)
  50. if err != nil {
  51. return
  52. }
  53. postDataMap["sign"] = sign //签名
  54. bytesData, err = json.Marshal(postDataMap)
  55. return
  56. }
  57. // 将请求传入的数据格式转换成签名需要的格式(目前只能处理简单的类型,数组、对象暂不支持)
  58. func convertParamInterface(params map[string]interface{}) (signData map[string]string) {
  59. signData = make(map[string]string)
  60. for key := range params {
  61. val := ``
  62. //fmt.Println("key", key, ";val:", params[key], ";type:", reflect.TypeOf(params[key]))
  63. //signData[key] = params[key][0]
  64. tmpVal := params[key]
  65. switch reflect.TypeOf(tmpVal).Kind() {
  66. case reflect.String:
  67. val = fmt.Sprint(tmpVal)
  68. case reflect.Int, reflect.Int16, reflect.Int64, reflect.Int32, reflect.Int8:
  69. val = fmt.Sprint(tmpVal)
  70. case reflect.Uint, reflect.Uint32, reflect.Uint16, reflect.Uint8, reflect.Uint64:
  71. val = fmt.Sprint(tmpVal)
  72. case reflect.Bool:
  73. val = fmt.Sprint(tmpVal)
  74. case reflect.Float64:
  75. decimalNum := decimal.NewFromFloat(tmpVal.(float64))
  76. val = decimalNum.String()
  77. //val = strconv.FormatFloat(tmpVal.(float64), 'E', -1, 64) //float64
  78. case reflect.Float32:
  79. decimalNum := decimal.NewFromFloat32(tmpVal.(float32))
  80. val = decimalNum.String()
  81. }
  82. signData[key] = val
  83. }
  84. return signData
  85. }
  86. // getSignData 获取参数签名
  87. func getSignData(postData map[string]string) (sign string, err error) {
  88. appid := utils.APPID
  89. if appid == "" {
  90. err = errors.New("参数异常,缺少appid")
  91. return
  92. }
  93. secret := utils.SECRET
  94. if secret == "" {
  95. err = errors.New("参数异常,缺少secret")
  96. return
  97. }
  98. if postData["nonce_str"] == "" {
  99. err = errors.New("参数异常,缺少随机字符串")
  100. return
  101. }
  102. if postData["timestamp"] == "" {
  103. err = errors.New("参数异常,缺少时间戳")
  104. return
  105. }
  106. //先取出除sign外的所有的提交的参数key
  107. var keys []string
  108. for k := range postData {
  109. if k != "sign" {
  110. keys = append(keys, k)
  111. }
  112. }
  113. //1,根据参数名称的ASCII码表的顺序排序
  114. sort.Strings(keys)
  115. //2 根据排序后的参数名称,取出对应的值,并拼接字符串
  116. var signStr string
  117. for _, v := range keys {
  118. signStr += v + "=" + postData[v] + "&"
  119. }
  120. //3,全转小写(md5(拼装的字符串后+分配给你的app_secret))
  121. //sign := strings.ToLower(fmt.Sprintf("%x", md5.Sum([]byte(strings.Trim(signStr, "&")+key))))
  122. //md5.Sum([]byte(signStr+"key="+key)) 这是md5加密出来后的每个字符的ascall码,需要再转换成对应的字符
  123. //3,全转大写(md5(拼装的字符串后+分配给你的app_secret))
  124. sign = strings.ToUpper(fmt.Sprintf("%x", md5.Sum([]byte(signStr+"secret="+secret))))
  125. return
  126. }