实现统一退货相关API功能
- 新增合单退货、退货、退货查询等API接口 - 使用gtime替代time包处理时间格式 - 新增客户端签名生成和HTTP请求发送逻辑 - 定义统一退货相关请求和响应数据结构 - 实现拉卡拉SDK统一退货服务实例化方法menu
parent
7974b4d570
commit
290d3e6ba7
|
|
@ -46,3 +46,15 @@ const (
|
||||||
// LKL_RECONF_SUBMIT 拉卡拉商户进件复议提交
|
// LKL_RECONF_SUBMIT 拉卡拉商户进件复议提交
|
||||||
LKL_RECONF_SUBMIT = "/v2/mms/openApi/reconsiderSubmit"
|
LKL_RECONF_SUBMIT = "/v2/mms/openApi/reconsiderSubmit"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// unifiedReturn 统一退货API地址
|
||||||
|
const (
|
||||||
|
// LKL_UNIFIED_RETURN_MERGE_REFUND_URL 拉卡拉合单退货
|
||||||
|
LKL_UNIFIED_RETURN_MERGE_REFUND_URL = "/v3/rfd/refund_front/merge_refund"
|
||||||
|
// LKL_UNIFIED_RETURN_REFUND_URL 拉卡拉退货
|
||||||
|
LKL_UNIFIED_RETURN_REFUND_URL = "/v3/rfd/refund_front/refund"
|
||||||
|
// LKL_UNIFIED_RETURN_REFUND_QUERY_URL 拉卡拉退货查询
|
||||||
|
LKL_UNIFIED_RETURN_REFUND_QUERY_URL = "/v3/rfd/refund_front/refund_query"
|
||||||
|
// LKL_UNIFIED_RETURN_REFUND_FEE_URL 拉卡拉退货手续费查询
|
||||||
|
LKL_UNIFIED_RETURN_REFUND_FEE_URL = "/v3/rfd/refund_front/refund_fee"
|
||||||
|
)
|
||||||
|
|
|
||||||
|
|
@ -1,10 +1,9 @@
|
||||||
package lklsdk
|
package lklsdk
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"time"
|
|
||||||
|
|
||||||
"github.com/black1552/lkl_sdk/consts"
|
"github.com/black1552/lkl_sdk/consts"
|
||||||
"github.com/black1552/lkl_sdk/model"
|
"github.com/black1552/lkl_sdk/model"
|
||||||
|
"github.com/gogf/gf/v2/os/gtime"
|
||||||
)
|
)
|
||||||
|
|
||||||
// AccountService 账户服务
|
// AccountService 账户服务
|
||||||
|
|
@ -26,7 +25,7 @@ func (a *AccountService[T]) BalanceQuery(req *model.BalanceQueryReqData) (*T, er
|
||||||
|
|
||||||
// 构建BaseModel请求
|
// 构建BaseModel请求
|
||||||
baseReq := model.BalanceQueryRequest{
|
baseReq := model.BalanceQueryRequest{
|
||||||
ReqTime: time.Now().Format("20060102150405"),
|
ReqTime: gtime.Now().Format("YmdHis"),
|
||||||
Version: "3.0",
|
Version: "3.0",
|
||||||
ReqData: req,
|
ReqData: req,
|
||||||
}
|
}
|
||||||
|
|
@ -46,7 +45,7 @@ func (a *AccountService[T]) Withdraw(req *model.WithdrawReqData) (*T, error) {
|
||||||
|
|
||||||
// 构建BaseModel请求
|
// 构建BaseModel请求
|
||||||
baseReq := model.WithdrawRequest{
|
baseReq := model.WithdrawRequest{
|
||||||
ReqTime: time.Now().Format("20060102150405"),
|
ReqTime: gtime.Now().Format("YmdHis"),
|
||||||
Version: "3.0",
|
Version: "3.0",
|
||||||
ReqData: req,
|
ReqData: req,
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,112 @@
|
||||||
|
package common
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"crypto"
|
||||||
|
"crypto/rand"
|
||||||
|
"crypto/rsa"
|
||||||
|
"crypto/sha256"
|
||||||
|
"crypto/x509"
|
||||||
|
"encoding/base64"
|
||||||
|
"encoding/json"
|
||||||
|
"encoding/pem"
|
||||||
|
"fmt"
|
||||||
|
ran "math/rand"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/black1552/base-common/utils"
|
||||||
|
"github.com/gogf/gf/v2/os/gfile"
|
||||||
|
"github.com/gogf/gf/v2/util/gconv"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Client 拉卡拉SDK客户端
|
||||||
|
type Client[T any] struct {
|
||||||
|
config *Config
|
||||||
|
response T
|
||||||
|
ctx context.Context
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewClient 创建拉卡拉SDK客户端
|
||||||
|
func NewClient[T any](ctx context.Context, cfgJson string) *Client[T] {
|
||||||
|
var config *Config
|
||||||
|
err := gconv.Struct(cfgJson, &config)
|
||||||
|
if err != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return &Client[T]{
|
||||||
|
config: config,
|
||||||
|
ctx: ctx,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
func (c *Client[T]) generateNonceStr() string {
|
||||||
|
const letterBytes = "abcdefghijklmnopqrstuvwxyz0123456789"
|
||||||
|
nonce := make([]byte, 12)
|
||||||
|
for i := range nonce {
|
||||||
|
nonce[i] = letterBytes[ran.Intn(len(letterBytes))]
|
||||||
|
}
|
||||||
|
return string(nonce)
|
||||||
|
}
|
||||||
|
|
||||||
|
// generateSign 生成签名
|
||||||
|
func (c *Client[T]) generateSign(request []byte) (string, error) {
|
||||||
|
// 生成随机字符串
|
||||||
|
nonceStr := c.generateNonceStr()
|
||||||
|
// 获取当前时间戳(秒)
|
||||||
|
timestamp := fmt.Sprintf("%d", time.Now().Unix())
|
||||||
|
|
||||||
|
// 构建待签名报文
|
||||||
|
signData := fmt.Sprintf("%s\n%s\n%s\n%s\n%s\n", c.config.AppId, c.config.SerialNo, timestamp, nonceStr, string(request))
|
||||||
|
|
||||||
|
// 计算签名
|
||||||
|
hashed := sha256.Sum256([]byte(signData))
|
||||||
|
|
||||||
|
privateKey, err := c.loadPrivateKey()
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
signature, err := rsa.SignPKCS1v15(rand.Reader, privateKey, crypto.SHA256, hashed[:])
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
signatureBase64 := base64.StdEncoding.EncodeToString(signature)
|
||||||
|
|
||||||
|
// 构建Authorization头
|
||||||
|
authorization := fmt.Sprintf("LKLAPI-SHA256withRSA appid=\"%s\",serial_no=\"%s\",timestamp=\"%s\",nonce_str=\"%s\",signature=\"%s\"",
|
||||||
|
c.config.AppId, c.config.SerialNo, timestamp, nonceStr, signatureBase64)
|
||||||
|
return authorization, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Client[T]) loadPrivateKey() (*rsa.PrivateKey, error) {
|
||||||
|
block, _ := pem.Decode(gfile.GetBytes(c.config.PrivateKey))
|
||||||
|
if block == nil {
|
||||||
|
return nil, fmt.Errorf("无法解码私钥PEM数据")
|
||||||
|
}
|
||||||
|
// 解析PKCS#8格式私钥
|
||||||
|
privateKey, err := x509.ParsePKCS8PrivateKey(block.Bytes)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return privateKey.(*rsa.PrivateKey), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// DoRequest 发送HTTP请求
|
||||||
|
func (c *Client[T]) DoRequest(url string, reqData interface{}) (*T, error) {
|
||||||
|
// 序列化为JSON
|
||||||
|
jsonData, err := json.Marshal(reqData)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("序列化请求数据失败: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
auth, err := c.generateSign(jsonData)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("生成签名失败: %v", err)
|
||||||
|
}
|
||||||
|
header := map[string]string{
|
||||||
|
"Authorization": auth,
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
"Accept": "application/json",
|
||||||
|
}
|
||||||
|
// 设置其他必要的请求头
|
||||||
|
|
||||||
|
return utils.NewClient[T](jsonData, url, header).Post(c.ctx)
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,14 @@
|
||||||
|
package common
|
||||||
|
|
||||||
|
// Config 拉卡拉SDK配置
|
||||||
|
type Config struct {
|
||||||
|
PublicKey string `json:"public_key" dc:"公钥字符串"`
|
||||||
|
PrivateKey string `json:"private_key" dc:"私钥字符串"`
|
||||||
|
AppId string `json:"app_id" dc:"lakala应用ID"`
|
||||||
|
SerialNo string `json:"serial_no" dc:"序列号"`
|
||||||
|
SubAppId string `json:"sub_app_id" dc:"子应用ID 微信AppId"`
|
||||||
|
Version string `json:"version" dc:"lakala版本号"`
|
||||||
|
AccountType string `json:"account_type" dc:"账户类型"`
|
||||||
|
TransType string `json:"trans_type" dc:"交易类型"`
|
||||||
|
NotifyUrl string `json:"notify_url" dc:"回调地址"`
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,34 @@
|
||||||
|
package mergerefund
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/black1552/lkl_sdk/consts"
|
||||||
|
"github.com/black1552/lkl_sdk/lklsdk/common"
|
||||||
|
"github.com/gogf/gf/v2/os/gtime"
|
||||||
|
)
|
||||||
|
|
||||||
|
type MergeRefund[T any] struct {
|
||||||
|
client *common.Client[T]
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewMergeRefund[T any](client *common.Client[T]) *MergeRefund[T] {
|
||||||
|
return &MergeRefund[T]{
|
||||||
|
client: client,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
func (t *MergeRefund[T]) MergeRefund(req *RequestDataMergeRefund) (*T, error) {
|
||||||
|
// 构建请求参数
|
||||||
|
url := consts.BASE_URL + consts.LKL_UNIFIED_RETURN_MERGE_REFUND_URL
|
||||||
|
// 构建BaseModel请求
|
||||||
|
baseReq := RequestMergeRefund{
|
||||||
|
ReqTime: gtime.Now().Format("YmdHis"),
|
||||||
|
Version: "3.0",
|
||||||
|
ReqData: req,
|
||||||
|
}
|
||||||
|
|
||||||
|
// 发送请求
|
||||||
|
respBody, err := t.client.DoRequest(url, baseReq)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return respBody, nil
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,42 @@
|
||||||
|
package mergerefund
|
||||||
|
|
||||||
|
// RequestMergeRefund 统一退货基础请求结构体
|
||||||
|
type RequestMergeRefund struct {
|
||||||
|
ReqTime string `json:"req_time" dc:"ReqTime 请求时间 必填,格式yyyyMMddHHmmss"`
|
||||||
|
Version string `json:"version" dc:"Version 版本号 必填,固定为\"3\""`
|
||||||
|
ReqData *RequestDataMergeRefund `json:"req_data" dc:"ReqData 请求参数 必填,具体字段见对应的数据结构体"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// RequestDataMergeRefund 合单退货请求参数结构体
|
||||||
|
type RequestDataMergeRefund struct {
|
||||||
|
MerchantNo string `json:"merchant_no" dc:"MerchantNo 商户号 必填,拉卡拉分配的商户号"`
|
||||||
|
TermNo string `json:"term_no" dc:"TermNo 终端号 必填,拉卡拉分配的终端号"`
|
||||||
|
OutTradeNo string `json:"out_trade_no" dc:"OutTradeNo 商户请求流水号 必填,商户系统唯一"`
|
||||||
|
RefundAmount string `json:"refund_amount" dc:"RefundAmount 退款金额 必填,单位分,整数数字型字符"`
|
||||||
|
RefundReason string `json:"refund_reason,omitempty" dc:"RefundReason 退货原因 可选"`
|
||||||
|
OriginLogNo string `json:"origin_log_no,omitempty" dc:"OriginLogNo 拉卡拉对账单流水号 可选,正交易返回的拉卡拉对账单流水号"`
|
||||||
|
OriginOutTradeNo string `json:"origin_out_trade_no,omitempty" dc:"OriginOutTradeNo 原商户交易流水号 可选"`
|
||||||
|
OriginTradeNo string `json:"origin_trade_no,omitempty" dc:"OriginTradeNo 原交易拉卡拉交易订单号 可选"`
|
||||||
|
LocationInfo *LocationInfo `json:"location_info" dc:"LocationInfo 地址位置信息 必填"`
|
||||||
|
NotifyURL string `json:"notify_url,omitempty" dc:"NotifyURL 后台通知地址 可选,交易结果通知地址"`
|
||||||
|
RefundAccMode string `json:"refund_acc_mode,omitempty" dc:"RefundAccMode 退货账户模式 可选,00退货账户余额 05商户余额 06终端余额"`
|
||||||
|
RefundSplitInfo []*RelateOutSplitInfo `json:"refund_split_info,omitempty" dc:"RefundSplitInfo 请参合单域 可选"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// LocationInfo 地址位置信息结构体
|
||||||
|
type LocationInfo struct {
|
||||||
|
RequestIP string `json:"request_ip" dc:"RequestIP 请求方IP地址 必填,格式如36.45.36.95"`
|
||||||
|
Location string `json:"location,omitempty" dc:"Location 维度,经度 可选,商户终端的地理位置,格式:纬度,经度,+表示北纬、东经,-表示南纬、西经,精度最长支持小数点后9位"`
|
||||||
|
BaseStation string `json:"base_station,omitempty" dc:"BaseStation 基站信息 可选,客户端设备的基站信息"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// RelateOutSplitInfo 请求合单域结构体
|
||||||
|
type RelateOutSplitInfo struct {
|
||||||
|
OutSubTradeNo string `json:"out_sub_trade_no" dc:"OutSubTradeNo 外部子退款交易流水号 必填,商户子交易退款流水号,商户号下唯一"`
|
||||||
|
MerchantNo string `json:"merchant_no" dc:"MerchantNo 商户号 必填,拉卡拉分配的商户号"`
|
||||||
|
TermNo string `json:"term_no" dc:"TermNo 终端号 必填,拉卡拉分配的业务终端号"`
|
||||||
|
RefundAmount string `json:"refund_amount" dc:"RefundAmount 申请退款金额 必填,单位分,整数型字符"`
|
||||||
|
OriginOutSubTradeNo string `json:"origin_out_sub_trade_no,omitempty" dc:"OriginOutSubTradeNo 原商户交易流水号 可选,下单时的商户子单请求流水号"`
|
||||||
|
OriginSubTradeNo string `json:"origin_sub_trade_no,omitempty" dc:"OriginSubTradeNo 原拉卡拉子交易流水号 可选,下单成功时,返回的拉卡拉交易子流水"`
|
||||||
|
OriginSubLogNo string `json:"origin_sub_log_no,omitempty" dc:"OriginSubLogNo 原对账单子流水号 可选,原交易的tradeNo的后14位,必须是66开头的"`
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,53 @@
|
||||||
|
package mergerefund
|
||||||
|
|
||||||
|
// ResponseMergeRefund 统一退货响应结构体
|
||||||
|
type ResponseMergeRefund struct {
|
||||||
|
Code string `json:"code" dc:"Code 响应码"`
|
||||||
|
Msg string `json:"msg" dc:"Msg 响应描述"`
|
||||||
|
RespTime string `json:"resp_time" dc:"RespTime 响应时间"`
|
||||||
|
RespData *ResponseDataMergeRefund `json:"resp_data" dc:"RespData 响应数据"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *ResponseMergeRefund) SuccessOrFail() bool {
|
||||||
|
return r.Code == "000000"
|
||||||
|
}
|
||||||
|
|
||||||
|
// ResponseDataMergeRefund 合单退货响应数据结构体
|
||||||
|
type ResponseDataMergeRefund struct {
|
||||||
|
TradeState string `json:"trade_state" dc:"TradeState 交易状态 必填,INIT-初始化;SUCCESS-交易成功;FAIL-交易失败;DEAL-交易处理中/未知;PROCESSING-交易已受理;TIMEOUT-超时未知;EXCEPTION-异常"`
|
||||||
|
RefundType string `json:"refund_type" dc:"RefundType 退货模式 必填"`
|
||||||
|
MerchantNo string `json:"merchant_no" dc:"MerchantNo 商户号 必填,拉卡拉分配的商户号"`
|
||||||
|
OutTradeNo string `json:"out_trade_no" dc:"OutTradeNo 商户请求流水号 必填,请求中的商户请求流水号"`
|
||||||
|
TradeNo string `json:"trade_no" dc:"TradeNo 拉卡拉交易流水号 必填"`
|
||||||
|
LogNo string `json:"log_no" dc:"LogNo 拉卡拉对账单流水号 必填,tradeNo的后14位"`
|
||||||
|
AccTradeNo string `json:"acc_trade_no,omitempty" dc:"AccTradeNo 账户端交易订单号 可选"`
|
||||||
|
AccountType string `json:"account_type,omitempty" dc:"AccountType 钱包类型 可选,微信:WECHAT;支付宝:ALIPAY;银联:UQRCODEPAY;翼支付:BESTPAY;苏宁易付宝:SUNING"`
|
||||||
|
TotalAmount string `json:"total_amount" dc:"TotalAmount 交易金额 必填,单位分,整数数字型字符串"`
|
||||||
|
RefundAmount string `json:"refund_amount" dc:"RefundAmount 申请退款金额 必填,单位分,整数数字型字符串"`
|
||||||
|
PayerAmount string `json:"payer_amount" dc:"PayerAmount 实际退款金额 必填,单位分,整数数字型字符串"`
|
||||||
|
TradeTime string `json:"trade_time,omitempty" dc:"TradeTime 退款时间 可选,实际退款时间。yyyyMMddHHmmss"`
|
||||||
|
OriginLogNo string `json:"origin_log_no,omitempty" dc:"OriginLogNo 原拉卡拉对账单流水号 可选,如果请求中携带,则返回"`
|
||||||
|
OriginTradeNo string `json:"origin_trade_no,omitempty" dc:"OriginTradeNo 原拉卡拉交易流水号 可选,如果请求中携带,则返回"`
|
||||||
|
OriginOutTradeNo string `json:"origin_out_trade_no,omitempty" dc:"OriginOutTradeNo 原商户请求流水号 可选,如果请求中携带,则返回"`
|
||||||
|
UpIssAddnData string `json:"up_iss_addn_data,omitempty" dc:"UpIssAddnData 单品营销附加数据 可选,扫码交易,参与单品营销优惠时返回"`
|
||||||
|
UpCouponInfo string `json:"up_coupon_info,omitempty" dc:"UpCouponInfo 银联优惠信息 可选,扫码交易,参与单品营销优惠时返回"`
|
||||||
|
TradeInfo string `json:"trade_info,omitempty" dc:"TradeInfo 出资方信息 可选,扫码交易"`
|
||||||
|
ChannelRetDesc string `json:"channel_ret_desc" dc:"ChannelRetDesc 返回描述信息 必填"`
|
||||||
|
RefundSplitInfo []*RelateOutSplitRspInfo `json:"refund_split_info" dc:"RefundSplitInfo 响应合单域 必填"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// RelateOutSplitRspInfo 响应合单域结构体
|
||||||
|
type RelateOutSplitRspInfo struct {
|
||||||
|
OutSubTradeNo string `json:"out_sub_trade_no" dc:"OutSubTradeNo 外部子退款交易流水号 必填,请求中的商户子流水号"`
|
||||||
|
MerchantNo string `json:"merchant_no" dc:"MerchantNo 商户号 必填,拉卡拉分配的商户号"`
|
||||||
|
TermNo string `json:"term_no" dc:"TermNo 终端号 必填,拉卡拉分配的业务终端号"`
|
||||||
|
TradeState string `json:"trade_state" dc:"TradeState 交易状态 必填,INIT-初始化;SUCCESS-交易成功;FAIL-交易失败;DEAL-交易处理中/未知;PROCESSING-交易已受理;TIMEOUT-超时未知;EXCEPTION-异常"`
|
||||||
|
SubTradeNo string `json:"sub_trade_no" dc:"SubTradeNo 拉卡拉子交易流水号 必填"`
|
||||||
|
SubLogNo string `json:"sub_log_no" dc:"SubLogNo 拉卡拉子对账单流水号 必填"`
|
||||||
|
RefundAmount string `json:"refund_amount" dc:"RefundAmount 申请退款金额 必填,单位分,整数型字符"`
|
||||||
|
PayerAmount string `json:"payer_amount" dc:"PayerAmount 实际退款金额 必填,单位分,整数型字符"`
|
||||||
|
TotalAmount string `json:"total_amount" dc:"TotalAmount 交易金额 必填,单位分,整数型字符"`
|
||||||
|
OriginSubLogNo string `json:"origin_sub_log_no,omitempty" dc:"OriginSubLogNo 原拉卡拉子对账单流水号 可选,如果请求中携带,则返回"`
|
||||||
|
OriginSubTradeNo string `json:"origin_sub_trade_no,omitempty" dc:"OriginSubTradeNo 原拉卡拉子交易流水号 可选,如果请求中携带,则返回"`
|
||||||
|
OriginOutSubTradeNo string `json:"origin_out_sub_trade_no,omitempty" dc:"OriginOutSubTradeNo 原商户子交易流水号 可选,如果请求中携带,则返回"`
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,40 @@
|
||||||
|
package refund
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/black1552/lkl_sdk/consts"
|
||||||
|
"github.com/black1552/lkl_sdk/lklsdk/common"
|
||||||
|
"github.com/gogf/gf/v2/os/gtime"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Refund 统一退货API结构体
|
||||||
|
type Refund[T any] struct {
|
||||||
|
client *common.Client[T]
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewRefund 创建统一退货API实例
|
||||||
|
func NewRefund[T any](client *common.Client[T]) *Refund[T] {
|
||||||
|
return &Refund[T]{
|
||||||
|
client: client,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Refund 发起统一退货请求
|
||||||
|
// request: 统一退货请求参数
|
||||||
|
// 返回统一退货响应结果和错误信息
|
||||||
|
func (api *Refund[T]) Refund(req *RequestDataRefund) (*T, error) {
|
||||||
|
// 构建请求参数
|
||||||
|
url := consts.BASE_URL + consts.LKL_UNIFIED_RETURN_REFUND_URL
|
||||||
|
// 构建BaseModel请求
|
||||||
|
baseReq := RequestRefund{
|
||||||
|
ReqTime: gtime.Now().Format("YmdHis"),
|
||||||
|
Version: "3.0",
|
||||||
|
ReqData: req,
|
||||||
|
}
|
||||||
|
|
||||||
|
// 发送请求
|
||||||
|
respBody, err := api.client.DoRequest(url, baseReq)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return respBody, nil
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,37 @@
|
||||||
|
package refund
|
||||||
|
|
||||||
|
// ResponseRefund 统一退货响应结构体
|
||||||
|
// 参考文档:https://o.lakala.com/#/home/document/detail?id=892
|
||||||
|
type ResponseRefund struct {
|
||||||
|
Code string `json:"code" dc:"Code 返回业务代码 必填,通讯成功返回码:000000;交易未知:RFD11105/RFD11112;其他返回码可视为失败"`
|
||||||
|
Msg string `json:"msg" dc:"Msg 返回业务代码描述 必填,如\"成功\""`
|
||||||
|
RespTime string `json:"resp_time" dc:"RespTime 响应时间 必填,格式yyyyMMddHHmmss"`
|
||||||
|
RespData *ResponseDataRefund `json:"resp_data" dc:"RespData 响应数据 必填,具体字段根据接口返回"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *ResponseRefund) SuccessOrFail() bool {
|
||||||
|
return r.Code == "000000"
|
||||||
|
}
|
||||||
|
|
||||||
|
// ResponseDataRefund 统一退货响应数据结构体
|
||||||
|
type ResponseDataRefund struct {
|
||||||
|
TradeState string `json:"trade_state" dc:"TradeState 交易状态 必填,INIT-初始化;SUCCESS-交易成功;FAIL-交易失败;DEAL-交易处理中/未知;PROCESSING-交易已受理;TIMEOUT-超时未知;EXCEPTION-异常"`
|
||||||
|
RefundType string `json:"refund_type" dc:"RefundType 退货模式 必填"`
|
||||||
|
MerchantNo string `json:"merchant_no" dc:"MerchantNo 商户号 必填,拉卡拉分配的商户号"`
|
||||||
|
OutTradeNo string `json:"out_trade_no" dc:"OutTradeNo 商户请求流水号 必填,请求中的商户请求流水号"`
|
||||||
|
TradeNo string `json:"trade_no" dc:"TradeNo 拉卡拉交易流水号 必填"`
|
||||||
|
LogNo string `json:"log_no" dc:"LogNo 拉卡拉对账单流水号 必填,tradeNo的后14位"`
|
||||||
|
AccTradeNo string `json:"acc_trade_no,omitempty" dc:"AccTradeNo 账户端交易订单号 可选"`
|
||||||
|
AccountType string `json:"account_type,omitempty" dc:"AccountType 钱包类型 可选,微信:WECHAT;支付宝:ALIPAY;银联:UQRCODEPAY;翼支付:BESTPAY;苏宁易付宝:SUNING"`
|
||||||
|
TotalAmount string `json:"total_amount" dc:"TotalAmount 交易金额 必填,单位分,整数数字型字符串"`
|
||||||
|
RefundAmount string `json:"refund_amount" dc:"RefundAmount 申请退款金额 必填,单位分,整数数字型字符串"`
|
||||||
|
PayerAmount string `json:"payer_amount" dc:"PayerAmount 实际退款金额 必填,单位分,整数数字型字符串"`
|
||||||
|
TradeTime string `json:"trade_time,omitempty" dc:"TradeTime 退款时间 可选,实际退款时间。yyyyMMddHHmmss"`
|
||||||
|
OriginLogNo string `json:"origin_log_no,omitempty" dc:"OriginLogNo 原拉卡拉对账单流水号 可选,如果请求中携带,则返回"`
|
||||||
|
OriginTradeNo string `json:"origin_trade_no,omitempty" dc:"OriginTradeNo 原拉卡拉交易流水号 可选,如果请求中携带,则返回"`
|
||||||
|
OriginOutTradeNo string `json:"origin_out_trade_no,omitempty" dc:"OriginOutTradeNo 原商户请求流水号 可选,如果请求中携带,则返回"`
|
||||||
|
UpIssAddData string `json:"up_iss_add_data,omitempty" dc:"UpIssAddData 单品营销附加数据 可选,扫码交易,参与单品营销优惠时返回"`
|
||||||
|
UpCouponInfo string `json:"up_coupon_info,omitempty" dc:"UpCouponInfo 银联优惠信息 可选,扫码交易,参与单品营销优惠时返回"`
|
||||||
|
TradeInfo string `json:"trade_info,omitempty" dc:"TradeInfo 出资方信息 可选,扫码交易"`
|
||||||
|
ChannelRetDesc string `json:"channel_ret_desc" dc:"ChannelRetDesc 返回描述信息 必填"`
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,34 @@
|
||||||
|
package refund
|
||||||
|
|
||||||
|
// RequestRefund 统一退货基础请求结构体
|
||||||
|
// 参考文档:https://o.lakala.com/#/home/document/detail?id=894
|
||||||
|
type RequestRefund struct {
|
||||||
|
ReqTime string `json:"req_time"`
|
||||||
|
Version string `json:"version"`
|
||||||
|
ReqData *RequestDataRefund `json:"req_data"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// RequestDataRefund 统一退货请求数据结构体
|
||||||
|
// 注意:origin_out_trade_no、origin_log_no、origin_trade_no至少一个必填(调用收银台下单接口拉起交易后发起退款时至少要传两个)
|
||||||
|
// 优先级顺序:origin_trade_no > origin_log_no > origin_out_trade_no
|
||||||
|
type RequestDataRefund struct {
|
||||||
|
MerchantNo string `json:"merchant_no"` // merchant_no: 商户号 (必填,拉卡拉分配的商户号,String(15))
|
||||||
|
TermNo string `json:"term_no"` // term_no: 终端号 (必填,拉卡拉分配的终端号,String(8))
|
||||||
|
OutTradeNo string `json:"out_trade_no"` // out_trade_no: 商户请求流水号 (必填,商户系统唯一,String(32))
|
||||||
|
RefundAmount string `json:"refund_amount"` // refund_amount: 退款金额 (必填,单位分,整数数字型字符,String(12))
|
||||||
|
RefundReason string `json:"refund_reason,omitempty"` // refund_reason: 退货原因 (可选,String(32))
|
||||||
|
OriginLogNo string `json:"origin_log_no,omitempty"` // origin_log_no: 拉卡拉对账单流水号 (可选,正交易返回的拉卡拉对账单流水号,String(14))
|
||||||
|
OriginOutTradeNo string `json:"origin_out_trade_no,omitempty"` // origin_out_trade_no: 原商户交易流水号 (可选,String(32))
|
||||||
|
OriginTradeNo string `json:"origin_trade_no,omitempty"` // origin_trade_no: 原交易拉卡拉交易订单号 (可选,String(32))
|
||||||
|
LocationInfo *LocationInfo `json:"location_info"` // location_info: 地址位置信息 (必填)
|
||||||
|
RefundAccMode string `json:"refund_acc_mode,omitempty"` // refund_acc_mode: 退货账户模式 (可选,00退货账户余额 05商户余额 06终端余额 30终点账户,String(2))
|
||||||
|
NotifyURL string `json:"notify_url,omitempty"` // notify_url: 后台通知地址 (可选,交易结果通知地址,String(128))
|
||||||
|
RefundAmtSts string `json:"refund_amt_sts,omitempty"` // refund_amt_sts: 退货资金状态 (可选,00 分账前,01 分账后;分账交易部分退货的情况,需要前端上送交易的分账状态,String(2))
|
||||||
|
}
|
||||||
|
|
||||||
|
// LocationInfo 地址位置信息结构体
|
||||||
|
type LocationInfo struct {
|
||||||
|
RequestIP string `json:"request_ip"` // request_ip: 请求IP (必填)
|
||||||
|
Location string `json:"location"` // location: 位置信息 (必填,如经纬度等)
|
||||||
|
BaseStation string `json:"base_station,omitempty"` // base_station: 基站信息 (可选)
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1 @@
|
||||||
|
package refundfee
|
||||||
|
|
@ -0,0 +1 @@
|
||||||
|
package refundfee
|
||||||
|
|
@ -0,0 +1 @@
|
||||||
|
package refundfee
|
||||||
|
|
@ -0,0 +1,37 @@
|
||||||
|
package refundquery
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/black1552/lkl_sdk/consts"
|
||||||
|
"github.com/black1552/lkl_sdk/lklsdk/common"
|
||||||
|
"github.com/gogf/gf/v2/os/gtime"
|
||||||
|
)
|
||||||
|
|
||||||
|
type RefundQuery[T any] struct {
|
||||||
|
client *common.Client[T]
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewRefundQuery 创建统一退货查询API实例
|
||||||
|
func NewRefundQuery[T any](client *common.Client[T]) *RefundQuery[T] {
|
||||||
|
return &RefundQuery[T]{
|
||||||
|
client: client,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// RefundQuery 发起统一退货查询请求
|
||||||
|
func (api *RefundQuery[T]) RefundQuery(req *RequestDataRefundQuery) (*T, error) {
|
||||||
|
// 构建请求参数
|
||||||
|
url := consts.BASE_URL + consts.LKL_UNIFIED_RETURN_REFUND_QUERY_URL
|
||||||
|
// 构建BaseModel请求
|
||||||
|
baseReq := RequestRefundQuery{
|
||||||
|
ReqTime: gtime.Now().Format("YmdHis"),
|
||||||
|
Version: "3.0",
|
||||||
|
ReqData: req,
|
||||||
|
}
|
||||||
|
|
||||||
|
// 发送请求
|
||||||
|
respBody, err := api.client.DoRequest(url, baseReq)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return respBody, nil
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,18 @@
|
||||||
|
package refundquery
|
||||||
|
|
||||||
|
// RequestRefundQuery 统一退货查询请求结构体
|
||||||
|
// API文档: https://o.lakala.com/#/home/document/detail?id=893
|
||||||
|
type RequestRefundQuery struct {
|
||||||
|
ReqTime string `json:"req_time" dc:"ReqTime 请求时间 格式yyyyMMddHHmmss"`
|
||||||
|
Version string `json:"version" dc:"Version 版本号 固定值3.0"`
|
||||||
|
ReqData *RequestDataRefundQuery `json:"req_data" dc:"ReqData 请求数据"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// RequestDataRefundQuery 统一退货查询请求数据结构体
|
||||||
|
// API文档: https://o.lakala.com/#/home/document/detail?id=893
|
||||||
|
type RequestDataRefundQuery struct {
|
||||||
|
MerchantNo string `json:"merchant_no" dc:"MerchantNo 商户号 必填,拉卡拉分配的商户号,String(15)"`
|
||||||
|
TermNo string `json:"term_no" dc:"TermNo 终端号 必填,拉卡拉分配的业务终端号,String(8)"`
|
||||||
|
OutTradeNo string `json:"out_trade_no,omitempty" dc:"OutTradeNo 商户交易流水号 选填,下单时的商户请求流水号,String(32) 与拉卡拉交易流水号二选一"`
|
||||||
|
TradeNo string `json:"trade_no,omitempty" dc:"TradeNo 拉卡拉交易流水号 选填,拉卡拉交易流水号,String(32) 与商户交易流水号二选一"`
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,56 @@
|
||||||
|
package refundquery
|
||||||
|
|
||||||
|
// ResponseRefundQuery 统一退货查询响应结构体
|
||||||
|
// API文档: https://o.lakala.com/#/home/document/detail?id=893
|
||||||
|
type ResponseRefundQuery struct {
|
||||||
|
Code string `json:"code" dc:"Code 响应码 RFD00000#成功、RFD11112#网络请求超时"`
|
||||||
|
Msg string `json:"msg" dc:"Msg 响应描述"`
|
||||||
|
Data *ResponseDataRefundQuery `json:"data,omitempty" dc:"Data 响应数据"`
|
||||||
|
Sign string `json:"sign" dc:"Sign 签名"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// ResponseDataRefundQuery 统一退货查询响应数据结构体
|
||||||
|
// API文档: https://o.lakala.com/#/home/document/detail?id=893
|
||||||
|
type ResponseDataRefundQuery struct {
|
||||||
|
OutTradeNo string `json:"out_trade_no" dc:"OutTradeNo 商户请求流水号 必填,原退款交易商户请求流水号(扫码交易返回)"`
|
||||||
|
TradeTime string `json:"trade_time" dc:"TradeTime 交易时间 必填,交易时间yyyyMMddHHmmss"`
|
||||||
|
TradeState string `json:"trade_state" dc:"TradeState 交易状态 必填,INIT-初始化;SUCCESS-交易成功;FAIL-交易失败;DEAL-交易处理中/未知;TIMEOUT-超时未知;EXCEPTION-异常"`
|
||||||
|
TradeNo string `json:"trade_no" dc:"TradeNo 拉卡拉商户订单号 必填,拉卡拉生成的交易流水"`
|
||||||
|
LogNo string `json:"log_no" dc:"LogNo 拉卡拉对账单流水号 必填,拉卡拉生成的对账单流水号(新增)"`
|
||||||
|
AccTradeNo string `json:"acc_trade_no,omitempty" dc:"AccTradeNo 账户端交易订单号 选填,账户端交易流水号,String(32)"`
|
||||||
|
RefundAmount string `json:"refund_amount" dc:"RefundAmount 交易金额 必填"`
|
||||||
|
PayMode string `json:"pay_mode,omitempty" dc:"PayMode 支付方式 选填,00 借记卡 01 贷记卡 02 准贷记卡 银行卡交易返回"`
|
||||||
|
CrdNo string `json:"crd_no,omitempty" dc:"CrdNo 卡号 选填,脱敏卡号,前六后四,中间用*替换"`
|
||||||
|
AccountType string `json:"account_type,omitempty" dc:"AccountType 钱包类型 选填,微信:WECHAT 支付宝:ALIPAY 银联:UNION 翼支付: BESTPAY 苏宁易付宝: SUNING 扫码交易返回"`
|
||||||
|
PayerAmount string `json:"payer_amount,omitempty" dc:"PayerAmount 付款人实付金额 选填,实际退款金额,单位分 扫码交易返回"`
|
||||||
|
AccSettleAmount string `json:"acc_settle_amount,omitempty" dc:"AccSettleAmount 账户端结算金额 选填,账户端应结订单金额,单位分 扫码交易返回"`
|
||||||
|
AccMdiscountAmount string `json:"acc_mdiscount_amount,omitempty" dc:"AccMdiscountAmount 商户侧优惠金额(账户端) 选填,商户优惠金额,单位分 扫码交易返回"`
|
||||||
|
AccDiscountAmount string `json:"acc_discount_amount,omitempty" dc:"AccDiscountAmount 账户端优惠金额 选填,拉卡拉优惠金额, 扫码交易返回"`
|
||||||
|
ChannelRetDesc string `json:"channel_ret_desc" dc:"ChannelRetDesc 返回描述信息 必填,code#msg:RFD00000#成功、RFD11112#网络请求超时"`
|
||||||
|
RefundSplitInfo []*RelateOutSplitRspInfo `json:"refund_split_info,omitempty" dc:"RefundSplitInfo 退款拆单信息 选填,合单交易退款查询时返回"`
|
||||||
|
OriginLogNo string `json:"origin_log_no" dc:"OriginLogNo 拉卡拉对账单流水号 必填,原交易拉卡拉对账单流水号"`
|
||||||
|
OriginOutTradeNo string `json:"origin_out_trade_no" dc:"OriginOutTradeNo 原商户交易流水号 必填"`
|
||||||
|
OriginTradeNo string `json:"origin_trade_no" dc:"OriginTradeNo 原交易拉卡拉交易订单号 必填"`
|
||||||
|
OriginTotalAmount string `json:"origin_total_amount" dc:"OriginTotalAmount 原交易金额 必填,原正交易订单金额"`
|
||||||
|
RefundType string `json:"refund_type,omitempty" dc:"RefundType 退货模式 选填"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// RelateOutSplitRspInfo 退款拆单信息结构体
|
||||||
|
// API文档: https://o.lakala.com/#/home/document/detail?id=893
|
||||||
|
type RelateOutSplitRspInfo struct {
|
||||||
|
OutSubTradeNo string `json:"out_sub_trade_no" dc:"OutSubTradeNo 外部子退款交易流水号 必填,商户子交易流水号,商户号下唯一"`
|
||||||
|
MerchantNo string `json:"merchant_no" dc:"MerchantNo 商户号 必填,拉卡拉分配的商户号"`
|
||||||
|
TermNo string `json:"term_no" dc:"TermNo 终端号 必填,拉卡拉分配的业务终端号"`
|
||||||
|
RefundAmount string `json:"refund_amount" dc:"RefundAmount 申请退款金额 必填,单位分,整数型字符"`
|
||||||
|
SubTradeNo string `json:"sub_trade_no,omitempty" dc:"SubTradeNo 拉卡拉子交易流水号 选填"`
|
||||||
|
SubLogNo string `json:"sub_log_no,omitempty" dc:"SubLogNo 对账单子流水号 选填,sub_trade_no后14位"`
|
||||||
|
TradeState string `json:"trade_state,omitempty" dc:"TradeState 子交易状态 选填,SUCCESS-交易成功 FAIL-交易失败"`
|
||||||
|
ResultCode string `json:"result_code,omitempty" dc:"ResultCode 处理结果码 选填"`
|
||||||
|
ResultMsg string `json:"result_msg,omitempty" dc:"ResultMsg 处理描述 选填"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// SuccessOrFail 判断交易是否成功
|
||||||
|
// 返回true表示交易成功,false表示交易失败或处理中
|
||||||
|
func (r *ResponseRefundQuery) SuccessOrFail() bool {
|
||||||
|
return r.Code == "000000"
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,43 @@
|
||||||
|
package unifiedreturn
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
|
||||||
|
"github.com/black1552/lkl_sdk/lklsdk/common"
|
||||||
|
"github.com/black1552/lkl_sdk/lklsdk/unifiedreturn/mergerefund"
|
||||||
|
"github.com/black1552/lkl_sdk/lklsdk/unifiedreturn/refund"
|
||||||
|
"github.com/black1552/lkl_sdk/lklsdk/unifiedreturn/refundquery"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Server[T any] struct {
|
||||||
|
Client *common.Client[T]
|
||||||
|
MergeRefound *mergerefund.MergeRefund[T]
|
||||||
|
Refound *refund.Refund[T]
|
||||||
|
RefoundQuery *refundquery.RefundQuery[T]
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewServer 创建拉卡拉统一退货服务实例
|
||||||
|
func NewServer[T any](ctx context.Context, cfgJson string) *Server[T] {
|
||||||
|
client := common.NewClient[T](ctx, cfgJson)
|
||||||
|
return &Server[T]{
|
||||||
|
Client: client,
|
||||||
|
MergeRefound: mergerefund.NewMergeRefund[T](client),
|
||||||
|
Refound: refund.NewRefund[T](client),
|
||||||
|
RefoundQuery: refundquery.NewRefundQuery[T](client),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// MergeRefund 合单退货
|
||||||
|
func (u *Server[T]) MergeRefund(req *mergerefund.RequestDataMergeRefund) (*T, error) {
|
||||||
|
return u.MergeRefound.MergeRefund(req)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Refund 退货
|
||||||
|
func (u *Server[T]) Refund(req *refund.RequestDataRefund) (*T, error) {
|
||||||
|
return u.Refound.Refund(req)
|
||||||
|
}
|
||||||
|
|
||||||
|
// RefundQuery 退货查询
|
||||||
|
func (u *Server[T]) RefundQuery(req *refundquery.RequestDataRefundQuery) (*T, error) {
|
||||||
|
return u.RefoundQuery.RefundQuery(req)
|
||||||
|
}
|
||||||
Loading…
Reference in New Issue