use std::future::Future;
use futures_util::Stream;
use std::pin::Pin;
use uuid::Uuid;
/// A stream of jobs from the job queue.
pub type JobStream<J, E> = Pin<Box<dyn Stream<Item = Result<J, E>> + Send>>;
/// A job queue that can store and return jobs.
pub trait JobQueue<T: JobItem>: Send + Sync + Sized + Clone + 'static {
/// A type of job object that will be returned by the queue.
type Job: Job<T, Self>;
/// Error type that the queue can produce in its work.
type Error: std::error::Error + Send + Sync + Sized;
/// Get one item from the job queue, if the job queue has pending
/// items available.
///
/// # Errors
///
/// Returns an error if a job queue failed in some way. Having no
/// items is not a failure, in which case `Ok(None)` is returned.
#[must_use = "Undone jobs get put back into the queue."]
fn get_one(&self) -> impl Future<Output = Result<Option<Self::Job>, Self::Error>> + Send;
/// Put an item into a job queue, returning its UUID.
fn put(&self, item: &T) -> impl Future<Output = Result<Uuid, Self::Error>> + Send;
/*
/// Check the amount of pending and stuck items in the job queue.
fn len(&self) -> impl Future<Output = Result<(usize, usize), Self::Error>> + Send;
/// Returns whether the job queue has some pending items.
async fn is_empty(&self) -> Result<bool, Self::Error> {
Ok(self.len().await?.0 == 0)
}
/// Returns whether the job queue has some stuck items that
/// require manual cleanup.
async fn has_stuck(&self) -> Result<bool, Self::Error> {
Ok(self.len().await?.1 > 0)
}
*/
/// Consume the job queue object and return a stream of unstuck
/// items from the job queue.
///
/// Note that one item may be returned several times if it is not
/// marked as done.
fn into_stream(self) -> impl Future<Output = Result<JobStream<Self::Job, Self::Error>, Self::Error>> + Send;
}
/// A job description yielded from a job queue.
///
/// # Implementors
///
/// On [`Drop`], the job should be returned to a job queue. If your
/// job queue tracks attempts, the counter should be incremented by
/// one.
///
/// Figuring how to do this asynchronously from a synchronous trait
/// is left as an exercise to the reader.
pub trait Job<T: JobItem, Q: JobQueue<T>>: Send + Sync + Sized {
/// Get the object describing the task itself.
fn job(&self) -> &T;
/// Mark the job as done and remove it from the job queue.
fn done(self) -> impl Future<Output = Result<(), Q::Error>> + Send;
}
/// An object describing the job itself, returned as part of a
/// [`Job`].
pub trait JobItem: Send + Sync + Sized + std::fmt::Debug {}