- 支持试读
内容介绍
本专题将通过实现一个自动生成数据库 CRUD 的 Derive宏 来对过程宏 `proc-macro` 进行一步步的探讨。我们希望通过本专题的学习,能让你掌握 rust 过程宏的知识要点,并将其应用到实际开发中。 - 支持试读
解析 derive(Db)
本章我们将开始实现第一步:解析 `derive(Db)`。 - 支持试读
解析目标结构体的元数据
本章我们将讨论如何解析目标结构体的元数据,包括:结构体的名称、结构体的字段(包括可见性、字段名和数据类型)。 解析Derive和字段的属性,并实现 CRUD 操作中的插入
本章我们将实现 CRUD 操作中的【插入】。在实际开发中,有些字段是不需要插入的,比如自动编号的主键。我们可以通过宏属性来指定哪些字段不需要插入。同时,我们还要通过宏属性来指定目标结构体的表名、主键以及是否为视图等。- 支持试读
实现更新和删除方法
有了上一章的基础,我们实现更新和删除方法也不是难事。 实现单条数据的查找
本章我们将讨论如何用宏为目标结构体实现单条数据的查找。实现数据列表和分页
本章我们将讨论如何用宏为目标结构体实现数据列表和分页。
实现更新和删除方法
有了上一章的基础,我们实现更新和删除方法也不是难事。
实现更新方法
pub(crate) fn update_ts(dm: &DbMeta) -> proc_macro2::TokenStream {
let field_list = dm.update_fileds();
let field_list_str = field_list
.iter()
.map(|f| format!("{:?} = ", f.to_string()))
.collect::<Vec<_>>();
let field_list_com = field_list
.iter()
.enumerate()
.map(|(idx, _)| format!("{}", if idx < field_list.len() - 1 { ", " } else { "" }))
.collect::<Vec<_>>();
let table = dm.table.clone();
let sql = format!("UPDATE {:?} SET ", &table,);
let pk = dm.pk_ident().clone();
let pk_str = pk.to_string();
quote! {
pub async fn update<'a>(&self, e: impl ::sqlx::PgExecutor<'a>) -> ::sqlx::Result<u64> {
let sql = #sql;
let mut q = ::sqlx::QueryBuilder::new(sql);
#(
q.push(#field_list_str)
.push_bind(&self.#field_list)
.push(#field_list_com);
)*
q.push(format!(" WHERE {} = ", #pk_str)).push_bind(&self.#pk);
let aff = q.build().execute(e).await?.rows_affected();
Ok(aff)
}
}
}
pub(crate) fn del_ts(dm: &DbMeta) -> proc_macro2::TokenStream {
let table = dm.table.clone();
let pk = dm.pk_ident().clone();
let pk_str = pk.to_string();
let sql = format!("DELETE FROM {:?} WHERE {:?} = ", &table, &pk_str);
quote! {
pub async fn delete<'a>(&self, e: impl ::sqlx::PgExecutor<'a>) -> ::sqlx::Result<u64> {
let sql = #sql;
let mut q = ::sqlx::QueryBuilder::new(sql);
q.push_bind(&self.#pk);
let aff = q.build().execute(e).await?.rows_affected();
Ok(aff)
}
}
}