aboutsummaryrefslogtreecommitdiffstats
path: root/sonar-db
diff options
context:
space:
mode:
Diffstat (limited to 'sonar-db')
-rw-r--r--sonar-db/src/main/java/org/sonar/db/component/ComponentDao.java47
-rw-r--r--sonar-db/src/main/java/org/sonar/db/component/ComponentMapper.java5
-rw-r--r--sonar-db/src/main/resources/org/sonar/db/component/ComponentMapper.xml20
-rw-r--r--sonar-db/src/test/java/org/sonar/db/component/ComponentDaoTest.java38
4 files changed, 76 insertions, 34 deletions
diff --git a/sonar-db/src/main/java/org/sonar/db/component/ComponentDao.java b/sonar-db/src/main/java/org/sonar/db/component/ComponentDao.java
index ad049fc90df..e4d2527af9b 100644
--- a/sonar-db/src/main/java/org/sonar/db/component/ComponentDao.java
+++ b/sonar-db/src/main/java/org/sonar/db/component/ComponentDao.java
@@ -28,20 +28,24 @@ import java.util.HashSet;
import java.util.List;
import java.util.Locale;
import java.util.Map;
-import java.util.Objects;
import java.util.Set;
+import javax.annotation.CheckForNull;
import javax.annotation.Nullable;
import org.apache.ibatis.session.ResultHandler;
import org.apache.ibatis.session.RowBounds;
import org.sonar.api.resources.Qualifiers;
import org.sonar.api.resources.Scopes;
import org.sonar.db.Dao;
+import org.sonar.db.DatabaseUtils;
import org.sonar.db.DbSession;
import org.sonar.db.RowNotFoundException;
+import org.sonar.db.WildcardPosition;
import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.collect.Maps.newHashMapWithExpectedSize;
import static java.util.Collections.emptyList;
+import static java.util.Objects.requireNonNull;
+import static org.apache.commons.lang.StringUtils.isBlank;
import static org.sonar.db.DatabaseUtils.executeLargeInputs;
import static org.sonar.db.DatabaseUtils.executeLargeUpdates;
@@ -205,20 +209,37 @@ public class ComponentDao implements Dao {
return mapper(dbSession).selectAllRootsByOrganization(organizationUuid);
}
- public List<ComponentDto> selectProvisionedProjects(DbSession session, int offset, int limit, @Nullable String query) {
- Map<String, Object> parameters = newHashMapWithExpectedSize(2);
- addProjectQualifier(parameters);
- addPartialQueryParameterIfNotNull(parameters, query);
-
- return mapper(session).selectProvisionedProjects(parameters, new RowBounds(offset, limit));
+ /**
+ * Select a page of provisioned (root) components. Results are ordered by ascending name.
+ *
+ * @param dbSession
+ * @param textQuery optional text query to match component name or key
+ * @param qualifiers filter on qualifiers. Must not be null nor empty
+ * @param rowBounds pagination
+ */
+ public List<ComponentDto> selectProvisioned(DbSession dbSession, @Nullable String textQuery, Set<String> qualifiers, RowBounds rowBounds) {
+ checkArgument(!qualifiers.isEmpty(), "qualifiers must not be empty");
+ return mapper(dbSession).selectProvisioned(buildUpperLikeSql(textQuery), qualifiers, rowBounds);
}
- public int countProvisionedProjects(DbSession session, @Nullable String query) {
- Map<String, Object> parameters = newHashMapWithExpectedSize(2);
- addProjectQualifier(parameters);
- addPartialQueryParameterIfNotNull(parameters, query);
+ /**
+ * Count number of provisioned (root) components.
+ *
+ * @param dbSession
+ * @param textQuery optional text query to match component name or key
+ * @param qualifiers filter on qualifiers. Must not be null nor empty
+ */
+ public int countProvisioned(DbSession dbSession, @Nullable String textQuery, Set<String> qualifiers) {
+ checkArgument(!qualifiers.isEmpty(), "qualifiers must not be empty");
+ return mapper(dbSession).countProvisioned(buildUpperLikeSql(textQuery), qualifiers);
+ }
- return mapper(session).countProvisionedProjects(parameters);
+ @CheckForNull
+ private static String buildUpperLikeSql(@Nullable String textQuery) {
+ if (isBlank(textQuery)) {
+ return null;
+ }
+ return DatabaseUtils.buildLikeValue(textQuery.toUpperCase(Locale.ENGLISH), WildcardPosition.BEFORE_AND_AFTER);
}
public List<ComponentDto> selectGhostProjects(DbSession session, int offset, int limit, @Nullable String query) {
@@ -244,7 +265,7 @@ public class ComponentDao implements Dao {
* @param handler the action to be applied to every result
*/
public void selectForIndexing(DbSession session, @Nullable String projectUuid, ResultHandler handler) {
- Objects.requireNonNull(handler);
+ requireNonNull(handler);
mapper(session).selectForIndexing(projectUuid, handler);
}
diff --git a/sonar-db/src/main/java/org/sonar/db/component/ComponentMapper.java b/sonar-db/src/main/java/org/sonar/db/component/ComponentMapper.java
index 9ad0394deb4..101ac0f26c6 100644
--- a/sonar-db/src/main/java/org/sonar/db/component/ComponentMapper.java
+++ b/sonar-db/src/main/java/org/sonar/db/component/ComponentMapper.java
@@ -22,6 +22,7 @@ package org.sonar.db.component;
import java.util.Collection;
import java.util.List;
import java.util.Map;
+import java.util.Set;
import javax.annotation.CheckForNull;
import javax.annotation.Nullable;
import org.apache.ibatis.annotations.Param;
@@ -108,9 +109,9 @@ public interface ComponentMapper {
*/
List<String> selectProjectsFromView(@Param("viewUuidLikeQuery") String viewUuidLikeQuery, @Param("projectViewUuid") String projectViewUuid);
- List<ComponentDto> selectProvisionedProjects(Map<String, Object> parameters, RowBounds rowBounds);
+ List<ComponentDto> selectProvisioned(@Nullable @Param("keyOrNameLike") String keyOrNameLike, @Param("qualifiers") Set<String> qualifiers, RowBounds rowBounds);
- int countProvisionedProjects(Map<String, Object> parameters);
+ int countProvisioned(@Nullable @Param("keyOrNameLike") String keyOrNameLike, @Param("qualifiers") Set<String> qualifiers);
List<ComponentDto> selectGhostProjects(Map<String, Object> parameters, RowBounds rowBounds);
diff --git a/sonar-db/src/main/resources/org/sonar/db/component/ComponentMapper.xml b/sonar-db/src/main/resources/org/sonar/db/component/ComponentMapper.xml
index 6fba079f9d5..2f4a42d2835 100644
--- a/sonar-db/src/main/resources/org/sonar/db/component/ComponentMapper.xml
+++ b/sonar-db/src/main/resources/org/sonar/db/component/ComponentMapper.xml
@@ -368,30 +368,30 @@
</where>
</select>
- <select id="selectProvisionedProjects" parameterType="map" resultType="Component">
+ <select id="selectProvisioned" parameterType="map" resultType="Component">
select
<include refid="componentColumns"/>
from projects p
- <include refid="provisionClauses"/>
+ <include refid="provisionedClauses"/>
</select>
- <select id="countProvisionedProjects" parameterType="map" resultType="int">
+ <select id="countProvisioned" parameterType="map" resultType="int">
select count(p.id)
from projects p
- <include refid="provisionClauses"/>
+ <include refid="provisionedClauses"/>
</select>
- <sql id="provisionClauses">
+ <sql id="provisionedClauses">
left join snapshots s on s.component_uuid=p.uuid
where
s.id is null
- and p.enabled=${_true}
- and p.qualifier=#{qualifier}
+ and p.enabled = ${_true}
+ and p.qualifier in <foreach collection="qualifiers" open="(" close=")" item="qualifier" separator=",">#{qualifier}</foreach>
and p.copy_component_uuid is null
- <if test="query!=null">
+ <if test="keyOrNameLike != null">
and (
- UPPER(p.name) like #{query}
- or UPPER(p.kee) like #{query}
+ upper(p.name) like #{keyOrNameLike} ESCAPE '/'
+ or upper(p.kee) like #{keyOrNameLike} ESCAPE '/'
)
</if>
</sql>
diff --git a/sonar-db/src/test/java/org/sonar/db/component/ComponentDaoTest.java b/sonar-db/src/test/java/org/sonar/db/component/ComponentDaoTest.java
index 8d2b2d83c66..9bbaaf20aa7 100644
--- a/sonar-db/src/test/java/org/sonar/db/component/ComponentDaoTest.java
+++ b/sonar-db/src/test/java/org/sonar/db/component/ComponentDaoTest.java
@@ -24,9 +24,11 @@ import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.Map;
+import java.util.Set;
import javax.annotation.Nullable;
import org.apache.ibatis.session.ResultContext;
import org.apache.ibatis.session.ResultHandler;
+import org.apache.ibatis.session.RowBounds;
import org.assertj.core.api.ListAssert;
import org.junit.Rule;
import org.junit.Test;
@@ -529,23 +531,41 @@ public class ComponentDaoTest {
}
@Test
- public void select_provisioned_projects() {
+ public void select_provisioned() {
db.prepareDbUnit(getClass(), "select_provisioned_projects.xml");
- List<ComponentDto> result = underTest.selectProvisionedProjects(dbSession, 0, 10, null);
- ComponentDto project = result.get(0);
+ Set<String> projectQualifiers = newHashSet(Qualifiers.PROJECT);
+ assertProvisionedKeys(null, projectQualifiers, 0, 10).containsExactly("org.provisioned.project");
- assertThat(result).hasSize(1);
- assertThat(project.getKey()).isEqualTo("org.provisioned.project");
+ // pagination
+ assertProvisionedKeys(null, projectQualifiers, 2, 10).isEmpty();
+
+ // filter on qualifiers
+ assertProvisionedKeys(null, newHashSet("XXX"), 0, 10).isEmpty();
+ assertProvisionedKeys(null, newHashSet(Qualifiers.PROJECT, "XXX"), 0, 10).containsExactly("org.provisioned.project");
+
+ // match key
+ assertProvisionedKeys("org.provisioned.project", projectQualifiers, 0, 10).containsExactly("org.provisioned.project");
+ assertProvisionedKeys("ORG.provisioned.PROJECT", projectQualifiers, 0, 10).containsExactly("org.provisioned.project");
+ assertProvisionedKeys("missing", projectQualifiers, 0, 10).isEmpty();
+ assertProvisionedKeys("to be escaped '\"\\%", projectQualifiers, 0, 10).isEmpty();
+
+ // match name
+ assertProvisionedKeys("ned proj", projectQualifiers, 0, 10).containsExactly("org.provisioned.project");
+ }
+
+ private ListAssert<String> assertProvisionedKeys(@Nullable String textQuery, Set<String> qualifiers, int offset, int limit) {
+ List<ComponentDto> result = underTest.selectProvisioned(dbSession, textQuery, qualifiers, new RowBounds(offset, limit));
+ return assertThat(result).extracting(ComponentDto::getKey);
}
@Test
- public void count_provisioned_projects() {
+ public void count_provisioned() {
db.prepareDbUnit(getClass(), "select_provisioned_projects.xml");
- int numberOfProjects = underTest.countProvisionedProjects(dbSession, null);
-
- assertThat(numberOfProjects).isEqualTo(1);
+ assertThat(underTest.countProvisioned(dbSession, null, newHashSet(Qualifiers.PROJECT))).isEqualTo(1);
+ assertThat(underTest.countProvisioned(dbSession, null, newHashSet(Qualifiers.VIEW))).isEqualTo(0);
+ assertThat(underTest.countProvisioned(dbSession, null, newHashSet(Qualifiers.VIEW, Qualifiers.PROJECT))).isEqualTo(1);
}
@Test