322 lines
8.4 KiB
Go
322 lines
8.4 KiB
Go
|
package service
|
||
|
|
||
|
import (
|
||
|
"context"
|
||
|
"encoding/json"
|
||
|
"fmt"
|
||
|
"time"
|
||
|
|
||
|
"go-common/app/admin/main/videoup-task/model"
|
||
|
account "go-common/app/service/main/account/api"
|
||
|
"go-common/library/log"
|
||
|
"go-common/library/sync/errgroup"
|
||
|
"go-common/library/xstr"
|
||
|
)
|
||
|
|
||
|
// weightConf 所有激活的配置项
|
||
|
func (s *Service) weightConf(c context.Context) (wconfs map[int8]map[int64]*model.WCItem, err error) {
|
||
|
items, err := s.dao.WeightConf(c)
|
||
|
if err != nil {
|
||
|
log.Error("WeightConf error(%v)", err)
|
||
|
return
|
||
|
}
|
||
|
wconfs = map[int8]map[int64]*model.WCItem{}
|
||
|
for _, item := range items {
|
||
|
conf, ok := wconfs[item.Radio]
|
||
|
if !ok {
|
||
|
conf = make(map[int64]*model.WCItem)
|
||
|
}
|
||
|
conf[item.CID] = item
|
||
|
wconfs[item.Radio] = conf
|
||
|
}
|
||
|
return
|
||
|
}
|
||
|
|
||
|
// 同一类型的同id配置只能有一个
|
||
|
func (s *Service) checkConflict(c context.Context, mcases map[int64]*model.WCItem) (err error) {
|
||
|
wcf, err := s.weightConf(c)
|
||
|
if err != nil {
|
||
|
log.Error("s.weightConf error(%v)", err)
|
||
|
return
|
||
|
}
|
||
|
s.twConCache = wcf
|
||
|
for _, item := range mcases {
|
||
|
if cfgs, ok := s.twConCache[item.Radio]; ok {
|
||
|
if cfg, ok := cfgs[item.CID]; ok {
|
||
|
return fmt.Errorf("config(%d) conflict with (id=%d,desc=%s)", item.CID, cfg.ID, cfg.Desc)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return
|
||
|
}
|
||
|
|
||
|
// AddWeightConf 配置权重
|
||
|
func (s *Service) AddWeightConf(c context.Context, cfg *model.WeightConf, uid int64, uname string) (err error) {
|
||
|
//0. 一级分区转化为二级分区
|
||
|
if cfg.Radio == model.WConfType {
|
||
|
cfg.Ids = s.tarnsType(c, cfg.Ids)
|
||
|
}
|
||
|
|
||
|
//1. 解析表单提交的配置
|
||
|
mcases, istaskid, err := model.ParseWeightConf(cfg, uid, uname)
|
||
|
if err != nil {
|
||
|
log.Error("model.ParseWeightConf(%v, %v, %v) error(%v)", cfg, uname, err)
|
||
|
return
|
||
|
}
|
||
|
//2. 检查是否互相冲突
|
||
|
if err = s.checkConflict(c, mcases); err != nil {
|
||
|
log.Error("s.checkConflict error(%v)", err)
|
||
|
return
|
||
|
}
|
||
|
|
||
|
//3. 更新任务的权重配置
|
||
|
if istaskid {
|
||
|
if err = s.setWeightConf(c, cfg.Ids, mcases); err != nil {
|
||
|
log.Error("s.setWeightConf error(%v)", err)
|
||
|
return
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//4. 将配置存储到数据库表
|
||
|
if err = s.dao.InWeightConf(c, mcases); err != nil {
|
||
|
log.Error("s.dao.InWeightConf(%v) error(%v)", mcases, err)
|
||
|
return
|
||
|
}
|
||
|
|
||
|
return
|
||
|
}
|
||
|
|
||
|
// DelWeightConf 删除配置项
|
||
|
func (s *Service) DelWeightConf(c context.Context, id int64) (err error) {
|
||
|
if _, err = s.dao.DelWeightConf(c, id); err != nil {
|
||
|
log.Error("s.DelWeightConf(%d) error(%v)", id, err)
|
||
|
}
|
||
|
return
|
||
|
}
|
||
|
|
||
|
// ListWeightConf 列出配置
|
||
|
func (s *Service) ListWeightConf(c context.Context, v *model.Confs) (cfg []*model.WCItem, err error) {
|
||
|
if cfg, err = s.dao.ListWeightConf(c, v); err != nil {
|
||
|
log.Error("s.ListWeightConf(%+v) error(%v)", v, err)
|
||
|
return
|
||
|
}
|
||
|
// 根据taskid补充filename和title
|
||
|
switch v.Radio {
|
||
|
case model.WConfMid: // 补充昵称和粉丝数
|
||
|
s.singleIDtoName(c, cfg, func(c context.Context, cid int64) (res []interface{}, err error) {
|
||
|
stat, err := s.profile(c, cid)
|
||
|
if err != nil {
|
||
|
log.Error("s.profile(%d) error(%v)", cid, err)
|
||
|
return
|
||
|
}
|
||
|
res = []interface{}{stat.Profile.Name, stat.Follower}
|
||
|
return
|
||
|
}, false, "CID", "Creator", "Fans")
|
||
|
case model.WConfTaskID: //补充title和filename
|
||
|
s.mulIDtoName(c, cfg, s.dao.LWConfigHelp, "CID", "FileName", "Title", "Vid")
|
||
|
case model.WConfType: //补充分区名称
|
||
|
s.singleIDtoName(c, cfg, func(c context.Context, cid int64) (res []interface{}, err error) {
|
||
|
if item, ok := s.typeCache[int16(cid)]; ok {
|
||
|
res = []interface{}{item.Name}
|
||
|
}
|
||
|
return
|
||
|
}, false, "CID", "TypeName")
|
||
|
case model.WConfUpFrom: //补充投稿来源
|
||
|
s.singleIDtoName(c, cfg, func(c context.Context, cid int64) (res []interface{}, err error) {
|
||
|
res = []interface{}{model.UpFrom(int8(cid))}
|
||
|
return
|
||
|
}, false, "CID", "UpFrom")
|
||
|
default:
|
||
|
}
|
||
|
|
||
|
return
|
||
|
}
|
||
|
|
||
|
// ListWeightLogs 权重变更日志
|
||
|
func (s *Service) ListWeightLogs(c context.Context, taskid int64, page int) (cfg []*model.TaskWeightLog, items int64, err error) {
|
||
|
cfg, err = s.dao.WeightLog(c, taskid)
|
||
|
if err != nil {
|
||
|
log.Info("s.ListWeightLogs(%d) error(%v)", taskid, err)
|
||
|
return
|
||
|
}
|
||
|
cfg, items, err = s.weightLogHelp(c, taskid, cfg, page)
|
||
|
return
|
||
|
}
|
||
|
|
||
|
// MaxWeight 当前的权重最大值, 供用户配置做参考
|
||
|
func (s *Service) MaxWeight(c context.Context) (max int64, err error) {
|
||
|
return s.dao.GetMaxWeight(c)
|
||
|
}
|
||
|
|
||
|
// 补充查询日志需要的信息
|
||
|
func (s *Service) weightLogHelp(c context.Context, taskid int64, twl []*model.TaskWeightLog, page int) (retwl []*model.TaskWeightLog, items int64, err error) {
|
||
|
var (
|
||
|
mid int64
|
||
|
ps *account.ProfileStatReply
|
||
|
name string
|
||
|
fans int
|
||
|
upspecial []int8
|
||
|
)
|
||
|
|
||
|
task, err := s.dao.TaskByID(c, taskid)
|
||
|
if task == nil {
|
||
|
log.Error("s.dao.TaskByID(%d) err(%v)", taskid, err)
|
||
|
return
|
||
|
}
|
||
|
// 1.获取用户信息
|
||
|
items = int64(len(twl))
|
||
|
if items == 0 {
|
||
|
return
|
||
|
}
|
||
|
mid = twl[items-1].Mid
|
||
|
if mid == 0 {
|
||
|
log.Error("weightLogHelp taskid=%d miss mid", taskid)
|
||
|
goto EMPTYMID
|
||
|
}
|
||
|
|
||
|
ps, err = s.profile(c, mid)
|
||
|
if err != nil || ps == nil {
|
||
|
log.Error("can't get Account.Card(%d) err(%v)", mid, err)
|
||
|
err = nil
|
||
|
} else {
|
||
|
name = ps.Profile.Name
|
||
|
fans = int(ps.Follower)
|
||
|
}
|
||
|
|
||
|
upspecial = s.getSpecial(mid)
|
||
|
|
||
|
EMPTYMID:
|
||
|
retwl = []*model.TaskWeightLog{}
|
||
|
// 反转日志顺序
|
||
|
var count int
|
||
|
for i := len(twl) - 1 - (page-1)*20; i >= 0 && count <= 20; func() { i--; count++ }() {
|
||
|
twl[i].Creator = name
|
||
|
twl[i].UpSpecial = upspecial
|
||
|
twl[i].Fans = int64(fans)
|
||
|
twl[i].Wait = twl[i].Uptime.TimeValue().Sub(task.CTime.TimeValue()).Seconds() * 1000.0
|
||
|
if !task.PTime.TimeValue().IsZero() {
|
||
|
twl[i].Ptime = string(task.PTime)
|
||
|
}
|
||
|
|
||
|
if len(twl[i].CfItems) > 0 {
|
||
|
for _, item := range twl[i].CfItems {
|
||
|
var desc = item.Desc
|
||
|
var v = new(model.WCItem)
|
||
|
if err = json.Unmarshal([]byte(item.Desc), v); err == nil { //兼容新旧日志格式
|
||
|
desc = v.Desc
|
||
|
}
|
||
|
err = nil
|
||
|
twl[i].Desc += fmt.Sprintf("%s:%s; ", model.CfWeightDesc(item.Radio), desc)
|
||
|
}
|
||
|
}
|
||
|
retwl = append(retwl, twl[i])
|
||
|
}
|
||
|
return
|
||
|
}
|
||
|
|
||
|
func (s *Service) getSpecial(mid int64) []int8 {
|
||
|
upspecial := []int8{}
|
||
|
for k, v := range s.upperCache {
|
||
|
if _, ok := v[mid]; ok {
|
||
|
upspecial = append(upspecial, k)
|
||
|
}
|
||
|
}
|
||
|
return upspecial
|
||
|
}
|
||
|
|
||
|
func (s *Service) setWeightConf(c context.Context, ids string, mcases map[int64]*model.WCItem) (err error) {
|
||
|
var mitems map[int64]*model.TaskPriority
|
||
|
arrid, _ := xstr.SplitInts(ids)
|
||
|
// 1. 获取旧的配置
|
||
|
if mitems, err = s.getTWCache(c, arrid); err != nil {
|
||
|
log.Error("s.getTWCache(%v) error(%v)", arrid, err)
|
||
|
return
|
||
|
}
|
||
|
if len(mitems) == 0 {
|
||
|
return fmt.Errorf("没有找到任务(%v)", arrid)
|
||
|
}
|
||
|
// 2. 将新加配置加入
|
||
|
for _, item := range mitems {
|
||
|
if one, ok := mcases[item.TaskID]; ok && one != nil {
|
||
|
one.Mtime = model.NewFormatTime(time.Now())
|
||
|
item.CfItems = append(item.CfItems, one)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// 3. 更新配置到redis和数据库
|
||
|
var wg errgroup.Group
|
||
|
wg.Go(func() (err error) {
|
||
|
return s.dao.SetWeightRedis(c, mitems)
|
||
|
})
|
||
|
for _, item := range mitems {
|
||
|
var descb []byte
|
||
|
v := item
|
||
|
wg.Go(func() (err error) {
|
||
|
if descb, err = json.Marshal(v.CfItems); err != nil {
|
||
|
return
|
||
|
}
|
||
|
_, err = s.dao.UpCwAfterAdd(c, v.TaskID, string(descb))
|
||
|
return
|
||
|
})
|
||
|
}
|
||
|
err = wg.Wait()
|
||
|
|
||
|
return
|
||
|
}
|
||
|
|
||
|
// ShowWeightVC 展示权重配置值
|
||
|
func (s *Service) ShowWeightVC(c context.Context) (wvc *model.WeightVC, err error) {
|
||
|
if wvc, err = s.dao.WeightVC(c); err != nil {
|
||
|
log.Error("s.dao.WeightVC error(%v)", err)
|
||
|
return
|
||
|
}
|
||
|
|
||
|
if wvc == nil {
|
||
|
wvc = model.WLVConf
|
||
|
s.dao.InWeightVC(c, wvc, "默认权重配置")
|
||
|
}
|
||
|
return
|
||
|
}
|
||
|
|
||
|
// SetWeightVC 设置权重配置
|
||
|
func (s *Service) SetWeightVC(c context.Context, wvc *model.WeightVC) (err error) {
|
||
|
_, err = s.dao.SetWeightVC(c, wvc, "设置权重配置")
|
||
|
return
|
||
|
}
|
||
|
|
||
|
// 处理一级分区转二级分区
|
||
|
func (s *Service) tarnsType(c context.Context, ids string) (res string) {
|
||
|
var (
|
||
|
mapNoNeed = make(map[int16]struct{}) // 不需要的一级分区
|
||
|
mapNeed = make(map[int16]struct{}) // 需要的一级分区
|
||
|
idres = []int64{}
|
||
|
)
|
||
|
|
||
|
arrid, _ := xstr.SplitInts(ids)
|
||
|
for _, id := range arrid {
|
||
|
if mod, ok := s.typeCache[int16(id)]; ok {
|
||
|
if mod.PID == 0 {
|
||
|
mapNeed[int16(id)] = struct{}{}
|
||
|
} else {
|
||
|
mapNoNeed[mod.PID] = struct{}{}
|
||
|
idres = append(idres, id)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
for k := range mapNoNeed {
|
||
|
delete(mapNeed, k)
|
||
|
}
|
||
|
for k := range mapNeed {
|
||
|
idres = append(idres, s.typeCache2[k]...)
|
||
|
}
|
||
|
|
||
|
return xstr.JoinInts(idres)
|
||
|
}
|
||
|
|
||
|
func (s *Service) getTWCache(c context.Context, ids []int64) (mcases map[int64]*model.TaskPriority, err error) {
|
||
|
if mcases, err = s.dao.GetWeightRedis(c, ids); len(mcases) != len(ids) {
|
||
|
mcases, err = s.dao.GetWeightDB(c, ids)
|
||
|
}
|
||
|
return
|
||
|
}
|