1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
use super::FilterCollector;
use crate::diesel_ext::BoxableFilter;
use crate::query_builder::selection::filter::build_filter::BuildFilter;
use diesel::backend::Backend;
use diesel::query_builder::QueryFragment;
use diesel::{AppearsOnTable, BoolExpressionMethods};
use std::fmt::{self, Debug};
pub struct AndCollector<'a, T, DB>(
Option<Box<dyn BoxableFilter<T, DB, SqlType = ::diesel::sql_types::Bool> + 'a>>,
);
impl<'a, T, DB> Debug for AndCollector<'a, T, DB>
where
DB: Backend,
DB::QueryBuilder: Default,
{
fn fmt(&self, fmt: &mut fmt::Formatter<'_>) -> fmt::Result {
fmt.debug_tuple("AndCollector")
.field(&self.0.as_ref().map(|q| ::diesel::debug_query(q)))
.finish()
}
}
impl<'a, T, DB> Default for AndCollector<'a, T, DB> {
fn default() -> Self {
AndCollector(None)
}
}
impl<'a, T, DB> FilterCollector<'a, T, DB> for AndCollector<'a, T, DB>
where
DB: Backend + 'a,
T: 'a,
{
fn append_filter<F>(&mut self, f: F)
where
F: BuildFilter<DB> + 'a,
F::Ret: AppearsOnTable<T> + QueryFragment<DB> + 'a,
{
let f = f.into_filter();
let c = ::std::mem::replace(&mut self.0, None);
self.0 = match (c, f) {
(Some(c), Some(f)) => Some(Box::new(c.and(f)) as Box<_>),
(Some(c), None) => Some(c),
(None, Some(f)) => Some(Box::new(f) as Box<_>),
(None, None) => None,
};
}
}
impl<'a, T, DB> BuildFilter<DB> for AndCollector<'a, T, DB>
where
DB: Backend,
{
type Ret = Box<dyn BoxableFilter<T, DB, SqlType = ::diesel::sql_types::Bool> + 'a>;
fn into_filter(self) -> Option<Self::Ret> {
self.0
}
}