aboutsummaryrefslogtreecommitdiffstats
path: root/server
diff options
context:
space:
mode:
authorJacek <jacek.poreda@sonarsource.com>2021-08-12 16:55:04 +0200
committersonartech <sonartech@sonarsource.com>2021-09-08 20:03:34 +0000
commitcfe5faf0b14841bdad0cae24ac89832c98d1d32a (patch)
treeb089d578bc9b598f46c79e0c22d254fafafcc1fe /server
parent6ad72a24b224d7949ad790b70ef4ae8352ba1899 (diff)
downloadsonarqube-cfe5faf0b14841bdad0cae24ac89832c98d1d32a.tar.gz
sonarqube-cfe5faf0b14841bdad0cae24ac89832c98d1d32a.zip
SONAR-15259 Migrate portfolios from XML to DB
Diffstat (limited to 'server')
-rw-r--r--server/sonar-ce-common/src/main/java/org/sonar/ce/queue/CeTaskSubmit.java5
-rw-r--r--server/sonar-db-core/src/main/java/org/sonar/db/version/SqTables.java3
-rw-r--r--server/sonar-db-dao/src/main/java/org/sonar/db/DaoModule.java2
-rw-r--r--server/sonar-db-dao/src/main/java/org/sonar/db/DbClient.java7
-rw-r--r--server/sonar-db-dao/src/main/java/org/sonar/db/portfolio/PortfolioDao.java31
-rw-r--r--server/sonar-db-dao/src/main/java/org/sonar/db/portfolio/PortfolioDto.java5
-rw-r--r--server/sonar-db-dao/src/main/java/org/sonar/db/portfolio/PortfolioMapper.java8
-rw-r--r--server/sonar-db-dao/src/main/java/org/sonar/db/portfolio/PortfolioService.java212
-rw-r--r--server/sonar-db-dao/src/main/resources/org/sonar/db/portfolio/PortfolioMapper.xml44
-rw-r--r--server/sonar-db-dao/src/schema/schema-sq.ddl2
-rw-r--r--server/sonar-db-dao/src/test/java/org/sonar/db/portfolio/PortfolioDaoTest.java26
-rw-r--r--server/sonar-db-dao/src/testFixtures/java/org/sonar/db/component/ComponentDbTester.java67
-rw-r--r--server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v91/CreatePortfoliosTable.java2
-rw-r--r--server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v91/DbVersion91.java1
-rw-r--r--server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v91/MigratePortfoliosToNewTables.java581
-rw-r--r--server/sonar-db-migration/src/test/java/org/sonar/server/platform/db/migration/version/v91/DbVersion91Test.java2
-rw-r--r--server/sonar-db-migration/src/test/java/org/sonar/server/platform/db/migration/version/v91/MigratePortfoliosToNewTablesTest.java431
-rw-r--r--server/sonar-db-migration/src/test/resources/org/sonar/server/platform/db/migration/version/v91/MigratePortfoliosToNewTablesTest/schema.sql110
-rw-r--r--server/sonar-webserver-webapi/src/main/java/org/sonar/server/component/ComponentUpdater.java28
19 files changed, 1321 insertions, 246 deletions
diff --git a/server/sonar-ce-common/src/main/java/org/sonar/ce/queue/CeTaskSubmit.java b/server/sonar-ce-common/src/main/java/org/sonar/ce/queue/CeTaskSubmit.java
index 2cfef65aef5..b37cddb0a7f 100644
--- a/server/sonar-ce-common/src/main/java/org/sonar/ce/queue/CeTaskSubmit.java
+++ b/server/sonar-ce-common/src/main/java/org/sonar/ce/queue/CeTaskSubmit.java
@@ -27,6 +27,7 @@ import javax.annotation.Nullable;
import javax.annotation.concurrent.Immutable;
import org.sonar.db.component.BranchDto;
import org.sonar.db.component.ComponentDto;
+import org.sonar.db.portfolio.PortfolioDto;
import org.sonar.db.project.ProjectDto;
import static com.google.common.base.MoreObjects.firstNonNull;
@@ -134,6 +135,10 @@ public final class CeTaskSubmit {
return new Component(dto.getUuid(), dto.getUuid());
}
+ public static Component fromDto(PortfolioDto dto) {
+ return new Component(dto.getUuid(), dto.getUuid());
+ }
+
public static Component fromDto(BranchDto dto) {
return new Component(dto.getUuid(), dto.getProjectUuid());
}
diff --git a/server/sonar-db-core/src/main/java/org/sonar/db/version/SqTables.java b/server/sonar-db-core/src/main/java/org/sonar/db/version/SqTables.java
index 1102b02a8a7..8ca9b9f9405 100644
--- a/server/sonar-db-core/src/main/java/org/sonar/db/version/SqTables.java
+++ b/server/sonar-db-core/src/main/java/org/sonar/db/version/SqTables.java
@@ -72,6 +72,9 @@ public final class SqTables {
"perm_templates_groups",
"perm_tpl_characteristics",
"plugins",
+ "portfolios",
+ "portfolio_projects",
+ "portfolio_references",
"projects",
"project_alm_settings",
"project_branches",
diff --git a/server/sonar-db-dao/src/main/java/org/sonar/db/DaoModule.java b/server/sonar-db-dao/src/main/java/org/sonar/db/DaoModule.java
index bf6bda78ddb..f288a33053d 100644
--- a/server/sonar-db-dao/src/main/java/org/sonar/db/DaoModule.java
+++ b/server/sonar-db-dao/src/main/java/org/sonar/db/DaoModule.java
@@ -58,6 +58,7 @@ import org.sonar.db.permission.UserPermissionDao;
import org.sonar.db.permission.template.PermissionTemplateCharacteristicDao;
import org.sonar.db.permission.template.PermissionTemplateDao;
import org.sonar.db.plugin.PluginDao;
+import org.sonar.db.portfolio.PortfolioDao;
import org.sonar.db.project.ProjectDao;
import org.sonar.db.property.InternalComponentPropertiesDao;
import org.sonar.db.property.InternalPropertiesDao;
@@ -134,6 +135,7 @@ public class DaoModule extends Module {
PermissionTemplateDao.class,
PluginDao.class,
ProjectDao.class,
+ PortfolioDao.class,
ProjectLinkDao.class,
ProjectMappingsDao.class,
ProjectQgateAssociationDao.class,
diff --git a/server/sonar-db-dao/src/main/java/org/sonar/db/DbClient.java b/server/sonar-db-dao/src/main/java/org/sonar/db/DbClient.java
index ea6a0516646..3efb4aee142 100644
--- a/server/sonar-db-dao/src/main/java/org/sonar/db/DbClient.java
+++ b/server/sonar-db-dao/src/main/java/org/sonar/db/DbClient.java
@@ -56,6 +56,7 @@ import org.sonar.db.permission.UserPermissionDao;
import org.sonar.db.permission.template.PermissionTemplateCharacteristicDao;
import org.sonar.db.permission.template.PermissionTemplateDao;
import org.sonar.db.plugin.PluginDao;
+import org.sonar.db.portfolio.PortfolioDao;
import org.sonar.db.project.ProjectDao;
import org.sonar.db.property.InternalComponentPropertiesDao;
import org.sonar.db.property.InternalPropertiesDao;
@@ -156,6 +157,7 @@ public class DbClient {
private final ProjectMappingsDao projectMappingsDao;
private final NewCodePeriodDao newCodePeriodDao;
private final ProjectDao projectDao;
+ private final PortfolioDao portfolioDao;
private final SessionTokensDao sessionTokensDao;
private final SamlMessageIdDao samlMessageIdDao;
private final UserDismissedMessagesDao userDismissedMessagesDao;
@@ -232,6 +234,7 @@ public class DbClient {
internalComponentPropertiesDao = getDao(map, InternalComponentPropertiesDao.class);
newCodePeriodDao = getDao(map, NewCodePeriodDao.class);
projectDao = getDao(map, ProjectDao.class);
+ portfolioDao = getDao(map, PortfolioDao.class);
sessionTokensDao = getDao(map, SessionTokensDao.class);
samlMessageIdDao = getDao(map, SamlMessageIdDao.class);
userDismissedMessagesDao = getDao(map, UserDismissedMessagesDao.class);
@@ -314,6 +317,10 @@ public class DbClient {
return projectDao;
}
+ public PortfolioDao portfolioDao() {
+ return portfolioDao;
+ }
+
public ComponentKeyUpdaterDao componentKeyUpdaterDao() {
return componentKeyUpdaterDao;
}
diff --git a/server/sonar-db-dao/src/main/java/org/sonar/db/portfolio/PortfolioDao.java b/server/sonar-db-dao/src/main/java/org/sonar/db/portfolio/PortfolioDao.java
index 602ca4b6144..aaa87d1624d 100644
--- a/server/sonar-db-dao/src/main/java/org/sonar/db/portfolio/PortfolioDao.java
+++ b/server/sonar-db-dao/src/main/java/org/sonar/db/portfolio/PortfolioDao.java
@@ -19,15 +19,23 @@
*/
package org.sonar.db.portfolio;
-import static java.util.Collections.singleton;
-
+import java.util.Collection;
import java.util.List;
+import java.util.Map;
import java.util.Optional;
import java.util.Set;
+import java.util.stream.Collectors;
import org.sonar.api.utils.System2;
import org.sonar.core.util.UuidFactory;
import org.sonar.db.Dao;
import org.sonar.db.DbSession;
+import org.sonar.db.project.ProjectDto;
+
+import static java.util.Collections.singleton;
+import static java.util.stream.Collectors.groupingBy;
+import static java.util.stream.Collectors.mapping;
+import static java.util.stream.Collectors.toSet;
+import static org.sonar.db.DatabaseUtils.executeLargeInputs;
public class PortfolioDao implements Dao {
private final System2 system2;
@@ -54,6 +62,9 @@ public class PortfolioDao implements Dao {
}
public void insert(DbSession dbSession, PortfolioDto portfolio) {
+ if (portfolio.getUuid() == null) {
+ portfolio.setUuid(uuidFactory.create());
+ }
mapper(dbSession).insert(portfolio);
}
@@ -63,7 +74,6 @@ public class PortfolioDao implements Dao {
mapper(dbSession).deleteProjectsByPortfolioUuids(singleton(portfolioUuid));
}
-
public void deleteByUuids(DbSession dbSession, Set<String> portfolioUuids) {
if (portfolioUuids.isEmpty()) {
return;
@@ -95,12 +105,16 @@ public class PortfolioDao implements Dao {
return mapper(dbSession).selectReferencersByKey(referenceKey);
}
- public Set<String> getProjects(DbSession dbSession, String portfolioUuid) {
+ public List<ProjectDto> getProjects(DbSession dbSession, String portfolioUuid) {
return mapper(dbSession).selectProjects(portfolioUuid);
}
- Set<String> getAllProjectsInHierarchy(DbSession dbSession, String rootUuid) {
- return mapper(dbSession).selectAllProjectsInHierarchy(rootUuid);
+ public Map<String, Set<String>> getAllProjectsInHierarchy(DbSession dbSession, String rootUuid) {
+ return mapper(dbSession).selectAllProjectsInHierarchy(rootUuid)
+ .stream()
+ .collect(groupingBy(
+ PortfolioProjectDto::getProjectUuid,
+ mapping(PortfolioProjectDto::getPortfolioUuid, toSet())));
}
public void addProject(DbSession dbSession, String portfolioUuid, String projectUuid) {
@@ -123,4 +137,9 @@ public class PortfolioDao implements Dao {
public void deleteProjects(DbSession dbSession, String portfolioUuid) {
mapper(dbSession).deleteProjects(portfolioUuid);
}
+
+ public Map<String, String> selectKeysByUuids(DbSession dbSession, Collection<String> uuids) {
+ return executeLargeInputs(uuids, uuids1 -> mapper(dbSession).selectByUuids(uuids1)).stream()
+ .collect(Collectors.toMap(PortfolioDto::getUuid, PortfolioDto::getKey));
+ }
}
diff --git a/server/sonar-db-dao/src/main/java/org/sonar/db/portfolio/PortfolioDto.java b/server/sonar-db-dao/src/main/java/org/sonar/db/portfolio/PortfolioDto.java
index 80ae264af5e..9a06cc38d65 100644
--- a/server/sonar-db-dao/src/main/java/org/sonar/db/portfolio/PortfolioDto.java
+++ b/server/sonar-db-dao/src/main/java/org/sonar/db/portfolio/PortfolioDto.java
@@ -74,6 +74,11 @@ public class PortfolioDto {
return this;
}
+ public PortfolioDto setSelectionMode(SelectionMode selectionMode) {
+ this.selectionMode = selectionMode.name();
+ return this;
+ }
+
@CheckForNull
public String getSelectionExpression() {
return selectionExpression;
diff --git a/server/sonar-db-dao/src/main/java/org/sonar/db/portfolio/PortfolioMapper.java b/server/sonar-db-dao/src/main/java/org/sonar/db/portfolio/PortfolioMapper.java
index 929ab51f82e..720ecd99e76 100644
--- a/server/sonar-db-dao/src/main/java/org/sonar/db/portfolio/PortfolioMapper.java
+++ b/server/sonar-db-dao/src/main/java/org/sonar/db/portfolio/PortfolioMapper.java
@@ -19,10 +19,12 @@
*/
package org.sonar.db.portfolio;
+import java.util.Collection;
import java.util.List;
import java.util.Set;
import javax.annotation.CheckForNull;
import org.apache.ibatis.annotations.Param;
+import org.sonar.db.project.ProjectDto;
public interface PortfolioMapper {
@CheckForNull
@@ -53,11 +55,13 @@ public interface PortfolioMapper {
List<PortfolioDto> selectReferencersByKey(String referenceKey);
- Set<String> selectProjects(String portfolioUuid);
+ List<ProjectDto> selectProjects(String portfolioUuid);
List<ReferenceDto> selectAllReferencesToPortfolios();
- Set<String> selectAllProjectsInHierarchy(String rootUuid);
+ Set<PortfolioProjectDto> selectAllProjectsInHierarchy(String rootUuid);
+
+ List<PortfolioDto> selectByUuids(@Param("uuids") Collection<String> uuids);
void update(PortfolioDto portfolio);
diff --git a/server/sonar-db-dao/src/main/java/org/sonar/db/portfolio/PortfolioService.java b/server/sonar-db-dao/src/main/java/org/sonar/db/portfolio/PortfolioService.java
deleted file mode 100644
index 2c16358f579..00000000000
--- a/server/sonar-db-dao/src/main/java/org/sonar/db/portfolio/PortfolioService.java
+++ /dev/null
@@ -1,212 +0,0 @@
-/*
- * Copyright (C) 2016-2021 SonarSource SA
- * All rights reserved
- * mailto:info AT sonarsource DOT com
- */
-package org.sonar.db.portfolio;
-
-import java.util.ArrayList;
-import java.util.HashMap;
-import java.util.HashSet;
-import java.util.LinkedList;
-import java.util.List;
-import java.util.Map;
-import java.util.Set;
-import java.util.stream.Collectors;
-import javax.annotation.Nullable;
-import org.sonar.db.DbClient;
-import org.sonar.db.DbSession;
-import org.sonar.db.project.ProjectDao;
-import org.sonar.db.project.ProjectDto;
-
-import static java.lang.String.format;
-import static java.util.Collections.emptyList;
-
-public class PortfolioService {
- private final DbClient dbClient;
- private final PortfolioDao portfolioDao;
- private final ProjectDao projectDao;
-
- public PortfolioService(DbClient dbClient, PortfolioDao portfolioDao, ProjectDao projectDao) {
- this.dbClient = dbClient;
- this.portfolioDao = portfolioDao;
- this.projectDao = projectDao;
- }
-
- public void deleteReferencersTo(String referenceKey) {
- try (DbSession dbSession = dbClient.openSession(false)) {
- List<PortfolioDto> referencers = portfolioDao.selectReferencersByKey(dbSession, referenceKey);
- }
- }
-
- public List<PortfolioDto> getRoots() {
- try (DbSession dbSession = dbClient.openSession(false)) {
- return portfolioDao.selectAllRoots(dbSession);
- }
- }
-
- public List<PortfolioDto> getReferencersByKey(String referenceKey) {
- try (DbSession dbSession = dbClient.openSession(false)) {
- return portfolioDao.selectReferencersByKey(dbSession, referenceKey);
- }
- }
-
- public void addReferenceToPortfolio(String portfolioKey, PortfolioDto refPortfolio) {
- try (DbSession dbSession = dbClient.openSession(false)) {
- PortfolioDto portfolio = selectByKeyOrFail(dbSession, portfolioKey);
- Set<String> treeReferenceConnections = getTreeReferenceConnections(dbSession, portfolio.getRootUuid());
- if (treeReferenceConnections.contains(refPortfolio.getRootUuid())) {
- throw new IllegalArgumentException(format("Can't reference portfolio '%s' in portfolio '%s' - hierarchy would be inconsistent",
- refPortfolio.getKey(), portfolio.getKey()));
- }
- portfolioDao.addReference(dbSession, portfolio.getUuid(), refPortfolio.getUuid());
- }
- }
-
- public void addReferenceToApplication(String portfolioKey, ProjectDto app) {
- // TODO
- }
-
- public void refresh(List<PortfolioDto> portfolios) {
- Set<String> rootUuids = portfolios.stream().map(PortfolioDto::getRootUuid).collect(Collectors.toSet());
- // TODO eliminate duplicates based on references
-
-
- }
-
- public void appendProject(String portfolioKey, String projectKey) {
- try (DbSession dbSession = dbClient.openSession(false)) {
- PortfolioDto portfolio = selectByKeyOrFail(dbSession, portfolioKey);
- ProjectDto project = projectDao.selectProjectByKey(dbSession, projectKey).orElseThrow(() -> new IllegalArgumentException(format("Project '%s' not found", projectKey)));
-
- String rootUuid = portfolio.getRootUuid();
- Set<String> allProjectUuids = portfolioDao.getAllProjectsInHierarchy(dbSession, rootUuid);
- if (allProjectUuids.contains(project.getUuid())) {
- // TODO specify which portfolio?
- throw new IllegalArgumentException("The project with key %s is already selected in a portfolio in the hierarchy");
- }
- portfolioDao.addProject(dbSession, portfolio.getUuid(), project.getUuid());
- }
- }
-
- public void removeProject(String portfolioKey, String projectKey) {
-
- }
-
- public void updateSelectionMode(String portfolioKey, String selectionMode, @Nullable String selectionExpression) {
- PortfolioDto.SelectionMode mode = PortfolioDto.SelectionMode.valueOf(selectionMode);
- try (DbSession dbSession = dbClient.openSession(false)) {
- PortfolioDto portfolio = selectByKeyOrFail(dbSession, portfolioKey);
-
- if (mode.equals(PortfolioDto.SelectionMode.REST)) {
- for (PortfolioDto p : portfolioDao.selectTree(dbSession, portfolio.getUuid())) {
- if (!p.getUuid().equals(portfolio.getUuid()) && mode.name().equals(portfolio.getSelectionMode())) {
- p.setSelectionMode(PortfolioDto.SelectionMode.NONE.name());
- p.setSelectionExpression(null);
- portfolioDao.update(dbSession, p);
- }
- }
- }
-
- portfolio.setSelectionMode(selectionMode);
- portfolio.setSelectionExpression(selectionExpression);
-
- if (!mode.equals(PortfolioDto.SelectionMode.MANUAL)) {
- portfolioDao.deleteProjects(dbSession, portfolio.getUuid());
- }
-
- portfolioDao.update(dbSession, portfolio);
- }
- }
-
- /**
- * Deletes a portfolio and all of it's children.
- * Also deletes references from/to the deleted portfolios and the projects manually assigned to them.
- */
- public void delete(String portfolioKey) {
- try (DbSession dbSession = dbClient.openSession(false)) {
- PortfolioDto portfolio = selectByKeyOrFail(dbSession, portfolioKey);
- Set<String> subTree = getChildrenRecursively(dbSession, portfolio.getUuid()).stream().map(PortfolioDto::getUuid).collect(Collectors.toSet());
- portfolioDao.deleteByUuids(dbSession, subTree);
- // TODO trigger refresh
- }
- }
-
- private PortfolioDto selectByKeyOrFail(DbSession dbSession, String portfolioKey) {
- return portfolioDao.selectByKey(dbSession, portfolioKey).orElseThrow(() -> new IllegalArgumentException(format("Portfolio '%s' not found", portfolioKey)));
- }
-
- /**
- * Gets all portfolios belonging to a subtree, given its root
- */
- private List<PortfolioDto> getChildrenRecursively(DbSession dbSession, String portfolioUuid) {
- Map<String, PortfolioDto> tree = portfolioDao.selectTree(dbSession, portfolioUuid).stream().collect(Collectors.toMap(PortfolioDto::getUuid, x -> x));
- Map<String, Set<String>> childrenMap = new HashMap<>();
-
- for (PortfolioDto dto : tree.values()) {
- if (dto.getParentUuid() != null) {
- childrenMap.computeIfAbsent(dto.getParentUuid(), x -> new HashSet<>()).add(dto.getUuid());
- }
- }
-
- List<PortfolioDto> subTree = new ArrayList<>();
- LinkedList<String> stack = new LinkedList<>();
- stack.add(portfolioUuid);
-
- while (!stack.isEmpty()) {
- Set<String> children = childrenMap.get(stack.removeFirst());
- for (String child : children) {
- subTree.add(tree.get(child));
- stack.add(child);
- }
- }
-
- return subTree;
- }
-
- /**
- * Returns the root UUIDs of all trees with which the tree with the given root uuid is connected through a reference.
- * The connection can be incoming or outgoing, and it can be direct or indirect (through other trees).
- *
- * As an example, let's consider we have the following hierarchies of portfolios:
- *
- * A D - - -> F
- * |\ | |\
- * B C - - - > E G H
- *
- * Where C references E and D references F.
- * All 3 tree roots are connected to all the other roots.
- *
- * getTreeReferenceConnections(A) would return [D,F]
- * getTreeReferenceConnections(D) would return [A,F]
- * getTreeReferenceConnections(F) would return [A,D]
- */
- private Set<String> getTreeReferenceConnections(DbSession dbSession, String treeRootUuid) {
- List<ReferenceDto> references = portfolioDao.selectAllReferencesToPortfolios(dbSession);
-
- Map<String, List<String>> rootConnections = new HashMap<>();
-
- for (ReferenceDto ref : references) {
- rootConnections.computeIfAbsent(ref.getSourceRootUuid(), x -> new ArrayList<>()).add(ref.getTargetRootUuid());
- rootConnections.computeIfAbsent(ref.getTargetRootUuid(), x -> new ArrayList<>()).add(ref.getSourceRootUuid());
- }
-
- LinkedList<String> queue = new LinkedList<>();
- Set<String> transitiveReferences = new HashSet<>();
-
- // add all direct references
- queue.addAll(rootConnections.getOrDefault(treeRootUuid, emptyList()));
-
- // resolve all transitive references
- while (!queue.isEmpty()) {
- String uuid = queue.remove();
- if (!transitiveReferences.contains(uuid)) {
- queue.addAll(rootConnections.getOrDefault(treeRootUuid, emptyList()));
- transitiveReferences.add(uuid);
- }
- }
-
- return transitiveReferences;
- }
-
-}
diff --git a/server/sonar-db-dao/src/main/resources/org/sonar/db/portfolio/PortfolioMapper.xml b/server/sonar-db-dao/src/main/resources/org/sonar/db/portfolio/PortfolioMapper.xml
index e2a2451f349..663f69e003f 100644
--- a/server/sonar-db-dao/src/main/resources/org/sonar/db/portfolio/PortfolioMapper.xml
+++ b/server/sonar-db-dao/src/main/resources/org/sonar/db/portfolio/PortfolioMapper.xml
@@ -16,6 +16,18 @@
p.updated_at as updatedAt
</sql>
+ <sql id="projectColumns">
+ p.uuid as uuid,
+ p.kee as kee,
+ p.qualifier as qualifier,
+ p.name as name,
+ p.description as description,
+ p.tags as tagsString,
+ p.private as isPrivate,
+ p.created_at as createdAt,
+ p.updated_at as updatedAt
+ </sql>
+
<select id="selectByUuid" parameterType="String" resultType="Portfolio">
SELECT
<include refid="portfolioColumns"/>
@@ -24,6 +36,17 @@
p.uuid=#{uuid,jdbcType=VARCHAR}
</select>
+ <select id="selectByUuids" parameterType="String" resultType="Portfolio">
+ SELECT
+ <include refid="portfolioColumns"/>
+ FROM portfolios p
+ where
+ p.uuid in
+ <foreach collection="uuids" open="(" close=")" item="uuid" separator=",">
+ #{uuid,jdbcType=VARCHAR}
+ </foreach>
+ </select>
+
<select id="selectByKey" parameterType="String" resultType="Portfolio">
SELECT
<include refid="portfolioColumns"/>
@@ -40,17 +63,22 @@
p.parent_uuid is null
</select>
- <select id="selectProjects" resultType="String">
- SELECT
- p.project_uuid
- FROM portfolio_projects p
- where
- p.portfolio_uuid=#{portfolioUuid,jdbcType=VARCHAR}
+ <select id="selectProjects" resultType="Project">
+ SELECT
+ <include refid="projectColumns"/>
+ FROM portfolio_projects pp
+ INNER JOIN projects p on p.uuid = pp.project_uuid
+ WHERE
+ pp.portfolio_uuid=#{portfolioUuid,jdbcType=VARCHAR}
</select>
- <select id="selectAllProjectsInHierarchy" resultType="String">
+ <select id="selectAllProjectsInHierarchy" resultType="org.sonar.db.portfolio.PortfolioProjectDto">
SELECT
- pp.project_uuid
+ pp.uuid,
+ pp.project_uuid as projectUuid,
+ pp.portfolio_uuid as portfolioUuid,
+ pp.project_uuid as projectUuid,
+ pp.created_at as createdAt
FROM portfolio_projects pp
INNER JOIN portfolios p
ON pp.portfolio_uuid = p.uuid
diff --git a/server/sonar-db-dao/src/schema/schema-sq.ddl b/server/sonar-db-dao/src/schema/schema-sq.ddl
index 84adb575198..2913cd77daf 100644
--- a/server/sonar-db-dao/src/schema/schema-sq.ddl
+++ b/server/sonar-db-dao/src/schema/schema-sq.ddl
@@ -580,7 +580,7 @@ ALTER TABLE "PORTFOLIO_REFERENCES" ADD CONSTRAINT "PK_PORTFOLIO_REFERENCES" PRIM
CREATE TABLE "PORTFOLIOS"(
"UUID" VARCHAR(40) NOT NULL,
- "KEE" VARCHAR(40) NOT NULL,
+ "KEE" VARCHAR(400) NOT NULL,
"NAME" VARCHAR(2000) NOT NULL,
"DESCRIPTION" VARCHAR(2000),
"ROOT_UUID" VARCHAR(40) NOT NULL,
diff --git a/server/sonar-db-dao/src/test/java/org/sonar/db/portfolio/PortfolioDaoTest.java b/server/sonar-db-dao/src/test/java/org/sonar/db/portfolio/PortfolioDaoTest.java
index bc07b4287e5..17cffd59a4a 100644
--- a/server/sonar-db-dao/src/test/java/org/sonar/db/portfolio/PortfolioDaoTest.java
+++ b/server/sonar-db-dao/src/test/java/org/sonar/db/portfolio/PortfolioDaoTest.java
@@ -108,11 +108,10 @@ public class PortfolioDaoTest {
@Test
public void deleteByUuids() {
createPortfolio("p1");
- createPortfolio("p2" );
+ createPortfolio("p2");
createPortfolio("p3");
createPortfolio("p4");
-
portfolioDao.addProject(db.getSession(), "p1", "proj1");
portfolioDao.addProject(db.getSession(), "p2", "proj1");
@@ -166,23 +165,29 @@ public class PortfolioDaoTest {
@Test
public void insert_and_select_projects() {
+ db.components().insertPrivateProject("project1");
+ db.components().insertPrivateProject("project2");
+
assertThat(portfolioDao.getProjects(db.getSession(), "portfolio1")).isEmpty();
portfolioDao.addProject(db.getSession(), "portfolio1", "project1");
portfolioDao.addProject(db.getSession(), "portfolio1", "project2");
portfolioDao.addProject(db.getSession(), "portfolio2", "project2");
db.commit();
- assertThat(portfolioDao.getProjects(db.getSession(), "portfolio1")).containsExactlyInAnyOrder("project1", "project2");
- assertThat(portfolioDao.getProjects(db.getSession(), "portfolio2")).containsExactlyInAnyOrder("project2");
+ assertThat(portfolioDao.getProjects(db.getSession(), "portfolio1")).extracting(ProjectDto::getUuid).containsExactlyInAnyOrder("project1", "project2");
+ assertThat(portfolioDao.getProjects(db.getSession(), "portfolio2")).extracting(ProjectDto::getUuid).containsExactlyInAnyOrder("project2");
assertThat(portfolioDao.getProjects(db.getSession(), "portfolio3")).isEmpty();
assertThat(db.countRowsOfTable("portfolio_projects")).isEqualTo(3);
assertThat(db.select(db.getSession(), "select created_at from portfolio_projects"))
.extracting(m -> m.values().iterator().next())
- .containsExactlyInAnyOrder(1L, 2L, 3L);
+ .containsExactlyInAnyOrder(3L, 4L, 5L);
}
@Test
public void delete_projects() {
+ db.components().insertPrivateProject("project1");
+ db.components().insertPrivateProject("project2");
+
assertThat(portfolioDao.getProjects(db.getSession(), "portfolio1")).isEmpty();
portfolioDao.addProject(db.getSession(), "portfolio1", "project1");
portfolioDao.addProject(db.getSession(), "portfolio1", "project2");
@@ -190,12 +195,17 @@ public class PortfolioDaoTest {
portfolioDao.deleteProjects(db.getSession(), "portfolio1");
assertThat(portfolioDao.getProjects(db.getSession(), "portfolio1")).isEmpty();
- assertThat(portfolioDao.getProjects(db.getSession(), "portfolio2")).containsExactlyInAnyOrder("project2");
-
+ assertThat(portfolioDao.getProjects(db.getSession(), "portfolio2")).extracting(ProjectDto::getUuid)
+ .containsExactlyInAnyOrder("project2");
}
@Test
public void getAllProjectsInHierarchy() {
+ db.components().insertPrivateProject("p1");
+ db.components().insertPrivateProject("p2");
+ db.components().insertPrivateProject("p3");
+ db.components().insertPrivateProject("p4");
+
createPortfolio("root", null, "root");
createPortfolio("child1", null, "root");
createPortfolio("child11", "child1", "root");
@@ -207,7 +217,7 @@ public class PortfolioDaoTest {
portfolioDao.addProject(db.getSession(), "child11", "p3");
portfolioDao.addProject(db.getSession(), "root2", "p4");
- assertThat(portfolioDao.getAllProjectsInHierarchy(db.getSession(), "root")).containsOnly("p1", "p2", "p3");
+ assertThat(portfolioDao.getAllProjectsInHierarchy(db.getSession(), "root").keySet()).containsExactly("p1", "p2", "p3");
assertThat(portfolioDao.getAllProjectsInHierarchy(db.getSession(), "nonexisting")).isEmpty();
}
diff --git a/server/sonar-db-dao/src/testFixtures/java/org/sonar/db/component/ComponentDbTester.java b/server/sonar-db-dao/src/testFixtures/java/org/sonar/db/component/ComponentDbTester.java
index 9901e834140..2f330626468 100644
--- a/server/sonar-db-dao/src/testFixtures/java/org/sonar/db/component/ComponentDbTester.java
+++ b/server/sonar-db-dao/src/testFixtures/java/org/sonar/db/component/ComponentDbTester.java
@@ -27,12 +27,14 @@ import org.sonar.api.utils.System2;
import org.sonar.db.DbClient;
import org.sonar.db.DbSession;
import org.sonar.db.DbTester;
+import org.sonar.db.portfolio.PortfolioDto;
import org.sonar.db.project.ProjectDto;
import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.base.Preconditions.checkState;
import static java.util.Arrays.asList;
import static org.sonar.db.component.BranchType.BRANCH;
+import static org.sonar.db.portfolio.PortfolioDto.SelectionMode.NONE;
public class ComponentDbTester {
private final DbTester db;
@@ -161,27 +163,67 @@ public class ComponentDbTester {
}
public final ComponentDto insertPublicPortfolio() {
- return insertPublicPortfolio(defaults());
+ return insertComponentAndPortfolio(ComponentTesting.newPortfolio().setPrivate(false), false, defaults(), defaults());
}
public final ComponentDto insertPublicPortfolio(String uuid, Consumer<ComponentDto> dtoPopulator) {
- return insertComponentImpl(ComponentTesting.newPortfolio(uuid).setPrivate(false), false, dtoPopulator);
+ return insertComponentAndPortfolio(ComponentTesting.newPortfolio(uuid).setPrivate(false), false, dtoPopulator, defaults());
}
public final ComponentDto insertPublicPortfolio(Consumer<ComponentDto> dtoPopulator) {
- return insertComponentImpl(ComponentTesting.newPortfolio().setPrivate(false), false, dtoPopulator);
+ return insertComponentAndPortfolio(ComponentTesting.newPortfolio().setPrivate(false), false, dtoPopulator, defaults());
+ }
+
+ public final ComponentDto insertPublicPortfolio(Consumer<ComponentDto> dtoPopulator, Consumer<PortfolioDto> portfolioPopulator) {
+ return insertComponentAndPortfolio(ComponentTesting.newPortfolio().setPrivate(false), false, dtoPopulator, portfolioPopulator);
+ }
+
+ public ComponentDto insertComponentAndPortfolio(ComponentDto componentDto, boolean isPrivate, Consumer<ComponentDto> componentPopulator,
+ Consumer<PortfolioDto> portfolioPopulator) {
+ insertComponentImpl(componentDto, isPrivate, componentPopulator);
+
+ PortfolioDto portfolioDto = toPortfolioDto(componentDto, System2.INSTANCE.now());
+ portfolioPopulator.accept(portfolioDto);
+ dbClient.portfolioDao().insert(dbSession, portfolioDto);
+ db.commit();
+ return componentDto;
}
public final ComponentDto insertPrivatePortfolio() {
- return insertComponentImpl(ComponentTesting.newPortfolio().setPrivate(true), true, defaults());
+ return insertComponentAndPortfolio(ComponentTesting.newPortfolio().setPrivate(true), true, defaults(), defaults());
}
public final ComponentDto insertPrivatePortfolio(String uuid, Consumer<ComponentDto> dtoPopulator) {
- return insertComponentImpl(ComponentTesting.newPortfolio(uuid).setPrivate(true), true, dtoPopulator);
+ return insertComponentAndPortfolio(ComponentTesting.newPortfolio(uuid).setPrivate(true), true, dtoPopulator, defaults());
}
public final ComponentDto insertPrivatePortfolio(Consumer<ComponentDto> dtoPopulator) {
- return insertComponentImpl(ComponentTesting.newPortfolio().setPrivate(true), true, dtoPopulator);
+ return insertComponentAndPortfolio(ComponentTesting.newPortfolio().setPrivate(true), true, dtoPopulator, defaults());
+ }
+
+ public final ComponentDto insertPrivatePortfolio(Consumer<ComponentDto> dtoPopulator, Consumer<PortfolioDto> portfolioPopulator) {
+ return insertComponentAndPortfolio(ComponentTesting.newPortfolio().setPrivate(true), true, dtoPopulator, portfolioPopulator);
+ }
+
+ public void addPortfolioProject(ComponentDto portfolio, String... projectUuids) {
+ for (String uuid : projectUuids) {
+ dbClient.portfolioDao().addProject(dbSession, portfolio.uuid(), uuid);
+ }
+ db.commit();
+ }
+
+ public void addPortfolioProject(ComponentDto portfolio, ComponentDto... projects) {
+ for (ComponentDto project : projects) {
+ dbClient.portfolioDao().addProject(dbSession, portfolio.uuid(), project.uuid());
+ }
+ db.commit();
+ }
+
+ public void addPortfolioProject(PortfolioDto portfolioDto, ProjectDto... projects) {
+ for (ProjectDto project : projects) {
+ dbClient.portfolioDao().addProject(dbSession, portfolioDto.getUuid(), project.getUuid());
+ }
+ db.commit();
}
public final ComponentDto insertPublicApplication() {
@@ -386,6 +428,19 @@ public class ComponentDbTester {
.setName(componentDto.name());
}
+ public static PortfolioDto toPortfolioDto(ComponentDto componentDto, long createTime) {
+ return new PortfolioDto()
+ .setUuid(componentDto.uuid())
+ .setKey(componentDto.getDbKey())
+ .setRootUuid(componentDto.projectUuid())
+ .setSelectionMode(NONE.name())
+ .setCreatedAt(createTime)
+ .setUpdatedAt(createTime)
+ .setPrivate(componentDto.isPrivate())
+ .setDescription(componentDto.description())
+ .setName(componentDto.name());
+ }
+
private static <T> Consumer<T> defaults() {
return t -> {
};
diff --git a/server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v91/CreatePortfoliosTable.java b/server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v91/CreatePortfoliosTable.java
index a25d3b1adcf..26d9035fcbc 100644
--- a/server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v91/CreatePortfoliosTable.java
+++ b/server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v91/CreatePortfoliosTable.java
@@ -40,7 +40,7 @@ public class CreatePortfoliosTable extends CreateTableChange {
public void execute(Context context, String tableName) throws SQLException {
context.execute(new CreateTableBuilder(getDialect(), tableName)
.addPkColumn(newVarcharColumnDefBuilder().setColumnName("uuid").setIsNullable(false).setLimit(UUID_SIZE).build())
- .addColumn(newVarcharColumnDefBuilder().setColumnName("kee").setIsNullable(false).setLimit(UUID_SIZE).build())
+ .addColumn(newVarcharColumnDefBuilder().setColumnName("kee").setIsNullable(false).setLimit(400).build())
.addColumn(newVarcharColumnDefBuilder().setColumnName("name").setIsNullable(false).setLimit(2000).build())
.addColumn(newVarcharColumnDefBuilder().setColumnName("description").setIsNullable(true).setLimit(2000).build())
.addColumn(newVarcharColumnDefBuilder().setColumnName("root_uuid").setIsNullable(false).setLimit(UUID_SIZE).build())
diff --git a/server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v91/DbVersion91.java b/server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v91/DbVersion91.java
index 45dc9309a74..6a000293a25 100644
--- a/server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v91/DbVersion91.java
+++ b/server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v91/DbVersion91.java
@@ -39,6 +39,7 @@ public class DbVersion91 implements DbVersion {
.add(6011, "Create 'portfolios' table", CreatePortfoliosTable.class)
.add(6012, "Create 'portfolio_references' table", CreatePortfolioReferencesTable.class)
.add(6013, "Create 'portfolio_projects' table", CreatePortfolioProjectsTable.class)
+ .add(6014, "Migrate portfolios to new tables", MigratePortfoliosToNewTables.class)
;
}
}
diff --git a/server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v91/MigratePortfoliosToNewTables.java b/server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v91/MigratePortfoliosToNewTables.java
new file mode 100644
index 00000000000..377f7261c28
--- /dev/null
+++ b/server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v91/MigratePortfoliosToNewTables.java
@@ -0,0 +1,581 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2021 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.v91;
+
+import com.google.common.collect.Sets;
+import java.io.IOException;
+import java.io.InputStream;
+import java.io.InputStreamReader;
+import java.io.StringReader;
+import java.sql.SQLException;
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.HashSet;
+import java.util.LinkedHashMap;
+import java.util.LinkedList;
+import java.util.List;
+import java.util.Map;
+import java.util.Optional;
+import java.util.Set;
+import java.util.TreeSet;
+import java.util.regex.Pattern;
+import java.util.stream.Collectors;
+import javax.annotation.CheckForNull;
+import javax.annotation.Nullable;
+import javax.xml.XMLConstants;
+import javax.xml.parsers.ParserConfigurationException;
+import javax.xml.parsers.SAXParser;
+import javax.xml.parsers.SAXParserFactory;
+import javax.xml.stream.XMLInputFactory;
+import javax.xml.stream.XMLStreamException;
+import javax.xml.transform.sax.SAXSource;
+import javax.xml.validation.SchemaFactory;
+import org.apache.commons.lang.StringUtils;
+import org.codehaus.staxmate.SMInputFactory;
+import org.codehaus.staxmate.in.SMHierarchicCursor;
+import org.codehaus.staxmate.in.SMInputCursor;
+import org.sonar.api.utils.System2;
+import org.sonar.core.util.UuidFactory;
+import org.sonar.db.Database;
+import org.sonar.db.DatabaseUtils;
+import org.sonar.server.platform.db.migration.step.DataChange;
+import org.sonar.server.platform.db.migration.step.MassUpdate;
+import org.sonar.server.platform.db.migration.step.Select;
+import org.sonar.server.platform.db.migration.step.Upsert;
+import org.sonar.server.platform.db.migration.version.v91.MigratePortfoliosToNewTables.ViewXml.ViewDef;
+import org.xml.sax.InputSource;
+import org.xml.sax.SAXException;
+import org.xml.sax.SAXParseException;
+import org.xml.sax.helpers.DefaultHandler;
+
+import static java.nio.charset.StandardCharsets.UTF_8;
+import static java.util.Optional.ofNullable;
+import static javax.xml.XMLConstants.FEATURE_SECURE_PROCESSING;
+import static org.apache.commons.io.IOUtils.toInputStream;
+import static org.apache.commons.lang.StringUtils.trim;
+
+public class MigratePortfoliosToNewTables extends DataChange {
+ private static final String SELECT_DEFAULT_VISIBILITY = "select text_value from properties where prop_key = 'projects.default.visibility'";
+ private static final String SELECT_UUID_VISIBILITY_BY_COMPONENT_KEY = "select c.uuid, c.private from components c where c.kee = ?";
+ private static final String SELECT_PORTFOLIO_UUID_AND_SELECTION_MODE_BY_KEY = "select uuid,selection_mode from portfolios where kee = ?";
+ private static final String SELECT_PROJECT_KEYS_BY_PORTFOLIO_UUID = "select p.kee from portfolio_projects pp "
+ + "join projects p on pp.project_uuid = p.uuid where pp.portfolio_uuid = ?";
+ private static final String SELECT_PROJECT_UUIDS_BY_KEYS = "select p.uuid,p.kee from projects p where p.kee in (PLACEHOLDER)";
+ private static final String VIEWS_DEF_KEY = "views.def";
+ private static final String PLACEHOLDER = "PLACEHOLDER";
+
+ private final UuidFactory uuidFactory;
+ private final System2 system;
+
+ private boolean defaultPrivateFlag;
+
+ public enum SelectionMode {
+ NONE, MANUAL, REGEXP, REST, TAGS
+ }
+
+ public MigratePortfoliosToNewTables(Database db, UuidFactory uuidFactory, System2 system) {
+ super(db);
+
+ this.uuidFactory = uuidFactory;
+ this.system = system;
+ }
+
+ @Override
+ protected void execute(Context context) throws SQLException {
+ String xml = getViewsDefinition(context);
+ // skip migration if `views.def` does not exist in the db
+ if (xml == null) {
+ return;
+ }
+
+ this.defaultPrivateFlag = ofNullable(context.prepareSelect(SELECT_DEFAULT_VISIBILITY)
+ .get(row -> "private".equals(row.getString(1))))
+ .orElse(false);
+
+ try {
+ Map<String, ViewXml.ViewDef> portfolioXmlMap = ViewXml.parse(xml);
+ List<ViewXml.ViewDef> portfolios = new LinkedList<>(portfolioXmlMap.values());
+
+ Map<String, PortfolioDb> portfolioDbMap = new HashMap<>();
+ for (ViewXml.ViewDef portfolio : portfolios) {
+ PortfolioDb createdPortfolio = insertPortfolio(context, portfolio);
+ if (createdPortfolio.selectionMode == SelectionMode.MANUAL) {
+ insertPortfolioProjects(context, portfolio, createdPortfolio);
+ }
+ portfolioDbMap.put(createdPortfolio.kee, createdPortfolio);
+ }
+ // all portfolio has been created and new uuids assigned
+ // update portfolio hierarchy parent/root
+ insertReferences(context, portfolioXmlMap, portfolioDbMap);
+ updateHierarchy(context, portfolioXmlMap, portfolioDbMap);
+ } catch (Exception e) {
+ throw new IllegalStateException("Failed to migrate views definitions property.", e);
+ }
+ }
+
+ private PortfolioDb insertPortfolio(Context context, ViewXml.ViewDef portfolioFromXml) throws SQLException {
+ long now = system.now();
+ PortfolioDb portfolioDb = context.prepareSelect(SELECT_PORTFOLIO_UUID_AND_SELECTION_MODE_BY_KEY)
+ .setString(1, portfolioFromXml.key)
+ .get(r -> new PortfolioDb(r.getString(1), portfolioFromXml.key, SelectionMode.valueOf(r.getString(2))));
+
+ Optional<ComponentDb> componentDbOpt = ofNullable(context.prepareSelect(SELECT_UUID_VISIBILITY_BY_COMPONENT_KEY)
+ .setString(1, portfolioFromXml.key)
+ .get(row -> new ComponentDb(row.getString(1), row.getBoolean(2))));
+
+ // no portfolio -> insert
+ if (portfolioDb == null) {
+ Upsert insertPortfolioQuery = context.prepareUpsert("insert into " +
+ "portfolios(uuid, kee, private, name, description, root_uuid, parent_uuid, selection_mode, selection_expression, updated_at, created_at) " +
+ "values (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)");
+
+ String portfolioUuid = componentDbOpt.map(c -> c.uuid).orElse(uuidFactory.create());
+ insertPortfolioQuery.setString(1, portfolioUuid)
+ .setString(2, portfolioFromXml.key)
+ .setBoolean(3, componentDbOpt.map(c -> c.visibility).orElse(this.defaultPrivateFlag))
+ .setString(4, portfolioFromXml.name)
+ .setString(5, portfolioFromXml.desc)
+ .setString(6, PLACEHOLDER)
+ .setString(7, PLACEHOLDER);
+ SelectionMode selectionMode = SelectionMode.NONE;
+ String selectionExpression = null;
+ if (portfolioFromXml.getProjects() != null && !portfolioFromXml.getProjects().isEmpty()) {
+ selectionMode = SelectionMode.MANUAL;
+ } else if (portfolioFromXml.regexp != null && !portfolioFromXml.regexp.isBlank()) {
+ selectionMode = SelectionMode.REGEXP;
+ selectionExpression = portfolioFromXml.regexp;
+ } else if (portfolioFromXml.def) {
+ selectionMode = SelectionMode.REST;
+ } else if (portfolioFromXml.tagsAssociation != null && !portfolioFromXml.tagsAssociation.isEmpty()) {
+ selectionMode = SelectionMode.TAGS;
+ selectionExpression = String.join(",", portfolioFromXml.tagsAssociation);
+ }
+
+ insertPortfolioQuery.setString(8, selectionMode.name())
+ .setString(9, selectionExpression)
+ // set dates
+ .setLong(10, now)
+ .setLong(11, now)
+ .execute()
+ .commit();
+ return new PortfolioDb(portfolioUuid, portfolioFromXml.key, selectionMode);
+ }
+ return portfolioDb;
+ }
+
+ private void insertPortfolioProjects(Context context, ViewDef portfolio, PortfolioDb createdPortfolio) throws SQLException {
+ long now = system.now();
+ // select all already added project uuids
+ Set<String> alreadyAddedPortfolioProjects = new HashSet<>(
+ context.prepareSelect(SELECT_PROJECT_KEYS_BY_PORTFOLIO_UUID)
+ .setString(1, createdPortfolio.uuid)
+ .list(r -> r.getString(1)));
+
+ Set<String> projectKeysFromXml = new HashSet<>(portfolio.getProjects());
+ Set<String> projectKeysToBeAdded = Sets.difference(projectKeysFromXml, alreadyAddedPortfolioProjects);
+
+ if (!projectKeysToBeAdded.isEmpty()) {
+ List<ProjectDb> projects = findPortfolioProjects(context, projectKeysToBeAdded);
+
+ var upsert = context.prepareUpsert("insert into " +
+ "portfolio_projects(uuid, portfolio_uuid, project_uuid, created_at) " +
+ "values (?, ?, ?, ?)");
+ for (ProjectDb projectDb : projects) {
+ upsert.setString(1, uuidFactory.create())
+ .setString(2, createdPortfolio.uuid)
+ .setString(3, projectDb.uuid)
+ .setLong(4, now)
+ .addBatch();
+ }
+ if (!projects.isEmpty()) {
+ upsert.execute()
+ .commit();
+ }
+ }
+ }
+
+ private static List<ProjectDb> findPortfolioProjects(Context context, Set<String> projectKeysToBeAdded) {
+ return DatabaseUtils.executeLargeInputs(projectKeysToBeAdded, keys -> {
+ try {
+ String selectQuery = SELECT_PROJECT_UUIDS_BY_KEYS.replace(PLACEHOLDER,
+ keys.stream().map(key -> "'" + key + "'").collect(
+ Collectors.joining(",")));
+ return context.prepareSelect(selectQuery)
+ .list(r -> new ProjectDb(r.getString(1), r.getString(2)));
+ } catch (SQLException e) {
+ throw new IllegalStateException("Could not execute 'in' query", e);
+ }
+ });
+ }
+
+ private void insertReferences(Context context, Map<String, ViewDef> portfolioXmlMap,
+ Map<String, PortfolioDb> portfolioDbMap) throws SQLException {
+ Upsert insertQuery = context.prepareUpsert("insert into portfolio_references(uuid, portfolio_uuid, reference_uuid, created_at) values (?, ?, ?, ?)");
+
+ long now = system.now();
+ boolean shouldExecuteQuery = false;
+ for (ViewDef portfolio : portfolioXmlMap.values()) {
+ var currentPortfolioUuid = portfolioDbMap.get(portfolio.key).uuid;
+ Set<String> referencesFromXml = new HashSet<>(portfolio.getReferences());
+ Set<String> referencesFromDb = new HashSet<>(context.prepareSelect("select pr.reference_uuid from portfolio_references pr where pr.portfolio_uuid = ?")
+ .setString(1, currentPortfolioUuid)
+ .list(row -> row.getString(1)));
+
+ for (String appOrPortfolio : referencesFromXml) {
+ // if portfolio and hasn't been added already
+ if (portfolioDbMap.containsKey(appOrPortfolio) && !referencesFromDb.contains(portfolioDbMap.get(appOrPortfolio).uuid)) {
+ insertQuery
+ .setString(1, uuidFactory.create())
+ .setString(2, currentPortfolioUuid)
+ .setString(3, portfolioDbMap.get(appOrPortfolio).uuid)
+ .setLong(4, now)
+ .addBatch();
+ shouldExecuteQuery = true;
+ } else {
+ // if application exist and haven't been added
+ String appUuid = context.prepareSelect("select p.uuid from projects p where p.kee = ?")
+ .setString(1, appOrPortfolio)
+ .get(row -> row.getString(1));
+ if (appUuid != null && !referencesFromDb.contains(appUuid)) {
+ insertQuery
+ .setString(1, uuidFactory.create())
+ .setString(2, currentPortfolioUuid)
+ .setString(3, appUuid)
+ .setLong(4, now)
+ .addBatch();
+ shouldExecuteQuery = true;
+ }
+ }
+ }
+ }
+ if (shouldExecuteQuery) {
+ insertQuery
+ .execute()
+ .commit();
+ }
+
+ }
+
+ private static void updateHierarchy(Context context, Map<String, ViewXml.ViewDef> defs, Map<String, PortfolioDb> portfolioDbMap) throws SQLException {
+ MassUpdate massUpdate = context.prepareMassUpdate();
+ massUpdate.select("select uuid, kee from portfolios where root_uuid = ? or parent_uuid = ?")
+ .setString(1, PLACEHOLDER)
+ .setString(2, PLACEHOLDER);
+ massUpdate.update("update portfolios set root_uuid = ?, parent_uuid = ? where uuid = ?");
+ massUpdate.execute((row, update) -> {
+ String currentPortfolioUuid = row.getString(1);
+ String currentPortfolioKey = row.getString(2);
+
+ var currentPortfolio = defs.get(currentPortfolioKey);
+ String parentUuid = ofNullable(currentPortfolio.parent).map(parent -> portfolioDbMap.get(parent).uuid).orElse(null);
+ String rootUuid = ofNullable(currentPortfolio.root).map(root -> portfolioDbMap.get(root).uuid).orElse(currentPortfolioUuid);
+ update.setString(1, rootUuid)
+ .setString(2, parentUuid)
+ .setString(3, currentPortfolioUuid);
+ return true;
+ });
+ }
+
+ @CheckForNull
+ private static String getViewsDefinition(DataChange.Context context) throws SQLException {
+ Select select = context.prepareSelect("select text_value,clob_value from internal_properties where kee=?");
+ select.setString(1, VIEWS_DEF_KEY);
+ return select.get(row -> {
+ String v = row.getString(1);
+ if (v != null) {
+ return v;
+ } else {
+ return row.getString(2);
+ }
+ });
+ }
+
+ private static class ComponentDb {
+ String uuid;
+ boolean visibility;
+
+ public ComponentDb(String uuid, boolean visibility) {
+ this.uuid = uuid;
+ this.visibility = visibility;
+ }
+ }
+
+ private static class PortfolioDb {
+ String uuid;
+ String kee;
+ SelectionMode selectionMode;
+
+ PortfolioDb(String uuid, String kee,
+ SelectionMode selectionMode) {
+ this.uuid = uuid;
+ this.kee = kee;
+ this.selectionMode = selectionMode;
+ }
+ }
+
+ private static class ProjectDb {
+ String uuid;
+ String kee;
+
+ ProjectDb(String uuid, String kee) {
+ this.uuid = uuid;
+ this.kee = kee;
+ }
+ }
+
+ static class ViewXml {
+ static final String SCHEMA_VIEWS = "/static/views.xsd";
+ static final String VIEWS_HEADER_BARE = "<views>";
+ static final Pattern VIEWS_HEADER_BARE_PATTERN = Pattern.compile(VIEWS_HEADER_BARE);
+ static final String VIEWS_HEADER_FQ = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>"
+ + "<views xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\" xmlns=\"http://sonarsource.com/schema/views\">";
+
+ private ViewXml() {
+ // nothing to do here
+ }
+
+ private static Map<String, ViewDef> parse(String xml) throws ParserConfigurationException, SAXException, IOException, XMLStreamException {
+ if (StringUtils.isEmpty(xml)) {
+ return new LinkedHashMap<>(0);
+ }
+
+ List<ViewDef> views;
+ validate(xml);
+ SMInputFactory inputFactory = initStax();
+ SMHierarchicCursor rootC = inputFactory.rootElementCursor(new StringReader(xml));
+ rootC.advance(); // <views>
+ SMInputCursor cursor = rootC.childElementCursor();
+ views = parseViewDefinitions(cursor);
+
+ Map<String, ViewDef> result = new LinkedHashMap<>(views.size());
+ for (ViewDef def : views) {
+ result.put(def.key, def);
+ }
+
+ return result;
+ }
+
+ private static void validate(String xml) throws IOException, SAXException, ParserConfigurationException {
+ // Replace bare, namespace unaware header with fully qualified header (with schema declaration)
+ String fullyQualifiedXml = VIEWS_HEADER_BARE_PATTERN.matcher(xml).replaceFirst(VIEWS_HEADER_FQ);
+ try (InputStream xsd = MigratePortfoliosToNewTables.class.getResourceAsStream(SCHEMA_VIEWS)) {
+ InputSource viewsDefinition = new InputSource(new InputStreamReader(toInputStream(fullyQualifiedXml, UTF_8), UTF_8));
+
+ SchemaFactory saxSchemaFactory = SchemaFactory.newInstance(XMLConstants.W3C_XML_SCHEMA_NS_URI);
+ saxSchemaFactory.setProperty(XMLConstants.ACCESS_EXTERNAL_SCHEMA, "");
+ saxSchemaFactory.setProperty(XMLConstants.ACCESS_EXTERNAL_DTD, "");
+
+ SAXParserFactory parserFactory = SAXParserFactory.newInstance();
+ parserFactory.setFeature(FEATURE_SECURE_PROCESSING, true);
+ parserFactory.setNamespaceAware(true);
+ parserFactory.setSchema(saxSchemaFactory.newSchema(new SAXSource(new InputSource(xsd))));
+
+ SAXParser saxParser = parserFactory.newSAXParser();
+ saxParser.setProperty(XMLConstants.ACCESS_EXTERNAL_DTD, "");
+ saxParser.setProperty(XMLConstants.ACCESS_EXTERNAL_SCHEMA, "");
+ saxParser.parse(viewsDefinition, new ViewsValidator());
+ }
+ }
+
+ private static List<ViewDef> parseViewDefinitions(SMInputCursor viewsCursor) throws XMLStreamException {
+ List<ViewDef> views = new ArrayList<>();
+ while (viewsCursor.getNext() != null) {
+ ViewDef viewDef = new ViewDef();
+ viewDef.setKey(viewsCursor.getAttrValue("key"));
+ viewDef.setDef(Boolean.parseBoolean(viewsCursor.getAttrValue("def")));
+ viewDef.setParent(viewsCursor.getAttrValue("parent"));
+ viewDef.setRoot(viewsCursor.getAttrValue("root"));
+ SMInputCursor viewCursor = viewsCursor.childElementCursor();
+ while (viewCursor.getNext() != null) {
+ String nodeName = viewCursor.getLocalName();
+ parseChildElement(viewDef, viewCursor, nodeName);
+ }
+ views.add(viewDef);
+ }
+ return views;
+ }
+
+ private static void parseChildElement(ViewDef viewDef, SMInputCursor viewCursor, String nodeName) throws XMLStreamException {
+ if (StringUtils.equals(nodeName, "name")) {
+ viewDef.setName(trim(viewCursor.collectDescendantText()));
+ } else if (StringUtils.equals(nodeName, "desc")) {
+ viewDef.setDesc(trim(viewCursor.collectDescendantText()));
+ } else if (StringUtils.equals(nodeName, "regexp")) {
+ viewDef.setRegexp(trim(viewCursor.collectDescendantText()));
+ } else if (StringUtils.equals(nodeName, "language")) {
+ viewDef.setLanguage(trim(viewCursor.collectDescendantText()));
+ } else if (StringUtils.equals(nodeName, "tag_key")) {
+ viewDef.setTagKey(trim(viewCursor.collectDescendantText()));
+ } else if (StringUtils.equals(nodeName, "tag_value")) {
+ viewDef.setTagValue(trim(viewCursor.collectDescendantText()));
+ } else if (StringUtils.equals(nodeName, "p")) {
+ viewDef.addProject(trim(viewCursor.collectDescendantText()));
+ } else if (StringUtils.equals(nodeName, "vw-ref")) {
+ viewDef.addReference(trim(viewCursor.collectDescendantText()));
+ } else if (StringUtils.equals(nodeName, "qualifier")) {
+ viewDef.setQualifier(trim(viewCursor.collectDescendantText()));
+ } else if (StringUtils.equals(nodeName, "tagsAssociation")) {
+ parseTagsAssociation(viewDef, viewCursor);
+ }
+ }
+
+ private static void parseTagsAssociation(ViewDef def, SMInputCursor viewCursor) throws XMLStreamException {
+ SMInputCursor projectCursor = viewCursor.childElementCursor();
+ while (projectCursor.getNext() != null) {
+ def.addTagAssociation(trim(projectCursor.collectDescendantText()));
+ }
+ }
+
+ private static SMInputFactory initStax() {
+ XMLInputFactory xmlFactory = XMLInputFactory.newInstance();
+ xmlFactory.setProperty(XMLInputFactory.IS_COALESCING, Boolean.TRUE);
+ xmlFactory.setProperty(XMLInputFactory.IS_NAMESPACE_AWARE, Boolean.FALSE);
+ // just so it won't try to load DTD in if there's DOCTYPE
+ xmlFactory.setProperty(XMLInputFactory.SUPPORT_DTD, Boolean.FALSE);
+ xmlFactory.setProperty(XMLInputFactory.IS_VALIDATING, Boolean.FALSE);
+ return new SMInputFactory(xmlFactory);
+ }
+
+ static class ViewDef {
+ String key = null;
+
+ String parent = null;
+
+ String root = null;
+
+ boolean def = false;
+
+ List<String> p = new ArrayList<>();
+
+ List<String> vwRef = new ArrayList<>();
+
+ String name = null;
+
+ String desc = null;
+
+ String regexp = null;
+
+ String language = null;
+
+ String tagKey = null;
+
+ String tagValue = null;
+
+ String qualifier = null;
+
+ Set<String> tagsAssociation = new TreeSet<>();
+
+ public List<String> getProjects() {
+ return p;
+ }
+
+ public List<String> getReferences() {
+ return vwRef;
+ }
+
+ public ViewDef setKey(String key) {
+ this.key = key;
+ return this;
+ }
+
+ public ViewDef setParent(String parent) {
+ this.parent = parent;
+ return this;
+ }
+
+ public ViewDef setRoot(@Nullable String root) {
+ this.root = root;
+ return this;
+ }
+
+ public ViewDef setDef(boolean def) {
+ this.def = def;
+ return this;
+ }
+
+ public ViewDef addProject(String project) {
+ this.p.add(project);
+ return this;
+ }
+
+ public ViewDef setName(String name) {
+ this.name = name;
+ return this;
+ }
+
+ public ViewDef setDesc(@Nullable String desc) {
+ this.desc = desc;
+ return this;
+ }
+
+ public ViewDef setRegexp(@Nullable String regexp) {
+ this.regexp = regexp;
+ return this;
+ }
+
+ public ViewDef setLanguage(@Nullable String language) {
+ this.language = language;
+ return this;
+ }
+
+ public ViewDef setTagKey(@Nullable String tagKey) {
+ this.tagKey = tagKey;
+ return this;
+ }
+
+ public ViewDef setTagValue(@Nullable String tagValue) {
+ this.tagValue = tagValue;
+ return this;
+ }
+
+ public ViewDef addReference(String reference) {
+ this.vwRef.add(reference);
+ return this;
+ }
+
+ public ViewDef setQualifier(@Nullable String qualifier) {
+ this.qualifier = qualifier;
+ return this;
+ }
+
+ public ViewDef addTagAssociation(String tag) {
+ this.tagsAssociation.add(tag);
+ return this;
+ }
+ }
+
+ private static final class ViewsValidator extends DefaultHandler {
+ @Override
+ public void error(SAXParseException exception) throws SAXException {
+ throw exception;
+ }
+
+ @Override
+ public void warning(SAXParseException exception) throws SAXException {
+ throw exception;
+ }
+
+ @Override
+ public void fatalError(SAXParseException exception) throws SAXException {
+ throw exception;
+ }
+ }
+ }
+}
diff --git a/server/sonar-db-migration/src/test/java/org/sonar/server/platform/db/migration/version/v91/DbVersion91Test.java b/server/sonar-db-migration/src/test/java/org/sonar/server/platform/db/migration/version/v91/DbVersion91Test.java
index e5b51a680b6..0a361304f1b 100644
--- a/server/sonar-db-migration/src/test/java/org/sonar/server/platform/db/migration/version/v91/DbVersion91Test.java
+++ b/server/sonar-db-migration/src/test/java/org/sonar/server/platform/db/migration/version/v91/DbVersion91Test.java
@@ -41,7 +41,7 @@ public class DbVersion91Test {
@Test
public void verify_migration_count() {
- verifyMigrationCount(underTest, 13);
+ verifyMigrationCount(underTest, 14);
}
}
diff --git a/server/sonar-db-migration/src/test/java/org/sonar/server/platform/db/migration/version/v91/MigratePortfoliosToNewTablesTest.java b/server/sonar-db-migration/src/test/java/org/sonar/server/platform/db/migration/version/v91/MigratePortfoliosToNewTablesTest.java
new file mode 100644
index 00000000000..e8c8e48d9be
--- /dev/null
+++ b/server/sonar-db-migration/src/test/java/org/sonar/server/platform/db/migration/version/v91/MigratePortfoliosToNewTablesTest.java
@@ -0,0 +1,431 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2021 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.v91;
+
+import java.sql.SQLException;
+import java.util.stream.Collectors;
+import javax.annotation.Nullable;
+import org.junit.Rule;
+import org.junit.Test;
+import org.sonar.api.impl.utils.TestSystem2;
+import org.sonar.api.resources.Qualifiers;
+import org.sonar.api.utils.System2;
+import org.sonar.core.util.UuidFactory;
+import org.sonar.core.util.UuidFactoryFast;
+import org.sonar.db.CoreDbTester;
+import org.sonar.server.platform.db.migration.step.DataChange;
+
+import static org.assertj.core.api.Assertions.assertThat;
+import static org.assertj.core.api.Assertions.assertThatCode;
+import static org.assertj.core.api.Assertions.tuple;
+
+public class MigratePortfoliosToNewTablesTest {
+ static final int TEXT_VALUE_MAX_LENGTH = 4000;
+ private static final long NOW = 10_000_000L;
+
+ @Rule
+ public final CoreDbTester db = CoreDbTester.createForSchema(MigratePortfoliosToNewTablesTest.class, "schema.sql");
+
+ private final UuidFactory uuidFactory = UuidFactoryFast.getInstance();
+ private final System2 system2 = new TestSystem2().setNow(NOW);
+
+ private final DataChange underTest = new MigratePortfoliosToNewTables(db.database(), uuidFactory, system2);
+
+ private final String SIMPLE_XML_ONE_PORTFOLIO = "<views>" +
+ "<vw key=\"port1\" def=\"false\">\n" +
+ " <name><![CDATA[name]]></name>\n" +
+ " <desc><![CDATA[description]]></desc>\n" +
+ " <qualifier><![CDATA[VW]]></qualifier>\n" +
+ " </vw>\n" +
+ "<vw key=\"port2\" def=\"false\">\n" +
+ " <name><![CDATA[name2]]></name>\n" +
+ " <desc><![CDATA[description2]]></desc>\n" +
+ " <qualifier><![CDATA[VW]]></qualifier>\n" +
+ " </vw>\n" +
+ "</views>";
+
+ private final String SIMPLE_XML_ONLY_PORTFOLIOS = "<views>" +
+ "<vw key=\"port1\" def=\"true\">\n" +
+ " <name><![CDATA[port1]]></name>\n" +
+ " <desc><![CDATA[]]></desc>\n" +
+ " <qualifier><![CDATA[VW]]></qualifier>\n" +
+ " </vw>\n" +
+ "<vw key=\"port3\" def=\"false\">\n" +
+ " <name><![CDATA[port3]]></name>\n" +
+ " <desc><![CDATA[]]></desc>\n" +
+ " <qualifier><![CDATA[VW]]></qualifier>\n" +
+ " </vw>\n" +
+ "<vw key=\"port2\" def=\"false\">\n" +
+ " <name><![CDATA[port2]]></name>\n" +
+ " <desc><![CDATA[]]></desc>\n" +
+ " <qualifier><![CDATA[VW]]></qualifier>\n" +
+ " <vw-ref><![CDATA[port1]]></vw-ref>\n" +
+ " <vw-ref><![CDATA[port3]]></vw-ref>\n" +
+ " </vw>\n" +
+ "</views>";
+
+ private final String SIMPLE_XML_PORTFOLIOS_AND_APP = "<views>" +
+ "<vw key=\"port1\" def=\"true\">\n" +
+ " <name><![CDATA[port1]]></name>\n" +
+ " <desc><![CDATA[]]></desc>\n" +
+ " <qualifier><![CDATA[VW]]></qualifier>\n" +
+ " </vw>\n" +
+ "<vw key=\"port3\" def=\"false\">\n" +
+ " <name><![CDATA[port3]]></name>\n" +
+ " <desc><![CDATA[]]></desc>\n" +
+ " <qualifier><![CDATA[VW]]></qualifier>\n" +
+ " </vw>\n" +
+ "<vw key=\"port2\" def=\"false\">\n" +
+ " <name><![CDATA[port2]]></name>\n" +
+ " <desc><![CDATA[]]></desc>\n" +
+ " <qualifier><![CDATA[VW]]></qualifier>\n" +
+ " <vw-ref><![CDATA[port1]]></vw-ref>\n" +
+ " <vw-ref><![CDATA[port3]]></vw-ref>\n" +
+ " </vw>\n" +
+ "<vw key=\"port4\" def=\"false\">\n" +
+ " <name><![CDATA[port2]]></name>\n" +
+ " <desc><![CDATA[]]></desc>\n" +
+ " <qualifier><![CDATA[VW]]></qualifier>\n" +
+ " <vw-ref><![CDATA[port2]]></vw-ref>\n" +
+ " <vw-ref><![CDATA[app1-key]]></vw-ref>\n" +
+ " <vw-ref><![CDATA[app2-key]]></vw-ref>\n" +
+ " <vw-ref><![CDATA[app3-key-not-existing]]></vw-ref>\n" +
+ " </vw>\n" +
+ "</views>";
+
+ private final String SIMPLE_XML_PORTFOLIOS_HIERARCHY = "<views>" +
+ " <vw key=\"port1\" def=\"false\">\n"
+ + " <name><![CDATA[port1]]></name>\n"
+ + " <desc><![CDATA[port1]]></desc>\n"
+ + " <qualifier><![CDATA[VW]]></qualifier>\n"
+ + " </vw>\n"
+ + " <vw key=\"port2\" def=\"false\" root=\"port1\" parent=\"port1\">\n"
+ + " <name><![CDATA[port2]]></name>\n"
+ + " <desc><![CDATA[port2]]></desc>\n"
+ + " </vw>\n"
+ + " <vw key=\"port3\" def=\"false\" root=\"port1\" parent=\"port1\">\n"
+ + " <name><![CDATA[port3]]></name>\n"
+ + " <desc><![CDATA[port3]]></desc>\n"
+ + " </vw>\n"
+ + " <vw key=\"port4\" def=\"false\" root=\"port1\" parent=\"port1\">\n"
+ + " <name><![CDATA[port4]]></name>\n"
+ + " <desc><![CDATA[port4]]></desc>\n"
+ + " </vw>\n"
+ + " <vw key=\"port5\" def=\"false\" root=\"port1\" parent=\"port2\">\n"
+ + " <name><![CDATA[port5]]></name>\n"
+ + " </vw>\n"
+ + " <vw key=\"port6\" def=\"false\" root=\"port1\" parent=\"port3\">\n"
+ + " <name><![CDATA[port6]]></name>\n"
+ + " <desc><![CDATA[port6]]></desc>\n"
+ + " </vw>\n"
+ + " <vw key=\"port7\" def=\"false\" root=\"port1\" parent=\"port3\">\n"
+ + " <name><![CDATA[port7]]></name>\n"
+ + " <desc><![CDATA[port7]]></desc>\n"
+ + " </vw>" +
+ "</views>";
+
+ private final String SIMPLE_XML_PORTFOLIOS_DIFFERENT_SELECTIONS = "<views>"
+ + " <vw key=\"port-regexp\" def=\"false\">\n"
+ + " <name><![CDATA[port-regexp]]></name>\n"
+ + " <desc><![CDATA[port-regexp]]></desc>\n"
+ + " <regexp><![CDATA[.*port.*]]></regexp>\n"
+ + " <qualifier><![CDATA[VW]]></qualifier>\n"
+ + " </vw>"
+ + " <vw key=\"port-tags\" def=\"false\">\n"
+ + " <name><![CDATA[port-tags]]></name>\n"
+ + " <desc><![CDATA[port-tags]]></desc>\n"
+ + " <qualifier><![CDATA[VW]]></qualifier>\n"
+ + " <tagsAssociation>\n"
+ + " <tag>tag1</tag>\n"
+ + " <tag>tag2</tag>\n"
+ + " <tag>tag3</tag>\n"
+ + " </tagsAssociation>\n"
+ + " </vw>"
+ + " <vw key=\"port-projects\" def=\"false\">\n"
+ + " <name><![CDATA[port-projects]]></name>\n"
+ + " <desc><![CDATA[port-projects]]></desc>\n"
+ + " <qualifier><![CDATA[VW]]></qualifier>\n"
+ + " <p>p1-key</p>\n"
+ + " <p>p2-key</p>\n"
+ + " <p>p3-key</p>\n"
+ + " <p>p4-key-not-existing</p>\n"
+ + " </vw>"
+ + "</views>";
+
+ private final String XML_PORTFOLIOS_LEGACY_SELECTIONS = "<views>"
+ + " <vw key=\"port-language\" def=\"false\">\n"
+ + " <name><![CDATA[port-language]]></name>\n"
+ + " <desc><![CDATA[port-language]]></desc>\n"
+ + " <language><![CDATA[javascript]]></language>\n"
+ + " <qualifier><![CDATA[VW]]></qualifier>\n"
+ + " </vw>"
+ + " <vw key=\"port-tag-value\" def=\"false\">\n"
+ + " <name><![CDATA[port-tag-value]]></name>\n"
+ + " <desc><![CDATA[port-tag-value]]></desc>\n"
+ + " <qualifier><![CDATA[VW]]></qualifier>\n"
+ + " <tag_key>tag-key</tag_key>\n"
+ + " <tag_value>tag-value</tag_value>\n"
+ + " </vw>"
+ + "</views>";
+
+ @Test
+ public void does_not_fail_when_nothing_to_migrate() throws SQLException {
+ assertThatCode(underTest::execute)
+ .doesNotThrowAnyException();
+ }
+
+ @Test
+ public void migrate_single_portfolio_with_default_visibility() throws SQLException {
+ insertViewsDefInternalProperty(SIMPLE_XML_ONE_PORTFOLIO);
+ insertDefaultVisibilityProperty(true);
+ insertComponent("uuid", "port2", false);
+
+ underTest.execute();
+ // re-entrant
+ underTest.execute();
+
+ assertThat(db.select("select kee, private, name, description, "
+ + "selection_mode, selection_expression, "
+ + "updated_at, created_at from portfolios"))
+ .extracting(
+ row -> row.get("KEE"),
+ row -> row.get("PRIVATE"),
+ row -> row.get("NAME"),
+ row -> row.get("DESCRIPTION"),
+ row -> row.get("SELECTION_MODE"),
+ row -> row.get("SELECTION_EXPRESSION"),
+ row -> row.get("UPDATED_AT"),
+ row -> row.get("CREATED_AT"))
+ .containsExactlyInAnyOrder(
+ tuple("port1", true, "name", "description", "NONE", null, NOW, NOW),
+ tuple("port2", false, "name2", "description2", "NONE", null, NOW, NOW));
+ }
+
+ @Test
+ public void migrate_portfolios_should_assign_component_uuid() throws SQLException {
+ insertViewsDefInternalProperty(SIMPLE_XML_ONE_PORTFOLIO);
+ insertComponent("uuid1", "port1", true);
+ insertComponent("uuid2", "port2", false);
+
+ underTest.execute();
+ // re-entrant
+ underTest.execute();
+
+ assertThat(db.select("select uuid, kee, private, name, description, "
+ + "selection_mode, selection_expression, "
+ + "updated_at, created_at from portfolios"))
+ .extracting(
+ row -> row.get("UUID"),
+ row -> row.get("KEE"),
+ row -> row.get("PRIVATE"),
+ row -> row.get("NAME"),
+ row -> row.get("DESCRIPTION"),
+ row -> row.get("SELECTION_MODE"),
+ row -> row.get("SELECTION_EXPRESSION"),
+ row -> row.get("UPDATED_AT"),
+ row -> row.get("CREATED_AT"))
+ .containsExactlyInAnyOrder(
+ tuple("uuid1", "port1", true, "name", "description", "NONE", null, NOW, NOW),
+ tuple("uuid2", "port2", false, "name2", "description2", "NONE", null, NOW, NOW));
+ }
+
+ @Test
+ public void migrate_simple_xml_only_portfolios_references() throws SQLException {
+ insertViewsDefInternalProperty(SIMPLE_XML_ONLY_PORTFOLIOS);
+
+ underTest.execute();
+ // re-entrant
+ underTest.execute();
+
+ assertThat(db.select("select kee from portfolios"))
+ .extracting(row -> row.get("KEE"))
+ .containsExactlyInAnyOrder("port1", "port2", "port3");
+
+ var portfolioKeyByUuid = db.select("select uuid, kee from portfolios").stream()
+ .collect(Collectors.toMap(row -> row.get("KEE"), row -> row.get("UUID").toString()));
+
+ String portfolio1Uuid = portfolioKeyByUuid.get("port1");
+ String portfolio2Uuid = portfolioKeyByUuid.get("port2");
+ String portfolio3Uuid = portfolioKeyByUuid.get("port3");
+
+ assertThat(db.select("select portfolio_uuid, reference_uuid from portfolio_references"))
+ .extracting(row -> row.get("PORTFOLIO_UUID"), row -> row.get("REFERENCE_UUID"))
+ .containsExactlyInAnyOrder(tuple(portfolio2Uuid, portfolio1Uuid), tuple(portfolio2Uuid, portfolio3Uuid));
+ }
+
+ @Test
+ public void migrate_simple_xml_portfolios_and_apps_references() throws SQLException {
+ insertViewsDefInternalProperty(SIMPLE_XML_PORTFOLIOS_AND_APP);
+
+ insertProject("app1-uuid", "app1-key", Qualifiers.APP);
+ insertProject("app2-uuid", "app2-key", Qualifiers.APP);
+ insertProject("proj1", "proj1-key", Qualifiers.PROJECT);
+
+ underTest.execute();
+ // re-entrant
+ underTest.execute();
+
+ assertThat(db.select("select kee from portfolios"))
+ .extracting(row -> row.get("KEE"))
+ .containsExactlyInAnyOrder("port1", "port2", "port3", "port4");
+
+ var portfolioKeyByUuid = db.select("select uuid, kee from portfolios").stream()
+ .collect(Collectors.toMap(row -> row.get("KEE"), row -> row.get("UUID").toString()));
+
+ String portfolio1Uuid = portfolioKeyByUuid.get("port1");
+ String portfolio2Uuid = portfolioKeyByUuid.get("port2");
+ String portfolio3Uuid = portfolioKeyByUuid.get("port3");
+ String portfolio4Uuid = portfolioKeyByUuid.get("port4");
+
+ assertThat(db.select("select portfolio_uuid, reference_uuid from portfolio_references"))
+ .extracting(row -> row.get("PORTFOLIO_UUID"), row -> row.get("REFERENCE_UUID"))
+ .containsExactlyInAnyOrder(
+ tuple(portfolio2Uuid, portfolio1Uuid),
+ tuple(portfolio2Uuid, portfolio3Uuid),
+ tuple(portfolio4Uuid, portfolio2Uuid),
+ tuple(portfolio4Uuid, "app1-uuid"),
+ tuple(portfolio4Uuid, "app2-uuid"))
+ .doesNotContain(tuple(portfolio4Uuid, "app3-key-not-existing"));
+ }
+
+ @Test
+ public void migrate_xml_portfolios_hierarchy() throws SQLException {
+ insertViewsDefInternalProperty(SIMPLE_XML_PORTFOLIOS_HIERARCHY);
+
+ underTest.execute();
+ // re-entrant
+ underTest.execute();
+
+ assertThat(db.select("select kee from portfolios"))
+ .extracting(row -> row.get("KEE"))
+ .containsExactlyInAnyOrder("port1", "port2", "port3", "port4", "port5", "port6", "port7");
+
+ var portfolioKeyByUuid = db.select("select uuid, kee from portfolios").stream()
+ .collect(Collectors.toMap(row -> row.get("KEE"), row -> row.get("UUID").toString()));
+
+ String portfolio1Uuid = portfolioKeyByUuid.get("port1");
+ String portfolio2Uuid = portfolioKeyByUuid.get("port2");
+ String portfolio3Uuid = portfolioKeyByUuid.get("port3");
+ String portfolio4Uuid = portfolioKeyByUuid.get("port4");
+ String portfolio5Uuid = portfolioKeyByUuid.get("port5");
+ String portfolio6Uuid = portfolioKeyByUuid.get("port6");
+ String portfolio7Uuid = portfolioKeyByUuid.get("port7");
+
+ assertThat(db.select("select uuid, parent_uuid, root_uuid from portfolios"))
+ .extracting(row -> row.get("UUID"), row -> row.get("PARENT_UUID"), row -> row.get("ROOT_UUID"))
+ .containsExactlyInAnyOrder(
+ tuple(portfolio1Uuid, null, portfolio1Uuid),
+ tuple(portfolio2Uuid, portfolio1Uuid, portfolio1Uuid),
+ tuple(portfolio3Uuid, portfolio1Uuid, portfolio1Uuid),
+ tuple(portfolio4Uuid, portfolio1Uuid, portfolio1Uuid),
+ tuple(portfolio5Uuid, portfolio2Uuid, portfolio1Uuid),
+ tuple(portfolio6Uuid, portfolio3Uuid, portfolio1Uuid),
+ tuple(portfolio7Uuid, portfolio3Uuid, portfolio1Uuid));
+ }
+
+ @Test
+ public void migrate_xml_portfolios_different_selections() throws SQLException {
+ insertViewsDefInternalProperty(SIMPLE_XML_PORTFOLIOS_DIFFERENT_SELECTIONS);
+
+ insertProject("p1-uuid", "p1-key", Qualifiers.PROJECT);
+ insertProject("p2-uuid", "p2-key", Qualifiers.PROJECT);
+ insertProject("p3-uuid", "p3-key", Qualifiers.PROJECT);
+
+ underTest.execute();
+ // re-entrant
+ underTest.execute();
+
+ assertThat(db.select("select kee, selection_mode, selection_expression from portfolios"))
+ .extracting(row -> row.get("KEE"), row -> row.get("SELECTION_MODE"), row -> row.get("SELECTION_EXPRESSION"))
+ .containsExactlyInAnyOrder(
+ tuple("port-regexp", "REGEXP", ".*port.*"),
+ tuple("port-projects", "MANUAL", null),
+ tuple("port-tags", "TAGS", "tag1,tag2,tag3"));
+
+ // verify projects
+ assertThat(db.select("select p.kee, pp.project_uuid from "
+ + "portfolio_projects pp join portfolios p on pp.portfolio_uuid = p.uuid where p.kee = 'port-projects'"))
+ .extracting(row -> row.get("KEE"), row -> row.get("PROJECT_UUID"))
+ .containsExactlyInAnyOrder(
+ tuple("port-projects", "p1-uuid"),
+ tuple("port-projects", "p2-uuid"),
+ tuple("port-projects", "p3-uuid"))
+ .doesNotContain(tuple("port-projects", "p4-uuid-not-existing"));
+ }
+
+ @Test
+ public void migrate_xml_portfolios_legacy_selections() throws SQLException {
+ insertViewsDefInternalProperty(XML_PORTFOLIOS_LEGACY_SELECTIONS);
+ underTest.execute();
+ // re-entrant
+ underTest.execute();
+
+ assertThat(db.select("select kee, selection_mode, selection_expression from portfolios"))
+ .extracting(row -> row.get("KEE"), row -> row.get("SELECTION_MODE"), row -> row.get("SELECTION_EXPRESSION"))
+ .containsExactlyInAnyOrder(
+ tuple("port-language", "NONE", null),
+ tuple("port-tag-value", "NONE", null));
+ }
+
+ private void insertProject(String uuid, String key, String qualifier) {
+ db.executeInsert("PROJECTS",
+ "UUID", uuid,
+ "KEE", key,
+ "QUALIFIER", qualifier,
+ "ORGANIZATION_UUID", uuid + "-key",
+ "TAGS", "tag1",
+ "PRIVATE", Boolean.toString(false),
+ "UPDATED_AT", System2.INSTANCE.now());
+ }
+
+ private void insertViewsDefInternalProperty(@Nullable String xml) {
+ String valueColumn = "text_value";
+ if (xml != null && xml.length() > TEXT_VALUE_MAX_LENGTH) {
+ valueColumn = "clob_value";
+ }
+
+ db.executeInsert("internal_properties",
+ "kee", "views.def",
+ "is_empty", "false",
+ valueColumn, xml,
+ "created_at", system2.now());
+ }
+
+ private void insertComponent(String uuid, String kee, boolean isPrivate) {
+ db.executeInsert("components",
+ "uuid", uuid,
+ "kee", kee,
+ "enabled", false,
+ "private", isPrivate,
+ "root_uuid", uuid,
+ "uuid_path", uuid,
+ "project_uuid", uuid);
+ }
+
+ private void insertDefaultVisibilityProperty(boolean isPrivate) {
+ db.executeInsert("properties",
+ "uuid", uuidFactory.create(),
+ "prop_key", "projects.default.visibility",
+ "IS_EMPTY", false,
+ "text_value", isPrivate ? "private" : "public",
+ "created_at", system2.now());
+ }
+
+}
diff --git a/server/sonar-db-migration/src/test/resources/org/sonar/server/platform/db/migration/version/v91/MigratePortfoliosToNewTablesTest/schema.sql b/server/sonar-db-migration/src/test/resources/org/sonar/server/platform/db/migration/version/v91/MigratePortfoliosToNewTablesTest/schema.sql
new file mode 100644
index 00000000000..3cd223894dd
--- /dev/null
+++ b/server/sonar-db-migration/src/test/resources/org/sonar/server/platform/db/migration/version/v91/MigratePortfoliosToNewTablesTest/schema.sql
@@ -0,0 +1,110 @@
+CREATE TABLE "PROJECTS"(
+ "UUID" VARCHAR(40) NOT NULL,
+ "KEE" VARCHAR(400) NOT NULL,
+ "QUALIFIER" VARCHAR(10) NOT NULL,
+ "ORGANIZATION_UUID" VARCHAR(40) NOT NULL,
+ "NAME" VARCHAR(2000),
+ "DESCRIPTION" VARCHAR(2000),
+ "PRIVATE" BOOLEAN NOT NULL,
+ "TAGS" VARCHAR(500),
+ "CREATED_AT" BIGINT,
+ "UPDATED_AT" BIGINT NOT NULL
+);
+ALTER TABLE "PROJECTS" ADD CONSTRAINT "PK_NEW_PROJECTS" PRIMARY KEY("UUID");
+CREATE UNIQUE INDEX "UNIQ_PROJECTS_KEE" ON "PROJECTS"("KEE");
+CREATE INDEX "IDX_QUALIFIER" ON "PROJECTS"("QUALIFIER");
+
+CREATE TABLE "INTERNAL_PROPERTIES"(
+ "KEE" VARCHAR(20) NOT NULL,
+ "IS_EMPTY" BOOLEAN NOT NULL,
+ "TEXT_VALUE" VARCHAR(4000),
+ "CLOB_VALUE" CLOB,
+ "CREATED_AT" BIGINT NOT NULL
+);
+ALTER TABLE "INTERNAL_PROPERTIES" ADD CONSTRAINT "PK_INTERNAL_PROPERTIES" PRIMARY KEY("KEE");
+
+CREATE TABLE "PORTFOLIO_PROJECTS"(
+ "UUID" VARCHAR(40) NOT NULL,
+ "PORTFOLIO_UUID" VARCHAR(40) NOT NULL,
+ "PROJECT_UUID" VARCHAR(40) NOT NULL,
+ "CREATED_AT" BIGINT NOT NULL
+);
+ALTER TABLE "PORTFOLIO_PROJECTS" ADD CONSTRAINT "PK_PORTFOLIO_PROJECTS" PRIMARY KEY("UUID");
+
+CREATE TABLE "PORTFOLIO_REFERENCES"(
+ "UUID" VARCHAR(40) NOT NULL,
+ "PORTFOLIO_UUID" VARCHAR(40) NOT NULL,
+ "REFERENCE_UUID" VARCHAR(40) NOT NULL,
+ "CREATED_AT" BIGINT NOT NULL
+);
+ALTER TABLE "PORTFOLIO_REFERENCES" ADD CONSTRAINT "PK_PORTFOLIO_REFERENCES" PRIMARY KEY("UUID");
+
+CREATE TABLE "PORTFOLIOS"(
+ "UUID" VARCHAR(40) NOT NULL,
+ "KEE" VARCHAR(400) NOT NULL,
+ "NAME" VARCHAR(2000) NOT NULL,
+ "DESCRIPTION" VARCHAR(2000),
+ "ROOT_UUID" VARCHAR(40) NOT NULL,
+ "PARENT_UUID" VARCHAR(40),
+ "PRIVATE" BOOLEAN NOT NULL,
+ "SELECTION_MODE" VARCHAR(50) NOT NULL,
+ "SELECTION_EXPRESSION" VARCHAR(50),
+ "CREATED_AT" BIGINT NOT NULL,
+ "UPDATED_AT" BIGINT NOT NULL
+);
+ALTER TABLE "PORTFOLIOS" ADD CONSTRAINT "PK_PORTFOLIOS" PRIMARY KEY("UUID");
+
+CREATE TABLE "PROPERTIES"(
+ "UUID" VARCHAR(40) NOT NULL,
+ "PROP_KEY" VARCHAR(512) NOT NULL,
+ "IS_EMPTY" BOOLEAN NOT NULL,
+ "TEXT_VALUE" VARCHAR(4000),
+ "CLOB_VALUE" CLOB,
+ "CREATED_AT" BIGINT NOT NULL,
+ "COMPONENT_UUID" VARCHAR(40),
+ "USER_UUID" VARCHAR(255)
+);
+ALTER TABLE "PROPERTIES" ADD CONSTRAINT "PK_PROPERTIES" PRIMARY KEY("UUID");
+CREATE INDEX "PROPERTIES_KEY" ON "PROPERTIES"("PROP_KEY");
+
+CREATE TABLE "COMPONENTS"(
+ "UUID" VARCHAR(50) NOT NULL,
+ "KEE" VARCHAR(400),
+ "DEPRECATED_KEE" VARCHAR(400),
+ "NAME" VARCHAR(2000),
+ "LONG_NAME" VARCHAR(2000),
+ "DESCRIPTION" VARCHAR(2000),
+ "ENABLED" BOOLEAN DEFAULT TRUE NOT NULL,
+ "SCOPE" VARCHAR(3),
+ "QUALIFIER" VARCHAR(10),
+ "PRIVATE" BOOLEAN NOT NULL,
+ "ROOT_UUID" VARCHAR(50) NOT NULL,
+ "LANGUAGE" VARCHAR(20),
+ "COPY_COMPONENT_UUID" VARCHAR(50),
+ "PATH" VARCHAR(2000),
+ "UUID_PATH" VARCHAR(1500) NOT NULL,
+ "PROJECT_UUID" VARCHAR(50) NOT NULL,
+ "MODULE_UUID" VARCHAR(50),
+ "MODULE_UUID_PATH" VARCHAR(1500),
+ "MAIN_BRANCH_PROJECT_UUID" VARCHAR(50),
+ "B_CHANGED" BOOLEAN,
+ "B_NAME" VARCHAR(500),
+ "B_LONG_NAME" VARCHAR(500),
+ "B_DESCRIPTION" VARCHAR(2000),
+ "B_ENABLED" BOOLEAN,
+ "B_QUALIFIER" VARCHAR(10),
+ "B_LANGUAGE" VARCHAR(20),
+ "B_COPY_COMPONENT_UUID" VARCHAR(50),
+ "B_PATH" VARCHAR(2000),
+ "B_UUID_PATH" VARCHAR(1500),
+ "B_MODULE_UUID" VARCHAR(50),
+ "B_MODULE_UUID_PATH" VARCHAR(1500),
+ "CREATED_AT" TIMESTAMP
+);
+CREATE UNIQUE INDEX "PROJECTS_KEE" ON "COMPONENTS"("KEE");
+CREATE INDEX "PROJECTS_MODULE_UUID" ON "COMPONENTS"("MODULE_UUID");
+CREATE INDEX "PROJECTS_PROJECT_UUID" ON "COMPONENTS"("PROJECT_UUID");
+CREATE INDEX "PROJECTS_QUALIFIER" ON "COMPONENTS"("QUALIFIER");
+CREATE INDEX "PROJECTS_ROOT_UUID" ON "COMPONENTS"("ROOT_UUID");
+CREATE INDEX "PROJECTS_UUID" ON "COMPONENTS"("UUID");
+CREATE INDEX "IDX_MAIN_BRANCH_PRJ_UUID" ON "COMPONENTS"("MAIN_BRANCH_PROJECT_UUID");
diff --git a/server/sonar-webserver-webapi/src/main/java/org/sonar/server/component/ComponentUpdater.java b/server/sonar-webserver-webapi/src/main/java/org/sonar/server/component/ComponentUpdater.java
index ef052363271..9da3b7edd93 100644
--- a/server/sonar-webserver-webapi/src/main/java/org/sonar/server/component/ComponentUpdater.java
+++ b/server/sonar-webserver-webapi/src/main/java/org/sonar/server/component/ComponentUpdater.java
@@ -36,6 +36,8 @@ import org.sonar.db.DbSession;
import org.sonar.db.component.BranchDto;
import org.sonar.db.component.BranchType;
import org.sonar.db.component.ComponentDto;
+import org.sonar.db.portfolio.PortfolioDto;
+import org.sonar.db.portfolio.PortfolioDto.SelectionMode;
import org.sonar.db.project.ProjectDto;
import org.sonar.server.es.ProjectIndexer.Cause;
import org.sonar.server.es.ProjectIndexers;
@@ -90,7 +92,7 @@ public class ComponentUpdater {
* Don't forget to call commitAndIndex(...) when ready to commit.
*/
public ComponentDto createWithoutCommit(DbSession dbSession, NewComponent newComponent,
- @Nullable String userUuid, @Nullable String userLogin, Consumer<ComponentDto> componentModifier) {
+ @Nullable String userUuid, @Nullable String userLogin, Consumer<ComponentDto> componentModifier) {
return createWithoutCommit(dbSession, newComponent, userUuid, userLogin, null, componentModifier);
}
@@ -144,6 +146,12 @@ public class ComponentUpdater {
ProjectDto projectDto = toProjectDto(component, now);
dbClient.projectDao().insert(session, projectDto);
}
+
+ if (isRootView(component)) {
+ PortfolioDto portfolioDto = toPortfolioDto(component, now);
+ dbClient.portfolioDao().insert(session, portfolioDto);
+ }
+
return component;
}
@@ -159,11 +167,29 @@ public class ComponentUpdater {
.setCreatedAt(now);
}
+ private static PortfolioDto toPortfolioDto(ComponentDto component, long now) {
+ return new PortfolioDto()
+ .setUuid(component.uuid())
+ .setRootUuid(component.projectUuid())
+ .setKey(component.getKey())
+ .setName(component.name())
+ .setPrivate(component.isPrivate())
+ .setDescription(component.description())
+ .setSelectionMode(SelectionMode.NONE.name())
+ .setUpdatedAt(now)
+ .setCreatedAt(now);
+ }
+
private static boolean isRootProject(ComponentDto componentDto) {
return Scopes.PROJECT.equals(componentDto.scope())
&& MAIN_BRANCH_QUALIFIERS.contains(componentDto.qualifier());
}
+ private static boolean isRootView(ComponentDto componentDto) {
+ return Scopes.PROJECT.equals(componentDto.scope())
+ && Qualifiers.VIEW.contains(componentDto.qualifier());
+ }
+
private void createMainBranch(DbSession session, String componentUuid, @Nullable String mainBranch) {
BranchDto branch = new BranchDto()
.setBranchType(BranchType.BRANCH)