userset.go 2.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101
  1. package user
  2. import (
  3. "sync"
  4. "time"
  5. "github.com/v2ray/v2ray-core/common/collect"
  6. "github.com/v2ray/v2ray-core/common/log"
  7. )
  8. const (
  9. updateIntervalSec = 10
  10. cacheDurationSec = 120
  11. )
  12. type UserSet interface {
  13. AddUser(user User) error
  14. GetUser(timeHash []byte) (*ID, int64, bool)
  15. }
  16. type TimedUserSet struct {
  17. validUserIds []ID
  18. userHash map[string]indexTimePair
  19. userHashDeleteQueue *collect.TimedQueue
  20. access sync.RWMutex
  21. }
  22. type indexTimePair struct {
  23. index int
  24. timeSec int64
  25. }
  26. func NewTimedUserSet() UserSet {
  27. tus := &TimedUserSet{
  28. validUserIds: make([]ID, 0, 16),
  29. userHash: make(map[string]indexTimePair, 512),
  30. userHashDeleteQueue: collect.NewTimedQueue(updateIntervalSec),
  31. access: sync.RWMutex{},
  32. }
  33. go tus.updateUserHash(time.Tick(updateIntervalSec * time.Second))
  34. go tus.removeEntries(tus.userHashDeleteQueue.RemovedEntries())
  35. return tus
  36. }
  37. func (us *TimedUserSet) removeEntries(entries <-chan interface{}) {
  38. for {
  39. entry := <-entries
  40. us.access.Lock()
  41. delete(us.userHash, entry.(string))
  42. us.access.Unlock()
  43. }
  44. }
  45. func (us *TimedUserSet) generateNewHashes(lastSec, nowSec int64, idx int, id ID) {
  46. idHash := NewTimeHash(HMACHash{})
  47. for lastSec < nowSec+cacheDurationSec {
  48. idHash := idHash.Hash(id.Bytes[:], lastSec)
  49. log.Debug("Valid User Hash: %v", idHash)
  50. us.access.Lock()
  51. us.userHash[string(idHash)] = indexTimePair{idx, lastSec}
  52. us.access.Unlock()
  53. us.userHashDeleteQueue.Add(string(idHash), lastSec+2*cacheDurationSec)
  54. lastSec++
  55. }
  56. }
  57. func (us *TimedUserSet) updateUserHash(tick <-chan time.Time) {
  58. now := time.Now().UTC()
  59. lastSec := now.Unix()
  60. for {
  61. now := <-tick
  62. nowSec := now.UTC().Unix()
  63. for idx, id := range us.validUserIds {
  64. us.generateNewHashes(lastSec, nowSec, idx, id)
  65. }
  66. lastSec = nowSec
  67. }
  68. }
  69. func (us *TimedUserSet) AddUser(user User) error {
  70. id := user.Id
  71. idx := len(us.validUserIds)
  72. us.validUserIds = append(us.validUserIds, id)
  73. nowSec := time.Now().UTC().Unix()
  74. lastSec := nowSec - cacheDurationSec
  75. us.generateNewHashes(lastSec, nowSec, idx, id)
  76. return nil
  77. }
  78. func (us TimedUserSet) GetUser(userHash []byte) (*ID, int64, bool) {
  79. defer us.access.RUnlock()
  80. us.access.RLock()
  81. pair, found := us.userHash[string(userHash)]
  82. if found {
  83. return &us.validUserIds[pair.index], pair.timeSec, true
  84. }
  85. return nil, 0, false
  86. }