From 959146dbc44c9e8e48aa073bf9be2fe4bfee0ad7 Mon Sep 17 00:00:00 2001 From: Yiu Tin Cheung Ivan Date: Tue, 20 Jun 2023 11:44:38 +0800 Subject: [PATCH 1/2] added Conflicted to TryInsertResult clippy changes fmt change from basic if statement to matches --- src/executor/insert.rs | 25 ++++++++++++++++++++----- src/query/insert.rs | 10 +++++++++- tests/upsert_tests.rs | 16 +++++++++++++++- 3 files changed, 44 insertions(+), 7 deletions(-) diff --git a/src/executor/insert.rs b/src/executor/insert.rs index be00dbed3..1348f8fba 100644 --- a/src/executor/insert.rs +++ b/src/executor/insert.rs @@ -30,9 +30,9 @@ where /// The types of results for an INSERT operation #[derive(Debug)] pub enum TryInsertResult { - /// The INSERT operation did not insert any value + /// The INSERT statement did not have any value to insert Empty, - /// Reserved + /// The INSERT operation did not insert any valid value Conflicted, /// Successfully inserted Inserted(T), @@ -52,7 +52,12 @@ where if self.insert_struct.columns.is_empty() { TryInsertResult::Empty } else { - TryInsertResult::Inserted(self.insert_struct.exec(db).await) + let temp = self.insert_struct.exec(db).await; + if matches!(temp, Err(DbErr::RecordNotInserted)) { + TryInsertResult::Conflicted + } else { + TryInsertResult::Inserted(temp) + } } } @@ -70,7 +75,12 @@ where if self.insert_struct.columns.is_empty() { TryInsertResult::Empty } else { - TryInsertResult::Inserted(self.insert_struct.exec_without_returning(db).await) + let temp = self.insert_struct.exec_without_returning(db).await; + if matches!(temp, Err(DbErr::RecordNotInserted)) { + TryInsertResult::Conflicted + } else { + TryInsertResult::Inserted(temp) + } } } @@ -87,7 +97,12 @@ where if self.insert_struct.columns.is_empty() { TryInsertResult::Empty } else { - TryInsertResult::Inserted(self.insert_struct.exec_with_returning(db).await) + let temp = self.insert_struct.exec_with_returning(db).await; + if matches!(temp, Err(DbErr::RecordNotInserted)) { + TryInsertResult::Conflicted + } else { + TryInsertResult::Inserted(temp) + } } } } diff --git a/src/query/insert.rs b/src/query/insert.rs index 3fdb343c3..cf878dcd1 100644 --- a/src/query/insert.rs +++ b/src/query/insert.rs @@ -211,6 +211,14 @@ where /// Allow insert statement return safely if inserting nothing. /// The database will not be affected. + pub fn do_nothing(self) -> TryInsert + where + A: ActiveModelTrait, + { + TryInsert::from_insert(self) + } + + /// alias to do_nothing pub fn on_empty_do_nothing(self) -> TryInsert where A: ActiveModelTrait, @@ -309,7 +317,7 @@ where self } - // helper function for on_empty_do_nothing in Insert + // helper function for do_nothing in Insert pub fn from_insert(insert: Insert) -> Self { Self { insert_struct: insert, diff --git a/tests/upsert_tests.rs b/tests/upsert_tests.rs index 748d5b0a3..05009b5b9 100644 --- a/tests/upsert_tests.rs +++ b/tests/upsert_tests.rs @@ -3,6 +3,7 @@ pub mod common; pub use common::{features::*, setup::*, TestContext}; use pretty_assertions::assert_eq; use sea_orm::entity::prelude::*; +use sea_orm::TryInsertResult; use sea_orm::{sea_query::OnConflict, Set}; #[sea_orm_macros::test] @@ -50,11 +51,24 @@ pub async fn create_insert_default(db: &DatabaseConnection) -> Result<(), DbErr> ActiveModel { id: Set(3) }, ActiveModel { id: Set(4) }, ]) - .on_conflict(on_conflict) + .on_conflict(on_conflict.clone()) .exec(db) .await; assert_eq!(res.err(), Some(DbErr::RecordNotInserted)); + let res = Entity::insert_many([ + ActiveModel { id: Set(1) }, + ActiveModel { id: Set(2) }, + ActiveModel { id: Set(3) }, + ActiveModel { id: Set(4) }, + ]) + .on_conflict(on_conflict) + .do_nothing() + .exec(db) + .await; + + assert!(matches!(res, TryInsertResult::Conflicted)); + Ok(()) } From b882ce93941f56c2733b43855371ec42e8b467d7 Mon Sep 17 00:00:00 2001 From: Yiu Tin Cheung Ivan Date: Wed, 21 Jun 2023 18:17:59 +0800 Subject: [PATCH 2/2] changed to early return --- src/executor/insert.rs | 42 ++++++++++++++++++------------------------ 1 file changed, 18 insertions(+), 24 deletions(-) diff --git a/src/executor/insert.rs b/src/executor/insert.rs index 1348f8fba..1af3c3384 100644 --- a/src/executor/insert.rs +++ b/src/executor/insert.rs @@ -50,15 +50,13 @@ where A: 'a, { if self.insert_struct.columns.is_empty() { - TryInsertResult::Empty - } else { - let temp = self.insert_struct.exec(db).await; - if matches!(temp, Err(DbErr::RecordNotInserted)) { - TryInsertResult::Conflicted - } else { - TryInsertResult::Inserted(temp) - } + return TryInsertResult::Empty; + } + let temp = self.insert_struct.exec(db).await; + if matches!(temp, Err(DbErr::RecordNotInserted)) { + return TryInsertResult::Conflicted; } + TryInsertResult::Inserted(temp) } /// Execute an insert operation without returning (don't use `RETURNING` syntax) @@ -73,15 +71,13 @@ where A: 'a, { if self.insert_struct.columns.is_empty() { - TryInsertResult::Empty - } else { - let temp = self.insert_struct.exec_without_returning(db).await; - if matches!(temp, Err(DbErr::RecordNotInserted)) { - TryInsertResult::Conflicted - } else { - TryInsertResult::Inserted(temp) - } + return TryInsertResult::Empty; } + let temp = self.insert_struct.exec_without_returning(db).await; + if matches!(temp, Err(DbErr::RecordNotInserted)) { + return TryInsertResult::Conflicted; + } + TryInsertResult::Inserted(temp) } /// Execute an insert operation and return the inserted model (use `RETURNING` syntax if database supported) @@ -95,15 +91,13 @@ where A: 'a, { if self.insert_struct.columns.is_empty() { - TryInsertResult::Empty - } else { - let temp = self.insert_struct.exec_with_returning(db).await; - if matches!(temp, Err(DbErr::RecordNotInserted)) { - TryInsertResult::Conflicted - } else { - TryInsertResult::Inserted(temp) - } + return TryInsertResult::Empty; + } + let temp = self.insert_struct.exec_with_returning(db).await; + if matches!(temp, Err(DbErr::RecordNotInserted)) { + return TryInsertResult::Conflicted; } + TryInsertResult::Inserted(temp) } }