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
130
131
132
133
134
135
136
137
138
139
140
141
142
143
use connector_interface::AggregationSelection;
use indexmap::IndexMap;
use query_structure::{
    ast::FieldArity, DefaultKind, FieldSelection, PrismaValue, ScalarFieldRef, SelectedField, TypeIdentifier,
};

/// Maps field db field names to their meta information.
pub type OutputMetaMapping = IndexMap<String, OutputMeta>;

/// `OutputMeta` contains information that is required to process the output of
/// Mongo queries. With this information, we can correctly parse information and
/// coerce values as necessary / fill missing data.
#[derive(Debug, Clone)]
pub enum OutputMeta {
    Scalar(ScalarOutputMeta),
    Composite(CompositeOutputMeta),
}

#[derive(Debug, Clone)]
pub struct ScalarOutputMeta {
    pub ident: TypeIdentifier,
    pub default: Option<PrismaValue>,
    pub list: bool,
}

impl ScalarOutputMeta {
    pub fn strip_list(&self) -> Self {
        Self {
            ident: self.ident,
            default: self.default.clone(),
            list: false,
        }
    }
}

#[derive(Debug, Clone)]
pub struct CompositeOutputMeta {
    pub list: bool,
    pub inner: OutputMetaMapping,
}

impl CompositeOutputMeta {
    pub fn strip_list(&self) -> Self {
        Self {
            list: false,
            inner: self.inner.clone(),
        }
    }
}

pub fn from_selected_fields(selected_fields: &FieldSelection) -> OutputMetaMapping {
    let selections: Vec<_> = selected_fields.selections().collect();
    from_selections(&selections)
}

pub fn from_selections(selected_fields: &[&SelectedField]) -> OutputMetaMapping {
    let mut map = OutputMetaMapping::new();

    for selection in selected_fields {
        match selection {
            SelectedField::Scalar(sf) => {
                map.insert(sf.db_name().to_owned(), from_scalar_field(sf));
            }

            SelectedField::Composite(cs) => {
                let selections: Vec<&SelectedField> = cs.selections.iter().collect();
                let inner = from_selections(&selections);

                map.insert(
                    cs.field.db_name().to_owned(),
                    OutputMeta::Composite(CompositeOutputMeta {
                        list: cs.field.is_list(),
                        inner,
                    }),
                );
            }

            SelectedField::Relation(_) => unreachable!(),

            SelectedField::Virtual(vs) => {
                let (ident, arity) = vs.type_identifier_with_arity();

                map.insert(
                    vs.db_alias(),
                    OutputMeta::Scalar(ScalarOutputMeta {
                        ident,
                        default: None,
                        list: matches!(arity, FieldArity::List),
                    }),
                );
            }
        }
    }

    map
}

pub fn from_scalar_field(field: &ScalarFieldRef) -> OutputMeta {
    let (ident, field_arity) = field.type_identifier_with_arity();

    // Only add a possible default return if the field is required.
    let default = field.default_value().and_then(|dv| match dv {
        DefaultKind::Single(pv) if field.is_required() => Some(pv),
        _ => None,
    });

    OutputMeta::Scalar(ScalarOutputMeta {
        ident,
        default,
        list: matches!(field_arity, FieldArity::List),
    })
}

/// Mapping valid for one specific selection.
/// Field name -> OutputMeta
pub fn from_aggregation_selection(selection: &AggregationSelection) -> OutputMetaMapping {
    let mut map = OutputMetaMapping::new();

    for (name, ident, field_arity) in selection.identifiers() {
        map.insert(
            name,
            OutputMeta::Scalar(ScalarOutputMeta {
                ident,
                default: None,
                list: matches!(field_arity, FieldArity::List),
            }),
        );
    }

    map
}

impl From<ScalarOutputMeta> for OutputMeta {
    fn from(s: ScalarOutputMeta) -> Self {
        Self::Scalar(s)
    }
}

impl From<CompositeOutputMeta> for OutputMeta {
    fn from(c: CompositeOutputMeta) -> Self {
        Self::Composite(c)
    }
}