package com.ruoyi.order.service.impl;
|
import java.io.InputStream;
|
import java.util.Date;
|
|
import com.alibaba.fastjson2.JSONObject;
|
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
|
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
|
import com.ruoyi.common.core.constant.Constants;
|
import com.ruoyi.common.core.constant.OrderConstants;
|
import com.ruoyi.common.core.constant.WechatConstants;
|
import com.ruoyi.common.core.domain.R;
|
import com.ruoyi.common.core.exception.GlobalException;
|
import com.ruoyi.common.core.utils.SnowflakeIdWorker;
|
import com.ruoyi.common.core.utils.StringUtils;
|
import com.ruoyi.common.core.vo.UserDto;
|
import com.ruoyi.common.redis.service.RedisService;
|
import com.ruoyi.order.entity.Order;
|
import com.ruoyi.order.entity.Withdraw;
|
import com.ruoyi.order.entity.WithdrawDetail;
|
import com.ruoyi.order.entity.WithdrawalSetting;
|
import com.ruoyi.order.mapper.WithdrawMapper;
|
import com.ruoyi.order.request.WithdrawExportRequest;
|
import com.ruoyi.order.service.OrderService;
|
import com.ruoyi.order.service.WithdrawDetailService;
|
import com.ruoyi.order.service.WithdrawService;
|
import com.ruoyi.order.service.WithdrawalSettingService;
|
import com.ruoyi.order.vo.MoneyQueryRequest;
|
import com.ruoyi.order.vo.UserWithdrawRecordVO;
|
import com.ruoyi.order.vx.HttpUtil;
|
import com.ruoyi.user.api.feignClient.UserClient;
|
import org.springframework.beans.factory.annotation.Value;
|
import org.springframework.core.io.ClassPathResource;
|
import org.springframework.stereotype.Service;
|
|
import javax.annotation.Resource;
|
import java.math.BigDecimal;
|
import java.math.RoundingMode;
|
import java.time.LocalDate;
|
import java.time.Month;
|
import java.time.Year;
|
import java.time.temporal.TemporalAdjusters;
|
import java.util.*;
|
import java.util.concurrent.TimeUnit;
|
import java.util.stream.Collectors;
|
|
/**
|
* <p>
|
* 用户提现申请记录表 服务实现类
|
* </p>
|
*
|
* @author hjl
|
* @since 2024-07-09
|
*/
|
@Service
|
public class WithdrawServiceImpl extends ServiceImpl<WithdrawMapper, Withdraw> implements WithdrawService {
|
|
@Resource
|
private WithdrawalSettingService withdrawalSettingService;
|
@Resource
|
private OrderService orderService;
|
@Resource
|
private RedisService redisService;
|
/**
|
* 雪花算法类
|
*/
|
private static final SnowflakeIdWorker SNOW_FLAKE_ID_WORKER = new SnowflakeIdWorker(5, 5);
|
|
/**
|
* 小程序id
|
*/
|
@Value("wx.appid")
|
private String appId;
|
/**
|
* 转账名称
|
*/
|
@Value("wx.batchName")
|
private String batchName;
|
/**
|
* 商户号
|
*/
|
@Value("wx.mchId")
|
private String mchId;
|
/**
|
* 支付证书序列号
|
*/
|
@Value("wx.wechatPayserialNo")
|
private String wechatPayserialNo;
|
/**
|
* 转账备注
|
*/
|
@Value("wx.transferRemark")
|
private String transferRemark;
|
@Resource
|
private UserClient userClient;
|
|
@Override
|
public Page<UserWithdrawRecordVO> withdrawPage(List<Integer> userIds,String nickname, String userPhone, String applyForTime, Integer state, Page<UserWithdrawRecordVO> page) {
|
Page<UserWithdrawRecordVO> userWithdrawRecordVOPage = baseMapper.withdrawPage(userIds,nickname, userPhone, applyForTime, state, page);
|
|
return userWithdrawRecordVOPage;
|
}
|
@Override
|
public Page<UserWithdrawRecordVO> withdrawPage1(List<String> cityList,List<Integer> userIds,String nickname, String userPhone, String applyForTime, Integer state, Page<UserWithdrawRecordVO> page,List<String> siteIds) {
|
Page<UserWithdrawRecordVO> userWithdrawRecordVOPage = baseMapper.withdrawPage1(cityList,userIds,nickname, userPhone, applyForTime, state, page,siteIds);
|
|
return userWithdrawRecordVOPage;
|
}
|
|
@Override
|
public List<UserWithdrawRecordVO> excelExport(WithdrawExportRequest exportRequest) {
|
String nickname = exportRequest.getNickname();
|
String userPhone = exportRequest.getUserPhone();
|
String applyForTime = exportRequest.getApplyForTime();
|
Integer state = exportRequest.getState();
|
List<String> ids = exportRequest.getIdList();
|
// 结果封装
|
List<UserWithdrawRecordVO> list;
|
if (null != ids && !ids.isEmpty()) {
|
list = baseMapper.exportByIdList(ids);
|
} else {
|
list = baseMapper.exportList(nickname, userPhone, applyForTime, state);
|
}
|
for (UserWithdrawRecordVO userWithdrawRecordVO : list) {
|
R<UserDto> user = userClient.getUser(userWithdrawRecordVO.getUserId());
|
userWithdrawRecordVO.setUserNo(user.getData().getUserNo());
|
userWithdrawRecordVO.setNickname(user.getData().getNickname());
|
}
|
return list;
|
}
|
|
@Override
|
public Boolean confirmWithdraw(String openId, Integer userId, Order order) {
|
// 校验提现
|
List<Withdraw> list = this.lambdaQuery().eq(Withdraw::getUserId, userId)
|
.eq(Withdraw::getOrderId, order.getId()).list();
|
List<Integer> stateList = list.stream().map(Withdraw::getState).collect(Collectors.toList());
|
if (stateList.contains(Constants.ONE)) {
|
throw new GlobalException("当前订单提现申请已通过!");
|
}
|
|
return weChatPay(order.getOrderMoney(), openId,list.get(0).getId());
|
}
|
|
@Override
|
public BigDecimal withdrawalTotalMoney(MoneyQueryRequest request) {
|
List<String> cityList = request.getCityList();
|
String type = request.getType();
|
// 计算当前季度时间所包含时间
|
int currentYear = Year.now().getValue();
|
// 获取当前月份
|
Month currentMonth = LocalDate.now().getMonth();
|
// 计算当前季度的开始时间和结束时间
|
LocalDate startDate = LocalDate.of(currentYear, getStartMonthOfQuarter(currentMonth), 1);
|
LocalDate endDate = startDate.plusMonths(2).with(TemporalAdjusters.lastDayOfMonth());
|
String startDateStr = String.valueOf(startDate);
|
String endDateStr = String.valueOf(endDate);
|
BigDecimal withdrawalTotalMoney;
|
if (request.getSiteIds()==null){
|
request.setSiteIds(new ArrayList<>());
|
}
|
if (OrderConstants.QUARTER.equals(type)) {
|
// 用户提现总额
|
withdrawalTotalMoney = baseMapper.withdrawalTotalMoney(cityList, startDateStr, endDateStr,request.getSiteIds());
|
} else if (OrderConstants.YEAR.equals(type)) {
|
// 用户提现总额
|
withdrawalTotalMoney = baseMapper.withdrawalTotalMoneyByYear(cityList,request.getSiteIds());
|
} else if (OrderConstants.MONTH.equals(type)) {
|
// 用户提现总额
|
withdrawalTotalMoney = baseMapper.withdrawalTotalMoneyByMonth(cityList,request.getSiteIds());
|
} else {
|
// 数量初始化
|
withdrawalTotalMoney = BigDecimal.ZERO;
|
}
|
return withdrawalTotalMoney;
|
}
|
|
@Override
|
public Boolean enableProcess(Integer enableProcess) {
|
Integer[] state = {0, 1};
|
boolean contains = Arrays.stream(state).collect(Collectors.toList()).contains(enableProcess);
|
if (!contains) {
|
throw new GlobalException("系统设置关闭/开启审核状态异常!");
|
}
|
return withdrawalSettingService.lambdaUpdate().set(WithdrawalSetting::getEnableProcess, enableProcess)
|
.update();
|
}
|
|
@Override
|
public WithdrawalSetting withdrawProcess() {
|
return withdrawalSettingService.lambdaQuery().one();
|
}
|
|
@Override
|
public Page<UserWithdrawRecordVO> withdrawList(Integer userId, Page<UserWithdrawRecordVO> page) {
|
Page<UserWithdrawRecordVO> userWithdrawRecordVOPage = baseMapper.withdrawList(userId, page);
|
for (UserWithdrawRecordVO record : userWithdrawRecordVOPage.getRecords()) {
|
R<UserDto> user = userClient.getUser(record.getUserId());
|
System.err.println("==="+user.getData());
|
record.setNickname(user.getData().getNickname());
|
record.setProfilePicture(user.getData().getProfilePicture());
|
record.setUserPhone(user.getData().getPhone());
|
record.setUserNo(record.getUserNo());
|
}
|
return userWithdrawRecordVOPage;
|
}
|
|
@Override
|
public Page<UserWithdrawRecordVO> withdrawList1(List<String> cityList,Integer userId, Page<UserWithdrawRecordVO> page,List<String> siteIds) {
|
Page<UserWithdrawRecordVO> userWithdrawRecordVOPage = baseMapper.withdrawList1(cityList,userId, page,siteIds);
|
for (UserWithdrawRecordVO record : userWithdrawRecordVOPage.getRecords()) {
|
R<UserDto> user = userClient.getUser(record.getUserId());
|
System.err.println("==="+user.getData());
|
record.setNickname(user.getData().getNickname());
|
record.setProfilePicture(user.getData().getProfilePicture());
|
record.setUserPhone(user.getData().getPhone());
|
record.setUserNo(user.getData().getUserNo());
|
}
|
return userWithdrawRecordVOPage;
|
}
|
|
@Override
|
public Boolean confirmWithdrawByUser(String orderId, Integer userId, String openId, String userPhone) {
|
Order order = orderService.lambdaQuery()
|
.eq(Order::getId, orderId)
|
.eq(Order::getIsDelete, 0).one();
|
if (null == order) {
|
throw new GlobalException("订单信息异常!");
|
}
|
// 校验提现
|
List<Withdraw> list = this.lambdaQuery().eq(Withdraw::getUserId, userId)
|
.eq(Withdraw::getOrderId, orderId).list();
|
List<Integer> stateList = list.stream().map(Withdraw::getState).collect(Collectors.toList());
|
if (stateList.contains(Constants.ZERO)) {
|
return false;
|
// throw new GlobalException("当前订单已提交提现申请,请等待审核!");
|
} else if (stateList.contains(Constants.ONE)) {
|
throw new GlobalException("当前订单已完成提现,请勿重复提现!");
|
}
|
// 系统审核设置
|
WithdrawalSetting withdrawalSetting = withdrawalSettingService.lambdaQuery().one();
|
Withdraw withdraw = new Withdraw();
|
withdraw.setUserId(userId);
|
withdraw.setUserPhone(userPhone);
|
withdraw.setApplyForTime(new Date());
|
withdraw.setApplyForMoney(order.getOrderMoney());
|
withdraw.setOrderId(orderId);
|
withdraw.setCityCode(order.getCityCode());
|
|
this.save(withdraw);
|
// 未开启全局 提现审核,则用户提现不需要后台审核
|
if (Constants.ZERO.equals(withdrawalSetting.getEnableProcess())) {
|
// 已通过
|
withdraw.setState(Constants.ONE);
|
// 商家微信打款至微信零钱
|
boolean update = weChatPay(order.getOrderMoney(), openId,withdraw.getId());
|
// if (!update) {
|
// throw new GlobalException("交易提现失败,请检查是否绑定微信!");
|
// }
|
} else {
|
// 待审核
|
withdraw.setState(Constants.ZERO);
|
}
|
|
|
|
order.setIsWithdrawal(1);
|
orderService.updateById(order);
|
redisService.deleteObject(orderId);
|
|
return this.updateById(withdraw);
|
}
|
@Resource
|
private WithdrawDetailService withdrawDetailService;
|
|
private boolean weChatPay(BigDecimal orderMoney, String openId,String withdrawId) {
|
if (StringUtils.isBlank(openId)) {
|
return false;
|
}
|
|
BigDecimal maxTransferAmount = new BigDecimal("20000"); // 单次转账限额,单位为分
|
int totalTransfers = orderMoney.multiply(new BigDecimal("100")).divide(maxTransferAmount, 0, RoundingMode.UP).intValue();
|
boolean allTransfersSuccessful = true;
|
|
for (int i = 0; i < totalTransfers; i++) {
|
BigDecimal transferAmount;
|
if (i < totalTransfers - 1) {
|
transferAmount = maxTransferAmount;
|
} else {
|
// 最后一笔转账,金额为剩余金额
|
transferAmount = orderMoney.multiply(new BigDecimal("100")).subtract(maxTransferAmount.multiply(new BigDecimal(i))).setScale(0, RoundingMode.DOWN);
|
}
|
|
Map<String, Object> postMap = new HashMap<>(8);
|
postMap.put(WechatConstants.APP_ID, "wx98563d0ec9cf21c8");
|
postMap.put(WechatConstants.OUT_BATCH_NO, String.valueOf(UUID.randomUUID()).replaceAll("-", ""));
|
System.err.println("====="+postMap.get(WechatConstants.OUT_BATCH_NO));
|
postMap.put(WechatConstants.BATCH_NAME, "二手回收提现");
|
postMap.put(WechatConstants.BATCH_REMARK, "二手回收提现");
|
postMap.put(WechatConstants.TOTAL_AMOUNT, transferAmount);
|
postMap.put(WechatConstants.TOTAL_NUM, Constants.ONE);
|
|
List<Map<String, Object>> list = new ArrayList<>();
|
Map<String, Object> subMap = new HashMap<>(4);
|
subMap.put(WechatConstants.OUT_DETAIL_NO, String.valueOf(SNOW_FLAKE_ID_WORKER.nextId()));
|
subMap.put(WechatConstants.TRANSFER_AMOUNT, transferAmount);
|
subMap.put(WechatConstants.TRANSFER_REMARK, "二手回收提现到账");
|
subMap.put(WechatConstants.OPEN_ID, openId);
|
list.add(subMap);
|
postMap.put(WechatConstants.TRANSFER_DETAIL_LIST, list);
|
|
// 使用类加载器获取资源 URL
|
// String path = ClassLoader.getSystemResource("/usr/local/vx/apiclient_key.pem").getPath();
|
|
// String result = HttpUtil.postTransBatRequest(
|
// WechatConstants.WE_CHAT_URL_PRE,
|
// JSONObject.toJSONString(postMap),
|
// "7EEA04429B006E12AAA421C002EC48BBEED5BE94",
|
// "1665330417",
|
// "/usr/local/vx/apiclient_key.pem", WechatConstants.WE_CHAT_URL_SUF);
|
String result = HttpUtil.postTransBatRequest(
|
WechatConstants.WE_CHAT_URL_PRE,
|
JSONObject.toJSONString(postMap),
|
"7EEA04429B006E12AAA421C002EC48BBEED5BE94",
|
"1665330417",
|
"/usr/local/vx/apiclient_key.pem", WechatConstants.WE_CHAT_URL_SUF);
|
|
JSONObject jsonObject = JSONObject.parseObject(result);
|
|
WithdrawDetail withdrawDetail = new WithdrawDetail();
|
withdrawDetail.setWithdrawId(withdrawId);
|
withdrawDetail.setMoney(transferAmount);
|
withdrawDetail.setOutBatchNo((String) postMap.get(WechatConstants.OUT_BATCH_NO));
|
withdrawDetailService.save(withdrawDetail);
|
|
if (jsonObject.containsKey(WechatConstants.CREATE_TIME)) {
|
// 转账成功
|
//保存转账明细
|
// WithdrawDetail withdrawDetail = new WithdrawDetail();
|
// withdrawDetail.setWithdrawId(withdrawId);
|
// withdrawDetail.setMoney(transferAmount);
|
// withdrawDetail.setOutBatchNo((String) postMap.get(WechatConstants.OUT_BATCH_NO));
|
// withdrawDetailService.save(withdrawDetail);
|
|
|
} else {
|
allTransfersSuccessful = false;
|
break;
|
}
|
|
}
|
|
return allTransfersSuccessful;
|
}
|
|
public static void main(String[] args) {
|
Map<String, Object> postMap = new HashMap<>(8);
|
postMap.put(WechatConstants.APP_ID, "wx98563d0ec9cf21c8");
|
postMap.put(WechatConstants.OUT_BATCH_NO, String.valueOf(UUID.randomUUID()).replaceAll("-", ""));
|
System.err.println("====="+postMap.get(WechatConstants.OUT_BATCH_NO));
|
postMap.put(WechatConstants.BATCH_NAME, "二手回收提现");
|
postMap.put(WechatConstants.BATCH_REMARK, "二手回收提现");
|
postMap.put(WechatConstants.TOTAL_AMOUNT, 1);
|
postMap.put(WechatConstants.TOTAL_NUM, Constants.ONE);
|
|
List<Map<String, Object>> list = new ArrayList<>();
|
Map<String, Object> subMap = new HashMap<>(4);
|
subMap.put(WechatConstants.OUT_DETAIL_NO, String.valueOf(SNOW_FLAKE_ID_WORKER.nextId()));
|
subMap.put(WechatConstants.TRANSFER_AMOUNT, 1);
|
subMap.put(WechatConstants.TRANSFER_REMARK, "二手回收提现到账");
|
subMap.put(WechatConstants.OPEN_ID, "ouqOk6-Bp6PnNnlHoQSV-6lzoVoU");
|
list.add(subMap);
|
postMap.put(WechatConstants.TRANSFER_DETAIL_LIST, list);
|
String result = HttpUtil.postTransBatRequest(
|
WechatConstants.WE_CHAT_URL_PRE,
|
JSONObject.toJSONString(postMap),
|
"7EEA04429B006E12AAA421C002EC48BBEED5BE94",
|
"1665330417",
|
"E:\\ershoucert\\apiclient_key.pem", WechatConstants.WE_CHAT_URL_SUF);
|
|
JSONObject jsonObject = JSONObject.parseObject(result);
|
System.err.println(jsonObject);
|
}
|
|
/**
|
* 根据当前月份获取当前季度的开始月份
|
*/
|
private static Month getStartMonthOfQuarter(Month currentMonth) {
|
if (currentMonth.compareTo(Month.JANUARY) >= 0 && currentMonth.compareTo(Month.MARCH) <= 0) {
|
return Month.JANUARY;
|
} else if (currentMonth.compareTo(Month.APRIL) >= 0 && currentMonth.compareTo(Month.JUNE) <= 0) {
|
return Month.APRIL;
|
} else if (currentMonth.compareTo(Month.JULY) >= 0 && currentMonth.compareTo(Month.SEPTEMBER) <= 0) {
|
return Month.JULY;
|
} else {
|
return Month.OCTOBER;
|
}
|
}
|
|
}
|