diff --git a/product/pom.xml b/product/pom.xml
index 960d54248..eaf283845 100644
--- a/product/pom.xml
+++ b/product/pom.xml
@@ -20,6 +20,7 @@
product-rest
product-biz
product-biz-api
+ product-mq
diff --git a/product/product-biz-api/src/main/java/cn/iocoder/mall/product/biz/enums/category/ProductCategoryConstants.java b/product/product-biz-api/src/main/java/cn/iocoder/mall/product/biz/enums/category/ProductCategoryConstants.java
index 0613d0b76..4039f3cdf 100644
--- a/product/product-biz-api/src/main/java/cn/iocoder/mall/product/biz/enums/category/ProductCategoryConstants.java
+++ b/product/product-biz-api/src/main/java/cn/iocoder/mall/product/biz/enums/category/ProductCategoryConstants.java
@@ -1,10 +1,19 @@
package cn.iocoder.mall.product.biz.enums.category;
-public class ProductCategoryConstants {
+public interface ProductCategoryConstants {
+
+ /**
+ * 状态 - 开启
+ */
+ Integer STATUS_ENABLE = 1;
+ /**
+ * 状态 - 关闭
+ */
+ Integer STATUS_DISABLE = 2;
/**
* 父分类编号 - 根节点
*/
- public static final Integer PID_ROOT = 0;
+ Integer PID_ROOT = 0;
}
\ No newline at end of file
diff --git a/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/convert/product/ProductSpuConvert.java b/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/convert/sku/ProductSpuConvert.java
similarity index 93%
rename from product/product-biz/src/main/java/cn/iocoder/mall/product/biz/convert/product/ProductSpuConvert.java
rename to product/product-biz/src/main/java/cn/iocoder/mall/product/biz/convert/sku/ProductSpuConvert.java
index bc4f871d7..624980de3 100644
--- a/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/convert/product/ProductSpuConvert.java
+++ b/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/convert/sku/ProductSpuConvert.java
@@ -1,13 +1,13 @@
-package cn.iocoder.mall.product.biz.convert.product;
+package cn.iocoder.mall.product.biz.convert.sku;
import cn.iocoder.common.framework.util.StringUtil;
import cn.iocoder.mall.product.biz.bo.product.*;
import cn.iocoder.mall.product.biz.dataobject.category.ProductCategoryDO;
import cn.iocoder.mall.product.biz.dataobject.spu.ProductSkuDO;
import cn.iocoder.mall.product.biz.dataobject.spu.ProductSpuDO;
-import cn.iocoder.mall.product.biz.dto.product.ProductSkuAddOrUpdateDTO;
-import cn.iocoder.mall.product.biz.dto.product.ProductSpuAddDTO;
-import cn.iocoder.mall.product.biz.dto.product.ProductSpuUpdateDTO;
+import cn.iocoder.mall.product.biz.dto.sku.ProductSkuAddOrUpdateDTO;
+import cn.iocoder.mall.product.biz.dto.sku.ProductSpuAddDTO;
+import cn.iocoder.mall.product.biz.dto.sku.ProductSpuUpdateDTO;
import org.mapstruct.Mapper;
import org.mapstruct.Mapping;
import org.mapstruct.Mappings;
@@ -24,6 +24,17 @@ public interface ProductSpuConvert {
ProductSpuConvert INSTANCE = Mappers.getMapper(ProductSpuConvert.class);
+ @Mappings({
+ @Mapping(source = "picUrls", target = "picUrls", ignore = true)
+ })
+ ProductSpuDO convertToSpuDO(ProductSpuAddDTO productSpuAddDTO);
+
+
+ @Mappings({
+ @Mapping(source = "attrs", target = "attrs", ignore = true)
+ })
+ ProductSkuDO convertToSkuDO(ProductSkuAddOrUpdateDTO productSkuAddDTO);
+
@Mappings({
@Mapping(source = "picUrls", target = "picUrls", qualifiedByName = "translatePicUrlsFromString")
})
@@ -37,16 +48,6 @@ public interface ProductSpuConvert {
@Mappings({})
List convert(List spus);
- @Mappings({
- @Mapping(source = "picUrls", target = "picUrls", ignore = true)
- })
- ProductSpuDO convert(ProductSpuAddDTO productSpuAddDTO);
-
- @Mappings({
- @Mapping(source = "attrs", target = "attrs", ignore = true)
- })
- ProductSkuDO convert(ProductSkuAddOrUpdateDTO productSkuAddDTO);
-
@Mappings({
@Mapping(source = "picUrls", target = "picUrls", ignore = true)
diff --git a/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/dao/spu/ProductSkuMapper.java b/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/dao/sku/ProductSkuMapper.java
similarity index 79%
rename from product/product-biz/src/main/java/cn/iocoder/mall/product/biz/dao/spu/ProductSkuMapper.java
rename to product/product-biz/src/main/java/cn/iocoder/mall/product/biz/dao/sku/ProductSkuMapper.java
index da316d205..9a2179205 100644
--- a/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/dao/spu/ProductSkuMapper.java
+++ b/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/dao/sku/ProductSkuMapper.java
@@ -1,8 +1,9 @@
-package cn.iocoder.mall.product.biz.dao.spu;
+package cn.iocoder.mall.product.biz.dao.sku;
import cn.iocoder.mall.product.biz.dataobject.spu.ProductSkuDO;
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
import com.baomidou.mybatisplus.core.toolkit.Wrappers;
+import org.apache.ibatis.annotations.Param;
import org.springframework.stereotype.Repository;
import java.util.List;
@@ -16,4 +17,6 @@ public interface ProductSkuMapper extends BaseMapper {
.eq(ProductSkuDO::getStatus, status)
.eq(ProductSkuDO::getDeleted, false));
}
+
+ void insertList(@Param("productSkuDOs") List productSkuDOs);
}
diff --git a/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/dao/spu/ProductSpuMapper.java b/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/dao/sku/ProductSpuMapper.java
similarity index 85%
rename from product/product-biz/src/main/java/cn/iocoder/mall/product/biz/dao/spu/ProductSpuMapper.java
rename to product/product-biz/src/main/java/cn/iocoder/mall/product/biz/dao/sku/ProductSpuMapper.java
index 3e21a2d0e..4b001141c 100644
--- a/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/dao/spu/ProductSpuMapper.java
+++ b/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/dao/sku/ProductSpuMapper.java
@@ -1,4 +1,4 @@
-package cn.iocoder.mall.product.biz.dao.spu;
+package cn.iocoder.mall.product.biz.dao.sku;
import cn.iocoder.mall.product.biz.dataobject.spu.ProductSpuDO;
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
diff --git a/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/dao/spu/UserProductSpuCollectionsMapper.java b/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/dao/sku/UserProductSpuCollectionsMapper.java
similarity index 97%
rename from product/product-biz/src/main/java/cn/iocoder/mall/product/biz/dao/spu/UserProductSpuCollectionsMapper.java
rename to product/product-biz/src/main/java/cn/iocoder/mall/product/biz/dao/sku/UserProductSpuCollectionsMapper.java
index 9e1037d2b..ca604ec7e 100644
--- a/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/dao/spu/UserProductSpuCollectionsMapper.java
+++ b/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/dao/sku/UserProductSpuCollectionsMapper.java
@@ -1,4 +1,4 @@
-package cn.iocoder.mall.product.biz.dao.spu;
+package cn.iocoder.mall.product.biz.dao.sku;
import cn.iocoder.mall.product.biz.dataobject.spu.UserProductSpuCollectionsDO;
import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
diff --git a/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/dto/sku/ProductSkuAddOrUpdateDTO.java b/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/dto/sku/ProductSkuAddOrUpdateDTO.java
new file mode 100644
index 000000000..42a88e8b3
--- /dev/null
+++ b/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/dto/sku/ProductSkuAddOrUpdateDTO.java
@@ -0,0 +1,35 @@
+package cn.iocoder.mall.product.biz.dto.sku;
+
+import lombok.Data;
+import lombok.experimental.Accessors;
+
+import javax.validation.constraints.Min;
+import javax.validation.constraints.NotNull;
+import java.util.List;
+
+/**
+ * 商品 Sku 添加 DTO
+ */
+@Data
+@Accessors(chain = true)
+public class ProductSkuAddOrUpdateDTO {
+
+ /**
+ * 规格值数组
+ */
+ @NotNull(message = "规格值数组不能为空")
+ private List attrs;
+ /**
+ * 价格,单位:分
+ */
+ @NotNull(message = "价格不能为空")
+ @Min(value = 1L, message = "最小价格为 1")
+ private Integer price;
+ /**
+ * 库存数量
+ */
+ @NotNull(message = "库存数量不能为空")
+ @Min(value = 1L, message = "最小库存为 1")
+ private Integer quantity;
+
+}
diff --git a/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/dto/sku/ProductSpuAddDTO.java b/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/dto/sku/ProductSpuAddDTO.java
new file mode 100644
index 000000000..d4e0455d1
--- /dev/null
+++ b/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/dto/sku/ProductSpuAddDTO.java
@@ -0,0 +1,62 @@
+package cn.iocoder.mall.product.biz.dto.sku;
+
+import lombok.Data;
+import lombok.experimental.Accessors;
+
+import javax.validation.constraints.NotEmpty;
+import javax.validation.constraints.NotNull;
+import java.util.List;
+
+/**
+ * 商品 SPU + SKU 添加 DTO
+ */
+@Data
+@Accessors(chain = true)
+public class ProductSpuAddDTO {
+
+ // ========== 基本信息 =========
+ /**
+ * SPU 名字
+ */
+ @NotEmpty(message = "SPU 名字不能为空")
+ private String name;
+ /**
+ * 卖点
+ */
+ @NotEmpty(message = "卖点不能为空")
+ private String sellPoint;
+ /**
+ * 描述
+ */
+ @NotEmpty(message = "描述不能为空")
+ private String description;
+ /**
+ * 分类编号
+ */
+ @NotNull(message = "分类不能为空")
+ private Integer cid;
+ /**
+ * 商品主图地址
+ */
+ @NotNull(message = "商品主图不能为空")
+ private List picUrls;
+
+ // ========== 其他信息 =========
+ /**
+ * 是否上架商品(是否可见)。
+ *
+ * true 为已上架
+ * false 为已下架
+ */
+ @NotNull(message = "是否上架不能为空")
+ private Boolean visible;
+
+ // ========== SKU =========
+
+ /**
+ * SKU 数组
+ */
+ @NotNull(message = "SKU 不能为空")
+ private List skus;
+
+}
diff --git a/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/dto/sku/ProductSpuUpdateDTO.java b/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/dto/sku/ProductSpuUpdateDTO.java
new file mode 100644
index 000000000..1bf191478
--- /dev/null
+++ b/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/dto/sku/ProductSpuUpdateDTO.java
@@ -0,0 +1,69 @@
+package cn.iocoder.mall.product.biz.dto.sku;
+
+import cn.iocoder.mall.product.biz.dto.product.ProductSkuAddOrUpdateDTO;
+import lombok.Data;
+import lombok.experimental.Accessors;
+
+import javax.validation.constraints.NotEmpty;
+import javax.validation.constraints.NotNull;
+import java.util.List;
+
+/**
+ * 商品 SPU + SKU 更新 DTO
+ */
+@Data
+@Accessors(chain = true)
+public class ProductSpuUpdateDTO {
+
+ /**
+ * Spu 编号
+ */
+ @NotNull(message = "SPU 编号不能为空")
+ private Integer id;
+
+ // ========== 基本信息 =========
+ /**
+ * SPU 名字
+ */
+ @NotEmpty(message = "SPU 名字不能为空")
+ private String name;
+ /**
+ * 卖点
+ */
+ @NotEmpty(message = "卖点不能为空")
+ private String sellPoint;
+ /**
+ * 描述
+ */
+ @NotEmpty(message = "描述不能为空")
+ private String description;
+ /**
+ * 分类编号
+ */
+ @NotNull(message = "分类不能为空")
+ private Integer cid;
+ /**
+ * 商品主图地址
+ */
+ @NotNull(message = "商品主图不能为空")
+ private List picUrls;
+
+ // ========== 其他信息 =========
+ /**
+ * 是否上架商品(是否可见)。
+ *
+ * true 为已上架
+ * false 为已下架
+ */
+ @NotNull(message = "是否上架不能为空")
+ private Boolean visible;
+
+ // ========== SKU =========
+
+ /**
+ * SKU 数组
+ */
+ @NotNull(message = "SKU 不能为空")
+ private List skus;
+
+}
diff --git a/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/service/category/ProductCategoryService.java b/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/service/category/ProductCategoryService.java
index 2590d78bc..124b4f00e 100644
--- a/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/service/category/ProductCategoryService.java
+++ b/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/service/category/ProductCategoryService.java
@@ -1,6 +1,7 @@
package cn.iocoder.mall.product.biz.service.category;
import cn.iocoder.mall.product.biz.bo.category.ProductCategoryBO;
+import cn.iocoder.mall.product.biz.dataobject.category.ProductCategoryDO;
import cn.iocoder.mall.product.biz.dto.category.ProductCategoryAddDTO;
import cn.iocoder.mall.product.biz.dto.category.ProductCategoryDeleteDTO;
import cn.iocoder.mall.product.biz.dto.category.ProductCategoryUpdateDTO;
@@ -21,12 +22,14 @@ public interface ProductCategoryService {
/**
* 获取所有商品分类
+ *
* @return
*/
List getAllProductCategory();
/**
* 新增商品分类
+ *
* @param productCategoryAddDTO
* @return
*/
@@ -34,6 +37,7 @@ public interface ProductCategoryService {
/**
* 更新商品分类
+ *
* @param productCategoryUpdateDTO
* @return
*/
@@ -41,6 +45,7 @@ public interface ProductCategoryService {
/**
* 更新商品分类状态
+ *
* @param productCategoryUpdateStatusDTO
* @return
*/
@@ -48,8 +53,17 @@ public interface ProductCategoryService {
/**
* 删除商品分类
+ *
* @param productCategoryDeleteDTO
* @return
*/
Boolean deleteProductCategory(@Valid ProductCategoryDeleteDTO productCategoryDeleteDTO);
+
+ /**
+ * 校验分类是否可用
+ *
+ * @param productCategoryId 分类ID
+ * @return 商品分类
+ */
+ ProductCategoryDO validProductCategory(Integer productCategoryId);
}
diff --git a/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/service/category/ProductCategoryServiceImpl.java b/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/service/category/ProductCategoryServiceImpl.java
index 31e34bcef..c540ab8fb 100644
--- a/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/service/category/ProductCategoryServiceImpl.java
+++ b/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/service/category/ProductCategoryServiceImpl.java
@@ -10,11 +10,13 @@ import cn.iocoder.mall.product.biz.dto.category.ProductCategoryAddDTO;
import cn.iocoder.mall.product.biz.dto.category.ProductCategoryDeleteDTO;
import cn.iocoder.mall.product.biz.dto.category.ProductCategoryUpdateDTO;
import cn.iocoder.mall.product.biz.dto.category.ProductCategoryUpdateStatusDTO;
+import cn.iocoder.mall.product.biz.enums.ProductErrorCodeEnum;
import cn.iocoder.mall.product.biz.enums.category.ProductCategoryConstants;
import cn.iocoder.mall.product.biz.enums.category.ProductCategoryStatusEnum;
import com.baomidou.mybatisplus.core.toolkit.Wrappers;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
+
import java.util.Arrays;
import java.util.Date;
import java.util.List;
@@ -32,21 +34,12 @@ public class ProductCategoryServiceImpl implements ProductCategoryService {
@Autowired
private ProductCategoryMapper productCategoryMapper;
- /**
- * 获取所有商品分类
- * @return
- */
@Override
public List getAllProductCategory() {
List categoryList = productCategoryMapper.selectList(null);
return ProductCategoryConvert.INSTANCE.convertToAllListBO(categoryList);
}
- /**
- * 新增商品分类
- * @param productCategoryAddDTO
- * @return
- */
@Override
public ProductCategoryBO addProductCategory(ProductCategoryAddDTO productCategoryAddDTO) {
// 校验父分类
@@ -62,11 +55,6 @@ public class ProductCategoryServiceImpl implements ProductCategoryService {
return ProductCategoryConvert.INSTANCE.convertToBO(productCategory);
}
- /**
- * 更新商品分类
- * @param productCategoryUpdateDTO
- * @return
- */
@Override
public Boolean updateProductCategory(ProductCategoryUpdateDTO productCategoryUpdateDTO) {
// 校验当前分类是否存在
@@ -91,11 +79,6 @@ public class ProductCategoryServiceImpl implements ProductCategoryService {
return true;
}
- /**
- * 更新商品分类状态
- * @param productCategoryUpdateStatusDTO
- * @return
- */
@Override
public Boolean updateProductCategoryStatus(ProductCategoryUpdateStatusDTO productCategoryUpdateStatusDTO) {
// 校验商品分类是否存在
@@ -118,11 +101,6 @@ public class ProductCategoryServiceImpl implements ProductCategoryService {
return true;
}
- /**
- * 删除商品分类
- * @param productCategoryDeleteDTO
- * @return
- */
@Override
public Boolean deleteProductCategory(ProductCategoryDeleteDTO productCategoryDeleteDTO) {
Integer productCategoryId = productCategoryDeleteDTO.getId();
@@ -149,10 +127,6 @@ public class ProductCategoryServiceImpl implements ProductCategoryService {
return true;
}
- /**
- * 校验商品分类的父分类
- * @param pid
- */
private void validParent(Integer pid) {
if (!ProductCategoryConstants.PID_ROOT.equals(pid)) {
ProductCategoryDO parentCategory = productCategoryMapper.selectById(pid);
@@ -167,4 +141,18 @@ public class ProductCategoryServiceImpl implements ProductCategoryService {
}
}
+ @Override
+ public ProductCategoryDO validProductCategory(Integer productCategoryId) {
+ // 校验分类是否存在
+ ProductCategoryDO productCategory = productCategoryMapper.selectById(productCategoryId);
+ if (productCategory == null) {
+ throw ServiceExceptionUtil.exception(ProductErrorCodeEnum.PRODUCT_CATEGORY_NOT_EXISTS.getCode());
+ }
+ // 只有禁用的商品分类才可以删除
+ if (ProductCategoryConstants.STATUS_DISABLE.equals(productCategory.getStatus())) {
+ throw ServiceExceptionUtil.exception(ProductErrorCodeEnum.PRODUCT_CATEGORY_MUST_ENABLE.getCode());
+ }
+ // 返回结果
+ return productCategory;
+ }
}
diff --git a/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/service/spu/ProductSpuServiceImpl.java b/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/service/spu/ProductSpuServiceImpl.java
index c4f111329..a03d39452 100644
--- a/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/service/spu/ProductSpuServiceImpl.java
+++ b/product/product-biz/src/main/java/cn/iocoder/mall/product/biz/service/spu/ProductSpuServiceImpl.java
@@ -2,25 +2,30 @@ package cn.iocoder.mall.product.biz.service.spu;
import cn.iocoder.common.framework.util.ServiceExceptionUtil;
import cn.iocoder.common.framework.util.StringUtil;
+import cn.iocoder.mall.mybatis.enums.DeletedStatusEnum;
import cn.iocoder.mall.product.biz.bo.product.ProductAttrAndValuePairBO;
import cn.iocoder.mall.product.biz.bo.product.ProductSpuDetailBO;
-import cn.iocoder.mall.product.biz.convert.product.ProductSpuConvert;
+import cn.iocoder.mall.product.biz.convert.sku.ProductSpuConvert;
import cn.iocoder.mall.product.biz.dao.category.ProductCategoryMapper;
-import cn.iocoder.mall.product.biz.dao.spu.ProductSkuMapper;
-import cn.iocoder.mall.product.biz.dao.spu.ProductSpuMapper;
+import cn.iocoder.mall.product.biz.dao.sku.ProductSkuMapper;
+import cn.iocoder.mall.product.biz.dao.sku.ProductSpuMapper;
import cn.iocoder.mall.product.biz.dataobject.category.ProductCategoryDO;
import cn.iocoder.mall.product.biz.dataobject.spu.ProductSkuDO;
import cn.iocoder.mall.product.biz.dataobject.spu.ProductSpuDO;
+import cn.iocoder.mall.product.biz.dto.sku.ProductSkuAddOrUpdateDTO;
+import cn.iocoder.mall.product.biz.dto.sku.ProductSpuAddDTO;
import cn.iocoder.mall.product.biz.enums.ProductErrorCodeEnum;
+import cn.iocoder.mall.product.biz.enums.category.ProductCategoryConstants;
import cn.iocoder.mall.product.biz.enums.spu.ProductSpuConstants;
import cn.iocoder.mall.product.biz.service.attr.ProductAttrService;
+import cn.iocoder.mall.product.biz.service.category.ProductCategoryService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
+import org.springframework.transaction.annotation.Transactional;
import org.springframework.util.Assert;
-import java.util.HashSet;
-import java.util.List;
-import java.util.Set;
+import java.util.*;
+import java.util.stream.Collectors;
@Service
public class ProductSpuServiceImpl implements ProductSpuService {
@@ -33,6 +38,8 @@ public class ProductSpuServiceImpl implements ProductSpuService {
private ProductCategoryMapper productCategoryMapper;
@Autowired
private ProductAttrService productAttrService;
+ @Autowired
+ private ProductCategoryService productCategoryService;
@Override
public ProductSpuDetailBO getProductSpuDetail(Integer spuId) {
@@ -55,4 +62,109 @@ public class ProductSpuServiceImpl implements ProductSpuService {
return ProductSpuConvert.INSTANCE.convert2(spu, skus, attrAndValuePairList, category);
}
+ public ProductSpuDetailBO addProductSpu(Integer adminId, ProductSpuAddDTO productSpuAddDTO) {
+ ProductSpuDetailBO productSpuDetailBO = addProductSpu0(adminId, productSpuAddDTO);
+ // 如果新增生成,发送创建商品 Topic 消息
+ // TODO 芋艿,先不考虑事务的问题。等后面的 fescar 一起搞
+// sendProductUpdateMessage(productSpuDetailBO.getId());
+ // 返回成功
+ return productSpuDetailBO;
+ }
+
+ @SuppressWarnings("Duplicates")
+ @Transactional
+ public ProductSpuDetailBO addProductSpu0(Integer adminId, ProductSpuAddDTO productSpuAddDTO) {
+ // 校验商品分类分类存在
+ ProductCategoryDO category = productCategoryService.validProductCategory(productSpuAddDTO.getCid());
+ if (ProductCategoryConstants.PID_ROOT.equals(category.getPid())) {
+ // 商品只能添加到二级分类下
+ throw ServiceExceptionUtil.exception(ProductErrorCodeEnum.PRODUCT_SPU_CATEGORY_MUST_BE_LEVEL2.getCode());
+ }
+ // 校验规格是否存在
+ Set productAttrValueIds = new HashSet<>();
+ productSpuAddDTO.getSkus().forEach(productSkuAddDTO -> productAttrValueIds.addAll(productSkuAddDTO.getAttrs()));
+ // 读取规格时,需要考虑规格是否被禁用
+ List attrAndValuePairList = productAttrService.validProductAttrAndValue(productAttrValueIds, true);
+ // 保存 Spu
+ ProductSpuDO spu = ProductSpuConvert.INSTANCE.convertToSpuDO(productSpuAddDTO)
+ .setPicUrls(StringUtil.join(productSpuAddDTO.getPicUrls(), ","))
+ .setSort(0); // 排序为 0
+ spu.setCreateTime(new Date());
+ spu.setDeleted(DeletedStatusEnum.DELETED_NO.getValue());
+ // 初始化 sku 相关信息到 spu 中
+ initSpuFromSkus(spu, productSpuAddDTO.getSkus());
+ productSpuMapper.insert(spu);
+ // 保存 Sku
+ List skus = productSpuAddDTO.getSkus().stream().map(productSkuAddDTO -> {
+ ProductSkuDO sku = ProductSpuConvert.INSTANCE.convertToSkuDO(productSkuAddDTO)
+ .setSpuId(spu.getId())
+ .setStatus(ProductSpuConstants.SKU_STATUS_ENABLE)
+ .setAttrs(StringUtil.join(productSkuAddDTO.getAttrs(), ","));
+ sku.setCreateTime(new Date());
+ sku.setDeleted(DeletedStatusEnum.DELETED_NO.getValue());
+ return sku;
+ }).collect(Collectors.toList());
+ // 校验 Sku 规格
+ validProductSku(productSpuAddDTO.getSkus(), attrAndValuePairList);
+ // 插入 SKU 到数据库
+ productSkuMapper.insertList(skus);
+ // 返回成功
+ return ProductSpuConvert.INSTANCE.convert2(spu, skus, attrAndValuePairList, category);
+ }
+
+ /**
+ * 根据 sku 数组,计算相关的字段到 spu 中。
+ *
+ * @param spu spu
+ * @param skus sku 数组
+ */
+ private void initSpuFromSkus(ProductSpuDO spu, List skus) {
+ assert skus.size() > 0; // 写个断言,避免下面警告
+ spu.setPrice(skus.stream().min(Comparator.comparing(ProductSkuAddOrUpdateDTO::getPrice)).get().getPrice()); // 求最小价格
+ spu.setQuantity(skus.stream().mapToInt(ProductSkuAddOrUpdateDTO::getQuantity).sum()); // 求库存之和
+ }
+
+// private boolean sendProductUpdateMessage(Integer id) {
+// // 创建 Message 对象
+// ProductUpdateMessage message = new ProductUpdateMessage().setId(id);
+// // 创建 Spring Message 对象
+// Message springMessage = MessageBuilder.withPayload(message)
+// .build();
+// // 发送消息
+// return mqStreamProducer.productUpdateOutput().send(springMessage);
+// }
+
+ /**
+ * 校验 sku 是否合法
+ *
+ * @param productSkuAddDTOs sku 添加或修改信息
+ * @param productAttrDetailBOs 商品规格明细数组
+ */
+ private void validProductSku(List productSkuAddDTOs, List productAttrDetailBOs) {
+ // 创建 ProductAttrDetailBO 的映射。其中,KEY 为 ProductAttrDetailBO.attrValueId ,即规格值的编号
+ Map productAttrDetailBOMap = productAttrDetailBOs.stream().collect(
+ Collectors.toMap(ProductAttrAndValuePairBO::getAttrValueId, productAttrDetailBO -> productAttrDetailBO));
+ // 1. 先校验,一个 Sku 下,没有重复的规格。校验方式是,遍历每个 Sku ,看看是否有重复的规格 attrId
+ for (ProductSkuAddOrUpdateDTO sku : productSkuAddDTOs) {
+ Set attrIds = sku.getAttrs().stream().map(attrValueId -> productAttrDetailBOMap.get(attrValueId).getAttrId())
+ .collect(Collectors.toSet());
+ if (attrIds.size() != sku.getAttrs().size()) {
+ throw ServiceExceptionUtil.exception(ProductErrorCodeEnum.PRODUCT_SKU_ATTR_CANT_NOT_DUPLICATE.getCode());
+ }
+ }
+ // 2. 再校验,每个 Sku 的规格值的数量,是一致的。
+ int attrSize = productSkuAddDTOs.get(0).getAttrs().size();
+ for (int i = 1; i < productSkuAddDTOs.size(); i++) {
+ if (attrSize != productSkuAddDTOs.get(i).getAttrs().size()) {
+ throw ServiceExceptionUtil.exception(ProductErrorCodeEnum.PRODUCT_SPU_ATTR_NUMBERS_MUST_BE_EQUALS.getCode());
+ }
+ }
+ // 3. 最后校验,每个 Sku 之间不是重复的
+ Set> skuAttrValues = new HashSet<>(); // 每个元素,都是一个 Sku 的 attrValueId 集合。这样,通过最外层的 Set ,判断是否有重复的.
+ for (ProductSkuAddOrUpdateDTO sku : productSkuAddDTOs) {
+ if (!skuAttrValues.add(new HashSet<>(sku.getAttrs()))) { // 添加失败,说明重复
+ throw ServiceExceptionUtil.exception(ProductErrorCodeEnum.PRODUCT_SPU_SKU__NOT_DUPLICATE.getCode());
+ }
+ }
+ }
}
diff --git a/product/product-biz/src/main/resources/mapper/ProductSkuMapper.xml b/product/product-biz/src/main/resources/mapper/ProductSkuMapper.xml
new file mode 100644
index 000000000..8ce298bcd
--- /dev/null
+++ b/product/product-biz/src/main/resources/mapper/ProductSkuMapper.xml
@@ -0,0 +1,19 @@
+
+
+
+
+
+
+ INSERT INTO product_sku (
+ spu_id, status, pic_url, attrs, price,
+ quantity, deleted, create_time
+ ) VALUES
+
+ (#{productSkuDO.spuId}, #{productSkuDO.status}, #{productSkuDO.picUrl}, #{productSkuDO.attrs}, #{productSkuDO.price},
+ #{productSkuDO.quantity}, #{productSkuDO.deleted}, #{productSkuDO.createTime}
+ )
+
+
+
+
+
diff --git a/product/product-biz/src/main/resources/mapper/ProductSpuMapper.xml b/product/product-biz/src/main/resources/mapper/ProductSpuMapper.xml
index 7289e7d6d..c6f19b77c 100644
--- a/product/product-biz/src/main/resources/mapper/ProductSpuMapper.xml
+++ b/product/product-biz/src/main/resources/mapper/ProductSpuMapper.xml
@@ -1,6 +1,6 @@
-
+
diff --git a/product/product-mq/pom.xml b/product/product-mq/pom.xml
new file mode 100644
index 000000000..857d7b9c4
--- /dev/null
+++ b/product/product-mq/pom.xml
@@ -0,0 +1,15 @@
+
+
+
+ demo
+ cn.iocoder.mall
+ 1.0-SNAPSHOT
+
+ 4.0.0
+
+ product-mq
+
+
+
diff --git a/product/product-mq/src/main/java/cn/iocoder/mall/demo/mq/package-info.java b/product/product-mq/src/main/java/cn/iocoder/mall/demo/mq/package-info.java
new file mode 100644
index 000000000..8172d958c
--- /dev/null
+++ b/product/product-mq/src/main/java/cn/iocoder/mall/demo/mq/package-info.java
@@ -0,0 +1 @@
+package cn.iocoder.mall.demo.mq;