plugin 云闪付,下单完成
This commit is contained in:
parent
eabf8b7b65
commit
aa2d29032f
|
@ -0,0 +1,37 @@
|
|||
module plugins/union_pay
|
||||
|
||||
go 1.21
|
||||
|
||||
require (
|
||||
codeup.aliyun.com/6552e56cc3b2728a4557fc18/plugin v0.0.0-20240703025635-39098bff7a4c
|
||||
github.com/carlmjohnson/requests v0.23.5
|
||||
github.com/go-playground/validator/v10 v10.22.0
|
||||
github.com/hashicorp/go-plugin v1.6.1
|
||||
github.com/stretchr/testify v1.9.0
|
||||
)
|
||||
|
||||
require (
|
||||
github.com/davecgh/go-spew v1.1.1 // indirect
|
||||
github.com/envoyproxy/protoc-gen-validate v1.0.4 // indirect
|
||||
github.com/fatih/color v1.7.0 // indirect
|
||||
github.com/gabriel-vasile/mimetype v1.4.3 // indirect
|
||||
github.com/go-playground/locales v0.14.1 // indirect
|
||||
github.com/go-playground/universal-translator v0.18.1 // indirect
|
||||
github.com/golang/protobuf v1.5.4 // indirect
|
||||
github.com/hashicorp/go-hclog v0.14.1 // indirect
|
||||
github.com/hashicorp/yamux v0.1.1 // indirect
|
||||
github.com/leodido/go-urn v1.4.0 // indirect
|
||||
github.com/mattn/go-colorable v0.1.4 // indirect
|
||||
github.com/mattn/go-isatty v0.0.10 // indirect
|
||||
github.com/mitchellh/go-testing-interface v0.0.0-20171004221916-a61a99592b77 // indirect
|
||||
github.com/oklog/run v1.0.0 // indirect
|
||||
github.com/pmezard/go-difflib v1.0.0 // indirect
|
||||
golang.org/x/crypto v0.21.0 // indirect
|
||||
golang.org/x/net v0.22.0 // indirect
|
||||
golang.org/x/sys v0.18.0 // indirect
|
||||
golang.org/x/text v0.14.0 // indirect
|
||||
google.golang.org/genproto/googleapis/rpc v0.0.0-20240318140521-94a12d6c2237 // indirect
|
||||
google.golang.org/grpc v1.64.0 // indirect
|
||||
google.golang.org/protobuf v1.34.2 // indirect
|
||||
gopkg.in/yaml.v3 v3.0.1 // indirect
|
||||
)
|
|
@ -0,0 +1,70 @@
|
|||
codeup.aliyun.com/6552e56cc3b2728a4557fc18/plugin v0.0.0-20240703025635-39098bff7a4c h1:JnLzT0jm8viTbNMdMuC+sVQJJnVHo9K+OZ6JouLq118=
|
||||
codeup.aliyun.com/6552e56cc3b2728a4557fc18/plugin v0.0.0-20240703025635-39098bff7a4c/go.mod h1:QdW8HjHYQN8LCkFAB9e4oh7HziePCYnDXnUaKtmb8iQ=
|
||||
github.com/bufbuild/protocompile v0.4.0 h1:LbFKd2XowZvQ/kajzguUp2DC9UEIQhIq77fZZlaQsNA=
|
||||
github.com/bufbuild/protocompile v0.4.0/go.mod h1:3v93+mbWn/v3xzN+31nwkJfrEpAUwp+BagBSZWx+TP8=
|
||||
github.com/carlmjohnson/requests v0.23.5 h1:NPANcAofwwSuC6SIMwlgmHry2V3pLrSqRiSBKYbNHHA=
|
||||
github.com/carlmjohnson/requests v0.23.5/go.mod h1:zG9P28thdRnN61aD7iECFhH5iGGKX2jIjKQD9kqYH+o=
|
||||
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
|
||||
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||
github.com/envoyproxy/protoc-gen-validate v1.0.4 h1:gVPz/FMfvh57HdSJQyvBtF00j8JU4zdyUgIUNhlgg0A=
|
||||
github.com/envoyproxy/protoc-gen-validate v1.0.4/go.mod h1:qys6tmnRsYrQqIhm2bvKZH4Blx/1gTIZ2UKVY1M+Yew=
|
||||
github.com/fatih/color v1.7.0 h1:DkWD4oS2D8LGGgTQ6IvwJJXSL5Vp2ffcQg58nFV38Ys=
|
||||
github.com/fatih/color v1.7.0/go.mod h1:Zm6kSWBoL9eyXnKyktHP6abPY2pDugNf5KwzbycvMj4=
|
||||
github.com/gabriel-vasile/mimetype v1.4.3 h1:in2uUcidCuFcDKtdcBxlR0rJ1+fsokWf+uqxgUFjbI0=
|
||||
github.com/gabriel-vasile/mimetype v1.4.3/go.mod h1:d8uq/6HKRL6CGdk+aubisF/M5GcPfT7nKyLpA0lbSSk=
|
||||
github.com/go-playground/assert/v2 v2.2.0 h1:JvknZsQTYeFEAhQwI4qEt9cyV5ONwRHC+lYKSsYSR8s=
|
||||
github.com/go-playground/assert/v2 v2.2.0/go.mod h1:VDjEfimB/XKnb+ZQfWdccd7VUvScMdVu0Titje2rxJ4=
|
||||
github.com/go-playground/locales v0.14.1 h1:EWaQ/wswjilfKLTECiXz7Rh+3BjFhfDFKv/oXslEjJA=
|
||||
github.com/go-playground/locales v0.14.1/go.mod h1:hxrqLVvrK65+Rwrd5Fc6F2O76J/NuW9t0sjnWqG1slY=
|
||||
github.com/go-playground/universal-translator v0.18.1 h1:Bcnm0ZwsGyWbCzImXv+pAJnYK9S473LQFuzCbDbfSFY=
|
||||
github.com/go-playground/universal-translator v0.18.1/go.mod h1:xekY+UJKNuX9WP91TpwSH2VMlDf28Uj24BCp08ZFTUY=
|
||||
github.com/go-playground/validator/v10 v10.22.0 h1:k6HsTZ0sTnROkhS//R0O+55JgM8C4Bx7ia+JlgcnOao=
|
||||
github.com/go-playground/validator/v10 v10.22.0/go.mod h1:dbuPbCMFw/DrkbEynArYaCwl3amGuJotoKCe95atGMM=
|
||||
github.com/golang/protobuf v1.5.4 h1:i7eJL8qZTpSEXOPTxNKhASYpMn+8e5Q6AdndVa1dWek=
|
||||
github.com/golang/protobuf v1.5.4/go.mod h1:lnTiLA8Wa4RWRcIUkrtSVa5nRhsEGBg48fD6rSs7xps=
|
||||
github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI=
|
||||
github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY=
|
||||
github.com/hashicorp/go-hclog v0.14.1 h1:nQcJDQwIAGnmoUWp8ubocEX40cCml/17YkF6csQLReU=
|
||||
github.com/hashicorp/go-hclog v0.14.1/go.mod h1:whpDNt7SSdeAju8AWKIWsul05p54N/39EeqMAyrmvFQ=
|
||||
github.com/hashicorp/go-plugin v1.6.1 h1:P7MR2UP6gNKGPp+y7EZw2kOiq4IR9WiqLvp0XOsVdwI=
|
||||
github.com/hashicorp/go-plugin v1.6.1/go.mod h1:XPHFku2tFo3o3QKFgSYo+cghcUhw1NA1hZyMK0PWAw0=
|
||||
github.com/hashicorp/yamux v0.1.1 h1:yrQxtgseBDrq9Y652vSRDvsKCJKOUD+GzTS4Y0Y8pvE=
|
||||
github.com/hashicorp/yamux v0.1.1/go.mod h1:CtWFDAQgb7dxtzFs4tWbplKIe2jSi3+5vKbgIO0SLnQ=
|
||||
github.com/jhump/protoreflect v1.15.1 h1:HUMERORf3I3ZdX05WaQ6MIpd/NJ434hTp5YiKgfCL6c=
|
||||
github.com/jhump/protoreflect v1.15.1/go.mod h1:jD/2GMKKE6OqX8qTjhADU1e6DShO+gavG9e0Q693nKo=
|
||||
github.com/leodido/go-urn v1.4.0 h1:WT9HwE9SGECu3lg4d/dIA+jxlljEa1/ffXKmRjqdmIQ=
|
||||
github.com/leodido/go-urn v1.4.0/go.mod h1:bvxc+MVxLKB4z00jd1z+Dvzr47oO32F/QSNjSBOlFxI=
|
||||
github.com/mattn/go-colorable v0.1.4 h1:snbPLB8fVfU9iwbbo30TPtbLRzwWu6aJS6Xh4eaaviA=
|
||||
github.com/mattn/go-colorable v0.1.4/go.mod h1:U0ppj6V5qS13XJ6of8GYAs25YV2eR4EVcfRqFIhoBtE=
|
||||
github.com/mattn/go-isatty v0.0.8/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hdxcsrc5s=
|
||||
github.com/mattn/go-isatty v0.0.10 h1:qxFzApOv4WsAL965uUPIsXzAKCZxN2p9UqdhFS4ZW10=
|
||||
github.com/mattn/go-isatty v0.0.10/go.mod h1:qgIWMr58cqv1PHHyhnkY9lrL7etaEgOFcMEpPG5Rm84=
|
||||
github.com/mitchellh/go-testing-interface v0.0.0-20171004221916-a61a99592b77 h1:7GoSOOW2jpsfkntVKaS2rAr1TJqfcxotyaUcuxoZSzg=
|
||||
github.com/mitchellh/go-testing-interface v0.0.0-20171004221916-a61a99592b77/go.mod h1:kRemZodwjscx+RGhAo8eIhFbs2+BFgRtFPeD/KE+zxI=
|
||||
github.com/oklog/run v1.0.0 h1:Ru7dDtJNOyC66gQ5dQmaCa0qIsAUFY3sFpK1Xk8igrw=
|
||||
github.com/oklog/run v1.0.0/go.mod h1:dlhp/R75TPv97u0XWUtDeV/lRKWPKSdTuV0TZvrmrQA=
|
||||
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
||||
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
||||
github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs=
|
||||
github.com/stretchr/testify v1.9.0 h1:HtqpIVDClZ4nwg75+f6Lvsy/wHu+3BoSGCbBAcpTsTg=
|
||||
github.com/stretchr/testify v1.9.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY=
|
||||
golang.org/x/crypto v0.21.0 h1:X31++rzVUdKhX5sWmSOFZxx8UW/ldWx55cbf08iNAMA=
|
||||
golang.org/x/crypto v0.21.0/go.mod h1:0BP7YvVV9gBbVKyeTG0Gyn+gZm94bibOW5BjDEYAOMs=
|
||||
golang.org/x/net v0.22.0 h1:9sGLhx7iRIHEiX0oAJ3MRZMUCElJgy7Br1nO+AMN3Tc=
|
||||
golang.org/x/net v0.22.0/go.mod h1:JKghWKKOSdJwpW2GEx0Ja7fmaKnMsbu+MWVZTokSYmg=
|
||||
golang.org/x/sys v0.0.0-20190222072716-a9d3bda3a223/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
||||
golang.org/x/sys v0.0.0-20191008105621-543471e840be/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.18.0 h1:DBdB3niSjOA/O0blCZBqDefyWNYveAYMNF1Wum0DYQ4=
|
||||
golang.org/x/sys v0.18.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
|
||||
golang.org/x/text v0.14.0 h1:ScX5w1eTa3QqT8oi6+ziP7dTV1S2+ALU0bI+0zXKWiQ=
|
||||
golang.org/x/text v0.14.0/go.mod h1:18ZOQIKpY8NJVqYksKHtTdi31H5itFRjB5/qKTNYzSU=
|
||||
google.golang.org/genproto/googleapis/rpc v0.0.0-20240318140521-94a12d6c2237 h1:NnYq6UN9ReLM9/Y01KWNOWyI5xQ9kbIms5GGJVwS/Yc=
|
||||
google.golang.org/genproto/googleapis/rpc v0.0.0-20240318140521-94a12d6c2237/go.mod h1:WtryC6hu0hhx87FDGxWCDptyssuo68sk10vYjF+T9fY=
|
||||
google.golang.org/grpc v1.64.0 h1:KH3VH9y/MgNQg1dE7b3XfVK0GsPSIzJwdF617gUSbvY=
|
||||
google.golang.org/grpc v1.64.0/go.mod h1:oxjF8E3FBnjp+/gVFYdWacaLDx9na1aqy9oovLpxQYg=
|
||||
google.golang.org/protobuf v1.34.2 h1:6xV6lTsCfpGD21XK49h7MhtcApnLqkfYgPcdHftf6hg=
|
||||
google.golang.org/protobuf v1.34.2/go.mod h1:qYOHts0dSfpeUzUFpOMr/WGzszTmLH+DiWniOlNbLDw=
|
||||
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM=
|
||||
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
||||
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
|
||||
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
|
@ -0,0 +1,11 @@
|
|||
package po
|
||||
|
||||
import "plugins/union_pay/internal/vo"
|
||||
|
||||
type Notify struct {
|
||||
MerchantId int `json:"merchantId" validate:"required"`
|
||||
OutTradeNo string `json:"outTradeNo" validate:"required"`
|
||||
RechargeAccount string `json:"rechargeAccount" validate:"required"`
|
||||
Status vo.OrderStatus `json:"status" validate:"required"`
|
||||
Sign string `json:"sign" validate:"required"`
|
||||
}
|
|
@ -0,0 +1,38 @@
|
|||
package po
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"plugins/union_pay/internal/vo"
|
||||
)
|
||||
|
||||
type OrderReq struct {
|
||||
ChNlId string `validate:"required" json:"chnlId"` // 渠道方代码-必填
|
||||
Qid string `validate:"required" json:"qid"` // 外部接入方定义的主键,保证唯一,即流水号,用于幂等校验
|
||||
OrderDt string `validate:"required" json:"orderDt"` // 请求日期-格式为yyyyMMdd
|
||||
// 银联营销平台用于上下文串联,取值可与QID相同,不强制校验是否唯一,-请求跟踪号
|
||||
TraceId string `validate:"required" json:"traceId"`
|
||||
// 优惠券ID-票券活动ID,由银联营销平台生成,31打头的16位数字 ,同优惠券状态通知接口的couponId
|
||||
DiscountId string `validate:"required" json:"discountId"`
|
||||
// 01:手机号维度;02:卡号维度;03:云闪付注册用户ID维度;05:外部用户ID维度;06:网络支付平台用户。
|
||||
// 07:证件号用户维度。票券的账户维度,同一个优惠券ID下的优惠券,其发放维度与核销维度一致,银行需确保支付交易中上送对应的要素。
|
||||
EntityTp string `validate:"required" json:"entityTp"`
|
||||
// 当entityTp为03时,cardNo、mobile、openId三选一必填;当entityTp为01时,mobile必填 该参数已配置加密规则:SM4
|
||||
Mobile string `validate:"required" json:"mobile"`
|
||||
Cmd string `validate:"required" json:"cmd"`
|
||||
AccessId string `validate:"required" json:"accessId"`
|
||||
DiscountNum uint32 `validate:"required" json:"discountNum"`
|
||||
}
|
||||
|
||||
type OrderResp struct {
|
||||
Code vo.Code `json:"code"`
|
||||
Msg string `json:"msg"`
|
||||
SubCode vo.SubCode `json:"subCode"`
|
||||
SubMsg string `json:"subMsg"`
|
||||
DiscountId string `json:"discountId"`
|
||||
Qid string `json:"qid"`
|
||||
CouponCd string `json:"couponCd"`
|
||||
}
|
||||
|
||||
func (o *OrderResp) GetMsg() string {
|
||||
return fmt.Sprintf("Msg:%s,SubMsg:%s,自定处理msg:%s", o.Msg, o.SubMsg, o.SubCode.GetMsg())
|
||||
}
|
|
@ -0,0 +1,74 @@
|
|||
package po
|
||||
|
||||
import (
|
||||
"encoding/json"
|
||||
"fmt"
|
||||
"github.com/go-playground/validator/v10"
|
||||
)
|
||||
|
||||
type Req interface {
|
||||
Validate() error
|
||||
GetReId() string
|
||||
ToJson() []byte
|
||||
}
|
||||
|
||||
var _ Req = (*OrderReq)(nil)
|
||||
var _ Req = (*QueryReq)(nil)
|
||||
var _ Req = (*Notify)(nil)
|
||||
|
||||
func (req *OrderReq) Validate() error {
|
||||
err := validator.New().Struct(req)
|
||||
if err != nil {
|
||||
for _, err = range err.(validator.ValidationErrors) {
|
||||
return fmt.Errorf("参数有误:" + err.Error())
|
||||
}
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (req *OrderReq) GetReId() string {
|
||||
return req.Qid
|
||||
}
|
||||
|
||||
func (req *OrderReq) ToJson() []byte {
|
||||
b, _ := json.Marshal(req)
|
||||
return b
|
||||
}
|
||||
|
||||
func (req *QueryReq) Validate() error {
|
||||
err := validator.New().Struct(req)
|
||||
if err != nil {
|
||||
for _, err = range err.(validator.ValidationErrors) {
|
||||
return fmt.Errorf("参数有误:" + err.Error())
|
||||
}
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (req *QueryReq) GetReId() string {
|
||||
return ""
|
||||
}
|
||||
|
||||
func (req *QueryReq) ToJson() []byte {
|
||||
b, _ := json.Marshal(req)
|
||||
return b
|
||||
}
|
||||
|
||||
func (req *Notify) Validate() error {
|
||||
err := validator.New().Struct(req)
|
||||
if err != nil {
|
||||
for _, err = range err.(validator.ValidationErrors) {
|
||||
return fmt.Errorf("参数有误:" + err.Error())
|
||||
}
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (req *Notify) GetReId() string {
|
||||
return ""
|
||||
}
|
||||
|
||||
func (req *Notify) ToJson() []byte {
|
||||
b, _ := json.Marshal(req)
|
||||
return b
|
||||
}
|
|
@ -0,0 +1,18 @@
|
|||
package po
|
||||
|
||||
import "plugins/union_pay/internal/vo"
|
||||
|
||||
type QueryReq struct {
|
||||
MerchantId string `validate:"required"`
|
||||
OutTradeNo string `validate:"required"`
|
||||
TimeStamp int64 `validate:"required"`
|
||||
Version string `validate:"required"`
|
||||
Sign string
|
||||
}
|
||||
|
||||
type QueryResp struct {
|
||||
Code vo.Code `json:"code"`
|
||||
Status vo.OrderStatus `json:"status"`
|
||||
Message string `json:"message"`
|
||||
OutTradeNo string `json:"outTradeNo"`
|
||||
}
|
|
@ -0,0 +1,71 @@
|
|||
package internal
|
||||
|
||||
import (
|
||||
"codeup.aliyun.com/6552e56cc3b2728a4557fc18/plugin/proto"
|
||||
"encoding/json"
|
||||
"plugins/union_pay/internal/po"
|
||||
)
|
||||
|
||||
type Config struct {
|
||||
Config *proto.Config
|
||||
Extra *ConfigExtra
|
||||
}
|
||||
|
||||
type ConfigExtra struct {
|
||||
ChNlId string `json:"chnlId"` // 渠道方代码
|
||||
IV string `json:"iv"` // 加密密钥
|
||||
KEY string `json:"key"` // 加密密钥
|
||||
RsaPrk string `json:"rsa_prk"` // 私钥
|
||||
RsaNpk string `json:"rsa_npk"` // 回调公钥
|
||||
}
|
||||
|
||||
func transformConfig(config *proto.Config) (*Config, error) {
|
||||
var extra ConfigExtra
|
||||
err := json.Unmarshal(config.Extra, &extra)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return &Config{
|
||||
Config: config,
|
||||
Extra: &extra,
|
||||
}, nil
|
||||
}
|
||||
|
||||
func rechargeReq(in *proto.OrderRequest, chNlId string) *po.OrderReq {
|
||||
type Extra struct {
|
||||
OrderDt string `json:"orderDt"`
|
||||
}
|
||||
var e Extra
|
||||
_ = json.Unmarshal(in.Order.Extra, &e)
|
||||
|
||||
return &po.OrderReq{
|
||||
Cmd: "couponAcquire",
|
||||
AccessId: "UP",
|
||||
EntityTp: "03",
|
||||
|
||||
ChNlId: chNlId,
|
||||
Qid: in.Order.OrderNo,
|
||||
OrderDt: e.OrderDt,
|
||||
TraceId: in.Order.OrderNo,
|
||||
DiscountId: in.Product.ProductNo,
|
||||
Mobile: in.Order.Account,
|
||||
DiscountNum: in.Order.Quantity,
|
||||
}
|
||||
}
|
||||
|
||||
func rechargeResp(request *proto.OrderRequest, resp po.OrderResp) *proto.OrderResponse {
|
||||
data, _ := json.Marshal(resp)
|
||||
result := &proto.Result{
|
||||
OrderNo: request.Order.OrderNo,
|
||||
TradeNo: resp.CouponCd,
|
||||
Message: resp.Msg,
|
||||
Data: data,
|
||||
}
|
||||
if !resp.Code.IsSuccess() {
|
||||
result.Status = proto.Status_FAIL
|
||||
result.Message = resp.GetMsg()
|
||||
} else {
|
||||
result.Status = proto.Status_ING
|
||||
}
|
||||
return &proto.OrderResponse{Result: result}
|
||||
}
|
|
@ -0,0 +1,54 @@
|
|||
package internal
|
||||
|
||||
import (
|
||||
"codeup.aliyun.com/6552e56cc3b2728a4557fc18/plugin/proto"
|
||||
"context"
|
||||
"fmt"
|
||||
"github.com/carlmjohnson/requests"
|
||||
"plugins/union_pay/internal/po"
|
||||
)
|
||||
|
||||
// 插件通信信息,若不对应则会报错panic
|
||||
const (
|
||||
Tag = "union_pay"
|
||||
Version = 1
|
||||
CookieKey = "union_pay"
|
||||
CookieValue = "union_pay"
|
||||
)
|
||||
|
||||
const (
|
||||
orderMethod = "/upapi/mkt/cpn/couponAcquire/v1"
|
||||
orderBizMethod = "mkt.cpn.couponAcquire.v1"
|
||||
)
|
||||
|
||||
const (
|
||||
queryMethod = "/upapi/mkt/cpn/couponAcqQuery/v1"
|
||||
queryBizMethod = "mkt.cpn.couponAcqQuery.v1"
|
||||
)
|
||||
|
||||
type UnionPayService struct{}
|
||||
|
||||
func (p *UnionPayService) Order(ctx context.Context, request *proto.OrderRequest) (*proto.OrderResponse, error) {
|
||||
conf, err := transformConfig(request.Config)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
uv := rechargeReq(request, conf.Extra.ChNlId)
|
||||
|
||||
var response po.OrderResp
|
||||
url := fmt.Sprintf("%s%s", request.Config.BaseUri, orderMethod)
|
||||
err = requests.URL(url).Headers(headers(conf, uv, orderBizMethod)).BodyJSON(uv).ToJSON(&response).Fetch(ctx)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("请求异常,msg:" + err.Error())
|
||||
}
|
||||
|
||||
return rechargeResp(request, response), nil
|
||||
}
|
||||
|
||||
func (p *UnionPayService) Query(ctx context.Context, request *proto.QueryRequest) (*proto.QueryResponse, error) {
|
||||
return nil, nil
|
||||
}
|
||||
|
||||
func (p *UnionPayService) Notify(_ context.Context, request *proto.NotifyRequest) (*proto.NotifyResponse, error) {
|
||||
return nil, nil
|
||||
}
|
|
@ -0,0 +1,87 @@
|
|||
package internal
|
||||
|
||||
import (
|
||||
"codeup.aliyun.com/6552e56cc3b2728a4557fc18/plugin/proto"
|
||||
"context"
|
||||
"fmt"
|
||||
"github.com/stretchr/testify/assert"
|
||||
"testing"
|
||||
)
|
||||
|
||||
var appId = "up_49pau3fu6latj_03h"
|
||||
var appKey = ""
|
||||
var baseUri = "https://openapi.unionpay.com"
|
||||
var notifyUrl = ""
|
||||
|
||||
var server = &UnionPayService{}
|
||||
|
||||
var config = &proto.Config{
|
||||
AppId: appId,
|
||||
AppKey: appKey,
|
||||
BaseUri: baseUri,
|
||||
NotifyUrl: notifyUrl,
|
||||
Extra: []byte(`{"chnlId":"8091","iv":"0123456789123456","key":"16fb034dead44ff6f599b1ca373ccfd1",
|
||||
"rsa_prk":"MIIEogIBAAKCAQEAllAo3AbVfNg1iDT3CisgueIm2Pya2C6qatF0S8/aAKW8A6HBIuvrk1sb2TjbI0UDVowaNDsPa7nNdZHkQa6CQKf2BwFRy7gB8vAD1YSq23NTwplGe5IpuRe2Cbljh5diQ/+0zwZPZQEvjSMqZJILYVluGBw656JSndRdXHCs608CBib14l35A6D7rSZNd53nM9yXj/WOpLuY/jX0sbKJrBUIhcQcMiCFK/Nx06vhhCGZSWwaj0TTSMEOj9nyU+56MMx+N7527JUnOYMyVbbEx5h1S5sPjntMVVzz8c8FC5MKbgXCgjTt5Depl6YgrvoSCNNXZqnXN2l9dtBxKGSEdQIDAQABAoIBAD7r606/pZqPP0l3MnqVNyvY2X43r/ITBs+UQmSB67Ydqqiyi5C8xW530x6JSGJpP055c3atynTD4Jf4rF46WNgL91dTXBQ1QMubYV9+G6+lhUiOtQyBUOnkXRWQ/3MBEed5IQ0QjSew6WDheZuD7zOfJhD9sELKll7vSO2rVRdi5Kf3pps0K4aeYLvlelHjPtu2EkMyAD4d37QO2q8A/+acOf7wyx1uZ55nobe0jVgQBa2Tuuwhd2JilmsgMvcVxeAJoocyQxPVTU+iMidgZKxyW/fJUaR7V2KgKfP0H2OZbSruD2HyHL7o70+/XBg250msh691oTCXSGWiITFXRDkCgYEA1KF3daSzWK2vTZeNMxj++VRNZfyigNqBwgRZxrSGIUeNGpDdDp7flldM/fO/0RijKnh+nXLBCMZlxffSI7W3jkkjosEq0uyXv6Ez4WlhwHEAE6KrZYQ8VDU3We1LXERu/3SMU16Uq2Vme8J6kpshr7OLGeuTNfsSG79lpHU/OO8CgYEAtPjF/AY/t85u4JV0w6FJ/K7Iya+AyIyZhsgnxsPv5NwVtNsQzTRbXlpgnVmaKUXzLBTBJA2P6Gie35R6kJuOuMnvHjtG7umI5lOUJaqYk5GZEXHxbIKTXUxAbv+WKRHcFwcH54Ppurs7bN6ultWdE+yyU5PvabFRAidL+rV+MNsCgYBkqx2XwSZ4MaLxpXLgYlE1UkhipL17K/iLFuw6O3XVg762eSdvqKBK2YJGKYUDRhraHLo1aQMjWGsO00nWRk5lOxN4FuRjgEhIHdYmPnJdXiKfKUDkizrU020lUkd/o5a5s0BMqFbS3usqZOULTA8i1GpjFzlXMUtdf7xrpGcOWQKBgFJXWuX4Befg8owXFKCqc1qYQF2cEBGv/27XBVon7rAH/4xhnHxx1RzSLGGL0lvzKtikvNhln1PUvmzmRvoi/UZRAw6TVAwCkDqUTrTwsSl7ttS8Lmf75Ycu2aZnfnBDYwvLNCp/18oRxNEdZPzWRvk42k6y4d0KQ9yh+q0hBUC/AoGASgJro5HgKWkSzYx1Fk3rvo7EWeZwXNfWuzIhtCiVDG4+df0ZdybkbjxbaRwL2rbTSBeqSaZw4UwpE1JyF8erfzx0qpHCY5V3ocnA8kg56mX9kE5H9WKjMe1tbuZpba4i08zvM8vpz9gASf3z/WrWWqgpvynxDWpy4xFAqrWC0VQ=",
|
||||
"rsa_npk":"MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAy05sCMKEYZQGryKKq7qxxMi4uIRcOuv0aQcptjEifc/TS56bU0iGlkuH9k90tPp6EoKYLUfnL1FdHQOWXboFktwoxjy372gc6CsQfU9ZNjsWMYI8CTZF6gWLk4HjfBO4HcCKJs7oGntgVZxfVqVxJ8N/M1ZC2sDuVkoHYymONgNwgO+yYXrAlPPfJBltG5QNOcGTHBZ4PTc6A0M17+2BiskKUJfLEbN2soEZR0b5v5qy0D1xe5w+CdNcvtEb5CemPRxxRYjIqfHlhQPO7ZCUFPmhtoIbtBYG+BIU+qK3Q1DDiBndOVkQGVBQcOnMfYpxp82hlA1TAVeRICWubH3nJwIDAQAB"}`),
|
||||
}
|
||||
|
||||
func TestOrder(t *testing.T) {
|
||||
request := &proto.OrderRequest{
|
||||
Config: config,
|
||||
Order: &proto.OrderRequest_Order{
|
||||
OrderNo: "240403164049635931",
|
||||
Account: "R2wVgntY8aP9QTvCN/bWfw==",
|
||||
Extra: []byte(`{orderDt:"20240704"}`),
|
||||
},
|
||||
Product: &proto.OrderRequest_Product{
|
||||
ProductNo: "3102024032977191",
|
||||
Extra: []byte(`{}`),
|
||||
},
|
||||
}
|
||||
t.Run("TestOrder", func(t *testing.T) {
|
||||
got, err := server.Order(context.Background(), request)
|
||||
if err != nil {
|
||||
t.Errorf("Order() error = %v", err)
|
||||
return
|
||||
}
|
||||
fmt.Printf("%+v", got)
|
||||
assert.Equal(t, int(proto.Status_ING), int(got.Result.Status))
|
||||
})
|
||||
}
|
||||
|
||||
func TestQuery(t *testing.T) {
|
||||
request := &proto.QueryRequest{
|
||||
Config: config,
|
||||
Order: &proto.QueryRequest_Order{
|
||||
OrderNo: "test_zltx_4",
|
||||
TradeNo: "",
|
||||
Account: "",
|
||||
Extra: nil,
|
||||
},
|
||||
}
|
||||
t.Run("TestQuery", func(t *testing.T) {
|
||||
got, err := server.Query(context.Background(), request)
|
||||
if err != nil {
|
||||
t.Errorf("Query() error = %v", err)
|
||||
return
|
||||
}
|
||||
fmt.Printf("%+v \n", got)
|
||||
assert.Equal(t, int(proto.Status_SUCCESS), int(got.Result.Status))
|
||||
})
|
||||
}
|
||||
|
||||
func TestNotify(t *testing.T) {
|
||||
in := &proto.NotifyRequest{
|
||||
Queries: nil,
|
||||
Headers: nil,
|
||||
Body: []byte(`{"merchantId":10, "outTradeNo":"123", "rechargeAccount":"1866666666", "status":"01", "sign":"sign"}`),
|
||||
}
|
||||
t.Run("TestNotify", func(t *testing.T) {
|
||||
got, err := server.Notify(context.Background(), in)
|
||||
if assert.Nil(t, err) {
|
||||
assert.Equal(t, true, got.Result)
|
||||
}
|
||||
fmt.Printf("%+v \n", got)
|
||||
assert.Equal(t, int(proto.Status_SUCCESS), int(got.Result.Status))
|
||||
})
|
||||
}
|
|
@ -0,0 +1,35 @@
|
|||
package internal
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"net/http"
|
||||
"plugins/union_pay/internal/po"
|
||||
"plugins/union_pay/internal/utils"
|
||||
"time"
|
||||
)
|
||||
|
||||
func headers(config *Config, req po.Req, bizMethod string) map[string][]string {
|
||||
version := "1.0.0"
|
||||
h := make(http.Header)
|
||||
h.Add("Content-type", "application/json")
|
||||
h.Add("version", version)
|
||||
h.Add("appType", "01")
|
||||
h.Add("signMethod", "RSA2")
|
||||
|
||||
h.Add("appId", config.Config.AppId)
|
||||
h.Add("bizMethod", bizMethod)
|
||||
|
||||
h.Add("reqId", req.GetReId())
|
||||
|
||||
now := time.Now()
|
||||
milliseconds := now.Unix()*1000 + int64(now.Nanosecond())/1e6
|
||||
h.Add("reqTs", fmt.Sprintf("%d", milliseconds))
|
||||
|
||||
encodedHash := utils.Sha(version, config.Config.AppId, bizMethod, req.GetReId(), string(req.ToJson()))
|
||||
signValue, err := utils.Sign(encodedHash, utils.FormatPEMPrivateKey(config.Extra.RsaPrk))
|
||||
if err != nil {
|
||||
return nil
|
||||
}
|
||||
h.Add("sign", signValue)
|
||||
return h
|
||||
}
|
|
@ -0,0 +1,28 @@
|
|||
package utils
|
||||
|
||||
import (
|
||||
"strings"
|
||||
)
|
||||
|
||||
func FormatPEMPrivateKey(privateKeyStr string) []byte {
|
||||
const pemHeader = "-----BEGIN RSA PRIVATE KEY-----"
|
||||
const pemFooter = "-----END RSA PRIVATE KEY-----"
|
||||
|
||||
return buildKey(pemHeader, privateKeyStr, pemFooter)
|
||||
}
|
||||
|
||||
func FormatPEMPublicKey(publicKeyStr string) []byte {
|
||||
const pemHeader = "-----BEGIN PUBLIC KEY-----"
|
||||
const pemFooter = "-----END PUBLIC KEY-----"
|
||||
|
||||
return buildKey(pemHeader, publicKeyStr, pemFooter)
|
||||
}
|
||||
|
||||
func buildKey(pemHeader, keyStr, pemFooter string) []byte {
|
||||
var lines []string
|
||||
lines = append(lines, pemHeader)
|
||||
lines = append(lines, keyStr)
|
||||
lines = append(lines, pemFooter)
|
||||
|
||||
return []byte(strings.Join(lines, "\n"))
|
||||
}
|
|
@ -0,0 +1,29 @@
|
|||
package utils
|
||||
|
||||
import (
|
||||
"crypto/sha256"
|
||||
"fmt"
|
||||
)
|
||||
|
||||
func Sha(version, appId, bizMethod, reId, body string) string {
|
||||
s := fmt.Sprintf("version=%s&appId=%s&bizMethod=%s&reqId=%s&body=%s", version, appId, bizMethod, reId, body)
|
||||
return encodeBySHA256([]byte(s))
|
||||
}
|
||||
|
||||
func encodeBySHA256(str []byte) string {
|
||||
hash := sha256.Sum256(str)
|
||||
return getFormattedText(hash[:])
|
||||
}
|
||||
|
||||
func getFormattedText(bytes []byte) string {
|
||||
HexDigits := "0123456789abcdef"
|
||||
|
||||
l := len(bytes)
|
||||
buf := make([]byte, l*2)
|
||||
for j := 0; j < l; j++ {
|
||||
buf[j*2] = HexDigits[(bytes[j]>>4)&0x0f]
|
||||
buf[j*2+1] = HexDigits[bytes[j]&0x0f]
|
||||
}
|
||||
|
||||
return string(buf)
|
||||
}
|
|
@ -0,0 +1,68 @@
|
|||
package utils
|
||||
|
||||
import (
|
||||
"crypto"
|
||||
"crypto/rand"
|
||||
"crypto/rsa"
|
||||
"crypto/sha256"
|
||||
"crypto/x509"
|
||||
"encoding/base64"
|
||||
"encoding/pem"
|
||||
"errors"
|
||||
"fmt"
|
||||
)
|
||||
|
||||
func Sign(data string, privateKeyPEM []byte) (string, error) {
|
||||
block, _ := pem.Decode(privateKeyPEM)
|
||||
if block == nil {
|
||||
return "", errors.New("failed to parse PEM block containing the private key")
|
||||
}
|
||||
|
||||
privyKey, err := x509.ParsePKCS1PrivateKey(block.Bytes)
|
||||
if err != nil {
|
||||
return "", fmt.Errorf("failed to parse DER encoded private key: %v", err)
|
||||
}
|
||||
|
||||
hashed := sha256.Sum256([]byte(data))
|
||||
signature, err := rsa.SignPKCS1v15(rand.Reader, privyKey, crypto.SHA256, hashed[:])
|
||||
if err != nil {
|
||||
return "", fmt.Errorf("failed to sign: %v", err)
|
||||
}
|
||||
|
||||
return base64.StdEncoding.EncodeToString(signature), nil
|
||||
}
|
||||
|
||||
func Verify(data, signature string, publicKeyPEM []byte) bool {
|
||||
block, _ := pem.Decode(publicKeyPEM)
|
||||
if block == nil {
|
||||
fmt.Println("failed to parse PEM block containing the public key")
|
||||
return false
|
||||
}
|
||||
|
||||
pubKey, err := x509.ParsePKIXPublicKey(block.Bytes)
|
||||
if err != nil {
|
||||
fmt.Println("error parsing public key:", err)
|
||||
return false
|
||||
}
|
||||
|
||||
rsaPubKey, ok := pubKey.(*rsa.PublicKey)
|
||||
if !ok {
|
||||
fmt.Println("not an RSA public key")
|
||||
return false
|
||||
}
|
||||
|
||||
hashed := sha256.Sum256([]byte(data))
|
||||
sig, err := base64.StdEncoding.DecodeString(signature)
|
||||
if err != nil {
|
||||
fmt.Println("error decoding signature:", err)
|
||||
return false
|
||||
}
|
||||
|
||||
err = rsa.VerifyPKCS1v15(rsaPubKey, crypto.SHA256, hashed[:], sig)
|
||||
if err != nil {
|
||||
fmt.Println("signature verification error:", err)
|
||||
return false
|
||||
}
|
||||
|
||||
return true
|
||||
}
|
|
@ -0,0 +1,22 @@
|
|||
package utils
|
||||
|
||||
import (
|
||||
"testing"
|
||||
)
|
||||
|
||||
var privateKeyPEM = FormatPEMPrivateKey("MIIEogIBAAKCAQEAllAo3AbVfNg1iDT3CisgueIm2Pya2C6qatF0S8/aAKW8A6HBIuvrk1sb2TjbI0UDVowaNDsPa7nNdZHkQa6CQKf2BwFRy7gB8vAD1YSq23NTwplGe5IpuRe2Cbljh5diQ/+0zwZPZQEvjSMqZJILYVluGBw656JSndRdXHCs608CBib14l35A6D7rSZNd53nM9yXj/WOpLuY/jX0sbKJrBUIhcQcMiCFK/Nx06vhhCGZSWwaj0TTSMEOj9nyU+56MMx+N7527JUnOYMyVbbEx5h1S5sPjntMVVzz8c8FC5MKbgXCgjTt5Depl6YgrvoSCNNXZqnXN2l9dtBxKGSEdQIDAQABAoIBAD7r606/pZqPP0l3MnqVNyvY2X43r/ITBs+UQmSB67Ydqqiyi5C8xW530x6JSGJpP055c3atynTD4Jf4rF46WNgL91dTXBQ1QMubYV9+G6+lhUiOtQyBUOnkXRWQ/3MBEed5IQ0QjSew6WDheZuD7zOfJhD9sELKll7vSO2rVRdi5Kf3pps0K4aeYLvlelHjPtu2EkMyAD4d37QO2q8A/+acOf7wyx1uZ55nobe0jVgQBa2Tuuwhd2JilmsgMvcVxeAJoocyQxPVTU+iMidgZKxyW/fJUaR7V2KgKfP0H2OZbSruD2HyHL7o70+/XBg250msh691oTCXSGWiITFXRDkCgYEA1KF3daSzWK2vTZeNMxj++VRNZfyigNqBwgRZxrSGIUeNGpDdDp7flldM/fO/0RijKnh+nXLBCMZlxffSI7W3jkkjosEq0uyXv6Ez4WlhwHEAE6KrZYQ8VDU3We1LXERu/3SMU16Uq2Vme8J6kpshr7OLGeuTNfsSG79lpHU/OO8CgYEAtPjF/AY/t85u4JV0w6FJ/K7Iya+AyIyZhsgnxsPv5NwVtNsQzTRbXlpgnVmaKUXzLBTBJA2P6Gie35R6kJuOuMnvHjtG7umI5lOUJaqYk5GZEXHxbIKTXUxAbv+WKRHcFwcH54Ppurs7bN6ultWdE+yyU5PvabFRAidL+rV+MNsCgYBkqx2XwSZ4MaLxpXLgYlE1UkhipL17K/iLFuw6O3XVg762eSdvqKBK2YJGKYUDRhraHLo1aQMjWGsO00nWRk5lOxN4FuRjgEhIHdYmPnJdXiKfKUDkizrU020lUkd/o5a5s0BMqFbS3usqZOULTA8i1GpjFzlXMUtdf7xrpGcOWQKBgFJXWuX4Befg8owXFKCqc1qYQF2cEBGv/27XBVon7rAH/4xhnHxx1RzSLGGL0lvzKtikvNhln1PUvmzmRvoi/UZRAw6TVAwCkDqUTrTwsSl7ttS8Lmf75Ycu2aZnfnBDYwvLNCp/18oRxNEdZPzWRvk42k6y4d0KQ9yh+q0hBUC/AoGASgJro5HgKWkSzYx1Fk3rvo7EWeZwXNfWuzIhtCiVDG4+df0ZdybkbjxbaRwL2rbTSBeqSaZw4UwpE1JyF8erfzx0qpHCY5V3ocnA8kg56mX9kE5H9WKjMe1tbuZpba4i08zvM8vpz9gASf3z/WrWWqgpvynxDWpy4xFAqrWC0VQ=")
|
||||
var publicKeyPEM = FormatPEMPublicKey("MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEA3ErEoBiGNPT3iIt9ZIuf8znScpOpVk8cToo810EGP8AE/Z6M2s7r18kHCLPCBHvxOjcOTJNWwD2bbDnfY5HNHCXSTWwQUDtFpPAsm4orEaB4Qsh6ohTXTcyNhWp9cCIg42i7jVi5Eq7TVzKuVezCXk3XeLltT56wuCVRfe35SNPe8RDakvbE+fl46+yorz0AKXBoqvsgR8q9QKc6HDSUcFEDZkuFCt7PnMek6L0dQdmb7NHQdG2+8RnUAR4wMEVKuGAX5nCp4IeDE4L23658KPijK1Hg6u6dAqKTfCarj4Act+WOjYYCaARcU/rNX8RGthdvLhUc4lT0ALnbmOEj2QIDAQAB")
|
||||
|
||||
func Test_sign_verify(t *testing.T) {
|
||||
data := "123456"
|
||||
t.Run("test_sign", func(t *testing.T) {
|
||||
got, err := Sign(data, privateKeyPEM)
|
||||
if err != nil {
|
||||
t.Errorf("sign() error = %v", err)
|
||||
return
|
||||
}
|
||||
t.Log(got)
|
||||
b := Verify(data, got, publicKeyPEM)
|
||||
t.Log(b)
|
||||
})
|
||||
}
|
|
@ -0,0 +1,9 @@
|
|||
package vo
|
||||
|
||||
type Code string
|
||||
|
||||
const CodeSuccess Code = "0000000000"
|
||||
|
||||
func (c Code) IsSuccess() bool {
|
||||
return c == CodeSuccess
|
||||
}
|
|
@ -0,0 +1,43 @@
|
|||
package vo
|
||||
|
||||
import (
|
||||
"codeup.aliyun.com/6552e56cc3b2728a4557fc18/plugin/proto"
|
||||
)
|
||||
|
||||
type OrderStatus string
|
||||
|
||||
const (
|
||||
OrderSuccess OrderStatus = "01" // 充值成功
|
||||
OrderPending OrderStatus = "02" // 充值处理中
|
||||
OrderFail OrderStatus = "03" // 充值失败
|
||||
OrderAbnormal OrderStatus = "04" // 充值异常,处理中
|
||||
)
|
||||
|
||||
var orderStatusMap = map[OrderStatus]proto.Status{
|
||||
OrderSuccess: proto.Status_SUCCESS,
|
||||
OrderPending: proto.Status_ING,
|
||||
OrderFail: proto.Status_FAIL,
|
||||
}
|
||||
|
||||
func (o OrderStatus) IsSuccess() bool {
|
||||
return o == OrderSuccess
|
||||
}
|
||||
|
||||
func (o OrderStatus) IsPending() bool {
|
||||
return o == OrderPending
|
||||
}
|
||||
|
||||
func (o OrderStatus) IsFail() bool {
|
||||
return o == OrderFail
|
||||
}
|
||||
|
||||
func (o OrderStatus) IsAbnormal() bool {
|
||||
return o == OrderAbnormal
|
||||
}
|
||||
|
||||
func (o OrderStatus) GetOrderStatus() proto.Status {
|
||||
if resultStatus, ok := orderStatusMap[o]; ok {
|
||||
return resultStatus
|
||||
}
|
||||
return proto.Status_ING
|
||||
}
|
|
@ -0,0 +1,109 @@
|
|||
package vo
|
||||
|
||||
type SubCode string
|
||||
|
||||
const (
|
||||
SubCodeUp02094 SubCode = "UP02094"
|
||||
SubCodeUp02099 SubCode = "UP02099"
|
||||
SubCodeUp06003 SubCode = "UP06003"
|
||||
SubCodeUp06004 SubCode = "UP06004"
|
||||
SubCodeUp06005 SubCode = "UP06005"
|
||||
SubCodeUp06006 SubCode = "UP06006"
|
||||
SubCodeUp06007 SubCode = "UP06007"
|
||||
SubCodeUp06008 SubCode = "UP06008"
|
||||
SubCodeUp06009 SubCode = "UP06009"
|
||||
SubCodeUp06010 SubCode = "UP06010"
|
||||
SubCodeUp06011 SubCode = "UP06011"
|
||||
SubCodeUp06012 SubCode = "UP06012"
|
||||
SubCodeUp06021 SubCode = "UP06021"
|
||||
SubCodeUp06022 SubCode = "UP06022"
|
||||
SubCodeUp06034 SubCode = "UP06034"
|
||||
SubCodeUp06035 SubCode = "UP06035"
|
||||
SubCodeUp06036 SubCode = "UP06036"
|
||||
SubCodeUp06037 SubCode = "UP06037"
|
||||
SubCodeUp06038 SubCode = "UP06038"
|
||||
SubCodeUp06039 SubCode = "UP06039"
|
||||
SubCodeUp06040 SubCode = "UP06040"
|
||||
SubCodeUp06041 SubCode = "UP06041"
|
||||
SubCodeUp06042 SubCode = "UP06042"
|
||||
SubCodeUp06043 SubCode = "UP06043"
|
||||
SubCodeUp06044 SubCode = "UP06044"
|
||||
SubCodeUp06045 SubCode = "UP06045"
|
||||
SubCodeUp07001 SubCode = "UP07001"
|
||||
SubCodeUp07002 SubCode = "UP07002"
|
||||
SubCodeUp07003 SubCode = "UP07003"
|
||||
SubCodeUp07004 SubCode = "UP07004"
|
||||
SubCodeUp07005 SubCode = "UP07005"
|
||||
SubCodeUp07010 SubCode = "UP07010"
|
||||
SubCodeUp07012 SubCode = "UP07012"
|
||||
SubCodeUp07013 SubCode = "UP07013"
|
||||
SubCodeUp07014 SubCode = "UP07014"
|
||||
SubCodeUp07015 SubCode = "UP07015"
|
||||
SubCodeUp07020 SubCode = "UP07020"
|
||||
SubCodeUp07030 SubCode = "UP07030"
|
||||
SubCodeUp07050 SubCode = "UP07050"
|
||||
SubCodeUp07051 SubCode = "UP07051"
|
||||
SubCodeUp07052 SubCode = "UP07052"
|
||||
SubCodeUp00030 SubCode = "UP00030"
|
||||
SubCodeUp00050 SubCode = "UP00050"
|
||||
SubCodeUp00080 SubCode = "UP00080"
|
||||
SubCodeUp02025 SubCode = "UP02025"
|
||||
SubCodeUp02071 SubCode = "UP02071"
|
||||
)
|
||||
|
||||
var codeMsg = map[SubCode]string{
|
||||
SubCodeUp02094: "交易重复",
|
||||
SubCodeUp02099: "交易单品信息无效",
|
||||
SubCodeUp06003: "线上折扣活动不存在",
|
||||
SubCodeUp06004: "线上折扣活动已下线",
|
||||
SubCodeUp06005: "线上折扣活动未开始",
|
||||
SubCodeUp06006: "线上折扣活动已结束",
|
||||
SubCodeUp06007: "线上折扣活动已过期(当前消费请求已不满足)",
|
||||
SubCodeUp06008: "线上折扣活动多方出资配置或者计算错误",
|
||||
SubCodeUp06009: "线上折扣活动查询结果不满足消费请求(适用于查询并承兑)",
|
||||
SubCodeUp06010: "折扣活动承兑金额检验错误",
|
||||
SubCodeUp06011: "线上折扣活动余额不足",
|
||||
SubCodeUp06012: "线上折扣活动当前周期余额不足",
|
||||
SubCodeUp06021: "线上折扣活动城市校验失败,定位城市非活动城市",
|
||||
SubCodeUp06022: "线上折扣活动城市校验失败,交易未上送定位城市",
|
||||
SubCodeUp06034: "线上折扣活动参与卡付失败封顶",
|
||||
SubCodeUp06035: "线上折扣活动参与手机封顶",
|
||||
SubCodeUp06036: "线上折扣活动参与银行卡封顶",
|
||||
SubCodeUp06037: "线上折扣活动参与迷你付终端封顶",
|
||||
SubCodeUp06038: "线上折扣活动参与用户封顶",
|
||||
SubCodeUp06039: "线上折扣活动参与账单号封顶",
|
||||
SubCodeUp06040: "线上折扣活动参与用户商品数量封顶",
|
||||
SubCodeUp06041: "线上折扣活动参与证件号封顶",
|
||||
SubCodeUp06042: "线上折扣活动参与商户号封顶",
|
||||
SubCodeUp06043: "线上折扣活动参与人脸证件号封顶",
|
||||
SubCodeUp06044: "线上折扣活动参与商户Pos终端封顶",
|
||||
SubCodeUp06045: "线上折扣活动参与活动封顶",
|
||||
SubCodeUp07001: "优惠券下载失败",
|
||||
SubCodeUp07002: "优惠券冻结失败",
|
||||
SubCodeUp07003: "优惠券删除失败",
|
||||
SubCodeUp07004: "优惠券操作结果查询失败",
|
||||
SubCodeUp07005: "优惠券开放平台权限校验失败",
|
||||
SubCodeUp07010: "用户优惠券张数超过限制",
|
||||
SubCodeUp07012: "优惠券下载封顶累计失败",
|
||||
SubCodeUp07013: "优惠券下载调用账务失败",
|
||||
SubCodeUp07014: "优惠券下载卡号反查用户ID失败",
|
||||
SubCodeUp07015: "优惠券下载手机号反查用户ID失败",
|
||||
SubCodeUp07020: "优惠券冻结调用账务失败",
|
||||
SubCodeUp07030: "优惠券删除调用账户失败",
|
||||
SubCodeUp07050: "优惠券下载维度无效",
|
||||
SubCodeUp07051: "用户送券维度:当前用户未实名认证",
|
||||
SubCodeUp07052: "用户送券维度:可用银行卡与当前账户不同名,需绑定同名卡",
|
||||
SubCodeUp00030: "请求报文错误(无效)",
|
||||
SubCodeUp00050: "线上折扣系统并发控制限制",
|
||||
SubCodeUp00080: "线上折扣系统内部错误",
|
||||
SubCodeUp02025: "原交易不存在",
|
||||
SubCodeUp02071: "交易无效",
|
||||
}
|
||||
|
||||
func (code SubCode) GetMsg() string {
|
||||
msg, ok := codeMsg[code]
|
||||
if !ok {
|
||||
return "未知错误代码"
|
||||
}
|
||||
return msg
|
||||
}
|
|
@ -0,0 +1,15 @@
|
|||
package main
|
||||
|
||||
import (
|
||||
"codeup.aliyun.com/6552e56cc3b2728a4557fc18/plugin/shared"
|
||||
"github.com/hashicorp/go-plugin"
|
||||
"plugins/union_pay/internal"
|
||||
)
|
||||
|
||||
func main() {
|
||||
plugin.Serve(&plugin.ServeConfig{
|
||||
HandshakeConfig: shared.HandshakeConfig(internal.Version, internal.CookieKey, internal.CookieValue),
|
||||
Plugins: shared.PluginSet(shared.NewPlugin(&internal.UnionPayService{}, internal.Tag)),
|
||||
GRPCServer: plugin.DefaultGRPCServer,
|
||||
})
|
||||
}
|
|
@ -23,8 +23,7 @@ const (
|
|||
queryMethod = "/recharge/query"
|
||||
)
|
||||
|
||||
type ZltxService struct {
|
||||
}
|
||||
type ZltxService struct{}
|
||||
|
||||
func (p *ZltxService) Order(ctx context.Context, request *proto.OrderRequest) (*proto.OrderResponse, error) {
|
||||
uv, err := req(rechargeReq(request), request.Config.AppKey)
|
||||
|
|
Loading…
Reference in New Issue