| | |
| | | import com.doumee.core.utils.DateUtil; |
| | | import com.doumee.core.utils.geocode.MapUtil; |
| | | import com.doumee.core.utils.Utils; |
| | | import com.doumee.core.utils.aliyun.AliSmsService; |
| | | import com.doumee.dao.business.*; |
| | | import com.doumee.dao.business.model.*; |
| | | import com.doumee.dao.system.SystemUserMapper; |
| | |
| | | import com.github.xiaoymin.knife4j.core.util.CollectionUtils; |
| | | import com.github.yulichang.wrapper.MPJLambdaWrapper; |
| | | import com.alibaba.fastjson.JSONObject; |
| | | import lombok.extern.slf4j.Slf4j; |
| | | import org.apache.commons.lang3.StringUtils; |
| | | import org.springframework.beans.factory.annotation.Autowired; |
| | | import org.springframework.data.redis.core.RedisTemplate; |
| | |
| | | * @author rk |
| | | * @date 2026/04/10 |
| | | */ |
| | | @Slf4j |
| | | @Service |
| | | public class OrdersServiceImpl implements OrdersService { |
| | | |
| | |
| | | |
| | | @Autowired |
| | | private NoticeService noticeService; |
| | | |
| | | @Autowired |
| | | private SmsrecordMapper smsrecordMapper; |
| | | |
| | | @Autowired |
| | | private WxPayV3Service wxPayV3Service; |
| | |
| | | return BigDecimal.ZERO; |
| | | } |
| | | String rateStr = systemDictDataBiz.queryByCode(Constants.OPERATION_CONFIG, Constants.OP_INSURANCE_RATE).getCode(); |
| | | BigDecimal rate = new BigDecimal(rateStr); |
| | | BigDecimal rate = new BigDecimal(rateStr).divide(new BigDecimal("100"), 4, BigDecimal.ROUND_HALF_UP); |
| | | return declaredValue.multiply(rate).setScale(2, BigDecimal.ROUND_HALF_UP); |
| | | } |
| | | |
| | |
| | | List<Category> categories = categoryMapper.selectBatchIds(categoryIds); |
| | | Map<Integer, String> categoryNameMap = new HashMap<>(); |
| | | Map<Integer, String> categoryDetailMap = new HashMap<>(); |
| | | Map<Integer, String> categoryOtherFieldMap = new HashMap<>(); |
| | | for (Category c : categories) { |
| | | categoryNameMap.put(c.getId(), c.getName()); |
| | | categoryDetailMap.put(c.getId(), c.getDetail()); |
| | | categoryOtherFieldMap.put(c.getId(),c.getOtherField()); |
| | | } |
| | | |
| | | // 计算每项物品费用:小计 = 单价 × 数量 × 天数 |
| | |
| | | ItemPriceVO vo = new ItemPriceVO(); |
| | | vo.setCategoryId(item.getCategoryId()); |
| | | vo.setCategoryName(categoryNameMap.getOrDefault(item.getCategoryId(), "")); |
| | | vo.setDetail(categoryDetailMap.get(item.getCategoryId())); |
| | | vo.setDetail(categoryOtherFieldMap.get(item.getCategoryId())); |
| | | vo.setQuantity(item.getQuantity()); |
| | | vo.setUnitPrice(unitPrice); |
| | | vo.setLocallyPrice(unitPrice); |
| | |
| | | String to = dto.getToLat() + "," + dto.getToLgt(); |
| | | JSONObject distanceResult = MapUtil.direction("driving", from, to); |
| | | BigDecimal distance = distanceResult.getBigDecimal("distance"); |
| | | // distance 单位为米,转为公里 |
| | | // distance 单位为米,转为公里(不足1公里按1公里计算) |
| | | BigDecimal distanceKm = distance.divide(new BigDecimal(1000), 2, RoundingMode.HALF_UP); |
| | | if (distanceKm.compareTo(BigDecimal.ONE) < 0) { |
| | | distanceKm = BigDecimal.ONE; |
| | | } |
| | | |
| | | // 收集所有物品类型ID |
| | | List<Integer> categoryIds = new ArrayList<>(); |
| | |
| | | List<Category> categories = categoryMapper.selectBatchIds(categoryIds); |
| | | Map<Integer, String> categoryNameMap = new HashMap<>(); |
| | | Map<Integer, String> categoryDetailMap = new HashMap<>(); |
| | | Map<Integer, String> categoryOtherFieldMap = new HashMap<>(); |
| | | for (Category c : categories) { |
| | | categoryNameMap.put(c.getId(), c.getName()); |
| | | categoryDetailMap.put(c.getId(), c.getDetail()); |
| | | categoryOtherFieldMap.put(c.getId(),c.getOtherField()); |
| | | } |
| | | |
| | | // 3. 逐项计算运费:起步价 + 超出部分阶梯价 |
| | |
| | | ItemPriceVO vo = new ItemPriceVO(); |
| | | vo.setCategoryId(item.getCategoryId()); |
| | | vo.setCategoryName(categoryNameMap.getOrDefault(item.getCategoryId(), "")); |
| | | vo.setDetail(categoryDetailMap.get(item.getCategoryId())); |
| | | vo.setDetail(categoryOtherFieldMap.get(item.getCategoryId())); |
| | | vo.setQuantity(item.getQuantity()); |
| | | vo.setUnitPrice(unitPrice); |
| | | vo.setLocallyPrice(locallyPrice); |
| | |
| | | |
| | | // ========== 3. 查询寄件店铺信息 ========== |
| | | ShopInfo depositShop = shopInfoMapper.selectById(dto.getDepositShopId()); |
| | | if (depositShop == null) { |
| | | if (depositShop == null || Constants.equalsInteger(depositShop.getDeleted(), Constants.ONE)) { |
| | | throw new BusinessException(ResponseStatus.BAD_REQUEST.getCode(), "寄件店铺不存在"); |
| | | } |
| | | if (depositShop.getStatus() == null || !Constants.equalsInteger(depositShop.getStatus(), Constants.ZERO)) { |
| | | throw new BusinessException(ResponseStatus.BAD_REQUEST.getCode(), "寄件店铺已停业,请选择其他门店"); |
| | | } |
| | | |
| | | // ========== 4. 计算费用 ========== |
| | |
| | | } |
| | | // 取件点:店铺 or 自选点,至少提供一组 |
| | | if (dto.getTakeShopId() != null) { |
| | | if (dto.getTakeShopId().equals(dto.getDepositShopId())) { |
| | | throw new BusinessException(ResponseStatus.BAD_REQUEST.getCode(), "异地寄存订单存件门店和取件门店不能相同"); |
| | | } |
| | | takeShop = shopInfoMapper.selectById(dto.getTakeShopId()); |
| | | if (takeShop == null) { |
| | | if (takeShop == null || Constants.equalsInteger(takeShop.getDeleted(), Constants.ONE)) { |
| | | throw new BusinessException(ResponseStatus.BAD_REQUEST.getCode(), "取件店铺不存在"); |
| | | } |
| | | if (takeShop.getStatus() == null || Constants.equalsInteger(takeShop.getStatus(), Constants.ONE)) { |
| | | throw new BusinessException(ResponseStatus.BAD_REQUEST.getCode(), "取件店铺已停业,请选择其他门店"); |
| | | } |
| | | takeLat = BigDecimal.valueOf(takeShop.getLatitude()); |
| | | takeLgt = BigDecimal.valueOf(takeShop.getLongitude()); |
| | |
| | | orders.setTakeLat(takeLat); |
| | | orders.setTakeLgt(takeLgt); |
| | | orders.setIsUrgent(dto.getIsUrgent()); |
| | | orders.setDistance(priceResult.getDistance()); |
| | | } else { |
| | | // 就地:取件点同寄件店铺 |
| | | orders.setTakeShopId(dto.getDepositShopId()); |
| | |
| | | |
| | | // 物品信息 |
| | | orders.setGoodType(dto.getGoodType()); |
| | | orders.setGoodLevel(goodTypeCategory.getRelationId()); |
| | | // 查询物品级别 type = 3 |
| | | Category levelCategory = categoryMapper.selectById(goodTypeCategory.getRelationId()); |
| | | if(Objects.isNull(levelCategory)){ |
| | | throw new BusinessException(ResponseStatus.DATA_EMPTY.getCode(),"未查询到物品等级信息:{}"+goodTypeCategory.getName()); |
| | | } |
| | | orders.setGoodLevel(Integer.valueOf(levelCategory.getDetail())); |
| | | // 拼接物品信息:物品类型名称、尺寸名称*数量(数组字符串) |
| | | List<String> goodsParts = new ArrayList<>(); |
| | | for (ItemPriceVO itemVO : priceResult.getItemList()) { |
| | |
| | | } |
| | | orders.setDeclaredFee(priceResult.getInsuranceFee()); |
| | | orders.setPrice(priceResult.getItemPrice()); |
| | | |
| | | // 薪酬计算与占比存储 |
| | | calculateAndSetFeeAllocation(orders, depositShop, takeShop); |
| | | |
| | | // 无人接单通知相关初始化 |
| | | orders.setPlatformSmsNotified(Constants.ZERO); |
| | | orders.setPlatformSmsNotifiedTime(now); |
| | | |
| | | ordersMapper.insert(orders); |
| | | Integer orderId = orders.getId(); |
| | |
| | | // 5. 重新生成第三方订单编号(避免重复) |
| | | String orderTradeNo = generateOrderTradeNo(); |
| | | orders.setOutTradeNo(orderTradeNo); |
| | | orders.setPlatformSmsNotifiedTime(new Date()); |
| | | orders.setUpdateTime(new Date()); |
| | | ordersMapper.updateById(orders); |
| | | // 6. 唤起微信支付 |
| | |
| | | // 取消/退款状态时查询退款记录 |
| | | Integer status = order.getStatus(); |
| | | if (status != null && (status == Constants.OrderStatus.overdue.getStatus() |
| | | || status == Constants.OrderStatus.closed.getStatus() |
| | | || status == Constants.OrderStatus.cancelOverdue.getStatus() |
| | | || status == Constants.OrderStatus.cancelling.getStatus() |
| | | || status == Constants.OrderStatus.cancelled.getStatus())) { |
| | | OrdersRefund ordersRefund = ordersRefundMapper.selectOne( |
| | | new QueryWrapper<OrdersRefund>().lambda() |
| | |
| | | .eq(Orders::getId, order.getId()) |
| | | .set(Orders::getIsUrgent, Constants.ONE) |
| | | .set(Orders::getPlatformRewardAmount, urgentFeeFen) |
| | | .set(Orders::getPlatformSmsNotified, Constants.ZERO) // 重置通知状态为未通知 |
| | | .set(Orders::getPlatformSmsNotifiedTime, new Date()) // 重置通知基准时间为当前 |
| | | .set(Orders::getUpdateTime, new Date()); |
| | | |
| | | // 异地寄存且有取件门店时,生成司机核销码 |
| | |
| | | item.setLuggageName(d.getLuggageName()); |
| | | item.setLuggageDetail(d.getLuggageDetail()); |
| | | item.setNum(d.getNum()); |
| | | double unitPriceYuan = d.getUnitPrice() != null ? Constants.getFormatMoney(d.getUnitPrice()) : 0; |
| | | item.setUnitPriceYuan(unitPriceYuan); |
| | | item.setSubtotal(unitPriceYuan * (d.getNum() != null ? d.getNum() : 0)); |
| | | long unitPriceFen = d.getUnitPrice() != null ? d.getUnitPrice() : 0L; |
| | | item.setUnitPrice(unitPriceFen); |
| | | item.setSubtotal(unitPriceFen * (d.getNum() != null ? d.getNum() : 0)); |
| | | items.add(item); |
| | | } |
| | | } |
| | |
| | | .eq(PricingRule::getCityId, cityId) |
| | | .eq(PricingRule::getFieldA, String.valueOf(fieldA)) |
| | | .last("limit 1")); |
| | | if (rule != null && StringUtils.isNotBlank(rule.getFieldC())) { |
| | | return new BigDecimal(rule.getFieldC()); |
| | | if (rule != null && StringUtils.isNotBlank(rule.getFieldB())) { |
| | | // fieldB 存储的是百分比整数(如15表示15%),转换为小数比例(0.15) |
| | | return new BigDecimal(rule.getFieldB()).divide(new BigDecimal("100"), 4, BigDecimal.ROUND_HALF_UP); |
| | | } |
| | | return BigDecimal.ZERO; |
| | | } |
| | |
| | | vo.setCode(o.getCode()); |
| | | vo.setType(o.getType()); |
| | | vo.setStatus(o.getStatus()); |
| | | vo.setStatusName(Constants.OrderStatus.getDescByKey(o.getStatus(), |
| | | Constants.equalsInteger(o.getType(), Constants.ZERO)?o.getType():Objects.nonNull(o.getTakeShopId())?Constants.ONE:Constants.TWO) |
| | | ); |
| | | vo.setCreateTime(o.getCreateTime()); |
| | | vo.setExpectedTakeTime(o.getExpectedTakeTime()); |
| | | vo.setMemberVerifyCode(o.getMemberVerifyCode()); |
| | | |
| | | // 存件门店(关联查询直接取值) |
| | | vo.setDepositShopId(o.getDepositShopId()); |
| | | vo.setDepositShopName(o.getDepositShopName()); |
| | | vo.setDepositShopLinkName(o.getDepositShopLinkName()); |
| | | vo.setDepositShopPhone(o.getDepositShopLinkPhone()); |
| | | |
| | | // 取件信息:有取件门店取门店,无则取用户自选取件点 |
| | | if (o.getTakeShopId() != null) { |
| | | vo.setTakeShopId(o.getTakeShopId()); |
| | | vo.setTakeShopName(o.getTakeShopName()); |
| | | vo.setTakeShopAddress(o.getTakeShopAddress()); |
| | | } else { |
| | |
| | | // 费用(分) |
| | | vo.setDeclaredFee(o.getDeclaredFee()); |
| | | vo.setEstimatedAmount(o.getEstimatedAmount()); |
| | | |
| | | // 评价状态 |
| | | vo.setCommentStatus(o.getCommentStatus()); |
| | | |
| | | // 查询物品明细(一次查询,同时用于物品列表和逾期计算) |
| | | List<OrdersDetail> details = ordersDetailMapper.selectList( |
| | |
| | | vo.setCode(o.getCode()); |
| | | vo.setType(o.getType()); |
| | | vo.setStatus(o.getStatus()); |
| | | vo.setStatusName(Constants.OrderStatus.getDescByKey(o.getStatus(), |
| | | Constants.equalsInteger(o.getType(), Constants.ZERO)?o.getType():Objects.nonNull(o.getTakeShopId())?Constants.ONE:Constants.TWO) |
| | | ); |
| | | vo.setCreateTime(o.getCreateTime()); |
| | | vo.setExpectedTakeTime(o.getExpectedTakeTime()); |
| | | |
| | |
| | | vo.setExpectedDepositTime(order.getExpectedDepositTime()); |
| | | vo.setExpectedTakeTime(order.getExpectedTakeTime()); |
| | | vo.setArriveTime(order.getArriveTime()); |
| | | vo.setStatusName(Constants.OrderStatus.getDescByKey(order.getStatus(),order.getType())); |
| | | |
| | | // 费用(分) |
| | | vo.setBasicAmount(order.getBasicAmount()); |
| | | vo.setDeclaredAmount(order.getDeclaredAmount()); |
| | | vo.setDeclaredFee(order.getDeclaredFee()); |
| | | vo.setUrgentAmount(order.getUrgentAmount()); |
| | | vo.setActualPayAmount(order.getPayAmount()); |
| | | vo.setActualPayAmount(Constants.equalsInteger(order.getPayStatus(), Constants.ONE)?order.getPayAmount():order.getEstimatedAmount()); |
| | | |
| | | // 标记 |
| | | vo.setExceptionStatus(order.getExceptionStatus()); |
| | |
| | | |
| | | // 取件信息 |
| | | if (order.getTakeShopId() != null) { |
| | | vo.setTakeShopId(order.getTakeShopId()); |
| | | ShopInfo takeShop = shopInfoMapper.selectById(order.getTakeShopId()); |
| | | if (takeShop != null) { |
| | | vo.setTakeShopName(takeShop.getName()); |
| | |
| | | // 取件联系人 |
| | | vo.setTakeUser(order.getTakeUser()); |
| | | vo.setTakePhone(order.getTakePhone()); |
| | | |
| | | // 司机信息 |
| | | if (order.getAcceptDriver() != null) { |
| | | DriverInfo driver = driverInfoMapper.selectById(order.getAcceptDriver()); |
| | | if (driver != null) { |
| | | vo.setDriverId(driver.getId()); |
| | | vo.setDriverName(driver.getName()); |
| | | vo.setDriverPhone(driver.getTelephone()); |
| | | } |
| | | } |
| | | |
| | | // 物品类型名称 |
| | | if (order.getGoodType() != null) { |
| | |
| | | .eq(OrdersDetail::getDeleted, Constants.ZERO)); |
| | | vo.setDetailList(buildDetailList(details)); |
| | | |
| | | // 逾期信息 |
| | | OverdueFeeVO overdueInfo = calculateOverdueFeeInternal(order, details); |
| | | vo.setOverdue(overdueInfo.getOverdue()); |
| | | vo.setOverdueDays(overdueInfo.getOverdueDays()); |
| | | vo.setOverdueFee(overdueInfo.getOverdueFee()); |
| | | Integer orderStatus = order.getStatus(); |
| | | if(Constants.equalsInteger(orderStatus, Constants.FIVE)){ |
| | | // 逾期信息 |
| | | OverdueFeeVO overdueInfo = calculateOverdueFeeInternal(order, details); |
| | | if (Constants.ONE.equals(order.getType()) |
| | | && order.getTakeShopId() != null) { |
| | | // 异地寄存 + 有取件门店: |
| | | // 根据行李转移到店时间(arriveTime)当天晚上12点判断是否逾期 |
| | | if (order.getArriveTime() != null) { |
| | | Calendar arriveCal = Calendar.getInstance(); |
| | | arriveCal.setTime(order.getArriveTime()); |
| | | arriveCal.set(Calendar.HOUR_OF_DAY, 23); |
| | | arriveCal.set(Calendar.MINUTE, 59); |
| | | arriveCal.set(Calendar.SECOND, 59); |
| | | Date arriveEndOfDay = arriveCal.getTime(); |
| | | boolean isOverdue = new Date().after(arriveEndOfDay); |
| | | vo.setOverdue(isOverdue); |
| | | if (isOverdue) { |
| | | vo.setOverdueDays(overdueInfo.getOverdueDays() > 0 ? overdueInfo.getOverdueDays() : 1); |
| | | vo.setOverdueFee(overdueInfo.getOverdueFee()); |
| | | } else { |
| | | vo.setOverdueDays(0); |
| | | vo.setOverdueFee(0L); |
| | | } |
| | | } else { |
| | | vo.setOverdue(false); |
| | | vo.setOverdueDays(0); |
| | | vo.setOverdueFee(0L); |
| | | } |
| | | } else if (Constants.ZERO.equals(order.getType())) { |
| | | // 就地寄存:保持原逻辑 |
| | | vo.setOverdue(overdueInfo.getOverdue()); |
| | | vo.setOverdueDays(overdueInfo.getOverdueDays()); |
| | | vo.setOverdueFee(overdueInfo.getOverdueFee()); |
| | | } |
| | | } else { |
| | | vo.setOverdue(false); |
| | | vo.setOverdueDays(0); |
| | | vo.setOverdueFee(0L); |
| | | } |
| | | |
| | | // 退款信息(status=96关闭/99取消时返回) |
| | | if (orderStatus != null && |
| | | (Constants.equalsInteger(orderStatus, Constants.OrderStatus.cancelled.getStatus()))) { |
| | | vo.setRefundApplyTime(order.getCancelTime()); |
| | | // 查询退款记录获取退款金额和备注 |
| | | OrdersRefund ordersRefund = ordersRefundMapper.selectOne( |
| | | new QueryWrapper<OrdersRefund>().lambda() |
| | | .eq(OrdersRefund::getOrderId, order.getId()) |
| | | .eq(OrdersRefund::getDeleted, Constants.ZERO) |
| | | .orderByDesc(OrdersRefund::getCreateTime) |
| | | .last("limit 1")); |
| | | if (ordersRefund != null) { |
| | | vo.setRefundAmount(ordersRefund.getRefundAmount() != null |
| | | ? ordersRefund.getRefundAmount() : order.getRefundAmount()); |
| | | vo.setRefundRemark(ordersRefund.getRemark()); |
| | | vo.setRefundTime(ordersRefund.getRefundTime()); |
| | | } else { |
| | | vo.setRefundAmount(order.getRefundAmount()); |
| | | } |
| | | } |
| | | |
| | | // 核销码 |
| | | Integer status = order.getStatus(); |
| | |
| | | vo.setMemberVerifyCode(order.getMemberVerifyCode()); |
| | | } |
| | | |
| | | // 异地寄存经纬度(就地寄存不返回) |
| | | if (Constants.ONE.equals(order.getType())) { |
| | | // status=3(已接单):返回存件门店经纬度 + 司机经纬度 |
| | | if (Constants.equalsInteger(status, Constants.OrderStatus.accepted.getStatus())) { |
| | | ShopInfo depositShop = shopInfoMapper.selectById(order.getDepositShopId()); |
| | | if (depositShop != null) { |
| | | vo.setDepositShopLng(depositShop.getLongitude()); |
| | | vo.setDepositShopLat(depositShop.getLatitude()); |
| | | } |
| | | if (order.getAcceptDriver() != null) { |
| | | DriverInfo driver = driverInfoMapper.selectById(order.getAcceptDriver()); |
| | | if (driver != null) { |
| | | vo.setDriverLng(driver.getLongitude()); |
| | | vo.setDriverLat(driver.getLatitude()); |
| | | } |
| | | } |
| | | } |
| | | // status=4(配送中):返回取件点经纬度 + 司机经纬度 |
| | | if (Constants.equalsInteger(status, Constants.OrderStatus.delivering.getStatus())) { |
| | | // 取件点经纬度(优先取件门店,否则订单上的取件坐标) |
| | | if (order.getTakeShopId() != null) { |
| | | ShopInfo takeShop = shopInfoMapper.selectById(order.getTakeShopId()); |
| | | if (takeShop != null) { |
| | | vo.setTakeLng(takeShop.getLongitude()); |
| | | vo.setTakeLat(takeShop.getLatitude()); |
| | | } |
| | | } else if (order.getTakeLgt() != null && order.getTakeLat() != null) { |
| | | vo.setTakeLng(order.getTakeLgt().doubleValue()); |
| | | vo.setTakeLat(order.getTakeLat().doubleValue()); |
| | | } |
| | | if (order.getAcceptDriver() != null) { |
| | | DriverInfo driver = driverInfoMapper.selectById(order.getAcceptDriver()); |
| | | if (driver != null) { |
| | | vo.setDriverLng(driver.getLongitude()); |
| | | vo.setDriverLat(driver.getLatitude()); |
| | | } |
| | | } |
| | | } |
| | | } |
| | | |
| | | // 评价信息 |
| | | vo.setCommentStatus(order.getCommentStatus()); |
| | | if (Constants.equalsInteger(order.getCommentStatus(), Constants.ONE)) { |
| | | vo.setCommentTime(order.getCommentTime()); |
| | | // 查询评价记录,获取各对象评分 |
| | | List<OrderComment> comments = orderCommentMapper.selectList(new QueryWrapper<OrderComment>().lambda() |
| | | .eq(OrderComment::getOrderId, order.getId()) |
| | | .eq(OrderComment::getDeleted, Constants.ZERO)); |
| | | for (OrderComment c : comments) { |
| | | if (Constants.equalsInteger(c.getTargetType(), Constants.ONE)) { |
| | | vo.setDepositScore(c.getScore()); |
| | | vo.setCommentContent(c.getContent()); |
| | | } else if (Constants.equalsInteger(c.getTargetType(), Constants.TWO)) { |
| | | vo.setTakeScore(c.getScore()); |
| | | } else if (Constants.equalsInteger(c.getTargetType(), Constants.THREE)) { |
| | | vo.setDriverScore(c.getScore()); |
| | | } |
| | | } |
| | | // 评价附件图片 |
| | | vo.setCommentImages(getFileUrls(order.getId(), Constants.FileType.COMMENT_ATTACH.getKey(), imgPrefix)); |
| | | } |
| | | |
| | | return vo; |
| | | } |
| | | |
| | |
| | | throw new BusinessException(ResponseStatus.DATA_EMPTY); |
| | | } |
| | | |
| | | // 仅异地寄存可取消 |
| | | if (!Constants.equalsInteger(order.getType(), Constants.ONE)) { |
| | | throw new BusinessException(ResponseStatus.BAD_REQUEST.getCode(), "仅异地寄存订单可取消"); |
| | | } |
| | | |
| | | Integer status = order.getStatus(); |
| | | if (status == null) { |
| | | throw new BusinessException(ResponseStatus.BAD_REQUEST.getCode(), "订单状态异常"); |
| | |
| | | |
| | | Date now = new Date(); |
| | | |
| | | // 待支付:直接取消 |
| | | // 待支付:直接取消(不限订单类型) |
| | | if (Constants.equalsInteger(status, Constants.OrderStatus.waitPay.getStatus())) { |
| | | order.setStatus(Constants.OrderStatus.cancelled.getStatus()); |
| | | order.setCancelTime(now); |
| | | ordersMapper.updateById(order); |
| | | saveCancelLog(order, "会员取消订单(待支付)", reason, memberId); |
| | | // 短信通知会员:订单已取消 |
| | | Member cancelMember1 = memberMapper.selectById(memberId); |
| | | sendSmsNotify(cancelMember1 != null ? cancelMember1.getTelephone() : null, |
| | | Constants.SmsNotify.MEMBER_CANCELLED, "orderNo", order.getCode()); |
| | | return; |
| | | } |
| | | |
| | | // 待寄存:直接取消,全额退款 |
| | | // 待寄存:直接取消,全额退款(不限订单类型) |
| | | if (Constants.equalsInteger(status, Constants.OrderStatus.waitDeposit.getStatus())) { |
| | | // 先标记订单已取消 |
| | | order.setStatus(Constants.OrderStatus.cancelled.getStatus()); |
| | | order.setCancelTime(now); |
| | | order.setRefundAmount(order.getPayAmount()); |
| | | ordersMapper.updateById(order); |
| | | |
| | | saveCancelLog(order, "会员取消订单(待寄存,全额退款)", reason, memberId); |
| | | // 通知会员:已取消 |
| | | sendOrderNotice(memberId, Constants.MemberOrderNotify.CANCELLED, orderId, |
| | | "orderNo", order.getCode()); |
| | | // 短信通知会员:订单已取消 |
| | | Member cancelMember2 = memberMapper.selectById(memberId); |
| | | sendSmsNotify(cancelMember2 != null ? cancelMember2.getTelephone() : null, |
| | | Constants.SmsNotify.MEMBER_CANCELLED, "orderNo", order.getCode()); |
| | | |
| | | // 调用微信退款V3,全额退款 |
| | | com.wechat.pay.java.service.refund.model.Refund refundResult; |
| | | try { |
| | | refundResult = wxPayV3Service.refund(order.getOutTradeNo(), order.getPayAmount(), order.getPayAmount(), |
| | | "订单退款", wxPayProperties.getV3RefundNotifyUrl()); |
| | | } catch (Exception e) { |
| | | log.error("待寄存订单退款调用异常, orderId={}", orderId, e); |
| | | throw new BusinessException(ResponseStatus.BAD_REQUEST.getCode(), "退款失败,请稍后重试"); |
| | | } |
| | | |
| | | com.wechat.pay.java.service.refund.model.Status refundStatus = refundResult.getStatus(); |
| | | |
| | | // 记录退款信息 |
| | | OrdersRefund refund = new OrdersRefund(); |
| | | refund.setOrderId(orderId); |
| | |
| | | refund.setCancelInfo(reason); |
| | | refund.setCreateTime(now); |
| | | refund.setDeleted(Constants.ZERO); |
| | | |
| | | // 调用微信退款V3,全额退款 |
| | | Refund refundResult = wxPayV3Service.refund(order.getOutTradeNo(), order.getPayAmount(), order.getPayAmount(), |
| | | "订单退款", wxPayProperties.getV3RefundNotifyUrl()); |
| | | refund.setBeforeStatus(Constants.OrderStatus.waitDeposit.getStatus()); |
| | | refund.setRefundAmount(order.getPayAmount()); |
| | | refund.setRefundCode(refundResult.getOutRefundNo()); |
| | | refund.setStatus(Constants.ZERO); // 退款中 |
| | | |
| | | if (com.wechat.pay.java.service.refund.model.Status.SUCCESS.equals(refundStatus)) { |
| | | // 退款成功 |
| | | refund.setStatus(Constants.ONE); |
| | | } else if (com.wechat.pay.java.service.refund.model.Status.PROCESSING.equals(refundStatus)) { |
| | | // 退款中,等回调处理 |
| | | refund.setStatus(Constants.ZERO); |
| | | } else { |
| | | // 退款失败/异常(CLOSED / ABNORMAL / 其他) |
| | | log.error("待寄存订单退款失败, orderId={}, refundStatus={}", orderId, refundStatus); |
| | | throw new BusinessException(ResponseStatus.BAD_REQUEST.getCode(), "退款失败,请联系客服处理"); |
| | | } |
| | | ordersRefundMapper.insert(refund); |
| | | |
| | | order.setStatus(Constants.OrderStatus.cancelled.getStatus()); |
| | | order.setCancelTime(now); |
| | | order.setRefundAmount(order.getPayAmount()); |
| | | ordersMapper.updateById(order); |
| | | |
| | | saveCancelLog(order, "会员取消订单(待寄存,全额退款)", reason, memberId); |
| | | // 通知会员:退款中 |
| | | sendOrderNotice(memberId, Constants.MemberOrderNotify.REFUNDING, orderId, |
| | | "orderNo", order.getCode()); |
| | | return; |
| | | } |
| | | |
| | | // 已寄存/已接单:进入取消中状态 |
| | | // 已寄存/已接单:仅异地寄存可取消 |
| | | if (!Constants.equalsInteger(order.getType(), Constants.ONE)) { |
| | | throw new BusinessException(ResponseStatus.BAD_REQUEST.getCode(), "仅异地寄存订单可取消"); |
| | | } |
| | | |
| | | // 已寄存/已接单:直接将订单类型改为就地寄存 |
| | | if (Constants.equalsInteger(status, Constants.OrderStatus.deposited.getStatus()) |
| | | || Constants.equalsInteger(status, Constants.OrderStatus.accepted.getStatus())) { |
| | | order.setStatus(Constants.OrderStatus.cancelling.getStatus()); |
| | | order.setCancelTime(now); |
| | | order.setType(Constants.ZERO); // 就地寄存 |
| | | ordersMapper.updateById(order); |
| | | saveCancelLog(order, "会员申请取消订单(已寄存/已接单)", reason, memberId); |
| | | // 通知存件门店:退款申请 |
| | | saveCancelLog(order, "会员取消异地寄存订单,转为就地寄存", reason, memberId); |
| | | // 通知存件门店 |
| | | if (order.getDepositShopId() != null) { |
| | | sendShopNotice(order.getDepositShopId(), Constants.ShopOrderNotify.REFUNDING, orderId, |
| | | "orderNo", order.getCode()); |
| | | ShopInfo depositShop = shopInfoMapper.selectById(order.getDepositShopId()); |
| | | sendSmsNotify(depositShop != null ? depositShop.getLinkPhone() : null, |
| | | Constants.SmsNotify.SHOP_REFUNDING, "orderNo", order.getCode()); |
| | | } |
| | | // 通知司机:订单已取消(已接单情况下司机需停止服务) |
| | | if (order.getAcceptDriver() != null && Constants.equalsInteger(status, Constants.OrderStatus.accepted.getStatus())) { |
| | | sendDriverNotice(order.getAcceptDriver(), Constants.DriverOrderNotify.REFUNDING, orderId, |
| | | "orderNo", order.getCode()); |
| | | DriverInfo driver = driverInfoMapper.selectById(order.getAcceptDriver()); |
| | | sendSmsNotify(driver != null ? driver.getTelephone() : null, |
| | | Constants.SmsNotify.DRIVER_REFUNDING, "orderNo", order.getCode()); |
| | | } |
| | | return; |
| | | } |
| | |
| | | } |
| | | } |
| | | |
| | | /** |
| | | * 发送司机站内信通知 |
| | | */ |
| | | private void sendDriverNotice(Integer driverId, Constants.DriverOrderNotify notify, Integer orderId, String... params) { |
| | | DriverInfo driver = driverInfoMapper.selectById(driverId); |
| | | if (driver == null || driver.getMemberId() == null) { |
| | | return; |
| | | } |
| | | Notice notice = new Notice(); |
| | | notice.setUserType(1); // 1=司机 |
| | | notice.setUserId(driver.getMemberId()); |
| | | notice.setTitle(notify.getTitle()); |
| | | notice.setContent(notify.format(params)); |
| | | notice.setObjId(orderId); |
| | | notice.setObjType(0); // 0=订单 |
| | | notice.setStatus(0); // 0=未读 |
| | | notice.setIsdeleted(Constants.ZERO); |
| | | notice.setCreateDate(new Date()); |
| | | noticeService.create(notice); |
| | | } |
| | | |
| | | @Override |
| | | @Transactional(rollbackFor = {Exception.class, BusinessException.class}) |
| | | public void handleStorageOrderPayNotify(String outTradeNo, String wxTradeNo) { |
| | |
| | | order.setPayStatus(Constants.ONE); // 已支付 |
| | | order.setPayTime(now); |
| | | order.setWxExternalNo(wxTradeNo); |
| | | order.setPayAmount(order.getTotalAmount()); |
| | | order.setUpdateTime(now); |
| | | // 生成会员核销码 |
| | | order.setMemberVerifyCode(generateVerifyCode()); |
| | |
| | | if (Constants.ZERO.equals(order.getType()) && order.getDepositShopId() != null) { |
| | | sendShopNotice(order.getDepositShopId(), Constants.ShopOrderNotify.WAIT_VERIFY, order.getId(), |
| | | "orderNo", order.getCode()); |
| | | } |
| | | |
| | | // 短信通知存件门店:有新订单待核验 |
| | | if (order.getDepositShopId() != null) { |
| | | ShopInfo depositShop = shopInfoMapper.selectById(order.getDepositShopId()); |
| | | if (depositShop != null) { |
| | | sendSmsNotify(depositShop.getLinkPhone(), Constants.SmsNotify.SHOP_WAIT_VERIFY, |
| | | "orderNo", order.getCode()); |
| | | } |
| | | } |
| | | } |
| | | |
| | |
| | | if (!Constants.equalsInteger(order.getMemberId(), memberId)) { |
| | | throw new BusinessException(ResponseStatus.NOT_ALLOWED.getCode(), "无权操作此订单"); |
| | | } |
| | | // 仅已完成(7)、已取消(99)、已退款(96)可删除 |
| | | // 仅已完成(7)、已取消(99) |
| | | int status = Constants.formatIntegerNum(order.getStatus()); |
| | | if (status != Constants.OrderStatus.finished.getStatus() |
| | | && status != Constants.OrderStatus.cancelled.getStatus() |
| | | && status != Constants.OrderStatus.closed.getStatus()) { |
| | | && status != Constants.OrderStatus.cancelled.getStatus()) { |
| | | throw new BusinessException(ResponseStatus.NOT_ALLOWED.getCode(), "当前订单状态不可删除"); |
| | | } |
| | | ordersMapper.update(new UpdateWrapper<Orders>().lambda() |
| | |
| | | "orderNo", order.getCode(), |
| | | "amount", String.valueOf(Constants.getFormatMoney( |
| | | order.getTotalAmount() != null ? order.getTotalAmount() : 0L))); |
| | | |
| | | // 通知司机:订单已结算 |
| | | if (order.getAcceptDriver() != null) { |
| | | sendDriverNotice(order.getAcceptDriver(), Constants.DriverOrderNotify.SETTLED, order.getId(), |
| | | "orderNo", order.getCode(), |
| | | "amount", String.valueOf(Constants.getFormatMoney( |
| | | order.getDriverFee() != null ? order.getDriverFee() : 0L))); |
| | | } |
| | | } |
| | | } |
| | | |
| | |
| | | orderCommentMapper.insert(driverComment); |
| | | } |
| | | |
| | | // 4.4 保存评价附件图片(obj_type=15,最多3张) |
| | | saveVerifyImages(order.getId(), dto.getImages(), Constants.FileType.COMMENT_ATTACH.getKey(), null); |
| | | |
| | | // 5. 更新门店/司机平均评分 |
| | | updateTargetScore(Constants.ONE, order.getDepositShopId()); |
| | | if (isRemote && order.getTakeShopId() != null) { |
| | |
| | | // 通知存件门店和取件门店:订单已评价 |
| | | notifyBothShops(order, Constants.ShopOrderNotify.EVALUATED, |
| | | "orderNo", order.getCode()); |
| | | |
| | | // 通知司机:订单已评价 |
| | | if (order.getAcceptDriver() != null) { |
| | | sendDriverNotice(order.getAcceptDriver(), Constants.DriverOrderNotify.EVALUATED, order.getId(), |
| | | "orderNo", order.getCode()); |
| | | } |
| | | } |
| | | |
| | | /** |
| | |
| | | notifyBothShops(order, Constants.ShopOrderNotify.FINISHED, |
| | | "orderNo", order.getCode(), |
| | | "settleDays", settleDays != null ? settleDays : "7"); |
| | | // 通知司机:订单已完成 |
| | | if (order.getAcceptDriver() != null) { |
| | | sendDriverNotice(order.getAcceptDriver(), Constants.DriverOrderNotify.FINISHED, order.getId(), |
| | | "orderNo", order.getCode(), |
| | | "settleDays", settleDays != null ? settleDays : "7"); |
| | | } |
| | | } else { |
| | | throw new BusinessException(ResponseStatus.BAD_REQUEST.getCode(), "当前订单状态不允许核销"); |
| | | } |
| | |
| | | refundRecord.setType(3); // 出库退款 |
| | | refundRecord.setCreateTime(now); |
| | | refundRecord.setRefundRemark(remark); |
| | | refundRecord.setRefundAmount(order.getRefundAmount()); |
| | | refundRecord.setDeleted(Constants.ZERO); |
| | | refundRecord.setBeforeStatus(order.getStatus()); |
| | | ordersRefundMapper.insert(refundRecord); |
| | | |
| | | // 调用微信退款V3(放在最后,确保前置操作全部成功) |
| | |
| | | notifyBothShops(order, Constants.ShopOrderNotify.FINISHED, |
| | | "orderNo", order.getCode(), |
| | | "settleDays", settleDays != null ? settleDays : "7"); |
| | | // 通知司机:订单已完成 |
| | | if (order.getAcceptDriver() != null) { |
| | | sendDriverNotice(order.getAcceptDriver(), Constants.DriverOrderNotify.FINISHED, order.getId(), |
| | | "orderNo", order.getCode(), |
| | | "settleDays", settleDays != null ? settleDays : "7"); |
| | | } |
| | | } |
| | | |
| | | @Override |
| | |
| | | notifyBothShops(order, Constants.ShopOrderNotify.FINISHED, |
| | | "orderNo", order.getCode(), |
| | | "settleDays", settleDays != null ? settleDays : "7"); |
| | | // 通知司机:订单已完成 |
| | | if (order.getAcceptDriver() != null) { |
| | | sendDriverNotice(order.getAcceptDriver(), Constants.DriverOrderNotify.FINISHED, order.getId(), |
| | | "orderNo", order.getCode(), |
| | | "settleDays", settleDays != null ? settleDays : "7"); |
| | | } |
| | | } |
| | | |
| | | @Override |
| | |
| | | Long takeShopFee = 0L; |
| | | Long driverFee = 0L; |
| | | |
| | | if (Constants.equalsInteger(order.getType(), Constants.TWO)) { |
| | | if (Constants.equalsInteger(order.getType(), Constants.ONE)) { |
| | | // 异地寄存:存件门店 + 司机 |
| | | driverFee = new BigDecimal(totalAmount) |
| | | .multiply(driverRate) |
| | |
| | | sendShopNotice(order.getTakeShopId(), Constants.ShopOrderNotify.ARRIVED, order.getId(), |
| | | "orderNo", order.getCode(), |
| | | "destination", destination); |
| | | } |
| | | |
| | | // 短信通知会员:行李已送达 |
| | | Member arrivedMember = memberMapper.selectById(order.getMemberId()); |
| | | if (arrivedMember != null) { |
| | | sendSmsNotify(arrivedMember.getTelephone(), Constants.SmsNotify.MEMBER_ARRIVED, |
| | | "orderNo", order.getCode(), |
| | | "address", destination, |
| | | "code", order.getMemberVerifyCode() != null ? order.getMemberVerifyCode() : ""); |
| | | } |
| | | } |
| | | |
| | |
| | | } |
| | | if (Constants.equalsInteger(status, Constants.OrderStatus.cancelled.getStatus())) { |
| | | return "订单已取消,感谢您的支持,欢迎下次再会!"; |
| | | } |
| | | if (Constants.equalsInteger(status, Constants.OrderStatus.cancelling.getStatus())) { |
| | | return "退款申请已提交,平台会尽快为您处理退款"; |
| | | } |
| | | if (Constants.equalsInteger(status, Constants.OrderStatus.closed.getStatus())) { |
| | | return "退款已成功原路返回,请注意查收"; |
| | | } |
| | | return ""; |
| | | } |
| | |
| | | } |
| | | |
| | | @Override |
| | | public ActiveOrderTipVO getActiveOrderTip(Integer memberId) { |
| | | // 查询状态为 0~5 的最新一条订单 |
| | | QueryWrapper<Orders> wrapper = new QueryWrapper<>(); |
| | | wrapper.eq("MEMBER_ID", memberId) |
| | | .in("STATUS", 0, 1, 2, 3, 4, 5) |
| | | .orderByDesc("CREATE_TIME") |
| | | .last("LIMIT 1"); |
| | | Orders order = ordersMapper.selectOne(wrapper); |
| | | if (order == null) { |
| | | return null; |
| | | } |
| | | |
| | | ActiveOrderTipVO vo = new ActiveOrderTipVO(); |
| | | vo.setOrderId(order.getId()); |
| | | vo.setStatus(order.getStatus()); |
| | | |
| | | // 构建提示文案 |
| | | boolean isLocal = Constants.equalsInteger(order.getType(), Constants.ZERO); |
| | | Integer status = order.getStatus(); |
| | | String tip = null; |
| | | |
| | | if (Constants.equalsInteger(status, Constants.OrderStatus.waitPay.getStatus())) { |
| | | // 待支付:提示支付倒计时 |
| | | String minutes = ""; |
| | | try { |
| | | minutes = operationConfigBiz.getConfig().getAutoCancelTime(); |
| | | } catch (Exception ignored) {} |
| | | tip = "请在" + (StringUtils.isNotBlank(minutes) ? minutes : "") + "分钟内完成支付,超时订单将自动取消"; |
| | | |
| | | // 计算支付倒计时 |
| | | if (StringUtils.isNotBlank(minutes) && order.getCreateTime() != null) { |
| | | long timeoutMs = Long.parseLong(minutes) * 60 * 1000; |
| | | long deadline = order.getCreateTime().getTime() + timeoutMs; |
| | | long remain = deadline - System.currentTimeMillis(); |
| | | vo.setPayCountdown(remain > 0 ? remain : -1L); |
| | | } else { |
| | | vo.setPayCountdown(-1L); |
| | | } |
| | | } else if (Constants.equalsInteger(status, Constants.OrderStatus.waitDeposit.getStatus())) { |
| | | tip = "订单已支付,请前往门店寄存"; |
| | | } else if (Constants.equalsInteger(status, Constants.OrderStatus.deposited.getStatus())) { |
| | | tip = isLocal ? "行李已寄存,请凭取件码前往指定门店取件" : "门店已接单,正在为您安排取件司机"; |
| | | } else if (Constants.equalsInteger(status, Constants.OrderStatus.accepted.getStatus())) { |
| | | tip = isLocal ? "行李已寄存,请凭取件码前往指定门店取件" : "已有司机抢单,正前往取件地点"; |
| | | } else if (Constants.equalsInteger(status, Constants.OrderStatus.delivering.getStatus())) { |
| | | tip = "司机已取件,正运往目的地"; |
| | | } else if (Constants.equalsInteger(status, Constants.OrderStatus.arrived.getStatus())) { |
| | | tip = "行李已送达服务点,请及时前往取件"; |
| | | } |
| | | |
| | | vo.setTip(tip); |
| | | return vo; |
| | | } |
| | | |
| | | @Override |
| | | public EstimatedDeliveryResultVO calculateEstimatedDelivery(Integer cityId, |
| | | Double fromLat, Double fromLng, |
| | | Double toLat, Double toLng) { |
| | |
| | | return vo; |
| | | } |
| | | |
| | | @Override |
| | | public int cancelTimeoutUnpaidOrders() { |
| | | // 获取超时配置(分钟) |
| | | String autoCancelTimeStr = operationConfigBiz.getConfig().getAutoCancelTime(); |
| | | if (StringUtils.isBlank(autoCancelTimeStr)) { |
| | | log.info("未配置超时取消时间,跳过"); |
| | | return 0; |
| | | } |
| | | int autoCancelMinutes; |
| | | try { |
| | | autoCancelMinutes = Integer.parseInt(autoCancelTimeStr); |
| | | } catch (NumberFormatException e) { |
| | | log.warn("超时取消时间配置异常: {}", autoCancelTimeStr); |
| | | return 0; |
| | | } |
| | | if (autoCancelMinutes <= 0) { |
| | | return 0; |
| | | } |
| | | |
| | | // 查询所有超时未支付订单:status=0 且 创建时间 + 配置分钟数 < 当前时间 |
| | | Date deadline = new Date(System.currentTimeMillis() - (long) autoCancelMinutes * 60 * 1000); |
| | | List<Orders> timeoutOrders = ordersMapper.selectList(new QueryWrapper<Orders>().lambda() |
| | | .eq(Orders::getStatus, Constants.OrderStatus.waitPay.getStatus()) |
| | | .eq(Orders::getDeleted, Constants.ZERO) |
| | | .lt(Orders::getCreateTime, deadline)); |
| | | |
| | | if (timeoutOrders == null || timeoutOrders.isEmpty()) { |
| | | return 0; |
| | | } |
| | | |
| | | int count = 0; |
| | | Date now = new Date(); |
| | | for (Orders order : timeoutOrders) { |
| | | try { |
| | | order.setStatus(Constants.OrderStatus.cancelled.getStatus()); |
| | | order.setCancelTime(now); |
| | | order.setUpdateTime(now); |
| | | ordersMapper.updateById(order); |
| | | |
| | | // 写入操作日志 |
| | | OrderLog orderLog = new OrderLog(); |
| | | orderLog.setOrderId(order.getId()); |
| | | orderLog.setTitle("系统自动取消(超时未支付)"); |
| | | orderLog.setLogInfo("订单超时" + autoCancelMinutes + "分钟未支付,系统自动取消"); |
| | | orderLog.setObjType(Constants.ORDER_LOG_CANCEL); |
| | | orderLog.setOrderStatus(Constants.OrderStatus.cancelled.getStatus()); |
| | | orderLog.setOptUserType(3); // 3=系统 |
| | | orderLog.setCreateTime(now); |
| | | orderLog.setDeleted(Constants.ZERO); |
| | | orderLogService.create(orderLog); |
| | | |
| | | // 短信通知会员:订单已取消 |
| | | if (order.getMemberId() != null) { |
| | | Member member = memberMapper.selectById(order.getMemberId()); |
| | | sendSmsNotify(member != null ? member.getTelephone() : null, |
| | | Constants.SmsNotify.MEMBER_CANCELLED, "orderNo", order.getCode()); |
| | | } |
| | | |
| | | count++; |
| | | } catch (Exception e) { |
| | | log.error("取消超时订单异常, orderId={}, error={}", order.getId(), e.getMessage()); |
| | | } |
| | | } |
| | | log.info("超时未支付订单自动取消完成,共取消{}单", count); |
| | | return count; |
| | | } |
| | | |
| | | @Override |
| | | public int notifyUngrabbedOrders() { |
| | | // 获取无人抢单通知时间配置(分钟) |
| | | String noGrabTimeStr = operationConfigBiz.getConfig().getNoGrabNotifyTime(); |
| | | if (StringUtils.isBlank(noGrabTimeStr)) { |
| | | return 0; |
| | | } |
| | | int noGrabMinutes; |
| | | try { |
| | | noGrabMinutes = Integer.parseInt(noGrabTimeStr); |
| | | } catch (NumberFormatException e) { |
| | | log.warn("无人抢单通知时间配置异常: {}", noGrabTimeStr); |
| | | return 0; |
| | | } |
| | | if (noGrabMinutes <= 0) { |
| | | return 0; |
| | | } |
| | | |
| | | // 获取通知人员主键列表 |
| | | String noGrabUsers = operationConfigBiz.getConfig().getNoGrabNotifyUsers(); |
| | | if (StringUtils.isBlank(noGrabUsers)) { |
| | | return 0; |
| | | } |
| | | List<String> userIdStrList = Arrays.asList(noGrabUsers.split(",")); |
| | | List<Integer> userIds = new ArrayList<>(); |
| | | for (String idStr : userIdStrList) { |
| | | if (StringUtils.isNotBlank(idStr.trim())) { |
| | | userIds.add(Integer.parseInt(idStr.trim())); |
| | | } |
| | | } |
| | | if (userIds.isEmpty()) { |
| | | return 0; |
| | | } |
| | | |
| | | // 查询通知人员手机号 |
| | | List<String> notifyPhones = new ArrayList<>(); |
| | | for (Integer userId : userIds) { |
| | | SystemUser user = systemUserMapper.selectById(userId); |
| | | if (user != null && StringUtils.isNotBlank(user.getMobile())) { |
| | | notifyPhones.add(user.getMobile()); |
| | | } |
| | | } |
| | | if (notifyPhones.isEmpty()) { |
| | | log.warn("无人抢单通知人员均无有效手机号"); |
| | | return 0; |
| | | } |
| | | |
| | | // 查询异地已寄存(status=2)、未通知(platformSmsNotified=0或null)、超时的订单 |
| | | Date deadline = new Date(System.currentTimeMillis() - (long) noGrabMinutes * 60 * 1000); |
| | | List<Orders> ungrabbedOrders = ordersMapper.selectList(new QueryWrapper<Orders>().lambda() |
| | | .eq(Orders::getType, Constants.ONE) // 异地寄存 |
| | | .eq(Orders::getStatus, Constants.OrderStatus.deposited.getStatus()) // 已寄存 |
| | | .eq(Orders::getDeleted, Constants.ZERO) |
| | | .ne(Orders::getPlatformSmsNotified, Constants.ONE) // 未通知 |
| | | .lt(Orders::getPlatformSmsNotifiedTime, deadline)); // 通知基准时间超过配置时间 |
| | | |
| | | if (ungrabbedOrders == null || ungrabbedOrders.isEmpty()) { |
| | | return 0; |
| | | } |
| | | |
| | | int count = 0; |
| | | Date now = new Date(); |
| | | for (Orders order : ungrabbedOrders) { |
| | | try { |
| | | // 给所有通知人员发短信 |
| | | for (String phone : notifyPhones) { |
| | | sendSmsNotify(phone, Constants.SmsNotify.PLATFORM_WAIT_GRAB, |
| | | "orderNo", order.getCode(), |
| | | "time", String.valueOf(noGrabMinutes)); |
| | | } |
| | | // 标记已通知 + 记录通知时间 |
| | | order.setPlatformSmsNotified(Constants.ONE); |
| | | order.setPlatformSmsNotifiedTime(now); |
| | | order.setUpdateTime(now); |
| | | ordersMapper.updateById(order); |
| | | count++; |
| | | } catch (Exception e) { |
| | | log.error("无人抢单短信通知异常, orderId={}, error={}", order.getId(), e.getMessage()); |
| | | } |
| | | } |
| | | log.info("无人抢单短信通知完成,共通知{}单", count); |
| | | return count; |
| | | } |
| | | |
| | | @Override |
| | | public int autoCompleteOrders() { |
| | | // 获取自动确认收货天数配置 |
| | | String autoConfirmDaysStr = operationConfigBiz.getConfig().getAutoConfirmReceipt(); |
| | | if (StringUtils.isBlank(autoConfirmDaysStr)) { |
| | | return 0; |
| | | } |
| | | int autoConfirmDays; |
| | | try { |
| | | autoConfirmDays = Integer.parseInt(autoConfirmDaysStr); |
| | | } catch (NumberFormatException e) { |
| | | log.warn("自动确认收货天数配置异常: {}", autoConfirmDaysStr); |
| | | return 0; |
| | | } |
| | | if (autoConfirmDays <= 0) { |
| | | return 0; |
| | | } |
| | | |
| | | // 查询已送达(status=5)且送达时间超过配置天数的订单 |
| | | Date deadline = new Date(System.currentTimeMillis() - (long) autoConfirmDays * 24 * 60 * 60 * 1000); |
| | | List<Orders> orders = ordersMapper.selectList(new QueryWrapper<Orders>().lambda() |
| | | .eq(Orders::getStatus, Constants.OrderStatus.arrived.getStatus()) |
| | | .eq(Orders::getDeleted, Constants.ZERO) |
| | | .isNotNull(Orders::getArriveTime) |
| | | .lt(Orders::getArriveTime, deadline)); |
| | | |
| | | if (orders == null || orders.isEmpty()) { |
| | | return 0; |
| | | } |
| | | |
| | | int count = 0; |
| | | Date now = new Date(); |
| | | for (Orders order : orders) { |
| | | try { |
| | | // 逾期未支付的不自动完成 |
| | | if (Constants.equalsInteger(order.getOverdueStatus(), Constants.ONE)) { |
| | | continue; |
| | | } |
| | | |
| | | // 更新订单状态为已完成 |
| | | order.setStatus(Constants.OrderStatus.finished.getStatus()); |
| | | order.setFinishTime(now); |
| | | order.setUpdateTime(now); |
| | | ordersMapper.updateById(order); |
| | | |
| | | // 释放核销码 |
| | | if (StringUtils.isNotBlank(order.getMemberVerifyCode())) { |
| | | releaseVerifyCode(order.getMemberVerifyCode()); |
| | | } |
| | | |
| | | // 生成收益记录 |
| | | calculateAndSaveOrderFees(order.getId()); |
| | | generateRevenueRecords(order.getId()); |
| | | |
| | | // 记录操作日志 |
| | | OrderLog orderLog = new OrderLog(); |
| | | orderLog.setOrderId(order.getId()); |
| | | orderLog.setTitle("系统自动完成"); |
| | | orderLog.setLogInfo("订单已送达超过" + autoConfirmDays + "天未确认,系统自动完成"); |
| | | orderLog.setObjType(Constants.ORDER_LOG_CANCEL); |
| | | orderLog.setOrderStatus(Constants.OrderStatus.finished.getStatus()); |
| | | orderLog.setOptUserType(3); // 3=系统 |
| | | orderLog.setCreateTime(now); |
| | | orderLog.setDeleted(Constants.ZERO); |
| | | orderLogService.create(orderLog); |
| | | |
| | | // 通知会员:订单已完成 |
| | | sendOrderNotice(order.getMemberId(), Constants.MemberOrderNotify.FINISHED, order.getId(), |
| | | "orderNo", order.getCode()); |
| | | // 通知存件门店和取件门店 |
| | | String settleDays = operationConfigBiz.getConfig().getSettlementDate(); |
| | | notifyBothShops(order, Constants.ShopOrderNotify.FINISHED, |
| | | "orderNo", order.getCode(), |
| | | "settleDays", settleDays != null ? settleDays : "7"); |
| | | // 通知司机 |
| | | if (order.getAcceptDriver() != null) { |
| | | sendDriverNotice(order.getAcceptDriver(), Constants.DriverOrderNotify.FINISHED, order.getId(), |
| | | "orderNo", order.getCode(), |
| | | "settleDays", settleDays != null ? settleDays : "7"); |
| | | } |
| | | |
| | | count++; |
| | | } catch (Exception e) { |
| | | log.error("自动完成订单异常, orderId={}, error={}", order.getId(), e.getMessage()); |
| | | } |
| | | } |
| | | log.info("自动完成超时订单完成,共完成{}单", count); |
| | | return count; |
| | | } |
| | | |
| | | /** |
| | | * 发送短信通知(失败不影响主业务) |
| | | * @param phone 接收手机号 |
| | | * @param smsNotify 短信模板枚举 |
| | | * @param paramPairs 模板参数,key-value 交替传入,如 "orderNo", "XL202604220001" |
| | | */ |
| | | private void sendSmsNotify(String phone, Constants.SmsNotify smsNotify, String... paramPairs) { |
| | | if (StringUtils.isBlank(phone)) { |
| | | return; |
| | | } |
| | | String content = smsNotify.format(paramPairs); |
| | | try { |
| | | JSONObject templateParam = new JSONObject(); |
| | | for (int i = 0; i < paramPairs.length - 1; i += 2) { |
| | | templateParam.put(paramPairs[i], paramPairs[i + 1]); |
| | | } |
| | | boolean result = AliSmsService.sendSms(phone, smsNotify.getTemplateCode(), |
| | | templateParam.toJSONString()); |
| | | if (result) { |
| | | log.info("短信发送成功: phone={}, template={}", phone, smsNotify.name()); |
| | | } else { |
| | | log.warn("短信发送失败: phone={}, template={}", phone, smsNotify.name()); |
| | | } |
| | | // 存储短信记录 |
| | | Smsrecord record = new Smsrecord(); |
| | | record.setPhone(phone); |
| | | record.setContent(content); |
| | | record.setType(Constants.ONE); // 1=订单通知 |
| | | record.setStatus(result ? Constants.ONE : Constants.ZERO); // 1=已发送, 0=发送失败 |
| | | record.setCreateTime(new Date()); |
| | | record.setDeleted(Constants.ZERO); |
| | | smsrecordMapper.insert(record); |
| | | } catch (Exception e) { |
| | | log.error("短信发送异常: phone={}, template={}, error={}", phone, smsNotify.name(), e.getMessage()); |
| | | // 异常也记录 |
| | | try { |
| | | Smsrecord record = new Smsrecord(); |
| | | record.setPhone(phone); |
| | | record.setContent(content); |
| | | record.setType(Constants.ONE); |
| | | record.setStatus(Constants.ZERO); // 发送失败 |
| | | record.setCreateTime(new Date()); |
| | | record.setDeleted(Constants.ZERO); |
| | | smsrecordMapper.insert(record); |
| | | } catch (Exception ignored) {} |
| | | } |
| | | } |
| | | |
| | | } |