Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fix concurrent rw error (#2027) #2028

Merged
merged 1 commit into from
Aug 15, 2024
Merged
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
27 changes: 20 additions & 7 deletions tools/walletextension/ratelimiter/rate_limiter.go
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ type RequestInterval struct {
// RateLimitUser represents a user with a map of current requests.
type RateLimitUser struct {
CurrentRequests map[uuid.UUID]RequestInterval
mu sync.RWMutex
}

// zeroUUID is a zero UUID returned when no new request is added.
Expand All @@ -33,17 +34,21 @@ func (rl *RateLimiter) AddRequest(userID common.Address, interval RequestInterva
return zeroUUID
}
rl.mu.Lock()
defer rl.mu.Unlock()

user, exists := rl.users[userID]
if !exists {
user = &RateLimitUser{
CurrentRequests: make(map[uuid.UUID]RequestInterval),
mu: sync.RWMutex{},
}
rl.users[userID] = user
}
rl.mu.Unlock()

id := uuid.New()
user.mu.Lock()
user.CurrentRequests[id] = interval
user.mu.Unlock()

return id
}

Expand All @@ -54,18 +59,22 @@ func (rl *RateLimiter) SetRequestEnd(userID common.Address, id uuid.UUID) {
return
}

if user, userExists := rl.users[userID]; userExists {
rl.mu.RLock()
user, userExists := rl.users[userID]
rl.mu.RUnlock()

if userExists {
user.mu.Lock()
if request, requestExists := user.CurrentRequests[id]; requestExists {
rl.mu.Lock()
defer rl.mu.Unlock()
now := time.Now()
request.End = &now
user.CurrentRequests[id] = request
} else {
rl.logger.Info("Request with ID %s not found for user %s.", id, userID.Hex())
rl.logger.Info("Request with ID not found for user.", "id", id, "user", userID)
}
user.mu.Unlock()
} else {
rl.logger.Info("User %s not found while trying to update the request.", userID.Hex())
rl.logger.Info("User not found while trying to update the request.", "user", userID)
}
}

Expand All @@ -76,11 +85,13 @@ func (rl *RateLimiter) CountOpenRequests(userID common.Address) int {

var count int
if user, exists := rl.users[userID]; exists {
user.mu.RLock()
for _, interval := range user.CurrentRequests {
if interval.End == nil {
count++
}
}
user.mu.RUnlock()
}
return count
}
Expand Down Expand Up @@ -202,6 +213,7 @@ func (rl *RateLimiter) PruneRequests() {
// delete all the requests that have
cutoff := time.Now().Add(-rl.window)
for userID, user := range rl.users {
user.mu.Lock()
for id, interval := range user.CurrentRequests {
if interval.End != nil && interval.End.Before(cutoff) {
delete(user.CurrentRequests, id)
Expand All @@ -210,6 +222,7 @@ func (rl *RateLimiter) PruneRequests() {
if len(user.CurrentRequests) == 0 {
delete(rl.users, userID)
}
user.mu.Unlock()
}
timeTaken := time.Since(startTime)
if timeTaken > 1*time.Second {
Expand Down