kosame_dsl 0.3.0

Macro-based Rust ORM focused on developer ergonomics
Documentation
use crate::{
    expr::{self, visit_bind_param},
    query::Query,
    statement::Statement,
    visit::Visit,
};
use proc_macro2::TokenStream;
use quote::{ToTokens, format_ident, quote};
use syn::Ident;

struct BindParamsBuilder<'a> {
    params: Vec<&'a Ident>,
}

impl BindParamsBuilder<'_> {
    fn new() -> Self {
        Self { params: Vec::new() }
    }
}

impl<'a> Visit<'a> for BindParamsBuilder<'a> {
    fn visit_bind_param(&mut self, bind_param: &'a expr::BindParam) {
        if !self.params.contains(&&bind_param.name) {
            self.params.push(&bind_param.name);
        }
        visit_bind_param(self, bind_param);
    }
}

pub struct BindParams<'a> {
    params: Vec<&'a Ident>,
}

impl BindParams<'_> {
    #[must_use]
    pub fn is_empty(&self) -> bool {
        self.params.is_empty()
    }
}

impl<'a> From<&'a Statement> for BindParams<'a> {
    fn from(value: &'a Statement) -> Self {
        let mut builder = BindParamsBuilder::new();
        builder.visit_statement(value);
        Self {
            params: builder.params,
        }
    }
}

impl<'a> From<&'a Query> for BindParams<'a> {
    fn from(value: &'a Query) -> Self {
        let mut builder = BindParamsBuilder::new();
        builder.visit_node(&value.body);
        Self {
            params: builder.params,
        }
    }
}

impl ToTokens for BindParams<'_> {
    fn to_tokens(&self, tokens: &mut TokenStream) {
        let mut modules = vec![];
        for (ordinal, name) in self.params.iter().enumerate() {
            let ordinal = u32::try_from(ordinal).unwrap();
            let name_string = name.to_string();
            modules.push(quote! {
                pub(super) mod #name {
                    pub const BIND_PARAM: ::kosame::repr::expr::BindParam<'_> = ::kosame::repr::expr::BindParam::new(#name_string, #ordinal);
                }
            });
        }

        let mut fields = vec![];
        for name in &self.params {
            fields.push(quote! {
                #name: &'a (dyn ::kosame::driver::postgres_types::ToSql + ::std::marker::Sync)
            });
        }
        let fields_len = fields.len();

        let lifetime = (fields_len > 0).then(|| quote! { <'a> });

        quote! {
            mod params {
                #(#modules)*
            }

            #[derive(Debug, Clone)]
            pub struct Params #lifetime {
                #(pub #fields),*
            }
        }
        .to_tokens(tokens);

        #[cfg(any(feature = "postgres", feature = "tokio-postgres"))]
        {
            let field_names = &self.params;
            quote! {
            impl<'a> ::kosame::params::Params<Vec<&'a (dyn ::kosame::driver::postgres_types::ToSql + ::std::marker::Sync + 'a)>> for Params #lifetime {
                fn to_driver(&self) -> Vec<&'a (dyn ::kosame::driver::postgres_types::ToSql + ::std::marker::Sync + 'a)> {
                    vec![#(self.#field_names),*]
                }
            }
        }.to_tokens(tokens);
        }
    }
}

pub struct BindParamsClosure<'a> {
    module_name: &'a Ident,
    bind_params: &'a BindParams<'a>,
}

impl<'a> BindParamsClosure<'a> {
    #[must_use]
    pub fn new(module_name: &'a Ident, bind_params: &'a BindParams<'a>) -> Self {
        Self {
            module_name,
            bind_params,
        }
    }
}

impl ToTokens for BindParamsClosure<'_> {
    fn to_tokens(&self, tokens: &mut TokenStream) {
        let module_name = self.module_name;

        let mut rename_vars = vec![];
        let mut struct_fields = vec![];
        for (ordinal, name) in self.bind_params.params.iter().enumerate() {
            let renamed = format_ident!("bind_param_{}", ordinal);
            rename_vars.push(quote! { let #renamed = &#name; });
            struct_fields.push(quote! { #name: #renamed });
        }

        quote! {
            #(#rename_vars)*

            let closure = #module_name::Params {
                #(#struct_fields,)*
            };
        }
        .to_tokens(tokens);
    }
}