package com.ruoyi.measuringinstrumentledger.service.impl; import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.baomidou.mybatisplus.core.metadata.IPage; import com.baomidou.mybatisplus.core.toolkit.CollectionUtils; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; import com.ruoyi.basic.dto.StorageBlobDTO; import com.ruoyi.basic.pojo.StorageAttachment; import com.ruoyi.basic.service.StorageAttachmentService; import com.ruoyi.common.enums.FileNameType; import com.ruoyi.common.enums.StorageAttachmentRecordType; import com.ruoyi.common.utils.StringUtils; import com.ruoyi.common.utils.poi.ExcelUtil; import com.ruoyi.measuringinstrumentledger.dto.MeasuringInstrumentLedgerRecordDTO; import com.ruoyi.measuringinstrumentledger.mapper.MeasuringInstrumentLedgerMapper; import com.ruoyi.measuringinstrumentledger.mapper.MeasuringInstrumentLedgerRecordMapper; import com.ruoyi.measuringinstrumentledger.pojo.MeasuringInstrumentLedger; import com.ruoyi.measuringinstrumentledger.pojo.MeasuringInstrumentLedgerRecord; import com.ruoyi.measuringinstrumentledger.service.MeasuringInstrumentLedgerService; import com.ruoyi.other.mapper.TempFileMapper; import com.ruoyi.other.pojo.TempFile; import com.ruoyi.project.system.domain.SysUser; import com.ruoyi.project.system.mapper.SysUserMapper; import com.ruoyi.sales.mapper.CommonFileMapper; import com.ruoyi.sales.pojo.CommonFile; import lombok.extern.slf4j.Slf4j; import org.apache.commons.io.FilenameUtils; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.annotation.Value; import org.springframework.stereotype.Service; import javax.servlet.http.HttpServletResponse; import java.io.IOException; import java.nio.file.Files; import java.nio.file.Path; import java.nio.file.Paths; import java.nio.file.StandardCopyOption; import java.time.LocalDate; import java.time.LocalDateTime; import java.time.format.DateTimeFormatter; import java.util.ArrayList; import java.util.List; import java.util.Map; import java.util.UUID; import java.util.stream.Collectors; import static com.ruoyi.common.constant.StorageAttachmentConstants.StorageAttachmentFile; /** * @author :yys * @date : 2025/8/5 9:23 */ @Service @Slf4j public class MeasuringInstrumentLedgerServiceImpl extends ServiceImpl implements MeasuringInstrumentLedgerService { @Autowired private MeasuringInstrumentLedgerMapper measuringInstrumentLedgerMapper; @Autowired private MeasuringInstrumentLedgerRecordMapper measuringInstrumentLedgerRecordMapper; @Autowired private TempFileMapper tempFileMapper; @Autowired private CommonFileMapper commonFileMapper; @Autowired private SysUserMapper sysUserMapper; @Autowired private StorageAttachmentService attachmentService; @Value("${file.upload-dir}") private String uploadDir; @Override public IPage listPage(Page page, MeasuringInstrumentLedger measuringInstrumentLedger) { IPage iPage = measuringInstrumentLedgerMapper.listPage(page, measuringInstrumentLedger); List records = iPage.getRecords(); List ids = records.stream().map(MeasuringInstrumentLedger::getId).collect(Collectors.toList()); List measuringInstrumentLedgerRecords = measuringInstrumentLedgerRecordMapper.selectList(new LambdaQueryWrapper() .in(MeasuringInstrumentLedgerRecord::getMeasuringInstrumentLedgerId, ids) .orderByDesc(MeasuringInstrumentLedgerRecord::getCreateTime)); if (!CollectionUtils.isEmpty(measuringInstrumentLedgerRecords)) { Map> collect = measuringInstrumentLedgerRecords.stream().collect(Collectors.groupingBy(MeasuringInstrumentLedgerRecord::getMeasuringInstrumentLedgerId)); for (MeasuringInstrumentLedger ledger : records) { if (collect.containsKey(ledger.getId())) { ledger.setMostDate(collect.get(ledger.getId()).get(0).getRecordDate()); } } } return iPage; } @Override public boolean verifying(MeasuringInstrumentLedgerRecordDTO req) throws IOException { MeasuringInstrumentLedger measuringInstrumentLedger = measuringInstrumentLedgerMapper.selectById(req.getMeasuringInstrumentLedgerId()); if (measuringInstrumentLedger == null) { throw new RuntimeException("计量器具台账不存在"); } measuringInstrumentLedgerRecordMapper.insert(req); if (req.getBlobs() != null && !req.getBlobs().isEmpty()) { List attachments = new ArrayList<>(); for (StorageBlobDTO storageBlobDTO : req.getBlobs()) { StorageAttachment storageAttachment = new StorageAttachment( StorageAttachmentFile, (long) StorageAttachmentRecordType.MeasuringInstrumentLedgerRecord.ordinal(), req.getId() ); storageAttachment.setStorageBlobDTO(storageBlobDTO); attachments.add(storageAttachment); } attachmentService.saveStorageAttachment(attachments, req.getId(), StorageAttachmentRecordType.MeasuringInstrumentLedgerRecord, StorageAttachmentFile); } return true; } @Override public void export(HttpServletResponse response) { List list = measuringInstrumentLedgerMapper.listPage(new MeasuringInstrumentLedger()); ExcelUtil util = new ExcelUtil(MeasuringInstrumentLedger.class); util.exportExcel(response, list , "客户档案数据"); } @Override public boolean add(MeasuringInstrumentLedger measuringInstrumentLedger) throws IOException { SysUser sysUser = sysUserMapper.selectUserById(measuringInstrumentLedger.getUserId()); if (sysUser == null) { return false; } measuringInstrumentLedger.setUserName(sysUser.getUserName()); measuringInstrumentLedgerMapper.insert(measuringInstrumentLedger); if(!CollectionUtils.isEmpty(measuringInstrumentLedger.getTempFileIds())){ migrateTempFilesToFormal(measuringInstrumentLedger.getId(), measuringInstrumentLedger.getTempFileIds(), FileNameType.MEASURING.getValue()); } return true; } /** * 将临时文件迁移到正式目录 * * @param businessId 业务ID(销售台账ID) * @param tempFileIds 临时文件ID列表 * @throws IOException 文件操作异常 */ private void migrateTempFilesToFormal(Long businessId, List tempFileIds,Integer fileType) throws IOException { if (CollectionUtils.isEmpty(tempFileIds)) { return; } // 构建正式目录路径(按业务类型和日期分组) String formalDir = uploadDir + LocalDate.now().format(DateTimeFormatter.ISO_LOCAL_DATE); Path formalDirPath = Paths.get(formalDir); // 确保正式目录存在(递归创建) if (!Files.exists(formalDirPath)) { Files.createDirectories(formalDirPath); } for (String tempFileId : tempFileIds) { // 查询临时文件记录 TempFile tempFile = tempFileMapper.selectById(tempFileId); if (tempFile == null) { log.warn("临时文件不存在,跳过处理: {}", tempFileId); continue; } // 构建正式文件名(包含业务ID和时间戳,避免冲突) String originalFilename = tempFile.getOriginalName(); String fileExtension = FilenameUtils.getExtension(originalFilename); String formalFilename = businessId + "_" + System.currentTimeMillis() + "_" + UUID.randomUUID().toString().substring(0, 8) + (StringUtils.hasText(fileExtension) ? "." + fileExtension : ""); Path formalFilePath = formalDirPath.resolve(formalFilename); try { // 执行文件迁移(使用原子操作确保安全性) // Files.move( // Paths.get(tempFile.getTempPath()), // formalFilePath, // StandardCopyOption.REPLACE_EXISTING, // StandardCopyOption.ATOMIC_MOVE // ); // 原子移动失败,使用复制+删除 Files.copy(Paths.get(tempFile.getTempPath()), formalFilePath, StandardCopyOption.REPLACE_EXISTING); Files.deleteIfExists(Paths.get(tempFile.getTempPath())); log.info("文件迁移成功: {} -> {}", tempFile.getTempPath(), formalFilePath); // 更新文件记录(关联到业务ID) CommonFile fileRecord = new CommonFile(); fileRecord.setCommonId(businessId); fileRecord.setName(originalFilename); fileRecord.setUrl(formalFilePath.toString()); fileRecord.setCreateTime(LocalDateTime.now()); fileRecord.setType(fileType); commonFileMapper.insert(fileRecord); // 删除临时文件记录 tempFileMapper.deleteById(tempFile); log.info("文件迁移成功: {} -> {}", tempFile.getTempPath(), formalFilePath); } catch (IOException e) { log.error("文件迁移失败: {}", tempFile.getTempPath(), e); // 可选择回滚事务或记录失败文件 throw new IOException("文件迁移异常", e); } } } }