印度包网
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 

337 lines
9.5 KiB

package richpay
import (
"errors"
"fmt"
"net/http"
"server/config"
"server/modules/pay/base"
"server/modules/pay/values"
"server/pb"
"server/util"
"time"
"github.com/gogo/protobuf/proto"
"github.com/liangdas/mqant/log"
)
func NewSub(b *base.Base) {
sub := &Sub{
Base: b,
}
b.HttpType = base.HttpTypeJson
if b.Opt == base.OPTPay {
b.Resp = new(PayResp)
b.ReqURL = baseURL + payURL
} else if b.Opt == base.OPTWithdraw {
b.Resp = new(WithdrawResp)
b.ReqURL = baseURL + withdrawURL
} else if b.Opt == base.OPTPayCB {
b.CallbackReq = new(PayCallbackReq)
b.CallbackResp.Msg = "success"
} else if b.Opt == base.OPTWithdrawCB {
b.CallbackReq = new(WithdrawCallbackReq)
b.CallbackResp.Msg = "success"
} else if b.Opt == base.OPTQueryWithdraw { // 查询
b.Resp = new(QueryWithdrawResp)
b.ReqURL = baseURL + queryWithdrawURL
} else if b.Opt == base.OPTQueryPay { // 查询
b.Resp = new(QueryPayResp)
b.ReqURL = baseURL + queryPayURL
} else {
return
}
b.Sub = sub
}
type Sub struct {
Base *base.Base
}
func (s *Sub) PackHeader(header http.Header) {
if accessToken == "" {
GetAccessToken()
}
header.Add("Access-Token", accessToken)
}
func (s *Sub) PackReq() interface{} {
if s.Base.Opt == base.OPTPay {
return s.PackPayReq()
} else if s.Base.Opt == base.OPTWithdraw {
return s.PackWithdrawReq()
} else if s.Base.Opt == base.OPTQueryWithdraw {
return s.PackQueryWithdrawReq()
} else if s.Base.Opt == base.OPTQueryPay {
return s.PackQueryPayReq()
}
return nil
}
func (s *Sub) GetResp() (proto.Message, error) {
log.Debug("resp:%v", s.Base.Resp)
if s.Base.Opt == base.OPTPay {
resp := s.Base.Resp.(*PayResp)
if resp.BusCode == "AO-000103" { // token错误
GetAccessToken()
return nil, errors.New("pay fail")
}
if resp.Code != 200 || resp.Data.PayURL == "" {
return nil, errors.New("pay fail")
}
return &pb.InnerRechargeResp{APIOrderID: resp.Data.OrderNo, URL: resp.Data.PayURL, Channel: uint32(values.RichPay)}, nil
} else if s.Base.Opt == base.OPTWithdraw {
resp := s.Base.Resp.(*WithdrawResp)
if resp.BusCode == "AO-000103" { // token错误
GetAccessToken()
return nil, errors.New("withdraw fail")
}
if resp.Msg == "SV-000000" {
return nil, errors.New("withdraw fail")
}
if s.Base.Status == 0 && resp.Code != 200 {
return nil, errors.New("withdraw fail")
}
return &pb.InnerWithdrawResp{APIOrderID: resp.Data.OrderNo, Channel: uint32(values.RichPay)}, nil
} else if s.Base.Opt == base.OPTQueryWithdraw {
resp := s.Base.Resp.(*QueryWithdrawResp)
ret := &pb.InnerQueryWithdrawResp{Msg: resp.Msg}
if resp.BusCode == "AO-000103" { // token错误
GetAccessToken()
s.Base.QueryWithdrawResp.Status = base.QueryStatusUnknown
return ret, nil
}
if resp.Code != 200 {
s.Base.QueryWithdrawResp.Status = base.QueryStatusUnknown
return ret, nil
}
s.Base.QueryWithdrawResp.OrderID = resp.Data.MerchantOrderNo
s.Base.QueryWithdrawResp.Msg = resp.Msg
if s.Base.Status != 0 {
s.Base.QueryWithdrawResp.Status = base.QueryStatusUnknown
} else {
if resp.Data.Status == "2" {
s.Base.QueryWithdrawResp.Status = base.QueryStatusOrderSuccess
} else if resp.Data.Status == "3" || resp.Data.Status == "4" || resp.Data.Status == "5" {
s.Base.QueryWithdrawResp.Status = base.QueryStatusFail
} else {
s.Base.QueryWithdrawResp.Status = base.QueryStatusSuccess
}
}
return ret, nil
} else if s.Base.Opt == base.OPTQueryPay {
resp := s.Base.Resp.(*QueryPayResp)
ret := &pb.InnerQueryWithdrawResp{Msg: resp.Msg}
if resp.BusCode == "AO-000103" { // token错误
GetAccessToken()
s.Base.QueryWithdrawResp.Status = base.QueryStatusUnknown
return ret, nil
}
if resp.Code != 200 {
s.Base.QueryPayResp.Status = base.QueryStatusUnknown
return ret, nil
}
s.Base.QueryWithdrawResp.OrderID = resp.Data.MerchantOrderNo
s.Base.QueryPayResp.Msg = resp.Msg
if s.Base.Status != 0 {
s.Base.QueryPayResp.Status = base.QueryStatusUnknown
} else {
if resp.Data.Status == "2" {
s.Base.QueryPayResp.Status = base.QueryStatusOrderSuccess
} else if resp.Data.Status == "3" {
s.Base.QueryPayResp.Status = base.QueryStatusFail
} else {
s.Base.QueryPayResp.Status = base.QueryStatusSuccess
}
}
return ret, nil
}
return nil, errors.New("unknown opt")
}
func (s *Sub) PackPayReq() interface{} {
r := s.Base.PayReq
mid := config.GetConfig().Pay.RichPay.MID
privateKey := config.GetConfig().Pay.RichPay.PrivateKey
send := &PayReq{
MerchantID: mid,
MerchantOrderNo: r.OrderID,
OrderAmount: fmt.Sprintf("%d", r.Amount),
Email: r.Email,
Name: r.Name,
Phone: r.Phone,
DeeplinkSwitch: "0",
NotifyURL: s.Base.GetPayCallbackURL(),
NonceStr: util.GenerateRandomString(6),
Action: "payin",
Timestamp: fmt.Sprintf("%d", time.Now().Unix()),
}
sign, err := base.SignDsa(mid+send.MerchantOrderNo+send.OrderAmount+send.NonceStr+send.Timestamp+send.Action, privateKey)
if err != nil {
log.Error("err:%v", err)
return nil
}
send.Sign = sign
return send
}
func (s *Sub) PackWithdrawReq() interface{} {
r := s.Base.WithdrawReq
mid := config.GetConfig().Pay.RichPay.MID
privateKey := config.GetConfig().Pay.RichPay.PrivateKey
send := &WithdrawReq{
MerchantID: mid,
MerchantOrderNo: r.OrderID,
OrderAmount: r.Amount,
Name: r.Name,
Email: r.Email,
Phone: r.Phone,
NotifyURL: s.Base.GetWithdrawCallbackURL(),
NonceStr: util.GenerateRandomString(6),
Action: "payout",
Timestamp: fmt.Sprintf("%d", time.Now().Unix()),
// Type: ,
// VPA: ,
// Account: ,
// IFSC: ,
}
//if fmt.Sprintf("%d", r.PayType) == common.WithdrawTypeBank {
send.Type = "BANK"
send.Account = r.CardNo
send.IFSC = r.PayCode
//} else if fmt.Sprintf("%d", r.PayType) == common.WithdrawTypeUPI {
// send.Type = "UPI"
// send.VPA = r.PayCode
//} else {
// return nil
//}
str := fmt.Sprintf("%s%s%d%s%s%s", mid, send.MerchantOrderNo, send.OrderAmount, send.NonceStr, send.Timestamp, send.Action)
sign, err := base.SignDsa(str, privateKey)
if err != nil {
log.Error("err:%v", err)
return nil
}
send.Sign = sign
return send
}
func (s *Sub) CheckSign(str string) bool {
log.Debug("callback:%v", s.Base.CallbackReq)
if s.Base.Opt == base.OPTPayCB {
req := s.Base.CallbackReq.(*PayCallbackReq)
s.Base.CallbackResp.OrderID = req.MerchantOrderNo
s.Base.CallbackResp.APIOrderID = req.OrderNo
if req.Status == "0" || req.Status == "1" {
return false
}
s.Base.CallbackResp.Success = req.Status == "2"
message := req.MerchantOrderNo + req.OrderNo + req.OrderAmount + req.NonceStr + req.Timestamp
pass, err := base.VerifyDsa(message, req.Sign, publicKey)
if err != nil {
log.Error("err:%v", err)
return false
}
if !pass {
log.Error("check sign fail,message:%s", message)
}
return pass
} else if s.Base.Opt == base.OPTWithdrawCB {
req := s.Base.CallbackReq.(*WithdrawCallbackReq)
s.Base.CallbackResp.OrderID = req.MerchantOrderNo
if req.Status == "0" || req.Status == "1" {
return false
}
s.Base.CallbackResp.Success = req.Status == "2"
message := req.MerchantOrderNo + req.OrderNo + req.OrderAmount + req.NonceStr + req.Timestamp
pass, err := base.VerifyDsa(message, req.Sign, publicKey)
if err != nil {
log.Error("err:%v", err)
return false
}
if !pass {
log.Error("check sign fail,message:%s", message)
}
return pass
}
return false
}
func (s *Sub) PackQueryWithdrawReq() interface{} {
r := s.Base.QueryWithdrawReq
send := &QueryWithdrawReq{
MerchantOrderNo: r.OrderID,
}
return send
}
func (s *Sub) PackQueryPayReq() interface{} {
r := s.Base.QueryPayReq
send := &QueryPayReq{
MerchantOrderNo: r.OrderID,
}
return send
}
func GetAccessToken() {
url := baseURL + tokenURL
resp := &TokenResp{}
header := map[string]string{"clientid": config.GetConfig().Pay.RichPay.MID, "secret": config.GetConfig().Pay.RichPay.Secret}
util.HttpPost(url, nil, resp, header)
if resp.Code == 200 {
accessToken = resp.AccessToken
refreshToken = resp.RefreshToken
next := resp.ExpiresIn - 200
if next <= 0 {
next = 60
}
if config.GetBase().ServerFlag == "a" { // 只有a服去刷新token
log.Debug("next refresh token:%d", next)
tokenTimerLock.Lock()
if tokenTimer != nil {
tokenTimer.Stop()
}
tokenTimer = time.AfterFunc(time.Duration(next)*time.Second, RefreshAccessToken)
tokenTimerLock.Unlock()
} else {
log.Debug("next GetAccessToken token:%d", next+1)
tokenTimerLock.Lock()
if tokenTimer != nil {
tokenTimer.Stop()
}
tokenTimer = time.AfterFunc(time.Duration(next+1)*time.Second, GetAccessToken)
tokenTimerLock.Unlock()
}
} else {
tokenTimerLock.Lock()
if tokenTimer != nil {
tokenTimer.Stop()
}
log.Debug("GetAccessToken fail resp:%+v", resp)
tokenTimer = time.AfterFunc(10*time.Second, GetAccessToken)
tokenTimerLock.Unlock()
}
}
func RefreshAccessToken() {
url := baseURL + refreshTokenURL
resp := &TokenResp{}
header := map[string]string{"Refresh-Token": refreshToken}
util.HttpPost(url, nil, resp, header)
if resp.Code == 200 {
accessToken = resp.AccessToken
refreshToken = resp.RefreshToken
tokenTimerLock.Lock()
if tokenTimer != nil {
tokenTimer.Stop()
}
tokenTimer = time.AfterFunc(time.Duration(resp.ExpiresIn-200)*time.Second, RefreshAccessToken)
tokenTimerLock.Unlock()
} else {
tokenTimerLock.Lock()
if tokenTimer != nil {
tokenTimer.Stop()
}
tokenTimer = time.AfterFunc(10*time.Second, GetAccessToken)
tokenTimerLock.Unlock()
}
}