mirror of
https://github.com/casdoor/casdoor.git
synced 2025-05-23 02:35:49 +08:00

* feat: support wechat jsapi payment * feat: add log * feat: update sign * feat: process wechat pay result * feat: process wechat pay result * feat: save wechat openid for different app * feat: save wechat openid for different app * feat: add SetUserOAuthProperties for signup * feat: fix openid for wechat * feat: get user extra property in buyproduct * feat: remove log * feat: remove log * feat: gofumpt code * feat: change lr->crlf * feat: change crlf->lf * feat: improve code
182 lines
5.6 KiB
Go
182 lines
5.6 KiB
Go
// Copyright 2022 The Casdoor Authors. All Rights Reserved.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package pp
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
|
|
"github.com/casdoor/casdoor/util"
|
|
"github.com/go-pay/gopay"
|
|
"github.com/go-pay/gopay/wechat/v3"
|
|
)
|
|
|
|
type WechatPayNotifyResponse struct {
|
|
Code string `json:"Code"`
|
|
Message string `json:"Message"`
|
|
}
|
|
|
|
type WechatPaymentProvider struct {
|
|
Client *wechat.ClientV3
|
|
AppId string
|
|
}
|
|
|
|
func NewWechatPaymentProvider(mchId string, apiV3Key string, appId string, serialNo string, privateKey string) (*WechatPaymentProvider, error) {
|
|
// https://pay.weixin.qq.com/docs/merchant/products/native-payment/preparation.html
|
|
// clientId => mchId
|
|
// clientSecret => apiV3Key
|
|
// clientId2 => appId
|
|
|
|
// appCertificate => serialNo
|
|
// appPrivateKey => privateKey
|
|
if appId == "" || mchId == "" || serialNo == "" || apiV3Key == "" || privateKey == "" {
|
|
return &WechatPaymentProvider{}, nil
|
|
}
|
|
|
|
clientV3, err := wechat.NewClientV3(mchId, serialNo, apiV3Key, privateKey)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
platformCert, serialNo, err := clientV3.GetAndSelectNewestCert()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
pp := &WechatPaymentProvider{
|
|
Client: clientV3.SetPlatformCert([]byte(platformCert), serialNo),
|
|
AppId: appId,
|
|
}
|
|
|
|
return pp, nil
|
|
}
|
|
|
|
func (pp *WechatPaymentProvider) Pay(r *PayReq) (*PayResp, error) {
|
|
bm := gopay.BodyMap{}
|
|
desc := joinAttachString([]string{r.ProductDisplayName, r.ProductName, r.ProviderName})
|
|
bm.Set("attach", desc)
|
|
bm.Set("appid", pp.AppId)
|
|
bm.Set("description", r.ProductDisplayName)
|
|
bm.Set("notify_url", r.NotifyUrl)
|
|
bm.Set("out_trade_no", r.PaymentName)
|
|
bm.SetBodyMap("amount", func(bm gopay.BodyMap) {
|
|
bm.Set("total", priceFloat64ToInt64(r.Price))
|
|
bm.Set("currency", r.Currency)
|
|
})
|
|
// In Wechat browser, we use JSAPI
|
|
if r.PaymentEnv == PaymentEnvWechatBrowser {
|
|
if r.PayerId == "" {
|
|
return nil, errors.New("failed to get the payer's openid, please retry login")
|
|
}
|
|
bm.SetBodyMap("payer", func(bm gopay.BodyMap) {
|
|
bm.Set("openid", r.PayerId) // If the account is signup via Wechat, the PayerId is the Wechat OpenId e.g.oxW9O1ZDvgreSHuBSQDiQ2F055PI
|
|
})
|
|
jsapiRsp, err := pp.Client.V3TransactionJsapi(context.Background(), bm)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if jsapiRsp.Code != wechat.Success {
|
|
return nil, errors.New(jsapiRsp.Error)
|
|
}
|
|
// use RSA256 to sign the pay request
|
|
params, err := pp.Client.PaySignOfJSAPI(pp.AppId, jsapiRsp.Response.PrepayId)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
payResp := &PayResp{
|
|
PayUrl: "",
|
|
OrderId: r.PaymentName, // Wechat can use paymentName as the OutTradeNo to query order status
|
|
AttachInfo: map[string]interface{}{
|
|
"appId": params.AppId,
|
|
"timeStamp": params.TimeStamp,
|
|
"nonceStr": params.NonceStr,
|
|
"package": params.Package,
|
|
"signType": "RSA",
|
|
"paySign": params.PaySign,
|
|
},
|
|
}
|
|
return payResp, nil
|
|
} else {
|
|
// In other case, we use NativeAPI
|
|
nativeRsp, err := pp.Client.V3TransactionNative(context.Background(), bm)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if nativeRsp.Code != wechat.Success {
|
|
return nil, errors.New(nativeRsp.Error)
|
|
}
|
|
payResp := &PayResp{
|
|
PayUrl: nativeRsp.Response.CodeUrl,
|
|
OrderId: r.PaymentName, // Wechat can use paymentName as the OutTradeNo to query order status
|
|
}
|
|
return payResp, nil
|
|
}
|
|
}
|
|
|
|
func (pp *WechatPaymentProvider) Notify(body []byte, orderId string) (*NotifyResult, error) {
|
|
notifyResult := &NotifyResult{}
|
|
queryRsp, err := pp.Client.V3TransactionQueryOrder(context.Background(), wechat.OutTradeNo, orderId)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if queryRsp.Code != wechat.Success {
|
|
return nil, errors.New(queryRsp.Error)
|
|
}
|
|
|
|
switch queryRsp.Response.TradeState {
|
|
case "SUCCESS":
|
|
// skip
|
|
case "CLOSED":
|
|
notifyResult.PaymentStatus = PaymentStateCanceled
|
|
return notifyResult, nil
|
|
case "NOTPAY", "USERPAYING": // not-pad: waiting for user to pay; user-paying: user is paying
|
|
notifyResult.PaymentStatus = PaymentStateCreated
|
|
return notifyResult, nil
|
|
default:
|
|
notifyResult.PaymentStatus = PaymentStateError
|
|
notifyResult.NotifyMessage = fmt.Sprintf("unexpected wechat trade state: %v", queryRsp.Response.TradeState)
|
|
return notifyResult, nil
|
|
}
|
|
productDisplayName, productName, providerName, _ := parseAttachString(queryRsp.Response.Attach)
|
|
notifyResult = &NotifyResult{
|
|
ProductName: productName,
|
|
ProductDisplayName: productDisplayName,
|
|
ProviderName: providerName,
|
|
OrderId: orderId,
|
|
Price: priceInt64ToFloat64(int64(queryRsp.Response.Amount.Total)),
|
|
PaymentStatus: PaymentStatePaid,
|
|
PaymentName: queryRsp.Response.OutTradeNo,
|
|
}
|
|
return notifyResult, nil
|
|
}
|
|
|
|
func (pp *WechatPaymentProvider) GetInvoice(paymentName string, personName string, personIdCard string, personEmail string, personPhone string, invoiceType string, invoiceTitle string, invoiceTaxId string) (string, error) {
|
|
return "", nil
|
|
}
|
|
|
|
func (pp *WechatPaymentProvider) GetResponseError(err error) string {
|
|
response := &WechatPayNotifyResponse{
|
|
Code: "SUCCESS",
|
|
Message: "",
|
|
}
|
|
|
|
if err != nil {
|
|
response.Code = "FAIL"
|
|
response.Message = err.Error()
|
|
}
|
|
|
|
return util.StructToJson(response)
|
|
}
|