|
@@ -3,10 +3,13 @@ package com.atguigu.tingshu.order.service.impl;
|
|
import cn.hutool.core.bean.BeanUtil;
|
|
import cn.hutool.core.bean.BeanUtil;
|
|
import cn.hutool.core.collection.CollUtil;
|
|
import cn.hutool.core.collection.CollUtil;
|
|
import cn.hutool.core.date.DateUtil;
|
|
import cn.hutool.core.date.DateUtil;
|
|
|
|
+import cn.hutool.core.lang.Assert;
|
|
import com.atguigu.tingshu.account.AccountFeignClient;
|
|
import com.atguigu.tingshu.account.AccountFeignClient;
|
|
import com.atguigu.tingshu.album.AlbumFeignClient;
|
|
import com.atguigu.tingshu.album.AlbumFeignClient;
|
|
import com.atguigu.tingshu.common.constant.RedisConstant;
|
|
import com.atguigu.tingshu.common.constant.RedisConstant;
|
|
import com.atguigu.tingshu.common.execption.GuiguException;
|
|
import com.atguigu.tingshu.common.execption.GuiguException;
|
|
|
|
+import com.atguigu.tingshu.common.rabbit.constant.MqConst;
|
|
|
|
+import com.atguigu.tingshu.common.rabbit.service.RabbitService;
|
|
import com.atguigu.tingshu.common.result.Result;
|
|
import com.atguigu.tingshu.common.result.Result;
|
|
import com.atguigu.tingshu.common.util.AuthContextHolder;
|
|
import com.atguigu.tingshu.common.util.AuthContextHolder;
|
|
import com.atguigu.tingshu.model.order.OrderDerate;
|
|
import com.atguigu.tingshu.model.order.OrderDerate;
|
|
@@ -27,11 +30,15 @@ import com.atguigu.tingshu.vo.order.OrderDetailVo;
|
|
import com.atguigu.tingshu.vo.order.OrderInfoVo;
|
|
import com.atguigu.tingshu.vo.order.OrderInfoVo;
|
|
import com.atguigu.tingshu.vo.order.TradeVo;
|
|
import com.atguigu.tingshu.vo.order.TradeVo;
|
|
import com.atguigu.tingshu.vo.user.UserPaidRecordVo;
|
|
import com.atguigu.tingshu.vo.user.UserPaidRecordVo;
|
|
|
|
+import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
|
|
import com.baomidou.mybatisplus.core.toolkit.IdWorker;
|
|
import com.baomidou.mybatisplus.core.toolkit.IdWorker;
|
|
|
|
+import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
|
|
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
|
|
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
|
|
import io.seata.spring.annotation.GlobalTransactional;
|
|
import io.seata.spring.annotation.GlobalTransactional;
|
|
import lombok.extern.slf4j.Slf4j;
|
|
import lombok.extern.slf4j.Slf4j;
|
|
import org.springframework.beans.factory.annotation.Autowired;
|
|
import org.springframework.beans.factory.annotation.Autowired;
|
|
|
|
+import org.springframework.beans.factory.annotation.Value;
|
|
|
|
+import org.springframework.cloud.context.config.annotation.RefreshScope;
|
|
import org.springframework.data.redis.core.RedisTemplate;
|
|
import org.springframework.data.redis.core.RedisTemplate;
|
|
import org.springframework.data.redis.core.script.DefaultRedisScript;
|
|
import org.springframework.data.redis.core.script.DefaultRedisScript;
|
|
import org.springframework.stereotype.Service;
|
|
import org.springframework.stereotype.Service;
|
|
@@ -45,6 +52,7 @@ import static com.atguigu.tingshu.common.constant.SystemConstant.*;
|
|
|
|
|
|
@Slf4j
|
|
@Slf4j
|
|
@Service
|
|
@Service
|
|
|
|
+@RefreshScope //实现Value注解读取实时刷新
|
|
@SuppressWarnings({"all"})
|
|
@SuppressWarnings({"all"})
|
|
public class OrderInfoServiceImpl extends ServiceImpl<OrderInfoMapper, OrderInfo> implements OrderInfoService {
|
|
public class OrderInfoServiceImpl extends ServiceImpl<OrderInfoMapper, OrderInfo> implements OrderInfoService {
|
|
|
|
|
|
@@ -76,6 +84,17 @@ public class OrderInfoServiceImpl extends ServiceImpl<OrderInfoMapper, OrderInfo
|
|
private TradeOrderStrategyFactory tradeOrderStrategyFactory;
|
|
private TradeOrderStrategyFactory tradeOrderStrategyFactory;
|
|
|
|
|
|
|
|
|
|
|
|
+ @Autowired
|
|
|
|
+ private RabbitService rabbitService;
|
|
|
|
+
|
|
|
|
+
|
|
|
|
+ /**
|
|
|
|
+ * 对购买商品(VIP会员、专辑、声音)封装订单结算页所需要数据
|
|
|
|
+ *
|
|
|
|
+ * @param userId 用户ID
|
|
|
|
+ * @param tradeVo {购买商品类别,商品ID,声音数量}
|
|
|
|
+ * @return
|
|
|
|
+ */
|
|
@Override
|
|
@Override
|
|
public OrderInfoVo trade(Long userId, TradeVo tradeVo) {
|
|
public OrderInfoVo trade(Long userId, TradeVo tradeVo) {
|
|
//1.创建订单VO
|
|
//1.创建订单VO
|
|
@@ -87,6 +106,7 @@ public class OrderInfoServiceImpl extends ServiceImpl<OrderInfoMapper, OrderInfo
|
|
tradeOrderStrategy.handleTrade(tradeVo, orderInfoVo);
|
|
tradeOrderStrategy.handleTrade(tradeVo, orderInfoVo);
|
|
return orderInfoVo;
|
|
return orderInfoVo;
|
|
}
|
|
}
|
|
|
|
+
|
|
/**
|
|
/**
|
|
* 对购买商品(VIP会员、专辑、声音)封装订单结算页所需要数据
|
|
* 对购买商品(VIP会员、专辑、声音)封装订单结算页所需要数据
|
|
*
|
|
*
|
|
@@ -249,6 +269,9 @@ public class OrderInfoServiceImpl extends ServiceImpl<OrderInfoMapper, OrderInfo
|
|
return orderInfoVo;
|
|
return orderInfoVo;
|
|
}*/
|
|
}*/
|
|
|
|
|
|
|
|
+ @Value("${order.cancel}")
|
|
|
|
+ private Integer cancelTTl;
|
|
|
|
+
|
|
/**
|
|
/**
|
|
* 提交订单
|
|
* 提交订单
|
|
*
|
|
*
|
|
@@ -327,15 +350,11 @@ public class OrderInfoServiceImpl extends ServiceImpl<OrderInfoMapper, OrderInfo
|
|
throw new GuiguException(result.getCode(), result.getMessage());
|
|
throw new GuiguException(result.getCode(), result.getMessage());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
-
|
|
|
|
- //模拟异常
|
|
|
|
- //if (orderInfo.getId() % 2 == 0) {
|
|
|
|
- // int i = 1 / 0;
|
|
|
|
- //}
|
|
|
|
}
|
|
}
|
|
if (ORDER_PAY_WAY_WEIXIN.equals(payWay)) {
|
|
if (ORDER_PAY_WAY_WEIXIN.equals(payWay)) {
|
|
//5. TODO 处理付款方式为:微信支付 采用延迟消息进行延迟关单
|
|
//5. TODO 处理付款方式为:微信支付 采用延迟消息进行延迟关单
|
|
-
|
|
|
|
|
|
+ //rabbitService.sendDelayMessage(MqConst.EXCHANGE_CANCEL_ORDER, MqConst.ROUTING_CANCEL_ORDER, orderInfo.getId(), MqConst.CANCEL_ORDER_DELAY_TIME);
|
|
|
|
+ rabbitService.sendDelayMessage(MqConst.EXCHANGE_CANCEL_ORDER, MqConst.ROUTING_CANCEL_ORDER, orderInfo.getId(), cancelTTl);
|
|
}
|
|
}
|
|
|
|
|
|
return orderInfo.getOrderNo();
|
|
return orderInfo.getOrderNo();
|
|
@@ -393,7 +412,7 @@ public class OrderInfoServiceImpl extends ServiceImpl<OrderInfoMapper, OrderInfo
|
|
}
|
|
}
|
|
//3.保存订单优惠明细列表
|
|
//3.保存订单优惠明细列表
|
|
List<OrderDerateVo> orderDerateVoList = orderInfoVo.getOrderDerateVoList();
|
|
List<OrderDerateVo> orderDerateVoList = orderInfoVo.getOrderDerateVoList();
|
|
- if (CollUtil.isNotEmpty(orderDetailVoList)) {
|
|
|
|
|
|
+ if (CollUtil.isNotEmpty(orderDerateVoList)) {
|
|
orderDerateVoList
|
|
orderDerateVoList
|
|
.stream()
|
|
.stream()
|
|
.forEach(orderDerateVo -> {
|
|
.forEach(orderDerateVo -> {
|
|
@@ -405,4 +424,93 @@ public class OrderInfoServiceImpl extends ServiceImpl<OrderInfoMapper, OrderInfo
|
|
return orderInfo;
|
|
return orderInfo;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+ /**
|
|
|
|
+ * 根据订单编号查询订单信息(包含订单明细,优惠列表)
|
|
|
|
+ *
|
|
|
|
+ * @param orderNo
|
|
|
|
+ * @return
|
|
|
|
+ */
|
|
|
|
+ @Override
|
|
|
|
+ public OrderInfo getOrderInfo(String orderNo) {
|
|
|
|
+ //1.根据订单编号查询订单信息
|
|
|
|
+ OrderInfo orderInfo = baseMapper.selectOne(
|
|
|
|
+ new LambdaQueryWrapper<OrderInfo>()
|
|
|
|
+ .eq(OrderInfo::getOrderNo, orderNo)
|
|
|
|
+ );
|
|
|
|
+ Assert.notNull(orderInfo, "订单:{}不存在", orderNo);
|
|
|
|
+ //2.根据订单ID查询订单明细列表
|
|
|
|
+ List<OrderDetail> orderDetailList = orderDetailMapper.selectList(
|
|
|
|
+ new LambdaQueryWrapper<OrderDetail>()
|
|
|
|
+ .eq(OrderDetail::getOrderId, orderInfo.getId())
|
|
|
|
+ );
|
|
|
|
+ if (CollUtil.isNotEmpty(orderDetailList)) {
|
|
|
|
+ orderInfo.setOrderDetailList(orderDetailList);
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+
|
|
|
|
+ //3.根据订单ID查询订单优惠列表
|
|
|
|
+ List<OrderDerate> orderDerateList = orderDerateMapper.selectList(
|
|
|
|
+ new LambdaQueryWrapper<OrderDerate>()
|
|
|
|
+ .eq(OrderDerate::getOrderId, orderInfo.getId())
|
|
|
|
+ );
|
|
|
|
+ if (CollUtil.isNotEmpty(orderDerateList)) {
|
|
|
|
+ orderInfo.setOrderDerateList(orderDerateList);
|
|
|
|
+ }
|
|
|
|
+ return orderInfo;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ /**
|
|
|
|
+ * 查询指定用户订单列表
|
|
|
|
+ *
|
|
|
|
+ * @param pageInfo
|
|
|
|
+ * @param userId
|
|
|
|
+ * @return
|
|
|
|
+ */
|
|
|
|
+ @Override
|
|
|
|
+ public Page<OrderInfo> findUserOrderPage(Page<OrderInfo> pageInfo, Long userId) {
|
|
|
|
+ //1.方式1:采用queryWrapper分页查询订单,遍历订单在查询订单明细
|
|
|
|
+ //2.方式2:采用自定义SQL分页查询订单,同时查询订单明细
|
|
|
|
+ //pageInfo = baseMapper.findUserOrderPage(pageInfo, userId);
|
|
|
|
+ //3.方式3:采用queryWrapper分页查询订单,获取订单ID列表根据订单ID列表查询订单详情,业务逻辑中处理订单明细
|
|
|
|
+ //3.1 分页查询订单
|
|
|
|
+ LambdaQueryWrapper<OrderInfo> queryWrapper = new LambdaQueryWrapper<>();
|
|
|
|
+ queryWrapper.eq(OrderInfo::getUserId, userId);
|
|
|
|
+ queryWrapper.orderByDesc(OrderInfo::getId);
|
|
|
|
+ pageInfo = this.page(pageInfo, queryWrapper);
|
|
|
|
+
|
|
|
|
+ //3.2 获取订单ID列表
|
|
|
|
+ List<OrderInfo> records = pageInfo.getRecords();
|
|
|
|
+ if (CollUtil.isNotEmpty(records)) {
|
|
|
|
+ List<Long> orderIdList = records.stream().map(OrderInfo::getId).collect(Collectors.toList());
|
|
|
|
+ List<OrderDetail> orderDetailList = orderDetailMapper.selectList(
|
|
|
|
+ new LambdaQueryWrapper<OrderDetail>()
|
|
|
|
+ .in(OrderDetail::getOrderId, orderIdList)
|
|
|
|
+ );
|
|
|
|
+ //3.3 采用Stream对查询到订单明细列表按照订单ID进行分组 Map中Key为订单ID Map中值为订单明细列表
|
|
|
|
+ Map<Long, List<OrderDetail>> collect = orderDetailList.stream().collect(Collectors.groupingBy(OrderDetail::getOrderId));
|
|
|
|
+ //4.遍历订单找出订单下包含订单明细
|
|
|
|
+ records.stream().forEach(orderInfo -> {
|
|
|
|
+ orderInfo.setOrderDetailList(collect.get(orderInfo.getId()));
|
|
|
|
+ });
|
|
|
|
+ }
|
|
|
|
+ return pageInfo;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ /**
|
|
|
|
+ * 根据订单ID检查订单支付状态,如果未支付则更新订单状态为已取消
|
|
|
|
+ *
|
|
|
|
+ * @param orderId
|
|
|
|
+ */
|
|
|
|
+ @Override
|
|
|
|
+ public void cancelOrder(Long orderId) {
|
|
|
|
+ //1.根据订单ID查询订单信息
|
|
|
|
+ OrderInfo orderInfo = baseMapper.selectById(orderId);
|
|
|
|
+ //2.判断订单状态:如果未支付则更新订单状态为已取消
|
|
|
|
+ if (orderInfo != null && ORDER_STATUS_UNPAID.equals(orderInfo.getOrderStatus())) {
|
|
|
|
+ orderInfo.setOrderStatus(ORDER_STATUS_CANCEL);
|
|
|
|
+ baseMapper.updateById(orderInfo);
|
|
|
|
+ //TODO 远程调用支付服务,修改本地交易记录状态:已关闭
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
}
|
|
}
|