-
-
Notifications
You must be signed in to change notification settings - Fork 277
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #1089 from AppFlowy-IO/quick-notes-api
feat: quick note CRUD APIs
- Loading branch information
Showing
15 changed files
with
646 additions
and
2 deletions.
There are no files selected for viewing
42 changes: 42 additions & 0 deletions
42
.sqlx/query-35622d4ebede28dd28b613edcf3970ad258286f176ce86e88bd662a602e4ad58.json
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
14 changes: 14 additions & 0 deletions
14
.sqlx/query-5cce5f82c0fb9237f724478e2167243bc772c092910f07b8226431a6dd70a7da.json
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
15 changes: 15 additions & 0 deletions
15
.sqlx/query-770a4979e137ca08c5ea625259221f9d397a56defb8e498eb92da7b3a8af612b.json
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,90 @@ | ||
use client_api_entity::{ | ||
CreateQuickNoteParams, ListQuickNotesQueryParams, QuickNote, QuickNotes, UpdateQuickNoteParams, | ||
}; | ||
use reqwest::Method; | ||
use shared_entity::response::{AppResponse, AppResponseError}; | ||
use uuid::Uuid; | ||
|
||
use crate::Client; | ||
|
||
fn quick_note_resources_url(base_url: &str, workspace_id: Uuid) -> String { | ||
format!("{base_url}/api/workspace/{workspace_id}/quick-note") | ||
} | ||
|
||
fn quick_note_resource_url(base_url: &str, workspace_id: Uuid, quick_note_id: Uuid) -> String { | ||
let quick_note_resources_prefix = quick_note_resources_url(base_url, workspace_id); | ||
format!("{quick_note_resources_prefix}/{quick_note_id}") | ||
} | ||
|
||
// Quick Note API | ||
impl Client { | ||
pub async fn create_quick_note( | ||
&self, | ||
workspace_id: Uuid, | ||
data: Option<serde_json::Value>, | ||
) -> Result<QuickNote, AppResponseError> { | ||
let url = quick_note_resources_url(&self.base_url, workspace_id); | ||
let resp = self | ||
.http_client_with_auth(Method::POST, &url) | ||
.await? | ||
.json(&CreateQuickNoteParams { data }) | ||
.send() | ||
.await?; | ||
AppResponse::<QuickNote>::from_response(resp) | ||
.await? | ||
.into_data() | ||
} | ||
|
||
pub async fn list_quick_notes( | ||
&self, | ||
workspace_id: Uuid, | ||
search_term: Option<String>, | ||
offset: Option<i32>, | ||
limit: Option<i32>, | ||
) -> Result<QuickNotes, AppResponseError> { | ||
let url = quick_note_resources_url(&self.base_url, workspace_id); | ||
let resp = self | ||
.http_client_with_auth(Method::GET, &url) | ||
.await? | ||
.query(&ListQuickNotesQueryParams { | ||
search_term, | ||
offset, | ||
limit, | ||
}) | ||
.send() | ||
.await?; | ||
AppResponse::<QuickNotes>::from_response(resp) | ||
.await? | ||
.into_data() | ||
} | ||
|
||
pub async fn update_quick_note( | ||
&self, | ||
workspace_id: Uuid, | ||
quick_note_id: Uuid, | ||
data: serde_json::Value, | ||
) -> Result<(), AppResponseError> { | ||
let url = quick_note_resource_url(&self.base_url, workspace_id, quick_note_id); | ||
let resp = self | ||
.http_client_with_auth(Method::PUT, &url) | ||
.await? | ||
.json(&UpdateQuickNoteParams { data }) | ||
.send() | ||
.await?; | ||
AppResponse::<()>::from_response(resp).await?.into_error() | ||
} | ||
|
||
pub async fn delete_quick_note( | ||
&self, | ||
workspace_id: Uuid, | ||
quick_note_id: Uuid, | ||
) -> Result<(), AppResponseError> { | ||
let url = quick_note_resource_url(&self.base_url, workspace_id, quick_note_id); | ||
let resp = self | ||
.http_client_with_auth(Method::DELETE, &url) | ||
.await? | ||
.send() | ||
.await?; | ||
AppResponse::<()>::from_response(resp).await?.into_error() | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,104 @@ | ||
use app_error::AppError; | ||
use database_entity::dto::QuickNote; | ||
use sqlx::{Executor, Postgres, QueryBuilder}; | ||
use uuid::Uuid; | ||
|
||
use crate::pg_row::AFQuickNoteRow; | ||
|
||
pub async fn insert_new_quick_note<'a, E: Executor<'a, Database = Postgres>>( | ||
executor: E, | ||
workspace_id: Uuid, | ||
uid: i64, | ||
data: &serde_json::Value, | ||
) -> Result<QuickNote, AppError> { | ||
let quick_note = sqlx::query_as!( | ||
QuickNote, | ||
r#" | ||
INSERT INTO af_quick_note (workspace_id, uid, data) VALUES ($1, $2, $3) | ||
RETURNING quick_note_id AS id, data, created_at AS "created_at!", updated_at AS "last_updated_at!" | ||
"#, | ||
workspace_id, | ||
uid, | ||
data | ||
) | ||
.fetch_one(executor) | ||
.await?; | ||
Ok(quick_note) | ||
} | ||
|
||
pub async fn select_quick_notes_with_one_more_than_limit< | ||
'a, | ||
E: Executor<'a, Database = Postgres>, | ||
>( | ||
executor: E, | ||
workspace_id: Uuid, | ||
uid: i64, | ||
search_term: Option<String>, | ||
offset: Option<i32>, | ||
limit: Option<i32>, | ||
) -> Result<Vec<QuickNote>, AppError> { | ||
let mut query_builder: QueryBuilder<Postgres> = QueryBuilder::new( | ||
r#" | ||
SELECT | ||
quick_note_id, | ||
data, | ||
created_at, | ||
updated_at | ||
FROM af_quick_note WHERE workspace_id = | ||
"#, | ||
); | ||
query_builder.push_bind(workspace_id); | ||
query_builder.push(" AND uid = "); | ||
query_builder.push_bind(uid); | ||
if let Some(search_term) = search_term.filter(|term| !term.is_empty()) { | ||
query_builder.push(" AND data @? "); | ||
let json_path_query = format!("'$.**.insert ? (@ like_regex \".*{}.*\")'", search_term); | ||
query_builder.push(json_path_query); | ||
} | ||
query_builder.push(" ORDER BY created_at DESC"); | ||
if let Some(limit) = limit { | ||
query_builder.push(" LIMIT "); | ||
query_builder.push_bind(limit); | ||
query_builder.push(" + 1 "); | ||
} | ||
if let Some(offset) = offset { | ||
query_builder.push(" OFFSET "); | ||
query_builder.push_bind(offset); | ||
} | ||
let query = query_builder.build_query_as::<AFQuickNoteRow>(); | ||
let quick_notes_with_one_more_than_limit = query | ||
.fetch_all(executor) | ||
.await? | ||
.into_iter() | ||
.map(Into::into) | ||
.collect(); | ||
Ok(quick_notes_with_one_more_than_limit) | ||
} | ||
|
||
pub async fn update_quick_note_by_id<'a, E: Executor<'a, Database = Postgres>>( | ||
executor: E, | ||
quick_note_id: Uuid, | ||
data: &serde_json::Value, | ||
) -> Result<(), AppError> { | ||
sqlx::query!( | ||
"UPDATE af_quick_note SET data = $1, updated_at = NOW() WHERE quick_note_id = $2", | ||
data, | ||
quick_note_id | ||
) | ||
.execute(executor) | ||
.await?; | ||
Ok(()) | ||
} | ||
|
||
pub async fn delete_quick_note_by_id<'a, E: Executor<'a, Database = Postgres>>( | ||
executor: E, | ||
quick_note_id: Uuid, | ||
) -> Result<(), AppError> { | ||
sqlx::query!( | ||
"DELETE FROM af_quick_note WHERE quick_note_id = $1", | ||
quick_note_id | ||
) | ||
.execute(executor) | ||
.await?; | ||
Ok(()) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,17 @@ | ||
CREATE TABLE IF NOT EXISTS af_quick_note ( | ||
quick_note_id UUID NOT NULL DEFAULT gen_random_uuid (), | ||
workspace_id UUID NOT NULL, | ||
uid BIGINT NOT NULL REFERENCES af_user (uid) ON DELETE CASCADE, | ||
updated_at TIMESTAMP | ||
WITH | ||
TIME ZONE DEFAULT CURRENT_TIMESTAMP, | ||
created_at TIMESTAMP | ||
WITH | ||
TIME ZONE DEFAULT CURRENT_TIMESTAMP, | ||
data JSONB NOT NULL, | ||
PRIMARY KEY (quick_note_id) | ||
); | ||
|
||
CREATE INDEX IF NOT EXISTS idx_workspace_id_on_af_quick_note ON af_quick_note (workspace_id); | ||
|
||
CREATE INDEX IF NOT EXISTS idx_uid_on_af_quick_note ON af_quick_note (uid); |
Oops, something went wrong.