launchpadlib 0.5.7

Rust library for accessing Launchpad.net
Documentation
//! Pagination support
use crate::Error;
use std::pin::Pin;

use futures::task::Context;
use futures::Future;
use futures::{task::Poll, Stream};

/// A page of items.
#[async_trait::async_trait]
pub trait Page: Sized {
    /// The type of item in the page.
    type Item;

    /// Return the next page, if any.
    async fn next<'a>(
        &'a self,
        client: &'a dyn wadl::r#async::Client,
    ) -> Result<Option<Self>, Error>
    where
        Self: Sized;

    /// Return the previous page, if any.
    async fn prev<'a>(
        &'a self,
        client: &'a dyn wadl::r#async::Client,
    ) -> Result<Option<Self>, Error>
    where
        Self: Sized;

    /// Return the index of the first entry on this page.
    fn start(&self) -> usize;

    /// Return the total number of entries in the page collection
    fn total_size(&self) -> Option<usize>;

    /// Get the entries on this page.
    fn entries(&self) -> Vec<Self::Item>;
}

/// A collection of items that may be paginated.
#[allow(dead_code)]
pub struct PagedCollection<'a, P: Page> {
    client: &'a dyn wadl::r#async::Client,
    pending: Vec<P::Item>,
    page: P,
}

impl<'a, P: Page> PagedCollection<'a, P> {
    /// Return the total number of entries in the collection.
    pub fn len(&self) -> Option<usize> {
        self.page.total_size()
    }

    /// Return true if the collection is empty.
    pub fn is_empty(&self) -> bool {
        self.len() == Some(0)
            || (self.len().is_none() && self.page.entries().is_empty() && self.page.start() == 0)
    }

    /// Get the item at the given index.
    ///
    /// This will fetch pages as needed to satisfy the request.
    pub async fn get(&mut self, index: usize) -> Result<Option<P::Item>, Error> {
        if let Some(total_size) = self.len() {
            if index >= total_size {
                return Ok(None);
            }
        }

        while index < self.page.start() {
            self.page = if let Some(page) = self.page.prev(self.client).await? {
                page
            } else {
                return Ok(None);
            };
        }

        while index >= self.page.start() + self.page.entries().len() {
            self.page = if let Some(page) = self.page.next(self.client).await? {
                page
            } else {
                return Ok(None);
            };
        }

        let mut entries = self.page.entries();

        Ok(Some(entries.remove(index - self.page.start())))
    }

    /// Create a new page
    pub fn new(client: &'a dyn wadl::r#async::Client, page: P) -> Self {
        let mut pending = page.entries();
        pending.reverse();
        Self {
            client,
            pending,
            page,
        }
    }
}

impl<'a, P: Clone> Stream for PagedCollection<'a, P>
where
    P: Page + Unpin,
    P::Item: Unpin,
{
    type Item = Result<P::Item, Error>;

    fn poll_next(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Option<Self::Item>> {
        let this = self.get_mut();

        // Return an item from the pending list if available
        if let Some(item) = this.pending.pop() {
            return Poll::Ready(Some(Ok(item)));
        }

        let page_clone = this.page.clone();

        // Create an async block to fetch the next page.
        let fut = page_clone.next(this.client);

        // Pin the future so we can poll it
        futures::pin_mut!(fut);

        match fut.poll(cx) {
            Poll::Pending => {
                // If the future is not yet ready, move `page` back to `this.page`
                Poll::Pending
            }
            Poll::Ready(Ok(Some(next_page))) => {
                // Update `this.page` with the newly fetched page
                this.page = next_page;
                let mut entries = this.page.entries();
                entries.reverse();
                this.pending = entries;

                // Return the next item from the newly fetched page
                if let Some(item) = this.pending.pop() {
                    Poll::Ready(Some(Ok(item)))
                } else {
                    Poll::Ready(None)
                }
            }
            Poll::Ready(Ok(None)) => {
                // No more pages to fetch, stream has ended
                Poll::Ready(None)
            }
            Poll::Ready(Err(e)) => {
                // On error, move `page` back to `this.page` and return the error
                Poll::Ready(Some(Err(e)))
            }
        }
    }
}

#[cfg(test)]
mod tests {
    use super::*;

    use futures::TryStreamExt;

    #[derive(Clone)]
    struct DummyMaster<I: Send + Sync> {
        entries: Vec<I>,
        chunk_size: usize,
    }

    #[derive(Clone)]
    struct DummyPage<I: Send + Sync> {
        start: usize,
        entries: std::sync::Arc<DummyMaster<I>>,
    }

    #[async_trait::async_trait]
    impl<I: Clone + Send + Sync> Page for DummyPage<I> {
        type Item = I;

        async fn next<'a>(
            &'a self,
            _: &'a dyn wadl::r#async::Client,
        ) -> Result<Option<Self>, Error> {
            if self.start + self.entries.chunk_size >= self.entries.entries.len() {
                Ok(None)
            } else {
                Ok(Some(Self {
                    start: self.start + self.entries.chunk_size,
                    entries: self.entries.clone(),
                }))
            }
        }

        async fn prev<'a>(
            &'a self,
            _: &'a dyn wadl::r#async::Client,
        ) -> Result<Option<Self>, Error> {
            if self.start == 0 {
                Ok(None)
            } else {
                Ok(Some(Self {
                    start: self.start - self.entries.chunk_size,
                    entries: self.entries.clone(),
                }))
            }
        }

        fn start(&self) -> usize {
            self.start
        }

        fn total_size(&self) -> Option<usize> {
            Some(self.entries.entries.len())
        }

        fn entries(&self) -> Vec<Self::Item> {
            self.entries.entries[self.start
                ..std::cmp::min(
                    self.start + self.entries.chunk_size,
                    self.entries.entries.len(),
                )]
                .to_vec()
        }
    }

    #[tokio::test]
    async fn test_iter() {
        let client = crate::r#async::client::Client::anonymous("just testing");
        let master = DummyMaster {
            entries: vec!["a", "b", "c"],
            chunk_size: 2,
        };
        let page = DummyPage {
            start: 0,
            entries: std::sync::Arc::new(master),
        };
        let mut collection = super::PagedCollection::new(&client, page);

        assert_eq!(collection.len(), Some(3));
        assert!(!collection.is_empty());
        assert_eq!("a", collection.get(0).await.unwrap().unwrap());
        assert_eq!("b", collection.get(1).await.unwrap().unwrap());
        assert_eq!("c", collection.get(2).await.unwrap().unwrap());
        assert_eq!(None, collection.get(3).await.unwrap());
    }

    #[tokio::test]
    async fn test_empty() {
        let client = crate::r#async::client::Client::anonymous("just testing");
        let master: DummyMaster<&str> = DummyMaster::<&str> {
            entries: vec![],
            chunk_size: 2,
        };
        let page = DummyPage {
            start: 0,
            entries: std::sync::Arc::new(master),
        };
        let collection = super::PagedCollection::new(&client, page);

        assert_eq!(collection.len(), Some(0));
        assert_eq!(collection.is_empty(), true);
    }

    #[tokio::test]
    async fn test_stream() {
        let client = crate::r#async::client::Client::anonymous("just testing");
        let master = DummyMaster {
            entries: vec!["a", "b", "c"],
            chunk_size: 2,
        };

        let page = DummyPage {
            entries: std::sync::Arc::new(master),
            start: 0,
        };

        let collection = super::PagedCollection::new(&client, page);

        let result: Vec<&str> = collection.try_collect::<Vec<&str>>().await.unwrap();

        assert_eq!(result, vec!["a", "b", "c"]);
    }
}