// Copyright 2022 ROC. All rights reserved. // Use of this source code is governed by a MIT style // license that can be found in the LICENSE file. package jinzhu import ( "fmt" "time" "github.com/rocboss/paopao-ce/internal/core" "github.com/rocboss/paopao-ce/internal/core/cs" "github.com/rocboss/paopao-ce/internal/core/ms" "github.com/rocboss/paopao-ce/internal/dao/jinzhu/dbr" "github.com/rocboss/paopao-ce/pkg/types" "gorm.io/gorm" ) var ( _ core.CommentService = (*commentSrv)(nil) _ core.CommentManageService = (*commentManageSrv)(nil) ) type commentSrv struct { db *gorm.DB } type commentManageSrv struct { db *gorm.DB cms core.CommentMetricServantA } func newCommentService(db *gorm.DB) core.CommentService { return &commentSrv{ db: db, } } func newCommentManageService(db *gorm.DB, cms core.CommentMetricServantA) core.CommentManageService { return &commentManageSrv{ db: db, cms: cms, } } func (s *commentSrv) GetCommentThumbsMap(userId int64, tweetId int64) (cs.CommentThumbsMap, cs.CommentThumbsMap, error) { if userId < 0 { return nil, nil, nil } commentThumbsList := cs.CommentThumbsList{} err := s.db.Model(&dbr.TweetCommentThumbs{}).Where("user_id=? AND tweet_id=?", userId, tweetId).Find(&commentThumbsList).Error if err != nil { return nil, nil, err } commentThumbs, replyThumbs := make(cs.CommentThumbsMap), make(cs.CommentThumbsMap) for _, thumbs := range commentThumbsList { if thumbs.CommentType == 0 { commentThumbs[thumbs.CommentID] = thumbs } else { replyThumbs[thumbs.ReplyID] = thumbs } } return commentThumbs, replyThumbs, nil } func (s *commentSrv) GetComments(tweetId int64, style cs.StyleCommentType, limit int, offset int) (res []*ms.Comment, total int64, err error) { db := s.db.Table(_comment_) sort := "is_essence DESC, id ASC" switch style { case cs.StyleCommentHots: // rank_score=评论回复数*2+点赞*4-点踩, order byrank_score DESC db = db.Joins(fmt.Sprintf("LEFT JOIN %s m ON %s.id=m.comment_id AND m.is_del=0", _commentMetric_, _comment_)) sort = fmt.Sprintf("is_essence DESC, m.rank_score DESC, %s.id DESC", _comment_) case cs.StyleCommentNewest: sort = "is_essence DESC, id DESC" case cs.StyleCommentDefault: fallthrough default: // nothing } db = db.Where("post_id=?", tweetId) if err = db.Count(&total).Error; err != nil { return } err = db.Order(sort).Limit(limit).Offset(offset).Find(&res).Error return } func (s *commentSrv) GetCommentByID(id int64) (*ms.Comment, error) { comment := &dbr.Comment{ Model: &dbr.Model{ ID: id, }, } return comment.Get(s.db) } func (s *commentSrv) GetCommentReplyByID(id int64) (*ms.CommentReply, error) { reply := &dbr.CommentReply{ Model: &dbr.Model{ ID: id, }, } return reply.Get(s.db) } func (s *commentSrv) GetCommentCount(conditions *ms.ConditionsT) (int64, error) { return (&dbr.Comment{}).Count(s.db, conditions) } func (s *commentSrv) GetCommentContentsByIDs(ids []int64) ([]*ms.CommentContent, error) { commentContent := &dbr.CommentContent{} return commentContent.List(s.db, &dbr.ConditionsT{ "comment_id IN ?": ids, }, 0, 0) } func (s *commentSrv) GetCommentRepliesByID(ids []int64) ([]*ms.CommentReplyFormated, error) { CommentReply := &dbr.CommentReply{} replies, err := CommentReply.List(s.db, &dbr.ConditionsT{ "comment_id IN ?": ids, "ORDER": "id ASC", }, 0, 0) if err != nil { return nil, err } userIds := []int64{} for _, reply := range replies { userIds = append(userIds, reply.UserID, reply.AtUserID) } users, err := getUsersByIDs(s.db, userIds) if err != nil { return nil, err } repliesFormated := []*ms.CommentReplyFormated{} for _, reply := range replies { replyFormated := reply.Format() for _, user := range users { if reply.UserID == user.ID { replyFormated.User = user.Format() } if reply.AtUserID == user.ID { replyFormated.AtUser = user.Format() } } repliesFormated = append(repliesFormated, replyFormated) } return repliesFormated, nil } func (s *commentManageSrv) HighlightComment(userId, commentId int64) (res int8, err error) { post := &dbr.Post{} comment := &dbr.Comment{} db := s.db.Model(comment) if err = db.Where("id=?", commentId).First(comment).Error; err != nil { return } if err = s.db.Table(_post_).Where("id=?", comment.PostID).First(post).Error; err != nil { return } if post.UserID != userId { return 0, cs.ErrNoPermission } comment.IsEssence = 1 - comment.IsEssence return comment.IsEssence, db.Save(comment).Error } func (s *commentManageSrv) DeleteComment(comment *ms.Comment) (err error) { db := s.db.Begin() defer db.Rollback() if err = comment.Delete(db); err != nil { return } err = db.Model(&dbr.TweetCommentThumbs{}).Where("user_id=? AND tweet_id=? AND comment_id=?", comment.UserID, comment.PostID, comment.ID).Updates(map[string]any{ "deleted_on": time.Now().Unix(), "is_del": 1, }).Error if err != nil { return } db.Commit() if err == nil { // 对错误宽松处理,暂时就不处理 s.cms.DeleteCommentMetric(comment.ID) } return } func (s *commentManageSrv) CreateComment(comment *ms.Comment) (res *ms.Comment, err error) { if res, err = comment.Create(s.db); err == nil { // 对错误宽松处理,暂时就不处理 s.cms.AddCommentMetric(res.ID) } return } func (s *commentManageSrv) CreateCommentReply(reply *ms.CommentReply) (res *ms.CommentReply, err error) { if res, err = reply.Create(s.db); err == nil { // 宽松处理错误 s.db.Table(_comment_).Where("id=?", reply.CommentID).Update("reply_count", gorm.Expr("reply_count+1")) onUpdateCommentMetricEvent(reply.CommentID, s.db, s.cms) } return } func (s *commentManageSrv) DeleteCommentReply(reply *ms.CommentReply) (err error) { db := s.db.Begin() defer db.Rollback() err = reply.Delete(s.db) if err != nil { return } err = db.Model(&dbr.TweetCommentThumbs{}). Where("user_id=? AND comment_id=? AND reply_id=?", reply.UserID, reply.CommentID, reply.ID).Updates(map[string]any{ "deleted_on": time.Now().Unix(), "is_del": 1, }).Error if err != nil { return } // 宽松处理错误 db.Table(_comment_).Where("id=?", reply.CommentID).Update("reply_count", gorm.Expr("reply_count-1")) onUpdateCommentMetricEvent(reply.CommentID, s.db, s.cms) db.Commit() return } func (s *commentManageSrv) CreateCommentContent(content *ms.CommentContent) (*ms.CommentContent, error) { return content.Create(s.db) } func (s *commentManageSrv) ThumbsUpComment(userId int64, tweetId, commentId int64) error { db := s.db.Begin() defer db.Rollback() var ( thumbsUpCount int32 = 0 thumbsDownCount int32 = 0 ) commentThumbs := &dbr.TweetCommentThumbs{} // 检查thumbs状态 err := s.db.Where("user_id=? AND tweet_id=? AND comment_id=? AND comment_type=0", userId, tweetId, commentId).Take(commentThumbs).Error if err == nil { switch { case commentThumbs.IsThumbsUp == types.Yes && commentThumbs.IsThumbsDown == types.No: thumbsUpCount, thumbsDownCount = -1, 0 case commentThumbs.IsThumbsUp == types.No && commentThumbs.IsThumbsDown == types.No: thumbsUpCount, thumbsDownCount = 1, 0 default: thumbsUpCount, thumbsDownCount = 1, -1 commentThumbs.IsThumbsDown = types.No } commentThumbs.IsThumbsUp = 1 - commentThumbs.IsThumbsUp commentThumbs.ModifiedOn = time.Now().Unix() } else { commentThumbs = &dbr.TweetCommentThumbs{ UserID: userId, TweetID: tweetId, CommentID: commentId, IsThumbsUp: types.Yes, IsThumbsDown: types.No, CommentType: 0, Model: &dbr.Model{ CreatedOn: time.Now().Unix(), }, } thumbsUpCount, thumbsDownCount = 1, 0 } // 更新thumbs状态 if err = s.db.Save(commentThumbs).Error; err != nil { return err } // 更新thumbsUpCount if err = s.updateCommentThumbsUpCount(&dbr.Comment{}, commentId, thumbsUpCount, thumbsDownCount); err != nil { return err } db.Commit() if err == nil { onUpdateCommentMetricEvent(commentId, s.db, s.cms) } return nil } func (s *commentManageSrv) ThumbsDownComment(userId int64, tweetId, commentId int64) error { db := s.db.Begin() defer db.Rollback() var ( thumbsUpCount int32 = 0 thumbsDownCount int32 = 0 ) commentThumbs := &dbr.TweetCommentThumbs{} // 检查thumbs状态 err := s.db.Where("user_id=? AND tweet_id=? AND comment_id=? AND comment_type=0", userId, tweetId, commentId).Take(commentThumbs).Error if err == nil { switch { case commentThumbs.IsThumbsDown == types.Yes: thumbsUpCount, thumbsDownCount = 0, -1 case commentThumbs.IsThumbsDown == types.No && commentThumbs.IsThumbsUp == types.No: thumbsUpCount, thumbsDownCount = 0, 1 default: thumbsUpCount, thumbsDownCount = -1, 1 commentThumbs.IsThumbsUp = types.No } commentThumbs.IsThumbsDown = 1 - commentThumbs.IsThumbsDown commentThumbs.ModifiedOn = time.Now().Unix() } else { commentThumbs = &dbr.TweetCommentThumbs{ UserID: userId, TweetID: tweetId, CommentID: commentId, IsThumbsUp: types.No, IsThumbsDown: types.Yes, CommentType: 0, Model: &dbr.Model{ CreatedOn: time.Now().Unix(), }, } thumbsUpCount, thumbsDownCount = 0, 1 } // 更新thumbs状态 if err = s.db.Save(commentThumbs).Error; err != nil { return err } // 更新thumbsUpCount if err = s.updateCommentThumbsUpCount(&dbr.Comment{}, commentId, thumbsUpCount, thumbsDownCount); err != nil { return err } db.Commit() if err == nil { onUpdateCommentMetricEvent(commentId, s.db, s.cms) } return nil } func (s *commentManageSrv) ThumbsUpReply(userId int64, tweetId, commentId, replyId int64) error { db := s.db.Begin() defer db.Rollback() var ( thumbsUpCount int32 = 0 thumbsDownCount int32 = 0 ) commentThumbs := &dbr.TweetCommentThumbs{} // 检查thumbs状态 err := s.db.Where("user_id=? AND tweet_id=? AND comment_id=? AND reply_id=? AND comment_type=1", userId, tweetId, commentId, replyId).Take(commentThumbs).Error if err == nil { switch { case commentThumbs.IsThumbsUp == types.Yes: thumbsUpCount, thumbsDownCount = -1, 0 case commentThumbs.IsThumbsUp == types.No && commentThumbs.IsThumbsDown == types.No: thumbsUpCount, thumbsDownCount = 1, 0 default: thumbsUpCount, thumbsDownCount = 1, -1 commentThumbs.IsThumbsDown = types.No } commentThumbs.IsThumbsUp = 1 - commentThumbs.IsThumbsUp commentThumbs.ModifiedOn = time.Now().Unix() } else { commentThumbs = &dbr.TweetCommentThumbs{ UserID: userId, TweetID: tweetId, CommentID: commentId, ReplyID: replyId, IsThumbsUp: types.Yes, IsThumbsDown: types.No, CommentType: 1, Model: &dbr.Model{ CreatedOn: time.Now().Unix(), }, } thumbsUpCount, thumbsDownCount = 1, 0 } // 更新thumbs状态 if err = s.db.Save(commentThumbs).Error; err != nil { return err } // 更新thumbsUpCount if err = s.updateCommentThumbsUpCount(&dbr.CommentReply{}, replyId, thumbsUpCount, thumbsDownCount); err != nil { return err } db.Commit() return nil } func (s *commentManageSrv) ThumbsDownReply(userId int64, tweetId, commentId, replyId int64) error { db := s.db.Begin() defer db.Rollback() var ( thumbsUpCount int32 = 0 thumbsDownCount int32 = 0 ) commentThumbs := &dbr.TweetCommentThumbs{} // 检查thumbs状态 err := s.db.Where("user_id=? AND tweet_id=? AND comment_id=? AND reply_id=? AND comment_type=1", userId, tweetId, commentId, replyId).Take(commentThumbs).Error if err == nil { switch { case commentThumbs.IsThumbsDown == types.Yes: thumbsUpCount, thumbsDownCount = 0, -1 case commentThumbs.IsThumbsUp == types.No && commentThumbs.IsThumbsDown == types.No: thumbsUpCount, thumbsDownCount = 0, 1 default: thumbsUpCount, thumbsDownCount = -1, 1 commentThumbs.IsThumbsUp = types.No } commentThumbs.IsThumbsDown = 1 - commentThumbs.IsThumbsDown commentThumbs.ModifiedOn = time.Now().Unix() } else { commentThumbs = &dbr.TweetCommentThumbs{ UserID: userId, TweetID: tweetId, CommentID: commentId, ReplyID: replyId, IsThumbsUp: types.No, IsThumbsDown: types.Yes, CommentType: 1, Model: &dbr.Model{ CreatedOn: time.Now().Unix(), }, } thumbsUpCount, thumbsDownCount = 0, 1 } // 更新thumbs状态 if err = s.db.Save(commentThumbs).Error; err != nil { return err } // 更新thumbsUpCount if err = s.updateCommentThumbsUpCount(&dbr.CommentReply{}, replyId, thumbsUpCount, thumbsDownCount); err != nil { return err } db.Commit() return nil } func (s *commentManageSrv) updateCommentThumbsUpCount(obj any, id int64, thumbsUpCount, thumbsDownCount int32) error { updateColumns := make(map[string]any, 2) if thumbsUpCount == 1 { updateColumns["thumbs_up_count"] = gorm.Expr("thumbs_up_count + 1") } else if thumbsUpCount == -1 { updateColumns["thumbs_up_count"] = gorm.Expr("thumbs_up_count - 1") } if thumbsDownCount == 1 { updateColumns["thumbs_down_count"] = gorm.Expr("thumbs_down_count + 1") } else if thumbsDownCount == -1 { updateColumns["thumbs_down_count"] = gorm.Expr("thumbs_down_count - 1") } if len(updateColumns) > 0 { updateColumns["modified_on"] = time.Now().Unix() return s.db.Model(obj).Where("id=?", id).UpdateColumns(updateColumns).Error } return nil }