From aca82f5926b2e955adada1cc50480b3e89bfc994 Mon Sep 17 00:00:00 2001
From: Nick Redfearn <97466325+nick-redfearn@users.noreply.github.com>
Date: Fri, 26 Jul 2024 12:52:27 +0100
Subject: [PATCH] fix truncate parsing to capture multiple tables (#2048)
* fix truncate parsing to capture multiple tables
* followup fixes including adding a lookahead
* replacng tabs with spaces
* increasing lookahead
* fixing after getting maven working locally
---
.../statement/truncate/Truncate.java | 26 ++++-
.../util/deparser/StatementDeParser.java | 12 +-
.../net/sf/jsqlparser/parser/JSqlParserCC.jjt | 23 +++-
.../truncate/TruncateMultipleTablesTest.java | 110 ++++++++++++++++++
.../statement/truncate/TruncateTest.java | 29 +++--
5 files changed, 181 insertions(+), 19 deletions(-)
create mode 100644 src/test/java/net/sf/jsqlparser/statement/truncate/TruncateMultipleTablesTest.java
diff --git a/src/main/java/net/sf/jsqlparser/statement/truncate/Truncate.java b/src/main/java/net/sf/jsqlparser/statement/truncate/Truncate.java
index 4388f3ddd..58442d38e 100644
--- a/src/main/java/net/sf/jsqlparser/statement/truncate/Truncate.java
+++ b/src/main/java/net/sf/jsqlparser/statement/truncate/Truncate.java
@@ -9,6 +9,9 @@
*/
package net.sf.jsqlparser.statement.truncate;
+import static java.util.stream.Collectors.joining;
+
+import java.util.List;
import net.sf.jsqlparser.schema.Table;
import net.sf.jsqlparser.statement.Statement;
import net.sf.jsqlparser.statement.StatementVisitor;
@@ -19,6 +22,7 @@ public class Truncate implements Statement {
boolean tableToken; // to support TRUNCATE without TABLE
boolean only; // to support TRUNCATE with ONLY
private Table table;
+ private List
tables;
@Override
public T accept(StatementVisitor statementVisitor, S context) {
@@ -29,10 +33,18 @@ public Table getTable() {
return table;
}
+ public List getTables() {
+ return tables;
+ }
+
public void setTable(Table table) {
this.table = table;
}
+ public void setTables(List tables) {
+ this.tables = tables;
+ }
+
public boolean getCascade() {
return cascade;
}
@@ -52,8 +64,13 @@ public String toString() {
sb.append(" ONLY");
}
sb.append(" ");
- sb.append(table);
-
+ if (tables != null && !tables.isEmpty()) {
+ sb.append(tables.stream()
+ .map(Table::toString)
+ .collect(joining(", ")));
+ } else {
+ sb.append(table);
+ }
if (cascade) {
sb.append(" CASCADE");
}
@@ -86,6 +103,11 @@ public Truncate withTable(Table table) {
return this;
}
+ public Truncate withTables(List tables) {
+ this.setTables(tables);
+ return this;
+ }
+
public Truncate withCascade(boolean cascade) {
this.setCascade(cascade);
return this;
diff --git a/src/main/java/net/sf/jsqlparser/util/deparser/StatementDeParser.java b/src/main/java/net/sf/jsqlparser/util/deparser/StatementDeParser.java
index 30cf27f79..47ecd643d 100644
--- a/src/main/java/net/sf/jsqlparser/util/deparser/StatementDeParser.java
+++ b/src/main/java/net/sf/jsqlparser/util/deparser/StatementDeParser.java
@@ -9,9 +9,12 @@
*/
package net.sf.jsqlparser.util.deparser;
+import static java.util.stream.Collectors.joining;
+
import java.lang.reflect.InvocationTargetException;
import java.util.stream.Collectors;
+import net.sf.jsqlparser.schema.Table;
import net.sf.jsqlparser.statement.Block;
import net.sf.jsqlparser.statement.Commit;
import net.sf.jsqlparser.statement.CreateFunctionalStatement;
@@ -180,8 +183,13 @@ public StringBuilder visit(Truncate truncate, S context) {
buffer.append(" ONLY");
}
buffer.append(" ");
- buffer.append(truncate.getTable());
-
+ if (truncate.getTables() != null && !truncate.getTables().isEmpty()) {
+ buffer.append(truncate.getTables().stream()
+ .map(Table::toString)
+ .collect(joining(", ")));
+ } else {
+ buffer.append(truncate.getTable());
+ }
if (truncate.getCascade()) {
buffer.append(" CASCADE");
}
diff --git a/src/main/jjtree/net/sf/jsqlparser/parser/JSqlParserCC.jjt b/src/main/jjtree/net/sf/jsqlparser/parser/JSqlParserCC.jjt
index a9f977dd9..d0e868446 100644
--- a/src/main/jjtree/net/sf/jsqlparser/parser/JSqlParserCC.jjt
+++ b/src/main/jjtree/net/sf/jsqlparser/parser/JSqlParserCC.jjt
@@ -6617,6 +6617,9 @@ Truncate Truncate():
{
Truncate truncate = new Truncate();
Table table;
+ List tables = new ArrayList();
+ boolean only = false;
+ boolean cascade = false;
}
{
/**
@@ -6627,14 +6630,24 @@ Truncate Truncate():
* [ RESTART IDENTITY | CONTINUE IDENTITY ] [ CASCADE | RESTRICT ]
*
*/
- [LOOKAHEAD(2) {truncate.setTableToken(true);}] [ {truncate.setOnly(true);}]
- table=Table() { truncate.setTable(table); truncate.setCascade(false); } [ {truncate.setCascade(true);} ]
- {
- return truncate;
+
+ [LOOKAHEAD(2) {truncate.setTableToken(true);}]
+ [ { only = true; }]
+ table=Table() { tables.add(table); } (LOOKAHEAD(2) "," table=Table() { tables.add(table); } )*
+ [ { cascade = true; }]
+ {
+ if (only && tables.size() > 1 ) {
+ throw new ParseException("Cannot TRUNCATE ONLY with multiple tables");
+ } else {
+ return truncate
+ .withTables(tables)
+ .withTable(table)
+ .withOnly(only)
+ .withCascade(cascade);
+ }
}
}
-
AlterExpression.ColumnDataType AlterExpressionColumnDataType():
{
String columnName = null;
diff --git a/src/test/java/net/sf/jsqlparser/statement/truncate/TruncateMultipleTablesTest.java b/src/test/java/net/sf/jsqlparser/statement/truncate/TruncateMultipleTablesTest.java
new file mode 100644
index 000000000..18894c8ca
--- /dev/null
+++ b/src/test/java/net/sf/jsqlparser/statement/truncate/TruncateMultipleTablesTest.java
@@ -0,0 +1,110 @@
+/*-
+ * #%L
+ * JSQLParser library
+ * %%
+ * Copyright (C) 2004 - 2019 JSQLParser
+ * %%
+ * Dual licensed under GNU LGPL 2.1 or Apache License 2.0
+ * #L%
+ */
+package net.sf.jsqlparser.statement.truncate;
+
+import static net.sf.jsqlparser.test.TestUtils.assertDeparse;
+import static net.sf.jsqlparser.test.TestUtils.assertSqlCanBeParsedAndDeparsed;
+import static org.junit.jupiter.api.Assertions.assertEquals;
+import static org.junit.jupiter.api.Assertions.assertNull;
+import static org.junit.jupiter.api.Assertions.assertThrows;
+import static org.junit.jupiter.api.Assertions.assertTrue;
+
+import java.io.StringReader;
+import java.util.List;
+import net.sf.jsqlparser.JSQLParserException;
+import net.sf.jsqlparser.parser.CCJSqlParserManager;
+import net.sf.jsqlparser.schema.Table;
+import org.junit.jupiter.api.Test;
+
+public class TruncateMultipleTablesTest {
+
+ private CCJSqlParserManager parserManager = new CCJSqlParserManager();
+
+ @Test
+ public void testTruncate2Tables() throws Exception {
+ String statement = "TRUncATE TABLE myschema.mytab, myschema2.mytab2";
+ Truncate truncate = (Truncate) parserManager.parse(new StringReader(statement));
+ assertEquals("myschema2", truncate.getTable().getSchemaName());
+ assertEquals("myschema2.mytab2", truncate.getTable().getFullyQualifiedName());
+ assertEquals(statement.toUpperCase(), truncate.toString().toUpperCase());
+ assertEquals("myschema.mytab", truncate.getTables().get(0).getFullyQualifiedName());
+ assertEquals("myschema2.mytab2", truncate.getTables().get(1).getFullyQualifiedName());
+
+ statement = "TRUncATE TABLE mytab, my2ndtab";
+ String toStringStatement = "TRUncATE TABLE mytab, my2ndtab";
+ truncate = (Truncate) parserManager.parse(new StringReader(statement));
+ assertEquals("my2ndtab", truncate.getTable().getName());
+ assertEquals(toStringStatement.toUpperCase(), truncate.toString().toUpperCase());
+ assertEquals("mytab", truncate.getTables().get(0).getFullyQualifiedName());
+ assertEquals("my2ndtab", truncate.getTables().get(1).getFullyQualifiedName());
+
+ statement = "TRUNCATE TABLE mytab, my2ndtab CASCADE";
+ truncate = (Truncate) parserManager.parse(new StringReader(statement));
+ assertNull(truncate.getTables().get(0).getSchemaName());
+ assertEquals("mytab", truncate.getTables().get(0).getFullyQualifiedName());
+ assertEquals("my2ndtab", truncate.getTables().get(1).getFullyQualifiedName());
+ assertTrue(truncate.getCascade());
+ assertEquals(statement, truncate.toString());
+ }
+
+ @Test
+ public void testTruncatePostgresqlWithoutTableNames() throws Exception {
+ String statement = "TRUncATE myschema.mytab, myschema2.mytab2";
+ Truncate truncate = (Truncate) parserManager.parse(new StringReader(statement));
+ assertEquals("myschema2", truncate.getTable().getSchemaName());
+ assertEquals("myschema2.mytab2", truncate.getTable().getFullyQualifiedName());
+ assertEquals(statement.toUpperCase(), truncate.toString().toUpperCase());
+ assertEquals("myschema.mytab", truncate.getTables().get(0).getFullyQualifiedName());
+ assertEquals("myschema2.mytab2", truncate.getTables().get(1).getFullyQualifiedName());
+
+ statement = "TRUncATE mytab, my2ndtab";
+ String toStringStatement = "TRUncATE mytab, my2ndtab";
+ truncate = (Truncate) parserManager.parse(new StringReader(statement));
+ assertEquals("my2ndtab", truncate.getTable().getName());
+ assertEquals(toStringStatement.toUpperCase(), truncate.toString().toUpperCase());
+ assertEquals("mytab", truncate.getTables().get(0).getFullyQualifiedName());
+ assertEquals("my2ndtab", truncate.getTables().get(1).getFullyQualifiedName());
+
+ statement = "TRUNCATE mytab, my2ndtab CASCADE";
+ truncate = (Truncate) parserManager.parse(new StringReader(statement));
+ assertNull(truncate.getTables().get(0).getSchemaName());
+ assertEquals("mytab", truncate.getTables().get(0).getFullyQualifiedName());
+ assertEquals("my2ndtab", truncate.getTables().get(1).getFullyQualifiedName());
+ assertTrue(truncate.getCascade());
+ assertEquals(statement, truncate.toString());
+ }
+
+ @Test
+ public void testTruncateDeparse() throws JSQLParserException {
+ String statement = "TRUNCATE TABLE foo, bar";
+ assertSqlCanBeParsedAndDeparsed(statement);
+ assertDeparse(new Truncate()
+ .withTables(List.of(new Table("foo"), new Table("bar")))
+ .withTableToken(true), statement);
+ }
+
+ @Test
+ public void testTruncateCascadeDeparse() throws JSQLParserException {
+ String statement = "TRUNCATE TABLE foo, bar CASCADE";
+ assertSqlCanBeParsedAndDeparsed(statement);
+ assertDeparse(new Truncate()
+ .withTables(List.of(new Table("foo"), new Table("bar")))
+ .withTableToken(true)
+ .withCascade(true), statement);
+ }
+
+ @Test
+ public void testTruncateDoesNotAllowOnlyWithMultipleTables() {
+ String statement = "TRUNCATE TABLE ONLY foo, bar";
+ assertThrows(JSQLParserException.class,
+ () -> parserManager.parse(new StringReader(statement)));
+ }
+
+}
diff --git a/src/test/java/net/sf/jsqlparser/statement/truncate/TruncateTest.java b/src/test/java/net/sf/jsqlparser/statement/truncate/TruncateTest.java
index 036aa64ad..5d8db40f6 100644
--- a/src/test/java/net/sf/jsqlparser/statement/truncate/TruncateTest.java
+++ b/src/test/java/net/sf/jsqlparser/statement/truncate/TruncateTest.java
@@ -71,8 +71,8 @@ public void testTruncateDeparse() throws JSQLParserException {
String statement = "TRUNCATE TABLE foo";
assertSqlCanBeParsedAndDeparsed(statement);
assertDeparse(new Truncate()
- .withTable(new Table("foo"))
- .withTableToken(true), statement);
+ .withTable(new Table("foo"))
+ .withTableToken(true), statement);
}
@Test
@@ -80,19 +80,28 @@ public void testTruncateCascadeDeparse() throws JSQLParserException {
String statement = "TRUNCATE TABLE foo CASCADE";
assertSqlCanBeParsedAndDeparsed(statement);
assertDeparse(new Truncate()
- .withTable(new Table("foo"))
- .withTableToken(true)
- .withCascade(true), statement);
+ .withTable(new Table("foo"))
+ .withTableToken(true)
+ .withCascade(true), statement);
}
@Test
public void testTruncateOnlyDeparse() throws JSQLParserException {
- String statement = "TRUNCATE TABLE ONLY foo CASCADE";
+ String statement = "TRUNCATE TABLE ONLY foo";
assertSqlCanBeParsedAndDeparsed(statement);
assertDeparse(new Truncate()
- .withTable(new Table("foo"))
- .withCascade(true)
- .withTableToken(true)
- .withOnly(true), statement);
+ .withTable(new Table("foo"))
+ .withTableToken(true)
+ .withOnly(true), statement);
+ }
+
+ @Test
+ public void testTruncateOnlyAndCascadeDeparse() throws JSQLParserException {
+ String statement = "TRUNCATE ONLY foo CASCADE";
+ assertSqlCanBeParsedAndDeparsed(statement);
+ assertDeparse(new Truncate()
+ .withTable(new Table("foo"))
+ .withCascade(true)
+ .withOnly(true), statement);
}
}