pay.go 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205
  1. package payermax
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. "io"
  6. "net/http"
  7. "strconv"
  8. "strings"
  9. "bet24.com/log"
  10. "bet24.com/public"
  11. "bet24.com/servers/common"
  12. coreClient "bet24.com/servers/coreservice/client"
  13. "bet24.com/servers/payment/config"
  14. "bet24.com/servers/payment/db"
  15. "bet24.com/servers/payment/shop"
  16. "github.com/gin-gonic/gin"
  17. )
  18. // 下单
  19. func PayOrder(c *gin.Context) {
  20. obj := db.NewOrder(db.SP_PayerMax_ORDER)
  21. if err := c.ShouldBind(&obj.In); err != nil {
  22. log.Debug("%s query params err %v", "payermax.PayOrder", err)
  23. c.String(http.StatusOK, "")
  24. return
  25. }
  26. obj.In.IpAddress = strings.Split(c.Request.RemoteAddr, ":")[0]
  27. //// 币种为空,根据ip获取
  28. //if obj.In.Currency == "" {
  29. // currency := shop.GetCurrencyRateByIp(obj.In.UserID, obj.In.IpAddress)
  30. // obj.In.Currency = currency
  31. //}
  32. // 获取产品信息
  33. item := shop.GetProduct(obj.In.ProductID)
  34. if item == nil {
  35. log.Error("%s query GetProduct productId=%s currency=%s is nil", "payermax.PayOrder", obj.In.ProductID, obj.In.Currency)
  36. c.String(http.StatusOK, "")
  37. return
  38. }
  39. //// 获取当前汇率信息
  40. //info := shop.GetExchangeRate(obj.In.Currency)
  41. //if info == nil {
  42. // log.Error("%s query GetExchangeRate obj.In.Currency=%s is nil", "payermax.PayOrder", obj.In.Currency)
  43. // c.String(http.StatusOK, "")
  44. // return
  45. //}
  46. //
  47. //// 计算价格
  48. //calPrice := info.Rate * item.Price
  49. //
  50. //// 检查价格是否篡改
  51. //if calPrice != obj.In.Price {
  52. // log.Error("%s obj.In.Price=%v info.Rate=%v calPrice=%v is invalid", "payermax.PayOrder", obj.In.Price, info.Rate, calPrice)
  53. // c.String(http.StatusOK, "")
  54. // return
  55. //}
  56. // 检查价格是否篡改
  57. if item.Price != obj.In.Price {
  58. log.Error("%s obj.In.Price=%v calPrice=%v is invalid", "payermax.PayOrder", obj.In.Price, item.Price)
  59. c.String(http.StatusOK, "")
  60. return
  61. }
  62. obj.DoAction(nil)
  63. if obj.Out.OrderID == "" {
  64. log.Debug("%s GenOrder fail obj.In=%+v", "payermax.PayOrder", obj.In)
  65. c.String(http.StatusOK, "")
  66. return
  67. }
  68. // 请求payOrder的代码
  69. req := pay_req{
  70. Version: "1.0",
  71. KeyVersion: "1",
  72. RequestTime: common.GetNowTime().Format(dateFormat),
  73. MerchantAppId: config.Server.PayerMax.MerchantAppId,
  74. MerchantNo: config.Server.PayerMax.MerchantNo,
  75. Data: payReqData{
  76. OutTradeNo: obj.Out.OrderID,
  77. Subject: obj.In.ProductID,
  78. TotalAmount: fmt.Sprintf("%d", int(obj.In.Price)),
  79. Currency: obj.In.Currency,
  80. Country: obj.In.Country,
  81. UserId: strconv.Itoa(obj.In.UserID),
  82. FrontCallbackUrl: config.Server.PayerMax.FrontCallbackUrl,
  83. NotifyUrl: config.Server.PayerMax.Url_pay_Notify,
  84. },
  85. }
  86. // 生成签名
  87. checkContent, err := json.Marshal(req)
  88. if err != nil {
  89. log.Error("payermax.PayOrder json marshal fail %v", err)
  90. return
  91. }
  92. sign := public.Sha256WithRsa(string(checkContent), config.Server.PayerMax.MCH_PRIVATE_KEY)
  93. log.Debug("payermax.payOrder checkContent=%s sign=%s", string(checkContent), sign)
  94. // POST请求
  95. respBody := httpPostByJson(config.Server.PayerMax.Url_pay_order, string(checkContent), sign)
  96. log.Debug("payermax.payOrder req ==> %+v resp ==> %+v", req, respBody)
  97. var resp pay_resp
  98. if err := json.Unmarshal([]byte(respBody), &resp); err != nil {
  99. log.Error("payermax.payOrder json unmarshal req ==> %+v resp ==> %+v fail %v", req, respBody, err)
  100. return
  101. }
  102. log.Debug("payermax.payOrder resp ==> %+v", resp)
  103. // 返回码,’APPLY_SUCCESS’代表成功
  104. if resp.Code != "APPLY_SUCCESS" {
  105. log.Error("payermax.payOrder post return resp fail ==> %+v", resp)
  106. return
  107. }
  108. // 跳转到支付页面,以便持卡人完成支付过程
  109. c.Redirect(http.StatusMovedPermanently, resp.Data.RedirectUrl)
  110. // c.String(http.StatusOK, "Success")
  111. return
  112. }
  113. // 回调通知
  114. func PayNotify(c *gin.Context) {
  115. bodyData, _ := io.ReadAll(c.Request.Body)
  116. log.Debug("payermax.PayNotify ==> ctx.Request.body: %v", string(bodyData))
  117. // 获取签名
  118. sign := c.Request.Header.Get("sign")
  119. log.Debug("payermax.PayNotify sign=%s", sign)
  120. var resp payNotify
  121. if err := json.Unmarshal(bodyData, &resp); err != nil {
  122. log.Debug("%s query params err %v", "payermax.PayNotify", err)
  123. c.String(http.StatusOK, "")
  124. return
  125. }
  126. log.Debug("payermax.PayNotify resp ==> %+v", resp)
  127. // 验证签名
  128. if public.VerifyRsaSignBySHA256(string(bodyData), sign, config.Server.PayerMax.PLAT_PUBLIC_KEY) != nil {
  129. log.Error("payermax.PayNotify 签名失败 ==> %+v", resp)
  130. return
  131. }
  132. log.Debug("payermax.PayNotify 签名成功")
  133. // 返回码,’APPLY_SUCCESS’代表成功
  134. if resp.Code != "APPLY_SUCCESS" {
  135. log.Error("payermax.PayNotify resp ==> %+v 失败", resp)
  136. c.JSON(http.StatusOK, ret{
  137. Code: "SUCCESS",
  138. Msg: "Success",
  139. })
  140. return
  141. }
  142. // 数据库操作
  143. obj := db.NewNotify(db.SP_PayerMax_NOTIFY)
  144. obj.In.OrderID = resp.Data.OutTradeNo
  145. obj.In.TradeID = resp.Data.TradeToken
  146. // obj.In.Price = resp.Data.TotalAmount
  147. obj.DoAction(nil)
  148. // 操作成功,给道具
  149. if obj.Out.RetCode == 1 {
  150. // 充值
  151. resp := coreClient.Recharge(obj.Out.UserID, obj.Out.ProductID)
  152. log.Debug("%s 充值成功 %+v", "payermax.PayNotify", resp)
  153. }
  154. c.JSON(http.StatusOK, ret{
  155. Code: "SUCCESS",
  156. Msg: "Success",
  157. })
  158. return
  159. }
  160. // 支付完成跳转处理
  161. func PayCallback(c *gin.Context) {
  162. var resp payReturn
  163. if err := c.ShouldBind(&resp); err != nil {
  164. log.Debug("%s query params err %v", "payermax.PayCallback", err)
  165. c.String(http.StatusOK, "")
  166. return
  167. }
  168. log.Debug("payermax.PayCallback ==> %+v", resp)
  169. c.HTML(http.StatusOK, "payermax.html", gin.H{
  170. "title": "PayerMax Recharge Result",
  171. "outTradeNo": resp.OutTradeNo,
  172. "tradeToken": resp.TradeToken,
  173. "status": resp.Status,
  174. })
  175. return
  176. }