userset.go 2.4 KB

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