Skip to main content

diesel_derives/
util.rs

1use proc_macro2::TokenStream;
2use quote::quote;
3use syn::parse::{Parse, ParseStream, Peek, Result};
4use syn::token::Eq;
5use syn::{Data, DeriveInput, GenericArgument, Ident, Type, parenthesized, parse_quote};
6
7use crate::model::Model;
8
9pub const COLUMN_NAME_NOTE: &str = "column_name = foo";
10pub const SQL_TYPE_NOTE: &str = "sql_type = Foo";
11pub const SERIALIZE_AS_NOTE: &str = "serialize_as = Foo";
12pub const DESERIALIZE_AS_NOTE: &str = "deserialize_as = Foo";
13pub const TABLE_NAME_NOTE: &str = "table_name = foo";
14pub const TREAT_NONE_AS_DEFAULT_VALUE_NOTE: &str = "treat_none_as_default_value = true";
15pub const TREAT_NONE_AS_NULL_NOTE: &str = "treat_none_as_null = true";
16pub const BELONGS_TO_NOTE: &str = "belongs_to(Foo, foreign_key = foo_id)";
17pub const MYSQL_TYPE_NOTE: &str = "mysql_type(name = \"foo\")";
18pub const SQLITE_TYPE_NOTE: &str = "sqlite_type(name = \"foo\")";
19pub const POSTGRES_TYPE_NOTE: &str = "postgres_type(name = \"foo\", schema = \"public\")";
20pub const POSTGRES_TYPE_NOTE_ID: &str = "postgres_type(oid = 37, array_oid = 54)";
21pub const SELECT_EXPRESSION_NOTE: &str =
22    "select_expression = schema::table_name::column_name.is_not_null()";
23pub const SELECT_EXPRESSION_TYPE_NOTE: &str =
24    "select_expression_type = dsl::IsNotNull<schema::table_name::column_name>";
25pub const CHECK_FOR_BACKEND_NOTE: &str = "diesel::pg::Pg";
26pub const BASE_QUERY_NOTE: &str =
27    "base_query = schema::table_name::table.order_by(schema::table_name::id)";
28pub const BASE_QUERY_TYPE_NOTE: &str =
29    "base_query_type = dsl::OrderBy<schema::table_name::table, schema::table_name::id>";
30pub const RENAME_ALL_NOTE: &str = "rename_all = \"camelCase\"";
31pub const RENAME_NOTE: &str = "rename = \"your_name\"";
32
33pub fn unknown_attribute(name: &Ident, valid: &[&str]) -> syn::Error {
34    let prefix = if valid.len() == 1 { "" } else { " one of" };
35
36    syn::Error::new(
37        name.span(),
38        ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("unknown attribute, expected{1} `{0}`",
                valid.join("`, `"), prefix))
    })format!(
39            "unknown attribute, expected{prefix} `{}`",
40            valid.join("`, `")
41        ),
42    )
43}
44
45pub fn parse_eq<T: Parse>(input: ParseStream, help: &str) -> Result<T> {
46    if input.is_empty() {
47        return Err(syn::Error::new(
48            input.span(),
49            ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("unexpected end of input, expected `=`\nhelp: the correct format looks like `#[diesel({0})]`",
                help))
    })format!(
50                "unexpected end of input, expected `=`\n\
51                 help: the correct format looks like `#[diesel({help})]`",
52            ),
53        ));
54    }
55
56    input.parse::<Eq>()?;
57    input.parse()
58}
59
60/// Specialized version of `parse_eq` for `syn::Type` with a customized error message for readability.
61/// This is useful because a great variety of tokens would be valid to parse as a `syn::Type`.
62pub fn parse_eq_type(input: ParseStream, help: &str) -> Result<Type> {
63    if input.is_empty() {
64        return Err(syn::Error::new(
65            input.span(),
66            ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("unexpected end of input, expected `=`\nhelp: the correct format looks like `#[diesel({0})]`",
                help))
    })format!(
67                "unexpected end of input, expected `=`\n\
68                 help: the correct format looks like `#[diesel({help})]`",
69            ),
70        ));
71    }
72
73    input.parse::<Eq>()?;
74    input
75        .parse::<Type>()
76        .map_err(|e| syn::Error::new(e.span(), "expected type"))
77}
78
79pub fn parse_paren<T: Parse>(input: ParseStream, help: &str) -> Result<T> {
80    if input.is_empty() {
81        return Err(syn::Error::new(
82            input.span(),
83            ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("unexpected end of input, expected parentheses\nhelp: the correct format looks like `#[diesel({0})]`",
                help))
    })format!(
84                "unexpected end of input, expected parentheses\n\
85                 help: the correct format looks like `#[diesel({help})]`",
86            ),
87        ));
88    }
89
90    let content;
91    match ::syn::__private::parse_parens(&input) {
    ::syn::__private::Ok(parens) => {
        content = parens.content;
        _ = content;
        parens.token
    }
    ::syn::__private::Err(error) => { return ::syn::__private::Err(error); }
};parenthesized!(content in input);
92    content.parse()
93}
94
95pub fn parse_paren_list<T, D>(
96    input: ParseStream,
97    help: &str,
98    sep: D,
99) -> Result<syn::punctuated::Punctuated<T, <D as Peek>::Token>>
100where
101    T: Parse,
102    D: Peek,
103    D::Token: Parse,
104{
105    if input.is_empty() {
106        return Err(syn::Error::new(
107            input.span(),
108            ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("unexpected end of input, expected parentheses\nhelp: the correct format looks like `#[diesel({0})]`",
                help))
    })format!(
109                "unexpected end of input, expected parentheses\n\
110                 help: the correct format looks like `#[diesel({help})]`",
111            ),
112        ));
113    }
114
115    let content;
116    match ::syn::__private::parse_parens(&input) {
    ::syn::__private::Ok(parens) => {
        content = parens.content;
        _ = content;
        parens.token
    }
    ::syn::__private::Err(error) => { return ::syn::__private::Err(error); }
};parenthesized!(content in input);
117    content.parse_terminated(T::parse, sep)
118}
119
120pub fn wrap_in_dummy_mod(item: TokenStream) -> TokenStream {
121    {
    let mut _s = ::quote::__private::TokenStream::new();
    ::quote::__private::push_ident(&mut _s, "const");
    ::quote::__private::push_underscore(&mut _s);
    ::quote::__private::push_colon(&mut _s);
    ::quote::__private::push_group(&mut _s,
        ::quote::__private::Delimiter::Parenthesis,
        ::quote::__private::TokenStream::new());
    ::quote::__private::push_eq(&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, "use");
            ::quote::__private::push_ident(&mut _s, "diesel");
            ::quote::__private::push_semi(&mut _s);
            ::quote::ToTokens::to_tokens(&item, &mut _s);
            _s
        });
    ::quote::__private::push_semi(&mut _s);
    _s
}quote! {
122        const _: () = {
123            // This import is not actually redundant. When using diesel_derives
124            // inside of diesel, `diesel` doesn't exist as an extern crate, and
125            // to work around that it contains a private
126            // `mod diesel { pub use super::*; }` that this import will then
127            // refer to. In all other cases, this imports refers to the extern
128            // crate diesel.
129            use diesel;
130
131            #item
132        };
133    }
134}
135
136pub fn inner_of_option_ty(ty: &Type) -> &Type {
137    option_ty_arg(ty).unwrap_or(ty)
138}
139
140pub fn is_option_ty(ty: &Type) -> bool {
141    option_ty_arg(ty).is_some()
142}
143
144fn option_ty_arg(mut ty: &Type) -> Option<&Type> {
145    use syn::PathArguments::AngleBracketed;
146
147    // Check the inner equivalent type
148    loop {
149        match ty {
150            Type::Group(group) => ty = &group.elem,
151            Type::Paren(paren) => ty = &paren.elem,
152            _ => break,
153        }
154    }
155
156    match *ty {
157        Type::Path(ref ty) => {
158            let last_segment = ty.path.segments.iter().next_back().unwrap();
159            match last_segment.arguments {
160                AngleBracketed(ref args) if last_segment.ident == "Option" => {
161                    match args.args.iter().next_back() {
162                        Some(GenericArgument::Type(ty)) => Some(ty),
163                        _ => None,
164                    }
165                }
166                _ => None,
167            }
168        }
169        _ => None,
170    }
171}
172
173pub fn ty_for_foreign_derive(item: &DeriveInput, model: &Model) -> Result<Type> {
174    if model.foreign_derive {
175        match item.data {
176            Data::Struct(ref body) => match body.fields.iter().next() {
177                Some(field) => Ok(field.ty.clone()),
178                None => Err(syn::Error::new(
179                    proc_macro2::Span::mixed_site(),
180                    "foreign_derive requires at least one field",
181                )),
182            },
183            _ => Err(syn::Error::new(
184                proc_macro2::Span::mixed_site(),
185                "foreign_derive can only be used with structs",
186            )),
187        }
188    } else {
189        let ident = &item.ident;
190        let (_, ty_generics, ..) = item.generics.split_for_impl();
191        Ok(::syn::__private::parse_quote({
        let mut _s = ::quote::__private::TokenStream::new();
        ::quote::ToTokens::to_tokens(&ident, &mut _s);
        ::quote::ToTokens::to_tokens(&ty_generics, &mut _s);
        _s
    })parse_quote!(#ident #ty_generics))
192    }
193}
194
195pub fn camel_to_snake(name: &str) -> String {
196    let mut result = String::with_capacity(name.len());
197    result.push_str(&name[..1].to_lowercase());
198    for character in name[1..].chars() {
199        if character.is_uppercase() {
200            result.push('_');
201            for lowercase in character.to_lowercase() {
202                result.push(lowercase);
203            }
204        } else {
205            result.push(character);
206        }
207    }
208    result
209}