use crate::{
any::{TypeName, WithContextLt},
bijective_higher_ranked_type,
effect::{Effect, Future},
hkt::AnySizedSend,
protocol::{walker::hint::HintMeta, Visitor},
Status,
};
use super::VisitResult;
pub trait Recoverable<'ctx, E: Effect> {
fn visit<'a>(
&'a mut self,
scope: DynRecoverableScope<'a, 'ctx, E>,
) -> Future<'a, VisitResult<DynRecoverableScope<'a, 'ctx, E>>, E>;
}
bijective_higher_ranked_type! {
pub type DynRecoverable['ctx][E]: WithContextLt['ctx][]
for<'a>
(dyn Recoverable<'ctx, E> + Send + 'a)
where {
E: Effect
}
}
bijective_higher_ranked_type! {
pub type [][E]: TypeName[][]
for<'ctx>
(DynRecoverable<'ctx, E>)
where {
E: Effect
}
}
pub trait RecoverableScope<'ctx, E: Effect> {
fn new_walk<'a>(&'a mut self, visitor: Visitor<'a, 'ctx>) -> Future<'a, Status, E>;
}
pub type DynRecoverableScope<'a, 'ctx, E> = &'a mut (dyn RecoverableScope<'ctx, E> + Send + 'a);
bijective_higher_ranked_type! {
pub type RecoverableKnownHkt[][]: AnySizedSend[][] for<'lt> (())
}
impl<'ctx, E: Effect> HintMeta<'ctx> for DynRecoverable<'ctx, E> {
type Known = RecoverableKnownHkt;
type Hint = ();
}
pub fn visit_recoverable<'a, 'ctx, E: Effect>(
visitor: Visitor<'a, 'ctx>,
scope: DynRecoverableScope<'a, 'ctx, E>,
) -> Future<'a, VisitResult<DynRecoverableScope<'a, 'ctx, E>>, E> {
if let Some(object) = visitor.upcast_mut::<DynRecoverable<'ctx, E>>() {
// Allow the visitor to give a hint if it wants.
object.visit(scope)
} else {
// If the visitor doesn't support request hint then we continue.
E::ready(VisitResult::Skipped(scope))
}
}