@@ -34,7 +34,7 @@ func TestAsyncMemoryQueue(t *testing.T) {
34
34
func TestAsyncMemoryQueueBlocking (t * testing.T ) {
35
35
consumed := & atomic.Int64 {}
36
36
ac := newAsyncQueue (
37
- newMemoryQueue [int64 ](memoryQueueSettings [int64 ]{sizer : sizerInt64 {}, capacity : 100 , blocking : true }),
37
+ newMemoryQueue [int64 ](memoryQueueSettings [int64 ]{sizer : sizerInt64 {}, capacity : 100 , blockOnOverflow : true }),
38
38
4 , func (_ context.Context , _ int64 , done Done ) {
39
39
consumed .Add (1 )
40
40
done .OnDone (nil )
@@ -58,7 +58,7 @@ func TestAsyncMemoryQueueBlocking(t *testing.T) {
58
58
func TestAsyncMemoryQueueBlockingCancelled (t * testing.T ) {
59
59
stop := make (chan struct {})
60
60
ac := newAsyncQueue (
61
- newMemoryQueue [int64 ](memoryQueueSettings [int64 ]{sizer : sizerInt64 {}, capacity : 10 , blocking : true }),
61
+ newMemoryQueue [int64 ](memoryQueueSettings [int64 ]{sizer : sizerInt64 {}, capacity : 10 , blockOnOverflow : true }),
62
62
1 , func (_ context.Context , _ int64 , done Done ) {
63
63
<- stop
64
64
done .OnDone (nil )
@@ -102,3 +102,114 @@ func BenchmarkAsyncMemoryQueue(b *testing.B) {
102
102
require .NoError (b , ac .Shutdown (context .Background ()))
103
103
assert .EqualValues (b , b .N , consumed .Load ())
104
104
}
105
+
106
+ func TestMemoryQueueMultiThread (t * testing.T ) {
107
+ buf := newBuffer ()
108
+ wg := sync.WaitGroup {}
109
+ q := newMemoryQueue [int64 ](memoryQueueSettings [int64 ]{sizer : sizerInt64 {}, capacity : 1000 , blockOnOverflow : true })
110
+ buf .start ()
111
+ ac := newAsyncQueue (q , 3 , buf .consume )
112
+ require .NoError (t , ac .Start (context .Background (), componenttest .NewNopHost ()))
113
+
114
+ for i := 0 ; i < 10 ; i ++ {
115
+ wg .Add (1 )
116
+ go func () {
117
+ defer wg .Done ()
118
+ for j := 0 ; j < 10_000 ; j ++ {
119
+ assert .NoError (t , ac .Offer (context .Background (), int64 (1 )))
120
+ }
121
+ }()
122
+ }
123
+
124
+ wg .Wait ()
125
+ require .NoError (t , ac .Shutdown (context .Background ()))
126
+ buf .shutdown ()
127
+ assert .Equal (t , int64 (10 * 10_000 ), buf .consumed ())
128
+ }
129
+
130
+ func TestMemoryQueueWaitForResponseMultiThread (t * testing.T ) {
131
+ buf := newBuffer ()
132
+ wg := sync.WaitGroup {}
133
+ q := newMemoryQueue [int64 ](memoryQueueSettings [int64 ]{sizer : sizerInt64 {}, capacity : 1000 , waitForResult : true , blockOnOverflow : true })
134
+ buf .start ()
135
+ ac := newAsyncQueue (q , 3 , buf .consume )
136
+ require .NoError (t , ac .Start (context .Background (), componenttest .NewNopHost ()))
137
+
138
+ for i := 0 ; i < 10 ; i ++ {
139
+ wg .Add (1 )
140
+ go func () {
141
+ defer wg .Done ()
142
+ for j := 0 ; j < 10_000 ; j ++ {
143
+ assert .NoError (t , ac .Offer (context .Background (), int64 (1 )))
144
+ }
145
+ }()
146
+ }
147
+
148
+ wg .Wait ()
149
+ require .NoError (t , ac .Shutdown (context .Background ()))
150
+ buf .shutdown ()
151
+ assert .Equal (t , int64 (10 * 10_000 ), buf .consumed ())
152
+ }
153
+
154
+ const flushNum = 5
155
+
156
+ type buffer struct {
157
+ ch chan Done
158
+ nr * atomic.Int64
159
+ wg sync.WaitGroup
160
+ dones []Done
161
+ }
162
+
163
+ func newBuffer () * buffer {
164
+ buf := & buffer {
165
+ ch : make (chan Done , 10 ),
166
+ nr : & atomic.Int64 {},
167
+ dones : make ([]Done , 0 , flushNum ),
168
+ }
169
+ return buf
170
+ }
171
+
172
+ func (buf * buffer ) consume (_ context.Context , _ int64 , done Done ) {
173
+ buf .ch <- done
174
+ }
175
+
176
+ func (buf * buffer ) start () {
177
+ buf .wg .Add (1 )
178
+ go func () {
179
+ defer buf .wg .Done ()
180
+ for {
181
+ select {
182
+ case done , ok := <- buf .ch :
183
+ if ! ok {
184
+ return
185
+ }
186
+ buf .dones = append (buf .dones , done )
187
+ if len (buf .dones ) == flushNum {
188
+ buf .flush ()
189
+ }
190
+ case <- time .After (10 * time .Millisecond ):
191
+ buf .flush ()
192
+ }
193
+ }
194
+ }()
195
+ }
196
+
197
+ func (buf * buffer ) shutdown () {
198
+ close (buf .ch )
199
+ buf .wg .Wait ()
200
+ }
201
+
202
+ func (buf * buffer ) flush () {
203
+ if len (buf .dones ) == 0 {
204
+ return
205
+ }
206
+ buf .nr .Add (int64 (len (buf .dones )))
207
+ for _ , done := range buf .dones {
208
+ done .OnDone (nil )
209
+ }
210
+ buf .dones = buf .dones [:0 ]
211
+ }
212
+
213
+ func (buf * buffer ) consumed () int64 {
214
+ return buf .nr .Load ()
215
+ }
0 commit comments