userset.go 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130
  1. package core
  2. import (
  3. "container/heap"
  4. "time"
  5. v2hash "github.com/v2ray/v2ray-core/hash"
  6. "github.com/v2ray/v2ray-core/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. userHashes map[string]indexTimePair
  19. hash2Remove hashEntrySet
  20. }
  21. type indexTimePair struct {
  22. index int
  23. timeSec int64
  24. }
  25. type hashEntry struct {
  26. hash string
  27. timeSec int64
  28. }
  29. type hashEntrySet []*hashEntry
  30. func (set hashEntrySet) Len() int {
  31. return len(set)
  32. }
  33. func (set hashEntrySet) Less(i, j int) bool {
  34. return set[i].timeSec < set[j].timeSec
  35. }
  36. func (set hashEntrySet) Swap(i, j int) {
  37. tmp := set[i]
  38. set[i] = set[j]
  39. set[j] = tmp
  40. }
  41. func (set *hashEntrySet) Push(value interface{}) {
  42. entry := value.(*hashEntry)
  43. *set = append(*set, entry)
  44. }
  45. func (set *hashEntrySet) Pop() interface{} {
  46. old := *set
  47. n := len(old)
  48. v := old[n-1]
  49. *set = old[:n-1]
  50. return v
  51. }
  52. func NewTimedUserSet() UserSet {
  53. vuSet := new(TimedUserSet)
  54. vuSet.validUserIds = make([]ID, 0, 16)
  55. vuSet.userHashes = make(map[string]indexTimePair)
  56. vuSet.hash2Remove = make(hashEntrySet, 0, cacheDurationSec*10)
  57. go vuSet.updateUserHash(time.Tick(updateIntervalSec * time.Second))
  58. return vuSet
  59. }
  60. func (us *TimedUserSet) generateNewHashes(lastSec, nowSec int64, idx int, id ID) {
  61. idHash := v2hash.NewTimeHash(v2hash.HMACHash{})
  62. for lastSec < nowSec+cacheDurationSec {
  63. idHash := idHash.Hash(id.Bytes, lastSec)
  64. log.Debug("Valid User Hash: %v", idHash)
  65. heap.Push(&us.hash2Remove, &hashEntry{string(idHash), lastSec})
  66. us.userHashes[string(idHash)] = indexTimePair{idx, lastSec}
  67. lastSec++
  68. }
  69. }
  70. func (us *TimedUserSet) updateUserHash(tick <-chan time.Time) {
  71. now := time.Now().UTC()
  72. lastSec := now.Unix()
  73. lastSec2Remove := now.Unix()
  74. for {
  75. now := <-tick
  76. nowSec := now.UTC().Unix()
  77. remove2Sec := nowSec - cacheDurationSec
  78. if remove2Sec > lastSec2Remove {
  79. for lastSec2Remove+1 < remove2Sec {
  80. front := heap.Pop(&us.hash2Remove)
  81. entry := front.(*hashEntry)
  82. lastSec2Remove = entry.timeSec
  83. delete(us.userHashes, entry.hash)
  84. }
  85. }
  86. for idx, id := range us.validUserIds {
  87. us.generateNewHashes(lastSec, nowSec, idx, id)
  88. }
  89. }
  90. }
  91. func (us *TimedUserSet) AddUser(user User) error {
  92. id := user.Id
  93. idx := len(us.validUserIds)
  94. us.validUserIds = append(us.validUserIds, id)
  95. nowSec := time.Now().UTC().Unix()
  96. lastSec := nowSec - cacheDurationSec
  97. us.generateNewHashes(lastSec, nowSec, idx, id)
  98. return nil
  99. }
  100. func (us TimedUserSet) GetUser(userHash []byte) (*ID, int64, bool) {
  101. pair, found := us.userHashes[string(userHash)]
  102. if found {
  103. return &us.validUserIds[pair.index], pair.timeSec, true
  104. }
  105. return nil, 0, false
  106. }