mastodon-async/src/entities/itemsiter.rs

104 lines
3.5 KiB
Rust
Raw Normal View History

use futures::{stream::unfold, Stream};
2022-12-07 20:58:28 +00:00
use log::{as_debug, as_serde, debug, info, warn};
use crate::page::Page;
2022-12-07 20:58:28 +00:00
use serde::{Deserialize, Serialize};
2022-11-27 14:44:43 +00:00
/// Abstracts away the `next_page` logic into a single stream of items
///
/// ```no_run,async
2022-12-22 18:19:49 +00:00
/// use mastodon_async::prelude::*;
/// use futures::stream::StreamExt;
/// use futures_util::pin_mut;
///
/// tokio_test::block_on(async {
/// let data = Data::default();
/// let client = Mastodon::from(data);
/// let statuses = client.statuses("user-id", None).await.unwrap().items_iter();
/// statuses.for_each(|status| async move {
/// // Do something with the status
/// }).await;
/// })
2022-11-27 14:44:43 +00:00
/// ```
///
/// See documentation for `futures::Stream::StreamExt` for available methods.
2022-11-27 14:44:43 +00:00
#[derive(Debug, Clone)]
2022-12-07 20:58:28 +00:00
pub(crate) struct ItemsIter<T: Clone + for<'de> Deserialize<'de> + Serialize> {
page: Page<T>,
2022-11-27 14:44:43 +00:00
buffer: Vec<T>,
cur_idx: usize,
use_initial: bool,
}
2022-12-07 20:58:28 +00:00
impl<'a, T: Clone + for<'de> Deserialize<'de> + Serialize> ItemsIter<T> {
pub(crate) fn new(page: Page<T>) -> ItemsIter<T> {
2022-11-27 14:44:43 +00:00
ItemsIter {
page,
buffer: vec![],
cur_idx: 0,
use_initial: true,
}
}
fn need_next_page(&self) -> bool {
2022-12-07 20:58:28 +00:00
if self.buffer.is_empty() || self.cur_idx == self.buffer.len() {
debug!(idx = self.cur_idx, buffer_len = self.buffer.len(); "next page needed");
true
} else {
false
}
2022-11-27 14:44:43 +00:00
}
async fn fill_next_page(&mut self) -> Option<()> {
2022-12-07 20:58:28 +00:00
match self.page.next_page().await {
Ok(Some(items)) => {
info!(item_count = items.len(); "next page received");
if items.is_empty() {
return None;
}
self.buffer = items;
self.cur_idx = 0;
Some(())
},
Err(err) => {
warn!(err = as_debug!(err); "error encountered filling next page");
None
},
_ => None,
2022-11-27 14:44:43 +00:00
}
}
pub(crate) fn stream(self) -> impl Stream<Item = T> {
unfold(self, |mut this| async move {
if this.use_initial {
2022-12-07 20:58:28 +00:00
let idx = this.cur_idx;
if this.page.initial_items.is_empty() || idx == this.page.initial_items.len() {
debug!(index = idx, n_initial_items = this.page.initial_items.len(); "exhausted initial items and no more pages are present");
2022-11-27 14:44:43 +00:00
return None;
}
2022-12-07 20:58:28 +00:00
if idx == this.page.initial_items.len() - 1 {
this.cur_idx = 0;
this.use_initial = false;
2022-12-07 20:58:28 +00:00
debug!(index = idx, n_initial_items = this.page.initial_items.len(); "exhausted initial items");
} else {
this.cur_idx += 1;
}
let item = this.page.initial_items[idx].clone();
2022-12-07 20:58:28 +00:00
debug!(item = as_serde!(item), index = idx; "yielding item from initial items");
// let item = Box::pin(item);
// pin_mut!(item);
Some((item, this))
} else {
if this.need_next_page() {
this.fill_next_page().await?;
}
let idx = this.cur_idx;
this.cur_idx += 1;
let item = this.buffer[idx].clone();
2022-12-07 20:58:28 +00:00
debug!(item = as_serde!(item), index = idx; "yielding item from initial stream");
Some((item, this))
2022-11-27 14:44:43 +00:00
}
})
2022-11-27 14:44:43 +00:00
}
}