wechat_client.go 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335
  1. package gopay
  2. import (
  3. "crypto/tls"
  4. "encoding/xml"
  5. "errors"
  6. "fmt"
  7. "strings"
  8. "sync"
  9. )
  10. type WeChatClient struct {
  11. AppId string
  12. MchId string
  13. ApiKey string
  14. BaseURL string
  15. CertFile []byte
  16. KeyFile []byte
  17. Pkcs12File []byte
  18. IsProd bool
  19. mu sync.RWMutex
  20. }
  21. // 初始化微信客户端
  22. // appId:应用ID
  23. // mchId:商户ID
  24. // ApiKey:API秘钥值
  25. // IsProd:是否是正式环境
  26. func NewWeChatClient(appId, mchId, apiKey string, isProd bool) (client *WeChatClient) {
  27. return &WeChatClient{
  28. AppId: appId,
  29. MchId: mchId,
  30. ApiKey: apiKey,
  31. IsProd: isProd}
  32. }
  33. // 提交付款码支付
  34. // 文档地址:https://pay.weixin.qq.com/wiki/doc/api/micropay.php?chapter=9_10&index=1
  35. func (w *WeChatClient) Micropay(body BodyMap) (wxRsp *WeChatMicropayResponse, err error) {
  36. var bs []byte
  37. if w.IsProd {
  38. bs, err = w.doWeChat(body, wxMicropay)
  39. } else {
  40. bs, err = w.doWeChat(body, wxSandboxMicropay)
  41. }
  42. if err != nil {
  43. return
  44. }
  45. wxRsp = new(WeChatMicropayResponse)
  46. if err = xml.Unmarshal(bs, wxRsp); err != nil {
  47. return nil, fmt.Errorf("xml.Unmarshal:%s", err.Error())
  48. }
  49. return
  50. }
  51. // 统一下单
  52. // 文档地址:https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=9_1
  53. func (w *WeChatClient) UnifiedOrder(body BodyMap) (wxRsp *WeChatUnifiedOrderResponse, err error) {
  54. var bs []byte
  55. if w.IsProd {
  56. bs, err = w.doWeChat(body, wxUnifiedorder)
  57. } else {
  58. body.Set("total_fee", 101)
  59. bs, err = w.doWeChat(body, wxSandboxUnifiedorder)
  60. }
  61. if err != nil {
  62. return
  63. }
  64. wxRsp = new(WeChatUnifiedOrderResponse)
  65. if err = xml.Unmarshal(bs, wxRsp); err != nil {
  66. return nil, fmt.Errorf("xml.Unmarshal:%s", err.Error())
  67. }
  68. return
  69. }
  70. // 查询订单
  71. // 文档地址:https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=9_2
  72. func (w *WeChatClient) QueryOrder(body BodyMap) (wxRsp *WeChatQueryOrderResponse, err error) {
  73. var bs []byte
  74. if w.IsProd {
  75. bs, err = w.doWeChat(body, wxOrderquery)
  76. } else {
  77. bs, err = w.doWeChat(body, wxSandboxOrderquery)
  78. }
  79. if err != nil {
  80. return
  81. }
  82. wxRsp = new(WeChatQueryOrderResponse)
  83. if err = xml.Unmarshal(bs, wxRsp); err != nil {
  84. return nil, fmt.Errorf("xml.Unmarshal:%s", err.Error())
  85. }
  86. return
  87. }
  88. // 关闭订单
  89. // 文档地址:https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=9_3
  90. func (w *WeChatClient) CloseOrder(body BodyMap) (wxRsp *WeChatCloseOrderResponse, err error) {
  91. var bs []byte
  92. if w.IsProd {
  93. bs, err = w.doWeChat(body, wxCloseorder)
  94. } else {
  95. bs, err = w.doWeChat(body, wxSandboxCloseorder)
  96. }
  97. if err != nil {
  98. return
  99. }
  100. wxRsp = new(WeChatCloseOrderResponse)
  101. if err = xml.Unmarshal(bs, wxRsp); err != nil {
  102. return nil, fmt.Errorf("xml.Unmarshal:%s", err.Error())
  103. }
  104. return
  105. }
  106. // 撤销订单
  107. // 注意:如已使用client.AddCertFilePath()或client.AddCertFileByte()添加过证书,参数certFilePath、keyFilePath、pkcs12FilePath全传空字符串 "",如方法需单独使用证书,则传证书Path
  108. // 文档地址:https://pay.weixin.qq.com/wiki/doc/api/micropay.php?chapter=9_11&index=3
  109. func (w *WeChatClient) Reverse(body BodyMap, certFilePath, keyFilePath, pkcs12FilePath string) (wxRsp *WeChatReverseResponse, err error) {
  110. var (
  111. bs []byte
  112. tlsConfig *tls.Config
  113. )
  114. if w.IsProd {
  115. if tlsConfig, err = w.addCertConfig(certFilePath, keyFilePath, pkcs12FilePath); err != nil {
  116. return nil, err
  117. }
  118. bs, err = w.doWeChat(body, wxReverse, tlsConfig)
  119. } else {
  120. bs, err = w.doWeChat(body, wxSandboxReverse)
  121. }
  122. if err != nil {
  123. return
  124. }
  125. wxRsp = new(WeChatReverseResponse)
  126. if err = xml.Unmarshal(bs, wxRsp); err != nil {
  127. return nil, fmt.Errorf("xml.Unmarshal:%s", err.Error())
  128. }
  129. return
  130. }
  131. // 申请退款
  132. // 注意:如已使用client.AddCertFilePath()或client.AddCertFileByte()添加过证书,参数certFilePath、keyFilePath、pkcs12FilePath全传空字符串 "",如方法需单独使用证书,则传证书Path
  133. // 文档地址:https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=9_4
  134. func (w *WeChatClient) Refund(body BodyMap, certFilePath, keyFilePath, pkcs12FilePath string) (wxRsp *WeChatRefundResponse, err error) {
  135. var (
  136. bs []byte
  137. tlsConfig *tls.Config
  138. )
  139. if w.IsProd {
  140. if tlsConfig, err = w.addCertConfig(certFilePath, keyFilePath, pkcs12FilePath); err != nil {
  141. return nil, err
  142. }
  143. bs, err = w.doWeChat(body, wxRefund, tlsConfig)
  144. } else {
  145. bs, err = w.doWeChat(body, wxSandboxRefund)
  146. }
  147. if err != nil {
  148. return
  149. }
  150. wxRsp = new(WeChatRefundResponse)
  151. if err = xml.Unmarshal(bs, wxRsp); err != nil {
  152. return nil, fmt.Errorf("xml.Unmarshal:%s", err.Error())
  153. }
  154. return
  155. }
  156. // 查询退款
  157. // 文档地址:https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=9_5
  158. func (w *WeChatClient) QueryRefund(body BodyMap) (wxRsp *WeChatQueryRefundResponse, err error) {
  159. var bs []byte
  160. if w.IsProd {
  161. bs, err = w.doWeChat(body, wxRefundquery)
  162. } else {
  163. bs, err = w.doWeChat(body, wxSandboxRefundquery)
  164. }
  165. if err != nil {
  166. return
  167. }
  168. wxRsp = new(WeChatQueryRefundResponse)
  169. if err = xml.Unmarshal(bs, wxRsp); err != nil {
  170. return nil, fmt.Errorf("xml.Unmarshal:%s", err.Error())
  171. }
  172. return
  173. }
  174. // 下载对账单
  175. // 文档地址:https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=9_6
  176. func (w *WeChatClient) DownloadBill(body BodyMap) (wxRsp string, err error) {
  177. var bs []byte
  178. if w.IsProd {
  179. bs, err = w.doWeChat(body, wxDownloadbill)
  180. } else {
  181. bs, err = w.doWeChat(body, wxSandboxDownloadbill)
  182. }
  183. if err != nil {
  184. return
  185. }
  186. wxRsp = string(bs)
  187. return
  188. }
  189. // 下载资金账单
  190. // 注意:如已使用client.AddCertFilePath()或client.AddCertFileByte()添加过证书,参数certFilePath、keyFilePath、pkcs12FilePath全传空字符串 "",如方法需单独使用证书,则传证书Path
  191. // 貌似不支持沙箱环境,因为沙箱环境默认需要用MD5签名,但是此接口仅支持HMAC-SHA256签名
  192. // 文档地址:https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=9_18&index=7
  193. func (w *WeChatClient) DownloadFundFlow(body BodyMap, certFilePath, keyFilePath, pkcs12FilePath string) (wxRsp string, err error) {
  194. var (
  195. bs []byte
  196. tlsConfig *tls.Config
  197. )
  198. if w.IsProd {
  199. if tlsConfig, err = w.addCertConfig(certFilePath, keyFilePath, pkcs12FilePath); err != nil {
  200. return null, err
  201. }
  202. bs, err = w.doWeChat(body, wxDownloadfundflow, tlsConfig)
  203. } else {
  204. bs, err = w.doWeChat(body, wxSandboxDownloadfundflow)
  205. }
  206. if err != nil {
  207. return
  208. }
  209. wxRsp = string(bs)
  210. return
  211. }
  212. // 拉取订单评价数据
  213. // 注意:如已使用client.AddCertFilePath()或client.AddCertFileByte()添加过证书,参数certFilePath、keyFilePath、pkcs12FilePath全传空字符串 "",如方法需单独使用证书,则传证书Path
  214. // 貌似不支持沙箱环境,因为沙箱环境默认需要用MD5签名,但是此接口仅支持HMAC-SHA256签名
  215. // 文档地址:https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=9_17&index=11
  216. func (w *WeChatClient) BatchQueryComment(body BodyMap, certFilePath, keyFilePath, pkcs12FilePath string) (wxRsp string, err error) {
  217. var (
  218. bs []byte
  219. tlsConfig *tls.Config
  220. )
  221. if w.IsProd {
  222. body.Set("sign_type", SignType_HMAC_SHA256)
  223. if tlsConfig, err = w.addCertConfig(certFilePath, keyFilePath, pkcs12FilePath); err != nil {
  224. return null, err
  225. }
  226. bs, err = w.doWeChat(body, wxBatchquerycomment, tlsConfig)
  227. } else {
  228. bs, err = w.doWeChat(body, wxSandboxBatchquerycomment)
  229. }
  230. if err != nil {
  231. return
  232. }
  233. wxRsp = string(bs)
  234. return
  235. }
  236. // 企业向微信用户个人付款
  237. // 注意:如已使用client.AddCertFilePath()或client.AddCertFileByte()添加过证书,参数certFilePath、keyFilePath、pkcs12FilePath全传空字符串 "",如方法需单独使用证书,则传证书Path
  238. // 注意:此方法未支持沙箱环境,默认正式环境,转账请慎重
  239. // 文档地址:https://pay.weixin.qq.com/wiki/doc/api/tools/mch_pay.php?chapter=14_1
  240. func (w *WeChatClient) Transfer(body BodyMap, certFilePath, keyFilePath, pkcs12FilePath string) (wxRsp *WeChatTransfersResponse, err error) {
  241. body.Set("mch_appid", w.AppId)
  242. body.Set("mchid", w.MchId)
  243. var (
  244. tlsConfig *tls.Config
  245. url = wxBaseUrlCh + wxTransfers
  246. )
  247. if tlsConfig, err = w.addCertConfig(certFilePath, keyFilePath, pkcs12FilePath); err != nil {
  248. return nil, err
  249. }
  250. body.Set("sign", getWeChatReleaseSign(w.ApiKey, SignType_MD5, body))
  251. httpClient := NewHttpClient().SetTLSConfig(tlsConfig).Type(TypeXML)
  252. if w.BaseURL != null {
  253. w.mu.RLock()
  254. url = w.BaseURL + wxTransfers
  255. w.mu.RUnlock()
  256. }
  257. wxRsp = new(WeChatTransfersResponse)
  258. res, errs := httpClient.Post(url).SendString(generateXml(body)).EndStruct(wxRsp)
  259. if len(errs) > 0 {
  260. return nil, errs[0]
  261. }
  262. if res.StatusCode != 200 {
  263. return nil, fmt.Errorf("HTTP Request Error, StatusCode = %d", res.StatusCode)
  264. }
  265. return wxRsp, nil
  266. }
  267. // 公众号纯签约(未完成)
  268. // 文档地址:https://pay.weixin.qq.com/wiki/doc/api/pap.php?chapter=18_1&index=1
  269. func (w *WeChatClient) EntrustPublic(body BodyMap) (bs []byte, err error) {
  270. bs, err = w.doWeChat(body, wxEntrustPublic)
  271. return nil, nil
  272. }
  273. // 向微信发送请求
  274. func (w *WeChatClient) doWeChat(body BodyMap, path string, tlsConfig ...*tls.Config) (bs []byte, err error) {
  275. body.Set("appid", w.AppId)
  276. body.Set("mch_id", w.MchId)
  277. var (
  278. sign string
  279. url = wxBaseUrlCh + path
  280. )
  281. if body.Get("sign") != null {
  282. goto GoRequest
  283. }
  284. if !w.IsProd {
  285. body.Set("sign_type", SignType_MD5)
  286. if sign, err = getWeChatSignBoxSign(w.MchId, w.ApiKey, body); err != nil {
  287. return nil, err
  288. }
  289. } else {
  290. sign = getWeChatReleaseSign(w.ApiKey, body.Get("sign_type"), body)
  291. }
  292. body.Set("sign", sign)
  293. GoRequest:
  294. httpClient := NewHttpClient()
  295. if w.IsProd && tlsConfig != nil {
  296. httpClient.SetTLSConfig(tlsConfig[0])
  297. }
  298. if w.BaseURL != null {
  299. w.mu.RLock()
  300. url = w.BaseURL + path
  301. w.mu.RUnlock()
  302. }
  303. res, bs, errs := httpClient.Type(TypeXML).Post(url).SendString(generateXml(body)).EndBytes()
  304. if len(errs) > 0 {
  305. return nil, errs[0]
  306. }
  307. if res.StatusCode != 200 {
  308. return nil, fmt.Errorf("HTTP Request Error, StatusCode = %d", res.StatusCode)
  309. }
  310. if strings.Contains(string(bs), "HTML") {
  311. return nil, errors.New(string(bs))
  312. }
  313. return bs, nil
  314. }