From ee8aea575b5dd4ba6a8df9c92461e41243b66589 Mon Sep 17 00:00:00 2001 From: Julien Lancelot Date: Thu, 16 Mar 2017 16:25:22 +0100 Subject: SONAR-8899 SONAR-8900 SONAR-8919 Add ITs --- it/it-tests/src/test/java/it/Category3Suite.java | 4 +- .../test/java/it/organization/IssueAssignTest.java | 179 +++++++++++++++++++++ .../src/test/java/it/organization/RootTest.java | 6 +- .../IssueAssignTest/one-issue-per-file-profile.xml | 12 ++ 4 files changed, 197 insertions(+), 4 deletions(-) create mode 100644 it/it-tests/src/test/java/it/organization/IssueAssignTest.java create mode 100644 it/it-tests/src/test/resources/organization/IssueAssignTest/one-issue-per-file-profile.xml (limited to 'it') diff --git a/it/it-tests/src/test/java/it/Category3Suite.java b/it/it-tests/src/test/java/it/Category3Suite.java index 6d9e32a1de2..24f7baf5ec8 100644 --- a/it/it-tests/src/test/java/it/Category3Suite.java +++ b/it/it-tests/src/test/java/it/Category3Suite.java @@ -34,9 +34,10 @@ import it.analysis.ScannerTest; import it.analysis.SettingsEncryptionTest; import it.analysis.TempFolderTest; import it.measure.DecimalScaleMetricTest; -import it.plugins.VersionPluginTest; +import it.organization.IssueAssignTest; import it.organization.OrganizationTest; import it.organization.RootTest; +import it.plugins.VersionPluginTest; import it.webhook.WebhooksTest; import org.junit.ClassRule; import org.junit.runner.RunWith; @@ -66,6 +67,7 @@ import static util.ItUtils.xooPlugin; DecimalScaleMetricTest.class, // organization OrganizationTest.class, + IssueAssignTest.class, // root users RootTest.class, WebhooksTest.class diff --git a/it/it-tests/src/test/java/it/organization/IssueAssignTest.java b/it/it-tests/src/test/java/it/organization/IssueAssignTest.java new file mode 100644 index 00000000000..c9e5c4234f0 --- /dev/null +++ b/it/it-tests/src/test/java/it/organization/IssueAssignTest.java @@ -0,0 +1,179 @@ +/* + * SonarQube + * Copyright (C) 2009-2017 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 it.organization; + +import com.sonar.orchestrator.Orchestrator; +import com.sonar.orchestrator.locator.FileLocation; +import it.Category3Suite; +import java.util.List; +import java.util.stream.Collectors; +import org.junit.After; +import org.junit.Before; +import org.junit.ClassRule; +import org.junit.Rule; +import org.junit.Test; +import org.junit.rules.ExpectedException; +import org.sonarqube.ws.Issues; +import org.sonarqube.ws.Issues.Issue; +import org.sonarqube.ws.client.HttpException; +import org.sonarqube.ws.client.PostRequest; +import org.sonarqube.ws.client.WsClient; +import org.sonarqube.ws.client.issue.AssignRequest; +import org.sonarqube.ws.client.issue.BulkChangeRequest; +import org.sonarqube.ws.client.issue.SearchWsRequest; +import org.sonarqube.ws.client.organization.CreateWsRequest; +import util.issue.IssueRule; +import util.user.UserRule; + +import static java.lang.String.format; +import static java.util.Collections.emptyMap; +import static java.util.Collections.singletonList; +import static org.assertj.core.api.Assertions.assertThat; +import static util.ItUtils.newAdminWsClient; +import static util.ItUtils.runProjectAnalysis; +import static util.ItUtils.setServerProperty; + +public class IssueAssignTest { + + private final static String SAMPLE_PROJECT_KEY = "sample"; + private final static String ORGANIZATION_KEY = "organization-key"; + private final static String OTHER_ORGANIZATION_KEY = "other-organization-key"; + private static final String ASSIGNEE_LOGIN = "bob"; + + @Rule + public ExpectedException expectedException = ExpectedException.none(); + + @ClassRule + public static Orchestrator orchestrator = Category3Suite.ORCHESTRATOR; + + @ClassRule + public static UserRule userRule = UserRule.from(orchestrator); + + @ClassRule + public static IssueRule issueRule = IssueRule.from(orchestrator); + + private WsClient adminClient = newAdminWsClient(orchestrator); + + @Before + public void setUp() throws Exception { + orchestrator.resetData(); + userRule.resetUsers(); + + orchestrator.getServer().post("api/organizations/enable_support", emptyMap()); + createOrganization(ORGANIZATION_KEY); + + orchestrator.getServer().restoreProfile(FileLocation.ofClasspath("/organization/IssueAssignTest/one-issue-per-file-profile.xml")); + } + + @After + public void tearDown() throws Exception { + adminClient.organizations().search(org.sonarqube.ws.client.organization.SearchWsRequest.builder().setOrganizations(ORGANIZATION_KEY, OTHER_ORGANIZATION_KEY).build()) + .getOrganizationsList() + .forEach(organization -> adminClient.organizations().delete(organization.getKey())); + } + + @Test + public void auto_assign_issues_to_user_if_default_assignee_is_member_of_project_organization() throws Exception { + userRule.createUser(ASSIGNEE_LOGIN, ASSIGNEE_LOGIN); + adminClient.organizations().addMember(ORGANIZATION_KEY, ASSIGNEE_LOGIN); + provisionProject(SAMPLE_PROJECT_KEY, ORGANIZATION_KEY); + setServerProperty(orchestrator, "sample", "sonar.issues.defaultAssigneeLogin", ASSIGNEE_LOGIN); + + analyseProject(SAMPLE_PROJECT_KEY, ORGANIZATION_KEY); + + assertThat(issueRule.getRandomIssue().getAssignee()).isEqualTo(ASSIGNEE_LOGIN); + } + + @Test + public void does_not_auto_assign_issues_to_user_if_default_assignee_is_not_member_of_project_organization() throws Exception { + createOrganization(OTHER_ORGANIZATION_KEY); + userRule.createUser(ASSIGNEE_LOGIN, ASSIGNEE_LOGIN); + adminClient.organizations().addMember(OTHER_ORGANIZATION_KEY, ASSIGNEE_LOGIN); + provisionProject(SAMPLE_PROJECT_KEY, ORGANIZATION_KEY); + setServerProperty(orchestrator, "sample", "sonar.issues.defaultAssigneeLogin", ASSIGNEE_LOGIN); + + analyseProject(SAMPLE_PROJECT_KEY, ORGANIZATION_KEY); + + assertThat(issueRule.getRandomIssue().hasAssignee()).isFalse(); + } + + @Test + public void assign_issue_to_user_being_member_of_same_organization_as_project_issue_organization() throws Exception { + userRule.createUser(ASSIGNEE_LOGIN, ASSIGNEE_LOGIN); + adminClient.organizations().addMember(ORGANIZATION_KEY, ASSIGNEE_LOGIN); + provisionAndAnalyseProject(SAMPLE_PROJECT_KEY, ORGANIZATION_KEY); + Issue issue = issueRule.getRandomIssue(); + + adminClient.issues().assign(new AssignRequest(issue.getKey(), ASSIGNEE_LOGIN)); + + assertThat(issueRule.getByKey(issue.getKey()).getAssignee()).isEqualTo(ASSIGNEE_LOGIN); + } + + @Test + public void fail_to_assign_issue_to_user_not_being_member_of_same_organization_as_project_issue_organization() throws Exception { + createOrganization(OTHER_ORGANIZATION_KEY); + userRule.createUser(ASSIGNEE_LOGIN, ASSIGNEE_LOGIN); + adminClient.organizations().addMember(OTHER_ORGANIZATION_KEY, ASSIGNEE_LOGIN); + provisionAndAnalyseProject(SAMPLE_PROJECT_KEY, ORGANIZATION_KEY); + Issue issue = issueRule.getRandomIssue(); + + expectedException.expect(HttpException.class); + expectedException.expectMessage(format("User 'bob' is not member of organization '%s'", ORGANIZATION_KEY)); + + adminClient.issues().assign(new AssignRequest(issue.getKey(), ASSIGNEE_LOGIN)); + } + + @Test + public void bulk_assign_issues_to_user_being_only_member_of_same_organization_as_project_issue_organization() throws Exception { + createOrganization(OTHER_ORGANIZATION_KEY); + userRule.createUser(ASSIGNEE_LOGIN, ASSIGNEE_LOGIN); + // User is only member of "organization-key", not of "other-organization-key" + adminClient.organizations().addMember(ORGANIZATION_KEY, ASSIGNEE_LOGIN); + provisionAndAnalyseProject(SAMPLE_PROJECT_KEY, ORGANIZATION_KEY); + provisionAndAnalyseProject("sample2", OTHER_ORGANIZATION_KEY); + List issues = issueRule.search(new org.sonarqube.ws.client.issue.SearchWsRequest()).getIssuesList().stream().map(Issue::getKey).collect(Collectors.toList()); + + Issues.BulkChangeWsResponse response = adminClient.issues().bulkChange(BulkChangeRequest.builder().setIssues(issues).setAssign(ASSIGNEE_LOGIN).build()); + + assertThat(response.getIgnored()).isGreaterThan(0); + assertThat(issueRule.search(new SearchWsRequest().setProjectKeys(singletonList("sample"))).getIssuesList()).extracting(Issue::getAssignee).containsOnly(ASSIGNEE_LOGIN); + assertThat(issueRule.search(new SearchWsRequest().setProjectKeys(singletonList("sample2"))).getIssuesList()).extracting(Issue::hasAssignee).containsOnly(false); + } + + private void createOrganization(String organizationKey) { + adminClient.organizations().create(new CreateWsRequest.Builder().setKey(organizationKey).setName(organizationKey).build()).getOrganization(); + } + + private void provisionAndAnalyseProject(String projectKey, String organization) { + provisionProject(projectKey, organization); + analyseProject(projectKey, organization); + } + + private void provisionProject(String projectKey, String organization) { + adminClient.wsConnector().call(new PostRequest("/api/projects/create").setParam("key", projectKey).setParam("name", projectKey).setParam("organization", organization)); + } + + private void analyseProject(String projectKey, String organization) { + orchestrator.getServer().associateProjectToQualityProfile(projectKey, "xoo", "one-issue-per-file-profile"); + runProjectAnalysis(orchestrator, "issue/xoo-with-scm", "sonar.projectKey", projectKey, + "sonar.organization", organization, "sonar.login", "admin", "sonar.password", "admin", "sonar.scm.disabled", "false", "sonar.scm.provider", "xoo"); + } +} diff --git a/it/it-tests/src/test/java/it/organization/RootTest.java b/it/it-tests/src/test/java/it/organization/RootTest.java index 89b3af24120..5283f3341b9 100644 --- a/it/it-tests/src/test/java/it/organization/RootTest.java +++ b/it/it-tests/src/test/java/it/organization/RootTest.java @@ -23,7 +23,7 @@ import com.sonar.orchestrator.Orchestrator; import it.Category3Suite; import java.sql.SQLException; import java.util.Collections; -import org.junit.After; +import org.junit.Before; import org.junit.ClassRule; import org.junit.Rule; import org.junit.Test; @@ -44,8 +44,8 @@ public class RootTest { @Rule public UserRule userRule = UserRule.from(orchestrator); - @After - public void tearDown() { + @Before + public void before() { orchestrator.resetData(); } diff --git a/it/it-tests/src/test/resources/organization/IssueAssignTest/one-issue-per-file-profile.xml b/it/it-tests/src/test/resources/organization/IssueAssignTest/one-issue-per-file-profile.xml new file mode 100644 index 00000000000..141921f02e7 --- /dev/null +++ b/it/it-tests/src/test/resources/organization/IssueAssignTest/one-issue-per-file-profile.xml @@ -0,0 +1,12 @@ + + + one-issue-per-file-profile + xoo + + + xoo + OneIssuePerFile + CRITICAL + + + -- cgit v1.2.3