From 181a045a13d3e7ae9bcfcc702577d4aedc3419a8 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Stefan=20Gr=C3=B6nke?= Date: Sun, 13 Oct 2024 15:58:55 +0200 Subject: [PATCH 1/2] implement IfElseStatement for MySQL backend --- src/backend/query_builder.rs | 15 ++++++++++++ src/expr.rs | 3 ++- src/if_else.rs | 33 ++++++++++++++++++++++++++ src/lib.rs | 2 ++ tests/mysql/if_else.rs | 45 ++++++++++++++++++++++++++++++++++++ tests/mysql/mod.rs | 1 + 6 files changed, 98 insertions(+), 1 deletion(-) create mode 100644 src/if_else.rs create mode 100644 tests/mysql/if_else.rs diff --git a/src/backend/query_builder.rs b/src/backend/query_builder.rs index 41cac862..85984700 100644 --- a/src/backend/query_builder.rs +++ b/src/backend/query_builder.rs @@ -387,9 +387,24 @@ pub trait QueryBuilder: SimpleExpr::Constant(val) => { self.prepare_constant(val, sql); } + SimpleExpr::IfElse(val) => { + self.prepare_if_else_statement(val, sql); + } } } + fn prepare_if_else_statement(&self, val: &Box, sql: &mut dyn SqlWriter) { + write!(sql, "IF ").unwrap(); + self.prepare_simple_expr(&val.when, sql); + write!(sql, " THEN\n").unwrap(); + self.prepare_simple_expr(&val.then, sql); + if let Some(otherwise) = &val.otherwise { + write!(sql, "\nELSE\n").unwrap(); + self.prepare_simple_expr(otherwise, sql); + }; + write!(sql, "\nEND IF").unwrap(); + } + /// Translate [`CaseStatement`] into SQL statement. fn prepare_case_statement(&self, stmts: &CaseStatement, sql: &mut dyn SqlWriter) { write!(sql, "(CASE").unwrap(); diff --git a/src/expr.rs b/src/expr.rs index b6894c94..11a95306 100644 --- a/src/expr.rs +++ b/src/expr.rs @@ -4,7 +4,7 @@ //! //! [`SimpleExpr`] is the expression common among select fields, where clauses and many other places. -use crate::{func::*, query::*, types::*, value::*}; +use crate::{func::*, if_else::*, query::*, types::*, value::*}; /// Helper to build a [`SimpleExpr`]. #[derive(Debug, Clone)] @@ -35,6 +35,7 @@ pub enum SimpleExpr { AsEnum(DynIden, Box), Case(Box), Constant(Value), + IfElse(Box), } /// "Operator" methods for building complex expressions. diff --git a/src/if_else.rs b/src/if_else.rs new file mode 100644 index 00000000..1d89611d --- /dev/null +++ b/src/if_else.rs @@ -0,0 +1,33 @@ +use crate::{QueryBuilder, SimpleExpr}; + +#[derive(Debug, Clone, PartialEq)] +pub struct IfElseStatement { + pub when: SimpleExpr, + pub then: SimpleExpr, + pub otherwise: Option, +} + +impl IfElseStatement { + pub fn new(when: SimpleExpr, then: SimpleExpr, otherwise: Option) -> Self { + Self { + when, + then, + otherwise, + } + } + + pub fn to_string(&self, query_builder: T) -> String { + let mut sql = String::with_capacity(256); + query_builder.prepare_if_else_statement(&Box::new(self.clone()), &mut sql); + sql + } +} +pub trait IfElseStatementBuilder { + /// Build corresponding SQL statement for certain database backend and return SQL string + fn build(&self, query_builder: T) -> String; + + /// Build corresponding SQL statement for certain database backend and return SQL string + fn to_string(&self, query_builder: T) -> String { + self.build(query_builder) + } +} diff --git a/src/lib.rs b/src/lib.rs index 15e1a189..7ee5411c 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -818,6 +818,7 @@ pub mod expr; pub mod extension; pub mod foreign_key; pub mod func; +pub mod if_else; pub mod index; pub mod prepare; pub mod query; @@ -835,6 +836,7 @@ pub use backend::*; pub use expr::*; pub use foreign_key::*; pub use func::*; +pub use if_else::*; pub use index::*; pub use prepare::*; pub use query::*; diff --git a/tests/mysql/if_else.rs b/tests/mysql/if_else.rs new file mode 100644 index 00000000..bd5e94b7 --- /dev/null +++ b/tests/mysql/if_else.rs @@ -0,0 +1,45 @@ +use super::*; +use pretty_assertions::assert_eq; + +#[rustfmt::skip] +#[test] +fn if_without_else() { + let query = Query::select().column(Asterisk).from(Glyph::Table).take(); + let then = SimpleExpr::SubQuery(None, Box::new(query.into_sub_query_statement())); + let if_statement = IfElseStatement::new( + Expr::col(Glyph::Id).eq(1), + then, + None + ); + assert_eq!( + if_statement.to_string(MysqlQueryBuilder), + [ + "IF `id` = 1 THEN", + "(SELECT * FROM `glyph`)", + "END IF" + ].join("\n") + ) +} + +#[rustfmt::skip] +#[test] +fn if_with_else() { + let query = Query::select().column(Asterisk).from(Glyph::Table).take(); + let then = SimpleExpr::SubQuery(None, Box::new(query.into_sub_query_statement())); + let if_statement = IfElseStatement::new( + Expr::col(Glyph::Id).eq(1), + then, + Some(Expr::val("23").into()), + ); + assert_eq!( + if_statement.to_string(MysqlQueryBuilder), + [ + "IF `id` = 1 THEN", + "(SELECT * FROM `glyph`)", + "ELSE", + "'23'", + "END IF" + ] + .join("\n") + ) +} diff --git a/tests/mysql/mod.rs b/tests/mysql/mod.rs index d717774f..d3a4fe85 100644 --- a/tests/mysql/mod.rs +++ b/tests/mysql/mod.rs @@ -1,6 +1,7 @@ use sea_query::{extension::mysql::*, tests_cfg::*, *}; mod foreign_key; +mod if_else; mod index; mod query; mod table; From 6bdcd1e352438b9d848a6251bfd3f590a26d00ff Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Stefan=20Gr=C3=B6nke?= Date: Mon, 14 Oct 2024 16:14:23 +0200 Subject: [PATCH 2/2] elseif support in IfElseStatement --- src/backend/mysql/query.rs | 5 +++ src/backend/postgres/query.rs | 5 +++ src/backend/query_builder.rs | 20 ++++++++-- src/backend/sqlite/query.rs | 4 ++ tests/mysql/if_else.rs | 54 +++++++++++++++++++++++++++ tests/postgres/if_else.rs | 70 +++++++++++++++++++++++++++++++++++ tests/postgres/mod.rs | 1 + tests/sqlite/mod.rs | 1 + tests/sqlite/unsupported.rs | 13 +++++++ 9 files changed, 169 insertions(+), 4 deletions(-) create mode 100644 tests/postgres/if_else.rs create mode 100644 tests/sqlite/unsupported.rs diff --git a/src/backend/mysql/query.rs b/src/backend/mysql/query.rs index 58126014..31e3d922 100644 --- a/src/backend/mysql/query.rs +++ b/src/backend/mysql/query.rs @@ -147,6 +147,11 @@ impl QueryBuilder for MysqlQueryBuilder { fn insert_default_keyword(&self) -> &str { "()" } + + /// Prefix of the ELSEIF (MySQL) + fn elseif_keyword_prefix(&self) -> &str { + "ELSE" + } } impl MysqlQueryBuilder { diff --git a/src/backend/postgres/query.rs b/src/backend/postgres/query.rs index f0650110..629ea510 100644 --- a/src/backend/postgres/query.rs +++ b/src/backend/postgres/query.rs @@ -174,6 +174,11 @@ impl QueryBuilder for PostgresQueryBuilder { fn if_null_function(&self) -> &str { "COALESCE" } + + /// Prefix of the ELSIF (Postgres) + fn elseif_keyword_prefix(&self) -> &str { + "ELS" + } } fn is_pg_comparison(b: &BinOper) -> bool { diff --git a/src/backend/query_builder.rs b/src/backend/query_builder.rs index 85984700..2e52f4c6 100644 --- a/src/backend/query_builder.rs +++ b/src/backend/query_builder.rs @@ -393,16 +393,28 @@ pub trait QueryBuilder: } } + /// Prefix of the ELSEIF (MySQL) vs ELSIF (Postgres) keyword + fn elseif_keyword_prefix(&self) -> &str { + panic!("ELSEIF/ELSIF keyword prefix not implemented for this backend"); + } + fn prepare_if_else_statement(&self, val: &Box, sql: &mut dyn SqlWriter) { write!(sql, "IF ").unwrap(); self.prepare_simple_expr(&val.when, sql); write!(sql, " THEN\n").unwrap(); self.prepare_simple_expr(&val.then, sql); - if let Some(otherwise) = &val.otherwise { - write!(sql, "\nELSE\n").unwrap(); - self.prepare_simple_expr(otherwise, sql); + match &val.otherwise { + Some(SimpleExpr::IfElse(value)) => { + write!(sql, "\n{}", self.elseif_keyword_prefix()).unwrap(); + self.prepare_if_else_statement(value, sql); + } + Some(otherwise) => { + write!(sql, "\nELSE\n").unwrap(); + self.prepare_simple_expr(otherwise, sql); + write!(sql, "\nEND IF").unwrap(); + } + None => write!(sql, "\nEND IF").unwrap(), }; - write!(sql, "\nEND IF").unwrap(); } /// Translate [`CaseStatement`] into SQL statement. diff --git a/src/backend/sqlite/query.rs b/src/backend/sqlite/query.rs index 0a062294..9a96e42b 100644 --- a/src/backend/sqlite/query.rs +++ b/src/backend/sqlite/query.rs @@ -92,4 +92,8 @@ impl QueryBuilder for SqliteQueryBuilder { // SQLite doesn't support inserting multiple rows with default values write!(sql, "DEFAULT VALUES").unwrap() } + + fn prepare_if_else_statement(&self, _val: &Box, _sql: &mut dyn SqlWriter) { + panic!("Sqlite doesn't support if-else statements") + } } diff --git a/tests/mysql/if_else.rs b/tests/mysql/if_else.rs index bd5e94b7..a5e64034 100644 --- a/tests/mysql/if_else.rs +++ b/tests/mysql/if_else.rs @@ -43,3 +43,57 @@ fn if_with_else() { .join("\n") ) } + +#[test] +fn if_with_elseif() { + let query = Query::select().column(Asterisk).from(Glyph::Table).take(); + let then = SimpleExpr::SubQuery(None, Box::new(query.into_sub_query_statement())); + let if_statement = IfElseStatement::new( + Expr::col(Glyph::Id).eq(1), + then, + Some(SimpleExpr::IfElse(Box::new(IfElseStatement::new( + Expr::col(Glyph::Id).eq(2), + Expr::val("42").into(), + None, + )))), + ); + assert_eq!( + if_statement.to_string(MysqlQueryBuilder), + [ + "IF `id` = 1 THEN", + "(SELECT * FROM `glyph`)", + "ELSEIF `id` = 2 THEN", + "'42'", + "END IF" + ] + .join("\n") + ) +} + +#[test] +fn if_with_elseif_and_else() { + let query = Query::select().column(Asterisk).from(Glyph::Table).take(); + let then = SimpleExpr::SubQuery(None, Box::new(query.into_sub_query_statement())); + let if_statement = IfElseStatement::new( + Expr::col(Glyph::Id).eq(1), + then, + Some(SimpleExpr::IfElse(Box::new(IfElseStatement::new( + Expr::col(Glyph::Id).eq(2), + Expr::val("42").into(), + Some(Expr::val("9000").into()), + )))), + ); + assert_eq!( + if_statement.to_string(MysqlQueryBuilder), + [ + "IF `id` = 1 THEN", + "(SELECT * FROM `glyph`)", + "ELSEIF `id` = 2 THEN", + "'42'", + "ELSE", + "'9000'", + "END IF" + ] + .join("\n") + ); +} diff --git a/tests/postgres/if_else.rs b/tests/postgres/if_else.rs new file mode 100644 index 00000000..9f9098bd --- /dev/null +++ b/tests/postgres/if_else.rs @@ -0,0 +1,70 @@ +use super::*; +use pretty_assertions::assert_eq; + +#[test] +#[rustfmt::skip] +fn if_without_else() { + let query = Query::select().column(Asterisk).from(Glyph::Table).take(); + let then = SimpleExpr::SubQuery(None, Box::new(query.into_sub_query_statement())); + let if_statement = IfElseStatement::new( + Expr::col(Glyph::Id).eq(1), + then, + None + ); + assert_eq!( + if_statement.to_string(MysqlQueryBuilder), + [ + "IF `id` = 1 THEN", + "(SELECT * FROM `glyph`)", + "END IF" + ].join("\n") + ) +} + +#[test] +#[rustfmt::skip] +fn if_with_else() { + let query = Query::select().column(Asterisk).from(Glyph::Table).take(); + let then = SimpleExpr::SubQuery(None, Box::new(query.into_sub_query_statement())); + let if_statement = IfElseStatement::new( + Expr::col(Glyph::Id).eq(1), + then, + Some(Expr::val("23").into()) + ); + assert_eq!( + if_statement.to_string(PostgresQueryBuilder), + [ + "IF \"id\" = 1 THEN", + "(SELECT * FROM \"glyph\")", + "ELSE", + "'23'", + "END IF" + ].join("\n") + ) +} + +#[test] +#[rustfmt::skip] +fn if_with_elseif() { + let query = Query::select().column(Asterisk).from(Glyph::Table).take(); + let then = SimpleExpr::SubQuery(None, Box::new(query.into_sub_query_statement())); + let if_statement = IfElseStatement::new( + Expr::col(Glyph::Id).eq(1), + then, + Some(SimpleExpr::IfElse(Box::new(IfElseStatement::new( + Expr::col(Glyph::Id).eq(2), + Expr::val("123").into(), + None + )))) + ); + assert_eq!( + if_statement.to_string(PostgresQueryBuilder), + [ + "IF \"id\" = 1 THEN", + "(SELECT * FROM \"glyph\")", + "ELSIF \"id\" = 2 THEN", + "'123'", + "END IF" + ].join("\n") + ) +} diff --git a/tests/postgres/mod.rs b/tests/postgres/mod.rs index 82b85df3..76390ae4 100644 --- a/tests/postgres/mod.rs +++ b/tests/postgres/mod.rs @@ -1,6 +1,7 @@ use sea_query::{tests_cfg::*, *}; mod foreign_key; +mod if_else; mod index; mod query; mod table; diff --git a/tests/sqlite/mod.rs b/tests/sqlite/mod.rs index fc7388cd..2c90d514 100644 --- a/tests/sqlite/mod.rs +++ b/tests/sqlite/mod.rs @@ -4,6 +4,7 @@ mod foreign_key; mod index; mod query; mod table; +mod unsupported; #[path = "../common.rs"] mod common; diff --git a/tests/sqlite/unsupported.rs b/tests/sqlite/unsupported.rs new file mode 100644 index 00000000..5bf704b9 --- /dev/null +++ b/tests/sqlite/unsupported.rs @@ -0,0 +1,13 @@ +use super::*; + +#[test] +#[should_panic] +#[rustfmt::skip] +fn if_else_statement_is_unsupported() { + let if_statement = IfElseStatement::new( + Expr::col(Glyph::Id).eq(1), + Expr::val("23").into(), + None + ); + if_statement.to_string(SqliteQueryBuilder); +}