use super::*;
use crate::{Relation, RelationType};
use heck::ToSnakeCase;
impl EntityWriter {
#[allow(clippy::too_many_arguments)]
pub fn gen_dense_code_blocks(
entity: &Entity,
with_serde: &WithSerde,
column_option: &ColumnOption,
schema_name: &Option<String>,
serde_skip_deserializing_primary_key: bool,
serde_skip_hidden_column: bool,
model_extra_derives: &TokenStream,
model_extra_attributes: &TokenStream,
_column_extra_derives: &TokenStream,
_seaography: bool,
impl_active_model_behavior: bool,
) -> Vec<TokenStream> {
let mut imports = Self::gen_import(with_serde);
imports.extend(Self::gen_import_active_enum(entity));
let mut code_blocks = vec![
imports,
Self::gen_dense_model_struct(
entity,
with_serde,
column_option,
schema_name,
serde_skip_deserializing_primary_key,
serde_skip_hidden_column,
model_extra_derives,
model_extra_attributes,
),
];
if impl_active_model_behavior {
code_blocks.push(Self::impl_active_model_behavior());
}
code_blocks
}
#[allow(clippy::too_many_arguments)]
pub fn gen_dense_model_struct(
entity: &Entity,
with_serde: &WithSerde,
column_option: &ColumnOption,
schema_name: &Option<String>,
serde_skip_deserializing_primary_key: bool,
serde_skip_hidden_column: bool,
model_extra_derives: &TokenStream,
model_extra_attributes: &TokenStream,
) -> TokenStream {
let table_name = entity.table_name.as_str();
let column_names_snake_case = entity.get_column_names_snake_case();
let column_rs_types = entity.get_column_rs_types(column_option);
let if_eq_needed = entity.get_eq_needed();
let primary_keys: Vec<String> = entity
.primary_keys
.iter()
.map(|pk| pk.name.clone())
.collect();
let attrs: Vec<TokenStream> = entity
.columns
.iter()
.map(|col| {
let mut attrs: Punctuated<_, Comma> = Punctuated::new();
let is_primary_key = primary_keys.contains(&col.name);
if !col.is_snake_case_name() {
let column_name = &col.name;
attrs.push(quote! { column_name = #column_name });
}
if is_primary_key {
attrs.push(quote! { primary_key });
if !col.auto_increment {
attrs.push(quote! { auto_increment = false });
}
}
if let Some(ts) = col.get_col_type_attrs() {
attrs.extend([ts]);
if !col.not_null {
attrs.push(quote! { nullable });
}
};
if col.unique {
attrs.push(quote! { unique });
} else if let Some(unique_key) = &col.unique_key {
attrs.push(quote! { unique_key = #unique_key });
}
let mut ts = quote! {};
if !attrs.is_empty() {
for (i, attr) in attrs.into_iter().enumerate() {
if i > 0 {
ts = quote! { #ts, };
}
ts = quote! { #ts #attr };
}
ts = quote! { #[sea_orm(#ts)] };
}
let serde_attribute = col.get_serde_attribute(
is_primary_key,
serde_skip_deserializing_primary_key,
serde_skip_hidden_column,
);
ts = quote! {
#ts
#serde_attribute
};
ts
})
.collect();
let schema_name = match Self::gen_schema_name(schema_name) {
Some(schema_name) => quote! {
schema_name = #schema_name,
},
None => quote! {},
};
let extra_derive = with_serde.extra_derive();
let mut compound_objects: Punctuated<_, Comma> = Punctuated::new();
let map_col = |a: &syn::Ident| -> String {
let a = a.to_string();
let b = a.to_snake_case();
if a != b.to_upper_camel_case() {
a
} else {
b
}
};
let map_punctuated = |punctuated: Vec<String>| -> String {
let len = punctuated.len();
let punctuated = punctuated.join(", ");
match len {
0..=1 => punctuated,
_ => format!("({punctuated})"),
}
};
let via_entities = entity.get_conjunct_relations_via_snake_case();
for rel in entity.relations.iter() {
if !rel.self_referencing && rel.impl_related {
let (rel_type, sea_orm_attr) = match rel.rel_type {
RelationType::HasOne => (format_ident!("HasOne"), quote!(#[sea_orm(has_one)])),
RelationType::HasMany => {
(format_ident!("HasMany"), quote!(#[sea_orm(has_many)]))
}
RelationType::BelongsTo => {
let (from, to) = rel.get_src_ref_columns(map_col, map_col, map_punctuated);
let on_update = if let Some(action) = &rel.on_update {
let action = Relation::get_foreign_key_action(action);
quote!(, on_update = #action)
} else {
quote!()
};
let on_delete = if let Some(action) = &rel.on_delete {
let action = Relation::get_foreign_key_action(action);
quote!(, on_delete = #action)
} else {
quote!()
};
let relation_enum = if rel.num_suffix > 0 {
let relation_enum = rel.get_enum_name().to_string();
quote!(relation_enum = #relation_enum,)
} else {
quote!()
};
(
format_ident!("HasOne"),
quote!(#[sea_orm(belongs_to, #relation_enum from = #from, to = #to #on_update #on_delete)]),
)
}
};
if let Some(to_entity) = rel.get_module_name()
&& !via_entities.contains(&to_entity)
{
let field = if matches!(rel.rel_type, RelationType::HasMany) {
format_ident!(
"{}",
pluralizer::pluralize(&to_entity.to_string(), 2, false)
)
} else {
to_entity.clone()
};
let field = if rel.num_suffix == 0 {
field
} else {
format_ident!("{field}_{}", rel.num_suffix)
};
compound_objects.push(quote! {
#sea_orm_attr
pub #field: #rel_type<super::#to_entity::Entity>
});
}
} else if rel.self_referencing {
let (from, to) = rel.get_src_ref_columns(map_col, map_col, map_punctuated);
let on_update = if let Some(action) = &rel.on_update {
let action = Relation::get_foreign_key_action(action);
quote!(, on_update = #action)
} else {
quote!()
};
let on_delete = if let Some(action) = &rel.on_delete {
let action = Relation::get_foreign_key_action(action);
quote!(, on_delete = #action)
} else {
quote!()
};
let relation_enum = rel.get_enum_name().to_string();
let field = format_ident!(
"{}{}",
entity.get_table_name_snake_case_ident(),
if rel.num_suffix > 0 {
format!("_{}", rel.num_suffix)
} else {
"".into()
}
);
compound_objects.push(quote! {
#[sea_orm(self_ref, relation_enum = #relation_enum, from = #from, to = #to #on_update #on_delete)]
pub #field: HasOne<Entity>
});
}
}
for (to_entity, via_entity) in entity
.get_conjunct_relations_to_snake_case()
.into_iter()
.zip(via_entities)
{
let field = format_ident!(
"{}",
pluralizer::pluralize(&to_entity.to_string(), 2, false)
);
let via_entity = via_entity.to_string();
compound_objects.push(quote! {
#[sea_orm(has_many, via = #via_entity)]
pub #field: HasMany<super::#to_entity::Entity>
});
}
if !compound_objects.is_empty() {
compound_objects.push_punct(<syn::Token![,]>::default());
}
quote! {
#[sea_orm::model]
#[derive(Clone, Debug, PartialEq #if_eq_needed, DeriveEntityModel #extra_derive #model_extra_derives)]
#[sea_orm(
#schema_name
table_name = #table_name
)]
#model_extra_attributes
pub struct Model {
#(
#attrs
pub #column_names_snake_case: #column_rs_types,
)*
#compound_objects
}
}
}
#[allow(dead_code)]
fn gen_dense_related_entity(entity: &Entity) -> TokenStream {
let via_entities = entity.get_conjunct_relations_via_snake_case();
let related_modules = entity.get_related_entity_modules();
let related_attrs = entity.get_related_entity_attrs();
let related_enum_names = entity.get_related_entity_enum_name();
let items = related_modules
.into_iter()
.zip(related_attrs)
.zip(related_enum_names)
.filter_map(|((related_module, related_attr), related_enum_name)| {
if !via_entities.contains(&related_module) {
Some(quote!(#related_attr #related_enum_name))
} else {
None
}
})
.collect::<Vec<_>>();
quote! {
#[derive(Copy, Clone, Debug, EnumIter, DeriveRelatedEntity)]
pub enum RelatedEntity {
#(#items),*
}
}
}
}
#[cfg(test)]
mod test {
#[test]
#[ignore]
fn test_name() {
panic!("{}", pluralizer::pluralize("filling", 2, false));
}
}