PointActivityServiceImpl.java 44 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055
  1. package com.ylx.point.service.impl;
  2. import cn.hutool.core.bean.BeanUtil;
  3. import cn.hutool.core.collection.CollUtil;
  4. import cn.hutool.core.collection.CollectionUtil;
  5. import cn.hutool.core.date.DateUtil;
  6. import cn.hutool.core.util.ObjectUtil;
  7. import cn.hutool.core.util.StrUtil;
  8. import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
  9. import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
  10. import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
  11. import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
  12. import com.ylx.common.core.domain.model.WxLoginUser;
  13. import com.ylx.common.exception.ServiceException;
  14. import com.ylx.common.utils.DateUtils;
  15. import com.ylx.common.utils.SecurityUtils;
  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.*;
  28. import java.time.temporal.ChronoUnit;
  29. import java.util.*;
  30. import java.util.stream.Collectors;
  31. /**
  32. * 积分活动主Service业务层处理
  33. *
  34. * @author wzj
  35. * @date 2026-03-25
  36. */
  37. @Slf4j
  38. @Service
  39. public class PointActivityServiceImpl extends ServiceImpl<PointActivityMapper, PointActivity> implements IPointActivityService {
  40. @Resource
  41. private PointActivityMapper pointActivityMapper;
  42. @Resource
  43. private IPointActivityTaskService pointActivityTaskService;
  44. @Resource
  45. private IPointSignTaskService pointSignTaskService;
  46. @Resource
  47. private IPointSignRewardService pointSignRewardService;
  48. @Resource
  49. private IPointUserActivityTaskCompletionService pointUserActivityTaskCompletionService;
  50. @Resource
  51. private IPointActivityExpirePolicyService pointActivityExpirePolicyService;
  52. @Resource
  53. private IPointUserSignLogService pointUserSignLogService;
  54. @Resource
  55. private IPointUserSignStatusService pointUserSignStatusService;
  56. @Resource
  57. private IPointAccountService pointAccountService;
  58. private static final int BATCH_SIZE = 1000;
  59. private static final int TARGET_COUNT = 2;
  60. private static final String INVALID_EXPIRE_POLICY_TYPE_MESSAGE = "无效的过期策略类型";
  61. private static final String EXPIRE_DAYS_REQUIRED_MESSAGE = "过期天数不能为空";
  62. private static final String EXPIRE_YEAR_REQUIRED_MESSAGE = "过期年数不能为空";
  63. private static final String UPDATE_FAILED_MESSAGE = "更新积分活动过期策略失败";
  64. // 优先级顺序
  65. private static final List<Integer> PRIORITY_TYPES = Arrays.asList(
  66. PointActivityTypeEnum.NEW_USER_ACTIVITY.getCode(),
  67. PointActivityTypeEnum.DAILY_ACTIVITY.getCode(),
  68. PointActivityTypeEnum.MONTHLY_ACTIVITY.getCode()); // 新手, 每日, 每月
  69. @Override
  70. public Page<PointActivityPageVo> list(Page<PointActivity> page, PointActivityPageDTO dto) {
  71. LambdaQueryWrapper<PointActivity> queryWrapper = new LambdaQueryWrapper<>();
  72. if (ObjectUtil.isNotNull(dto.getActivityType())) {
  73. queryWrapper.eq(PointActivity::getActivityType, dto.getActivityType());
  74. }
  75. if (StrUtil.isNotEmpty(dto.getCityCode())) {
  76. queryWrapper.eq(PointActivity::getCityCode, dto.getCityCode());
  77. }
  78. if (ObjectUtil.isNotNull(dto.getStatus())) {
  79. queryWrapper.eq(PointActivity::getStatus, dto.getStatus());
  80. }
  81. if (ObjectUtil.isNotNull(dto.getStartTime()) && ObjectUtil.isNotNull(dto.getEndTime())) {
  82. queryWrapper.le(PointActivity::getStartTime, dto.getEndTime());
  83. queryWrapper.ge(PointActivity::getEndTime, dto.getStartTime());
  84. }
  85. queryWrapper.orderByDesc(PointActivity::getCreateTime)
  86. .orderByDesc(PointActivity::getSortOrder);
  87. Page<PointActivityPageVo> pageData = new Page<>();
  88. pageData.setSize(page.getSize());
  89. pageData.setCurrent(page.getCurrent());
  90. Page<PointActivity> pointActivityPage = pointActivityMapper.selectPage(page, queryWrapper);
  91. if (ObjectUtil.isNotNull(pointActivityPage) && CollectionUtil.isNotEmpty(pointActivityPage.getRecords())) {
  92. List<PointActivityPageVo> voList = pointActivityPage.getRecords().stream()
  93. .map(this::convertToVo) // 抽取转换方法
  94. .collect(Collectors.toList());
  95. pageData.setRecords(voList);
  96. pageData.setTotal(pointActivityPage.getTotal());
  97. pageData.setPages(pointActivityPage.getPages());
  98. }
  99. return pageData;
  100. }
  101. @Override
  102. public PointActivityDetailsVo<?> getDetailsInfo(Long id) {
  103. PointActivity pointActivity = this.pointActivityMapper.selectPointActivityById(id);
  104. if (ObjectUtil.isNull(pointActivity)) {
  105. throw new IllegalArgumentException("参数有误,活动不存在");
  106. }
  107. Integer activityType = pointActivity.getActivityType();
  108. if (PointActivityTypeEnum.SIGN_TASK.getCode().equals(activityType)) {
  109. PointSignTaskVo vo = this.pointSignTaskService.selectPointSignTaskByActivityId(pointActivity.getId());
  110. BeanUtil.copyProperties(pointActivity, vo);
  111. return PointActivityDetailsVo.of(activityType, vo);
  112. } else {
  113. PointActivityTaskVo vo = new PointActivityTaskVo();
  114. BeanUtil.copyProperties(pointActivity, vo);
  115. List<PointActivityTaskDTO> activityTaskList = this.pointActivityTaskService.selectPointActivityTaskByActivityId(pointActivity.getId());
  116. vo.setActivityTaskList(activityTaskList);
  117. return PointActivityDetailsVo.of(activityType, vo);
  118. }
  119. }
  120. @Override
  121. @Transactional(rollbackFor = Exception.class)
  122. public void add(PointActivityDTO dto) {
  123. // 校验参数
  124. validateActivityDTO(dto);
  125. // 保存活动信息
  126. PointActivity entity = createActivityEntity(dto);
  127. saveActivity(entity);
  128. // 保存活动详情信息
  129. saveActivityDetails(dto, entity.getId());
  130. }
  131. @Override
  132. @Transactional(rollbackFor = Exception.class)
  133. public void edit(PointActivityDTO dto) {
  134. // 校验参数
  135. Long id = dto.getId();
  136. if (ObjectUtil.isNull(id)) {
  137. throw new IllegalArgumentException("活动主键id不能为空");
  138. }
  139. // 根据id获取活动详情
  140. PointActivity entity = this.pointActivityMapper.selectPointActivityById(id);
  141. if (ObjectUtil.isNull(entity)) {
  142. throw new IllegalArgumentException("参数有误,活动不存在");
  143. }
  144. // 校验活动状态
  145. if (PointActivityStatusEnum.IN_PROGRESS.getCode().equals(entity.getStatus())) {
  146. throw new ServiceException("活动进行中,不可编辑");
  147. }
  148. updateActivity(entity, dto);
  149. // 更新活动详情信息
  150. updateActivityDetails(dto, id);
  151. }
  152. @Override
  153. public PointActivityStatVo getStatInfo(Long id) {
  154. // 根据id获取活动详情
  155. PointActivity entity = this.pointActivityMapper.selectPointActivityById(id);
  156. if (ObjectUtil.isNull(entity)) {
  157. throw new IllegalArgumentException("参数有误,活动不存在");
  158. }
  159. PointActivityStatVo vo = new PointActivityStatVo();
  160. BeanUtil.copyProperties(entity, vo);
  161. // 数据总览
  162. PointActivityOverviewVO dataOverview = this.pointUserActivityTaskCompletionService.getPointActivityOverviewByActivityId(id);
  163. vo.setDataOverview(dataOverview);
  164. // 数据详情
  165. List<PointActivityTaskDetailVO> taskDetailList = pointActivityMapper.getTaskDetailListById(id);
  166. vo.setTaskDetailList(taskDetailList);
  167. return vo;
  168. }
  169. @Override
  170. @Transactional(rollbackFor = Exception.class)
  171. public void editValidity(PointActivityValidityDTO dto) {
  172. // 根据dto中activityType获取全部的活动list
  173. PointActivity pointActivity = new PointActivity();
  174. pointActivity.setActivityType(dto.getActivityType());
  175. List<PointActivity> activityList = pointActivityMapper.selectPointActivityList(pointActivity);
  176. // 根据类型获取积分活动过期策略表数据
  177. PointActivityExpirePolicy expirePolicy = pointActivityExpirePolicyService.selectPointActivityExpirePolicyByActivityType(dto.getActivityType());
  178. if (CollectionUtil.isEmpty(activityList)) {
  179. // 更新积分活动过期策略表
  180. this.saveOrUpdatePointActivityExpirePolicy(expirePolicy, dto);
  181. } else {
  182. // 判断活动list中只要有活动的状态为"进行中",修改就失败
  183. boolean hasOngoingActivity = activityList.stream()
  184. .anyMatch(activity -> PointActivityStatusEnum.IN_PROGRESS.getCode().equals(activity.getStatus()));
  185. if (hasOngoingActivity) {
  186. throw new ServiceException("存在进行中的活动,无法修改有效期");
  187. }
  188. // 更新积分活动过期策略表
  189. this.saveOrUpdatePointActivityExpirePolicy(expirePolicy, dto);
  190. // 批量更新活动的有效期策略
  191. batchUpdateExpirePolicy(activityList, expirePolicy.getId());
  192. }
  193. }
  194. @Override
  195. public List<PointActivityValidityVo> validityDetails() {
  196. try {
  197. // 查询所有过期策略
  198. List<PointActivityExpirePolicy> expirePolicies = this.pointActivityExpirePolicyService.list();
  199. // 如果没有数据,返回空列表
  200. if (CollectionUtil.isEmpty(expirePolicies)) {
  201. log.debug("没有查询到积分活动过期策略数据");
  202. return Collections.emptyList();
  203. }
  204. // 转换为VO对象
  205. List<PointActivityValidityVo> result = expirePolicies.stream()
  206. .map(this::convertToVo)
  207. .collect(Collectors.toList());
  208. log.info("成功查询到 {} 条积分活动有效期详情", result.size());
  209. return result;
  210. } catch (Exception e) {
  211. log.error("查询积分活动有效期详情失败", e);
  212. throw new ServiceException("查询积分活动有效期详情失败");
  213. }
  214. }
  215. @Override
  216. public Integer selectTotalActiveTasks(String cityCode) {
  217. return pointActivityMapper.selectTotalActiveTasks(cityCode);
  218. }
  219. @Override
  220. public Page<UserPointActivityVo> getUserPointActivityList(Page<PointActivity> page, UserPointActivityPageDTO dto) {
  221. // 当前登录用户信息
  222. WxLoginUser wxLoginUser = getCurrentWxLoginUser();
  223. String openId = wxLoginUser.getCOpenid();
  224. dto.setOpenId(openId);
  225. return pointActivityMapper.selectTaskWithProgress(page, dto);
  226. }
  227. @Override
  228. public List<UserPointActivityVo> activityList() {
  229. // 当前登录用户信息
  230. WxLoginUser wxLoginUser = getCurrentWxLoginUser();
  231. String openId = wxLoginUser.getCOpenid();
  232. List<UserPointActivityVo> resultList = new ArrayList<>();
  233. // 2. 按优先级遍历活动类型
  234. for (Integer activityType : PRIORITY_TYPES) {
  235. if (resultList.size() >= TARGET_COUNT) {
  236. break;
  237. }
  238. // 3. 查询该类型下 进行中 的任务
  239. List<PointActivityTask> taskList = pointActivityTaskService.selectTasksByActivityType(activityType);
  240. if (CollectionUtil.isEmpty(taskList)) {
  241. continue;
  242. }
  243. // 4. 批量查询用户完成记录
  244. List<Long> taskIds = taskList.stream().map(PointActivityTask::getId).collect(Collectors.toList());
  245. List<PointUserActivityTaskCompletion> completionList =
  246. pointUserActivityTaskCompletionService.selectCompletionsByOpenIdAndTaskIds(openId, taskIds);
  247. Map<Long, PointUserActivityTaskCompletion> completionMap = completionList.stream()
  248. .collect(Collectors.toMap(PointUserActivityTaskCompletion::getTaskId, c -> c));
  249. // 5. 对任务进行排序:未完成优先,创建时间正序
  250. taskList.sort((t1, t2) -> {
  251. PointUserActivityTaskCompletion comp1 = completionMap.get(t1.getId());
  252. PointUserActivityTaskCompletion comp2 = completionMap.get(t2.getId());
  253. // 判断是否已完成
  254. boolean isCompleted1 = comp1 != null;
  255. boolean isCompleted2 = comp2 != null;
  256. // 未完成优先
  257. if (isCompleted1 != isCompleted2) {
  258. return isCompleted1 ? 1 : -1;
  259. }
  260. // 同一状态按创建时间正序排列
  261. if (t1.getCreateTime() != null && t2.getCreateTime() != null) {
  262. return t1.getCreateTime().compareTo(t2.getCreateTime());
  263. }
  264. return 0;
  265. });
  266. // 6. 筛选任务并添加到结果列表
  267. for (PointActivityTask task : taskList) {
  268. if (resultList.size() >= TARGET_COUNT) break;
  269. PointUserActivityTaskCompletion completion = completionMap.get(task.getId());
  270. resultList.add(convertToVo(task, completion));
  271. }
  272. }
  273. return resultList.stream().limit(2).collect(Collectors.toList());
  274. }
  275. @Override
  276. public List<SignDayVo> getSignInfo(SignDTO dto) {
  277. // 1. 获取当前登录用户
  278. WxLoginUser wxLoginUser = getCurrentWxLoginUser();
  279. String openId = wxLoginUser.getCOpenid();
  280. // 2. 获取当前城市签到任务
  281. PointSignTask task = getEnabledTask(dto.getCityCode());
  282. if (ObjectUtil.isNull(task)) {
  283. throw new ServiceException("当前城市暂无签到活动");
  284. }
  285. Long taskId = task.getId();
  286. Long activityId = task.getActivityId();
  287. PointActivity activity = this.pointActivityMapper.selectPointActivityById(activityId);
  288. if (ObjectUtil.isNull(activity)) {
  289. throw new IllegalArgumentException("参数有误,活动不存在");
  290. }
  291. // 3. 基础配置
  292. ZoneId zoneId = ZoneId.of("Asia/Shanghai");
  293. LocalDate today = LocalDate.now(zoneId);
  294. int basePoints = task.getBasePoints() == null ? 0 : task.getBasePoints();
  295. LocalDate activityStartDate = activity.getStartTime() == null ? null
  296. : activity.getStartTime().toInstant().atZone(zoneId).toLocalDate();
  297. LocalDate activityEndDate = activity.getEndTime() == null ? null
  298. : activity.getEndTime().toInstant().atZone(zoneId).toLocalDate();
  299. boolean isPermanent = "1".equals(activity.getIsPermanent());
  300. // 4. 阶梯奖励配置(按 continue_days 升序,满足“累加阶梯奖”)
  301. List<PointSignReward> rewardList = pointSignRewardService.list(new LambdaQueryWrapper<PointSignReward>()
  302. .eq(PointSignReward::getSignTaskId, taskId)
  303. .eq(PointSignReward::getIsDeleted, 0)
  304. .orderByAsc(PointSignReward::getContinueDays));
  305. // 5. 用户当前连续天数(读接口不加锁)
  306. PointUserSignStatus status = pointUserSignStatusService.getOne(new LambdaQueryWrapper<PointUserSignStatus>()
  307. .eq(PointUserSignStatus::getOpenId, openId)
  308. .eq(PointUserSignStatus::getActivityId, activityId)
  309. .last("LIMIT 1"));
  310. int currentContinuousDays = (status == null || status.getCurrentContinuousDays() == null) ? 0 : status.getCurrentContinuousDays();
  311. // 今天是否已签到:直接让 DB 判断 DATE 是否为今天,避免 JDBC/时区导致的跨天偏移
  312. boolean signedToday = pointUserSignStatusService.count(new LambdaQueryWrapper<PointUserSignStatus>()
  313. .eq(PointUserSignStatus::getOpenId, openId)
  314. .eq(PointUserSignStatus::getActivityId, activityId)
  315. .apply("last_sign_date = CURDATE()")) > 0;
  316. // 6. 本周范围(周日-周六)
  317. DayOfWeek dayOfWeek = today.getDayOfWeek();
  318. int daysSinceSunday = (dayOfWeek.getValue() == 7) ? 0 : dayOfWeek.getValue();
  319. LocalDate weekStart = today.minusDays(daysSinceSunday);
  320. LocalDate weekEndExclusive = weekStart.plusDays(7);
  321. // 7. 本周签到日志(只取本周,避免历史污染)
  322. // 注意:历史数据里 PointUserSignLog.points 可能仅存“阶梯加成”(如 5/6/7),
  323. // 面板展示需要按 continuousDays + 当前配置重算“基础分+阶梯累加”。
  324. Map<LocalDate, PointUserSignLog> signedLogMap = new HashMap<>();
  325. Date weekStartDate = Date.from(weekStart.atStartOfDay(zoneId).toInstant());
  326. Date weekEndDateExclusive = Date.from(weekEndExclusive.atStartOfDay(zoneId).toInstant());
  327. // 显式列名,避免字段映射差异导致过滤失效
  328. List<PointUserSignLog> signLogList = pointUserSignLogService.list(new LambdaQueryWrapper<PointUserSignLog>()
  329. .eq(PointUserSignLog::getOpenId, openId)
  330. .eq(PointUserSignLog::getTaskId, taskId)
  331. .ge(PointUserSignLog::getSignDate, weekStartDate)
  332. .lt(PointUserSignLog::getSignDate, weekEndDateExclusive));
  333. if (CollUtil.isNotEmpty(signLogList)) {
  334. for (PointUserSignLog log : signLogList) {
  335. if (log == null || log.getSignDate() == null) {
  336. continue;
  337. }
  338. LocalDate signDate = log.getSignDate().toInstant().atZone(zoneId).toLocalDate();
  339. signedLogMap.put(signDate, log);
  340. }
  341. }
  342. boolean hasSignHistory = (status != null && status.getLastSignDate() != null) || CollUtil.isNotEmpty(signedLogMap);
  343. // 8. 生成 7 天面板数据
  344. List<SignDayVo> voList = new ArrayList<>(7);
  345. // 面板口径:今天未签到时,今天永远按“连续第 1 天”预估
  346. int virtualTodayContinuousDays = signedToday ? currentContinuousDays : 1;
  347. int nextContinuousDays = virtualTodayContinuousDays + 1;
  348. for (int i = 0; i < 7; i++) {
  349. LocalDate date = weekStart.plusDays(i);
  350. PointUserSignLog signedLog = signedLogMap.get(date);
  351. // 今天是否已签,以状态表为准(更权威且是 date 类型)
  352. boolean signed = date.equals(today) ? signedToday : signedLog != null;
  353. SignDayVo vo = new SignDayVo();
  354. vo.setDate(date);
  355. // 8.1 活动有效性校验:不在活动时间范围内,直接过期(status=4)
  356. if ((activityStartDate != null && date.isBefore(activityStartDate))
  357. || (!isPermanent && activityEndDate != null && date.isAfter(activityEndDate))) {
  358. vo.setStatus(4);
  359. vo.setPoints(basePoints);
  360. voList.add(vo);
  361. continue;
  362. }
  363. // 8.2 状态计算
  364. if (signed) {
  365. vo.setStatus(1);
  366. } else if (date.equals(today)) {
  367. vo.setStatus(2);
  368. } else if (date.isAfter(today)) {
  369. vo.setStatus(0);
  370. } else {
  371. // 过去未签:只有“存在历史签到”的情况下,昨天才算断签(3);否则一律过期(4)
  372. vo.setStatus(hasSignHistory && date.equals(today.minusDays(1)) ? 3 : 4);
  373. }
  374. // 8.3 积分计算(基础分 + 阶梯累加)
  375. int points;
  376. if (vo.getStatus() == 1) {
  377. Integer continuousDays = (signedLog == null) ? null : signedLog.getContinuousDays();
  378. // 优先按连续天数重算(兼容旧数据 points=5/6/7 的情况)
  379. if (continuousDays != null) {
  380. points = calcCumulativeSignPoints(basePoints, continuousDays, rewardList);
  381. } else if (signedLog != null && signedLog.getPoints() != null) {
  382. // 兜底:如果日志里就是实际发放值(新数据),则直接回显
  383. points = signedLog.getPoints();
  384. } else if (date.equals(today) && signedToday) {
  385. // 今天已签但本周日志未查到(极端情况):用状态表连续天数兜底
  386. points = calcCumulativeSignPoints(basePoints, currentContinuousDays, rewardList);
  387. } else {
  388. points = basePoints;
  389. }
  390. } else if (vo.getStatus() == 2) {
  391. points = calcCumulativeSignPoints(basePoints, virtualTodayContinuousDays, rewardList);
  392. } else if (vo.getStatus() == 0) {
  393. points = calcCumulativeSignPoints(basePoints, nextContinuousDays, rewardList);
  394. nextContinuousDays++;
  395. } else {
  396. // 断签/过期:展示基础分
  397. points = basePoints;
  398. }
  399. vo.setPoints(points);
  400. voList.add(vo);
  401. }
  402. return voList;
  403. }
  404. /**
  405. * 签到积分计算:基础分 + 阶梯奖励累加
  406. *
  407. * 配置口径:reward.continueDays 表示“本档位需要再连续签到 N 天”,需要做累计阈值:
  408. * 例如配置 1/2/3 天,对应阈值为 1、(1+2)=3、(1+2+3)=6。
  409. *
  410. * 业务规则:首日仅基础分,阶梯奖励从“超过阈值的下一天”开始生效,
  411. * 即连续天数满足 (continuousDays > threshold) 才加对应档位奖励。
  412. */
  413. private int calcCumulativeSignPoints(int basePoints, int continuousDays, List<PointSignReward> rewardList) {
  414. int total = Math.max(basePoints, 0);
  415. if (continuousDays <= 0 || CollUtil.isEmpty(rewardList)) {
  416. return total;
  417. }
  418. int threshold = 0;
  419. for (PointSignReward reward : rewardList) {
  420. if (reward == null || reward.getContinueDays() == null || reward.getRewardPoints() == null) {
  421. continue;
  422. }
  423. threshold += reward.getContinueDays();
  424. if (continuousDays > threshold) {
  425. total += reward.getRewardPoints();
  426. }
  427. }
  428. return total;
  429. }
  430. @Override
  431. @Transactional(rollbackFor = Exception.class)
  432. public void sign(SignDTO dto) {
  433. // 1. 用户登录校验
  434. WxLoginUser wxLoginUser = getCurrentWxLoginUser();
  435. String openId = wxLoginUser.getCOpenid();
  436. // 2. 获取有效签到任务
  437. PointSignTask task = getEnabledTask(dto.getCityCode());
  438. if (ObjectUtil.isNull(task)) {
  439. throw new ServiceException("当前城市暂无签到活动");
  440. }
  441. Long activityId = task.getActivityId();
  442. Long taskId = task.getId();
  443. String activityName = task.getActivityName();
  444. Integer taskType = PointActivityTypeEnum.SIGN_TASK.getCode();
  445. // ======================== 核心:统一使用任务完成表防重 ========================
  446. boolean todaySigned = pointUserActivityTaskCompletionService.isTodayCompleted(openId, activityId, taskId, taskType);
  447. if (todaySigned) {
  448. throw new ServiceException("今日已签到");
  449. }
  450. // 3. 悲观锁获取用户签到状态(并发安全)
  451. PointUserSignStatus status = pointUserSignStatusService.selectByOpenIdAndActivityIdForUpdate(openId, activityId);
  452. Date todayDate = DateUtil.date();
  453. Date yesterdayDate = DateUtil.yesterday();
  454. int newContinuousDays;
  455. if (ObjectUtil.isNull(status)) {
  456. // 首次签到
  457. newContinuousDays = 1;
  458. status = new PointUserSignStatus();
  459. status.setOpenId(openId);
  460. status.setActivityId(activityId);
  461. status.setLastSignDate(todayDate);
  462. status.setCurrentContinuousDays(newContinuousDays);
  463. status.setLastRewardCycleDays(0);
  464. pointUserSignStatusService.save(status);
  465. } else {
  466. // 计算连续天数(修复断签逻辑)
  467. boolean isContinue = DateUtil.isSameDay(status.getLastSignDate(), yesterdayDate);
  468. if (isContinue) {
  469. newContinuousDays = status.getCurrentContinuousDays() + 1;
  470. } else {
  471. // 断签规则:0重置 1保留
  472. newContinuousDays = task.getBreakRule() == 0 ? 1 : status.getCurrentContinuousDays();
  473. }
  474. status.setLastSignDate(todayDate);
  475. status.setCurrentContinuousDays(newContinuousDays);
  476. pointUserSignStatusService.updateById(status);
  477. }
  478. // 4. 计算奖励积分(基础分 + 阶梯奖励累加)
  479. int basePoints = task.getBasePoints() == null ? 0 : task.getBasePoints();
  480. List<PointSignReward> rewardList = pointSignRewardService.list(new LambdaQueryWrapper<PointSignReward>()
  481. .eq(PointSignReward::getSignTaskId, taskId)
  482. .eq(PointSignReward::getIsDeleted, 0)
  483. .orderByAsc(PointSignReward::getContinueDays));
  484. int rewardPoints = calcCumulativeSignPoints(basePoints, newContinuousDays, rewardList);
  485. // 5. 写入签到流水
  486. PointUserSignLog signLog = new PointUserSignLog();
  487. signLog.setOpenId(openId);
  488. signLog.setActivityId(activityId);
  489. signLog.setTaskId(taskId);
  490. signLog.setSignDate(todayDate);
  491. signLog.setContinuousDays(newContinuousDays);
  492. signLog.setPoints(rewardPoints);
  493. signLog.setIsMakeUp(0);
  494. pointUserSignLogService.save(signLog);
  495. // ======================== 统一记录任务完成(必须放在这里,事务内) ========================
  496. pointUserActivityTaskCompletionService.completeTask(openId, activityId, taskId, taskType);
  497. // 6. 发放积分
  498. try {
  499. pointAccountService.addPoints(openId, rewardPoints, activityName, null, activityId, taskId, taskType);
  500. } catch (Exception e) {
  501. log.error("签到发放积分异常 openid:{}", openId, e);
  502. throw new ServiceException("签到成功,但积分发放失败,请联系客服");
  503. }
  504. }
  505. /**
  506. * 组装 VO 对象
  507. */
  508. private UserPointActivityVo convertToVo(PointActivityTask task, PointUserActivityTaskCompletion completion) {
  509. UserPointActivityVo vo = new UserPointActivityVo();
  510. vo.setActivityId(task.getActivityId());
  511. vo.setTaskId(task.getId().toString());
  512. vo.setTaskName(task.getTaskName()); // 或者是从字典表查出的中文名称
  513. vo.setRewardPoints(task.getRewardPoints());
  514. // 完成次数:如果没有记录则为 "0"
  515. vo.setCompletedCount(completion != null ? String.valueOf(completion.getCompletedCount()) : "0");
  516. // 触发条件:直接从任务配置中获取,或者根据 taskCode 查字典
  517. vo.setTriggerValue(task.getTriggerValue());
  518. return vo;
  519. }
  520. private PointActivityValidityVo convertToVo(PointActivityExpirePolicy policy) {
  521. PointActivityValidityVo vo = new PointActivityValidityVo();
  522. BeanUtil.copyProperties(policy, vo);
  523. // 可以在这里添加额外的转换逻辑
  524. return vo;
  525. }
  526. private void validateActivityDTO(PointActivityDTO dto) {
  527. if (ObjectUtil.isNull(dto)) {
  528. throw new ServiceException("活动信息不能为空");
  529. }
  530. PointActivityDTO.ActivityTimeConfig timeConfig = dto.getTimeConfig();
  531. if (ObjectUtil.isNull(timeConfig) || (StrUtil.equals(timeConfig.getIsPermanent(), "0") &&
  532. (ObjectUtil.isNull(timeConfig.getStartTime()) || ObjectUtil.isNull(timeConfig.getEndTime())))) {
  533. throw new ServiceException("活动时间配置不能为空");
  534. }
  535. }
  536. private PointActivity createActivityEntity(PointActivityDTO dto) {
  537. // 创建基础实体
  538. PointActivity entity = new PointActivity();
  539. BeanUtil.copyProperties(dto, entity);
  540. // 处理时间配置
  541. PointActivityDTO.ActivityTimeConfig timeConfig = dto.getTimeConfig();
  542. entity.setIsPermanent(timeConfig.getIsPermanent());
  543. // 添加对活动时间的校验
  544. if (ObjectUtil.isNotEmpty(timeConfig.getStartTime())) {
  545. entity.setStartTime(DateUtil.beginOfDay(timeConfig.getStartTime()));
  546. }
  547. if (ObjectUtil.isNotEmpty(timeConfig.getEndTime())) {
  548. entity.setEndTime(processEndTime(timeConfig.getEndTime()));
  549. }
  550. // 处理状态
  551. if (PointActivityStatusEnum.PUBLISHED.getCode().equals(dto.getStatus())) {
  552. entity.setStatus(determineActivityStatus(
  553. entity.getStartTime(),
  554. entity.getEndTime(),
  555. timeConfig.getIsPermanent()
  556. ));
  557. }
  558. return entity;
  559. }
  560. private void saveActivity(PointActivity entity) {
  561. entity.setCreateBy(SecurityUtils.getUsername());
  562. entity.setCreateTime(new Date());
  563. int affectedRows = pointActivityMapper.insertPointActivity(entity);
  564. if (affectedRows <= 0) {
  565. throw new ServiceException("保存积分活动失败");
  566. }
  567. }
  568. private void saveOrUpdatePointActivityExpirePolicy(PointActivityExpirePolicy expirePolicy, PointActivityValidityDTO dto) {
  569. // 参数校验
  570. if (ObjectUtil.isNull(dto)) {
  571. throw new IllegalArgumentException("参数不能为空");
  572. }
  573. // 根据expirePolicy类型设置对应的过期时间
  574. Integer expirePolicyCode = dto.getExpirePolicy();
  575. if (ObjectUtil.isNull(expirePolicyCode)) {
  576. throw new IllegalArgumentException("过期策略类型不能为空");
  577. }
  578. // 准备要保存或更新的实体
  579. PointActivityExpirePolicy entity = expirePolicy != null ? expirePolicy : new PointActivityExpirePolicy();
  580. // 设置基本信息
  581. entity.setActivityType(dto.getActivityType());
  582. entity.setExpirePolicy(dto.getExpirePolicy());
  583. // 根据过期策略类型设置对应的过期时间
  584. setExpireTimeByPolicyType(entity, dto, expirePolicyCode);
  585. // 保存或更新
  586. saveOrUpdateExpirePolicy(entity, dto.getActivityType());
  587. }
  588. private void setExpireTimeByPolicyType(PointActivityExpirePolicy entity, PointActivityValidityDTO dto, Integer expirePolicyCode) {
  589. // 先清空所有过期时间字段
  590. entity.setExpireDays(null);
  591. entity.setExpireYear(null);
  592. // 根据策略类型设置对应的过期时间
  593. switch (expirePolicyCode) {
  594. case 1: // PERMANENT_VALID - 永久有效
  595. // 已经清空了过期天数和年数,无需额外操作
  596. break;
  597. case 2: // TRANSACTION_EXPIRE - 按交易过期
  598. if (dto.getExpireDays() == null) {
  599. throw new IllegalArgumentException(EXPIRE_DAYS_REQUIRED_MESSAGE);
  600. }
  601. entity.setExpireDays(dto.getExpireDays());
  602. break;
  603. case 3: // YEARLY_EXPIRE - 按年过期
  604. if (dto.getExpireYear() == null) {
  605. throw new IllegalArgumentException(EXPIRE_YEAR_REQUIRED_MESSAGE);
  606. }
  607. entity.setExpireYear(dto.getExpireYear());
  608. break;
  609. default:
  610. throw new IllegalArgumentException(INVALID_EXPIRE_POLICY_TYPE_MESSAGE);
  611. }
  612. }
  613. private void saveOrUpdateExpirePolicy(PointActivityExpirePolicy entity, Integer activityType) {
  614. try {
  615. entity.setUpdateBy(SecurityUtils.getUsername());
  616. entity.setUpdateTime(DateUtils.getNowDate());
  617. boolean isSuccess = pointActivityExpirePolicyService.updateById(entity);
  618. if (!isSuccess) {
  619. log.error("保存或更新积分活动过期策略失败, activityType: {}", activityType);
  620. throw new ServiceException(UPDATE_FAILED_MESSAGE);
  621. }
  622. } catch (Exception e) {
  623. log.error("保存或更新积分活动过期策略时发生异常, activityType: {}", activityType, e);
  624. throw new ServiceException(UPDATE_FAILED_MESSAGE);
  625. }
  626. }
  627. private void saveActivityDetails(PointActivityDTO dto, Long activityId) {
  628. if (PointActivityTypeEnum.SIGN_TASK.getCode().equals(dto.getActivityType())) {
  629. saveSignTaskDetails(dto, activityId);
  630. } else {
  631. saveActivityTaskDetails(dto, activityId);
  632. }
  633. }
  634. private void updateActivityDetails(PointActivityDTO dto, Long activityId) {
  635. if (PointActivityTypeEnum.SIGN_TASK.getCode().equals(dto.getActivityType())) {
  636. updateSignTaskDetails(dto, activityId);
  637. } else {
  638. updateActivityTaskDetails(dto, activityId);
  639. }
  640. }
  641. private void updateActivity(PointActivity entity, PointActivityDTO dto) {
  642. // 更新活动信息
  643. BeanUtil.copyProperties(dto, entity);
  644. // 处理时间配置
  645. PointActivityDTO.ActivityTimeConfig timeConfig = dto.getTimeConfig();
  646. if (timeConfig != null) {
  647. entity.setStartTime(DateUtil.beginOfDay(timeConfig.getStartTime()));
  648. entity.setEndTime(processEndTime(timeConfig.getEndTime()));
  649. entity.setIsPermanent(timeConfig.getIsPermanent());
  650. }
  651. // 处理状态
  652. if (PointActivityStatusEnum.PUBLISHED.getCode().equals(dto.getStatus())) {
  653. entity.setStatus(determineActivityStatus(
  654. entity.getStartTime(),
  655. entity.getEndTime(),
  656. timeConfig.getIsPermanent()
  657. ));
  658. }
  659. entity.setUpdateBy(SecurityUtils.getUsername());
  660. entity.setUpdateTime(DateUtils.getNowDate());
  661. int affectedRows = pointActivityMapper.updatePointActivity(entity);
  662. if (affectedRows <= 0) {
  663. throw new ServiceException("更新积分活动失败");
  664. }
  665. }
  666. private void saveSignTaskDetails(PointActivityDTO dto, Long activityId) {
  667. PointSignTaskDTO signTaskDTO = dto.getSignTaskDTO();
  668. if (ObjectUtil.isNull(signTaskDTO)) {
  669. throw new ServiceException("签到任务配置不能为空");
  670. }
  671. PointSignTask pointSignTask = new PointSignTask();
  672. pointSignTask.setActivityId(activityId);
  673. pointSignTask.setBasePoints(signTaskDTO.getBasePoints());
  674. pointSignTask.setBreakRule(signTaskDTO.getBreakRule());
  675. int affectedRows = pointSignTaskService.insertPointSignTask(pointSignTask);
  676. if (affectedRows <= 0) {
  677. throw new ServiceException("保存积分签到任务规则失败");
  678. }
  679. saveSignRewards(signTaskDTO.getSignRewardList(), pointSignTask.getId());
  680. }
  681. private void saveSignRewards(List<PointSignRewardDTO> rewardList, Long taskId) {
  682. if (CollectionUtil.isEmpty(rewardList)) {
  683. return;
  684. }
  685. List<PointSignReward> rewards = rewardList.stream()
  686. .map(dto -> {
  687. PointSignReward reward = new PointSignReward();
  688. reward.setRewardPoints(dto.getRewardPoints());
  689. reward.setContinueDays(dto.getContinueDays());
  690. reward.setCreateBy(SecurityUtils.getUsername());
  691. reward.setCreateTime(DateUtils.getNowDate());
  692. reward.setUpdateTime(reward.getCreateTime());
  693. reward.setIsDeleted(0);
  694. reward.setSignTaskId(taskId);
  695. return reward;
  696. })
  697. .collect(Collectors.toList());
  698. pointSignRewardService.batchInsertPointSignReward(rewards);
  699. }
  700. private void saveActivityTaskDetails(PointActivityDTO dto, Long activityId) {
  701. List<PointActivityTaskDTO> tasks = dto.getTasks();
  702. if (CollectionUtil.isEmpty(tasks)) {
  703. return;
  704. }
  705. List<PointActivityTask> activityTasks = tasks.stream()
  706. .map(task -> {
  707. PointActivityTask activityTask = new PointActivityTask();
  708. BeanUtil.copyProperties(task, activityTask);
  709. activityTask.setActivityId(activityId);
  710. activityTask.setIsDeleted(0);
  711. activityTask.setCreateBy(SecurityUtils.getUsername());
  712. activityTask.setCreateTime(DateUtils.getNowDate());
  713. activityTask.setUpdateTime(activityTask.getCreateTime());
  714. return activityTask;
  715. })
  716. .collect(Collectors.toList());
  717. pointActivityTaskService.batchInsertPointActivityTask(activityTasks);
  718. }
  719. /**
  720. * 处理结束时间,设置为当天的23:59:59
  721. */
  722. private Date processEndTime(Date endTime) {
  723. if (endTime == null) {
  724. return null;
  725. }
  726. try {
  727. LocalDateTime localDateTime = LocalDateTime.ofInstant(
  728. endTime.toInstant(),
  729. ZoneId.systemDefault()
  730. );
  731. LocalDateTime endOfDay = localDateTime.with(
  732. LocalTime.of(23, 59, 59));
  733. return Date.from(endOfDay.atZone(ZoneId.systemDefault()).toInstant());
  734. } catch (DateTimeException e) {
  735. throw new IllegalArgumentException("Invalid end time", e);
  736. }
  737. }
  738. /**
  739. * 确定活动状态
  740. */
  741. private Integer determineActivityStatus(Date startTime, Date endTime, String isPermanent) {
  742. if (StrUtil.equals(isPermanent, "1")) {
  743. return PointActivityStatusEnum.IN_PROGRESS.getCode();
  744. }
  745. Date now = DateUtils.getNowDate();
  746. if (startTime.after(now)) {
  747. return PointActivityStatusEnum.NOT_START.getCode();
  748. } else if (endTime != null && endTime.before(now)) {
  749. return PointActivityStatusEnum.FINISHED.getCode();
  750. } else {
  751. return PointActivityStatusEnum.IN_PROGRESS.getCode();
  752. }
  753. }
  754. /**
  755. * 更新签到任务详情
  756. */
  757. private void updateSignTaskDetails(PointActivityDTO dto, Long activityId) {
  758. PointSignTaskDTO signTaskDTO = dto.getSignTaskDTO();
  759. if (ObjectUtil.isNull(signTaskDTO)) {
  760. throw new ServiceException("签到任务配置不能为空");
  761. }
  762. // 获取现有的签到任务
  763. PointSignTaskVo existingTask = pointSignTaskService.selectPointSignTaskByActivityId(activityId);
  764. // 更新签到任务基本信息
  765. PointSignTask pointSignTask = new PointSignTask();
  766. pointSignTask.setId(existingTask.getSignTaskId());
  767. pointSignTask.setBasePoints(signTaskDTO.getBasePoints());
  768. pointSignTask.setBreakRule(signTaskDTO.getBreakRule());
  769. // 更新签到任务
  770. int affectedRows = pointSignTaskService.updatePointSignTask(pointSignTask);
  771. if (affectedRows <= 0) {
  772. throw new ServiceException("更新积分签到任务规则失败");
  773. }
  774. // 更新签到奖励配置
  775. updateSignRewards(signTaskDTO.getSignRewardList(), pointSignTask.getId());
  776. }
  777. /**
  778. * 更新签到奖励配置
  779. */
  780. private void updateSignRewards(List<PointSignRewardDTO> rewardList, Long taskId) {
  781. // 先删除原有的奖励配置
  782. PointSignReward query = new PointSignReward();
  783. query.setSignTaskId(taskId);
  784. List<PointSignReward> existingRewards = pointSignRewardService.selectPointSignRewardList(query);
  785. if (CollectionUtil.isNotEmpty(existingRewards)) {
  786. String[] ids = existingRewards.stream()
  787. .map(reward -> String.valueOf(reward.getId()))
  788. .toArray(String[]::new);
  789. pointSignRewardService.deletePointSignRewardByIds(ids);
  790. }
  791. // 重新插入新的奖励配置
  792. if (CollectionUtil.isEmpty(rewardList)) {
  793. return;
  794. }
  795. List<PointSignReward> rewards = rewardList.stream()
  796. .map(dto -> {
  797. PointSignReward reward = new PointSignReward();
  798. reward.setRewardPoints(dto.getRewardPoints());
  799. reward.setContinueDays(dto.getContinueDays());
  800. reward.setCreateTime(DateUtils.getNowDate());
  801. reward.setUpdateTime(reward.getCreateTime());
  802. reward.setIsDeleted(0);
  803. reward.setSignTaskId(taskId);
  804. return reward;
  805. })
  806. .collect(Collectors.toList());
  807. pointSignRewardService.batchInsertPointSignReward(rewards);
  808. }
  809. /**
  810. * 更新活动任务详情
  811. */
  812. private void updateActivityTaskDetails(PointActivityDTO dto, Long activityId) {
  813. // 先删除原有的活动任务
  814. pointActivityTaskService.deletePointActivityTaskByActivityId(activityId);
  815. // 重新插入新的活动任务
  816. List<PointActivityTaskDTO> tasks = dto.getTasks();
  817. if (CollectionUtil.isEmpty(tasks)) {
  818. return;
  819. }
  820. List<PointActivityTask> activityTasks = tasks.stream()
  821. .map(task -> {
  822. PointActivityTask activityTask = new PointActivityTask();
  823. BeanUtil.copyProperties(task, activityTask);
  824. activityTask.setActivityId(activityId);
  825. activityTask.setIsDeleted(0);
  826. activityTask.setUpdateBy(SecurityUtils.getUsername());
  827. activityTask.setUpdateTime(DateUtils.getNowDate());
  828. return activityTask;
  829. })
  830. .collect(Collectors.toList());
  831. pointActivityTaskService.batchInsertPointActivityTask(activityTasks);
  832. }
  833. private void batchUpdateExpirePolicy(List<PointActivity> activityList, Long expirePolicyId) {
  834. // 参数校验
  835. if (CollectionUtil.isEmpty(activityList)) {
  836. log.debug("活动列表为空,无需更新过期策略");
  837. return;
  838. }
  839. if (expirePolicyId == null) {
  840. throw new IllegalArgumentException("过期策略ID不能为空");
  841. }
  842. try {
  843. // 使用Stream处理活动列表,设置过期策略ID
  844. List<PointActivity> activitiesToUpdate = activityList.stream()
  845. .map(activity -> {
  846. activity.setExpirePolicyId(expirePolicyId);
  847. activity.setUpdateTime(DateUtils.getNowDate());
  848. activity.setUpdateBy(SecurityUtils.getUsername());
  849. return activity;
  850. })
  851. .collect(Collectors.toList());
  852. // 批量更新数据库
  853. super.updateBatchById(activitiesToUpdate, BATCH_SIZE);
  854. log.info("成功更新 {} 个活动的过期策略为 {}", activitiesToUpdate.size(), expirePolicyId);
  855. } catch (Exception e) {
  856. log.error("批量更新活动过期策略失败, expirePolicyId: {}", expirePolicyId, e);
  857. throw new ServiceException("批量更新活动过期策略失败");
  858. }
  859. }
  860. private PointSignTask getEnabledTask(String cityCode) {
  861. // 1. 根据城市查询当前生效的活动
  862. PointActivity activity = pointActivityMapper.selectOne(
  863. new QueryWrapper<PointActivity>()
  864. .eq("city_code", cityCode)
  865. .eq("status", 1) // 活动启用
  866. .last("LIMIT 1")
  867. );
  868. if (ObjectUtil.isNull(activity)) {
  869. throw new ServiceException("当前城市未配置签到活动");
  870. }
  871. // 2. 根据活动绑定的 taskId 查询签到任务
  872. LambdaQueryWrapper<PointSignTask> queryWrapper = new LambdaQueryWrapper<>();
  873. queryWrapper.eq(PointSignTask::getActivityId, activity.getId());
  874. queryWrapper.eq(PointSignTask::getIsDeleted, 0);
  875. List<PointSignTask> list = pointSignTaskService.list(queryWrapper);
  876. PointSignTask task = null;
  877. if (CollectionUtil.isNotEmpty(list)) {
  878. task = CollectionUtil.getLast(list);
  879. task.setActivityName(activity.getName());
  880. }
  881. return task;
  882. }
  883. private PointActivityPageVo convertToVo(PointActivity activity) {
  884. PointActivityPageVo vo = new PointActivityPageVo();
  885. // 属性拷贝(推荐)
  886. BeanUtil.copyProperties(activity, vo);
  887. return vo;
  888. }
  889. private WxLoginUser getCurrentWxLoginUser() {
  890. WxLoginUser loginUser = SecurityUtils.getWxLoginUser();
  891. if (ObjectUtil.isNull(loginUser)) {
  892. throw new ServiceException("用户未登录或登录已过期");
  893. }
  894. return loginUser;
  895. }
  896. }