blob: d64e204982ffd5b1526438c8afa017f51039c5f9 [file] [log] [blame]
// 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> {}