mirror of
				https://gitee.com/hhyykk/ipms-sjy.git
				synced 2025-10-31 18:28:43 +08:00 
			
		
		
		
	mall + trade:优化运费模版的代码
This commit is contained in:
		| @@ -45,19 +45,25 @@ public interface ErrorCodeConstants { | |||||||
|     // ==========  Cart 模块 1011002000 ========== |     // ==========  Cart 模块 1011002000 ========== | ||||||
|     ErrorCode CARD_ITEM_NOT_FOUND = new ErrorCode(1011002000, "购物车项不存在"); |     ErrorCode CARD_ITEM_NOT_FOUND = new ErrorCode(1011002000, "购物车项不存在"); | ||||||
|  |  | ||||||
|     // ==========  物流配送模块 1011003000 ========== |     // ========== Price 相关 1011003000 ============ | ||||||
|     ErrorCode DELIVERY_EXPRESS_NOT_EXISTS = new ErrorCode(1011003000, "快递公司不存在"); |     ErrorCode PRICE_CALCULATE_PAY_PRICE_ILLEGAL = new ErrorCode(1011003000, "支付价格计算异常,原因:价格小于等于 0"); | ||||||
|     // TODO @jason:最好每个模块一段哈。express 一个;exmpresstemplate 一个;pickup 一个 |     ErrorCode PRICE_CALCULATE_DELIVERY_PRICE_USER_ADDR_IS_EMPTY = new ErrorCode(1011003001, "计算快递运费异常,收件人地址编号为空"); | ||||||
|     ErrorCode EXPRESS_CODE_DUPLICATE = new ErrorCode(1011003001, "已经存在该编码的快递公司"); |     ErrorCode PRICE_CALCULATE_DELIVERY_PRICE_TEMPLATE_NOT_FOUND = new ErrorCode(1011003002, "计算快递运费异常,找不到对应的运费模板"); | ||||||
|     ErrorCode EXPRESS_TEMPLATE_NOT_EXISTS = new ErrorCode(1011003002, "运费模板不存在"); |  | ||||||
|     ErrorCode EXPRESS_TEMPLATE_NAME_DUPLICATE = new ErrorCode(1011003003, "已经存在该运费模板名"); |     // ==========  物流 Express 模块 1011004000 ========== | ||||||
|     ErrorCode EXPRESS_API_QUERY_ERROR = new ErrorCode(1011003004, "快递查询接口异常"); |     ErrorCode EXPRESS_NOT_EXISTS = new ErrorCode(1011004000, "快递公司不存在"); | ||||||
|     ErrorCode EXPRESS_API_QUERY_FAILED = new ErrorCode(1011003005, "快递查询返回失败, 原因:{}"); |     ErrorCode EXPRESS_CODE_DUPLICATE = new ErrorCode(1011004001, "已经存在该编码的快递公司"); | ||||||
|     ErrorCode EXPRESS_CLIENT_NOT_PROVIDE = new ErrorCode(1011003006, "需要接入快递服务商,比如【快递100】"); |     ErrorCode EXPRESS_API_QUERY_ERROR = new ErrorCode(1011004002, "快递查询接口异常"); | ||||||
|     ErrorCode PICK_UP_STORE_NOT_EXISTS = new ErrorCode(1011003007, "自提门店不存在"); |     ErrorCode EXPRESS_API_QUERY_FAILED = new ErrorCode(1011004003, "快递查询返回失败,原因:{}"); | ||||||
|  |     ErrorCode EXPRESS_CLIENT_NOT_PROVIDE = new ErrorCode(1011004004, "需要接入快递服务商,比如【快递100】"); | ||||||
|  |  | ||||||
|  |     // ==========  物流 Template 模块 1011005000 ========== | ||||||
|  |     ErrorCode EXPRESS_TEMPLATE_NAME_DUPLICATE = new ErrorCode(1011005000, "已经存在该运费模板名"); | ||||||
|  |     ErrorCode EXPRESS_TEMPLATE_NOT_EXISTS = new ErrorCode(1011005001, "运费模板不存在"); | ||||||
|  |  | ||||||
|  |     // ==========  物流 PICK_UP 模块 1011006000 ========== | ||||||
|  |  | ||||||
|  |     ErrorCode PICK_UP_STORE_NOT_EXISTS = new ErrorCode(1011006000, "自提门店不存在"); | ||||||
|  |  | ||||||
|  |  | ||||||
|     // ========== Price 相关 1011004000 ============ |  | ||||||
|     ErrorCode PRICE_CALCULATE_PAY_PRICE_ILLEGAL = new ErrorCode(1011004000, "支付价格计算异常,原因:价格小于等于 0"); |  | ||||||
|     ErrorCode PRICE_CALCULATE_DELIVERY_PRICE_USER_ADDR_IS_EMPTY = new ErrorCode(1011004001, "计算快递运费异常,收件人地址编号为空"); |  | ||||||
|     ErrorCode PRICE_CALCULATE_DELIVERY_PRICE_TEMPLATE_NOT_FOUND = new ErrorCode(1011004002, "计算快递运费异常,找不到对应的运费模板"); |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -7,10 +7,15 @@ import cn.iocoder.yudao.module.trade.dal.dataobject.delivery.DeliveryExpressTemp | |||||||
| import cn.iocoder.yudao.module.trade.dal.dataobject.delivery.DeliveryExpressTemplateDO; | import cn.iocoder.yudao.module.trade.dal.dataobject.delivery.DeliveryExpressTemplateDO; | ||||||
| import cn.iocoder.yudao.module.trade.dal.dataobject.delivery.DeliveryExpressTemplateFreeDO; | import cn.iocoder.yudao.module.trade.dal.dataobject.delivery.DeliveryExpressTemplateFreeDO; | ||||||
| import cn.iocoder.yudao.module.trade.service.delivery.bo.DeliveryExpressTemplateRespBO; | import cn.iocoder.yudao.module.trade.service.delivery.bo.DeliveryExpressTemplateRespBO; | ||||||
|  | import com.google.common.collect.Maps; | ||||||
| import org.mapstruct.Mapper; | import org.mapstruct.Mapper; | ||||||
| import org.mapstruct.factory.Mappers; | import org.mapstruct.factory.Mappers; | ||||||
|  |  | ||||||
| import java.util.List; | import java.util.List; | ||||||
|  | import java.util.Map; | ||||||
|  |  | ||||||
|  | import static cn.iocoder.yudao.framework.common.util.collection.CollectionUtils.convertMultiMap; | ||||||
|  | import static cn.iocoder.yudao.framework.common.util.collection.CollectionUtils.findFirst; | ||||||
|  |  | ||||||
| @Mapper | @Mapper | ||||||
| public interface DeliveryExpressTemplateConvert { | public interface DeliveryExpressTemplateConvert { | ||||||
| @@ -48,7 +53,7 @@ public interface DeliveryExpressTemplateConvert { | |||||||
|  |  | ||||||
|     DeliveryExpressTemplateChargeDO convertTemplateCharge(DeliveryExpressTemplateUpdateReqVO.ExpressTemplateChargeUpdateVO vo); |     DeliveryExpressTemplateChargeDO convertTemplateCharge(DeliveryExpressTemplateUpdateReqVO.ExpressTemplateChargeUpdateVO vo); | ||||||
|  |  | ||||||
|     DeliveryExpressTemplateRespBO.DeliveryExpressTemplateChargeBO convertTemplateCharge(DeliveryExpressTemplateChargeDO bean); |     DeliveryExpressTemplateRespBO.Charge convertTemplateCharge(DeliveryExpressTemplateChargeDO bean); | ||||||
|  |  | ||||||
|     default List<DeliveryExpressTemplateChargeDO> convertTemplateChargeList(Long templateId, Integer chargeMode, List<ExpressTemplateChargeBaseVO> list) { |     default List<DeliveryExpressTemplateChargeDO> convertTemplateChargeList(Long templateId, Integer chargeMode, List<ExpressTemplateChargeBaseVO> list) { | ||||||
|         return CollectionUtils.convertList(list, vo -> convertTemplateCharge(templateId, chargeMode, vo)); |         return CollectionUtils.convertList(list, vo -> convertTemplateCharge(templateId, chargeMode, vo)); | ||||||
| @@ -60,7 +65,7 @@ public interface DeliveryExpressTemplateConvert { | |||||||
|  |  | ||||||
|     DeliveryExpressTemplateFreeDO convertTemplateFree(DeliveryExpressTemplateUpdateReqVO.ExpressTemplateFreeUpdateVO vo); |     DeliveryExpressTemplateFreeDO convertTemplateFree(DeliveryExpressTemplateUpdateReqVO.ExpressTemplateFreeUpdateVO vo); | ||||||
|  |  | ||||||
|     DeliveryExpressTemplateRespBO.DeliveryExpressTemplateFreeBO convertTemplateFree(DeliveryExpressTemplateFreeDO bean); |     DeliveryExpressTemplateRespBO.Free convertTemplateFree(DeliveryExpressTemplateFreeDO bean); | ||||||
|  |  | ||||||
|     List<ExpressTemplateChargeBaseVO> convertTemplateChargeList(List<DeliveryExpressTemplateChargeDO> list); |     List<ExpressTemplateChargeBaseVO> convertTemplateChargeList(List<DeliveryExpressTemplateChargeDO> list); | ||||||
|  |  | ||||||
| @@ -70,4 +75,22 @@ public interface DeliveryExpressTemplateConvert { | |||||||
|         return CollectionUtils.convertList(list, vo -> convertTemplateFree(templateId, vo)); |         return CollectionUtils.convertList(list, vo -> convertTemplateFree(templateId, vo)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     default Map<Long, DeliveryExpressTemplateRespBO> convertMap(Integer areaId, List<DeliveryExpressTemplateDO> templateList, | ||||||
|  |                                                                 List<DeliveryExpressTemplateChargeDO> chargeList, | ||||||
|  |                                                                 List<DeliveryExpressTemplateFreeDO> freeList) { | ||||||
|  |         Map<Long, List<DeliveryExpressTemplateChargeDO>> templateIdChargeMap = convertMultiMap(chargeList, | ||||||
|  |                 DeliveryExpressTemplateChargeDO::getTemplateId); | ||||||
|  |         Map<Long, List<DeliveryExpressTemplateFreeDO>> templateIdFreeMap = convertMultiMap(freeList, | ||||||
|  |                 DeliveryExpressTemplateFreeDO::getTemplateId); | ||||||
|  |         // 组合运费模板配置 RespBO | ||||||
|  |         Map<Long, DeliveryExpressTemplateRespBO> result = Maps.newHashMapWithExpectedSize(templateList.size()); | ||||||
|  |         templateList.forEach(template -> { | ||||||
|  |             DeliveryExpressTemplateRespBO bo = new DeliveryExpressTemplateRespBO() | ||||||
|  |                     .setChargeMode(template.getChargeMode()) | ||||||
|  |                     .setCharge(convertTemplateCharge(findFirst(templateIdChargeMap.get(template.getId()), charge -> charge.getAreaIds().contains(areaId)))) | ||||||
|  |                     .setFree(convertTemplateFree(findFirst(templateIdFreeMap.get(template.getId()), free -> free.getAreaIds().contains(areaId)))); | ||||||
|  |             result.put(template.getId(), bo); | ||||||
|  |         }); | ||||||
|  |         return result; | ||||||
|  |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -2,7 +2,6 @@ package cn.iocoder.yudao.module.trade.dal.mysql.delivery; | |||||||
|  |  | ||||||
|  |  | ||||||
| import cn.iocoder.yudao.framework.mybatis.core.mapper.BaseMapperX; | import cn.iocoder.yudao.framework.mybatis.core.mapper.BaseMapperX; | ||||||
| import cn.iocoder.yudao.framework.mybatis.core.query.LambdaQueryWrapperX; |  | ||||||
| import cn.iocoder.yudao.module.trade.dal.dataobject.delivery.DeliveryExpressTemplateChargeDO; | import cn.iocoder.yudao.module.trade.dal.dataobject.delivery.DeliveryExpressTemplateChargeDO; | ||||||
| import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; | import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; | ||||||
| import org.apache.ibatis.annotations.Mapper; | import org.apache.ibatis.annotations.Mapper; | ||||||
| @@ -24,9 +23,9 @@ public interface DeliveryExpressTemplateChargeMapper extends BaseMapperX<Deliver | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     default List<DeliveryExpressTemplateChargeDO> selectByTemplateIds(Collection<Long> templateIds) { |     default List<DeliveryExpressTemplateChargeDO> selectByTemplateIds(Collection<Long> templateIds) { | ||||||
|         return selectList(new LambdaQueryWrapperX<DeliveryExpressTemplateChargeDO>() |         return selectList(DeliveryExpressTemplateChargeDO::getTemplateId, templateIds); | ||||||
|                 .inIfPresent(DeliveryExpressTemplateChargeDO::getTemplateId, templateIds)); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,7 +1,6 @@ | |||||||
| package cn.iocoder.yudao.module.trade.dal.mysql.delivery; | package cn.iocoder.yudao.module.trade.dal.mysql.delivery; | ||||||
|  |  | ||||||
| import cn.iocoder.yudao.framework.mybatis.core.mapper.BaseMapperX; | import cn.iocoder.yudao.framework.mybatis.core.mapper.BaseMapperX; | ||||||
| import cn.iocoder.yudao.framework.mybatis.core.query.LambdaQueryWrapperX; |  | ||||||
| import cn.iocoder.yudao.module.trade.dal.dataobject.delivery.DeliveryExpressTemplateFreeDO; | import cn.iocoder.yudao.module.trade.dal.dataobject.delivery.DeliveryExpressTemplateFreeDO; | ||||||
| import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; | import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; | ||||||
| import org.apache.ibatis.annotations.Mapper; | import org.apache.ibatis.annotations.Mapper; | ||||||
| @@ -22,9 +21,8 @@ public interface DeliveryExpressTemplateFreeMapper extends BaseMapperX<DeliveryE | |||||||
|                 .eq(DeliveryExpressTemplateFreeDO::getTemplateId, templateId)); |                 .eq(DeliveryExpressTemplateFreeDO::getTemplateId, templateId)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     default List<DeliveryExpressTemplateFreeDO> selectListByTemplateIds(Collection<Long> ids) { |     default List<DeliveryExpressTemplateFreeDO> selectListByTemplateIds(Collection<Long> templateIds) { | ||||||
|         return selectList(new LambdaQueryWrapperX<DeliveryExpressTemplateFreeDO>() |         return selectList(DeliveryExpressTemplateFreeDO::getTemplateId, templateIds); | ||||||
|                 .inIfPresent(DeliveryExpressTemplateFreeDO::getTemplateId, ids)); |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -75,7 +75,7 @@ public class DeliveryExpressServiceImpl implements DeliveryExpressService { | |||||||
|     } |     } | ||||||
|     private void validateDeliveryExpressExists(Long id) { |     private void validateDeliveryExpressExists(Long id) { | ||||||
|         if (deliveryExpressMapper.selectById(id) == null) { |         if (deliveryExpressMapper.selectById(id) == null) { | ||||||
|             throw exception(DELIVERY_EXPRESS_NOT_EXISTS); |             throw exception(EXPRESS_NOT_EXISTS); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -84,11 +84,12 @@ public interface DeliveryExpressTemplateService { | |||||||
|     DeliveryExpressTemplateDO validateDeliveryExpressTemplate(Long templateId); |     DeliveryExpressTemplateDO validateDeliveryExpressTemplate(Long templateId); | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 基于运费模板编号数组和收件人地址区域编号. 获取匹配运费模板 |      * 基于运费模板编号数组和收件人地址区域编号,获取匹配运费模板 | ||||||
|      * |      * | ||||||
|      * @param ids    编号列表 |      * @param ids    编号列表 | ||||||
|      * @param areaId 区域编号 |      * @param areaId 区域编号 | ||||||
|      * @return Map (templateId -> 运费模板设置) |      * @return Map (templateId -> 运费模板设置) | ||||||
|      */ |      */ | ||||||
|     Map<Long, DeliveryExpressTemplateRespBO> getExpressTemplateMapByIdsAndArea(Collection<Long> ids, Integer areaId); |     Map<Long, DeliveryExpressTemplateRespBO> getExpressTemplateMapByIdsAndArea(Collection<Long> ids, Integer areaId); | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -224,36 +224,26 @@ public class DeliveryExpressTemplateServiceImpl implements DeliveryExpressTempla | |||||||
|     @Override |     @Override | ||||||
|     public Map<Long, DeliveryExpressTemplateRespBO> getExpressTemplateMapByIdsAndArea(Collection<Long> ids, Integer areaId) { |     public Map<Long, DeliveryExpressTemplateRespBO> getExpressTemplateMapByIdsAndArea(Collection<Long> ids, Integer areaId) { | ||||||
|         Assert.notNull(areaId, "区域编号 {} 不能为空", areaId); |         Assert.notNull(areaId, "区域编号 {} 不能为空", areaId); | ||||||
|  |         // 查询 template 数组 | ||||||
|         if (CollUtil.isEmpty(ids)) { |         if (CollUtil.isEmpty(ids)) { | ||||||
|             return Collections.emptyMap(); |             return Collections.emptyMap(); | ||||||
|         } |         } | ||||||
|         List<DeliveryExpressTemplateDO> templateList = expressTemplateMapper.selectBatchIds(ids); |         List<DeliveryExpressTemplateDO> templateList = expressTemplateMapper.selectBatchIds(ids); | ||||||
|         // 查询 templateCharge |         // 查询 templateCharge 数组 | ||||||
|         List<DeliveryExpressTemplateChargeDO> templeChargeList = expressTemplateChargeMapper.selectByTemplateIds(ids); |         List<DeliveryExpressTemplateChargeDO> chargeList = expressTemplateChargeMapper.selectByTemplateIds(ids); | ||||||
|         Map<Long, List<DeliveryExpressTemplateChargeDO>> templateChargeMap = convertMultiMap(templeChargeList, |         // 查询 templateFree 数组 | ||||||
|                 DeliveryExpressTemplateChargeDO::getTemplateId); |         List<DeliveryExpressTemplateFreeDO> freeList = expressTemplateFreeMapper.selectListByTemplateIds(ids); | ||||||
|         // 查询 templateFree |  | ||||||
|         List<DeliveryExpressTemplateFreeDO> templateFreeList = expressTemplateFreeMapper.selectListByTemplateIds(ids); |  | ||||||
|         Map<Long, List<DeliveryExpressTemplateFreeDO>> templateFreeMap = convertMultiMap(templateFreeList, |  | ||||||
|                 DeliveryExpressTemplateFreeDO::getTemplateId); |  | ||||||
|         // 组合运费模板配置 RespBO |         // 组合运费模板配置 RespBO | ||||||
|         Map<Long, DeliveryExpressTemplateRespBO> result = new HashMap<>(templateList.size()); |         return INSTANCE.convertMap(areaId, templateList, chargeList, freeList); | ||||||
|         templateList.forEach(item -> { |  | ||||||
|             DeliveryExpressTemplateRespBO bo = new DeliveryExpressTemplateRespBO() |  | ||||||
|                     .setChargeMode(item.getChargeMode()) |  | ||||||
|                     .setTemplateCharge(findMatchExpressTemplateCharge(templateChargeMap.get(item.getId()), areaId)) |  | ||||||
|                     .setTemplateFree(findMatchExpressTemplateFree(templateFreeMap.get(item.getId()), areaId)); |  | ||||||
|             result.put(item.getId(), bo); |  | ||||||
|         }); |  | ||||||
|         return result; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private DeliveryExpressTemplateRespBO.DeliveryExpressTemplateChargeBO findMatchExpressTemplateCharge( |     private DeliveryExpressTemplateRespBO.Charge findMatchExpressTemplateCharge( | ||||||
|             List<DeliveryExpressTemplateChargeDO> templateChargeList, Integer areaId) { |             List<DeliveryExpressTemplateChargeDO> templateChargeList, Integer areaId) { | ||||||
|         return INSTANCE.convertTemplateCharge(findFirst(templateChargeList, item -> item.getAreaIds().contains(areaId))); |         return INSTANCE.convertTemplateCharge(findFirst(templateChargeList, item -> item.getAreaIds().contains(areaId))); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private DeliveryExpressTemplateRespBO.DeliveryExpressTemplateFreeBO findMatchExpressTemplateFree( |     private DeliveryExpressTemplateRespBO.Free findMatchExpressTemplateFree( | ||||||
|             List<DeliveryExpressTemplateFreeDO> templateFreeList, Integer areaId) { |             List<DeliveryExpressTemplateFreeDO> templateFreeList, Integer areaId) { | ||||||
|         return INSTANCE.convertTemplateFree(findFirst(templateFreeList, item -> item.getAreaIds().contains(areaId))); |         return INSTANCE.convertTemplateFree(findFirst(templateFreeList, item -> item.getAreaIds().contains(areaId))); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -21,12 +21,12 @@ public class DeliveryExpressTemplateRespBO { | |||||||
|     /** |     /** | ||||||
|      * 运费模板快递运费设置 |      * 运费模板快递运费设置 | ||||||
|      */ |      */ | ||||||
|     private DeliveryExpressTemplateChargeBO templateCharge; |     private Charge charge; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 运费模板包邮设置 |      * 运费模板包邮设置 | ||||||
|      */ |      */ | ||||||
|     private DeliveryExpressTemplateFreeBO templateFree; |     private Free free; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 快递运费模板费用配置 BO |      * 快递运费模板费用配置 BO | ||||||
| @@ -34,7 +34,7 @@ public class DeliveryExpressTemplateRespBO { | |||||||
|      * @author jason |      * @author jason | ||||||
|      */ |      */ | ||||||
|     @Data |     @Data | ||||||
|     public static class DeliveryExpressTemplateChargeBO { |     public static class Charge { | ||||||
|  |  | ||||||
|         /** |         /** | ||||||
|          * 首件数量(件数,重量,或体积) |          * 首件数量(件数,重量,或体积) | ||||||
| @@ -60,7 +60,7 @@ public class DeliveryExpressTemplateRespBO { | |||||||
|      * @author jason |      * @author jason | ||||||
|      */ |      */ | ||||||
|     @Data |     @Data | ||||||
|     public static class DeliveryExpressTemplateFreeBO { |     public static class Free { | ||||||
|  |  | ||||||
|         /** |         /** | ||||||
|          * 包邮金额,单位:分 |          * 包邮金额,单位:分 | ||||||
| @@ -76,4 +76,5 @@ public class DeliveryExpressTemplateRespBO { | |||||||
|          */ |          */ | ||||||
|         private Integer freeCount; |         private Integer freeCount; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -342,7 +342,7 @@ public class TradeOrderServiceImpl implements TradeOrderService { | |||||||
|         // TODO 芋艿:logisticsId 校验存在 发货物流公司 fix |         // TODO 芋艿:logisticsId 校验存在 发货物流公司 fix | ||||||
|         DeliveryExpressDO deliveryExpress = deliveryExpressService.getDeliveryExpress(deliveryReqVO.getLogisticsId()); |         DeliveryExpressDO deliveryExpress = deliveryExpressService.getDeliveryExpress(deliveryReqVO.getLogisticsId()); | ||||||
|         if (deliveryExpress == null) { |         if (deliveryExpress == null) { | ||||||
|             throw exception(DELIVERY_EXPRESS_NOT_EXISTS); |             throw exception(EXPRESS_NOT_EXISTS); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         // 更新 TradeOrderDO 状态为已发货,等待收货 |         // 更新 TradeOrderDO 状态为已发货,等待收货 | ||||||
|   | |||||||
| @@ -77,7 +77,7 @@ public class TradeDeliveryPriceCalculator implements TradePriceCalculator { | |||||||
|             List<OrderItem> orderItems = entry.getValue(); |             List<OrderItem> orderItems = entry.getValue(); | ||||||
|             DeliveryExpressTemplateRespBO templateBO = expressTemplateMap.get(templateId); |             DeliveryExpressTemplateRespBO templateBO = expressTemplateMap.get(templateId); | ||||||
|             if (templateBO == null) { |             if (templateBO == null) { | ||||||
|                 log.error("不能计算快递运费。不能找到 templateId : {}. 对应的运费模板配置 Resp BO", templateId); |                 log.error("[calculateDeliveryPrice][不能计算快递运费,找不到 templateId({}) 对应的运费模板配置]", templateId); | ||||||
|                 continue; |                 continue; | ||||||
|             } |             } | ||||||
|             // 总件数, 总金额, 总重量, 总体积 |             // 总件数, 总金额, 总重量, 总体积 | ||||||
| @@ -93,12 +93,12 @@ public class TradeDeliveryPriceCalculator implements TradePriceCalculator { | |||||||
|             } |             } | ||||||
|             // 优先判断是否包邮. 如果包邮不计算快递运费 |             // 优先判断是否包邮. 如果包邮不计算快递运费 | ||||||
|             if (isExpressFree(templateBO.getChargeMode(), totalCount, totalWeight, |             if (isExpressFree(templateBO.getChargeMode(), totalCount, totalWeight, | ||||||
|                             totalVolume, totalPrice, templateBO.getTemplateFree())) { |                             totalVolume, totalPrice, templateBO.getFree())) { | ||||||
|                 continue; |                 continue; | ||||||
|             } |             } | ||||||
|             // 计算快递运费 |             // 计算快递运费 | ||||||
|             calculateExpressFeeByChargeMode(totalCount, totalWeight, totalVolume, |             calculateExpressFeeByChargeMode(totalCount, totalWeight, totalVolume, | ||||||
|                     templateBO.getChargeMode(), templateBO.getTemplateCharge(), orderItems); |                     templateBO.getChargeMode(), templateBO.getCharge(), orderItems); | ||||||
|  |  | ||||||
|         } |         } | ||||||
|         TradePriceCalculatorHelper.recountAllPrice(result); |         TradePriceCalculatorHelper.recountAllPrice(result); | ||||||
| @@ -115,10 +115,10 @@ public class TradeDeliveryPriceCalculator implements TradePriceCalculator { | |||||||
|      * @param orderItems SKU 商品项目 |      * @param orderItems SKU 商品项目 | ||||||
|      */ |      */ | ||||||
|     private void calculateExpressFeeByChargeMode(double totalCount, double totalWeight, double totalVolume, |     private void calculateExpressFeeByChargeMode(double totalCount, double totalWeight, double totalVolume, | ||||||
|                                                  int chargeMode, DeliveryExpressTemplateRespBO.DeliveryExpressTemplateChargeBO templateCharge, |                                                  int chargeMode, DeliveryExpressTemplateRespBO.Charge templateCharge, | ||||||
|                                                  List<OrderItem> orderItems) { |                                                  List<OrderItem> orderItems) { | ||||||
|         if (templateCharge == null) { |         if (templateCharge == null) { | ||||||
|             log.error("计算快递运费时,不能找到对应的快递运费模板费用配置。无法计算以下商品 SKU 项目运费: {}", orderItems); |             log.error("[calculateExpressFeeByChargeMode][计算快递运费时,找不到 SKU({}) 对应的运费模版]", orderItems); | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         DeliveryExpressChargeModeEnum chargeModeEnum = DeliveryExpressChargeModeEnum.valueOf(chargeMode); |         DeliveryExpressChargeModeEnum chargeModeEnum = DeliveryExpressChargeModeEnum.valueOf(chargeMode); | ||||||
| @@ -145,7 +145,7 @@ public class TradeDeliveryPriceCalculator implements TradePriceCalculator { | |||||||
|      * @param templateCharge 快递运费配置 |      * @param templateCharge 快递运费配置 | ||||||
|      * @param orderItems     SKU 商品项目 |      * @param orderItems     SKU 商品项目 | ||||||
|      */ |      */ | ||||||
|     private void calculateExpressFee(double total, DeliveryExpressTemplateRespBO.DeliveryExpressTemplateChargeBO templateCharge, List<OrderItem> orderItems) { |     private void calculateExpressFee(double total, DeliveryExpressTemplateRespBO.Charge templateCharge, List<OrderItem> orderItems) { | ||||||
|         int deliveryPrice; |         int deliveryPrice; | ||||||
|         if (total <= templateCharge.getStartCount()) { |         if (total <= templateCharge.getStartCount()) { | ||||||
|             deliveryPrice = templateCharge.getStartPrice(); |             deliveryPrice = templateCharge.getStartPrice(); | ||||||
| @@ -174,7 +174,6 @@ public class TradeDeliveryPriceCalculator implements TradePriceCalculator { | |||||||
|         for (OrderItem item : orderItems) { |         for (OrderItem item : orderItems) { | ||||||
|             // 更新快递运费 |             // 更新快递运费 | ||||||
|             item.setDeliveryPrice(dividePrice); |             item.setDeliveryPrice(dividePrice); | ||||||
|  |  | ||||||
|             TradePriceCalculatorHelper.recountPayPrice(item); |             TradePriceCalculatorHelper.recountPayPrice(item); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @@ -190,7 +189,7 @@ public class TradeDeliveryPriceCalculator implements TradePriceCalculator { | |||||||
|      * @param templateFree 包邮配置 |      * @param templateFree 包邮配置 | ||||||
|      */ |      */ | ||||||
|     private boolean isExpressFree(Integer chargeMode, int totalCount, double totalWeight, |     private boolean isExpressFree(Integer chargeMode, int totalCount, double totalWeight, | ||||||
|                                   double totalVolume, int totalPrice, DeliveryExpressTemplateRespBO.DeliveryExpressTemplateFreeBO templateFree) { |                                   double totalVolume, int totalPrice, DeliveryExpressTemplateRespBO.Free templateFree) { | ||||||
|         if (templateFree == null) { |         if (templateFree == null) { | ||||||
|             return false; |             return false; | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -1,8 +1,11 @@ | |||||||
| package cn.iocoder.yudao.module.trade.service.price.calculator; | package cn.iocoder.yudao.module.trade.service.price.calculator; | ||||||
|  |  | ||||||
|  | import cn.hutool.core.map.MapUtil; | ||||||
| import cn.iocoder.yudao.framework.test.core.ut.BaseMockitoUnitTest; | import cn.iocoder.yudao.framework.test.core.ut.BaseMockitoUnitTest; | ||||||
| import cn.iocoder.yudao.module.member.api.address.AddressApi; | import cn.iocoder.yudao.module.member.api.address.AddressApi; | ||||||
| import cn.iocoder.yudao.module.member.api.address.dto.AddressRespDTO; | import cn.iocoder.yudao.module.member.api.address.dto.AddressRespDTO; | ||||||
|  | import cn.iocoder.yudao.module.trade.enums.delivery.DeliveryExpressChargeModeEnum; | ||||||
|  | import cn.iocoder.yudao.module.trade.enums.delivery.DeliveryTypeEnum; | ||||||
| import cn.iocoder.yudao.module.trade.service.delivery.DeliveryExpressTemplateService; | import cn.iocoder.yudao.module.trade.service.delivery.DeliveryExpressTemplateService; | ||||||
| import cn.iocoder.yudao.module.trade.service.delivery.bo.DeliveryExpressTemplateRespBO; | import cn.iocoder.yudao.module.trade.service.delivery.bo.DeliveryExpressTemplateRespBO; | ||||||
| import cn.iocoder.yudao.module.trade.service.price.bo.TradePriceCalculateReqBO; | import cn.iocoder.yudao.module.trade.service.price.bo.TradePriceCalculateReqBO; | ||||||
| @@ -14,20 +17,17 @@ import org.mockito.InjectMocks; | |||||||
| import org.mockito.Mock; | import org.mockito.Mock; | ||||||
|  |  | ||||||
| import java.util.ArrayList; | import java.util.ArrayList; | ||||||
| import java.util.HashMap; |  | ||||||
| import java.util.Map; |  | ||||||
|  |  | ||||||
| import static cn.iocoder.yudao.framework.common.util.collection.SetUtils.asSet; | import static cn.iocoder.yudao.framework.common.util.collection.SetUtils.asSet; | ||||||
| import static cn.iocoder.yudao.framework.test.core.util.RandomUtils.randomPojo; | import static cn.iocoder.yudao.framework.test.core.util.RandomUtils.randomPojo; | ||||||
| import static cn.iocoder.yudao.module.trade.enums.delivery.DeliveryExpressChargeModeEnum.PIECE; |  | ||||||
| import static cn.iocoder.yudao.module.trade.enums.delivery.DeliveryTypeEnum.EXPRESS; |  | ||||||
| import static java.util.Arrays.asList; | import static java.util.Arrays.asList; | ||||||
|  | import static org.assertj.core.api.Assertions.assertThat; | ||||||
| import static org.mockito.ArgumentMatchers.eq; | import static org.mockito.ArgumentMatchers.eq; | ||||||
| import static org.mockito.Mockito.when; | import static org.mockito.Mockito.when; | ||||||
|  |  | ||||||
| import static org.assertj.core.api.Assertions.assertThat; |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  |  * {@link TradeDeliveryPriceCalculator} 的单元测试 | ||||||
|  |  * | ||||||
|  * @author jason |  * @author jason | ||||||
|  */ |  */ | ||||||
| public class TradeDeliveryPriceCalculatorTest  extends BaseMockitoUnitTest { | public class TradeDeliveryPriceCalculatorTest  extends BaseMockitoUnitTest { | ||||||
| @@ -41,22 +41,22 @@ public class TradeDeliveryPriceCalculatorTest  extends BaseMockitoUnitTest { | |||||||
|  |  | ||||||
|     private TradePriceCalculateReqBO reqBO; |     private TradePriceCalculateReqBO reqBO; | ||||||
|     private TradePriceCalculateRespBO resultBO; |     private TradePriceCalculateRespBO resultBO; | ||||||
|     private AddressRespDTO addressResp; |  | ||||||
|     private DeliveryExpressTemplateRespBO.DeliveryExpressTemplateChargeBO chargeBO; |  | ||||||
|     private DeliveryExpressTemplateRespBO.DeliveryExpressTemplateFreeBO freeBO; |  | ||||||
|     private DeliveryExpressTemplateRespBO templateRespBO; |     private DeliveryExpressTemplateRespBO templateRespBO; | ||||||
|  |     private DeliveryExpressTemplateRespBO.Charge chargeBO; | ||||||
|  |     private DeliveryExpressTemplateRespBO.Free freeBO; | ||||||
|  |  | ||||||
|     @BeforeEach |     @BeforeEach | ||||||
|     public void init(){ |     public void init(){ | ||||||
|         // 准备参数 |         // 准备参数 | ||||||
|         reqBO = new TradePriceCalculateReqBO() |         reqBO = new TradePriceCalculateReqBO() | ||||||
|                 .setDeliveryType(EXPRESS.getMode()) |                 .setDeliveryType(DeliveryTypeEnum.EXPRESS.getMode()) | ||||||
|                 .setAddressId(10L) |                 .setAddressId(10L) | ||||||
|                 .setUserId(1L) |                 .setUserId(1L) | ||||||
|                 .setItems(asList( |                 .setItems(asList( | ||||||
|                         new TradePriceCalculateReqBO.Item().setSkuId(10L).setCount(2).setSelected(true), |                         new TradePriceCalculateReqBO.Item().setSkuId(10L).setCount(2).setSelected(true), | ||||||
|                         new TradePriceCalculateReqBO.Item().setSkuId(20L).setCount(10).setSelected(true), |                         new TradePriceCalculateReqBO.Item().setSkuId(20L).setCount(10).setSelected(true), | ||||||
|                         new TradePriceCalculateReqBO.Item().setSkuId(30L).setCount(4).setSelected(false) |                         new TradePriceCalculateReqBO.Item().setSkuId(30L).setCount(4).setSelected(false) // 未选中 | ||||||
|                 )); |                 )); | ||||||
|         resultBO = new TradePriceCalculateRespBO() |         resultBO = new TradePriceCalculateRespBO() | ||||||
|                 .setPrice(new TradePriceCalculateRespBO.Price()) |                 .setPrice(new TradePriceCalculateRespBO.Price()) | ||||||
| @@ -72,18 +72,21 @@ public class TradeDeliveryPriceCalculatorTest  extends BaseMockitoUnitTest { | |||||||
|         // 保证价格被初始化上 |         // 保证价格被初始化上 | ||||||
|         TradePriceCalculatorHelper.recountPayPrice(resultBO.getItems()); |         TradePriceCalculatorHelper.recountPayPrice(resultBO.getItems()); | ||||||
|         TradePriceCalculatorHelper.recountAllPrice(resultBO); |         TradePriceCalculatorHelper.recountAllPrice(resultBO); | ||||||
|  |  | ||||||
|         // 准备收件地址数据 |         // 准备收件地址数据 | ||||||
|         addressResp = randomPojo(AddressRespDTO.class, item -> item.setAreaId(10)); |         AddressRespDTO addressResp = randomPojo(AddressRespDTO.class, item -> item.setAreaId(10)); | ||||||
|  |         when(addressApi.getAddress(eq(10L), eq(1L))).thenReturn(addressResp); | ||||||
|  |  | ||||||
|         // 准备运费模板费用配置数据 |         // 准备运费模板费用配置数据 | ||||||
|         chargeBO = randomPojo(DeliveryExpressTemplateRespBO.DeliveryExpressTemplateChargeBO.class, |         chargeBO = randomPojo(DeliveryExpressTemplateRespBO.Charge.class, | ||||||
|                 item -> item.setStartCount(10D).setStartPrice(1000).setExtraCount(10D).setExtraPrice(2000)); |                 item -> item.setStartCount(10D).setStartPrice(1000).setExtraCount(10D).setExtraPrice(2000)); | ||||||
|         // 准备运费模板包邮配置数据 订单总件数 < 包邮件数时 12 < 20 |         // 准备运费模板包邮配置数据:订单总件数 < 包邮件数时 12 < 20 | ||||||
|         freeBO = randomPojo(DeliveryExpressTemplateRespBO.DeliveryExpressTemplateFreeBO.class, |         freeBO = randomPojo(DeliveryExpressTemplateRespBO.Free.class, | ||||||
|                 item -> item.setFreeCount(20).setFreePrice(100)); |                 item -> item.setFreeCount(20).setFreePrice(100)); | ||||||
|         // 准备 SP 运费模板数据 |         // 准备 SP 运费模板数据 | ||||||
|         templateRespBO = randomPojo(DeliveryExpressTemplateRespBO.class, |         templateRespBO = randomPojo(DeliveryExpressTemplateRespBO.class, | ||||||
|                 item -> item.setChargeMode(PIECE.getType()) |                 item -> item.setChargeMode(DeliveryExpressChargeModeEnum.PIECE.getType()) | ||||||
|                         .setTemplateCharge(chargeBO).setTemplateFree(freeBO)); |                         .setCharge(chargeBO).setFree(freeBO)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Test |     @Test | ||||||
| @@ -92,32 +95,27 @@ public class TradeDeliveryPriceCalculatorTest  extends BaseMockitoUnitTest { | |||||||
|         // SKU 1 : 100 * 2  = 200 |         // SKU 1 : 100 * 2  = 200 | ||||||
|         // SKU 2 :200 * 10 = 2000 |         // SKU 2 :200 * 10 = 2000 | ||||||
|         // 运费  首件 1000 +  续件 2000 = 3000 |         // 运费  首件 1000 +  续件 2000 = 3000 | ||||||
|         Map<Long, DeliveryExpressTemplateRespBO> respMap = new HashMap<>(); |  | ||||||
|         respMap.put(1L, templateRespBO); |  | ||||||
|  |  | ||||||
|         // mock 方法 |         // mock 方法 | ||||||
|         when(addressApi.getAddress(eq(10L), eq(1L))).thenReturn(addressResp); |  | ||||||
|         when(deliveryExpressTemplateService.getExpressTemplateMapByIdsAndArea(eq(asSet(1L)), eq(10))) |         when(deliveryExpressTemplateService.getExpressTemplateMapByIdsAndArea(eq(asSet(1L)), eq(10))) | ||||||
|                 .thenReturn(respMap); |                 .thenReturn(MapUtil.of(1L, templateRespBO)); | ||||||
|  |  | ||||||
|  |         // 调用 | ||||||
|         calculator.calculate(reqBO, resultBO); |         calculator.calculate(reqBO, resultBO); | ||||||
|  |         // 断言 | ||||||
|         TradePriceCalculateRespBO.Price price = resultBO.getPrice(); |         TradePriceCalculateRespBO.Price price = resultBO.getPrice(); | ||||||
|  |  | ||||||
|         assertThat(price) |         assertThat(price) | ||||||
|                 .extracting("totalPrice","discountPrice","couponPrice","pointPrice","deliveryPrice","payPrice") |                 .extracting("totalPrice","discountPrice","couponPrice","pointPrice","deliveryPrice","payPrice") | ||||||
|                 .containsExactly(2200, 0, 0, 0, 3000,  5200); |                 .containsExactly(2200, 0, 0, 0, 3000,  5200); | ||||||
|         // 断言:SKU |  | ||||||
|         assertThat(resultBO.getItems()).hasSize(3); |         assertThat(resultBO.getItems()).hasSize(3); | ||||||
|         // SKU1 |         // 断言:SKU1 | ||||||
|         assertThat(resultBO.getItems().get(0)) |         assertThat(resultBO.getItems().get(0)) | ||||||
|                 .extracting("price", "count","discountPrice" ,"couponPrice", "pointPrice","deliveryPrice","payPrice") |                 .extracting("price", "count","discountPrice" ,"couponPrice", "pointPrice","deliveryPrice","payPrice") | ||||||
|                 .containsExactly(100, 2, 0, 0, 0, 1500, 1700); |                 .containsExactly(100, 2, 0, 0, 0, 1500, 1700); | ||||||
|         // SKU2 |         // 断言:SKU2 | ||||||
|         assertThat(resultBO.getItems().get(1)) |         assertThat(resultBO.getItems().get(1)) | ||||||
|                 .extracting("price", "count","discountPrice" ,"couponPrice", "pointPrice","deliveryPrice","payPrice") |                 .extracting("price", "count","discountPrice" ,"couponPrice", "pointPrice","deliveryPrice","payPrice") | ||||||
|                 .containsExactly(200, 10, 0, 0, 0, 1500, 3500); |                 .containsExactly(200, 10, 0, 0, 0, 1500, 3500); | ||||||
|         // SKU3 未选中 |         // 断言:SKU3 未选中 | ||||||
|         assertThat(resultBO.getItems().get(2)) |         assertThat(resultBO.getItems().get(2)) | ||||||
|                 .extracting("price", "count","discountPrice" ,"couponPrice", "pointPrice","deliveryPrice","payPrice") |                 .extracting("price", "count","discountPrice" ,"couponPrice", "pointPrice","deliveryPrice","payPrice") | ||||||
|                 .containsExactly(300, 1, 0, 0, 0, 0, 300); |                 .containsExactly(300, 1, 0, 0, 0, 0, 300); | ||||||
| @@ -129,38 +127,33 @@ public class TradeDeliveryPriceCalculatorTest  extends BaseMockitoUnitTest { | |||||||
|         // SKU 1 : 100 * 2  = 200 |         // SKU 1 : 100 * 2  = 200 | ||||||
|         // SKU 2 :200 * 10 = 2000 |         // SKU 2 :200 * 10 = 2000 | ||||||
|         // 运费  0 |         // 运费  0 | ||||||
|         Map<Long, DeliveryExpressTemplateRespBO> respMap = new HashMap<>(); |  | ||||||
|         respMap.put(1L, templateRespBO); |  | ||||||
|         // 准备运费模板包邮配置数据 包邮 订单总件数 > 包邮件数时 12 > 10 |  | ||||||
|         freeBO = randomPojo(DeliveryExpressTemplateRespBO.DeliveryExpressTemplateFreeBO.class, |  | ||||||
|                 item -> item.setFreeCount(10).setFreePrice(1000)); |  | ||||||
|         templateRespBO.setTemplateFree(freeBO); |  | ||||||
|         // mock 方法 |         // mock 方法 | ||||||
|         when(addressApi.getAddress(eq(10L), eq(1L))).thenReturn(addressResp); |         // 准备运费模板包邮配置数据 包邮 订单总件数 > 包邮件数时 12 > 10 | ||||||
|  |         templateRespBO.setFree(randomPojo(DeliveryExpressTemplateRespBO.Free.class, | ||||||
|  |                 item -> item.setFreeCount(10).setFreePrice(1000))); | ||||||
|         when(deliveryExpressTemplateService.getExpressTemplateMapByIdsAndArea(eq(asSet(1L)), eq(10))) |         when(deliveryExpressTemplateService.getExpressTemplateMapByIdsAndArea(eq(asSet(1L)), eq(10))) | ||||||
|                 .thenReturn(respMap); |                 .thenReturn(MapUtil.of(1L, templateRespBO)); | ||||||
|  |  | ||||||
|  |         // 调用 | ||||||
|         calculator.calculate(reqBO, resultBO); |         calculator.calculate(reqBO, resultBO); | ||||||
|  |         // 断言 | ||||||
|         TradePriceCalculateRespBO.Price price = resultBO.getPrice(); |         TradePriceCalculateRespBO.Price price = resultBO.getPrice(); | ||||||
|  |  | ||||||
|         // 断言price |  | ||||||
|         assertThat(price) |         assertThat(price) | ||||||
|                 .extracting("totalPrice","discountPrice","couponPrice","pointPrice","deliveryPrice","payPrice") |                 .extracting("totalPrice","discountPrice","couponPrice","pointPrice","deliveryPrice","payPrice") | ||||||
|                 .containsExactly(2200, 0, 0, 0, 0,  2200); |                 .containsExactly(2200, 0, 0, 0, 0,  2200); | ||||||
|         // 断言:SKU |  | ||||||
|         assertThat(resultBO.getItems()).hasSize(3); |         assertThat(resultBO.getItems()).hasSize(3); | ||||||
|         // SKU1 |         // 断言:SKU1 | ||||||
|         assertThat(resultBO.getItems().get(0)) |         assertThat(resultBO.getItems().get(0)) | ||||||
|                 .extracting("price", "count","discountPrice" ,"couponPrice", "pointPrice","deliveryPrice","payPrice") |                 .extracting("price", "count","discountPrice" ,"couponPrice", "pointPrice","deliveryPrice","payPrice") | ||||||
|                 .containsExactly(100, 2, 0, 0, 0, 0, 200); |                 .containsExactly(100, 2, 0, 0, 0, 0, 200); | ||||||
|         // SKU2 |         // 断言:SKU2 | ||||||
|         assertThat(resultBO.getItems().get(1)) |         assertThat(resultBO.getItems().get(1)) | ||||||
|                 .extracting("price", "count","discountPrice" ,"couponPrice", "pointPrice","deliveryPrice","payPrice") |                 .extracting("price", "count","discountPrice" ,"couponPrice", "pointPrice","deliveryPrice","payPrice") | ||||||
|                 .containsExactly(200, 10, 0, 0, 0, 0, 2000); |                 .containsExactly(200, 10, 0, 0, 0, 0, 2000); | ||||||
|         // SKU3 未选中 |         // 断言:SKU3 未选中 | ||||||
|         assertThat(resultBO.getItems().get(2)) |         assertThat(resultBO.getItems().get(2)) | ||||||
|                 .extracting("price", "count","discountPrice" ,"couponPrice", "pointPrice","deliveryPrice","payPrice") |                 .extracting("price", "count","discountPrice" ,"couponPrice", "pointPrice","deliveryPrice","payPrice") | ||||||
|                 .containsExactly(300, 1, 0, 0, 0, 0, 300); |                 .containsExactly(300, 1, 0, 0, 0, 0, 300); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
		Reference in New Issue
	
	Block a user
	 YunaiV
					YunaiV