| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677 |
- package core
- import (
- "time"
- )
- const (
- updateIntervalSec = 10
- cacheDurationSec = 120
- )
- type UserSet interface {
- AddUser(user User) error
- GetUser(timeHash []byte) (*ID, bool)
- }
- type TimedUserSet struct {
- validUserIds []ID
- userHashes map[string]int
- }
- type hashEntry struct {
- hash string
- timeSec int64
- }
- func NewTimedUserSet() UserSet {
- vuSet := new(TimedUserSet)
- vuSet.validUserIds = make([]ID, 0, 16)
- vuSet.userHashes = make(map[string]int)
- go vuSet.updateUserHash(time.Tick(updateIntervalSec * time.Second))
- return vuSet
- }
- func (us *TimedUserSet) updateUserHash(tick <-chan time.Time) {
- now := time.Now().UTC()
- lastSec := now.Unix() - cacheDurationSec
- hash2Remove := make(chan hashEntry, updateIntervalSec*2)
- lastSec2Remove := now.Unix() + cacheDurationSec
- for {
- now := <-tick
- nowSec := now.UTC().Unix()
- remove2Sec := nowSec - cacheDurationSec
- if remove2Sec > lastSec2Remove {
- for lastSec2Remove+1 < remove2Sec {
- entry := <-hash2Remove
- lastSec2Remove = entry.timeSec
- delete(us.userHashes, entry.hash)
- }
- }
- for i := lastSec + 1; i <= nowSec; i++ {
- for idx, id := range us.validUserIds {
- idHash := id.TimeHash(i)
- hash2Remove <- hashEntry{string(idHash), i}
- us.userHashes[string(idHash)] = idx
- }
- }
- }
- }
- func (us *TimedUserSet) AddUser(user User) error {
- id := user.Id
- us.validUserIds = append(us.validUserIds, id)
- return nil
- }
- func (us TimedUserSet) GetUser(userHash []byte) (*ID, bool) {
- idIndex, found := us.userHashes[string(userHash)]
- if found {
- return &us.validUserIds[idIndex], true
- }
- return nil, false
- }
|