throttle.go 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132
  1. package ircclient
  2. import (
  3. "log"
  4. "time"
  5. )
  6. type FloodTrack struct {
  7. Pos int
  8. Track []time.Time
  9. }
  10. func (F *FloodTrack) Expire(timeout int) {
  11. var idx int
  12. ReCheck:
  13. for idx = 0; idx < F.Pos; idx++ {
  14. // log.Println(idx, time.Since(F.Track[idx]).Seconds())
  15. if time.Since(F.Track[idx]).Seconds() > float64(timeout) {
  16. // Remove this from the list
  17. F.Pos--
  18. for pos := idx; pos < F.Pos; pos++ {
  19. F.Track[pos] = F.Track[pos+1]
  20. }
  21. goto ReCheck
  22. }
  23. }
  24. }
  25. func (F *FloodTrack) Save() {
  26. F.Track[F.Pos] = time.Now()
  27. F.Pos++
  28. }
  29. type ThrottleBuffer struct {
  30. buffer map[string][]string
  31. targets []string
  32. last int
  33. Life_sucks bool
  34. }
  35. func (T *ThrottleBuffer) init() {
  36. T.buffer = make(map[string][]string, 0)
  37. T.targets = make([]string, 0)
  38. T.last = 0
  39. }
  40. /*
  41. Pop next available from buffer. If empty, remove from map and targets
  42. list. Adjust last if needed.
  43. */
  44. func (T *ThrottleBuffer) pop() string {
  45. // Get next target
  46. var t string = T.targets[T.last]
  47. T.last++
  48. if T.last == len(T.targets) {
  49. // We're past the end, start over at the beginning.
  50. T.last = 0
  51. }
  52. var msg string = T.buffer[t][0]
  53. if len(T.buffer[t]) == 1 {
  54. // This is the last entry for this target.
  55. delete(T.buffer, t)
  56. if len(T.targets) == 1 {
  57. // This is the last entry
  58. T.targets = make([]string, 0)
  59. T.Life_sucks = false
  60. log.Println("Flood control off.")
  61. } else {
  62. // Remove t from targets
  63. for x := 0; x < len(T.targets); x++ {
  64. if T.targets[x] == t {
  65. T.targets = append(T.targets[:x], T.targets[x+1:]...)
  66. if x <= T.last {
  67. T.last--
  68. }
  69. break
  70. }
  71. }
  72. }
  73. } else {
  74. // Delete the first entry for t
  75. T.buffer[t] = append(T.buffer[t][:0], T.buffer[t][1:]...)
  76. }
  77. return msg
  78. }
  79. func (T *ThrottleBuffer) push(To string, Output string) {
  80. if len(T.targets) == 0 {
  81. T.Life_sucks = true
  82. T.last = 0
  83. log.Println("Flood control enabled.")
  84. }
  85. _, has := T.buffer[To]
  86. if !has {
  87. T.buffer[To] = make([]string, 0)
  88. T.targets = append(T.targets, To)
  89. }
  90. T.buffer[To] = append(T.buffer[To], Output)
  91. }
  92. func (T *ThrottleBuffer) delete(To string) {
  93. _, has := T.buffer[To]
  94. if has {
  95. // Yes, the buffer has message(s) for To
  96. delete(T.buffer, To)
  97. if len(T.targets) == 1 {
  98. // Last entry
  99. T.targets = make([]string, 0)
  100. T.Life_sucks = false
  101. log.Println("Flood control off.")
  102. } else {
  103. // Remove To from targets
  104. for x := 0; x < len(T.targets); x++ {
  105. if T.targets[x] == To {
  106. T.targets = append(T.targets[:x], T.targets[x+1:]...)
  107. if x <= T.last {
  108. T.last--
  109. }
  110. break
  111. }
  112. }
  113. }
  114. }
  115. }