|
@@ -0,0 +1,180 @@
|
|
|
|
|
+package pay
|
|
|
|
|
+
|
|
|
|
|
+import (
|
|
|
|
|
+ "bytes"
|
|
|
|
|
+ "crypto/tls"
|
|
|
|
|
+ "encoding/pem"
|
|
|
|
|
+ "encoding/xml"
|
|
|
|
|
+ "fmt"
|
|
|
|
|
+ "github.com/akikistyle/wechat/util"
|
|
|
|
|
+ "golang.org/x/crypto/pkcs12"
|
|
|
|
|
+ "io/ioutil"
|
|
|
|
|
+ "log"
|
|
|
|
|
+ "net/http"
|
|
|
|
|
+)
|
|
|
|
|
+
|
|
|
|
|
+var refundGateway = "https://api.mch.weixin.qq.com/secapi/pay/refund"
|
|
|
|
|
+
|
|
|
|
|
+//Refund Parameter
|
|
|
|
|
+type RefundParams struct {
|
|
|
|
|
+ TransactionId string
|
|
|
|
|
+ OutRefundNo string
|
|
|
|
|
+ TotalFee string
|
|
|
|
|
+ RefundFee string
|
|
|
|
|
+ RefundDesc string
|
|
|
|
|
+ RootCa string //ca证书
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+//Refund request
|
|
|
|
|
+type refundRequest struct {
|
|
|
|
|
+ AppID string `xml:"appid"`
|
|
|
|
|
+ MchID string `xml:"mch_id"`
|
|
|
|
|
+ NonceStr string `xml:"nonce_str"`
|
|
|
|
|
+ Sign string `xml:"sign"`
|
|
|
|
|
+ SignType string `xml:"sign_type,omitempty"`
|
|
|
|
|
+ TransactionId string `xml:"transaction_id"`
|
|
|
|
|
+ OutRefundNo string `xml:"out_refund_no"`
|
|
|
|
|
+ TotalFee string `xml:"total_fee"`
|
|
|
|
|
+ RefundFee string `xml:"refund_fee"`
|
|
|
|
|
+ RefundDesc string `xml:"refund_desc,omitempty"`
|
|
|
|
|
+ //NotifyUrl string `xml:"notify_url,omitempty"`
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+//Refund Response
|
|
|
|
|
+type RefundResponse struct {
|
|
|
|
|
+ ReturnCode string `xml:"return_code"`
|
|
|
|
|
+ ReturnMsg string `xml:"return_msg"`
|
|
|
|
|
+ AppID string `xml:"appid,omitempty"`
|
|
|
|
|
+ MchID string `xml:"mch_id,omitempty"`
|
|
|
|
|
+ NonceStr string `xml:"nonce_str,omitempty"`
|
|
|
|
|
+ Sign string `xml:"sign,omitempty"`
|
|
|
|
|
+ ResultCode string `xml:"result_code,omitempty"`
|
|
|
|
|
+ ErrCode string `xml:"err_code,omitempty"`
|
|
|
|
|
+ ErrCodeDes string `xml:"err_code_des,omitempty"`
|
|
|
|
|
+ TransactionId string `xml:"transaction_id,omitempty"`
|
|
|
|
|
+ OutTradeNo string `xml:"out_trade_no,omitempty"`
|
|
|
|
|
+ OutRefundNo string `xml:"out_refund_no,omitempty"`
|
|
|
|
|
+ RefundId string `xml:"refund_id,omitempty"`
|
|
|
|
|
+ RefundFee string `xml:"refund_fee,omitempty"`
|
|
|
|
|
+ SettlementRefundFee string `xml:"settlement_refund_fee,omitempty"`
|
|
|
|
|
+ TotalFee string `xml:"total_fee,omitempty"`
|
|
|
|
|
+ SettlementTotalFee string `xml:"settlement_total_fee,omitempty"`
|
|
|
|
|
+ FeeType string `xml:"fee_type,omitempty"`
|
|
|
|
|
+ CashFee string `xml:"cash_fee,omitempty"`
|
|
|
|
|
+ CashFeeType string `xml:"cash_fee_type,omitempty"`
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+//退款申请
|
|
|
|
|
+func (pcf *Pay) Refund(p *RefundParams) (rsp RefundResponse, err error) {
|
|
|
|
|
+ nonceStr := util.RandomStr(32)
|
|
|
|
|
+ param := make(map[string]interface{})
|
|
|
|
|
+ param["appid"] = pcf.AppID
|
|
|
|
|
+ param["mch_id"] = pcf.PayMchID
|
|
|
|
|
+ param["nonce_str"] = nonceStr
|
|
|
|
|
+ param["out_refund_no"] = p.OutRefundNo
|
|
|
|
|
+ param["refund_desc"] = p.RefundDesc
|
|
|
|
|
+ param["refund_fee"] = p.RefundFee
|
|
|
|
|
+ param["total_fee"] = p.TotalFee
|
|
|
|
|
+ param["sign_type"] = "MD5"
|
|
|
|
|
+ param["transaction_id"] = p.TransactionId
|
|
|
|
|
+
|
|
|
|
|
+ bizKey := "&key=" + pcf.PayKey
|
|
|
|
|
+ str := orderParam(param, bizKey)
|
|
|
|
|
+ sign := util.MD5Sum(str)
|
|
|
|
|
+ request := refundRequest{
|
|
|
|
|
+ AppID: pcf.AppID,
|
|
|
|
|
+ MchID: pcf.PayMchID,
|
|
|
|
|
+ NonceStr: nonceStr,
|
|
|
|
|
+ Sign: sign,
|
|
|
|
|
+ SignType: "MD5",
|
|
|
|
|
+ TransactionId: p.TransactionId,
|
|
|
|
|
+ OutRefundNo: p.OutRefundNo,
|
|
|
|
|
+ TotalFee: p.TotalFee,
|
|
|
|
|
+ RefundFee: p.RefundFee,
|
|
|
|
|
+ RefundDesc: p.RefundDesc,
|
|
|
|
|
+ }
|
|
|
|
|
+ rawRet, err := postXMLWithTLS(refundGateway, request, p.RootCa, pcf.PayMchID)
|
|
|
|
|
+ if err != nil {
|
|
|
|
|
+ return
|
|
|
|
|
+ }
|
|
|
|
|
+ err = xml.Unmarshal(rawRet, &rsp)
|
|
|
|
|
+ if err != nil {
|
|
|
|
|
+ return
|
|
|
|
|
+ }
|
|
|
|
|
+ if rsp.ReturnCode == "SUCCESS" {
|
|
|
|
|
+ if rsp.ResultCode == "SUCCESS" {
|
|
|
|
|
+ err = nil
|
|
|
|
|
+ return
|
|
|
|
|
+ }
|
|
|
|
|
+ err = fmt.Errorf("refund error, errcode=%s,errmsg=%s", rsp.ErrCode, rsp.ErrCodeDes)
|
|
|
|
|
+ return
|
|
|
|
|
+ }
|
|
|
|
|
+ err = fmt.Errorf("[msg : xmlUnmarshalError] [rawReturn : %s] [params : %s] [sign : %s]",
|
|
|
|
|
+ string(rawRet), str, sign)
|
|
|
|
|
+ return
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+//http TLS
|
|
|
|
|
+func httpWithTLS(rootCa, key string) (*http.Client, error) {
|
|
|
|
|
+ var client *http.Client
|
|
|
|
|
+ certData, err := ioutil.ReadFile(rootCa)
|
|
|
|
|
+ if err != nil {
|
|
|
|
|
+ return nil, fmt.Errorf("unable to find cert path=%s, error=%v", rootCa, err)
|
|
|
|
|
+ }
|
|
|
|
|
+ cert := pkcs12ToPem(certData, key)
|
|
|
|
|
+ config := &tls.Config{
|
|
|
|
|
+ Certificates: []tls.Certificate{cert},
|
|
|
|
|
+ }
|
|
|
|
|
+ tr := &http.Transport{
|
|
|
|
|
+ TLSClientConfig: config,
|
|
|
|
|
+ DisableCompression: true,
|
|
|
|
|
+ }
|
|
|
|
|
+ client = &http.Client{Transport: tr}
|
|
|
|
|
+ return client, nil
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+//将Pkcs12转成Pem
|
|
|
|
|
+func pkcs12ToPem(p12 []byte, password string) tls.Certificate {
|
|
|
|
|
+ blocks, err := pkcs12.ToPEM(p12, password)
|
|
|
|
|
+ defer func() {
|
|
|
|
|
+ if x := recover(); x != nil {
|
|
|
|
|
+ log.Print(x)
|
|
|
|
|
+ }
|
|
|
|
|
+ }()
|
|
|
|
|
+ if err != nil {
|
|
|
|
|
+ panic(err)
|
|
|
|
|
+ }
|
|
|
|
|
+ var pemData []byte
|
|
|
|
|
+ for _, b := range blocks {
|
|
|
|
|
+ pemData = append(pemData, pem.EncodeToMemory(b)...)
|
|
|
|
|
+ }
|
|
|
|
|
+ cert, err := tls.X509KeyPair(pemData, pemData)
|
|
|
|
|
+ if err != nil {
|
|
|
|
|
+ panic(err)
|
|
|
|
|
+ }
|
|
|
|
|
+ return cert
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+//Post XML with TLS
|
|
|
|
|
+func postXMLWithTLS(uri string, obj interface{}, ca, key string) ([]byte, error) {
|
|
|
|
|
+ xmlData, err := xml.Marshal(obj)
|
|
|
|
|
+ if err != nil {
|
|
|
|
|
+ return nil, err
|
|
|
|
|
+ }
|
|
|
|
|
+
|
|
|
|
|
+ body := bytes.NewBuffer(xmlData)
|
|
|
|
|
+ client, err := httpWithTLS(ca, key)
|
|
|
|
|
+ if err != nil {
|
|
|
|
|
+ return nil, err
|
|
|
|
|
+ }
|
|
|
|
|
+ response, err := client.Post(uri, "application/xml;charset=utf-8", body)
|
|
|
|
|
+ if err != nil {
|
|
|
|
|
+ return nil, err
|
|
|
|
|
+ }
|
|
|
|
|
+ defer response.Body.Close()
|
|
|
|
|
+
|
|
|
|
|
+ if response.StatusCode != http.StatusOK {
|
|
|
|
|
+ return nil, fmt.Errorf("http code error : uri=%v , statusCode=%v", uri, response.StatusCode)
|
|
|
|
|
+ }
|
|
|
|
|
+ return ioutil.ReadAll(response.Body)
|
|
|
|
|
+}
|