* feat(server): moved an http server object into a pkg subdir to be reusable for http, spots, and integrations * feat(web): isolated web module (server, router, middleware, utils) used in spots and new integrations * feat(web): removed possible panic * feat(web): split all handlers from http service into different packages for better management. * feat(web): changed router's method signature * feat(web): added missing handlers interface * feat(web): added health middleware to remove unnecessary checks * feat(web): customizable middleware set for web servers * feat(web): simplified the handler's structure * feat(web): created an unified server.Run method for all web services (http, spot, integrations) * feat(web): fixed a json size limit issue * feat(web): removed Keys and PG connection from router * feat(web): simplified integration's main file * feat(web): simplified spot's main file * feat(web): simplified http's main file (builder) * feat(web): refactored audit trail functionality * feat(web): added ee version of audit trail * feat(web): added ee version of conditions module * feat(web): moved ee version of some web session structs * feat(web): new format of web metrics * feat(web): added new web metrics to all handlers * feat(web): added justExpired feature to web ingest handler * feat(web): added small integrations improvements
88 lines
1.7 KiB
Go
88 lines
1.7 KiB
Go
package limiter
|
|
|
|
import (
|
|
"sync"
|
|
"time"
|
|
)
|
|
|
|
type RateLimiter struct {
|
|
rate int
|
|
burst int
|
|
tokens int
|
|
lastToken time.Time
|
|
lastUsed time.Time
|
|
mu sync.Mutex
|
|
}
|
|
|
|
func NewRateLimiter(rate int, burst int) *RateLimiter {
|
|
return &RateLimiter{
|
|
rate: rate,
|
|
burst: burst,
|
|
tokens: burst,
|
|
lastToken: time.Now(),
|
|
lastUsed: time.Now(),
|
|
}
|
|
}
|
|
|
|
func (rl *RateLimiter) Allow() bool {
|
|
rl.mu.Lock()
|
|
defer rl.mu.Unlock()
|
|
|
|
now := time.Now()
|
|
elapsed := now.Sub(rl.lastToken)
|
|
|
|
rl.tokens += int(elapsed.Seconds()) * rl.rate
|
|
if rl.tokens > rl.burst {
|
|
rl.tokens = rl.burst
|
|
}
|
|
|
|
rl.lastToken = now
|
|
rl.lastUsed = now
|
|
|
|
if rl.tokens > 0 {
|
|
rl.tokens--
|
|
return true
|
|
}
|
|
return false
|
|
}
|
|
|
|
type UserRateLimiter struct {
|
|
rateLimiters sync.Map
|
|
rate int
|
|
burst int
|
|
cleanupInterval time.Duration
|
|
maxIdleTime time.Duration
|
|
}
|
|
|
|
func NewUserRateLimiter(rate int, burst int, cleanupInterval time.Duration, maxIdleTime time.Duration) *UserRateLimiter {
|
|
url := &UserRateLimiter{
|
|
rate: rate,
|
|
burst: burst,
|
|
cleanupInterval: cleanupInterval,
|
|
maxIdleTime: maxIdleTime,
|
|
}
|
|
go url.cleanup()
|
|
return url
|
|
}
|
|
|
|
func (url *UserRateLimiter) GetRateLimiter(user uint64) *RateLimiter {
|
|
value, _ := url.rateLimiters.LoadOrStore(user, NewRateLimiter(url.rate, url.burst))
|
|
return value.(*RateLimiter)
|
|
}
|
|
|
|
func (url *UserRateLimiter) cleanup() {
|
|
for {
|
|
time.Sleep(url.cleanupInterval)
|
|
now := time.Now()
|
|
|
|
url.rateLimiters.Range(func(key, value interface{}) bool {
|
|
rl := value.(*RateLimiter)
|
|
rl.mu.Lock()
|
|
if now.Sub(rl.lastUsed) > url.maxIdleTime {
|
|
url.rateLimiters.Delete(key)
|
|
}
|
|
rl.mu.Unlock()
|
|
return true
|
|
})
|
|
}
|
|
}
|