fp-library 0.17.0

A functional programming library for Rust featuring your favourite higher-kinded types and type classes.
Documentation
//! Deferred lazy evaluation using thread-safe thunks.
//!
//! ### Examples
//!
//! ```
//! use fp_library::{
//! 	brands::*,
//! 	functions::*,
//! 	types::*,
//! };
//!
//! let memo: ArcLazy<i32> = send_defer(|| ArcLazy::new(|| 42));
//! assert_eq!(*memo.evaluate(), 42);
//! ```

#[fp_macros::document_module]
mod inner {
	use fp_macros::*;
	/// A trait for deferred lazy evaluation with thread-safe thunks.
	///
	/// `SendDeferrable` is the thread-safe counterpart of
	/// [`Deferrable`](crate::classes::Deferrable). Where `Deferrable` accepts any
	/// `FnOnce() -> Self + 'a`, this trait requires `FnOnce() -> Self + Send + 'a`,
	/// ensuring the closure can be sent across thread boundaries. The two traits
	/// are independent: implementing one does not require implementing the other.
	///
	/// Unlike [`SendCloneFn`](crate::classes::SendCloneFn), which wraps multi-use
	/// `Fn` closures that are `Send + Sync`, this trait accepts a `FnOnce` closure that
	/// only needs to be `Send` (not `Sync`), since deferred computations are executed
	/// at most once.
	///
	/// ### Laws
	///
	/// `SendDeferrable` instances must satisfy the following law:
	/// * Transparency: `send_defer(|| x)` is observationally equivalent to `x` when evaluated.
	///
	/// ### Why there is no generic `fix`
	///
	/// As with [`Deferrable`](crate::classes::Deferrable), lazy self-reference requires
	/// shared ownership and interior mutability, which are properties specific to
	/// [`Lazy`](crate::types::Lazy). The concrete function
	/// [`arc_lazy_fix`](crate::types::lazy::arc_lazy_fix) provides this capability for
	/// `ArcLazy` specifically.
	#[document_type_parameters("The lifetime of the computation.")]
	#[document_examples]
	///
	/// Transparency law for [`ArcLazy`](crate::types::ArcLazy):
	///
	/// ```
	/// use fp_library::{
	/// 	functions::*,
	/// 	types::*,
	/// };
	///
	/// // Transparency: send_defer(|| x) is equivalent to x when evaluated.
	/// let x = ArcLazy::pure(42);
	/// let deferred: ArcLazy<i32> = send_defer(|| ArcLazy::pure(42));
	/// assert_eq!(*deferred.evaluate(), *x.evaluate());
	/// ```
	pub trait SendDeferrable<'a> {
		/// Creates a deferred value from a thread-safe thunk.
		#[document_signature]
		///
		#[document_parameters("The function that produces the value.")]
		///
		#[document_returns("A deferred value.")]
		#[document_examples]
		///
		/// ```
		/// use fp_library::{
		/// 	brands::*,
		/// 	functions::*,
		/// 	types::*,
		/// };
		///
		/// let memo: ArcLazy<i32> = send_defer(|| ArcLazy::new(|| 42));
		/// assert_eq!(*memo.evaluate(), 42);
		/// ```
		fn send_defer(f: impl FnOnce() -> Self + Send + 'a) -> Self
		where
			Self: Sized;
	}

	/// Creates a deferred value from a thread-safe thunk.
	///
	/// Free function version that dispatches to [the type class' associated function][`SendDeferrable::send_defer`].
	#[document_signature]
	///
	#[document_type_parameters(
		"The lifetime of the computation.",
		"The type of the deferred value."
	)]
	///
	#[document_parameters("The function that produces the value.")]
	///
	#[document_returns("A deferred value.")]
	#[document_examples]
	///
	/// ```
	/// use fp_library::{
	/// 	brands::*,
	/// 	functions::*,
	/// 	types::*,
	/// };
	///
	/// let memo: ArcLazy<i32> = send_defer(|| ArcLazy::new(|| 42));
	/// assert_eq!(*memo.evaluate(), 42);
	/// ```
	pub fn send_defer<'a, D: SendDeferrable<'a>>(f: impl FnOnce() -> D + Send + 'a) -> D {
		D::send_defer(f)
	}
}

