github.com/mvg-fi/go-limiter@v0.1.1/memorystore/store_test.go (about) 1 package memorystore 2 3 import ( 4 "context" 5 "crypto/rand" 6 "crypto/sha256" 7 "fmt" 8 "sort" 9 "testing" 10 "time" 11 12 "github.com/mvg-fi/go-limiter/internal/fasttime" 13 ) 14 15 func TestFillRate(t *testing.T) { 16 t.Parallel() 17 18 t.Run("many_tokens_small_interval", func(t *testing.T) { 19 t.Parallel() 20 21 s, _ := New(&Config{ 22 Tokens: 65525, 23 Interval: time.Second, 24 }) 25 26 for i := 0; i < 20; i++ { 27 limit, remaining, _, _, _ := s.Take(context.Background(), "key") 28 if remaining < limit-uint64(i)-1 { 29 t.Errorf("invalid remaining: run: %d limit: %d remaining: %d", i, limit, remaining) 30 } 31 time.Sleep(100 * time.Millisecond) 32 } 33 }) 34 } 35 36 func testKey(tb testing.TB) string { 37 tb.Helper() 38 39 var b [512]byte 40 if _, err := rand.Read(b[:]); err != nil { 41 tb.Fatalf("failed to generate random string: %v", err) 42 } 43 digest := fmt.Sprintf("%x", sha256.Sum256(b[:])) 44 return digest[:32] 45 } 46 47 func TestStore_Exercise(t *testing.T) { 48 t.Parallel() 49 50 ctx := context.Background() 51 52 s, err := New(&Config{ 53 Tokens: 5, 54 Interval: 3 * time.Second, 55 SweepInterval: 24 * time.Hour, 56 SweepMinTTL: 24 * time.Hour, 57 }) 58 if err != nil { 59 t.Fatal(err) 60 } 61 t.Cleanup(func() { 62 if err := s.Close(ctx); err != nil { 63 t.Fatal(err) 64 } 65 }) 66 67 key := testKey(t) 68 69 // Get when no config exists 70 { 71 limit, remaining, err := s.(*store).Get(ctx, key) 72 if err != nil { 73 t.Fatal(err) 74 } 75 76 if got, want := limit, uint64(0); got != want { 77 t.Errorf("expected %v to be %v", got, want) 78 } 79 if got, want := remaining, uint64(0); got != want { 80 t.Errorf("expected %v to be %v", got, want) 81 } 82 } 83 84 // Take with no key configuration - this should use the default values 85 { 86 limit, remaining, reset, ok, err := s.Take(ctx, key) 87 if err != nil { 88 t.Fatal(err) 89 } 90 if !ok { 91 t.Errorf("expected ok") 92 } 93 if got, want := limit, uint64(5); got != want { 94 t.Errorf("expected %v to be %v", got, want) 95 } 96 if got, want := remaining, uint64(4); got != want { 97 t.Errorf("expected %v to be %v", got, want) 98 } 99 if got, want := time.Until(time.Unix(0, int64(reset))), 3*time.Second; got > want { 100 t.Errorf("expected %v to less than %v", got, want) 101 } 102 } 103 104 // Get the value 105 { 106 limit, remaining, err := s.(*store).Get(ctx, key) 107 if err != nil { 108 t.Fatal(err) 109 } 110 if got, want := limit, uint64(5); got != want { 111 t.Errorf("expected %v to be %v", got, want) 112 } 113 if got, want := remaining, uint64(4); got != want { 114 t.Errorf("expected %v to be %v", got, want) 115 } 116 } 117 118 // Now set a value 119 { 120 if err := s.Set(ctx, key, 11, 5*time.Second); err != nil { 121 t.Fatal(err) 122 } 123 } 124 125 // Get the value again 126 { 127 limit, remaining, err := s.(*store).Get(ctx, key) 128 if err != nil { 129 t.Fatal(err) 130 } 131 if got, want := limit, uint64(11); got != want { 132 t.Errorf("expected %v to be %v", got, want) 133 } 134 if got, want := remaining, uint64(11); got != want { 135 t.Errorf("expected %v to be %v", got, want) 136 } 137 } 138 139 // Take again, this should use the new values 140 { 141 limit, remaining, reset, ok, err := s.Take(ctx, key) 142 if err != nil { 143 t.Fatal(err) 144 } 145 if !ok { 146 t.Errorf("expected ok") 147 } 148 if got, want := limit, uint64(11); got != want { 149 t.Errorf("expected %v to be %v", got, want) 150 } 151 if got, want := remaining, uint64(10); got != want { 152 t.Errorf("expected %v to be %v", got, want) 153 } 154 if got, want := time.Until(time.Unix(0, int64(reset))), 5*time.Second; got > want { 155 t.Errorf("expected %v to less than %v", got, want) 156 } 157 } 158 159 // Get the value again 160 { 161 limit, remaining, err := s.(*store).Get(ctx, key) 162 if err != nil { 163 t.Fatal(err) 164 } 165 if got, want := limit, uint64(11); got != want { 166 t.Errorf("expected %v to be %v", got, want) 167 } 168 if got, want := remaining, uint64(10); got != want { 169 t.Errorf("expected %v to be %v", got, want) 170 } 171 } 172 173 // Burst and take 174 { 175 if err := s.Burst(ctx, key, 5); err != nil { 176 t.Fatal(err) 177 } 178 179 limit, remaining, reset, ok, err := s.Take(ctx, key) 180 if err != nil { 181 t.Fatal(err) 182 } 183 if !ok { 184 t.Errorf("expected ok") 185 } 186 if got, want := limit, uint64(11); got != want { 187 t.Errorf("expected %v to be %v", got, want) 188 } 189 if got, want := remaining, uint64(14); got != want { 190 t.Errorf("expected %v to be %v", got, want) 191 } 192 if got, want := time.Until(time.Unix(0, int64(reset))), 5*time.Second; got > want { 193 t.Errorf("expected %v to less than %v", got, want) 194 } 195 } 196 197 // Get the value one final time 198 { 199 limit, remaining, err := s.(*store).Get(ctx, key) 200 if err != nil { 201 t.Fatal(err) 202 } 203 if got, want := limit, uint64(11); got != want { 204 t.Errorf("expected %v to be %v", got, want) 205 } 206 if got, want := remaining, uint64(14); got != want { 207 t.Errorf("expected %v to be %v", got, want) 208 } 209 } 210 } 211 212 func TestStore_Take(t *testing.T) { 213 t.Parallel() 214 215 ctx := context.Background() 216 217 cases := []struct { 218 name string 219 tokens uint64 220 interval time.Duration 221 }{ 222 { 223 name: "milli", 224 tokens: 5, 225 interval: 500 * time.Millisecond, 226 }, 227 { 228 name: "second", 229 tokens: 10, 230 interval: 1 * time.Second, 231 }, 232 } 233 234 for _, tc := range cases { 235 tc := tc 236 237 t.Run(tc.name, func(t *testing.T) { 238 t.Parallel() 239 240 key := testKey(t) 241 242 s, err := New(&Config{ 243 Interval: tc.interval, 244 Tokens: tc.tokens, 245 SweepInterval: 24 * time.Hour, 246 SweepMinTTL: 24 * time.Hour, 247 }) 248 if err != nil { 249 t.Fatal(err) 250 } 251 t.Cleanup(func() { 252 if err := s.Close(ctx); err != nil { 253 t.Fatal(err) 254 } 255 }) 256 257 type result struct { 258 limit, remaining uint64 259 reset time.Duration 260 ok bool 261 err error 262 } 263 264 // Take twice everything from the bucket. 265 takeCh := make(chan *result, 2*tc.tokens) 266 for i := uint64(1); i <= 2*tc.tokens; i++ { 267 go func() { 268 limit, remaining, reset, ok, err := s.Take(ctx, key) 269 takeCh <- &result{limit, remaining, time.Duration(fasttime.Now() - reset), ok, err} 270 }() 271 } 272 273 // Accumulate and sort results, since they could come in any order. 274 var results []*result 275 for i := uint64(1); i <= 2*tc.tokens; i++ { 276 select { 277 case result := <-takeCh: 278 results = append(results, result) 279 case <-time.After(5 * time.Second): 280 t.Fatal("timeout") 281 } 282 } 283 sort.Slice(results, func(i, j int) bool { 284 if results[i].remaining == results[j].remaining { 285 return !results[j].ok 286 } 287 return results[i].remaining > results[j].remaining 288 }) 289 290 for i, result := range results { 291 if err := result.err; err != nil { 292 t.Fatal(err) 293 } 294 295 if got, want := result.limit, tc.tokens; got != want { 296 t.Errorf("limit: expected %d to be %d", got, want) 297 } 298 if got, want := result.reset, tc.interval; got > want { 299 t.Errorf("reset: expected %d to be less than %d", got, want) 300 } 301 302 // first half should pass, second half should fail 303 if uint64(i) < tc.tokens { 304 if got, want := result.remaining, tc.tokens-uint64(i)-1; got != want { 305 t.Errorf("remaining: expected %d to be %d", got, want) 306 } 307 if got, want := result.ok, true; got != want { 308 t.Errorf("ok: expected %t to be %t", got, want) 309 } 310 } else { 311 if got, want := result.remaining, uint64(0); got != want { 312 t.Errorf("remaining: expected %d to be %d", got, want) 313 } 314 if got, want := result.ok, false; got != want { 315 t.Errorf("ok: expected %t to be %t", got, want) 316 } 317 } 318 } 319 320 // Wait for the bucket to have entries again. 321 time.Sleep(tc.interval) 322 323 // Verify we can take once more. 324 _, _, _, ok, err := s.Take(ctx, key) 325 if err != nil { 326 t.Fatal(err) 327 } 328 if !ok { 329 t.Errorf("expected %t to be %t", ok, true) 330 } 331 }) 332 } 333 } 334 335 func TestBucketedLimiter_tick(t *testing.T) { 336 t.Parallel() 337 338 cases := []struct { 339 name string 340 start uint64 341 curr uint64 342 interval time.Duration 343 exp uint64 344 }{ 345 { 346 name: "no_diff", 347 start: 0, 348 curr: 0, 349 interval: time.Second, 350 exp: 0, 351 }, 352 { 353 name: "half", 354 start: 0, 355 curr: uint64(500 * time.Millisecond), 356 interval: time.Second, 357 exp: 0, 358 }, 359 { 360 name: "almost", 361 start: 0, 362 curr: uint64(1*time.Second - time.Nanosecond), 363 interval: time.Second, 364 exp: 0, 365 }, 366 { 367 name: "exact", 368 start: 0, 369 curr: uint64(1 * time.Second), 370 interval: time.Second, 371 exp: 1, 372 }, 373 { 374 name: "multiple", 375 start: 0, 376 curr: uint64(50*time.Second - 500*time.Millisecond), 377 interval: time.Second, 378 exp: 49, 379 }, 380 { 381 name: "short", 382 start: 0, 383 curr: uint64(50*time.Second - 500*time.Millisecond), 384 interval: time.Millisecond, 385 exp: 49500, 386 }, 387 } 388 389 for _, tc := range cases { 390 tc := tc 391 392 t.Run(tc.name, func(t *testing.T) { 393 t.Parallel() 394 395 if got, want := tick(tc.start, tc.curr, tc.interval), tc.exp; got != want { 396 t.Errorf("expected %v to be %v", got, want) 397 } 398 }) 399 } 400 }