go-common/app/interface/main/app-resource/service/sidebar/service.go
2019-04-22 18:49:16 +08:00

158 lines
3.7 KiB
Go

package sidebar
import (
"context"
"fmt"
"sync"
"time"
"go-common/app/interface/main/app-resource/conf"
bplusdao "go-common/app/interface/main/app-resource/dao/bplus"
resdao "go-common/app/interface/main/app-resource/dao/resource"
whitedao "go-common/app/interface/main/app-resource/dao/white"
"go-common/app/interface/main/app-resource/model"
"go-common/app/interface/main/app-resource/model/sidebar"
resource "go-common/app/service/main/resource/model"
"go-common/library/log"
"go-common/library/sync/errgroup"
)
const (
_initSidebarKey = "sidebar_%d_%d_%s"
_defaultLanguageHans = "hans"
_defaultLanguageHant = "hant"
)
type Service struct {
c *conf.Config
//dao
res *resdao.Dao
bdao *bplusdao.Dao
wdao *whitedao.Dao
// sidebar
tick time.Duration
sidebarCache map[string][]*sidebar.SideBar
limitsCahce map[int64][]*resource.SideBarLimit
//limit ids
limitIDs map[int64]struct{}
}
func New(c *conf.Config) (s *Service) {
s = &Service{
c: c,
res: resdao.New(c),
bdao: bplusdao.New(c),
wdao: whitedao.New(c),
tick: time.Duration(c.Tick),
sidebarCache: map[string][]*sidebar.SideBar{},
limitsCahce: map[int64][]*resource.SideBarLimit{},
//limit ids
limitIDs: map[int64]struct{}{},
}
s.loadSidebar()
s.loadLimit(c.SideBarLimit)
go s.tickproc()
return s
}
// SideBar
func (s *Service) SideBar(c context.Context, plat int8, build, module int, mid int64, language string) (ss []*sidebar.SideBar) {
if key := fmt.Sprintf(_initSidebarKey, plat, module, language); len(s.sidebarCache[fmt.Sprintf(key)]) == 0 || language == "" {
if model.IsOverseas(plat) {
key = fmt.Sprintf(_initSidebarKey, plat, module, _defaultLanguageHant)
if len(s.sidebarCache[fmt.Sprintf(key)]) > 0 {
language = _defaultLanguageHant
} else {
language = _defaultLanguageHans
}
} else {
language = _defaultLanguageHans
}
}
var (
key = fmt.Sprintf(_initSidebarKey, plat, module, language)
verify = map[int64]bool{}
mutex sync.Mutex
)
if sidebars, ok := s.sidebarCache[key]; ok {
g, _ := errgroup.WithContext(c)
for _, v := range sidebars {
var (
vid = v.ID
vurl = v.WhiteURL
)
if vurl != "" && mid > 0 {
g.Go(func() (err error) {
var ok bool
if ok, err = s.wdao.WhiteVerify(context.TODO(), mid, vurl); err != nil {
log.Error("s.wdao.WhiteVerify uri(%s) error(%v)", vurl, err)
ok = false
err = nil
}
mutex.Lock()
verify[vid] = ok
mutex.Unlock()
return
})
} else if vurl != "" && mid == 0 {
verify[vid] = false
}
}
g.Wait()
LOOP:
for _, v := range sidebars {
for _, l := range s.limitsCahce[v.ID] {
if model.InvalidBuild(build, l.Build, l.Condition) {
continue LOOP
}
}
if verifybool, ok := verify[v.ID]; ok && !verifybool {
continue LOOP
}
ss = append(ss, v)
}
}
return
}
// tickproc tick load cache.
func (s *Service) tickproc() {
for {
time.Sleep(s.tick)
s.loadSidebar()
}
}
func (s *Service) loadSidebar() {
sideBars, err := s.res.ResSideBar(context.TODO())
if err != nil || sideBars == nil {
log.Error("s.sideDao.SideBar error(%v) or nil", err)
return
}
var (
tmp = map[int64]struct{}{}
ss = map[string][]*sidebar.SideBar{}
)
for _, v := range sideBars.SideBar {
if _, ok := tmp[v.ID]; ok {
continue
}
tmp[v.ID] = struct{}{}
t := &sidebar.SideBar{}
t.Change(v)
key := fmt.Sprintf(_initSidebarKey, t.Plat, t.Module, t.Language)
ss[key] = append(ss[key], t)
}
s.sidebarCache = ss
s.limitsCahce = sideBars.Limit
log.Info("loadSidebar cache success")
}
func (s *Service) loadLimit(limit []int64) {
tmp := map[int64]struct{}{}
for _, l := range limit {
tmp[l] = struct{}{}
}
s.limitIDs = tmp
}