aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--server/sonar-db-dao/src/schema/schema-sq.ddl2
-rw-r--r--server/sonar-db-migration/src/it/java/org/sonar/server/platform/db/migration/version/v104/PopulateRuleTagsTableIT.java17
-rw-r--r--server/sonar-db-migration/src/it/java/org/sonar/server/platform/db/migration/version/v106/ResizeValueColumnInRuleTagsTableIT.java65
-rw-r--r--server/sonar-db-migration/src/it/resources/org/sonar/server/platform/db/migration/version/v106/ResizeValueColumnInRuleTagsTableIT/schema.sql5
-rw-r--r--server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v104/CreateRuleTagsTable.java3
-rw-r--r--server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v104/PopulateRuleTagsTable.java68
-rw-r--r--server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v106/DbVersion106.java3
-rw-r--r--server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v106/ResizeValueColumnInRuleTagsTable.java46
8 files changed, 190 insertions, 19 deletions
diff --git a/server/sonar-db-dao/src/schema/schema-sq.ddl b/server/sonar-db-dao/src/schema/schema-sq.ddl
index 0825583478d..dbe7d4108fe 100644
--- a/server/sonar-db-dao/src/schema/schema-sq.ddl
+++ b/server/sonar-db-dao/src/schema/schema-sq.ddl
@@ -922,7 +922,7 @@ CREATE TABLE "RULE_REPOSITORIES"(
ALTER TABLE "RULE_REPOSITORIES" ADD CONSTRAINT "PK_RULE_REPOSITORIES" PRIMARY KEY("KEE");
CREATE TABLE "RULE_TAGS"(
- "VALUE" CHARACTER VARYING(40) NOT NULL,
+ "VALUE" CHARACTER VARYING(400) NOT NULL,
"RULE_UUID" CHARACTER VARYING(40) NOT NULL,
"IS_SYSTEM_TAG" BOOLEAN NOT NULL
);
diff --git a/server/sonar-db-migration/src/it/java/org/sonar/server/platform/db/migration/version/v104/PopulateRuleTagsTableIT.java b/server/sonar-db-migration/src/it/java/org/sonar/server/platform/db/migration/version/v104/PopulateRuleTagsTableIT.java
index f7bb7df824b..117cb025b86 100644
--- a/server/sonar-db-migration/src/it/java/org/sonar/server/platform/db/migration/version/v104/PopulateRuleTagsTableIT.java
+++ b/server/sonar-db-migration/src/it/java/org/sonar/server/platform/db/migration/version/v104/PopulateRuleTagsTableIT.java
@@ -75,6 +75,21 @@ class PopulateRuleTagsTableIT {
}
@Test
+ void execute_whenSystemAndCustomTagShareTheSameTag_removeDuplicates() throws SQLException {
+ insertRule("uuid-1", "test,other1", "test,other2");
+
+ migration.execute();
+
+ assertThat(db.select("select value, is_system_tag, rule_uuid from rule_tags"))
+ .extracting(t -> t.get("value"), t -> t.get("is_system_tag"), t -> t.get("rule_uuid"))
+ .containsExactlyInAnyOrder(
+ tuple("test", true, "uuid-1"),
+ tuple("other1", true, "uuid-1"),
+ tuple("other2", false, "uuid-1")
+ );
+ }
+
+ @Test
void execute_whenRunMoreThanOnce_shouldBeReentrant() throws SQLException {
insertRule("uuid-3", "sys_tag", "tag");
migration.execute();
@@ -86,7 +101,7 @@ class PopulateRuleTagsTableIT {
private void verifyMapping() {
assertThat(db.select("select value, is_system_tag, rule_uuid from rule_tags"))
.extracting(t -> t.get("value"), t -> t.get("is_system_tag"), t -> t.get("rule_uuid"))
- .containsExactly(
+ .containsExactlyInAnyOrder(
tuple("sys_tag", true, "uuid-3"),
tuple("tag", false, "uuid-3")
);
diff --git a/server/sonar-db-migration/src/it/java/org/sonar/server/platform/db/migration/version/v106/ResizeValueColumnInRuleTagsTableIT.java b/server/sonar-db-migration/src/it/java/org/sonar/server/platform/db/migration/version/v106/ResizeValueColumnInRuleTagsTableIT.java
new file mode 100644
index 00000000000..1335c194bb7
--- /dev/null
+++ b/server/sonar-db-migration/src/it/java/org/sonar/server/platform/db/migration/version/v106/ResizeValueColumnInRuleTagsTableIT.java
@@ -0,0 +1,65 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2024 SonarSource SA
+ * mailto:info AT sonarsource DOT com
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+package org.sonar.server.platform.db.migration.version.v106;
+
+import java.sql.SQLException;
+import java.sql.Types;
+import org.junit.jupiter.api.Test;
+import org.junit.jupiter.api.extension.RegisterExtension;
+import org.sonar.db.CoreDbTester;
+import org.sonar.db.MigrationDbTester;
+import org.sonar.server.platform.db.migration.version.v104.CreateRuleTagsTable;
+
+class ResizeValueColumnInRuleTagsTableIT {
+
+ private static final String EXPECTED_TABLE_NAME = "rule_tags";
+ private static final String EXPECTED_COLUMN_NAME = "value";
+
+ /**
+ * This is database that has run the new version of the {@link CreateRuleTagsTable} migration with the 400 limit of the value column.
+ */
+ @RegisterExtension
+ public final MigrationDbTester dbWith400LimitOnValueColumn = MigrationDbTester.createForMigrationStep(ResizeValueColumnInRuleTagsTable.class);
+
+ /**
+ * This is the database that has run the old version of the {@link CreateRuleTagsTable} migration with the 40 limit of the value column.
+ */
+ @RegisterExtension
+ public final CoreDbTester dbWith40LimitOnValueColumn = CoreDbTester.createForSchema(ResizeValueColumnInRuleTagsTableIT.class, "schema.sql");
+
+ private final ResizeValueColumnInRuleTagsTable underTestNoAction = new ResizeValueColumnInRuleTagsTable(dbWith400LimitOnValueColumn.database());
+ private final ResizeValueColumnInRuleTagsTable underTestThatFixesColumnSize = new ResizeValueColumnInRuleTagsTable(dbWith40LimitOnValueColumn.database());
+
+ @Test
+ void execute_whenColumnIsNotResized_shouldResizeTheColumn() throws SQLException {
+ dbWith40LimitOnValueColumn.assertColumnDefinition(EXPECTED_TABLE_NAME, EXPECTED_COLUMN_NAME, Types.VARCHAR, 40, false);
+ underTestThatFixesColumnSize.execute();
+ dbWith40LimitOnValueColumn.assertColumnDefinition(EXPECTED_TABLE_NAME, EXPECTED_COLUMN_NAME, Types.VARCHAR, 400, false);
+ }
+
+ @Test
+ void execute_whenColumnIsAlreadyResized_shouldDoNothing() throws SQLException {
+ dbWith400LimitOnValueColumn.assertColumnDefinition(EXPECTED_TABLE_NAME, EXPECTED_COLUMN_NAME, Types.VARCHAR, 400, false);
+ underTestNoAction.execute();
+ dbWith400LimitOnValueColumn.assertColumnDefinition(EXPECTED_TABLE_NAME, EXPECTED_COLUMN_NAME, Types.VARCHAR, 400, false);
+ }
+
+
+}
diff --git a/server/sonar-db-migration/src/it/resources/org/sonar/server/platform/db/migration/version/v106/ResizeValueColumnInRuleTagsTableIT/schema.sql b/server/sonar-db-migration/src/it/resources/org/sonar/server/platform/db/migration/version/v106/ResizeValueColumnInRuleTagsTableIT/schema.sql
new file mode 100644
index 00000000000..ed4f3118592
--- /dev/null
+++ b/server/sonar-db-migration/src/it/resources/org/sonar/server/platform/db/migration/version/v106/ResizeValueColumnInRuleTagsTableIT/schema.sql
@@ -0,0 +1,5 @@
+CREATE TABLE "RULE_TAGS"(
+ "RULE_UUID" CHARACTER VARYING(40) NOT NULL,
+ "IS_SYSTEM_TAG" BOOLEAN NOT NULL,
+ "VALUE" CHARACTER VARYING(40) NOT NULL
+); \ No newline at end of file
diff --git a/server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v104/CreateRuleTagsTable.java b/server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v104/CreateRuleTagsTable.java
index 7ca48fc304e..19852d1939a 100644
--- a/server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v104/CreateRuleTagsTable.java
+++ b/server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v104/CreateRuleTagsTable.java
@@ -32,11 +32,10 @@ public class CreateRuleTagsTable extends CreateTableChange {
static final String RULE_TAGS_TABLE_NAME = "rule_tags";
- static final String UUID_COLUMN_NAME = "uuid";
static final String VALUE_COLUMN_NAME = "value";
static final String IS_SYSTEM_TAG_COLUMN_NAME = "is_system_tag";
static final String RULE_UUID_COLUMN_NAME = "rule_uuid";
- static final int VALUE_COLUMN_SIZE = 40;
+ static final int VALUE_COLUMN_SIZE = 400;
public CreateRuleTagsTable(Database db) {
super(db, RULE_TAGS_TABLE_NAME);
diff --git a/server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v104/PopulateRuleTagsTable.java b/server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v104/PopulateRuleTagsTable.java
index 25aca8cbbc8..e9564a8faf6 100644
--- a/server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v104/PopulateRuleTagsTable.java
+++ b/server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v104/PopulateRuleTagsTable.java
@@ -20,8 +20,11 @@
package org.sonar.server.platform.db.migration.version.v104;
import java.sql.SQLException;
+import java.util.ArrayList;
import java.util.Arrays;
+import java.util.HashSet;
import java.util.List;
+import java.util.Map;
import java.util.Optional;
import java.util.Set;
import java.util.stream.Collectors;
@@ -33,14 +36,14 @@ import org.sonar.server.platform.db.migration.step.Upsert;
public class PopulateRuleTagsTable extends DataChange {
private static final String SELECT_QUERY = """
- SELECT uuid, system_tags AS tag, 1 as is_system_tag
- FROM rules
- WHERE system_tags IS NOT NULL
- UNION ALL
- SELECT uuid, tags AS tag, 0 as is_system_tag
- FROM rules
- WHERE tags IS NOT NULL
- """;
+ SELECT uuid, system_tags AS tag, 1 as is_system_tag
+ FROM rules
+ WHERE system_tags IS NOT NULL
+ UNION ALL
+ SELECT uuid, tags AS tag, 0 as is_system_tag
+ FROM rules
+ WHERE tags IS NOT NULL
+ """;
private static final String INSERT_QUERY = """
INSERT INTO rule_tags (rule_uuid, is_system_tag, value)
@@ -57,18 +60,55 @@ public class PopulateRuleTagsTable extends DataChange {
return;
}
- List<PopulateRuleTagsTable.Tag> allTags = findAllTags(context);
+ List<Tags> allTags = findAllTags(context);
if (allTags.isEmpty()) {
return;
}
+ allTags = removeDuplicatesForAllRule(allTags);
Upsert insertTagsQuery = context.prepareUpsert(INSERT_QUERY);
- for (PopulateRuleTagsTable.Tag tag : allTags) {
- insertEveryTag(insertTagsQuery, tag.ruleUuid(), tag.values(), tag.isSystemTag());
+ for (Tags tags : allTags) {
+ insertEveryTag(insertTagsQuery, tags.ruleUuid(), tags.values(), tags.isSystemTag());
}
insertTagsQuery.execute().commit();
}
+ /**
+ * System tags and custom tags can contain the same values. In this case, we keep only the system tag.
+ */
+ private static List<Tags> removeDuplicatesForAllRule(List<Tags> allTags) {
+ Map<String, List<Tags>> tagsByRuleUuid = allTags.stream().collect(Collectors.groupingBy(Tags::ruleUuid));
+ List<Tags> listWithoutDuplicates = new ArrayList<>();
+
+ for (Map.Entry<String, List<Tags>> entry : tagsByRuleUuid.entrySet()) {
+ listWithoutDuplicates.addAll(removeDuplicateForRule(entry.getValue()));
+ }
+ return listWithoutDuplicates;
+ }
+
+ private static List<Tags> removeDuplicateForRule(List<Tags> ruleTags) {
+ Optional<Tags> systemTags = ruleTags.stream().filter(Tags::isSystemTag).findFirst();
+ Optional<Tags> manualTags = ruleTags.stream().filter(t -> !t.isSystemTag()).findFirst();
+
+ if (systemTags.isEmpty()) {
+ return List.of(manualTags.orElseThrow());
+ } else if (manualTags.isEmpty()) {
+ return List.of(systemTags.orElseThrow());
+ } else {
+ Set<String> systemTagValues = systemTags.get().values();
+ Set<String> manualTagValues = manualTags.get().values();
+ Set<String> commonValues = new HashSet<>(systemTagValues);
+ commonValues.retainAll(manualTagValues);
+
+ if (commonValues.isEmpty()) {
+ return List.of(manualTags.orElseThrow(), systemTags.orElseThrow());
+ } else {
+ manualTagValues.removeAll(commonValues);
+ return List.of(systemTags.orElseThrow(), new Tags(manualTags.get().ruleUuid(), manualTagValues, false));
+ }
+ }
+ }
+
private static void insertEveryTag(Upsert insertRuleTags, String ruleUuid, Set<String> values, boolean isSystemTag) throws SQLException {
for (String tag : values) {
insertRuleTags
@@ -79,9 +119,9 @@ public class PopulateRuleTagsTable extends DataChange {
}
}
- private static List<PopulateRuleTagsTable.Tag> findAllTags(Context context) throws SQLException {
+ private static List<Tags> findAllTags(Context context) throws SQLException {
return context.prepareSelect(SELECT_QUERY)
- .list(r -> new PopulateRuleTagsTable.Tag(r.getString(1), parseTagString(r.getString(2)), r.getBoolean(3)));
+ .list(r -> new Tags(r.getString(1), parseTagString(r.getString(2)), r.getBoolean(3)));
}
private static boolean isTableAlreadyPopulated(Context context) throws SQLException {
@@ -97,7 +137,7 @@ public class PopulateRuleTagsTable extends DataChange {
.collect(Collectors.toSet());
}
- private record Tag(String ruleUuid, Set<String> values, boolean isSystemTag) {
+ private record Tags(String ruleUuid, Set<String> values, boolean isSystemTag) {
}
}
diff --git a/server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v106/DbVersion106.java b/server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v106/DbVersion106.java
index 61507231cf0..6f1ded53b60 100644
--- a/server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v106/DbVersion106.java
+++ b/server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v106/DbVersion106.java
@@ -42,6 +42,7 @@ public class DbVersion106 implements DbVersion {
public void addSteps(MigrationStepRegistry registry) {
registry
.add(10_6_000,"Add 'prioritized_rule' column to 'issues' table", AddPrioritizedRuleColumnToIssuesTable.class)
- .add(10_6_001,"Add 'prioritized_rule' column to 'active_rules' table", AddPrioritizedRuleColumnToActiveRulesTable.class);
+ .add(10_6_001,"Add 'prioritized_rule' column to 'active_rules' table", AddPrioritizedRuleColumnToActiveRulesTable.class)
+ .add(10_6_002,"Ensure 'value' column is resized to 400 in 'rule_tags' table", ResizeValueColumnInRuleTagsTable.class);
}
}
diff --git a/server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v106/ResizeValueColumnInRuleTagsTable.java b/server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v106/ResizeValueColumnInRuleTagsTable.java
new file mode 100644
index 00000000000..9c512b57d66
--- /dev/null
+++ b/server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v106/ResizeValueColumnInRuleTagsTable.java
@@ -0,0 +1,46 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2024 SonarSource SA
+ * mailto:info AT sonarsource DOT com
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+package org.sonar.server.platform.db.migration.version.v106;
+
+import java.sql.SQLException;
+import org.sonar.db.Database;
+import org.sonar.server.platform.db.migration.def.VarcharColumnDef;
+import org.sonar.server.platform.db.migration.sql.AlterColumnsBuilder;
+import org.sonar.server.platform.db.migration.step.DdlChange;
+
+public class ResizeValueColumnInRuleTagsTable extends DdlChange {
+
+ private static final VarcharColumnDef columnDefinition = VarcharColumnDef.newVarcharColumnDefBuilder()
+ .setColumnName("value")
+ .setIsNullable(false)
+ .setLimit(400)
+ .build();
+
+ public ResizeValueColumnInRuleTagsTable(Database db) {
+ super(db);
+ }
+
+ @Override
+ public void execute(Context context) throws SQLException {
+ context.execute(new AlterColumnsBuilder(getDialect(), "rule_tags")
+ .updateColumn(columnDefinition)
+ .build());
+ }
+}