qnc-server-tob/app/main/api/internal/queue/paySuccessNotify.go

176 lines
5.5 KiB
Go
Raw Normal View History

2025-01-10 00:09:25 +08:00
package queue
import (
"context"
"database/sql"
2025-01-10 00:09:25 +08:00
"encoding/hex"
"encoding/json"
"fmt"
2025-03-07 03:48:59 +08:00
"os"
2025-06-09 12:34:52 +08:00
"qnc-server/app/main/api/internal/svc"
"qnc-server/app/main/api/internal/types"
"qnc-server/app/main/model"
2025-04-11 13:10:17 +08:00
"qnc-server/pkg/lzkit/crypto"
"qnc-server/pkg/lzkit/lzUtils"
"time"
2025-04-08 14:50:24 +08:00
2025-05-24 14:26:20 +08:00
"github.com/bytedance/sonic"
2025-04-08 14:50:24 +08:00
"github.com/hibiken/asynq"
"github.com/zeromicro/go-zero/core/logx"
2025-01-10 00:09:25 +08:00
)
type PaySuccessNotifyUserHandler struct {
svcCtx *svc.ServiceContext
}
func NewPaySuccessNotifyUserHandler(svcCtx *svc.ServiceContext) *PaySuccessNotifyUserHandler {
return &PaySuccessNotifyUserHandler{
svcCtx: svcCtx,
}
}
var payload struct {
OrderID int64 `json:"order_id"`
}
func (l *PaySuccessNotifyUserHandler) ProcessTask(ctx context.Context, t *asynq.Task) error {
// 从任务的负载中解码数据
if err := json.Unmarshal(t.Payload(), &payload); err != nil {
return fmt.Errorf("解析任务负载失败: %w", err)
}
order, err := l.svcCtx.OrderModel.FindOne(ctx, payload.OrderID)
if err != nil {
return fmt.Errorf("无效的订单ID: %d, %v", payload.OrderID, err)
}
2025-03-07 03:48:59 +08:00
env := os.Getenv("ENV")
if order.Status != "paid" && env != "development" {
2025-01-10 00:09:25 +08:00
err = fmt.Errorf("无效的订单: %d", payload.OrderID)
logx.Errorf("处理任务失败,原因: %v", err)
return asynq.SkipRetry
}
product, err := l.svcCtx.ProductModel.FindOne(ctx, order.ProductId)
if err != nil {
return fmt.Errorf("找不到相关产品: orderID: %d, productID: %d", payload.OrderID, order.ProductId)
}
2025-04-15 22:52:02 +08:00
redisKey := fmt.Sprintf(types.QueryCacheKey, order.UserId, order.OrderNo)
2025-04-08 14:50:24 +08:00
cache, cacheErr := l.svcCtx.Redis.GetCtx(ctx, redisKey)
if cacheErr != nil {
2025-04-11 13:10:17 +08:00
return fmt.Errorf("获取缓存内容失败: %+v", cacheErr)
2025-01-10 00:09:25 +08:00
}
2025-04-08 14:50:24 +08:00
var data types.QueryCacheLoad
2025-05-24 14:26:20 +08:00
err = sonic.Unmarshal([]byte(cache), &data)
2025-04-08 14:50:24 +08:00
if err != nil {
2025-04-11 13:10:17 +08:00
return fmt.Errorf("解析缓存内容失败: %+v", err)
2025-01-10 00:09:25 +08:00
}
secretKey := l.svcCtx.Config.Encrypt.SecretKey
key, decodeErr := hex.DecodeString(secretKey)
if decodeErr != nil {
2025-04-11 13:10:17 +08:00
return fmt.Errorf("获取AES密钥失败: %+v", decodeErr)
2025-01-10 00:09:25 +08:00
}
2025-04-08 14:50:24 +08:00
decryptData, aesdecryptErr := crypto.AesDecrypt(data.Params, key)
2025-01-10 00:09:25 +08:00
if aesdecryptErr != nil {
2025-04-11 13:10:17 +08:00
return fmt.Errorf("解密参数失败: %+v", aesdecryptErr)
2025-04-08 14:50:24 +08:00
}
// 从数据库中查询完整的查询记录
2025-05-24 14:26:20 +08:00
query, err := l.svcCtx.QueryModel.FindOneByOrderId(ctx, order.Id)
2025-04-08 14:50:24 +08:00
if err != nil {
return fmt.Errorf("获取插入后的查询记录失败: %+v", err)
2025-01-10 00:09:25 +08:00
}
combinedResponse, err := l.svcCtx.ApiRequestService.ProcessRequests(decryptData, product.Id)
if err != nil {
return l.handleError(ctx, err, order, query)
}
// 加密返回响应
encryptData, aesEncryptErr := crypto.AesEncrypt(combinedResponse, key)
if aesEncryptErr != nil {
err = fmt.Errorf("加密响应信息失败: %v", aesEncryptErr)
2025-04-08 14:50:24 +08:00
return l.handleError(ctx, err, order, query)
2025-01-10 00:09:25 +08:00
}
query.QueryData = lzUtils.StringToNullString(encryptData)
updateErr := l.svcCtx.QueryModel.UpdateWithVersion(ctx, nil, query)
if updateErr != nil {
err = fmt.Errorf("保存响应数据失败: %v", updateErr)
2025-04-08 14:50:24 +08:00
return l.handleError(ctx, err, order, query)
2025-01-10 00:09:25 +08:00
}
query.QueryState = "success"
updateQueryErr := l.svcCtx.QueryModel.UpdateWithVersion(ctx, nil, query)
if updateQueryErr != nil {
updateQueryErr = fmt.Errorf("修改查询状态失败: %v", updateQueryErr)
return l.handleError(ctx, updateQueryErr, order, query)
}
2025-03-07 03:48:59 +08:00
err = l.svcCtx.AgentService.AgentProcess(ctx, order)
if err != nil {
return l.handleError(ctx, err, order, query)
}
2025-04-08 14:50:24 +08:00
_, delErr := l.svcCtx.Redis.DelCtx(ctx, redisKey)
if delErr != nil {
logx.Errorf("删除Redis缓存失败但任务已成功处理订单ID: %d, 错误: %v", order.Id, delErr)
}
2025-01-10 00:09:25 +08:00
return nil
}
// 定义一个中间件函数
func (l *PaySuccessNotifyUserHandler) handleError(ctx context.Context, err error, order *model.Order, query *model.Query) error {
logx.Errorf("处理任务失败,原因: %v", err)
2025-04-15 22:52:02 +08:00
redisKey := fmt.Sprintf(types.QueryCacheKey, order.UserId, order.OrderNo)
2025-04-08 14:50:24 +08:00
_, delErr := l.svcCtx.Redis.DelCtx(ctx, redisKey)
if delErr != nil {
logx.Errorf("删除Redis缓存失败订单ID: %d, 错误: %v", order.Id, delErr)
}
2025-01-10 00:09:25 +08:00
if order.Status == "paid" && query.QueryState == "pending" {
// 更新查询状态为失败
query.QueryState = "failed"
updateQueryErr := l.svcCtx.QueryModel.UpdateWithVersion(ctx, nil, query)
if updateQueryErr != nil {
logx.Errorf("更新查询状态失败订单ID: %d, 错误: %v", order.Id, updateQueryErr)
return asynq.SkipRetry
}
// 退款
if order.PaymentPlatform == "wechat" {
refundErr := l.svcCtx.WechatPayService.WeChatRefund(ctx, order.OrderNo, order.Amount, order.Amount)
if refundErr != nil {
logx.Error(refundErr)
return asynq.SkipRetry
}
} else {
refund, refundErr := l.svcCtx.AlipayService.AliRefund(ctx, order.OrderNo, order.Amount)
if refundErr != nil {
logx.Error(refundErr)
return asynq.SkipRetry
}
if refund.IsSuccess() {
logx.Errorf("支付宝退款成功, orderID: %d", order.Id)
// 更新订单状态为退款
order.Status = "refunded"
order.RefundTime = sql.NullTime{
Time: time.Now(),
Valid: true,
}
2025-01-10 00:09:25 +08:00
updateOrderErr := l.svcCtx.OrderModel.UpdateWithVersion(ctx, nil, order)
if updateOrderErr != nil {
logx.Errorf("更新订单状态失败订单ID: %d, 错误: %v", order.Id, updateOrderErr)
return fmt.Errorf("更新订单状态失败: %v", updateOrderErr)
}
return asynq.SkipRetry
} else {
logx.Errorf("支付宝退款失败:%v", refundErr)
return asynq.SkipRetry
}
// 直接成功
}
}
return asynq.SkipRetry
}