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
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
mod read;
mod write;

pub use read::*;
pub use write::*;

use crate::ToGraphviz;
use query_structure::{FieldSelection, Filter, Model, SelectionResult};

#[derive(Debug, Clone)]
#[allow(clippy::large_enum_variant)]
pub(crate) enum Query {
    Read(ReadQuery),
    Write(WriteQuery),
}

impl Query {
    pub(crate) fn satisfies(&self, fields: &FieldSelection) -> bool {
        match self {
            Self::Read(rq) => rq.satisfies(fields),
            Self::Write(wq) => wq.satisfies(fields),
        }
    }

    pub(crate) fn satisfy_dependency(&mut self, fields: FieldSelection) {
        match self {
            Self::Read(rq) => rq.satisfy_dependency(fields),
            Self::Write(wq) => wq.satisfy_dependency(fields),
        }
    }

    pub(crate) fn model(&self) -> Model {
        match self {
            Self::Read(rq) => rq.model(),
            Self::Write(wq) => wq.model(),
        }
    }

    pub(crate) fn is_update_one(&self) -> bool {
        matches!(self, Query::Write(WriteQuery::UpdateRecord(_)))
    }

    pub(crate) fn is_create_one(&self) -> bool {
        matches!(self, Query::Write(WriteQuery::CreateRecord(_)))
    }

    pub(crate) fn is_delete_one(&self) -> bool {
        matches!(self, Query::Write(WriteQuery::CreateRecord(_)))
    }
}

impl FilteredQuery for Query {
    fn get_filter(&mut self) -> Option<&mut Filter> {
        match self {
            Self::Read(rq) => rq.get_filter(),
            Self::Write(wq) => wq.get_filter(),
        }
    }

    fn set_filter(&mut self, filter: Filter) {
        match self {
            Self::Read(rq) => rq.set_filter(filter),
            Self::Write(wq) => wq.set_filter(filter),
        }
    }
}

pub trait FilteredQuery {
    fn add_filter<T>(&mut self, filter: T)
    where
        T: Into<Filter>,
    {
        let filter = filter.into();
        let existing_filter = self.get_filter();
        let filter = match existing_filter {
            Some(Filter::And(ref mut v)) => {
                v.push(filter);
                None
            }
            Some(Filter::Or(ref mut v)) => {
                v.push(filter);
                None
            }
            Some(Filter::Not(ref mut v)) => {
                v.push(filter);
                None
            }
            Some(Filter::Empty) => Some(filter),
            Some(other) => Some(Self::default_filter_behaviour(vec![other.clone(), filter])),
            None => Some(filter),
        };

        if let Some(filter) = filter {
            self.set_filter(filter);
        }
    }

    fn get_filter(&mut self) -> Option<&mut Filter>;
    fn set_filter(&mut self, filter: Filter);

    fn default_filter_behaviour(inner_filters: Vec<Filter>) -> Filter {
        Filter::Or(inner_filters)
    }
}

pub trait FilteredNestedMutation {
    /// Sets the parent id of a nested mutation.
    /// This indicates the connector that it doesn't need to refetch the child and that the mutation
    /// can directly be performed using that id.
    fn set_selectors(&mut self, selectors: Vec<SelectionResult>);
}

impl std::fmt::Display for Query {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        match self {
            Self::Read(q) => write!(f, "{q}"),
            Self::Write(q) => write!(f, "{q}"),
        }
    }
}

impl ToGraphviz for Query {
    fn to_graphviz(&self) -> String {
        match self {
            Self::Read(q) => q.to_graphviz(),
            Self::Write(q) => q.to_graphviz(),
        }
    }
}