From ed8d6d1ec276e5793cb168ee73d98dbb9cf37ee5 Mon Sep 17 00:00:00 2001 From: Yanjie Xu Date: Fri, 27 Oct 2023 20:42:17 -0700 Subject: [PATCH] Support TSQL PIVOT operator (#165) Add support for PIVOT in babelfish Task: BABEL-284 Signed-off-by: Yanjie Xu --- src/backend/parser/analyze.c | 10 ++++++++-- src/include/nodes/parsenodes.h | 8 ++++++++ src/include/parser/analyze.h | 4 ++++ 3 files changed, 20 insertions(+), 2 deletions(-) diff --git a/src/backend/parser/analyze.c b/src/backend/parser/analyze.c index 1e9202915f5..84094f64007 100644 --- a/src/backend/parser/analyze.c +++ b/src/backend/parser/analyze.c @@ -88,6 +88,9 @@ pre_transform_setop_tree_hook_type pre_transform_setop_tree_hook = NULL; /* Hook to reset a query's targetlist after modification in pre_transfrom_sort_clause */ pre_transform_setop_sort_clause_hook_type pre_transform_setop_sort_clause_hook = NULL; +/* Hooks for transform TSQL pivot clause in select stmt */ +transform_pivot_clause_hook_type transform_pivot_clause_hook = NULL; + static Query *transformOptionalSelectInto(ParseState *pstate, Node *parseTree); static Query *transformDeleteStmt(ParseState *pstate, DeleteStmt *stmt); static Query *transformInsertStmt(ParseState *pstate, InsertStmt *stmt); @@ -120,7 +123,6 @@ static void transformLockingClause(ParseState *pstate, Query *qry, static bool test_raw_expression_coverage(Node *node, void *context); #endif - /* * parse_analyze_fixedparams * Analyze a raw parse tree and transform it to Query form. @@ -1365,7 +1367,6 @@ count_rowexpr_columns(ParseState *pstate, Node *expr) return -1; } - /* * transformSelectStmt - * transforms a Select Statement @@ -1404,6 +1405,11 @@ transformSelectStmt(ParseState *pstate, SelectStmt *stmt) /* make WINDOW info available for window functions, too */ pstate->p_windowdefs = stmt->windowClause; + if (stmt->isPivot && transform_pivot_clause_hook) + { + (*transform_pivot_clause_hook)(pstate, stmt); + } + /* process the FROM clause */ transformFromClause(pstate, stmt->fromClause); diff --git a/src/include/nodes/parsenodes.h b/src/include/nodes/parsenodes.h index 0683ddfe828..d2ede9d3313 100644 --- a/src/include/nodes/parsenodes.h +++ b/src/include/nodes/parsenodes.h @@ -1753,6 +1753,14 @@ typedef struct SelectStmt struct SelectStmt *larg; /* left child */ struct SelectStmt *rarg; /* right child */ /* Eventually add fields for CORRESPONDING spec here */ + + /* These fields are used only in SelectStmt with PIVOT keyword */ + bool isPivot; + struct SelectStmt *srcSql; + struct SelectStmt *catSql; + List *value_col_strlist; + char *pivotCol; + Node *aggFunc; } SelectStmt; diff --git a/src/include/parser/analyze.h b/src/include/parser/analyze.h index b7fb76eaaaa..c165c929ef0 100644 --- a/src/include/parser/analyze.h +++ b/src/include/parser/analyze.h @@ -64,6 +64,10 @@ extern PGDLLIMPORT pre_transform_setop_tree_hook_type pre_transform_setop_tree_h typedef void (*pre_transform_setop_sort_clause_hook_type) (ParseState *pstate, Query *qry, List *sortClause, Query *leftmostQuery); extern PGDLLIMPORT pre_transform_setop_sort_clause_hook_type pre_transform_setop_sort_clause_hook; +/* Hook for transform pivot clause in tsql select stmt */ +typedef void (*transform_pivot_clause_hook_type)(ParseState *pstate, SelectStmt *stmt); +extern PGDLLIMPORT transform_pivot_clause_hook_type transform_pivot_clause_hook; + extern Query *parse_analyze_fixedparams(RawStmt *parseTree, const char *sourceText, const Oid *paramTypes, int numParams, QueryEnvironment *queryEnv); extern Query *parse_analyze(RawStmt *parseTree, const char *sourceText,