686 lines
22 KiB
Go
686 lines
22 KiB
Go
|
|
package service
|
|||
|
|
|
|||
|
|
import (
|
|||
|
|
"bytes"
|
|||
|
|
"context"
|
|||
|
|
"encoding/json"
|
|||
|
|
"fmt"
|
|||
|
|
"io"
|
|||
|
|
"jnc-server/app/main/api/internal/config"
|
|||
|
|
"net/http"
|
|||
|
|
"time"
|
|||
|
|
|
|||
|
|
"github.com/zeromicro/go-zero/core/logx"
|
|||
|
|
"github.com/zeromicro/go-zero/core/stores/redis"
|
|||
|
|
)
|
|||
|
|
|
|||
|
|
const (
|
|||
|
|
// Redis key 前缀
|
|||
|
|
YunYinSignPayTokenKey = "yunyin_sign_pay:token"
|
|||
|
|
YunYinSignPayUserIdKey = "yunyin_sign_pay:user_id"
|
|||
|
|
TokenCacheExpire = 2 * time.Hour // token 缓存2小时
|
|||
|
|
UserIdCacheExpire = 2 * time.Hour // 操作ID缓存2小时
|
|||
|
|
)
|
|||
|
|
|
|||
|
|
// YunYinSignPayService 云印签支付服务
|
|||
|
|
type YunYinSignPayService struct {
|
|||
|
|
config config.YunYinSignPayConfig
|
|||
|
|
client *http.Client
|
|||
|
|
redis *redis.Redis
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// NewYunYinSignPayService 创建云印签支付服务实例
|
|||
|
|
func NewYunYinSignPayService(c config.Config, redisClient *redis.Redis) *YunYinSignPayService {
|
|||
|
|
return &YunYinSignPayService{
|
|||
|
|
config: c.YunYinSignPay,
|
|||
|
|
client: &http.Client{
|
|||
|
|
Timeout: 30 * time.Second,
|
|||
|
|
},
|
|||
|
|
redis: redisClient,
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// GetAccessTokenRequest 获取token请求
|
|||
|
|
type GetAccessTokenRequest struct {
|
|||
|
|
AppID string `json:"appId"`
|
|||
|
|
AppSecret string `json:"appSecret"`
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// GetAccessTokenResponse 获取token响应
|
|||
|
|
type GetAccessTokenResponse struct {
|
|||
|
|
Code int `json:"code"`
|
|||
|
|
Msg string `json:"msg"`
|
|||
|
|
Data *AccessTokenData `json:"data,omitempty"`
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// AccessTokenData token数据
|
|||
|
|
type AccessTokenData struct {
|
|||
|
|
AccessToken string `json:"accessToken"`
|
|||
|
|
ExpireTime int64 `json:"expireTime,omitempty"`
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// GetAccessToken 获取访问token(带缓存)
|
|||
|
|
func (y *YunYinSignPayService) GetAccessToken(ctx context.Context) (string, error) {
|
|||
|
|
// 先从Redis获取
|
|||
|
|
token, err := y.redis.GetCtx(ctx, YunYinSignPayTokenKey)
|
|||
|
|
if err == nil && token != "" {
|
|||
|
|
logx.Infof("从Redis获取云印签token成功")
|
|||
|
|
return token, nil
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// Redis中没有,调用API获取
|
|||
|
|
logx.Infof("从API获取云印签token")
|
|||
|
|
reqBody := GetAccessTokenRequest{
|
|||
|
|
AppID: y.config.AppID,
|
|||
|
|
AppSecret: y.config.AppSecret,
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
jsonData, err := json.Marshal(reqBody)
|
|||
|
|
if err != nil {
|
|||
|
|
return "", fmt.Errorf("序列化请求参数失败: %v", err)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
url := fmt.Sprintf("%s/service/getAccessToken", y.config.ApiURL)
|
|||
|
|
req, err := http.NewRequestWithContext(ctx, "POST", url, bytes.NewReader(jsonData))
|
|||
|
|
if err != nil {
|
|||
|
|
return "", fmt.Errorf("创建请求失败: %v", err)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
req.Header.Set("Content-Type", "application/json")
|
|||
|
|
|
|||
|
|
resp, err := y.client.Do(req)
|
|||
|
|
if err != nil {
|
|||
|
|
return "", fmt.Errorf("请求失败: %v", err)
|
|||
|
|
}
|
|||
|
|
defer resp.Body.Close()
|
|||
|
|
|
|||
|
|
body, err := io.ReadAll(resp.Body)
|
|||
|
|
if err != nil {
|
|||
|
|
return "", fmt.Errorf("读取响应失败: %v", err)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
var tokenResp GetAccessTokenResponse
|
|||
|
|
if err := json.Unmarshal(body, &tokenResp); err != nil {
|
|||
|
|
return "", fmt.Errorf("解析响应失败: %v, 响应内容: %s", err, string(body))
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
if tokenResp.Code != 0 {
|
|||
|
|
return "", fmt.Errorf("获取token失败: %s", tokenResp.Msg)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
if tokenResp.Data == nil || tokenResp.Data.AccessToken == "" {
|
|||
|
|
return "", fmt.Errorf("token数据为空")
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
accessToken := tokenResp.Data.AccessToken
|
|||
|
|
|
|||
|
|
// 存储到Redis,2小时过期
|
|||
|
|
err = y.redis.SetexCtx(ctx, YunYinSignPayTokenKey, accessToken, int(TokenCacheExpire.Seconds()))
|
|||
|
|
if err != nil {
|
|||
|
|
logx.Errorf("存储token到Redis失败: %v", err)
|
|||
|
|
// 不返回错误,因为token已经获取成功
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
logx.Infof("获取云印签token成功,已缓存到Redis")
|
|||
|
|
return accessToken, nil
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// GetUserIdRequest 获取操作ID请求
|
|||
|
|
type GetUserIdRequest struct {
|
|||
|
|
Mobile string `json:"mobile"`
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// GetUserIdResponse 获取操作ID响应
|
|||
|
|
type GetUserIdResponse struct {
|
|||
|
|
Code int `json:"code"`
|
|||
|
|
Msg string `json:"msg"`
|
|||
|
|
Data *UserIdData `json:"data,omitempty"`
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// UserIdData 操作ID数据
|
|||
|
|
type UserIdData struct {
|
|||
|
|
UserId string `json:"userId"`
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// GetUserId 获取操作ID(带缓存)
|
|||
|
|
func (y *YunYinSignPayService) GetUserId(ctx context.Context, accessToken string) (string, error) {
|
|||
|
|
// 先从Redis获取
|
|||
|
|
userId, err := y.redis.GetCtx(ctx, YunYinSignPayUserIdKey)
|
|||
|
|
if err == nil && userId != "" {
|
|||
|
|
logx.Infof("从Redis获取云印签操作ID成功")
|
|||
|
|
return userId, nil
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// Redis中没有,调用API获取
|
|||
|
|
logx.Infof("从API获取云印签操作ID")
|
|||
|
|
reqBody := GetUserIdRequest{
|
|||
|
|
Mobile: y.config.Mobile,
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
jsonData, err := json.Marshal(reqBody)
|
|||
|
|
if err != nil {
|
|||
|
|
return "", fmt.Errorf("序列化请求参数失败: %v", err)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
url := fmt.Sprintf("%s/member/infoByCorpId", y.config.ApiURL)
|
|||
|
|
req, err := http.NewRequestWithContext(ctx, "POST", url, bytes.NewReader(jsonData))
|
|||
|
|
if err != nil {
|
|||
|
|
return "", fmt.Errorf("创建请求失败: %v", err)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
req.Header.Set("Content-Type", "application/json")
|
|||
|
|
req.Header.Set("appId", y.config.AppID)
|
|||
|
|
req.Header.Set("accessToken", accessToken)
|
|||
|
|
|
|||
|
|
resp, err := y.client.Do(req)
|
|||
|
|
if err != nil {
|
|||
|
|
return "", fmt.Errorf("请求失败: %v", err)
|
|||
|
|
}
|
|||
|
|
defer resp.Body.Close()
|
|||
|
|
|
|||
|
|
body, err := io.ReadAll(resp.Body)
|
|||
|
|
if err != nil {
|
|||
|
|
return "", fmt.Errorf("读取响应失败: %v", err)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
var userIdResp GetUserIdResponse
|
|||
|
|
if err := json.Unmarshal(body, &userIdResp); err != nil {
|
|||
|
|
return "", fmt.Errorf("解析响应失败: %v, 响应内容: %s", err, string(body))
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
if userIdResp.Code != 0 {
|
|||
|
|
return "", fmt.Errorf("获取操作ID失败: %s", userIdResp.Msg)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
if userIdResp.Data == nil || userIdResp.Data.UserId == "" {
|
|||
|
|
return "", fmt.Errorf("操作ID数据为空")
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// 获取操作ID
|
|||
|
|
operationUserId := userIdResp.Data.UserId
|
|||
|
|
|
|||
|
|
// 存储到Redis,2小时过期
|
|||
|
|
err = y.redis.SetexCtx(ctx, YunYinSignPayUserIdKey, operationUserId, int(UserIdCacheExpire.Seconds()))
|
|||
|
|
if err != nil {
|
|||
|
|
logx.Errorf("存储操作ID到Redis失败: %v", err)
|
|||
|
|
// 不返回错误,因为操作ID已经获取成功
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
logx.Infof("获取云印签操作ID成功,已缓存到Redis")
|
|||
|
|
return operationUserId, nil
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// ParticipantInfo 参与者信息
|
|||
|
|
type ParticipantInfo struct {
|
|||
|
|
ParticipantFlag string `json:"participantFlag"`
|
|||
|
|
PsnAccount string `json:"psnAccount"`
|
|||
|
|
PsnName string `json:"psnName"`
|
|||
|
|
ParticipantCorpName string `json:"participantCorpName,omitempty"`
|
|||
|
|
ParticipantType int `json:"participantType"`
|
|||
|
|
PayeeContractFlag int `json:"payeeContractFlag,omitempty"`
|
|||
|
|
Payee *PayeeInfo `json:"payee,omitempty"`
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// PayeeInfo 收款方信息
|
|||
|
|
type PayeeInfo struct {
|
|||
|
|
Amount float64 `json:"amount"`
|
|||
|
|
Priority string `json:"priority"`
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// FillComponent 填充组件
|
|||
|
|
type FillComponent struct {
|
|||
|
|
ComponentKey string `json:"componentKey"`
|
|||
|
|
ComponentValue string `json:"componentValue"`
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// StartSignFlowRequest 发起签署请求
|
|||
|
|
type StartSignFlowRequest struct {
|
|||
|
|
TemplateCode string `json:"templateCode"`
|
|||
|
|
TemplateName string `json:"templateName"`
|
|||
|
|
AutoFill int `json:"autoFill"`
|
|||
|
|
SourceOrderCode string `json:"sourceOrderCode"`
|
|||
|
|
ParticipantList []ParticipantInfo `json:"participantList"`
|
|||
|
|
FillComponents []FillComponent `json:"fillComponents"`
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// StartSignFlowResponse 发起签署响应
|
|||
|
|
type StartSignFlowResponse struct {
|
|||
|
|
Code interface{} `json:"code"` // 可能是 int 或 string
|
|||
|
|
Msg string `json:"msg"`
|
|||
|
|
Data *StartSignFlowData `json:"data,omitempty"`
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// StartSignFlowData 发起签署数据(API响应)
|
|||
|
|
type StartSignFlowData struct {
|
|||
|
|
FlowID string `json:"flowId,omitempty"`
|
|||
|
|
FlowStatus int `json:"flowStatus,omitempty"`
|
|||
|
|
FlowDesc string `json:"flowDesc,omitempty"`
|
|||
|
|
FlowTitle string `json:"flowTitle,omitempty"`
|
|||
|
|
NextStatus int `json:"nextStatus,omitempty"`
|
|||
|
|
ResultStatus interface{} `json:"resultStatus,omitempty"`
|
|||
|
|
ErrorMessage interface{} `json:"errorMessage,omitempty"`
|
|||
|
|
SourceOrderCode string `json:"sourceOrderCode,omitempty"`
|
|||
|
|
ParticipantList []ParticipantItem `json:"participantList,omitempty"`
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// StartSignFlowResult 发起签署流程返回结果
|
|||
|
|
type StartSignFlowResult struct {
|
|||
|
|
ParticipantID string // 签署方2的参与者ID
|
|||
|
|
TaskID string // 流程ID(flowId)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// ParticipantItem 参与者项(响应中的)
|
|||
|
|
type ParticipantItem struct {
|
|||
|
|
ParticipantID string `json:"participantId"`
|
|||
|
|
ParticipantFlag string `json:"participantFlag"`
|
|||
|
|
SignStatus interface{} `json:"signStatus,omitempty"`
|
|||
|
|
ParticipantCorpID string `json:"participantCorpId,omitempty"`
|
|||
|
|
ParticipantCorpName string `json:"participantCorpName,omitempty"`
|
|||
|
|
ParticipantType int `json:"participantType"`
|
|||
|
|
ParticipateBizType []string `json:"participateBizType,omitempty"`
|
|||
|
|
PsnID string `json:"psnId,omitempty"`
|
|||
|
|
PsnName string `json:"psnName,omitempty"`
|
|||
|
|
PayeeContractFlag interface{} `json:"payeeContractFlag,omitempty"`
|
|||
|
|
Payee interface{} `json:"payee,omitempty"`
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// GetCodeInt 获取 code 的 int 值
|
|||
|
|
func (r *StartSignFlowResponse) GetCodeInt() int {
|
|||
|
|
switch v := r.Code.(type) {
|
|||
|
|
case int:
|
|||
|
|
return v
|
|||
|
|
case float64:
|
|||
|
|
return int(v)
|
|||
|
|
case string:
|
|||
|
|
if v == "200" {
|
|||
|
|
return 200
|
|||
|
|
}
|
|||
|
|
return 0
|
|||
|
|
default:
|
|||
|
|
return 0
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// StartSignFlow 发起签署流程
|
|||
|
|
func (y *YunYinSignPayService) StartSignFlow(ctx context.Context, accessToken, userId string, req *StartSignFlowRequest) (*StartSignFlowResult, error) {
|
|||
|
|
jsonData, err := json.Marshal(req)
|
|||
|
|
if err != nil {
|
|||
|
|
return nil, fmt.Errorf("序列化请求参数失败: %v", err)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
url := fmt.Sprintf("%s/signTask/startSignFlow", y.config.ApiURL)
|
|||
|
|
httpReq, err := http.NewRequestWithContext(ctx, "POST", url, bytes.NewReader(jsonData))
|
|||
|
|
if err != nil {
|
|||
|
|
return nil, fmt.Errorf("创建请求失败: %v", err)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
httpReq.Header.Set("Content-Type", "application/json")
|
|||
|
|
httpReq.Header.Set("appId", y.config.AppID)
|
|||
|
|
httpReq.Header.Set("accessToken", accessToken)
|
|||
|
|
httpReq.Header.Set("userId", userId)
|
|||
|
|
httpReq.Header.Set("source", "pc")
|
|||
|
|
|
|||
|
|
resp, err := y.client.Do(httpReq)
|
|||
|
|
if err != nil {
|
|||
|
|
return nil, fmt.Errorf("请求失败: %v", err)
|
|||
|
|
}
|
|||
|
|
defer resp.Body.Close()
|
|||
|
|
|
|||
|
|
body, err := io.ReadAll(resp.Body)
|
|||
|
|
if err != nil {
|
|||
|
|
return nil, fmt.Errorf("读取响应失败: %v", err)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
var signFlowResp StartSignFlowResponse
|
|||
|
|
if err := json.Unmarshal(body, &signFlowResp); err != nil {
|
|||
|
|
return nil, fmt.Errorf("解析响应失败: %v, 响应内容: %s", err, string(body))
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// 检查响应码(可能是字符串"200"或数字200)
|
|||
|
|
codeInt := signFlowResp.GetCodeInt()
|
|||
|
|
if codeInt != 200 {
|
|||
|
|
return nil, fmt.Errorf("发起签署失败: %s", signFlowResp.Msg)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
if signFlowResp.Data == nil {
|
|||
|
|
return nil, fmt.Errorf("签署数据为空")
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// 从 participantList 中找到签署方2的 participantId
|
|||
|
|
var participantID2 string
|
|||
|
|
for _, participant := range signFlowResp.Data.ParticipantList {
|
|||
|
|
if participant.ParticipantFlag == "签署方2" {
|
|||
|
|
participantID2 = participant.ParticipantID
|
|||
|
|
break
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
if participantID2 == "" {
|
|||
|
|
return nil, fmt.Errorf("未找到签署方2的参与者ID")
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
logx.Infof("发起云印签签署流程成功,订单号: %s, 流程ID: %s, 签署方2参与者ID: %s", req.SourceOrderCode, signFlowResp.Data.FlowID, participantID2)
|
|||
|
|
|
|||
|
|
// 返回结果,包含签署方2的参与者ID和流程ID
|
|||
|
|
return &StartSignFlowResult{
|
|||
|
|
ParticipantID: participantID2,
|
|||
|
|
TaskID: signFlowResp.Data.FlowID, // 使用 flowId 作为 taskId
|
|||
|
|
}, nil
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// GetPaymentURLRequest 获取支付链接请求
|
|||
|
|
type GetPaymentURLRequest struct {
|
|||
|
|
ParticipantID string `json:"participantId"`
|
|||
|
|
Type int `json:"type"` // 0=微信支付,1=支付宝支付
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// GetPaymentURLResponse 获取支付链接响应
|
|||
|
|
type GetPaymentURLResponse struct {
|
|||
|
|
Code int `json:"code"`
|
|||
|
|
Msg string `json:"msg"`
|
|||
|
|
Data *PaymentURLData `json:"data,omitempty"`
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// PaymentURLData 支付链接数据
|
|||
|
|
type PaymentURLData struct {
|
|||
|
|
PayURL string `json:"payUrl,omitempty"`
|
|||
|
|
URL string `json:"url,omitempty"`
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// GetPaymentURL 获取支付链接
|
|||
|
|
func (y *YunYinSignPayService) GetPaymentURL(ctx context.Context, accessToken, userId string, participantID string, payType int) (string, error) {
|
|||
|
|
reqBody := GetPaymentURLRequest{
|
|||
|
|
ParticipantID: participantID,
|
|||
|
|
Type: payType,
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
jsonData, err := json.Marshal(reqBody)
|
|||
|
|
if err != nil {
|
|||
|
|
return "", fmt.Errorf("序列化请求参数失败: %v", err)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
url := fmt.Sprintf("%s/signTask/contract/appletForParticipant", y.config.ApiURL)
|
|||
|
|
httpReq, err := http.NewRequestWithContext(ctx, "POST", url, bytes.NewReader(jsonData))
|
|||
|
|
if err != nil {
|
|||
|
|
return "", fmt.Errorf("创建请求失败: %v", err)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
httpReq.Header.Set("Content-Type", "application/json")
|
|||
|
|
httpReq.Header.Set("appId", y.config.AppID)
|
|||
|
|
httpReq.Header.Set("accessToken", accessToken)
|
|||
|
|
httpReq.Header.Set("userId", userId)
|
|||
|
|
httpReq.Header.Set("source", "pc")
|
|||
|
|
|
|||
|
|
resp, err := y.client.Do(httpReq)
|
|||
|
|
if err != nil {
|
|||
|
|
return "", fmt.Errorf("请求失败: %v", err)
|
|||
|
|
}
|
|||
|
|
defer resp.Body.Close()
|
|||
|
|
|
|||
|
|
body, err := io.ReadAll(resp.Body)
|
|||
|
|
if err != nil {
|
|||
|
|
return "", fmt.Errorf("读取响应失败: %v", err)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
var paymentURLResp GetPaymentURLResponse
|
|||
|
|
if err := json.Unmarshal(body, &paymentURLResp); err != nil {
|
|||
|
|
return "", fmt.Errorf("解析响应失败: %v, 响应内容: %s", err, string(body))
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
if paymentURLResp.Code != 0 {
|
|||
|
|
return "", fmt.Errorf("获取支付链接失败: %s", paymentURLResp.Msg)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
if paymentURLResp.Data == nil {
|
|||
|
|
return "", fmt.Errorf("支付链接数据为空")
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// 优先返回 PayURL,如果没有则返回 URL
|
|||
|
|
payURL := paymentURLResp.Data.PayURL
|
|||
|
|
if payURL == "" {
|
|||
|
|
payURL = paymentURLResp.Data.URL
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
if payURL == "" {
|
|||
|
|
return "", fmt.Errorf("支付链接为空")
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
logx.Infof("获取云印签支付链接成功,参与者ID: %s", participantID)
|
|||
|
|
return payURL, nil
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// CreateYunYinSignPayOrderResult 创建云印签支付订单结果
|
|||
|
|
type CreateYunYinSignPayOrderResult struct {
|
|||
|
|
PayURL string // 支付链接
|
|||
|
|
ParticipantID string // 参与者ID
|
|||
|
|
TaskID string // 任务ID
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// CreateYunYinSignPayOrder 创建云印签支付订单(封装完整流程)
|
|||
|
|
func (y *YunYinSignPayService) CreateYunYinSignPayOrder(ctx context.Context, userMobile, userName string, amount float64, outTradeNo string, payType int) (*CreateYunYinSignPayOrderResult, error) {
|
|||
|
|
// 1. 获取token和操作ID(带缓存)
|
|||
|
|
accessToken, err := y.GetAccessToken(ctx)
|
|||
|
|
if err != nil {
|
|||
|
|
return nil, fmt.Errorf("获取云印签token失败: %v", err)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
operationUserId, err := y.GetUserId(ctx, accessToken)
|
|||
|
|
if err != nil {
|
|||
|
|
return nil, fmt.Errorf("获取云印签操作ID失败: %v", err)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// 2. 构建参与者列表
|
|||
|
|
participantList := []ParticipantInfo{
|
|||
|
|
// 签署方1:我方
|
|||
|
|
{
|
|||
|
|
ParticipantFlag: "签署方1",
|
|||
|
|
PsnAccount: y.config.Mobile,
|
|||
|
|
PsnName: y.config.Name,
|
|||
|
|
ParticipantCorpName: y.config.CorpName,
|
|||
|
|
ParticipantType: 1, // 1表示企业
|
|||
|
|
},
|
|||
|
|
// 签署方2:用户(支付方)
|
|||
|
|
{
|
|||
|
|
ParticipantFlag: "签署方2",
|
|||
|
|
PsnAccount: userMobile,
|
|||
|
|
PsnName: func() string {
|
|||
|
|
if userName != "" {
|
|||
|
|
return userName
|
|||
|
|
}
|
|||
|
|
return "用户"
|
|||
|
|
}(),
|
|||
|
|
ParticipantType: 0, // 0表示个人
|
|||
|
|
PayeeContractFlag: 1,
|
|||
|
|
Payee: &PayeeInfo{
|
|||
|
|
Amount: amount,
|
|||
|
|
Priority: "1",
|
|||
|
|
},
|
|||
|
|
},
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// 3. 发起签署流程
|
|||
|
|
startSignFlowReq := &StartSignFlowRequest{
|
|||
|
|
TemplateCode: y.config.TemplateCode,
|
|||
|
|
TemplateName: y.config.TemplateName,
|
|||
|
|
AutoFill: 1,
|
|||
|
|
SourceOrderCode: outTradeNo,
|
|||
|
|
ParticipantList: participantList,
|
|||
|
|
FillComponents: []FillComponent{}, // 可以根据需要填充
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
signFlowData, err := y.StartSignFlow(ctx, accessToken, operationUserId, startSignFlowReq)
|
|||
|
|
if err != nil {
|
|||
|
|
return nil, fmt.Errorf("发起云印签签署失败: %v", err)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
if signFlowData.ParticipantID == "" {
|
|||
|
|
return nil, fmt.Errorf("签署流程返回的参与者ID为空")
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// 4. 获取支付链接
|
|||
|
|
payURL, err := y.GetPaymentURL(ctx, accessToken, operationUserId, signFlowData.ParticipantID, payType)
|
|||
|
|
if err != nil {
|
|||
|
|
return nil, fmt.Errorf("获取云印签支付链接失败: %v", err)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
return &CreateYunYinSignPayOrderResult{
|
|||
|
|
PayURL: payURL,
|
|||
|
|
ParticipantID: signFlowData.ParticipantID,
|
|||
|
|
TaskID: signFlowData.TaskID,
|
|||
|
|
}, nil
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// QueryPayeeBillRequest 查询收款单请求
|
|||
|
|
type QueryPayeeBillRequest struct {
|
|||
|
|
SourceOrderCode string `json:"sourceOrderCode,omitempty"` // 来源订单号
|
|||
|
|
PayOrderCode string `json:"payOrderCode,omitempty"` // 支付单号
|
|||
|
|
ContractCode string `json:"contractCode,omitempty"` // 合同号
|
|||
|
|
PayStatus int `json:"payStatus,omitempty"` // 支付状态: 1-待支付, 2-支付成功, 3-退款成功
|
|||
|
|
ChannelOrderNo string `json:"channelOrderNo,omitempty"` // 渠道单号
|
|||
|
|
PayName string `json:"payName,omitempty"` // 付款方名称
|
|||
|
|
ParticipantType int `json:"participantType,omitempty"` // 付款方类型: 1-企业, 0-个人
|
|||
|
|
PayeeName string `json:"payeeName,omitempty"` // 收款方名称
|
|||
|
|
PayeeCorpName string `json:"payeeCorpName,omitempty"` // 收款方企业名称
|
|||
|
|
FlowStartTime string `json:"flowStartTime,omitempty"` // 查询范围的开始时间
|
|||
|
|
FlowFinishTime string `json:"flowFinishTime,omitempty"` // 查询范围的结束时间
|
|||
|
|
ListPageNo int `json:"listPageNo,omitempty"` // 当前页码,从1开始
|
|||
|
|
ListPageSize int `json:"listPageSize,omitempty"` // 每页返回的记录数量
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// QueryPayeeBillResponse 查询收款单响应
|
|||
|
|
type QueryPayeeBillResponse struct {
|
|||
|
|
Code interface{} `json:"code"` // 返回码,可能是字符串"200"或数字200
|
|||
|
|
Msg string `json:"msg"` // 返回码的描述信息
|
|||
|
|
TotalCount int `json:"totalCount"` // 满足查询条件的总记录数
|
|||
|
|
ListPageCount int `json:"listPageCount"` // 总页数
|
|||
|
|
Data []PayeeBillItem `json:"data"` // 收款单记录列表
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// PayeeBillItem 收款单记录
|
|||
|
|
type PayeeBillItem struct {
|
|||
|
|
ID int64 `json:"id"` // 消息ID
|
|||
|
|
ContractName string `json:"contractName"` // 合同名称
|
|||
|
|
ContractCode string `json:"contractCode"` // 合同号
|
|||
|
|
FlowCode string `json:"flowCode"` // 签署流程编码
|
|||
|
|
PayOrderCode string `json:"payOrderCode"` // 支付单号
|
|||
|
|
Amount float64 `json:"amount"` // 支付金额
|
|||
|
|
PayStatus int `json:"payStatus"` // 支付状态: 0-订单生成, 1-支付中, 2-支付成功, 3-支付失败, 4-已退款
|
|||
|
|
RefundAmount float64 `json:"refundAmount"` // 退款金额
|
|||
|
|
ParticipateID int64 `json:"participateId"` // 付款参与方ID
|
|||
|
|
PsnName string `json:"psnName"` // 付款方经办人/个人名称
|
|||
|
|
PsnAccount string `json:"psnAccount"` // 付款方经办人手机
|
|||
|
|
ParticipantType int `json:"participantType"` // 付款方类型 (1-企业, 2-个人)
|
|||
|
|
ParticipantCorpName string `json:"participantCorpName"` // 付款方企业名称
|
|||
|
|
ParticipateBizType string `json:"participateBizType"` // 付款方参与方式
|
|||
|
|
PayeeCorpName string `json:"payeeCorpName"` // 收款方企业名称
|
|||
|
|
CreateTime string `json:"createTime"` // 订单创建时间
|
|||
|
|
ChannelOrderNo string `json:"channelOrderNo"` // 渠道单号 (如微信或支付宝的交易号)
|
|||
|
|
ProviderCode string `json:"providerCode"` // 支付方式编码
|
|||
|
|
CallbackStatus int `json:"callbackStatus"` // 异步回调通知状态
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// QueryPayeeBillResult 查询收款单结果
|
|||
|
|
type QueryPayeeBillResult struct {
|
|||
|
|
PayStatus int // 支付状态: 0-订单生成, 1-支付中, 2-支付成功, 3-支付失败, 4-已退款
|
|||
|
|
PayOrderCode string // 支付单号
|
|||
|
|
ChannelOrderNo string // 渠道单号
|
|||
|
|
Amount float64 // 支付金额
|
|||
|
|
RefundAmount float64 // 退款金额
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// GetCodeInt 获取 code 的 int 值
|
|||
|
|
func (r *QueryPayeeBillResponse) GetCodeInt() int {
|
|||
|
|
switch v := r.Code.(type) {
|
|||
|
|
case int:
|
|||
|
|
return v
|
|||
|
|
case float64:
|
|||
|
|
return int(v)
|
|||
|
|
case string:
|
|||
|
|
if v == "200" {
|
|||
|
|
return 200
|
|||
|
|
}
|
|||
|
|
return 0
|
|||
|
|
default:
|
|||
|
|
return 0
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// QueryPayeeBill 查询收款单(根据sourceOrderCode查询)
|
|||
|
|
func (y *YunYinSignPayService) QueryPayeeBill(ctx context.Context, sourceOrderCode string) (*QueryPayeeBillResult, error) {
|
|||
|
|
// 1. 获取token和操作ID(带缓存)
|
|||
|
|
accessToken, err := y.GetAccessToken(ctx)
|
|||
|
|
if err != nil {
|
|||
|
|
return nil, fmt.Errorf("获取云印签token失败: %v", err)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
operationUserId, err := y.GetUserId(ctx, accessToken)
|
|||
|
|
if err != nil {
|
|||
|
|
return nil, fmt.Errorf("获取云印签操作ID失败: %v", err)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// 2. 构建查询请求
|
|||
|
|
reqBody := QueryPayeeBillRequest{
|
|||
|
|
SourceOrderCode: sourceOrderCode,
|
|||
|
|
ListPageNo: 1,
|
|||
|
|
ListPageSize: 10, // 只需要查询第一条匹配的记录
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
jsonData, err := json.Marshal(reqBody)
|
|||
|
|
if err != nil {
|
|||
|
|
return nil, fmt.Errorf("序列化请求参数失败: %v", err)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// 3. 调用查询API
|
|||
|
|
url := fmt.Sprintf("%s/signFlowBill/payeeBillList", y.config.ApiURL)
|
|||
|
|
httpReq, err := http.NewRequestWithContext(ctx, "POST", url, bytes.NewReader(jsonData))
|
|||
|
|
if err != nil {
|
|||
|
|
return nil, fmt.Errorf("创建请求失败: %v", err)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
httpReq.Header.Set("Content-Type", "application/json")
|
|||
|
|
httpReq.Header.Set("appId", y.config.AppID)
|
|||
|
|
httpReq.Header.Set("accessToken", accessToken)
|
|||
|
|
httpReq.Header.Set("userId", operationUserId)
|
|||
|
|
httpReq.Header.Set("source", "pc")
|
|||
|
|
|
|||
|
|
resp, err := y.client.Do(httpReq)
|
|||
|
|
if err != nil {
|
|||
|
|
return nil, fmt.Errorf("请求失败: %v", err)
|
|||
|
|
}
|
|||
|
|
defer resp.Body.Close()
|
|||
|
|
|
|||
|
|
body, err := io.ReadAll(resp.Body)
|
|||
|
|
if err != nil {
|
|||
|
|
return nil, fmt.Errorf("读取响应失败: %v", err)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
var queryResp QueryPayeeBillResponse
|
|||
|
|
if err := json.Unmarshal(body, &queryResp); err != nil {
|
|||
|
|
return nil, fmt.Errorf("解析响应失败: %v, 响应内容: %s", err, string(body))
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// 4. 检查响应码
|
|||
|
|
codeInt := queryResp.GetCodeInt()
|
|||
|
|
if codeInt != 200 {
|
|||
|
|
return nil, fmt.Errorf("查询收款单失败: %s", queryResp.Msg)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// 5. 查找匹配的记录(根据sourceOrderCode)
|
|||
|
|
if len(queryResp.Data) == 0 {
|
|||
|
|
return nil, fmt.Errorf("未找到匹配的收款单记录")
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// 取第一条记录(因为我们已经用sourceOrderCode精确查询)
|
|||
|
|
billItem := queryResp.Data[0]
|
|||
|
|
|
|||
|
|
logx.Infof("查询云印签收款单成功,订单号: %s, 支付状态: %d", sourceOrderCode, billItem.PayStatus)
|
|||
|
|
|
|||
|
|
return &QueryPayeeBillResult{
|
|||
|
|
PayStatus: billItem.PayStatus,
|
|||
|
|
PayOrderCode: billItem.PayOrderCode,
|
|||
|
|
ChannelOrderNo: billItem.ChannelOrderNo,
|
|||
|
|
Amount: billItem.Amount,
|
|||
|
|
RefundAmount: billItem.RefundAmount,
|
|||
|
|
}, nil
|
|||
|
|
}
|