forked from nccapo/rate-limiter
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathlimiter_test.go
More file actions
124 lines (105 loc) · 2.86 KB
/
limiter_test.go
File metadata and controls
124 lines (105 loc) · 2.86 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
package rrl
import (
"context"
"io"
"log"
"testing"
"time"
"github.com/alicebob/miniredis/v2"
"github.com/redis/go-redis/v9"
"github.com/stretchr/testify/assert"
)
func setupRedisClient(t *testing.T) (*redis.Client, *miniredis.Miniredis) {
mr, err := miniredis.Run()
if err != nil {
t.Fatalf("Error creating mock Redis server: %v", err)
}
client := redis.NewClient(&redis.Options{
Addr: mr.Addr(),
DB: 0,
})
return client, mr
}
func TestRateLimiter_Allow(t *testing.T) {
client, mr := setupRedisClient(t)
defer mr.Close()
// Create a logger that discards output during tests
testLogger := log.New(io.Discard, "", 0)
// Create a mutable time reference for testing
currentTime := time.Now()
// Create Redis Store
redisStore := NewRedisStore(client, false)
redisStore.timeNow = func() time.Time {
return currentTime
}
limiter, err := NewRateLimiter(
WithRate(1),
WithMaxTokens(5),
WithRefillInterval(1*time.Second),
WithStore(redisStore),
WithLogger(testLogger),
)
// Override limiter timeNow as well
if limiter != nil {
limiter.timeNow = func() time.Time {
return currentTime
}
}
if err != nil {
t.Fatal(err)
}
tests := []struct {
name string
key string
expected bool
advance time.Duration
}{
{"First Request", "user1", true, 0},
{"Second Request", "user1", true, 0},
{"Third Request", "user1", true, 0},
{"Fourth Request", "user1", true, 0},
{"Fifth Request", "user1", true, 0},
{"Sixth Request", "user1", false, 0},
{"Wait for Refill", "user1", true, 1 * time.Second},
}
for _, tt := range tests {
if tt.advance > 0 {
// Advance the time for the test
currentTime = currentTime.Add(tt.advance)
}
t.Run(tt.name, func(t *testing.T) {
result := limiter.IsRequestAllowed(tt.key)
assert.Equal(t, tt.expected, result)
})
}
}
func TestRateLimiter_Wait(t *testing.T) {
client, mr := setupRedisClient(t)
defer mr.Close()
// 1 token per second, max 1
store := NewRedisStore(client, false)
// We need to inject timeNow into store if we want deterministic wait,
// but Wait uses time.After/Timer which depends on real time.
// So we can only test that it blocks approximately correctly or use a very short interval.
limiter, err := NewRateLimiter(
WithRate(1),
WithMaxTokens(1),
WithRefillInterval(100*time.Millisecond), // Fast refill for testing
WithStore(store),
)
if err != nil {
t.Fatalf("Failed to create limiter: %v", err)
}
ctx := context.Background()
// 1. First request immediate
start := time.Now()
err = limiter.Wait(ctx, "wait-user")
assert.NoError(t, err)
assert.True(t, time.Since(start) < 50*time.Millisecond)
// 2. Second request should wait ~100ms
start = time.Now()
err = limiter.Wait(ctx, "wait-user")
assert.NoError(t, err)
elapsed := time.Since(start)
assert.True(t, elapsed >= 80*time.Millisecond, "Should wait at least 80ms, got %v", elapsed)
}