From 3a68c953d25577f962272a59ba9edd180e31b05a Mon Sep 17 00:00:00 2001 From: lingo-xp <36907211+lingo-xp@users.noreply.github.com> Date: Mon, 16 Dec 2024 22:35:02 +0800 Subject: [PATCH] Rewrite merge insert row. (#6290) * improved sql parser bigquery support * Rewrite merge insert into row. --------- Co-authored-by: wenshao --- .../sql/ast/statement/SQLMergeStatement.java | 9 +++++++ .../odps/visitor/OdpsOutputVisitor.java | 5 ++++ .../druid/sql/parser/SQLStatementParser.java | 5 ++-- .../sql/visitor/SQLASTOutputVisitor.java | 27 ++++++++++++------- .../test/resources/bvt/parser/bigquery/0.txt | 6 +++++ core/src/test/resources/bvt/parser/odps/0.txt | 10 ++++++- 6 files changed, 49 insertions(+), 13 deletions(-) diff --git a/core/src/main/java/com/alibaba/druid/sql/ast/statement/SQLMergeStatement.java b/core/src/main/java/com/alibaba/druid/sql/ast/statement/SQLMergeStatement.java index 3e8a873277..a9e3256096 100644 --- a/core/src/main/java/com/alibaba/druid/sql/ast/statement/SQLMergeStatement.java +++ b/core/src/main/java/com/alibaba/druid/sql/ast/statement/SQLMergeStatement.java @@ -156,6 +156,7 @@ public boolean replace(SQLExpr expr, SQLExpr target) { } public static class WhenInsert extends When { + private boolean insertRow; private List columns = new ArrayList(); private List values = new ArrayList(); @@ -195,6 +196,14 @@ public void setColumns(List columns) { this.columns = columns; } + public boolean isInsertRow() { + return insertRow; + } + + public void setInsertRow(boolean insertRow) { + this.insertRow = insertRow; + } + public List getValues() { return values; } diff --git a/core/src/main/java/com/alibaba/druid/sql/dialect/odps/visitor/OdpsOutputVisitor.java b/core/src/main/java/com/alibaba/druid/sql/dialect/odps/visitor/OdpsOutputVisitor.java index 967664000a..195a43533d 100644 --- a/core/src/main/java/com/alibaba/druid/sql/dialect/odps/visitor/OdpsOutputVisitor.java +++ b/core/src/main/java/com/alibaba/druid/sql/dialect/odps/visitor/OdpsOutputVisitor.java @@ -1120,4 +1120,9 @@ protected void printMethodParameters(SQLMethodInvokeExpr x) { } super.printMethodParameters(x); } + + @Override + public void printMergeInsertRow() { + print(" *"); + } } diff --git a/core/src/main/java/com/alibaba/druid/sql/parser/SQLStatementParser.java b/core/src/main/java/com/alibaba/druid/sql/parser/SQLStatementParser.java index 5983e0a966..689d71afa8 100644 --- a/core/src/main/java/com/alibaba/druid/sql/parser/SQLStatementParser.java +++ b/core/src/main/java/com/alibaba/druid/sql/parser/SQLStatementParser.java @@ -5672,13 +5672,14 @@ protected boolean parseMergeWhen(SQLMergeStatement stmt) { exprParser.exprList(insertClause.getColumns(), insertClause); accept(Token.RPAREN); } - if (lexer.nextIfIdentifier("ROW")) { - insertClause.getValues().add(new SQLIdentifierExpr("ROW")); + if (lexer.nextIfIdentifier("ROW") || lexer.nextIfIdentifier("*") || lexer.nextIf(Token.STAR)) { + insertClause.setInsertRow(true); } else { accept(Token.VALUES); accept(Token.LPAREN); exprParser.exprList(insertClause.getValues(), insertClause); accept(Token.RPAREN); + insertClause.setInsertRow(false); } if (lexer.token == Token.WHERE) { diff --git a/core/src/main/java/com/alibaba/druid/sql/visitor/SQLASTOutputVisitor.java b/core/src/main/java/com/alibaba/druid/sql/visitor/SQLASTOutputVisitor.java index fff1ee9b9e..2eed446d09 100644 --- a/core/src/main/java/com/alibaba/druid/sql/visitor/SQLASTOutputVisitor.java +++ b/core/src/main/java/com/alibaba/druid/sql/visitor/SQLASTOutputVisitor.java @@ -8593,17 +8593,21 @@ public boolean visit(WhenInsert x) { by.accept(this); } print0(ucase ? " THEN INSERT" : " then insert"); - if (x.getColumns().size() > 0) { - printAndAccept(" (", ")", x.getColumns(), ", ", 5); + if (x.isInsertRow()) { + printMergeInsertRow(); + } else { + if (x.getColumns().size() > 0) { + printAndAccept(" (", ")", x.getColumns(), ", ", 5); + } + println(); + printAndAccept( + ucase ? "VALUES (" : "values (", + ")", + x.getValues(), + ", ", + 5 + ); } - println(); - printAndAccept( - ucase ? "VALUES (" : "values (", - ")", - x.getValues(), - ", ", - 5 - ); if (x.getWhere() != null) { this.indentCount++; println(); @@ -8615,6 +8619,9 @@ public boolean visit(WhenInsert x) { return false; } + public void printMergeInsertRow() { + print(ucase ? " ROW" : "row"); + } @Override public boolean visit(SQLErrorLoggingClause x) { print0(ucase ? "LOG ERRORS " : "log errors "); diff --git a/core/src/test/resources/bvt/parser/bigquery/0.txt b/core/src/test/resources/bvt/parser/bigquery/0.txt index f0b67de991..88af927d3c 100644 --- a/core/src/test/resources/bvt/parser/bigquery/0.txt +++ b/core/src/test/resources/bvt/parser/bigquery/0.txt @@ -1,3 +1,9 @@ +merge into a as target using b as source on a.id = b.id when not matched then insert row +-------------------- +MERGE INTO a target +USING b source ON a.id = b.id +WHEN NOT MATCHED THEN INSERT ROW +------------------------------------------------------------------------------------------------------------------------ SELECT * EXCEPT (predicted_label), predicted_label AS predicted_label1 FROM diff --git a/core/src/test/resources/bvt/parser/odps/0.txt b/core/src/test/resources/bvt/parser/odps/0.txt index 36f2f5b99e..93341b4f52 100644 --- a/core/src/test/resources/bvt/parser/odps/0.txt +++ b/core/src/test/resources/bvt/parser/odps/0.txt @@ -6,4 +6,12 @@ FROM a select named_struct(a.b, b, a.c, c, a.d, d) from a -------------------- SELECT named_struct(a.b, b, a.c, c, a.d, d) -FROM a \ No newline at end of file +FROM a +------------------------------------------------------------------------------------------------------------------------ +MERGE INTO a target +USING b source ON a.id = b.id +WHEN NOT MATCHED THEN INSERT * +-------------------- +MERGE INTO a target +USING b source ON a.id = b.id +WHEN NOT MATCHED THEN INSERT * \ No newline at end of file