jssdk.go 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246
  1. package pay
  2. import (
  3. "bytes"
  4. "encoding/json"
  5. "fmt"
  6. "time"
  7. "unicode/utf8"
  8. "github.com/yaotian/gowechat/mch/base"
  9. "github.com/yaotian/gowechat/util"
  10. )
  11. //OrderInput 下单
  12. //官网文档 https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=9_1
  13. type OrderInput struct {
  14. OpenID string //trade_type=JSAPI时(即公众号支付),此参数必传,此参数为微信用户在商户对应appid下的唯一标识
  15. Body string //String(128)
  16. OutTradeNum string //String(32) 20150806125346 商户系统内部订单号,要求32个字符内,只能是数字、大小写字母_-|*@ ,且在同一个商户号下唯一。
  17. TotalFee int //分为单位
  18. IP string
  19. NotifyURL string //异步接收微信支付结果通知的回调地址,通知url必须为外网可访问的url,不能携带参数
  20. ProductID string //trade_type=NATIVE时(即扫码支付),此参数必传
  21. tradeType string //JSAPI,NATIVE,APP
  22. }
  23. //SetTradeType 设置TradeType
  24. func (c *OrderInput) setTradeType(tradeType string) {
  25. c.tradeType = tradeType
  26. }
  27. //WxPayInfo 统一下单后,返回的信息,这些信息是前端jssdk支付时需要的配置
  28. type WxPayInfo struct {
  29. AppID string `json:"appId"`
  30. TimeStamp string `json:"timeStamp"`
  31. NonceStr string `json:"nonceStr"`
  32. Package string `json:"package"`
  33. SignType string `json:"signType"`
  34. PaySign string `json:"paySign"`
  35. resultMap map[string]string
  36. }
  37. //ToJSON WeixinJSBridge json content
  38. func (c *WxPayInfo) ToJSON() (str string) {
  39. js, err := json.Marshal(c)
  40. if err == nil {
  41. return string(js)
  42. }
  43. return
  44. }
  45. //ToMap result map[string]string
  46. func (c *WxPayInfo) ToMap() (m map[string]string) {
  47. return c.resultMap
  48. }
  49. /*GetJsAPIConfig 前端JsAPI支付时,需要提交的信息
  50. */
  51. func (c *Pay) GetJsAPIConfig(order OrderInput) (config *WxPayInfo, err error) {
  52. order.setTradeType("JSAPI")
  53. err = c.checkOrder(order)
  54. if err != nil {
  55. return
  56. }
  57. var prepayID string
  58. prepayID, err = c.getPrepayID(order)
  59. if err != nil {
  60. return
  61. }
  62. nocestr := util.RandomStr(8)
  63. timestamp := fmt.Sprint(time.Now().Unix())
  64. result := make(map[string]string)
  65. result["appId"] = c.AppID
  66. result["timeStamp"] = timestamp
  67. result["nonceStr"] = nocestr
  68. result["package"] = "prepay_id=" + prepayID
  69. result["signType"] = "MD5"
  70. sign := base.Sign(result, c.MchAPIKey, nil)
  71. result["paySign"] = sign
  72. config = new(WxPayInfo)
  73. config.NonceStr = util.RandomStr(8)
  74. config.TimeStamp = fmt.Sprint(time.Now().Unix())
  75. config.AppID = c.AppID
  76. config.Package = "prepay_id=" + prepayID
  77. config.SignType = "MD5"
  78. config.PaySign = sign
  79. config.resultMap = result
  80. return
  81. }
  82. //GetNativePayQrcodePicURL native支付时二维码图片的url
  83. func (c *Pay) GetNativePayQrcodePicURL(order OrderInput) (qrcodeURL string, err error) {
  84. order.setTradeType("NATIVE")
  85. input := c.createUnifiedOrderMap(order)
  86. var result map[string]string
  87. if result, err = c.UnifiedOrder(input); err == nil { //有prepay_id
  88. qrcodeURL = result["code_url"]
  89. if len(qrcodeURL) == 0 {
  90. err = fmt.Errorf("native pay Qrcode url is empty")
  91. }
  92. }
  93. return
  94. }
  95. // 调用 UnifiedOrder 获得 prepayID
  96. func (c *Pay) getPrepayID(order OrderInput) (prepayID string, err error) {
  97. input := c.createUnifiedOrderMap(order)
  98. var result map[string]string
  99. if result, err = c.UnifiedOrder(input); err == nil { //有prepay_id
  100. prepayID := result["prepay_id"]
  101. if prepayID != "" {
  102. return prepayID, nil
  103. }
  104. err = fmt.Errorf("prepayID is empty")
  105. }
  106. return
  107. }
  108. func (c *Pay) createUnifiedOrderMap(order OrderInput) (input map[string]string) {
  109. input = make(map[string]string)
  110. input["appid"] = c.AppID //设置微信分配的公众账号ID
  111. input["mch_id"] = c.MchID //设置微信支付分配的商户号
  112. input["nonce_str"] = util.RandomStr(5) //设置随机字符串,不长于32位。推荐随机数生成算法
  113. input["body"] = order.Body //获取商品或支付单简要描述的值
  114. input["out_trade_no"] = order.OutTradeNum //设置商户系统内部的订单号,32个字符内、可包含字母, 其他说明见商户订单号
  115. input["total_fee"] = util.ToStr(order.TotalFee) //设置订单总金额,只能为整数,详见支付金额
  116. input["spbill_create_ip"] = order.IP //设置APP和网页支付提交用户端ip,Native支付填调用微信支付API的机器IP。
  117. input["notify_url"] = order.NotifyURL //设置接收微信支付异步通知回调地址
  118. input["trade_type"] = order.tradeType
  119. //设置取值如下:JSAPI,NATIVE,APP,详细说明见参数规定
  120. if order.ProductID != "" {
  121. input["product_id"] = order.ProductID //这个
  122. }
  123. input["openid"] = order.OpenID //设置trade_type=JSAPI,此参数必传,用户在商户appid下的唯一标识。下单前需要调用【网页授权获取用户信息】接口获取到用户的Openid
  124. //sign
  125. sign := base.Sign(input, c.MchAPIKey, nil)
  126. input["sign"] = sign
  127. return
  128. }
  129. func (c *Pay) checkOrder(order OrderInput) (err error) {
  130. tradeType := order.tradeType
  131. if tradeType != "JSAPI" && tradeType != "APP" && tradeType != "NATIVE" {
  132. return fmt.Errorf("tradeType is invalid")
  133. }
  134. if tradeType == "NATIVE" {
  135. if order.ProductID == "" {
  136. err = fmt.Errorf("Native TradeType need ProductID")
  137. return
  138. }
  139. }
  140. if tradeType == "JSAPI" {
  141. if order.OpenID == "" {
  142. err = fmt.Errorf("OpenID can not be empty when pay mode is JSAPI")
  143. return
  144. }
  145. }
  146. if utf8.RuneCountInString(order.Body) > 128 || order.Body == "" {
  147. err = fmt.Errorf("Body is invalid. Size can not exceed 128.")
  148. return
  149. }
  150. if utf8.RuneCountInString(order.OutTradeNum) > 32 || order.OutTradeNum == "" {
  151. err = fmt.Errorf("OutTradeNum is invalid. Size can not exceed 128.")
  152. return
  153. }
  154. if order.TotalFee <= 0 {
  155. err = fmt.Errorf("Order TotalFee is invalid.")
  156. return
  157. }
  158. if order.IP == "" {
  159. err = fmt.Errorf("Order IP is invalid.")
  160. return
  161. }
  162. if order.NotifyURL == "" {
  163. err = fmt.Errorf("Notify URL is invalid.")
  164. return
  165. }
  166. return
  167. }
  168. //CheckPayNotifyData 检查pay notify url收到的消息,是否是返回成功
  169. func (c *Pay) CheckPayNotifyData(data []byte) (isSuccess bool, err error) {
  170. msg, err := base.ParseXMLToMap(bytes.NewReader(data))
  171. if err != nil {
  172. return
  173. }
  174. ReturnCode, ok := msg["return_code"]
  175. if ReturnCode == base.ReturnCodeSuccess || !ok {
  176. haveAppId := msg["appid"]
  177. if haveAppId != c.AppID {
  178. err = fmt.Errorf("get appid is not same as mine. AppID from response is %s. My server AppID is %s,", haveAppId, c.AppID)
  179. return
  180. }
  181. haveMchId := msg["mch_id"]
  182. if haveMchId != c.MchID {
  183. err = fmt.Errorf("get Mch id is not same as mine. MchID from response is %s. My server MchID is %s,", haveMchId, c.MchID)
  184. return
  185. }
  186. signature1, ok := msg["sign"]
  187. if !ok {
  188. err = fmt.Errorf("no sign got")
  189. return
  190. }
  191. signature2 := base.Sign(msg, c.MchAPIKey, nil)
  192. if signature1 != signature2 {
  193. err = fmt.Errorf("Sign is not same. sige_got is %s, sign_mine is %s", signature1, signature2)
  194. return
  195. }
  196. outTradeNum, ok := msg["out_trade_no"]
  197. if !ok || outTradeNum == "" {
  198. err = fmt.Errorf("no out_trade_no")
  199. return
  200. }
  201. result_code, ok := msg["result_code"]
  202. if !ok {
  203. err = fmt.Errorf("no result_code")
  204. return
  205. }
  206. if result_code == base.ResultCodeSuccess {
  207. isSuccess = true
  208. }
  209. }
  210. return
  211. }