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.

RuleIndexer.java 6.3KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178
  1. /*
  2. * SonarQube
  3. * Copyright (C) 2009-2023 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.rule.index;
  21. import com.google.common.collect.ImmutableSet;
  22. import com.google.common.collect.ListMultimap;
  23. import java.util.Collection;
  24. import java.util.List;
  25. import java.util.Optional;
  26. import java.util.Set;
  27. import java.util.stream.Collectors;
  28. import java.util.stream.Stream;
  29. import org.slf4j.Logger;
  30. import org.slf4j.LoggerFactory;
  31. import org.sonar.core.util.stream.MoreCollectors;
  32. import org.sonar.db.DbClient;
  33. import org.sonar.db.DbSession;
  34. import org.sonar.db.es.EsQueueDto;
  35. import org.sonar.db.rule.RuleForIndexingDto;
  36. import org.sonar.server.es.BulkIndexer;
  37. import org.sonar.server.es.BulkIndexer.Size;
  38. import org.sonar.server.es.EsClient;
  39. import org.sonar.server.es.IndexType;
  40. import org.sonar.server.es.IndexingListener;
  41. import org.sonar.server.es.IndexingResult;
  42. import org.sonar.server.es.OneToOneResilientIndexingListener;
  43. import org.sonar.server.es.ResilientIndexer;
  44. import org.sonar.server.security.SecurityStandards;
  45. import static java.util.Arrays.asList;
  46. import static java.util.stream.Collectors.joining;
  47. import static java.util.stream.Stream.concat;
  48. import static org.sonar.server.rule.index.RuleIndexDefinition.TYPE_RULE;
  49. import static org.sonar.server.security.SecurityStandards.SQ_CATEGORY_KEYS_ORDERING;
  50. public class RuleIndexer implements ResilientIndexer {
  51. private static final Logger LOG = LoggerFactory.getLogger(RuleIndexer.class);
  52. private final EsClient esClient;
  53. private final DbClient dbClient;
  54. public RuleIndexer(EsClient esClient, DbClient dbClient) {
  55. this.esClient = esClient;
  56. this.dbClient = dbClient;
  57. }
  58. @Override
  59. public Set<IndexType> getIndexTypes() {
  60. return ImmutableSet.of(TYPE_RULE);
  61. }
  62. @Override
  63. public void indexOnStartup(Set<IndexType> uninitializedIndexTypes) {
  64. if (uninitializedIndexTypes.contains(TYPE_RULE)) {
  65. indexAll(Size.LARGE);
  66. }
  67. }
  68. public void indexAll() {
  69. indexAll(Size.REGULAR);
  70. }
  71. private void indexAll(Size bulkSize) {
  72. try (DbSession dbSession = dbClient.openSession(false)) {
  73. BulkIndexer bulk = createBulkIndexer(bulkSize, IndexingListener.FAIL_ON_ERROR);
  74. bulk.start();
  75. dbClient.ruleDao().selectIndexingRules(dbSession, dto -> bulk.add(ruleDocOf(dto).toIndexRequest()));
  76. bulk.stop();
  77. }
  78. }
  79. public void commitAndIndex(DbSession dbSession, Collection<String> ruleUuids) {
  80. List<EsQueueDto> items = ruleUuids.stream()
  81. .map(RuleIndexer::createQueueDtoForRule)
  82. .toList();
  83. dbClient.esQueueDao().insert(dbSession, items);
  84. dbSession.commit();
  85. postCommit(dbSession, items);
  86. }
  87. /**
  88. * Commit a change on a rule and its extension
  89. */
  90. public void commitAndIndex(DbSession dbSession, String ruleUuid) {
  91. List<EsQueueDto> items = asList(createQueueDtoForRule(ruleUuid));
  92. dbClient.esQueueDao().insert(dbSession, items);
  93. dbSession.commit();
  94. postCommit(dbSession, items);
  95. }
  96. /**
  97. * This method is used by the Byteman script of integration tests.
  98. */
  99. private void postCommit(DbSession dbSession, List<EsQueueDto> items) {
  100. index(dbSession, items);
  101. }
  102. @Override
  103. public IndexingResult index(DbSession dbSession, Collection<EsQueueDto> items) {
  104. IndexingResult result = new IndexingResult();
  105. if (!items.isEmpty()) {
  106. ListMultimap<String, EsQueueDto> itemsByType = groupItemsByIndexTypeFormat(items);
  107. doIndexRules(dbSession, itemsByType.get(TYPE_RULE.format())).ifPresent(result::add);
  108. }
  109. return result;
  110. }
  111. private Optional<IndexingResult> doIndexRules(DbSession dbSession, List<EsQueueDto> items) {
  112. if (items.isEmpty()) {
  113. return Optional.empty();
  114. }
  115. BulkIndexer bulkIndexer = createBulkIndexer(Size.REGULAR, new OneToOneResilientIndexingListener(dbClient, dbSession, items));
  116. bulkIndexer.start();
  117. Set<String> ruleUuids = items
  118. .stream()
  119. .map(EsQueueDto::getDocId)
  120. .collect(Collectors.toSet());
  121. dbClient.ruleDao().selectIndexingRulesByKeys(dbSession, ruleUuids,
  122. r -> {
  123. bulkIndexer.add(ruleDocOf(r).toIndexRequest());
  124. ruleUuids.remove(r.getUuid());
  125. });
  126. // the remaining items reference rows that don't exist in db. They must be deleted from index.
  127. ruleUuids.forEach(ruleUuid -> bulkIndexer.addDeletion(TYPE_RULE, ruleUuid, ruleUuid));
  128. return Optional.of(bulkIndexer.stop());
  129. }
  130. private static RuleDoc ruleDocOf(RuleForIndexingDto dto) {
  131. SecurityStandards securityStandards = SecurityStandards.fromSecurityStandards(dto.getSecurityStandards());
  132. if (!securityStandards.getIgnoredSQCategories().isEmpty()) {
  133. LOG.debug(
  134. "Rule {} with CWEs '{}' maps to multiple SQ Security Categories: {}",
  135. dto.getRuleKey(),
  136. String.join(", ", securityStandards.getCwe()),
  137. concat(Stream.of(securityStandards.getSqCategory()), securityStandards.getIgnoredSQCategories().stream())
  138. .map(SecurityStandards.SQCategory::getKey)
  139. .sorted(SQ_CATEGORY_KEYS_ORDERING)
  140. .collect(joining(", ")));
  141. }
  142. return RuleDoc.createFrom(dto, securityStandards);
  143. }
  144. private BulkIndexer createBulkIndexer(Size bulkSize, IndexingListener listener) {
  145. return new BulkIndexer(esClient, TYPE_RULE, bulkSize, listener);
  146. }
  147. private static ListMultimap<String, EsQueueDto> groupItemsByIndexTypeFormat(Collection<EsQueueDto> items) {
  148. return items.stream().collect(MoreCollectors.index(EsQueueDto::getDocType));
  149. }
  150. private static EsQueueDto createQueueDtoForRule(String ruleUuid) {
  151. return EsQueueDto.create(TYPE_RULE.format(), ruleUuid, null, ruleUuid);
  152. }
  153. }