Skip to main content

diesel_derives/
selectable.rs

1use proc_macro2::{Span, TokenStream};
2use quote::{quote, quote_spanned};
3use std::borrow::Cow;
4use syn::spanned::Spanned;
5use syn::{DeriveInput, Ident, Result, parse_quote};
6
7use crate::field::Field;
8use crate::model::{CheckForBackend, Model};
9use crate::util::wrap_in_dummy_mod;
10
11type DefaultCheckCallback = fn(
12    &Model,
13    &syn::ImplGenerics<'_>,
14    Option<&syn::WhereClause>,
15    &[FieldSelectExpressionTyBuilder<'_>],
16) -> std::prelude::v1::Result<TokenStream, syn::Error>;
17
18pub fn derive(
19    item: DeriveInput,
20    default_check: Option<DefaultCheckCallback>,
21) -> Result<TokenStream> {
22    let model = Model::from_item(&item, false, false)?;
23
24    let (original_impl_generics, ty_generics, original_where_clause) =
25        item.generics.split_for_impl();
26
27    let mut generics = item.generics.clone();
28    generics
29        .params
30        .push(::syn::__private::parse_quote({
        let mut _s = ::quote::__private::TokenStream::new();
        ::quote::__private::push_ident(&mut _s, "__DB");
        ::quote::__private::push_colon(&mut _s);
        ::quote::__private::push_ident(&mut _s, "diesel");
        ::quote::__private::push_colon2(&mut _s);
        ::quote::__private::push_ident(&mut _s, "backend");
        ::quote::__private::push_colon2(&mut _s);
        ::quote::__private::push_ident(&mut _s, "Backend");
        _s
    })parse_quote!(__DB: diesel::backend::Backend));
31
32    for embed_field in model.fields().iter().filter(|f| f.embed()) {
33        let embed_ty = &embed_field.ty;
34        generics
35            .make_where_clause()
36            .predicates
37            .push(::syn::__private::parse_quote({
        let mut _s = ::quote::__private::TokenStream::new();
        ::quote::ToTokens::to_tokens(&embed_ty, &mut _s);
        ::quote::__private::push_colon(&mut _s);
        ::quote::__private::push_ident(&mut _s, "Selectable");
        ::quote::__private::push_lt(&mut _s);
        ::quote::__private::push_ident(&mut _s, "__DB");
        ::quote::__private::push_gt(&mut _s);
        _s
    })parse_quote!(#embed_ty: Selectable<__DB>));
38    }
39
40    let (impl_generics, _, where_clause) = generics.split_for_impl();
41
42    let struct_name = &item.ident;
43
44    let mut compile_errors: Vec<syn::Error> = Vec::new();
45    let field_select_expression_type_builders = model
46        .fields()
47        .iter()
48        .map(|f| field_select_expression_ty_builder(f, &model, &mut compile_errors))
49        .collect::<Result<Vec<_>>>()?;
50    let field_select_expression_types = field_select_expression_type_builders
51        .iter()
52        .map(|f| f.type_with_backend(&::syn::__private::parse_quote({
        let mut _s = ::quote::__private::TokenStream::new();
        ::quote::__private::push_ident(&mut _s, "__DB");
        _s
    })parse_quote!(__DB)))
53        .collect::<Vec<_>>();
54    let field_select_expressions = model
55        .fields()
56        .iter()
57        .map(|f| field_column_inst(f, &model))
58        .collect::<Result<Vec<_>>>()?;
59
60    let check_function = match model.check_for_backend.as_ref() {
61        Some(CheckForBackend::Backends(backends)) => Some(generate_check_function(
62            &model,
63            &original_impl_generics,
64            original_where_clause,
65            &field_select_expression_type_builders,
66            backends,
67            ::syn::__private::parse_quote({
        let mut _s = ::quote::__private::TokenStream::new();
        ::quote::__private::push_ident(&mut _s, "_check_field_compatibility");
        _s
    })parse_quote!(_check_field_compatibility),
68        )?),
69        Some(CheckForBackend::Disabled(_lit)) => None,
70        None => default_check
71            .map(|c| {
72                c(
73                    &model,
74                    &original_impl_generics,
75                    original_where_clause,
76                    &field_select_expression_type_builders,
77                )
78            })
79            .transpose()?,
80    };
81
82    let errors: TokenStream = compile_errors
83        .into_iter()
84        .map(|e| e.into_compile_error())
85        .collect();
86
87    Ok(wrap_in_dummy_mod({
    let mut _s = ::quote::__private::TokenStream::new();
    ::quote::__private::push_ident(&mut _s, "use");
    ::quote::__private::push_ident(&mut _s, "diesel");
    ::quote::__private::push_colon2(&mut _s);
    ::quote::__private::push_ident(&mut _s, "expression");
    ::quote::__private::push_colon2(&mut _s);
    ::quote::__private::push_ident(&mut _s, "Selectable");
    ::quote::__private::push_semi(&mut _s);
    ::quote::__private::push_ident(&mut _s, "impl");
    ::quote::ToTokens::to_tokens(&impl_generics, &mut _s);
    ::quote::__private::push_ident(&mut _s, "Selectable");
    ::quote::__private::push_lt(&mut _s);
    ::quote::__private::push_ident(&mut _s, "__DB");
    ::quote::__private::push_gt(&mut _s);
    ::quote::__private::push_ident(&mut _s, "for");
    ::quote::ToTokens::to_tokens(&struct_name, &mut _s);
    ::quote::ToTokens::to_tokens(&ty_generics, &mut _s);
    ::quote::ToTokens::to_tokens(&where_clause, &mut _s);
    ::quote::__private::push_group(&mut _s,
        ::quote::__private::Delimiter::Brace,
        {
            let mut _s = ::quote::__private::TokenStream::new();
            ::quote::__private::push_ident(&mut _s, "type");
            ::quote::__private::push_ident(&mut _s, "SelectExpression");
            ::quote::__private::push_eq(&mut _s);
            ::quote::__private::push_group(&mut _s,
                ::quote::__private::Delimiter::Parenthesis,
                {
                    let mut _s = ::quote::__private::TokenStream::new();
                    {
                        use ::quote::__private::ext::*;
                        let has_iter = ::quote::__private::HasIterator::<false>;
                        #[allow(unused_mut)]
                        let (mut field_select_expression_types, i) =
                            field_select_expression_types.quote_into_iter();
                        let has_iter = has_iter | i;
                        <_ as
                                ::quote::__private::CheckHasIterator<true>>::check(has_iter);
                        while true {
                            let field_select_expression_types =
                                match field_select_expression_types.next() {
                                    Some(_x) => ::quote::__private::RepInterp(_x),
                                    None => break,
                                };
                            ::quote::ToTokens::to_tokens(&field_select_expression_types,
                                &mut _s);
                            ::quote::__private::push_comma(&mut _s);
                        }
                    }
                    _s
                });
            ::quote::__private::push_semi(&mut _s);
            ::quote::__private::push_ident(&mut _s, "fn");
            ::quote::__private::push_ident(&mut _s, "construct_selection");
            ::quote::__private::push_group(&mut _s,
                ::quote::__private::Delimiter::Parenthesis,
                ::quote::__private::TokenStream::new());
            ::quote::__private::push_rarrow(&mut _s);
            ::quote::__private::push_ident(&mut _s, "Self");
            ::quote::__private::push_colon2(&mut _s);
            ::quote::__private::push_ident(&mut _s, "SelectExpression");
            ::quote::__private::push_group(&mut _s,
                ::quote::__private::Delimiter::Brace,
                {
                    let mut _s = ::quote::__private::TokenStream::new();
                    ::quote::__private::push_group(&mut _s,
                        ::quote::__private::Delimiter::Parenthesis,
                        {
                            let mut _s = ::quote::__private::TokenStream::new();
                            {
                                use ::quote::__private::ext::*;
                                let has_iter = ::quote::__private::HasIterator::<false>;
                                #[allow(unused_mut)]
                                let (mut field_select_expressions, i) =
                                    field_select_expressions.quote_into_iter();
                                let has_iter = has_iter | i;
                                <_ as
                                        ::quote::__private::CheckHasIterator<true>>::check(has_iter);
                                while true {
                                    let field_select_expressions =
                                        match field_select_expressions.next() {
                                            Some(_x) => ::quote::__private::RepInterp(_x),
                                            None => break,
                                        };
                                    ::quote::ToTokens::to_tokens(&field_select_expressions,
                                        &mut _s);
                                    ::quote::__private::push_comma(&mut _s);
                                }
                            }
                            _s
                        });
                    _s
                });
            _s
        });
    ::quote::ToTokens::to_tokens(&check_function, &mut _s);
    ::quote::ToTokens::to_tokens(&errors, &mut _s);
    _s
}quote! {
88        use diesel::expression::Selectable;
89
90        impl #impl_generics Selectable<__DB>
91            for #struct_name #ty_generics
92        #where_clause
93        {
94            type SelectExpression = (#(#field_select_expression_types,)*);
95
96            fn construct_selection() -> Self::SelectExpression {
97                (#(#field_select_expressions,)*)
98            }
99        }
100
101        #check_function
102
103        #errors
104    }))
105}
106
107pub fn generate_check_function(
108    model: &Model,
109    original_impl_generics: &syn::ImplGenerics<'_>,
110    original_where_clause: Option<&syn::WhereClause>,
111    field_select_expression_type_builders: &[FieldSelectExpressionTyBuilder<'_>],
112    backends: &syn::punctuated::Punctuated<syn::TypePath, syn::token::Comma>,
113    function_name: Ident,
114) -> Result<TokenStream> {
115    let field_check_bound = model
116        .fields()
117        .iter()
118        .zip(field_select_expression_type_builders)
119        .flat_map(|(f, ty_builder)| {
120            backends.iter().map(move |b| {
121                let span = Span::mixed_site().located_at(f.ty.span());
122                let field_ty = to_field_ty_bound(f.ty_for_deserialize())?;
123                let ty = ty_builder.type_with_backend(b);
124                Ok(::syn::__private::parse_quote({
        let mut _s = ::quote::__private::TokenStream::new();
        let _span: ::quote::__private::Span =
            ::quote::__private::get_span(span).__into_span();
        ::quote::ToTokens::to_tokens(&field_ty, &mut _s);
        ::quote::__private::push_colon_spanned(&mut _s, _span);
        ::quote::__private::push_ident_spanned(&mut _s, _span, "diesel");
        ::quote::__private::push_colon2_spanned(&mut _s, _span);
        ::quote::__private::push_ident_spanned(&mut _s, _span, "deserialize");
        ::quote::__private::push_colon2_spanned(&mut _s, _span);
        ::quote::__private::push_ident_spanned(&mut _s, _span, "FromSqlRow");
        ::quote::__private::push_lt_spanned(&mut _s, _span);
        ::quote::__private::push_ident_spanned(&mut _s, _span, "diesel");
        ::quote::__private::push_colon2_spanned(&mut _s, _span);
        ::quote::__private::push_ident_spanned(&mut _s, _span, "dsl");
        ::quote::__private::push_colon2_spanned(&mut _s, _span);
        ::quote::__private::push_ident_spanned(&mut _s, _span, "SqlTypeOf");
        ::quote::__private::push_lt_spanned(&mut _s, _span);
        ::quote::ToTokens::to_tokens(&ty, &mut _s);
        ::quote::__private::push_gt_spanned(&mut _s, _span);
        ::quote::__private::push_comma_spanned(&mut _s, _span);
        ::quote::ToTokens::to_tokens(&b, &mut _s);
        ::quote::__private::push_gt_spanned(&mut _s, _span);
        _s
    })syn::parse_quote_spanned! {span =>
125                    #field_ty: diesel::deserialize::FromSqlRow<diesel::dsl::SqlTypeOf<#ty>, #b>
126                })
127            })
128        })
129        .collect::<Result<Vec<_>>>()?;
130    let where_clause = &mut original_where_clause.cloned();
131    let where_clause = where_clause.get_or_insert_with(|| ::syn::__private::parse_quote({
        let mut _s = ::quote::__private::TokenStream::new();
        ::quote::__private::push_ident(&mut _s, "where");
        _s
    })parse_quote!(where));
132    for field_check in field_check_bound {
133        where_clause.predicates.push(field_check);
134    }
135    Ok({
    let mut _s = ::quote::__private::TokenStream::new();
    ::quote::__private::push_ident(&mut _s, "fn");
    ::quote::ToTokens::to_tokens(&function_name, &mut _s);
    ::quote::ToTokens::to_tokens(&original_impl_generics, &mut _s);
    ::quote::__private::push_group(&mut _s,
        ::quote::__private::Delimiter::Parenthesis,
        ::quote::__private::TokenStream::new());
    ::quote::ToTokens::to_tokens(&where_clause, &mut _s);
    ::quote::__private::push_group(&mut _s,
        ::quote::__private::Delimiter::Brace,
        ::quote::__private::TokenStream::new());
    _s
}quote::quote! {
136        fn #function_name #original_impl_generics()
137            #where_clause
138        {}
139    })
140}
141
142fn to_field_ty_bound(field_ty: &syn::Type) -> Result<TokenStream> {
143    match field_ty {
144        syn::Type::Reference(r) => {
145            use crate::quote::ToTokens;
146            // references are not supported for checking for now
147            //
148            // (How ever you can even have references in a `Queryable` struct anyway)
149            Err(syn::Error::new(
150                field_ty.span(),
151                ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("references are not supported in `Queryable` types\nconsider using `std::borrow::Cow<\'{0}, {1}>` instead",
                r.lifetime.as_ref().expect("It's a struct field so it must have a named lifetime").ident,
                r.elem.to_token_stream()))
    })format!(
152                    "references are not supported in `Queryable` types\n\
153                         consider using `std::borrow::Cow<'{}, {}>` instead",
154                    r.lifetime
155                        .as_ref()
156                        .expect("It's a struct field so it must have a named lifetime")
157                        .ident,
158                    r.elem.to_token_stream()
159                ),
160            ))
161        }
162        field_ty => Ok({
    let mut _s = ::quote::__private::TokenStream::new();
    ::quote::ToTokens::to_tokens(&field_ty, &mut _s);
    _s
}quote::quote! {
163            #field_ty
164        }),
165    }
166}
167
168fn field_select_expression_ty_builder<'a>(
169    field: &'a Field,
170    model: &Model,
171    compile_errors: &mut Vec<syn::Error>,
172) -> Result<FieldSelectExpressionTyBuilder<'a>> {
173    if let Some(ref select_expression) = field.select_expression {
174        use dsl_auto_type::auto_type::expression_type_inference as type_inference;
175        let expr = &select_expression.item;
176        let (inferred_type, errors) = type_inference::infer_expression_type(
177            expr,
178            field.select_expression_type.as_ref().map(|t| &t.item),
179            &type_inference::InferrerSettings::builder()
180                .dsl_path(::syn::__private::parse_quote({
        let mut _s = ::quote::__private::TokenStream::new();
        ::quote::__private::push_ident(&mut _s, "diesel");
        ::quote::__private::push_colon2(&mut _s);
        ::quote::__private::push_ident(&mut _s, "dsl");
        _s
    })parse_quote!(diesel::dsl))
181                .function_types_case(crate::AUTO_TYPE_DEFAULT_FUNCTION_TYPE_CASE)
182                .method_types_case(crate::AUTO_TYPE_DEFAULT_METHOD_TYPE_CASE)
183                .build(),
184        );
185        compile_errors.extend(errors);
186        Ok(FieldSelectExpressionTyBuilder::Always(
187            {
    let mut _s = ::quote::__private::TokenStream::new();
    ::quote::ToTokens::to_tokens(&inferred_type, &mut _s);
    _s
}quote::quote!(#inferred_type),
188        ))
189    } else if let Some(ref select_expression_type) = field.select_expression_type {
190        let ty = &select_expression_type.item;
191        Ok(FieldSelectExpressionTyBuilder::Always({
    let mut _s = ::quote::__private::TokenStream::new();
    ::quote::ToTokens::to_tokens(&ty, &mut _s);
    _s
}quote!(#ty)))
192    } else if field.embed() {
193        Ok(FieldSelectExpressionTyBuilder::EmbedSelectable {
194            embed_ty: &field.ty,
195        })
196    } else {
197        let table_name = &model.table_names()[0];
198        let column_name = field.column_name()?.to_ident()?;
199        let span = Span::call_site();
200        Ok(FieldSelectExpressionTyBuilder::Always(
201            {
    let mut _s = ::quote::__private::TokenStream::new();
    let _span: ::quote::__private::Span =
        ::quote::__private::get_span(span).__into_span();
    ::quote::ToTokens::to_tokens(&table_name, &mut _s);
    ::quote::__private::push_colon2_spanned(&mut _s, _span);
    ::quote::ToTokens::to_tokens(&column_name, &mut _s);
    _s
}quote_spanned!(span=> #table_name::#column_name),
202        ))
203    }
204}
205
206pub enum FieldSelectExpressionTyBuilder<'a> {
207    Always(TokenStream),
208    EmbedSelectable { embed_ty: &'a syn::Type },
209}
210
211impl FieldSelectExpressionTyBuilder<'_> {
212    fn type_with_backend(&self, backend: &syn::TypePath) -> Cow<'_, TokenStream> {
213        match self {
214            FieldSelectExpressionTyBuilder::Always(ty) => Cow::Borrowed(ty),
215            FieldSelectExpressionTyBuilder::EmbedSelectable { embed_ty } => {
216                Cow::Owned({
    let mut _s = ::quote::__private::TokenStream::new();
    ::quote::__private::push_lt(&mut _s);
    ::quote::ToTokens::to_tokens(&embed_ty, &mut _s);
    ::quote::__private::push_ident(&mut _s, "as");
    ::quote::__private::push_ident(&mut _s, "Selectable");
    ::quote::__private::push_lt(&mut _s);
    ::quote::ToTokens::to_tokens(&backend, &mut _s);
    ::quote::__private::push_shr(&mut _s);
    ::quote::__private::push_colon2(&mut _s);
    ::quote::__private::push_ident(&mut _s, "SelectExpression");
    _s
}quote!(<#embed_ty as Selectable<#backend>>::SelectExpression))
217            }
218        }
219    }
220}
221
222fn field_column_inst(field: &Field, model: &Model) -> Result<TokenStream> {
223    if let Some(ref select_expression) = field.select_expression {
224        let expr = &select_expression.item;
225        Ok({
    let mut _s = ::quote::__private::TokenStream::new();
    ::quote::ToTokens::to_tokens(&expr, &mut _s);
    _s
}quote!(#expr))
226    } else if field.embed() {
227        let embed_ty = &field.ty;
228        Ok({
    let mut _s = ::quote::__private::TokenStream::new();
    ::quote::__private::push_lt(&mut _s);
    ::quote::ToTokens::to_tokens(&embed_ty, &mut _s);
    ::quote::__private::push_ident(&mut _s, "as");
    ::quote::__private::push_ident(&mut _s, "Selectable");
    ::quote::__private::push_lt(&mut _s);
    ::quote::__private::push_ident(&mut _s, "__DB");
    ::quote::__private::push_shr(&mut _s);
    ::quote::__private::push_colon2(&mut _s);
    ::quote::__private::push_ident(&mut _s, "construct_selection");
    ::quote::__private::push_group(&mut _s,
        ::quote::__private::Delimiter::Parenthesis,
        ::quote::__private::TokenStream::new());
    _s
}quote!(<#embed_ty as Selectable<__DB>>::construct_selection()))
229    } else {
230        let table_name = &model.table_names()[0];
231        let column_name = field.column_name()?.to_ident()?;
232        let span = Span::call_site();
233        Ok({
    let mut _s = ::quote::__private::TokenStream::new();
    let _span: ::quote::__private::Span =
        ::quote::__private::get_span(span).__into_span();
    ::quote::ToTokens::to_tokens(&table_name, &mut _s);
    ::quote::__private::push_colon2_spanned(&mut _s, _span);
    ::quote::ToTokens::to_tokens(&column_name, &mut _s);
    _s
}quote_spanned!(span=> #table_name::#column_name))
234    }
235}