PointActivityServiceImpl.java 39 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980
  1. package com.ylx.point.service.impl;
  2. import cn.hutool.core.bean.BeanUtil;
  3. import cn.hutool.core.collection.CollectionUtil;
  4. import cn.hutool.core.date.DateTime;
  5. import cn.hutool.core.date.DateUtil;
  6. import cn.hutool.core.util.ObjectUtil;
  7. import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
  8. import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
  9. import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
  10. import com.ylx.common.core.domain.model.WxLoginUser;
  11. import com.ylx.common.exception.base.BaseException;
  12. import com.ylx.common.utils.DateUtils;
  13. import com.ylx.common.utils.SecurityUtils;
  14. import com.ylx.massage.domain.TWxUser;
  15. import com.ylx.massage.service.TWxUserService;
  16. import com.ylx.point.domain.*;
  17. import com.ylx.point.domain.dto.*;
  18. import com.ylx.point.domain.vo.*;
  19. import com.ylx.point.enums.PointActivityStatusEnum;
  20. import com.ylx.point.enums.PointActivityTypeEnum;
  21. import com.ylx.point.mapper.PointActivityMapper;
  22. import com.ylx.point.service.*;
  23. import lombok.extern.slf4j.Slf4j;
  24. import org.springframework.stereotype.Service;
  25. import org.springframework.transaction.annotation.Transactional;
  26. import javax.annotation.Resource;
  27. import java.time.DateTimeException;
  28. import java.time.LocalDate;
  29. import java.time.LocalDateTime;
  30. import java.time.ZoneId;
  31. import java.time.format.DateTimeFormatter;
  32. import java.util.*;
  33. import java.util.stream.Collectors;
  34. /**
  35. * 积分活动主Service业务层处理
  36. *
  37. * @author wzj
  38. * @date 2026-03-25
  39. */
  40. @Slf4j
  41. @Service
  42. public class PointActivityServiceImpl extends ServiceImpl<PointActivityMapper, PointActivity> implements IPointActivityService {
  43. @Resource
  44. private PointActivityMapper pointActivityMapper;
  45. @Resource
  46. private IPointActivityTaskService pointActivityTaskService;
  47. @Resource
  48. private IPointSignTaskService pointSignTaskService;
  49. @Resource
  50. private IPointSignRewardService pointSignRewardService;
  51. @Resource
  52. private IPointUserActivityTaskCompletionService pointUserActivityTaskCompletionService;
  53. @Resource
  54. private IPointActivityExpirePolicyService pointActivityExpirePolicyService;
  55. @Resource
  56. private TWxUserService wxUserService;
  57. @Resource
  58. private IPointUserSignLogService pointUserSignLogService;
  59. @Resource
  60. private IPointUserSignStatusService pointUserSignStatusService;
  61. @Resource
  62. private IPointAccountService pointAccountService;
  63. private static final int BATCH_SIZE = 1000;
  64. private static final int TARGET_COUNT = 2;
  65. private static final String INVALID_EXPIRE_POLICY_TYPE_MESSAGE = "无效的过期策略类型";
  66. private static final String EXPIRE_DAYS_REQUIRED_MESSAGE = "过期天数不能为空";
  67. private static final String EXPIRE_YEAR_REQUIRED_MESSAGE = "过期年数不能为空";
  68. private static final String UPDATE_FAILED_MESSAGE = "更新积分活动过期策略失败";
  69. // 优先级顺序
  70. private static final List<Integer> PRIORITY_TYPES = Arrays.asList(
  71. PointActivityTypeEnum.NEW_USER_ACTIVITY.getCode(),
  72. PointActivityTypeEnum.DAILY_ACTIVITY.getCode(),
  73. PointActivityTypeEnum.MONTHLY_ACTIVITY.getCode()); // 新手, 每日, 每月
  74. @Override
  75. public List<PointActivityPageVo> list(PointActivityPageDTO dto) {
  76. return pointActivityMapper.list(dto);
  77. }
  78. @Override
  79. public PointActivityDetailsVo<?> getDetailsInfo(Long id) {
  80. PointActivity pointActivity = this.pointActivityMapper.selectPointActivityById(id);
  81. if (ObjectUtil.isNull(pointActivity)) {
  82. throw new BaseException("参数有误,活动不存在");
  83. }
  84. Integer activityType = pointActivity.getActivityType();
  85. if (PointActivityTypeEnum.SIGN_TASK.getCode().equals(activityType)) {
  86. PointSignTaskVo vo = this.pointSignTaskService.selectPointSignTaskByActivityId(pointActivity.getId());
  87. BeanUtil.copyProperties(pointActivity, vo);
  88. return PointActivityDetailsVo.of(activityType, vo);
  89. } else {
  90. PointActivityTaskVo vo = new PointActivityTaskVo();
  91. BeanUtil.copyProperties(pointActivity, vo);
  92. List<PointActivityTaskDTO> activityTaskList = this.pointActivityTaskService.selectPointActivityTaskByActivityId(pointActivity.getId());
  93. vo.setActivityTaskList(activityTaskList);
  94. return PointActivityDetailsVo.of(activityType, vo);
  95. }
  96. }
  97. @Override
  98. @Transactional(rollbackFor = Exception.class)
  99. public void add(PointActivityDTO dto) {
  100. // 校验参数
  101. validateActivityDTO(dto);
  102. // 保存活动信息
  103. PointActivity entity = createActivityEntity(dto);
  104. saveActivity(entity);
  105. // 保存活动详情信息
  106. saveActivityDetails(dto, entity.getId());
  107. }
  108. @Override
  109. @Transactional(rollbackFor = Exception.class)
  110. public void edit(PointActivityDTO dto) {
  111. // 校验参数
  112. Long id = dto.getId();
  113. if (ObjectUtil.isNull(id)) {
  114. throw new BaseException("活动主键id不能为空");
  115. }
  116. // 根据id获取活动详情
  117. PointActivity entity = this.pointActivityMapper.selectPointActivityById(id);
  118. if (ObjectUtil.isNull(entity)) {
  119. throw new BaseException("参数有误,活动不存在");
  120. }
  121. // 校验活动状态
  122. if (PointActivityStatusEnum.IN_PROGRESS.getCode().equals(entity.getStatus())) {
  123. throw new BaseException("活动进行中,不可编辑");
  124. }
  125. updateActivity(entity, dto);
  126. // 更新活动详情信息
  127. updateActivityDetails(dto, id);
  128. }
  129. @Override
  130. public PointActivityStatVo getStatInfo(Long id) {
  131. // 根据id获取活动详情
  132. PointActivity entity = this.pointActivityMapper.selectPointActivityById(id);
  133. if (ObjectUtil.isNull(entity)) {
  134. throw new BaseException("参数有误,活动不存在");
  135. }
  136. PointActivityStatVo vo = new PointActivityStatVo();
  137. BeanUtil.copyProperties(entity, vo);
  138. // 数据总览
  139. PointActivityOverviewVO dataOverview = this.pointUserActivityTaskCompletionService.getPointActivityOverviewByActivityId(id);
  140. vo.setDataOverview(dataOverview);
  141. // 数据详情
  142. List<PointActivityTaskDetailVO> taskDetailList = pointActivityMapper.getTaskDetailListById(id);
  143. vo.setTaskDetailList(taskDetailList);
  144. return vo;
  145. }
  146. @Override
  147. @Transactional(rollbackFor = Exception.class)
  148. public void editValidity(PointActivityValidityDTO dto) {
  149. // 根据dto中activityType获取全部的活动list
  150. PointActivity pointActivity = new PointActivity();
  151. pointActivity.setActivityType(dto.getActivityType());
  152. List<PointActivity> activityList = pointActivityMapper.selectPointActivityList(pointActivity);
  153. // 根据类型获取积分活动过期策略表数据
  154. PointActivityExpirePolicy expirePolicy = pointActivityExpirePolicyService.selectPointActivityExpirePolicyByActivityType(dto.getActivityType());
  155. if (CollectionUtil.isEmpty(activityList)) {
  156. // 更新积分活动过期策略表
  157. this.saveOrUpdatePointActivityExpirePolicy(expirePolicy, dto);
  158. } else {
  159. // 判断活动list中只要有活动的状态为"进行中",修改就失败
  160. boolean hasOngoingActivity = activityList.stream()
  161. .anyMatch(activity -> PointActivityStatusEnum.IN_PROGRESS.getCode().equals(activity.getStatus()));
  162. if (hasOngoingActivity) {
  163. throw new BaseException("存在进行中的活动,无法修改有效期");
  164. }
  165. // 更新积分活动过期策略表
  166. this.saveOrUpdatePointActivityExpirePolicy(expirePolicy, dto);
  167. // 批量更新活动的有效期策略
  168. batchUpdateExpirePolicy(activityList, expirePolicy.getId());
  169. }
  170. }
  171. @Override
  172. public List<PointActivityValidityVo> validityDetails() {
  173. try {
  174. // 查询所有过期策略
  175. List<PointActivityExpirePolicy> expirePolicies = this.pointActivityExpirePolicyService.list();
  176. // 如果没有数据,返回空列表
  177. if (CollectionUtil.isEmpty(expirePolicies)) {
  178. log.debug("没有查询到积分活动过期策略数据");
  179. return Collections.emptyList();
  180. }
  181. // 转换为VO对象
  182. List<PointActivityValidityVo> result = expirePolicies.stream()
  183. .map(this::convertToVo)
  184. .collect(Collectors.toList());
  185. log.info("成功查询到 {} 条积分活动有效期详情", result.size());
  186. return result;
  187. } catch (Exception e) {
  188. log.error("查询积分活动有效期详情失败", e);
  189. throw new BaseException("查询积分活动有效期详情失败");
  190. }
  191. }
  192. @Override
  193. public Integer selectTotalActiveTasks(String cityCode) {
  194. return pointActivityMapper.selectTotalActiveTasks(cityCode);
  195. }
  196. @Override
  197. public List<UserPointActivityVo> getUserPointActivityList(UserPointActivityPageDTO dto) {
  198. // 当前登录用户信息
  199. WxLoginUser wxLoginUser = SecurityUtils.getWxLoginUser();
  200. TWxUser user = wxUserService.getByOpenId(wxLoginUser.getCOpenid());
  201. if (user == null) {
  202. throw new RuntimeException("用户不存在");
  203. }
  204. String userId = user.getId();
  205. dto.setUserId(userId);
  206. // TODO 根据cityName换cityCode
  207. return pointActivityMapper.selectTaskWithProgress(dto);
  208. }
  209. @Override
  210. public List<UserPointActivityVo> activityList() {
  211. // 当前登录用户信息
  212. WxLoginUser wxLoginUser = SecurityUtils.getWxLoginUser();
  213. TWxUser user = wxUserService.getByOpenId(wxLoginUser.getCOpenid());
  214. if (user == null) {
  215. throw new RuntimeException("用户不存在");
  216. }
  217. String userId = user.getId();
  218. List<UserPointActivityVo> resultList = new ArrayList<>();
  219. // 2. 按优先级遍历活动类型
  220. for (Integer activityType : PRIORITY_TYPES) {
  221. if (resultList.size() >= TARGET_COUNT) {
  222. break;
  223. }
  224. // 3. 查询该类型下 进行中 的任务
  225. List<PointActivityTask> taskList = pointActivityTaskService.selectTasksByActivityType(activityType);
  226. if (CollectionUtil.isEmpty(taskList)) {
  227. continue;
  228. }
  229. // 4. 批量查询用户完成记录
  230. List<Long> taskIds = taskList.stream().map(PointActivityTask::getId).collect(Collectors.toList());
  231. List<PointUserActivityTaskCompletion> completionList =
  232. pointUserActivityTaskCompletionService.selectCompletionsByUserAndTaskIds(userId, taskIds);
  233. Map<Long, PointUserActivityTaskCompletion> completionMap = completionList.stream()
  234. .collect(Collectors.toMap(PointUserActivityTaskCompletion::getTaskId, c -> c));
  235. // 5. 筛选【未完成】的任务
  236. for (PointActivityTask task : taskList) {
  237. if (resultList.size() >= TARGET_COUNT) break;
  238. PointUserActivityTaskCompletion completion = completionMap.get(task.getId());
  239. if (isTaskIncomplete(task, completion)) {
  240. resultList.add(convertToVo(task, completion));
  241. }
  242. }
  243. }
  244. return resultList.stream().limit(2).collect(Collectors.toList());
  245. }
  246. @Override
  247. public List<SignDayVo> getSignInfo(SignDTO dto) {
  248. // 1. 获取用户信息
  249. WxLoginUser wxLoginUser = SecurityUtils.getWxLoginUser();
  250. if (ObjectUtil.isNull(wxLoginUser)) {
  251. throw new RuntimeException("用户不存在");
  252. }
  253. String openId = wxLoginUser.getCOpenid();
  254. // 2. 获取任务ID (模拟城市逻辑)
  255. PointSignTask task = getEnabledTask(dto.getCityCode());
  256. if (task == null) {
  257. throw new RuntimeException("当前城市暂无签到活动");
  258. }
  259. Long taskId = task.getId();
  260. // 3. 查询奖励配置 (按天数正序查询)
  261. LambdaQueryWrapper<PointSignReward> rewardQuery = new LambdaQueryWrapper<>();
  262. rewardQuery.eq(PointSignReward::getSignTaskId, taskId);
  263. rewardQuery.orderByAsc(PointSignReward::getContinueDays);
  264. rewardQuery.last("LIMIT 7");
  265. List<PointSignReward> rewardList = pointSignRewardService.list(rewardQuery);
  266. // 4. 获取用户当前状态
  267. boolean signedToday = pointUserSignLogService.countTodaySign(openId, taskId) > 0;
  268. LambdaQueryWrapper<PointUserSignStatus> statusQuery = new LambdaQueryWrapper<>();
  269. statusQuery.eq(PointUserSignStatus::getOpenId, openId);
  270. statusQuery.eq(PointUserSignStatus::getActivityId, taskId);
  271. PointUserSignStatus status = pointUserSignStatusService.getOne(statusQuery);
  272. // 修正点:处理空指针,如果没记录则默认为0
  273. int continuousDays = (status == null) ? 0 : status.getCurrentContinuousDays();
  274. // 5. 组装返回数据
  275. List<SignDayVo> voList = new ArrayList<>();
  276. LocalDate today = LocalDate.now();
  277. DateTimeFormatter formatter = DateTimeFormatter.ofPattern("MM-dd");
  278. // 注意:UI通常要求从左到右是 [过去 -> 今天],所以我们需要倒序循环 (从第6天前 -> 今天)
  279. for (int i = 6; i >= 0; i--) {
  280. LocalDate date = today.minusDays(i);
  281. int dayIndex = 7 - i; // 计算这是第几天 (1, 2, 3... 7)
  282. SignDayVo vo = new SignDayVo();
  283. // 修正点:格式化日期为字符串,避免前端处理麻烦
  284. vo.setDate(date);
  285. // --- 核心逻辑修正:状态判断 ---
  286. // 逻辑:
  287. // 1. 如果是今天 (i==0):看 signedToday
  288. // 2. 如果是过去 (i>0):看 continuousDays 是否覆盖了这一天
  289. // 注意:只有当今天还没签时,continuousDays才代表截止到昨天的进度。
  290. // 如果今天已签,continuousDays 包含了今天。
  291. boolean isSigned;
  292. if (i == 0) {
  293. // 今天的情况
  294. isSigned = signedToday;
  295. } else {
  296. // 过去的情况
  297. // 如果今天已签,continuousDays 包含了今天,所以 i <= continuousDays - 1
  298. // 如果今天没签,continuousDays 截止到昨天,所以 i <= continuousDays
  299. // 简化逻辑:只要 (i < continuousDays) 或者 (i == continuousDays 且 今天已签)
  300. // 最通用的判断:当前的循环索引 i (倒数第几天) 小于 总连续天数,说明已签
  301. // 特殊情况:如果今天没签,continuousDays 是昨天的进度。
  302. // 简单写法:
  303. if (signedToday) {
  304. isSigned = i <= continuousDays; // 包含今天在内的已签天数
  305. } else {
  306. isSigned = i < continuousDays; // 不包含今天
  307. }
  308. }
  309. if (isSigned) {
  310. vo.setStatus(1); // 已签
  311. } else if (i == 0) {
  312. vo.setStatus(2); // 今天未签 (可签到)
  313. } else {
  314. vo.setStatus(0); // 未来/未签
  315. }
  316. // --- 奖励匹配 ---
  317. // 根据 dayIndex (1-7) 去 rewardList 找对应的配置
  318. PointSignReward reward = rewardList.stream()
  319. .filter(r -> r.getContinueDays() == dayIndex)
  320. .findFirst()
  321. .orElse(null);
  322. if (reward != null) {
  323. vo.setPoints(reward.getRewardPoints());
  324. // 如果有其他字段如图片,也可以在这里set
  325. } else {
  326. vo.setPoints(0);
  327. }
  328. voList.add(vo);
  329. }
  330. return voList;
  331. }
  332. @Override
  333. @Transactional(rollbackFor = Exception.class)
  334. public void sign(SignDTO dto) {
  335. // 1. 获取用户信息
  336. WxLoginUser wxLoginUser = SecurityUtils.getWxLoginUser();
  337. if (ObjectUtil.isNull(wxLoginUser)) {
  338. throw new RuntimeException("用户未登录");
  339. }
  340. String openId = wxLoginUser.getCOpenid();
  341. // 2. 获取当前城市有效的签到任务配置
  342. // 建议:这里应该加缓存,避免每次签到都查库
  343. PointSignTask task = getEnabledTask(dto.getCityCode());
  344. if (ObjectUtil.isNull(task)) {
  345. throw new RuntimeException("当前城市暂无签到活动");
  346. }
  347. Long activityId = task.getActivityId();
  348. Long taskId = task.getId();
  349. // 3. 防重复签到 (利用数据库唯一索引兜底,这里做快速失败)
  350. // 优化:直接查库比 count 快,且逻辑更清晰
  351. PointUserSignLog todayLog = pointUserSignLogService.getOne(new LambdaQueryWrapper<PointUserSignLog>()
  352. .eq(PointUserSignLog::getOpenId, openId)
  353. .eq(PointUserSignLog::getTaskId, taskId)
  354. .eq(PointUserSignLog::getSignDate, DateUtils.getNowDate()) // 假设日志表有 sign_date 字段,如果没有用 create_time 转 date
  355. .last("LIMIT 1"));
  356. if (ObjectUtil.isNotNull(todayLog)) {
  357. throw new RuntimeException("今日已签到");
  358. }
  359. // 4. 获取并锁定用户签到状态 (悲观锁)
  360. // 注意:selectByOpenIdAndActivityIdForUpdate 必须使用 FOR UPDATE 语法
  361. PointUserSignStatus status = pointUserSignStatusService.selectByOpenIdAndActivityIdForUpdate(openId, activityId);
  362. int newContinuousDays = 1;
  363. DateTime today = DateUtil.date();
  364. DateTime yesterday = DateUtil.yesterday();
  365. if (ObjectUtil.isNull(status)) {
  366. // --- 首次签到 ---
  367. status = new PointUserSignStatus();
  368. status.setOpenId(openId);
  369. status.setActivityId(activityId);
  370. status.setLastSignDate(today);
  371. status.setCurrentContinuousDays(1);
  372. status.setLastRewardCycleDays(0);
  373. pointUserSignStatusService.save(status);
  374. } else {
  375. // --- 非首次签到:计算连续天数 ---
  376. DateTime lastSignDate = status.getLastSignDate();
  377. // 判断是否断签:昨天不是最后签到日期,即为断签
  378. boolean isBreak = !DateUtil.isSameDay(lastSignDate, yesterday);
  379. if (isBreak) {
  380. if (task.getBreakRule() == 0) {
  381. // 规则0:断签重置
  382. newContinuousDays = 1;
  383. } else {
  384. // 规则1:断签保留进度 (这里逻辑通常是 +1,除非你想做补签逻辑)
  385. newContinuousDays = status.getCurrentContinuousDays() + 1;
  386. }
  387. } else {
  388. // 正常连续
  389. newContinuousDays = status.getCurrentContinuousDays() + 1;
  390. }
  391. // 更新状态
  392. status.setLastSignDate(today);
  393. status.setCurrentContinuousDays(newContinuousDays);
  394. pointUserSignStatusService.updateById(status);
  395. }
  396. // 5. 计算奖励积分
  397. // 优化:匹配 <= 当前天数的最大奖励配置 (例如配置了3天、7天,第5天应该拿3天的奖,或者拿基础奖)
  398. int rewardPoints = task.getBasePoints(); // 默认基础积分
  399. PointSignReward reward = pointSignRewardService.getOne(new LambdaQueryWrapper<PointSignReward>()
  400. .eq(PointSignReward::getSignTaskId, taskId)
  401. .le(PointSignReward::getContinueDays, newContinuousDays) // 小于等于当前天数
  402. .orderByDesc(PointSignReward::getContinueDays)
  403. .last("LIMIT 1"));
  404. if (ObjectUtil.isNotNull(reward)) {
  405. rewardPoints = reward.getRewardPoints();
  406. }
  407. // 6. 写入签到流水日志
  408. PointUserSignLog signLog = new PointUserSignLog();
  409. signLog.setOpenId(openId);
  410. signLog.setActivityId(activityId);
  411. signLog.setTaskId(taskId);
  412. signLog.setSignDate(today); // 需确保数据库字段支持 Date 类型
  413. signLog.setContinuousDays(newContinuousDays);
  414. signLog.setPoints(rewardPoints);
  415. signLog.setIsMakeUp(0);
  416. pointUserSignLogService.save(signLog);
  417. // 7. 增加用户积分账户余额
  418. // 注意:这里假设 pointAccountService 内部会写入 point_user_log (积分总流水表)
  419. try {
  420. pointAccountService.addPoints(openId, rewardPoints, null, activityId, taskId, PointActivityTypeEnum.SIGN_TASK.getCode());
  421. } catch (Exception e) {
  422. log.error("签到发放积分失败", e);
  423. throw new RuntimeException("签到成功,但积分发放失败,请联系客服");
  424. }
  425. }
  426. /**
  427. * 判断任务是否未完成
  428. */
  429. private boolean isTaskIncomplete(PointActivityTask task, PointUserActivityTaskCompletion completion) {
  430. // 1. 获取规则要求的次数
  431. int requiredCount = 0;
  432. try {
  433. requiredCount = Integer.parseInt(task.getTriggerValue());
  434. } catch (NumberFormatException e) {
  435. // 如果配置错误,为了安全起见,默认视为未完成(或根据业务抛出异常)
  436. return true;
  437. }
  438. // 2. 获取用户实际完成的次数
  439. int actualCount = 0;
  440. if (ObjectUtil.isNotNull(completion) && ObjectUtil.isNotNull(completion.getCompletedCount())) {
  441. actualCount = completion.getCompletedCount();
  442. }
  443. // 3. 核心判断逻辑
  444. // 如果 实际次数 < 要求次数,则任务“未完成” (返回 true)
  445. return actualCount < requiredCount;
  446. }
  447. /**
  448. * 组装 VO 对象
  449. */
  450. private UserPointActivityVo convertToVo(PointActivityTask task, PointUserActivityTaskCompletion completion) {
  451. UserPointActivityVo vo = new UserPointActivityVo();
  452. vo.setActivityId(task.getActivityId());
  453. vo.setTaskName(task.getTaskName()); // 或者是从字典表查出的中文名称
  454. vo.setRewardPoints(task.getRewardPoints());
  455. // 完成次数:如果没有记录则为 "0"
  456. vo.setCompletedCount(completion != null ? String.valueOf(completion.getCompletedCount()) : "0");
  457. // 触发条件:直接从任务配置中获取,或者根据 taskCode 查字典
  458. vo.setTriggerValue(task.getTriggerValue());
  459. return vo;
  460. }
  461. private PointActivityValidityVo convertToVo(PointActivityExpirePolicy policy) {
  462. PointActivityValidityVo vo = new PointActivityValidityVo();
  463. BeanUtil.copyProperties(policy, vo);
  464. // 可以在这里添加额外的转换逻辑
  465. return vo;
  466. }
  467. private void validateActivityDTO(PointActivityDTO dto) {
  468. if (ObjectUtil.isNull(dto)) {
  469. throw new BaseException("活动信息不能为空");
  470. }
  471. PointActivityDTO.ActivityTimeConfig timeConfig = dto.getTimeConfig();
  472. if (ObjectUtil.isNull(timeConfig) || (!timeConfig.getIsPermanent() &&
  473. (ObjectUtil.isNull(timeConfig.getStartTime()) || ObjectUtil.isNull(timeConfig.getEndTime())))) {
  474. throw new BaseException("活动时间配置不能为空");
  475. }
  476. }
  477. private PointActivity createActivityEntity(PointActivityDTO dto) {
  478. // 创建基础实体
  479. PointActivity entity = new PointActivity();
  480. BeanUtil.copyProperties(dto, entity);
  481. // 处理时间配置
  482. PointActivityDTO.ActivityTimeConfig timeConfig = dto.getTimeConfig();
  483. entity.setStartTime(DateUtil.beginOfDay(timeConfig.getStartTime()));
  484. entity.setEndTime(processEndTime(timeConfig.getEndTime()));
  485. entity.setIsPermanent(timeConfig.getIsPermanent() ? 1 : 0);
  486. // 处理状态
  487. if (PointActivityStatusEnum.PUBLISHED.getCode().equals(dto.getStatus())) {
  488. entity.setStatus(determineActivityStatus(
  489. entity.getStartTime(),
  490. entity.getEndTime(),
  491. timeConfig.getIsPermanent()
  492. ));
  493. }
  494. return entity;
  495. }
  496. private void saveActivity(PointActivity entity) {
  497. entity.setCreateBy(SecurityUtils.getUsername());
  498. entity.setCreateTime(new Date());
  499. int affectedRows = pointActivityMapper.insertPointActivity(entity);
  500. if (affectedRows <= 0) {
  501. throw new BaseException("保存积分活动失败");
  502. }
  503. }
  504. private void saveOrUpdatePointActivityExpirePolicy(PointActivityExpirePolicy expirePolicy, PointActivityValidityDTO dto) {
  505. // 参数校验
  506. if (ObjectUtil.isNull(dto)) {
  507. throw new IllegalArgumentException("参数不能为空");
  508. }
  509. // 根据expirePolicy类型设置对应的过期时间
  510. Integer expirePolicyCode = dto.getExpirePolicy();
  511. if (ObjectUtil.isNull(expirePolicyCode)) {
  512. throw new IllegalArgumentException("过期策略类型不能为空");
  513. }
  514. // 准备要保存或更新的实体
  515. PointActivityExpirePolicy entity = expirePolicy != null ? expirePolicy : new PointActivityExpirePolicy();
  516. // 设置基本信息
  517. entity.setActivityType(dto.getActivityType());
  518. entity.setExpirePolicy(dto.getExpirePolicy());
  519. // 根据过期策略类型设置对应的过期时间
  520. setExpireTimeByPolicyType(entity, dto, expirePolicyCode);
  521. // 保存或更新
  522. saveOrUpdateExpirePolicy(entity, dto.getActivityType());
  523. }
  524. private void setExpireTimeByPolicyType(PointActivityExpirePolicy entity, PointActivityValidityDTO dto, Integer expirePolicyCode) {
  525. // 先清空所有过期时间字段
  526. entity.setExpireDays(null);
  527. entity.setExpireYear(null);
  528. // 根据策略类型设置对应的过期时间
  529. switch (expirePolicyCode) {
  530. case 1: // PERMANENT_VALID - 永久有效
  531. // 已经清空了过期天数和年数,无需额外操作
  532. break;
  533. case 2: // TRANSACTION_EXPIRE - 按交易过期
  534. if (dto.getExpireDays() == null) {
  535. throw new IllegalArgumentException(EXPIRE_DAYS_REQUIRED_MESSAGE);
  536. }
  537. entity.setExpireDays(dto.getExpireDays());
  538. break;
  539. case 3: // YEARLY_EXPIRE - 按年过期
  540. if (dto.getExpireYear() == null) {
  541. throw new IllegalArgumentException(EXPIRE_YEAR_REQUIRED_MESSAGE);
  542. }
  543. entity.setExpireYear(dto.getExpireYear());
  544. break;
  545. default:
  546. throw new IllegalArgumentException(INVALID_EXPIRE_POLICY_TYPE_MESSAGE);
  547. }
  548. }
  549. private void saveOrUpdateExpirePolicy(PointActivityExpirePolicy entity, Integer activityType) {
  550. try {
  551. entity.setUpdateBy(SecurityUtils.getUsername());
  552. entity.setUpdateTime(DateUtils.getNowDate());
  553. boolean isSuccess = pointActivityExpirePolicyService.updateById(entity);
  554. if (!isSuccess) {
  555. log.error("保存或更新积分活动过期策略失败, activityType: {}", activityType);
  556. throw new BaseException(UPDATE_FAILED_MESSAGE);
  557. }
  558. } catch (Exception e) {
  559. log.error("保存或更新积分活动过期策略时发生异常, activityType: {}", activityType, e);
  560. throw new BaseException(UPDATE_FAILED_MESSAGE);
  561. }
  562. }
  563. private void saveActivityDetails(PointActivityDTO dto, Long activityId) {
  564. if (PointActivityTypeEnum.SIGN_TASK.getCode().equals(dto.getActivityType())) {
  565. saveSignTaskDetails(dto, activityId);
  566. } else {
  567. saveActivityTaskDetails(dto, activityId);
  568. }
  569. }
  570. private void updateActivityDetails(PointActivityDTO dto, Long activityId) {
  571. if (PointActivityTypeEnum.SIGN_TASK.getCode().equals(dto.getActivityType())) {
  572. updateSignTaskDetails(dto, activityId);
  573. } else {
  574. updateActivityTaskDetails(dto, activityId);
  575. }
  576. }
  577. private void updateActivity(PointActivity entity, PointActivityDTO dto) {
  578. // 更新活动信息
  579. BeanUtil.copyProperties(dto, entity);
  580. // 处理时间配置
  581. PointActivityDTO.ActivityTimeConfig timeConfig = dto.getTimeConfig();
  582. if (timeConfig != null) {
  583. entity.setStartTime(DateUtil.beginOfDay(timeConfig.getStartTime()));
  584. entity.setEndTime(processEndTime(timeConfig.getEndTime()));
  585. entity.setIsPermanent(timeConfig.getIsPermanent() ? 1 : 0);
  586. }
  587. // 处理状态
  588. if (PointActivityStatusEnum.PUBLISHED.getCode().equals(dto.getStatus())) {
  589. entity.setStatus(determineActivityStatus(
  590. entity.getStartTime(),
  591. entity.getEndTime(),
  592. timeConfig.getIsPermanent()
  593. ));
  594. }
  595. entity.setUpdateBy(SecurityUtils.getUsername());
  596. entity.setUpdateTime(DateUtils.getNowDate());
  597. int affectedRows = pointActivityMapper.updatePointActivity(entity);
  598. if (affectedRows <= 0) {
  599. throw new BaseException("更新积分活动失败");
  600. }
  601. }
  602. private void saveSignTaskDetails(PointActivityDTO dto, Long activityId) {
  603. PointSignTaskDTO signTaskDTO = dto.getSignTaskDTO();
  604. if (ObjectUtil.isNull(signTaskDTO)) {
  605. throw new BaseException("签到任务配置不能为空");
  606. }
  607. PointSignTask pointSignTask = new PointSignTask();
  608. pointSignTask.setActivityId(activityId);
  609. pointSignTask.setBasePoints(signTaskDTO.getBasePoints());
  610. pointSignTask.setBreakRule(signTaskDTO.getBreakRule());
  611. int affectedRows = pointSignTaskService.insertPointSignTask(pointSignTask);
  612. if (affectedRows <= 0) {
  613. throw new BaseException("保存积分签到任务规则失败");
  614. }
  615. saveSignRewards(signTaskDTO.getSignRewardList(), pointSignTask.getId());
  616. }
  617. private void saveSignRewards(List<PointSignRewardDTO> rewardList, Long taskId) {
  618. if (CollectionUtil.isEmpty(rewardList)) {
  619. return;
  620. }
  621. List<PointSignReward> rewards = rewardList.stream()
  622. .map(dto -> {
  623. PointSignReward reward = new PointSignReward();
  624. reward.setRewardPoints(dto.getRewardPoints());
  625. reward.setContinueDays(dto.getContinueDays());
  626. reward.setCreateBy(SecurityUtils.getUsername());
  627. reward.setCreateTime(DateUtils.getNowDate());
  628. reward.setUpdateTime(reward.getCreateTime());
  629. reward.setIsDeleted(0);
  630. reward.setSignTaskId(taskId);
  631. return reward;
  632. })
  633. .collect(Collectors.toList());
  634. pointSignRewardService.batchInsertPointSignReward(rewards);
  635. }
  636. private void saveActivityTaskDetails(PointActivityDTO dto, Long activityId) {
  637. List<PointActivityTaskDTO> tasks = dto.getTasks();
  638. if (CollectionUtil.isEmpty(tasks)) {
  639. return;
  640. }
  641. List<PointActivityTask> activityTasks = tasks.stream()
  642. .map(task -> {
  643. PointActivityTask activityTask = new PointActivityTask();
  644. BeanUtil.copyProperties(task, activityTask);
  645. activityTask.setActivityId(activityId);
  646. activityTask.setIsDeleted(0);
  647. activityTask.setCreateBy(SecurityUtils.getUsername());
  648. activityTask.setCreateTime(DateUtils.getNowDate());
  649. activityTask.setUpdateTime(activityTask.getCreateTime());
  650. return activityTask;
  651. })
  652. .collect(Collectors.toList());
  653. pointActivityTaskService.batchInsertPointActivityTask(activityTasks);
  654. }
  655. /**
  656. * 处理结束时间,设置为当天的23:59:59
  657. */
  658. private Date processEndTime(Date endTime) {
  659. if (endTime == null) {
  660. return null;
  661. }
  662. try {
  663. LocalDateTime localDateTime = LocalDateTime.ofInstant(
  664. endTime.toInstant(),
  665. ZoneId.systemDefault()
  666. );
  667. LocalDateTime endOfDay = localDateTime.with(
  668. java.time.LocalTime.of(23, 59, 59));
  669. return Date.from(endOfDay.atZone(ZoneId.systemDefault()).toInstant());
  670. } catch (DateTimeException e) {
  671. throw new IllegalArgumentException("Invalid end time", e);
  672. }
  673. }
  674. /**
  675. * 确定活动状态
  676. */
  677. private Integer determineActivityStatus(Date startTime, Date endTime, Boolean isPermanent) {
  678. if (isPermanent) {
  679. return PointActivityStatusEnum.IN_PROGRESS.getCode();
  680. }
  681. Date now = DateUtils.getNowDate();
  682. if (startTime.after(now)) {
  683. return PointActivityStatusEnum.NOT_START.getCode();
  684. } else if (endTime != null && endTime.before(now)) {
  685. return PointActivityStatusEnum.FINISHED.getCode();
  686. } else {
  687. return PointActivityStatusEnum.IN_PROGRESS.getCode();
  688. }
  689. }
  690. /**
  691. * 更新签到任务详情
  692. */
  693. private void updateSignTaskDetails(PointActivityDTO dto, Long activityId) {
  694. PointSignTaskDTO signTaskDTO = dto.getSignTaskDTO();
  695. if (ObjectUtil.isNull(signTaskDTO)) {
  696. throw new BaseException("签到任务配置不能为空");
  697. }
  698. // 获取现有的签到任务
  699. PointSignTaskVo existingTask = pointSignTaskService.selectPointSignTaskByActivityId(activityId);
  700. // 更新签到任务基本信息
  701. PointSignTask pointSignTask = new PointSignTask();
  702. pointSignTask.setId(existingTask.getSignTaskId());
  703. pointSignTask.setBasePoints(signTaskDTO.getBasePoints());
  704. pointSignTask.setBreakRule(signTaskDTO.getBreakRule());
  705. // 更新签到任务
  706. int affectedRows = pointSignTaskService.updatePointSignTask(pointSignTask);
  707. if (affectedRows <= 0) {
  708. throw new BaseException("更新积分签到任务规则失败");
  709. }
  710. // 更新签到奖励配置
  711. updateSignRewards(signTaskDTO.getSignRewardList(), pointSignTask.getId());
  712. }
  713. /**
  714. * 更新签到奖励配置
  715. */
  716. private void updateSignRewards(List<PointSignRewardDTO> rewardList, Long taskId) {
  717. // 先删除原有的奖励配置
  718. PointSignReward query = new PointSignReward();
  719. query.setSignTaskId(taskId);
  720. List<PointSignReward> existingRewards = pointSignRewardService.selectPointSignRewardList(query);
  721. if (CollectionUtil.isNotEmpty(existingRewards)) {
  722. String[] ids = existingRewards.stream()
  723. .map(reward -> String.valueOf(reward.getId()))
  724. .toArray(String[]::new);
  725. pointSignRewardService.deletePointSignRewardByIds(ids);
  726. }
  727. // 重新插入新的奖励配置
  728. if (CollectionUtil.isEmpty(rewardList)) {
  729. return;
  730. }
  731. List<PointSignReward> rewards = rewardList.stream()
  732. .map(dto -> {
  733. PointSignReward reward = new PointSignReward();
  734. reward.setRewardPoints(dto.getRewardPoints());
  735. reward.setContinueDays(dto.getContinueDays());
  736. reward.setCreateTime(DateUtils.getNowDate());
  737. reward.setUpdateTime(reward.getCreateTime());
  738. reward.setIsDeleted(0);
  739. reward.setSignTaskId(taskId);
  740. return reward;
  741. })
  742. .collect(Collectors.toList());
  743. pointSignRewardService.batchInsertPointSignReward(rewards);
  744. }
  745. /**
  746. * 更新活动任务详情
  747. */
  748. private void updateActivityTaskDetails(PointActivityDTO dto, Long activityId) {
  749. // 先删除原有的活动任务
  750. pointActivityTaskService.deletePointActivityTaskByActivityId(activityId);
  751. // 重新插入新的活动任务
  752. List<PointActivityTaskDTO> tasks = dto.getTasks();
  753. if (CollectionUtil.isEmpty(tasks)) {
  754. return;
  755. }
  756. List<PointActivityTask> activityTasks = tasks.stream()
  757. .map(task -> {
  758. PointActivityTask activityTask = new PointActivityTask();
  759. BeanUtil.copyProperties(task, activityTask);
  760. activityTask.setActivityId(activityId);
  761. activityTask.setIsDeleted(0);
  762. activityTask.setUpdateBy(SecurityUtils.getUsername());
  763. activityTask.setUpdateTime(DateUtils.getNowDate());
  764. return activityTask;
  765. })
  766. .collect(Collectors.toList());
  767. pointActivityTaskService.batchInsertPointActivityTask(activityTasks);
  768. }
  769. private void batchUpdateExpirePolicy(List<PointActivity> activityList, Long expirePolicyId) {
  770. // 参数校验
  771. if (CollectionUtil.isEmpty(activityList)) {
  772. log.debug("活动列表为空,无需更新过期策略");
  773. return;
  774. }
  775. if (expirePolicyId == null) {
  776. throw new IllegalArgumentException("过期策略ID不能为空");
  777. }
  778. try {
  779. // 使用Stream处理活动列表,设置过期策略ID
  780. List<PointActivity> activitiesToUpdate = activityList.stream()
  781. .map(activity -> {
  782. activity.setExpirePolicyId(expirePolicyId);
  783. activity.setUpdateTime(DateUtils.getNowDate());
  784. activity.setUpdateBy(SecurityUtils.getUsername());
  785. return activity;
  786. })
  787. .collect(Collectors.toList());
  788. // 批量更新数据库
  789. super.updateBatchById(activitiesToUpdate, BATCH_SIZE);
  790. log.info("成功更新 {} 个活动的过期策略为 {}", activitiesToUpdate.size(), expirePolicyId);
  791. } catch (Exception e) {
  792. log.error("批量更新活动过期策略失败, expirePolicyId: {}", expirePolicyId, e);
  793. throw new BaseException("批量更新活动过期策略失败");
  794. }
  795. }
  796. private PointSignTask getEnabledTask(String cityCode) {
  797. // 1. 根据城市查询当前生效的活动
  798. PointActivity activity = pointActivityMapper.selectOne(
  799. new QueryWrapper<PointActivity>()
  800. .eq("cityCode", cityCode)
  801. .eq("status", 1) // 活动启用
  802. .last("LIMIT 1")
  803. );
  804. if (ObjectUtil.isNull(activity)) {
  805. throw new RuntimeException("当前城市未配置签到活动");
  806. }
  807. // 2. 根据活动绑定的 taskId 查询签到任务
  808. LambdaQueryWrapper<PointSignTask> queryWrapper = new LambdaQueryWrapper<>();
  809. queryWrapper.eq(PointSignTask::getActivityId, activity.getId());
  810. queryWrapper.eq(PointSignTask::getIsDeleted, 1);
  811. List<PointSignTask> list = pointSignTaskService.list(queryWrapper);
  812. PointSignTask task = new PointSignTask();
  813. ;
  814. if (CollectionUtil.isNotEmpty(list)) {
  815. task = CollectionUtil.getLast(list);
  816. }
  817. return task;
  818. }
  819. }