pub use super::members::Contains;
use sp_std::marker::PhantomData;
pub trait FilterStack<T>: Contains<T> {
type Stack;
fn push(constraint: impl Fn(&T) -> bool + 'static);
fn pop();
fn take() -> Self::Stack;
fn restore(taken: Self::Stack);
}
pub struct FilterStackGuard<F: FilterStack<T>, T>(PhantomData<(F, T)>);
pub struct ClearFilterGuard<F: FilterStack<T>, T>(Option<F::Stack>, PhantomData<T>);
impl<F: FilterStack<T>, T> FilterStackGuard<F, T> {
pub fn new(constraint: impl Fn(&T) -> bool + 'static) -> Self {
F::push(constraint);
Self(PhantomData)
}
}
impl<F: FilterStack<T>, T> Drop for FilterStackGuard<F, T> {
fn drop(&mut self) {
F::pop();
}
}
impl<F: FilterStack<T>, T> ClearFilterGuard<F, T> {
pub fn new() -> Self {
Self(Some(F::take()), PhantomData)
}
}
impl<F: FilterStack<T>, T> Drop for ClearFilterGuard<F, T> {
fn drop(&mut self) {
if let Some(taken) = self.0.take() {
F::restore(taken);
}
}
}
pub trait InstanceFilter<T>: Sized + Send + Sync {
fn filter(&self, _: &T) -> bool;
fn is_superset(&self, _o: &Self) -> bool {
false
}
}
impl<T> InstanceFilter<T> for () {
fn filter(&self, _: &T) -> bool {
true
}
fn is_superset(&self, _o: &Self) -> bool {
true
}
}
#[doc(hidden)]
pub use sp_std::{
boxed::Box,
cell::RefCell,
mem::{swap, take},
vec::Vec,
};
#[macro_export]
macro_rules! impl_filter_stack {
($target:ty, $base:ty, $call:ty, $module:ident) => {
#[cfg(feature = "std")]
mod $module {
#[allow(unused_imports)]
use super::*;
use $crate::traits::filter::{swap, take, RefCell, Vec, Box, Contains, FilterStack};
thread_local! {
static FILTER: RefCell<Vec<Box<dyn Fn(&$call) -> bool + 'static>>> = RefCell::new(Vec::new());
}
impl Contains<$call> for $target {
fn contains(call: &$call) -> bool {
<$base>::contains(call) &&
FILTER.with(|filter| filter.borrow().iter().all(|f| f(call)))
}
}
impl FilterStack<$call> for $target {
type Stack = Vec<Box<dyn Fn(&$call) -> bool + 'static>>;
fn push(f: impl Fn(&$call) -> bool + 'static) {
FILTER.with(|filter| filter.borrow_mut().push(Box::new(f)));
}
fn pop() {
FILTER.with(|filter| filter.borrow_mut().pop());
}
fn take() -> Self::Stack {
FILTER.with(|filter| take(filter.borrow_mut().as_mut()))
}
fn restore(mut s: Self::Stack) {
FILTER.with(|filter| swap(filter.borrow_mut().as_mut(), &mut s));
}
}
}
#[cfg(not(feature = "std"))]
mod $module {
#[allow(unused_imports)]
use super::*;
use $crate::traits::{swap, take, RefCell, Vec, Box, Contains, FilterStack};
struct ThisFilter(RefCell<Vec<Box<dyn Fn(&$call) -> bool + 'static>>>);
unsafe impl Send for ThisFilter {}
unsafe impl Sync for ThisFilter {}
static FILTER: ThisFilter = ThisFilter(RefCell::new(Vec::new()));
impl Contains<$call> for $target {
fn contains(call: &$call) -> bool {
<$base>::contains(call) && FILTER.0.borrow().iter().all(|f| f(call))
}
}
impl FilterStack<$call> for $target {
type Stack = Vec<Box<dyn Fn(&$call) -> bool + 'static>>;
fn push(f: impl Fn(&$call) -> bool + 'static) {
FILTER.0.borrow_mut().push(Box::new(f));
}
fn pop() {
FILTER.0.borrow_mut().pop();
}
fn take() -> Self::Stack {
take(FILTER.0.borrow_mut().as_mut())
}
fn restore(mut s: Self::Stack) {
swap(FILTER.0.borrow_mut().as_mut(), &mut s);
}
}
}
}
}
#[cfg(test)]
pub mod test_impl_filter_stack {
use super::*;
pub struct IsCallable;
pub struct BaseFilter;
impl Contains<u32> for BaseFilter {
fn contains(x: &u32) -> bool {
x % 2 == 0
}
}
impl_filter_stack!(
crate::traits::filter::test_impl_filter_stack::IsCallable,
crate::traits::filter::test_impl_filter_stack::BaseFilter,
u32,
is_callable
);
#[test]
fn impl_filter_stack_should_work() {
assert!(IsCallable::contains(&36));
assert!(IsCallable::contains(&40));
assert!(IsCallable::contains(&42));
assert!(!IsCallable::contains(&43));
IsCallable::push(|x| *x < 42);
assert!(IsCallable::contains(&36));
assert!(IsCallable::contains(&40));
assert!(!IsCallable::contains(&42));
IsCallable::push(|x| *x % 3 == 0);
assert!(IsCallable::contains(&36));
assert!(!IsCallable::contains(&40));
IsCallable::pop();
assert!(IsCallable::contains(&36));
assert!(IsCallable::contains(&40));
assert!(!IsCallable::contains(&42));
let saved = IsCallable::take();
assert!(IsCallable::contains(&36));
assert!(IsCallable::contains(&40));
assert!(IsCallable::contains(&42));
assert!(!IsCallable::contains(&43));
IsCallable::restore(saved);
assert!(IsCallable::contains(&36));
assert!(IsCallable::contains(&40));
assert!(!IsCallable::contains(&42));
IsCallable::pop();
assert!(IsCallable::contains(&36));
assert!(IsCallable::contains(&40));
assert!(IsCallable::contains(&42));
assert!(!IsCallable::contains(&43));
}
#[test]
fn guards_should_work() {
assert!(IsCallable::contains(&36));
assert!(IsCallable::contains(&40));
assert!(IsCallable::contains(&42));
assert!(!IsCallable::contains(&43));
{
let _guard_1 = FilterStackGuard::<IsCallable, u32>::new(|x| *x < 42);
assert!(IsCallable::contains(&36));
assert!(IsCallable::contains(&40));
assert!(!IsCallable::contains(&42));
{
let _guard_2 = FilterStackGuard::<IsCallable, u32>::new(|x| *x % 3 == 0);
assert!(IsCallable::contains(&36));
assert!(!IsCallable::contains(&40));
}
assert!(IsCallable::contains(&36));
assert!(IsCallable::contains(&40));
assert!(!IsCallable::contains(&42));
{
let _guard_2 = ClearFilterGuard::<IsCallable, u32>::new();
assert!(IsCallable::contains(&36));
assert!(IsCallable::contains(&40));
assert!(IsCallable::contains(&42));
assert!(!IsCallable::contains(&43));
}
assert!(IsCallable::contains(&36));
assert!(IsCallable::contains(&40));
assert!(!IsCallable::contains(&42));
}
assert!(IsCallable::contains(&36));
assert!(IsCallable::contains(&40));
assert!(IsCallable::contains(&42));
assert!(!IsCallable::contains(&43));
}
}