package cn.lili.modules.lmk.service.impl; import cn.lili.common.exception.ServiceException; import cn.lili.common.security.AuthUser; import cn.lili.common.security.context.UserContext; import cn.lili.modules.lmk.domain.entity.StoreCoupon; import cn.lili.modules.lmk.domain.entity.StoreCouponSingle; import cn.lili.modules.lmk.enums.general.PrizeStatusEnum; import cn.lili.modules.lmk.enums.general.StoreCouponStausEnum; import cn.lili.modules.lmk.service.StoreCouponService; import cn.lili.modules.lmk.service.StoreCouponSingleService; import cn.lili.modules.order.order.entity.enums.ClaimStatusEnum; import cn.lili.modules.promotion.entity.dos.MemberCoupon; import cn.lili.modules.promotion.service.MemberCouponService; import cn.lili.rocketmq.RocketmqSendCallbackBuilder; import com.alibaba.fastjson.JSON; import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.baomidou.mybatisplus.core.conditions.update.LambdaUpdateWrapper; import com.baomidou.mybatisplus.core.metadata.IPage; import cn.lili.modules.lmk.domain.entity.StoreCouponClaimRecord; import cn.lili.modules.lmk.mapper.StoreCouponClaimRecordMapper; import cn.lili.modules.lmk.service.StoreCouponClaimRecordService; import cn.lili.base.Result; import com.baomidou.mybatisplus.core.toolkit.Wrappers; import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; import cn.lili.modules.lmk.domain.form.StoreCouponClaimRecordForm; import cn.lili.modules.lmk.domain.vo.StoreCouponClaimRecordVO; import cn.lili.modules.lmk.domain.query.StoreCouponClaimRecordQuery; import org.redisson.api.RLock; import org.redisson.api.RedissonClient; import org.springframework.stereotype.Service; import lombok.RequiredArgsConstructor; import cn.lili.utils.PageUtil; import org.springframework.beans.BeanUtils; import org.springframework.transaction.annotation.Transactional; import org.springframework.transaction.support.TransactionSynchronization; import org.springframework.transaction.support.TransactionSynchronizationManager; import org.springframework.util.Assert; import java.util.List; import java.util.stream.Collectors; /** * 店铺优惠卷领取记录 服务实现类 * * @author peng * @since 2025-09-25 */ @Service @RequiredArgsConstructor public class StoreCouponClaimRecordServiceImpl extends ServiceImpl implements StoreCouponClaimRecordService { private final StoreCouponClaimRecordMapper storeCouponClaimRecordMapper; private final RedissonClient redissonClient; private static final String STORE_COUPON_CLAIM = "store_coupon_claim:"; private final MemberCouponService memberCouponService; private final StoreCouponService storeCouponService; private final StoreCouponSingleService storeCouponSingleService; /** * 添加 * @param form * @return */ @Override public Result add(StoreCouponClaimRecordForm form) { StoreCouponClaimRecord entity = StoreCouponClaimRecordForm.getEntityByForm(form, null); baseMapper.insert(entity); return Result.ok("添加成功"); } /** * 修改 * @param form * @return */ @Override public Result update(StoreCouponClaimRecordForm form) { StoreCouponClaimRecord entity = baseMapper.selectById(form.getId()); // 为空抛IllegalArgumentException,做全局异常处理 Assert.notNull(entity, "记录不存在"); BeanUtils.copyProperties(form, entity); baseMapper.updateById(entity); return Result.ok("修改成功"); } /** * 批量删除 * @param ids * @return */ @Override public Result remove(List ids) { baseMapper.deleteBatchIds(ids); return Result.ok("删除成功"); } /** * id删除 * @param id * @return */ @Override public Result removeById(String id) { baseMapper.deleteById(id); return Result.ok("删除成功"); } /** * 分页查询 * @param query * @return */ @Override public Result page(StoreCouponClaimRecordQuery query) { IPage page = PageUtil.getPage(query, StoreCouponClaimRecordVO.class); baseMapper.getPage(page, query); return Result.ok().data(page.getRecords()).total(page.getTotal()); } /** * 根据id查找 * @param id * @return */ @Override public Result detail(String id) { StoreCouponClaimRecordVO vo = baseMapper.getById(id); Assert.notNull(vo, "记录不存在"); return Result.ok().data(vo); } /** * 列表 * @return */ @Override public Result all() { List entities = baseMapper.selectList(null); List vos = entities.stream() .map(entity -> StoreCouponClaimRecordVO.getVoByEntity(entity, null)) .collect(Collectors.toList()); return Result.ok().data(vos); } @Override @Transactional(rollbackFor = Exception.class) public Result claimCoupon(String id) { AuthUser currentUser = UserContext.getCurrentUser(); if (currentUser == null) { throw new ServiceException("当前用户没有登录无法领取"); } String userId = currentUser.getId(); String nickName = currentUser.getNickName(); //锁住礼品码id RLock redissonLock = redissonClient.getLock(STORE_COUPON_CLAIM + id); try { redissonLock.lock(); LambdaQueryWrapper forUpdate = Wrappers.lambdaQuery() .eq(StoreCouponSingle::getId, id).last("FOR UPDATE"); StoreCouponSingle storeCouponSingle = storeCouponSingleService.getOne(forUpdate); if (storeCouponSingle == null) { throw new ServiceException("当前礼品码不存在"); } if (!ClaimStatusEnum.NOT_CLAIM.name().equals(storeCouponSingle.getClaimStatus())) { throw new ServiceException("当前礼品码状态异常"); } LambdaQueryWrapper claimListQuery = Wrappers.lambdaQuery() .eq(StoreCouponSingle::getClaimUserId, userId) .eq(StoreCouponSingle::getStoreCoupRef, storeCouponSingle.getStoreCoupRef()) .eq(StoreCouponSingle::getClaimStatus, ClaimStatusEnum.CLAIM.name()); List claimList = storeCouponSingleService.list(claimListQuery); if (!claimList.isEmpty()) { throw new ServiceException("已经领取过该类型的礼品码无法领取"); } //处理幂等问题限制一个用户只能该店铺领取一种优惠卷 LambdaQueryWrapper memCoupon = Wrappers.lambdaQuery() .eq(StoreCouponClaimRecord::getUserId, userId) .eq(StoreCouponClaimRecord::getCouponId, storeCouponSingle.getCouponId()); List list = this.list(memCoupon); if (!list.isEmpty()){ throw new ServiceException("当前用户已经领取过了无法再次领取"); } //更新单品被领取的记录 storeCouponSingle.setClaimStatus(ClaimStatusEnum.CLAIM.name()); storeCouponSingle.setClaimUserId(userId); storeCouponSingle.setClaimUserName(nickName); storeCouponSingleService.updateById(storeCouponSingle); //校验是否在单品卷类领取过 LambdaQueryWrapper storeCoupQuery = Wrappers.lambdaQuery() .eq(StoreCoupon::getId, storeCouponSingle.getStoreCoupRef()).last("FOR UPDATE"); StoreCoupon storeCoupon = storeCouponService.getOne(storeCoupQuery); if (storeCoupon == null) { throw new ServiceException("当前店铺优惠卷不存在"); } if (!StoreCouponStausEnum.ENABLE.name().equals(storeCoupon.getStatus())) { throw new ServiceException("当前店铺优惠卷状态异常"); } //领取对应的优惠卷写入记录 memberCouponService.receiveCoupon(storeCouponSingle.getCouponId(),userId , nickName); StoreCouponClaimRecord storeCouponClaimRecord = getStoreCouponClaimRecord(storeCouponSingle, userId); this.save(storeCouponClaimRecord); LambdaUpdateWrapper updateStoreCoupon = Wrappers.lambdaUpdate().eq(StoreCoupon::getId, storeCoupon.getId()) .set(StoreCoupon::getCouponClaimNum, storeCoupon.getCouponClaimNum() + 1) .gt(StoreCoupon::getCouponClaimNum, storeCoupon.getCouponClaimNum() + 1); boolean update = storeCouponService.update(updateStoreCoupon); if (!update) { throw new ServiceException("更新失败"); } //领取成功返回优惠卷id用于跳转购物使用 return Result.ok().data(storeCouponSingle.getCouponId()); } finally { TransactionSynchronizationManager.registerSynchronization(new TransactionSynchronization() { @Override public void afterCommit() { if (redissonLock.isHeldByCurrentThread()) { redissonLock.unlock(); } } @Override public void afterCompletion(int status) { // 确保即使在事务回滚的情况下也能释放锁 if (redissonLock.isHeldByCurrentThread()) { redissonLock.unlock(); } } }); } } private static StoreCouponClaimRecord getStoreCouponClaimRecord(StoreCouponSingle storeCouponSingle, String userId) { StoreCouponClaimRecord storeCouponClaimRecord = new StoreCouponClaimRecord(); storeCouponClaimRecord.setCouponId(storeCouponSingle.getCouponId()); storeCouponClaimRecord.setCouponName(storeCouponSingle.getCouponName()); storeCouponClaimRecord.setStoreId(storeCouponSingle.getStoreId()); storeCouponClaimRecord.setStoreName(storeCouponSingle.getStoreName()); storeCouponClaimRecord.setUserId(userId); storeCouponClaimRecord.setStoreCouponId(storeCouponSingle.getStoreCoupRef()); return storeCouponClaimRecord; } }