Skip to content

Instantly share code, notes, and snippets.

@alexedwards
Created February 26, 2025 11:13
Show Gist options
  • Save alexedwards/660633e386448e2d550ba58cea79a7e0 to your computer and use it in GitHub Desktop.
Save alexedwards/660633e386448e2d550ba58cea79a7e0 to your computer and use it in GitHub Desktop.
func (app *application) rateLimit(next http.Handler) http.Handler {
if !app.config.limiter.enabled {
return next
}
type client struct {
limiter *rate.Limiter
lastSeen time.Time
}
var (
mu sync.Mutex
clients = make(map[string]*client)
)
go func() {
for {
time.Sleep(time.Minute)
mu.Lock()
for ip, client := range clients {
if time.Since(client.lastSeen) > 3*time.Minute {
delete(clients, ip)
}
}
mu.Unlock()
}
}()
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
ip := realip.FromRequest(r)
mu.Lock()
if _, found := clients[ip]; !found {
clients[ip] = &client{
limiter: rate.NewLimiter(rate.Limit(app.config.limiter.rps), app.config.limiter.burst),
}
}
clients[ip].lastSeen = time.Now()
if !clients[ip].limiter.Allow() {
mu.Unlock()
app.rateLimitExceededResponse(w, r)
return
}
mu.Unlock()
next.ServeHTTP(w, r)
})
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment