From a912559273aef7ca7eaaee3183a5464d04311d20 Mon Sep 17 00:00:00 2001 From: Weijun-H Date: Wed, 25 Sep 2024 18:15:10 +0800 Subject: [PATCH 01/30] feat: Support for views --- src/hooks/utility.rs | 17 +++++++++++++++-- tests/tests/scan.rs | 26 +++++++++++++++++++++++++- 2 files changed, 40 insertions(+), 3 deletions(-) diff --git a/src/hooks/utility.rs b/src/hooks/utility.rs index d53dbd41..07d2491e 100644 --- a/src/hooks/utility.rs +++ b/src/hooks/utility.rs @@ -26,9 +26,13 @@ use anyhow::{bail, Result}; use pgrx::{pg_sys, AllocatedByRust, HookResult, PgBox}; use sqlparser::{ast::Statement, dialect::PostgreSqlDialect, parser::Parser}; +use crate::duckdb::connection::execute; use explain::explain_query; +use pgrx::pg_sys::NodeTag; use prepare::*; +use super::query::*; + type ProcessUtilityHook = fn( pstmt: PgBox, query_string: &core::ffi::CStr, @@ -114,6 +118,7 @@ pub async fn process_utility_hook( pstmt.utilityStmt as *mut pg_sys::ExplainStmt, dest.as_ptr(), )?, + pg_sys::NodeTag::T_ViewStmt => view_query(query_string)?, _ => bail!("unexpected statement type in utility hook"), }; @@ -133,13 +138,21 @@ pub async fn process_utility_hook( Ok(()) } -fn is_support_utility(stmt_type: pg_sys::NodeTag) -> bool { +fn is_support_utility(stmt_type: NodeTag) -> bool { stmt_type == pg_sys::NodeTag::T_ExplainStmt - || stmt_type == pg_sys::NodeTag::T_PrepareStmt + || stmt_type == pg_sys::NodeTag::T_ViewStmt || stmt_type == pg_sys::NodeTag::T_DeallocateStmt || stmt_type == pg_sys::NodeTag::T_ExecuteStmt } +fn view_query(query_string: &core::ffi::CStr) -> Result { + // Set DuckDB search path according search path in Postgres + set_search_path_by_pg()?; + // Push down the view creation query to DuckDB + execute(query_string.to_str().unwrap(), [])?; + Ok(true) +} + fn parse_query_from_utility_stmt(query_string: &core::ffi::CStr) -> Result { let query_string = query_string.to_str()?; diff --git a/tests/tests/scan.rs b/tests/tests/scan.rs index 425ced84..cf32bcc8 100644 --- a/tests/tests/scan.rs +++ b/tests/tests/scan.rs @@ -560,7 +560,7 @@ async fn test_executor_hook_search_path(mut conn: PgConnection, tempdir: TempDir } #[rstest] -async fn test_prepare_stmt_execute(#[future(awt)] s3: S3, mut conn: PgConnection) -> Result<()> { +async fn test_view_foreign_table(#[future(awt)] s3: S3, mut conn: PgConnection) -> Result<()> { NycTripsTable::setup().execute(&mut conn); let rows: Vec = "SELECT * FROM nyc_trips".fetch(&mut conn); s3.client @@ -646,6 +646,30 @@ async fn test_prepare_search_path(mut conn: PgConnection, tempdir: TempDir) -> R "DEALLOCATE q1".execute(&mut conn); assert!("EXECUTE q1(true)".execute_result(&mut conn).is_err()); + Ok(()) +} + +#[rstest] +async fn test_prepare_stmt_execute(#[future(awt)] s3: S3, mut conn: PgConnection) -> Result<()> { + NycTripsTable::setup().execute(&mut conn); + let rows: Vec = "SELECT * FROM nyc_trips".fetch(&mut conn); + s3.client + .create_bucket() + .bucket(S3_TRIPS_BUCKET) + .send() + .await?; + s3.create_bucket(S3_TRIPS_BUCKET).await?; + s3.put_rows(S3_TRIPS_BUCKET, S3_TRIPS_KEY, &rows).await?; + + NycTripsTable::setup_s3_listing_fdw( + &s3.url.clone(), + &format!("s3://{S3_TRIPS_BUCKET}/{S3_TRIPS_KEY}"), + ) + .execute(&mut conn); + "CREATE VIEW trips_view AS SELECT * FROM trips".execute(&mut conn); + let res: (i64,) = "SELECT * FROM trips_view".fetch_one(&mut conn); + + assert_eq!(res.0, 2964624); Ok(()) } From 9aef1895dd8d7f4b90838e1e22e3af6987dc74ac Mon Sep 17 00:00:00 2001 From: Weijun-H Date: Wed, 25 Sep 2024 18:16:26 +0800 Subject: [PATCH 02/30] chore: Add comments for tests --- tests/tests/scan.rs | 1 + 1 file changed, 1 insertion(+) diff --git a/tests/tests/scan.rs b/tests/tests/scan.rs index cf32bcc8..311ba197 100644 --- a/tests/tests/scan.rs +++ b/tests/tests/scan.rs @@ -559,6 +559,7 @@ async fn test_executor_hook_search_path(mut conn: PgConnection, tempdir: TempDir Ok(()) } +// Test view creation with foreign table #[rstest] async fn test_view_foreign_table(#[future(awt)] s3: S3, mut conn: PgConnection) -> Result<()> { NycTripsTable::setup().execute(&mut conn); From 0a57e300bc954463b753bb5d8b5cb65f86331112 Mon Sep 17 00:00:00 2001 From: Weijun-H Date: Wed, 25 Sep 2024 18:23:37 +0800 Subject: [PATCH 03/30] chore --- src/hooks/utility.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/hooks/utility.rs b/src/hooks/utility.rs index 07d2491e..27b7e443 100644 --- a/src/hooks/utility.rs +++ b/src/hooks/utility.rs @@ -149,7 +149,7 @@ fn view_query(query_string: &core::ffi::CStr) -> Result { // Set DuckDB search path according search path in Postgres set_search_path_by_pg()?; // Push down the view creation query to DuckDB - execute(query_string.to_str().unwrap(), [])?; + execute(query_string.to_str()?, [])?; Ok(true) } From e76495d38990ef5ff156ebefc7caf4aeb61a6db8 Mon Sep 17 00:00:00 2001 From: Weijun-H Date: Wed, 25 Sep 2024 18:35:56 +0800 Subject: [PATCH 04/30] chore: Update tests --- tests/tests/scan.rs | 53 +++++++++++++++++++++++++++------------------ 1 file changed, 32 insertions(+), 21 deletions(-) diff --git a/tests/tests/scan.rs b/tests/tests/scan.rs index 311ba197..203d2012 100644 --- a/tests/tests/scan.rs +++ b/tests/tests/scan.rs @@ -559,9 +559,9 @@ async fn test_executor_hook_search_path(mut conn: PgConnection, tempdir: TempDir Ok(()) } -// Test view creation with foreign table + #[rstest] -async fn test_view_foreign_table(#[future(awt)] s3: S3, mut conn: PgConnection) -> Result<()> { +async fn test_prepare_stmt_execute(#[future(awt)] s3: S3, mut conn: PgConnection) -> Result<()> { NycTripsTable::setup().execute(&mut conn); let rows: Vec = "SELECT * FROM nyc_trips".fetch(&mut conn); s3.client @@ -647,30 +647,41 @@ async fn test_prepare_search_path(mut conn: PgConnection, tempdir: TempDir) -> R "DEALLOCATE q1".execute(&mut conn); assert!("EXECUTE q1(true)".execute_result(&mut conn).is_err()); + Ok(()) } +// Test view creation with foreign table #[rstest] -async fn test_prepare_stmt_execute(#[future(awt)] s3: S3, mut conn: PgConnection) -> Result<()> { - NycTripsTable::setup().execute(&mut conn); - let rows: Vec = "SELECT * FROM nyc_trips".fetch(&mut conn); - s3.client - .create_bucket() - .bucket(S3_TRIPS_BUCKET) - .send() - .await?; - s3.create_bucket(S3_TRIPS_BUCKET).await?; - s3.put_rows(S3_TRIPS_BUCKET, S3_TRIPS_KEY, &rows).await?; +async fn test_view_foreign_table(mut conn: PgConnection, tempdir: TempDir) -> Result<()> { + let stored_batch = primitive_record_batch()?; + let parquet_path = tempdir.path().join("test_arrow_types.parquet"); + let parquet_file = File::create(&parquet_path)?; - NycTripsTable::setup_s3_listing_fdw( - &s3.url.clone(), - &format!("s3://{S3_TRIPS_BUCKET}/{S3_TRIPS_KEY}"), - ) - .execute(&mut conn); - "CREATE VIEW trips_view AS SELECT * FROM trips".execute(&mut conn); - let res: (i64,) = "SELECT * FROM trips_view".fetch_one(&mut conn); + let mut writer = ArrowWriter::try_new(parquet_file, stored_batch.schema(), None).unwrap(); + writer.write(&stored_batch)?; + writer.close()?; - assert_eq!(res.0, 2964624); + primitive_setup_fdw_local_file_listing(parquet_path.as_path().to_str().unwrap(), "primitive") + .execute(&mut conn); + + // fully pushdown to the DuckDB + "CREATE VIEW primitive_view AS SELECT * FROM primitive".execute(&mut conn); + let res: (bool,) = "SELECT boolean_col FROM primitive_view".fetch_one(&mut conn); + assert!(res.0); + + // cannot fully pushdown to the DuckDB + "CREATE TABLE t1 (a int);".execute(&mut conn); + "INSERT INTO t1 VALUES (1);".execute(&mut conn); + r#" + CREATE VIEW primitive_join_view AS + SELECT * + FROM primitive + JOIN t1 ON t1.a = primitive.int32_col + "# + .execute(&mut conn); + let res: (i32,) = "SELECT int32_col FROM primitive_join_view".fetch_one(&mut conn); + assert_eq!(res.0, 1); Ok(()) -} +} \ No newline at end of file From fc0d1ef0c14fddeb6ed67c0f44db298d44dc18e8 Mon Sep 17 00:00:00 2001 From: Weijun-H Date: Thu, 26 Sep 2024 14:41:22 +0800 Subject: [PATCH 05/30] fix: Check if the table exists in the DuckDB --- src/hooks/query.rs | 24 ++++++++++++++++++++++++ src/hooks/utility.rs | 38 +++++++++++++++++++++++++++++++++++--- 2 files changed, 59 insertions(+), 3 deletions(-) diff --git a/src/hooks/query.rs b/src/hooks/query.rs index 863a2f97..e57b3c0d 100644 --- a/src/hooks/query.rs +++ b/src/hooks/query.rs @@ -116,6 +116,30 @@ fn get_postgres_search_path() -> Vec { schema_vec } +pub fn get_postgres_current_schema() -> String { + let active_schemas = + unsafe { PgList::::from_pg(pg_sys::fetch_search_path(false)) }; + + let schema_oid = active_schemas.get_oid(0).unwrap(); + + let mut current_schema = String::new(); + let tuple = unsafe { + pg_sys::SearchSysCache1( + pg_sys::SysCacheIdentifier::NAMESPACEOID as i32, + schema_oid.into_datum().unwrap(), + ) + }; + + if !tuple.is_null() { + let pg_namespace = unsafe { pg_sys::GETSTRUCT(tuple) as pg_sys::Form_pg_namespace }; + let name = pg_sys::name_data_to_str(unsafe { &(*pg_namespace).nspname }); + current_schema = name.to_string(); + + unsafe { pg_sys::ReleaseSysCache(tuple) }; + } + current_schema +} + pub fn is_duckdb_query(relations: &[PgRelation]) -> bool { !relations.is_empty() && relations.iter().all(|pg_relation| { diff --git a/src/hooks/utility.rs b/src/hooks/utility.rs index 27b7e443..76dec5e4 100644 --- a/src/hooks/utility.rs +++ b/src/hooks/utility.rs @@ -20,13 +20,14 @@ mod explain; mod prepare; +use std::ffi::CStr; use std::ptr::null_mut; use anyhow::{bail, Result}; use pgrx::{pg_sys, AllocatedByRust, HookResult, PgBox}; use sqlparser::{ast::Statement, dialect::PostgreSqlDialect, parser::Parser}; -use crate::duckdb::connection::execute; +use crate::duckdb::connection::{execute, view_exists}; use explain::explain_query; use pgrx::pg_sys::NodeTag; use prepare::*; @@ -118,7 +119,9 @@ pub async fn process_utility_hook( pstmt.utilityStmt as *mut pg_sys::ExplainStmt, dest.as_ptr(), )?, - pg_sys::NodeTag::T_ViewStmt => view_query(query_string)?, + pg_sys::NodeTag::T_ViewStmt => { + view_query(query_string, pstmt.utilityStmt as *mut pg_sys::ViewStmt)? + } _ => bail!("unexpected statement type in utility hook"), }; @@ -145,9 +148,38 @@ fn is_support_utility(stmt_type: NodeTag) -> bool { || stmt_type == pg_sys::NodeTag::T_ExecuteStmt } -fn view_query(query_string: &core::ffi::CStr) -> Result { +fn view_query(query_string: &core::ffi::CStr, stmt: *mut pg_sys::ViewStmt) -> Result { + // Get the current schema in Postgres + let current_schema = get_postgres_current_schema(); // Set DuckDB search path according search path in Postgres set_search_path_by_pg()?; + + let query = unsafe { (*stmt).query as *mut pg_sys::SelectStmt }; + let from_clause = unsafe { (*query).fromClause }; + unsafe { + let elements: *mut pg_sys::ListCell = (*from_clause).elements; + for i in 0..(*from_clause).length { + let rv = (*elements.offset(i as isize)).ptr_value as *mut pg_sys::RangeVar; + + // if the schema name is not provided, the current schema is used + let relation_name = if (*rv).relname.is_null() { + current_schema.as_str() + } else { + CStr::from_ptr((*rv).relname).to_str()? + }; + let schema_name = if (*rv).schemaname.is_null() { + current_schema.as_str() + } else { + CStr::from_ptr((*rv).schemaname).to_str()? + }; + + if !view_exists(relation_name, schema_name)? { + pgrx::warning!("{schema_name}.{relation_name} does not exist in DuckDB, this query won't be pushed down to DuckDB."); + return Ok(true); + } + } + } + // Push down the view creation query to DuckDB execute(query_string.to_str()?, [])?; Ok(true) From 64aac71078508269f06592b4f477b57b7a49eaef Mon Sep 17 00:00:00 2001 From: Weijun-H Date: Fri, 27 Sep 2024 11:01:15 +0800 Subject: [PATCH 06/30] fix: Not fully pushed to DuckDB --- src/hooks/utility.rs | 43 ++++++++++++++++++++++++------------------- tests/tests/scan.rs | 25 +++++++++++++++++++++++++ 2 files changed, 49 insertions(+), 19 deletions(-) diff --git a/src/hooks/utility.rs b/src/hooks/utility.rs index 76dec5e4..82614493 100644 --- a/src/hooks/utility.rs +++ b/src/hooks/utility.rs @@ -27,7 +27,7 @@ use anyhow::{bail, Result}; use pgrx::{pg_sys, AllocatedByRust, HookResult, PgBox}; use sqlparser::{ast::Statement, dialect::PostgreSqlDialect, parser::Parser}; -use crate::duckdb::connection::{execute, view_exists}; +use crate::duckdb::connection::{execute}; use explain::explain_query; use pgrx::pg_sys::NodeTag; use prepare::*; @@ -157,25 +157,30 @@ fn view_query(query_string: &core::ffi::CStr, stmt: *mut pg_sys::ViewStmt) -> Re let query = unsafe { (*stmt).query as *mut pg_sys::SelectStmt }; let from_clause = unsafe { (*query).fromClause }; unsafe { - let elements: *mut pg_sys::ListCell = (*from_clause).elements; + let elements = (*from_clause).elements; for i in 0..(*from_clause).length { - let rv = (*elements.offset(i as isize)).ptr_value as *mut pg_sys::RangeVar; - - // if the schema name is not provided, the current schema is used - let relation_name = if (*rv).relname.is_null() { - current_schema.as_str() - } else { - CStr::from_ptr((*rv).relname).to_str()? - }; - let schema_name = if (*rv).schemaname.is_null() { - current_schema.as_str() - } else { - CStr::from_ptr((*rv).schemaname).to_str()? - }; - - if !view_exists(relation_name, schema_name)? { - pgrx::warning!("{schema_name}.{relation_name} does not exist in DuckDB, this query won't be pushed down to DuckDB."); - return Ok(true); + let element = (*elements.offset(i as isize)).ptr_value as *mut pg_sys::Node; + + match (*element).type_ { + pg_sys::NodeTag::T_RangeVar => { + if !analyze_range_var( + element as *mut pg_sys::RangeVar, + current_schema.as_str(), + )? { + return Ok(true); + } + } + pg_sys::NodeTag::T_JoinExpr => { + if !analyze_join_expr( + element as *mut pg_sys::JoinExpr, + current_schema.as_str(), + )? { + return Ok(true); + } + } + _ => { + continue; + } } } } diff --git a/tests/tests/scan.rs b/tests/tests/scan.rs index 203d2012..5c361ce7 100644 --- a/tests/tests/scan.rs +++ b/tests/tests/scan.rs @@ -591,6 +591,31 @@ async fn test_prepare_stmt_execute(#[future(awt)] s3: S3, mut conn: PgConnection assert!("EXECUTE test_query(3)".execute_result(&mut conn).is_err()); + // cannot fully pushdown to DuckDB + let warning_res: (String,) = "CREATE TABLE rate_code ( + id INT PRIMARY KEY, + name TEXT + ); + + INSERT INTO rate_code + (id, name) + VALUES + (1, 'one'), + (2, 'two'), + (3, 'three'), + (4, 'four'), + (5, 'five'), + (99, 'ninety nine'); + + create view trips_with_rate_code as + select * + from trips + join rate_code + on trips.ratecodeid = rate_code.id; + " + .fetch_one(&mut conn); + + assert_eq!(warning_res.0, ""); Ok(()) } From 13eb303851b9f3a88542b225b5e55659c5db4ff7 Mon Sep 17 00:00:00 2001 From: Weijun-H Date: Fri, 27 Sep 2024 11:12:08 +0800 Subject: [PATCH 07/30] chore --- tests/tests/scan.rs | 21 ++++++++++++++++----- 1 file changed, 16 insertions(+), 5 deletions(-) diff --git a/tests/tests/scan.rs b/tests/tests/scan.rs index 5c361ce7..51c97775 100644 --- a/tests/tests/scan.rs +++ b/tests/tests/scan.rs @@ -592,7 +592,8 @@ async fn test_prepare_stmt_execute(#[future(awt)] s3: S3, mut conn: PgConnection assert!("EXECUTE test_query(3)".execute_result(&mut conn).is_err()); // cannot fully pushdown to DuckDB - let warning_res: (String,) = "CREATE TABLE rate_code ( + r#" + CREATE TABLE rate_code ( id INT PRIMARY KEY, name TEXT ); @@ -606,16 +607,26 @@ async fn test_prepare_stmt_execute(#[future(awt)] s3: S3, mut conn: PgConnection (4, 'four'), (5, 'five'), (99, 'ninety nine'); + "# + .execute(&mut conn); + let warning_res = r#" create view trips_with_rate_code as select * from trips join rate_code - on trips.ratecodeid = rate_code.id; - " - .fetch_one(&mut conn); + on trips.vendorid = rate_code.id; + "# + .execute_result(&mut conn); - assert_eq!(warning_res.0, ""); + if let Err(e) = warning_res { + assert_eq!( + e.to_string(), + "WARNING: publicrate_code does not exist in DuckDB".to_string() + ); + } else { + panic!("Expecting an warning but got success"); + } Ok(()) } From fa61a4cfc0714de0886353e8b7843e6f454c778f Mon Sep 17 00:00:00 2001 From: Weijun-H Date: Fri, 27 Sep 2024 13:04:17 +0800 Subject: [PATCH 08/30] chore:test --- tests/tests/scan.rs | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/tests/tests/scan.rs b/tests/tests/scan.rs index 51c97775..a24d291a 100644 --- a/tests/tests/scan.rs +++ b/tests/tests/scan.rs @@ -591,6 +591,15 @@ async fn test_prepare_stmt_execute(#[future(awt)] s3: S3, mut conn: PgConnection assert!("EXECUTE test_query(3)".execute_result(&mut conn).is_err()); + let test: Vec<(String,)> = r#" + SELECT column_name + FROM information_schema.columns + WHERE table_name = 'trips'; + "# + .fetch(&mut conn); + + panic!("{:?}", test); + // cannot fully pushdown to DuckDB r#" CREATE TABLE rate_code ( @@ -622,7 +631,7 @@ async fn test_prepare_stmt_execute(#[future(awt)] s3: S3, mut conn: PgConnection if let Err(e) = warning_res { assert_eq!( e.to_string(), - "WARNING: publicrate_code does not exist in DuckDB".to_string() + "WARNING: public.rate_code does not exist in DuckDB".to_string() ); } else { panic!("Expecting an warning but got success"); From bcefae51e25e9af20d14bbf4d7ff1feaa5f7961f Mon Sep 17 00:00:00 2001 From: Weijun-H Date: Fri, 27 Sep 2024 13:14:34 +0800 Subject: [PATCH 09/30] chore:test --- tests/tests/scan.rs | 11 +---------- 1 file changed, 1 insertion(+), 10 deletions(-) diff --git a/tests/tests/scan.rs b/tests/tests/scan.rs index a24d291a..77c2f364 100644 --- a/tests/tests/scan.rs +++ b/tests/tests/scan.rs @@ -591,15 +591,6 @@ async fn test_prepare_stmt_execute(#[future(awt)] s3: S3, mut conn: PgConnection assert!("EXECUTE test_query(3)".execute_result(&mut conn).is_err()); - let test: Vec<(String,)> = r#" - SELECT column_name - FROM information_schema.columns - WHERE table_name = 'trips'; - "# - .fetch(&mut conn); - - panic!("{:?}", test); - // cannot fully pushdown to DuckDB r#" CREATE TABLE rate_code ( @@ -624,7 +615,7 @@ async fn test_prepare_stmt_execute(#[future(awt)] s3: S3, mut conn: PgConnection select * from trips join rate_code - on trips.vendorid = rate_code.id; + on trips.VendorID = rate_code.id; "# .execute_result(&mut conn); From eef518fea8fb82076b679136009ec00646821129 Mon Sep 17 00:00:00 2001 From: Weijun-H Date: Tue, 1 Oct 2024 16:38:31 +0800 Subject: [PATCH 10/30] chore: Used sqlparser to parse create view statement --- Cargo.lock | 11 +++++-- Cargo.toml | 2 +- src/hooks/utility.rs | 71 ++++++++++++++++++++++---------------------- 3 files changed, 45 insertions(+), 39 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index b2b958ea..7ad8f342 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -3879,7 +3879,13 @@ dependencies = [ "pgrx", "serde_json", "signal-hook", +<<<<<<< HEAD "sqlparser 0.50.0", +======= + "soa_derive", + "sqlparser 0.51.0", + "sqlx", +>>>>>>> f052c25 (chore: Used sqlparser to parse create view statement) "strum 0.26.3", "supabase-wrappers", "thiserror", @@ -5114,11 +5120,12 @@ dependencies = [ [[package]] name = "sqlparser" -version = "0.50.0" +version = "0.51.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b2e5b515a2bd5168426033e9efbfd05500114833916f1d5c268f938b4ee130ac" +checksum = "5fe11944a61da0da3f592e19a45ebe5ab92dc14a779907ff1f08fbb797bfefc7" dependencies = [ "log", + "sqlparser_derive", ] [[package]] diff --git a/Cargo.toml b/Cargo.toml index 2812ef34..f6b59b27 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -28,7 +28,7 @@ duckdb = { git = "https://github.com/paradedb/duckdb-rs.git", features = [ pgrx = "0.12.6" serde_json = "1.0.128" signal-hook = "0.3.17" -sqlparser = "0.50.0" +sqlparser = { version = "0.51.0", features = ["visitor"] } strum = { version = "0.26.3", features = ["derive"] } supabase-wrappers = { git = "https://github.com/paradedb/wrappers.git", default-features = false, rev = "f5ecb8d" } thiserror = "1.0.63" diff --git a/src/hooks/utility.rs b/src/hooks/utility.rs index 82614493..8dc3083e 100644 --- a/src/hooks/utility.rs +++ b/src/hooks/utility.rs @@ -20,18 +20,22 @@ mod explain; mod prepare; -use std::ffi::CStr; use std::ptr::null_mut; use anyhow::{bail, Result}; use pgrx::{pg_sys, AllocatedByRust, HookResult, PgBox}; use sqlparser::{ast::Statement, dialect::PostgreSqlDialect, parser::Parser}; +use std::ops::ControlFlow; + +use pg_sys::NodeTag; +use pgrx::*; +use sqlparser::ast::visit_relations; -use crate::duckdb::connection::{execute}; use explain::explain_query; -use pgrx::pg_sys::NodeTag; use prepare::*; +use crate::duckdb::connection::{execute, view_exists}; + use super::query::*; type ProcessUtilityHook = fn( @@ -119,9 +123,7 @@ pub async fn process_utility_hook( pstmt.utilityStmt as *mut pg_sys::ExplainStmt, dest.as_ptr(), )?, - pg_sys::NodeTag::T_ViewStmt => { - view_query(query_string, pstmt.utilityStmt as *mut pg_sys::ViewStmt)? - } + pg_sys::NodeTag::T_ViewStmt => view_query(query_string)?, _ => bail!("unexpected statement type in utility hook"), }; @@ -148,43 +150,40 @@ fn is_support_utility(stmt_type: NodeTag) -> bool { || stmt_type == pg_sys::NodeTag::T_ExecuteStmt } -fn view_query(query_string: &core::ffi::CStr, stmt: *mut pg_sys::ViewStmt) -> Result { +fn view_query(query_string: &core::ffi::CStr) -> Result { // Get the current schema in Postgres let current_schema = get_postgres_current_schema(); // Set DuckDB search path according search path in Postgres set_search_path_by_pg()?; - let query = unsafe { (*stmt).query as *mut pg_sys::SelectStmt }; - let from_clause = unsafe { (*query).fromClause }; - unsafe { - let elements = (*from_clause).elements; - for i in 0..(*from_clause).length { - let element = (*elements.offset(i as isize)).ptr_value as *mut pg_sys::Node; - - match (*element).type_ { - pg_sys::NodeTag::T_RangeVar => { - if !analyze_range_var( - element as *mut pg_sys::RangeVar, - current_schema.as_str(), - )? { - return Ok(true); - } - } - pg_sys::NodeTag::T_JoinExpr => { - if !analyze_join_expr( - element as *mut pg_sys::JoinExpr, - current_schema.as_str(), - )? { - return Ok(true); - } - } - _ => { - continue; - } - } + let dialect = PostgreSqlDialect {}; + let statements = Parser::parse_sql(&dialect, query_string.to_str()?)?; + // visit statements, capturing relations (table names) + let mut visited = vec![]; + + visit_relations(&statements, |relation| { + visited.push(relation.clone()); + ControlFlow::<()>::Continue(()) + }); + + for relation in visited.iter() { + let (schema_name, relation_name) = if relation.0.len() == 1 { + (current_schema.clone(), relation.0[0].to_string()) + } else if relation.0.len() == 2 { + (relation.0[0].to_string(), relation.0[1].to_string()) + } else { + error!( + "it is not possible to create a view with more than 2 parts in the relation name" + ); + }; + + if !view_exists(&relation_name, &schema_name)? { + fallback_warning!(format!( + "{schema_name}.{relation_name} does not exist in DuckDB" + )); + return Ok(true); } } - // Push down the view creation query to DuckDB execute(query_string.to_str()?, [])?; Ok(true) From ab1b4af85cd44e6117ca9570df83827ded85032b Mon Sep 17 00:00:00 2001 From: Weijun-H Date: Tue, 1 Oct 2024 17:35:33 +0800 Subject: [PATCH 11/30] chore: Updated tests --- tests/tests/scan.rs | 49 +++++++++++---------------------------------- 1 file changed, 12 insertions(+), 37 deletions(-) diff --git a/tests/tests/scan.rs b/tests/tests/scan.rs index 77c2f364..73e754c2 100644 --- a/tests/tests/scan.rs +++ b/tests/tests/scan.rs @@ -572,11 +572,9 @@ async fn test_prepare_stmt_execute(#[future(awt)] s3: S3, mut conn: PgConnection s3.create_bucket(S3_TRIPS_BUCKET).await?; s3.put_rows(S3_TRIPS_BUCKET, S3_TRIPS_KEY, &rows).await?; - NycTripsTable::setup_s3_listing_fdw( - &s3.url.clone(), - &format!("s3://{S3_TRIPS_BUCKET}/{S3_TRIPS_KEY}"), - ) - .execute(&mut conn); + let mut writer = ArrowWriter::try_new(parquet_file, stored_batch.schema(), None).unwrap(); + writer.write(&stored_batch)?; + writer.close()?; r#"PREPARE test_query(int) AS SELECT count(*) FROM trips WHERE "VendorID" = $1;"# .execute(&mut conn); @@ -591,42 +589,19 @@ async fn test_prepare_stmt_execute(#[future(awt)] s3: S3, mut conn: PgConnection assert!("EXECUTE test_query(3)".execute_result(&mut conn).is_err()); - // cannot fully pushdown to DuckDB + // cannot fully pushdown to the DuckDB + "CREATE TABLE t1 (a int);".execute(&mut conn); + "INSERT INTO t1 VALUES (1);".execute(&mut conn); r#" - CREATE TABLE rate_code ( - id INT PRIMARY KEY, - name TEXT - ); - - INSERT INTO rate_code - (id, name) - VALUES - (1, 'one'), - (2, 'two'), - (3, 'three'), - (4, 'four'), - (5, 'five'), - (99, 'ninety nine'); + CREATE VIEW primitive_join_view AS + SELECT * + FROM primitive + JOIN t1 ON t1.a = primitive.int32_col "# .execute(&mut conn); - let warning_res = r#" - create view trips_with_rate_code as - select * - from trips - join rate_code - on trips.VendorID = rate_code.id; - "# - .execute_result(&mut conn); - - if let Err(e) = warning_res { - assert_eq!( - e.to_string(), - "WARNING: public.rate_code does not exist in DuckDB".to_string() - ); - } else { - panic!("Expecting an warning but got success"); - } + let res: (i32,) = "SELECT int32_col FROM primitive_join_view".fetch_one(&mut conn); + assert_eq!(res.0, 1); Ok(()) } From 32a80717b8f126eddc32dd0e117f580176214a38 Mon Sep 17 00:00:00 2001 From: Weijun-H Date: Fri, 11 Oct 2024 10:24:43 +0800 Subject: [PATCH 12/30] chore: Add comments --- src/hooks/utility.rs | 11 ++++++++--- 1 file changed, 8 insertions(+), 3 deletions(-) diff --git a/src/hooks/utility.rs b/src/hooks/utility.rs index 8dc3083e..118c4a7b 100644 --- a/src/hooks/utility.rs +++ b/src/hooks/utility.rs @@ -151,7 +151,7 @@ fn is_support_utility(stmt_type: NodeTag) -> bool { } fn view_query(query_string: &core::ffi::CStr) -> Result { - // Get the current schema in Postgres + // if the schema is not set in the query, use the current schema in Postgres by default let current_schema = get_postgres_current_schema(); // Set DuckDB search path according search path in Postgres set_search_path_by_pg()?; @@ -171,12 +171,17 @@ fn view_query(query_string: &core::ffi::CStr) -> Result { (current_schema.clone(), relation.0[0].to_string()) } else if relation.0.len() == 2 { (relation.0[0].to_string(), relation.0[1].to_string()) - } else { + } else if relation.0.len() == 3 { + // pg_analytics does not create view with database name now error!( - "it is not possible to create a view with more than 2 parts in the relation name" + "pg_analytics does not support creating view with database name: {}", + relation.0[0].to_string() ); + } else { + bail!("unexpected relation name: {:?}", relation.0); }; + // If the table does not exist in DuckDB, do not push down the query to DuckDB if !view_exists(&relation_name, &schema_name)? { fallback_warning!(format!( "{schema_name}.{relation_name} does not exist in DuckDB" From 918ea6c67e75a32ef2c693b5feb67474833beb0a Mon Sep 17 00:00:00 2001 From: Weijun-H Date: Fri, 11 Oct 2024 10:32:20 +0800 Subject: [PATCH 13/30] chore: Add comments --- src/hooks/utility.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/hooks/utility.rs b/src/hooks/utility.rs index 118c4a7b..c38762a1 100644 --- a/src/hooks/utility.rs +++ b/src/hooks/utility.rs @@ -151,7 +151,7 @@ fn is_support_utility(stmt_type: NodeTag) -> bool { } fn view_query(query_string: &core::ffi::CStr) -> Result { - // if the schema is not set in the query, use the current schema in Postgres by default + // Use the current scheme if the schema is not provided in the query. let current_schema = get_postgres_current_schema(); // Set DuckDB search path according search path in Postgres set_search_path_by_pg()?; From 5fbf120ae74ba1060a7200bdaa0af067872fb8ff Mon Sep 17 00:00:00 2001 From: Weijun-H Date: Mon, 14 Oct 2024 20:41:40 +0800 Subject: [PATCH 14/30] chore: Fix rebase --- src/hooks/utility.rs | 51 ++-------------------------- src/hooks/utility/view.rs | 71 +++++++++++++++++++++++++++++++++++++++ tests/tests/scan.rs | 24 ++++--------- 3 files changed, 79 insertions(+), 67 deletions(-) create mode 100644 src/hooks/utility/view.rs diff --git a/src/hooks/utility.rs b/src/hooks/utility.rs index c38762a1..ee732367 100644 --- a/src/hooks/utility.rs +++ b/src/hooks/utility.rs @@ -19,23 +19,20 @@ #![allow(deprecated)] mod explain; mod prepare; +mod view; use std::ptr::null_mut; use anyhow::{bail, Result}; use pgrx::{pg_sys, AllocatedByRust, HookResult, PgBox}; use sqlparser::{ast::Statement, dialect::PostgreSqlDialect, parser::Parser}; -use std::ops::ControlFlow; +use view::view_query; use pg_sys::NodeTag; -use pgrx::*; -use sqlparser::ast::visit_relations; use explain::explain_query; use prepare::*; -use crate::duckdb::connection::{execute, view_exists}; - use super::query::*; type ProcessUtilityHook = fn( @@ -150,50 +147,6 @@ fn is_support_utility(stmt_type: NodeTag) -> bool { || stmt_type == pg_sys::NodeTag::T_ExecuteStmt } -fn view_query(query_string: &core::ffi::CStr) -> Result { - // Use the current scheme if the schema is not provided in the query. - let current_schema = get_postgres_current_schema(); - // Set DuckDB search path according search path in Postgres - set_search_path_by_pg()?; - - let dialect = PostgreSqlDialect {}; - let statements = Parser::parse_sql(&dialect, query_string.to_str()?)?; - // visit statements, capturing relations (table names) - let mut visited = vec![]; - - visit_relations(&statements, |relation| { - visited.push(relation.clone()); - ControlFlow::<()>::Continue(()) - }); - - for relation in visited.iter() { - let (schema_name, relation_name) = if relation.0.len() == 1 { - (current_schema.clone(), relation.0[0].to_string()) - } else if relation.0.len() == 2 { - (relation.0[0].to_string(), relation.0[1].to_string()) - } else if relation.0.len() == 3 { - // pg_analytics does not create view with database name now - error!( - "pg_analytics does not support creating view with database name: {}", - relation.0[0].to_string() - ); - } else { - bail!("unexpected relation name: {:?}", relation.0); - }; - - // If the table does not exist in DuckDB, do not push down the query to DuckDB - if !view_exists(&relation_name, &schema_name)? { - fallback_warning!(format!( - "{schema_name}.{relation_name} does not exist in DuckDB" - )); - return Ok(true); - } - } - // Push down the view creation query to DuckDB - execute(query_string.to_str()?, [])?; - Ok(true) -} - fn parse_query_from_utility_stmt(query_string: &core::ffi::CStr) -> Result { let query_string = query_string.to_str()?; diff --git a/src/hooks/utility/view.rs b/src/hooks/utility/view.rs new file mode 100644 index 00000000..8cee26b2 --- /dev/null +++ b/src/hooks/utility/view.rs @@ -0,0 +1,71 @@ +// Copyright (c) 2023-2024 Retake, Inc. +// +// This file is part of ParadeDB - Postgres for Search and Analytics +// +// This program is free software: you can redistribute it and/or modify +// it under the terms of the GNU Affero General Public License as published by +// the Free Software Foundation, either version 3 of the License, or +// (at your option) any later version. +// +// This program is distributed in the hope that it will be useful +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU Affero General Public License for more details. +// +// You should have received a copy of the GNU Affero General Public License +// along with this program. If not, see . + +use anyhow::{bail, Result}; +use sqlparser::{dialect::PostgreSqlDialect, parser::Parser}; +use std::ops::ControlFlow; + +use pgrx::*; +use sqlparser::ast::visit_relations; + +use crate::duckdb::connection::{execute, view_exists}; + +use super::{get_postgres_current_schema, set_search_path_by_pg}; + +pub fn view_query(query_string: &core::ffi::CStr) -> Result { + // Use the current scheme if the schema is not provided in the query. + let current_schema = get_postgres_current_schema(); + // Set DuckDB search path according search path in Postgres + set_search_path_by_pg()?; + + let dialect = PostgreSqlDialect {}; + let statements = Parser::parse_sql(&dialect, query_string.to_str()?)?; + // visit statements, capturing relations (table names) + let mut visited = vec![]; + + visit_relations(&statements, |relation| { + visited.push(relation.clone()); + ControlFlow::<()>::Continue(()) + }); + + for relation in visited.iter() { + let (schema_name, relation_name) = if relation.0.len() == 1 { + (current_schema.clone(), relation.0[0].to_string()) + } else if relation.0.len() == 2 { + (relation.0[0].to_string(), relation.0[1].to_string()) + } else if relation.0.len() == 3 { + // pg_analytics does not create view with database name now + error!( + "pg_analytics does not support creating view with database name: {}", + relation.0[0].to_string() + ); + } else { + bail!("unexpected relation name: {:?}", relation.0); + }; + + // If the table does not exist in DuckDB, do not push down the query to DuckDB + if !view_exists(&relation_name, &schema_name)? { + fallback_warning!(format!( + "{schema_name}.{relation_name} does not exist in DuckDB" + )); + return Ok(true); + } + } + // Push down the view creation query to DuckDB + execute(query_string.to_str()?, [])?; + Ok(true) +} diff --git a/tests/tests/scan.rs b/tests/tests/scan.rs index 73e754c2..2835fdf6 100644 --- a/tests/tests/scan.rs +++ b/tests/tests/scan.rs @@ -559,7 +559,6 @@ async fn test_executor_hook_search_path(mut conn: PgConnection, tempdir: TempDir Ok(()) } - #[rstest] async fn test_prepare_stmt_execute(#[future(awt)] s3: S3, mut conn: PgConnection) -> Result<()> { NycTripsTable::setup().execute(&mut conn); @@ -572,9 +571,11 @@ async fn test_prepare_stmt_execute(#[future(awt)] s3: S3, mut conn: PgConnection s3.create_bucket(S3_TRIPS_BUCKET).await?; s3.put_rows(S3_TRIPS_BUCKET, S3_TRIPS_KEY, &rows).await?; - let mut writer = ArrowWriter::try_new(parquet_file, stored_batch.schema(), None).unwrap(); - writer.write(&stored_batch)?; - writer.close()?; + NycTripsTable::setup_s3_listing_fdw( + &s3.url.clone(), + &format!("s3://{S3_TRIPS_BUCKET}/{S3_TRIPS_KEY}"), + ) + .execute(&mut conn); r#"PREPARE test_query(int) AS SELECT count(*) FROM trips WHERE "VendorID" = $1;"# .execute(&mut conn); @@ -589,19 +590,6 @@ async fn test_prepare_stmt_execute(#[future(awt)] s3: S3, mut conn: PgConnection assert!("EXECUTE test_query(3)".execute_result(&mut conn).is_err()); - // cannot fully pushdown to the DuckDB - "CREATE TABLE t1 (a int);".execute(&mut conn); - "INSERT INTO t1 VALUES (1);".execute(&mut conn); - r#" - CREATE VIEW primitive_join_view AS - SELECT * - FROM primitive - JOIN t1 ON t1.a = primitive.int32_col - "# - .execute(&mut conn); - - let res: (i32,) = "SELECT int32_col FROM primitive_join_view".fetch_one(&mut conn); - assert_eq!(res.0, 1); Ok(()) } @@ -695,4 +683,4 @@ async fn test_view_foreign_table(mut conn: PgConnection, tempdir: TempDir) -> Re let res: (i32,) = "SELECT int32_col FROM primitive_join_view".fetch_one(&mut conn); assert_eq!(res.0, 1); Ok(()) -} \ No newline at end of file +} From a2f0e671e2e32dcfddc49bf0bce24013b3e2c249 Mon Sep 17 00:00:00 2001 From: Weijun-H Date: Mon, 14 Oct 2024 20:42:58 +0800 Subject: [PATCH 15/30] chore --- src/hooks/utility.rs | 1 + 1 file changed, 1 insertion(+) diff --git a/src/hooks/utility.rs b/src/hooks/utility.rs index ee732367..2b0afa9b 100644 --- a/src/hooks/utility.rs +++ b/src/hooks/utility.rs @@ -143,6 +143,7 @@ pub async fn process_utility_hook( fn is_support_utility(stmt_type: NodeTag) -> bool { stmt_type == pg_sys::NodeTag::T_ExplainStmt || stmt_type == pg_sys::NodeTag::T_ViewStmt + || stmt_type == pg_sys::NodeTag::T_PrepareStmt || stmt_type == pg_sys::NodeTag::T_DeallocateStmt || stmt_type == pg_sys::NodeTag::T_ExecuteStmt } From d32385529ca26b1a50667ea02750c0a802fd48c3 Mon Sep 17 00:00:00 2001 From: Weijun-H Date: Wed, 16 Oct 2024 21:02:47 +0800 Subject: [PATCH 16/30] refactor: check pgcatalog --- src/hooks/query.rs | 24 ------- src/hooks/utility.rs | 4 +- src/hooks/utility/view.rs | 132 ++++++++++++++++++++++++-------------- 3 files changed, 87 insertions(+), 73 deletions(-) diff --git a/src/hooks/query.rs b/src/hooks/query.rs index e57b3c0d..863a2f97 100644 --- a/src/hooks/query.rs +++ b/src/hooks/query.rs @@ -116,30 +116,6 @@ fn get_postgres_search_path() -> Vec { schema_vec } -pub fn get_postgres_current_schema() -> String { - let active_schemas = - unsafe { PgList::::from_pg(pg_sys::fetch_search_path(false)) }; - - let schema_oid = active_schemas.get_oid(0).unwrap(); - - let mut current_schema = String::new(); - let tuple = unsafe { - pg_sys::SearchSysCache1( - pg_sys::SysCacheIdentifier::NAMESPACEOID as i32, - schema_oid.into_datum().unwrap(), - ) - }; - - if !tuple.is_null() { - let pg_namespace = unsafe { pg_sys::GETSTRUCT(tuple) as pg_sys::Form_pg_namespace }; - let name = pg_sys::name_data_to_str(unsafe { &(*pg_namespace).nspname }); - current_schema = name.to_string(); - - unsafe { pg_sys::ReleaseSysCache(tuple) }; - } - current_schema -} - pub fn is_duckdb_query(relations: &[PgRelation]) -> bool { !relations.is_empty() && relations.iter().all(|pg_relation| { diff --git a/src/hooks/utility.rs b/src/hooks/utility.rs index 2b0afa9b..ffbe9e34 100644 --- a/src/hooks/utility.rs +++ b/src/hooks/utility.rs @@ -120,7 +120,9 @@ pub async fn process_utility_hook( pstmt.utilityStmt as *mut pg_sys::ExplainStmt, dest.as_ptr(), )?, - pg_sys::NodeTag::T_ViewStmt => view_query(query_string)?, + pg_sys::NodeTag::T_ViewStmt => { + view_query(query_string, pstmt.utilityStmt as *mut pg_sys::ViewStmt)? + } _ => bail!("unexpected statement type in utility hook"), }; diff --git a/src/hooks/utility/view.rs b/src/hooks/utility/view.rs index 8cee26b2..721296ec 100644 --- a/src/hooks/utility/view.rs +++ b/src/hooks/utility/view.rs @@ -15,57 +15,93 @@ // You should have received a copy of the GNU Affero General Public License // along with this program. If not, see . -use anyhow::{bail, Result}; -use sqlparser::{dialect::PostgreSqlDialect, parser::Parser}; -use std::ops::ControlFlow; +use anyhow::Result; +use pg_sys::{get_relname_relid, RangeVarGetCreationNamespace}; +use std::ffi::CStr; use pgrx::*; -use sqlparser::ast::visit_relations; - -use crate::duckdb::connection::{execute, view_exists}; - -use super::{get_postgres_current_schema, set_search_path_by_pg}; - -pub fn view_query(query_string: &core::ffi::CStr) -> Result { - // Use the current scheme if the schema is not provided in the query. - let current_schema = get_postgres_current_schema(); - // Set DuckDB search path according search path in Postgres - set_search_path_by_pg()?; - - let dialect = PostgreSqlDialect {}; - let statements = Parser::parse_sql(&dialect, query_string.to_str()?)?; - // visit statements, capturing relations (table names) - let mut visited = vec![]; - - visit_relations(&statements, |relation| { - visited.push(relation.clone()); - ControlFlow::<()>::Continue(()) - }); - - for relation in visited.iter() { - let (schema_name, relation_name) = if relation.0.len() == 1 { - (current_schema.clone(), relation.0[0].to_string()) - } else if relation.0.len() == 2 { - (relation.0[0].to_string(), relation.0[1].to_string()) - } else if relation.0.len() == 3 { - // pg_analytics does not create view with database name now - error!( - "pg_analytics does not support creating view with database name: {}", - relation.0[0].to_string() - ); - } else { - bail!("unexpected relation name: {:?}", relation.0); - }; - - // If the table does not exist in DuckDB, do not push down the query to DuckDB - if !view_exists(&relation_name, &schema_name)? { - fallback_warning!(format!( - "{schema_name}.{relation_name} does not exist in DuckDB" - )); - return Ok(true); + +use crate::{duckdb::connection::execute, hooks::query::is_duckdb_query}; + +use super::set_search_path_by_pg; + +pub fn view_query(query_string: &core::ffi::CStr, stmt: *mut pg_sys::ViewStmt) -> Result { + if analyze_query(stmt)? { + // Push down the view creation query to DuckDB + set_search_path_by_pg()?; + execute(query_string.to_str()?, [])?; + } + Ok(true) +} + +fn analyze_query(stmt: *mut pg_sys::ViewStmt) -> Result { + let query = unsafe { (*stmt).query as *mut pg_sys::SelectStmt }; + let from_clause = unsafe { (*query).fromClause }; + + analyze_from_clause(from_clause) +} + +fn analyze_from_clause(from_clause: *mut pg_sys::List) -> Result { + unsafe { + let elements = (*from_clause).elements; + for i in 0..(*from_clause).length { + let element = (*elements.offset(i as isize)).ptr_value as *mut pg_sys::Node; + + match (*element).type_ { + pg_sys::NodeTag::T_RangeVar => { + return analyze_range_var(element as *mut pg_sys::RangeVar); + } + pg_sys::NodeTag::T_JoinExpr => { + return analyze_join_expr(element as *mut pg_sys::JoinExpr); + } + _ => continue, + } + } + } + Ok(false) +} + +fn analyze_range_var(rv: *mut pg_sys::RangeVar) -> Result { + let pg_relation = unsafe { + let schema_id = RangeVarGetCreationNamespace(rv); + let relid = get_relname_relid((*rv).relname, schema_id); + + pgrx::warning!( + "Relation table name: {:?}", + CStr::from_ptr((*rv).relname).to_str() + ); + + let relation = pg_sys::RelationIdGetRelation(relid); + PgRelation::from_pg_owned(relation) + }; + + Ok(is_duckdb_query(&[pg_relation])) +} + +fn analyze_join_expr(join_expr: *mut pg_sys::JoinExpr) -> Result { + pgrx::warning!("Analyzing JoinExpr"); + + unsafe { + let ltree = (*join_expr).larg; + let rtree = (*join_expr).rarg; + + Ok(analyze_tree(ltree)? && analyze_tree(rtree)?) + } +} + +fn analyze_tree(mut tree: *mut pg_sys::Node) -> Result { + while !tree.is_null() { + unsafe { + match (*tree).type_ { + pg_sys::NodeTag::T_RangeVar => { + return analyze_range_var(tree as *mut pg_sys::RangeVar); + } + pg_sys::NodeTag::T_JoinExpr => { + tree = (*(tree as *mut pg_sys::JoinExpr)).larg; + } + _ => break, + } } } - // Push down the view creation query to DuckDB - execute(query_string.to_str()?, [])?; Ok(true) } From a5d8d9f2deddcd1fd33a295a4630633a18ccafd4 Mon Sep 17 00:00:00 2001 From: Weijun-H Date: Wed, 16 Oct 2024 21:05:21 +0800 Subject: [PATCH 17/30] chore: Fix typo --- .codespellignore | 2 ++ 1 file changed, 2 insertions(+) diff --git a/.codespellignore b/.codespellignore index 2a20e345..4d8836db 100644 --- a/.codespellignore +++ b/.codespellignore @@ -1,2 +1,4 @@ crate socio-economic +larg +rlarg From f9bf692db6c1e71053a294a4a4ba220f3264786b Mon Sep 17 00:00:00 2001 From: Weijun-H Date: Wed, 16 Oct 2024 21:25:34 +0800 Subject: [PATCH 18/30] chore: Clean up --- Cargo.lock | 7 +++---- Cargo.toml | 2 +- src/hooks/utility/view.rs | 26 ++++++++++---------------- 3 files changed, 14 insertions(+), 21 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 7ad8f342..8385ee9e 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -3883,7 +3883,7 @@ dependencies = [ "sqlparser 0.50.0", ======= "soa_derive", - "sqlparser 0.51.0", + "sqlparser 0.50.0", "sqlx", >>>>>>> f052c25 (chore: Used sqlparser to parse create view statement) "strum 0.26.3", @@ -5120,12 +5120,11 @@ dependencies = [ [[package]] name = "sqlparser" -version = "0.51.0" +version = "0.50.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5fe11944a61da0da3f592e19a45ebe5ab92dc14a779907ff1f08fbb797bfefc7" +checksum = "b2e5b515a2bd5168426033e9efbfd05500114833916f1d5c268f938b4ee130ac" dependencies = [ "log", - "sqlparser_derive", ] [[package]] diff --git a/Cargo.toml b/Cargo.toml index f6b59b27..2812ef34 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -28,7 +28,7 @@ duckdb = { git = "https://github.com/paradedb/duckdb-rs.git", features = [ pgrx = "0.12.6" serde_json = "1.0.128" signal-hook = "0.3.17" -sqlparser = { version = "0.51.0", features = ["visitor"] } +sqlparser = "0.50.0" strum = { version = "0.26.3", features = ["derive"] } supabase-wrappers = { git = "https://github.com/paradedb/wrappers.git", default-features = false, rev = "f5ecb8d" } thiserror = "1.0.63" diff --git a/src/hooks/utility/view.rs b/src/hooks/utility/view.rs index 721296ec..7b8ff9ca 100644 --- a/src/hooks/utility/view.rs +++ b/src/hooks/utility/view.rs @@ -17,7 +17,6 @@ use anyhow::Result; use pg_sys::{get_relname_relid, RangeVarGetCreationNamespace}; -use std::ffi::CStr; use pgrx::*; @@ -26,21 +25,20 @@ use crate::{duckdb::connection::execute, hooks::query::is_duckdb_query}; use super::set_search_path_by_pg; pub fn view_query(query_string: &core::ffi::CStr, stmt: *mut pg_sys::ViewStmt) -> Result { - if analyze_query(stmt)? { + let query = unsafe { (*stmt).query as *mut pg_sys::SelectStmt }; + let from_clause = unsafe { (*query).fromClause }; + + if analyze_from_clause(from_clause)? { // Push down the view creation query to DuckDB set_search_path_by_pg()?; execute(query_string.to_str()?, [])?; + } else { + fallback_warning!("relation is not a foreign table from DuckDB"); } Ok(true) } -fn analyze_query(stmt: *mut pg_sys::ViewStmt) -> Result { - let query = unsafe { (*stmt).query as *mut pg_sys::SelectStmt }; - let from_clause = unsafe { (*query).fromClause }; - - analyze_from_clause(from_clause) -} - +/// Analyze the from clause to find the RangeVar node to check if it's a DuckDB query fn analyze_from_clause(from_clause: *mut pg_sys::List) -> Result { unsafe { let elements = (*from_clause).elements; @@ -61,16 +59,12 @@ fn analyze_from_clause(from_clause: *mut pg_sys::List) -> Result { Ok(false) } +/// Check if the RangeVar is a DuckDB query fn analyze_range_var(rv: *mut pg_sys::RangeVar) -> Result { let pg_relation = unsafe { let schema_id = RangeVarGetCreationNamespace(rv); let relid = get_relname_relid((*rv).relname, schema_id); - pgrx::warning!( - "Relation table name: {:?}", - CStr::from_ptr((*rv).relname).to_str() - ); - let relation = pg_sys::RelationIdGetRelation(relid); PgRelation::from_pg_owned(relation) }; @@ -78,9 +72,8 @@ fn analyze_range_var(rv: *mut pg_sys::RangeVar) -> Result { Ok(is_duckdb_query(&[pg_relation])) } +/// Check if the JoinExpr is a DuckDB query fn analyze_join_expr(join_expr: *mut pg_sys::JoinExpr) -> Result { - pgrx::warning!("Analyzing JoinExpr"); - unsafe { let ltree = (*join_expr).larg; let rtree = (*join_expr).rarg; @@ -89,6 +82,7 @@ fn analyze_join_expr(join_expr: *mut pg_sys::JoinExpr) -> Result { } } +/// Analyze the tree recursively to find the RangeVar node to check if it's a DuckDB query fn analyze_tree(mut tree: *mut pg_sys::Node) -> Result { while !tree.is_null() { unsafe { From 834cfbd2804f4f1239dce8478ff65bb50bce33c1 Mon Sep 17 00:00:00 2001 From: Weijun-H Date: Thu, 17 Oct 2024 21:55:40 +0800 Subject: [PATCH 19/30] chore: Warning the table which is not a foreign table from DuckDB --- src/hooks/utility/view.rs | 22 +++++++++++++++++----- 1 file changed, 17 insertions(+), 5 deletions(-) diff --git a/src/hooks/utility/view.rs b/src/hooks/utility/view.rs index 7b8ff9ca..e797df23 100644 --- a/src/hooks/utility/view.rs +++ b/src/hooks/utility/view.rs @@ -15,6 +15,8 @@ // You should have received a copy of the GNU Affero General Public License // along with this program. If not, see . +use std::ffi::CStr; + use anyhow::Result; use pg_sys::{get_relname_relid, RangeVarGetCreationNamespace}; @@ -32,9 +34,8 @@ pub fn view_query(query_string: &core::ffi::CStr, stmt: *mut pg_sys::ViewStmt) - // Push down the view creation query to DuckDB set_search_path_by_pg()?; execute(query_string.to_str()?, [])?; - } else { - fallback_warning!("relation is not a foreign table from DuckDB"); } + Ok(true) } @@ -61,15 +62,26 @@ fn analyze_from_clause(from_clause: *mut pg_sys::List) -> Result { /// Check if the RangeVar is a DuckDB query fn analyze_range_var(rv: *mut pg_sys::RangeVar) -> Result { - let pg_relation = unsafe { + let (pg_relation, rel_name) = unsafe { let schema_id = RangeVarGetCreationNamespace(rv); let relid = get_relname_relid((*rv).relname, schema_id); let relation = pg_sys::RelationIdGetRelation(relid); - PgRelation::from_pg_owned(relation) + ( + PgRelation::from_pg_owned(relation), + CStr::from_ptr((*rv).relname), + ) }; - Ok(is_duckdb_query(&[pg_relation])) + if is_duckdb_query(&[pg_relation]) { + Ok(true) + } else { + fallback_warning!(format!( + "{} is not a foreign table from DuckDB", + rel_name.to_string_lossy() + )); + Ok(false) + } } /// Check if the JoinExpr is a DuckDB query From 8248a74cac0ef6673527db640a234331e077d9a1 Mon Sep 17 00:00:00 2001 From: Weijun-H Date: Fri, 18 Oct 2024 23:27:33 +0800 Subject: [PATCH 20/30] chore: Rebase --- Cargo.lock | 683 +++++++++++++++++++++++++---------------------------- 1 file changed, 322 insertions(+), 361 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 8385ee9e..c9381f55 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -4,18 +4,18 @@ version = 3 [[package]] name = "addr2line" -version = "0.22.0" +version = "0.24.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6e4503c46a5c0c7844e948c9a4d6acd9f50cccb4de1c48eb9e291ea17470c678" +checksum = "dfbe277e56a376000877090da837660b4427aad530e3028d44e0bffe4f89a1c1" dependencies = [ "gimli", ] [[package]] -name = "adler" -version = "1.0.2" +name = "adler2" +version = "2.0.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f26201604c87b1e01bd3d98f8d5d9a8fcbb815e8cedb41ffccbeb4bf593a35fe" +checksum = "512761e0bb2578dd7380c6baaa0f4ce03e84f95e960231d1dec8bf4d7d6e2627" [[package]] name = "ahash" @@ -39,7 +39,7 @@ dependencies = [ "getrandom", "once_cell", "version_check", - "zerocopy 0.7.35", + "zerocopy", ] [[package]] @@ -99,9 +99,9 @@ dependencies = [ [[package]] name = "anyhow" -version = "1.0.88" +version = "1.0.90" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4e1496f8fb1fbf272686b8d37f523dab3e4a7443300055e74cdaa449f3114356" +checksum = "37bf3594c4c988a53154954629820791dde498571819ae4ca50ca811e060cc95" [[package]] name = "approx" @@ -114,15 +114,15 @@ dependencies = [ [[package]] name = "arrayref" -version = "0.3.8" +version = "0.3.9" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9d151e35f61089500b617991b791fc8bfd237ae50cd5950803758a179b41e67a" +checksum = "76a2e8124351fda1ef8aaaa3bbd7ebbcb486bbcd4225aca0aa0d84bb2db8fecb" [[package]] name = "arrayvec" -version = "0.7.4" +version = "0.7.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "96d30a06541fbafbc7f82ed10c06164cfbd2c401138f6addd8404629c4b16711" +checksum = "7c02d123df017efcdfbd739ef81735b36c5ba83ec3c59c80a9d7ecc718f92e50" [[package]] name = "arrow" @@ -362,7 +362,7 @@ dependencies = [ "arrow-schema 51.0.0", "chrono", "half 2.4.1", - "indexmap 2.3.0", + "indexmap 2.6.0", "lexical-core", "num", "serde", @@ -543,9 +543,9 @@ dependencies = [ [[package]] name = "async-compression" -version = "0.4.12" +version = "0.4.16" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fec134f64e2bc57411226dfc4e52dec859ddfc7e711fc5e07b612584f000e4aa" +checksum = "103db485efc3e41214fe4fda9f3dbeae2eb9082f48fd236e6095627a9422066e" dependencies = [ "bzip2", "flate2", @@ -561,13 +561,13 @@ dependencies = [ [[package]] name = "async-executor" -version = "1.13.0" +version = "1.13.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d7ebdfa2ebdab6b1760375fa7d6f382b9f486eac35fc994625a00e89280bdbb7" +checksum = "30ca9a001c1e8ba5149f91a74362376cc6bc5b919d92d988668657bd570bdcec" dependencies = [ "async-task", "concurrent-queue", - "fastrand 2.1.0", + "fastrand 2.1.1", "futures-lite 2.3.0", "slab", ] @@ -580,7 +580,7 @@ checksum = "05b1b633a2115cd122d73b955eadd9916c18c8f510ec9cd1686404c60ad1c29c" dependencies = [ "async-channel 2.3.1", "async-executor", - "async-io 2.3.3", + "async-io 2.3.4", "async-lock 3.4.0", "blocking", "futures-lite 2.3.0", @@ -610,9 +610,9 @@ dependencies = [ [[package]] name = "async-io" -version = "2.3.3" +version = "2.3.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0d6baa8f0178795da0e71bc42c9e5d13261aac7ee549853162e66a241ba17964" +checksum = "444b0228950ee6501b3568d3c93bf1176a1fdbc3b758dcd9475046d30f4dc7e8" dependencies = [ "async-lock 3.4.0", "cfg-if", @@ -620,11 +620,11 @@ dependencies = [ "futures-io", "futures-lite 2.3.0", "parking", - "polling 3.7.2", - "rustix 0.38.34", + "polling 3.7.3", + "rustix 0.38.37", "slab", "tracing", - "windows-sys 0.52.0", + "windows-sys 0.59.0", ] [[package]] @@ -656,7 +656,7 @@ dependencies = [ "async-attributes", "async-channel 1.9.0", "async-global-executor", - "async-io 2.3.3", + "async-io 2.3.4", "async-lock 3.4.0", "crossbeam-utils", "futures-channel", @@ -682,9 +682,9 @@ checksum = "8b75356056920673b02621b35afd0f7dda9306d03c79a30f5c56c44cf256e3de" [[package]] name = "async-trait" -version = "0.1.81" +version = "0.1.83" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6e0c28dcc82d7c8ead5cb13beb15405b57b8546e93215673ff8ca0349a028107" +checksum = "721cae7de5c34fbb2acd27e21e6d2cf7b886dce0c27388d46c4e6c47ea4318dd" dependencies = [ "proc-macro2", "quote", @@ -718,15 +718,15 @@ checksum = "1505bd5d3d116872e7271a6d4e16d81d0c8570876c8de68093a09ac269d8aac0" [[package]] name = "autocfg" -version = "1.3.0" +version = "1.4.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0c4b4d0bd25bd0b74681c0ad21497610ce1b7c91b1022cd21c80c6fbdd9476b0" +checksum = "ace50bade8e6234aa140d9a2f552bbee1db4d353f69b8217bc503490fc1a9f26" [[package]] name = "aws-config" -version = "1.5.6" +version = "1.5.8" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "848d7b9b605720989929279fa644ce8f244d0ce3146fcca5b70e4eb7b3c020fc" +checksum = "7198e6f03240fdceba36656d8be440297b6b82270325908c7381f37d826a74f6" dependencies = [ "aws-credential-types", "aws-runtime", @@ -741,7 +741,7 @@ dependencies = [ "aws-smithy-types", "aws-types", "bytes", - "fastrand 2.1.0", + "fastrand 2.1.1", "hex", "http 0.2.12", "ring", @@ -780,7 +780,7 @@ dependencies = [ "aws-smithy-types", "aws-types", "bytes", - "fastrand 2.1.0", + "fastrand 2.1.1", "http 0.2.12", "http-body 0.4.6", "once_cell", @@ -792,9 +792,9 @@ dependencies = [ [[package]] name = "aws-sdk-s3" -version = "1.49.0" +version = "1.57.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e518950d4ac43508c8bfc2fe4e24b0752d99eab80134461d5e162dcda0214b55" +checksum = "8888c238bf93c77c5df8274b3999fd7fc1bb3fb658616f40dfde9e4fcd9efd94" dependencies = [ "ahash 0.8.11", "aws-credential-types", @@ -811,7 +811,7 @@ dependencies = [ "aws-smithy-xml", "aws-types", "bytes", - "fastrand 2.1.0", + "fastrand 2.1.1", "hex", "hmac", "http 0.2.12", @@ -827,9 +827,9 @@ dependencies = [ [[package]] name = "aws-sdk-sso" -version = "1.42.0" +version = "1.46.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "27bf24cd0d389daa923e974b0e7c38daf308fc21e963c049f57980235017175e" +checksum = "0dc2faec3205d496c7e57eff685dd944203df7ce16a4116d0281c44021788a7b" dependencies = [ "aws-credential-types", "aws-runtime", @@ -849,9 +849,9 @@ dependencies = [ [[package]] name = "aws-sdk-ssooidc" -version = "1.43.0" +version = "1.47.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3b43b3220f1c46ac0e9dcc0a97d94b93305dacb36d1dd393996300c6b9b74364" +checksum = "c93c241f52bc5e0476e259c953234dab7e2a35ee207ee202e86c0095ec4951dc" dependencies = [ "aws-credential-types", "aws-runtime", @@ -871,9 +871,9 @@ dependencies = [ [[package]] name = "aws-sdk-sts" -version = "1.42.0" +version = "1.46.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d1c46924fb1add65bba55636e12812cae2febf68c0f37361766f627ddcca91ce" +checksum = "b259429be94a3459fa1b00c5684faee118d74f9577cc50aebadc36e507c63b5f" dependencies = [ "aws-credential-types", "aws-runtime", @@ -1006,22 +1006,22 @@ dependencies = [ [[package]] name = "aws-smithy-runtime" -version = "1.7.1" +version = "1.7.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d1ce695746394772e7000b39fe073095db6d45a862d0767dd5ad0ac0d7f8eb87" +checksum = "a065c0fe6fdbdf9f11817eb68582b2ab4aff9e9c39e986ae48f7ec576c6322db" dependencies = [ "aws-smithy-async", "aws-smithy-http", "aws-smithy-runtime-api", "aws-smithy-types", "bytes", - "fastrand 2.1.0", + "fastrand 2.1.1", "h2", "http 0.2.12", "http-body 0.4.6", "http-body 1.0.1", "httparse", - "hyper 0.14.30", + "hyper 0.14.31", "hyper-rustls 0.24.2", "once_cell", "pin-project-lite", @@ -1050,9 +1050,9 @@ dependencies = [ [[package]] name = "aws-smithy-types" -version = "1.2.6" +version = "1.2.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "03701449087215b5369c7ea17fef0dd5d24cb93439ec5af0c7615f58c3f22605" +checksum = "147100a7bea70fa20ef224a6bad700358305f5dc0f84649c53769761395b355b" dependencies = [ "base64-simd", "bytes", @@ -1099,17 +1099,17 @@ dependencies = [ [[package]] name = "backtrace" -version = "0.3.73" +version = "0.3.74" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5cc23269a4f8976d0a4d2e7109211a419fe30e8d88d677cd60b6bc79c5732e0a" +checksum = "8d82cb332cdfaed17ae235a638438ac4d4839913cc2af585c3c6746e8f8bee1a" dependencies = [ "addr2line", - "cc", "cfg-if", "libc", "miniz_oxide", "object", "rustc-demangle", + "windows-targets 0.52.6", ] [[package]] @@ -1168,7 +1168,7 @@ dependencies = [ "bitflags 2.6.0", "cexpr", "clang-sys", - "itertools", + "itertools 0.13.0", "proc-macro2", "quote", "regex", @@ -1215,9 +1215,9 @@ dependencies = [ [[package]] name = "blake3" -version = "1.5.3" +version = "1.5.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e9ec96fe9a81b5e365f9db71fe00edc4fe4ca2cc7dcb7861f0603012a7caa210" +checksum = "d82033247fd8e890df8f740e407ad4d038debb9eb1f40533fffb32e7d17dc6f7" dependencies = [ "arrayref", "arrayvec", @@ -1263,7 +1263,7 @@ dependencies = [ "home", "http 1.1.0", "http-body-util", - "hyper 1.4.1", + "hyper 1.5.0", "hyper-named-pipe", "hyper-rustls 0.26.0", "hyper-util", @@ -1271,8 +1271,8 @@ dependencies = [ "log", "pin-project-lite", "rustls 0.22.4", - "rustls-native-certs 0.7.1", - "rustls-pemfile 2.1.2", + "rustls-native-certs 0.7.3", + "rustls-pemfile 2.2.0", "rustls-pki-types", "serde", "serde_derive", @@ -1373,18 +1373,18 @@ dependencies = [ [[package]] name = "bytemuck" -version = "1.16.3" +version = "1.19.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "102087e286b4677862ea56cf8fc58bb2cdfa8725c40ffb80fe3a008eb7f2fc83" +checksum = "8334215b81e418a0a7bdb8ef0849474f40bb10c8b71f1c4ed315cff49f32494d" dependencies = [ "bytemuck_derive", ] [[package]] name = "bytemuck_derive" -version = "1.7.0" +version = "1.8.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1ee891b04274a59bd38b412188e24b849617b2e45a0fd8d057deb63e7403761b" +checksum = "bcfcc3cd946cb52f0bbfdbbcfa2f4e24f75ebb6c0e1002f7c25904fada18b9ec" dependencies = [ "proc-macro2", "quote", @@ -1399,9 +1399,9 @@ checksum = "1fd0f2584146f6f2ef48085050886acf353beff7305ebd1ae69500e27c67f64b" [[package]] name = "bytes" -version = "1.7.1" +version = "1.7.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8318a53db07bb3f8dca91a600466bdb3f2eaadeedfdbcf02e1accbad9271ba50" +checksum = "428d9aa8fbc0670b7b8d6030a7fadd0f86151cae55e4dbbece15f3780a3dfaf3" [[package]] name = "bytes-utils" @@ -1467,12 +1467,13 @@ checksum = "37b2a672a2cb129a2e41c10b1224bb368f9f37a2b16b612598138befd7b37eb5" [[package]] name = "cc" -version = "1.1.7" +version = "1.1.30" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "26a5c3fd7bfa1ce3897a3a3501d362b2d87b7f2583ebcb4a949ec25911025cbc" +checksum = "b16803a61b81d9eabb7eae2588776c4c1e584b738ede45fdbb4c972cec1e9945" dependencies = [ "jobserver", "libc", + "shlex", ] [[package]] @@ -1633,9 +1634,9 @@ dependencies = [ [[package]] name = "constant_time_eq" -version = "0.3.0" +version = "0.3.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f7144d30dcf0fafbce74250a3963025d8d52177934239851c917d29f1df280c2" +checksum = "7c74b8349d32d297c9134b8c88677813a227df8f779daa29bfc29c183fe3dca6" [[package]] name = "convert_case" @@ -1658,15 +1659,15 @@ dependencies = [ [[package]] name = "core-foundation-sys" -version = "0.8.6" +version = "0.8.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "06ea2b9bc92be3c2baa9334a323ebca2d6f074ff852cd1d7b11064035cd3868f" +checksum = "773648b94d0e5d620f64f280777445740e61fe701025087ec8b57f45c791888b" [[package]] name = "cpufeatures" -version = "0.2.12" +version = "0.2.14" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "53fe5e26ff1b7aef8bca9c6080520cfb8d9333c7568e1829cef191a9723e5504" +checksum = "608697df725056feaccfa42cffdaeeec3fccc4ffc38358ecd19b243e716a78e0" dependencies = [ "libc", ] @@ -1858,8 +1859,8 @@ dependencies = [ "glob", "half 2.4.1", "hashbrown 0.14.5", - "indexmap 2.3.0", - "itertools", + "indexmap 2.6.0", + "itertools 0.12.1", "log", "num_cpus", "object_store", @@ -1961,7 +1962,7 @@ dependencies = [ "datafusion-expr", "datafusion-physical-expr", "hex", - "itertools", + "itertools 0.12.1", "log", "md-5", "regex", @@ -1985,7 +1986,7 @@ dependencies = [ "datafusion-execution", "datafusion-expr", "datafusion-functions", - "itertools", + "itertools 0.12.1", "log", "paste", ] @@ -2003,7 +2004,7 @@ dependencies = [ "datafusion-expr", "datafusion-physical-expr", "hashbrown 0.14.5", - "itertools", + "itertools 0.12.1", "log", "regex-syntax", ] @@ -2031,8 +2032,8 @@ dependencies = [ "half 2.4.1", "hashbrown 0.14.5", "hex", - "indexmap 2.3.0", - "itertools", + "indexmap 2.6.0", + "itertools 0.12.1", "log", "md-5", "paste", @@ -2064,8 +2065,8 @@ dependencies = [ "futures", "half 2.4.1", "hashbrown 0.14.5", - "indexmap 2.3.0", - "itertools", + "indexmap 2.6.0", + "itertools 0.12.1", "log", "once_cell", "parking_lot", @@ -2149,8 +2150,8 @@ dependencies = [ "fix-hidden-lifetime-bug", "futures", "hashbrown 0.14.5", - "indexmap 2.3.0", - "itertools", + "indexmap 2.6.0", + "itertools 0.12.1", "lazy_static", "libc", "maplit", @@ -2453,9 +2454,9 @@ dependencies = [ [[package]] name = "fastrand" -version = "2.1.0" +version = "2.1.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9fc0510504f03c51ada170672ac806f1f105a88aa97a5281117e1ddc3368e51a" +checksum = "e8c02a5121d4ea3eb16a80748c74f5549a5665e4c21333c6098f283870fbdea6" [[package]] name = "ff" @@ -2469,30 +2470,30 @@ dependencies = [ [[package]] name = "filetime" -version = "0.2.23" +version = "0.2.25" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1ee447700ac8aa0b2f2bd7bc4462ad686ba06baa6727ac149a2d6277f0d240fd" +checksum = "35c0522e981e68cbfa8c3f978441a5f34b30b96e146b33cd3359176b50fe8586" dependencies = [ "cfg-if", "libc", - "redox_syscall 0.4.1", - "windows-sys 0.52.0", + "libredox", + "windows-sys 0.59.0", ] [[package]] name = "fix-hidden-lifetime-bug" -version = "0.2.6" +version = "0.2.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "40976db4d694ad47c9cdf86bd978cf4cfbe1c29469d5d1fc5b730a75e3d43a9b" +checksum = "ab7b4994e93dd63050356bdde7d417591d1b348523638dc1c1f539f16e338d55" dependencies = [ "fix-hidden-lifetime-bug-proc_macros", ] [[package]] name = "fix-hidden-lifetime-bug-proc_macros" -version = "0.2.6" +version = "0.2.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "75ab4fc77a21a7e7704c9ddbceaeb00ad8512b71b83738f3c0f17a0f44ee24a3" +checksum = "e8f0de9daf465d763422866d0538f07be1596e05623e120b37b4f715f5585200" dependencies = [ "proc-macro2", "quote", @@ -2517,9 +2518,9 @@ dependencies = [ [[package]] name = "flate2" -version = "1.0.30" +version = "1.0.34" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5f54427cfd1c7829e2a139fcefea601bf088ebca651d2bf53ebc600eac295dae" +checksum = "a1b589b4dc103969ad3cf85c950899926ec64300a1a46d76c03a6072957036f0" dependencies = [ "crc32fast", "miniz_oxide", @@ -2542,6 +2543,12 @@ version = "1.0.7" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3f9eec918d3f24069decb9af1554cad7c880e2da24a9afd88aca000531ab82c1" +[[package]] +name = "foldhash" +version = "0.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f81ec6369c545a7d40e4589b5597581fa1c441fe1cce96dd1de43159910a36a2" + [[package]] name = "form_urlencoded" version = "1.2.1" @@ -2559,9 +2566,9 @@ checksum = "e6d5a32815ae3f33302d95fdcb2ce17862f8c65363dcfd29360480ba1001fc9c" [[package]] name = "futures" -version = "0.3.30" +version = "0.3.31" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "645c6916888f6cb6350d2550b80fb63e734897a8498abe35cfb732b6487804b0" +checksum = "65bc07b1a8bc7c85c5f2e110c476c7389b4554ba72af57d8445ea63a576b0876" dependencies = [ "futures-channel", "futures-core", @@ -2574,9 +2581,9 @@ dependencies = [ [[package]] name = "futures-channel" -version = "0.3.30" +version = "0.3.31" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "eac8f7d7865dcb88bd4373ab671c8cf4508703796caa2b1985a9ca867b3fcb78" +checksum = "2dff15bf788c671c1934e366d07e30c1814a8ef514e1af724a602e8a2fbe1b10" dependencies = [ "futures-core", "futures-sink", @@ -2584,15 +2591,15 @@ dependencies = [ [[package]] name = "futures-core" -version = "0.3.30" +version = "0.3.31" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "dfc6580bb841c5a68e9ef15c77ccc837b40a7504914d52e47b8b0e9bbda25a1d" +checksum = "05f29059c0c2090612e8d742178b0580d2dc940c837851ad723096f87af6663e" [[package]] name = "futures-executor" -version = "0.3.30" +version = "0.3.31" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a576fc72ae164fca6b9db127eaa9a9dda0d61316034f33a0a0d4eda41f02b01d" +checksum = "1e28d1d997f585e54aebc3f97d39e72338912123a67330d723fdbb564d646c9f" dependencies = [ "futures-core", "futures-task", @@ -2612,9 +2619,9 @@ dependencies = [ [[package]] name = "futures-io" -version = "0.3.30" +version = "0.3.31" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a44623e20b9681a318efdd71c299b6b222ed6f231972bfe2f224ebad6311f0c1" +checksum = "9e5c1b78ca4aae1ac06c48a526a655760685149f0d465d21f37abfe57ce075c6" [[package]] name = "futures-lite" @@ -2637,7 +2644,7 @@ version = "2.3.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "52527eb5074e35e9339c6b4e8d12600c7128b68fb25dcb9fa9dec18f7c25f3a5" dependencies = [ - "fastrand 2.1.0", + "fastrand 2.1.1", "futures-core", "futures-io", "parking", @@ -2646,9 +2653,9 @@ dependencies = [ [[package]] name = "futures-macro" -version = "0.3.30" +version = "0.3.31" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "87750cf4b7a4c0625b1529e4c543c2182106e4dedc60a2a6455e00d212c489ac" +checksum = "162ee34ebcb7c64a8abebc059ce0fee27c2262618d7b60ed8faf72fef13c3650" dependencies = [ "proc-macro2", "quote", @@ -2657,15 +2664,15 @@ dependencies = [ [[package]] name = "futures-sink" -version = "0.3.30" +version = "0.3.31" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9fb8e00e87438d937621c1c6269e53f536c14d3fbd6a042bb24879e57d474fb5" +checksum = "e575fab7d1e0dcb8d0c7bcf9a63ee213816ab51902e6d244a95819acacf1d4f7" [[package]] name = "futures-task" -version = "0.3.30" +version = "0.3.31" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "38d84fa142264698cdce1a9f9172cf383a0c82de1bddcf3092901442c4097004" +checksum = "f90f7dce0722e95104fcb095585910c0977252f286e354b5e3bd38902cd99988" [[package]] name = "futures-timer" @@ -2675,9 +2682,9 @@ checksum = "f288b0a4f20f9a56b5d1da57e2227c661b7b16168e2f72365f57b63326e29b24" [[package]] name = "futures-util" -version = "0.3.30" +version = "0.3.31" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3d6401deb83407ab3da39eba7e33987a73c3df0c82b4bb5813ee871c19c41d48" +checksum = "9fa08315bb612088cc391249efdc3bc77536f16c91f6cf495e6fbe85b20a4a81" dependencies = [ "futures-channel", "futures-core", @@ -2738,9 +2745,9 @@ dependencies = [ [[package]] name = "gimli" -version = "0.29.0" +version = "0.31.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "40ecd4077b5ae9fd2e9e169b102c6c330d0605168eb0e8bf79952b256dbefffd" +checksum = "07e28edb80900c19c28f1072f2e8aeca7fa06b23cd4169cefe1af5aa3260783f" [[package]] name = "glob" @@ -2783,7 +2790,7 @@ dependencies = [ "futures-sink", "futures-util", "http 0.2.12", - "indexmap 2.3.0", + "indexmap 2.6.0", "slab", "tokio", "tokio-util", @@ -2836,6 +2843,17 @@ dependencies = [ "allocator-api2", ] +[[package]] +name = "hashbrown" +version = "0.15.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e087f84d4f86bf4b218b927129862374b72199ae7d8657835f1e89000eea4fb" +dependencies = [ + "allocator-api2", + "equivalent", + "foldhash", +] + [[package]] name = "hashlink" version = "0.8.4" @@ -2973,9 +2991,9 @@ dependencies = [ [[package]] name = "httparse" -version = "1.9.4" +version = "1.9.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0fcc0b4a115bf80b728eb8ea024ad5bd707b615bfed49e0665b6e0f86fd082d9" +checksum = "7d71d3574edd2771538b901e6549113b4006ece66150fb69c0fb6d9a2adae946" [[package]] name = "httpdate" @@ -2991,9 +3009,9 @@ checksum = "9a3a5bfb195931eeb336b2a7b4d761daec841b97f947d34394601737a7bba5e4" [[package]] name = "hyper" -version = "0.14.30" +version = "0.14.31" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a152ddd61dfaec7273fe8419ab357f33aee0d914c5f4efbf0d96fa749eea5ec9" +checksum = "8c08302e8fa335b151b788c775ff56e7a03ae64ff85c548ee820fecb70356e85" dependencies = [ "bytes", "futures-channel", @@ -3015,9 +3033,9 @@ dependencies = [ [[package]] name = "hyper" -version = "1.4.1" +version = "1.5.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "50dfd22e0e76d0f662d429a5f80fcaf3855009297eab6a0a9f8543834744ba05" +checksum = "bbbff0a806a4728c99295b254c8838933b5b082d75e3cb70c8dab21fdfbcfa9a" dependencies = [ "bytes", "futures-channel", @@ -3039,7 +3057,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "73b7d8abf35697b81a825e386fc151e0d503e8cb5fcb93cc8669c376dfd6f278" dependencies = [ "hex", - "hyper 1.4.1", + "hyper 1.5.0", "hyper-util", "pin-project-lite", "tokio", @@ -3055,7 +3073,7 @@ checksum = "ec3efd23720e2049821a693cbc7e65ea87c72f1c58ff2f9522ff332b1491e590" dependencies = [ "futures-util", "http 0.2.12", - "hyper 0.14.30", + "hyper 0.14.31", "log", "rustls 0.21.12", "rustls-native-certs 0.6.3", @@ -3071,11 +3089,11 @@ checksum = "a0bea761b46ae2b24eb4aef630d8d1c398157b6fc29e6350ecf090a0b70c952c" dependencies = [ "futures-util", "http 1.1.0", - "hyper 1.4.1", + "hyper 1.5.0", "hyper-util", "log", "rustls 0.22.4", - "rustls-native-certs 0.7.1", + "rustls-native-certs 0.7.3", "rustls-pki-types", "tokio", "tokio-rustls 0.25.0", @@ -3084,20 +3102,19 @@ dependencies = [ [[package]] name = "hyper-util" -version = "0.1.6" +version = "0.1.9" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3ab92f4f49ee4fb4f997c784b7a2e0fa70050211e0b6a287f898c3c9785ca956" +checksum = "41296eb09f183ac68eec06e03cdbea2e759633d4067b2f6552fc2e009bcad08b" dependencies = [ "bytes", "futures-channel", "futures-util", "http 1.1.0", "http-body 1.0.1", - "hyper 1.4.1", + "hyper 1.5.0", "pin-project-lite", "socket2 0.5.7", "tokio", - "tower", "tower-service", "tracing", ] @@ -3110,7 +3127,7 @@ checksum = "acf569d43fa9848e510358c07b80f4adf34084ddc28c6a4a651ee8474c070dcc" dependencies = [ "hex", "http-body-util", - "hyper 1.4.1", + "hyper 1.5.0", "hyper-util", "pin-project-lite", "tokio", @@ -3119,9 +3136,9 @@ dependencies = [ [[package]] name = "iana-time-zone" -version = "0.1.60" +version = "0.1.61" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e7ffbb5a1b541ea2561f8c41c087286cc091e21e556a4f09a8f6cbf17b69b141" +checksum = "235e081f3925a06703c2d0117ea8b91f042756fd6e7a6e5d901e8ca1a996b220" dependencies = [ "android_system_properties", "core-foundation-sys", @@ -3175,12 +3192,12 @@ dependencies = [ [[package]] name = "indexmap" -version = "2.3.0" +version = "2.6.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "de3fc2e30ba82dd1b3911c8de1ffc143c74a914a14e99514d7637e3099df5ea0" +checksum = "707907fe3c25f5424cce2cb7e1cbcafee6bdbe735ca90ef77c29e84591e5b9da" dependencies = [ "equivalent", - "hashbrown 0.14.5", + "hashbrown 0.15.0", "serde", ] @@ -3239,6 +3256,15 @@ dependencies = [ "either", ] +[[package]] +name = "itertools" +version = "0.13.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "413ee7dfc52ee1a4949ceeb7dbc8a33f2d6c088194d9f922fb8318faf1f01186" +dependencies = [ + "either", +] + [[package]] name = "itoa" version = "1.0.11" @@ -3256,9 +3282,9 @@ dependencies = [ [[package]] name = "js-sys" -version = "0.3.69" +version = "0.3.72" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "29c15563dc2726973df627357ce0c9ddddbea194836909d655df6a75d2cf296d" +checksum = "6a88f1bda2bd75b0452a14784937d796722fdebfe50df998aeb3f0b7603019a9" dependencies = [ "wasm-bindgen", ] @@ -3347,9 +3373,9 @@ dependencies = [ [[package]] name = "libc" -version = "0.2.155" +version = "0.2.161" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "97b3888a4aecf77e811145cadf6eef5901f4782c53886191b2f693f24761847c" +checksum = "8e9489c2807c139ffd9c1794f4af0ebe86a828db53ecdc7fea2111d0fed085d1" [[package]] name = "libduckdb-sys" @@ -3390,6 +3416,7 @@ checksum = "c0ff37bd590ca25063e35af745c343cb7a0271906fb7b37e4813e8f79f00268d" dependencies = [ "bitflags 2.6.0", "libc", + "redox_syscall", ] [[package]] @@ -3436,11 +3463,11 @@ dependencies = [ [[package]] name = "lru" -version = "0.12.4" +version = "0.12.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "37ee39891760e7d94734f6f63fedc29a2e4a152f836120753a72503f09fcf904" +checksum = "234cf4f4a04dc1f57e24b96cc0cd600cf2af460d4161ac5ecdd0af8e1f3b2a38" dependencies = [ - "hashbrown 0.14.5", + "hashbrown 0.15.0", ] [[package]] @@ -3493,18 +3520,18 @@ checksum = "68354c5c6bd36d73ff3feceb05efa59b6acb7626617f4962be322a825e61f79a" [[package]] name = "miniz_oxide" -version = "0.7.4" +version = "0.8.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b8a240ddb74feaf34a79a7add65a741f3167852fba007066dcac1ca548d89c08" +checksum = "e2d80299ef12ff69b16a84bb182e3b9df68b5a91574d3d4fa6e41b65deec4df1" dependencies = [ - "adler", + "adler2", ] [[package]] name = "mio" -version = "1.0.1" +version = "1.0.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4569e456d394deccd22ce1c1913e6ea0e54519f577285001215d33557431afe4" +checksum = "80e04d1dcff3aae0704555fe5fee3bcfaf3d1fdf8a7e521d5b9d2b42acb52cec" dependencies = [ "hermit-abi 0.3.9", "libc", @@ -3631,9 +3658,9 @@ dependencies = [ [[package]] name = "object" -version = "0.36.2" +version = "0.36.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3f203fa8daa7bb185f760ae12bd8e097f63d17041dcdcaf675ac54cdf863170e" +checksum = "aedf0a2d09c573ed1d8d85b30c119153926a2b36dce0ab28322c09a117a4683e" dependencies = [ "memchr", ] @@ -3649,7 +3676,7 @@ dependencies = [ "chrono", "futures", "humantime", - "itertools", + "itertools 0.12.1", "parking_lot", "percent-encoding", "snafu", @@ -3661,9 +3688,9 @@ dependencies = [ [[package]] name = "once_cell" -version = "1.19.0" +version = "1.20.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3fdb12b2476b595f9358c5161aa467c2438859caa136dec86c26fdd2efe17b92" +checksum = "1261fe7e33c73b354eab43b1273a57c8f967d0391e80353e51f764ac02cf6775" [[package]] name = "openssl-probe" @@ -3694,11 +3721,12 @@ checksum = "4030760ffd992bef45b0ae3f10ce1aba99e33464c90d14dd7c039884963ddc7a" [[package]] name = "owo-colors" -version = "4.0.0" +version = "4.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "caff54706df99d2a78a5a4e3455ff45448d81ef1bb63c22cd14052ca0e993a3f" +checksum = "fb37767f6569cd834a413442455e0f066d0d522de8630436e2a1761d9726ba56" dependencies = [ - "supports-color", + "supports-color 2.1.0", + "supports-color 3.0.1", ] [[package]] @@ -3714,9 +3742,9 @@ dependencies = [ [[package]] name = "parking" -version = "2.2.0" +version = "2.2.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bb813b8af86854136c6922af0598d719255ecb2179515e6e7730d468f05c9cae" +checksum = "f38d5652c16fde515bb1ecef450ab0f6a219d619a7274976324d5e377f7dceba" [[package]] name = "parking_lot" @@ -3736,7 +3764,7 @@ checksum = "1e401f977ab385c9e4e3ab30627d6f26d00e2c73eef317493c4ec6d468726cf8" dependencies = [ "cfg-if", "libc", - "redox_syscall 0.5.3", + "redox_syscall", "smallvec", "windows-targets 0.52.6", ] @@ -3849,9 +3877,9 @@ checksum = "df202b0b0f5b8e389955afd5f27b007b00fb948162953f1db9c70d2c7e3157d7" [[package]] name = "pest" -version = "2.7.11" +version = "2.7.14" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "cd53dff83f26735fdc1ca837098ccf133605d794cdae66acfc2bfac3ec809d95" +checksum = "879952a81a83930934cbf1786752d6dedc3b1f29e8f8fb2ad1d0a36f377cf442" dependencies = [ "memchr", "thiserror", @@ -3865,7 +3893,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "b4c5cc86750666a3ed20bdaf5ca2a0344f9c67674cae0515bec2da16fbaa47db" dependencies = [ "fixedbitset", - "indexmap 2.3.0", + "indexmap 2.6.0", ] [[package]] @@ -3879,13 +3907,7 @@ dependencies = [ "pgrx", "serde_json", "signal-hook", -<<<<<<< HEAD "sqlparser 0.50.0", -======= - "soa_derive", - "sqlparser 0.50.0", - "sqlx", ->>>>>>> f052c25 (chore: Used sqlparser to parse create view statement) "strum 0.26.3", "supabase-wrappers", "thiserror", @@ -4033,26 +4055,6 @@ dependencies = [ "siphasher", ] -[[package]] -name = "pin-project" -version = "1.1.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b6bf43b791c5b9e34c3d182969b4abb522f9343702850a2e57f460d00d09b4b3" -dependencies = [ - "pin-project-internal", -] - -[[package]] -name = "pin-project-internal" -version = "1.1.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2f38a4412a78282e09a2cf38d195ea5420d15ba0602cb375210efbc877243965" -dependencies = [ - "proc-macro2", - "quote", - "syn 2.0.79", -] - [[package]] name = "pin-project-lite" version = "0.2.14" @@ -4067,12 +4069,12 @@ checksum = "8b870d8c151b6f2fb93e84a13146138f05d02ed11c7e7c54f8826aaaf7c9f184" [[package]] name = "piper" -version = "0.2.3" +version = "0.2.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ae1d5c74c9876f070d3e8fd503d748c7d974c3e48da8f41350fa5222ef9b4391" +checksum = "96c8c490f422ef9a4efd2cb5b42b76c8613d7e7dfc1caf667b8a3350a5acc066" dependencies = [ "atomic-waker", - "fastrand 2.1.0", + "fastrand 2.1.1", "futures-io", ] @@ -4109,9 +4111,9 @@ dependencies = [ [[package]] name = "pkg-config" -version = "0.3.30" +version = "0.3.31" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d231b230927b5e4ad203db57bbcbee2802f6bce620b1e4a9024a07d94e2907ec" +checksum = "953ec861398dccce10c670dfeaf3ec4911ca479e9c02154b3a215178c5f566f2" [[package]] name = "polling" @@ -4131,17 +4133,17 @@ dependencies = [ [[package]] name = "polling" -version = "3.7.2" +version = "3.7.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a3ed00ed3fbf728b5816498ecd316d1716eecaced9c0c8d2c5a6740ca214985b" +checksum = "cc2790cd301dec6cd3b7a025e4815cf825724a51c98dccfe6a3e55f05ffb6511" dependencies = [ "cfg-if", "concurrent-queue", "hermit-abi 0.4.0", "pin-project-lite", - "rustix 0.38.34", + "rustix 0.38.37", "tracing", - "windows-sys 0.52.0", + "windows-sys 0.59.0", ] [[package]] @@ -4152,11 +4154,11 @@ checksum = "439ee305def115ba05938db6eb1644ff94165c5ab5e9420d1c1bcedbba909391" [[package]] name = "ppv-lite86" -version = "0.2.18" +version = "0.2.20" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "dee4364d9f3b902ef14fab8a1ddffb783a1cb6b4bba3bfc1fa3922732c7de97f" +checksum = "77957b295656769bb8ad2b6a6b09d897d94f05c41b069aede1fcdaa675eaea04" dependencies = [ - "zerocopy 0.6.6", + "zerocopy", ] [[package]] @@ -4193,9 +4195,9 @@ dependencies = [ [[package]] name = "proc-macro2" -version = "1.0.86" +version = "1.0.88" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5e719e8df665df0d1c8fbfd238015744736151d4445ec0836b8e628aae103b77" +checksum = "7c3a7fc5db1e57d5a779a352c8cdb57b29aa4c40cc69c3a68a7fedc815fbf2f9" dependencies = [ "unicode-ident", ] @@ -4217,7 +4219,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "81bddcdb20abf9501610992b6759a4c888aef7d1a7247ef75e2404275ac24af1" dependencies = [ "anyhow", - "itertools", + "itertools 0.12.1", "proc-macro2", "quote", "syn 2.0.79", @@ -4300,27 +4302,18 @@ dependencies = [ [[package]] name = "redox_syscall" -version = "0.4.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4722d768eff46b75989dd134e5c353f0d6296e5aaa3132e776cbdb56be7731aa" -dependencies = [ - "bitflags 1.3.2", -] - -[[package]] -name = "redox_syscall" -version = "0.5.3" +version = "0.5.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2a908a6e00f1fdd0dfd9c0eb08ce85126f6d8bbda50017e74bc4a4b7d4a926a4" +checksum = "9b6dfecf2c74bce2466cabf93f6664d6998a69eb21e39f4207930065b27b771f" dependencies = [ "bitflags 2.6.0", ] [[package]] name = "redox_users" -version = "0.4.5" +version = "0.4.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bd283d9651eeda4b2a83a43c1c91b266c40fd76ecd39a50a8c630ae69dc72891" +checksum = "ba009ff324d1fc1b900bd1fdb31564febe58a8ccc8a6fdbb93b543d33b13ca43" dependencies = [ "getrandom", "libredox", @@ -4329,9 +4322,9 @@ dependencies = [ [[package]] name = "regex" -version = "1.10.6" +version = "1.11.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4219d74c6b67a3654a9fbebc4b419e22126d13d2f3c4a07ee0cb61ff79a79619" +checksum = "38200e5ee88914975b69f657f0801b6f6dccafd44fd9326302a4aaeecfacb1d8" dependencies = [ "aho-corasick", "memchr", @@ -4341,9 +4334,9 @@ dependencies = [ [[package]] name = "regex-automata" -version = "0.4.7" +version = "0.4.8" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "38caf58cc5ef2fed281f89292ef23f6365465ed9a41b7a7754eb4e26496c92df" +checksum = "368758f23274712b504848e9d5a6f010445cc8b87a7cdb4d7cbee666c1288da3" dependencies = [ "aho-corasick", "memchr", @@ -4358,9 +4351,9 @@ checksum = "53a49587ad06b26609c52e423de037e7f57f20d53535d66e08c695f347df952a" [[package]] name = "regex-syntax" -version = "0.8.4" +version = "0.8.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7a66a03ae7c801facd77a29370b4faec201768915ac14a721ba36f20bc9c209b" +checksum = "2b15c43186be67a4fd63bee50d0303afffcef381492ebe2c5d87f324e1b8815c" [[package]] name = "relative-path" @@ -4405,9 +4398,9 @@ dependencies = [ [[package]] name = "rkyv" -version = "0.7.44" +version = "0.7.45" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5cba464629b3394fc4dbc6f940ff8f5b4ff5c7aef40f29166fd4ad12acbc99c0" +checksum = "9008cd6385b9e161d8229e1f6549dd23c3d022f132a2ea37ac3a10ac4935779b" dependencies = [ "bitvec", "bytecheck", @@ -4423,9 +4416,9 @@ dependencies = [ [[package]] name = "rkyv_derive" -version = "0.7.44" +version = "0.7.45" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a7dddfff8de25e6f62b9d64e6e432bf1c6736c57d20323e15ee10435fbda7c65" +checksum = "503d1d27590a2b0a3a4ca4c94755aa2875657196ecbf401a42eff41d7de532c0" dependencies = [ "proc-macro2", "quote", @@ -4493,9 +4486,9 @@ dependencies = [ [[package]] name = "rust_decimal" -version = "1.35.0" +version = "1.36.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1790d1c4c0ca81211399e0e0af16333276f375209e71a37b67698a373db5b47a" +checksum = "b082d80e3e3cc52b2ed634388d436fe1f4de6af5786cc2de9ba9737527bdf555" dependencies = [ "arrayvec", "borsh", @@ -4553,9 +4546,9 @@ dependencies = [ [[package]] name = "rustix" -version = "0.38.34" +version = "0.38.37" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "70dc5ec042f7a43c4a73241207cecc9873a06d45debb38b329f8541d85c2730f" +checksum = "8acb788b847c24f28525660c4d7758620a7210875711f79e7f663cc152726811" dependencies = [ "bitflags 2.6.0", "errno", @@ -4585,7 +4578,7 @@ dependencies = [ "log", "ring", "rustls-pki-types", - "rustls-webpki 0.102.6", + "rustls-webpki 0.102.8", "subtle", "zeroize", ] @@ -4604,12 +4597,12 @@ dependencies = [ [[package]] name = "rustls-native-certs" -version = "0.7.1" +version = "0.7.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a88d6d420651b496bdd98684116959239430022a115c1240e6c3993be0b15fba" +checksum = "e5bfb394eeed242e909609f56089eecfe5fda225042e8b171791b9c95f5931e5" dependencies = [ "openssl-probe", - "rustls-pemfile 2.1.2", + "rustls-pemfile 2.2.0", "rustls-pki-types", "schannel", "security-framework", @@ -4626,19 +4619,18 @@ dependencies = [ [[package]] name = "rustls-pemfile" -version = "2.1.2" +version = "2.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "29993a25686778eb88d4189742cd713c9bce943bc54251a33509dc63cbacf73d" +checksum = "dce314e5fee3f39953d46bb63bb8a46d40c2f8fb7cc5a3b6cab2bde9721d6e50" dependencies = [ - "base64 0.22.1", "rustls-pki-types", ] [[package]] name = "rustls-pki-types" -version = "1.7.0" +version = "1.10.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "976295e77ce332211c0d24d92c0e83e50f5c5f046d11082cea19f3df13a3562d" +checksum = "16f1201b3c9a7ee8039bcadc17b7e605e2945b27eee7631788c1bd2b0643674b" [[package]] name = "rustls-webpki" @@ -4652,9 +4644,9 @@ dependencies = [ [[package]] name = "rustls-webpki" -version = "0.102.6" +version = "0.102.8" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8e6b52d4fda176fd835fdc55a835d4a89b8499cad995885a21149d5ad62f852e" +checksum = "64ca1bc8749bd4cf37b5ce386cc146580777b4e8572c7b97baf22c83f444bee9" dependencies = [ "ring", "rustls-pki-types", @@ -4663,9 +4655,9 @@ dependencies = [ [[package]] name = "rustversion" -version = "1.0.17" +version = "1.0.18" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "955d28af4278de8121b7ebeb796b6a45735dc01436d898801014aced2773a3d6" +checksum = "0e819f2bc632f285be6d7cd36e25940d45b2391dd6d9b939e79de557f7014248" [[package]] name = "ryu" @@ -4684,11 +4676,11 @@ dependencies = [ [[package]] name = "schannel" -version = "0.1.23" +version = "0.1.26" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fbc91545643bcf3a0bbb6569265615222618bdf33ce4ffbbd13c4bbd4c093534" +checksum = "01227be5826fa0690321a2ba6c5cd57a19cf3f6a09e76973b58e61de6ab9d1c1" dependencies = [ - "windows-sys 0.52.0", + "windows-sys 0.59.0", ] [[package]] @@ -4742,9 +4734,9 @@ dependencies = [ [[package]] name = "security-framework-sys" -version = "2.11.1" +version = "2.12.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "75da29fe9b9b08fe9d6b22b5b4bcbc75d8db3aa31e639aa56bb62e9d46bfceaf" +checksum = "ea4a292869320c0272d7bc55a5a6aafaff59b4f63404a003887b679a2e05b4b6" dependencies = [ "core-foundation-sys", "libc", @@ -4791,9 +4783,9 @@ dependencies = [ [[package]] name = "serde_arrow" -version = "0.11.7" +version = "0.11.8" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2118d57cd864dbd4fc0a54afcd1adc245051edf03c7419b99e4b7fb47bb3052a" +checksum = "f11dc39a704b214e72e4cec092fff98180ac432f5f7850dd0d55e9012c29fba9" dependencies = [ "arrow-array 51.0.0", "arrow-buffer 51.0.0", @@ -4828,9 +4820,9 @@ dependencies = [ [[package]] name = "serde_json" -version = "1.0.128" +version = "1.0.129" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6ff5456707a1de34e7e37f2a6fd3d3f808c318259cbd01ab6377795054b483d8" +checksum = "6dbcf9b78a125ee667ae19388837dd12294b858d101fdd393cb9d5501ef09eb2" dependencies = [ "itoa", "memchr", @@ -4851,9 +4843,9 @@ dependencies = [ [[package]] name = "serde_spanned" -version = "0.6.7" +version = "0.6.8" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "eb5b1b31579f3811bf615c144393417496f152e12ac8b7663bf664f4a815306d" +checksum = "87607cb1398ed59d48732e575a4c28a7a8ebf2454b964fe3f224f2afc07909e1" dependencies = [ "serde", ] @@ -4872,15 +4864,15 @@ dependencies = [ [[package]] name = "serde_with" -version = "3.9.0" +version = "3.11.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "69cecfa94848272156ea67b2b1a53f20fc7bc638c4a46d2f8abde08f05f4b857" +checksum = "8e28bdad6db2b8340e449f7108f020b3b092e8583a9e3fb82713e1d4e71fe817" dependencies = [ "base64 0.22.1", "chrono", "hex", "indexmap 1.9.3", - "indexmap 2.3.0", + "indexmap 2.6.0", "serde", "serde_derive", "serde_json", @@ -4890,9 +4882,9 @@ dependencies = [ [[package]] name = "serde_with_macros" -version = "3.9.0" +version = "3.11.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a8fee4991ef4f274617a51ad4af30519438dacb2f56ac773b08a1922ff743350" +checksum = "9d846214a9854ef724f3da161b426242d8de7c1fc7de2f89bb1efcb154dca79d" dependencies = [ "darling", "proc-macro2", @@ -4969,9 +4961,9 @@ dependencies = [ [[package]] name = "simdutf8" -version = "0.1.4" +version = "0.1.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f27f6278552951f1f2b8cf9da965d10969b2efdea95a6ec47987ab46edfe263a" +checksum = "e3a9fe34e3e7a50316060351f37187a3f546bce95496156754b601a5fa71b76e" [[package]] name = "siphasher" @@ -5100,9 +5092,9 @@ checksum = "3b9b39299b249ad65f3b7e96443bad61c02ca5cd3589f46cb6d610a0fd6c0d6a" [[package]] name = "sqlformat" -version = "0.2.4" +version = "0.2.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f895e3734318cc55f1fe66258926c9b910c124d47520339efecbb6c59cec7c1f" +checksum = "7bba3a93db0cc4f7bdece8bb09e77e2e785c20bfebf79eb8340ed80708048790" dependencies = [ "nom", "unicode_categories", @@ -5176,7 +5168,7 @@ dependencies = [ "futures-util", "hashlink", "hex", - "indexmap 2.3.0", + "indexmap 2.6.0", "log", "memchr", "once_cell", @@ -5482,6 +5474,15 @@ dependencies = [ "is_ci", ] +[[package]] +name = "supports-color" +version = "3.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8775305acf21c96926c900ad056abeef436701108518cf890020387236ac5a77" +dependencies = [ + "is_ci", +] + [[package]] name = "syn" version = "1.0.109" @@ -5524,9 +5525,9 @@ checksum = "55937e1799185b12863d447f42597ed69d9928686b8d88a1df17376a097d8369" [[package]] name = "tar" -version = "0.4.41" +version = "0.4.42" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "cb797dad5fb5b76fcf519e702f4a589483b5ef06567f160c392832c1f5e44909" +checksum = "4ff6c40d3aedb5e06b57c6f669ad17ab063dd1e63d977c6a88e7f4dfa4f04020" dependencies = [ "filetime", "libc", @@ -5535,14 +5536,14 @@ dependencies = [ [[package]] name = "tempfile" -version = "3.12.0" +version = "3.13.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "04cbcdd0c794ebb0d4cf35e88edd2f7d2c4c3e9a5a6dab322839b321c6a87a64" +checksum = "f0f2c9fc62d0beef6951ccffd757e241266a2c833136efbe35af6cd2567dca5b" dependencies = [ "cfg-if", - "fastrand 2.1.0", + "fastrand 2.1.1", "once_cell", - "rustix 0.38.34", + "rustix 0.38.37", "windows-sys 0.59.0", ] @@ -5612,18 +5613,18 @@ dependencies = [ [[package]] name = "thiserror" -version = "1.0.63" +version = "1.0.64" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c0342370b38b6a11b6cc11d6a805569958d54cfa061a29969c3b5ce2ea405724" +checksum = "d50af8abc119fb8bb6dbabcfa89656f46f84aa0ac7688088608076ad2b459a84" dependencies = [ "thiserror-impl", ] [[package]] name = "thiserror-impl" -version = "1.0.63" +version = "1.0.64" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a4558b58466b9ad7ca0f102865eccc95938dca1a74a856f2b57b6629050da261" +checksum = "08904e7672f5eb876eaaf87e0ce17857500934f4981c4a0ab2b4aa98baac7fc3" dependencies = [ "proc-macro2", "quote", @@ -5748,9 +5749,9 @@ dependencies = [ [[package]] name = "tokio-util" -version = "0.7.11" +version = "0.7.12" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9cf6b47b3771c49ac75ad09a6162f53ad4b8088b76ac60e8ec1455b31a189fe1" +checksum = "61e7c3654c13bcd040d4a03abee2c75b1d14a37b423cf5a813ceae1cc903ec6a" dependencies = [ "bytes", "futures-core", @@ -5782,43 +5783,22 @@ dependencies = [ [[package]] name = "toml_edit" -version = "0.22.20" +version = "0.22.22" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "583c44c02ad26b0c3f3066fe629275e50627026c51ac2e595cca4c230ce1ce1d" +checksum = "4ae48d6208a266e853d946088ed816055e556cc6028c5e8e2b84d9fa5dd7c7f5" dependencies = [ - "indexmap 2.3.0", + "indexmap 2.6.0", "serde", "serde_spanned", "toml_datetime", "winnow", ] -[[package]] -name = "tower" -version = "0.4.13" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b8fa9be0de6cf49e536ce1851f987bd21a43b771b09473c3549a6c853db37c1c" -dependencies = [ - "futures-core", - "futures-util", - "pin-project", - "pin-project-lite", - "tokio", - "tower-layer", - "tower-service", -] - -[[package]] -name = "tower-layer" -version = "0.3.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c20c8dbed6283a09604c3e69b4b7eeb54e298b8a600d4d5ecb5ad39de609f1d0" - [[package]] name = "tower-service" -version = "0.3.2" +version = "0.3.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b6bc1c9ce2b5135ac7f93c72918fc37feb872bdc6a5533a8b85eb4b86bfdae52" +checksum = "8df9b6e13f2d32c91b9bd719c00d1958837bc7dec474d94952798cc8e69eeec3" [[package]] name = "tracing" @@ -5876,9 +5856,9 @@ checksum = "42ff0bf0c66b8238c6f3b578df37d0b7848e55df8577b3f74f92a69acceeb825" [[package]] name = "ucd-trie" -version = "0.1.6" +version = "0.1.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ed646292ffc8188ef8ea4d1e0e0150fb15a5c2e12ad9b8fc191ae7a8a7f3c4b9" +checksum = "2896d95c02a80c6d6a5d6e953d479f5ddf2dfdb6a244441010e373ac0fb88971" [[package]] name = "unescape" @@ -5888,9 +5868,9 @@ checksum = "ccb97dac3243214f8d8507998906ca3e2e0b900bf9bf4870477f125b82e68f6e" [[package]] name = "unicode-bidi" -version = "0.3.15" +version = "0.3.17" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "08f95100a766bf4f8f28f90d77e0a5461bbdb219042e7679bebe79004fed8d75" +checksum = "5ab17db44d7388991a428b2ee655ce0c212e862eff1768a455c58f9aad6e7893" [[package]] name = "unicode-ident" @@ -5900,30 +5880,30 @@ checksum = "e91b56cd4cadaeb79bbf1a5645f6b4f8dc5bde8834ad5894a8db35fda9efa1fe" [[package]] name = "unicode-normalization" -version = "0.1.23" +version = "0.1.24" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a56d1686db2308d901306f92a263857ef59ea39678a5458e7cb17f01415101f5" +checksum = "5033c97c4262335cded6d6fc3e5c18ab755e1a3dc96376350f3d8e9f009ad956" dependencies = [ "tinyvec", ] [[package]] name = "unicode-properties" -version = "0.1.1" +version = "0.1.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e4259d9d4425d9f0661581b804cb85fe66a4c631cadd8f490d1c13a35d5d9291" +checksum = "e70f2a8b45122e719eb623c01822704c4e0907e7e426a05927e1a1cfff5b75d0" [[package]] name = "unicode-segmentation" -version = "1.11.0" +version = "1.12.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d4c87d22b6e3f4a18d4d40ef354e97c90fcb14dd91d7dc0aa9d8a1172ebf7202" +checksum = "f6ccf251212114b54433ec949fd6a7841275f9ada20dddd2f29e9ceea4501493" [[package]] name = "unicode-width" -version = "0.1.13" +version = "0.1.14" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0336d538f7abc86d282a4189614dfaa90810dfc2c6f6427eaf88e16311dd225d" +checksum = "7dd6e30e90baa6f72411720665d41d89b9a3d039dc45b8faea1ddd07f617f6af" [[package]] name = "unicode_categories" @@ -5957,9 +5937,9 @@ checksum = "daf8dba3b7eb870caf1ddeed7bc9d2a049f3cfdfae7cb521b087cc33ae4c49da" [[package]] name = "uuid" -version = "1.10.0" +version = "1.11.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "81dfa00651efa65069b0b6b651f4aaa31ba9e3c3ce0137aaad053604ee7e0314" +checksum = "f8c5f0a0af699448548ad1a2fbf920fb4bee257eae39953ba95cb84891a0446a" dependencies = [ "getrandom", "serde", @@ -6028,19 +6008,20 @@ checksum = "b8dad83b4f25e74f184f64c43b150b91efe7647395b42289f38e50566d82855b" [[package]] name = "wasm-bindgen" -version = "0.2.92" +version = "0.2.95" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4be2531df63900aeb2bca0daaaddec08491ee64ceecbee5076636a3b026795a8" +checksum = "128d1e363af62632b8eb57219c8fd7877144af57558fb2ef0368d0087bddeb2e" dependencies = [ "cfg-if", + "once_cell", "wasm-bindgen-macro", ] [[package]] name = "wasm-bindgen-backend" -version = "0.2.92" +version = "0.2.95" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "614d787b966d3989fa7bb98a654e369c762374fd3213d212cfc0251257e747da" +checksum = "cb6dd4d3ca0ddffd1dd1c9c04f94b868c37ff5fac97c30b97cff2d74fce3a358" dependencies = [ "bumpalo", "log", @@ -6053,9 +6034,9 @@ dependencies = [ [[package]] name = "wasm-bindgen-futures" -version = "0.4.42" +version = "0.4.45" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "76bc14366121efc8dbb487ab05bcc9d346b3b5ec0eaa76e46594cabbe51762c0" +checksum = "cc7ec4f8827a71586374db3e87abdb5a2bb3a15afed140221307c3ec06b1f63b" dependencies = [ "cfg-if", "js-sys", @@ -6065,9 +6046,9 @@ dependencies = [ [[package]] name = "wasm-bindgen-macro" -version = "0.2.92" +version = "0.2.95" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a1f8823de937b71b9460c0c34e25f3da88250760bec0ebac694b49997550d726" +checksum = "e79384be7f8f5a9dd5d7167216f022090cf1f9ec128e6e6a482a2cb5c5422c56" dependencies = [ "quote", "wasm-bindgen-macro-support", @@ -6075,9 +6056,9 @@ dependencies = [ [[package]] name = "wasm-bindgen-macro-support" -version = "0.2.92" +version = "0.2.95" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e94f17b526d0a461a191c78ea52bbce64071ed5c04c9ffe424dcb38f74171bb7" +checksum = "26c6ab57572f7a24a4985830b120de1594465e5d500f24afe89e16b4e833ef68" dependencies = [ "proc-macro2", "quote", @@ -6088,15 +6069,15 @@ dependencies = [ [[package]] name = "wasm-bindgen-shared" -version = "0.2.92" +version = "0.2.95" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "af190c94f2773fdb3729c55b007a722abb5384da03bc0986df4c289bf5567e96" +checksum = "65fc09f10666a9f147042251e0dda9c18f166ff7de300607007e96bdebc1068d" [[package]] name = "web-sys" -version = "0.3.69" +version = "0.3.72" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "77afa9a11836342370f4817622a2f0f418b134426d91a82dfb48f532d2ec13ef" +checksum = "f6488b90108c040df0fe62fa815cbdee25124641df01814dd7282749234c6112" dependencies = [ "js-sys", "wasm-bindgen", @@ -6104,11 +6085,11 @@ dependencies = [ [[package]] name = "whoami" -version = "1.5.1" +version = "1.5.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a44ab49fad634e88f55bf8f9bb3abd2f27d7204172a112c7c9987e01c1c94ea9" +checksum = "372d5b87f58ec45c384ba03563b03544dc5fadc3983e434b286913f5b4a9bb6d" dependencies = [ - "redox_syscall 0.4.1", + "redox_syscall", "wasite", ] @@ -6130,11 +6111,11 @@ checksum = "ac3b87c63620426dd9b991e5ce0329eff545bccbbb34f3be09ff6fb6ab51b7b6" [[package]] name = "winapi-util" -version = "0.1.8" +version = "0.1.9" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4d4cc384e1e73b93bafa6fb4f1df8c41695c8a91cf9c4c64358067d15a7b6c6b" +checksum = "cf221c93e13a30d793f7645a0e7762c55d169dbb0a49671918a2319d289b10bb" dependencies = [ - "windows-sys 0.52.0", + "windows-sys 0.59.0", ] [[package]] @@ -6302,9 +6283,9 @@ checksum = "589f6da84c646204747d1270a2a5661ea66ed1cced2631d546fdfb155959f9ec" [[package]] name = "winnow" -version = "0.6.18" +version = "0.6.20" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "68a9bda4691f099d435ad181000724da8e5899daa10713c2d432552b9ccd3a6f" +checksum = "36c1fec1a2bb5866f07c25f68c26e565c4c200aebb96d7e55710c19d3e8ac49b" dependencies = [ "memchr", ] @@ -6326,7 +6307,7 @@ checksum = "8da84f1a25939b27f6820d92aed108f83ff920fdf11a7b19366c27c4cda81d4f" dependencies = [ "libc", "linux-raw-sys 0.4.14", - "rustix 0.38.34", + "rustix 0.38.37", ] [[package]] @@ -6359,34 +6340,14 @@ version = "3.0.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "2a599daf1b507819c1121f0bf87fa37eb19daac6aff3aefefd4e6e2e0f2020fc" -[[package]] -name = "zerocopy" -version = "0.6.6" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "854e949ac82d619ee9a14c66a1b674ac730422372ccb759ce0c39cabcf2bf8e6" -dependencies = [ - "byteorder", - "zerocopy-derive 0.6.6", -] - [[package]] name = "zerocopy" version = "0.7.35" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "1b9b4fd18abc82b8136838da5d50bae7bdea537c574d8dc1a34ed098d6c166f0" dependencies = [ - "zerocopy-derive 0.7.35", -] - -[[package]] -name = "zerocopy-derive" -version = "0.6.6" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "125139de3f6b9d625c39e2efdd73d41bdac468ccd556556440e322be0e1bbd91" -dependencies = [ - "proc-macro2", - "quote", - "syn 2.0.79", + "byteorder", + "zerocopy-derive", ] [[package]] From 01cdc6c4f3eeb4bd27188779f42e4149052b1535 Mon Sep 17 00:00:00 2001 From: Weijun-H Date: Tue, 22 Oct 2024 17:10:47 +0800 Subject: [PATCH 21/30] refactor: Using parse_analyze_fixedparams --- src/hooks/utility.rs | 10 +++- src/hooks/utility/view.rs | 117 ++++++++++++-------------------------- 2 files changed, 42 insertions(+), 85 deletions(-) diff --git a/src/hooks/utility.rs b/src/hooks/utility.rs index ffbe9e34..a94f6bd7 100644 --- a/src/hooks/utility.rs +++ b/src/hooks/utility.rs @@ -120,9 +120,13 @@ pub async fn process_utility_hook( pstmt.utilityStmt as *mut pg_sys::ExplainStmt, dest.as_ptr(), )?, - pg_sys::NodeTag::T_ViewStmt => { - view_query(query_string, pstmt.utilityStmt as *mut pg_sys::ViewStmt)? - } + pg_sys::NodeTag::T_ViewStmt => view_query( + query_string, + parse_state, + pstmt.utilityStmt as *mut pg_sys::ViewStmt, + pstmt.stmt_location, + pstmt.stmt_len, + )?, _ => bail!("unexpected statement type in utility hook"), }; diff --git a/src/hooks/utility/view.rs b/src/hooks/utility/view.rs index e797df23..e0a4bb4a 100644 --- a/src/hooks/utility/view.rs +++ b/src/hooks/utility/view.rs @@ -15,99 +15,52 @@ // You should have received a copy of the GNU Affero General Public License // along with this program. If not, see . -use std::ffi::CStr; +use std::ptr::null_mut; use anyhow::Result; -use pg_sys::{get_relname_relid, RangeVarGetCreationNamespace}; +use pg_sys::parse_analyze_fixedparams; use pgrx::*; use crate::{duckdb::connection::execute, hooks::query::is_duckdb_query}; -use super::set_search_path_by_pg; - -pub fn view_query(query_string: &core::ffi::CStr, stmt: *mut pg_sys::ViewStmt) -> Result { - let query = unsafe { (*stmt).query as *mut pg_sys::SelectStmt }; - let from_clause = unsafe { (*query).fromClause }; - - if analyze_from_clause(from_clause)? { - // Push down the view creation query to DuckDB - set_search_path_by_pg()?; - execute(query_string.to_str()?, [])?; - } - - Ok(true) -} - -/// Analyze the from clause to find the RangeVar node to check if it's a DuckDB query -fn analyze_from_clause(from_clause: *mut pg_sys::List) -> Result { - unsafe { - let elements = (*from_clause).elements; - for i in 0..(*from_clause).length { - let element = (*elements.offset(i as isize)).ptr_value as *mut pg_sys::Node; - - match (*element).type_ { - pg_sys::NodeTag::T_RangeVar => { - return analyze_range_var(element as *mut pg_sys::RangeVar); - } - pg_sys::NodeTag::T_JoinExpr => { - return analyze_join_expr(element as *mut pg_sys::JoinExpr); - } - _ => continue, - } - } - } - Ok(false) -} - -/// Check if the RangeVar is a DuckDB query -fn analyze_range_var(rv: *mut pg_sys::RangeVar) -> Result { - let (pg_relation, rel_name) = unsafe { - let schema_id = RangeVarGetCreationNamespace(rv); - let relid = get_relname_relid((*rv).relname, schema_id); - - let relation = pg_sys::RelationIdGetRelation(relid); - ( - PgRelation::from_pg_owned(relation), - CStr::from_ptr((*rv).relname), +use super::{get_query_relations, set_search_path_by_pg}; + +pub fn view_query( + query_string: &core::ffi::CStr, + pstate: *mut pg_sys::ParseState, + stmt: *mut pg_sys::ViewStmt, + stmt_location: i32, + stmt_len: i32, +) -> Result { + // Perform parsing and analysis to get the Query + let query = unsafe { + let mut raw_stmt = pg_sys::RawStmt { + type_: pg_sys::NodeTag::T_RawStmt, + stmt: (*stmt).query, + stmt_location, + stmt_len, + }; + + parse_analyze_fixedparams( + &mut raw_stmt, + (*pstate).p_sourcetext, + null_mut(), + 0, + null_mut(), ) }; - if is_duckdb_query(&[pg_relation]) { - Ok(true) - } else { - fallback_warning!(format!( - "{} is not a foreign table from DuckDB", - rel_name.to_string_lossy() - )); - Ok(false) - } -} - -/// Check if the JoinExpr is a DuckDB query -fn analyze_join_expr(join_expr: *mut pg_sys::JoinExpr) -> Result { - unsafe { - let ltree = (*join_expr).larg; - let rtree = (*join_expr).rarg; - - Ok(analyze_tree(ltree)? && analyze_tree(rtree)?) - } -} + let query_relations = get_query_relations(unsafe { (*query).rtable }); -/// Analyze the tree recursively to find the RangeVar node to check if it's a DuckDB query -fn analyze_tree(mut tree: *mut pg_sys::Node) -> Result { - while !tree.is_null() { - unsafe { - match (*tree).type_ { - pg_sys::NodeTag::T_RangeVar => { - return analyze_range_var(tree as *mut pg_sys::RangeVar); - } - pg_sys::NodeTag::T_JoinExpr => { - tree = (*(tree as *mut pg_sys::JoinExpr)).larg; - } - _ => break, - } - } + if unsafe { (*query).commandType } != pg_sys::CmdType::CMD_SELECT + || !is_duckdb_query(&query_relations) + { + fallback_warning!("Some relations are not in DuckDB"); + return Ok(true); } + // Push down the view creation query to DuckDB + set_search_path_by_pg()?; + execute(query_string.to_str()?, [])?; Ok(true) } From 4b69e43b56d4611cbbe6f9e42ca858510a3b6757 Mon Sep 17 00:00:00 2001 From: Weijun-H Date: Tue, 22 Oct 2024 17:34:28 +0800 Subject: [PATCH 22/30] refactor: Update parse_analyze_fixedparams usage in view_query function --- src/hooks/utility/view.rs | 31 ++++++++++++++++++++++--------- 1 file changed, 22 insertions(+), 9 deletions(-) diff --git a/src/hooks/utility/view.rs b/src/hooks/utility/view.rs index e0a4bb4a..208110f0 100644 --- a/src/hooks/utility/view.rs +++ b/src/hooks/utility/view.rs @@ -18,9 +18,8 @@ use std::ptr::null_mut; use anyhow::Result; -use pg_sys::parse_analyze_fixedparams; -use pgrx::*; +use pgrx::{pg_sys, warning}; use crate::{duckdb::connection::execute, hooks::query::is_duckdb_query}; @@ -42,13 +41,27 @@ pub fn view_query( stmt_len, }; - parse_analyze_fixedparams( - &mut raw_stmt, - (*pstate).p_sourcetext, - null_mut(), - 0, - null_mut(), - ) + #[cfg(any(feature = "pg15", feature = "pg16", feature = "pg17"))] + { + pg_sys::parse_analyze_fixedparams( + &mut raw_stmt, + (*pstate).p_sourcetext, + null_mut(), + 0, + null_mut(), + ) + } + + #[cfg(any(feature = "pg13", feature = "pg14"))] + { + pg_sys::parse_analyze( + &mut raw_stmt, + (*pstate).p_sourcetext, + null_mut(), + 0, + null_mut(), + ) + } }; let query_relations = get_query_relations(unsafe { (*query).rtable }); From b6f90c3cf83011f3054adf2acf65eebda563f531 Mon Sep 17 00:00:00 2001 From: Weijun-H Date: Tue, 22 Oct 2024 17:54:40 +0800 Subject: [PATCH 23/30] refactor: Update parse_analyze_fixedparams usage in view_query function --- src/hooks/utility.rs | 1 - src/hooks/utility/view.rs | 5 ++--- 2 files changed, 2 insertions(+), 4 deletions(-) diff --git a/src/hooks/utility.rs b/src/hooks/utility.rs index a94f6bd7..4cb69e14 100644 --- a/src/hooks/utility.rs +++ b/src/hooks/utility.rs @@ -122,7 +122,6 @@ pub async fn process_utility_hook( )?, pg_sys::NodeTag::T_ViewStmt => view_query( query_string, - parse_state, pstmt.utilityStmt as *mut pg_sys::ViewStmt, pstmt.stmt_location, pstmt.stmt_len, diff --git a/src/hooks/utility/view.rs b/src/hooks/utility/view.rs index 208110f0..d61b25b4 100644 --- a/src/hooks/utility/view.rs +++ b/src/hooks/utility/view.rs @@ -27,7 +27,6 @@ use super::{get_query_relations, set_search_path_by_pg}; pub fn view_query( query_string: &core::ffi::CStr, - pstate: *mut pg_sys::ParseState, stmt: *mut pg_sys::ViewStmt, stmt_location: i32, stmt_len: i32, @@ -45,7 +44,7 @@ pub fn view_query( { pg_sys::parse_analyze_fixedparams( &mut raw_stmt, - (*pstate).p_sourcetext, + query_string.as_ptr(), null_mut(), 0, null_mut(), @@ -56,7 +55,7 @@ pub fn view_query( { pg_sys::parse_analyze( &mut raw_stmt, - (*pstate).p_sourcetext, + query_string.as_ptr(), null_mut(), 0, null_mut(), From 7e0df9951446f99d72a0540f7c3f9074e8953513 Mon Sep 17 00:00:00 2001 From: Weijun-H Date: Tue, 22 Oct 2024 22:48:54 +0800 Subject: [PATCH 24/30] refactor: Update parse_analyze_fixedparams usage in view_query function and use pg_analyze_and_rewrite_fixedparams instead --- src/hooks/utility/view.rs | 37 +++++++++++++++++++++++++++---------- 1 file changed, 27 insertions(+), 10 deletions(-) diff --git a/src/hooks/utility/view.rs b/src/hooks/utility/view.rs index d61b25b4..9ff3d6ff 100644 --- a/src/hooks/utility/view.rs +++ b/src/hooks/utility/view.rs @@ -32,7 +32,7 @@ pub fn view_query( stmt_len: i32, ) -> Result { // Perform parsing and analysis to get the Query - let query = unsafe { + let query_list = unsafe { let mut raw_stmt = pg_sys::RawStmt { type_: pg_sys::NodeTag::T_RawStmt, stmt: (*stmt).query, @@ -42,7 +42,7 @@ pub fn view_query( #[cfg(any(feature = "pg15", feature = "pg16", feature = "pg17"))] { - pg_sys::parse_analyze_fixedparams( + pg_sys::pg_analyze_and_rewrite_fixedparams( &mut raw_stmt, query_string.as_ptr(), null_mut(), @@ -53,9 +53,9 @@ pub fn view_query( #[cfg(any(feature = "pg13", feature = "pg14"))] { - pg_sys::parse_analyze( + pg_sys::pg_analyze_and_rewrite( &mut raw_stmt, - query_string.as_ptr(), + (*pstate).p_sourcetext, null_mut(), 0, null_mut(), @@ -63,14 +63,31 @@ pub fn view_query( } }; - let query_relations = get_query_relations(unsafe { (*query).rtable }); + let plan_list = unsafe { + pg_sys::pg_plan_queries( + query_list, + query_string.as_ptr(), + pg_sys::CURSOR_OPT_PARALLEL_OK as i32, + null_mut(), + ) + }; + + unsafe { + for i in 0..(*plan_list).length { + let planned_stmt: *mut pg_sys::PlannedStmt = + (*(*plan_list).elements.offset(i as isize)).ptr_value as *mut pg_sys::PlannedStmt; - if unsafe { (*query).commandType } != pg_sys::CmdType::CMD_SELECT - || !is_duckdb_query(&query_relations) - { - fallback_warning!("Some relations are not in DuckDB"); - return Ok(true); + let query_relations = get_query_relations((*planned_stmt).rtable); + + if (*planned_stmt).commandType != pg_sys::CmdType::CMD_SELECT + || !is_duckdb_query(&query_relations) + { + fallback_warning!("Some relations are not in DuckDB"); + return Ok(true); + } + } } + // Push down the view creation query to DuckDB set_search_path_by_pg()?; execute(query_string.to_str()?, [])?; From 59356354495e7a9900f4000ec7d6652f3bf7abbe Mon Sep 17 00:00:00 2001 From: Weijun-H Date: Tue, 22 Oct 2024 22:55:37 +0800 Subject: [PATCH 25/30] fix: CI --- src/hooks/utility/view.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/hooks/utility/view.rs b/src/hooks/utility/view.rs index 9ff3d6ff..5037b5da 100644 --- a/src/hooks/utility/view.rs +++ b/src/hooks/utility/view.rs @@ -55,7 +55,7 @@ pub fn view_query( { pg_sys::pg_analyze_and_rewrite( &mut raw_stmt, - (*pstate).p_sourcetext, + query_string.as_ptr(), null_mut(), 0, null_mut(), From 795dc23a69af3a9d31b4474b9d487b185a3f8900 Mon Sep 17 00:00:00 2001 From: Weijun-H Date: Fri, 25 Oct 2024 21:56:37 +0800 Subject: [PATCH 26/30] chore --- src/hooks/utility.rs | 30 ++++++++++++++++-------------- src/hooks/utility/view.rs | 23 ++++++++++++----------- tests/tests/scan.rs | 3 ++- 3 files changed, 30 insertions(+), 26 deletions(-) diff --git a/src/hooks/utility.rs b/src/hooks/utility.rs index 4cb69e14..b42ef984 100644 --- a/src/hooks/utility.rs +++ b/src/hooks/utility.rs @@ -23,18 +23,14 @@ mod view; use std::ptr::null_mut; +use super::query::*; use anyhow::{bail, Result}; +use explain::explain_query; use pgrx::{pg_sys, AllocatedByRust, HookResult, PgBox}; +use prepare::*; use sqlparser::{ast::Statement, dialect::PostgreSqlDialect, parser::Parser}; use view::view_query; -use pg_sys::NodeTag; - -use explain::explain_query; -use prepare::*; - -use super::query::*; - type ProcessUtilityHook = fn( pstmt: PgBox, query_string: &core::ffi::CStr, @@ -120,12 +116,18 @@ pub async fn process_utility_hook( pstmt.utilityStmt as *mut pg_sys::ExplainStmt, dest.as_ptr(), )?, - pg_sys::NodeTag::T_ViewStmt => view_query( - query_string, - pstmt.utilityStmt as *mut pg_sys::ViewStmt, - pstmt.stmt_location, - pstmt.stmt_len, - )?, + pg_sys::NodeTag::T_ViewStmt => { + let utility_stmt = unsafe { + pg_sys::copyObjectImpl(pstmt.utilityStmt as *const std::ffi::c_void) + as *mut pg_sys::Node + }; + view_query( + query_string, + utility_stmt as *mut pg_sys::ViewStmt, + pstmt.stmt_location, + pstmt.stmt_len, + )? + } _ => bail!("unexpected statement type in utility hook"), }; @@ -145,7 +147,7 @@ pub async fn process_utility_hook( Ok(()) } -fn is_support_utility(stmt_type: NodeTag) -> bool { +fn is_support_utility(stmt_type: pg_sys::NodeTag) -> bool { stmt_type == pg_sys::NodeTag::T_ExplainStmt || stmt_type == pg_sys::NodeTag::T_ViewStmt || stmt_type == pg_sys::NodeTag::T_PrepareStmt diff --git a/src/hooks/utility/view.rs b/src/hooks/utility/view.rs index 5037b5da..3138faed 100644 --- a/src/hooks/utility/view.rs +++ b/src/hooks/utility/view.rs @@ -19,7 +19,10 @@ use std::ptr::null_mut; use anyhow::Result; -use pgrx::{pg_sys, warning}; +use pgrx::{ + pg_sys::{self}, + warning, +}; use crate::{duckdb::connection::execute, hooks::query::is_duckdb_query}; @@ -32,18 +35,16 @@ pub fn view_query( stmt_len: i32, ) -> Result { // Perform parsing and analysis to get the Query - let query_list = unsafe { - let mut raw_stmt = pg_sys::RawStmt { - type_: pg_sys::NodeTag::T_RawStmt, - stmt: (*stmt).query, - stmt_location, - stmt_len, - }; + let rewritten_queries = unsafe { + let mut raw_stmt = pgrx::PgBox::::alloc_node(pg_sys::NodeTag::T_RawStmt); + raw_stmt.stmt = (*stmt).query; + raw_stmt.stmt_location = stmt_location; + raw_stmt.stmt_len = stmt_len; #[cfg(any(feature = "pg15", feature = "pg16", feature = "pg17"))] { pg_sys::pg_analyze_and_rewrite_fixedparams( - &mut raw_stmt, + raw_stmt.as_ptr(), query_string.as_ptr(), null_mut(), 0, @@ -54,7 +55,7 @@ pub fn view_query( #[cfg(any(feature = "pg13", feature = "pg14"))] { pg_sys::pg_analyze_and_rewrite( - &mut raw_stmt, + raw_stmt.as_ptr(), query_string.as_ptr(), null_mut(), 0, @@ -65,7 +66,7 @@ pub fn view_query( let plan_list = unsafe { pg_sys::pg_plan_queries( - query_list, + rewritten_queries, query_string.as_ptr(), pg_sys::CURSOR_OPT_PARALLEL_OK as i32, null_mut(), diff --git a/tests/tests/scan.rs b/tests/tests/scan.rs index 2835fdf6..8e3453fa 100644 --- a/tests/tests/scan.rs +++ b/tests/tests/scan.rs @@ -672,11 +672,12 @@ async fn test_view_foreign_table(mut conn: PgConnection, tempdir: TempDir) -> Re // cannot fully pushdown to the DuckDB "CREATE TABLE t1 (a int);".execute(&mut conn); "INSERT INTO t1 VALUES (1);".execute(&mut conn); + r#" CREATE VIEW primitive_join_view AS SELECT * FROM primitive - JOIN t1 ON t1.a = primitive.int32_col + JOIN t1 ON t1.a = primitive.int32_col; "# .execute(&mut conn); From 07c46b367bcbfbcdf6e3c05888070592fe35af81 Mon Sep 17 00:00:00 2001 From: Weijun-H Date: Sun, 27 Oct 2024 22:32:27 +0800 Subject: [PATCH 27/30] chore --- src/hooks/utility/view.rs | 19 +++++-------------- 1 file changed, 5 insertions(+), 14 deletions(-) diff --git a/src/hooks/utility/view.rs b/src/hooks/utility/view.rs index 3138faed..3f6659d9 100644 --- a/src/hooks/utility/view.rs +++ b/src/hooks/utility/view.rs @@ -64,23 +64,14 @@ pub fn view_query( } }; - let plan_list = unsafe { - pg_sys::pg_plan_queries( - rewritten_queries, - query_string.as_ptr(), - pg_sys::CURSOR_OPT_PARALLEL_OK as i32, - null_mut(), - ) - }; - unsafe { - for i in 0..(*plan_list).length { - let planned_stmt: *mut pg_sys::PlannedStmt = - (*(*plan_list).elements.offset(i as isize)).ptr_value as *mut pg_sys::PlannedStmt; + for i in 0..(*rewritten_queries).length { + let query: *mut pg_sys::Query = + (*(*rewritten_queries).elements.offset(i as isize)).ptr_value as *mut pg_sys::Query; - let query_relations = get_query_relations((*planned_stmt).rtable); + let query_relations = get_query_relations((*query).rtable); - if (*planned_stmt).commandType != pg_sys::CmdType::CMD_SELECT + if (*query).commandType != pg_sys::CmdType::CMD_SELECT || !is_duckdb_query(&query_relations) { fallback_warning!("Some relations are not in DuckDB"); From 797cd8b4e71dfae9c8c280ea1ee8be511a549f43 Mon Sep 17 00:00:00 2001 From: Weijun-H Date: Sun, 27 Oct 2024 22:42:34 +0800 Subject: [PATCH 28/30] chore: Add nested test --- tests/tests/scan.rs | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/tests/tests/scan.rs b/tests/tests/scan.rs index 8e3453fa..decc9934 100644 --- a/tests/tests/scan.rs +++ b/tests/tests/scan.rs @@ -669,6 +669,11 @@ async fn test_view_foreign_table(mut conn: PgConnection, tempdir: TempDir) -> Re let res: (bool,) = "SELECT boolean_col FROM primitive_view".fetch_one(&mut conn); assert!(res.0); + // nested view, fully pushdown to the DuckDB + "CREATE VIEW nested_primitive_view AS SELECT * FROM primitive_view".execute(&mut conn); + let res: (bool,) = "SELECT boolean_col FROM nested_primitive_view".fetch_one(&mut conn); + assert!(res.0); + // cannot fully pushdown to the DuckDB "CREATE TABLE t1 (a int);".execute(&mut conn); "INSERT INTO t1 VALUES (1);".execute(&mut conn); From f275502e6f7d20a287a65c4c718f625392f4e004 Mon Sep 17 00:00:00 2001 From: Weijun-H Date: Sun, 27 Oct 2024 23:12:13 +0800 Subject: [PATCH 29/30] chore: Update pg_analytics --- .github/workflows/test-pg_analytics.yml | 23 +++++++++++++++-------- 1 file changed, 15 insertions(+), 8 deletions(-) diff --git a/.github/workflows/test-pg_analytics.yml b/.github/workflows/test-pg_analytics.yml index b2ec88d7..3e06a726 100644 --- a/.github/workflows/test-pg_analytics.yml +++ b/.github/workflows/test-pg_analytics.yml @@ -120,6 +120,17 @@ jobs: working-directory: /home/runner/.pgrx/data-${{ matrix.pg_version }}/ run: sed -i "s/^#shared_preload_libraries = .*/shared_preload_libraries = 'pg_analytics'/" postgresql.conf + - name: Compile & install pg_analytics extension + if: steps.check_skip.outputs.skip_remaining_steps != 'true' + run: cargo pgrx install --pg-config="/usr/lib/postgresql/${{ matrix.pg_version }}/bin/pg_config" + + - name: Start Postgres via cargo-pgrx + if: steps.check_skip.outputs.skip_remaining_steps != 'true' + run: | + RUST_BACKTRACE=1 cargo pgrx start pg${{ matrix.pg_version }} + # Necessary for the ephemeral Postgres test to have proper permissions + sudo chown -R $(whoami) /var/run/postgresql/ + # The SHA hash here must exactly match the Image::Tag that is referenced in the # testcontainers modules Rust crate for localstack. - name: Pull localstack image @@ -140,16 +151,12 @@ jobs: echo -e "\n# Enable code coverage on Linux only, for CI builds\n[target.'cfg(target_os=\"linux\")']\nrustflags = [\"-Cinstrument-coverage\"]" >> .cargo/config.toml mkdir -p target/coverage target/coverage-report - echo "" - echo "Building pg_analytics..." - cargo pgrx install --pg-config="/usr/lib/postgresql/${{ matrix.pg_version }}/bin/pg_config" --release - - echo "" - echo "Starting Postgres..." - cargo pgrx start pg${{ matrix.pg_version }} - echo "" echo "Running Rust tests..." export DATABASE_URL=postgresql://localhost:288${{ matrix.pg_version }}/postgres export RUST_BACKTRACE=1 cargo test --package tests --features "pg${{ matrix.pg_version }}" --no-default-features + + - name: Print the Postgres Logs + if: steps.check_skip.outputs.skip_remaining_steps != 'true' && always() + run: cat ~/.pgrx/${{ matrix.pg_version}}.log From d51c7fab410a2493b96dad1c5f615bd9be9267bf Mon Sep 17 00:00:00 2001 From: Weijun-H Date: Sun, 27 Oct 2024 23:32:48 +0800 Subject: [PATCH 30/30] chore --- src/hooks/utility/view.rs | 19 ++++++++++++++----- 1 file changed, 14 insertions(+), 5 deletions(-) diff --git a/src/hooks/utility/view.rs b/src/hooks/utility/view.rs index 3f6659d9..3138faed 100644 --- a/src/hooks/utility/view.rs +++ b/src/hooks/utility/view.rs @@ -64,14 +64,23 @@ pub fn view_query( } }; + let plan_list = unsafe { + pg_sys::pg_plan_queries( + rewritten_queries, + query_string.as_ptr(), + pg_sys::CURSOR_OPT_PARALLEL_OK as i32, + null_mut(), + ) + }; + unsafe { - for i in 0..(*rewritten_queries).length { - let query: *mut pg_sys::Query = - (*(*rewritten_queries).elements.offset(i as isize)).ptr_value as *mut pg_sys::Query; + for i in 0..(*plan_list).length { + let planned_stmt: *mut pg_sys::PlannedStmt = + (*(*plan_list).elements.offset(i as isize)).ptr_value as *mut pg_sys::PlannedStmt; - let query_relations = get_query_relations((*query).rtable); + let query_relations = get_query_relations((*planned_stmt).rtable); - if (*query).commandType != pg_sys::CmdType::CMD_SELECT + if (*planned_stmt).commandType != pg_sys::CmdType::CMD_SELECT || !is_duckdb_query(&query_relations) { fallback_warning!("Some relations are not in DuckDB");