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
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() |
|
} |
|
}
|
|
|