You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

ModuleIssuesTest.java 9.7KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258
  1. /*
  2. * SonarQube
  3. * Copyright (C) 2009-2019 SonarSource SA
  4. * mailto:info AT sonarsource DOT com
  5. *
  6. * This program is free software; you can redistribute it and/or
  7. * modify it under the terms of the GNU Lesser General Public
  8. * License as published by the Free Software Foundation; either
  9. * version 3 of the License, or (at your option) any later version.
  10. *
  11. * This program is distributed in the hope that it will be useful,
  12. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  14. * Lesser General Public License for more details.
  15. *
  16. * You should have received a copy of the GNU Lesser General Public License
  17. * along with this program; if not, write to the Free Software Foundation,
  18. * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
  19. */
  20. package org.sonar.scanner.issue;
  21. import java.io.IOException;
  22. import java.util.Collections;
  23. import java.util.HashSet;
  24. import org.junit.Before;
  25. import org.junit.Rule;
  26. import org.junit.Test;
  27. import org.junit.rules.TemporaryFolder;
  28. import org.junit.runner.RunWith;
  29. import org.mockito.ArgumentCaptor;
  30. import org.mockito.Mock;
  31. import org.mockito.runners.MockitoJUnitRunner;
  32. import org.sonar.api.batch.bootstrap.ProjectDefinition;
  33. import org.sonar.api.batch.fs.internal.DefaultInputFile;
  34. import org.sonar.api.batch.fs.internal.DefaultInputModule;
  35. import org.sonar.api.batch.fs.internal.TestInputFileBuilder;
  36. import org.sonar.api.batch.rule.internal.ActiveRulesBuilder;
  37. import org.sonar.api.batch.rule.internal.NewActiveRule;
  38. import org.sonar.api.batch.rule.internal.RulesBuilder;
  39. import org.sonar.api.batch.sensor.issue.internal.DefaultExternalIssue;
  40. import org.sonar.api.batch.sensor.issue.internal.DefaultIssue;
  41. import org.sonar.api.batch.sensor.issue.internal.DefaultIssueLocation;
  42. import org.sonar.api.rule.RuleKey;
  43. import org.sonar.api.rule.Severity;
  44. import org.sonar.api.rules.RuleType;
  45. import org.sonar.scanner.protocol.output.ScannerReport;
  46. import org.sonar.scanner.report.ReportPublisher;
  47. import static org.assertj.core.api.Assertions.assertThat;
  48. import static org.mockito.ArgumentMatchers.any;
  49. import static org.mockito.ArgumentMatchers.anyString;
  50. import static org.mockito.ArgumentMatchers.eq;
  51. import static org.mockito.Mockito.RETURNS_DEEP_STUBS;
  52. import static org.mockito.Mockito.mock;
  53. import static org.mockito.Mockito.verify;
  54. import static org.mockito.Mockito.verifyZeroInteractions;
  55. import static org.mockito.Mockito.when;
  56. @RunWith(MockitoJUnitRunner.class)
  57. public class ModuleIssuesTest {
  58. static final RuleKey SQUID_RULE_KEY = RuleKey.of("squid", "AvoidCycle");
  59. static final String SQUID_RULE_NAME = "Avoid Cycle";
  60. private static final RuleKey NOSONAR_RULE_KEY = RuleKey.of("squid", "NoSonarCheck");
  61. private DefaultInputModule projectRoot;
  62. @Rule
  63. public TemporaryFolder temp = new TemporaryFolder();
  64. @Mock
  65. ModuleIssueFilters filters;
  66. ActiveRulesBuilder activeRulesBuilder = new ActiveRulesBuilder();
  67. RulesBuilder ruleBuilder = new RulesBuilder();
  68. ModuleIssues moduleIssues;
  69. DefaultInputFile file = new TestInputFileBuilder("foo", "src/Foo.php").initMetadata("Foo\nBar\nBiz\n").build();
  70. ReportPublisher reportPublisher = mock(ReportPublisher.class, RETURNS_DEEP_STUBS);
  71. @Before
  72. public void prepare() throws IOException {
  73. projectRoot = new DefaultInputModule(ProjectDefinition.create()
  74. .setKey("foo")
  75. .setBaseDir(temp.newFolder())
  76. .setWorkDir(temp.newFolder()));
  77. }
  78. @Test
  79. public void ignore_null_active_rule() {
  80. ruleBuilder.add(SQUID_RULE_KEY).setName(SQUID_RULE_NAME);
  81. initModuleIssues();
  82. DefaultIssue issue = new DefaultIssue(projectRoot)
  83. .at(new DefaultIssueLocation().on(file).at(file.selectLine(3)).message("Foo"))
  84. .forRule(SQUID_RULE_KEY);
  85. boolean added = moduleIssues.initAndAddIssue(issue);
  86. assertThat(added).isFalse();
  87. verifyZeroInteractions(reportPublisher);
  88. }
  89. @Test
  90. public void ignore_null_rule_of_active_rule() {
  91. ruleBuilder.add(SQUID_RULE_KEY).setName(SQUID_RULE_NAME);
  92. activeRulesBuilder.addRule(new NewActiveRule.Builder().setRuleKey(SQUID_RULE_KEY).setQProfileKey("qp-1").build());
  93. initModuleIssues();
  94. DefaultIssue issue = new DefaultIssue(projectRoot)
  95. .at(new DefaultIssueLocation().on(file).at(file.selectLine(3)).message("Foo"))
  96. .forRule(SQUID_RULE_KEY);
  97. boolean added = moduleIssues.initAndAddIssue(issue);
  98. assertThat(added).isFalse();
  99. verifyZeroInteractions(reportPublisher);
  100. }
  101. @Test
  102. public void add_issue_to_cache() {
  103. ruleBuilder.add(SQUID_RULE_KEY).setName(SQUID_RULE_NAME);
  104. activeRulesBuilder.addRule(new NewActiveRule.Builder()
  105. .setRuleKey(SQUID_RULE_KEY)
  106. .setSeverity(Severity.INFO)
  107. .setQProfileKey("qp-1")
  108. .build());
  109. initModuleIssues();
  110. DefaultIssue issue = new DefaultIssue(projectRoot)
  111. .at(new DefaultIssueLocation().on(file).at(file.selectLine(3)).message("Foo"))
  112. .forRule(SQUID_RULE_KEY)
  113. .overrideSeverity(org.sonar.api.batch.rule.Severity.CRITICAL);
  114. when(filters.accept(anyString(), any(ScannerReport.Issue.class))).thenReturn(true);
  115. boolean added = moduleIssues.initAndAddIssue(issue);
  116. assertThat(added).isTrue();
  117. ArgumentCaptor<ScannerReport.Issue> argument = ArgumentCaptor.forClass(ScannerReport.Issue.class);
  118. verify(reportPublisher.getWriter()).appendComponentIssue(eq(file.batchId()), argument.capture());
  119. assertThat(argument.getValue().getSeverity()).isEqualTo(org.sonar.scanner.protocol.Constants.Severity.CRITICAL);
  120. }
  121. @Test
  122. public void add_external_issue_to_cache() {
  123. ruleBuilder.add(SQUID_RULE_KEY).setName(SQUID_RULE_NAME);
  124. initModuleIssues();
  125. DefaultExternalIssue issue = new DefaultExternalIssue(projectRoot)
  126. .at(new DefaultIssueLocation().on(file).at(file.selectLine(3)).message("Foo"))
  127. .type(RuleType.BUG)
  128. .forRule(SQUID_RULE_KEY)
  129. .severity(org.sonar.api.batch.rule.Severity.CRITICAL);
  130. moduleIssues.initAndAddExternalIssue(issue);
  131. ArgumentCaptor<ScannerReport.ExternalIssue> argument = ArgumentCaptor.forClass(ScannerReport.ExternalIssue.class);
  132. verify(reportPublisher.getWriter()).appendComponentExternalIssue(eq(file.batchId()), argument.capture());
  133. assertThat(argument.getValue().getSeverity()).isEqualTo(org.sonar.scanner.protocol.Constants.Severity.CRITICAL);
  134. }
  135. @Test
  136. public void use_severity_from_active_rule_if_no_severity_on_issue() {
  137. ruleBuilder.add(SQUID_RULE_KEY).setName(SQUID_RULE_NAME);
  138. activeRulesBuilder.addRule(new NewActiveRule.Builder()
  139. .setRuleKey(SQUID_RULE_KEY)
  140. .setSeverity(Severity.INFO)
  141. .setQProfileKey("qp-1")
  142. .build());
  143. initModuleIssues();
  144. DefaultIssue issue = new DefaultIssue(projectRoot)
  145. .at(new DefaultIssueLocation().on(file).at(file.selectLine(3)).message("Foo"))
  146. .forRule(SQUID_RULE_KEY);
  147. when(filters.accept(anyString(), any(ScannerReport.Issue.class))).thenReturn(true);
  148. moduleIssues.initAndAddIssue(issue);
  149. ArgumentCaptor<ScannerReport.Issue> argument = ArgumentCaptor.forClass(ScannerReport.Issue.class);
  150. verify(reportPublisher.getWriter()).appendComponentIssue(eq(file.batchId()), argument.capture());
  151. assertThat(argument.getValue().getSeverity()).isEqualTo(org.sonar.scanner.protocol.Constants.Severity.INFO);
  152. }
  153. @Test
  154. public void filter_issue() {
  155. ruleBuilder.add(SQUID_RULE_KEY).setName(SQUID_RULE_NAME);
  156. activeRulesBuilder.addRule(new NewActiveRule.Builder()
  157. .setRuleKey(SQUID_RULE_KEY)
  158. .setSeverity(Severity.INFO)
  159. .setQProfileKey("qp-1")
  160. .build());
  161. initModuleIssues();
  162. DefaultIssue issue = new DefaultIssue(projectRoot)
  163. .at(new DefaultIssueLocation().on(file).at(file.selectLine(3)).message(""))
  164. .forRule(SQUID_RULE_KEY);
  165. when(filters.accept(anyString(), any(ScannerReport.Issue.class))).thenReturn(false);
  166. boolean added = moduleIssues.initAndAddIssue(issue);
  167. assertThat(added).isFalse();
  168. verifyZeroInteractions(reportPublisher);
  169. }
  170. @Test
  171. public void should_ignore_lines_commented_with_nosonar() {
  172. ruleBuilder.add(SQUID_RULE_KEY).setName(SQUID_RULE_NAME);
  173. activeRulesBuilder.addRule(new NewActiveRule.Builder()
  174. .setRuleKey(SQUID_RULE_KEY)
  175. .setSeverity(Severity.INFO)
  176. .setQProfileKey("qp-1")
  177. .build());
  178. initModuleIssues();
  179. DefaultIssue issue = new DefaultIssue(projectRoot)
  180. .at(new DefaultIssueLocation().on(file).at(file.selectLine(3)).message(""))
  181. .forRule(SQUID_RULE_KEY);
  182. file.noSonarAt(new HashSet<>(Collections.singletonList(3)));
  183. boolean added = moduleIssues.initAndAddIssue(issue);
  184. assertThat(added).isFalse();
  185. verifyZeroInteractions(reportPublisher);
  186. }
  187. @Test
  188. public void should_accept_issues_on_no_sonar_rules() {
  189. // The "No Sonar" rule logs violations on the lines that are flagged with "NOSONAR" !!
  190. ruleBuilder.add(NOSONAR_RULE_KEY).setName("No Sonar");
  191. activeRulesBuilder.addRule(new NewActiveRule.Builder()
  192. .setRuleKey(NOSONAR_RULE_KEY)
  193. .setSeverity(Severity.INFO)
  194. .setQProfileKey("qp-1")
  195. .build());
  196. initModuleIssues();
  197. file.noSonarAt(new HashSet<>(Collections.singletonList(3)));
  198. DefaultIssue issue = new DefaultIssue(projectRoot)
  199. .at(new DefaultIssueLocation().on(file).at(file.selectLine(3)).message(""))
  200. .forRule(NOSONAR_RULE_KEY);
  201. when(filters.accept(anyString(), any(ScannerReport.Issue.class))).thenReturn(true);
  202. boolean added = moduleIssues.initAndAddIssue(issue);
  203. assertThat(added).isTrue();
  204. verify(reportPublisher.getWriter()).appendComponentIssue(eq(file.batchId()), any());
  205. }
  206. /**
  207. * Every rules and active rules has to be added in builders before creating ModuleIssues
  208. */
  209. private void initModuleIssues() {
  210. moduleIssues = new ModuleIssues(activeRulesBuilder.build(), filters, reportPublisher);
  211. }
  212. }