fix(coupon): 修复订单取消时金额券未退还的bug

订单取消退券逻辑依赖 used_order_id 匹配,但金额券在下单时
不设置 used_order_id(仅在支付确认后设置),导致未支付订单
取消时 WHERE 条件匹配不到行,退券静默失败。

修复:去掉 used_order_id 条件,按券 ID 直接退还,增加幂等
校验和错误处理,兜底从流水回推预扣金额。
This commit is contained in:
Zuncle 2026-03-17 21:15:33 +08:00
parent d1ee319f0e
commit b9a40df5c5

View File

@ -59,32 +59,66 @@ func (s *service) CancelOrder(ctx context.Context, userID int64, orderID int64,
// 4. 退还优惠券(恢复预扣的余额和状态)
if order.CouponID > 0 {
var oc struct {
AppliedAmount int64
// 幂等校验:若已记录过 cancel_refund 流水则跳过
refundExists, err := tx.UserCouponLedger.WithContext(ctx).Where(
tx.UserCouponLedger.UserCouponID.Eq(order.CouponID),
tx.UserCouponLedger.OrderID.Eq(order.ID),
tx.UserCouponLedger.Action.Eq("cancel_refund"),
).Count()
if err != nil {
return err
}
// 获取该订单实际扣减的优惠券金额
_ = tx.OrderCoupons.WithContext(ctx).Where(tx.OrderCoupons.OrderID.Eq(order.ID), tx.OrderCoupons.UserCouponID.Eq(order.CouponID)).Scan(&oc)
// 执行原子回退:增加余额 + 重置状态 + 清除占用订单
res := tx.UserCoupons.WithContext(ctx).UnderlyingDB().Exec(`
UPDATE user_coupons
SET balance_amount = balance_amount + ?,
status = 1,
used_order_id = 0,
used_at = NULL
WHERE id = ? AND used_order_id = ?
`, oc.AppliedAmount, order.CouponID, order.ID)
if refundExists == 0 {
var oc struct {
AppliedAmount int64
}
// 优先从 order_coupons 获取实际抵扣金额
if err := tx.OrderCoupons.WithContext(ctx).Where(
tx.OrderCoupons.OrderID.Eq(order.ID),
tx.OrderCoupons.UserCouponID.Eq(order.CouponID),
).Scan(&oc); err != nil {
return err
}
if res.RowsAffected > 0 {
// 记录退还流水
_ = tx.UserCouponLedger.WithContext(ctx).Create(&model.UserCouponLedger{
UserID: userID,
UserCouponID: order.CouponID,
ChangeAmount: oc.AppliedAmount,
OrderID: order.ID,
Action: "cancel_refund",
CreatedAt: time.Now(),
})
// 兜底order_coupons 无记录时,从流水中回推预扣金额
if oc.AppliedAmount <= 0 {
if err := tx.UserCouponLedger.WithContext(ctx).UnderlyingDB().Raw(`
SELECT COALESCE(SUM(CASE WHEN change_amount < 0 THEN -change_amount ELSE 0 END), 0) AS applied_amount
FROM user_coupon_ledger
WHERE user_id = ? AND user_coupon_id = ? AND order_id = ? AND action IN ('reserve', 'usage')
`, userID, order.CouponID, order.ID).Scan(&oc).Error; err != nil {
return err
}
}
if oc.AppliedAmount > 0 {
// 恢复余额 + 重置状态(不依赖 used_order_id
res := tx.UserCoupons.WithContext(ctx).UnderlyingDB().Exec(`
UPDATE user_coupons
SET balance_amount = balance_amount + ?,
status = 1,
used_order_id = 0,
used_at = NULL
WHERE id = ?
`, oc.AppliedAmount, order.CouponID)
if res.Error != nil {
return res.Error
}
if res.RowsAffected > 0 {
if err := tx.UserCouponLedger.WithContext(ctx).Create(&model.UserCouponLedger{
UserID: userID,
UserCouponID: order.CouponID,
ChangeAmount: oc.AppliedAmount,
OrderID: order.ID,
Action: "cancel_refund",
CreatedAt: time.Now(),
}); err != nil {
return err
}
}
}
}
}