From d40cadd64f7879ab3f2f608f0c3724fcd753cc64 Mon Sep 17 00:00:00 2001 From: Ning Sun Date: Fri, 6 Dec 2024 16:22:25 +0800 Subject: [PATCH] feat: add more transaction related statement for postgres interface (#5081) * fix: add match for start and abort transactions * feat: add commit transaction statement * feat: add warning on transaction start * chore: update message --- src/servers/src/postgres/fixtures.rs | 45 +++++++++++++++++++++++----- 1 file changed, 38 insertions(+), 7 deletions(-) diff --git a/src/servers/src/postgres/fixtures.rs b/src/servers/src/postgres/fixtures.rs index 18c3661b9334..895f5c03e4a9 100644 --- a/src/servers/src/postgres/fixtures.rs +++ b/src/servers/src/postgres/fixtures.rs @@ -51,22 +51,40 @@ static VAR_VALUES: Lazy> = Lazy::new(|| { static SHOW_PATTERN: Lazy = Lazy::new(|| Regex::new("(?i)^SHOW (.*?);?$").unwrap()); static SET_TRANSACTION_PATTERN: Lazy = Lazy::new(|| Regex::new("(?i)^SET TRANSACTION (.*?);?$").unwrap()); -static TRANSACTION_PATTERN: Lazy = - Lazy::new(|| Regex::new("(?i)^(BEGIN|ROLLBACK|COMMIT);?").unwrap()); +static START_TRANSACTION_PATTERN: Lazy = + Lazy::new(|| Regex::new("(?i)^(START TRANSACTION.*|BEGIN);?").unwrap()); +static COMMIT_TRANSACTION_PATTERN: Lazy = + Lazy::new(|| Regex::new("(?i)^(COMMIT TRANSACTION|COMMIT);?").unwrap()); +static ABORT_TRANSACTION_PATTERN: Lazy = + Lazy::new(|| Regex::new("(?i)^(ABORT TRANSACTION|ROLLBACK);?").unwrap()); /// Test if given query statement matches the patterns pub(crate) fn matches(query: &str) -> bool { - TRANSACTION_PATTERN.captures(query).is_some() + START_TRANSACTION_PATTERN.is_match(query) + || COMMIT_TRANSACTION_PATTERN.is_match(query) + || ABORT_TRANSACTION_PATTERN.is_match(query) || SHOW_PATTERN.captures(query).is_some() || SET_TRANSACTION_PATTERN.is_match(query) } +fn set_transaction_warning(query_ctx: QueryContextRef) { + query_ctx.set_warning("Please note transaction is not supported in GreptimeDB.".to_string()); +} + /// Process unsupported SQL and return fixed result as a compatibility solution -pub(crate) fn process<'a>(query: &str, _query_ctx: QueryContextRef) -> Option>> { +pub(crate) fn process<'a>(query: &str, query_ctx: QueryContextRef) -> Option>> { // Transaction directives: - if let Some(tx) = TRANSACTION_PATTERN.captures(query) { - let tx_tag = &tx[1]; - Some(vec![Response::Execution(Tag::new(&tx_tag.to_uppercase()))]) + if START_TRANSACTION_PATTERN.is_match(query) { + set_transaction_warning(query_ctx); + if query.to_lowercase().starts_with("begin") { + Some(vec![Response::Execution(Tag::new("BEGIN"))]) + } else { + Some(vec![Response::Execution(Tag::new("START TRANSACTION"))]) + } + } else if ABORT_TRANSACTION_PATTERN.is_match(query) { + Some(vec![Response::Execution(Tag::new("ROLLBACK"))]) + } else if COMMIT_TRANSACTION_PATTERN.is_match(query) { + Some(vec![Response::Execution(Tag::new("COMMIT"))]) } else if let Some(show_var) = SHOW_PATTERN.captures(query) { let show_var = show_var[1].to_lowercase(); if let Some(value) = VAR_VALUES.get(&show_var.as_ref()) { @@ -150,6 +168,19 @@ mod test { "SET", query_context.clone(), ); + assert_tag( + "START TRANSACTION isolation level READ COMMITTED;", + "START TRANSACTION", + query_context.clone(), + ); + assert_tag( + "start transaction isolation level READ COMMITTED;", + "START TRANSACTION", + query_context.clone(), + ); + assert_tag("abort transaction;", "ROLLBACK", query_context.clone()); + assert_tag("commit transaction;", "COMMIT", query_context.clone()); + assert_tag("COMMIT transaction;", "COMMIT", query_context.clone()); let resp = get_data("SHOW transaction isolation level", query_context.clone()); assert_eq!(1, resp.row_schema().len());