go-common/app/admin/main/videoup-task/service/weight.go
2019-04-22 18:49:16 +08:00

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
}