aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/issue/IssueQueryService.java123
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/issue/IssueQueryServiceTest.java182
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/issue/ws/SearchActionComponentsMediumTest.java2
3 files changed, 245 insertions, 62 deletions
diff --git a/server/sonar-server/src/main/java/org/sonar/server/issue/IssueQueryService.java b/server/sonar-server/src/main/java/org/sonar/server/issue/IssueQueryService.java
index 895a3aa8cad..c2a59641dc5 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/issue/IssueQueryService.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/issue/IssueQueryService.java
@@ -92,13 +92,18 @@ public class IssueQueryService implements ServerComponent {
.createdAfter(RubyUtils.toDate(params.get(IssueFilterParameters.CREATED_AFTER)))
.createdBefore(RubyUtils.toDate(params.get(IssueFilterParameters.CREATED_BEFORE)));
- addComponentParameters(builder, session,
- RubyUtils.toBoolean(params.get(IssueFilterParameters.ON_COMPONENT_ONLY)),
+ Set<String> allComponentUuids = Sets.newHashSet();
+ boolean effectiveOnComponentOnly = mergeComponentParameters(session, RubyUtils.toBoolean(params.get(IssueFilterParameters.ON_COMPONENT_ONLY)),
RubyUtils.toStrings(params.get(IssueFilterParameters.COMPONENTS)),
RubyUtils.toStrings(params.get(IssueFilterParameters.COMPONENT_UUIDS)),
RubyUtils.toStrings(params.get(IssueFilterParameters.COMPONENT_KEYS)),
RubyUtils.toStrings(params.get(IssueFilterParameters.COMPONENT_ROOT_UUIDS)),
RubyUtils.toStrings(params.get(IssueFilterParameters.COMPONENT_ROOTS)),
+ allComponentUuids);
+
+ addComponentParameters(builder, session,
+ effectiveOnComponentOnly,
+ allComponentUuids,
RubyUtils.toStrings(params.get(IssueFilterParameters.PROJECT_UUIDS)),
RubyUtils.toStrings(
ObjectUtils.defaultIfNull(
@@ -149,13 +154,18 @@ public class IssueQueryService implements ServerComponent {
.createdBefore(request.paramAsDateTime(IssueFilterParameters.CREATED_BEFORE))
.ignorePaging(request.paramAsBoolean(IssueFilterParameters.IGNORE_PAGING));
- addComponentParameters(builder, session,
- request.paramAsBoolean(IssueFilterParameters.ON_COMPONENT_ONLY),
+ Set<String> allComponentUuids = Sets.newHashSet();
+ boolean effectiveOnComponentOnly = mergeComponentParameters(session, request.paramAsBoolean(IssueFilterParameters.ON_COMPONENT_ONLY),
request.paramAsStrings(IssueFilterParameters.COMPONENTS),
request.paramAsStrings(IssueFilterParameters.COMPONENT_UUIDS),
request.paramAsStrings(IssueFilterParameters.COMPONENT_KEYS),
request.paramAsStrings(IssueFilterParameters.COMPONENT_ROOT_UUIDS),
request.paramAsStrings(IssueFilterParameters.COMPONENT_ROOTS),
+ allComponentUuids);
+
+ addComponentParameters(builder, session,
+ effectiveOnComponentOnly,
+ allComponentUuids,
request.paramAsStrings(IssueFilterParameters.PROJECT_UUIDS), request.paramAsStrings(IssueFilterParameters.PROJECT_KEYS),
request.paramAsStrings(IssueFilterParameters.MODULE_UUIDS),
request.paramAsStrings(IssueFilterParameters.DIRECTORIES),
@@ -174,33 +184,62 @@ public class IssueQueryService implements ServerComponent {
}
}
- private void addComponentParameters(IssueQuery.Builder builder, DbSession session,
- @Nullable Boolean onComponentOnly,
+ private boolean mergeComponentParameters(DbSession session, Boolean onComponentOnly,
@Nullable Collection<String> components,
@Nullable Collection<String> componentUuids,
@Nullable Collection<String> componentKeys,
- /*
- * Since 5.1, search of issues is recursive by default (module + submodules),
- * but "componentKeys" parameter already deprecates "components" parameter,
- * so queries specifying "componentRoots" must be handled manually
- */
@Nullable Collection<String> componentRootUuids,
@Nullable Collection<String> componentRoots,
+ Set<String> allComponentUuids) {
+ boolean effectiveOnComponentOnly = false;
+
+ failOnIncompatibleParameters(components, componentUuids, componentKeys, componentRootUuids, componentRoots);
+
+ if (componentRootUuids != null) {
+ allComponentUuids.addAll(componentRootUuids);
+ effectiveOnComponentOnly = false;
+ } else if (componentRoots != null) {
+ allComponentUuids.addAll(componentUuids(session, componentRoots));
+ effectiveOnComponentOnly = false;
+ } else if (components != null) {
+ allComponentUuids.addAll(componentUuids(session, components));
+ effectiveOnComponentOnly = true;
+ } else if (componentUuids != null) {
+ allComponentUuids.addAll(componentUuids);
+ effectiveOnComponentOnly = BooleanUtils.isTrue(onComponentOnly);
+ } else if (componentKeys != null) {
+ allComponentUuids.addAll(componentUuids(session, componentKeys));
+ effectiveOnComponentOnly = BooleanUtils.isTrue(onComponentOnly);
+ }
+ return effectiveOnComponentOnly;
+ }
+
+ protected void failOnIncompatibleParameters(
+ @Nullable Collection<String> components,
+ @Nullable Collection<String> componentUuids,
+ @Nullable Collection<String> componentKeys,
+ @Nullable Collection<String> componentRootUuids,
+ @Nullable Collection<String> componentRoots) {
+ failIfBothParametersSet(componentRootUuids, componentRoots, "componentRoots and componentRootUuids cannot be set simultaneously");
+ failIfBothParametersSet(componentUuids, components, "components and componentUuids cannot be set simultaneously");
+ failIfBothParametersSet(componentKeys, componentUuids, "componentKeys and componentUuids cannot be set simultaneously");
+ }
+
+ private void failIfBothParametersSet(@Nullable Collection<String> uuids, @Nullable Collection<String> keys, String message) {
+ if (uuids != null && keys != null) {
+ throw new IllegalArgumentException(message);
+ }
+ }
+
+ private void addComponentParameters(IssueQuery.Builder builder, DbSession session,
+ boolean effectiveOnComponentOnly,
+ Collection<String> allComponentUuids,
@Nullable Collection<String> projectUuids, @Nullable Collection<String> projects,
@Nullable Collection<String> moduleUuids,
@Nullable Collection<String> directories,
@Nullable Collection<String> fileUuids,
@Nullable Collection<String> authors) {
- Set<String> allComponentUuids = Sets.newHashSet();
- boolean effectiveOnComponentOnly = mergeComponentParameters(session, onComponentOnly,
- components,
- componentUuids,
- componentKeys,
- componentRootUuids,
- componentRoots,
- allComponentUuids);
-
builder.onComponentOnly(effectiveOnComponentOnly);
if (allComponentUuids.isEmpty()) {
@@ -237,7 +276,7 @@ public class IssueQueryService implements ServerComponent {
addComponentsBelowView(builder, session, authors, projects, projectUuids, moduleUuids, directories, fileUuids);
} else if ("DEV".equals(uniqueQualifier)) {
// XXX No constant for developer !!!
- Collection<String> actualAuthors = authors == null ? dbClient.authorDao().selectScmAccountsByDeveloperUuids(allComponentUuids) : authors;
+ Collection<String> actualAuthors = authors == null ? dbClient.authorDao().selectScmAccountsByDeveloperUuids(session, allComponentUuids) : authors;
addComponentsBelowView(builder, session, actualAuthors, projects, projectUuids, moduleUuids, directories, fileUuids);
} else if (Qualifiers.PROJECT.equals(uniqueQualifier)) {
builder.projectUuids(allComponentUuids);
@@ -263,53 +302,15 @@ public class IssueQueryService implements ServerComponent {
}
}
- private boolean mergeComponentParameters(DbSession session, Boolean onComponentOnly,
- Collection<String> components,
- Collection<String> componentUuids,
- Collection<String> componentKeys,
- Collection<String> componentRootUuids,
- Collection<String> componentRoots,
- Set<String> allComponentUuids) {
- boolean effectiveOnComponentOnly = false;
-
- if (componentRootUuids != null) {
- if (componentRoots != null) {
- throw new IllegalArgumentException("componentRoots and componentRootUuids cannot be set simultaneously");
- }
- allComponentUuids.addAll(componentRootUuids);
- effectiveOnComponentOnly = false;
- } else if (componentRoots != null) {
- allComponentUuids.addAll(componentUuids(session, componentRoots));
- effectiveOnComponentOnly = false;
- } else if (components != null) {
- if (componentUuids != null) {
- throw new IllegalArgumentException("components and componentUuids cannot be set simultaneously");
- }
- allComponentUuids.addAll(componentUuids(session, components));
- effectiveOnComponentOnly = true;
- } else if (componentUuids != null) {
- if (componentKeys != null) {
- throw new IllegalArgumentException("componentUuids and componentKeys cannot be set simultaneously");
- }
- allComponentUuids.addAll(componentUuids);
- effectiveOnComponentOnly = BooleanUtils.isTrue(onComponentOnly);
- } else if (componentKeys != null) {
- allComponentUuids.addAll(componentUuids(session, componentKeys));
- effectiveOnComponentOnly = BooleanUtils.isTrue(onComponentOnly);
- }
- return effectiveOnComponentOnly;
- }
-
private void addComponentsBelowView(Builder builder, DbSession session, @Nullable Collection<String> authors,
@Nullable Collection<String> projects, @Nullable Collection<String> projectUuids,
@Nullable Collection<String> moduleUuids, Collection<String> directories, Collection<String> fileUuids) {
builder.authors(authors);
+ failIfBothParametersSet(projectUuids, projects, "projects and projectUuids cannot be set simultaneously");
+
if (projectUuids != null) {
- if (projects != null) {
- throw new IllegalArgumentException("projects and projectUuids cannot be set simultaneously");
- }
builder.projectUuids(projectUuids);
} else {
builder.projectUuids(componentUuids(session, projects));
diff --git a/server/sonar-server/src/test/java/org/sonar/server/issue/IssueQueryServiceTest.java b/server/sonar-server/src/test/java/org/sonar/server/issue/IssueQueryServiceTest.java
index 7c8a05e7f85..8ad75641895 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/issue/IssueQueryServiceTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/issue/IssueQueryServiceTest.java
@@ -20,7 +20,10 @@
package org.sonar.server.issue;
+import com.google.common.collect.Lists;
import com.google.common.collect.Sets;
+import org.assertj.core.api.Fail;
+import org.junit.After;
import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
@@ -32,11 +35,14 @@ import org.mockito.stubbing.Answer;
import org.sonar.api.resources.Qualifiers;
import org.sonar.api.rule.RuleKey;
import org.sonar.api.utils.DateUtils;
+import org.sonar.api.web.UserRole;
+import org.sonar.core.component.ComponentDto;
import org.sonar.core.persistence.DbSession;
import org.sonar.core.user.AuthorDao;
import org.sonar.server.component.ComponentService;
import org.sonar.server.component.db.ComponentDao;
import org.sonar.server.db.DbClient;
+import org.sonar.server.user.MockUserSession;
import java.util.ArrayList;
import java.util.Arrays;
@@ -49,7 +55,9 @@ import static java.util.Arrays.asList;
import static org.assertj.core.api.Assertions.assertThat;
import static org.junit.Assert.fail;
import static org.mockito.Matchers.any;
+import static org.mockito.Matchers.anyCollection;
import static org.mockito.Matchers.eq;
+import static org.mockito.Matchers.isA;
import static org.mockito.Mockito.when;
@RunWith(MockitoJUnitRunner.class)
@@ -89,6 +97,11 @@ public class IssueQueryServiceTest {
issueQueryService = new IssueQueryService(dbClient, componentService);
}
+ @After
+ public void tearDown() {
+ MockUserSession.set();
+ }
+
@Test
public void create_query_from_parameters() {
Map<String, Object> map = newHashMap();
@@ -227,4 +240,173 @@ public class IssueQueryServiceTest {
}
}
+ @Test
+ public void should_search_in_tree_with_component_root_uuids_but_unknown_qualifiers() throws Exception {
+ Map<String, Object> map = newHashMap();
+ map.put("componentRootUuids", newArrayList("ABCD"));
+
+ when(componentService.getDistinctQualifiers(isA(DbSession.class), anyCollection())).thenReturn(Sets.<String>newHashSet());
+
+ IssueQuery query = issueQueryService.createFromMap(map);
+ assertThat(query.onComponentOnly()).isFalse();
+ assertThat(query.componentUuids()).contains("ABCD");
+ }
+
+ @Test
+ public void should_search_in_tree_with_component_roots_but_different_qualifiers() throws Exception {
+ Map<String, Object> map = newHashMap();
+ map.put("componentRoots", newArrayList("org.apache.struts:struts", "org.codehaus.sonar:sonar-server"));
+
+ when(componentService.componentUuids(isA(DbSession.class), anyCollection(), eq(true))).thenReturn(Sets.newHashSet("ABCD", "BCDE"));
+ when(componentService.getDistinctQualifiers(isA(DbSession.class), anyCollection())).thenReturn(Sets.newTreeSet(Arrays.asList("TRK", "BRC")));
+
+ try {
+ issueQueryService.createFromMap(map);
+ Fail.failBecauseExceptionWasNotThrown(IllegalArgumentException.class);
+ } catch (IllegalArgumentException exception) {
+ assertThat(exception).hasMessage("All components must have the same qualifier, found BRC,TRK");
+ }
+ }
+
+ @Test
+ public void should_search_in_tree_with_view() throws Exception {
+ String viewUuid = "ABCD";
+ Map<String, Object> map = newHashMap();
+ map.put("componentRootUuids", newArrayList(viewUuid));
+
+ when(componentService.getDistinctQualifiers(isA(DbSession.class), anyCollection())).thenReturn(Sets.newHashSet(Qualifiers.VIEW));
+
+ MockUserSession.set().addProjectUuidPermissions(UserRole.USER, viewUuid);
+
+ IssueQuery query = issueQueryService.createFromMap(map);
+ assertThat(query.viewUuids()).containsExactly(viewUuid);
+ assertThat(query.onComponentOnly()).isFalse();
+ }
+
+ @Test
+ public void should_search_in_tree_with_subview_but_bad_permissions() throws Exception {
+ String subViewUuid = "ABCD";
+ Map<String, Object> map = newHashMap();
+ map.put("componentRootUuids", newArrayList(subViewUuid));
+
+ when(componentService.getDistinctQualifiers(isA(DbSession.class), anyCollection())).thenReturn(Sets.newHashSet(Qualifiers.VIEW));
+
+ MockUserSession.set();
+
+ IssueQuery query = issueQueryService.createFromMap(map);
+ assertThat(query.viewUuids()).isNotEmpty().doesNotContain(subViewUuid);
+ }
+
+ @Test
+ public void should_search_in_tree_with_project_uuid() throws Exception {
+ String projectUuid = "ABCD";
+ Map<String, Object> map = newHashMap();
+ map.put("componentUuids", newArrayList(projectUuid));
+
+ when(componentService.getDistinctQualifiers(isA(DbSession.class), anyCollection())).thenReturn(Sets.newHashSet(Qualifiers.PROJECT));
+
+ IssueQuery query = issueQueryService.createFromMap(map);
+ assertThat(query.projectUuids()).containsExactly(projectUuid);
+ assertThat(query.onComponentOnly()).isFalse();
+ }
+
+ @Test
+ public void should_search_on_component_only_with_project_key() throws Exception {
+ String projectKey = "org.apache.struts:struts";
+ String projectUuid = "ABCD";
+ Map<String, Object> map = newHashMap();
+ map.put("componentKeys", newArrayList(projectKey));
+ map.put("onComponentOnly", true);
+
+ when(componentService.componentUuids(isA(DbSession.class), anyCollection(), eq(true))).thenReturn(Sets.newHashSet(projectUuid));
+ when(componentService.getDistinctQualifiers(isA(DbSession.class), anyCollection())).thenReturn(Sets.newHashSet(Qualifiers.PROJECT));
+
+ IssueQuery query = issueQueryService.createFromMap(map);
+ assertThat(query.projectUuids()).isEmpty();
+ assertThat(query.componentUuids()).containsExactly(projectUuid);
+ assertThat(query.onComponentOnly()).isTrue();
+ }
+
+ @Test
+ public void should_search_on_developer() throws Exception {
+ String devUuid = "DEV:anakin.skywalker";
+ String login1 = "anakin@skywalker.name";
+ String login2 = "darth.vader";
+ Map<String, Object> map = newHashMap();
+ map.put("componentUuids", newArrayList(devUuid));
+
+ when(componentService.getDistinctQualifiers(isA(DbSession.class), anyCollection())).thenReturn(Sets.newHashSet("DEV"));
+ when(authorDao.selectScmAccountsByDeveloperUuids(isA(DbSession.class), anyCollection())).thenReturn(Lists.newArrayList(login1, login2));
+
+ IssueQuery query = issueQueryService.createFromMap(map);
+ assertThat(query.authors()).containsExactly(login1, login2);
+ }
+
+ @Test
+ public void should_override_authors_when_searching_on_developer() throws Exception {
+ String devUuid = "DEV:anakin.skywalker";
+ String login = "anakin@skywalker.name";
+ Map<String, Object> map = newHashMap();
+ map.put("componentUuids", newArrayList(devUuid));
+ map.put("authors", newArrayList(login));
+
+ when(componentService.getDistinctQualifiers(isA(DbSession.class), anyCollection())).thenReturn(Sets.newHashSet("DEV"));
+
+ IssueQuery query = issueQueryService.createFromMap(map);
+ assertThat(query.authors()).containsExactly(login);
+ }
+
+ @Test
+ public void should_search_in_tree_with_module_uuid() throws Exception {
+ String moduleUuid = "ABCD";
+ Map<String, Object> map = newHashMap();
+ map.put("componentUuids", newArrayList(moduleUuid));
+
+ when(componentService.getDistinctQualifiers(isA(DbSession.class), anyCollection())).thenReturn(Sets.newHashSet(Qualifiers.MODULE));
+
+ IssueQuery query = issueQueryService.createFromMap(map);
+ assertThat(query.moduleRootUuids()).containsExactly(moduleUuid);
+ assertThat(query.onComponentOnly()).isFalse();
+ }
+
+ @Test
+ public void should_search_in_tree_with_directory_uuid() throws Exception {
+ String directoryUuid = "ABCD";
+ String directoryPath = "/some/module/relative/path";
+ String moduleUuid = "BCDE";
+ Map<String, Object> map = newHashMap();
+ map.put("componentUuids", newArrayList(directoryUuid));
+
+ when(componentService.getDistinctQualifiers(isA(DbSession.class), anyCollection())).thenReturn(Sets.newHashSet(Qualifiers.DIRECTORY));
+ when(componentService.getByUuids(isA(DbSession.class), anyCollection())).thenReturn(Arrays.asList(new ComponentDto().setModuleUuid(moduleUuid).setPath(directoryPath)));
+
+ IssueQuery query = issueQueryService.createFromMap(map);
+ assertThat(query.moduleUuids()).contains(moduleUuid);
+ assertThat(query.directories()).containsExactly(directoryPath);
+ assertThat(query.onComponentOnly()).isFalse();
+ }
+
+ @Test
+ public void should_search_on_source_file() throws Exception {
+ String fileUuid = "ABCD";
+ Map<String, Object> map = newHashMap();
+ map.put("componentUuids", newArrayList(fileUuid));
+
+ when(componentService.getDistinctQualifiers(isA(DbSession.class), anyCollection())).thenReturn(Sets.newHashSet(Qualifiers.FILE));
+
+ IssueQuery query = issueQueryService.createFromMap(map);
+ assertThat(query.fileUuids()).containsExactly(fileUuid);
+ }
+
+ @Test
+ public void should_search_on_test_file() throws Exception {
+ String fileUuid = "ABCD";
+ Map<String, Object> map = newHashMap();
+ map.put("componentUuids", newArrayList(fileUuid));
+
+ when(componentService.getDistinctQualifiers(isA(DbSession.class), anyCollection())).thenReturn(Sets.newHashSet(Qualifiers.UNIT_TEST_FILE));
+
+ IssueQuery query = issueQueryService.createFromMap(map);
+ assertThat(query.fileUuids()).containsExactly(fileUuid);
+ }
}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/issue/ws/SearchActionComponentsMediumTest.java b/server/sonar-server/src/test/java/org/sonar/server/issue/ws/SearchActionComponentsMediumTest.java
index 294ebb12463..5b3ed067794 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/issue/ws/SearchActionComponentsMediumTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/issue/ws/SearchActionComponentsMediumTest.java
@@ -342,7 +342,7 @@ public class SearchActionComponentsMediumTest {
tester.get(IssueIndexer.class).indexAll();
wsTester.newGetRequest(IssuesWs.API_ENDPOINT, SearchAction.SEARCH_ACTION)
- .setParam(IssueFilterParameters.COMPONENT_UUIDS, project.uuid())
+ .setParam(IssueFilterParameters.COMPONENT_UUIDS, module.uuid())
.setParam(IssueFilterParameters.MODULE_UUIDS, subModule1.uuid() + "," + subModule3.uuid())
.setParam(WebService.Param.FACETS, "moduleUuids")
.execute()