订单支付异常情况处理

正常情况下我们选择商品然后购买,完成支付后就发货了,但是因为大多采用的是http协议,如果出现了网络异常等情况就会掉单之类的,比如顾客付完钱了但是订单状态并没有改变,还有就是支付成功后回调失败,导致没有发货。对于真正的服务,这些异常情况我们也得考虑进去。

上面基本就是订单服务的主流程了,但是我们可以思考一下异常流程:

  1. 用户订单创建成功,但创建支付单 HTTP 超时失败。
  2. 支付回调时,系统宕机或者本身服务出问题。
  3. 支付成功后发送MQ消息,消息丢失,用户支付掉单。
  4. 长时间未支付,超时订单。

那么,这些就都是可能出现的异常流程。虽然概率很低,但随着使用规模的增加,很低概率的问题,也会产生较大规模的客诉问题。所以要针对这些流程做补偿处理。 

  1. 针对1~4提到异常流程,一条支付链路就会被扩展为现在的样子,在各个流程中需要穿插进入异常补偿流程。
  2. 用户下单,但可能存在之前下的残单,那么就要对应给予补充的流程后,再返回回去。
  3. 支付回调,仍然可能有异常。所以要有掉单补偿和发货补偿。两条任务处理。

 

因此,我们通过使用定时任务完成补偿

 1. 掉单补偿,检测未接收到或未正确处理的支付回调通知(即用户已经扫码支付成功但是订单状态没有改变)

@Component
@Slf4j
public class NoPayNotifyOrderJob {

    @Resource
    private IOrderRepository orderRepository;

    @Resource
    private IOrderService orderService;

    @Resource
    private RabbitTemplate rabbitTemplate;


    @Value("${pay.alipay.APP_ID}")
    private String APP_ID;

    @Value("${pay.alipay.APP_PRIVATE_KEY}")
    private String APP_PRIVATE_KEY;

    @Value("${pay.alipay.ALIPAY_PUBLIC_KEY}")
    private String ALIPAY_PUBLIC_KEY;

    private final SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ssXXX");

    @Timed(value = "no_pay_notify_order_job", description = "定时任务,订单支付状态更新")
    @Scheduled(cron = "0/3 * * * * ?")
    public void exec(){
        try {
            List<String> orderIds = orderRepository.queryNoPayNotifyOrder();
            if (orderIds.isEmpty()) {
                log.info("定时任务,订单支付状态更新,暂无未更新订单 orderId is null");
                return;
            }

            for (String orderId: orderIds) {
                AlipayClient alipayClient = new DefaultAlipayClient(AlipayConfig.URL,
                        APP_ID,
                        APP_PRIVATE_KEY,
                        AlipayConfig.FORMAT,
                        AlipayConfig.CHARSET,
                        ALIPAY_PUBLIC_KEY,
                        AlipayConfig.SIGNTYPE);

                AlipayTradeQueryRequest request = new AlipayTradeQueryRequest();
                JSONObject bizContent = new JSONObject();
                bizContent.put("out_trade_no", orderId);
                request.setBizContent(bizContent.toString());
                AlipayTradeQueryResponse response = null;
                try {
                    response = alipayClient.execute(request);
                    if (!response.isSuccess()) {
                        new ChatGPTException("请求支付查询查询失败");
                    }
                } catch (AlipayApiException e) {
                    log.error("请求支付宝查询支付结果异常:{}", e.toString(), e);
                    new ChatGPTException("请求支付查询查询失败");
                }

                //获取支付结果
                String resultJson = response.getBody();
                //转map
                Map resultMap = JSON.parseObject(resultJson, Map.class);
                Map alipay_trade_query_response = (Map) resultMap.get("alipay_trade_query_response");
                //支付结果
                Double total_amount = Double.valueOf(alipay_trade_query_response.get("total_amount").toString());
                String trade_no = (String) alipay_trade_query_response.get("trade_no");
                String successTime = (String)alipay_trade_query_response.get("gmt_payment");

                boolean isSuccess = orderService.changeOrderPaySuccess(orderId, trade_no, BigDecimal.valueOf(total_amount), dateFormat.parse(successTime));

                if (isSuccess) {
                    // 发布消息
                    rabbitTemplate.convertAndSend(PayNotifyConfig.PAYNOTIFY_EXCHANGE_FANOUT,"", orderId);
                }
            }
        }catch (Exception e){
            log.error("定时任务,订单支付状态更新失败", e);
            e.printStackTrace();
        }
    }

}

2.订单补货任务

@Component
@Slf4j
public class OrderReplenishmentJob {

    @Resource
    private IOrderService orderService;

    @Resource
    private RabbitTemplate rabbitTemplate;


    /**
     * 执行订单补货,超时3分钟,已支付,待发货未发货的订单
     */
    @Scheduled(cron = "0 0/3 * * * ?")
    public void exec() {
        try {
            List<String> orderIds = orderService.queryReplenishmentOrder();
            if (orderIds.isEmpty()) {
                log.info("定时任务,订单补货不存在,查询 orderIds is null");
                return;
            }
            for (String orderId : orderIds) {
                log.info("定时任务,订单补货开始。orderId: {}", orderId);
                rabbitTemplate.convertAndSend(PayNotifyConfig.PAYNOTIFY_EXCHANGE_FANOUT,"", orderId);
            }
        } catch (Exception e) {
            log.error("定时任务,订单补货失败。", e);
        }
    }
}

3. 超时关单任务

@Component
@Slf4j
public class TimeoutCloseOrderJob {

    @Resource
    private IOrderService orderService;

    @Scheduled(cron = "0 0/30 * * * ?")
    public void exec() {
       try {
           List<String> orderIds = orderService.queryTimeoutCloseOrderList();
           for (String orderId: orderIds) {
               boolean status = orderService.changeOrderClose(orderId);
               log.info("定时任务,超时30分钟订单关闭 orderId: {} status:{}", orderId, status);
           }
       }catch (Exception e) {
           log.error("定时任务,超时15分钟订单关闭失败", e);
       }
    }
}

  • 5
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值