service/library/payclients/wxpaycli/client.go

338 lines
9.1 KiB
Go
Raw Normal View History

2023-12-21 22:17:40 +08:00
package wxpaycli
import (
2024-02-03 16:04:56 +08:00
"context"
2024-02-20 23:05:24 +08:00
"crypto"
cryptorand "crypto/rand"
"crypto/rsa"
"crypto/sha256"
"crypto/sha512"
"crypto/x509"
"encoding/base64"
"encoding/pem"
"errors"
"fmt"
2024-02-05 21:16:49 +08:00
"net/http"
2024-02-05 21:55:44 +08:00
"os"
2024-03-13 20:35:18 +08:00
"service/library/idgenerator"
2024-02-03 16:04:56 +08:00
"time"
"github.com/go-pay/gopay"
2024-02-20 23:05:24 +08:00
wxpayv2 "github.com/go-pay/gopay/wechat"
wxpayv3 "github.com/go-pay/gopay/wechat/v3"
2024-02-03 16:04:56 +08:00
2023-12-21 22:17:40 +08:00
"service/bizcommon/util"
"service/library/configcenter"
"service/library/logger"
)
2024-02-03 16:04:56 +08:00
const (
DefaultOrderTimeoutSeconds = 900 // 默认订单超时时间,单位: s
)
2024-05-03 02:26:48 +08:00
const (
2024-05-07 19:24:47 +08:00
AppIdXinYiDaoLe = "1665016206"
AppIdTieFenZone = "1675813721"
2024-05-03 02:26:48 +08:00
)
var allWxpayClients = map[string]*WxpayClient{}
func GetDefaultWxpayClient() *WxpayClient {
2024-05-07 19:24:47 +08:00
return allWxpayClients[AppIdTieFenZone]
2024-05-03 02:26:48 +08:00
}
2024-02-03 16:04:56 +08:00
2023-12-21 22:17:40 +08:00
type WxpayClient struct {
2024-02-20 23:05:24 +08:00
clientV3 *wxpayv3.ClientV3
2024-05-07 19:24:47 +08:00
MchId string `json:"mch_id"`
2024-02-20 23:05:24 +08:00
AppSecret string `json:"app_secret"`
AppId string `json:"app_id"`
NotifyUrl string `json:"notify_url"`
PrivateKeyPath string `json:"private_key_path"`
2023-12-21 22:17:40 +08:00
}
2024-05-03 02:26:48 +08:00
func InitMulti(cfgList ...*configcenter.WxpayClientConfig) (err error) {
for _, cfg := range cfgList {
var cli *WxpayClient
cli, err = NewWxpayClient(cfg)
if err != nil {
return
}
if cli == nil {
err = errors.New("NewAlipayClient fail")
return
}
2024-05-07 19:24:47 +08:00
allWxpayClients[cli.MchId] = cli
2024-05-03 02:26:48 +08:00
}
return
2024-02-03 16:04:56 +08:00
}
2024-05-03 02:26:48 +08:00
func NewWxpayClient(cfg *configcenter.WxpayClientConfig) (ret *WxpayClient, err error) {
2024-02-05 21:55:44 +08:00
// private key
bs, err := os.ReadFile(cfg.PrivateKeyPath)
if err != nil {
logger.Error("real PrivateKeyPath fail, cfg: %v, err: %v", util.ToJson(cfg), err)
return
}
privateKey := string(bs)
2024-02-20 23:05:24 +08:00
wxpayCliV3, err := wxpayv3.NewClientV3(cfg.MchId, cfg.SerialNo, cfg.ApiV3Key, privateKey)
2023-12-21 22:17:40 +08:00
if err != nil {
logger.Error("NewClientV3 fail, cfg: %v, err: %v", util.ToJson(cfg), err)
return
}
2024-02-03 16:04:56 +08:00
2024-05-03 02:26:48 +08:00
ret = &WxpayClient{
2024-02-20 23:05:24 +08:00
clientV3: wxpayCliV3,
2024-05-07 19:24:47 +08:00
MchId: cfg.MchId,
2024-02-20 23:05:24 +08:00
AppSecret: cfg.AppSecret,
AppId: cfg.AppId,
NotifyUrl: cfg.NotifyUrl,
PrivateKeyPath: cfg.PrivateKeyPath,
2024-02-03 16:04:56 +08:00
}
return
}
// 验签
2024-02-20 23:05:24 +08:00
func (c *WxpayClient) ParseNotify(req *http.Request) (notify *wxpayv3.V3DecryptResult, err error) {
notifyReq, err := wxpayv3.V3ParseNotify(req)
2024-02-05 21:16:49 +08:00
if err != nil {
logger.Error("V3ParseNotify fail, notifyReq: %v, err: %v", util.ToJson(notifyReq), err)
return
}
if notifyReq == nil {
logger.Error("V3ParseNotify nil, err: %v", err)
return
}
2024-02-20 23:05:24 +08:00
notifyTmp, err := notifyReq.DecryptCipherText(string(c.clientV3.ApiV3Key))
2024-02-05 21:16:49 +08:00
if err != nil {
logger.Error("DecryptCipherText fail, notifyTmp: %v, err: %v", util.ToJson(notifyTmp), err)
return
}
if notifyTmp == nil {
logger.Error("DecryptCipherText nil, err: %v", err)
return
}
logger.Info("Wxpay ParseNotify, %v", util.ToJson(notifyTmp))
notify = notifyTmp
return
}
2024-02-03 16:04:56 +08:00
// 微信支付 native支付
type NativePayParam struct {
Description string
OutTradeNo string // 商家订单id我们自己的订单id
TotalAmount int64 // 金额,单位:分
TimeOutSeconds int // 订单有效时间,单位:秒
}
2024-02-05 21:16:49 +08:00
func (c *WxpayClient) NativePay(ctx context.Context, param *NativePayParam) (wxpayNativeParamStr string, err error) {
2024-02-03 16:04:56 +08:00
if param.TimeOutSeconds <= 0 {
param.TimeOutSeconds = DefaultOrderTimeoutSeconds
}
bm := gopay.BodyMap{
"appid": c.AppId,
"description": param.Description,
"out_trade_no": param.OutTradeNo,
"time_expire": time.Now().Add(time.Second * time.Duration(param.TimeOutSeconds)).Format(time.RFC3339),
"notify_url": c.NotifyUrl,
"amount": gopay.BodyMap{
"total": param.TotalAmount,
"currency": "CNY",
},
}
2024-02-20 23:05:24 +08:00
resp, err := c.clientV3.V3TransactionNative(ctx, bm)
2024-02-03 16:04:56 +08:00
if err != nil {
return
}
2024-02-20 23:05:24 +08:00
if resp.Code != wxpayv3.Success {
logger.Info("wxpayv3 NativePay fail, code: %v, error: %v, response: %v", resp.Code, resp.Error, util.ToJson(resp.Response))
2024-02-03 16:04:56 +08:00
return
2023-12-21 22:17:40 +08:00
}
2024-02-05 21:16:49 +08:00
wxpayNativeParamStr = resp.Response.CodeUrl
2024-02-20 23:05:24 +08:00
logger.Info("wxpayv3 NativePay success, code: %v, error: %v, response: %v", resp.Code, resp.Error, util.ToJson(resp.Response))
2023-12-21 22:17:40 +08:00
return
}
2024-02-20 23:05:24 +08:00
// 通过authcode获取openid
func (c *WxpayClient) GetOpenIdByAuthCode(ctx context.Context, authCode string) (openid string, err error) {
at, err := wxpayv2.GetOauth2AccessToken(ctx, c.AppId, c.AppSecret, authCode)
if err != nil {
return
}
if len(at.Openid) <= 0 {
err = errors.New(fmt.Sprintf("fail, %s", util.ToJson(at)))
return
}
openid = at.Openid
return
}
// 微信支付 jsapi支付
type JsapiPayParam struct {
Description string
OutTradeNo string // 商家订单id我们自己的订单id
TotalAmount int64 // 金额,单位:分
TimeOutSeconds int // 订单有效时间,单位:秒
OpenId string
}
type JsapiPayResp struct {
PrepayId string `json:"-"`
AppId string `json:"appId"`
TimeStamp string `json:"timeStamp"`
NonceStr string `json:"nonceStr"`
Package string `json:"package"`
SignType string `json:"signType"`
PaySign string `json:"paySign"`
}
func (c *WxpayClient) JsapiPay(ctx context.Context, param *JsapiPayParam) (wxpayJsapiResp JsapiPayResp, err error) {
if param.TimeOutSeconds <= 0 {
param.TimeOutSeconds = DefaultOrderTimeoutSeconds
}
bm := gopay.BodyMap{
"appid": c.AppId,
"description": param.Description,
"out_trade_no": param.OutTradeNo,
"time_expire": time.Now().Add(time.Second * time.Duration(param.TimeOutSeconds)).Format(time.RFC3339),
"notify_url": c.NotifyUrl,
"amount": gopay.BodyMap{
"total": param.TotalAmount,
"currency": "CNY",
},
"payer": gopay.BodyMap{
"openid": param.OpenId,
},
}
resp, err := c.clientV3.V3TransactionJsapi(ctx, bm)
if err != nil {
return
}
if resp.Code != wxpayv3.Success {
logger.Info("wxpayv3 NativePay fail, code: %v, error: %v, response: %v", resp.Code, resp.Error, util.ToJson(resp.Response))
return
}
r := JsapiPayResp{
PrepayId: resp.Response.PrepayId,
AppId: c.AppId,
TimeStamp: fmt.Sprintf("%d", time.Now().Unix()),
NonceStr: util.RandomString(32),
Package: "prepay_id=" + resp.Response.PrepayId,
SignType: "RSA",
PaySign: resp.SignInfo.SignBody,
}
var (
cipherText = fmt.Sprintf("%s\n%s\n%s\n%s\n", r.AppId, r.TimeStamp, r.NonceStr, r.Package)
keyBytes, _ = os.ReadFile(c.PrivateKeyPath)
)
paySignBytes, err := rsaSign(keyBytes, crypto.SHA256, []byte(cipherText))
if err != nil {
return
}
r.PaySign = base64.StdEncoding.EncodeToString(paySignBytes)
wxpayJsapiResp = r
logger.Info("wxpayv3 JsapiPay success, code: %v, error: %v, response: %v", resp.Code, resp.Error, util.ToJson(wxpayJsapiResp))
return
}
func rsaSign(prvkey []byte, hash crypto.Hash, data []byte) ([]byte, error) {
block, _ := pem.Decode(prvkey)
if block == nil {
return nil, errors.New("decode private key error")
}
privateKey, err := x509.ParsePKCS8PrivateKey(block.Bytes)
if err != nil {
return nil, err
}
// MD5 and SHA1 are not supported as they are not secure.
var hashed []byte
switch hash {
case crypto.SHA224:
h := sha256.Sum224(data)
hashed = h[:]
case crypto.SHA256:
h := sha256.Sum256(data)
hashed = h[:]
case crypto.SHA384:
h := sha512.Sum384(data)
hashed = h[:]
case crypto.SHA512:
h := sha512.Sum512(data)
hashed = h[:]
}
return rsa.SignPKCS1v15(cryptorand.Reader, privateKey.(*rsa.PrivateKey), hash, hashed)
}
2024-02-26 19:28:03 +08:00
// 微信支付 h5支付
type H5PayParam struct {
Description string
OutTradeNo string // 商家订单id我们自己的订单id
TotalAmount int64 // 金额,单位:分
TimeOutSeconds int // 订单有效时间,单位:秒
Cip string // 客户端ip
}
func (c *WxpayClient) H5Pay(ctx context.Context, param *H5PayParam) (wxpayH5ParamStr string, err error) {
if param.TimeOutSeconds <= 0 {
param.TimeOutSeconds = DefaultOrderTimeoutSeconds
}
bm := gopay.BodyMap{
"appid": c.AppId,
"description": param.Description,
"out_trade_no": param.OutTradeNo,
"time_expire": time.Now().Add(time.Second * time.Duration(param.TimeOutSeconds)).Format(time.RFC3339),
"notify_url": c.NotifyUrl,
"amount": gopay.BodyMap{
"total": param.TotalAmount,
"currency": "CNY",
},
"scene_info": gopay.BodyMap{
"payer_client_ip": param.Cip,
"h5_info": gopay.BodyMap{
"type": "Wap",
},
},
}
resp, err := c.clientV3.V3TransactionH5(ctx, bm)
if err != nil {
return
}
if resp.Code != wxpayv3.Success {
logger.Info("wxpayv3 NativePay fail, code: %v, error: %v, response: %v", resp.Code, resp.Error, util.ToJson(resp.Response))
return
}
wxpayH5ParamStr = resp.Response.H5Url
logger.Info("wxpayv3 H5 success, code: %v, error: %v, response: %v", resp.Code, resp.Error, util.ToJson(resp.Response))
return
}
2024-03-13 20:35:18 +08:00
// 退款
type RefundOneParam struct {
OutTradeNo string // 商家订单id我们自己的订单id
RefundAmount int64 // 退款金额,单位:分
RefundReason string // 退款理由
}
func (c *WxpayClient) RefundOne(ctx context.Context, param *RefundOneParam) (resp *wxpayv3.RefundRsp, err error) {
bm := gopay.BodyMap{
"out_trade_no": param.OutTradeNo,
"out_refund_no": idgenerator.GenWxpayRefundId(),
"reason": param.RefundReason,
"amount": gopay.BodyMap{
"refund": param.RefundAmount,
"total": param.RefundAmount,
"currency": "CNY",
},
}
resp, err = c.clientV3.V3Refund(ctx, bm)
if err != nil {
return
}
return
}