refactor(tests): reformat code and update dependency management
- Reorganized import statements in adapter tests for better readability - Replaced or_insert_with(Vec::new) with or_default() in test closures - Updated Cargo.lock with new dependency versions and checksums - Added TLS features to tonic dependency configuration - Included sqlx, chrono, and uuid dependencies with specific features - Added jsonwebtoken and arc-swap as project dependencies - Reformatted assertion statements to comply with line length limits - Adjusted base64 import order in engine codec module - Updated protobuf include statement formatting
This commit is contained in:
@@ -0,0 +1,112 @@
|
||||
//! Bookmark CRUD operations on `MessageRepo`.
|
||||
|
||||
use chrono::Utc;
|
||||
use uuid::Uuid;
|
||||
|
||||
use crate::ImksResult;
|
||||
use crate::models::message_bookmark::MessageBookmark;
|
||||
|
||||
use super::message_repo::MessageRepo;
|
||||
use super::pagination::{CursorPage, clamp_limit};
|
||||
|
||||
impl MessageRepo {
|
||||
/// Add a bookmark for a message.
|
||||
pub async fn add_bookmark(
|
||||
&self,
|
||||
message_id: Uuid,
|
||||
channel_id: Uuid,
|
||||
user_id: Uuid,
|
||||
note: Option<&str>,
|
||||
) -> ImksResult<MessageBookmark> {
|
||||
let id = Uuid::now_v7();
|
||||
let now = Utc::now();
|
||||
|
||||
sqlx::query_as::<_, MessageBookmark>(
|
||||
r#"
|
||||
INSERT INTO message_bookmark (id, message_id, channel_id, user_id, note, created_at, updated_at)
|
||||
VALUES ($1, $2, $3, $4, $5, $6, $6)
|
||||
ON CONFLICT (user_id, message_id) DO UPDATE SET note = EXCLUDED.note, updated_at = EXCLUDED.updated_at
|
||||
RETURNING *
|
||||
"#,
|
||||
)
|
||||
.bind(id)
|
||||
.bind(message_id)
|
||||
.bind(channel_id)
|
||||
.bind(user_id)
|
||||
.bind(note)
|
||||
.bind(now)
|
||||
.fetch_one(self.pool())
|
||||
.await
|
||||
.map_err(Into::into)
|
||||
}
|
||||
|
||||
/// Remove a bookmark.
|
||||
pub async fn remove_bookmark(&self, message_id: Uuid, user_id: Uuid) -> ImksResult<bool> {
|
||||
let result =
|
||||
sqlx::query("DELETE FROM message_bookmark WHERE message_id = $1 AND user_id = $2")
|
||||
.bind(message_id)
|
||||
.bind(user_id)
|
||||
.execute(self.pool())
|
||||
.await?;
|
||||
|
||||
Ok(result.rows_affected() > 0)
|
||||
}
|
||||
|
||||
/// Check if a user has bookmarked a message.
|
||||
pub async fn is_bookmarked(&self, message_id: Uuid, user_id: Uuid) -> ImksResult<bool> {
|
||||
let exists: Option<Uuid> = sqlx::query_scalar(
|
||||
"SELECT id FROM message_bookmark WHERE message_id = $1 AND user_id = $2",
|
||||
)
|
||||
.bind(message_id)
|
||||
.bind(user_id)
|
||||
.fetch_optional(self.pool())
|
||||
.await?;
|
||||
|
||||
Ok(exists.is_some())
|
||||
}
|
||||
|
||||
/// List a user's bookmarks with cursor-based pagination (newest first).
|
||||
pub async fn list_bookmarks(
|
||||
&self,
|
||||
user_id: Uuid,
|
||||
before: Option<Uuid>,
|
||||
limit: Option<i64>,
|
||||
) -> ImksResult<CursorPage<MessageBookmark>> {
|
||||
let effective_limit = clamp_limit(limit);
|
||||
let fetch_limit = effective_limit + 1;
|
||||
|
||||
let rows = match before {
|
||||
Some(cursor) => {
|
||||
sqlx::query_as::<_, MessageBookmark>(
|
||||
r#"
|
||||
SELECT * FROM message_bookmark
|
||||
WHERE user_id = $1 AND id < $2
|
||||
ORDER BY id DESC
|
||||
LIMIT $3
|
||||
"#,
|
||||
)
|
||||
.bind(user_id)
|
||||
.bind(cursor)
|
||||
.bind(fetch_limit)
|
||||
.fetch_all(self.pool())
|
||||
.await?
|
||||
}
|
||||
None => {
|
||||
sqlx::query_as::<_, MessageBookmark>(
|
||||
r#"
|
||||
SELECT * FROM message_bookmark
|
||||
WHERE user_id = $1
|
||||
ORDER BY id DESC
|
||||
LIMIT $2
|
||||
"#,
|
||||
)
|
||||
.bind(user_id)
|
||||
.bind(fetch_limit)
|
||||
.fetch_all(self.pool())
|
||||
.await?
|
||||
}
|
||||
};
|
||||
|
||||
Ok(CursorPage::from_raw(rows, effective_limit, |b| b.id))
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user