diff options
Diffstat (limited to 'src/boxed.rs')
-rw-r--r-- | src/boxed.rs | 117 |
1 files changed, 117 insertions, 0 deletions
diff --git a/src/boxed.rs b/src/boxed.rs new file mode 100644 index 0000000..d64e204 --- /dev/null +++ b/src/boxed.rs @@ -0,0 +1,117 @@ +// Copyright (c) 2018 The predicates-rs Project Developers. +// +// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or +// http://www.apache.org/license/LICENSE-2.0> or the MIT license +// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your +// option. This file may not be copied, modified, or distributed +// except according to those terms. + +//! Predicate that can wrap other dynamically-called predicates in an +//! easy-to-manage type. + +use std::fmt; + +use crate::reflection; +use crate::utils; +use crate::Predicate; + +/// `Predicate` that wraps another `Predicate` as a trait object, allowing +/// sized storage of predicate types. +pub struct BoxPredicate<Item: ?Sized>(Box<dyn Predicate<Item> + Send + Sync>); + +impl<Item> BoxPredicate<Item> +where + Item: ?Sized, +{ + /// Creates a new `BoxPredicate`, a wrapper around a dynamically-dispatched + /// `Predicate` type with useful trait impls. + pub fn new<P: Predicate<Item>>(inner: P) -> BoxPredicate<Item> + where + P: Send + Sync + 'static, + { + BoxPredicate(Box::new(inner)) + } +} + +impl<Item> fmt::Debug for BoxPredicate<Item> +where + Item: ?Sized, +{ + fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { + f.debug_struct("BoxPredicate").finish() + } +} + +impl<Item> reflection::PredicateReflection for BoxPredicate<Item> +where + Item: ?Sized, +{ + fn parameters<'a>(&'a self) -> Box<dyn Iterator<Item = reflection::Parameter<'a>> + 'a> { + self.0.parameters() + } + + fn children<'a>(&'a self) -> Box<dyn Iterator<Item = reflection::Child<'a>> + 'a> { + self.0.children() + } +} + +impl<Item> fmt::Display for BoxPredicate<Item> +where + Item: ?Sized, +{ + fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { + self.0.fmt(f) + } +} + +impl<Item> Predicate<Item> for BoxPredicate<Item> +where + Item: ?Sized, +{ + fn eval(&self, variable: &Item) -> bool { + self.0.eval(variable) + } + + fn find_case<'a>(&'a self, expected: bool, variable: &Item) -> Option<reflection::Case<'a>> { + utils::default_find_case(self, expected, variable) + } +} + +/// `Predicate` extension for boxing a `Predicate`. +pub trait PredicateBoxExt<Item: ?Sized> +where + Self: Predicate<Item>, +{ + /// Returns a `BoxPredicate` wrapper around this `Predicate` type. + /// + /// Returns a `BoxPredicate` wrapper around this `Predicate type. The + /// `BoxPredicate` type has a number of useful properties: + /// + /// - It stores the inner predicate as a trait object, so the type of + /// `BoxPredicate` will always be the same even if steps are added or + /// removed from the predicate. + /// - It is a common type, allowing it to be stored in vectors or other + /// collection types. + /// - It implements `Debug` and `Display`. + /// + /// # Examples + /// + /// ``` + /// use predicates::prelude::*; + /// + /// let predicates = vec![ + /// predicate::always().boxed(), + /// predicate::never().boxed(), + /// ]; + /// assert_eq!(true, predicates[0].eval(&4)); + /// assert_eq!(false, predicates[1].eval(&4)); + /// ``` + fn boxed(self) -> BoxPredicate<Item> + where + Self: Sized + Send + Sync + 'static, + { + BoxPredicate::new(self) + } +} + +impl<P, Item> PredicateBoxExt<Item> for P where P: Predicate<Item> {} |