/
delayed_bucket_client.go
101 lines (77 loc) · 2.49 KB
/
delayed_bucket_client.go
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
// SPDX-License-Identifier: AGPL-3.0-only
package bucket
import (
"context"
"io"
"math/rand"
"time"
"github.com/thanos-io/objstore"
)
// DelayedBucketClient wraps objstore.InstrumentedBucket and add a random delay to each API call.
// This client is intended to be used only for testing purposes.
type DelayedBucketClient struct {
wrapped objstore.Bucket
minDelay time.Duration
maxDelay time.Duration
}
func NewDelayedBucketClient(wrapped objstore.Bucket, minDelay, maxDelay time.Duration) objstore.Bucket {
if minDelay < 0 || maxDelay < 0 || maxDelay < minDelay {
// We're fine just panicking, because we expect this client to be used only for testing purposes.
panic("invalid delay configuration")
}
return &DelayedBucketClient{
wrapped: wrapped,
minDelay: minDelay,
maxDelay: maxDelay,
}
}
func (m *DelayedBucketClient) Upload(ctx context.Context, name string, r io.Reader) error {
m.delay()
defer m.delay()
return m.wrapped.Upload(ctx, name, r)
}
func (m *DelayedBucketClient) Delete(ctx context.Context, name string) error {
m.delay()
defer m.delay()
return m.wrapped.Delete(ctx, name)
}
func (m *DelayedBucketClient) Name() string {
return m.wrapped.Name()
}
func (m *DelayedBucketClient) Iter(ctx context.Context, dir string, f func(string) error, options ...objstore.IterOption) error {
m.delay()
defer m.delay()
return m.wrapped.Iter(ctx, dir, f, options...)
}
func (m *DelayedBucketClient) Get(ctx context.Context, name string) (io.ReadCloser, error) {
m.delay()
defer m.delay()
return m.wrapped.Get(ctx, name)
}
func (m *DelayedBucketClient) GetRange(ctx context.Context, name string, off, length int64) (io.ReadCloser, error) {
m.delay()
defer m.delay()
return m.wrapped.GetRange(ctx, name, off, length)
}
func (m *DelayedBucketClient) Exists(ctx context.Context, name string) (bool, error) {
m.delay()
defer m.delay()
return m.wrapped.Exists(ctx, name)
}
func (m *DelayedBucketClient) IsObjNotFoundErr(err error) bool {
return m.wrapped.IsObjNotFoundErr(err)
}
func (m *DelayedBucketClient) IsAccessDeniedErr(err error) bool {
return m.wrapped.IsAccessDeniedErr(err)
}
func (m *DelayedBucketClient) Attributes(ctx context.Context, name string) (objstore.ObjectAttributes, error) {
m.delay()
defer m.delay()
return m.wrapped.Attributes(ctx, name)
}
func (m *DelayedBucketClient) Close() error {
return m.wrapped.Close()
}
func (m *DelayedBucketClient) delay() {
time.Sleep(m.minDelay + time.Duration(rand.Int63n(m.maxDelay.Nanoseconds()-m.minDelay.Nanoseconds())))
}