| | |
| | | package com.ruoyi.system.service.impl; |
| | | |
| | | import cn.hutool.core.date.DateUtil; |
| | | import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; |
| | | import com.ruoyi.common.basic.PageInfo; |
| | | import com.ruoyi.common.config.SmsProperties; |
| | | import com.ruoyi.common.constant.AmountConstant; |
| | | import com.ruoyi.common.constant.CacheConstants; |
| | | import com.ruoyi.common.core.redis.RedisCache; |
| | | import com.ruoyi.common.enums.DisabledEnum; |
| | | import com.ruoyi.common.exception.ServiceException; |
| | | import com.ruoyi.common.utils.*; |
| | | import com.ruoyi.common.utils.uuid.UUID; |
| | | import com.ruoyi.system.dto.*; |
| | | import com.ruoyi.system.dto.BillStatisticsDto; |
| | | import com.ruoyi.system.dto.CachPayDto; |
| | | import com.ruoyi.system.dto.OfflinePayCheckDto; |
| | | import com.ruoyi.system.dto.SmsByBillDto; |
| | | import com.ruoyi.system.dto.TBillDto; |
| | | import com.ruoyi.system.dto.TbillSaveDto; |
| | | import com.ruoyi.system.mapper.TBillMapper; |
| | | import com.ruoyi.system.model.*; |
| | | import com.ruoyi.system.model.TBankFlow; |
| | | import com.ruoyi.system.model.TBill; |
| | | import com.ruoyi.system.model.TBillDetail; |
| | | import com.ruoyi.system.model.TFlowManagement; |
| | | import com.ruoyi.system.model.TInvoiceToBill; |
| | | import com.ruoyi.system.model.TOrderBill; |
| | | import com.ruoyi.system.model.TPayOrder; |
| | | import com.ruoyi.system.query.TBillQuery; |
| | | import com.ruoyi.system.query.TInvoiceToBillQuery; |
| | | import com.ruoyi.system.service.*; |
| | | import com.ruoyi.system.service.TBankFlowService; |
| | | import com.ruoyi.system.service.TBillConfirmService; |
| | | import com.ruoyi.system.service.TBillDetailService; |
| | | import com.ruoyi.system.service.TBillService; |
| | | import com.ruoyi.system.service.TFlowManagementService; |
| | | import com.ruoyi.system.service.TInvoiceToBillService; |
| | | import com.ruoyi.system.service.TOrderBillService; |
| | | import com.ruoyi.system.service.TPayOrderService; |
| | | import com.ruoyi.system.vo.ScreenRentRankVO; |
| | | import com.taxi591.bankapi.dto.ChargeBillRequest; |
| | | import lombok.extern.slf4j.Slf4j; |
| | | import org.jetbrains.annotations.NotNull; |
| | | import org.joda.time.LocalDateTime; |
| | | import org.springframework.beans.BeanUtils; |
| | | import org.springframework.beans.factory.annotation.Autowired; |
| | | import org.springframework.stereotype.Service; |
| | |
| | | import javax.validation.constraints.NotEmpty; |
| | | import java.math.BigDecimal; |
| | | import java.text.ParseException; |
| | | import java.time.LocalDateTime; |
| | | import java.util.ArrayList; |
| | | import java.util.Comparator; |
| | | import java.util.Date; |
| | |
| | | return info; |
| | | } |
| | | |
| | | public PageInfo<TBillDto> queryPageForApplet(TBillQuery query){ |
| | | PageInfo<TBill> pageInfo = new PageInfo<>(query.getPageNum(), query.getPageSize()); |
| | | PageInfo<TBillDto> info = tBillMapper.pageForApplet(pageInfo, query); |
| | | return info; |
| | | } |
| | | |
| | | @Override |
| | | public List<String> getBillIds(TBillQuery query) { |
| | | List<TBillDto> billDtos = tBillMapper.getBillList(query); |
| | |
| | | if (isok){ |
| | | try { |
| | | TBill save = new TBill(); |
| | | save.setId(tBill.getId()); |
| | | TBill presist = getById(tBill.getId()); |
| | | //如果账单是已缴费状态,本方法不再进行更新账单 |
| | | if (presist.getPayFeesStatus().equals("3")){ |
| | | throw new ServiceException("该账单已缴费完成"); |
| | | } |
| | | |
| | | switch (type){ |
| | | // 仅更新除金额字段外的属性 |
| | | case 1: |
| | |
| | | BigDecimal outstand = presist.getPayableFeesMoney() |
| | | .add(save.getPayableFeesPenalty()) |
| | | .subtract(presist.getPayFeesMoney()); |
| | | |
| | | save.setOutstandingMoney(outstand); |
| | | } |
| | | //处理缴费金额 |
| | |
| | | .subtract(save.getPayFeesMoney()); |
| | | save.setOutstandingMoney(outstand); |
| | | //抵扣金额就是缴费金额 |
| | | save.setDeductionMoney(tBill.getPayableFeesMoney()); |
| | | save.setDeductionMoney(tBill.getPayFeesMoney()); |
| | | save.setPreOutstand(presist.getOutstandingMoney()); |
| | | if (outstand.compareTo(BigDecimal.ZERO)<=0){ |
| | | save.setPayFeesStatus("3"); |
| | |
| | | break; |
| | | |
| | | } |
| | | save.setBusinessDeptId(presist.getBusinessDeptId()); |
| | | updateById(save); |
| | | return save; |
| | | }finally { |
| | |
| | | @Override |
| | | @Transactional(rollbackFor = Exception.class) |
| | | public void saveBill(TbillSaveDto bill) { |
| | | bill.setManualAddition(DisabledEnum.YES.getCode()); |
| | | save(bill); |
| | | if (bill.getBillType().equals("3")){ |
| | | if (bill.getDetails()==null || bill.getDetails().size()==0){ |
| | |
| | | @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.setBusinessDeptId(bill.getBusinessDeptId()); |
| | | 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.setBusinessDeptId(bill.getBusinessDeptId()); |
| | | 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()); |
| | |
| | | String uuid = UUID.fastUUID().toString(); |
| | | boolean lock = redisCache.trylockLoop(CacheConstants.COMPLETE_PAY_LOCK_KEY + orderNo, uuid, 60); |
| | | if (lock){ |
| | | TPayOrder order = tPayOrderService.getById(orderNo); |
| | | if (order==null){ |
| | | throw new ServiceException("订单不存在"); |
| | | } |
| | | if (StringUtils.isNotEmpty(order.getPayNo())){ |
| | | log.info("订单号已处理:{}",orderNo); |
| | | return; |
| | | } |
| | | /** |
| | | * 更新订单状态 |
| | | */ |
| | | TPayOrder save = new TPayOrder(); |
| | | save.setId(order.getId()); |
| | | save.setStatus(1); |
| | | save.setPayNo(billRequest.getMessage().getInfo().getTraceNo()); |
| | | save.setPayType(billRequest.getMessage().getHead().getChannel()); |
| | | try { |
| | | save.setPayTime(DateUtils.parseDate(billRequest.getMessage().getHead().getTimeStamp(),"yyyyMMddHHmmssSSS")); |
| | | } catch (ParseException e) { |
| | | throw new ServiceException("日期格式化错误"); |
| | | TPayOrder order = tPayOrderService.getById(orderNo); |
| | | if (order==null){ |
| | | throw new ServiceException("订单不存在"); |
| | | } |
| | | if (StringUtils.isNotEmpty(order.getPayNo())){ |
| | | log.info("订单号已处理:{}",orderNo); |
| | | return; |
| | | } |
| | | /** |
| | | * 更新订单状态 |
| | | */ |
| | | TPayOrder save = new TPayOrder(); |
| | | save.setId(order.getId()); |
| | | save.setStatus(1); |
| | | save.setPayNo(billRequest.getMessage().getInfo().getTraceNo()); |
| | | save.setPayType(billRequest.getMessage().getHead().getChannel()); |
| | | try { |
| | | save.setPayTime(DateUtils.parseDate(billRequest.getMessage().getHead().getTimeStamp(),"yyyyMMddHHmmssSSS")); |
| | | } catch (ParseException e) { |
| | | throw new ServiceException("日期格式化错误"); |
| | | } |
| | | save.setCallbackTime(new Date()); |
| | | BigDecimal payAmount = new BigDecimal(billRequest.getMessage().getInfo().getPayBillAmt()); |
| | | save.setActPayAmount(payAmount |
| | | .multiply(AmountConstant.b100).longValue()); |
| | | save.setStatus(1); |
| | | save.setPayInfo(billRequest.getMessage().toString()); |
| | | tPayOrderService.updateById(save); |
| | | /** |
| | | * 更新账单状态 |
| | | */ |
| | | List<TOrderBill> orderBills = orderBillService.getByOrderNo(order.getId()); |
| | | List<TBill> bills = orderBills.stream().map(ob -> getById(ob.getBillId())).collect(Collectors.toList()); |
| | | lockAndUpdateByAmountBatch(bills,payAmount,(bill)->{ |
| | | TFlowManagement saveFlow = new TFlowManagement(); |
| | | saveFlow.setPayType(1); |
| | | saveFlow.setPayer(order.getUserId()); |
| | | saveFlow.setBusinessDeptId(bill.getBusinessDeptId()); |
| | | saveFlow.setPayTime(DateUtils.dateToLocalDateTime(save.getPayTime())); |
| | | saveFlow.setSysSerialNumber(OrderNos.getDid(30)); |
| | | saveFlow.setBankSerialNumber(save.getPayNo()); |
| | | saveFlow.setFlowType(2); |
| | | saveFlow.setPaymentBillId(bill.getId()); |
| | | saveFlow.setDeductionMoney(bill.getDeductionMoney()); |
| | | saveFlow.setFlowMoney(payAmount); |
| | | saveFlow.setRemainingMoney(bill.getOutstandingMoney()); |
| | | saveFlow.setPreOutstand(bill.getPreOutstand()); |
| | | tFlowManagementService.save(saveFlow); |
| | | }); |
| | | // TBankFlow bankFlow = new TBankFlow(); |
| | | // bankFlow.setPayType(1); |
| | | // bankFlow.setPayer(order.getUserId()); |
| | | // bankFlow.setPayTime(DateUtils.dateToLocalDateTime(save.getPayTime())); |
| | | // bankFlow.setBankSerialNumber(save.getPayNo()); |
| | | // bankFlow.setFlowMoney(payAmount); |
| | | // bankFlow.setFlowStatus(1); |
| | | // tBankFlowService.save(bankFlow); |
| | | }finally { |
| | | redisCache.unlock(CacheConstants.COMPLETE_PAY_LOCK_KEY + orderNo,uuid); |
| | | } |
| | | save.setCallbackTime(new Date()); |
| | | BigDecimal payAmount = new BigDecimal(billRequest.getMessage().getInfo().getPayBillAmt()); |
| | | save.setActPayAmount(payAmount |
| | | .multiply(AmountConstant.b100).longValue()); |
| | | save.setStatus(1); |
| | | save.setPayInfo(billRequest.getMessage().toString()); |
| | | tPayOrderService.updateById(save); |
| | | /** |
| | | * 更新账单状态 |
| | | */ |
| | | List<TOrderBill> orderBills = orderBillService.getByOrderNo(order.getId()); |
| | | List<TBill> bills = orderBills.stream().map(ob -> getById(ob.getBillId())).collect(Collectors.toList()); |
| | | lockAndUpdateByAmountBatch(bills,payAmount,(bill)->{ |
| | | TFlowManagement saveFlow = new TFlowManagement(); |
| | | saveFlow.setPayType(1); |
| | | saveFlow.setPayer(order.getUserId()); |
| | | saveFlow.setPayTime(DateUtils.dateToLocalDateTime(save.getPayTime())); |
| | | saveFlow.setSysSerialNumber(OrderNos.getDid(30)); |
| | | saveFlow.setBankSerialNumber(save.getPayNo()); |
| | | saveFlow.setFlowType(2); |
| | | saveFlow.setPaymentBillId(bill.getId()); |
| | | saveFlow.setDeductionMoney(bill.getDeductionMoney()); |
| | | saveFlow.setFlowMoney(payAmount); |
| | | saveFlow.setRemainingMoney(bill.getOutstandingMoney()); |
| | | saveFlow.setPreOutstand(bill.getPreOutstand()); |
| | | tFlowManagementService.save(saveFlow); |
| | | }); |
| | | TBankFlow bankFlow = new TBankFlow(); |
| | | bankFlow.setPayType(1); |
| | | bankFlow.setPayer(order.getUserId()); |
| | | bankFlow.setPayTime(DateUtils.dateToLocalDateTime(save.getPayTime())); |
| | | bankFlow.setBankSerialNumber(save.getPayNo()); |
| | | bankFlow.setFlowMoney(payAmount); |
| | | bankFlow.setFlowStatus(1); |
| | | tBankFlowService.save(bankFlow); |
| | | |
| | | } |
| | | |
| | | } |
| | | |
| | | |
| | | /** |
| | | * 根据发票编号查询账单列表 |
| | |
| | | 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); |
| | | } |
| | | //存流水 |
| | | TFlowManagement save = new TFlowManagement(); |
| | | save.setPayType(3); |
| | | save.setPayer(dto.getPayer()); |
| | | save.setBusinessDeptId(bill.getBusinessDeptId()); |
| | | save.setPayTime(bankflow!=null?bankflow.getPayTime():DateUtils.dateToLocalDateTime(new Date())); |
| | | save.setSysSerialNumber(OrderNos.getDid()); |
| | | save.setBankSerialNumber(bankflow!=null?bankflow.getBankSerialNumber():null); |
| | |
| | | } |
| | | |
| | | @Override |
| | | public BillStatisticsDto statistics() { |
| | | public BillStatisticsDto statistics(TBillQuery query) { |
| | | BillStatisticsDto dto = new BillStatisticsDto(); |
| | | dto.setRent(getBaseMapper().statisticsAllRent()); |
| | | dto.setNopay(getBaseMapper().statisticsNoPay()); |
| | | dto.setPayed(getBaseMapper().statisticsPayed()); |
| | | dto.setOverdue(getBaseMapper().statisticsOverdue()); |
| | | dto.setRent(getBaseMapper().statisticsAllRent(query)); |
| | | dto.setNopay(getBaseMapper().statisticsNoPay(query)); |
| | | dto.setPayed(getBaseMapper().statisticsPayed(query)); |
| | | dto.setOverdue(getBaseMapper().statisticsOverdue(query)); |
| | | return dto; |
| | | } |
| | | |
| | | @Override |
| | | public Integer batchBillCount(String userId, List<String> billIds) { |
| | | return this.baseMapper.batchBillCount(userId,billIds); |
| | | } |
| | | |
| | | /** |
| | | * 街道租金排行 |
| | | * @return |
| | | */ |
| | | @Override |
| | | public List<ScreenRentRankVO> getStreetRentRank(String businessDeptId) { |
| | | return baseMapper.getStreetRentRank(businessDeptId); |
| | | } |
| | | /** |
| | | * 查询季付账单 |
| | | * @param businessDeptId |
| | | * @return |
| | | */ |
| | | @Override |
| | | public List<TBill> getJiFuBillList(String businessDeptId) { |
| | | return baseMapper.getJiFuBillList(businessDeptId,null,null); |
| | | } |
| | | |
| | | /** |
| | | * 查询当前季度的季付账单 |
| | | * @param businessDeptId |
| | | * @param first |
| | | * @param last |
| | | * @return |
| | | */ |
| | | @Override |
| | | public List<TBill> getJiFuBillListByTime(String businessDeptId, Date first, Date last) { |
| | | return baseMapper.getJiFuBillList(businessDeptId,first,last); |
| | | } |
| | | |
| | | @Override |
| | | public void editAmount(TbillSaveDto bill) { |
| | | String requestId = UUID.fastUUID().toString(); |
| | | String lockkey = CacheConstants.BILL_UPDATE_LOCK_KEY + bill.getId(); |
| | | boolean isok = redisCache.trylockLoop(lockkey, requestId, 60); |
| | | if (isok){ |
| | | try { |
| | | TBill presist = getById(bill.getId()); |
| | | TBill save = new TBill(); |
| | | save.setId(bill.getId()); |
| | | BigDecimal preOutstand = presist.getOutstandingMoney(); |
| | | // 如果传入的金额小于0,则是扣减,如果大于0,是增加金额,增加金额只加入欠费金额中 |
| | | if (bill.getEditAmount().compareTo(BigDecimal.ZERO)<0 |
| | | && presist.getPayableFeesPenalty().compareTo(BigDecimal.ZERO)>0){ |
| | | BigDecimal prePayableFeesPenalty = presist.getPayableFeesPenalty(); |
| | | //违约金大于调整金额,够减 |
| | | if (prePayableFeesPenalty.compareTo(bill.getEditAmount().abs())>=0){ |
| | | BigDecimal afterPenalty = prePayableFeesPenalty.add(bill.getEditAmount()); |
| | | save.setPayableFeesPenalty(afterPenalty); |
| | | }else{ |
| | | BigDecimal afterPenalty = BigDecimal.ZERO; |
| | | save.setPayableFeesPenalty(afterPenalty); |
| | | } |
| | | } |
| | | BigDecimal afterOutstand = preOutstand.add(bill.getEditAmount()); |
| | | save.setOutstandingMoney(afterOutstand); |
| | | updateById(save); |
| | | }finally { |
| | | redisCache.unlock(lockkey,requestId); |
| | | } |
| | | } |
| | | |
| | | //todo 记录金额修改记录 |
| | | // TFlowManagement flow = new TFlowManagement(); |
| | | // flow.setPayType(3); |
| | | // flow.setPayer("管理员修改"); |
| | | // flow.setBusinessDeptId(presist.getBusinessDeptId()); |
| | | // flow.setPayTime(DateUtils.dateToLocalDateTime(new Date())); |
| | | // flow.setSysSerialNumber(OrderNos.getDid(30)); |
| | | // flow.setFlowType(1); |
| | | // flow.setPaymentBillId(back.getId()); |
| | | // flow.setDeductionMoney(back.getDeductionMoney()); |
| | | // flow.setFlowMoney(save.getOutstandingMoney()); |
| | | // flow.setRemainingMoney(back.getOutstandingMoney()); |
| | | // flow.setPreOutstand(back.getPreOutstand()); |
| | | // flow.setCreateBy(SecurityUtils.getUsername()); |
| | | // tFlowManagementService.save(flow); |
| | | } |
| | | |
| | | |
| | | } |