341 lines
14 KiB
Go
341 lines
14 KiB
Go
|
package wechat
|
|||
|
|
|||
|
import (
|
|||
|
"crypto/rsa"
|
|||
|
"encoding/json"
|
|||
|
"errors"
|
|||
|
"fmt"
|
|||
|
"io"
|
|||
|
"io/ioutil"
|
|||
|
"net/http"
|
|||
|
|
|||
|
"github.com/go-pay/gopay"
|
|||
|
"github.com/go-pay/gopay/pkg/xlog"
|
|||
|
)
|
|||
|
|
|||
|
type Resource struct {
|
|||
|
OriginalType string `json:"original_type,omitempty"`
|
|||
|
Algorithm string `json:"algorithm"`
|
|||
|
Ciphertext string `json:"ciphertext"`
|
|||
|
AssociatedData string `json:"associated_data"`
|
|||
|
Nonce string `json:"nonce"`
|
|||
|
}
|
|||
|
|
|||
|
type V3DecryptResult struct {
|
|||
|
Appid string `json:"appid"`
|
|||
|
Mchid string `json:"mchid"`
|
|||
|
OutTradeNo string `json:"out_trade_no"`
|
|||
|
TransactionId string `json:"transaction_id"`
|
|||
|
TradeType string `json:"trade_type"`
|
|||
|
TradeState string `json:"trade_state"`
|
|||
|
TradeStateDesc string `json:"trade_state_desc"`
|
|||
|
BankType string `json:"bank_type"`
|
|||
|
Attach string `json:"attach"`
|
|||
|
SuccessTime string `json:"success_time"`
|
|||
|
Payer *Payer `json:"payer"`
|
|||
|
Amount *Amount `json:"amount"`
|
|||
|
SceneInfo *SceneInfo `json:"scene_info"`
|
|||
|
PromotionDetail []*PromotionDetail `json:"promotion_detail"`
|
|||
|
}
|
|||
|
|
|||
|
type V3DecryptPartnerResult struct {
|
|||
|
SpAppid string `json:"sp_appid"`
|
|||
|
SpMchid string `json:"sp_mchid"`
|
|||
|
SubAppid string `json:"sub_appid"`
|
|||
|
SubMchid string `json:"sub_mchid"`
|
|||
|
OutTradeNo string `json:"out_trade_no"`
|
|||
|
TransactionId string `json:"transaction_id"`
|
|||
|
TradeType string `json:"trade_type"`
|
|||
|
TradeState string `json:"trade_state"`
|
|||
|
TradeStateDesc string `json:"trade_state_desc"`
|
|||
|
BankType string `json:"bank_type"`
|
|||
|
Attach string `json:"attach"`
|
|||
|
SuccessTime string `json:"success_time"`
|
|||
|
Payer *PartnerPayer `json:"payer"`
|
|||
|
Amount *Amount `json:"amount"`
|
|||
|
SceneInfo *SceneInfo `json:"scene_info"`
|
|||
|
PromotionDetail []*PromotionDetail `json:"promotion_detail"`
|
|||
|
}
|
|||
|
|
|||
|
type V3DecryptRefundResult struct {
|
|||
|
Mchid string `json:"mchid"`
|
|||
|
OutTradeNo string `json:"out_trade_no"`
|
|||
|
TransactionId string `json:"transaction_id"`
|
|||
|
OutRefundNo string `json:"out_refund_no"`
|
|||
|
RefundId string `json:"refund_id"`
|
|||
|
RefundStatus string `json:"refund_status"`
|
|||
|
SuccessTime string `json:"success_time"`
|
|||
|
UserReceivedAccount string `json:"user_received_account"`
|
|||
|
Amount *RefundAmount `json:"amount"`
|
|||
|
}
|
|||
|
|
|||
|
type V3DecryptPartnerRefundResult struct {
|
|||
|
SpMchid string `json:"sp_mchid"`
|
|||
|
SubMchid string `json:"sub_mchid"`
|
|||
|
OutTradeNo string `json:"out_trade_no"`
|
|||
|
TransactionId string `json:"transaction_id"`
|
|||
|
OutRefundNo string `json:"out_refund_no"`
|
|||
|
RefundId string `json:"refund_id"`
|
|||
|
RefundStatus string `json:"refund_status"`
|
|||
|
SuccessTime string `json:"success_time"`
|
|||
|
UserReceivedAccount string `json:"user_received_account"`
|
|||
|
Amount *RefundAmount `json:"amount"`
|
|||
|
}
|
|||
|
|
|||
|
type V3DecryptCombineResult struct {
|
|||
|
CombineAppid string `json:"combine_appid"`
|
|||
|
CombineMchid string `json:"combine_mchid"`
|
|||
|
CombineOutTradeNo string `json:"combine_out_trade_no"`
|
|||
|
SceneInfo *SceneInfo `json:"scene_info"`
|
|||
|
SubOrders []*SubOrders `json:"sub_orders"` // 最多支持子单条数:50
|
|||
|
CombinePayerInfo *Payer `json:"combine_payer_info"` // 支付者信息
|
|||
|
}
|
|||
|
|
|||
|
type V3DecryptScoreResult struct {
|
|||
|
Appid string `json:"appid"`
|
|||
|
Mchid string `json:"mchid"`
|
|||
|
OutOrderNo string `json:"out_order_no"`
|
|||
|
ServiceId string `json:"service_id"`
|
|||
|
Openid string `json:"openid"`
|
|||
|
State string `json:"state"`
|
|||
|
StateDescription string `json:"state_description"`
|
|||
|
TotalAmount int `json:"total_amount"`
|
|||
|
ServiceIntroduction string `json:"service_introduction"`
|
|||
|
PostPayments []*PostPayments `json:"post_payments"`
|
|||
|
PostDiscounts []*PostDiscounts `json:"post_discounts"`
|
|||
|
RiskFund *RiskFund `json:"risk_fund"`
|
|||
|
TimeRange *TimeRange `json:"time_range"`
|
|||
|
Location *Location `json:"location"`
|
|||
|
Attach string `json:"attach"`
|
|||
|
NotifyUrl string `json:"notify_url"`
|
|||
|
OrderId string `json:"order_id"`
|
|||
|
NeedCollection bool `json:"need_collection"`
|
|||
|
Collection *Collection `json:"collection"`
|
|||
|
}
|
|||
|
|
|||
|
type V3DecryptProfitShareResult struct {
|
|||
|
SpMchid string `json:"sp_mchid"` // 服务商商户号
|
|||
|
SubMchid string `json:"sub_mchid"` // 子商户号
|
|||
|
TransactionId string `json:"transaction_id"` // 微信订单号
|
|||
|
OrderId string `json:"order_id"` // 微信分账/回退单号
|
|||
|
OutOrderNo string `json:"out_order_no"` // 商户分账/回退单号
|
|||
|
Receiver *Receiver `json:"receiver"`
|
|||
|
SuccessTime string `json:"success_time"` // 成功时间
|
|||
|
}
|
|||
|
|
|||
|
type Receiver struct {
|
|||
|
Type string `json:"type"` // 分账接收方类型
|
|||
|
Account string `json:"account"` // 分账接收方账号
|
|||
|
Amount int `json:"amount"` // 分账动账金额
|
|||
|
Description string `json:"description"` // 分账/回退描述
|
|||
|
}
|
|||
|
|
|||
|
type V3DecryptBusifavorResult struct {
|
|||
|
EventType string `json:"event_type"` // 事件类型
|
|||
|
CouponCode string `json:"coupon_code"` // 券code
|
|||
|
StockId string `json:"stock_id"` // 批次号
|
|||
|
SendTime string `json:"send_time"` // 发放时间
|
|||
|
Openid string `json:"openid"` // 用户标识
|
|||
|
Unionid string `json:"unionid"` // 用户统一标识
|
|||
|
SendChannel string `json:"send_channel"` // 发放渠道
|
|||
|
SendMerchant string `json:"send_merchant"` // 发券商户号
|
|||
|
AttachInfo *BusifavorAttachInfo `json:"attach_info"` // 发券附加信息
|
|||
|
}
|
|||
|
|
|||
|
type BusifavorAttachInfo struct {
|
|||
|
TransactionId string `json:"transaction_id"` // 交易订单编号
|
|||
|
ActCode string `json:"act_code"` // 支付有礼活动编号/营销馆活动ID
|
|||
|
HallCode string `json:"hall_code"` // 营销馆ID
|
|||
|
HallBelongMchID int `json:"hall_belong_mch_id"` // 营销馆所属商户号
|
|||
|
CardID string `json:"card_id"` // 会员卡ID
|
|||
|
Code string `json:"code"` // 会员卡code
|
|||
|
ActivityID string `json:"activity_id"` // 会员活动ID
|
|||
|
}
|
|||
|
|
|||
|
type V3NotifyReq struct {
|
|||
|
Id string `json:"id"`
|
|||
|
CreateTime string `json:"create_time"`
|
|||
|
ResourceType string `json:"resource_type"`
|
|||
|
EventType string `json:"event_type"`
|
|||
|
Summary string `json:"summary"`
|
|||
|
Resource *Resource `json:"resource"`
|
|||
|
SignInfo *SignInfo `json:"-"`
|
|||
|
}
|
|||
|
|
|||
|
type V3NotifyRsp struct {
|
|||
|
Code string `json:"code"`
|
|||
|
Message string `json:"message"`
|
|||
|
}
|
|||
|
|
|||
|
// =====================================================================================================================
|
|||
|
|
|||
|
// 解析微信回调请求的参数到 V3NotifyReq 结构体
|
|||
|
func V3ParseNotify(req *http.Request) (notifyReq *V3NotifyReq, err error) {
|
|||
|
bs, err := ioutil.ReadAll(io.LimitReader(req.Body, int64(5<<20))) // default 5MB change the size you want;
|
|||
|
defer req.Body.Close()
|
|||
|
if err != nil {
|
|||
|
return nil, fmt.Errorf("read request body error:%w", err)
|
|||
|
}
|
|||
|
si := &SignInfo{
|
|||
|
HeaderTimestamp: req.Header.Get(HeaderTimestamp),
|
|||
|
HeaderNonce: req.Header.Get(HeaderNonce),
|
|||
|
HeaderSignature: req.Header.Get(HeaderSignature),
|
|||
|
HeaderSerial: req.Header.Get(HeaderSerial),
|
|||
|
SignBody: string(bs),
|
|||
|
}
|
|||
|
notifyReq = &V3NotifyReq{SignInfo: si}
|
|||
|
if err = json.Unmarshal(bs, notifyReq); err != nil {
|
|||
|
return nil, fmt.Errorf("json.Unmarshal(%s, %+v):%w", string(bs), notifyReq, err)
|
|||
|
}
|
|||
|
return notifyReq, nil
|
|||
|
}
|
|||
|
|
|||
|
// Deprecated
|
|||
|
// 推荐使用 VerifySignByPK()
|
|||
|
func (v *V3NotifyReq) VerifySign(wxPkContent string) (err error) {
|
|||
|
if v.SignInfo != nil {
|
|||
|
return V3VerifySign(v.SignInfo.HeaderTimestamp, v.SignInfo.HeaderNonce, v.SignInfo.SignBody, v.SignInfo.HeaderSignature, wxPkContent)
|
|||
|
}
|
|||
|
return errors.New("verify notify sign, bug SignInfo is nil")
|
|||
|
}
|
|||
|
|
|||
|
// 异步通知验签
|
|||
|
// wxPublicKey:微信平台证书公钥内容,通过 client.WxPublicKeyMap() 获取,然后根据 signInfo.HeaderSerial 获取相应的公钥
|
|||
|
// 推荐使用 VerifySignByPKMap()
|
|||
|
func (v *V3NotifyReq) VerifySignByPK(wxPublicKey *rsa.PublicKey) (err error) {
|
|||
|
if v.SignInfo != nil {
|
|||
|
return V3VerifySignByPK(v.SignInfo.HeaderTimestamp, v.SignInfo.HeaderNonce, v.SignInfo.SignBody, v.SignInfo.HeaderSignature, wxPublicKey)
|
|||
|
}
|
|||
|
return errors.New("verify notify sign, bug SignInfo is nil")
|
|||
|
}
|
|||
|
|
|||
|
// 异步通知验签
|
|||
|
// wxPublicKey:微信平台证书公钥内容,通过 client.WxPublicKeyMap() 获取
|
|||
|
func (v *V3NotifyReq) VerifySignByPKMap(wxPublicKeyMap map[string]*rsa.PublicKey) (err error) {
|
|||
|
if v.SignInfo != nil && wxPublicKeyMap != nil {
|
|||
|
return V3VerifySignByPK(v.SignInfo.HeaderTimestamp, v.SignInfo.HeaderNonce, v.SignInfo.SignBody, v.SignInfo.HeaderSignature, wxPublicKeyMap[v.SignInfo.HeaderSerial])
|
|||
|
}
|
|||
|
return errors.New("verify notify sign, bug SignInfo or wxPublicKeyMap is nil")
|
|||
|
}
|
|||
|
|
|||
|
// 解密 普通支付 回调中的加密信息
|
|||
|
func (v *V3NotifyReq) DecryptCipherText(apiV3Key string) (result *V3DecryptResult, err error) {
|
|||
|
if v.Resource != nil {
|
|||
|
result, err = V3DecryptNotifyCipherText(v.Resource.Ciphertext, v.Resource.Nonce, v.Resource.AssociatedData, apiV3Key)
|
|||
|
if err != nil {
|
|||
|
bytes, _ := json.Marshal(v)
|
|||
|
return nil, fmt.Errorf("V3NotifyReq(%s) decrypt cipher text error(%w)", string(bytes), err)
|
|||
|
}
|
|||
|
return result, nil
|
|||
|
}
|
|||
|
return nil, errors.New("notify data Resource is nil")
|
|||
|
}
|
|||
|
|
|||
|
// 解密 服务商支付 回调中的加密信息
|
|||
|
func (v *V3NotifyReq) DecryptPartnerCipherText(apiV3Key string) (result *V3DecryptPartnerResult, err error) {
|
|||
|
if v.Resource != nil {
|
|||
|
result, err = V3DecryptPartnerNotifyCipherText(v.Resource.Ciphertext, v.Resource.Nonce, v.Resource.AssociatedData, apiV3Key)
|
|||
|
if err != nil {
|
|||
|
bytes, _ := json.Marshal(v)
|
|||
|
return nil, fmt.Errorf("V3NotifyReq(%s) decrypt cipher text error(%w)", string(bytes), err)
|
|||
|
}
|
|||
|
return result, nil
|
|||
|
}
|
|||
|
return nil, errors.New("notify data Resource is nil")
|
|||
|
}
|
|||
|
|
|||
|
// 解密 普通退款 回调中的加密信息
|
|||
|
func (v *V3NotifyReq) DecryptRefundCipherText(apiV3Key string) (result *V3DecryptRefundResult, err error) {
|
|||
|
if v.Resource != nil {
|
|||
|
result, err = V3DecryptRefundNotifyCipherText(v.Resource.Ciphertext, v.Resource.Nonce, v.Resource.AssociatedData, apiV3Key)
|
|||
|
if err != nil {
|
|||
|
bytes, _ := json.Marshal(v)
|
|||
|
return nil, fmt.Errorf("V3NotifyReq(%s) decrypt cipher text error(%w)", string(bytes), err)
|
|||
|
}
|
|||
|
return result, nil
|
|||
|
}
|
|||
|
return nil, errors.New("notify data Resource is nil")
|
|||
|
}
|
|||
|
|
|||
|
// 解密 服务商退款 回调中的加密信息
|
|||
|
func (v *V3NotifyReq) DecryptPartnerRefundCipherText(apiV3Key string) (result *V3DecryptPartnerRefundResult, err error) {
|
|||
|
if v.Resource != nil {
|
|||
|
result, err = V3DecryptPartnerRefundNotifyCipherText(v.Resource.Ciphertext, v.Resource.Nonce, v.Resource.AssociatedData, apiV3Key)
|
|||
|
if err != nil {
|
|||
|
bytes, _ := json.Marshal(v)
|
|||
|
return nil, fmt.Errorf("V3NotifyReq(%s) decrypt cipher text error(%w)", string(bytes), err)
|
|||
|
}
|
|||
|
return result, nil
|
|||
|
}
|
|||
|
return nil, errors.New("notify data Resource is nil")
|
|||
|
}
|
|||
|
|
|||
|
// 解密 合单支付 回调中的加密信息
|
|||
|
func (v *V3NotifyReq) DecryptCombineCipherText(apiV3Key string) (result *V3DecryptCombineResult, err error) {
|
|||
|
if v.Resource != nil {
|
|||
|
result, err = V3DecryptCombineNotifyCipherText(v.Resource.Ciphertext, v.Resource.Nonce, v.Resource.AssociatedData, apiV3Key)
|
|||
|
if err != nil {
|
|||
|
bytes, _ := json.Marshal(v)
|
|||
|
return nil, fmt.Errorf("V3NotifyReq(%s) decrypt cipher text error(%w)", string(bytes), err)
|
|||
|
}
|
|||
|
return result, nil
|
|||
|
}
|
|||
|
return nil, errors.New("notify data Resource is nil")
|
|||
|
}
|
|||
|
|
|||
|
// 解密 支付分 回调中的加密信息
|
|||
|
func (v *V3NotifyReq) DecryptScoreCipherText(apiV3Key string) (result *V3DecryptScoreResult, err error) {
|
|||
|
if v.Resource != nil {
|
|||
|
result, err = V3DecryptScoreNotifyCipherText(v.Resource.Ciphertext, v.Resource.Nonce, v.Resource.AssociatedData, apiV3Key)
|
|||
|
if err != nil {
|
|||
|
bytes, _ := json.Marshal(v)
|
|||
|
return nil, fmt.Errorf("V3NotifyReq(%s) decrypt cipher text error(%w)", string(bytes), err)
|
|||
|
}
|
|||
|
return result, nil
|
|||
|
}
|
|||
|
return nil, errors.New("notify data Resource is nil")
|
|||
|
}
|
|||
|
|
|||
|
// 解密分账动账回调中的加密信息
|
|||
|
func (v *V3NotifyReq) DecryptProfitShareCipherText(apiV3Key string) (result *V3DecryptProfitShareResult, err error) {
|
|||
|
if v.Resource != nil {
|
|||
|
result, err = V3DecryptProfitShareNotifyCipherText(v.Resource.Ciphertext, v.Resource.Nonce, v.Resource.AssociatedData, apiV3Key)
|
|||
|
if err != nil {
|
|||
|
bytes, _ := json.Marshal(v)
|
|||
|
return nil, fmt.Errorf("V3NotifyReq(%s) decrypt cipher text error(%w)", string(bytes), err)
|
|||
|
}
|
|||
|
return result, nil
|
|||
|
}
|
|||
|
return nil, errors.New("notify data Resource is nil")
|
|||
|
}
|
|||
|
|
|||
|
// 解密商家券回调中的加密信息
|
|||
|
func (v *V3NotifyReq) DecryptBusifavorCipherText(apiV3Key string) (result *V3DecryptBusifavorResult, err error) {
|
|||
|
if v.Resource != nil {
|
|||
|
result, err = V3DecryptBusifavorNotifyCipherText(v.Resource.Ciphertext, v.Resource.Nonce, v.Resource.AssociatedData, apiV3Key)
|
|||
|
if err != nil {
|
|||
|
bytes, _ := json.Marshal(v)
|
|||
|
return nil, fmt.Errorf("V3NotifyReq(%s) decrypt cipher text error(%w)", string(bytes), err)
|
|||
|
}
|
|||
|
return result, nil
|
|||
|
}
|
|||
|
return nil, errors.New("notify data Resource is nil")
|
|||
|
}
|
|||
|
|
|||
|
// Deprecated
|
|||
|
// 暂时不推荐此方法,请使用 wechat.V3ParseNotify()
|
|||
|
// 解析微信回调请求的参数到 gopay.BodyMap
|
|||
|
func V3ParseNotifyToBodyMap(req *http.Request) (bm gopay.BodyMap, err error) {
|
|||
|
bs, err := ioutil.ReadAll(io.LimitReader(req.Body, int64(3<<20))) // default 3MB change the size you want;
|
|||
|
defer req.Body.Close()
|
|||
|
if err != nil {
|
|||
|
xlog.Error("err:", err)
|
|||
|
return
|
|||
|
}
|
|||
|
bm = make(gopay.BodyMap)
|
|||
|
if err = json.Unmarshal(bs, &bm); err != nil {
|
|||
|
return nil, fmt.Errorf("json.Unmarshal(%s):%w", string(bs), err)
|
|||
|
}
|
|||
|
return bm, nil
|
|||
|
}
|