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.

RegisterQualityProfilesNotificationTest.java 18KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365
  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.server.qualityprofile;
  21. import com.google.common.collect.Multimap;
  22. import java.security.SecureRandom;
  23. import java.util.Arrays;
  24. import java.util.List;
  25. import java.util.Map;
  26. import java.util.Random;
  27. import java.util.function.Consumer;
  28. import org.junit.Rule;
  29. import org.junit.Test;
  30. import org.junit.rules.ExpectedException;
  31. import org.mockito.ArgumentCaptor;
  32. import org.sonar.api.rule.RuleKey;
  33. import org.sonar.api.rule.Severity;
  34. import org.sonar.api.rules.RulePriority;
  35. import org.sonar.api.server.profile.BuiltInQualityProfilesDefinition;
  36. import org.sonar.api.server.profile.BuiltInQualityProfilesDefinition.BuiltInActiveRule;
  37. import org.sonar.api.server.profile.BuiltInQualityProfilesDefinition.NewBuiltInQualityProfile;
  38. import org.sonar.api.utils.System2;
  39. import org.sonar.api.utils.log.LogTester;
  40. import org.sonar.core.util.UuidFactoryFast;
  41. import org.sonar.core.util.stream.MoreCollectors;
  42. import org.sonar.db.DbClient;
  43. import org.sonar.db.DbTester;
  44. import org.sonar.db.organization.OrganizationDto;
  45. import org.sonar.db.qualityprofile.ActiveRuleDto;
  46. import org.sonar.db.qualityprofile.QProfileDto;
  47. import org.sonar.db.qualityprofile.RulesProfileDto;
  48. import org.sonar.db.rule.RuleDefinitionDto;
  49. import org.sonar.server.qualityprofile.index.ActiveRuleIndexer;
  50. import org.sonar.server.rule.index.RuleIndex;
  51. import org.sonar.server.tester.UserSessionRule;
  52. import org.sonar.server.util.TypeValidations;
  53. import static com.google.common.base.Preconditions.checkState;
  54. import static java.util.Collections.singleton;
  55. import static org.apache.commons.lang.RandomStringUtils.randomAlphanumeric;
  56. import static org.apache.commons.lang.math.RandomUtils.nextLong;
  57. import static org.assertj.core.api.Assertions.assertThat;
  58. import static org.assertj.core.api.Assertions.tuple;
  59. import static org.mockito.ArgumentMatchers.any;
  60. import static org.mockito.ArgumentMatchers.anyLong;
  61. import static org.mockito.ArgumentMatchers.eq;
  62. import static org.mockito.Mockito.mock;
  63. import static org.mockito.Mockito.verify;
  64. import static org.mockito.Mockito.verifyZeroInteractions;
  65. import static org.mockito.Mockito.when;
  66. import static org.sonar.api.rules.RulePriority.MAJOR;
  67. import static org.sonar.db.qualityprofile.QualityProfileTesting.newRuleProfileDto;
  68. import static org.sonar.server.language.LanguageTesting.newLanguage;
  69. import static org.sonar.server.qualityprofile.ActiveRuleChange.Type.ACTIVATED;
  70. import static org.sonar.server.qualityprofile.ActiveRuleChange.Type.DEACTIVATED;
  71. import static org.sonar.server.qualityprofile.ActiveRuleChange.Type.UPDATED;
  72. public class RegisterQualityProfilesNotificationTest {
  73. private static final Random RANDOM = new SecureRandom();
  74. private System2 system2 = mock(System2.class);
  75. @Rule
  76. public DbTester db = DbTester.create(system2);
  77. @Rule
  78. public UserSessionRule userSessionRule = UserSessionRule.standalone();
  79. @Rule
  80. public ExpectedException expectedException = ExpectedException.none();
  81. @Rule
  82. public BuiltInQProfileRepositoryRule builtInQProfileRepositoryRule = new BuiltInQProfileRepositoryRule();
  83. @Rule
  84. public LogTester logTester = new LogTester();
  85. private DbClient dbClient = db.getDbClient();
  86. private TypeValidations typeValidations = mock(TypeValidations.class);
  87. private ActiveRuleIndexer activeRuleIndexer = mock(ActiveRuleIndexer.class);
  88. private BuiltInQProfileInsert builtInQProfileInsert = new BuiltInQProfileInsertImpl(dbClient, system2, UuidFactoryFast.getInstance(), typeValidations, activeRuleIndexer);
  89. private RuleActivator ruleActivator = new RuleActivator(system2, dbClient, typeValidations, userSessionRule);
  90. private QProfileRules qProfileRules = new QProfileRulesImpl(dbClient, ruleActivator, mock(RuleIndex.class), activeRuleIndexer);
  91. private BuiltInQProfileUpdate builtInQProfileUpdate = new BuiltInQProfileUpdateImpl(dbClient, ruleActivator, activeRuleIndexer);
  92. private BuiltInQualityProfilesUpdateListener builtInQualityProfilesNotification = mock(BuiltInQualityProfilesUpdateListener.class);
  93. private RegisterQualityProfiles underTest = new RegisterQualityProfiles(builtInQProfileRepositoryRule, dbClient,
  94. builtInQProfileInsert, builtInQProfileUpdate, builtInQualityProfilesNotification, system2);
  95. @Test
  96. public void do_not_send_notification_on_new_profile() {
  97. String language = newLanguageKey();
  98. builtInQProfileRepositoryRule.add(newLanguage(language), "Sonar way");
  99. builtInQProfileRepositoryRule.initialize();
  100. underTest.start();
  101. verifyZeroInteractions(builtInQualityProfilesNotification);
  102. }
  103. @Test
  104. public void do_not_send_notification_when_profile_is_not_updated() {
  105. String language = newLanguageKey();
  106. RuleDefinitionDto dbRule = db.rules().insert(r -> r.setLanguage(language));
  107. RulesProfileDto dbProfile = insertBuiltInProfile(language);
  108. activateRuleInDb(dbProfile, dbRule, MAJOR);
  109. addPluginProfile(dbProfile, dbRule);
  110. builtInQProfileRepositoryRule.initialize();
  111. underTest.start();
  112. verifyZeroInteractions(builtInQualityProfilesNotification);
  113. }
  114. @Test
  115. public void send_notification_when_a_new_rule_is_activated() {
  116. String language = newLanguageKey();
  117. RuleDefinitionDto existingRule = db.rules().insert(r -> r.setLanguage(language));
  118. RulesProfileDto dbProfile = insertBuiltInProfile(language);
  119. activateRuleInDb(dbProfile, existingRule, MAJOR);
  120. RuleDefinitionDto newRule = db.rules().insert(r -> r.setLanguage(language));
  121. addPluginProfile(dbProfile, existingRule, newRule);
  122. builtInQProfileRepositoryRule.initialize();
  123. underTest.start();
  124. ArgumentCaptor<Multimap> captor = ArgumentCaptor.forClass(Multimap.class);
  125. verify(builtInQualityProfilesNotification).onChange(captor.capture(), anyLong(), anyLong());
  126. Multimap<QProfileName, ActiveRuleChange> updatedProfiles = captor.<Multimap<QProfileName, ActiveRuleChange>>getValue();
  127. assertThat(updatedProfiles.keySet())
  128. .extracting(QProfileName::getName, QProfileName::getLanguage)
  129. .containsExactlyInAnyOrder(tuple(dbProfile.getName(), dbProfile.getLanguage()));
  130. assertThat(updatedProfiles.values())
  131. .extracting(value -> value.getActiveRule().getRuleId(), ActiveRuleChange::getType)
  132. .containsExactlyInAnyOrder(tuple(newRule.getId(), ACTIVATED));
  133. }
  134. @Test
  135. public void send_notification_when_a_rule_is_deactivated() {
  136. String language = newLanguageKey();
  137. RuleDefinitionDto existingRule = db.rules().insert(r -> r.setLanguage(language));
  138. RulesProfileDto dbProfile = insertBuiltInProfile(language);
  139. activateRuleInDb(dbProfile, existingRule, MAJOR);
  140. addPluginProfile(dbProfile);
  141. builtInQProfileRepositoryRule.initialize();
  142. underTest.start();
  143. ArgumentCaptor<Multimap> captor = ArgumentCaptor.forClass(Multimap.class);
  144. verify(builtInQualityProfilesNotification).onChange(captor.capture(), anyLong(), anyLong());
  145. Multimap<QProfileName, ActiveRuleChange> updatedProfiles = captor.<Multimap<QProfileName, ActiveRuleChange>>getValue();
  146. assertThat(updatedProfiles.keySet())
  147. .extracting(QProfileName::getName, QProfileName::getLanguage)
  148. .containsExactlyInAnyOrder(tuple(dbProfile.getName(), dbProfile.getLanguage()));
  149. assertThat(updatedProfiles.values())
  150. .extracting(value -> value.getActiveRule().getRuleId(), ActiveRuleChange::getType)
  151. .containsExactlyInAnyOrder(tuple(existingRule.getId(), DEACTIVATED));
  152. }
  153. @Test
  154. public void send_a_single_notification_when_multiple_rules_are_activated() {
  155. String language = newLanguageKey();
  156. RuleDefinitionDto existingRule1 = db.rules().insert(r -> r.setLanguage(language));
  157. RuleDefinitionDto newRule1 = db.rules().insert(r -> r.setLanguage(language));
  158. RulesProfileDto dbProfile1 = insertBuiltInProfile(language);
  159. activateRuleInDb(dbProfile1, existingRule1, MAJOR);
  160. addPluginProfile(dbProfile1, existingRule1, newRule1);
  161. RuleDefinitionDto existingRule2 = db.rules().insert(r -> r.setLanguage(language));
  162. RuleDefinitionDto newRule2 = db.rules().insert(r -> r.setLanguage(language));
  163. RulesProfileDto dbProfile2 = insertBuiltInProfile(language);
  164. activateRuleInDb(dbProfile2, existingRule2, MAJOR);
  165. addPluginProfile(dbProfile2, existingRule2, newRule2);
  166. builtInQProfileRepositoryRule.initialize();
  167. underTest.start();
  168. ArgumentCaptor<Multimap> captor = ArgumentCaptor.forClass(Multimap.class);
  169. verify(builtInQualityProfilesNotification).onChange(captor.capture(), anyLong(), anyLong());
  170. Multimap<QProfileName, ActiveRuleChange> updatedProfiles = captor.<Multimap<QProfileName, ActiveRuleChange>>getValue();
  171. assertThat(updatedProfiles.keySet())
  172. .extracting(QProfileName::getName, QProfileName::getLanguage)
  173. .containsExactlyInAnyOrder(
  174. tuple(dbProfile1.getName(), dbProfile1.getLanguage()),
  175. tuple(dbProfile2.getName(), dbProfile2.getLanguage()));
  176. assertThat(updatedProfiles.values())
  177. .extracting(value -> value.getActiveRule().getRuleId(), ActiveRuleChange::getType)
  178. .containsExactlyInAnyOrder(
  179. tuple(newRule1.getId(), ACTIVATED),
  180. tuple(newRule2.getId(), ACTIVATED));
  181. }
  182. @Test
  183. public void notification_does_not_include_inherited_profiles_when_rule_is_added() {
  184. String language = newLanguageKey();
  185. RuleDefinitionDto newRule = db.rules().insert(r -> r.setLanguage(language));
  186. OrganizationDto organization = db.organizations().insert();
  187. QProfileDto builtInQProfileDto = insertProfile(organization, orgQProfile -> orgQProfile.setIsBuiltIn(true).setLanguage(language));
  188. QProfileDto childQProfileDto = insertProfile(organization, orgQProfile -> orgQProfile.setIsBuiltIn(false).setLanguage(language).setParentKee(builtInQProfileDto.getKee()));
  189. addPluginProfile(builtInQProfileDto, newRule);
  190. builtInQProfileRepositoryRule.initialize();
  191. underTest.start();
  192. ArgumentCaptor<Multimap> captor = ArgumentCaptor.forClass(Multimap.class);
  193. verify(builtInQualityProfilesNotification).onChange(captor.capture(), anyLong(), anyLong());
  194. Multimap<QProfileName, ActiveRuleChange> updatedProfiles = captor.<Multimap<QProfileName, ActiveRuleChange>>getValue();
  195. assertThat(updatedProfiles.keySet())
  196. .extracting(QProfileName::getName, QProfileName::getLanguage)
  197. .containsExactlyInAnyOrder(tuple(builtInQProfileDto.getName(), builtInQProfileDto.getLanguage()));
  198. assertThat(updatedProfiles.values())
  199. .extracting(value -> value.getActiveRule().getRuleId(), ActiveRuleChange::getType)
  200. .containsExactlyInAnyOrder(tuple(newRule.getId(), ACTIVATED));
  201. }
  202. @Test
  203. public void notification_does_not_include_inherited_profiled_when_rule_is_changed() {
  204. String language = newLanguageKey();
  205. RuleDefinitionDto rule = db.rules().insert(r -> r.setLanguage(language).setSeverity(Severity.MINOR));
  206. OrganizationDto organization = db.organizations().insert();
  207. QProfileDto builtInProfile = insertProfile(organization, orgQProfile -> orgQProfile.setIsBuiltIn(true).setLanguage(language));
  208. db.qualityProfiles().activateRule(builtInProfile, rule, ar -> ar.setSeverity(Severity.MINOR));
  209. QProfileDto childProfile = insertProfile(organization, orgQProfile -> orgQProfile.setIsBuiltIn(false).setLanguage(language).setParentKee(builtInProfile.getKee()));
  210. db.qualityProfiles().activateRule(childProfile, rule, ar -> ar.setInheritance(ActiveRuleDto.INHERITED).setSeverity(Severity.MINOR));
  211. addPluginProfile(builtInProfile, rule);
  212. builtInQProfileRepositoryRule.initialize();
  213. db.commit();
  214. underTest.start();
  215. ArgumentCaptor<Multimap> captor = ArgumentCaptor.forClass(Multimap.class);
  216. verify(builtInQualityProfilesNotification).onChange(captor.capture(), anyLong(), anyLong());
  217. Multimap<QProfileName, ActiveRuleChange> updatedProfiles = captor.<Multimap<QProfileName, ActiveRuleChange>>getValue();
  218. assertThat(updatedProfiles.keySet())
  219. .extracting(QProfileName::getName, QProfileName::getLanguage)
  220. .containsExactlyInAnyOrder(tuple(builtInProfile.getName(), builtInProfile.getLanguage()));
  221. assertThat(updatedProfiles.values())
  222. .extracting(value -> value.getActiveRule().getRuleId(), ActiveRuleChange::getType)
  223. .containsExactlyInAnyOrder(tuple(rule.getId(), UPDATED));
  224. }
  225. @Test
  226. public void notification_does_not_include_inherited_profiles_when_rule_is_deactivated() {
  227. String language = newLanguageKey();
  228. RuleDefinitionDto rule = db.rules().insert(r -> r.setLanguage(language).setSeverity(Severity.MINOR));
  229. OrganizationDto organization = db.organizations().insert();
  230. QProfileDto builtInQProfileDto = insertProfile(organization,
  231. orgQProfile -> orgQProfile.setIsBuiltIn(true).setLanguage(language));
  232. db.qualityProfiles().activateRule(builtInQProfileDto, rule);
  233. QProfileDto childQProfileDto = insertProfile(organization,
  234. orgQProfile -> orgQProfile.setIsBuiltIn(false).setLanguage(language).setParentKee(builtInQProfileDto.getKee()));
  235. qProfileRules.activateAndCommit(db.getSession(), childQProfileDto, singleton(RuleActivation.create(rule.getId())));
  236. db.commit();
  237. addPluginProfile(builtInQProfileDto);
  238. builtInQProfileRepositoryRule.initialize();
  239. underTest.start();
  240. ArgumentCaptor<Multimap> captor = ArgumentCaptor.forClass(Multimap.class);
  241. verify(builtInQualityProfilesNotification).onChange(captor.capture(), anyLong(), anyLong());
  242. Multimap<QProfileName, ActiveRuleChange> updatedProfiles = captor.<Multimap<QProfileName, ActiveRuleChange>>getValue();
  243. assertThat(updatedProfiles.keySet())
  244. .extracting(QProfileName::getName, QProfileName::getLanguage)
  245. .containsExactlyInAnyOrder(tuple(builtInQProfileDto.getName(), builtInQProfileDto.getLanguage()));
  246. assertThat(updatedProfiles.values())
  247. .extracting(value -> value.getActiveRule().getRuleId(), ActiveRuleChange::getType)
  248. .containsExactlyInAnyOrder(tuple(rule.getId(), DEACTIVATED));
  249. }
  250. @Test
  251. public void notification_contains_send_start_and_end_date() {
  252. String language = newLanguageKey();
  253. RuleDefinitionDto existingRule = db.rules().insert(r -> r.setLanguage(language));
  254. RulesProfileDto dbProfile = insertBuiltInProfile(language);
  255. activateRuleInDb(dbProfile, existingRule, MAJOR);
  256. RuleDefinitionDto newRule = db.rules().insert(r -> r.setLanguage(language));
  257. addPluginProfile(dbProfile, existingRule, newRule);
  258. builtInQProfileRepositoryRule.initialize();
  259. long startDate = RANDOM.nextInt(5000);
  260. long endDate = startDate + RANDOM.nextInt(5000);
  261. when(system2.now()).thenReturn(startDate, endDate);
  262. underTest.start();
  263. verify(builtInQualityProfilesNotification).onChange(any(), eq(startDate), eq(endDate));
  264. }
  265. private void addPluginProfile(RulesProfileDto dbProfile, RuleDefinitionDto... dbRules) {
  266. BuiltInQualityProfilesDefinition.Context context = new BuiltInQualityProfilesDefinition.Context();
  267. NewBuiltInQualityProfile newQp = context.createBuiltInQualityProfile(dbProfile.getName(), dbProfile.getLanguage());
  268. Arrays.stream(dbRules).forEach(dbRule -> newQp.activateRule(dbRule.getRepositoryKey(), dbRule.getRuleKey()).overrideSeverity(Severity.MAJOR));
  269. newQp.done();
  270. List<BuiltInActiveRule> rules = context.profile(dbProfile.getLanguage(), dbProfile.getName()).rules();
  271. BuiltInQProfile.ActiveRule[] activeRules = toActiveRules(rules, dbRules);
  272. builtInQProfileRepositoryRule.add(newLanguage(dbProfile.getLanguage()), dbProfile.getName(), false, activeRules);
  273. }
  274. private static BuiltInQProfile.ActiveRule[] toActiveRules(List<BuiltInActiveRule> rules, RuleDefinitionDto[] dbRules) {
  275. Map<RuleKey, RuleDefinitionDto> dbRulesByRuleKey = Arrays.stream(dbRules)
  276. .collect(MoreCollectors.uniqueIndex(RuleDefinitionDto::getKey));
  277. return rules.stream()
  278. .map(r -> {
  279. RuleKey ruleKey = RuleKey.of(r.repoKey(), r.ruleKey());
  280. RuleDefinitionDto ruleDefinitionDto = dbRulesByRuleKey.get(ruleKey);
  281. checkState(ruleDefinitionDto != null, "Rule '%s' not found", ruleKey);
  282. return new BuiltInQProfile.ActiveRule(ruleDefinitionDto.getId(), r);
  283. }).toArray(BuiltInQProfile.ActiveRule[]::new);
  284. }
  285. private void addPluginProfile(QProfileDto profile, RuleDefinitionDto... dbRules) {
  286. BuiltInQualityProfilesDefinition.Context context = new BuiltInQualityProfilesDefinition.Context();
  287. NewBuiltInQualityProfile newQp = context.createBuiltInQualityProfile(profile.getName(), profile.getLanguage());
  288. Arrays.stream(dbRules).forEach(dbRule -> newQp.activateRule(dbRule.getRepositoryKey(), dbRule.getRuleKey()).overrideSeverity(Severity.MAJOR));
  289. newQp.done();
  290. BuiltInQProfile.ActiveRule[] activeRules = toActiveRules(context.profile(profile.getLanguage(), profile.getName()).rules(), dbRules);
  291. builtInQProfileRepositoryRule.add(newLanguage(profile.getLanguage()), profile.getName(), false, activeRules);
  292. }
  293. private RulesProfileDto insertBuiltInProfile(String language) {
  294. RulesProfileDto ruleProfileDto = newRuleProfileDto(rp -> rp.setIsBuiltIn(true).setLanguage(language));
  295. db.getDbClient().qualityProfileDao().insert(db.getSession(), ruleProfileDto);
  296. db.commit();
  297. return ruleProfileDto;
  298. }
  299. private void activateRuleInDb(RulesProfileDto profile, RuleDefinitionDto rule, RulePriority severity) {
  300. ActiveRuleDto dto = new ActiveRuleDto()
  301. .setProfileId(profile.getId())
  302. .setSeverity(severity.name())
  303. .setRuleId(rule.getId())
  304. .setCreatedAt(nextLong())
  305. .setUpdatedAt(nextLong());
  306. db.getDbClient().activeRuleDao().insert(db.getSession(), dto);
  307. db.commit();
  308. }
  309. private QProfileDto insertProfile(OrganizationDto organization, Consumer<QProfileDto> consumer) {
  310. QProfileDto builtInQProfileDto = db.qualityProfiles().insert(organization, consumer);
  311. db.commit();
  312. return builtInQProfileDto;
  313. }
  314. private static String newLanguageKey() {
  315. return randomAlphanumeric(20).toLowerCase();
  316. }
  317. }