From 70bac9077e1116e3e7b28e1628e9bbb043ad5f77 Mon Sep 17 00:00:00 2001 From: wucongxing8150 <815046773@qq.com> Date: Wed, 28 Aug 2024 18:31:18 +0800 Subject: [PATCH] =?UTF-8?q?=E5=88=9D=E5=A7=8B=E5=8C=96=E6=8F=90=E4=BA=A4?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- api/controller/Article.go | 289 +++++++++++++++++++++++++++++++++++++ api/controller/Base.go | 5 +- api/controller/Public.go | 38 +++++ api/controller/Video.go | 289 +++++++++++++++++++++++++++++++++++++ api/dao/Article.go | 93 ++++++++++++ api/dao/BaseArea.go | 72 +++++++++ api/dao/SystemTime.go | 117 +++++++++++++++ api/dao/Video.go | 93 ++++++++++++ api/dto/Article.go | 84 +++++++++++ api/dto/ArticleAuthor.go | 42 ++++++ api/dto/Video.go | 85 +++++++++++ api/dto/VideoAuthor.go | 42 ++++++ api/middlewares/auth.go | 34 ++--- api/middlewares/jwt.go | 7 +- api/model/Article.go | 14 +- api/model/ArticleAuthor.go | 1 + api/model/BaseArea.go | 26 ++++ api/model/BaseHospital.go | 46 ++++++ api/model/Data.go | 2 +- api/model/SystemTime.go | 33 +++++ api/model/Video.go | 13 +- api/model/VideoAuthor.go | 1 + api/requests/Article.go | 12 ++ api/requests/Video.go | 12 ++ api/router/router.go | 42 ++++++ api/service/SystemTime.go | 49 +++++++ api/service/User.go | 109 ++++++++++++++ 27 files changed, 1608 insertions(+), 42 deletions(-) create mode 100644 api/controller/Article.go create mode 100644 api/controller/Public.go create mode 100644 api/controller/Video.go create mode 100644 api/dao/BaseArea.go create mode 100644 api/dao/SystemTime.go create mode 100644 api/dto/Article.go create mode 100644 api/dto/ArticleAuthor.go create mode 100644 api/dto/Video.go create mode 100644 api/dto/VideoAuthor.go create mode 100644 api/model/BaseArea.go create mode 100644 api/model/BaseHospital.go create mode 100644 api/model/SystemTime.go create mode 100644 api/requests/Article.go create mode 100644 api/requests/Video.go create mode 100644 api/service/SystemTime.go create mode 100644 api/service/User.go diff --git a/api/controller/Article.go b/api/controller/Article.go new file mode 100644 index 0000000..8de6ed2 --- /dev/null +++ b/api/controller/Article.go @@ -0,0 +1,289 @@ +package controller + +import ( + "fmt" + "github.com/gin-gonic/gin" + "net/http" + "strconv" + "time" + "vote-api/api/dao" + "vote-api/api/dto" + "vote-api/api/model" + "vote-api/api/requests" + "vote-api/api/responses" + "vote-api/api/service" + "vote-api/consts" + "vote-api/global" + "vote-api/utils" +) + +type Article struct{} + +// GetArticlePage 获取图文列表-分页 +func (r *Article) GetArticlePage(c *gin.Context) { + articleRequest := requests.ArticleRequest{} + req := articleRequest.GetArticlePage + if err := c.ShouldBind(&req); err != nil { + responses.FailWithMessage(err.Error(), c) + return + } + + // 参数验证 + if err := global.Validate.Struct(req); err != nil { + responses.FailWithMessage(utils.Translate(err), c) + return + } + + if req.Page == 0 { + req.Page = 1 + } + + if req.PageSize == 0 { + req.PageSize = 20 + } + + // 获取数据 + articleDao := dao.ArticleDao{} + articles, total, err := articleDao.GetArticlePageSearch(req, req.Page, req.PageSize) + if err != nil { + responses.FailWithMessage(err.Error(), c) + return + } + + // 处理返回值 + g := dto.GetArticleListDto(articles) + + // 检测用户今日是否投票 + userId := c.GetInt64("UserId") + if userId != 0 { + userService := service.UserService{} + for _, articleDto := range g { + articleId, err := strconv.ParseInt(articleDto.ArticleId, 10, 64) + if err != nil { + continue + } + + articleDto.IsVote = userService.CheckUserVoteDay(userId, articleId, 1) + } + } + + result := make(map[string]interface{}) + result["page"] = req.Page + result["page_size"] = req.PageSize + result["total"] = total + result["data"] = g + responses.OkWithData(result, c) +} + +// GetArticle 获取图文详情 +func (r *Article) GetArticle(c *gin.Context) { + id := c.Param("article_id") + if id == "" { + responses.FailWithMessage("缺少参数", c) + return + } + + // 将 id 转换为 int64 类型 + articleId, err := strconv.ParseInt(id, 10, 64) + if err != nil { + responses.Fail(c) + return + } + + // 获取数据 + articleDao := dao.ArticleDao{} + article, err := articleDao.GetArticlePreloadById(articleId) + if err != nil { + responses.FailWithMessage("文章错误", c) + return + } + + // 获取排名 + rank, _ := articleDao.GetArticleRank(article.ArticleId) + + // 处理返回值 + g := dto.GetArticleDto(article) + + // 加载数据-作者 + g.LoadArticleAuthor(article.ArticleAuthor) + + // 加载数据-作者排名 + g.LoadRank(rank) + + // 检测用户今日是否投票 + userId := c.GetInt64("UserId") + if userId != 0 { + userService := service.UserService{} + isVote := userService.CheckUserVoteDay(userId, articleId, 1) + + // 加载数据-投票状态 + g.LoadVoteStatus(isVote) + } + + responses.OkWithData(g, c) +} + +// GetArticleRankList 获取文章排名列表 +func (r *Article) GetArticleRankList(c *gin.Context) { + // 检测投票有效期 + systemTimeService := service.SystemTimeService{} + isValid := systemTimeService.CheckVoteValidStatus() + if isValid == false { + responses.OkWithData(nil, c) + return + } + + // 获取数据 + articleDao := dao.ArticleDao{} + maps := make(map[string]interface{}) + maps["article_status"] = 1 + articles, err := articleDao.GetArticleOrderList(maps, "vote_num desc", 15) + if err != nil { + responses.OkWithData(nil, c) + return + } + + // 处理返回值 + g := dto.GetArticleListDto(articles) + + // 检测用户今日是否投票 + userId := c.GetInt64("UserId") + if userId != 0 { + userService := service.UserService{} + for _, articleDto := range g { + articleId, err := strconv.ParseInt(articleDto.ArticleId, 10, 64) + if err != nil { + continue + } + + articleDto.IsVote = userService.CheckUserVoteDay(userId, articleId, 1) + } + } + + responses.OkWithData(g, c) +} + +// AddArticleVote 文章投票 +func (r *Article) AddArticleVote(c *gin.Context) { + id := c.Param("article_id") + if id == "" { + responses.FailWithMessage("缺少参数", c) + return + } + + // 将 id 转换为 int64 类型 + articleId, err := strconv.ParseInt(id, 10, 64) + if err != nil { + responses.Fail(c) + return + } + + userId := c.GetInt64("UserId") + if userId == 0 { + c.JSON(http.StatusUnauthorized, gin.H{ + "message": "请登录后投票", + "code": consts.TokenError, + "data": "", + }) + return + } + + // 检测并发请求 + redisKey := "AddArticleVote" + fmt.Sprintf("%d", userId) + fmt.Sprintf("%d", articleId) + res, _ := global.Redis.Get(c, redisKey).Result() + if res != "" { + responses.FailWithMessage("请勿重复操作", c) + return + } + + defer func(redisKey string) { + global.Redis.Del(c, redisKey) + }(redisKey) + + // 添加缓存 + _, err = global.Redis.Set(c, redisKey, "1", (1)*time.Second).Result() + if err != nil { + responses.FailWithMessage("投票失败", c) + return + } + + // 获取数据 + articleDao := dao.ArticleDao{} + article, err := articleDao.GetArticleById(articleId) + if err != nil || article == nil { + responses.FailWithMessage("非法数据", c) + return + } + + // 检测投票有效期 + systemTimeService := service.SystemTimeService{} + isValid := systemTimeService.CheckVoteValidStatus() + if isValid == false { + responses.FailWithMessage("投票已结束", c) + return + } + + // 检测用户今日是否投票 + userService := service.UserService{} + isVote := userService.CheckUserVoteDay(userId, articleId, 1) + if isVote == true { + responses.FailWithMessage("请勿重复投票", c) + return + } + + // 开始事务 + tx := global.Db.Begin() + defer func() { + if r := recover(); r != nil { + tx.Rollback() + utils.LogJsonErr("投票失败", r) + responses.FailWithMessage("投票失败", c) + return + } + }() + + // 增加投票数 + err = articleDao.Inc(tx, articleId, "vote_num", 1) + if err != nil { + tx.Rollback() + responses.FailWithMessage("投票失败", c) + return + } + + // 增加投票记录 + nowDay, _ := time.Parse("2006-01-02", time.Now().Format("2006-01-02")) + votedAt := model.LocalTime(nowDay) + articleVoteDay := &model.ArticleVoteDay{ + ArticleId: articleId, + UserId: userId, + VotedAt: &votedAt, + } + + articleVoteDayDao := dao.ArticleVoteDayDao{} + articleVoteDay, err = articleVoteDayDao.AddArticleVoteDay(tx, articleVoteDay) + if err != nil { + tx.Rollback() + responses.FailWithMessage("投票失败", c) + return + } + + // 增加数据表-投票数量 + dataDao := dao.DataDao{} + err = dataDao.Inc(tx, 1, "vote_num", 1) + if err != nil { + tx.Rollback() + responses.FailWithMessage("投票失败", c) + return + } + + // 新增投票缓存 + result := userService.AddUserVoteDayCache(userId, articleId, 1) + if result == false { + tx.Rollback() + responses.FailWithMessage("投票失败", c) + return + } + + tx.Commit() + responses.Ok(c) +} diff --git a/api/controller/Base.go b/api/controller/Base.go index ec6d740..f293e8a 100644 --- a/api/controller/Base.go +++ b/api/controller/Base.go @@ -2,5 +2,8 @@ package controller // Api api接口 type Api struct { - Login // 登录 + Login // 登录 + Article // 图文 + Video // 视频 + Public // 公共方法 } diff --git a/api/controller/Public.go b/api/controller/Public.go new file mode 100644 index 0000000..7b0156f --- /dev/null +++ b/api/controller/Public.go @@ -0,0 +1,38 @@ +package controller + +import ( + "github.com/gin-gonic/gin" + "vote-api/api/dao" + "vote-api/api/responses" + "vote-api/global" + "vote-api/utils" +) + +type Public struct { +} + +// AddBrowse 增加浏览数量 +func (r *Public) AddBrowse(c *gin.Context) { + // 开始事务 + tx := global.Db.Begin() + defer func() { + if r := recover(); r != nil { + tx.Rollback() + utils.LogJsonErr("增加浏览数量失败", r) + responses.Ok(c) + return + } + }() + + // 增加数据表-投票数量 + dataDao := dao.DataDao{} + err := dataDao.Inc(tx, 1, "view_num", 1) + if err != nil { + tx.Rollback() + responses.Ok(c) + return + } + + tx.Commit() + responses.Ok(c) +} diff --git a/api/controller/Video.go b/api/controller/Video.go new file mode 100644 index 0000000..4325aef --- /dev/null +++ b/api/controller/Video.go @@ -0,0 +1,289 @@ +package controller + +import ( + "fmt" + "github.com/gin-gonic/gin" + "net/http" + "strconv" + "time" + "vote-api/api/dao" + "vote-api/api/dto" + "vote-api/api/model" + "vote-api/api/requests" + "vote-api/api/responses" + "vote-api/api/service" + "vote-api/consts" + "vote-api/global" + "vote-api/utils" +) + +type Video struct{} + +// GetVideoPage 获取视频列表-分页 +func (r *Video) GetVideoPage(c *gin.Context) { + videoRequest := requests.VideoRequest{} + req := videoRequest.GetVideoPage + if err := c.ShouldBind(&req); err != nil { + responses.FailWithMessage(err.Error(), c) + return + } + + // 参数验证 + if err := global.Validate.Struct(req); err != nil { + responses.FailWithMessage(utils.Translate(err), c) + return + } + + if req.Page == 0 { + req.Page = 1 + } + + if req.PageSize == 0 { + req.PageSize = 20 + } + + // 获取数据 + videoDao := dao.VideoDao{} + videos, total, err := videoDao.GetVideoPageSearch(req, req.Page, req.PageSize) + if err != nil { + responses.FailWithMessage(err.Error(), c) + return + } + + // 处理返回值 + g := dto.GetVideoListDto(videos) + + // 检测用户今日是否投票 + userId := c.GetInt64("UserId") + if userId != 0 { + userService := service.UserService{} + for _, videoDto := range g { + videoId, err := strconv.ParseInt(videoDto.VideoId, 10, 64) + if err != nil { + continue + } + + videoDto.IsVote = userService.CheckUserVoteDay(userId, videoId, 2) + } + } + + result := make(map[string]interface{}) + result["page"] = req.Page + result["page_size"] = req.PageSize + result["total"] = total + result["data"] = g + responses.OkWithData(result, c) +} + +// GetVideo 获取视频详情 +func (r *Video) GetVideo(c *gin.Context) { + id := c.Param("video_id") + if id == "" { + responses.FailWithMessage("缺少参数", c) + return + } + + // 将 id 转换为 int64 类型 + videoId, err := strconv.ParseInt(id, 10, 64) + if err != nil { + responses.Fail(c) + return + } + + // 获取数据 + videoDao := dao.VideoDao{} + video, err := videoDao.GetVideoPreloadById(videoId) + if err != nil { + responses.FailWithMessage("文章错误", c) + return + } + + // 获取排名 + rank, _ := videoDao.GetVideoRank(video.VideoId) + + // 处理返回值 + g := dto.GetVideoDto(video) + + // 加载数据-作者 + g.LoadVideoAuthor(video.VideoAuthor) + + // 加载数据-作者排名 + g.LoadRank(rank) + + // 检测用户今日是否投票 + userId := c.GetInt64("UserId") + if userId != 0 { + userService := service.UserService{} + isVote := userService.CheckUserVoteDay(userId, videoId, 2) + + // 加载数据-投票状态 + g.LoadVoteStatus(isVote) + } + + responses.OkWithData(g, c) +} + +// GetVideoRankList 获取视频排名列表 +func (r *Video) GetVideoRankList(c *gin.Context) { + // 检测投票状态 + systemTimeService := service.SystemTimeService{} + isValid := systemTimeService.CheckVoteValidStatus() + if isValid == false { + responses.OkWithData(nil, c) + return + } + + // 获取数据 + videoDao := dao.VideoDao{} + maps := make(map[string]interface{}) + maps["video_status"] = 1 + videos, err := videoDao.GetVideoOrderList(maps, "vote_num desc", 15) + if err != nil { + responses.OkWithData(nil, c) + return + } + + // 处理返回值 + g := dto.GetVideoListDto(videos) + + // 检测用户今日是否投票 + userId := c.GetInt64("UserId") + if userId != 0 { + userService := service.UserService{} + for _, videoDto := range g { + videoId, err := strconv.ParseInt(videoDto.VideoId, 10, 64) + if err != nil { + continue + } + + videoDto.IsVote = userService.CheckUserVoteDay(userId, videoId, 2) + } + } + + responses.OkWithData(g, c) +} + +// AddVideoVote 视频投票 +func (r *Video) AddVideoVote(c *gin.Context) { + id := c.Param("video_id") + if id == "" { + responses.FailWithMessage("缺少参数", c) + return + } + + // 将 id 转换为 int64 类型 + videoId, err := strconv.ParseInt(id, 10, 64) + if err != nil { + responses.Fail(c) + return + } + + userId := c.GetInt64("UserId") + if userId == 0 { + c.JSON(http.StatusUnauthorized, gin.H{ + "message": "请登录后投票", + "code": consts.TokenError, + "data": "", + }) + return + } + + // 检测并发请求 + redisKey := "AddVideoVote" + fmt.Sprintf("%d", userId) + fmt.Sprintf("%d", videoId) + res, _ := global.Redis.Get(c, redisKey).Result() + if res != "" { + responses.FailWithMessage("请勿重复操作", c) + return + } + + defer func(redisKey string) { + global.Redis.Del(c, redisKey) + }(redisKey) + + // 添加缓存 + _, err = global.Redis.Set(c, redisKey, "1", (1)*time.Second).Result() + if err != nil { + responses.FailWithMessage("投票失败", c) + return + } + + // 获取数据 + videoDao := dao.VideoDao{} + video, err := videoDao.GetVideoById(videoId) + if err != nil || video == nil { + responses.FailWithMessage("非法数据", c) + return + } + + // 检测投票有效期 + systemTimeService := service.SystemTimeService{} + isValid := systemTimeService.CheckVoteValidStatus() + if isValid == false { + responses.FailWithMessage("投票已结束", c) + return + } + + // 检测用户今日是否投票 + userService := service.UserService{} + isVote := userService.CheckUserVoteDay(userId, videoId, 2) + if isVote == true { + responses.FailWithMessage("请勿重复投票", c) + return + } + + // 开始事务 + tx := global.Db.Begin() + defer func() { + if r := recover(); r != nil { + tx.Rollback() + utils.LogJsonErr("投票失败", r) + responses.FailWithMessage("投票失败", c) + return + } + }() + + // 增加投票数 + err = videoDao.Inc(tx, videoId, "vote_num", 1) + if err != nil { + tx.Rollback() + responses.FailWithMessage("投票失败", c) + return + } + + // 增加投票记录 + nowDay, _ := time.Parse("2006-01-02", time.Now().Format("2006-01-02")) + votedAt := model.LocalTime(nowDay) + videoVoteDay := &model.VideoVoteDay{ + VideoId: videoId, + UserId: userId, + VotedAt: &votedAt, + } + + videoVoteDayDao := dao.VideoVoteDayDao{} + videoVoteDay, err = videoVoteDayDao.AddVideoVoteDay(tx, videoVoteDay) + if err != nil { + tx.Rollback() + responses.FailWithMessage("投票失败", c) + return + } + + // 增加数据表-投票数量 + dataDao := dao.DataDao{} + err = dataDao.Inc(tx, 1, "vote_num", 1) + if err != nil { + tx.Rollback() + responses.FailWithMessage("投票失败", c) + return + } + + // 新增投票缓存 + result := userService.AddUserVoteDayCache(userId, videoId, 2) + if result == false { + tx.Rollback() + responses.FailWithMessage("投票失败", c) + return + } + + tx.Commit() + responses.Ok(c) +} diff --git a/api/dao/Article.go b/api/dao/Article.go index 64c372a..e32fe8e 100644 --- a/api/dao/Article.go +++ b/api/dao/Article.go @@ -4,6 +4,7 @@ import ( "gorm.io/gorm" "gorm.io/gorm/clause" "vote-api/api/model" + "vote-api/api/requests" "vote-api/global" ) @@ -124,3 +125,95 @@ func (r *ArticleDao) Dec(tx *gorm.DB, ArticleId int64, field string, numeral int } return nil } + +// GetArticlePageSearch 获取图文列表-分页 +func (r *ArticleDao) GetArticlePageSearch(req requests.GetArticlePage, page, pageSize int) (m []*model.Article, total int64, err error) { + var totalRecords int64 + + // 构建查询条件 + query := global.Db.Model(&model.Article{}) + + // 作者 + query = query.Preload("ArticleAuthor") + + // 作者关联医院 + query = query.Preload("ArticleAuthor.BaseHospital") + + // 文章状态(1:正常 2:禁用) + query = query.Where("article_status = ?", 1) + + // 搜索关键字 + if req.Keyword != "" { + keyword := "%" + req.Keyword + "%" // + + // 标题 + orQuery := global.Db.Model(&model.Article{}).Or("article_title LIKE ?", keyword) + + // 医院名称 + hospitalSubQuery := global.Db.Model(&model.BaseHospital{}). + Select("hospital_id"). + Where("hospital_name LIKE ?", keyword) + + articleAuthorSubQuery := global.Db.Model(&model.ArticleAuthor{}). + Select("article_id"). + Where(gorm.Expr("hospital_id IN (?)", hospitalSubQuery)) + + orQuery = orQuery.Or(gorm.Expr("article_id IN (?)", articleAuthorSubQuery)) + + // 作者姓名 + subQuery := global.Db.Model(&model.ArticleAuthor{}). + Select("article_id"). + Where("author_name LIKE ?", keyword) + + orQuery = orQuery.Or(gorm.Expr("article_id IN (?)", subQuery)) + + // 执行组建 + query = query.Where(orQuery) + } + + // 排序 + query = query.Order("created_at asc") + + // 查询总数量 + if err := query.Count(&totalRecords).Error; err != nil { + return nil, 0, err + } + + err = query.Scopes(model.Paginate(page, pageSize)).Find(&m).Error + if err != nil { + return nil, 0, err + } + return m, totalRecords, nil +} + +// GetArticleOrderList 获取列表-排序 +func (r *ArticleDao) GetArticleOrderList(maps interface{}, orderField string, limit int) (m []*model.Article, err error) { + err = global.Db.Where(maps).Preload(clause.Associations).Order(orderField).Limit(limit).Find(&m).Error + if err != nil { + return nil, err + } + return m, nil +} + +// GetArticleRank 获取某一条数据的排名 +func (r *ArticleDao) GetArticleRank(articleID int64) (int, error) { + var rank int + + // 定义子查询 + subQuery := global.Db.Model(&model.Article{}). + Select("article_id, vote_num, (@rank := @rank + 1) AS rank"). + Where("article_status = ?", 1). + Order("vote_num DESC"). + Joins(", (SELECT @rank := 0) AS r") + + // 将子查询作为命名子查询的一部分进行查询 + err := global.Db.Table("(?) AS sub", subQuery). + Where("sub.article_id = ?", articleID). + Pluck("sub.rank", &rank).Error + + if err != nil { + return 0, err + } + + return rank, nil +} diff --git a/api/dao/BaseArea.go b/api/dao/BaseArea.go new file mode 100644 index 0000000..36fe1d4 --- /dev/null +++ b/api/dao/BaseArea.go @@ -0,0 +1,72 @@ +package dao + +import ( + "gorm.io/gorm" + "vote-api/api/model" + "vote-api/global" +) + +type BaseAreaDao struct { +} + +// GetBaseAreaById 获取地区-地区id +func (r *BaseAreaDao) GetBaseAreaById(BaseAreaId int) (m *model.BaseArea, err error) { + err = global.Db.First(&m, BaseAreaId).Error + if err != nil { + return nil, err + } + return m, nil +} + +// DeleteBaseArea 删除地区 +func (r *BaseAreaDao) DeleteBaseArea(tx *gorm.DB, maps interface{}) error { + err := tx.Where(maps).Delete(&model.BaseArea{}).Error + if err != nil { + return err + } + return nil +} + +// EditBaseArea 修改地区 +func (r *BaseAreaDao) EditBaseArea(tx *gorm.DB, maps interface{}, data interface{}) error { + err := tx.Model(&model.BaseArea{}).Where(maps).Updates(data).Error + if err != nil { + return err + } + return nil +} + +// EditBaseAreaById 修改地区-医生id +func (r *BaseAreaDao) EditBaseAreaById(tx *gorm.DB, BaseAreaId int, data interface{}) error { + err := tx.Model(&model.BaseArea{}).Where("BaseArea_id = ?", BaseAreaId).Updates(data).Error + if err != nil { + return err + } + return nil +} + +// GetBaseAreaList 获取地区列表 +func (r *BaseAreaDao) GetBaseAreaList(maps interface{}) (m []*model.BaseArea, err error) { + err = global.Db.Where(maps).Find(&m).Error + if err != nil { + return nil, err + } + return m, nil +} + +// AddBaseArea 新增地区 +func (r *BaseAreaDao) AddBaseArea(tx *gorm.DB, model *model.BaseArea) (*model.BaseArea, error) { + if err := tx.Create(model).Error; err != nil { + return nil, err + } + return model, nil +} + +// AddBaseAreaByMap 新增地区-map +func (r *BaseAreaDao) AddBaseAreaByMap(tx *gorm.DB, data map[string]interface{}) (*model.BaseArea, error) { + userDoctorInfo := &model.BaseArea{} + if err := tx.Model(&model.BaseArea{}).Create(data).Error; err != nil { + return nil, err + } + return userDoctorInfo, nil +} diff --git a/api/dao/SystemTime.go b/api/dao/SystemTime.go new file mode 100644 index 0000000..e9cecdd --- /dev/null +++ b/api/dao/SystemTime.go @@ -0,0 +1,117 @@ +package dao + +import ( + "gorm.io/gorm" + "gorm.io/gorm/clause" + "vote-api/api/model" + "vote-api/global" +) + +type SystemTimeDao struct { +} + +// GetSystemTimeById 获取数据-id +func (r *SystemTimeDao) GetSystemTimeById(SystemTimeId int64) (m *model.SystemTime, err error) { + err = global.Db.First(&m, SystemTimeId).Error + if err != nil { + return nil, err + } + return m, nil +} + +// GetSystemTimePreloadById 获取数据-加载全部关联-id +func (r *SystemTimeDao) GetSystemTimePreloadById(SystemTimeId int64) (m *model.SystemTime, err error) { + err = global.Db.Preload(clause.Associations).First(&m, SystemTimeId).Error + if err != nil { + return nil, err + } + return m, nil +} + +// DeleteSystemTime 删除 +func (r *SystemTimeDao) DeleteSystemTime(tx *gorm.DB, maps interface{}) error { + err := tx.Where(maps).Delete(&model.SystemTime{}).Error + if err != nil { + return err + } + return nil +} + +// DeleteSystemTimeById 删除-id +func (r *SystemTimeDao) DeleteSystemTimeById(tx *gorm.DB, SystemTimeId int64) error { + if err := tx.Delete(&model.SystemTime{}, SystemTimeId).Error; err != nil { + return err + } + return nil +} + +// EditSystemTime 修改 +func (r *SystemTimeDao) EditSystemTime(tx *gorm.DB, maps interface{}, data interface{}) error { + err := tx.Model(&model.SystemTime{}).Where(maps).Updates(data).Error + if err != nil { + return err + } + return nil +} + +// EditSystemTimeById 修改-id +func (r *SystemTimeDao) EditSystemTimeById(tx *gorm.DB, SystemTimeId int64, data interface{}) error { + err := tx.Model(&model.SystemTime{}).Where("system_time_id = ?", SystemTimeId).Updates(data).Error + if err != nil { + return err + } + return nil +} + +// GetSystemTimeList 获取列表 +func (r *SystemTimeDao) GetSystemTimeList(maps interface{}) (m []*model.SystemTime, err error) { + err = global.Db.Where(maps).Find(&m).Error + if err != nil { + return nil, err + } + return m, nil +} + +// GetSystemTimeCount 获取数量 +func (r *SystemTimeDao) GetSystemTimeCount(maps interface{}) (total int64, err error) { + err = global.Db.Model(&model.SystemTime{}).Where(maps).Count(&total).Error + if err != nil { + return 0, err + } + return total, nil +} + +// GetSystemTimeListRand 获取列表-随机 +func (r *SystemTimeDao) GetSystemTimeListRand(maps interface{}, limit int) (m []*model.SystemTime, err error) { + err = global.Db.Where(maps).Order("rand()").Limit(limit).Find(&m).Error + if err != nil { + return nil, err + } + return m, nil +} + +// AddSystemTime 新增 +func (r *SystemTimeDao) AddSystemTime(tx *gorm.DB, model *model.SystemTime) (*model.SystemTime, error) { + if err := tx.Create(model).Error; err != nil { + return nil, err + } + return model, nil +} + +// Inc 自增 +func (r *SystemTimeDao) Inc(tx *gorm.DB, SystemTimeId int64, field string, numeral int) error { + err := tx.Model(&model.SystemTime{}).Where("SystemTime_id = ?", SystemTimeId).UpdateColumn(field, gorm.Expr(field+" + ?", numeral)).Error + if err != nil { + return err + } + return nil +} + +// Dec 自减 +func (r *SystemTimeDao) Dec(tx *gorm.DB, SystemTimeId int64, field string, numeral int) error { + err := tx.Model(&model.SystemTime{}).Where("SystemTime_id = ?", SystemTimeId).UpdateColumn(field, gorm.Expr(field+" - ?", numeral)).Error + if err != nil { + return err + } + return nil +} diff --git a/api/dao/Video.go b/api/dao/Video.go index a975c7e..3cb616d 100644 --- a/api/dao/Video.go +++ b/api/dao/Video.go @@ -4,6 +4,7 @@ import ( "gorm.io/gorm" "gorm.io/gorm/clause" "vote-api/api/model" + "vote-api/api/requests" "vote-api/global" ) @@ -124,3 +125,95 @@ func (r *VideoDao) Dec(tx *gorm.DB, VideoId int64, field string, numeral int) er } return nil } + +// GetVideoPageSearch 获取视频列表-分页 +func (r *VideoDao) GetVideoPageSearch(req requests.GetVideoPage, page, pageSize int) (m []*model.Video, total int64, err error) { + var totalRecords int64 + + // 构建查询条件 + query := global.Db.Model(&model.Video{}) + + // 作者 + query = query.Preload("VideoAuthor") + + // 作者关联医院 + query = query.Preload("VideoAuthor.BaseHospital") + + // 文章状态(1:正常 2:禁用) + query = query.Where("video_status = ?", 1) + + // 搜索关键字 + if req.Keyword != "" { + keyword := "%" + req.Keyword + "%" // + + // 标题 + orQuery := global.Db.Model(&model.Video{}).Or("video_title LIKE ?", keyword) + + // 医院名称 + hospitalSubQuery := global.Db.Model(&model.BaseHospital{}). + Select("hospital_id"). + Where("hospital_name LIKE ?", keyword) + + articleAuthorSubQuery := global.Db.Model(&model.VideoAuthor{}). + Select("video_id"). + Where(gorm.Expr("hospital_id IN (?)", hospitalSubQuery)) + + orQuery = orQuery.Or(gorm.Expr("video_id IN (?)", articleAuthorSubQuery)) + + // 作者姓名 + subQuery := global.Db.Model(&model.VideoAuthor{}). + Select("video_id"). + Where("author_name LIKE ?", keyword) + + orQuery = orQuery.Or(gorm.Expr("video_id IN (?)", subQuery)) + + // 执行组建 + query = query.Where(orQuery) + } + + // 排序 + query = query.Order("created_at asc") + + // 查询总数量 + if err := query.Count(&totalRecords).Error; err != nil { + return nil, 0, err + } + + err = query.Scopes(model.Paginate(page, pageSize)).Find(&m).Error + if err != nil { + return nil, 0, err + } + return m, totalRecords, nil +} + +// GetVideoOrderList 获取列表-排序 +func (r *VideoDao) GetVideoOrderList(maps interface{}, orderField string, limit int) (m []*model.Video, err error) { + err = global.Db.Where(maps).Preload(clause.Associations).Order(orderField).Limit(limit).Find(&m).Error + if err != nil { + return nil, err + } + return m, nil +} + +// GetVideoRank 获取某一条数据的排名 +func (r *VideoDao) GetVideoRank(videoID int64) (int, error) { + var rank int + + // 定义子查询 + subQuery := global.Db.Model(&model.Video{}). + Select("video_id, vote_num, (@rank := @rank + 1) AS rank"). + Where("video_status = ?", 1). + Order("vote_num DESC"). + Joins(", (SELECT @rank := 0) AS r") + + // 将子查询作为命名子查询的一部分进行查询 + err := global.Db.Table("(?) AS sub", subQuery). + Where("sub.video_id = ?", videoID). + Pluck("sub.rank", &rank).Error + + if err != nil { + return 0, err + } + + return rank, nil +} diff --git a/api/dto/Article.go b/api/dto/Article.go new file mode 100644 index 0000000..d745e75 --- /dev/null +++ b/api/dto/Article.go @@ -0,0 +1,84 @@ +package dto + +import ( + "fmt" + "vote-api/api/model" +) + +// ArticleDto 文章表 +type ArticleDto struct { + ArticleId string `json:"article_id"` // 主键id + ArticleTitle string `json:"article_title"` // 文章标题 + ArticleStatus int `json:"article_status"` // 文章状态(1:正常 2:禁用) + VoteNum uint `json:"vote_num"` // 总票数 + ArticleContent string `json:"article_content"` // 文章内容 + CreatedAt model.LocalTime `json:"created_at"` // 创建时间 + UpdatedAt model.LocalTime `json:"updated_at"` // 更新时间 + ArticleAuthor []*ArticleAuthorDto `json:"article_author"` // 作者 + Rank *int `json:"rank"` // 排名 + IsVote bool `json:"is_vote"` // 是否已投票(false:否 true:是) +} + +// GetArticleListDto 列表-分页 +func GetArticleListDto(m []*model.Article) []*ArticleDto { + // 处理返回值 + responses := make([]*ArticleDto, len(m)) + + if len(m) > 0 { + for i, v := range m { + response := &ArticleDto{ + ArticleId: fmt.Sprintf("%d", v.ArticleId), + ArticleTitle: v.ArticleTitle, + ArticleStatus: v.ArticleStatus, + VoteNum: v.VoteNum, + CreatedAt: v.CreatedAt, + UpdatedAt: v.UpdatedAt, + } + + // 加载数据-作者 + if v.ArticleAuthor != nil { + response = response.LoadArticleAuthor(v.ArticleAuthor) + } + + // 将转换后的结构体添加到新切片中 + responses[i] = response + } + } + + return responses +} + +// GetArticleDto 详情 +func GetArticleDto(m *model.Article) *ArticleDto { + return &ArticleDto{ + ArticleId: fmt.Sprintf("%d", m.ArticleId), + ArticleTitle: m.ArticleTitle, + ArticleStatus: m.ArticleStatus, + VoteNum: m.VoteNum, + ArticleContent: m.ArticleContent, + CreatedAt: m.CreatedAt, + UpdatedAt: m.UpdatedAt, + } +} + +// LoadArticleAuthor 加载数据-作者 +func (r *ArticleDto) LoadArticleAuthor(m []*model.ArticleAuthor) *ArticleDto { + if len(m) > 0 { + r.ArticleAuthor = GetArticleAuthorListDto(m) + } + return r +} + +// LoadRank 加载数据-排名 +func (r *ArticleDto) LoadRank(m int) *ArticleDto { + if m > 0 { + r.Rank = &m + } + return r +} + +// LoadVoteStatus 加载数据-投票状态 +func (r *ArticleDto) LoadVoteStatus(m bool) *ArticleDto { + r.IsVote = m + return r +} diff --git a/api/dto/ArticleAuthor.go b/api/dto/ArticleAuthor.go new file mode 100644 index 0000000..ea2605c --- /dev/null +++ b/api/dto/ArticleAuthor.go @@ -0,0 +1,42 @@ +package dto + +import ( + "vote-api/api/model" +) + +type ArticleAuthorDto struct { + AuthorName string `json:"author_name"` // 作者姓名 + HospitalName string `json:"hospital_name"` // 作者所属医院 +} + +// GetArticleAuthorListDto 列表-分页 +func GetArticleAuthorListDto(m []*model.ArticleAuthor) []*ArticleAuthorDto { + // 处理返回值 + responses := make([]*ArticleAuthorDto, len(m)) + + if len(m) > 0 { + for i, v := range m { + response := &ArticleAuthorDto{ + AuthorName: v.AuthorName, + } + + // 加载数据-医院属性 + if v.BaseHospital != nil { + response = response.LoadBaseHospitalAttr(v.BaseHospital) + } + + // 将转换后的结构体添加到新切片中 + responses[i] = response + } + } + + return responses +} + +// LoadBaseHospitalAttr 加载数据-医院属性 +func (r *ArticleAuthorDto) LoadBaseHospitalAttr(m *model.BaseHospital) *ArticleAuthorDto { + if m != nil { + r.HospitalName = m.HospitalName + } + return r +} diff --git a/api/dto/Video.go b/api/dto/Video.go new file mode 100644 index 0000000..6f9a0bc --- /dev/null +++ b/api/dto/Video.go @@ -0,0 +1,85 @@ +package dto + +import ( + "fmt" + "vote-api/api/model" +) + +// VideoDto 视频表 +type VideoDto struct { + VideoId string `json:"video_id"` // 主键id + VideoTitle string `json:"video_title"` // 视频标题 + VideoStatus int `json:"article_status"` // 视频状态(1:正常 2:禁用) + VoteNum uint `json:"vote_num"` // 总票数 + VideoUrl string `json:"video_url"` // 视频地址 + CreatedAt model.LocalTime `json:"created_at"` // 创建时间 + UpdatedAt model.LocalTime `json:"updated_at"` // 更新时间 + VideoAuthor []*VideoAuthorDto `json:"video_author"` // 作者 + Rank *int `json:"rank"` // 排名 + IsVote bool `json:"is_vote"` // 是否已投票(false:否 true:是) +} + +// GetVideoListDto 列表-分页 +func GetVideoListDto(m []*model.Video) []*VideoDto { + // 处理返回值 + responses := make([]*VideoDto, len(m)) + + if len(m) > 0 { + for i, v := range m { + response := &VideoDto{ + VideoId: fmt.Sprintf("%d", v.VideoId), + VideoTitle: v.VideoTitle, + VideoStatus: v.VideoStatus, + VoteNum: v.VoteNum, + VideoUrl: v.VideoUrl, + CreatedAt: v.CreatedAt, + UpdatedAt: v.UpdatedAt, + } + + // 加载数据-作者 + if v.VideoAuthor != nil { + response = response.LoadVideoAuthor(v.VideoAuthor) + } + + // 将转换后的结构体添加到新切片中 + responses[i] = response + } + } + + return responses +} + +// GetVideoDto 详情 +func GetVideoDto(m *model.Video) *VideoDto { + return &VideoDto{ + VideoId: fmt.Sprintf("%d", m.VideoId), + VideoTitle: m.VideoTitle, + VideoStatus: m.VideoStatus, + VoteNum: m.VoteNum, + VideoUrl: m.VideoUrl, + CreatedAt: m.CreatedAt, + UpdatedAt: m.UpdatedAt, + } +} + +// LoadVideoAuthor 加载数据-作者 +func (r *VideoDto) LoadVideoAuthor(m []*model.VideoAuthor) *VideoDto { + if len(m) > 0 { + r.VideoAuthor = GetVideoAuthorListDto(m) + } + return r +} + +// LoadRank 加载数据-排名 +func (r *VideoDto) LoadRank(m int) *VideoDto { + if m > 0 { + r.Rank = &m + } + return r +} + +// LoadVoteStatus 加载数据-投票状态 +func (r *VideoDto) LoadVoteStatus(m bool) *VideoDto { + r.IsVote = m + return r +} diff --git a/api/dto/VideoAuthor.go b/api/dto/VideoAuthor.go new file mode 100644 index 0000000..2625183 --- /dev/null +++ b/api/dto/VideoAuthor.go @@ -0,0 +1,42 @@ +package dto + +import ( + "vote-api/api/model" +) + +type VideoAuthorDto struct { + AuthorName string `json:"author_name"` // 作者姓名 + HospitalName string `json:"hospital_name"` // 作者所属医院 +} + +// GetVideoAuthorListDto 列表-分页 +func GetVideoAuthorListDto(m []*model.VideoAuthor) []*VideoAuthorDto { + // 处理返回值 + responses := make([]*VideoAuthorDto, len(m)) + + if len(m) > 0 { + for i, v := range m { + response := &VideoAuthorDto{ + AuthorName: v.AuthorName, + } + + // 加载数据-医院属性 + if v.BaseHospital != nil { + response = response.LoadBaseHospitalAttr(v.BaseHospital) + } + + // 将转换后的结构体添加到新切片中 + responses[i] = response + } + } + + return responses +} + +// LoadBaseHospitalAttr 加载数据-医院属性 +func (r *VideoAuthorDto) LoadBaseHospitalAttr(m *model.BaseHospital) *VideoAuthorDto { + if m != nil { + r.HospitalName = m.HospitalName + } + return r +} diff --git a/api/middlewares/auth.go b/api/middlewares/auth.go index 8a9760c..639a60f 100644 --- a/api/middlewares/auth.go +++ b/api/middlewares/auth.go @@ -11,29 +11,23 @@ func Auth() gin.HandlerFunc { return func(c *gin.Context) { // 获取用户id userId := c.GetInt64("UserId") - if userId == 0 { - responses.Fail(c) - c.Abort() - return - } + if userId != 0 { + // 获取用户数据 + userDao := dao.UserDao{} + user, err := userDao.GetUserById(userId) + if err != nil || user == nil { + responses.FailWithMessage("用户数据错误", c) + c.Abort() + return + } - // 获取用户数据 - userDao := dao.UserDao{} - user, err := userDao.GetUserById(userId) - if err != nil || user == nil { - responses.FailWithMessage("用户数据错误", c) - c.Abort() - return + if user.UserStatus == 2 { + responses.FailWithMessage("用户已禁用", c) + c.Abort() + return + } } - if user.UserStatus == 2 { - responses.FailWithMessage("用户已禁用", c) - c.Abort() - return - } - - c.Set("UserId", userId) // 用户id - c.Next() } } diff --git a/api/middlewares/jwt.go b/api/middlewares/jwt.go index 1b6e659..7336b8b 100644 --- a/api/middlewares/jwt.go +++ b/api/middlewares/jwt.go @@ -15,12 +15,7 @@ func Jwt() gin.HandlerFunc { return func(c *gin.Context) { authorization := c.Request.Header.Get("Authorization") if authorization == "" || !strings.HasPrefix(authorization, "Bearer ") { - c.JSON(http.StatusUnauthorized, gin.H{ - "message": "请求未授权", - "code": consts.TokenError, - "data": "", - }) - c.Abort() + c.Next() return } diff --git a/api/model/Article.go b/api/model/Article.go index 73c7efd..fa3763a 100644 --- a/api/model/Article.go +++ b/api/model/Article.go @@ -8,14 +8,14 @@ import ( // Article 文章表 type Article struct { - ArticleId int64 `gorm:"column:article_id;type:bigint(19);primary_key;comment:主键id" json:"article_id"` - ArticleTitle string `gorm:"column:article_title;type:varchar(200);comment:文章标题" json:"article_title"` - ArticleStatus int `gorm:"column:article_status;type:tinyint(1);default:1;comment:文章状态(1:正常 2:禁用)" json:"article_status"` - StartTime *LocalTime `gorm:"column:start_time;type:datetime;comment:开始投票时间" json:"start_time"` - EndTime *LocalTime `gorm:"column:end_time;type:datetime;comment:结束投票时间" json:"end_time"` - VoteNum uint `gorm:"column:vote_num;type:int(10) unsigned;default:0;comment:总票数" json:"vote_num"` - ArticleContent string `gorm:"column:article_content;type:text;comment:文章内容" json:"article_content"` + ArticleId int64 `gorm:"column:article_id;type:bigint(19);primary_key;comment:主键id" json:"article_id"` + ArticleTitle string `gorm:"column:article_title;type:varchar(200);comment:文章标题" json:"article_title"` + ArticleStatus int `gorm:"column:article_status;type:tinyint(1);default:1;comment:文章状态(1:正常 2:禁用)" json:"article_status"` + VoteNum uint `gorm:"column:vote_num;type:int(10) unsigned;default:0;comment:总票数" json:"vote_num"` + ArticleContent string `gorm:"column:article_content;type:text;comment:文章内容" json:"article_content"` Model + ArticleAuthor []*ArticleAuthor `gorm:"foreignKey:ArticleId;references:article_id" json:"article_author"` + Rank *int `gorm:"column:rank;type:tinyint(1) unsigned;comment:排名" json:"rank"` } func (m *Article) TableName() string { diff --git a/api/model/ArticleAuthor.go b/api/model/ArticleAuthor.go index 3fcf20d..90963cb 100644 --- a/api/model/ArticleAuthor.go +++ b/api/model/ArticleAuthor.go @@ -13,6 +13,7 @@ type ArticleAuthor struct { AuthorName string `gorm:"column:author_name;type:varchar(100);comment:作者姓名" json:"author_name"` HospitalId int64 `gorm:"column:hospital_id;type:bigint(19);comment:作者所属医院id" json:"hospital_id"` Model + BaseHospital *BaseHospital `gorm:"foreignKey:HospitalId;references:hospital_id" json:"base_hospital"` } func (m *ArticleAuthor) TableName() string { diff --git a/api/model/BaseArea.go b/api/model/BaseArea.go new file mode 100644 index 0000000..da68b22 --- /dev/null +++ b/api/model/BaseArea.go @@ -0,0 +1,26 @@ +package model + +import ( + "gorm.io/gorm" + "vote-api/global" +) + +// BaseArea 地区表 +type BaseArea struct { + AreaId int64 `gorm:"column:area_id;type:bigint(19);primary_key;comment:地区编号" json:"area_id"` + AreaName string `gorm:"column:area_name;type:varchar(255);comment:名称" json:"area_name"` + ParentId int64 `gorm:"column:parent_id;type:bigint(19);comment:上级编号" json:"parent_id"` + Zip string `gorm:"column:zip;type:varchar(10);comment:邮编" json:"zip"` + AreaType int `gorm:"column:area_type;type:tinyint(4);comment:类型(1:国家,2:省,3:市,4:区县)" json:"area_type"` +} + +func (m *BaseArea) TableName() string { + return "base_area" +} + +func (m *BaseArea) BeforeCreate(tx *gorm.DB) error { + if m.AreaId == 0 { + m.AreaId = global.Snowflake.Generate().Int64() + } + return nil +} diff --git a/api/model/BaseHospital.go b/api/model/BaseHospital.go new file mode 100644 index 0000000..4d4ec6f --- /dev/null +++ b/api/model/BaseHospital.go @@ -0,0 +1,46 @@ +package model + +import ( + "gorm.io/gorm" + "time" + "vote-api/global" +) + +// BaseHospital 医院表 +type BaseHospital struct { + HospitalId int64 `gorm:"column:hospital_id;type:bigint(19);primary_key;comment:主键id" json:"hospital_id"` + HospitalName string `gorm:"column:hospital_name;type:varchar(255);comment:医院名称" json:"hospital_name"` + HospitalStatus int `gorm:"column:hospital_status;type:tinyint(1);default:1;comment:状态(0:禁用 1:正常 2:删除)" json:"hospital_status"` + HospitalLevelName string `gorm:"column:hospital_level_name;type:varchar(20);comment:医院等级名称" json:"hospital_level_name"` + PostCode string `gorm:"column:post_code;type:varchar(50);comment:邮政编码" json:"post_code"` + TelePhone string `gorm:"column:tele_phone;type:varchar(20);comment:电话" json:"tele_phone"` + ProvinceId int `gorm:"column:province_id;type:int(11);comment:省份id" json:"province_id"` + Province string `gorm:"column:province;type:varchar(50);comment:省份" json:"province"` + CityId int `gorm:"column:city_id;type:int(11);comment:城市id" json:"city_id"` + City string `gorm:"column:city;type:varchar(50);comment:城市" json:"city"` + CountyId int `gorm:"column:county_id;type:int(11);comment:区县id" json:"county_id"` + County string `gorm:"column:county;type:varchar(50);comment:区县" json:"county"` + Address string `gorm:"column:address;type:varchar(255);comment:地址" json:"address"` + Lat string `gorm:"column:lat;type:varchar(255);comment:纬度" json:"lat"` + Lng string `gorm:"column:lng;type:varchar(255);comment:经度" json:"lng"` + Desc string `gorm:"column:desc;type:varchar(255);comment:简介" json:"desc"` + Model +} + +func (m *BaseHospital) TableName() string { + return "base_hospital" +} + +func (m *BaseHospital) BeforeCreate(tx *gorm.DB) error { + if m.HospitalId == 0 { + m.HospitalId = global.Snowflake.Generate().Int64() + } + + m.CreatedAt = LocalTime(time.Now()) + tx.Statement.SetColumn("CreatedAt", m.CreatedAt) + + m.UpdatedAt = LocalTime(time.Now()) + tx.Statement.SetColumn("UpdatedAt", m.UpdatedAt) + + return nil +} diff --git a/api/model/Data.go b/api/model/Data.go index c8905c5..46940f6 100644 --- a/api/model/Data.go +++ b/api/model/Data.go @@ -10,7 +10,7 @@ import ( type Data struct { DataId int64 `gorm:"column:data_id;type:bigint(19);primary_key;comment:主键id" json:"data_id"` ViewNum uint `gorm:"column:view_num;type:int(10) unsigned;default:0;comment:浏览数量" json:"view_num"` - VoteNum uint `gorm:"column:vote_num;type:int(10) unsigned;comment:投票数量" json:"vote_num"` + VoteNum uint `gorm:"column:vote_num;type:int(10) unsigned;default:0;comment:投票数量" json:"vote_num"` Model } diff --git a/api/model/SystemTime.go b/api/model/SystemTime.go new file mode 100644 index 0000000..3900bdc --- /dev/null +++ b/api/model/SystemTime.go @@ -0,0 +1,33 @@ +package model + +import ( + "gorm.io/gorm" + "time" + "vote-api/global" +) + +// SystemTime 配置-时间 +type SystemTime struct { + SystemTimeId int64 `gorm:"column:system_time_id;type:bigint(19);primary_key;comment:主键id" json:"system_time_id"` + StartTime *LocalTime `gorm:"column:start_time;type:datetime;comment:开始投票时间;NOT NULL" json:"start_time"` + EndTime *LocalTime `gorm:"column:end_time;type:datetime;comment:结束投票时间;NOT NULL" json:"end_time"` + Model +} + +func (m *SystemTime) TableName() string { + return "system_time" +} + +func (m *SystemTime) BeforeCreate(tx *gorm.DB) error { + if m.SystemTimeId == 0 { + m.SystemTimeId = global.Snowflake.Generate().Int64() + } + + m.CreatedAt = LocalTime(time.Now()) + tx.Statement.SetColumn("CreatedAt", m.CreatedAt) + + m.UpdatedAt = LocalTime(time.Now()) + tx.Statement.SetColumn("UpdatedAt", m.UpdatedAt) + + return nil +} diff --git a/api/model/Video.go b/api/model/Video.go index fff33c7..8f3b07b 100644 --- a/api/model/Video.go +++ b/api/model/Video.go @@ -8,14 +8,13 @@ import ( // Video 视频表 type Video struct { - VideoId int64 `gorm:"column:video_id;type:bigint(19);primary_key;comment:主键id" json:"video_id"` - VideoTitle string `gorm:"column:video_title;type:varchar(200);comment:视频标题" json:"video_title"` - VideoStatus int `gorm:"column:video_status;type:tinyint(1);default:1;comment:视频状态(1:正常 2:禁用)" json:"video_status"` - StartTime *LocalTime `gorm:"column:start_time;type:datetime;comment:开始投票时间" json:"start_time"` - EndTime *LocalTime `gorm:"column:end_time;type:datetime;comment:结束投票时间" json:"end_time"` - VoteNum uint `gorm:"column:vote_num;type:int(10) unsigned;default:0;comment:总票数" json:"vote_num"` - VideoUrl string `gorm:"column:video_url;type:varchar(255);comment:视频地址" json:"video_url"` + VideoId int64 `gorm:"column:video_id;type:bigint(19);primary_key;comment:主键id" json:"video_id"` + VideoTitle string `gorm:"column:video_title;type:varchar(200);comment:视频标题" json:"video_title"` + VideoStatus int `gorm:"column:video_status;type:tinyint(1);default:1;comment:视频状态(1:正常 2:禁用)" json:"video_status"` + VoteNum uint `gorm:"column:vote_num;type:int(10) unsigned;default:0;comment:总票数" json:"vote_num"` + VideoUrl string `gorm:"column:video_url;type:varchar(255);comment:视频地址" json:"video_url"` Model + VideoAuthor []*VideoAuthor `gorm:"foreignKey:VideoId;references:video_id" json:"video_author"` } func (m *Video) TableName() string { diff --git a/api/model/VideoAuthor.go b/api/model/VideoAuthor.go index bb06838..80f32af 100644 --- a/api/model/VideoAuthor.go +++ b/api/model/VideoAuthor.go @@ -13,6 +13,7 @@ type VideoAuthor struct { AuthorName string `gorm:"column:author_name;type:varchar(100);comment:作者姓名" json:"author_name"` HospitalId int64 `gorm:"column:hospital_id;type:bigint(19);comment:作者所属医院id" json:"hospital_id"` Model + BaseHospital *BaseHospital `gorm:"foreignKey:HospitalId;references:hospital_id" json:"base_hospital"` } func (m *VideoAuthor) TableName() string { diff --git a/api/requests/Article.go b/api/requests/Article.go new file mode 100644 index 0000000..07df0c0 --- /dev/null +++ b/api/requests/Article.go @@ -0,0 +1,12 @@ +package requests + +type ArticleRequest struct { + GetArticlePage // 获取图文列表-分页 +} + +// GetArticlePage 获取图文列表-分页 +type GetArticlePage struct { + Page int `json:"page" form:"page" label:"页码"` + PageSize int `json:"page_size" form:"page_size" label:"每页个数"` + Keyword string `json:"keyword" form:"keyword" label:"搜索关键字"` +} diff --git a/api/requests/Video.go b/api/requests/Video.go new file mode 100644 index 0000000..95470a0 --- /dev/null +++ b/api/requests/Video.go @@ -0,0 +1,12 @@ +package requests + +type VideoRequest struct { + GetVideoPage // 获取视频列表-分页 +} + +// GetVideoPage 获取视频列表-分页 +type GetVideoPage struct { + Page int `json:"page" form:"page" label:"页码"` + PageSize int `json:"page_size" form:"page_size" label:"每页个数"` + Keyword string `json:"keyword" form:"keyword" label:"搜索关键字"` +} diff --git a/api/router/router.go b/api/router/router.go index 5656d5b..e66305b 100644 --- a/api/router/router.go +++ b/api/router/router.go @@ -74,6 +74,9 @@ func Init() *gin.Engine { func publicRouter(r *gin.Engine, api controller.Api) { // 登录 r.POST("/login", api.Login.Login) + + // 增加浏览数量 + r.POST("/browse", api.Public.AddBrowse) } // adminRouter 公共路由-验证权限 @@ -88,4 +91,43 @@ func basicRouter(r *gin.Engine, api controller.Api) { // privateRouter 私有路由-验证权限 func privateRouter(r *gin.Engine, api controller.Api) { + // 图文 + articleGroup := r.Group("/article") + { + // 获取图文列表-分页 + articleGroup.GET("/page", api.Article.GetArticlePage) + + // 获取图文详情 + articleGroup.GET("/:article_id", api.Article.GetArticle) + } + + // 视频 + videoGroup := r.Group("/video") + { + // 获取视频列表-分页 + videoGroup.GET("/page", api.Video.GetVideoPage) + + // 获取视频详情 + videoGroup.GET("/:video_id", api.Video.GetVideo) + } + + // 排名 + rankGroup := r.Group("/rank") + { + // 获取文章排名列表 + rankGroup.GET("/article", api.Article.GetArticleRankList) + + // 获取视频排名列表 + rankGroup.GET("/video", api.Video.GetVideoRankList) + } + + // 投票 + voteGroup := r.Group("/vote") + { + // 文章投票 + voteGroup.POST("/article/:article_id", api.Article.AddArticleVote) + + // 视频投票 + voteGroup.POST("/video/:video_id", api.Video.AddVideoVote) + } } diff --git a/api/service/SystemTime.go b/api/service/SystemTime.go new file mode 100644 index 0000000..74840f8 --- /dev/null +++ b/api/service/SystemTime.go @@ -0,0 +1,49 @@ +package service + +import ( + "context" + "time" + "vote-api/api/dao" + "vote-api/global" +) + +type SystemTimeService struct { +} + +// CheckVoteValidStatus 检测投票有效期 +// bool true:未结束 false:已结束 +func (r *SystemTimeService) CheckVoteValidStatus() bool { + redisKey := "VoteSystemTime" + res, _ := global.Redis.Get(context.Background(), redisKey).Result() + if res == "" { + // 获取配置-时间 + systemTimeDao := dao.SystemTimeDao{} + systemTime, err := systemTimeDao.GetSystemTimeById(1) + if err != nil { + return false + } + + if systemTime.EndTime == nil { + return false + } + + // 结束时间 + endTime := time.Time(*systemTime.EndTime) + + // 当前时间 + now := time.Now() + + duration := endTime.Sub(now) + if duration < 0 { + return false + } + + // 添加缓存 + _, err = global.Redis.Set(context.Background(), redisKey, "1", duration).Result() + if err != nil { + return false + } + } + + return true +} diff --git a/api/service/User.go b/api/service/User.go new file mode 100644 index 0000000..3c84166 --- /dev/null +++ b/api/service/User.go @@ -0,0 +1,109 @@ +package service + +import ( + "context" + "fmt" + "time" + "vote-api/config" + "vote-api/global" +) + +type UserService struct { +} + +// AddUserVoteDayCache 新增用户每日投票记录缓存 +// t 1:文章 2:视频 +func (r *UserService) AddUserVoteDayCache(userId, id int64, t int) bool { + now := time.Now() + + redisKey := "UserVoteDay" + fmt.Sprintf("%d", userId) + fmt.Sprintf("%d", id) + now.Format("2006-01-02") + fmt.Sprintf("%d", t) + + // 缓存过期时间 + year, month, day := now.Date() + location := now.Location() + validTime := time.Date(year, month, day, 23, 59, 59, 0, location) + duration := validTime.Sub(now) + if duration < 0 { + return false + } + + if config.C.Env == "dev" { + duration = 60 * 5 * time.Second + } + + // 添加缓存 + _, err := global.Redis.Set(context.Background(), redisKey, "1", duration).Result() + if err != nil { + return false + } + + return true +} + +// CheckUserVoteDay 检测用户今日是否投票 +// t 1:文章 2:视频 +func (r *UserService) CheckUserVoteDay(userId, id int64, t int) bool { + if t != 1 && t != 2 { + return false + } + + now := time.Now() + redisKey := "UserVoteDay" + fmt.Sprintf("%d", userId) + fmt.Sprintf("%d", id) + now.Format("2006-01-02") + fmt.Sprintf("%d", t) + res, _ := global.Redis.Get(context.Background(), redisKey).Result() + if res == "" { + return false + } + + if res == "1" { + return true + } + + return false + //if res == "" { + // // 是否已投票(1:已投票 0:未投票) + // isVote := "0" + // if t == 1 { + // // 文章 + // maps := make(map[string]interface{}) + // maps["article_id"] = id + // maps["user_id"] = userId + // maps["voted_at"] = now.Format("2006-01-02") + // + // articleVoteDayDao := dao.ArticleVoteDayDao{} + // articleVoteDay, _ := articleVoteDayDao.GetArticleVoteDayById(id) + // if articleVoteDay != nil { + // isVote = "1" + // } + // } + // + // if t == 2 { + // // 视频 + // maps := make(map[string]interface{}) + // maps["article_id"] = id + // maps["user_id"] = userId + // maps["voted_at"] = now.Format("2006-01-02") + // + // videoVoteDayDao := dao.VideoVoteDayDao{} + // videoVoteDay, _ := videoVoteDayDao.GetVideoVoteDayById(id) + // if videoVoteDay != nil { + // isVote = "1" + // } + // } + // + // // 添加缓存 + // if isVote == "1" { + // result := r.AddUserVoteDayCache(userId, id, t) + // if result == false { + // return false + // } + // } + // + // res = isVote + //} + // + //if res == "0" { + // return false + //} else { + // return true + //} +}