From 5706757a0bcec3318c8c460d34c1827788065d7c Mon Sep 17 00:00:00 2001 From: xuhy <3313886187@qq.com> Date: 星期三, 12 三月 2025 13:53:47 +0800 Subject: [PATCH] Merge branch 'xizang-changyun' of https://gitee.com/xiaochen991015/xizang --- ruoyi-common/src/main/java/com/ruoyi/common/utils/TencentMailUtil.java | 124 +++++++++++++++++++- generator/src/test/java/com/xizang/CodeGeneratorTests.java | 6 ruoyi-system/src/main/java/com/ruoyi/system/service/TInvoiceService.java | 1 ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TBillServiceImpl.java | 82 +++++++++---- ruoyi-admin/src/main/java/com/ruoyi/web/controller/api/TInvoiceController.java | 9 - ruoyi-system/src/main/java/com/ruoyi/system/dto/OfflinePayCheckDto.java | 6 ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TInvoiceServiceImpl.java | 99 ++++++++++++++++ 7 files changed, 282 insertions(+), 45 deletions(-) diff --git a/generator/src/test/java/com/xizang/CodeGeneratorTests.java b/generator/src/test/java/com/xizang/CodeGeneratorTests.java index 64dd914..8a5e34c 100644 --- a/generator/src/test/java/com/xizang/CodeGeneratorTests.java +++ b/generator/src/test/java/com/xizang/CodeGeneratorTests.java @@ -35,9 +35,9 @@ // 全局配置 GlobalConfig gc = new GlobalConfig(); - String projectPath = "F:\\workSpace\\xizang\\generator"; + String projectPath = "D:\\workspaces\\Project\\company\\changyun\\xizang\\xizang\\generator"; gc.setOutputDir(projectPath + "/src/main/java") - .setAuthor("xiaochen") + .setAuthor("yupeng") .setMapperName("%sMapper") .setXmlName("%sMapper") .setServiceName("%sService") @@ -144,7 +144,7 @@ // strategy.setTablePrefix(pc.getModuleName() + ""); // strategy.setLikeTable(new LikeTable("room")); //strategy.setLikeTable(new LikeTable("member")); - strategy.setLikeTable(new LikeTable("t_bank_flow"));// 生成表名 + strategy.setLikeTable(new LikeTable("sys_file"));// 生成表名 // strategy.setLikeTable(new LikeTable("t_hotel"));// 生成表名 // strategy.setLikeTable(new LikeTable("t_scan_message"));// 生成表名 // strategy.setNotLikeTable(new LikeTable("hotel_info"));// 不生成表名 diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/api/TInvoiceController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/api/TInvoiceController.java index f074175..157d180 100644 --- a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/api/TInvoiceController.java +++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/api/TInvoiceController.java @@ -5,6 +5,7 @@ import com.ruoyi.common.basic.PageInfo; import com.ruoyi.common.core.domain.R; import com.ruoyi.common.enums.BusinessType; +import com.ruoyi.common.utils.TencentMailUtil; import com.ruoyi.system.dto.TBillDto; import com.ruoyi.system.model.TBill; import com.ruoyi.system.model.TInvoice; @@ -57,13 +58,7 @@ @PostMapping("/uploadVoucher") @PreAuthorize("@ss.hasPermi('invoice:list:payment')") public R<Boolean> uploadVoucher(@RequestBody TInvoiceQuery query) { - TInvoice tInvoice = new TInvoice(); - tInvoice.setId(query.getId()); - tInvoice.setInvoiceVoucher(query.getInvoiceVoucher()); - tInvoice.setInvoiceVoucherName(query.getInvoiceVoucherName()); - tInvoice.setInvoiceTime(query.getInvoiceTime()); - tInvoice.setStatus(2); - return R.ok(invoiceService.updateById(tInvoice)); + return R.ok(invoiceService.uploadVoucher(query)); } } diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/TencentMailUtil.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/TencentMailUtil.java index 99d7bb9..4ce4421 100644 --- a/ruoyi-common/src/main/java/com/ruoyi/common/utils/TencentMailUtil.java +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/TencentMailUtil.java @@ -6,11 +6,20 @@ import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Component; +import javax.activation.DataHandler; +import javax.activation.FileDataSource; +import java.net.URLEncoder; +import javax.activation.URLDataSource; import javax.mail.*; -import javax.mail.internet.InternetAddress; -import javax.mail.internet.MimeMessage; +import javax.mail.internet.*; import java.io.UnsupportedEncodingException; +import java.net.MalformedURLException; +import java.net.URL; +import java.util.List; +import java.util.Map; +import java.util.Objects; import java.util.Properties; +import java.util.concurrent.CompletableFuture; @Component @Slf4j @@ -43,7 +52,8 @@ // 访问SMTP服务时需要提供的密码(在控制台选择发信地址进行设置) props.put("mail.password", properties.getPassword()); props.setProperty("mail.smtp.socketFactory.fallback", "false"); - props.put("mail.smtp.ssl.enable", "false"); + props.put("mail.smtp.ssl.enable", "true"); + props.put("mail.smtp.ssl.protocols", "TLSv1.2"); // 构建授权信息,用于进行SMTP进行身份验证 Authenticator authenticator = new Authenticator() { @Override @@ -94,17 +104,113 @@ e.printStackTrace(); log.error("发送邮件发生异常",e); } - } - public static void main(String[] args) { - TencentMailUtil tencentMailUtil = new TencentMailUtil(); - MailProperties properties = new MailProperties(); - tencentMailUtil.properties = properties; - tencentMailUtil.send("214491528@qq.com","大学城揽院24栋"); + public void sendInvoice(String emailAddress, List<Map<String, String>> list) { + // 异步发送邮件 + CompletableFuture.runAsync(() -> { + try { + sendEmail(emailAddress, list); + } catch (ServiceException e) { + log.error("邮件发送失败", e); + } + }); + } + private void sendEmail(String emailAddress, List<Map<String, String>> list) throws ServiceException { + try { + // 创建邮件会话 + Session mailSession = Session.getInstance(getMailProperties(), getAuthenticator()); + + // 创建邮件消息 + MimeMessage message = new MimeMessage(mailSession); + + // 设置发件人 + InternetAddress from = new InternetAddress(properties.getUserAddr(), properties.getUserName()); + message.setFrom(from); + + // 设置收件人 + InternetAddress to = new InternetAddress(emailAddress); + message.setRecipient(MimeMessage.RecipientType.TO, to); + + // 设置邮件标题 + message.setSubject("发票"); + + // 创建邮件内容 + Multipart multipart = createMultipart(list); + + // 设置邮件内容 + message.setContent(multipart); + + // 发送邮件 + Transport.send(message); + } catch (MessagingException | UnsupportedEncodingException | MalformedURLException e) { + log.error("发送邮件发生异常", e); + throw new ServiceException("发送邮件失败, 请检查"); + } + } + + private Properties getMailProperties() { + Properties props = new Properties(); + props.put("mail.smtp.auth", "true"); + props.put("mail.smtp.host", properties.getSmtpHost()); + props.put("mail.smtp.socketFactory.class", "javax.net.ssl.SSLSocketFactory"); + props.put("mail.smtp.socketFactory.port", properties.getSmtpPort()); + props.put("mail.smtp.port", properties.getSmtpPort()); + props.put("mail.user", properties.getUserAddr()); + props.put("mail.password", properties.getPassword()); + props.setProperty("mail.smtp.socketFactory.fallback", "false"); + props.put("mail.smtp.ssl.enable", "false"); + return props; + } + + private Authenticator getAuthenticator() { + return new Authenticator() { + @Override + protected PasswordAuthentication getPasswordAuthentication() { + String userName = properties.getUserAddr(); + String password = properties.getPassword(); + return new PasswordAuthentication(userName, password); + } + }; } + + private Multipart createMultipart(List<Map<String, String>> list) throws MessagingException, UnsupportedEncodingException, MalformedURLException { + Multipart multipart = new MimeMultipart(); + + // 添加文本消息部分 + BodyPart messageBodyPart = new MimeBodyPart(); + messageBodyPart.setHeader("Content-Type", "text/plain;charset=utf-8"); + multipart.addBodyPart(messageBodyPart); + + // 添加附件部分 + for (Map<String, String> map : list) { + messageBodyPart = new MimeBodyPart(); + String url = map.get("url"); + String fileName = map.get("fileName"); + + URLDataSource source = new URLDataSource(new URL(url)); + messageBodyPart.setDataHandler(new DataHandler(source)); + + String filenameEncode = MimeUtility.encodeText(fileName, "UTF-8", "base64"); + messageBodyPart.setFileName(filenameEncode); + messageBodyPart.setHeader("Content-Transfer-Encoding", "base64"); + messageBodyPart.setHeader("Content-Disposition", "attachment"); + messageBodyPart.setHeader("Content-Type", "application/octet-stream;name=\"" + filenameEncode + "\""); + + multipart.addBodyPart(messageBodyPart); + } + + return multipart; + } + + public static void main(String[] args) throws UnsupportedEncodingException { + TencentMailUtil tencentMailUtil = new TencentMailUtil(); + MailProperties properties = new MailProperties(); + tencentMailUtil.properties = properties; + tencentMailUtil.send("645025773@qq.com","大学城揽院24栋"); + } } diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/dto/OfflinePayCheckDto.java b/ruoyi-system/src/main/java/com/ruoyi/system/dto/OfflinePayCheckDto.java index 8a78c2d..51a9e92 100644 --- a/ruoyi-system/src/main/java/com/ruoyi/system/dto/OfflinePayCheckDto.java +++ b/ruoyi-system/src/main/java/com/ruoyi/system/dto/OfflinePayCheckDto.java @@ -38,7 +38,11 @@ @ApiModelProperty("支付凭证") private String voucher; @ApiModelProperty("银行流水ID") - @NotEmpty(message = "银行流水ID不能为空") private String flowId; + @ApiModelProperty("支付类型") + @NotEmpty(message = "支付类型不能为空") + private Integer payType; + + } diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/TInvoiceService.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/TInvoiceService.java index c637d9d..62a2e34 100644 --- a/ruoyi-system/src/main/java/com/ruoyi/system/service/TInvoiceService.java +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/TInvoiceService.java @@ -19,4 +19,5 @@ public interface TInvoiceService extends IService<TInvoice> { PageInfo<TInvoice> pageList(TInvoiceQuery query); List<TInvoice> makeQuery(TInvoiceQuery query); + Boolean uploadVoucher(TInvoiceQuery query); } diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TBillServiceImpl.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TBillServiceImpl.java index 6fab5c1..0376220 100644 --- a/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TBillServiceImpl.java +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TBillServiceImpl.java @@ -241,42 +241,72 @@ @Override @Transactional(rollbackFor = Exception.class) public boolean checkOfflinePay(OfflinePayCheckDto dto) { - TBankFlow bankflow = tBankFlowService.getById(dto.getFlowId()); TBillDto bill = getDetailByBillId(dto.getBillId()); - if (bankflow.getRemainingMoney().compareTo(BigDecimal.ZERO)<=0){ - throw new ServiceException("该流水已无可抵扣剩余金额"); - } - if (bankflow.getRemainingMoney().compareTo(dto.getAmount())<0){ - throw new ServiceException("实付金额不能高于于流水可抵扣剩余金额"); - } - //如果实付金额大于欠费金额 - if (dto.getAmount().compareTo(bill.getOutstandingMoney())>0){ - throw new ServiceException("实付金额不能高于该账单欠费金额"); - } + if (dto.getPayType()==1){ //银行 + if (StringUtils.isEmpty(dto.getFlowId())){ + throw new ServiceException("银行流水ID不能为空"); + } + TBankFlow bankflow = tBankFlowService.getById(dto.getFlowId()); + if (bankflow.getRemainingMoney().compareTo(BigDecimal.ZERO)<=0){ + throw new ServiceException("该流水已无可抵扣剩余金额"); + } + if (bankflow.getRemainingMoney().compareTo(dto.getAmount())<0){ + throw new ServiceException("实付金额不能高于于流水可抵扣剩余金额"); + } + //如果实付金额大于欠费金额 + if (dto.getAmount().compareTo(bill.getOutstandingMoney())>0){ + throw new ServiceException("实付金额不能高于该账单欠费金额"); + } + TBill billSave = new TBill(); + billSave.setId(bill.getId()); + billSave.setPayFeesMoney(dto.getAmount()); + billSave.setBankSerialNumber(bankflow.getBankSerialNumber()); + billSave.setPayFeesTime(bankflow.getPayTime()); + billSave.setVoucher(dto.getVoucher()); + billSave.setPayFeesType(2); + TBill back = lockAndUpdateInfo(billSave, 2); + TBankFlow saveBankFlow = new TBankFlow(); + saveBankFlow.setId(bankflow.getId()); + saveBankFlow.setDeductionMoney(bankflow.getDeductionMoney().add(dto.getAmount())); + BigDecimal subtract = bankflow.getRemainingMoney().subtract(dto.getAmount()); + saveBankFlow.setRemainingMoney(subtract); + if (BigDecimal.ZERO.compareTo(subtract) == 0){ + saveBankFlow.setFlowStatus(1); + } + tBankFlowService.updateById(saveBankFlow); + //更新银行流水的已抵扣金额和剩余可抵扣金额 + //存流水 + TFlowManagement save = new TFlowManagement(); + save.setPayType(3); + save.setPayer(dto.getPayer()); + save.setPayTime(bankflow.getPayTime()); + save.setSysSerialNumber(OrderNos.getDid(30)); + save.setBankSerialNumber(bankflow.getBankSerialNumber()); + save.setFlowType(2); + save.setPaymentBillId(back.getId()); + save.setDeductionMoney(back.getDeductionMoney()); + save.setFlowMoney(dto.getAmount()); + save.setRemainingMoney(back.getOutstandingMoney()); + save.setPreOutstand(back.getPreOutstand()); + tFlowManagementService.save(save); + return true; + } + //现金支付 TBill billSave = new TBill(); billSave.setId(bill.getId()); billSave.setPayFeesMoney(dto.getAmount()); - billSave.setBankSerialNumber(bankflow.getBankSerialNumber()); - billSave.setPayFeesTime(bankflow.getPayTime()); + billSave.setPayFeesTime(dto.getPayTime()!=null?dto.getPayTime():DateUtils.dateToLocalDateTime(new Date())); billSave.setVoucher(dto.getVoucher()); billSave.setPayFeesType(2); TBill back = lockAndUpdateInfo(billSave, 2); - //更新银行流水的已抵扣金额和剩余可抵扣金额 - TBankFlow saveBankFlow = new TBankFlow(); - saveBankFlow.setId(bankflow.getId()); - saveBankFlow.setDeductionMoney(bankflow.getDeductionMoney().add(dto.getAmount())); - saveBankFlow.setRemainingMoney(bankflow.getRemainingMoney().subtract(dto.getAmount())); - tBankFlowService.updateById(saveBankFlow); - //存流水 TFlowManagement save = new TFlowManagement(); save.setPayType(3); save.setPayer(dto.getPayer()); - save.setPayTime(bankflow.getPayTime()); + save.setPayTime(billSave.getPayFeesTime()); save.setSysSerialNumber(OrderNos.getDid(30)); - save.setBankSerialNumber(bankflow.getBankSerialNumber()); - save.setFlowType(2); + save.setFlowType(1); save.setPaymentBillId(back.getId()); save.setDeductionMoney(back.getDeductionMoney()); save.setFlowMoney(dto.getAmount()); @@ -483,7 +513,11 @@ TBankFlow saveBankFlow = new TBankFlow(); saveBankFlow.setId(bankflow.getId()); saveBankFlow.setDeductionMoney(bankflow.getDeductionMoney().add(dto.getAmount())); - saveBankFlow.setRemainingMoney(bankflow.getRemainingMoney().subtract(dto.getAmount())); + BigDecimal subtract = bankflow.getRemainingMoney().subtract(dto.getAmount()); + saveBankFlow.setRemainingMoney(subtract); + if (BigDecimal.ZERO.compareTo(subtract) == 0){ + saveBankFlow.setFlowStatus(1); + } tBankFlowService.updateById(saveBankFlow); } //存流水 diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TInvoiceServiceImpl.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TInvoiceServiceImpl.java index 36dbad6..c0be87b 100644 --- a/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TInvoiceServiceImpl.java +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TInvoiceServiceImpl.java @@ -3,7 +3,9 @@ import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; import com.ruoyi.common.basic.PageInfo; +import com.ruoyi.common.core.exception.ServiceException; import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.common.utils.TencentMailUtil; import com.ruoyi.system.mapper.TInvoiceMapper; import com.ruoyi.system.model.TInvoice; import com.ruoyi.system.query.TInvoiceQuery; @@ -12,7 +14,10 @@ import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Service; -import java.util.List; +import javax.annotation.Resource; +import java.util.*; +import java.util.concurrent.CompletableFuture; +import java.util.stream.Collectors; /** * <p> @@ -28,6 +33,8 @@ TInvoiceToBillServiceImpl tInvoiceToBillService; @Autowired TBillService tBillService; + @Resource + TencentMailUtil mailUtil; @Override public PageInfo<TInvoice> pageList(TInvoiceQuery query) { @@ -49,4 +56,94 @@ .orderByDesc(TInvoice::getCreateTime); return this.baseMapper.selectList(queryWrapper); } + + @Override + public Boolean uploadVoucher(TInvoiceQuery query) { + // 检查是否存在对应的发票记录 + TInvoice preexist = getById(query.getId()); + if (preexist == null) { + log.error("未找到对应的发票记录,ID: {}"+query.getId()); + return false; + } + + // 更新发票信息 + TInvoice tInvoice = new TInvoice(); + tInvoice.setId(query.getId()); + tInvoice.setInvoiceVoucher(query.getInvoiceVoucher()); + tInvoice.setInvoiceVoucherName(query.getInvoiceVoucherName()); + tInvoice.setInvoiceTime(query.getInvoiceTime()); + tInvoice.setStatus(2); + + // 处理附件信息 + List<Map<String, String>> attachments = buildAttachments(query.getInvoiceVoucher(), query.getInvoiceVoucherName()); + if (attachments.isEmpty()) { + log.warn("未找到有效的附件信息"); + return updateById(tInvoice); + } + + // 异步发送邮件 + CompletableFuture.runAsync(() -> { + try { + mailUtil.sendInvoice(preexist.getEmail(), attachments); + } catch (ServiceException e) { + log.error("邮件发送失败", e); + } + }); + + // 更新数据库 + return updateById(tInvoice); + } + + private List<Map<String, String>> buildAttachments(String invoiceVoucher, String invoiceVoucherName) { + if (invoiceVoucher == null || invoiceVoucherName == null) { + return Collections.emptyList(); + } + + String[] voucherUrls = invoiceVoucher.split(","); + String[] voucherNames = invoiceVoucherName.split(","); + + // 确保两个数组长度一致 + int length = Math.min(voucherUrls.length, voucherNames.length); + if (length == 0) { + return Collections.emptyList(); + } + + // 构建附件列表 + List<Map<String, String>> attachments = new ArrayList<>(length); + for (int i = 0; i < length; i++) { + Map<String, String> attachment = new HashMap<>(2); // 初始容量为2,避免扩容 + attachment.put("url", voucherUrls[i]); + attachment.put("fileName", voucherNames[i]); + attachments.add(attachment); + } + + return attachments; + } + + // @Override + // public Boolean uploadVoucher(TInvoiceQuery query) { + // TInvoice preexist = getById(query.getId()); + // if (preexist == null){ + // return false; + // } + // TInvoice tInvoice = new TInvoice(); + // tInvoice.setId(query.getId()); + // tInvoice.setInvoiceVoucher(query.getInvoiceVoucher()); + // tInvoice.setInvoiceVoucherName(query.getInvoiceVoucherName()); + // tInvoice.setInvoiceTime(query.getInvoiceTime()); + // tInvoice.setStatus(2); + // List<Map<String, String>> mapArrayList = new ArrayList<>(); + // String invoiceVoucher = query.getInvoiceVoucher(); + // String invoiceVoucherName = query.getInvoiceVoucherName(); + // + // List<String> list = Arrays.stream(invoiceVoucher.split(",")).collect(Collectors.toList()); + // for (int i = 0; i < list.size()-1; i++) { + // Map<String, String> map = new HashMap<>(); + // map.put("url", list.get(i)); + // map.put("fileName",invoiceVoucherName.split(",")[i]); + // mapArrayList.add(map); + // } + // mailUtil.sendInvoice(preexist.getEmail(),mapArrayList); + // return updateById(tInvoice); + // } } -- Gitblit v1.7.1