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 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190
// Copyright Materialize, Inc. and contributors. All rights reserved.
//
// Use of this software is governed by the Business Source License
// included in the LICENSE file.
//
// As of the Change Date specified in that file, in accordance with
// the Business Source License, use of this software will be governed
// by the Apache License, Version 2.0.
//! Defines a "lending iterator" for [`Row`]
use std::fmt::Debug;
use crate::row::{Row, RowRef};
/// An iterator that can borrow from `self` and yield [`RowRef`]s.
///
/// This trait is a "lending iterator" for [`Row`]s, in other words, an iterator that borrows from
/// self (e.g. an underlying memory buffer) to return a [`RowRef`]. The [`std::iter::Iterator`]
/// trait does not currently support this pattern because there is no way to name the lifetime of
/// the borrow on its associated `Item` type. Generic Associated Types (GATs) would allow this but
/// so far no new trait has been introduced with this API.
///
/// There are a few open source crates that provide a trait:
///
/// * [`streaming_iterator`](https://docs.rs/streaming-iterator/latest/streaming_iterator/)
/// * [`lending-iterator`](https://docs.rs/lending-iterator/latest/lending_iterator/)
///
/// Neither have an `IntoLendingIterator` trait that is useful for our interface, nor do they work
/// well with trait objects.
pub trait RowIterator: Debug {
/// Returns the next [`RowRef`] advancing the iterator.
fn next(&mut self) -> Option<&RowRef>;
/// Returns the next [`RowRef`] without advancing the iterator.
fn peek(&mut self) -> Option<&RowRef>;
/// The total number of [`Row`]s this iterator could ever yield.
///
/// Note: it _does not_ return the number of rows _remaining_, in otherwords calling `.next()`
/// will not change the value returned from this method.
fn count(&self) -> usize;
/// Maps the returned [`RowRef`]s from this [`RowIterator`].
fn map<T, F>(self, f: F) -> MappedRowIterator<Self, F>
where
Self: Sized,
F: FnMut(&RowRef) -> T,
{
MappedRowIterator {
inner: self,
func: f,
}
}
}
impl<I: RowIterator + ?Sized> RowIterator for Box<I> {
fn next(&mut self) -> Option<&RowRef> {
(**self).next()
}
fn peek(&mut self) -> Option<&RowRef> {
(**self).peek()
}
fn count(&self) -> usize {
(**self).count()
}
}
impl<I: RowIterator + ?Sized> RowIterator for &mut I {
fn next(&mut self) -> Option<&RowRef> {
(**self).next()
}
fn peek(&mut self) -> Option<&RowRef> {
(**self).peek()
}
fn count(&self) -> usize {
(**self).count()
}
}
#[derive(Debug)]
pub struct MappedRowIterator<I: RowIterator, F> {
inner: I,
func: F,
}
impl<T, F, I: RowIterator> Iterator for MappedRowIterator<I, F>
where
F: FnMut(&RowRef) -> T,
{
type Item = T;
fn next(&mut self) -> Option<Self::Item> {
let row_ref = self.inner.next()?;
Some((self.func)(row_ref))
}
}
/// Convert a type into a [`RowIterator`].
pub trait IntoRowIterator {
type Iter: RowIterator;
fn into_row_iter(self) -> Self::Iter;
}
impl<T: RowIterator> IntoRowIterator for T {
type Iter = Self;
fn into_row_iter(self) -> Self::Iter {
self
}
}
/// A [`RowIterator`] for a single [`Row`].
#[derive(Debug)]
pub struct SingleRowIter {
row: Row,
finished: bool,
}
impl RowIterator for SingleRowIter {
fn next(&mut self) -> Option<&RowRef> {
if self.finished {
None
} else {
self.finished = true;
Some(self.row.as_ref())
}
}
fn peek(&mut self) -> Option<&RowRef> {
if self.finished {
None
} else {
Some(self.row.as_ref())
}
}
fn count(&self) -> usize {
1
}
}
impl IntoRowIterator for Row {
type Iter = SingleRowIter;
fn into_row_iter(self) -> Self::Iter {
SingleRowIter {
row: self,
finished: false,
}
}
}
/// A [`RowIterator`] for a [`Vec`] of [`Row`]s.
#[derive(Debug, Clone)]
pub struct VecRowIter {
rows: Vec<Row>,
index: usize,
}
impl RowIterator for VecRowIter {
fn next(&mut self) -> Option<&RowRef> {
let row = self.rows.get(self.index).map(|r| r.as_ref())?;
self.index = self.index.saturating_add(1);
Some(row)
}
fn peek(&mut self) -> Option<&RowRef> {
self.rows.get(self.index).map(|r| r.as_ref())
}
fn count(&self) -> usize {
self.rows.len()
}
}
impl IntoRowIterator for Vec<Row> {
type Iter = VecRowIter;
fn into_row_iter(self) -> Self::Iter {
VecRowIter {
rows: self,
index: 0,
}
}
}