refactor(postgres): format FeedSqlBuilder for improved readability

This commit is contained in:
2026-05-29 00:13:55 +02:00
parent 145b07d636
commit 8229285a2f

View File

@@ -100,13 +100,17 @@ fn row_to_entry(r: FeedRow, viewer: Option<uuid::Uuid>) -> Result<FeedEntry, Dom
struct FeedSqlBuilder<'a> { struct FeedSqlBuilder<'a> {
options: &'a FeedOptions, options: &'a FeedOptions,
scope: &'a FeedScope, scope: &'a FeedScope,
viewer: Option<uuid::Uuid>, viewer: Option<uuid::Uuid>,
} }
impl<'a> FeedSqlBuilder<'a> { impl<'a> FeedSqlBuilder<'a> {
fn new(options: &'a FeedOptions, scope: &'a FeedScope, viewer: Option<uuid::Uuid>) -> Self { fn new(options: &'a FeedOptions, scope: &'a FeedScope, viewer: Option<uuid::Uuid>) -> Self {
Self { options, scope, viewer } Self {
options,
scope,
viewer,
}
} }
fn select(&self) -> String { fn select(&self) -> String {
@@ -165,10 +169,18 @@ impl<'a> FeedSqlBuilder<'a> {
fn filter_sql(&self) -> String { fn filter_sql(&self) -> String {
let f = &self.options.filter; let f = &self.options.filter;
let mut s = String::new(); let mut s = String::new();
if f.originals_only { s += " AND t.in_reply_to_id IS NULL"; } if f.originals_only {
if f.replies_only { s += " AND t.in_reply_to_id IS NOT NULL"; } s += " AND t.in_reply_to_id IS NULL";
if f.local_only { s += " AND t.local = true"; } }
if f.hide_sensitive { s += " AND t.sensitive = false"; } if f.replies_only {
s += " AND t.in_reply_to_id IS NOT NULL";
}
if f.local_only {
s += " AND t.local = true";
}
if f.hide_sensitive {
s += " AND t.sensitive = false";
}
s s
} }
@@ -177,37 +189,40 @@ impl<'a> FeedSqlBuilder<'a> {
return "ORDER BY similarity(t.content, $1) DESC"; return "ORDER BY similarity(t.content, $1) DESC";
} }
match &self.options.sort { match &self.options.sort {
FeedSort::Newest => "ORDER BY t.created_at DESC", FeedSort::Newest => "ORDER BY t.created_at DESC",
FeedSort::Oldest => "ORDER BY t.created_at ASC", FeedSort::Oldest => "ORDER BY t.created_at ASC",
FeedSort::MostLiked => "ORDER BY like_count DESC, t.created_at DESC", FeedSort::MostLiked => "ORDER BY like_count DESC, t.created_at DESC",
FeedSort::MostBoosted => "ORDER BY boost_count DESC, t.created_at DESC", FeedSort::MostBoosted => "ORDER BY boost_count DESC, t.created_at DESC",
FeedSort::MostDiscussed => "ORDER BY reply_count DESC, t.created_at DESC", FeedSort::MostDiscussed => "ORDER BY reply_count DESC, t.created_at DESC",
} }
} }
fn public(&self) -> (String, String) { fn public(&self) -> (String, String) {
let filter = self.filter_sql(); let filter = self.filter_sql();
let order = self.order_sql(); let order = self.order_sql();
let count = format!( let count = format!(
"SELECT COUNT(*) FROM thoughts t WHERE t.local=true AND t.visibility='public'{}", "SELECT COUNT(*) FROM thoughts t WHERE t.local=true AND t.visibility='public'{}",
filter filter
); );
let data = format!( let data = format!(
"{} WHERE t.local=true AND t.visibility='public'{} {} LIMIT $1 OFFSET $2", "{} WHERE t.local=true AND t.visibility='public'{} {} LIMIT $1 OFFSET $2",
self.select(), filter, order self.select(),
filter,
order
); );
(count, data) (count, data)
} }
fn home(&self) -> (String, String) { fn home(&self) -> (String, String) {
let fed = self.fed_clause(); let fed = self.fed_clause();
let filter = self.filter_sql(); let filter = self.filter_sql();
let order = self.order_sql(); let order = self.order_sql();
let count = format!( let count = format!(
"SELECT COUNT(*) FROM thoughts t WHERE (t.user_id=ANY($1){}) AND t.visibility != 'direct'{}", "SELECT COUNT(*) FROM thoughts t WHERE (t.user_id=ANY($1){}) AND t.visibility != 'direct'{}",
fed, filter fed, filter
); );
let data = format!( let data =
format!(
"{} WHERE (t.user_id=ANY($1){}) AND t.visibility != 'direct'{} {} LIMIT $2 OFFSET $3", "{} WHERE (t.user_id=ANY($1){}) AND t.visibility != 'direct'{} {} LIMIT $2 OFFSET $3",
self.select(), fed, filter, order self.select(), fed, filter, order
); );
@@ -216,22 +231,24 @@ impl<'a> FeedSqlBuilder<'a> {
fn search(&self) -> (String, String) { fn search(&self) -> (String, String) {
let filter = self.filter_sql(); let filter = self.filter_sql();
let order = self.order_sql(); let order = self.order_sql();
let count = format!( let count = format!(
"SELECT COUNT(*) FROM thoughts t WHERE t.content % $1 AND t.visibility='public'{}", "SELECT COUNT(*) FROM thoughts t WHERE t.content % $1 AND t.visibility='public'{}",
filter filter
); );
let data = format!( let data = format!(
"{} WHERE t.content % $1 AND t.visibility='public'{} {} LIMIT $2 OFFSET $3", "{} WHERE t.content % $1 AND t.visibility='public'{} {} LIMIT $2 OFFSET $3",
self.select(), filter, order self.select(),
filter,
order
); );
(count, data) (count, data)
} }
fn tag(&self) -> (String, String) { fn tag(&self) -> (String, String) {
let filter = self.filter_sql(); let filter = self.filter_sql();
let order = self.order_sql(); let order = self.order_sql();
let count = format!( let count = format!(
"SELECT COUNT(*) FROM thoughts t "SELECT COUNT(*) FROM thoughts t
JOIN thought_tags tt ON tt.thought_id = t.id JOIN thought_tags tt ON tt.thought_id = t.id
JOIN tags tg ON tg.id = tt.tag_id JOIN tags tg ON tg.id = tt.tag_id
@@ -243,14 +260,16 @@ impl<'a> FeedSqlBuilder<'a> {
JOIN thought_tags tt ON tt.thought_id = t.id JOIN thought_tags tt ON tt.thought_id = t.id
JOIN tags tg ON tg.id = tt.tag_id JOIN tags tg ON tg.id = tt.tag_id
WHERE tg.name = $1 AND t.visibility = 'public'{} {} LIMIT $2 OFFSET $3", WHERE tg.name = $1 AND t.visibility = 'public'{} {} LIMIT $2 OFFSET $3",
self.select(), filter, order self.select(),
filter,
order
); );
(count, data) (count, data)
} }
fn user(&self) -> (String, String) { fn user(&self) -> (String, String) {
let filter = self.filter_sql(); let filter = self.filter_sql();
let order = self.order_sql(); let order = self.order_sql();
let count = format!( let count = format!(
"SELECT COUNT(*) FROM thoughts t WHERE t.user_id = $1 AND ($2::uuid = $1 OR (t.visibility != 'direct' AND (t.visibility IN ('public', 'unlisted') OR (t.visibility = 'followers' AND EXISTS(SELECT 1 FROM follows WHERE follower_id = $2 AND following_id = $1 AND state = 'accepted'))))){}", "SELECT COUNT(*) FROM thoughts t WHERE t.user_id = $1 AND ($2::uuid = $1 OR (t.visibility != 'direct' AND (t.visibility IN ('public', 'unlisted') OR (t.visibility = 'followers' AND EXISTS(SELECT 1 FROM follows WHERE follower_id = $2 AND following_id = $1 AND state = 'accepted'))))){}",
filter filter
@@ -266,8 +285,8 @@ impl<'a> FeedSqlBuilder<'a> {
#[async_trait] #[async_trait]
impl FeedRepository for PgFeedRepository { impl FeedRepository for PgFeedRepository {
async fn query(&self, req: &FeedRequest) -> Result<Paginated<FeedEntry>, DomainError> { async fn query(&self, req: &FeedRequest) -> Result<Paginated<FeedEntry>, DomainError> {
let viewer = req.query.viewer_id.as_ref().map(|v| v.as_uuid()); let viewer = req.query.viewer_id.as_ref().map(|v| v.as_uuid());
let page = &req.query.page; let page = &req.query.page;
let builder = FeedSqlBuilder::new(&req.options, &req.query.scope, viewer); let builder = FeedSqlBuilder::new(&req.options, &req.query.scope, viewer);
match &req.query.scope { match &req.query.scope {
@@ -287,8 +306,13 @@ impl FeedRepository for PgFeedRepository {
.await .await
.into_domain()?; .into_domain()?;
Ok(Paginated { Ok(Paginated {
items: rows.into_iter().map(|r| row_to_entry(r, viewer)).collect::<Result<Vec<_>, _>>()?, items: rows
total, page: page.page, per_page: page.per_page, .into_iter()
.map(|r| row_to_entry(r, viewer))
.collect::<Result<Vec<_>, _>>()?,
total,
page: page.page,
per_page: page.per_page,
}) })
} }
@@ -305,8 +329,13 @@ impl FeedRepository for PgFeedRepository {
.await .await
.into_domain()?; .into_domain()?;
Ok(Paginated { Ok(Paginated {
items: rows.into_iter().map(|r| row_to_entry(r, viewer)).collect::<Result<Vec<_>, _>>()?, items: rows
total, page: page.page, per_page: page.per_page, .into_iter()
.map(|r| row_to_entry(r, viewer))
.collect::<Result<Vec<_>, _>>()?,
total,
page: page.page,
per_page: page.per_page,
}) })
} }
@@ -325,8 +354,13 @@ impl FeedRepository for PgFeedRepository {
.await .await
.into_domain()?; .into_domain()?;
Ok(Paginated { Ok(Paginated {
items: rows.into_iter().map(|r| row_to_entry(r, viewer)).collect::<Result<Vec<_>, _>>()?, items: rows
total, page: page.page, per_page: page.per_page, .into_iter()
.map(|r| row_to_entry(r, viewer))
.collect::<Result<Vec<_>, _>>()?,
total,
page: page.page,
per_page: page.per_page,
}) })
} }
@@ -345,13 +379,18 @@ impl FeedRepository for PgFeedRepository {
.await .await
.into_domain()?; .into_domain()?;
Ok(Paginated { Ok(Paginated {
items: rows.into_iter().map(|r| row_to_entry(r, viewer)).collect::<Result<Vec<_>, _>>()?, items: rows
total, page: page.page, per_page: page.per_page, .into_iter()
.map(|r| row_to_entry(r, viewer))
.collect::<Result<Vec<_>, _>>()?,
total,
page: page.page,
per_page: page.per_page,
}) })
} }
FeedScope::User { user_id } => { FeedScope::User { user_id } => {
let uid = user_id.as_uuid(); let uid = user_id.as_uuid();
let viewer_uuid = viewer.unwrap_or(uuid::Uuid::nil()); let viewer_uuid = viewer.unwrap_or(uuid::Uuid::nil());
let (count_sql, data_sql) = builder.user(); let (count_sql, data_sql) = builder.user();
let total: i64 = sqlx::query_scalar(&count_sql) let total: i64 = sqlx::query_scalar(&count_sql)
@@ -369,8 +408,13 @@ impl FeedRepository for PgFeedRepository {
.await .await
.into_domain()?; .into_domain()?;
Ok(Paginated { Ok(Paginated {
items: rows.into_iter().map(|r| row_to_entry(r, viewer)).collect::<Result<Vec<_>, _>>()?, items: rows
total, page: page.page, per_page: page.per_page, .into_iter()
.map(|r| row_to_entry(r, viewer))
.collect::<Result<Vec<_>, _>>()?,
total,
page: page.page,
per_page: page.per_page,
}) })
} }
} }