Skip to content

Commit 926e78d

Browse files
authored
Add robust test case & run it against Redis in CI (#10)
* Add robust test case & run it against Redis in CI * Improve example, add go module to fix CI
1 parent 925f2a6 commit 926e78d

File tree

7 files changed

+227
-3
lines changed

7 files changed

+227
-3
lines changed

.github/workflows/ci.yml

+37
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
name: CI
2+
3+
on:
4+
push:
5+
pull_request:
6+
7+
jobs:
8+
tests:
9+
name: Tests
10+
runs-on: ubuntu-latest
11+
12+
services:
13+
redis:
14+
image: redis:6
15+
ports:
16+
- 6379:6379
17+
18+
steps:
19+
- name: Set up Go
20+
uses: actions/setup-go@v5
21+
with:
22+
go-version: ^1.17
23+
24+
- name: Check out code into the Go module directory
25+
uses: actions/checkout@v4
26+
27+
- name: Get dependencies
28+
run: go get -v -t -d ./...
29+
30+
- name: Build
31+
run: go build -v ./
32+
33+
- name: Build example
34+
run: cd ./_example && go build -v ./
35+
36+
- name: Test
37+
run: go test -v ./...

_example/go.mod

+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
module github.com/go-chi/httprate-redis/_example
2+
3+
go 1.22.5
4+
5+
replace github.com/go-chi/httprate-redis => ../
6+
7+
require (
8+
github.com/cespare/xxhash/v2 v2.3.0 // indirect
9+
github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f // indirect
10+
github.com/go-chi/chi/v5 v5.1.0 // indirect
11+
github.com/go-chi/httprate v0.11.0 // indirect
12+
github.com/go-chi/httprate-redis v0.3.0 // indirect
13+
github.com/redis/go-redis/v9 v9.6.1 // indirect
14+
)

_example/go.sum

+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
github.com/cespare/xxhash/v2 v2.3.0 h1:UL815xU9SqsFlibzuggzjXhog7bL6oX9BbNZnL2UFvs=
2+
github.com/cespare/xxhash/v2 v2.3.0/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs=
3+
github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f h1:lO4WD4F/rVNCu3HqELle0jiPLLBs70cWOduZpkS1E78=
4+
github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f/go.mod h1:cuUVRXasLTGF7a8hSLbxyZXjz+1KgoB3wDUb6vlszIc=
5+
github.com/go-chi/chi/v5 v5.1.0 h1:acVI1TYaD+hhedDJ3r54HyA6sExp3HfXq7QWEEY/xMw=
6+
github.com/go-chi/chi/v5 v5.1.0/go.mod h1:DslCQbL2OYiznFReuXYUmQ2hGd1aDpCnlMNITLSKoi8=
7+
github.com/go-chi/httprate v0.11.0 h1:GKsQF0qlVSU1Zg/D07tUUnNrYTTkrtpMihwO6myxtzc=
8+
github.com/go-chi/httprate v0.11.0/go.mod h1:TUepLXaz/pCjmCtf/obgOQJ2Sz6rC8fSf5cAt5cnTt0=
9+
github.com/redis/go-redis/v9 v9.6.0 h1:NLck+Rab3AOTHw21CGRpvQpgTrAU4sgdCswqGtlhGRA=
10+
github.com/redis/go-redis/v9 v9.6.0/go.mod h1:hdY0cQFCN4fnSYT6TkisLufl/4W5UIXyv0b/CLO2V2M=
11+
github.com/redis/go-redis/v9 v9.6.1 h1:HHDteefn6ZkTtY5fGUE8tj8uy85AHk6zP7CpzIAM0y4=
12+
github.com/redis/go-redis/v9 v9.6.1/go.mod h1:0C0c6ycQsdpVNQpxb1njEQIqkx5UcsM8FJCQLgE9+RA=

_example/main.go

+9-3
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
package main
22

33
import (
4+
"log"
45
"net/http"
56
"time"
67

@@ -16,18 +17,23 @@ func main() {
1617

1718
r.Group(func(r chi.Router) {
1819
r.Use(httprate.Limit(
19-
10,
20-
10*time.Second,
20+
5,
21+
time.Minute,
2122
httprate.WithKeyByIP(),
2223
httprateredis.WithRedisLimitCounter(&httprateredis.Config{
2324
Host: "127.0.0.1", Port: 6379,
2425
}),
2526
))
2627

2728
r.Get("/", func(w http.ResponseWriter, r *http.Request) {
28-
w.Write([]byte("."))
29+
w.Write([]byte("5 req/min\n"))
2930
})
3031
})
3132

33+
log.Printf("Serving at localhost:3333")
34+
log.Println()
35+
log.Printf("Try running:")
36+
log.Printf("curl -v http://localhost:3333")
37+
3238
http.ListenAndServe(":3333", r)
3339
}

go.mod

+1
Original file line numberDiff line numberDiff line change
@@ -10,4 +10,5 @@ require (
1010
require (
1111
github.com/cespare/xxhash/v2 v2.3.0 // indirect
1212
github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f // indirect
13+
golang.org/x/sync v0.7.0 // indirect
1314
)

go.sum

+2
Original file line numberDiff line numberDiff line change
@@ -14,3 +14,5 @@ github.com/redis/go-redis/v9 v9.3.0 h1:RiVDjmig62jIWp7Kk4XVLs0hzV6pI3PyTnnL0cnn0
1414
github.com/redis/go-redis/v9 v9.3.0/go.mod h1:hdY0cQFCN4fnSYT6TkisLufl/4W5UIXyv0b/CLO2V2M=
1515
github.com/redis/go-redis/v9 v9.6.0 h1:NLck+Rab3AOTHw21CGRpvQpgTrAU4sgdCswqGtlhGRA=
1616
github.com/redis/go-redis/v9 v9.6.0/go.mod h1:hdY0cQFCN4fnSYT6TkisLufl/4W5UIXyv0b/CLO2V2M=
17+
golang.org/x/sync v0.7.0 h1:YsImfSBoP9QPYL0xyKJPq0gcaJdG3rInoqxTWbfQu9M=
18+
golang.org/x/sync v0.7.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk=

httprateredis_test.go

+152
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,152 @@
1+
package httprateredis_test
2+
3+
import (
4+
"fmt"
5+
"math/rand"
6+
"testing"
7+
"time"
8+
9+
httprateredis "github.com/go-chi/httprate-redis"
10+
"golang.org/x/sync/errgroup"
11+
)
12+
13+
func TestRedisCounter(t *testing.T) {
14+
limitCounter, err := httprateredis.NewRedisLimitCounter(&httprateredis.Config{
15+
Host: "localhost",
16+
Port: 6379,
17+
MaxIdle: 500,
18+
MaxActive: 500,
19+
DBIndex: 0,
20+
ClientName: "httprateredis_test",
21+
PrefixKey: fmt.Sprintf("httprate:test:%v", rand.Int31n(100000)), // Unique key for each test
22+
})
23+
if err != nil {
24+
t.Fatalf("redis not available: %v", err)
25+
}
26+
27+
limitCounter.Config(1000, time.Minute)
28+
29+
currentWindow := time.Now().UTC().Truncate(time.Minute)
30+
previousWindow := currentWindow.Add(-time.Minute)
31+
32+
type test struct {
33+
name string // In each test do the following:
34+
advanceTime time.Duration // 1. advance time
35+
incrBy int // 2. increase counter
36+
prev int // 3. check previous window counter
37+
curr int // and current window counter
38+
}
39+
40+
tests := []test{
41+
{
42+
name: "t=0m: init",
43+
prev: 0,
44+
curr: 0,
45+
},
46+
{
47+
name: "t=0m: increment 1",
48+
incrBy: 1,
49+
prev: 0,
50+
curr: 1,
51+
},
52+
{
53+
name: "t=0m: increment by 99",
54+
incrBy: 99,
55+
prev: 0,
56+
curr: 100,
57+
},
58+
{
59+
name: "t=1m: move clock by 1m",
60+
advanceTime: time.Minute,
61+
prev: 100,
62+
curr: 0,
63+
},
64+
{
65+
name: "t=1m: increment by 20",
66+
incrBy: 20,
67+
prev: 100,
68+
curr: 20,
69+
},
70+
{
71+
name: "t=1m: increment by 20",
72+
incrBy: 20,
73+
prev: 100,
74+
curr: 40,
75+
},
76+
{
77+
name: "t=2m: move clock by 1m",
78+
advanceTime: time.Minute,
79+
prev: 40,
80+
curr: 0,
81+
},
82+
{
83+
name: "t=2m: incr++",
84+
incrBy: 1,
85+
prev: 40,
86+
curr: 1,
87+
},
88+
{
89+
name: "t=2m: incr+=9",
90+
incrBy: 9,
91+
prev: 40,
92+
curr: 10,
93+
},
94+
{
95+
name: "t=2m: incr+=20",
96+
incrBy: 20,
97+
prev: 40,
98+
curr: 30,
99+
},
100+
{
101+
name: "t=4m: move clock by 2m",
102+
advanceTime: 2 * time.Minute,
103+
prev: 0,
104+
curr: 0,
105+
},
106+
}
107+
108+
concurrentRequests := 1000
109+
110+
for _, tt := range tests {
111+
if tt.advanceTime > 0 {
112+
currentWindow = currentWindow.Add(tt.advanceTime)
113+
previousWindow = previousWindow.Add(tt.advanceTime)
114+
}
115+
116+
if tt.incrBy > 0 {
117+
var g errgroup.Group
118+
for i := 0; i < concurrentRequests; i++ {
119+
i := i
120+
g.Go(func() error {
121+
key := fmt.Sprintf("key:%v", i)
122+
return limitCounter.IncrementBy(key, currentWindow, tt.incrBy)
123+
})
124+
}
125+
if err := g.Wait(); err != nil {
126+
t.Errorf("%s: %v", tt.name, err)
127+
}
128+
}
129+
130+
var g errgroup.Group
131+
for i := 0; i < concurrentRequests; i++ {
132+
i := i
133+
g.Go(func() error {
134+
key := fmt.Sprintf("key:%v", i)
135+
curr, prev, err := limitCounter.Get(key, currentWindow, previousWindow)
136+
if err != nil {
137+
return fmt.Errorf("%q: %w", key, err)
138+
}
139+
if curr != tt.curr {
140+
return fmt.Errorf("%q: unexpected curr = %v, expected %v", key, curr, tt.curr)
141+
}
142+
if prev != tt.prev {
143+
return fmt.Errorf("%q: unexpected prev = %v, expected %v", key, prev, tt.prev)
144+
}
145+
return nil
146+
})
147+
}
148+
if err := g.Wait(); err != nil {
149+
t.Errorf("%s: %v", tt.name, err)
150+
}
151+
}
152+
}

0 commit comments

Comments
 (0)