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
//! Utilities useful across different store impls.

use super::{AuthorFilter, KeyFilter, Query, QueryKind, SortBy};
use crate::SignedEntry;

/// A helper for stores that have by-author and by-key indexes for records.
#[derive(Debug)]
pub enum IndexKind {
    AuthorKey {
        range: AuthorFilter,
        key_filter: KeyFilter,
    },
    KeyAuthor {
        range: KeyFilter,
        author_filter: AuthorFilter,
        latest_per_key: bool,
    },
}

impl From<&Query> for IndexKind {
    fn from(query: &Query) -> Self {
        match &query.kind {
            QueryKind::Flat(details) => match (&query.filter_author, details.sort_by) {
                (AuthorFilter::Any, SortBy::KeyAuthor) => IndexKind::KeyAuthor {
                    range: query.filter_key.clone(),
                    author_filter: AuthorFilter::Any,
                    latest_per_key: false,
                },
                _ => IndexKind::AuthorKey {
                    range: query.filter_author.clone(),
                    key_filter: query.filter_key.clone(),
                },
            },
            QueryKind::SingleLatestPerKey(_) => IndexKind::KeyAuthor {
                range: query.filter_key.clone(),
                author_filter: query.filter_author.clone(),
                latest_per_key: true,
            },
        }
    }
}

/// Helper to extract the latest entry per key from an iterator that yields [`SignedEntry`] items.
///
/// Items must be pushed in key-sorted order.
#[derive(Debug, Default)]
pub struct LatestPerKeySelector(Option<SignedEntry>);

pub enum SelectorRes {
    /// The iterator is finished.
    Finished,
    /// The selection is not yet finished, keep pushing more items.
    Continue,
    /// The selection yielded an entry.
    Some(SignedEntry),
}

impl LatestPerKeySelector {
    /// Push an entry into the selector.
    ///
    /// Entries must be sorted by key beforehand.
    pub fn push(&mut self, entry: Option<SignedEntry>) -> SelectorRes {
        let Some(entry) = entry else {
            return match self.0.take() {
                Some(entry) => SelectorRes::Some(entry),
                None => SelectorRes::Finished,
            };
        };
        match self.0.take() {
            None => {
                self.0 = Some(entry);
                SelectorRes::Continue
            }
            Some(last) if last.key() == entry.key() => {
                if entry.timestamp() > last.timestamp() {
                    self.0 = Some(entry);
                } else {
                    self.0 = Some(last);
                }
                SelectorRes::Continue
            }
            Some(last) => {
                self.0 = Some(entry);
                SelectorRes::Some(last)
            }
        }
    }
}