Skip to main content

mito2/read/
scan_util.rs

1// Copyright 2023 Greptime Team
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7//     http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15//! Utilities for scanners.
16
17use std::collections::{BinaryHeap, HashMap, VecDeque};
18use std::fmt;
19use std::pin::Pin;
20use std::sync::{Arc, Mutex};
21use std::task::{Context, Poll};
22use std::time::{Duration, Instant};
23
24use async_stream::try_stream;
25use common_telemetry::tracing;
26use datafusion::physical_plan::metrics::{ExecutionPlanMetricsSet, MetricBuilder, Time};
27use datatypes::arrow::record_batch::RecordBatch;
28use datatypes::timestamp::timestamp_array_to_primitive;
29use futures::Stream;
30use prometheus::IntGauge;
31use smallvec::SmallVec;
32use snafu::OptionExt;
33use store_api::storage::RegionId;
34
35use crate::error::{Result, UnexpectedSnafu};
36use crate::memtable::MemScanMetrics;
37use crate::metrics::{
38    IN_PROGRESS_SCAN, PRECISE_FILTER_ROWS_TOTAL, READ_BATCHES_RETURN, READ_ROW_GROUPS_TOTAL,
39    READ_ROWS_IN_ROW_GROUP_TOTAL, READ_ROWS_RETURN, READ_STAGE_ELAPSED,
40};
41use crate::read::dedup::{DedupMetrics, DedupMetricsReport};
42use crate::read::merge::{MergeMetrics, MergeMetricsReport};
43use crate::read::pruner::PartitionPruner;
44use crate::read::range::{RangeMeta, RowGroupIndex};
45use crate::read::scan_region::StreamContext;
46use crate::read::{Batch, BoxedBatchStream, BoxedRecordBatchStream, ScannerMetrics, Source};
47use crate::sst::file::{FileTimeRange, RegionFileId};
48use crate::sst::index::bloom_filter::applier::BloomFilterIndexApplyMetrics;
49use crate::sst::index::fulltext_index::applier::FulltextIndexApplyMetrics;
50use crate::sst::index::inverted_index::applier::InvertedIndexApplyMetrics;
51use crate::sst::parquet::DEFAULT_ROW_GROUP_SIZE;
52use crate::sst::parquet::file_range::FileRange;
53use crate::sst::parquet::flat_format::time_index_column_index;
54use crate::sst::parquet::reader::{MetadataCacheMetrics, ReaderFilterMetrics, ReaderMetrics};
55use crate::sst::parquet::row_group::ParquetFetchMetrics;
56
57/// Per-file scan metrics.
58#[derive(Default, Clone)]
59pub struct FileScanMetrics {
60    /// Number of ranges (row groups) read from this file.
61    pub num_ranges: usize,
62    /// Number of rows read from this file.
63    pub num_rows: usize,
64    /// Time spent building file ranges/parts (file-level preparation).
65    pub build_part_cost: Duration,
66    /// Time spent building readers for this file (accumulated across all ranges).
67    pub build_reader_cost: Duration,
68    /// Time spent scanning this file (accumulated across all ranges).
69    pub scan_cost: Duration,
70}
71
72impl fmt::Debug for FileScanMetrics {
73    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
74        write!(f, "{{\"build_part_cost\":\"{:?}\"", self.build_part_cost)?;
75
76        if self.num_ranges > 0 {
77            write!(f, ", \"num_ranges\":{}", self.num_ranges)?;
78        }
79        if self.num_rows > 0 {
80            write!(f, ", \"num_rows\":{}", self.num_rows)?;
81        }
82        if !self.build_reader_cost.is_zero() {
83            write!(
84                f,
85                ", \"build_reader_cost\":\"{:?}\"",
86                self.build_reader_cost
87            )?;
88        }
89        if !self.scan_cost.is_zero() {
90            write!(f, ", \"scan_cost\":\"{:?}\"", self.scan_cost)?;
91        }
92
93        write!(f, "}}")
94    }
95}
96
97impl FileScanMetrics {
98    /// Merges another FileMetrics into this one.
99    pub(crate) fn merge_from(&mut self, other: &FileScanMetrics) {
100        self.num_ranges += other.num_ranges;
101        self.num_rows += other.num_rows;
102        self.build_part_cost += other.build_part_cost;
103        self.build_reader_cost += other.build_reader_cost;
104        self.scan_cost += other.scan_cost;
105    }
106}
107
108/// Verbose scan metrics for a partition.
109#[derive(Default)]
110pub(crate) struct ScanMetricsSet {
111    /// Duration to prepare the scan task.
112    prepare_scan_cost: Duration,
113    /// Duration to build the (merge) reader.
114    build_reader_cost: Duration,
115    /// Duration to scan data.
116    scan_cost: Duration,
117    /// Duration while waiting for `yield`.
118    yield_cost: Duration,
119    /// Duration to convert [`Batch`]es.
120    convert_cost: Option<Time>,
121    /// Duration of the scan.
122    total_cost: Duration,
123    /// Number of rows returned.
124    num_rows: usize,
125    /// Number of batches returned.
126    num_batches: usize,
127    /// Number of mem ranges scanned.
128    num_mem_ranges: usize,
129    /// Number of file ranges scanned.
130    num_file_ranges: usize,
131
132    // Memtable related metrics:
133    /// Duration to scan memtables.
134    mem_scan_cost: Duration,
135    /// Number of rows read from memtables.
136    mem_rows: usize,
137    /// Number of batches read from memtables.
138    mem_batches: usize,
139    /// Number of series read from memtables.
140    mem_series: usize,
141    /// Duration of prefilter in memtable scan.
142    mem_prefilter_cost: Duration,
143    /// Number of rows filtered by prefilter in memtable scan.
144    mem_prefilter_rows_filtered: usize,
145
146    // SST related metrics:
147    /// Duration to build file ranges.
148    build_parts_cost: Duration,
149    /// Duration to scan SST files.
150    sst_scan_cost: Duration,
151    /// Number of row groups before filtering.
152    rg_total: usize,
153    /// Number of row groups filtered by fulltext index.
154    rg_fulltext_filtered: usize,
155    /// Number of row groups filtered by inverted index.
156    rg_inverted_filtered: usize,
157    /// Number of row groups filtered by min-max index.
158    rg_minmax_filtered: usize,
159    /// Number of row groups filtered by bloom filter index.
160    rg_bloom_filtered: usize,
161    /// Number of row groups filtered by vector index.
162    rg_vector_filtered: usize,
163    /// Number of rows in row group before filtering.
164    rows_before_filter: usize,
165    /// Number of rows in row group filtered by fulltext index.
166    rows_fulltext_filtered: usize,
167    /// Number of rows in row group filtered by inverted index.
168    rows_inverted_filtered: usize,
169    /// Number of rows in row group filtered by bloom filter index.
170    rows_bloom_filtered: usize,
171    /// Number of rows filtered by vector index.
172    rows_vector_filtered: usize,
173    /// Number of rows selected by vector index.
174    rows_vector_selected: usize,
175    /// Number of rows filtered by precise filter.
176    rows_precise_filtered: usize,
177    /// Number of index result cache hits for fulltext index.
178    fulltext_index_cache_hit: usize,
179    /// Number of index result cache misses for fulltext index.
180    fulltext_index_cache_miss: usize,
181    /// Number of index result cache hits for inverted index.
182    inverted_index_cache_hit: usize,
183    /// Number of index result cache misses for inverted index.
184    inverted_index_cache_miss: usize,
185    /// Number of index result cache hits for bloom filter index.
186    bloom_filter_cache_hit: usize,
187    /// Number of index result cache misses for bloom filter index.
188    bloom_filter_cache_miss: usize,
189    /// Number of index result cache hits for minmax pruning.
190    minmax_cache_hit: usize,
191    /// Number of index result cache misses for minmax pruning.
192    minmax_cache_miss: usize,
193    /// Number of pruner builder cache hits.
194    pruner_cache_hit: usize,
195    /// Number of pruner builder cache misses.
196    pruner_cache_miss: usize,
197    /// Duration spent waiting for pruner to build file ranges.
198    pruner_prune_cost: Duration,
199    /// Number of record batches read from SST.
200    num_sst_record_batches: usize,
201    /// Number of batches decoded from SST.
202    num_sst_batches: usize,
203    /// Number of rows read from SST.
204    num_sst_rows: usize,
205
206    /// Elapsed time before the first poll operation.
207    first_poll: Duration,
208
209    /// Number of send timeout in SeriesScan.
210    num_series_send_timeout: usize,
211    /// Number of send full in SeriesScan.
212    num_series_send_full: usize,
213    /// Number of rows the series distributor scanned.
214    num_distributor_rows: usize,
215    /// Number of batches the series distributor scanned.
216    num_distributor_batches: usize,
217    /// Duration of the series distributor to scan.
218    distributor_scan_cost: Duration,
219    /// Duration of the series distributor to yield.
220    distributor_yield_cost: Duration,
221    /// Duration spent in divider operations.
222    distributor_divider_cost: Duration,
223
224    /// Merge metrics.
225    merge_metrics: MergeMetrics,
226    /// Dedup metrics.
227    dedup_metrics: DedupMetrics,
228
229    /// The stream reached EOF
230    stream_eof: bool,
231
232    // Optional verbose metrics:
233    /// Inverted index apply metrics.
234    inverted_index_apply_metrics: Option<InvertedIndexApplyMetrics>,
235    /// Bloom filter index apply metrics.
236    bloom_filter_apply_metrics: Option<BloomFilterIndexApplyMetrics>,
237    /// Fulltext index apply metrics.
238    fulltext_index_apply_metrics: Option<FulltextIndexApplyMetrics>,
239    /// Parquet fetch metrics.
240    fetch_metrics: Option<ParquetFetchMetrics>,
241    /// Metadata cache metrics.
242    metadata_cache_metrics: Option<MetadataCacheMetrics>,
243    /// Per-file scan metrics, only populated when explain_verbose is true.
244    per_file_metrics: Option<HashMap<RegionFileId, FileScanMetrics>>,
245
246    /// Current memory usage for file range builders.
247    build_ranges_mem_size: isize,
248    /// Peak memory usage for file range builders.
249    build_ranges_peak_mem_size: isize,
250    /// Current number of file range builders.
251    num_range_builders: isize,
252    /// Peak number of file range builders.
253    num_peak_range_builders: isize,
254    /// Total bytes added to the range cache during this scan.
255    range_cache_size: usize,
256    /// Number of range cache hits during this scan.
257    range_cache_hit: usize,
258    /// Number of range cache misses during this scan.
259    range_cache_miss: usize,
260}
261
262/// Wrapper for file metrics that compares by total cost in reverse order.
263/// This allows using BinaryHeap as a min-heap for efficient top-K selection.
264struct CompareCostReverse<'a> {
265    total_cost: Duration,
266    file_id: RegionFileId,
267    metrics: &'a FileScanMetrics,
268}
269
270impl Ord for CompareCostReverse<'_> {
271    fn cmp(&self, other: &Self) -> std::cmp::Ordering {
272        // Reverse comparison: smaller costs are "greater"
273        other.total_cost.cmp(&self.total_cost)
274    }
275}
276
277impl PartialOrd for CompareCostReverse<'_> {
278    fn partial_cmp(&self, other: &Self) -> Option<std::cmp::Ordering> {
279        Some(self.cmp(other))
280    }
281}
282
283impl Eq for CompareCostReverse<'_> {}
284
285impl PartialEq for CompareCostReverse<'_> {
286    fn eq(&self, other: &Self) -> bool {
287        self.total_cost == other.total_cost
288    }
289}
290
291impl fmt::Debug for ScanMetricsSet {
292    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
293        let ScanMetricsSet {
294            prepare_scan_cost,
295            build_reader_cost,
296            scan_cost,
297            yield_cost,
298            convert_cost,
299            total_cost,
300            num_rows,
301            num_batches,
302            num_mem_ranges,
303            num_file_ranges,
304            build_parts_cost,
305            sst_scan_cost,
306            rg_total,
307            rg_fulltext_filtered,
308            rg_inverted_filtered,
309            rg_minmax_filtered,
310            rg_bloom_filtered,
311            rg_vector_filtered,
312            rows_before_filter,
313            rows_fulltext_filtered,
314            rows_inverted_filtered,
315            rows_bloom_filtered,
316            rows_vector_filtered,
317            rows_vector_selected,
318            rows_precise_filtered,
319            fulltext_index_cache_hit,
320            fulltext_index_cache_miss,
321            inverted_index_cache_hit,
322            inverted_index_cache_miss,
323            bloom_filter_cache_hit,
324            bloom_filter_cache_miss,
325            minmax_cache_hit,
326            minmax_cache_miss,
327            pruner_cache_hit,
328            pruner_cache_miss,
329            pruner_prune_cost,
330            num_sst_record_batches,
331            num_sst_batches,
332            num_sst_rows,
333            first_poll,
334            num_series_send_timeout,
335            num_series_send_full,
336            num_distributor_rows,
337            num_distributor_batches,
338            distributor_scan_cost,
339            distributor_yield_cost,
340            distributor_divider_cost,
341            merge_metrics,
342            dedup_metrics,
343            stream_eof,
344            mem_scan_cost,
345            mem_rows,
346            mem_batches,
347            mem_series,
348            mem_prefilter_cost,
349            mem_prefilter_rows_filtered,
350            inverted_index_apply_metrics,
351            bloom_filter_apply_metrics,
352            fulltext_index_apply_metrics,
353            fetch_metrics,
354            metadata_cache_metrics,
355            per_file_metrics,
356            build_ranges_mem_size: _,
357            build_ranges_peak_mem_size,
358            num_range_builders: _,
359            num_peak_range_builders,
360            range_cache_size,
361            range_cache_hit,
362            range_cache_miss,
363        } = self;
364
365        // Write core metrics
366        write!(
367            f,
368            "{{\"prepare_scan_cost\":\"{prepare_scan_cost:?}\", \
369            \"build_reader_cost\":\"{build_reader_cost:?}\", \
370            \"scan_cost\":\"{scan_cost:?}\", \
371            \"yield_cost\":\"{yield_cost:?}\", \
372            \"total_cost\":\"{total_cost:?}\", \
373            \"num_rows\":{num_rows}, \
374            \"num_batches\":{num_batches}, \
375            \"num_mem_ranges\":{num_mem_ranges}, \
376            \"num_file_ranges\":{num_file_ranges}, \
377            \"build_parts_cost\":\"{build_parts_cost:?}\", \
378            \"sst_scan_cost\":\"{sst_scan_cost:?}\", \
379            \"rg_total\":{rg_total}, \
380            \"rows_before_filter\":{rows_before_filter}, \
381            \"num_sst_record_batches\":{num_sst_record_batches}, \
382            \"num_sst_batches\":{num_sst_batches}, \
383            \"num_sst_rows\":{num_sst_rows}, \
384            \"first_poll\":\"{first_poll:?}\""
385        )?;
386
387        // Write convert_cost if present
388        if let Some(time) = convert_cost {
389            let duration = Duration::from_nanos(time.value() as u64);
390            write!(f, ", \"convert_cost\":\"{duration:?}\"")?;
391        }
392
393        // Write non-zero filter counters
394        if *rg_fulltext_filtered > 0 {
395            write!(f, ", \"rg_fulltext_filtered\":{rg_fulltext_filtered}")?;
396        }
397        if *rg_inverted_filtered > 0 {
398            write!(f, ", \"rg_inverted_filtered\":{rg_inverted_filtered}")?;
399        }
400        if *rg_minmax_filtered > 0 {
401            write!(f, ", \"rg_minmax_filtered\":{rg_minmax_filtered}")?;
402        }
403        if *rg_bloom_filtered > 0 {
404            write!(f, ", \"rg_bloom_filtered\":{rg_bloom_filtered}")?;
405        }
406        if *rg_vector_filtered > 0 {
407            write!(f, ", \"rg_vector_filtered\":{rg_vector_filtered}")?;
408        }
409        if *rows_fulltext_filtered > 0 {
410            write!(f, ", \"rows_fulltext_filtered\":{rows_fulltext_filtered}")?;
411        }
412        if *rows_inverted_filtered > 0 {
413            write!(f, ", \"rows_inverted_filtered\":{rows_inverted_filtered}")?;
414        }
415        if *rows_bloom_filtered > 0 {
416            write!(f, ", \"rows_bloom_filtered\":{rows_bloom_filtered}")?;
417        }
418        if *rows_vector_filtered > 0 {
419            write!(f, ", \"rows_vector_filtered\":{rows_vector_filtered}")?;
420        }
421        if *rows_vector_selected > 0 {
422            write!(f, ", \"rows_vector_selected\":{rows_vector_selected}")?;
423        }
424        if *rows_precise_filtered > 0 {
425            write!(f, ", \"rows_precise_filtered\":{rows_precise_filtered}")?;
426        }
427        if *fulltext_index_cache_hit > 0 {
428            write!(
429                f,
430                ", \"fulltext_index_cache_hit\":{fulltext_index_cache_hit}"
431            )?;
432        }
433        if *fulltext_index_cache_miss > 0 {
434            write!(
435                f,
436                ", \"fulltext_index_cache_miss\":{fulltext_index_cache_miss}"
437            )?;
438        }
439        if *inverted_index_cache_hit > 0 {
440            write!(
441                f,
442                ", \"inverted_index_cache_hit\":{inverted_index_cache_hit}"
443            )?;
444        }
445        if *inverted_index_cache_miss > 0 {
446            write!(
447                f,
448                ", \"inverted_index_cache_miss\":{inverted_index_cache_miss}"
449            )?;
450        }
451        if *bloom_filter_cache_hit > 0 {
452            write!(f, ", \"bloom_filter_cache_hit\":{bloom_filter_cache_hit}")?;
453        }
454        if *bloom_filter_cache_miss > 0 {
455            write!(f, ", \"bloom_filter_cache_miss\":{bloom_filter_cache_miss}")?;
456        }
457        if *minmax_cache_hit > 0 {
458            write!(f, ", \"minmax_cache_hit\":{minmax_cache_hit}")?;
459        }
460        if *minmax_cache_miss > 0 {
461            write!(f, ", \"minmax_cache_miss\":{minmax_cache_miss}")?;
462        }
463        if *pruner_cache_hit > 0 {
464            write!(f, ", \"pruner_cache_hit\":{pruner_cache_hit}")?;
465        }
466        if *pruner_cache_miss > 0 {
467            write!(f, ", \"pruner_cache_miss\":{pruner_cache_miss}")?;
468        }
469        if !pruner_prune_cost.is_zero() {
470            write!(f, ", \"pruner_prune_cost\":\"{pruner_prune_cost:?}\"")?;
471        }
472
473        // Write non-zero distributor metrics
474        if *num_series_send_timeout > 0 {
475            write!(f, ", \"num_series_send_timeout\":{num_series_send_timeout}")?;
476        }
477        if *num_series_send_full > 0 {
478            write!(f, ", \"num_series_send_full\":{num_series_send_full}")?;
479        }
480        if *num_distributor_rows > 0 {
481            write!(f, ", \"num_distributor_rows\":{num_distributor_rows}")?;
482        }
483        if *num_distributor_batches > 0 {
484            write!(f, ", \"num_distributor_batches\":{num_distributor_batches}")?;
485        }
486        if !distributor_scan_cost.is_zero() {
487            write!(
488                f,
489                ", \"distributor_scan_cost\":\"{distributor_scan_cost:?}\""
490            )?;
491        }
492        if !distributor_yield_cost.is_zero() {
493            write!(
494                f,
495                ", \"distributor_yield_cost\":\"{distributor_yield_cost:?}\""
496            )?;
497        }
498        if !distributor_divider_cost.is_zero() {
499            write!(
500                f,
501                ", \"distributor_divider_cost\":\"{distributor_divider_cost:?}\""
502            )?;
503        }
504
505        // Write non-zero memtable metrics
506        if *mem_rows > 0 {
507            write!(f, ", \"mem_rows\":{mem_rows}")?;
508        }
509        if *mem_batches > 0 {
510            write!(f, ", \"mem_batches\":{mem_batches}")?;
511        }
512        if *mem_series > 0 {
513            write!(f, ", \"mem_series\":{mem_series}")?;
514        }
515        if !mem_scan_cost.is_zero() {
516            write!(f, ", \"mem_scan_cost\":\"{mem_scan_cost:?}\"")?;
517        }
518        if !mem_prefilter_cost.is_zero() {
519            write!(f, ", \"mem_prefilter_cost\":\"{mem_prefilter_cost:?}\"")?;
520        }
521        if *mem_prefilter_rows_filtered > 0 {
522            write!(
523                f,
524                ", \"mem_prefilter_rows_filtered\":{mem_prefilter_rows_filtered}"
525            )?;
526        }
527
528        // Write optional verbose metrics if they are not empty
529        if let Some(metrics) = inverted_index_apply_metrics
530            && !metrics.is_empty()
531        {
532            write!(f, ", \"inverted_index_apply_metrics\":{:?}", metrics)?;
533        }
534        if let Some(metrics) = bloom_filter_apply_metrics
535            && !metrics.is_empty()
536        {
537            write!(f, ", \"bloom_filter_apply_metrics\":{:?}", metrics)?;
538        }
539        if let Some(metrics) = fulltext_index_apply_metrics
540            && !metrics.is_empty()
541        {
542            write!(f, ", \"fulltext_index_apply_metrics\":{:?}", metrics)?;
543        }
544        if let Some(metrics) = fetch_metrics
545            && !metrics.is_empty()
546        {
547            write!(f, ", \"fetch_metrics\":{:?}", metrics)?;
548        }
549        if let Some(metrics) = metadata_cache_metrics
550            && !metrics.is_empty()
551        {
552            write!(f, ", \"metadata_cache_metrics\":{:?}", metrics)?;
553        }
554
555        // Write merge metrics if not empty
556        if !merge_metrics.scan_cost.is_zero() {
557            write!(f, ", \"merge_metrics\":{:?}", merge_metrics)?;
558        }
559
560        // Write dedup metrics if not empty
561        if !dedup_metrics.dedup_cost.is_zero() {
562            write!(f, ", \"dedup_metrics\":{:?}", dedup_metrics)?;
563        }
564
565        // Write top file metrics if present and non-empty
566        if let Some(file_metrics) = per_file_metrics
567            && !file_metrics.is_empty()
568        {
569            // Use min-heap (BinaryHeap with reverse comparison) to keep only top 10
570            let mut heap = BinaryHeap::new();
571            for (file_id, metrics) in file_metrics.iter() {
572                let total_cost =
573                    metrics.build_part_cost + metrics.build_reader_cost + metrics.scan_cost;
574
575                // If the file has been pruned by a pruner, the build part cost may be zero.
576                // If we didn't read any ranges from it, we don't output the file.
577                if total_cost.is_zero() && metrics.num_ranges == 0 {
578                    continue;
579                }
580
581                if heap.len() < 10 {
582                    // Haven't reached 10 yet, just push
583                    heap.push(CompareCostReverse {
584                        total_cost,
585                        file_id: *file_id,
586                        metrics,
587                    });
588                } else if let Some(min_entry) = heap.peek() {
589                    // If current cost is higher than the minimum in our top-10, replace it
590                    if total_cost > min_entry.total_cost {
591                        heap.pop();
592                        heap.push(CompareCostReverse {
593                            total_cost,
594                            file_id: *file_id,
595                            metrics,
596                        });
597                    }
598                }
599            }
600
601            let top_files = heap.into_sorted_vec();
602            write!(f, ", \"top_file_metrics\": {{")?;
603            for (i, item) in top_files.iter().enumerate() {
604                let CompareCostReverse {
605                    total_cost: _,
606                    file_id,
607                    metrics,
608                } = item;
609                if i > 0 {
610                    write!(f, ", ")?;
611                }
612                write!(f, "\"{}\": {:?}", file_id, metrics)?;
613            }
614            write!(f, "}}")?;
615        }
616
617        if *range_cache_size > 0 {
618            write!(f, ", \"range_cache_size\":{range_cache_size}")?;
619        }
620        if *range_cache_hit > 0 {
621            write!(f, ", \"range_cache_hit\":{range_cache_hit}")?;
622        }
623        if *range_cache_miss > 0 {
624            write!(f, ", \"range_cache_miss\":{range_cache_miss}")?;
625        }
626
627        write!(
628            f,
629            ", \"build_ranges_peak_mem_size\":{build_ranges_peak_mem_size}, \
630             \"num_peak_range_builders\":{num_peak_range_builders}, \
631             \"stream_eof\":{stream_eof}}}"
632        )
633    }
634}
635impl ScanMetricsSet {
636    /// Attaches the `prepare_scan_cost` to the metrics set.
637    fn with_prepare_scan_cost(mut self, cost: Duration) -> Self {
638        self.prepare_scan_cost += cost;
639        self
640    }
641
642    /// Attaches the `convert_cost` to the metrics set.
643    fn with_convert_cost(mut self, time: Time) -> Self {
644        self.convert_cost = Some(time);
645        self
646    }
647
648    /// Merges the local scanner metrics.
649    fn merge_scanner_metrics(&mut self, other: &ScannerMetrics) {
650        let ScannerMetrics {
651            scan_cost,
652            yield_cost,
653            num_batches,
654            num_rows,
655        } = other;
656
657        self.scan_cost += *scan_cost;
658        self.yield_cost += *yield_cost;
659        self.num_rows += *num_rows;
660        self.num_batches += *num_batches;
661    }
662
663    /// Merges the local reader metrics.
664    fn merge_reader_metrics(&mut self, other: &ReaderMetrics) {
665        let ReaderMetrics {
666            build_cost,
667            filter_metrics:
668                ReaderFilterMetrics {
669                    rg_total,
670                    rg_fulltext_filtered,
671                    rg_inverted_filtered,
672                    rg_minmax_filtered,
673                    rg_bloom_filtered,
674                    rg_vector_filtered,
675                    rows_total,
676                    rows_fulltext_filtered,
677                    rows_inverted_filtered,
678                    rows_bloom_filtered,
679                    rows_vector_filtered,
680                    rows_vector_selected,
681                    rows_precise_filtered,
682                    fulltext_index_cache_hit,
683                    fulltext_index_cache_miss,
684                    inverted_index_cache_hit,
685                    inverted_index_cache_miss,
686                    bloom_filter_cache_hit,
687                    bloom_filter_cache_miss,
688                    minmax_cache_hit,
689                    minmax_cache_miss,
690                    pruner_cache_hit,
691                    pruner_cache_miss,
692                    pruner_prune_cost,
693                    inverted_index_apply_metrics,
694                    bloom_filter_apply_metrics,
695                    fulltext_index_apply_metrics,
696                },
697            num_record_batches,
698            num_batches,
699            num_rows,
700            scan_cost,
701            metadata_cache_metrics,
702            fetch_metrics,
703            metadata_mem_size,
704            num_range_builders,
705        } = other;
706
707        self.build_parts_cost += *build_cost;
708        self.sst_scan_cost += *scan_cost;
709
710        self.rg_total += *rg_total;
711        self.rg_fulltext_filtered += *rg_fulltext_filtered;
712        self.rg_inverted_filtered += *rg_inverted_filtered;
713        self.rg_minmax_filtered += *rg_minmax_filtered;
714        self.rg_bloom_filtered += *rg_bloom_filtered;
715        self.rg_vector_filtered += *rg_vector_filtered;
716
717        self.rows_before_filter += *rows_total;
718        self.rows_fulltext_filtered += *rows_fulltext_filtered;
719        self.rows_inverted_filtered += *rows_inverted_filtered;
720        self.rows_bloom_filtered += *rows_bloom_filtered;
721        self.rows_vector_filtered += *rows_vector_filtered;
722        self.rows_vector_selected += *rows_vector_selected;
723        self.rows_precise_filtered += *rows_precise_filtered;
724
725        self.fulltext_index_cache_hit += *fulltext_index_cache_hit;
726        self.fulltext_index_cache_miss += *fulltext_index_cache_miss;
727        self.inverted_index_cache_hit += *inverted_index_cache_hit;
728        self.inverted_index_cache_miss += *inverted_index_cache_miss;
729        self.bloom_filter_cache_hit += *bloom_filter_cache_hit;
730        self.bloom_filter_cache_miss += *bloom_filter_cache_miss;
731        self.minmax_cache_hit += *minmax_cache_hit;
732        self.minmax_cache_miss += *minmax_cache_miss;
733        self.pruner_cache_hit += *pruner_cache_hit;
734        self.pruner_cache_miss += *pruner_cache_miss;
735        self.pruner_prune_cost += *pruner_prune_cost;
736
737        self.num_sst_record_batches += *num_record_batches;
738        self.num_sst_batches += *num_batches;
739        self.num_sst_rows += *num_rows;
740
741        // Merge optional verbose metrics
742        if let Some(metrics) = inverted_index_apply_metrics {
743            self.inverted_index_apply_metrics
744                .get_or_insert_with(InvertedIndexApplyMetrics::default)
745                .merge_from(metrics);
746        }
747        if let Some(metrics) = bloom_filter_apply_metrics {
748            self.bloom_filter_apply_metrics
749                .get_or_insert_with(BloomFilterIndexApplyMetrics::default)
750                .merge_from(metrics);
751        }
752        if let Some(metrics) = fulltext_index_apply_metrics {
753            self.fulltext_index_apply_metrics
754                .get_or_insert_with(FulltextIndexApplyMetrics::default)
755                .merge_from(metrics);
756        }
757        if let Some(metrics) = fetch_metrics {
758            self.fetch_metrics
759                .get_or_insert_with(ParquetFetchMetrics::default)
760                .merge_from(metrics);
761        }
762        self.metadata_cache_metrics
763            .get_or_insert_with(MetadataCacheMetrics::default)
764            .merge_from(metadata_cache_metrics);
765
766        // Track memory usage and update peak.
767        self.build_ranges_mem_size += *metadata_mem_size;
768        if self.build_ranges_mem_size > self.build_ranges_peak_mem_size {
769            self.build_ranges_peak_mem_size = self.build_ranges_mem_size;
770        }
771
772        // Track number of builders and update peak.
773        self.num_range_builders += *num_range_builders;
774        if self.num_range_builders > self.num_peak_range_builders {
775            self.num_peak_range_builders = self.num_range_builders;
776        }
777    }
778
779    /// Merges per-file metrics.
780    fn merge_per_file_metrics(&mut self, other: &HashMap<RegionFileId, FileScanMetrics>) {
781        let self_file_metrics = self.per_file_metrics.get_or_insert_with(HashMap::new);
782        for (file_id, metrics) in other {
783            self_file_metrics
784                .entry(*file_id)
785                .or_default()
786                .merge_from(metrics);
787        }
788    }
789
790    /// Sets distributor metrics.
791    fn set_distributor_metrics(&mut self, distributor_metrics: &SeriesDistributorMetrics) {
792        let SeriesDistributorMetrics {
793            num_series_send_timeout,
794            num_series_send_full,
795            num_rows,
796            num_batches,
797            scan_cost,
798            yield_cost,
799            divider_cost,
800        } = distributor_metrics;
801
802        self.num_series_send_timeout += *num_series_send_timeout;
803        self.num_series_send_full += *num_series_send_full;
804        self.num_distributor_rows += *num_rows;
805        self.num_distributor_batches += *num_batches;
806        self.distributor_scan_cost += *scan_cost;
807        self.distributor_yield_cost += *yield_cost;
808        self.distributor_divider_cost += *divider_cost;
809    }
810
811    /// Observes metrics.
812    fn observe_metrics(&self) {
813        READ_STAGE_ELAPSED
814            .with_label_values(&["prepare_scan"])
815            .observe(self.prepare_scan_cost.as_secs_f64());
816        READ_STAGE_ELAPSED
817            .with_label_values(&["build_reader"])
818            .observe(self.build_reader_cost.as_secs_f64());
819        READ_STAGE_ELAPSED
820            .with_label_values(&["scan"])
821            .observe(self.scan_cost.as_secs_f64());
822        READ_STAGE_ELAPSED
823            .with_label_values(&["yield"])
824            .observe(self.yield_cost.as_secs_f64());
825        if let Some(time) = &self.convert_cost {
826            READ_STAGE_ELAPSED
827                .with_label_values(&["convert"])
828                .observe(Duration::from_nanos(time.value() as u64).as_secs_f64());
829        }
830        READ_STAGE_ELAPSED
831            .with_label_values(&["total"])
832            .observe(self.total_cost.as_secs_f64());
833        READ_ROWS_RETURN.observe(self.num_rows as f64);
834        READ_BATCHES_RETURN.observe(self.num_batches as f64);
835
836        READ_STAGE_ELAPSED
837            .with_label_values(&["build_parts"])
838            .observe(self.build_parts_cost.as_secs_f64());
839
840        READ_ROW_GROUPS_TOTAL
841            .with_label_values(&["before_filtering"])
842            .inc_by(self.rg_total as u64);
843        READ_ROW_GROUPS_TOTAL
844            .with_label_values(&["fulltext_index_filtered"])
845            .inc_by(self.rg_fulltext_filtered as u64);
846        READ_ROW_GROUPS_TOTAL
847            .with_label_values(&["inverted_index_filtered"])
848            .inc_by(self.rg_inverted_filtered as u64);
849        READ_ROW_GROUPS_TOTAL
850            .with_label_values(&["minmax_index_filtered"])
851            .inc_by(self.rg_minmax_filtered as u64);
852        READ_ROW_GROUPS_TOTAL
853            .with_label_values(&["bloom_filter_index_filtered"])
854            .inc_by(self.rg_bloom_filtered as u64);
855        #[cfg(feature = "vector_index")]
856        READ_ROW_GROUPS_TOTAL
857            .with_label_values(&["vector_index_filtered"])
858            .inc_by(self.rg_vector_filtered as u64);
859
860        PRECISE_FILTER_ROWS_TOTAL
861            .with_label_values(&["parquet"])
862            .inc_by(self.rows_precise_filtered as u64);
863        READ_ROWS_IN_ROW_GROUP_TOTAL
864            .with_label_values(&["before_filtering"])
865            .inc_by(self.rows_before_filter as u64);
866        READ_ROWS_IN_ROW_GROUP_TOTAL
867            .with_label_values(&["fulltext_index_filtered"])
868            .inc_by(self.rows_fulltext_filtered as u64);
869        READ_ROWS_IN_ROW_GROUP_TOTAL
870            .with_label_values(&["inverted_index_filtered"])
871            .inc_by(self.rows_inverted_filtered as u64);
872        READ_ROWS_IN_ROW_GROUP_TOTAL
873            .with_label_values(&["bloom_filter_index_filtered"])
874            .inc_by(self.rows_bloom_filtered as u64);
875        #[cfg(feature = "vector_index")]
876        READ_ROWS_IN_ROW_GROUP_TOTAL
877            .with_label_values(&["vector_index_filtered"])
878            .inc_by(self.rows_vector_filtered as u64);
879    }
880}
881
882struct PartitionMetricsInner {
883    region_id: RegionId,
884    /// Index of the partition to scan.
885    partition: usize,
886    /// Label to distinguish different scan operation.
887    scanner_type: &'static str,
888    /// Query start time.
889    query_start: Instant,
890    /// Whether to use verbose logging.
891    explain_verbose: bool,
892    /// Verbose scan metrics that only log to debug logs by default.
893    metrics: Mutex<ScanMetricsSet>,
894    in_progress_scan: IntGauge,
895
896    // Normal metrics that always report to the [ExecutionPlanMetricsSet]:
897    /// Duration to build file ranges.
898    build_parts_cost: Time,
899    /// Duration to build the (merge) reader.
900    build_reader_cost: Time,
901    /// Duration to scan data.
902    scan_cost: Time,
903    /// Duration while waiting for `yield`.
904    yield_cost: Time,
905    /// Duration to convert [`Batch`]es.
906    convert_cost: Time,
907    /// Aggregated compute time reported to DataFusion.
908    elapsed_compute: Time,
909}
910
911impl PartitionMetricsInner {
912    fn on_finish(&self, stream_eof: bool) {
913        let mut metrics = self.metrics.lock().unwrap();
914        if metrics.total_cost.is_zero() {
915            metrics.total_cost = self.query_start.elapsed();
916        }
917        if !metrics.stream_eof {
918            metrics.stream_eof = stream_eof;
919        }
920    }
921}
922
923impl MergeMetricsReport for PartitionMetricsInner {
924    fn report(&self, metrics: &mut MergeMetrics) {
925        let mut scan_metrics = self.metrics.lock().unwrap();
926        // Merge the metrics into scan_metrics
927        scan_metrics.merge_metrics.merge(metrics);
928
929        // Reset the input metrics
930        *metrics = MergeMetrics::default();
931    }
932}
933
934impl DedupMetricsReport for PartitionMetricsInner {
935    fn report(&self, metrics: &mut DedupMetrics) {
936        let mut scan_metrics = self.metrics.lock().unwrap();
937        // Merge the metrics into scan_metrics
938        scan_metrics.dedup_metrics.merge(metrics);
939
940        // Reset the input metrics
941        *metrics = DedupMetrics::default();
942    }
943}
944
945impl Drop for PartitionMetricsInner {
946    fn drop(&mut self) {
947        self.on_finish(false);
948        let metrics = self.metrics.lock().unwrap();
949        metrics.observe_metrics();
950        self.in_progress_scan.dec();
951
952        if self.explain_verbose {
953            common_telemetry::info!(
954                "{} finished, region_id: {}, partition: {}, scan_metrics: {:?}",
955                self.scanner_type,
956                self.region_id,
957                self.partition,
958                metrics,
959            );
960        } else {
961            common_telemetry::debug!(
962                "{} finished, region_id: {}, partition: {}, scan_metrics: {:?}",
963                self.scanner_type,
964                self.region_id,
965                self.partition,
966                metrics,
967            );
968        }
969    }
970}
971
972/// List of PartitionMetrics.
973#[derive(Default)]
974pub(crate) struct PartitionMetricsList(Mutex<Vec<Option<PartitionMetrics>>>);
975
976impl PartitionMetricsList {
977    /// Sets a new [PartitionMetrics] at the specified partition.
978    pub(crate) fn set(&self, partition: usize, metrics: PartitionMetrics) {
979        let mut list = self.0.lock().unwrap();
980        if list.len() <= partition {
981            list.resize(partition + 1, None);
982        }
983        list[partition] = Some(metrics);
984    }
985
986    /// Format verbose metrics for each partition for explain.
987    pub(crate) fn format_verbose_metrics(&self, f: &mut fmt::Formatter) -> fmt::Result {
988        let list = self.0.lock().unwrap();
989        write!(f, ", \"metrics_per_partition\": ")?;
990        f.debug_list()
991            .entries(list.iter().filter_map(|p| p.as_ref()))
992            .finish()?;
993        write!(f, "}}")
994    }
995}
996
997/// Metrics while reading a partition.
998#[derive(Clone)]
999pub struct PartitionMetrics(Arc<PartitionMetricsInner>);
1000
1001impl PartitionMetrics {
1002    pub(crate) fn new(
1003        region_id: RegionId,
1004        partition: usize,
1005        scanner_type: &'static str,
1006        query_start: Instant,
1007        explain_verbose: bool,
1008        metrics_set: &ExecutionPlanMetricsSet,
1009    ) -> Self {
1010        let partition_str = partition.to_string();
1011        let in_progress_scan = IN_PROGRESS_SCAN.with_label_values(&[scanner_type, &partition_str]);
1012        in_progress_scan.inc();
1013        let convert_cost = MetricBuilder::new(metrics_set).subset_time("convert_cost", partition);
1014        let metrics = ScanMetricsSet::default()
1015            .with_prepare_scan_cost(query_start.elapsed())
1016            .with_convert_cost(convert_cost.clone());
1017        let inner = PartitionMetricsInner {
1018            region_id,
1019            partition,
1020            scanner_type,
1021            query_start,
1022            explain_verbose,
1023            metrics: Mutex::new(metrics),
1024            in_progress_scan,
1025            build_parts_cost: MetricBuilder::new(metrics_set)
1026                .subset_time("build_parts_cost", partition),
1027            build_reader_cost: MetricBuilder::new(metrics_set)
1028                .subset_time("build_reader_cost", partition),
1029            scan_cost: MetricBuilder::new(metrics_set).subset_time("scan_cost", partition),
1030            yield_cost: MetricBuilder::new(metrics_set).subset_time("yield_cost", partition),
1031            convert_cost,
1032            elapsed_compute: MetricBuilder::new(metrics_set).elapsed_compute(partition),
1033        };
1034        Self(Arc::new(inner))
1035    }
1036
1037    pub(crate) fn on_first_poll(&self) {
1038        let mut metrics = self.0.metrics.lock().unwrap();
1039        metrics.first_poll = self.0.query_start.elapsed();
1040    }
1041
1042    pub(crate) fn inc_num_mem_ranges(&self, num: usize) {
1043        let mut metrics = self.0.metrics.lock().unwrap();
1044        metrics.num_mem_ranges += num;
1045    }
1046
1047    pub fn inc_num_file_ranges(&self, num: usize) {
1048        let mut metrics = self.0.metrics.lock().unwrap();
1049        metrics.num_file_ranges += num;
1050    }
1051
1052    fn record_elapsed_compute(&self, duration: Duration) {
1053        if duration.is_zero() {
1054            return;
1055        }
1056        self.0.elapsed_compute.add_duration(duration);
1057    }
1058
1059    /// Merges `build_reader_cost`.
1060    pub(crate) fn inc_build_reader_cost(&self, cost: Duration) {
1061        self.0.build_reader_cost.add_duration(cost);
1062
1063        let mut metrics = self.0.metrics.lock().unwrap();
1064        metrics.build_reader_cost += cost;
1065    }
1066
1067    pub(crate) fn inc_convert_batch_cost(&self, cost: Duration) {
1068        self.0.convert_cost.add_duration(cost);
1069        self.record_elapsed_compute(cost);
1070    }
1071
1072    /// Reports memtable scan metrics.
1073    pub(crate) fn report_mem_scan_metrics(&self, data: &crate::memtable::MemScanMetricsData) {
1074        let mut metrics = self.0.metrics.lock().unwrap();
1075        metrics.mem_scan_cost += data.scan_cost;
1076        metrics.mem_rows += data.num_rows;
1077        metrics.mem_batches += data.num_batches;
1078        metrics.mem_series += data.total_series;
1079        metrics.mem_prefilter_cost += data.prefilter_cost;
1080        metrics.mem_prefilter_rows_filtered += data.prefilter_rows_filtered;
1081    }
1082
1083    /// Merges [ScannerMetrics], `build_reader_cost`, `scan_cost` and `yield_cost`.
1084    pub(crate) fn merge_metrics(&self, metrics: &ScannerMetrics) {
1085        self.0.scan_cost.add_duration(metrics.scan_cost);
1086        self.record_elapsed_compute(metrics.scan_cost);
1087        self.0.yield_cost.add_duration(metrics.yield_cost);
1088        self.record_elapsed_compute(metrics.yield_cost);
1089
1090        let mut metrics_set = self.0.metrics.lock().unwrap();
1091        metrics_set.merge_scanner_metrics(metrics);
1092    }
1093
1094    /// Merges [ReaderMetrics] and `build_reader_cost`.
1095    pub fn merge_reader_metrics(
1096        &self,
1097        metrics: &ReaderMetrics,
1098        per_file_metrics: Option<&HashMap<RegionFileId, FileScanMetrics>>,
1099    ) {
1100        self.0.build_parts_cost.add_duration(metrics.build_cost);
1101
1102        let mut metrics_set = self.0.metrics.lock().unwrap();
1103        metrics_set.merge_reader_metrics(metrics);
1104
1105        // Merge per-file metrics if provided
1106        if let Some(file_metrics) = per_file_metrics {
1107            metrics_set.merge_per_file_metrics(file_metrics);
1108        }
1109    }
1110
1111    /// Finishes the query.
1112    pub(crate) fn on_finish(&self) {
1113        self.0.on_finish(true);
1114    }
1115
1116    /// Sets the distributor metrics.
1117    pub(crate) fn set_distributor_metrics(&self, metrics: &SeriesDistributorMetrics) {
1118        let mut metrics_set = self.0.metrics.lock().unwrap();
1119        metrics_set.set_distributor_metrics(metrics);
1120    }
1121
1122    /// Returns whether verbose explain is enabled.
1123    pub(crate) fn explain_verbose(&self) -> bool {
1124        self.0.explain_verbose
1125    }
1126
1127    /// Returns a MergeMetricsReport trait object for reporting merge metrics.
1128    pub(crate) fn merge_metrics_reporter(&self) -> Arc<dyn MergeMetricsReport> {
1129        self.0.clone()
1130    }
1131
1132    /// Returns a DedupMetricsReport trait object for reporting dedup metrics.
1133    pub(crate) fn dedup_metrics_reporter(&self) -> Arc<dyn DedupMetricsReport> {
1134        self.0.clone()
1135    }
1136
1137    /// Increments the total bytes added to the range cache.
1138    #[allow(dead_code)]
1139    pub(crate) fn inc_range_cache_size(&self, size: usize) {
1140        let mut metrics = self.0.metrics.lock().unwrap();
1141        metrics.range_cache_size += size;
1142    }
1143
1144    /// Increments the range cache hit counter.
1145    #[allow(dead_code)]
1146    pub(crate) fn inc_range_cache_hit(&self) {
1147        let mut metrics = self.0.metrics.lock().unwrap();
1148        metrics.range_cache_hit += 1;
1149    }
1150
1151    /// Increments the range cache miss counter.
1152    #[allow(dead_code)]
1153    pub(crate) fn inc_range_cache_miss(&self) {
1154        let mut metrics = self.0.metrics.lock().unwrap();
1155        metrics.range_cache_miss += 1;
1156    }
1157}
1158
1159impl fmt::Debug for PartitionMetrics {
1160    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
1161        let metrics = self.0.metrics.lock().unwrap();
1162        write!(
1163            f,
1164            r#"{{"partition":{}, "metrics":{:?}}}"#,
1165            self.0.partition, metrics
1166        )
1167    }
1168}
1169
1170/// Metrics for the series distributor.
1171#[derive(Default)]
1172pub(crate) struct SeriesDistributorMetrics {
1173    /// Number of send timeout in SeriesScan.
1174    pub(crate) num_series_send_timeout: usize,
1175    /// Number of send full in SeriesScan.
1176    pub(crate) num_series_send_full: usize,
1177    /// Number of rows the series distributor scanned.
1178    pub(crate) num_rows: usize,
1179    /// Number of batches the series distributor scanned.
1180    pub(crate) num_batches: usize,
1181    /// Duration of the series distributor to scan.
1182    pub(crate) scan_cost: Duration,
1183    /// Duration of the series distributor to yield.
1184    pub(crate) yield_cost: Duration,
1185    /// Duration spent in divider operations.
1186    pub(crate) divider_cost: Duration,
1187}
1188
1189/// Scans memtable ranges at `index`.
1190#[tracing::instrument(
1191    skip_all,
1192    fields(
1193        region_id = %stream_ctx.input.region_metadata().region_id,
1194        file_or_mem_index = %index.index,
1195        row_group_index = %index.row_group_index,
1196        source = "mem"
1197    )
1198)]
1199pub(crate) fn scan_mem_ranges(
1200    stream_ctx: Arc<StreamContext>,
1201    part_metrics: PartitionMetrics,
1202    index: RowGroupIndex,
1203    time_range: FileTimeRange,
1204) -> impl Stream<Item = Result<Batch>> {
1205    try_stream! {
1206        let ranges = stream_ctx.input.build_mem_ranges(index);
1207        part_metrics.inc_num_mem_ranges(ranges.len());
1208        for range in ranges {
1209            let build_reader_start = Instant::now();
1210            let mem_scan_metrics = Some(MemScanMetrics::default());
1211            let iter = range.build_prune_iter(time_range, mem_scan_metrics.clone())?;
1212            part_metrics.inc_build_reader_cost(build_reader_start.elapsed());
1213
1214            let mut source = Source::Iter(iter);
1215            while let Some(batch) = source.next_batch().await? {
1216                yield batch;
1217            }
1218
1219            // Report the memtable scan metrics to partition metrics
1220            if let Some(ref metrics) = mem_scan_metrics {
1221                let data = metrics.data();
1222                part_metrics.report_mem_scan_metrics(&data);
1223            }
1224        }
1225    }
1226}
1227
1228/// Scans memtable ranges at `index` using flat format that returns RecordBatch.
1229#[tracing::instrument(
1230    skip_all,
1231    fields(
1232        region_id = %stream_ctx.input.region_metadata().region_id,
1233        row_group_index = %index.index,
1234        source = "mem_flat"
1235    )
1236)]
1237pub(crate) fn scan_flat_mem_ranges(
1238    stream_ctx: Arc<StreamContext>,
1239    part_metrics: PartitionMetrics,
1240    index: RowGroupIndex,
1241    time_range: FileTimeRange,
1242) -> impl Stream<Item = Result<RecordBatch>> {
1243    try_stream! {
1244        let ranges = stream_ctx.input.build_mem_ranges(index);
1245        part_metrics.inc_num_mem_ranges(ranges.len());
1246        for range in ranges {
1247            let build_reader_start = Instant::now();
1248            let mem_scan_metrics = Some(MemScanMetrics::default());
1249            let mut iter = range.build_record_batch_iter(Some(time_range), mem_scan_metrics.clone())?;
1250            part_metrics.inc_build_reader_cost(build_reader_start.elapsed());
1251
1252            while let Some(record_batch) = iter.next().transpose()? {
1253                yield record_batch;
1254            }
1255
1256            // Report the memtable scan metrics to partition metrics
1257            if let Some(ref metrics) = mem_scan_metrics {
1258                let data = metrics.data();
1259                part_metrics.report_mem_scan_metrics(&data);
1260            }
1261        }
1262    }
1263}
1264
1265/// Files with row count greater than this threshold can contribute to the estimation.
1266const SPLIT_ROW_THRESHOLD: u64 = DEFAULT_ROW_GROUP_SIZE as u64;
1267/// Number of series threshold for splitting batches.
1268const NUM_SERIES_THRESHOLD: u64 = 10240;
1269/// Minimum batch size after splitting. The batch size is less than 60 because a series may only have
1270/// 60 samples per hour.
1271const BATCH_SIZE_THRESHOLD: u64 = 50;
1272
1273/// Returns true if splitting flat record batches may improve merge performance.
1274pub(crate) fn should_split_flat_batches_for_merge(
1275    stream_ctx: &Arc<StreamContext>,
1276    range_meta: &RangeMeta,
1277) -> bool {
1278    // Number of files to split and scan.
1279    let mut num_files_to_split = 0;
1280    let mut num_mem_rows = 0;
1281    let mut num_mem_series = 0;
1282    // Checks each file range, returns early if any range is not splittable.
1283    // For mem ranges, we collect the total number of rows and series because the number of rows in a
1284    // mem range may be too small.
1285    for index in &range_meta.row_group_indices {
1286        if stream_ctx.is_mem_range_index(*index) {
1287            let memtable = &stream_ctx.input.memtables[index.index];
1288            // Is mem range
1289            let stats = memtable.stats();
1290            num_mem_rows += stats.num_rows();
1291            num_mem_series += stats.series_count();
1292        } else if stream_ctx.is_file_range_index(*index) {
1293            // This is a file range.
1294            let file_index = index.index - stream_ctx.input.num_memtables();
1295            let file = &stream_ctx.input.files[file_index];
1296            if file.meta_ref().num_rows < SPLIT_ROW_THRESHOLD || file.meta_ref().num_series == 0 {
1297                // If the file doesn't have enough rows, or the number of series is unavailable, skips it.
1298                continue;
1299            }
1300            debug_assert!(file.meta_ref().num_rows > 0);
1301            if !can_split_series(file.meta_ref().num_rows, file.meta_ref().num_series) {
1302                // We can't split batches in a file.
1303                return false;
1304            } else {
1305                num_files_to_split += 1;
1306            }
1307        }
1308        // Skips non-file and non-mem ranges.
1309    }
1310
1311    if num_files_to_split > 0 {
1312        // We mainly consider file ranges because they have enough data for sampling.
1313        true
1314    } else if num_mem_series > 0 && num_mem_rows > 0 {
1315        // If we don't have files to scan, we check whether to split by the memtable.
1316        can_split_series(num_mem_rows as u64, num_mem_series as u64)
1317    } else {
1318        false
1319    }
1320}
1321
1322fn can_split_series(num_rows: u64, num_series: u64) -> bool {
1323    assert!(num_series > 0);
1324    assert!(num_rows > 0);
1325
1326    // It doesn't have too many series or it will have enough rows for each batch.
1327    num_series < NUM_SERIES_THRESHOLD || num_rows / num_series >= BATCH_SIZE_THRESHOLD
1328}
1329
1330/// Creates a new [ReaderFilterMetrics] with optional apply metrics initialized
1331/// based on the `explain_verbose` flag.
1332fn new_filter_metrics(explain_verbose: bool) -> ReaderFilterMetrics {
1333    if explain_verbose {
1334        ReaderFilterMetrics {
1335            inverted_index_apply_metrics: Some(InvertedIndexApplyMetrics::default()),
1336            bloom_filter_apply_metrics: Some(BloomFilterIndexApplyMetrics::default()),
1337            fulltext_index_apply_metrics: Some(FulltextIndexApplyMetrics::default()),
1338            ..Default::default()
1339        }
1340    } else {
1341        ReaderFilterMetrics::default()
1342    }
1343}
1344
1345/// Scans file ranges at `index`.
1346#[tracing::instrument(
1347    skip_all,
1348    fields(
1349        region_id = %stream_ctx.input.region_metadata().region_id,
1350        row_group_index = %index.index,
1351        source = read_type
1352    )
1353)]
1354pub(crate) async fn scan_file_ranges(
1355    stream_ctx: Arc<StreamContext>,
1356    part_metrics: PartitionMetrics,
1357    index: RowGroupIndex,
1358    read_type: &'static str,
1359    partition_pruner: Arc<PartitionPruner>,
1360) -> Result<impl Stream<Item = Result<Batch>>> {
1361    let mut reader_metrics = ReaderMetrics {
1362        filter_metrics: new_filter_metrics(part_metrics.explain_verbose()),
1363        ..Default::default()
1364    };
1365    let ranges = partition_pruner
1366        .build_file_ranges(index, &part_metrics, &mut reader_metrics)
1367        .await?;
1368    part_metrics.inc_num_file_ranges(ranges.len());
1369    part_metrics.merge_reader_metrics(&reader_metrics, None);
1370
1371    // Creates initial per-file metrics with build_part_cost.
1372    let init_per_file_metrics = if part_metrics.explain_verbose() {
1373        let file = stream_ctx.input.file_from_index(index);
1374        let file_id = file.file_id();
1375
1376        let mut map = HashMap::new();
1377        map.insert(
1378            file_id,
1379            FileScanMetrics {
1380                build_part_cost: reader_metrics.build_cost,
1381                ..Default::default()
1382            },
1383        );
1384        Some(map)
1385    } else {
1386        None
1387    };
1388
1389    Ok(build_file_range_scan_stream(
1390        stream_ctx,
1391        part_metrics,
1392        read_type,
1393        ranges,
1394        init_per_file_metrics,
1395    ))
1396}
1397
1398/// Scans file ranges at `index` using flat reader that returns RecordBatch.
1399#[tracing::instrument(
1400    skip_all,
1401    fields(
1402        region_id = %stream_ctx.input.region_metadata().region_id,
1403        row_group_index = %index.index,
1404        source = read_type
1405    )
1406)]
1407pub(crate) async fn scan_flat_file_ranges(
1408    stream_ctx: Arc<StreamContext>,
1409    part_metrics: PartitionMetrics,
1410    index: RowGroupIndex,
1411    read_type: &'static str,
1412    partition_pruner: Arc<PartitionPruner>,
1413) -> Result<impl Stream<Item = Result<RecordBatch>>> {
1414    let mut reader_metrics = ReaderMetrics {
1415        filter_metrics: new_filter_metrics(part_metrics.explain_verbose()),
1416        ..Default::default()
1417    };
1418    let ranges = partition_pruner
1419        .build_file_ranges(index, &part_metrics, &mut reader_metrics)
1420        .await?;
1421    part_metrics.inc_num_file_ranges(ranges.len());
1422    part_metrics.merge_reader_metrics(&reader_metrics, None);
1423
1424    // Creates initial per-file metrics with build_part_cost.
1425    let init_per_file_metrics = if part_metrics.explain_verbose() {
1426        let file = stream_ctx.input.file_from_index(index);
1427        let file_id = file.file_id();
1428
1429        let mut map = HashMap::new();
1430        map.insert(
1431            file_id,
1432            FileScanMetrics {
1433                build_part_cost: reader_metrics.build_cost,
1434                ..Default::default()
1435            },
1436        );
1437        Some(map)
1438    } else {
1439        None
1440    };
1441
1442    Ok(build_flat_file_range_scan_stream(
1443        stream_ctx,
1444        part_metrics,
1445        read_type,
1446        ranges,
1447        init_per_file_metrics,
1448    ))
1449}
1450
1451/// Build the stream of scanning the input [`FileRange`]s.
1452#[tracing::instrument(
1453    skip_all,
1454    fields(read_type = read_type, range_count = ranges.len())
1455)]
1456pub fn build_file_range_scan_stream(
1457    stream_ctx: Arc<StreamContext>,
1458    part_metrics: PartitionMetrics,
1459    read_type: &'static str,
1460    ranges: SmallVec<[FileRange; 2]>,
1461    mut per_file_metrics: Option<HashMap<RegionFileId, FileScanMetrics>>,
1462) -> impl Stream<Item = Result<Batch>> {
1463    try_stream! {
1464        let fetch_metrics = if part_metrics.explain_verbose() {
1465            Some(Arc::new(ParquetFetchMetrics::default()))
1466        } else {
1467            None
1468        };
1469        let reader_metrics = &mut ReaderMetrics {
1470            fetch_metrics: fetch_metrics.clone(),
1471            ..Default::default()
1472        };
1473        for range in ranges {
1474            let build_reader_start = Instant::now();
1475            let Some(reader) = range.reader(stream_ctx.input.series_row_selector, fetch_metrics.as_deref()).await? else {
1476                continue;
1477            };
1478            let build_cost = build_reader_start.elapsed();
1479            part_metrics.inc_build_reader_cost(build_cost);
1480            let compat_batch = range.compat_batch();
1481            let mut source = Source::PruneReader(reader);
1482            while let Some(mut batch) = source.next_batch().await? {
1483                if let Some(compact_batch) = compat_batch {
1484                    batch = compact_batch.as_primary_key().unwrap().compat_batch(batch)?;
1485                }
1486                yield batch;
1487            }
1488            if let Source::PruneReader(reader) = source {
1489                let prune_metrics = reader.metrics();
1490
1491                // Update per-file metrics if tracking is enabled
1492                if let Some(file_metrics_map) = per_file_metrics.as_mut() {
1493                    let file_id = range.file_handle().file_id();
1494                    let file_metrics = file_metrics_map
1495                        .entry(file_id)
1496                        .or_insert_with(FileScanMetrics::default);
1497
1498                    file_metrics.num_ranges += 1;
1499                    file_metrics.num_rows += prune_metrics.num_rows;
1500                    file_metrics.build_reader_cost += build_cost;
1501                    file_metrics.scan_cost += prune_metrics.scan_cost;
1502                }
1503
1504                reader_metrics.merge_from(&prune_metrics);
1505            }
1506        }
1507
1508        // Reports metrics.
1509        reader_metrics.observe_rows(read_type);
1510        reader_metrics.filter_metrics.observe();
1511        part_metrics.merge_reader_metrics(reader_metrics, per_file_metrics.as_ref());
1512    }
1513}
1514
1515/// Build the stream of scanning the input [`FileRange`]s using flat reader that returns RecordBatch.
1516#[tracing::instrument(
1517    skip_all,
1518    fields(read_type = read_type, range_count = ranges.len())
1519)]
1520pub fn build_flat_file_range_scan_stream(
1521    _stream_ctx: Arc<StreamContext>,
1522    part_metrics: PartitionMetrics,
1523    read_type: &'static str,
1524    ranges: SmallVec<[FileRange; 2]>,
1525    mut per_file_metrics: Option<HashMap<RegionFileId, FileScanMetrics>>,
1526) -> impl Stream<Item = Result<RecordBatch>> {
1527    try_stream! {
1528        let fetch_metrics = if part_metrics.explain_verbose() {
1529            Some(Arc::new(ParquetFetchMetrics::default()))
1530        } else {
1531            None
1532        };
1533        let reader_metrics = &mut ReaderMetrics {
1534            fetch_metrics: fetch_metrics.clone(),
1535            ..Default::default()
1536        };
1537        for range in ranges {
1538            let build_reader_start = Instant::now();
1539            let Some(mut reader) = range.flat_reader(_stream_ctx.input.series_row_selector, fetch_metrics.as_deref()).await? else{continue};
1540            let build_cost = build_reader_start.elapsed();
1541            part_metrics.inc_build_reader_cost(build_cost);
1542
1543            let may_compat = range
1544                .compat_batch()
1545                .map(|compat| {
1546                    compat.as_flat().context(UnexpectedSnafu {
1547                        reason: "Invalid compat for flat format",
1548                    })
1549                })
1550                .transpose()?;
1551
1552            let mapper = range.compaction_projection_mapper();
1553            while let Some(record_batch) = reader.next_batch().await? {
1554                let record_batch = if let Some(mapper) = mapper {
1555                    let batch = mapper.project(record_batch)?;
1556                    batch
1557                } else {
1558                    record_batch
1559                };
1560
1561                if let Some(flat_compat) = may_compat {
1562                    let batch = flat_compat.compat(record_batch)?;
1563                    yield batch;
1564                } else {
1565                    yield record_batch;
1566                }
1567            }
1568
1569            let prune_metrics = reader.metrics();
1570
1571            // Update per-file metrics if tracking is enabled
1572            if let Some(file_metrics_map) = per_file_metrics.as_mut() {
1573                let file_id = range.file_handle().file_id();
1574                let file_metrics = file_metrics_map
1575                    .entry(file_id)
1576                    .or_insert_with(FileScanMetrics::default);
1577
1578                file_metrics.num_ranges += 1;
1579                file_metrics.num_rows += prune_metrics.num_rows;
1580                file_metrics.build_reader_cost += build_cost;
1581                file_metrics.scan_cost += prune_metrics.scan_cost;
1582            }
1583
1584            reader_metrics.merge_from(&prune_metrics);
1585        }
1586
1587        // Reports metrics.
1588        reader_metrics.observe_rows(read_type);
1589        reader_metrics.filter_metrics.observe();
1590        part_metrics.merge_reader_metrics(reader_metrics, per_file_metrics.as_ref());
1591    }
1592}
1593
1594/// Build the stream of scanning the extension range denoted by the [`RowGroupIndex`].
1595#[cfg(feature = "enterprise")]
1596pub(crate) async fn scan_extension_range(
1597    context: Arc<StreamContext>,
1598    index: RowGroupIndex,
1599    partition_metrics: PartitionMetrics,
1600) -> Result<BoxedBatchStream> {
1601    use snafu::ResultExt;
1602
1603    let range = context.input.extension_range(index.index);
1604    let reader = range.reader(context.as_ref());
1605    let stream = reader
1606        .read(context, partition_metrics, index)
1607        .await
1608        .context(crate::error::ScanExternalRangeSnafu)?;
1609    Ok(stream)
1610}
1611
1612pub(crate) async fn maybe_scan_other_ranges(
1613    context: &Arc<StreamContext>,
1614    index: RowGroupIndex,
1615    metrics: &PartitionMetrics,
1616) -> Result<BoxedBatchStream> {
1617    #[cfg(feature = "enterprise")]
1618    {
1619        scan_extension_range(context.clone(), index, metrics.clone()).await
1620    }
1621
1622    #[cfg(not(feature = "enterprise"))]
1623    {
1624        let _ = context;
1625        let _ = index;
1626        let _ = metrics;
1627
1628        crate::error::UnexpectedSnafu {
1629            reason: "no other ranges scannable",
1630        }
1631        .fail()
1632    }
1633}
1634
1635/// Build the stream of scanning the extension range in flat format denoted by the [`RowGroupIndex`].
1636#[cfg(feature = "enterprise")]
1637pub(crate) async fn scan_flat_extension_range(
1638    context: Arc<StreamContext>,
1639    index: RowGroupIndex,
1640    partition_metrics: PartitionMetrics,
1641) -> Result<BoxedRecordBatchStream> {
1642    use snafu::ResultExt;
1643
1644    let range = context.input.extension_range(index.index);
1645    let reader = range.flat_reader(context.as_ref());
1646    let stream = reader
1647        .read(context, partition_metrics, index)
1648        .await
1649        .context(crate::error::ScanExternalRangeSnafu)?;
1650    Ok(stream)
1651}
1652
1653pub(crate) async fn maybe_scan_flat_other_ranges(
1654    context: &Arc<StreamContext>,
1655    index: RowGroupIndex,
1656    metrics: &PartitionMetrics,
1657) -> Result<BoxedRecordBatchStream> {
1658    #[cfg(feature = "enterprise")]
1659    {
1660        scan_flat_extension_range(context.clone(), index, metrics.clone()).await
1661    }
1662
1663    #[cfg(not(feature = "enterprise"))]
1664    {
1665        let _ = context;
1666        let _ = index;
1667        let _ = metrics;
1668
1669        crate::error::UnexpectedSnafu {
1670            reason: "no other ranges scannable in flat format",
1671        }
1672        .fail()
1673    }
1674}
1675
1676/// A stream wrapper that splits record batches from an inner stream.
1677pub(crate) struct SplitRecordBatchStream<S> {
1678    /// The inner stream that yields record batches.
1679    inner: S,
1680    /// Buffer for split batches.
1681    batches: VecDeque<RecordBatch>,
1682}
1683
1684impl<S> SplitRecordBatchStream<S> {
1685    /// Creates a new splitting stream wrapper.
1686    pub(crate) fn new(inner: S) -> Self {
1687        Self {
1688            inner,
1689            batches: VecDeque::new(),
1690        }
1691    }
1692}
1693
1694impl<S> Stream for SplitRecordBatchStream<S>
1695where
1696    S: Stream<Item = Result<RecordBatch>> + Unpin,
1697{
1698    type Item = Result<RecordBatch>;
1699
1700    fn poll_next(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Option<Self::Item>> {
1701        loop {
1702            // First, check if we have buffered split batches
1703            if let Some(batch) = self.batches.pop_front() {
1704                return Poll::Ready(Some(Ok(batch)));
1705            }
1706
1707            // Poll the inner stream for the next batch
1708            let record_batch = match futures::ready!(Pin::new(&mut self.inner).poll_next(cx)) {
1709                Some(Ok(batch)) => batch,
1710                Some(Err(e)) => return Poll::Ready(Some(Err(e))),
1711                None => return Poll::Ready(None),
1712            };
1713
1714            // Split the batch and buffer the results
1715            split_record_batch(record_batch, &mut self.batches);
1716            // Continue the loop to return the first split batch
1717        }
1718    }
1719}
1720
1721/// Splits the batch by timestamps.
1722///
1723/// # Panics
1724/// Panics if the timestamp array is invalid.
1725pub(crate) fn split_record_batch(record_batch: RecordBatch, batches: &mut VecDeque<RecordBatch>) {
1726    let batch_rows = record_batch.num_rows();
1727    if batch_rows == 0 {
1728        return;
1729    }
1730    if batch_rows < 2 {
1731        batches.push_back(record_batch);
1732        return;
1733    }
1734
1735    let time_index_pos = time_index_column_index(record_batch.num_columns());
1736    let timestamps = record_batch.column(time_index_pos);
1737    let (ts_values, _unit) = timestamp_array_to_primitive(timestamps).unwrap();
1738    let mut offsets = Vec::with_capacity(16);
1739    offsets.push(0);
1740    let values = ts_values.values();
1741    for (i, &value) in values.iter().take(batch_rows - 1).enumerate() {
1742        if value > values[i + 1] {
1743            offsets.push(i + 1);
1744        }
1745    }
1746    offsets.push(values.len());
1747
1748    // Splits the batch by offsets.
1749    for (i, &start) in offsets[..offsets.len() - 1].iter().enumerate() {
1750        let end = offsets[i + 1];
1751        let rows_in_batch = end - start;
1752        batches.push_back(record_batch.slice(start, rows_in_batch));
1753    }
1754}