@@ -116,6 +116,7 @@ func (i *ClientConfig) Validate() error {
116
116
117
117
type Client interface {
118
118
FilterChunks (ctx context.Context , tenant string , interval bloomshipper.Interval , blocks []blockWithSeries , plan plan.QueryPlan ) ([]* logproto.GroupedChunkRefs , error )
119
+ PrefetchBloomBlocks (ctx context.Context , blocks []bloomshipper.BlockRef ) error
119
120
}
120
121
121
122
// clientPool is a minimal interface that is satisfied by the JumpHashClientPool.
@@ -204,6 +205,47 @@ func (c *GatewayClient) Close() {
204
205
c .dnsProvider .Stop ()
205
206
}
206
207
208
+ func (c * GatewayClient ) PrefetchBloomBlocks (ctx context.Context , blocks []bloomshipper.BlockRef ) error {
209
+ if len (blocks ) == 0 {
210
+ return nil
211
+ }
212
+
213
+ pos := make (map [string ]int )
214
+ servers := make ([]addrWithBlocks , 0 , len (blocks ))
215
+ for _ , block := range blocks {
216
+ addr , err := c .pool .Addr (block .String ())
217
+ if err != nil {
218
+ level .Error (c .logger ).Log ("msg" , "failed to resolve server address for block" , "block" , block , "err" , err )
219
+ continue
220
+ }
221
+
222
+ if idx , found := pos [addr ]; found {
223
+ servers [idx ].blocks = append (servers [idx ].blocks , block .String ())
224
+ } else {
225
+ pos [addr ] = len (servers )
226
+ servers = append (servers , addrWithBlocks {
227
+ addr : addr ,
228
+ blocks : []string {block .String ()},
229
+ })
230
+ }
231
+ }
232
+
233
+ return concurrency .ForEachJob (ctx , len (servers ), len (servers ), func (ctx context.Context , i int ) error {
234
+ rs := servers [i ]
235
+ return c .doForAddrs ([]string {rs .addr }, func (client logproto.BloomGatewayClient ) error {
236
+ req := & logproto.PrefetchBloomBlocksRequest {Blocks : rs .blocks }
237
+ _ , err := client .PrefetchBloomBlocks (ctx , req )
238
+ if err != nil {
239
+ level .Error (c .logger ).Log ("msg" , "block prefetch failed for instance, skipping" , "addr" , rs .addr , "blocks" , len (rs .blocks ), "err" , err )
240
+ c .metrics .clientRequests .WithLabelValues (routePrefectBlocks , typeError ).Inc ()
241
+ } else {
242
+ c .metrics .clientRequests .WithLabelValues (routePrefectBlocks , typeSuccess ).Inc ()
243
+ }
244
+ return err
245
+ })
246
+ })
247
+ }
248
+
207
249
// FilterChunks implements Client
208
250
func (c * GatewayClient ) FilterChunks (ctx context.Context , _ string , interval bloomshipper.Interval , blocks []blockWithSeries , plan plan.QueryPlan ) ([]* logproto.GroupedChunkRefs , error ) {
209
251
// no block and therefore no series with chunks
@@ -268,10 +310,10 @@ func (c *GatewayClient) FilterChunks(ctx context.Context, _ string, interval blo
268
310
"err" , err ,
269
311
)
270
312
// filter none of the results on failed request
271
- c .metrics .clientRequests .WithLabelValues (typeError ).Inc ()
313
+ c .metrics .clientRequests .WithLabelValues (routeFilterChunks , typeError ).Inc ()
272
314
results [i ] = rs .groups
273
315
} else {
274
- c .metrics .clientRequests .WithLabelValues (typeSuccess ).Inc ()
316
+ c .metrics .clientRequests .WithLabelValues (routeFilterChunks , typeSuccess ).Inc ()
275
317
results [i ] = resp .ChunkRefs
276
318
}
277
319
@@ -390,6 +432,11 @@ func (c *GatewayClient) doForAddrs(addrs []string, fn func(logproto.BloomGateway
390
432
return err
391
433
}
392
434
435
+ type addrWithBlocks struct {
436
+ addr string
437
+ blocks []string
438
+ }
439
+
393
440
type addrWithGroups struct {
394
441
addr string
395
442
blocks []string
0 commit comments