From 5cbce2bac61130e3aefc7e73a9bc9080bd2b5a32 Mon Sep 17 00:00:00 2001 From: Mingwei Samuel Date: Thu, 1 Feb 2024 20:26:01 -0800 Subject: [PATCH] feat(hydroflow): add `StateHandleErased` (no type param) for storage in data structures, fix #1048 (#1054) Co-authored-by: Discord9 --- hydroflow/src/scheduled/mod.rs | 2 +- hydroflow/src/scheduled/state.rs | 74 +++++++++++++++++++++++++++++++- 2 files changed, 74 insertions(+), 2 deletions(-) diff --git a/hydroflow/src/scheduled/mod.rs b/hydroflow/src/scheduled/mod.rs index 17417788dddf..e414fdfab324 100644 --- a/hydroflow/src/scheduled/mod.rs +++ b/hydroflow/src/scheduled/mod.rs @@ -43,6 +43,6 @@ impl Display for HandoffId { /// A staten handle's ID. Invalid if used in a different [`graph::Hydroflow`] /// instance than the original that created it. -#[derive(Clone, Copy, Debug, PartialEq, Eq, Hash)] +#[derive(Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Ord, Hash)] #[repr(transparent)] pub struct StateId(pub(crate) usize); diff --git a/hydroflow/src/scheduled/state.rs b/hydroflow/src/scheduled/state.rs index 05c862cd06d1..f75076c2a377 100644 --- a/hydroflow/src/scheduled/state.rs +++ b/hydroflow/src/scheduled/state.rs @@ -1,13 +1,17 @@ //! Module for [`StateHandle`], part of the "state API". +use std::any::{Any, TypeId}; use std::marker::PhantomData; use super::StateId; /// A handle into a particular [`Hydroflow`](super::graph::Hydroflow) instance, referring to data /// inserted by [`add_state`](super::graph::Hydroflow::add_state). +/// +/// If you need to store state handles in a data structure see [`StateHandleErased`] which hides +/// the generic type parameter. #[must_use] -#[derive(Debug)] +#[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Hash)] pub struct StateHandle { pub(crate) state_id: StateId, pub(crate) _phantom: PhantomData<*mut T>, @@ -18,3 +22,71 @@ impl Clone for StateHandle { *self } } + +/// A state handle with the generic type parameter erased, allowing it to be stored in omogenous +/// data structures. The type is tracked internally as data via [`TypeId`]. +/// +/// Use [`StateHandleErased::from(state_handle)`](StateHandleErased::from) to create an instance +/// from a typed [`StateHandle`]. +/// +/// Use [`StateHandle::::try_from()`](StateHandle::try_from) to convert the `StateHandleErased` +/// back into a `StateHandle` of the given type `T`. If `T` is the wrong type then the original +/// `StateHandleErased` will be returned as the `Err`. +#[derive(Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Ord, Hash)] +pub struct StateHandleErased { + state_id: StateId, + type_id: TypeId, +} + +/// See [`StateHandleErased`]. +impl TryFrom for StateHandle +where + T: Any, +{ + type Error = StateHandleErased; + + fn try_from(value: StateHandleErased) -> Result { + if TypeId::of::() == value.type_id { + Ok(Self { + state_id: value.state_id, + _phantom: PhantomData, + }) + } else { + Err(value) + } + } +} +/// See [`StateHandleErased`]. +impl From> for StateHandleErased +where + T: Any, +{ + fn from(value: StateHandle) -> Self { + Self { + state_id: value.state_id, + type_id: TypeId::of::(), + } + } +} + +#[cfg(test)] +mod test { + use std::marker::PhantomData; + + use super::*; + use crate::scheduled::StateId; + + #[test] + fn test_erasure() { + let handle = StateHandle:: { + state_id: StateId(0), + _phantom: PhantomData, + }; + let handle_erased = StateHandleErased::from(handle); + let handle_good = StateHandle::::try_from(handle_erased); + let handle_bad = StateHandle::<&'static str>::try_from(handle_erased); + + assert_eq!(Ok(handle), handle_good); + assert_eq!(Err(handle_erased), handle_bad); + } +}