aboutsummaryrefslogtreecommitdiffstats
path: root/server/sonar-db-core
diff options
context:
space:
mode:
authorEric Giffon <eric.giffon@sonarsource.com>2023-11-15 17:00:36 +0100
committersonartech <sonartech@sonarsource.com>2023-11-17 20:03:01 +0000
commit4a2b9ead33736817c9b5d4eb7a81349f70f73efa (patch)
tree511eac321e69a7daf7b2ba4e9ddc547f988f2773 /server/sonar-db-core
parent80defaa50e05f78142f830f0a18d1f135659d137 (diff)
downloadsonarqube-4a2b9ead33736817c9b5d4eb7a81349f70f73efa.tar.gz
sonarqube-4a2b9ead33736817c9b5d4eb7a81349f70f73efa.zip
SONAR-21002 Adapt migration tests for all db vendors
Co-authored-by: Léo Geoffroy <leo.geoffroy@sonarsource.com>
Diffstat (limited to 'server/sonar-db-core')
-rw-r--r--server/sonar-db-core/build.gradle3
-rw-r--r--server/sonar-db-core/src/testFixtures/java/org/sonar/db/AbstractDbTester.java134
-rw-r--r--server/sonar-db-core/src/testFixtures/java/org/sonar/db/DatabaseTestUtils.java10
3 files changed, 118 insertions, 29 deletions
diff --git a/server/sonar-db-core/build.gradle b/server/sonar-db-core/build.gradle
index cc70946e9fd..3d32c75be42 100644
--- a/server/sonar-db-core/build.gradle
+++ b/server/sonar-db-core/build.gradle
@@ -40,6 +40,9 @@ dependencies {
testFixturesApi 'commons-dbutils:commons-dbutils'
testFixturesApi 'junit:junit'
testFixturesApi 'org.assertj:assertj-core'
+ testFixturesApi 'org.apache.commons:commons-collections4'
+
+ testFixturesImplementation 'com.oracle.database.jdbc:ojdbc11'
testFixturesCompileOnly 'com.google.code.findbugs:jsr305'
}
diff --git a/server/sonar-db-core/src/testFixtures/java/org/sonar/db/AbstractDbTester.java b/server/sonar-db-core/src/testFixtures/java/org/sonar/db/AbstractDbTester.java
index 0bd97c25111..95ca66ea221 100644
--- a/server/sonar-db-core/src/testFixtures/java/org/sonar/db/AbstractDbTester.java
+++ b/server/sonar-db-core/src/testFixtures/java/org/sonar/db/AbstractDbTester.java
@@ -37,30 +37,54 @@ import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.Date;
-import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.Set;
+import java.util.regex.Matcher;
+import java.util.regex.Pattern;
import javax.annotation.CheckForNull;
import javax.annotation.Nullable;
+import org.apache.commons.collections4.map.CaseInsensitiveMap;
import org.apache.commons.dbutils.QueryRunner;
import org.apache.commons.io.IOUtils;
import org.apache.commons.lang.StringUtils;
import org.junit.rules.ExternalResource;
import org.slf4j.LoggerFactory;
+import org.sonar.db.dialect.H2;
+import org.sonar.db.dialect.MsSql;
+import org.sonar.db.dialect.Oracle;
+import org.sonar.db.dialect.PostgreSql;
import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.collect.Lists.asList;
-import static java.sql.ResultSetMetaData.columnNoNulls;
-import static java.sql.ResultSetMetaData.columnNullable;
+import static java.sql.Types.BIGINT;
+import static java.sql.Types.BIT;
+import static java.sql.Types.BOOLEAN;
+import static java.sql.Types.INTEGER;
+import static java.sql.Types.NUMERIC;
+import static java.sql.Types.NVARCHAR;
+import static java.sql.Types.VARCHAR;
+import static junit.framework.TestCase.fail;
import static org.assertj.core.api.Assertions.assertThat;
-import static org.junit.Assert.fail;
public class AbstractDbTester<T extends TestDb> extends ExternalResource {
private static final Joiner COMMA_JOINER = Joiner.on(", ");
+
+ private static final Pattern INDEX_NAME_PATTERN = Pattern.compile("COALESCE\\(([\\w_]*),");
+
+ private static final Map<Integer, Integer> POSTGRES_TYPE_SUBSTITUTION = Map.of(BOOLEAN, BIT);
+ private static final Map<Integer, Integer> MSSQL_TYPE_SUBSTITUTION = Map.of(
+ BOOLEAN, BIT,
+ VARCHAR, NVARCHAR);
+
+ private static final Map<Integer, Integer> ORACLE_TYPE_SUBSTITUTION = Map.of(
+ BOOLEAN, NUMERIC,
+ BIGINT, NUMERIC,
+ INTEGER, NUMERIC);
+
protected final T db;
public AbstractDbTester(T db) {
@@ -130,10 +154,10 @@ public class AbstractDbTester<T extends TestDb> extends ExternalResource {
}
String sql = "insert into " + table.toLowerCase(Locale.ENGLISH) + " (" +
- COMMA_JOINER.join(valuesByColumn.keySet().stream().map(t -> t.toLowerCase(Locale.ENGLISH)).toArray(String[]::new)) +
- ") values (" +
- COMMA_JOINER.join(Collections.nCopies(valuesByColumn.size(), '?')) +
- ")";
+ COMMA_JOINER.join(valuesByColumn.keySet().stream().map(t -> t.toLowerCase(Locale.ENGLISH)).toArray(String[]::new)) +
+ ") values (" +
+ COMMA_JOINER.join(Collections.nCopies(valuesByColumn.size(), '?')) +
+ ")";
executeUpdateSql(sql, valuesByColumn.values().toArray(new Object[valuesByColumn.size()]));
}
@@ -209,7 +233,7 @@ public class AbstractDbTester<T extends TestDb> extends ExternalResource {
int colCount = metaData.getColumnCount();
List<Map<String, Object>> rows = new ArrayList<>();
while (resultSet.next()) {
- Map<String, Object> columns = new HashMap<>();
+ Map<String, Object> columns = new CaseInsensitiveMap<>();
for (int i = 1; i <= colCount; i++) {
Object value = resultSet.getObject(i);
if (value instanceof Clob) {
@@ -219,7 +243,10 @@ public class AbstractDbTester<T extends TestDb> extends ExternalResource {
} else if (value instanceof BigDecimal) {
// In Oracle, INTEGER types are mapped as BigDecimal
BigDecimal bgValue = ((BigDecimal) value);
- if (bgValue.scale() == 0) {
+ if (metaData.getPrecision(i) == 1) {
+ // In Oracle, Boolean are considered BigDecimal with a precision of 1.
+ value = bgValue.compareTo(BigDecimal.ONE) == 0;
+ } else if (bgValue.scale() == 0) {
value = bgValue.longValue();
} else {
value = bgValue.doubleValue();
@@ -232,6 +259,8 @@ public class AbstractDbTester<T extends TestDb> extends ExternalResource {
value = byteValue.intValue();
} else if (value instanceof Timestamp) {
value = new Date(((Timestamp) value).getTime());
+ } else if (value instanceof oracle.sql.TIMESTAMP) {
+ value = ((oracle.sql.TIMESTAMP) value).dateValue();
}
columns.put(metaData.getColumnLabel(i), value);
}
@@ -240,27 +269,50 @@ public class AbstractDbTester<T extends TestDb> extends ExternalResource {
return rows;
}
- public void assertColumnDefinition(String table, String column, int expectedType, @Nullable Integer expectedSize, @Nullable Boolean isNullable) {
+ public void renameIndex(String tableName, String indexName, String newIndexName) {
+ if (db.getDatabase().getDialect().getId().equals(MsSql.ID)) {
+ executeUpdateSql(String.format("EXEC sp_rename N'%s.%s', N'%s', N'INDEX'", tableName, indexName, newIndexName));
+ } else {
+ executeUpdateSql(String.format("ALTER INDEX %s RENAME TO %s", indexName, newIndexName));
+ }
+ }
+
+ public void assertColumnDefinition(String table, String column, int expectedType, @Nullable Integer expectedSize,
+ @Nullable Boolean isNullable) {
try (Connection connection = getConnection();
- PreparedStatement stmt = connection.prepareStatement("select * from " + table);
- ResultSet res = stmt.executeQuery()) {
- Integer columnIndex = getColumnIndex(res, column);
- if (columnIndex == null) {
- fail("The column '" + column + "' does not exist");
- }
+ ResultSet rs = connection.getMetaData().getColumns(null, null, toVendorCase(table), toVendorCase(column))) {
+ boolean exists = false;
- assertThat(res.getMetaData().getColumnType(columnIndex)).isEqualTo(expectedType);
- if (expectedSize != null) {
- assertThat(res.getMetaData().getColumnDisplaySize(columnIndex)).isEqualTo(expectedSize);
+ while (rs.next()) {
+ if (column.equalsIgnoreCase(rs.getString("COLUMN_NAME"))) {
+ exists = true;
+ assertThat(rs.getInt("DATA_TYPE")).isEqualTo(getDBType(expectedType));
+ if (expectedSize != null) {
+ assertThat(rs.getInt("COLUMN_SIZE")).isEqualTo(expectedSize);
+ }
+ if (isNullable != null) {
+ assertThat(rs.getString("IS_NULLABLE")).isEqualTo(isNullable ? "YES" : "NO");
+ }
+ }
}
- if (isNullable != null) {
- assertThat(res.getMetaData().isNullable(columnIndex)).isEqualTo(isNullable ? columnNullable : columnNoNulls);
+
+ if (!exists) {
+ fail("The column '" + column + "' does not exist");
}
} catch (Exception e) {
throw new IllegalStateException("Fail to check column", e);
}
}
+ private int getDBType(int expectedType) {
+ return switch (db.getDatabase().getDialect().getId()) {
+ case PostgreSql.ID -> POSTGRES_TYPE_SUBSTITUTION.getOrDefault(expectedType, expectedType);
+ case MsSql.ID -> MSSQL_TYPE_SUBSTITUTION.getOrDefault(expectedType, expectedType);
+ case Oracle.ID -> ORACLE_TYPE_SUBSTITUTION.getOrDefault(expectedType, expectedType);
+ default -> expectedType;
+ };
+ }
+
public void assertColumnDoesNotExist(String table, String column) throws SQLException {
try (Connection connection = getConnection();
PreparedStatement stmt = connection.prepareStatement("select * from " + table);
@@ -302,13 +354,16 @@ public class AbstractDbTester<T extends TestDb> extends ExternalResource {
private void assertIndexImpl(String tableName, String indexName, boolean expectedUnique, String expectedColumn, String... expectedSecondaryColumns) {
try (Connection connection = getConnection();
- ResultSet rs = connection.getMetaData().getIndexInfo(null, null, tableName.toUpperCase(Locale.ENGLISH), false, false)) {
+ ResultSet rs = connection.getMetaData().getIndexInfo(null, null, toVendorCase(tableName), false, false)) {
List<String> onColumns = new ArrayList<>();
+
while (rs.next()) {
if (indexName.equalsIgnoreCase(rs.getString("INDEX_NAME"))) {
assertThat(rs.getBoolean("NON_UNIQUE")).isEqualTo(!expectedUnique);
int position = rs.getInt("ORDINAL_POSITION");
- onColumns.add(position - 1, rs.getString("COLUMN_NAME").toLowerCase(Locale.ENGLISH));
+ String columnDefinition = rs.getString("COLUMN_NAME");
+ String columnName = getColumnNameFromDefinition(columnDefinition);
+ onColumns.add(position - 1, columnName.toLowerCase(Locale.ENGLISH));
}
}
assertThat(onColumns).containsExactlyInAnyOrderElementsOf(asList(expectedColumn, expectedSecondaryColumns));
@@ -318,6 +373,21 @@ public class AbstractDbTester<T extends TestDb> extends ExternalResource {
}
/**
+ * Column definition is not necessarly the column name, as we also create index with special expression
+ * see {@link org.sonar.server.platform.db.migration.sql.CreateIndexBuilder#createSqlStatement()}
+ */
+ private static String getColumnNameFromDefinition(String columnDefinition) {
+ Matcher matcher = INDEX_NAME_PATTERN.matcher(columnDefinition);
+ String columnName;
+ if (matcher.find()) {
+ columnName = matcher.group(1);
+ } else {
+ columnName = columnDefinition;
+ }
+ return columnName;
+ }
+
+ /**
* Verify that index with name {@code indexName} does not exist on the table {@code tableName}
*/
public void assertIndexDoesNotExist(String tableName, String indexName) {
@@ -325,7 +395,9 @@ public class AbstractDbTester<T extends TestDb> extends ExternalResource {
ResultSet rs = connection.getMetaData().getIndexInfo(null, null, tableName.toUpperCase(Locale.ENGLISH), false, false)) {
List<String> indices = new ArrayList<>();
while (rs.next()) {
- indices.add(rs.getString("INDEX_NAME").toLowerCase(Locale.ENGLISH));
+ if (rs.getString("INDEX_NAME") != null) {
+ indices.add(rs.getString("INDEX_NAME").toLowerCase(Locale.ENGLISH));
+ }
}
assertThat(indices).doesNotContain(indexName);
} catch (SQLException e) {
@@ -335,10 +407,7 @@ public class AbstractDbTester<T extends TestDb> extends ExternalResource {
public void assertPrimaryKey(String tableName, @Nullable String expectedPkName, String columnName, String... otherColumnNames) {
try (Connection connection = getConnection()) {
- PK pk = pkOf(connection, tableName.toUpperCase(Locale.ENGLISH));
- if (pk == null) {
- pkOf(connection, tableName.toLowerCase(Locale.ENGLISH));
- }
+ PK pk = pkOf(connection, toVendorCase(tableName));
assertThat(pk).as("No primary key is defined on table %s", tableName).isNotNull();
if (expectedPkName != null) {
assertThat(pk.getName()).isEqualToIgnoringCase(expectedPkName);
@@ -394,6 +463,13 @@ public class AbstractDbTester<T extends TestDb> extends ExternalResource {
}
}
+ public String toVendorCase(String input) {
+ if (List.of(H2.ID, Oracle.ID).contains(db.getDatabase().getDialect().getId())) {
+ return input.toUpperCase(Locale.ENGLISH);
+ }
+ return input.toLowerCase(Locale.ENGLISH);
+ }
+
private static final class PkColumn {
private static final Ordering<PkColumn> ORDERING_BY_INDEX = Ordering.natural().onResultOf(PkColumn::getIndex);
diff --git a/server/sonar-db-core/src/testFixtures/java/org/sonar/db/DatabaseTestUtils.java b/server/sonar-db-core/src/testFixtures/java/org/sonar/db/DatabaseTestUtils.java
index 4afeb5a7cd5..06c90abc5a6 100644
--- a/server/sonar-db-core/src/testFixtures/java/org/sonar/db/DatabaseTestUtils.java
+++ b/server/sonar-db-core/src/testFixtures/java/org/sonar/db/DatabaseTestUtils.java
@@ -24,6 +24,8 @@ import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;
import javax.sql.DataSource;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
import org.sonar.db.version.SqTables;
/**
@@ -31,6 +33,8 @@ import org.sonar.db.version.SqTables;
*/
public class DatabaseTestUtils {
+ private static final Logger LOGGER = LoggerFactory.getLogger(DatabaseTestUtils.class);
+
private DatabaseTestUtils() {
}
@@ -63,6 +67,12 @@ public class DatabaseTestUtils {
} catch (SQLException ignored) {
// probably because table does not exist. That's the case with H2 tests.
+ // Connection needs to be rollback to leave it in a clean state
+ try {
+ connection.rollback();
+ } catch (SQLException e) {
+ LOGGER.warn("Fail to rollback transaction when truncating table %s".formatted(table), e);
+ }
}
return false;
}