wechat_client.go 11 KB

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