pub use inner::*;

#[cfg(test)]
mod tests {
	use {
		crate::{
			functions::*,
			types::*,
		},
		quickcheck_macros::quickcheck,
	};

	/// SendDeferrable transparency law for `SendThunk`: `send_defer(|| x).evaluate() == x`.
	#[quickcheck]
	fn prop_send_deferrable_transparency_send_thunk(x: i32) -> bool {
		let deferred: SendThunk<i32> = send_defer(|| SendThunk::pure(x));
		deferred.evaluate() == x
	}

	/// SendDeferrable nesting law for `SendThunk`:
	/// `send_defer(|| send_defer(|| x)).evaluate() == send_defer(|| x).evaluate()`.
	#[quickcheck]
	fn prop_send_deferrable_nesting_send_thunk(x: i32) -> bool {
		let nested: SendThunk<i32> = send_defer(|| send_defer(|| SendThunk::pure(x)));
		let single: SendThunk<i32> = send_defer(|| SendThunk::pure(x));
		nested.evaluate() == single.evaluate()
	}

	/// SendDeferrable transparency law for `ArcLazy`: `send_defer(|| x).evaluate() == x`.
	#[quickcheck]
	fn prop_send_deferrable_transparency_arc_lazy(x: i32) -> bool {
		let deferred: ArcLazy<i32> = send_defer(|| ArcLazy::pure(x));
		*deferred.evaluate() == x
	}

	/// SendDeferrable nesting law for `ArcLazy`:
	/// `send_defer(|| send_defer(|| x)).evaluate() == send_defer(|| x).evaluate()`.
	#[quickcheck]
	fn prop_send_deferrable_nesting_arc_lazy(x: i32) -> bool {
		let nested: ArcLazy<i32> = send_defer(|| send_defer(|| ArcLazy::pure(x)));
		let single: ArcLazy<i32> = send_defer(|| ArcLazy::pure(x));
		*nested.evaluate() == *single.evaluate()
	}

	/// SendDeferrable transparency law for `TrySendThunk`:
	/// `send_defer(|| x).evaluate() == Ok(x)`.
	#[quickcheck]
	fn prop_send_deferrable_transparency_try_send_thunk(x: i32) -> bool {
		let deferred: TrySendThunk<i32, String> = send_defer(|| TrySendThunk::pure(x));
		deferred.evaluate() == Ok(x)
	}

	/// SendDeferrable nesting law for `TrySendThunk`:
	/// `send_defer(|| send_defer(|| x)).evaluate() == send_defer(|| x).evaluate()`.
	#[quickcheck]
	fn prop_send_deferrable_nesting_try_send_thunk(x: i32) -> bool {
		let nested: TrySendThunk<i32, String> = send_defer(|| send_defer(|| TrySendThunk::pure(x)));
		let single: TrySendThunk<i32, String> = send_defer(|| TrySendThunk::pure(x));
		nested.evaluate() == single.evaluate()
	}

	/// SendDeferrable transparency law for `ArcTryLazy`:
	/// `send_defer(|| x).evaluate() == Ok(&x)`.
	#[quickcheck]
	fn prop_send_deferrable_transparency_arc_try_lazy(x: i32) -> bool {
		let deferred: ArcTryLazy<i32, String> = send_defer(|| ArcTryLazy::ok(x));
		deferred.evaluate() == Ok(&x)
	}

	/// SendDeferrable nesting law for `ArcTryLazy`:
	/// `send_defer(|| send_defer(|| x)).evaluate() == send_defer(|| x).evaluate()`.
	#[quickcheck]
	fn prop_send_deferrable_nesting_arc_try_lazy(x: i32) -> bool {
		let nested: ArcTryLazy<i32, String> = send_defer(|| send_defer(|| ArcTryLazy::ok(x)));
		let single: ArcTryLazy<i32, String> = send_defer(|| ArcTryLazy::ok(x));
		nested.evaluate() == single.evaluate()
	}
}