diff --git a/api/src/main/java/com/datastrato/gravitino/rel/TableChange.java b/api/src/main/java/com/datastrato/gravitino/rel/TableChange.java
index 69023caa24e..01d6697dc7b 100644
--- a/api/src/main/java/com/datastrato/gravitino/rel/TableChange.java
+++ b/api/src/main/java/com/datastrato/gravitino/rel/TableChange.java
@@ -20,6 +20,7 @@
package com.datastrato.gravitino.rel;
+import com.datastrato.gravitino.rel.indexes.Index;
import com.datastrato.gravitino.rel.types.Type;
import java.util.Arrays;
import java.util.Objects;
@@ -284,6 +285,30 @@ static TableChange updateColumnNullability(String[] fieldName, boolean nullable)
return new UpdateColumnNullability(fieldName, nullable);
}
+ /**
+ * Create a TableChange for adding an index.
+ *
+ * @param type The type of the index.
+ * @param name The name of the index.
+ * @param fieldNames The field names of the index.
+ * @return A TableChange for the add index.
+ */
+ static TableChange addIndex(Index.IndexType type, String name, String[][] fieldNames) {
+ return new AddIndex(type, name, fieldNames);
+ }
+
+ /**
+ * Create a TableChange for deleting an index.
+ *
+ * @param name The name of the index to be dropped.
+ * @param ifExists If true, silence the error if column does not exist during drop. Otherwise, an
+ * {@link IllegalArgumentException} will be thrown.
+ * @return A TableChange for the delete index.
+ */
+ static TableChange deleteIndex(String name, Boolean ifExists) {
+ return new DeleteIndex(name, ifExists);
+ }
+
/** A TableChange to rename a table. */
final class RenameTable implements TableChange {
private final String newName;
@@ -529,6 +554,103 @@ public String toString() {
}
}
+ /**
+ * A TableChange to add an index. Add an index key based on the type and field name passed in as
+ * well as the name.
+ */
+ final class AddIndex implements TableChange {
+
+ private final Index.IndexType type;
+ private final String name;
+
+ private final String[][] fieldNames;
+
+ /**
+ * @param type The type of the index.
+ * @param name The name of the index.
+ * @param fieldNames The field names of the index.
+ */
+ public AddIndex(Index.IndexType type, String name, String[][] fieldNames) {
+ this.type = type;
+ this.name = name;
+ this.fieldNames = fieldNames;
+ }
+
+ /** @return The type of the index. */
+ public Index.IndexType getType() {
+ return type;
+ }
+
+ /** @return The name of the index. */
+ public String getName() {
+ return name;
+ }
+
+ /** @return The field names of the index. */
+ public String[][] getFieldNames() {
+ return fieldNames;
+ }
+
+ @Override
+ public boolean equals(Object o) {
+ if (this == o) return true;
+ if (o == null || getClass() != o.getClass()) return false;
+ AddIndex addIndex = (AddIndex) o;
+ return type == addIndex.type
+ && Objects.equals(name, addIndex.name)
+ && Arrays.equals(fieldNames, addIndex.fieldNames);
+ }
+
+ @Override
+ public int hashCode() {
+ int result = Objects.hash(type, name);
+ result = 31 * result + Arrays.hashCode(fieldNames);
+ return result;
+ }
+ }
+
+ /**
+ * A TableChange to delete an index.
+ *
+ *
If the index does not exist, the change must result in an {@link IllegalArgumentException}.
+ */
+ final class DeleteIndex implements TableChange {
+ private final String name;
+ private final boolean ifExists;
+
+ /**
+ * @param name name of the index.
+ * @param ifExists If true, silence the error if index does not exist during drop.
+ */
+ public DeleteIndex(String name, boolean ifExists) {
+ this.name = name;
+ this.ifExists = ifExists;
+ }
+
+ /** @return The name of the index to be deleted. */
+ public String getName() {
+ return name;
+ }
+
+ /** @return If true, silence the error if index does not exist during drop. */
+ public boolean isIfExists() {
+ return ifExists;
+ }
+
+ @Override
+ public boolean equals(Object o) {
+ if (this == o) return true;
+ if (o == null || getClass() != o.getClass()) return false;
+ DeleteIndex that = (DeleteIndex) o;
+ return Objects.equals(name, that.name) && Objects.equals(ifExists, that.ifExists);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(name, ifExists);
+ }
+ }
+
/**
* The interface for all column positions. Column positions are used to specify the position of a
* column when adding a new column to a table.
diff --git a/catalogs/catalog-jdbc-mysql/src/main/java/com/datastrato/gravitino/catalog/mysql/operation/MysqlTableOperations.java b/catalogs/catalog-jdbc-mysql/src/main/java/com/datastrato/gravitino/catalog/mysql/operation/MysqlTableOperations.java
index 415061d61f2..674b3ac9630 100644
--- a/catalogs/catalog-jdbc-mysql/src/main/java/com/datastrato/gravitino/catalog/mysql/operation/MysqlTableOperations.java
+++ b/catalogs/catalog-jdbc-mysql/src/main/java/com/datastrato/gravitino/catalog/mysql/operation/MysqlTableOperations.java
@@ -20,6 +20,7 @@
import com.datastrato.gravitino.rel.expressions.transforms.Transform;
import com.datastrato.gravitino.rel.indexes.Index;
import com.datastrato.gravitino.rel.indexes.Indexes;
+import com.google.common.annotations.VisibleForTesting;
import com.google.common.base.Preconditions;
import java.sql.Connection;
import java.sql.PreparedStatement;
@@ -170,17 +171,7 @@ private static void validateIncrementCol(JdbcColumn[] columns, Index[] indexes)
public static void appendIndexesSql(Index[] indexes, StringBuilder sqlBuilder) {
for (Index index : indexes) {
- String fieldStr =
- Arrays.stream(index.fieldNames())
- .map(
- colNames -> {
- if (colNames.length > 1) {
- throw new IllegalArgumentException(
- "Index does not support complex fields in MySQL");
- }
- return BACK_QUOTE + colNames[0] + BACK_QUOTE;
- })
- .collect(Collectors.joining(", "));
+ String fieldStr = getIndexFieldStr(index.fieldNames());
sqlBuilder.append(",\n");
switch (index.type()) {
case PRIMARY_KEY:
@@ -204,6 +195,19 @@ public static void appendIndexesSql(Index[] indexes, StringBuilder sqlBuilder) {
}
}
+ private static String getIndexFieldStr(String[][] fieldNames) {
+ return Arrays.stream(fieldNames)
+ .map(
+ colNames -> {
+ if (colNames.length > 1) {
+ throw new IllegalArgumentException(
+ "Index does not support complex fields in MySQL");
+ }
+ return BACK_QUOTE + colNames[0] + BACK_QUOTE;
+ })
+ .collect(Collectors.joining(", "));
+ }
+
@Override
protected boolean getAutoIncrementInfo(ResultSet resultSet) throws SQLException {
return "YES".equalsIgnoreCase(resultSet.getString("IS_AUTOINCREMENT"));
@@ -319,6 +323,11 @@ protected String generateAlterTableSql(
alterSql.add(
updateColumnNullabilityDefinition(
(TableChange.UpdateColumnNullability) change, lazyLoadTable));
+ } else if (change instanceof TableChange.AddIndex) {
+ alterSql.add(addIndexDefinition((TableChange.AddIndex) change));
+ } else if (change instanceof TableChange.DeleteIndex) {
+ lazyLoadTable = getOrCreateTable(databaseName, tableName, lazyLoadTable);
+ alterSql.add(deleteIndexDefinition(lazyLoadTable, (TableChange.DeleteIndex) change));
} else {
throw new IllegalArgumentException(
"Unsupported table change type: " + change.getClass().getName());
@@ -355,6 +364,18 @@ protected String generateAlterTableSql(
return result;
}
+ @VisibleForTesting
+ static String deleteIndexDefinition(
+ JdbcTable lazyLoadTable, TableChange.DeleteIndex deleteIndex) {
+ if (deleteIndex.isIfExists()) {
+ if (Arrays.stream(lazyLoadTable.index())
+ .anyMatch(index -> index.name().equals(deleteIndex.getName()))) {
+ throw new IllegalArgumentException("Index does not exist");
+ }
+ }
+ return "DROP INDEX " + BACK_QUOTE + deleteIndex.getName() + BACK_QUOTE;
+ }
+
private String updateColumnNullabilityDefinition(
TableChange.UpdateColumnNullability change, JdbcTable table) {
validateUpdateColumnNullable(change, table);
@@ -376,6 +397,33 @@ private String updateColumnNullabilityDefinition(
+ appendColumnDefinition(updateColumn, new StringBuilder());
}
+ @VisibleForTesting
+ static String addIndexDefinition(TableChange.AddIndex addIndex) {
+ StringBuilder sqlBuilder = new StringBuilder();
+ sqlBuilder.append("ADD ");
+ switch (addIndex.getType()) {
+ case PRIMARY_KEY:
+ if (null != addIndex.getName()
+ && !StringUtils.equalsIgnoreCase(
+ addIndex.getName(), Indexes.DEFAULT_MYSQL_PRIMARY_KEY_NAME)) {
+ throw new IllegalArgumentException("Primary key name must be PRIMARY in MySQL");
+ }
+ sqlBuilder.append("PRIMARY KEY ");
+ break;
+ case UNIQUE_KEY:
+ sqlBuilder
+ .append("UNIQUE INDEX ")
+ .append(BACK_QUOTE)
+ .append(addIndex.getName())
+ .append(BACK_QUOTE);
+ break;
+ default:
+ break;
+ }
+ sqlBuilder.append(" (").append(getIndexFieldStr(addIndex.getFieldNames())).append(")");
+ return sqlBuilder.toString();
+ }
+
private String generateTableProperties(List setProperties) {
return setProperties.stream()
.map(
diff --git a/catalogs/catalog-jdbc-mysql/src/test/java/com/datastrato/gravitino/catalog/mysql/operation/TestMysqlTableOperations.java b/catalogs/catalog-jdbc-mysql/src/test/java/com/datastrato/gravitino/catalog/mysql/operation/TestMysqlTableOperations.java
index 4462b66c2d6..34e596f35b2 100644
--- a/catalogs/catalog-jdbc-mysql/src/test/java/com/datastrato/gravitino/catalog/mysql/operation/TestMysqlTableOperations.java
+++ b/catalogs/catalog-jdbc-mysql/src/test/java/com/datastrato/gravitino/catalog/mysql/operation/TestMysqlTableOperations.java
@@ -4,6 +4,7 @@
*/
package com.datastrato.gravitino.catalog.mysql.operation;
+import com.datastrato.gravitino.rel.TableChange;
import com.datastrato.gravitino.rel.indexes.Index;
import com.datastrato.gravitino.rel.indexes.Indexes;
import org.junit.jupiter.api.Assertions;
@@ -48,4 +49,36 @@ public void testAppendIndexesBuilder() {
+ "CONSTRAINT `uk_3` UNIQUE (`col_4`, `col_5`, `col_6`, `col_7`)";
Assertions.assertEquals(expectedStr, sql.toString());
}
+
+ @Test
+ public void testOperationIndexDefinition() {
+ TableChange.AddIndex failIndex =
+ new TableChange.AddIndex(Index.IndexType.PRIMARY_KEY, "pk_1", new String[][] {{"col_1"}});
+ IllegalArgumentException illegalArgumentException =
+ Assertions.assertThrows(
+ IllegalArgumentException.class,
+ () -> MysqlTableOperations.addIndexDefinition(failIndex));
+ Assertions.assertTrue(
+ illegalArgumentException
+ .getMessage()
+ .contains("Primary key name must be PRIMARY in MySQL"));
+
+ TableChange.AddIndex successIndex =
+ new TableChange.AddIndex(
+ Index.IndexType.UNIQUE_KEY, "uk_1", new String[][] {{"col_1"}, {"col_2"}});
+ String sql = MysqlTableOperations.addIndexDefinition(successIndex);
+ Assertions.assertEquals("ADD UNIQUE INDEX `uk_1` (`col_1`, `col_2`)", sql);
+
+ successIndex =
+ new TableChange.AddIndex(
+ Index.IndexType.PRIMARY_KEY,
+ Indexes.DEFAULT_MYSQL_PRIMARY_KEY_NAME,
+ new String[][] {{"col_1"}, {"col_2"}});
+ sql = MysqlTableOperations.addIndexDefinition(successIndex);
+ Assertions.assertEquals("ADD PRIMARY KEY (`col_1`, `col_2`)", sql);
+
+ TableChange.DeleteIndex deleteIndex = new TableChange.DeleteIndex("uk_1", false);
+ sql = MysqlTableOperations.deleteIndexDefinition(null, deleteIndex);
+ Assertions.assertEquals("DROP INDEX `uk_1`", sql);
+ }
}
diff --git a/catalogs/catalog-jdbc-postgresql/src/main/java/com/datastrato/gravitino/catalog/postgresql/operation/PostgreSqlTableOperations.java b/catalogs/catalog-jdbc-postgresql/src/main/java/com/datastrato/gravitino/catalog/postgresql/operation/PostgreSqlTableOperations.java
index b0462a68d70..11905628c9b 100644
--- a/catalogs/catalog-jdbc-postgresql/src/main/java/com/datastrato/gravitino/catalog/postgresql/operation/PostgreSqlTableOperations.java
+++ b/catalogs/catalog-jdbc-postgresql/src/main/java/com/datastrato/gravitino/catalog/postgresql/operation/PostgreSqlTableOperations.java
@@ -143,17 +143,7 @@ protected String generateCreateTableSql(
@VisibleForTesting
static void appendIndexesSql(Index[] indexes, StringBuilder sqlBuilder) {
for (Index index : indexes) {
- String fieldStr =
- Arrays.stream(index.fieldNames())
- .map(
- colNames -> {
- if (colNames.length > 1) {
- throw new IllegalArgumentException(
- "Index does not support complex fields in PostgreSQL");
- }
- return PG_QUOTE + colNames[0] + PG_QUOTE;
- })
- .collect(Collectors.joining(", "));
+ String fieldStr = getIndexFieldStr(index.fieldNames());
sqlBuilder.append(",").append(NEW_LINE);
switch (index.type()) {
case PRIMARY_KEY:
@@ -174,6 +164,19 @@ static void appendIndexesSql(Index[] indexes, StringBuilder sqlBuilder) {
}
}
+ private static String getIndexFieldStr(String[][] fieldNames) {
+ return Arrays.stream(fieldNames)
+ .map(
+ colNames -> {
+ if (colNames.length > 1) {
+ throw new IllegalArgumentException(
+ "Index does not support complex fields in PostgreSQL");
+ }
+ return PG_QUOTE + colNames[0] + PG_QUOTE;
+ })
+ .collect(Collectors.joining(", "));
+ }
+
private void appendColumnDefinition(JdbcColumn column, StringBuilder sqlBuilder) {
// Add data type
sqlBuilder
@@ -270,6 +273,10 @@ protected String generateAlterTableSql(
validateUpdateColumnNullable(updateColumnNullability, lazyLoadTable);
alterSql.add(updateColumnNullabilityDefinition(updateColumnNullability, tableName));
+ } else if (change instanceof TableChange.AddIndex) {
+ alterSql.add(addIndexDefinition(tableName, (TableChange.AddIndex) change));
+ } else if (change instanceof TableChange.DeleteIndex) {
+ alterSql.add(deleteIndexDefinition(tableName, (TableChange.DeleteIndex) change));
} else {
throw new IllegalArgumentException(
"Unsupported table change type: " + change.getClass().getName());
@@ -287,6 +294,63 @@ protected String generateAlterTableSql(
return result;
}
+ @VisibleForTesting
+ static String deleteIndexDefinition(String tableName, TableChange.DeleteIndex deleteIndex) {
+ StringBuilder sqlBuilder = new StringBuilder();
+ sqlBuilder
+ .append("ALTER TABLE ")
+ .append(PG_QUOTE)
+ .append(tableName)
+ .append(PG_QUOTE)
+ .append(" DROP CONSTRAINT ")
+ .append(PG_QUOTE)
+ .append(deleteIndex.getName())
+ .append(PG_QUOTE)
+ .append(";\n");
+ if (deleteIndex.isIfExists()) {
+ sqlBuilder
+ .append("DROP INDEX IF EXISTS ")
+ .append(PG_QUOTE)
+ .append(deleteIndex.getName())
+ .append(PG_QUOTE)
+ .append(";");
+ } else {
+ sqlBuilder
+ .append("DROP INDEX ")
+ .append(PG_QUOTE)
+ .append(deleteIndex.getName())
+ .append(PG_QUOTE)
+ .append(";");
+ }
+ return sqlBuilder.toString();
+ }
+
+ @VisibleForTesting
+ static String addIndexDefinition(String tableName, TableChange.AddIndex addIndex) {
+ StringBuilder sqlBuilder = new StringBuilder();
+ sqlBuilder
+ .append("ALTER TABLE ")
+ .append(PG_QUOTE)
+ .append(tableName)
+ .append(PG_QUOTE)
+ .append(" ADD CONSTRAINT ")
+ .append(PG_QUOTE)
+ .append(addIndex.getName())
+ .append(PG_QUOTE);
+ switch (addIndex.getType()) {
+ case PRIMARY_KEY:
+ sqlBuilder.append(" PRIMARY KEY ");
+ break;
+ case UNIQUE_KEY:
+ sqlBuilder.append(" UNIQUE ");
+ break;
+ default:
+ throw new IllegalArgumentException("Unsupported index type: " + addIndex.getType());
+ }
+ sqlBuilder.append("(").append(getIndexFieldStr(addIndex.getFieldNames())).append(");");
+ return sqlBuilder.toString();
+ }
+
private String updateColumnNullabilityDefinition(
TableChange.UpdateColumnNullability updateColumnNullability, String tableName) {
if (updateColumnNullability.fieldName().length > 1) {
diff --git a/catalogs/catalog-jdbc-postgresql/src/test/java/com/datastrato/gravitino/catalog/postgresql/operation/TestPostgreSqlTableOperations.java b/catalogs/catalog-jdbc-postgresql/src/test/java/com/datastrato/gravitino/catalog/postgresql/operation/TestPostgreSqlTableOperations.java
index 5b442c4e6c1..0db95e3dfde 100644
--- a/catalogs/catalog-jdbc-postgresql/src/test/java/com/datastrato/gravitino/catalog/postgresql/operation/TestPostgreSqlTableOperations.java
+++ b/catalogs/catalog-jdbc-postgresql/src/test/java/com/datastrato/gravitino/catalog/postgresql/operation/TestPostgreSqlTableOperations.java
@@ -4,6 +4,7 @@
*/
package com.datastrato.gravitino.catalog.postgresql.operation;
+import com.datastrato.gravitino.rel.TableChange;
import com.datastrato.gravitino.rel.indexes.Index;
import com.datastrato.gravitino.rel.indexes.Indexes;
import org.apache.commons.lang3.StringUtils;
@@ -64,4 +65,36 @@ public void testAppendIndexesSql() {
illegalArgumentException.getMessage(),
"Index does not support complex fields in PostgreSQL"));
}
+
+ @Test
+ public void testOperationIndexDefinition() {
+ String tableName = "t1";
+ // Test add index definition success.
+ TableChange.AddIndex addIndex =
+ new TableChange.AddIndex(
+ Index.IndexType.PRIMARY_KEY, "test_pk", new String[][] {{"col_1"}});
+ String result = PostgreSqlTableOperations.addIndexDefinition(tableName, addIndex);
+ Assertions.assertEquals(
+ "ALTER TABLE \"t1\" ADD CONSTRAINT \"test_pk\" PRIMARY KEY (\"col_1\");", result);
+
+ addIndex =
+ new TableChange.AddIndex(
+ Index.IndexType.UNIQUE_KEY, "test_uk", new String[][] {{"col_1"}, {"col_2"}});
+ result = PostgreSqlTableOperations.addIndexDefinition(tableName, addIndex);
+ Assertions.assertEquals(
+ "ALTER TABLE \"t1\" ADD CONSTRAINT \"test_uk\" UNIQUE (\"col_1\", \"col_2\");", result);
+
+ // Test delete index definition.
+ TableChange.DeleteIndex deleteIndex = new TableChange.DeleteIndex("test_pk", false);
+ result = PostgreSqlTableOperations.deleteIndexDefinition(tableName, deleteIndex);
+ Assertions.assertEquals(
+ "ALTER TABLE \"t1\" DROP CONSTRAINT \"test_pk\";\n" + "DROP INDEX \"test_pk\";", result);
+
+ deleteIndex = new TableChange.DeleteIndex("test_2_pk", true);
+ result = PostgreSqlTableOperations.deleteIndexDefinition(tableName, deleteIndex);
+ Assertions.assertEquals(
+ "ALTER TABLE \"t1\" DROP CONSTRAINT \"test_2_pk\";\n"
+ + "DROP INDEX IF EXISTS \"test_2_pk\";",
+ result);
+ }
}
diff --git a/clients/client-java/src/main/java/com/datastrato/gravitino/client/DTOConverters.java b/clients/client-java/src/main/java/com/datastrato/gravitino/client/DTOConverters.java
index 0be41467554..4ce7f982003 100644
--- a/clients/client-java/src/main/java/com/datastrato/gravitino/client/DTOConverters.java
+++ b/clients/client-java/src/main/java/com/datastrato/gravitino/client/DTOConverters.java
@@ -147,6 +147,15 @@ static TableUpdateRequest toTableUpdateRequest(TableChange change) {
} else if (change instanceof TableChange.ColumnChange) {
return toColumnUpdateRequest((TableChange.ColumnChange) change);
+ } else if (change instanceof TableChange.AddIndex) {
+ return new TableUpdateRequest.AddTableIndexRequest(
+ ((TableChange.AddIndex) change).getType(),
+ ((TableChange.AddIndex) change).getName(),
+ ((TableChange.AddIndex) change).getFieldNames());
+ } else if (change instanceof TableChange.DeleteIndex) {
+ return new TableUpdateRequest.DeleteTableIndexRequest(
+ ((TableChange.DeleteIndex) change).getName(),
+ ((TableChange.DeleteIndex) change).isIfExists());
} else {
throw new IllegalArgumentException(
"Unknown change type: " + change.getClass().getSimpleName());
diff --git a/common/src/main/java/com/datastrato/gravitino/dto/requests/TableUpdateRequest.java b/common/src/main/java/com/datastrato/gravitino/dto/requests/TableUpdateRequest.java
index a346065ec7c..fdec6e496e5 100644
--- a/common/src/main/java/com/datastrato/gravitino/dto/requests/TableUpdateRequest.java
+++ b/common/src/main/java/com/datastrato/gravitino/dto/requests/TableUpdateRequest.java
@@ -6,6 +6,8 @@
import com.datastrato.gravitino.json.JsonUtils;
import com.datastrato.gravitino.rel.TableChange;
+import com.datastrato.gravitino.rel.indexes.Index;
+import com.datastrato.gravitino.rel.indexes.Indexes;
import com.datastrato.gravitino.rel.types.Type;
import com.datastrato.gravitino.rest.RESTRequest;
import com.fasterxml.jackson.annotation.JsonIgnoreProperties;
@@ -54,7 +56,11 @@
name = "updateColumnNullability"),
@JsonSubTypes.Type(
value = TableUpdateRequest.DeleteTableColumnRequest.class,
- name = "deleteColumn")
+ name = "deleteColumn"),
+ @JsonSubTypes.Type(value = TableUpdateRequest.AddTableIndexRequest.class, name = "addTableIndex"),
+ @JsonSubTypes.Type(
+ value = TableUpdateRequest.DeleteTableIndexRequest.class,
+ name = "deleteTableIndex")
})
public interface TableUpdateRequest extends RESTRequest {
@@ -672,4 +678,91 @@ public TableChange tableChange() {
return TableChange.deleteColumn(fieldName, ifExists);
}
}
+
+ /** Represents a request to add an index to a table. */
+ @EqualsAndHashCode
+ @ToString
+ class AddTableIndexRequest implements TableUpdateRequest {
+
+ @JsonProperty("index")
+ @JsonSerialize(using = JsonUtils.IndexSerializer.class)
+ @JsonDeserialize(using = JsonUtils.IndexDeserializer.class)
+ private Index index;
+
+ /** Default constructor for Jackson deserialization. */
+ public AddTableIndexRequest() {}
+
+ /**
+ * The constructor of the add table index request.
+ *
+ * @param type The type of the index
+ * @param name The name of the index
+ * @param fieldNames The field names under the table contained in the index.
+ */
+ public AddTableIndexRequest(Index.IndexType type, String name, String[][] fieldNames) {
+ this.index = Indexes.of(type, name, fieldNames);
+ }
+
+ /**
+ * Validates the request.
+ *
+ * @throws IllegalArgumentException If the request is invalid, this exception is thrown.
+ */
+ @Override
+ public void validate() throws IllegalArgumentException {
+ Preconditions.checkNotNull(index, "Index cannot be null");
+ Preconditions.checkArgument(index.type() != null, "Index type cannot be null");
+ Preconditions.checkArgument(
+ index.fieldNames() != null && index.fieldNames().length > 0,
+ "The index must be set with corresponding column names");
+ }
+
+ /** @return An instance of TableChange. */
+ @Override
+ public TableChange tableChange() {
+ return TableChange.addIndex(index.type(), index.name(), index.fieldNames());
+ }
+ }
+
+ /** Represents a request to delete an index from a table. */
+ @EqualsAndHashCode
+ @ToString
+ class DeleteTableIndexRequest implements TableUpdateRequest {
+
+ @JsonProperty("name")
+ private String name;
+
+ @JsonProperty("ifExists")
+ private Boolean ifExists;
+
+ /** Default constructor for Jackson deserialization. */
+ public DeleteTableIndexRequest() {}
+
+ /**
+ * The constructor of the delete table index request.
+ *
+ * @param name The name of the index
+ * @param ifExists Whether to delete the index if it exists
+ */
+ public DeleteTableIndexRequest(String name, Boolean ifExists) {
+ this.name = name;
+ this.ifExists = ifExists;
+ }
+
+ /**
+ * Validates the request.
+ *
+ * @throws IllegalArgumentException If the request is invalid, this exception is thrown.
+ */
+ @Override
+ public void validate() throws IllegalArgumentException {
+ Preconditions.checkNotNull(name, "Index name cannot be null");
+ }
+
+ /** @return An instance of TableChange. */
+ @Override
+ public TableChange tableChange() {
+ return TableChange.deleteIndex(name, ifExists);
+ }
+ }
}
diff --git a/common/src/test/java/com/datastrato/gravitino/dto/requests/TestTableUpdatesRequest.java b/common/src/test/java/com/datastrato/gravitino/dto/requests/TestTableUpdatesRequest.java
index 42c6fc8a95a..9ff8584533a 100644
--- a/common/src/test/java/com/datastrato/gravitino/dto/requests/TestTableUpdatesRequest.java
+++ b/common/src/test/java/com/datastrato/gravitino/dto/requests/TestTableUpdatesRequest.java
@@ -6,6 +6,8 @@
import com.datastrato.gravitino.json.JsonUtils;
import com.datastrato.gravitino.rel.TableChange;
+import com.datastrato.gravitino.rel.indexes.Index;
+import com.datastrato.gravitino.rel.indexes.Indexes;
import com.datastrato.gravitino.rel.types.Types;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.google.common.collect.ImmutableList;
@@ -193,4 +195,36 @@ public void testAddTableColumnRequest() throws JsonProcessingException {
.getPosition()
instanceof TableChange.Default);
}
+
+ @Test
+ public void testOperationTableIndexRequest() throws JsonProcessingException {
+ // check add index request
+ TableUpdateRequest tableUpdateRequest =
+ new TableUpdateRequest.AddTableIndexRequest(
+ Index.IndexType.PRIMARY_KEY,
+ Indexes.DEFAULT_MYSQL_PRIMARY_KEY_NAME,
+ new String[][] {{"column1"}});
+ String jsonString = JsonUtils.objectMapper().writeValueAsString(tableUpdateRequest);
+ String expected =
+ "{\"@type\":\"addTableIndex\",\"index\":{\"indexType\":\"PRIMARY_KEY\",\"name\":\"PRIMARY\",\"fieldNames\":[[\"column1\"]]}}";
+ Assertions.assertEquals(
+ JsonUtils.objectMapper().readTree(expected), JsonUtils.objectMapper().readTree(jsonString));
+
+ tableUpdateRequest =
+ new TableUpdateRequest.AddTableIndexRequest(
+ Index.IndexType.UNIQUE_KEY, "uk_2", new String[][] {{"column2"}});
+ jsonString = JsonUtils.objectMapper().writeValueAsString(tableUpdateRequest);
+ expected =
+ "{\"@type\":\"addTableIndex\",\"index\":{\"indexType\":\"UNIQUE_KEY\",\"name\":\"uk_2\",\"fieldNames\":[[\"column2\"]]}}";
+ Assertions.assertEquals(
+ JsonUtils.objectMapper().readTree(expected), JsonUtils.objectMapper().readTree(jsonString));
+
+ // check delete index request
+ TableUpdateRequest.DeleteTableIndexRequest deleteTableIndexRequest =
+ new TableUpdateRequest.DeleteTableIndexRequest("uk_2", true);
+ jsonString = JsonUtils.objectMapper().writeValueAsString(deleteTableIndexRequest);
+ expected = "{\"@type\":\"deleteTableIndex\",\"name\":\"uk_2\",\"ifExists\":true}";
+ Assertions.assertEquals(
+ JsonUtils.objectMapper().readTree(expected), JsonUtils.objectMapper().readTree(jsonString));
+ }
}
diff --git a/integration-test/src/test/java/com/datastrato/gravitino/integration/test/catalog/jdbc/mysql/CatalogMysqlIT.java b/integration-test/src/test/java/com/datastrato/gravitino/integration/test/catalog/jdbc/mysql/CatalogMysqlIT.java
index 276d5405c2d..7160b967b1a 100644
--- a/integration-test/src/test/java/com/datastrato/gravitino/integration/test/catalog/jdbc/mysql/CatalogMysqlIT.java
+++ b/integration-test/src/test/java/com/datastrato/gravitino/integration/test/catalog/jdbc/mysql/CatalogMysqlIT.java
@@ -1301,4 +1301,85 @@ void testUnparsedTypeConverter() {
.loadTable(NameIdentifier.of(metalakeName, catalogName, schemaName, tableName));
Assertions.assertEquals(Types.UnparsedType.of("BIT"), loadedTable.columns()[0].dataType());
}
+
+ @Test
+ void testOperationTableIndex() {
+ String tableName = GravitinoITUtils.genRandomName("test_add_index");
+ Column col1 = Column.of("col_1", Types.LongType.get(), "id", false, false, null);
+ Column col2 = Column.of("col_2", Types.VarCharType.of(255), "code", false, false, null);
+ Column col3 = Column.of("col_3", Types.VarCharType.of(255), "config", false, false, null);
+ Column[] newColumns = new Column[] {col1, col2, col3};
+ TableCatalog tableCatalog = catalog.asTableCatalog();
+ tableCatalog.createTable(
+ NameIdentifier.of(metalakeName, catalogName, schemaName, tableName),
+ newColumns,
+ table_comment,
+ createProperties(),
+ Transforms.EMPTY_TRANSFORM,
+ Distributions.NONE,
+ new SortOrder[0],
+ Indexes.EMPTY_INDEXES);
+
+ // add index test.
+ tableCatalog.alterTable(
+ NameIdentifier.of(metalakeName, catalogName, schemaName, tableName),
+ TableChange.addIndex(
+ Index.IndexType.UNIQUE_KEY, "u1_key", new String[][] {{"col_2"}, {"col_3"}}),
+ TableChange.addIndex(
+ Index.IndexType.PRIMARY_KEY,
+ Indexes.DEFAULT_MYSQL_PRIMARY_KEY_NAME,
+ new String[][] {{"col_1"}}));
+
+ Table table =
+ tableCatalog.loadTable(NameIdentifier.of(metalakeName, catalogName, schemaName, tableName));
+ Index[] indexes =
+ new Index[] {
+ Indexes.unique("u1_key", new String[][] {{"col_2"}, {"col_3"}}),
+ Indexes.createMysqlPrimaryKey(new String[][] {{"col_1"}})
+ };
+ assertionsTableInfo(
+ tableName, table_comment, Arrays.asList(newColumns), createProperties(), indexes, table);
+
+ // delete index and add new column and index.
+ tableCatalog.alterTable(
+ NameIdentifier.of(metalakeName, catalogName, schemaName, tableName),
+ TableChange.deleteIndex("u1_key", false),
+ TableChange.addColumn(
+ new String[] {"col_4"},
+ Types.VarCharType.of(255),
+ TableChange.ColumnPosition.defaultPos()),
+ TableChange.addIndex(Index.IndexType.UNIQUE_KEY, "u2_key", new String[][] {{"col_4"}}));
+
+ indexes =
+ new Index[] {
+ Indexes.createMysqlPrimaryKey(new String[][] {{"col_1"}}),
+ Indexes.unique("u2_key", new String[][] {{"col_4"}})
+ };
+ table =
+ tableCatalog.loadTable(NameIdentifier.of(metalakeName, catalogName, schemaName, tableName));
+ Column col4 = Column.of("col_4", Types.VarCharType.of(255), null, true, false, null);
+ newColumns = new Column[] {col1, col2, col3, col4};
+ assertionsTableInfo(
+ tableName, table_comment, Arrays.asList(newColumns), createProperties(), indexes, table);
+
+ // Add a previously existing index
+ tableCatalog.alterTable(
+ NameIdentifier.of(metalakeName, catalogName, schemaName, tableName),
+ TableChange.addIndex(
+ Index.IndexType.UNIQUE_KEY, "u1_key", new String[][] {{"col_2"}, {"col_3"}}),
+ TableChange.addIndex(
+ Index.IndexType.UNIQUE_KEY, "u3_key", new String[][] {{"col_1"}, {"col_4"}}));
+
+ indexes =
+ new Index[] {
+ Indexes.createMysqlPrimaryKey(new String[][] {{"col_1"}}),
+ Indexes.unique("u2_key", new String[][] {{"col_4"}}),
+ Indexes.unique("u1_key", new String[][] {{"col_2"}, {"col_3"}}),
+ Indexes.unique("u3_key", new String[][] {{"col_1"}, {"col_4"}})
+ };
+ table =
+ tableCatalog.loadTable(NameIdentifier.of(metalakeName, catalogName, schemaName, tableName));
+ assertionsTableInfo(
+ tableName, table_comment, Arrays.asList(newColumns), createProperties(), indexes, table);
+ }
}
diff --git a/integration-test/src/test/java/com/datastrato/gravitino/integration/test/catalog/jdbc/postgresql/CatalogPostgreSqlIT.java b/integration-test/src/test/java/com/datastrato/gravitino/integration/test/catalog/jdbc/postgresql/CatalogPostgreSqlIT.java
index 844919d3e98..f00ea380ee2 100644
--- a/integration-test/src/test/java/com/datastrato/gravitino/integration/test/catalog/jdbc/postgresql/CatalogPostgreSqlIT.java
+++ b/integration-test/src/test/java/com/datastrato/gravitino/integration/test/catalog/jdbc/postgresql/CatalogPostgreSqlIT.java
@@ -1193,4 +1193,82 @@ void testUnparsedTypeConverter() {
.loadTable(NameIdentifier.of(metalakeName, catalogName, schemaName, tableName));
Assertions.assertEquals(Types.UnparsedType.of("bit"), loadedTable.columns()[0].dataType());
}
+
+ @Test
+ void testOperationTableIndex() {
+ String tableName = GravitinoITUtils.genRandomName("test_add_index");
+ Column col1 = Column.of("col_1", Types.LongType.get(), "id", false, false, null);
+ Column col2 = Column.of("col_2", Types.VarCharType.of(255), "code", false, false, null);
+ Column col3 = Column.of("col_3", Types.VarCharType.of(255), "config", false, false, null);
+ Column[] newColumns = new Column[] {col1, col2, col3};
+ TableCatalog tableCatalog = catalog.asTableCatalog();
+ tableCatalog.createTable(
+ NameIdentifier.of(metalakeName, catalogName, schemaName, tableName),
+ newColumns,
+ table_comment,
+ createProperties(),
+ Transforms.EMPTY_TRANSFORM,
+ Distributions.NONE,
+ new SortOrder[0],
+ Indexes.EMPTY_INDEXES);
+
+ // add index test.
+ tableCatalog.alterTable(
+ NameIdentifier.of(metalakeName, catalogName, schemaName, tableName),
+ TableChange.addIndex(
+ Index.IndexType.UNIQUE_KEY, "u1_key", new String[][] {{"col_2"}, {"col_3"}}),
+ TableChange.addIndex(Index.IndexType.PRIMARY_KEY, "pk1_key", new String[][] {{"col_1"}}));
+
+ Table table =
+ tableCatalog.loadTable(NameIdentifier.of(metalakeName, catalogName, schemaName, tableName));
+ Index[] indexes =
+ new Index[] {
+ Indexes.unique("u1_key", new String[][] {{"col_2"}, {"col_3"}}),
+ Indexes.primary("pk1_key", new String[][] {{"col_1"}})
+ };
+ assertionsTableInfo(
+ tableName, table_comment, Arrays.asList(newColumns), createProperties(), indexes, table);
+
+ // delete index and add new column and index.
+ tableCatalog.alterTable(
+ NameIdentifier.of(metalakeName, catalogName, schemaName, tableName),
+ TableChange.deleteIndex("u1_key", true),
+ TableChange.addColumn(
+ new String[] {"col_4"},
+ Types.VarCharType.of(255),
+ TableChange.ColumnPosition.defaultPos()),
+ TableChange.addIndex(Index.IndexType.UNIQUE_KEY, "u2_key", new String[][] {{"col_4"}}));
+
+ indexes =
+ new Index[] {
+ Indexes.primary("pk1_key", new String[][] {{"col_1"}}),
+ Indexes.unique("u2_key", new String[][] {{"col_4"}})
+ };
+ table =
+ tableCatalog.loadTable(NameIdentifier.of(metalakeName, catalogName, schemaName, tableName));
+ Column col4 = Column.of("col_4", Types.VarCharType.of(255), null, true, false, null);
+ newColumns = new Column[] {col1, col2, col3, col4};
+ assertionsTableInfo(
+ tableName, table_comment, Arrays.asList(newColumns), createProperties(), indexes, table);
+
+ // Add a previously existing index
+ tableCatalog.alterTable(
+ NameIdentifier.of(metalakeName, catalogName, schemaName, tableName),
+ TableChange.addIndex(
+ Index.IndexType.UNIQUE_KEY, "u1_key", new String[][] {{"col_2"}, {"col_3"}}),
+ TableChange.addIndex(
+ Index.IndexType.UNIQUE_KEY, "u3_key", new String[][] {{"col_1"}, {"col_4"}}));
+
+ indexes =
+ new Index[] {
+ Indexes.primary("pk1_key", new String[][] {{"col_1"}}),
+ Indexes.unique("u2_key", new String[][] {{"col_4"}}),
+ Indexes.unique("u1_key", new String[][] {{"col_2"}, {"col_3"}}),
+ Indexes.unique("u3_key", new String[][] {{"col_1"}, {"col_4"}})
+ };
+ table =
+ tableCatalog.loadTable(NameIdentifier.of(metalakeName, catalogName, schemaName, tableName));
+ assertionsTableInfo(
+ tableName, table_comment, Arrays.asList(newColumns), createProperties(), indexes, table);
+ }
}
diff --git a/server/src/test/java/com/datastrato/gravitino/server/web/rest/TestTableOperations.java b/server/src/test/java/com/datastrato/gravitino/server/web/rest/TestTableOperations.java
index 0e4b3e6c0b2..c2dd309d99b 100644
--- a/server/src/test/java/com/datastrato/gravitino/server/web/rest/TestTableOperations.java
+++ b/server/src/test/java/com/datastrato/gravitino/server/web/rest/TestTableOperations.java
@@ -48,6 +48,8 @@
import com.datastrato.gravitino.rel.expressions.sorts.SortDirection;
import com.datastrato.gravitino.rel.expressions.sorts.SortOrder;
import com.datastrato.gravitino.rel.expressions.transforms.Transform;
+import com.datastrato.gravitino.rel.indexes.Index;
+import com.datastrato.gravitino.rel.indexes.Indexes;
import com.datastrato.gravitino.rel.types.Type;
import com.datastrato.gravitino.rel.types.Types;
import com.datastrato.gravitino.rest.RESTUtils;
@@ -634,6 +636,35 @@ public void testUpdateTableColumnPosition() {
testAlterTableRequest(req, table);
}
+ @Test
+ public void testAddTableIndex() {
+ TableUpdateRequest.AddTableIndexRequest req =
+ new TableUpdateRequest.AddTableIndexRequest(
+ Index.IndexType.PRIMARY_KEY,
+ Indexes.DEFAULT_MYSQL_PRIMARY_KEY_NAME,
+ new String[][] {{"col1"}});
+ Column[] columns =
+ new Column[] {
+ mockColumn("col2", Types.ByteType.get()), mockColumn("col1", Types.StringType.get())
+ };
+ Table table =
+ mockTable("table1", columns, "mock comment", ImmutableMap.of("k1", "v1"), new Transform[0]);
+ testAlterTableRequest(req, table);
+ }
+
+ @Test
+ public void testDeleteTableIndex() {
+ TableUpdateRequest.DeleteTableIndexRequest req =
+ new TableUpdateRequest.DeleteTableIndexRequest("test", false);
+ Column[] columns =
+ new Column[] {
+ mockColumn("col2", Types.ByteType.get()), mockColumn("col1", Types.StringType.get())
+ };
+ Table table =
+ mockTable("table1", columns, "mock comment", ImmutableMap.of("k1", "v1"), new Transform[0]);
+ testAlterTableRequest(req, table);
+ }
+
@Test
public void testDropTable() {
when(dispatcher.dropTable(any())).thenReturn(true);
@@ -779,6 +810,7 @@ private void testAlterTableRequest(TableUpdateRequest req, Table updatedTable) {
Assertions.assertEquals(tableDTO.distribution(), updatedTable.distribution());
Assertions.assertArrayEquals(tableDTO.sortOrder(), updatedTable.sortOrder());
+ Assertions.assertArrayEquals(tableDTO.index(), updatedTable.index());
}
private static String tablePath(String metalake, String catalog, String schema) {
@@ -854,6 +886,7 @@ public static Table mockTable(
when(table.partitioning()).thenReturn(transforms);
when(table.sortOrder()).thenReturn(new SortOrder[0]);
when(table.distribution()).thenReturn(DistributionDTO.NONE);
+ when(table.index()).thenReturn(Indexes.EMPTY_INDEXES);
Audit mockAudit = mock(Audit.class);
when(mockAudit.creator()).thenReturn("gravitino");