| package com.doumee.service.business.impl; | 
|   | 
| import com.alibaba.fastjson.JSONObject; | 
| import com.doumee.config.wx.SendWxMessage; | 
| import com.doumee.config.wx.TransferToUser; | 
| import com.doumee.config.wx.WXPayUtility; | 
| import com.doumee.config.wx.WxMiniConfig; | 
| import com.doumee.core.constants.Constants; | 
| import com.doumee.core.constants.ResponseStatus; | 
| import com.doumee.core.exception.BusinessException; | 
| import com.doumee.core.model.PageData; | 
| import com.doumee.core.model.PageWrap; | 
| import com.doumee.core.utils.DateUtil; | 
| import com.doumee.core.utils.Utils; | 
| import com.doumee.dao.business.MemberMapper; | 
| import com.doumee.dao.business.MemberRevenueMapper; | 
| import com.doumee.dao.business.WithdrawalOrdersMapper; | 
| import com.doumee.dao.business.model.Member; | 
| import com.doumee.dao.business.model.MemberRevenue; | 
| import com.doumee.dao.business.model.WithdrawalOrders; | 
| import com.doumee.dao.dto.WithdrawalDTO; | 
| import com.doumee.service.business.WithdrawalOrdersService; | 
| import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; | 
| import com.baomidou.mybatisplus.core.conditions.update.UpdateWrapper; | 
| import com.baomidou.mybatisplus.core.metadata.IPage; | 
| import com.baomidou.mybatisplus.extension.plugins.pagination.Page; | 
| import org.springframework.beans.factory.annotation.Autowired; | 
| import org.springframework.stereotype.Service; | 
| import org.springframework.transaction.annotation.Transactional; | 
| import org.springframework.util.CollectionUtils; | 
| import org.springframework.util.StringUtils; | 
|   | 
| import java.math.BigDecimal; | 
| import java.util.Date; | 
| import java.util.List; | 
| import java.util.Objects; | 
| import java.util.UUID; | 
|   | 
| /** | 
|  * 提现申请记录Service实现 | 
|  * @author 江蹄蹄 | 
|  * @date 2025/07/09 12:00 | 
|  */ | 
| @Service | 
| public class WithdrawalOrdersServiceImpl implements WithdrawalOrdersService { | 
|   | 
|     @Autowired | 
|     private WithdrawalOrdersMapper withdrawalOrdersMapper; | 
|   | 
|     @Autowired | 
|     private MemberMapper memberMapper; | 
|   | 
|     @Autowired | 
|     private MemberRevenueMapper memberRevenueMapper; | 
|   | 
|     @Autowired | 
|     private SendWxMessage sendWxMessage; | 
|   | 
|     @Override | 
|     public Integer create(WithdrawalOrders withdrawalOrders) { | 
|         withdrawalOrdersMapper.insert(withdrawalOrders); | 
|         return withdrawalOrders.getId(); | 
|     } | 
|   | 
|     @Override | 
|     public void deleteById(Integer id) { | 
|         withdrawalOrdersMapper.deleteById(id); | 
|     } | 
|   | 
|     @Override | 
|     public void delete(WithdrawalOrders withdrawalOrders) { | 
|         UpdateWrapper<WithdrawalOrders> deleteWrapper = new UpdateWrapper<>(withdrawalOrders); | 
|         withdrawalOrdersMapper.delete(deleteWrapper); | 
|     } | 
|   | 
|     @Override | 
|     public void deleteByIdInBatch(List<Integer> ids) { | 
|         if (CollectionUtils.isEmpty(ids)) { | 
|             return; | 
|         } | 
|         withdrawalOrdersMapper.deleteBatchIds(ids); | 
|     } | 
|   | 
|     @Override | 
|     public void updateById(WithdrawalOrders withdrawalOrders) { | 
|         withdrawalOrdersMapper.updateById(withdrawalOrders); | 
|     } | 
|   | 
|     @Override | 
|     public void updateByIdInBatch(List<WithdrawalOrders> withdrawalOrderss) { | 
|         if (CollectionUtils.isEmpty(withdrawalOrderss)) { | 
|             return; | 
|         } | 
|         for (WithdrawalOrders withdrawalOrders: withdrawalOrderss) { | 
|             this.updateById(withdrawalOrders); | 
|         } | 
|     } | 
|   | 
|     @Override | 
|     public WithdrawalOrders findById(Integer id) { | 
|         return withdrawalOrdersMapper.selectById(id); | 
|     } | 
|   | 
|     @Override | 
|     public WithdrawalOrders findOne(WithdrawalOrders withdrawalOrders) { | 
|         QueryWrapper<WithdrawalOrders> wrapper = new QueryWrapper<>(withdrawalOrders); | 
|         return withdrawalOrdersMapper.selectOne(wrapper); | 
|     } | 
|   | 
|     @Override | 
|     public List<WithdrawalOrders> findList(WithdrawalOrders withdrawalOrders) { | 
|         QueryWrapper<WithdrawalOrders> wrapper = new QueryWrapper<>(withdrawalOrders); | 
|         return withdrawalOrdersMapper.selectList(wrapper); | 
|     } | 
|    | 
|     @Override | 
|     public PageData<WithdrawalOrders> findPage(PageWrap<WithdrawalOrders> pageWrap) { | 
|         IPage<WithdrawalOrders> page = new Page<>(pageWrap.getPage(), pageWrap.getCapacity()); | 
|         QueryWrapper<WithdrawalOrders> queryWrapper = new QueryWrapper<>(); | 
|         Utils.MP.blankToNull(pageWrap.getModel()); | 
|         if (pageWrap.getModel().getId() != null) { | 
|             queryWrapper.lambda().eq(WithdrawalOrders::getId, pageWrap.getModel().getId()); | 
|         } | 
|         if (pageWrap.getModel().getDeleted() != null) { | 
|             queryWrapper.lambda().eq(WithdrawalOrders::getDeleted, pageWrap.getModel().getDeleted()); | 
|         } | 
|         if (pageWrap.getModel().getCreateUser() != null) { | 
|             queryWrapper.lambda().eq(WithdrawalOrders::getCreateUser, pageWrap.getModel().getCreateUser()); | 
|         } | 
|         if (pageWrap.getModel().getCreateTime() != null) { | 
|             queryWrapper.lambda().ge(WithdrawalOrders::getCreateTime, Utils.Date.getStart(pageWrap.getModel().getCreateTime())); | 
|             queryWrapper.lambda().le(WithdrawalOrders::getCreateTime, Utils.Date.getEnd(pageWrap.getModel().getCreateTime())); | 
|         } | 
|         if (pageWrap.getModel().getUpdateUser() != null) { | 
|             queryWrapper.lambda().eq(WithdrawalOrders::getUpdateUser, pageWrap.getModel().getUpdateUser()); | 
|         } | 
|         if (pageWrap.getModel().getUpdateTime() != null) { | 
|             queryWrapper.lambda().ge(WithdrawalOrders::getUpdateTime, Utils.Date.getStart(pageWrap.getModel().getUpdateTime())); | 
|             queryWrapper.lambda().le(WithdrawalOrders::getUpdateTime, Utils.Date.getEnd(pageWrap.getModel().getUpdateTime())); | 
|         } | 
|         if (pageWrap.getModel().getRemark() != null) { | 
|             queryWrapper.lambda().eq(WithdrawalOrders::getRemark, pageWrap.getModel().getRemark()); | 
|         } | 
|         if (pageWrap.getModel().getMemberId() != null) { | 
|             queryWrapper.lambda().eq(WithdrawalOrders::getMemberId, pageWrap.getModel().getMemberId()); | 
|         } | 
|         if (pageWrap.getModel().getAmount() != null) { | 
|             queryWrapper.lambda().eq(WithdrawalOrders::getAmount, pageWrap.getModel().getAmount()); | 
|         } | 
|         if (pageWrap.getModel().getWxExternalNo() != null) { | 
|             queryWrapper.lambda().eq(WithdrawalOrders::getWxExternalNo, pageWrap.getModel().getWxExternalNo()); | 
|         } | 
|         if (pageWrap.getModel().getStatus() != null) { | 
|             queryWrapper.lambda().eq(WithdrawalOrders::getStatus, pageWrap.getModel().getStatus()); | 
|         } | 
|         if (pageWrap.getModel().getDoneTime() != null) { | 
|             queryWrapper.lambda().ge(WithdrawalOrders::getDoneTime, Utils.Date.getStart(pageWrap.getModel().getDoneTime())); | 
|             queryWrapper.lambda().le(WithdrawalOrders::getDoneTime, Utils.Date.getEnd(pageWrap.getModel().getDoneTime())); | 
|         } | 
|         if (pageWrap.getModel().getDoneInfo() != null) { | 
|             queryWrapper.lambda().eq(WithdrawalOrders::getDoneInfo, pageWrap.getModel().getDoneInfo()); | 
|         } | 
|         for(PageWrap.SortData sortData: pageWrap.getSorts()) { | 
|             if (sortData.getDirection().equalsIgnoreCase(PageWrap.DESC)) { | 
|                 queryWrapper.orderByDesc(sortData.getProperty()); | 
|             } else { | 
|                 queryWrapper.orderByAsc(sortData.getProperty()); | 
|             } | 
|         } | 
|         return PageData.from(withdrawalOrdersMapper.selectPage(page, queryWrapper)); | 
|     } | 
|   | 
|     @Override | 
|     public long count(WithdrawalOrders withdrawalOrders) { | 
|         QueryWrapper<WithdrawalOrders> wrapper = new QueryWrapper<>(withdrawalOrders); | 
|         return withdrawalOrdersMapper.selectCount(wrapper); | 
|     } | 
|   | 
|   | 
|     /************************************移动端接口*******************************************/ | 
|   | 
|     /** | 
|      * 提现申请 | 
|      * @param withdrawalDTO | 
|      */ | 
|     @Override | 
|     @Transactional(rollbackFor = {BusinessException.class,Exception.class}) | 
|     public TransferToUser.TransferToUserResponse  applyWithdrawal(WithdrawalDTO withdrawalDTO){ | 
|         if(Objects.isNull(withdrawalDTO) | 
|         || Objects.isNull(withdrawalDTO.getAmount()) | 
|         || org.apache.commons.lang3.StringUtils.isBlank(withdrawalDTO.getName()) | 
|         || withdrawalDTO.getAmount().compareTo(BigDecimal.ZERO)<=Constants.ZERO | 
|         ){ | 
|             throw new BusinessException(ResponseStatus.BAD_REQUEST); | 
|         } | 
|         Member member = memberMapper.selectById(withdrawalDTO.getMember().getId()); | 
|         if(Objects.isNull(member)){ | 
|             throw new BusinessException(ResponseStatus.NOT_ALLOWED.getCode(),"用户信息异常,请联系管理员"); | 
|         } | 
|         if(member.getAmount() < withdrawalDTO.getAmount().multiply(new BigDecimal("100")).intValue()){ | 
|             throw new BusinessException(ResponseStatus.NOT_ALLOWED.getCode(),"对不起,可提现余额不足。"); | 
|         } | 
|         if(StringUtils.isEmpty(member.getName())){ | 
|             throw new BusinessException(ResponseStatus.NOT_ALLOWED.getCode(),"请先去维护真实姓名后进行提现申请"); | 
|         } | 
|         WithdrawalOrders withdrawalOrders = new WithdrawalOrders(); | 
|         withdrawalOrders.setCreateTime(new Date()); | 
|         withdrawalOrders.setMemberId(member.getId()); | 
|         withdrawalOrders.setOutBillNo(UUID.randomUUID().toString().replace("-","")); | 
|         withdrawalOrders.setAmount((withdrawalDTO.getAmount().multiply(new BigDecimal("100"))).longValue()); | 
|         withdrawalOrders.setStatus(Constants.ZERO); | 
|         withdrawalOrders.setType(Constants.ZERO); | 
|         withdrawalOrders.setDeleted(Constants.ZERO); | 
|   | 
|         //发起提现申请 | 
|         TransferToUser.TransferToUserRequest transferToUserRequest = new TransferToUser.TransferToUserRequest(); | 
|         transferToUserRequest.openid = member.getOpenid(); | 
|         transferToUserRequest.outBillNo =  withdrawalOrders.getOutBillNo(); | 
|         transferToUserRequest.transferAmount = withdrawalOrders.getAmount(); | 
|         transferToUserRequest.transferRemark = "提现申请"; | 
|         try { | 
|             TransferToUser.TransferToUserResponse response =  WxMiniConfig.transferToUser.run(transferToUserRequest,withdrawalDTO.getName()); | 
|             withdrawalOrders.setRemark(JSONObject.toJSONString(response)); | 
|             if(response.state.name().equals("WAIT_USER_CONFIRM") || response.state.name().equals("ACCEPTED")){ | 
|                 withdrawalOrders.setWxExternalNo(response.transferBillNo); | 
|             } | 
|             withdrawalOrdersMapper.insert(withdrawalOrders); | 
|             //更新用户余额 | 
|             memberMapper.update(new UpdateWrapper<Member>().lambda().setSql(" AMOUNT = AMOUNT -  " + withdrawalOrders.getAmount() ).eq(Member::getId,member.getId())); | 
|             //存储流水记录 | 
|             MemberRevenue memberRevenue = new MemberRevenue(); | 
|             memberRevenue.setCreateTime(new Date()); | 
|             memberRevenue.setTransactionNo(withdrawalOrders.getOutBillNo()); | 
|             memberRevenue.setDeleted(Constants.ZERO); | 
|             memberRevenue.setMemberId(member.getId()); | 
|             memberRevenue.setType(Constants.THREE); | 
|             memberRevenue.setOptType(-Constants.ONE); | 
|             memberRevenue.setBeforeAmount(member.getAmount()); | 
|             memberRevenue.setAmount(withdrawalOrders.getAmount()); | 
|             memberRevenue.setAfterAmount(member.getAmount() - withdrawalOrders.getAmount()); | 
|             memberRevenue.setObjId(withdrawalOrders.getId()); | 
|             memberRevenue.setRemark(Constants.RevenueType.getInfo(memberRevenue.getType())); | 
|             memberRevenue.setObjType(Constants.ONE); | 
|             memberRevenue.setDeleted(Constants.ZERO); | 
|             memberRevenue.setStatus(Constants.TWO); | 
|             memberRevenueMapper.insert(memberRevenue); | 
|             return response; | 
|         } catch (WXPayUtility.ApiException e) { | 
|             String message = JSONObject.parseObject(e.getBody()).getString("message"); | 
|             throw new BusinessException(ResponseStatus.NOT_ALLOWED.getCode(),message); | 
|         } | 
|     } | 
|   | 
|   | 
|     @Override | 
|     public void transferSuccess(String outBillNo,Boolean isSuccess){ | 
|         WithdrawalOrders withdrawalOrders = withdrawalOrdersMapper.selectOne(new QueryWrapper<WithdrawalOrders>().lambda() | 
|                 .eq(WithdrawalOrders::getOutBillNo,outBillNo).last("limit 1")); | 
|         Member member = memberMapper.selectById(withdrawalOrders.getMemberId()); | 
|         if(Objects.isNull(member)){ | 
|             return; | 
|         } | 
|         if(Objects.nonNull(withdrawalOrders)){ | 
|             //转账成功 | 
|             if(isSuccess){ | 
|                 withdrawalOrdersMapper.update(new UpdateWrapper<WithdrawalOrders>().lambda() | 
|                         .set(WithdrawalOrders::getStatus,Constants.ONE) | 
|                         .set(WithdrawalOrders::getUpdateTime, DateUtil.getCurrDateTime()) | 
|                         .eq(WithdrawalOrders::getOutBillNo,outBillNo) | 
|                 ); | 
|                 withdrawalOrders.setUpdateTime(new Date()); | 
|                 sendWxMessage.withdrawalMessage(member.getOpenid(),withdrawalOrders); | 
|             }else { | 
|                 //更新用户余额 | 
|                 memberMapper.update(new UpdateWrapper<Member>().lambda().setSql(" AMOUNT = AMOUNT +  " + withdrawalOrders.getAmount() ).eq(Member::getId,withdrawalOrders.getMemberId())); | 
|                 //存储流水记录 | 
|                 MemberRevenue memberRevenue = new MemberRevenue(); | 
|                 memberRevenue.setCreateTime(new Date()); | 
|                 memberRevenue.setTransactionNo(withdrawalOrders.getOutBillNo()); | 
|                 memberRevenue.setDeleted(Constants.ZERO); | 
|                 memberRevenue.setMemberId(withdrawalOrders.getMemberId()); | 
|                 memberRevenue.setType(Constants.FOUR); | 
|                 memberRevenue.setOptType(Constants.ONE); | 
|                 memberRevenue.setBeforeAmount(member.getAmount()); | 
|                 memberRevenue.setAmount(withdrawalOrders.getAmount()); | 
|                 memberRevenue.setAfterAmount(member.getAmount() + withdrawalOrders.getAmount()); | 
|                 memberRevenue.setObjId(withdrawalOrders.getId()); | 
|                 memberRevenue.setRemark(Constants.RevenueType.getInfo(memberRevenue.getType())); | 
|                 memberRevenue.setObjType(Constants.ONE); | 
|                 memberRevenue.setDeleted(Constants.ZERO); | 
|                 memberRevenue.setStatus(Constants.ZERO); | 
|                 memberRevenueMapper.insert(memberRevenue); | 
|             } | 
|         } | 
|   | 
|     } | 
|   | 
|     @Override | 
|     public void cancelTransfer(TransferToUser.CancelTransferRequest request){ | 
|         if(Objects.isNull(request) | 
|         || StringUtils.isEmpty(request.outBillNo)){ | 
|             throw new BusinessException(ResponseStatus.BAD_REQUEST); | 
|         } | 
|         WithdrawalOrders withdrawalOrders = withdrawalOrdersMapper.selectOne(new QueryWrapper<WithdrawalOrders>().lambda().eq(WithdrawalOrders::getOutBillNo,request.outBillNo).last("limit 1")); | 
|         if(Objects.isNull(withdrawalOrders)){ | 
|             throw new BusinessException(ResponseStatus.DATA_EMPTY); | 
|         } | 
|         Member member = memberMapper.selectById(withdrawalOrders.getMemberId()); | 
|         if(Objects.isNull(member)){ | 
|             throw new BusinessException(ResponseStatus.DATA_EMPTY); | 
|         } | 
|         try { | 
|             TransferToUser.CancelTransferResponse response =  WxMiniConfig.transferToUser.cancelRun(request); | 
|             if(response.state.equals("CANCELING")){ | 
|                 //标记提现失败 加回对应的提现金额 | 
|                 withdrawalOrdersMapper.update(new UpdateWrapper<WithdrawalOrders>() | 
|                         .lambda().set(WithdrawalOrders::getStatus,Constants.TWO) | 
|                         .eq(WithdrawalOrders::getId,withdrawalOrders.getId())); | 
|   | 
|                 memberRevenueMapper.update(new UpdateWrapper<MemberRevenue>().lambda().set(MemberRevenue::getStatus,Constants.ONE) | 
|                                 .set(MemberRevenue::getUpdateTime,new Date()) | 
|                         .eq(MemberRevenue::getObjId,withdrawalOrders.getId())) | 
|                 ; | 
|   | 
|                 //更新用户余额 | 
|                 memberMapper.update(new UpdateWrapper<Member>().lambda().setSql(" AMOUNT = AMOUNT + " + withdrawalOrders.getAmount() ).eq(Member::getId,withdrawalOrders.getMemberId())); | 
|   | 
|                 //存储流水记录 | 
|                 MemberRevenue memberRevenue = new MemberRevenue(); | 
|                 memberRevenue.setCreateTime(new Date()); | 
|                 memberRevenue.setMemberId(withdrawalOrders.getMemberId()); | 
|                 memberRevenue.setType(Constants.FOUR); | 
|                 memberRevenue.setOptType(Constants.ONE); | 
|                 memberRevenue.setBeforeAmount(member.getAmount()); | 
|                 memberRevenue.setAmount(withdrawalOrders.getAmount()); | 
|                 memberRevenue.setAfterAmount(member.getAmount() + withdrawalOrders.getAmount()); | 
|                 memberRevenue.setObjId(withdrawalOrders.getId()); | 
|                 memberRevenue.setObjType(Constants.ONE); | 
|                 memberRevenue.setDeleted(Constants.ZERO); | 
|                 memberRevenue.setStatus(Constants.ZERO); | 
|                 memberRevenueMapper.insert(memberRevenue); | 
|   | 
|             } | 
|             System.out.println(JSONObject.toJSONString(response)); | 
|         } catch (WXPayUtility.ApiException e) { | 
|             String message = JSONObject.parseObject(e.getBody()).getString("message"); | 
|             throw new BusinessException(ResponseStatus.NOT_ALLOWED.getCode(),message); | 
|         } | 
|   | 
|     } | 
|   | 
|   | 
|   | 
| } |