squire 0.0.1-alpha.7

Safe and idiomatic SQLite bindings
//! [`Iterator`] implementations.
//!
//! These concrete types are exposed in Squire’s API, but they don’t normally
//! need to be referenced by name.

use crate::{
    column::{ColumnIndexes, Columns},
    error::Result,
    row::Rows,
    statement::{Binding, Execute},
    types::BindIndex,
};

/// Map over [`Rows`].
///
/// Returned by [`Rows::map`].
#[derive(Debug)]
pub struct Map<'c, 's, C, F, S = Binding<'c, 's>>
where
    C: ColumnIndexes,
    S: Execute<'c, 's>,
    'c: 's,
{
    pub(crate) rows: Rows<'c, 's, C, S>,
    pub(crate) f: F,
}

impl<'c, 's, 'r, C, F, T, S> Iterator for Map<'c, 's, C, F, S>
where
    C: Columns<'r>,
    F: FnMut(C) -> T,
    T: 's,
    S: Execute<'c, 's>,
    'c: 's,
    's: 'r,
{
    type Item = Result<T>;

    fn next(&mut self) -> Option<Self::Item> {
        // SAFETY: We never hold onto row data across loop iterations
        match unsafe { self.rows.advance() } {
            Ok(Some(item)) => Some(Ok((self.f)(item))),
            Ok(None) => None,
            Err(e) => Some(Err(e)),
        }
    }
}

/// Map over [`Rows`] and filter the results.
///
/// Returned by [`Rows::filter_map`].
#[derive(Debug)]
pub struct FilterMap<'c, 's, C, F, S = Binding<'c, 's>>
where
    C: ColumnIndexes,
    S: Execute<'c, 's>,
    'c: 's,
{
    pub(crate) rows: Rows<'c, 's, C, S>,
    pub(crate) f: F,
}

impl<'c, 's, 'r, C, F, T, S> Iterator for FilterMap<'c, 's, C, F, S>
where
    C: Columns<'r>,
    F: FnMut(C) -> Option<T>,
    T: 's,
    S: Execute<'c, 's>,
    'c: 's,
    's: 'r,
{
    type Item = Result<T>;

    fn next(&mut self) -> Option<Self::Item> {
        loop {
            // SAFETY: We never hold onto row data across loop iterations
            match unsafe { self.rows.advance() } {
                Ok(Some(item)) => {
                    if let Some(mapped) = (self.f)(item) {
                        return Some(Ok(mapped));
                    }
                }
                Ok(None) => return None,
                Err(e) => return Some(Err(e)),
            }
        }
    }
}

/// An [`Iterator`] of [parameter indexes](BindIndex).
pub struct BindIndexes {
    current: BindIndex,
}

impl BindIndexes {
    pub(crate) const fn new(initial: BindIndex) -> Self {
        Self { current: initial }
    }
}

impl Iterator for BindIndexes {
    type Item = BindIndex;

    fn next(&mut self) -> Option<Self::Item> {
        let current = self.current;
        self.current = self.current.next();
        Some(current)
    }

    fn size_hint(&self) -> (usize, Option<usize>) {
        (usize::MAX, None)
    }
}