api.go 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225
  1. package asign
  2. import (
  3. "bytes"
  4. "crypto"
  5. "crypto/md5"
  6. "crypto/rand"
  7. "crypto/rsa"
  8. "crypto/sha1"
  9. "crypto/x509"
  10. "encoding/base64"
  11. "encoding/hex"
  12. "encoding/pem"
  13. "errors"
  14. "fmt"
  15. "io"
  16. "mime/multipart"
  17. "mtp2_if/config"
  18. "mtp2_if/logger"
  19. "net/http"
  20. "reflect"
  21. "sort"
  22. "strconv"
  23. "strings"
  24. "time"
  25. "github.com/bytedance/sonic"
  26. "github.com/fatih/structs"
  27. )
  28. func APIPost[T_REQ APIReqData, T_RSP APIRspData](req APIReq[T_REQ], apiPort APIURL) (rsp *APIRsp[T_RSP], err error) {
  29. apiUrl := config.SerCfg.AsignCfg.Url + string(apiPort)
  30. var rspBody []byte
  31. rspBody, err = httpPost(apiUrl, req.Data)
  32. if err != nil {
  33. return
  34. }
  35. rspStr := string(rspBody)
  36. if len(rspStr) == 0 {
  37. logger.GetLogger().Error("调用接口 " + apiUrl + " 错误, response为空")
  38. err = errors.New("调用接口返回内容为空")
  39. return
  40. }
  41. rsp = new(APIRsp[T_RSP])
  42. err = sonic.Unmarshal(rspBody, rsp)
  43. return
  44. }
  45. func httpPost(apiUrl string, bizData ...interface{}) (rspBody []byte, err error) {
  46. appId := config.SerCfg.AsignCfg.AppId
  47. privateKey := fmt.Sprintf("-----BEGIN PRIVATE KEY-----\n%s\n-----END PRIVATE KEY-----", config.SerCfg.AsignCfg.PrivateKey)
  48. // fuck asign dev.
  49. timestamp := strconv.Itoa(int(time.Now().UnixMilli() + 10000*60))
  50. // 排序key
  51. // var b []byte
  52. var sortedData string
  53. if len(bizData) == 1 {
  54. // b, err = encoder.Encode(bizData[0], encoder.SortMapKeys)
  55. sortedData = SortMapByKey(structs.Map(bizData[0]))
  56. } else {
  57. // b, err = encoder.Encode(bizData, encoder.SortMapKeys)
  58. sortedData := "["
  59. for i, item := range bizData {
  60. if i > 0 {
  61. sortedData += ","
  62. }
  63. sortedData += SortMapByKey(structs.Map(item))
  64. }
  65. sortedData += "]"
  66. }
  67. if err != nil {
  68. logger.GetLogger().Errorf("处理入参排序失败:" + err.Error())
  69. return
  70. }
  71. // sortedData := string(b)
  72. // 签名
  73. signature, err := getSignature(sortedData, appId, timestamp, privateKey)
  74. if err != nil {
  75. logger.GetLogger().Errorf("签名失败:" + err.Error())
  76. return
  77. }
  78. // 构建form-data请求参数
  79. var requestBody bytes.Buffer
  80. multipartWriter := multipart.NewWriter(&requestBody)
  81. multipartWriter.WriteField("appId", appId)
  82. multipartWriter.WriteField("timestamp", timestamp)
  83. multipartWriter.WriteField("bizData", sortedData)
  84. multipartWriter.Close()
  85. // 构建请求
  86. req, err := http.NewRequest("POST", apiUrl, &requestBody)
  87. // 设置请求头
  88. req.Header.Set("sign", signature)
  89. req.Header.Set("timestamp", timestamp)
  90. req.Header.Set("Content-Type", multipartWriter.FormDataContentType())
  91. // 调用接口
  92. logger.GetLogger().Info("调用接口 "+apiUrl+" 请求, request:", sortedData)
  93. client := &http.Client{}
  94. rsp, err := client.Do(req)
  95. if err != nil {
  96. logger.GetLogger().Error("调用接口 "+apiUrl+" 错误, error:", err.Error())
  97. return
  98. }
  99. defer rsp.Body.Close()
  100. rspBody, err = io.ReadAll(rsp.Body)
  101. logger.GetLogger().Info("调用接口 "+apiUrl+" 返回, response:", string(rspBody))
  102. return
  103. }
  104. // 签名规范:
  105. // 1、表单提交方式:form-data
  106. // 示例:1B2M2Y8AsgTpgAmY7PhCfg==
  107. // 2、请求头部参数
  108. // 参数1:sign(签名值,具体算法参考一下的前面算法)
  109. // 参数2:timestamp(时间戳,13位)
  110. // 3、请求体参数:
  111. // 参数1:appId(appId值,每个接入者唯一一个)
  112. // 参数2:timestamp(时间戳,13位,与上述一致)
  113. // 参数3:bizData(json字符串,举个例子,比方说要传合同编号如:{"contractNo":"0001"})
  114. // 4、签名算法:
  115. // 4.1、将上述3所属的bizData(json字符串),按照阿拉伯字母排序(如:{"ba":1,"ac":2}--->{"ac":2,"ba":1}),
  116. // 4.2、将4.1排序后的字符串,将【bizData+md5(bizData)+ appId + timestatmp】拼接后利用RSA非对称加密算法(SHA1withRSA),计算出最后的签名sign,对其base64编码,放入head的key(sign)中。
  117. //
  118. // String sign ;
  119. // String rsaSuffix = jsonStr + DigestUtils.md5Hex ( jsonStr ) + appId + timestatmp ;
  120. // byte [] bytes = RSAUtils.generateSHA1withRSASignature ( rsaSuffix, privateKey ) ;
  121. // try {
  122. // sign = Base64Utils.encode ( bytes ) ;
  123. // sign = sign.replaceAll ( " \r\n" , "" ) ;
  124. // } catch ( Exception e ) {
  125. // log.error( "sdk异常", e ) ;
  126. // return ApiRespBody.create ( ApiResponseInfo. _ERROR ) ;
  127. // }
  128. func getSignature(sortedData string, appId, timestamp, privateKey string) (signature string, err error) {
  129. pem10, _ := pem.Decode([]byte(privateKey))
  130. privateKey10I, _ := x509.ParsePKCS8PrivateKey(pem10.Bytes)
  131. privateKey10 := privateKey10I.(*rsa.PrivateKey)
  132. // md5(bizData)
  133. m := md5.New()
  134. m.Write([]byte(sortedData))
  135. bdMd5Hx := hex.EncodeToString(m.Sum(nil))
  136. // 待签内容
  137. message := sortedData + bdMd5Hx + appId + timestamp
  138. h := sha1.New()
  139. h.Write([]byte(message))
  140. sum := h.Sum(nil)
  141. // 使用私钥进行签名
  142. sign, err := rsa.SignPKCS1v15(rand.Reader, privateKey10, crypto.SHA1, sum)
  143. if err != nil {
  144. return
  145. }
  146. // fmt.Println(signature)
  147. signature = base64.StdEncoding.EncodeToString(sign)
  148. signature = strings.ReplaceAll(signature, "\r\n", "")
  149. return
  150. }
  151. // SortMapByKey 按key排序map返回string -- 已经不使用的方法
  152. func SortMapByKey(data map[string]interface{}) (sortedData string) {
  153. keys := make([]string, 0, len(data))
  154. for k := range data {
  155. keys = append(keys, k)
  156. }
  157. sort.Strings(keys)
  158. for i, k := range keys {
  159. if i > 0 {
  160. sortedData += ","
  161. }
  162. // 判断是否指针
  163. v := reflect.ValueOf(data[k])
  164. if v.Kind() == reflect.Ptr {
  165. switch data[k].(type) {
  166. case *string:
  167. sortedData += fmt.Sprintf(`"%s":"%s"`, k, *(data[k].(*string)))
  168. case *map[string]interface{}:
  169. sortedData += fmt.Sprintf(`"%s":%s`, k, SortMapByKey(*(data[k].(*map[string]interface{}))))
  170. case *[]interface{}:
  171. list := data[k].([]interface{})
  172. sortedData += fmt.Sprintf(`"%s":[`, k)
  173. for j, item := range list {
  174. if j > 0 {
  175. sortedData += ","
  176. }
  177. sortedData += SortMapByKey(item.(map[string]interface{}))
  178. }
  179. sortedData += "]"
  180. default:
  181. sortedData += fmt.Sprintf(`"%s":%v`, k, reflect.ValueOf(data[k]).Elem())
  182. }
  183. } else {
  184. switch data[k].(type) {
  185. case string:
  186. sortedData += fmt.Sprintf(`"%s":"%s"`, k, data[k].(string))
  187. case map[string]interface{}:
  188. sortedData += fmt.Sprintf(`"%s":%s`, k, SortMapByKey(data[k].(map[string]interface{})))
  189. case []interface{}:
  190. list := data[k].([]interface{})
  191. sortedData += fmt.Sprintf(`"%s":[`, k)
  192. for j, item := range list {
  193. if j > 0 {
  194. sortedData += ","
  195. }
  196. sortedData += SortMapByKey(item.(map[string]interface{}))
  197. }
  198. sortedData += "]"
  199. default:
  200. sortedData += fmt.Sprintf(`"%s":%v`, k, data[k])
  201. }
  202. }
  203. }
  204. return fmt.Sprintf("{%s}", sortedData)
  205. }