debug:设备充值成功,但订单未支付成功。

查询日志后发现,是由于死锁导致的数据库异常。
This commit is contained in:
墨大叔 2024-09-21 16:51:18 +08:00
parent 8b746cca1c
commit ddb0b75468
6 changed files with 75 additions and 42 deletions

View File

@ -22,7 +22,8 @@ public enum RedisLockKey {
ADD_TIME_BILL("add_time_bill", "创建时长订单"), ADD_TIME_BILL("add_time_bill", "创建时长订单"),
PAY_BILL("pay_bill", "支付订单"), PAY_BILL("pay_bill", "支付订单"),
PREPAY_DEPOSIT("prepay_deposit", "支付押金"), PREPAY_DEPOSIT("prepay_deposit", "支付押金"),
ADD_RECHARGE_ORDER("add_recharge_order", "创建充值订单"); ADD_RECHARGE_ORDER("add_recharge_order", "创建充值订单"),
PAY_BILL_SUCCESS("pay_bill_success", "支付订单成功处理");
private final String key; private final String key;

View File

@ -85,7 +85,7 @@ public class HttpUtils
{ {
result.append(line); result.append(line);
} }
log.debug("recv - {}", result); // log.debug("recv - {}", result);
} }
catch (ConnectException e) catch (ConnectException e)
{ {
@ -153,7 +153,7 @@ public class HttpUtils
{ {
result.append(line); result.append(line);
} }
log.debug("recv - {}", result); // log.debug("recv - {}", result);
} }
catch (ConnectException e) catch (ConnectException e)
{ {
@ -224,7 +224,7 @@ public class HttpUtils
result.append(new String(ret.getBytes(StandardCharsets.ISO_8859_1), StandardCharsets.UTF_8)); result.append(new String(ret.getBytes(StandardCharsets.ISO_8859_1), StandardCharsets.UTF_8));
} }
} }
log.debug("recv - {}", result); // log.debug("recv - {}", result);
conn.disconnect(); conn.disconnect();
br.close(); br.close();
} }
@ -329,7 +329,7 @@ public class HttpUtils
{ {
result.append(line); result.append(line);
} }
log.debug("recv - {}", result); // log.debug("recv - {}", result);
} }
catch (ConnectException e) catch (ConnectException e)
{ {
@ -396,7 +396,7 @@ public class HttpUtils
{ {
result.append(line); result.append(line);
} }
log.debug("recv - {}", result); // log.debug("recv - {}", result);
} }
catch (ConnectException e) catch (ConnectException e)
{ {

View File

@ -105,10 +105,10 @@ public class IotServiceImpl implements IotService {
String sendUrl = iotHost + IotConstants.ADDS_HISTORY_DATAPOINTS + "?"+param; String sendUrl = iotHost + IotConstants.ADDS_HISTORY_DATAPOINTS + "?"+param;
String token = Token.getToken(); String token = Token.getToken();
log.info("IOT获取到Authorization:【{}】",token); // log.info("IOT获取到Authorization:【{}】",token);
String result = HttpUtils.sendGetWithToken(sendUrl, null, token); String result = HttpUtils.sendGetWithToken(sendUrl, null, token);
log.info("IOT返回的结果【{}】",result); // log.info("IOT返回的结果【{}】",result);
if (!StringUtils.hasText(result)) { if (!StringUtils.hasText(result)) {
log.error("与OneNet通信异常"); log.error("与OneNet通信异常");
return null; return null;
@ -159,7 +159,7 @@ public class IotServiceImpl implements IotService {
String param = "device_name=" + deviceName + "&product_id=" + productId; String param = "device_name=" + deviceName + "&product_id=" + productId;
String token = Token.getToken(); String token = Token.getToken();
log.info("IOT获取到Authorization:【{}】",token); // log.info("IOT获取到Authorization:【{}】",token);
String result = HttpUtils.sendGetWithToken(sendUrl, param, token); String result = HttpUtils.sendGetWithToken(sendUrl, param, token);
if (!StringUtils.hasText(result)) { if (!StringUtils.hasText(result)) {

View File

@ -277,35 +277,41 @@ public class PayBillServiceImpl implements PayBillService
private void handleSuccess(PayBillVO payBill, LocalDateTime payTime) { private void handleSuccess(PayBillVO payBill, LocalDateTime payTime) {
ServiceUtil.assertion(payBill == null, "支付订单不存在"); ServiceUtil.assertion(payBill == null, "支付订单不存在");
log.info("获取支付订单:{}", payBill.getPayNo()); Long lockKey = payBill.getPayId();
// 若已经支付成功则跳过
if (PayBillStatus.payedList().contains(payBill.getStatus())) {
return;
}
ServiceUtil.assertion(!PayBillStatus.PAYING.getStatus().equals(payBill.getStatus()), "该支付订单不是正在支付的支付订单");
transactionTemplate.execute(status -> { ServiceUtil.assertion(redisLock.lock(RedisLockKey.PAY_BILL_SUCCESS, lockKey), "支付订单正在处理中:payId=" + payBill.getPayId());
// 修改支付订单状态 try {
PayBill data = new PayBill(); // 若已经支付成功则跳过
data.setStatus(PayBillStatus.PAY_SUCCESS.getStatus()); if (PayBillStatus.payedList().contains(payBill.getStatus())) {
data.setPayTime(payTime); return;
PayBillQuery query = new PayBillQuery();
query.setStatus(PayBillStatus.PAYING.getStatus());
query.setPayId(payBill.getPayId());
int update = this.updateByQuery(data, query);
ServiceUtil.assertion(update != 1, "支付订单状态已改变,请稍后再试");
// 处理业务
PayBillBstType bstType = PayBillBstType.parse(payBill.getBstType());
if (bstType != null && bstType.getAfterPay() != null) {
PayBillVO newPayBill = selectPayBillByPayId(payBill.getPayId());
AfterPay afterPay = SpringUtils.getBean(bstType.getAfterPay());
int bstResult = afterPay.onPaySuccess(newPayBill);
ServiceUtil.assertion(bstResult == 0, "业务处理失败");
} }
ServiceUtil.assertion(!PayBillStatus.PAYING.getStatus().equals(payBill.getStatus()), "该支付订单不是正在支付的支付订单");
return update; transactionTemplate.execute(status -> {
}); // 修改支付订单状态
PayBill data = new PayBill();
data.setStatus(PayBillStatus.PAY_SUCCESS.getStatus());
data.setPayTime(payTime);
PayBillQuery query = new PayBillQuery();
query.setStatus(PayBillStatus.PAYING.getStatus());
query.setPayId(payBill.getPayId());
int update = this.updateByQuery(data, query);
ServiceUtil.assertion(update != 1, "支付订单状态已改变,请稍后再试");
// 处理业务
PayBillBstType bstType = PayBillBstType.parse(payBill.getBstType());
if (bstType != null && bstType.getAfterPay() != null) {
PayBillVO newPayBill = selectPayBillByPayId(payBill.getPayId());
AfterPay afterPay = SpringUtils.getBean(bstType.getAfterPay());
int bstResult = afterPay.onPaySuccess(newPayBill);
ServiceUtil.assertion(bstResult == 0, "业务处理失败");
}
return update;
});
} finally {
redisLock.unlock(RedisLockKey.PAY_BILL_SUCCESS, lockKey);
}
} }
@Override @Override

View File

@ -723,13 +723,18 @@ public class TransactionBillServiceImpl implements TransactionBillService, After
TransactionBillVO bill = transactionBillMapper.selectSmTransactionBillByBillId(billId); TransactionBillVO bill = transactionBillMapper.selectSmTransactionBillByBillId(billId);
ServiceUtil.assertion(bill == null || !TransactionBillType.RECHARGE.getType().equals(bill.getType()), "不存在的充值订单"); ServiceUtil.assertion(bill == null || !TransactionBillType.RECHARGE.getType().equals(bill.getType()), "不存在的充值订单");
ServiceUtil.assertion(!TransactionBillStatus.SUCCESS.getStatus().equals(bill.getStatus()), "订单未支付"); ServiceUtil.assertion(!TransactionBillStatus.SUCCESS.getStatus().equals(bill.getStatus()), "订单未支付");
ServiceUtil.assertion(TransactionBillDeviceRechargeStatus.SUCCESS.getStatus().equals(bill.getDeviceRechargeStatus()), "设备已充值成功,不允许再次充值");
if (TransactionBillDeviceRechargeStatus.SUCCESS.getStatus().equals(bill.getDeviceRechargeStatus())) {
log.warn( "设备已充值成功,不允许再次充值:billId={}", billId);
return false;
}
ServiceUtil.assertion(TransactionBillDeviceRechargeStatus.BLUETOOTH.getStatus().equals(bill.getDeviceRechargeStatus()), "设备已选择蓝牙充值,请使用蓝牙进行充值"); ServiceUtil.assertion(TransactionBillDeviceRechargeStatus.BLUETOOTH.getStatus().equals(bill.getDeviceRechargeStatus()), "设备已选择蓝牙充值,请使用蓝牙进行充值");
// 刷新设备数据 Boolean result = transactionTemplate.execute(status -> {
deviceService.pullDeviceInfo(bill.getDeviceId()); // 刷新设备数据
deviceService.pullDeviceInfo(bill.getDeviceId());
Boolean result = (Boolean) transactionTemplate.execute(status -> {
DeviceVO device = deviceService.selectSmDeviceByDeviceId(bill.getDeviceId()); DeviceVO device = deviceService.selectSmDeviceByDeviceId(bill.getDeviceId());
// 如果设备离线则直接返回失败 // 如果设备离线则直接返回失败
@ -743,7 +748,7 @@ public class TransactionBillServiceImpl implements TransactionBillService, After
// 修改设备充值状态成功 // 修改设备充值状态成功
int updateRecharge = transactionBillMapper.updateDeviceRechargeStatus(bill.getBillId(), TransactionBillDeviceRechargeStatus.SUCCESS.getStatus()); int updateRecharge = transactionBillMapper.updateDeviceRechargeStatus(bill.getBillId(), TransactionBillDeviceRechargeStatus.SUCCESS.getStatus());
ServiceUtil.assertion(updateRecharge != 1, "订单状态发生变化,请稍后重试"); ServiceUtil.assertion(updateRecharge != 1, "设备充值状态发生变化,请稍后重试");
try { try {
if (SuitFeeType.TIME.getType().equals(bill.getSuitFeeType())) { if (SuitFeeType.TIME.getType().equals(bill.getSuitFeeType())) {
@ -849,14 +854,33 @@ public class TransactionBillServiceImpl implements TransactionBillService, After
return updateCount; return updateCount;
}); });
// 异步充值设备尝试3次
if (result != null && result == 1) { if (result != null && result == 1) {
// 设备充值 this.tryRechargeDevice(bill.getBillId(), 3);
rechargeDevice(bill.getBillId());
} }
return result == null ? 0 : result; return result == null ? 0 : result;
} }
/**
* 异步尝试充值设备
* @param billId
* @param tryCount
*/
private void tryRechargeDevice(Long billId, int tryCount) {
if (tryCount <= 0) {
return;
}
scheduledExecutorService.schedule(()-> {
try {
boolean result = rechargeDevice(billId);
ServiceUtil.assertion(!result, String.format("尝试充值设备失败:billId=%s:剩余次数:%s,", billId, tryCount - 1));
} catch (Exception e) {
this.tryRechargeDevice(billId, tryCount - 1);
}
}, 0, TimeUnit.SECONDS);
}
@Override @Override
public DoPayVO payDeposit(RechargePayDepositBO bo) { public DoPayVO payDeposit(RechargePayDepositBO bo) {
// 校验 // 校验

View File

@ -272,6 +272,7 @@ public class AppTransactionBillController extends BaseController
@ApiOperation("支付押金") @ApiOperation("支付押金")
@PutMapping("/payDeposit") @PutMapping("/payDeposit")
public AjaxResult payDeposit(@RequestBody @Validated PayDepositDTO dto) { public AjaxResult payDeposit(@RequestBody @Validated PayDepositDTO dto) {
dto.setChannelId(1L);
return success(transactionBillService.payDeposit(transactionBillConverter.toRechargePayDepositBO(dto))); return success(transactionBillService.payDeposit(transactionBillConverter.toRechargePayDepositBO(dto)));
} }
@ -344,6 +345,7 @@ public class AppTransactionBillController extends BaseController
@ApiOperation("支付订单") @ApiOperation("支付订单")
@PutMapping("/pay") @PutMapping("/pay")
public AjaxResult pay(@RequestBody BillPayDTO dto) { public AjaxResult pay(@RequestBody BillPayDTO dto) {
dto.setChannelId(1L);
TransactionBillVO bill = transactionBillService.selectSmTransactionBillByBillNo(dto.getBillNo()); TransactionBillVO bill = transactionBillService.selectSmTransactionBillByBillNo(dto.getBillNo());
if (!transactionBillValidator.isUser(bill, getUserId())) { if (!transactionBillValidator.isUser(bill, getUserId())) {
return error("这不是您的订单,无法支付"); return error("这不是您的订单,无法支付");