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 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}