package cn.iocoder.yudao.module.erp.service.stock; import cn.hutool.core.collection.CollUtil; import cn.iocoder.yudao.framework.common.pojo.PageResult; import cn.iocoder.yudao.framework.common.util.number.MoneyUtils; import cn.iocoder.yudao.framework.common.util.object.BeanUtils; import cn.iocoder.yudao.module.erp.controller.admin.stock.vo.move.ErpStockMovePageReqVO; import cn.iocoder.yudao.module.erp.controller.admin.stock.vo.move.ErpStockMoveSaveReqVO; import cn.iocoder.yudao.module.erp.dal.dataobject.product.ErpProductDO; import cn.iocoder.yudao.module.erp.dal.dataobject.stock.ErpStockMoveDO; import cn.iocoder.yudao.module.erp.dal.dataobject.stock.ErpStockMoveItemDO; import cn.iocoder.yudao.module.erp.dal.mysql.stock.ErpStockMoveItemMapper; import cn.iocoder.yudao.module.erp.dal.mysql.stock.ErpStockMoveMapper; import cn.iocoder.yudao.module.erp.dal.redis.no.ErpNoRedisDAO; import cn.iocoder.yudao.module.erp.enums.ErpAuditStatus; import cn.iocoder.yudao.module.erp.enums.stock.ErpStockRecordBizTypeEnum; import cn.iocoder.yudao.module.erp.service.product.ErpProductService; import cn.iocoder.yudao.module.erp.service.stock.bo.ErpStockRecordCreateReqBO; import jakarta.annotation.Resource; import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional; import org.springframework.validation.annotation.Validated; import java.math.BigDecimal; import java.util.Collection; import java.util.Collections; import java.util.List; import java.util.Map; import java.util.stream.Stream; import static cn.iocoder.yudao.framework.common.exception.util.ServiceExceptionUtil.exception; import static cn.iocoder.yudao.framework.common.util.collection.CollectionUtils.*; import static cn.iocoder.yudao.module.erp.enums.ErrorCodeConstants.*; // TODO 芋艿:记录操作日志 /** * ERP 库存调拨单 Service 实现类 * * @author 芋道源码 */ @Service @Validated public class ErpStockMoveServiceImpl implements ErpStockMoveService { @Resource private ErpStockMoveMapper stockMoveMapper; @Resource private ErpStockMoveItemMapper stockMoveItemMapper; @Resource private ErpNoRedisDAO noRedisDAO; @Resource private ErpProductService productService; @Resource private ErpWarehouseService warehouseService; @Resource private ErpStockRecordService stockRecordService; @Override @Transactional(rollbackFor = Exception.class) public Long createStockMove(ErpStockMoveSaveReqVO createReqVO) { // 1.1 校验出库项的有效性 List stockMoveItems = validateStockMoveItems(createReqVO.getItems()); // 1.2 生成调拨单号,并校验唯一性 String no = noRedisDAO.generate(ErpNoRedisDAO.STOCK_MOVE_NO_PREFIX); if (stockMoveMapper.selectByNo(no) != null) { throw exception(STOCK_MOVE_NO_EXISTS); } // 2.1 插入出库单 ErpStockMoveDO stockMove = BeanUtils.toBean(createReqVO, ErpStockMoveDO.class, in -> in .setNo(no).setStatus(ErpAuditStatus.PROCESS.getStatus()) .setTotalCount(getSumValue(stockMoveItems, ErpStockMoveItemDO::getCount, BigDecimal::add)) .setTotalPrice(getSumValue(stockMoveItems, ErpStockMoveItemDO::getTotalPrice, BigDecimal::add, BigDecimal.ZERO))); stockMoveMapper.insert(stockMove); // 2.2 插入出库单项 stockMoveItems.forEach(o -> o.setMoveId(stockMove.getId())); stockMoveItemMapper.insertBatch(stockMoveItems); return stockMove.getId(); } @Override @Transactional(rollbackFor = Exception.class) public void updateStockMove(ErpStockMoveSaveReqVO updateReqVO) { // 1.1 校验存在 ErpStockMoveDO stockMove = validateStockMoveExists(updateReqVO.getId()); if (ErpAuditStatus.APPROVE.getStatus().equals(stockMove.getStatus())) { throw exception(STOCK_MOVE_UPDATE_FAIL_APPROVE, stockMove.getNo()); } // 1.2 校验出库项的有效性 List stockMoveItems = validateStockMoveItems(updateReqVO.getItems()); // 2.1 更新出库单 ErpStockMoveDO updateObj = BeanUtils.toBean(updateReqVO, ErpStockMoveDO.class, in -> in .setTotalCount(getSumValue(stockMoveItems, ErpStockMoveItemDO::getCount, BigDecimal::add)) .setTotalPrice(getSumValue(stockMoveItems, ErpStockMoveItemDO::getTotalPrice, BigDecimal::add))); stockMoveMapper.updateById(updateObj); // 2.2 更新出库单项 updateStockMoveItemList(updateReqVO.getId(), stockMoveItems); } @Override @Transactional(rollbackFor = Exception.class) public void updateStockMoveStatus(Long id, Integer status) { boolean approve = ErpAuditStatus.APPROVE.getStatus().equals(status); // 1.1 校验存在 ErpStockMoveDO stockMove = validateStockMoveExists(id); // 1.2 校验状态 if (stockMove.getStatus().equals(status)) { throw exception(approve ? STOCK_MOVE_APPROVE_FAIL : STOCK_MOVE_PROCESS_FAIL); } // 2. 更新状态 int updateCount = stockMoveMapper.updateByIdAndStatus(id, stockMove.getStatus(), new ErpStockMoveDO().setStatus(status)); if (updateCount == 0) { throw exception(approve ? STOCK_MOVE_APPROVE_FAIL : STOCK_MOVE_PROCESS_FAIL); } // 3. 变更库存 List stockMoveItems = stockMoveItemMapper.selectListByMoveId(id); Integer fromBizType = approve ? ErpStockRecordBizTypeEnum.MOVE_OUT.getType() : ErpStockRecordBizTypeEnum.MOVE_OUT_CANCEL.getType(); Integer toBizType = approve ? ErpStockRecordBizTypeEnum.MOVE_IN.getType() : ErpStockRecordBizTypeEnum.MOVE_IN_CANCEL.getType(); stockMoveItems.forEach(stockMoveItem -> { BigDecimal fromCount = approve ? stockMoveItem.getCount().negate() : stockMoveItem.getCount(); BigDecimal toCount = approve ? stockMoveItem.getCount() : stockMoveItem.getCount().negate(); stockRecordService.createStockRecord(new ErpStockRecordCreateReqBO( stockMoveItem.getProductId(), stockMoveItem.getFromWarehouseId(), fromCount, fromBizType, stockMoveItem.getMoveId(), stockMoveItem.getId(), stockMove.getNo())); stockRecordService.createStockRecord(new ErpStockRecordCreateReqBO( stockMoveItem.getProductId(), stockMoveItem.getToWarehouseId(), toCount, toBizType, stockMoveItem.getMoveId(), stockMoveItem.getId(), stockMove.getNo())); }); } private List validateStockMoveItems(List list) { // 1.1 校验产品存在 List productList = productService.validProductList( convertSet(list, ErpStockMoveSaveReqVO.Item::getProductId)); Map productMap = convertMap(productList, ErpProductDO::getId); // 1.2 校验仓库存在 warehouseService.validWarehouseList(convertSetByFlatMap(list, item -> Stream.of(item.getFromWarehouseId(), item.getToWarehouseId()))); // 2. 转化为 ErpStockMoveItemDO 列表 return convertList(list, o -> BeanUtils.toBean(o, ErpStockMoveItemDO.class, item -> item .setProductUnitId(productMap.get(item.getProductId()).getUnitId()) .setTotalPrice(MoneyUtils.priceMultiply(item.getProductPrice(), item.getCount())))); } private void updateStockMoveItemList(Long id, List newList) { // 第一步,对比新老数据,获得添加、修改、删除的列表 List oldList = stockMoveItemMapper.selectListByMoveId(id); List> diffList = diffList(oldList, newList, // id 不同,就认为是不同的记录 (oldVal, newVal) -> oldVal.getId().equals(newVal.getId())); // 第二步,批量添加、修改、删除 if (CollUtil.isNotEmpty(diffList.get(0))) { diffList.get(0).forEach(o -> o.setMoveId(id)); stockMoveItemMapper.insertBatch(diffList.get(0)); } if (CollUtil.isNotEmpty(diffList.get(1))) { stockMoveItemMapper.updateBatch(diffList.get(1)); } if (CollUtil.isNotEmpty(diffList.get(2))) { stockMoveItemMapper.deleteBatchIds(convertList(diffList.get(2), ErpStockMoveItemDO::getId)); } } @Override @Transactional(rollbackFor = Exception.class) public void deleteStockMove(List ids) { // 1. 校验不处于已审批 List stockMoves = stockMoveMapper.selectBatchIds(ids); if (CollUtil.isEmpty(stockMoves)) { return; } stockMoves.forEach(stockMove -> { if (ErpAuditStatus.APPROVE.getStatus().equals(stockMove.getStatus())) { throw exception(STOCK_MOVE_DELETE_FAIL_APPROVE, stockMove.getNo()); } }); // 2. 遍历删除,并记录操作日志 stockMoves.forEach(stockMove -> { // 2.1 删除出库单 stockMoveMapper.deleteById(stockMove.getId()); // 2.2 删除出库单项 stockMoveItemMapper.deleteByMoveId(stockMove.getId()); }); } private ErpStockMoveDO validateStockMoveExists(Long id) { ErpStockMoveDO stockMove = stockMoveMapper.selectById(id); if (stockMove == null) { throw exception(STOCK_MOVE_NOT_EXISTS); } return stockMove; } @Override public ErpStockMoveDO getStockMove(Long id) { return stockMoveMapper.selectById(id); } @Override public PageResult getStockMovePage(ErpStockMovePageReqVO pageReqVO) { return stockMoveMapper.selectPage(pageReqVO); } // ==================== 出库项 ==================== @Override public List getStockMoveItemListByMoveId(Long moveId) { return stockMoveItemMapper.selectListByMoveId(moveId); } @Override public List getStockMoveItemListByMoveIds(Collection moveIds) { if (CollUtil.isEmpty(moveIds)) { return Collections.emptyList(); } return stockMoveItemMapper.selectListByMoveIds(moveIds); } }