package pay import ( "context" "database/sql" "net/http" "qnc-server/app/main/api/internal/svc" "qnc-server/app/main/model" "strings" "time" "github.com/pkg/errors" "github.com/wechatpay-apiv3/wechatpay-go/services/refunddomestic" "github.com/zeromicro/go-zero/core/logx" "github.com/zeromicro/go-zero/core/stores/sqlx" ) type WechatPayRefundCallbackLogic struct { logx.Logger ctx context.Context svcCtx *svc.ServiceContext } func NewWechatPayRefundCallbackLogic(ctx context.Context, svcCtx *svc.ServiceContext) *WechatPayRefundCallbackLogic { return &WechatPayRefundCallbackLogic{ Logger: logx.WithContext(ctx), ctx: ctx, svcCtx: svcCtx, } } // handleQueryOrderRefund 处理查询订单退款 func (l *WechatPayRefundCallbackLogic) handleQueryOrderRefund(orderNo string, status refunddomestic.Status) error { order, err := l.svcCtx.OrderModel.FindOneByOrderNo(l.ctx, orderNo) if err != nil { return errors.Wrapf(err, "查找查询订单信息失败: %s", orderNo) } // 只处理成功和失败状态 var orderStatus, refundStatus string switch status { case refunddomestic.STATUS_SUCCESS: orderStatus = model.OrderStatusRefunded refundStatus = model.OrderRefundStatusSuccess case refunddomestic.STATUS_CLOSED: // 退款关闭,保持订单原状态,更新退款记录为失败 refundStatus = model.OrderRefundStatusFailed case refunddomestic.STATUS_ABNORMAL: // 退款异常,保持订单原状态,更新退款记录为失败 refundStatus = model.OrderRefundStatusFailed default: // 其他状态暂不处理 return nil } // 使用事务同时更新订单和退款记录 err = l.svcCtx.OrderModel.Trans(l.ctx, func(ctx context.Context, session sqlx.Session) error { // 更新订单状态(仅在退款成功时更新) if status == refunddomestic.STATUS_SUCCESS { order.Status = orderStatus order.RefundTime = sql.NullTime{ Time: time.Now(), Valid: true, } if err := l.svcCtx.OrderModel.UpdateWithVersion(ctx, session, order); err != nil { return errors.Wrapf(err, "更新查询订单状态失败: %s", orderNo) } } // 更新退款记录状态 refund, err := l.svcCtx.OrderRefundModel.FindOneByOrderId(ctx, order.Id) if err != nil { if err == model.ErrNotFound { logx.Errorf("未找到订单对应的退款记录: orderNo=%s, orderId=%d", orderNo, order.Id) return nil // 没有退款记录时不报错,只记录警告 } return errors.Wrapf(err, "查找退款记录失败: orderNo=%s", orderNo) } refund.Status = refundStatus if status == refunddomestic.STATUS_SUCCESS { refund.RefundTime = sql.NullTime{ Time: time.Now(), Valid: true, } } else if status == refunddomestic.STATUS_CLOSED { refund.CloseTime = sql.NullTime{ Time: time.Now(), Valid: true, } } if _, err := l.svcCtx.OrderRefundModel.Update(ctx, session, refund); err != nil { return errors.Wrapf(err, "更新退款记录状态失败: orderNo=%s", orderNo) } return nil }) if err != nil { return errors.Wrapf(err, "更新订单和退款记录失败: %s", orderNo) } return nil } // handleAgentOrderRefund 处理代理会员订单退款 func (l *WechatPayRefundCallbackLogic) handleAgentOrderRefund(orderNo string, status refunddomestic.Status) error { order, err := l.svcCtx.AgentMembershipRechargeOrderModel.FindOneByOrderNo(l.ctx, orderNo) if err != nil { return errors.Wrapf(err, "查找代理会员订单信息失败: %s", orderNo) } if status == refunddomestic.STATUS_SUCCESS { order.Status = "refunded" } else if status == refunddomestic.STATUS_ABNORMAL { return nil // 异常状态直接返回 } else { return nil // 其他状态直接返回 } if err := l.svcCtx.AgentMembershipRechargeOrderModel.UpdateWithVersion(l.ctx, nil, order); err != nil { return errors.Wrapf(err, "更新代理会员订单状态失败: %s", orderNo) } return nil } // sendSuccessResponse 发送成功响应 func (l *WechatPayRefundCallbackLogic) sendSuccessResponse(w http.ResponseWriter) { w.WriteHeader(http.StatusOK) _, _ = w.Write([]byte("success")) } func (l *WechatPayRefundCallbackLogic) WechatPayRefundCallback(w http.ResponseWriter, r *http.Request) error { // 1. 处理微信退款通知 notification, err := l.svcCtx.WechatPayService.HandleRefundNotification(l.ctx, r) if err != nil { logx.Errorf("微信退款回调处理失败: %v", err) l.sendSuccessResponse(w) return nil } orderNo := *notification.OutTradeNo var processErr error // 2. 根据订单号前缀处理不同类型的订单 switch { case strings.HasPrefix(orderNo, "Q_"): processErr = l.handleQueryOrderRefund(orderNo, *notification.Status) case strings.HasPrefix(orderNo, "A_"): processErr = l.handleAgentOrderRefund(orderNo, *notification.Status) default: // 兼容旧订单,假设没有前缀的是查询订单 processErr = l.handleQueryOrderRefund(orderNo, *notification.Status) } // 3. 处理错误并响应 if processErr != nil { logx.Errorf("处理退款订单失败: %v", processErr) } // 无论处理是否成功,都返回成功响应给微信 l.sendSuccessResponse(w) return nil }