3个版本
0.0.3 | 2023年2月5日 |
---|---|
0.0.2 | 2023年2月5日 |
0.0.1 | 2023年1月8日 |
#2 in #querystring
48KB
1K SLoC
seaqs
一个将查询字符串转换为seaquery条件的迷你工具。
描述
假设我们有一个用户表,我们想要为某些管理面板提供一个REST端点。使用这个crate,我们可以定义一个过滤器结构体,并用它来与sea_query(或sea_orm,sqlx也应该可以)一起使用。
use serde::Deserialize;
use sea_query::{Iden, Cond, Query, PostgresQueryBuilder};
use seaqs::{ApplyConds, ToCond, ToFieldCond, filters::*};
use serde_querystring::{from_str, de::ParseMode};
// It's part of the sea_query definition of a table.
#[derive(Iden)]
enum User {
Table,
Id,
Name,
Age,
Birthday,
CreatedAt
}
// And we define a filter struct like below
#[derive(Deserialize)]
struct UserFilters<'a> {
id: Option<UuidFilterSet>,
name: Option<StringFilterSet<'a>>,
age: Option<NumberFilterSet>,
birthday: Option<DateFilterSet>,
created_at: Option<DateTimeFilterSet>,
}
// Then we should impl the 'ToCond' trait, which should be done using a macro but there isn't one yet.
impl<'a> ToCond for UserFilters<'a> {
fn to_cond(&self) -> Cond {
let mut cond = Cond::all();
if let Some(id) = self.id.to_cond(User::Id) {
cond = cond.add(id)
}
if let Some(name) = self.name.to_cond(User::Name) {
cond = cond.add(name)
}
if let Some(age) = self.age.to_cond(User::Age) {
cond = cond.add(age)
}
if let Some(birthday) = self.birthday.to_cond(User::Birthday) {
cond = cond.add(birthday)
}
if let Some(created_at) = self.created_at.to_cond(User::CreatedAt) {
cond = cond.add(created_at)
}
cond
}
}
// I'm using serde_querystring here, but serde_json works too(whatever works with serde_with, works here)
let filters = from_str::<UserFilters>(
"age[lt]=50&age[gte]=20&name[contains]=John",
ParseMode::Brackets,
)
.unwrap();
// And create your query normally
let q = Query::select()
.column(User::Name)
.from(User::Table)
// Just use ApplyConds trait from seaqs
.apply_conds(&filters)
// You shouldn't use to_string, it's just here for the test
.to_string(PostgresQueryBuilder);
assert_eq!(
q,
r#"SELECT "name" FROM "user" WHERE "name" LIKE '%John%' AND ("age" >= 20 AND "age" < 50)"#
);
// You can also use the provided QueryFilter to add sort/order/page/limit to your query. It's designed to work well with react-admin or similar admin panels.
use seaqs::{ApplyFilters, QueryFilter, Filter};
// You need to impl Filter for it to work
impl<'a> Filter for UserFilters<'a> {
const SORTABLE_FIELDS: &'static [&'static str] = &["name", "age", "created_at"];
fn get_max_limit() -> i32 {
100
}
}
// Notice that we need to use the `filter` key now.
let filters = from_str::<QueryFilter<UserFilters>>(
"filter[age][lt]=50&filter[age][gte]=20&filter[name][contains]=John&start=10&end=100&sort=age&order=DESC",
ParseMode::Brackets,
)
.unwrap();
// And create your query normally
let q = Query::select()
.column(User::Name)
.from(User::Table)
// Just use ApplyFilters trait from seaqs
.apply_filters(&filters)
// You shouldn't use to_string, it's just here for the test
.to_string(PostgresQueryBuilder);
assert_eq!(
q,
r#"SELECT "name" FROM "user" WHERE "name" LIKE '%John%' AND ("age" >= 20 AND "age" < 50) ORDER BY "age" DESC LIMIT 90 OFFSET 10"#
)
依赖
~4–10MB
~103K SLoC