package cn.lili.modules.lmk.service.impl; import cn.lili.cache.Cache; import cn.lili.cache.CachePrefix; import cn.lili.common.properties.RocketmqCustomProperties; import cn.lili.common.security.AuthUser; import cn.lili.common.security.context.UserContext; import cn.lili.common.sensitive.SensitiveWordsFilter; import cn.lili.modules.lmk.constant.RedisKeyExpireConstant; import cn.lili.modules.lmk.domain.entity.ThumbsUpRecord; import cn.lili.modules.lmk.domain.form.ThumbsUpRecordForm; import cn.lili.modules.lmk.domain.vo.CollectTypeNumVO; import cn.lili.modules.lmk.enums.general.VideoCommentStatusEnum; import cn.lili.modules.lmk.event.event.VideoCommentNumCacheEvent; import cn.lili.modules.lmk.service.ThumbsUpRecordService; import cn.lili.rocketmq.RocketmqSendCallbackBuilder; import cn.lili.rocketmq.tags.CommentTagsEnum; import cn.lili.rocketmq.tags.OrderTagsEnum; import com.alibaba.fastjson.JSON; import com.baomidou.mybatisplus.core.metadata.IPage; import cn.lili.modules.lmk.domain.entity.VideoComment; import cn.lili.modules.lmk.mapper.VideoCommentMapper; import cn.lili.modules.lmk.service.VideoCommentService; import cn.lili.base.Result; import com.baomidou.mybatisplus.extension.conditions.query.LambdaQueryChainWrapper; import com.baomidou.mybatisplus.extension.conditions.update.LambdaUpdateChainWrapper; import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; import cn.lili.modules.lmk.domain.form.VideoCommentForm; import cn.lili.modules.lmk.domain.vo.VideoCommentVO; import cn.lili.modules.lmk.domain.query.VideoCommentQuery; import org.apache.commons.collections4.ListUtils; import org.apache.commons.lang3.StringUtils; import org.apache.rocketmq.client.producer.DefaultMQProducer; import org.apache.rocketmq.common.MixAll; import org.apache.rocketmq.spring.core.RocketMQTemplate; import org.springframework.context.ApplicationEventPublisher; 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.util.Assert; import java.util.List; import java.util.Objects; import java.util.concurrent.TimeUnit; import java.util.stream.Collectors; /** * 视频评论 服务实现类 * * @author xp * @since 2025-05-27 */ @Service @RequiredArgsConstructor public class VideoCommentServiceImpl extends ServiceImpl implements VideoCommentService { private final VideoCommentMapper videoCommentMapper; private final Cache cache; private final ThumbsUpRecordService thumbsUpRecordService; private final RocketMQTemplate rocketMQTemplate; private final RocketmqCustomProperties rocketmqCustomProperties; private final ApplicationEventPublisher eventPublisher; /** * 添加 * @param form * @return */ @Override public Result comment(VideoCommentForm form) { // 监测内容是否包含敏感词 if (SensitiveWordsFilter.includeSentenceWord(form.getCommentContent())) { return Result.error("评论含敏感内容"); } if (StringUtils.isNotBlank(form.getReplyId())) { VideoComment beReplyComment = baseMapper.selectById(form.getReplyId()); if (Objects.isNull(beReplyComment)) { throw new RuntimeException("您回复的评论已被删除"); } else if (VideoCommentStatusEnum.INVALID.getValue().equals(beReplyComment.getStatus())) { throw new RuntimeException("您回复的评论已失效"); } } VideoComment entity = VideoCommentForm.getEntityByForm(form, null); entity.setStatus(VideoCommentStatusEnum.NORMAL.getValue()); entity.setUserId(UserContext.getCurrentUserId()); AuthUser currentUser = UserContext.getCurrentUser(); entity.setUserNickname(currentUser.getNickName()); entity.setUserAvatar(currentUser.getFace()); baseMapper.insert(entity); // 初始化redis中评论的点赞数量 cache.put(CachePrefix.VIDEO_COMMENT_LIKE_NUM.getPrefixWithId(entity.getId()), 0, RedisKeyExpireConstant.COMMENT_LIKE_NUM_EXPIRE, RedisKeyExpireConstant.EXPIRE_DAY); // 处理视频评论数 eventPublisher.publishEvent(new VideoCommentNumCacheEvent(this, entity.getVideoId())); return Result.ok("添加成功").data(this.detail(entity.getId()).get("data")); } /** * 批量删除 * @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(VideoCommentQuery query) { IPage page = PageUtil.getPage(query, VideoCommentVO.class); baseMapper.getPage(page, query); return Result.ok().data(page.getRecords()).total(page.getTotal()); } /** * 根据id查找 * @param id * @return */ @Override public Result detail(String id) { VideoCommentVO 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 -> VideoCommentVO.getVoByEntity(entity, null)) .collect(Collectors.toList()); return Result.ok().data(vos); } @Override public Result wxPage(VideoCommentQuery query) { query.setUserId(UserContext.getCurrentUserId()); IPage page = PageUtil.getPage(query, VideoCommentVO.class); if (StringUtils.isNotBlank(query.getMasterCommentId())) { // 加载子评论的情况 baseMapper.replyCommentPage(page, query); for (VideoCommentVO comment : page.getRecords()) { comment.setThumbsUpNum(this.getCommentThumbsUpNum(comment.getId(), comment.getThumbsUpNum())); } return Result.ok().data(page.getRecords()); } else { // 加载主评论的情况。主评论id = masterCommentId baseMapper.masterCommentPage(page, query); for (VideoCommentVO comment : page.getRecords()) { comment.setThumbsUpNum(this.getCommentThumbsUpNum(comment.getId(), comment.getThumbsUpNum())); } } return Result.ok().data(page.getRecords()).total(page.getTotal()); } @Override public List countNumGroupByVideo() { return baseMapper.countNumGroupByVideo(); } @Override public Result thumbsUp(ThumbsUpRecordForm form) { ThumbsUpRecord record = ThumbsUpRecordForm.getEntityByForm(form, null); record.setUserId(UserContext.getCurrentUserId()); // 走mq异步处理 String destination = rocketmqCustomProperties.getCommentTopic() + ":" + CommentTagsEnum.THUMBS_UP.name(); rocketMQTemplate.asyncSend(destination, JSON.toJSONString(record), RocketmqSendCallbackBuilder.commonCallback()); return Result.ok(); } @Override @Transactional(rollbackFor = Exception.class) public void mqThumbsUp(ThumbsUpRecord record) { boolean exists = new LambdaQueryChainWrapper<>(thumbsUpRecordService.getBaseMapper()) .eq(ThumbsUpRecord::getRefId, record.getRefId()) .eq(ThumbsUpRecord::getUserId, record.getUserId()) .exists(); if (exists) { return; } thumbsUpRecordService.save(record); VideoComment comment = this.getById(record.getRefId()); if (cache.exist(CachePrefix.VIDEO_COMMENT_LIKE_NUM.getPrefixWithId(record.getRefId()))) { cache.incr(CachePrefix.VIDEO_COMMENT_LIKE_NUM.getPrefixWithId(record.getRefId())); } else { if (Objects.nonNull(comment)) { cache.put(CachePrefix.VIDEO_COMMENT_LIKE_NUM.getPrefixWithId(comment.getId()), comment.getThumbsUpNum() + 1, RedisKeyExpireConstant.COMMENT_LIKE_NUM_EXPIRE, RedisKeyExpireConstant.EXPIRE_DAY); } } // 标识该评论需要通过定时任务统计点赞数 if (Objects.nonNull(comment) && ! comment.getThumbsUpJob()) { new LambdaUpdateChainWrapper<>(baseMapper) .eq(VideoComment::getId, comment.getId()) .set(VideoComment::getThumbsUpJob, Boolean.TRUE) .update(); } } @Override public Result cancelThumbsUp(ThumbsUpRecordForm form) { ThumbsUpRecord record = ThumbsUpRecordForm.getEntityByForm(form, null); record.setUserId(UserContext.getCurrentUserId()); // 走mq异步处理 String destination = rocketmqCustomProperties.getCommentTopic() + ":" + CommentTagsEnum.CANCEL_THUMBS_UP.name(); rocketMQTemplate.asyncSend(destination, JSON.toJSONString(record), RocketmqSendCallbackBuilder.commonCallback()); return Result.ok(); } @Override @Transactional(rollbackFor = Exception.class) public void mqCancelThumbsUp(ThumbsUpRecord record) { new LambdaUpdateChainWrapper<>(thumbsUpRecordService.getBaseMapper()) .eq(ThumbsUpRecord::getRefId, record.getRefId()) .eq(ThumbsUpRecord::getThumbsUpType, record.getThumbsUpType()) .eq(ThumbsUpRecord::getUserId, record.getUserId()) .remove(); // redis数量减一 VideoComment comment = this.getById(record.getRefId()); if (cache.exist(CachePrefix.VIDEO_COMMENT_LIKE_NUM.getPrefixWithId(record.getRefId()))) { cache.decr(CachePrefix.VIDEO_COMMENT_LIKE_NUM.getPrefixWithId(record.getRefId())); } else { if (Objects.nonNull(comment)) { cache.put(CachePrefix.VIDEO_COMMENT_LIKE_NUM.getPrefixWithId(comment.getId()), comment.getThumbsUpNum() - 1, RedisKeyExpireConstant.COMMENT_LIKE_NUM_EXPIRE, RedisKeyExpireConstant.EXPIRE_DAY); } } // 标识该评论需要通过定时任务统计点赞数 if (Objects.nonNull(comment) && ! comment.getThumbsUpJob()) { new LambdaUpdateChainWrapper<>(baseMapper) .eq(VideoComment::getId, comment.getId()) .set(VideoComment::getThumbsUpJob, Boolean.TRUE) .update(); } } /** * 从redis中获取评论点赞数量,如果redis中没有则将mysql中的数量写入到redis * * @param commentId * @param mysqlNum * @return */ private Integer getCommentThumbsUpNum(String commentId, Integer mysqlNum) { Object redisNum = cache.get(CachePrefix.VIDEO_COMMENT_LIKE_NUM.getPrefixWithId(commentId)); if (Objects.isNull(redisNum)) { // redis中没有就把数据库的写到redis中 cache.put(CachePrefix.VIDEO_COMMENT_LIKE_NUM.getPrefixWithId(commentId), mysqlNum, RedisKeyExpireConstant.COMMENT_LIKE_NUM_EXPIRE, RedisKeyExpireConstant.EXPIRE_DAY); return mysqlNum; } return (Integer) redisNum; } @Override @Transactional(rollbackFor = Exception.class) public void updateCommentThumbsUpNumBatch(List numList) { // 按500条数据进行拆分 List> chunks = ListUtils.partition(numList, 500); for (List chunk : chunks) { baseMapper.updateCommentThumbsUpNumBatch(chunk); new LambdaUpdateChainWrapper<>(baseMapper) .in(VideoComment::getId, chunk.stream().map(CollectTypeNumVO::getId).collect(Collectors.toList())) .set(VideoComment::getThumbsUpJob, Boolean.FALSE) .update(); } } }