aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--server/sonar-db-core/src/main/resources/org/sonar/db/version/schema-h2.ddl2
-rw-r--r--server/sonar-db-dao/src/main/java/org/sonar/db/es/EsQueueDto.java34
-rw-r--r--server/sonar-db-dao/src/main/java/org/sonar/db/qualityprofile/ActiveRuleDao.java26
-rw-r--r--server/sonar-db-dao/src/main/java/org/sonar/db/qualityprofile/ActiveRuleMapper.java6
-rw-r--r--server/sonar-db-dao/src/main/java/org/sonar/db/qualityprofile/IndexedActiveRuleDto.java (renamed from server/sonar-server/src/main/java/org/sonar/server/rule/index/RuleDocWithSystemScope.java)38
-rw-r--r--server/sonar-db-dao/src/main/java/org/sonar/db/qualityprofile/QualityProfileDao.java5
-rw-r--r--server/sonar-db-dao/src/main/java/org/sonar/db/qualityprofile/QualityProfileMapper.java3
-rw-r--r--server/sonar-db-dao/src/main/java/org/sonar/db/rule/RuleDao.java24
-rw-r--r--server/sonar-db-dao/src/main/java/org/sonar/db/rule/RuleForIndexingDto.java6
-rw-r--r--server/sonar-db-dao/src/main/java/org/sonar/db/rule/RuleMapper.java8
-rw-r--r--server/sonar-db-dao/src/main/java/org/sonar/db/rule/RuleMetadataDto.java4
-rw-r--r--server/sonar-db-dao/src/main/resources/org/sonar/db/es/EsQueueMapper.xml6
-rw-r--r--server/sonar-db-dao/src/main/resources/org/sonar/db/qualityprofile/ActiveRuleMapper.xml27
-rw-r--r--server/sonar-db-dao/src/main/resources/org/sonar/db/qualityprofile/QualityProfileMapper.xml6
-rw-r--r--server/sonar-db-dao/src/main/resources/org/sonar/db/rule/RuleMapper.xml76
-rw-r--r--server/sonar-db-dao/src/test/java/org/sonar/db/qualityprofile/ActiveRuleDaoTest.java66
-rw-r--r--server/sonar-db-dao/src/test/java/org/sonar/db/qualityprofile/QualityProfileDaoTest.java20
-rw-r--r--server/sonar-db-dao/src/test/java/org/sonar/db/rule/RuleDaoTest.java300
-rw-r--r--server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v65/CreateEsQueueTable.java10
-rw-r--r--server/sonar-db-migration/src/test/java/org/sonar/server/platform/db/migration/version/v65/CreateEsQueueTableTest.java2
-rw-r--r--server/sonar-db-migration/src/test/resources/org/sonar/server/platform/db/migration/version/v65/AddIndexOnEsQueueCreatedAtTest/initial.sql4
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/es/BulkIndexer.java79
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/es/IndexingListener.java (renamed from server/sonar-server/src/main/java/org/sonar/server/rule/index/RuleIterator.java)12
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/es/IndexingResult.java (renamed from server/sonar-server/src/main/java/org/sonar/server/es/ResilientIndexerResult.java)52
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/es/RecoveryIndexer.java16
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/es/ResiliencyIndexingListener.java64
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/es/ResilientIndexer.java2
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/es/queue/package-info.java23
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/platform/platformlevel/PlatformLevel4.java2
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/qualityprofile/BuiltInQProfileInsertImpl.java8
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/qualityprofile/BuiltInQProfileUpdateImpl.java3
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/qualityprofile/QProfileFactoryImpl.java8
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/qualityprofile/QProfileResetImpl.java3
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/qualityprofile/RuleActivator.java26
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/qualityprofile/index/ActiveRuleIndexer.java203
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/qualityprofile/index/ActiveRuleIterator.java27
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/qualityprofile/index/ActiveRuleIteratorFactory.java48
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/qualityprofile/index/ActiveRuleIteratorForMultipleChunks.java69
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/qualityprofile/index/ActiveRuleIteratorForSingleChunk.java132
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/qualityprofile/ws/ActivateRuleAction.java11
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/qualityprofile/ws/ActivateRulesAction.java2
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/qualityprofile/ws/ChangeParentAction.java4
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/qualityprofile/ws/CreateAction.java3
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/qualityprofile/ws/DeactivateRuleAction.java2
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/qualityprofile/ws/DeactivateRulesAction.java2
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/rule/RegisterRules.java2
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/rule/index/RuleDoc.java40
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/rule/index/RuleIndexer.java125
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/rule/index/RuleIteratorForMultipleChunks.java80
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/rule/index/RuleIteratorForSingleChunk.java211
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/rule/index/RuleMetadataIterator.java126
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/user/index/UserIndexer.java19
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/user/index/UserResultSetIterator.java121
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/es/BulkIndexerTest.java7
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/es/EsTester.java8
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/es/IndexingResultTest.java78
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/es/RecoveryIndexerTest.java39
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/es/ResilientIndexerResultTest.java62
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/issue/index/IssueIndexTest.java2
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/organization/ws/EnableSupportActionTest.java13
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/qualityprofile/QProfileFactoryImplTest.java19
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/qualityprofile/RuleActivatorTest.java34
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/qualityprofile/index/ActiveRuleIndexerTest.java138
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/qualityprofile/ws/ActivateRuleActionTest.java8
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/qualityprofile/ws/ChangeParentActionTest.java9
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/qualityprofile/ws/CreateActionTest.java3
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/qualityprofile/ws/DeactivateRuleActionTest.java4
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/qualityprofile/ws/InheritanceActionTest.java5
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/qualityprofile/ws/ShowActionTest.java3
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/rule/RegisterRulesTest.java3
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/rule/index/RuleIndexTest.java3
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/rule/index/RuleIteratorForSingleChunkTest.java201
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/rule/ws/DeleteActionTest.java4
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/rule/ws/SearchActionTest.java3
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/rule/ws/ShowActionTest.java5
-rw-r--r--tests/resilience/active_rule_indexer.btm12
-rw-r--r--tests/src/test/java/org/sonarqube/tests/Category5Suite.java2
-rw-r--r--tests/src/test/java/org/sonarqube/tests/qualityProfile/ActiveRuleEsResilienceTest.java107
78 files changed, 1264 insertions, 1706 deletions
diff --git a/server/sonar-db-core/src/main/resources/org/sonar/db/version/schema-h2.ddl b/server/sonar-db-core/src/main/resources/org/sonar/db/version/schema-h2.ddl
index 3942cb98f86..81c8fe51945 100644
--- a/server/sonar-db-core/src/main/resources/org/sonar/db/version/schema-h2.ddl
+++ b/server/sonar-db-core/src/main/resources/org/sonar/db/version/schema-h2.ddl
@@ -663,6 +663,8 @@ CREATE TABLE "ES_QUEUE" (
"UUID" VARCHAR(40) NOT NULL PRIMARY KEY,
"DOC_TYPE" VARCHAR(40) NOT NULL,
"DOC_ID" VARCHAR(4000) NOT NULL,
+ "DOC_ID_TYPE" VARCHAR(20),
+ "DOC_ROUTING" VARCHAR(4000),
"CREATED_AT" BIGINT NOT NULL
);
CREATE UNIQUE INDEX "PK_ES_QUEUE" ON "ES_QUEUE" ("UUID");
diff --git a/server/sonar-db-dao/src/main/java/org/sonar/db/es/EsQueueDto.java b/server/sonar-db-dao/src/main/java/org/sonar/db/es/EsQueueDto.java
index 35c51c7f2e5..26f310d34ca 100644
--- a/server/sonar-db-dao/src/main/java/org/sonar/db/es/EsQueueDto.java
+++ b/server/sonar-db-dao/src/main/java/org/sonar/db/es/EsQueueDto.java
@@ -19,15 +19,20 @@
*/
package org.sonar.db.es;
+import javax.annotation.CheckForNull;
+import javax.annotation.Nullable;
+
public final class EsQueueDto {
public enum Type {
- USER, RULE, RULE_EXTENSION
+ USER, RULE, RULE_EXTENSION, ACTIVE_RULE
}
private String uuid;
private Type docType;
private String docId;
+ private String docIdType;
+ private String docRouting;
public String getUuid() {
return uuid;
@@ -56,12 +61,34 @@ public final class EsQueueDto {
return this;
}
+ @CheckForNull
+ public String getDocIdType() {
+ return docIdType;
+ }
+
+ private EsQueueDto setDocIdType(@Nullable String s) {
+ this.docIdType = s;
+ return this;
+ }
+
+ @CheckForNull
+ public String getDocRouting() {
+ return docRouting;
+ }
+
+ private EsQueueDto setDocRouting(@Nullable String s) {
+ this.docRouting = s;
+ return this;
+ }
+
@Override
public String toString() {
StringBuilder sb = new StringBuilder("EsQueueDto{");
sb.append("uuid='").append(uuid).append('\'');
sb.append(", docType=").append(docType);
sb.append(", docId='").append(docId).append('\'');
+ sb.append(", docIdType='").append(docIdType).append('\'');
+ sb.append(", docRouting='").append(docRouting).append('\'');
sb.append('}');
return sb.toString();
}
@@ -88,4 +115,9 @@ public final class EsQueueDto {
public static EsQueueDto create(Type docType, String docUuid) {
return new EsQueueDto().setDocType(docType).setDocId(docUuid);
}
+
+ public static EsQueueDto create(Type docType, String docId, @Nullable String docIdType, @Nullable String docRouting) {
+ return new EsQueueDto().setDocType(docType)
+ .setDocId(docId).setDocIdType(docIdType).setDocRouting(docRouting);
+ }
}
diff --git a/server/sonar-db-dao/src/main/java/org/sonar/db/qualityprofile/ActiveRuleDao.java b/server/sonar-db-dao/src/main/java/org/sonar/db/qualityprofile/ActiveRuleDao.java
index a699eab7870..b19f4ee848f 100644
--- a/server/sonar-db-dao/src/main/java/org/sonar/db/qualityprofile/ActiveRuleDao.java
+++ b/server/sonar-db-dao/src/main/java/org/sonar/db/qualityprofile/ActiveRuleDao.java
@@ -24,6 +24,7 @@ import java.util.Collection;
import java.util.List;
import java.util.Map;
import java.util.Optional;
+import java.util.function.Consumer;
import org.sonar.db.Dao;
import org.sonar.db.DatabaseUtils;
import org.sonar.db.DbSession;
@@ -31,6 +32,7 @@ import org.sonar.db.organization.OrganizationDto;
import org.sonar.db.rule.RuleParamDto;
import static org.sonar.db.DatabaseUtils.executeLargeInputs;
+import static org.sonar.db.DatabaseUtils.executeLargeInputsWithoutOutput;
import static org.sonar.db.KeyLongValue.toMap;
public class ActiveRuleDao implements Dao {
@@ -170,6 +172,30 @@ public class ActiveRuleDao implements Dao {
partition -> mapper(dbSession).countActiveRulesByQuery(query.getOrganization().getUuid(), partition, query.getRuleStatus(), query.getInheritance())));
}
+ public void scrollAllForIndexing(DbSession dbSession, Consumer<IndexedActiveRuleDto> consumer) {
+ mapper(dbSession).scrollAllForIndexing(context -> {
+ IndexedActiveRuleDto dto = (IndexedActiveRuleDto) context.getResultObject();
+ consumer.accept(dto);
+ });
+ }
+
+ public void scrollByIdsForIndexing(DbSession dbSession, Collection<Long> ids, Consumer<IndexedActiveRuleDto> consumer) {
+ ActiveRuleMapper mapper = mapper(dbSession);
+ executeLargeInputsWithoutOutput(ids,
+ pageOfIds -> mapper
+ .scrollByIdsForIndexing(pageOfIds, context -> {
+ IndexedActiveRuleDto dto = (IndexedActiveRuleDto) context.getResultObject();
+ consumer.accept(dto);
+ }));
+ }
+
+ public void scrollByRuleProfileForIndexing(DbSession dbSession, String ruleProfileUuid, Consumer<IndexedActiveRuleDto> consumer) {
+ mapper(dbSession).scrollByRuleProfileUuidForIndexing(ruleProfileUuid, context -> {
+ IndexedActiveRuleDto dto = (IndexedActiveRuleDto) context.getResultObject();
+ consumer.accept(dto);
+ });
+ }
+
private static ActiveRuleMapper mapper(DbSession dbSession) {
return dbSession.getMapper(ActiveRuleMapper.class);
}
diff --git a/server/sonar-db-dao/src/main/java/org/sonar/db/qualityprofile/ActiveRuleMapper.java b/server/sonar-db-dao/src/main/java/org/sonar/db/qualityprofile/ActiveRuleMapper.java
index 6a64c21ee4e..2821d3ae0c8 100644
--- a/server/sonar-db-dao/src/main/java/org/sonar/db/qualityprofile/ActiveRuleMapper.java
+++ b/server/sonar-db-dao/src/main/java/org/sonar/db/qualityprofile/ActiveRuleMapper.java
@@ -24,6 +24,7 @@ import java.util.List;
import javax.annotation.CheckForNull;
import javax.annotation.Nullable;
import org.apache.ibatis.annotations.Param;
+import org.apache.ibatis.session.ResultHandler;
import org.sonar.api.rule.RuleStatus;
import org.sonar.db.KeyLongValue;
@@ -73,4 +74,9 @@ public interface ActiveRuleMapper {
List<KeyLongValue> countActiveRulesByQuery(@Param("organizationUuid") String organizationUuid, @Param("profileUuids") List<String> profileUuids,
@Nullable @Param("ruleStatus") RuleStatus ruleStatus, @Param("inheritance") String inheritance);
+ void scrollAllForIndexing(ResultHandler handler);
+
+ void scrollByIdsForIndexing(@Param("ids") Collection<Long> ids, ResultHandler handler);
+
+ void scrollByRuleProfileUuidForIndexing(@Param("ruleProfileUuid") String ruleProfileUuid, ResultHandler handler);
}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/rule/index/RuleDocWithSystemScope.java b/server/sonar-db-dao/src/main/java/org/sonar/db/qualityprofile/IndexedActiveRuleDto.java
index 1e581a9ff21..b34d33ffeb0 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/rule/index/RuleDocWithSystemScope.java
+++ b/server/sonar-db-dao/src/main/java/org/sonar/db/qualityprofile/IndexedActiveRuleDto.java
@@ -17,24 +17,40 @@
* along with this program; if not, write to the Free Software Foundation,
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
+package org.sonar.db.qualityprofile;
-package org.sonar.server.rule.index;
+import javax.annotation.CheckForNull;
-public class RuleDocWithSystemScope {
+public class IndexedActiveRuleDto {
+ private long id;
+ private int severity;
+ private String inheritance;
+ private String repository;
+ private String key;
+ private String ruleProfileUuid;
- private final RuleDoc ruleDoc;
- private final RuleExtensionDoc ruleExtensionDoc;
+ public long getId() {
+ return id;
+ }
+
+ public int getSeverity() {
+ return severity;
+ }
+
+ @CheckForNull
+ public String getInheritance() {
+ return inheritance;
+ }
- public RuleDocWithSystemScope(RuleDoc ruleDoc, RuleExtensionDoc ruleExtensionDoc) {
- this.ruleDoc = ruleDoc;
- this.ruleExtensionDoc = ruleExtensionDoc;
+ public String getRepository() {
+ return repository;
}
- public RuleDoc getRuleDoc() {
- return ruleDoc;
+ public String getKey() {
+ return key;
}
- public RuleExtensionDoc getRuleExtensionDoc() {
- return ruleExtensionDoc;
+ public String getRuleProfileUuid() {
+ return ruleProfileUuid;
}
}
diff --git a/server/sonar-db-dao/src/main/java/org/sonar/db/qualityprofile/QualityProfileDao.java b/server/sonar-db-dao/src/main/java/org/sonar/db/qualityprofile/QualityProfileDao.java
index 600be7ac278..ee81f42b8d1 100644
--- a/server/sonar-db-dao/src/main/java/org/sonar/db/qualityprofile/QualityProfileDao.java
+++ b/server/sonar-db-dao/src/main/java/org/sonar/db/qualityprofile/QualityProfileDao.java
@@ -74,6 +74,11 @@ public class QualityProfileDao implements Dao {
return mapper(dbSession).selectBuiltInRuleProfiles();
}
+ @CheckForNull
+ public RulesProfileDto selectRuleProfile(DbSession dbSession, String ruleProfileUuid) {
+ return mapper(dbSession).selectRuleProfile(ruleProfileUuid);
+ }
+
public void insert(DbSession dbSession, RulesProfileDto dto) {
QualityProfileMapper mapper = mapper(dbSession);
mapper.insertRuleProfile(dto, new Date(system.now()));
diff --git a/server/sonar-db-dao/src/main/java/org/sonar/db/qualityprofile/QualityProfileMapper.java b/server/sonar-db-dao/src/main/java/org/sonar/db/qualityprofile/QualityProfileMapper.java
index 04def0160fd..4f9e9c9df56 100644
--- a/server/sonar-db-dao/src/main/java/org/sonar/db/qualityprofile/QualityProfileMapper.java
+++ b/server/sonar-db-dao/src/main/java/org/sonar/db/qualityprofile/QualityProfileMapper.java
@@ -42,6 +42,9 @@ public interface QualityProfileMapper {
List<RulesProfileDto> selectBuiltInRuleProfiles();
+ @CheckForNull
+ RulesProfileDto selectRuleProfile(@Param("uuid") String ruleProfileUuid);
+
List<QProfileDto> selectOrderedByOrganizationUuid(@Param("organizationUuid") String organizationUuid);
@CheckForNull
diff --git a/server/sonar-db-dao/src/main/java/org/sonar/db/rule/RuleDao.java b/server/sonar-db-dao/src/main/java/org/sonar/db/rule/RuleDao.java
index 94e47de330f..1af37e8cfd2 100644
--- a/server/sonar-db-dao/src/main/java/org/sonar/db/rule/RuleDao.java
+++ b/server/sonar-db-dao/src/main/java/org/sonar/db/rule/RuleDao.java
@@ -25,12 +25,12 @@ import java.util.Optional;
import java.util.function.Consumer;
import javax.annotation.Nullable;
import org.apache.ibatis.session.ResultHandler;
-import org.sonar.db.es.RuleExtensionId;
import org.sonar.api.rule.RuleKey;
import org.sonar.api.rules.RuleQuery;
import org.sonar.db.Dao;
import org.sonar.db.DbSession;
import org.sonar.db.RowNotFoundException;
+import org.sonar.db.es.RuleExtensionId;
import org.sonar.db.organization.OrganizationDto;
import static com.google.common.base.Preconditions.checkNotNull;
@@ -152,24 +152,38 @@ public class RuleDao implements Dao {
}
}
- public void scrollRuleExtensionByRuleKeys(DbSession dbSession, Collection<RuleExtensionId> ruleExtensionIds, Consumer<RuleExtensionForIndexingDto> consumer) {
+ public void scrollIndexingRuleExtensionsByIds(DbSession dbSession, Collection<RuleExtensionId> ruleExtensionIds, Consumer<RuleExtensionForIndexingDto> consumer) {
RuleMapper mapper = mapper(dbSession);
executeLargeInputsWithoutOutput(ruleExtensionIds,
pageOfRuleExtensionIds -> mapper
- .selectRuleExtensionForIndexingByKeys(pageOfRuleExtensionIds)
+ .selectIndexingRuleExtensionsByIds(pageOfRuleExtensionIds)
.forEach(consumer));
}
- public void scrollRuleByRuleKeys(DbSession dbSession, Collection<RuleKey> ruleKeys, Consumer<RuleForIndexingDto> consumer) {
+ public void scrollIndexingRuleExtensions(DbSession dbSession, Consumer<RuleExtensionForIndexingDto> consumer) {
+ mapper(dbSession).scrollIndexingRuleExtensions(context -> {
+ RuleExtensionForIndexingDto dto = (RuleExtensionForIndexingDto) context.getResultObject();
+ consumer.accept(dto);
+ });
+ }
+
+ public void scrollIndexingRulesByKeys(DbSession dbSession, Collection<RuleKey> ruleKeys, Consumer<RuleForIndexingDto> consumer) {
RuleMapper mapper = mapper(dbSession);
executeLargeInputsWithoutOutput(ruleKeys,
pageOfRuleKeys -> mapper
- .selectRuleForIndexingByKeys(pageOfRuleKeys)
+ .selectIndexingRulesByKeys(pageOfRuleKeys)
.forEach(consumer));
}
+ public void scrollIndexingRules(DbSession dbSession, Consumer<RuleForIndexingDto> consumer) {
+ mapper(dbSession).scrollIndexingRules(context -> {
+ RuleForIndexingDto dto = (RuleForIndexingDto) context.getResultObject();
+ consumer.accept(dto);
+ });
+ }
+
private static RuleMapper mapper(DbSession session) {
return session.getMapper(RuleMapper.class);
}
diff --git a/server/sonar-db-dao/src/main/java/org/sonar/db/rule/RuleForIndexingDto.java b/server/sonar-db-dao/src/main/java/org/sonar/db/rule/RuleForIndexingDto.java
index 52459e6338e..d255d5c2c38 100644
--- a/server/sonar-db-dao/src/main/java/org/sonar/db/rule/RuleForIndexingDto.java
+++ b/server/sonar-db-dao/src/main/java/org/sonar/db/rule/RuleForIndexingDto.java
@@ -40,7 +40,7 @@ public class RuleForIndexingDto {
private boolean isTemplate;
private String systemTags;
private String templateRuleKey;
- private String templateName;
+ private String templateRepository;
private String internalKey;
private String language;
private int type;
@@ -93,8 +93,8 @@ public class RuleForIndexingDto {
return templateRuleKey;
}
- public String getTemplateName() {
- return templateName;
+ public String getTemplateRepository() {
+ return templateRepository;
}
public String getInternalKey() {
diff --git a/server/sonar-db-dao/src/main/java/org/sonar/db/rule/RuleMapper.java b/server/sonar-db-dao/src/main/java/org/sonar/db/rule/RuleMapper.java
index d5a98e60200..350bd23b3e3 100644
--- a/server/sonar-db-dao/src/main/java/org/sonar/db/rule/RuleMapper.java
+++ b/server/sonar-db-dao/src/main/java/org/sonar/db/rule/RuleMapper.java
@@ -52,9 +52,13 @@ public interface RuleMapper {
List<RuleDefinitionDto> selectDefinitionByKeys(@Param("ruleKeys") List<RuleKey> keys);
- List<RuleForIndexingDto> selectRuleForIndexingByKeys(@Param("ruleKeys") List<RuleKey> keys);
+ void scrollIndexingRules(ResultHandler handler);
- List<RuleExtensionForIndexingDto> selectRuleExtensionForIndexingByKeys(@Param("ruleExtensionIds") List<RuleExtensionId> ruleExtensionIds);
+ List<RuleForIndexingDto> selectIndexingRulesByKeys(@Param("ruleKeys") List<RuleKey> keys);
+
+ void scrollIndexingRuleExtensions(ResultHandler handler);
+
+ List<RuleExtensionForIndexingDto> selectIndexingRuleExtensionsByIds(@Param("ruleExtensionIds") List<RuleExtensionId> ruleExtensionIds);
List<RuleDto> selectByQuery(@Param("organizationUuid") String organizationUuid, @Param("query") RuleQuery ruleQuery);
diff --git a/server/sonar-db-dao/src/main/java/org/sonar/db/rule/RuleMetadataDto.java b/server/sonar-db-dao/src/main/java/org/sonar/db/rule/RuleMetadataDto.java
index 594611079fc..17f27c40d8f 100644
--- a/server/sonar-db-dao/src/main/java/org/sonar/db/rule/RuleMetadataDto.java
+++ b/server/sonar-db-dao/src/main/java/org/sonar/db/rule/RuleMetadataDto.java
@@ -135,6 +135,10 @@ public class RuleMetadataDto {
return tags == null ? new HashSet<>() : new TreeSet<>(Arrays.asList(StringUtils.split(tags, ',')));
}
+ String getTagsAsString() {
+ return tags;
+ }
+
public RuleMetadataDto setTags(Set<String> tags) {
String raw = tags.isEmpty() ? null : StringUtils.join(tags, ',');
checkArgument(raw == null || raw.length() <= 4000, "Rule tags are too long: %s", raw);
diff --git a/server/sonar-db-dao/src/main/resources/org/sonar/db/es/EsQueueMapper.xml b/server/sonar-db-dao/src/main/resources/org/sonar/db/es/EsQueueMapper.xml
index 62ec6e3421b..e1be847595c 100644
--- a/server/sonar-db-dao/src/main/resources/org/sonar/db/es/EsQueueMapper.xml
+++ b/server/sonar-db-dao/src/main/resources/org/sonar/db/es/EsQueueMapper.xml
@@ -7,6 +7,8 @@
uuid,
doc_type as docType,
doc_id as docId,
+ doc_id_type as docIdType,
+ doc_routing as docRouting,
created_at as createdAt
</sql>
@@ -15,11 +17,15 @@
uuid,
doc_type,
doc_id,
+ doc_id_type,
+ doc_routing,
created_at
) values (
#{dto.uuid, jdbcType=VARCHAR},
#{dto.docType, jdbcType=VARCHAR},
#{dto.docId, jdbcType=VARCHAR},
+ #{dto.docIdType, jdbcType=VARCHAR},
+ #{dto.docRouting, jdbcType=VARCHAR},
#{now, jdbcType=BIGINT}
)
</insert>
diff --git a/server/sonar-db-dao/src/main/resources/org/sonar/db/qualityprofile/ActiveRuleMapper.xml b/server/sonar-db-dao/src/main/resources/org/sonar/db/qualityprofile/ActiveRuleMapper.xml
index 29ac1dc7481..5965b14ebde 100644
--- a/server/sonar-db-dao/src/main/resources/org/sonar/db/qualityprofile/ActiveRuleMapper.xml
+++ b/server/sonar-db-dao/src/main/resources/org/sonar/db/qualityprofile/ActiveRuleMapper.xml
@@ -272,5 +272,32 @@
group by oqp.uuid
</select>
+ <select id="scrollAllForIndexing" resultType="org.sonar.db.qualityprofile.IndexedActiveRuleDto" fetchSize="${_scrollFetchSize}" resultSetType="FORWARD_ONLY">
+ <include refid="scrollAllForIndexingSql"/>
+ </select>
+
+ <select id="scrollByIdsForIndexing" parameterType="map" resultType="org.sonar.db.qualityprofile.IndexedActiveRuleDto" fetchSize="${_scrollFetchSize}" resultSetType="FORWARD_ONLY">
+ <include refid="scrollAllForIndexingSql"/>
+ where ar.id in
+ <foreach collection="ids" open="(" close=")" item="id" separator=",">#{id, jdbcType=BIGINT}</foreach>
+ </select>
+
+ <select id="scrollByRuleProfileUuidForIndexing" parameterType="String" resultType="org.sonar.db.qualityprofile.IndexedActiveRuleDto" fetchSize="${_scrollFetchSize}" resultSetType="FORWARD_ONLY">
+ <include refid="scrollAllForIndexingSql"/>
+ where rp.kee = #{ruleProfileUuid, jdbcType=VARCHAR}
+ </select>
+
+ <sql id="scrollAllForIndexingSql">
+ select
+ ar.id as "id",
+ ar.failure_level as "severity",
+ ar.inheritance as "inheritance",
+ r.plugin_name as "repository",
+ r.plugin_rule_key as "key",
+ rp.kee as "ruleProfileUuid"
+ from active_rules ar
+ inner join rules_profiles rp on rp.id = ar.profile_id
+ inner join rules r on r.id = ar.rule_id
+ </sql>
</mapper>
diff --git a/server/sonar-db-dao/src/main/resources/org/sonar/db/qualityprofile/QualityProfileMapper.xml b/server/sonar-db-dao/src/main/resources/org/sonar/db/qualityprofile/QualityProfileMapper.xml
index 3f85132f1d4..4d75daa1067 100644
--- a/server/sonar-db-dao/src/main/resources/org/sonar/db/qualityprofile/QualityProfileMapper.xml
+++ b/server/sonar-db-dao/src/main/resources/org/sonar/db/qualityprofile/QualityProfileMapper.xml
@@ -109,6 +109,12 @@
where rp.is_built_in = ${_true}
</select>
+ <select id="selectRuleProfile" resultType="org.sonar.db.qualityprofile.RulesProfileDto">
+ select <include refid="ruleProfileColumns"/>
+ from rules_profiles rp
+ where rp.kee = #{uuid, jdbcType=VARCHAR}
+ </select>
+
<select id="selectOrderedByOrganizationUuid" parameterType="map" resultType="org.sonar.db.qualityprofile.QProfileDto">
select
<include refid="qProfileColumns"/>
diff --git a/server/sonar-db-dao/src/main/resources/org/sonar/db/rule/RuleMapper.xml b/server/sonar-db-dao/src/main/resources/org/sonar/db/rule/RuleMapper.xml
index 0e29a3e0d94..222fea7a639 100644
--- a/server/sonar-db-dao/src/main/resources/org/sonar/db/rule/RuleMapper.xml
+++ b/server/sonar-db-dao/src/main/resources/org/sonar/db/rule/RuleMapper.xml
@@ -139,26 +139,45 @@
and r.plugin_rule_key=#{rule,jdbcType=VARCHAR}
</select>
- <select id="selectRuleExtensionForIndexingByKeys" parameterType="map" resultType="org.sonar.db.rule.RuleExtensionForIndexingDto">
- select
- r.plugin_name as "pluginName",
- r.plugin_rule_key as "pluginRuleKey",
- rm.organization_uuid as "organizationUuid",
- rm.tags as "tags"
- from
- rules r
- inner join
- rules_metadata rm on rm.rule_id = r.id
- where
- rm.tags is not null and
- rm.tags != '' and
+ <select id="selectIndexingRuleExtensionsByIds" parameterType="map" resultType="org.sonar.db.rule.RuleExtensionForIndexingDto">
+ <include refid="sqlSelectIndexingRuleExtensions" />
+ and
<foreach collection="ruleExtensionIds" index="index" item="ruleExtId" open="" separator=" or " close="">
- ( r.plugin_name=#{ruleExtId.repositoryName,jdbcType=VARCHAR} and
- r.plugin_rule_key=#{ruleExtId.ruleKey,jdbcType=VARCHAR} and
- rm.organization_uuid=#{ruleExtId.organizationUuid,jdbcType=VARCHAR} )
+ ( r.plugin_name = #{ruleExtId.repositoryName, jdbcType=VARCHAR} and
+ r.plugin_rule_key = #{ruleExtId.ruleKey, jdbcType=VARCHAR} and
+ rm.organization_uuid = #{ruleExtId.organizationUuid, jdbcType=VARCHAR} )
</foreach>
</select>
+ <select id="scrollIndexingRuleExtensions" resultType="org.sonar.db.rule.RuleExtensionForIndexingDto" fetchSize="${_scrollFetchSize}" resultSetType="FORWARD_ONLY">
+ <include refid="sqlSelectIndexingRuleExtensions" />
+ </select>
+
+ <sql id="sqlSelectIndexingRuleExtensions">
+ select
+ r.plugin_name as "pluginName",
+ r.plugin_rule_key as "pluginRuleKey",
+ rm.organization_uuid as "organizationUuid",
+ rm.tags as "tags"
+ from rules r
+ inner join rules_metadata rm on rm.rule_id = r.id
+ where
+ rm.tags is not null and
+ rm.tags != ''
+ </sql>
+
+ <sql id="sqlSelectIndexingRuleExtensions" databaseId="oracle">
+ select
+ r.plugin_name as "pluginName",
+ r.plugin_rule_key as "pluginRuleKey",
+ rm.organization_uuid as "organizationUuid",
+ rm.tags as "tags"
+ from rules r
+ inner join rules_metadata rm on rm.rule_id = r.id
+ where
+ rm.tags is not null
+ </sql>
+
<select id="selectMetadataByKey" parameterType="map" resultType="org.sonar.db.rule.RuleMetadataDto">
select
rm.rule_id as "ruleId",
@@ -205,7 +224,19 @@
</foreach>
</select>
- <select id="selectRuleForIndexingByKeys" parameterType="map" resultType="org.sonar.db.rule.RuleForIndexingDto">
+ <select id="selectIndexingRulesByKeys" parameterType="map" resultType="org.sonar.db.rule.RuleForIndexingDto">
+ <include refid="sqlSelectIndexingRules"/>
+ where
+ <foreach collection="ruleKeys" index="index" item="ruleKey" open="" separator=" or " close="">
+ (r.plugin_name=#{ruleKey.repository,jdbcType=VARCHAR} and r.plugin_rule_key=#{ruleKey.rule,jdbcType=VARCHAR})
+ </foreach>
+ </select>
+
+ <select id="scrollIndexingRules" resultType="org.sonar.db.rule.RuleForIndexingDto" fetchSize="${_scrollFetchSize}" resultSetType="FORWARD_ONLY">
+ <include refid="sqlSelectIndexingRules"/>
+ </select>
+
+ <sql id="sqlSelectIndexingRules">
select
r.id as "id",
r.plugin_name as "repository",
@@ -218,20 +249,15 @@
r.is_template as "isTemplate",
r.system_tags as "systemTags",
t.plugin_rule_key as "templateRuleKey",
- t.plugin_name as "templateName",
+ t.plugin_name as "templateRepository",
r.plugin_config_key as "internalKey",
r.language as "language",
r.rule_type as "type",
r.created_at as "createdAt",
r.updated_at as "updatedAt"
- from
- rules r
+ from rules r
left outer join rules t on t.id = r.template_id
- where
- <foreach collection="ruleKeys" index="index" item="ruleKey" open="" separator=" or " close="">
- (r.plugin_name=#{ruleKey.repository,jdbcType=VARCHAR} and r.plugin_rule_key=#{ruleKey.rule,jdbcType=VARCHAR})
- </foreach>
- </select>
+ </sql>
<select id="selectByQuery" parameterType="map" resultType="Rule">
select
diff --git a/server/sonar-db-dao/src/test/java/org/sonar/db/qualityprofile/ActiveRuleDaoTest.java b/server/sonar-db-dao/src/test/java/org/sonar/db/qualityprofile/ActiveRuleDaoTest.java
index 8eb040b9d0b..0add6763deb 100644
--- a/server/sonar-db-dao/src/test/java/org/sonar/db/qualityprofile/ActiveRuleDaoTest.java
+++ b/server/sonar-db-dao/src/test/java/org/sonar/db/qualityprofile/ActiveRuleDaoTest.java
@@ -19,8 +19,10 @@
*/
package org.sonar.db.qualityprofile;
+import java.util.ArrayList;
import java.util.List;
import java.util.Objects;
+import java.util.function.Consumer;
import org.junit.Before;
import org.junit.Rule;
import org.junit.Test;
@@ -578,4 +580,68 @@ public class ActiveRuleDaoTest {
.containsOnly(entry(profile1.getKee(), 1L));
}
+ @Test
+ public void scrollAllForIndexing_empty_table() {
+ Accumulator accumulator = new Accumulator();
+ underTest.scrollAllForIndexing(dbSession, accumulator);
+ assertThat(accumulator.list).isEmpty();
+ }
+
+ @Test
+ public void scrollAllForIndexing() {
+ ActiveRuleDto ar1 = db.qualityProfiles().activateRule(profile1, rule1);
+ ActiveRuleDto ar2 = db.qualityProfiles().activateRule(profile2, rule1);
+ ActiveRuleDto ar3 = db.qualityProfiles().activateRule(profile2, rule2);
+
+ Accumulator accumulator = new Accumulator();
+ underTest.scrollAllForIndexing(dbSession, accumulator);
+ assertThat(accumulator.list)
+ .extracting(IndexedActiveRuleDto::getId, IndexedActiveRuleDto::getRepository, IndexedActiveRuleDto::getKey, IndexedActiveRuleDto::getRuleProfileUuid,
+ IndexedActiveRuleDto::getSeverity, IndexedActiveRuleDto::getInheritance)
+ .containsExactlyInAnyOrder(
+ tuple((long)ar1.getId(), ar1.getRuleKey().repository(), ar1.getRuleKey().rule(), profile1.getRulesProfileUuid(), ar1.getSeverity(), ar1.getInheritance()),
+ tuple((long)ar2.getId(), ar2.getRuleKey().repository(), ar2.getRuleKey().rule(), profile2.getRulesProfileUuid(), ar2.getSeverity(), ar2.getInheritance()),
+ tuple((long)ar3.getId(), ar3.getRuleKey().repository(), ar3.getRuleKey().rule(), profile2.getRulesProfileUuid(), ar3.getSeverity(), ar3.getInheritance()));
+ }
+
+ @Test
+ public void scrollByIdsForIndexing() {
+ ActiveRuleDto ar1 = db.qualityProfiles().activateRule(profile1, rule1);
+ ActiveRuleDto ar2 = db.qualityProfiles().activateRule(profile2, rule1);
+ ActiveRuleDto ar3 = db.qualityProfiles().activateRule(profile2, rule2);
+
+ Accumulator accumulator = new Accumulator();
+ underTest.scrollByIdsForIndexing(dbSession, asList((long)ar1.getId(), (long)ar2.getId()), accumulator);
+ assertThat(accumulator.list)
+ .extracting(IndexedActiveRuleDto::getId, IndexedActiveRuleDto::getRepository, IndexedActiveRuleDto::getKey, IndexedActiveRuleDto::getRuleProfileUuid,
+ IndexedActiveRuleDto::getSeverity)
+ .containsExactlyInAnyOrder(
+ tuple((long)ar1.getId(), ar1.getRuleKey().repository(), ar1.getRuleKey().rule(), profile1.getRulesProfileUuid(), ar1.getSeverity()),
+ tuple((long)ar2.getId(), ar2.getRuleKey().repository(), ar2.getRuleKey().rule(), profile2.getRulesProfileUuid(), ar2.getSeverity()));
+ }
+
+ @Test
+ public void scrollByRuleProfileForIndexing() {
+ ActiveRuleDto ar1 = db.qualityProfiles().activateRule(profile1, rule1);
+ ActiveRuleDto ar2 = db.qualityProfiles().activateRule(profile2, rule1);
+ ActiveRuleDto ar3 = db.qualityProfiles().activateRule(profile2, rule2);
+
+ Accumulator accumulator = new Accumulator();
+ underTest.scrollByRuleProfileForIndexing(dbSession, profile2.getRulesProfileUuid(), accumulator);
+ assertThat(accumulator.list)
+ .extracting(IndexedActiveRuleDto::getId, IndexedActiveRuleDto::getRepository, IndexedActiveRuleDto::getKey, IndexedActiveRuleDto::getRuleProfileUuid,
+ IndexedActiveRuleDto::getSeverity)
+ .containsExactlyInAnyOrder(
+ tuple((long)ar2.getId(), ar2.getRuleKey().repository(), ar2.getRuleKey().rule(), profile2.getRulesProfileUuid(), ar2.getSeverity()),
+ tuple((long)ar3.getId(), ar3.getRuleKey().repository(), ar3.getRuleKey().rule(), profile2.getRulesProfileUuid(), ar3.getSeverity()));
+ }
+
+ private static class Accumulator implements Consumer<IndexedActiveRuleDto> {
+ private final List<IndexedActiveRuleDto> list = new ArrayList<>();
+
+ @Override
+ public void accept(IndexedActiveRuleDto dto) {
+ list.add(dto);
+ }
+ }
}
diff --git a/server/sonar-db-dao/src/test/java/org/sonar/db/qualityprofile/QualityProfileDaoTest.java b/server/sonar-db-dao/src/test/java/org/sonar/db/qualityprofile/QualityProfileDaoTest.java
index d254c2b412c..3d8255b6005 100644
--- a/server/sonar-db-dao/src/test/java/org/sonar/db/qualityprofile/QualityProfileDaoTest.java
+++ b/server/sonar-db-dao/src/test/java/org/sonar/db/qualityprofile/QualityProfileDaoTest.java
@@ -148,6 +148,14 @@ public class QualityProfileDaoTest {
}
@Test
+ public void selectRuleProfile() {
+ RulesProfileDto rp = insertRulesProfile();
+
+ assertThat(underTest.selectRuleProfile(dbSession, rp.getKee()).getId()).isEqualTo(rp.getId());
+ assertThat(underTest.selectRuleProfile(dbSession, "missing")).isNull();
+ }
+
+ @Test
public void deleteRulesProfilesByUuids() {
RulesProfileDto rp1 = insertRulesProfile();
RulesProfileDto rp2 = insertRulesProfile();
@@ -245,7 +253,7 @@ public class QualityProfileDaoTest {
}
@Test
- public void find_all_is_sorted_by_profile_name() {
+ public void selectOrderedByOrganizationUuid_is_sorted_by_profile_name() {
QProfileDto dto1 = new QProfileDto()
.setKee("js_first")
.setRulesProfileUuid("rp-js_first")
@@ -315,7 +323,7 @@ public class QualityProfileDaoTest {
}
@Test
- public void get_by_name_and_language() {
+ public void selectByNameAndLanguage() {
List<QProfileDto> sharedData = createSharedData();
QProfileDto dto = underTest.selectByNameAndLanguage(dbSession, organization, "Sonar Way", "java");
@@ -328,7 +336,7 @@ public class QualityProfileDaoTest {
}
@Test
- public void get_by_name_and_languages() {
+ public void selectByNameAndLanguages() {
createSharedData();
List<QProfileDto> dtos = underTest.selectByNameAndLanguages(dbSession, organization, "Sonar Way", singletonList("java"));
@@ -362,7 +370,7 @@ public class QualityProfileDaoTest {
}
@Test
- public void should_not_find_by_language_in_wrong_organization() {
+ public void should_not_selectByLanguage_in_wrong_organization() {
QProfileDto profile = QualityProfileTesting.newQualityProfileDto()
.setOrganizationUuid(organization.getUuid());
underTest.insert(dbSession, profile);
@@ -372,7 +380,7 @@ public class QualityProfileDaoTest {
}
@Test
- public void should_not_find_by_language_with_wrong_language() {
+ public void should_not_selectByLanguage_with_wrong_language() {
QProfileDto profile = QualityProfileTesting.newQualityProfileDto()
.setOrganizationUuid(organization.getUuid());
underTest.insert(dbSession, profile);
@@ -382,7 +390,7 @@ public class QualityProfileDaoTest {
}
@Test
- public void find_children() {
+ public void selectChildren() {
QProfileDto original1 = new QProfileDto()
.setKee("java_child1")
.setRulesProfileUuid("rp-java_child1")
diff --git a/server/sonar-db-dao/src/test/java/org/sonar/db/rule/RuleDaoTest.java b/server/sonar-db-dao/src/test/java/org/sonar/db/rule/RuleDaoTest.java
index a79d4d7c141..ea1f43510e2 100644
--- a/server/sonar-db-dao/src/test/java/org/sonar/db/rule/RuleDaoTest.java
+++ b/server/sonar-db-dao/src/test/java/org/sonar/db/rule/RuleDaoTest.java
@@ -25,6 +25,7 @@ import java.util.Arrays;
import java.util.Collections;
import java.util.List;
import java.util.Optional;
+import java.util.function.Consumer;
import org.apache.ibatis.session.ResultHandler;
import org.junit.Before;
import org.junit.Rule;
@@ -40,6 +41,7 @@ import org.sonar.api.utils.DateUtils;
import org.sonar.api.utils.System2;
import org.sonar.db.DbTester;
import org.sonar.db.RowNotFoundException;
+import org.sonar.db.es.RuleExtensionId;
import org.sonar.db.organization.OrganizationDto;
import org.sonar.db.organization.OrganizationTesting;
@@ -47,6 +49,7 @@ import static com.google.common.collect.Sets.newHashSet;
import static java.util.Arrays.asList;
import static java.util.Collections.singletonList;
import static org.assertj.core.api.Assertions.assertThat;
+import static org.assertj.core.api.Assertions.tuple;
public class RuleDaoTest {
@@ -55,219 +58,219 @@ public class RuleDaoTest {
@Rule
public ExpectedException thrown = ExpectedException.none();
@Rule
- public DbTester dbTester = DbTester.create(System2.INSTANCE);
+ public DbTester db = DbTester.create(System2.INSTANCE);
- private RuleDao underTest = dbTester.getDbClient().ruleDao();
+ private RuleDao underTest = db.getDbClient().ruleDao();
private OrganizationDto organization;
@Before
public void before() {
- organization = dbTester.organizations().insert(o -> o.setUuid(ORGANIZATION_UUID));
+ organization = db.organizations().insert(o -> o.setUuid(ORGANIZATION_UUID));
}
@Test
public void selectByKey() {
- dbTester.prepareDbUnit(getClass(), "shared.xml");
+ db.prepareDbUnit(getClass(), "shared.xml");
- assertThat(underTest.selectByKey(dbTester.getSession(), organization, RuleKey.of("NOT", "FOUND")).isPresent()).isFalse();
+ assertThat(underTest.selectByKey(db.getSession(), organization, RuleKey.of("NOT", "FOUND")).isPresent()).isFalse();
- Optional<RuleDto> rule = underTest.selectByKey(dbTester.getSession(), organization, RuleKey.of("java", "S001"));
+ Optional<RuleDto> rule = underTest.selectByKey(db.getSession(), organization, RuleKey.of("java", "S001"));
assertThat(rule.isPresent()).isTrue();
assertThat(rule.get().getId()).isEqualTo(1);
}
@Test
public void selectByKey_populates_organizationUuid_even_when_organization_has_no_metadata() {
- dbTester.prepareDbUnit(getClass(), "shared.xml");
+ db.prepareDbUnit(getClass(), "shared.xml");
- assertThat(underTest.selectByKey(dbTester.getSession(), organization, RuleKey.of("java", "S001")).get().getOrganizationUuid())
+ assertThat(underTest.selectByKey(db.getSession(), organization, RuleKey.of("java", "S001")).get().getOrganizationUuid())
.isEqualTo(ORGANIZATION_UUID);
}
@Test
public void selectDefinitionByKey() {
- dbTester.prepareDbUnit(getClass(), "shared.xml");
+ db.prepareDbUnit(getClass(), "shared.xml");
- assertThat(underTest.selectDefinitionByKey(dbTester.getSession(), RuleKey.of("NOT", "FOUND")).isPresent()).isFalse();
+ assertThat(underTest.selectDefinitionByKey(db.getSession(), RuleKey.of("NOT", "FOUND")).isPresent()).isFalse();
- Optional<RuleDefinitionDto> rule = underTest.selectDefinitionByKey(dbTester.getSession(), RuleKey.of("java", "S001"));
+ Optional<RuleDefinitionDto> rule = underTest.selectDefinitionByKey(db.getSession(), RuleKey.of("java", "S001"));
assertThat(rule.isPresent()).isTrue();
assertThat(rule.get().getId()).isEqualTo(1);
}
@Test
public void selectById() {
- dbTester.prepareDbUnit(getClass(), "shared.xml");
+ db.prepareDbUnit(getClass(), "shared.xml");
String organizationUuid = "org-1";
- assertThat(underTest.selectById(55l, organizationUuid, dbTester.getSession())).isEmpty();
- Optional<RuleDto> ruleDtoOptional = underTest.selectById(1l, organizationUuid, dbTester.getSession());
+ assertThat(underTest.selectById(55l, organizationUuid, db.getSession())).isEmpty();
+ Optional<RuleDto> ruleDtoOptional = underTest.selectById(1l, organizationUuid, db.getSession());
assertThat(ruleDtoOptional).isPresent();
assertThat(ruleDtoOptional.get().getId()).isEqualTo(1);
}
@Test
public void selectById_populates_organizationUuid_even_when_organization_has_no_metadata() {
- dbTester.prepareDbUnit(getClass(), "shared.xml");
+ db.prepareDbUnit(getClass(), "shared.xml");
String organizationUuid = "org-1";
- assertThat(underTest.selectById(1l, organizationUuid, dbTester.getSession()).get().getOrganizationUuid())
+ assertThat(underTest.selectById(1l, organizationUuid, db.getSession()).get().getOrganizationUuid())
.isEqualTo(organizationUuid);
}
@Test
public void selectDefinitionById() {
- dbTester.prepareDbUnit(getClass(), "shared.xml");
+ db.prepareDbUnit(getClass(), "shared.xml");
- assertThat(underTest.selectDefinitionById(55l, dbTester.getSession())).isEmpty();
- Optional<RuleDefinitionDto> ruleDtoOptional = underTest.selectDefinitionById(1l, dbTester.getSession());
+ assertThat(underTest.selectDefinitionById(55l, db.getSession())).isEmpty();
+ Optional<RuleDefinitionDto> ruleDtoOptional = underTest.selectDefinitionById(1l, db.getSession());
assertThat(ruleDtoOptional).isPresent();
assertThat(ruleDtoOptional.get().getId()).isEqualTo(1);
}
@Test
public void selectByIds() {
- dbTester.prepareDbUnit(getClass(), "shared.xml");
+ db.prepareDbUnit(getClass(), "shared.xml");
String organizationUuid = "org-1";
- assertThat(underTest.selectByIds(dbTester.getSession(), organizationUuid, asList(1))).hasSize(1);
- assertThat(underTest.selectByIds(dbTester.getSession(), organizationUuid, asList(1, 2))).hasSize(2);
- assertThat(underTest.selectByIds(dbTester.getSession(), organizationUuid, asList(1, 2, 3))).hasSize(2);
+ assertThat(underTest.selectByIds(db.getSession(), organizationUuid, asList(1))).hasSize(1);
+ assertThat(underTest.selectByIds(db.getSession(), organizationUuid, asList(1, 2))).hasSize(2);
+ assertThat(underTest.selectByIds(db.getSession(), organizationUuid, asList(1, 2, 3))).hasSize(2);
- assertThat(underTest.selectByIds(dbTester.getSession(), organizationUuid, asList(123))).isEmpty();
+ assertThat(underTest.selectByIds(db.getSession(), organizationUuid, asList(123))).isEmpty();
}
@Test
public void selectByIds_populates_organizationUuid_even_when_organization_has_no_metadata() {
- dbTester.prepareDbUnit(getClass(), "shared.xml");
+ db.prepareDbUnit(getClass(), "shared.xml");
String organizationUuid = "org-1";
- assertThat(underTest.selectByIds(dbTester.getSession(), organizationUuid, asList(1, 2)))
+ assertThat(underTest.selectByIds(db.getSession(), organizationUuid, asList(1, 2)))
.extracting(RuleDto::getOrganizationUuid)
.containsExactly(organizationUuid, organizationUuid);
}
@Test
public void selectDefinitionByIds() {
- dbTester.prepareDbUnit(getClass(), "shared.xml");
+ db.prepareDbUnit(getClass(), "shared.xml");
- assertThat(underTest.selectDefinitionByIds(dbTester.getSession(), asList(1))).hasSize(1);
- assertThat(underTest.selectDefinitionByIds(dbTester.getSession(), asList(1, 2))).hasSize(2);
- assertThat(underTest.selectDefinitionByIds(dbTester.getSession(), asList(1, 2, 3))).hasSize(2);
+ assertThat(underTest.selectDefinitionByIds(db.getSession(), asList(1))).hasSize(1);
+ assertThat(underTest.selectDefinitionByIds(db.getSession(), asList(1, 2))).hasSize(2);
+ assertThat(underTest.selectDefinitionByIds(db.getSession(), asList(1, 2, 3))).hasSize(2);
- assertThat(underTest.selectDefinitionByIds(dbTester.getSession(), asList(123))).isEmpty();
+ assertThat(underTest.selectDefinitionByIds(db.getSession(), asList(123))).isEmpty();
}
@Test
public void selectOrFailByKey() {
- dbTester.prepareDbUnit(getClass(), "shared.xml");
+ db.prepareDbUnit(getClass(), "shared.xml");
OrganizationDto organization = OrganizationTesting.newOrganizationDto().setUuid("org-1");
- RuleDto rule = underTest.selectOrFailByKey(dbTester.getSession(), organization, RuleKey.of("java", "S001"));
+ RuleDto rule = underTest.selectOrFailByKey(db.getSession(), organization, RuleKey.of("java", "S001"));
assertThat(rule.getId()).isEqualTo(1);
}
@Test
public void selectOrFailByKey_fails_if_rule_not_found() {
- dbTester.prepareDbUnit(getClass(), "shared.xml");
+ db.prepareDbUnit(getClass(), "shared.xml");
thrown.expect(RowNotFoundException.class);
thrown.expectMessage("Rule with key 'NOT:FOUND' does not exist");
OrganizationDto organization = OrganizationTesting.newOrganizationDto().setUuid("org-1");
- underTest.selectOrFailByKey(dbTester.getSession(), organization, RuleKey.of("NOT", "FOUND"));
+ underTest.selectOrFailByKey(db.getSession(), organization, RuleKey.of("NOT", "FOUND"));
}
@Test
public void selectOrFailByKey_populates_organizationUuid_even_when_organization_has_no_metadata() {
- dbTester.prepareDbUnit(getClass(), "shared.xml");
+ db.prepareDbUnit(getClass(), "shared.xml");
String organizationUuid = "org-1";
OrganizationDto organization = OrganizationTesting.newOrganizationDto().setUuid(organizationUuid);
- assertThat(underTest.selectOrFailByKey(dbTester.getSession(), organization, RuleKey.of("java", "S001")).getOrganizationUuid())
+ assertThat(underTest.selectOrFailByKey(db.getSession(), organization, RuleKey.of("java", "S001")).getOrganizationUuid())
.isEqualTo(organizationUuid);
}
@Test
public void selectOrFailDefinitionByKey_fails_if_rule_not_found() {
- dbTester.prepareDbUnit(getClass(), "shared.xml");
+ db.prepareDbUnit(getClass(), "shared.xml");
thrown.expect(RowNotFoundException.class);
thrown.expectMessage("Rule with key 'NOT:FOUND' does not exist");
- underTest.selectOrFailDefinitionByKey(dbTester.getSession(), RuleKey.of("NOT", "FOUND"));
+ underTest.selectOrFailDefinitionByKey(db.getSession(), RuleKey.of("NOT", "FOUND"));
}
@Test
public void selectByKeys() {
- dbTester.prepareDbUnit(getClass(), "shared.xml");
+ db.prepareDbUnit(getClass(), "shared.xml");
String organizationUuid = "org-1";
- assertThat(underTest.selectByKeys(dbTester.getSession(), organizationUuid, Collections.emptyList())).isEmpty();
- assertThat(underTest.selectByKeys(dbTester.getSession(), organizationUuid, asList(RuleKey.of("NOT", "FOUND")))).isEmpty();
+ assertThat(underTest.selectByKeys(db.getSession(), organizationUuid, Collections.emptyList())).isEmpty();
+ assertThat(underTest.selectByKeys(db.getSession(), organizationUuid, asList(RuleKey.of("NOT", "FOUND")))).isEmpty();
- List<RuleDto> rules = underTest.selectByKeys(dbTester.getSession(), organizationUuid, asList(RuleKey.of("java", "S001"), RuleKey.of("java", "OTHER")));
+ List<RuleDto> rules = underTest.selectByKeys(db.getSession(), organizationUuid, asList(RuleKey.of("java", "S001"), RuleKey.of("java", "OTHER")));
assertThat(rules).hasSize(1);
assertThat(rules.get(0).getId()).isEqualTo(1);
}
@Test
public void selectByKeys_populates_organizationUuid_even_when_organization_has_no_metadata() {
- dbTester.prepareDbUnit(getClass(), "shared.xml");
+ db.prepareDbUnit(getClass(), "shared.xml");
String organizationUuid = "org-1";
- assertThat(underTest.selectByKeys(dbTester.getSession(), organizationUuid, asList(RuleKey.of("java", "S001"), RuleKey.of("java", "OTHER"))))
+ assertThat(underTest.selectByKeys(db.getSession(), organizationUuid, asList(RuleKey.of("java", "S001"), RuleKey.of("java", "OTHER"))))
.extracting(RuleDto::getOrganizationUuid)
.containsExactly(organizationUuid);
}
@Test
public void selectDefinitionByKeys() {
- dbTester.prepareDbUnit(getClass(), "shared.xml");
+ db.prepareDbUnit(getClass(), "shared.xml");
- assertThat(underTest.selectDefinitionByKeys(dbTester.getSession(), Collections.emptyList())).isEmpty();
- assertThat(underTest.selectDefinitionByKeys(dbTester.getSession(), asList(RuleKey.of("NOT", "FOUND")))).isEmpty();
+ assertThat(underTest.selectDefinitionByKeys(db.getSession(), Collections.emptyList())).isEmpty();
+ assertThat(underTest.selectDefinitionByKeys(db.getSession(), asList(RuleKey.of("NOT", "FOUND")))).isEmpty();
- List<RuleDefinitionDto> rules = underTest.selectDefinitionByKeys(dbTester.getSession(), asList(RuleKey.of("java", "S001"), RuleKey.of("java", "OTHER")));
+ List<RuleDefinitionDto> rules = underTest.selectDefinitionByKeys(db.getSession(), asList(RuleKey.of("java", "S001"), RuleKey.of("java", "OTHER")));
assertThat(rules).hasSize(1);
assertThat(rules.get(0).getId()).isEqualTo(1);
}
@Test
public void selectAll() {
- dbTester.prepareDbUnit(getClass(), "shared.xml");
+ db.prepareDbUnit(getClass(), "shared.xml");
- assertThat(underTest.selectAll(dbTester.getSession(), "org-1"))
+ assertThat(underTest.selectAll(db.getSession(), "org-1"))
.extracting(RuleDto::getId)
.containsOnly(1, 2, 10);
}
@Test
public void selectAll_populates_organizationUuid_even_when_organization_has_no_metadata() {
- dbTester.prepareDbUnit(getClass(), "shared.xml");
+ db.prepareDbUnit(getClass(), "shared.xml");
String organizationUuid = "org-1";
- assertThat(underTest.selectAll(dbTester.getSession(), organizationUuid))
+ assertThat(underTest.selectAll(db.getSession(), organizationUuid))
.extracting(RuleDto::getOrganizationUuid)
.containsExactly(organizationUuid, organizationUuid, organizationUuid);
}
@Test
public void selectAllDefinitions() {
- dbTester.prepareDbUnit(getClass(), "shared.xml");
+ db.prepareDbUnit(getClass(), "shared.xml");
- List<RuleDefinitionDto> ruleDtos = underTest.selectAllDefinitions(dbTester.getSession());
+ List<RuleDefinitionDto> ruleDtos = underTest.selectAllDefinitions(db.getSession());
assertThat(ruleDtos).extracting("id").containsOnly(1, 2, 10);
}
@Test
public void selectEnabled_with_ResultHandler() {
- dbTester.prepareDbUnit(getClass(), "selectEnabled.xml");
+ db.prepareDbUnit(getClass(), "selectEnabled.xml");
final List<RuleDefinitionDto> rules = new ArrayList<>();
ResultHandler resultHandler = resultContext -> rules.add((RuleDefinitionDto) resultContext.getResultObject());
- underTest.selectEnabled(dbTester.getSession(), resultHandler);
+ underTest.selectEnabled(db.getSession(), resultHandler);
assertThat(rules.size()).isEqualTo(1);
RuleDefinitionDto ruleDto = rules.get(0);
@@ -285,23 +288,23 @@ public class RuleDaoTest {
@Test
public void select_by_query() {
- dbTester.prepareDbUnit(getClass(), "shared.xml");
+ db.prepareDbUnit(getClass(), "shared.xml");
String organizationUuid = "org-1";
- assertThat(underTest.selectByQuery(dbTester.getSession(), organizationUuid, RuleQuery.create())).hasSize(2);
- assertThat(underTest.selectByQuery(dbTester.getSession(), organizationUuid, RuleQuery.create().withKey("S001"))).hasSize(1);
- assertThat(underTest.selectByQuery(dbTester.getSession(), organizationUuid, RuleQuery.create().withConfigKey("S1"))).hasSize(1);
- assertThat(underTest.selectByQuery(dbTester.getSession(), organizationUuid, RuleQuery.create().withRepositoryKey("java"))).hasSize(2);
- assertThat(underTest.selectByQuery(dbTester.getSession(), organizationUuid,
+ assertThat(underTest.selectByQuery(db.getSession(), organizationUuid, RuleQuery.create())).hasSize(2);
+ assertThat(underTest.selectByQuery(db.getSession(), organizationUuid, RuleQuery.create().withKey("S001"))).hasSize(1);
+ assertThat(underTest.selectByQuery(db.getSession(), organizationUuid, RuleQuery.create().withConfigKey("S1"))).hasSize(1);
+ assertThat(underTest.selectByQuery(db.getSession(), organizationUuid, RuleQuery.create().withRepositoryKey("java"))).hasSize(2);
+ assertThat(underTest.selectByQuery(db.getSession(), organizationUuid,
RuleQuery.create().withKey("S001").withConfigKey("S1").withRepositoryKey("java"))).hasSize(1);
}
@Test
public void select_by_query_populates_organizationUuid_even_when_organization_has_no_metadata() {
- dbTester.prepareDbUnit(getClass(), "shared.xml");
+ db.prepareDbUnit(getClass(), "shared.xml");
String organizationUuid = "org-1";
- assertThat(underTest.selectByQuery(dbTester.getSession(), organizationUuid, RuleQuery.create()))
+ assertThat(underTest.selectByQuery(db.getSession(), organizationUuid, RuleQuery.create()))
.extracting(RuleDto::getOrganizationUuid)
.containsExactly(organizationUuid, organizationUuid);
}
@@ -328,10 +331,10 @@ public class RuleDaoTest {
.setType(RuleType.BUG)
.setCreatedAt(1_500_000_000_000L)
.setUpdatedAt(2_000_000_000_000L);
- underTest.insert(dbTester.getSession(), newRule);
- dbTester.getSession().commit();
+ underTest.insert(db.getSession(), newRule);
+ db.getSession().commit();
- RuleDefinitionDto ruleDto = underTest.selectOrFailDefinitionByKey(dbTester.getSession(), RuleKey.of("plugin", "NewRuleKey"));
+ RuleDefinitionDto ruleDto = underTest.selectOrFailDefinitionByKey(db.getSession(), RuleKey.of("plugin", "NewRuleKey"));
assertThat(ruleDto.getId()).isNotNull();
assertThat(ruleDto.getName()).isEqualTo("new name");
assertThat(ruleDto.getDescription()).isEqualTo("new description");
@@ -356,7 +359,7 @@ public class RuleDaoTest {
@Test
public void update_RuleDefinitionDto() {
- dbTester.prepareDbUnit(getClass(), "update.xml");
+ db.prepareDbUnit(getClass(), "update.xml");
RuleDefinitionDto ruleToUpdate = new RuleDefinitionDto()
.setId(1)
@@ -379,10 +382,10 @@ public class RuleDaoTest {
.setType(RuleType.BUG)
.setUpdatedAt(2_000_000_000_000L);
- underTest.update(dbTester.getSession(), ruleToUpdate);
- dbTester.getSession().commit();
+ underTest.update(db.getSession(), ruleToUpdate);
+ db.getSession().commit();
- RuleDefinitionDto ruleDto = underTest.selectOrFailDefinitionByKey(dbTester.getSession(), RuleKey.of("plugin", "NewRuleKey"));
+ RuleDefinitionDto ruleDto = underTest.selectOrFailDefinitionByKey(db.getSession(), RuleKey.of("plugin", "NewRuleKey"));
assertThat(ruleDto.getName()).isEqualTo("new name");
assertThat(ruleDto.getDescription()).isEqualTo("new description");
assertThat(ruleDto.getDescriptionFormat()).isEqualTo(RuleDto.Format.MARKDOWN);
@@ -406,7 +409,7 @@ public class RuleDaoTest {
@Test
public void update_RuleMetadataDto_inserts_row_in_RULE_METADATA_if_not_exists_yet() {
- dbTester.prepareDbUnit(getClass(), "update.xml");
+ db.prepareDbUnit(getClass(), "update.xml");
String organizationUuid = "org-1";
RuleMetadataDto metadataToUpdate = new RuleMetadataDto()
@@ -423,11 +426,11 @@ public class RuleDaoTest {
.setCreatedAt(3_500_000_000_000L)
.setUpdatedAt(4_000_000_000_000L);
- underTest.insertOrUpdate(dbTester.getSession(), metadataToUpdate);
- dbTester.getSession().commit();
+ underTest.insertOrUpdate(db.getSession(), metadataToUpdate);
+ db.getSession().commit();
OrganizationDto organization = OrganizationTesting.newOrganizationDto().setUuid(organizationUuid);
- RuleDto ruleDto = underTest.selectOrFailByKey(dbTester.getSession(), organization, RuleKey.of("checkstyle", "AvoidNull"));
+ RuleDto ruleDto = underTest.selectOrFailByKey(db.getSession(), organization, RuleKey.of("checkstyle", "AvoidNull"));
assertThat(ruleDto.getName()).isEqualTo("Avoid Null");
assertThat(ruleDto.getDescription()).isEqualTo("Should avoid NULL");
assertThat(ruleDto.getDescriptionFormat()).isNull();
@@ -459,7 +462,7 @@ public class RuleDaoTest {
@Test
public void update_RuleMetadataDto_updates_row_in_RULE_METADATA_if_already_exists() {
- dbTester.prepareDbUnit(getClass(), "update.xml");
+ db.prepareDbUnit(getClass(), "update.xml");
String organizationUuid = "org-1";
OrganizationDto organization = OrganizationTesting.newOrganizationDto().setUuid(organizationUuid);
RuleMetadataDto metadataV1 = new RuleMetadataDto()
@@ -481,11 +484,11 @@ public class RuleDaoTest {
.setCreatedAt(6_500_000_000_000L)
.setUpdatedAt(7_000_000_000_000L);
- underTest.insertOrUpdate(dbTester.getSession(), metadataV1);
- dbTester.commit();
+ underTest.insertOrUpdate(db.getSession(), metadataV1);
+ db.commit();
- assertThat(dbTester.countRowsOfTable("RULES_METADATA")).isEqualTo(1);
- RuleDto ruleDto = underTest.selectOrFailByKey(dbTester.getSession(), organization, RuleKey.of("checkstyle", "AvoidNull"));
+ assertThat(db.countRowsOfTable("RULES_METADATA")).isEqualTo(1);
+ RuleDto ruleDto = underTest.selectOrFailByKey(db.getSession(), organization, RuleKey.of("checkstyle", "AvoidNull"));
assertThat(ruleDto.getName()).isEqualTo("Avoid Null");
assertThat(ruleDto.getDescription()).isEqualTo("Should avoid NULL");
assertThat(ruleDto.getDescriptionFormat()).isNull();
@@ -514,10 +517,10 @@ public class RuleDaoTest {
assertThat(ruleDto.getCreatedAt()).isEqualTo(3_500_000_000_000L);
assertThat(ruleDto.getUpdatedAt()).isEqualTo(4_000_000_000_000L);
- underTest.insertOrUpdate(dbTester.getSession(), metadataV2);
- dbTester.commit();
+ underTest.insertOrUpdate(db.getSession(), metadataV2);
+ db.commit();
- ruleDto = underTest.selectOrFailByKey(dbTester.getSession(), organization, RuleKey.of("checkstyle", "AvoidNull"));
+ ruleDto = underTest.selectOrFailByKey(db.getSession(), organization, RuleKey.of("checkstyle", "AvoidNull"));
assertThat(ruleDto.getName()).isEqualTo("Avoid Null");
assertThat(ruleDto.getDescription()).isEqualTo("Should avoid NULL");
assertThat(ruleDto.getDescriptionFormat()).isNull();
@@ -549,8 +552,8 @@ public class RuleDaoTest {
@Test
public void select_parameters_by_rule_key() {
- dbTester.prepareDbUnit(getClass(), "select_parameters_by_rule_key.xml");
- List<RuleParamDto> ruleDtos = underTest.selectRuleParamsByRuleKey(dbTester.getSession(), RuleKey.of("checkstyle", "AvoidNull"));
+ db.prepareDbUnit(getClass(), "select_parameters_by_rule_key.xml");
+ List<RuleParamDto> ruleDtos = underTest.selectRuleParamsByRuleKey(db.getSession(), RuleKey.of("checkstyle", "AvoidNull"));
assertThat(ruleDtos.size()).isEqualTo(1);
RuleParamDto ruleDto = ruleDtos.get(0);
@@ -563,19 +566,19 @@ public class RuleDaoTest {
@Test
public void select_parameters_by_rule_keys() {
- dbTester.prepareDbUnit(getClass(), "select_parameters_by_rule_key.xml");
+ db.prepareDbUnit(getClass(), "select_parameters_by_rule_key.xml");
- assertThat(underTest.selectRuleParamsByRuleKeys(dbTester.getSession(),
+ assertThat(underTest.selectRuleParamsByRuleKeys(db.getSession(),
Arrays.asList(RuleKey.of("checkstyle", "AvoidNull"), RuleKey.of("unused", "Unused")))).hasSize(2);
- assertThat(underTest.selectRuleParamsByRuleKeys(dbTester.getSession(),
+ assertThat(underTest.selectRuleParamsByRuleKeys(db.getSession(),
singletonList(RuleKey.of("unknown", "Unknown")))).isEmpty();
}
@Test
public void insert_parameter() {
- dbTester.prepareDbUnit(getClass(), "insert_parameter.xml");
- RuleDefinitionDto rule1 = underTest.selectOrFailDefinitionByKey(dbTester.getSession(), RuleKey.of("plugin", "NewRuleKey"));
+ db.prepareDbUnit(getClass(), "insert_parameter.xml");
+ RuleDefinitionDto rule1 = underTest.selectOrFailDefinitionByKey(db.getSession(), RuleKey.of("plugin", "NewRuleKey"));
RuleParamDto param = RuleParamDto.createFor(rule1)
.setName("max")
@@ -583,19 +586,19 @@ public class RuleDaoTest {
.setDefaultValue("30")
.setDescription("My Parameter");
- underTest.insertRuleParam(dbTester.getSession(), rule1, param);
- dbTester.getSession().commit();
+ underTest.insertRuleParam(db.getSession(), rule1, param);
+ db.getSession().commit();
- dbTester.assertDbUnit(getClass(), "insert_parameter-result.xml", "rules_parameters");
+ db.assertDbUnit(getClass(), "insert_parameter-result.xml", "rules_parameters");
}
@Test
public void update_parameter() {
- dbTester.prepareDbUnit(getClass(), "update_parameter.xml");
+ db.prepareDbUnit(getClass(), "update_parameter.xml");
- RuleDefinitionDto rule1 = underTest.selectOrFailDefinitionByKey(dbTester.getSession(), RuleKey.of("checkstyle", "AvoidNull"));
+ RuleDefinitionDto rule1 = underTest.selectOrFailDefinitionByKey(db.getSession(), RuleKey.of("checkstyle", "AvoidNull"));
- List<RuleParamDto> params = underTest.selectRuleParamsByRuleKey(dbTester.getSession(), rule1.getKey());
+ List<RuleParamDto> params = underTest.selectRuleParamsByRuleKey(db.getSession(), rule1.getKey());
assertThat(params).hasSize(1);
RuleParamDto param = Iterables.getFirst(params, null);
@@ -605,20 +608,107 @@ public class RuleDaoTest {
.setDefaultValue("^[a-z]+(\\.[a-z][a-z0-9]*)*$")
.setDescription("Regular expression used to check the package names against.");
- underTest.updateRuleParam(dbTester.getSession(), rule1, param);
- dbTester.getSession().commit();
+ underTest.updateRuleParam(db.getSession(), rule1, param);
+ db.getSession().commit();
- dbTester.assertDbUnit(getClass(), "update_parameter-result.xml", "rules_parameters");
+ db.assertDbUnit(getClass(), "update_parameter-result.xml", "rules_parameters");
}
@Test
public void delete_parameter() {
- dbTester.prepareDbUnit(getClass(), "select_parameters_by_rule_key.xml");
- assertThat(underTest.selectRuleParamsByRuleKey(dbTester.getSession(), RuleKey.of("checkstyle", "AvoidNull"))).hasSize(1);
+ db.prepareDbUnit(getClass(), "select_parameters_by_rule_key.xml");
+ assertThat(underTest.selectRuleParamsByRuleKey(db.getSession(), RuleKey.of("checkstyle", "AvoidNull"))).hasSize(1);
- underTest.deleteRuleParam(dbTester.getSession(), 1);
- dbTester.getSession().commit();
+ underTest.deleteRuleParam(db.getSession(), 1);
+ db.getSession().commit();
- assertThat(underTest.selectRuleParamsByRuleKey(dbTester.getSession(), RuleKey.of("checkstyle", "AvoidNull"))).isEmpty();
+ assertThat(underTest.selectRuleParamsByRuleKey(db.getSession(), RuleKey.of("checkstyle", "AvoidNull"))).isEmpty();
+ }
+
+ @Test
+ public void scrollIndexingRules_on_empty_table() {
+ Accumulator<RuleForIndexingDto> accumulator = new Accumulator<>();
+
+ underTest.scrollIndexingRules(db.getSession(), accumulator);
+
+ assertThat(accumulator.list).isEmpty();
+ }
+
+ @Test
+ public void scrollIndexingRules() {
+ Accumulator<RuleForIndexingDto> accumulator = new Accumulator<>();
+ RuleDefinitionDto r1 = db.rules().insert();
+ RuleDefinitionDto r2 = db.rules().insert();
+
+ underTest.scrollIndexingRules(db.getSession(), accumulator);
+
+ assertThat(accumulator.list)
+ .extracting(RuleForIndexingDto::getId, RuleForIndexingDto::getRuleKey)
+ .containsExactlyInAnyOrder(tuple(r1.getId(), r1.getKey()), tuple(r2.getId(), r2.getKey()));
+ }
+
+ @Test
+ public void scrollIndexingRulesByKeys() {
+ Accumulator<RuleForIndexingDto> accumulator = new Accumulator<>();
+ RuleDefinitionDto r1 = db.rules().insert();
+ RuleDefinitionDto r2 = db.rules().insert();
+
+ underTest.scrollIndexingRulesByKeys(db.getSession(), singletonList(r1.getKey()), accumulator);
+
+ assertThat(accumulator.list)
+ .extracting(RuleForIndexingDto::getId, RuleForIndexingDto::getRuleKey)
+ .containsExactlyInAnyOrder(tuple(r1.getId(), r1.getKey()));
+ }
+
+ @Test
+ public void scrollIndexingRulesByKeys_scrolls_nothing_if_key_does_not_exist() {
+ Accumulator<RuleForIndexingDto> accumulator = new Accumulator<>();
+ RuleDefinitionDto r1 = db.rules().insert();
+
+ underTest.scrollIndexingRulesByKeys(db.getSession(), singletonList(RuleKey.of("does", "not exist")), accumulator);
+
+ assertThat(accumulator.list).isEmpty();
+ }
+
+ @Test
+ public void scrollIndexingRuleExtensions() {
+ Accumulator<RuleExtensionForIndexingDto> accumulator = new Accumulator<>();
+ RuleDefinitionDto r1 = db.rules().insert();
+ RuleMetadataDto r1Extension = db.rules().insertOrUpdateMetadata(r1, organization, r -> r.setTagsField("t1,t2"));
+ RuleDefinitionDto r2 = db.rules().insert();
+ RuleMetadataDto r2Extension = db.rules().insertOrUpdateMetadata(r2, organization, r -> r.setTagsField("t1,t3"));
+
+ underTest.scrollIndexingRuleExtensions(db.getSession(), accumulator);
+
+ assertThat(accumulator.list)
+ .extracting(RuleExtensionForIndexingDto::getRuleKey, RuleExtensionForIndexingDto::getOrganizationUuid, RuleExtensionForIndexingDto::getTags)
+ .containsExactlyInAnyOrder(
+ tuple(r1.getKey(), organization.getUuid(), r1Extension.getTagsAsString()),
+ tuple(r2.getKey(), organization.getUuid(), r2Extension.getTagsAsString()));
+ }
+
+ @Test
+ public void scrollIndexingRuleExtensionsByIds() {
+ Accumulator<RuleExtensionForIndexingDto> accumulator = new Accumulator<>();
+ RuleDefinitionDto r1 = db.rules().insert();
+ RuleMetadataDto r1Extension = db.rules().insertOrUpdateMetadata(r1, organization, r -> r.setTagsField("t1,t2"));
+ RuleExtensionId r1ExtensionId = new RuleExtensionId(organization.getUuid(), r1.getRepositoryKey(), r1.getRuleKey());
+ RuleDefinitionDto r2 = db.rules().insert();
+ RuleMetadataDto r2Extension = db.rules().insertOrUpdateMetadata(r2, organization, r -> r.setTagsField("t1,t3"));
+
+ underTest.scrollIndexingRuleExtensionsByIds(db.getSession(), singletonList(r1ExtensionId), accumulator);
+
+ assertThat(accumulator.list)
+ .extracting(RuleExtensionForIndexingDto::getRuleKey, RuleExtensionForIndexingDto::getOrganizationUuid, RuleExtensionForIndexingDto::getTags)
+ .containsExactlyInAnyOrder(
+ tuple(r1.getKey(), organization.getUuid(), r1Extension.getTagsAsString()));
+ }
+
+ private static class Accumulator<T> implements Consumer<T> {
+ private final List<T> list = new ArrayList<>();
+ @Override
+ public void accept(T dto) {
+ list.add(dto);
+ }
}
}
diff --git a/server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v65/CreateEsQueueTable.java b/server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v65/CreateEsQueueTable.java
index 78eaf82f6f5..2f0ec9feeb5 100644
--- a/server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v65/CreateEsQueueTable.java
+++ b/server/sonar-db-migration/src/main/java/org/sonar/server/platform/db/migration/version/v65/CreateEsQueueTable.java
@@ -53,6 +53,16 @@ public class CreateEsQueueTable extends DdlChange {
.setIsNullable(false)
.setLimit(4000)
.build())
+ .addColumn(VarcharColumnDef.newVarcharColumnDefBuilder()
+ .setColumnName("doc_id_type")
+ .setIsNullable(true)
+ .setLimit(20)
+ .build())
+ .addColumn(VarcharColumnDef.newVarcharColumnDefBuilder()
+ .setColumnName("doc_routing")
+ .setIsNullable(true)
+ .setLimit(4000)
+ .build())
.addColumn(BigIntegerColumnDef.newBigIntegerColumnDefBuilder()
.setColumnName("created_at")
.setIsNullable(false)
diff --git a/server/sonar-db-migration/src/test/java/org/sonar/server/platform/db/migration/version/v65/CreateEsQueueTableTest.java b/server/sonar-db-migration/src/test/java/org/sonar/server/platform/db/migration/version/v65/CreateEsQueueTableTest.java
index aa8603a17e1..de6e762983e 100644
--- a/server/sonar-db-migration/src/test/java/org/sonar/server/platform/db/migration/version/v65/CreateEsQueueTableTest.java
+++ b/server/sonar-db-migration/src/test/java/org/sonar/server/platform/db/migration/version/v65/CreateEsQueueTableTest.java
@@ -49,6 +49,8 @@ public class CreateEsQueueTableTest {
db.assertColumnDefinition(TABLE, "uuid", Types.VARCHAR, 40, false);
db.assertColumnDefinition(TABLE, "doc_type", Types.VARCHAR, 40, false);
db.assertColumnDefinition(TABLE, "doc_id", Types.VARCHAR, 4000, false);
+ db.assertColumnDefinition(TABLE, "doc_id_type", Types.VARCHAR, 20, true);
+ db.assertColumnDefinition(TABLE, "doc_routing", Types.VARCHAR, 4000, true);
db.assertColumnDefinition(TABLE, "created_at", Types.BIGINT, null, false);
}
}
diff --git a/server/sonar-db-migration/src/test/resources/org/sonar/server/platform/db/migration/version/v65/AddIndexOnEsQueueCreatedAtTest/initial.sql b/server/sonar-db-migration/src/test/resources/org/sonar/server/platform/db/migration/version/v65/AddIndexOnEsQueueCreatedAtTest/initial.sql
index a6a15d71fa2..4c0d2b532b2 100644
--- a/server/sonar-db-migration/src/test/resources/org/sonar/server/platform/db/migration/version/v65/AddIndexOnEsQueueCreatedAtTest/initial.sql
+++ b/server/sonar-db-migration/src/test/resources/org/sonar/server/platform/db/migration/version/v65/AddIndexOnEsQueueCreatedAtTest/initial.sql
@@ -1,7 +1,9 @@
CREATE TABLE "ES_QUEUE" (
"UUID" VARCHAR(40) NOT NULL PRIMARY KEY,
"DOC_TYPE" VARCHAR(40) NOT NULL,
- "DOC_UUID" VARCHAR(255) NOT NULL,
+ "DOC_ID" VARCHAR(4000) NOT NULL,
+ "DOC_ID_TYPE" VARCHAR(20),
+ "DOC_ROUTING" VARCHAR(4000),
"CREATED_AT" BIGINT NOT NULL
);
CREATE UNIQUE INDEX "PK_ES_QUEUE" ON "ES_QUEUE" ("UUID");
diff --git a/server/sonar-server/src/main/java/org/sonar/server/es/BulkIndexer.java b/server/sonar-server/src/main/java/org/sonar/server/es/BulkIndexer.java
index dbf1e2132b9..ee167116e17 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/es/BulkIndexer.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/es/BulkIndexer.java
@@ -20,14 +20,11 @@
package org.sonar.server.es;
import com.google.common.annotations.VisibleForTesting;
-import java.util.Arrays;
-import java.util.Collection;
+import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
-import java.util.Objects;
import java.util.concurrent.TimeUnit;
-import java.util.concurrent.atomic.AtomicLong;
import javax.annotation.Nullable;
import org.elasticsearch.action.ActionRequest;
import org.elasticsearch.action.admin.indices.settings.get.GetSettingsResponse;
@@ -51,12 +48,8 @@ import org.elasticsearch.search.sort.SortOrder;
import org.sonar.api.utils.log.Logger;
import org.sonar.api.utils.log.Loggers;
import org.sonar.core.util.ProgressLogger;
-import org.sonar.db.DbClient;
-import org.sonar.db.DbSession;
-import org.sonar.db.es.EsQueueDto;
import static java.lang.String.format;
-import static java.util.stream.Collectors.toList;
/**
* Helper to bulk requests in an efficient way :
@@ -76,22 +69,20 @@ public class BulkIndexer {
private final EsClient client;
private final String indexName;
private final BulkProcessor bulkProcessor;
- private final AtomicLong counter = new AtomicLong(0L);
- private final ResilientIndexerResult successCounter = new ResilientIndexerResult();
+ private final IndexingResult result = new IndexingResult();
+ private final IndexingListener indexingListener;
private final SizeHandler sizeHandler;
- private final BulkProcessorListener bulkProcessorListener;
- @Nullable
- private DbClient dbClient;
- @Nullable
- private DbSession dbSession;
- private Collection<EsQueueDto> esQueueDtos;
public BulkIndexer(EsClient client, String indexName, Size size) {
- this.dbClient = null;
+ this(client, indexName, size, IndexingListener.noop());
+ }
+
+ public BulkIndexer(EsClient client, String indexName, Size size, IndexingListener indexingListener) {
this.client = client;
this.indexName = indexName;
this.sizeHandler = size.createHandler(Runtime2.INSTANCE);
- this.bulkProcessorListener = new BulkProcessorListener();
+ this.indexingListener = indexingListener;
+ BulkProcessorListener bulkProcessorListener = new BulkProcessorListener();
this.bulkProcessor = BulkProcessor.builder(client.nativeClient(), bulkProcessorListener)
.setBackoffPolicy(BackoffPolicy.exponentialBackoff())
.setBulkSize(FLUSH_BYTE_SIZE)
@@ -101,42 +92,32 @@ public class BulkIndexer {
}
public void start() {
+ result.clear();
sizeHandler.beforeStart(this);
- counter.set(0L);
- successCounter.clear();
- }
-
- public void start(DbSession dbSession, DbClient dbClient, Collection<EsQueueDto> esQueueDtos) {
- this.dbClient = dbClient;
- this.dbSession = dbSession;
- this.esQueueDtos = esQueueDtos;
- sizeHandler.beforeStart(this);
- counter.set(0L);
- successCounter.clear();
}
/**
* @return the number of documents successfully indexed
*/
- public ResilientIndexerResult stop() {
+ public IndexingResult stop() {
try {
bulkProcessor.awaitClose(1, TimeUnit.MINUTES);
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
throw new IllegalStateException("Elasticsearch bulk requests still being executed after 1 minute", e);
- } finally {
- dbSession = null;
}
client.prepareRefresh(indexName).get();
sizeHandler.afterStop(this);
- return successCounter;
+ return result;
}
public void add(ActionRequest<?> request) {
+ result.incrementRequests();
bulkProcessor.add(request);
}
public void addDeletion(SearchRequestBuilder searchRequest) {
+ // TODO to be replaced by delete_by_query that is back in ES5
searchRequest
.addSort("_doc", SortOrder.ASC)
.setScroll(TimeValue.timeValueMinutes(5))
@@ -171,10 +152,10 @@ public class BulkIndexer {
}
public void addDeletion(IndexType indexType, String id) {
- add(client.prepareDelete(indexType, id).setRouting(id).request());
+ add(client.prepareDelete(indexType, id).request());
}
- public void addDeletion(IndexType indexType, String id, String routing) {
+ public void addDeletion(IndexType indexType, String id, @Nullable String routing) {
add(client.prepareDelete(indexType, id).setRouting(routing).request());
}
@@ -184,11 +165,11 @@ public class BulkIndexer {
*
* Note that the parameter indexName could be removed if progress logs are not needed.
*/
- public static void delete(EsClient client, String indexName, SearchRequestBuilder searchRequest) {
+ public static IndexingResult delete(EsClient client, String indexName, SearchRequestBuilder searchRequest) {
BulkIndexer bulk = new BulkIndexer(client, indexName, Size.REGULAR);
bulk.start();
bulk.addDeletion(searchRequest);
- bulk.stop();
+ return bulk.stop();
}
private final class BulkProcessorListener implements Listener {
@@ -199,37 +180,23 @@ public class BulkIndexer {
@Override
public void afterBulk(long executionId, BulkRequest request, BulkResponse response) {
- counter.addAndGet(response.getItems().length);
-
+ List<String> successDocIds = new ArrayList<>();
for (BulkItemResponse item : response.getItems()) {
if (item.isFailed()) {
LOGGER.error("index [{}], type [{}], id [{}], message [{}]", item.getIndex(), item.getType(), item.getId(), item.getFailureMessage());
- successCounter.increaseFailure();
} else {
- successCounter.increaseSuccess();
+ result.incrementSuccess();
+ successDocIds.add(item.getId());
}
}
- deleteSuccessfulItems(response);
+ indexingListener.onSuccess(successDocIds);
}
@Override
public void afterBulk(long executionId, BulkRequest req, Throwable e) {
LOGGER.error("Fail to execute bulk index request: " + req, e);
}
-
- private void deleteSuccessfulItems(BulkResponse bulkResponse) {
- if (esQueueDtos != null) {
- List<EsQueueDto> itemsToDelete = Arrays.stream(bulkResponse.getItems())
- .filter(b -> !b.isFailed())
- .map(b -> esQueueDtos.stream().filter(t -> b.getId().equals(t.getDocId())).findFirst().orElse(null))
- .filter(Objects::nonNull)
- .collect(toList());
-
- dbClient.esQueueDao().delete(dbSession, itemsToDelete);
- dbSession.commit();
- }
- }
}
public enum Size {
@@ -303,7 +270,7 @@ public class BulkIndexer {
@Override
void beforeStart(BulkIndexer bulkIndexer) {
- this.progress = new ProgressLogger(format("Progress[BulkIndexer[%s]]", bulkIndexer.indexName), bulkIndexer.counter, LOGGER)
+ this.progress = new ProgressLogger(format("Progress[BulkIndexer[%s]]", bulkIndexer.indexName), bulkIndexer.result.total, LOGGER)
.setPluralLabel("requests");
this.progress.start();
Map<String, Object> temporarySettings = new HashMap<>();
diff --git a/server/sonar-server/src/main/java/org/sonar/server/rule/index/RuleIterator.java b/server/sonar-server/src/main/java/org/sonar/server/es/IndexingListener.java
index 3c5cb50bd03..639931780bd 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/rule/index/RuleIterator.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/es/IndexingListener.java
@@ -17,13 +17,15 @@
* along with this program; if not, write to the Free Software Foundation,
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
-package org.sonar.server.rule.index;
+package org.sonar.server.es;
-import java.util.Iterator;
+import java.util.Collection;
-public interface RuleIterator extends Iterator<RuleDocWithSystemScope>, AutoCloseable {
+public interface IndexingListener {
- @Override
- void close();
+ void onSuccess(Collection<String> docIds);
+ static IndexingListener noop() {
+ return docIds -> {};
+ }
}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/es/ResilientIndexerResult.java b/server/sonar-server/src/main/java/org/sonar/server/es/IndexingResult.java
index c176a819bda..5571f509989 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/es/ResilientIndexerResult.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/es/IndexingResult.java
@@ -20,40 +20,44 @@
package org.sonar.server.es;
-/**
- * The type Resilient indexer result.
- */
-public class ResilientIndexerResult {
- private long total = 0L;
- private long failures = 0L;
+import java.util.concurrent.atomic.AtomicLong;
+
+public class IndexingResult {
+
+ // FIXME should be private
+ AtomicLong total = new AtomicLong(0L);
+ private long successes = 0L;
- public ResilientIndexerResult clear() {
- total = 0L;
- failures = 0L;
+ IndexingResult clear() {
+ total.set(0L);
+ successes = 0L;
return this;
}
- public ResilientIndexerResult increaseFailure() {
- failures += 1;
- total += 1;
- return this;
+ void incrementRequests() {
+ total.incrementAndGet();
}
- public ResilientIndexerResult increaseSuccess() {
- total += 1;
+ IndexingResult incrementSuccess() {
+ successes += 1;
return this;
}
+ public void add(IndexingResult other) {
+ total.addAndGet(other.total.get());
+ successes += other.successes;
+ }
+
public long getFailures() {
- return failures;
+ return total.get() - successes;
}
public long getTotal() {
- return total;
+ return total.get();
}
public long getSuccess() {
- return total - failures;
+ return successes;
}
/**
@@ -62,16 +66,10 @@ public class ResilientIndexerResult {
* @see {@link RecoveryIndexer#recover()}
*/
public double getFailureRatio() {
- return total == 0 ? 1 : 1.0d * failures / total;
- }
-
- public double getSuccessRatio() {
- return total == 0 ? 0 : 1 - 1.0d * failures / total;
+ return total.get() == 0 ? 1 : (1.0d * getFailures()) / total.get();
}
- public ResilientIndexerResult add(ResilientIndexerResult other) {
- this.total += other.getTotal();
- this.failures += other.getFailures();
- return this;
+ public boolean isSuccess() {
+ return total.get() == successes;
}
}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/es/RecoveryIndexer.java b/server/sonar-server/src/main/java/org/sonar/server/es/RecoveryIndexer.java
index 3c7c5ab6c69..985143d03b1 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/es/RecoveryIndexer.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/es/RecoveryIndexer.java
@@ -38,6 +38,7 @@ import org.sonar.core.util.stream.MoreCollectors;
import org.sonar.db.DbClient;
import org.sonar.db.DbSession;
import org.sonar.db.es.EsQueueDto;
+import org.sonar.server.qualityprofile.index.ActiveRuleIndexer;
import org.sonar.server.rule.index.RuleIndexer;
import org.sonar.server.user.index.UserIndexer;
@@ -66,15 +67,18 @@ public class RecoveryIndexer implements Startable {
private final DbClient dbClient;
private final UserIndexer userIndexer;
private final RuleIndexer ruleIndexer;
+ private final ActiveRuleIndexer activeRuleIndexer;
private final long minAgeInMs;
private final long loopLimit;
- public RecoveryIndexer(System2 system2, Settings settings, DbClient dbClient, UserIndexer userIndexer, RuleIndexer ruleIndexer) {
+ public RecoveryIndexer(System2 system2, Settings settings, DbClient dbClient,
+ UserIndexer userIndexer, RuleIndexer ruleIndexer, ActiveRuleIndexer activeRuleIndexer) {
this.system2 = system2;
this.settings = settings;
this.dbClient = dbClient;
this.userIndexer = userIndexer;
this.ruleIndexer = ruleIndexer;
+ this.activeRuleIndexer = activeRuleIndexer;
this.minAgeInMs = getSetting(PROPERTY_MIN_AGE, DEFAULT_MIN_AGE_IN_MS);
this.loopLimit = getSetting(PROPERTY_LOOP_LIMIT, DEFAULT_LOOP_LIMIT);
}
@@ -110,11 +114,11 @@ public class RecoveryIndexer implements Startable {
try (DbSession dbSession = dbClient.openSession(false)) {
Profiler profiler = Profiler.create(LOGGER).start();
long beforeDate = system2.now() - minAgeInMs;
- ResilientIndexerResult result = new ResilientIndexerResult();
+ IndexingResult result = new IndexingResult();
Collection<EsQueueDto> items = dbClient.esQueueDao().selectForRecovery(dbSession, beforeDate, loopLimit);
while (!items.isEmpty()) {
- ResilientIndexerResult loopResult = new ResilientIndexerResult();
+ IndexingResult loopResult = new IndexingResult();
ListMultimap<EsQueueDto.Type, EsQueueDto> itemsByType = groupItemsByType(items);
for (Map.Entry<EsQueueDto.Type, Collection<EsQueueDto>> entry : itemsByType.asMap().entrySet()) {
@@ -136,7 +140,7 @@ public class RecoveryIndexer implements Startable {
}
}
- private ResilientIndexerResult doIndex(DbSession dbSession, EsQueueDto.Type type, Collection<EsQueueDto> typeItems) {
+ private IndexingResult doIndex(DbSession dbSession, EsQueueDto.Type type, Collection<EsQueueDto> typeItems) {
LOGGER.trace(LOG_PREFIX + "processing {} {}", typeItems.size(), type);
switch (type) {
case USER:
@@ -144,9 +148,11 @@ public class RecoveryIndexer implements Startable {
case RULE_EXTENSION:
case RULE:
return ruleIndexer.index(dbSession, typeItems);
+ case ACTIVE_RULE:
+ return activeRuleIndexer.index(dbSession, typeItems);
default:
LOGGER.error(LOG_PREFIX + "ignore {} documents with unsupported type {}", typeItems.size(), type);
- return new ResilientIndexerResult();
+ return new IndexingResult();
}
}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/es/ResiliencyIndexingListener.java b/server/sonar-server/src/main/java/org/sonar/server/es/ResiliencyIndexingListener.java
new file mode 100644
index 00000000000..b9d1d27ff14
--- /dev/null
+++ b/server/sonar-server/src/main/java/org/sonar/server/es/ResiliencyIndexingListener.java
@@ -0,0 +1,64 @@
+/*
+ * 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 org.sonar.server.es;
+
+import java.util.Collection;
+import java.util.Map;
+import java.util.Objects;
+import java.util.function.Function;
+import org.sonar.core.util.stream.MoreCollectors;
+import org.sonar.db.DbClient;
+import org.sonar.db.DbSession;
+import org.sonar.db.es.EsQueueDto;
+
+import static java.util.stream.Collectors.toMap;
+
+/**
+ * Clean-up the db table es_queue when documents
+ * are successfully indexed so that the recovery
+ * daemon does not re-index them.
+ */
+public class ResiliencyIndexingListener implements IndexingListener {
+
+ private final DbClient dbClient;
+ private final DbSession dbSession;
+ private final Collection<EsQueueDto> items;
+
+ public ResiliencyIndexingListener(DbClient dbClient, DbSession dbSession, Collection<EsQueueDto> items) {
+ this.dbClient = dbClient;
+ this.dbSession = dbSession;
+ this.items = items;
+ }
+
+ @Override
+ public void onSuccess(Collection<String> docIds) {
+ if (!docIds.isEmpty()) {
+ Map<String, EsQueueDto> itemsById = items.stream().collect(toMap(EsQueueDto::getDocId, Function.identity()));
+
+ Collection<EsQueueDto> itemsToDelete = docIds
+ .stream()
+ .map(itemsById::get)
+ .filter(Objects::nonNull)
+ .collect(MoreCollectors.toArrayList(docIds.size()));
+ dbClient.esQueueDao().delete(dbSession, itemsToDelete);
+ dbSession.commit();
+ }
+ }
+}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/es/ResilientIndexer.java b/server/sonar-server/src/main/java/org/sonar/server/es/ResilientIndexer.java
index 5e818aac748..b717c831e09 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/es/ResilientIndexer.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/es/ResilientIndexer.java
@@ -37,5 +37,5 @@ public interface ResilientIndexer {
* @param items the items to be indexed
* @return the number of successful indexation
*/
- ResilientIndexerResult index(DbSession dbSession, Collection<EsQueueDto> items);
+ IndexingResult index(DbSession dbSession, Collection<EsQueueDto> items);
}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/es/queue/package-info.java b/server/sonar-server/src/main/java/org/sonar/server/es/queue/package-info.java
deleted file mode 100644
index 89f951738d6..00000000000
--- a/server/sonar-server/src/main/java/org/sonar/server/es/queue/package-info.java
+++ /dev/null
@@ -1,23 +0,0 @@
-/*
- * 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.
- */
-@ParametersAreNonnullByDefault
-package org.sonar.server.es.queue;
-
-import javax.annotation.ParametersAreNonnullByDefault;
diff --git a/server/sonar-server/src/main/java/org/sonar/server/platform/platformlevel/PlatformLevel4.java b/server/sonar-server/src/main/java/org/sonar/server/platform/platformlevel/PlatformLevel4.java
index 434eab1d84e..8145619988e 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/platform/platformlevel/PlatformLevel4.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/platform/platformlevel/PlatformLevel4.java
@@ -153,7 +153,6 @@ import org.sonar.server.qualityprofile.QProfileResetImpl;
import org.sonar.server.qualityprofile.RuleActivator;
import org.sonar.server.qualityprofile.RuleActivatorContextFactory;
import org.sonar.server.qualityprofile.index.ActiveRuleIndexer;
-import org.sonar.server.qualityprofile.index.ActiveRuleIteratorFactory;
import org.sonar.server.qualityprofile.ws.OldRestoreAction;
import org.sonar.server.qualityprofile.ws.ProfilesWs;
import org.sonar.server.qualityprofile.ws.QProfilesWsModule;
@@ -260,7 +259,6 @@ public class PlatformLevel4 extends PlatformLevel {
// quality profile
BuiltInQProfileRepositoryImpl.class,
- ActiveRuleIteratorFactory.class,
ActiveRuleIndexer.class,
XMLProfileParser.class,
XMLProfileSerializer.class,
diff --git a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/BuiltInQProfileInsertImpl.java b/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/BuiltInQProfileInsertImpl.java
index 4fc3efa1bda..9698e5f5fde 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/BuiltInQProfileInsertImpl.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/BuiltInQProfileInsertImpl.java
@@ -81,19 +81,19 @@ public class BuiltInQProfileInsertImpl implements BuiltInQProfileInsert {
Date now = new Date(system2.now());
RulesProfileDto ruleProfile = insertRulesProfile(dbSession, builtInQProfile, now);
- List<ActiveRuleChange> localChanges = builtInQProfile.getActiveRules()
+ List<ActiveRuleChange> changes = builtInQProfile.getActiveRules()
.stream()
.map(activeRule -> insertActiveRule(dbSession, ruleProfile, activeRule, now.getTime()))
.collect(MoreCollectors.toList());
- localChanges.forEach(change -> dbClient.qProfileChangeDao().insert(batchDbSession, change.toDto(null)));
+ changes.forEach(change -> dbClient.qProfileChangeDao().insert(batchDbSession, change.toDto(null)));
associateToOrganizations(dbSession, batchDbSession, builtInQProfile, ruleProfile);
- dbSession.commit();
+ // TODO batch statements should be executed through dbSession
batchDbSession.commit();
- activeRuleIndexer.indexRuleProfile(dbSession, ruleProfile);
+ activeRuleIndexer.commitAndIndex(dbSession, changes);
}
private void associateToOrganizations(DbSession dbSession, DbSession batchDbSession, BuiltInQProfile builtIn, RulesProfileDto rulesProfileDto) {
diff --git a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/BuiltInQProfileUpdateImpl.java b/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/BuiltInQProfileUpdateImpl.java
index 3338faacb33..1a1d0bd507b 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/BuiltInQProfileUpdateImpl.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/BuiltInQProfileUpdateImpl.java
@@ -62,8 +62,7 @@ public class BuiltInQProfileUpdateImpl implements BuiltInQProfileUpdate {
toBeDeactivated.forEach(ruleKey ->
changes.addAll(ruleActivator.deactivateOnBuiltInRulesProfile(dbSession, ruleProfile, ruleKey, false)));
- dbSession.commit();
- activeRuleIndexer.indexChanges(dbSession, changes);
+ activeRuleIndexer.commitAndIndex(dbSession, changes);
return changes;
}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/QProfileFactoryImpl.java b/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/QProfileFactoryImpl.java
index 5210dfff3df..90bd1b1dacb 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/QProfileFactoryImpl.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/QProfileFactoryImpl.java
@@ -128,14 +128,16 @@ public class QProfileFactoryImpl implements QProfileFactory {
db.qualityProfileDao().deleteOrgQProfilesByUuids(dbSession, uuids);
// tables related to rules_profiles and active_rules are deleted
- // only for custom profiles
+ // only for custom profiles. Built-in profiles are never
+ // deleted from table rules_profiles.
if (!rulesProfileUuidsOfCustomProfiles.isEmpty()) {
db.activeRuleDao().deleteParametersByRuleProfileUuids(dbSession, rulesProfileUuidsOfCustomProfiles);
db.activeRuleDao().deleteByRuleProfileUuids(dbSession, rulesProfileUuidsOfCustomProfiles);
db.qProfileChangeDao().deleteByRulesProfileUuids(dbSession, rulesProfileUuidsOfCustomProfiles);
db.qualityProfileDao().deleteRulesProfilesByUuids(dbSession, rulesProfileUuidsOfCustomProfiles);
+ activeRuleIndexer.commitDeletionOfProfiles(dbSession, customProfiles);
+ } else {
+ dbSession.commit();
}
- dbSession.commit();
- activeRuleIndexer.deleteByProfiles(customProfiles);
}
}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/QProfileResetImpl.java b/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/QProfileResetImpl.java
index 8c9583fd301..2d7e13bda02 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/QProfileResetImpl.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/QProfileResetImpl.java
@@ -84,8 +84,7 @@ public class QProfileResetImpl implements QProfileReset {
// ignore, probably a rule inherited from parent that can't be deactivated
}
}
- dbSession.commit();
- activeRuleIndexer.indexChanges(dbSession, changes);
+ activeRuleIndexer.commitAndIndex(dbSession, changes);
return result;
}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/RuleActivator.java b/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/RuleActivator.java
index c6dc54582bb..3df832ffd8e 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/RuleActivator.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/RuleActivator.java
@@ -83,6 +83,12 @@ public class RuleActivator {
return doActivate(dbSession, activation, context);
}
+ public List<ActiveRuleChange> activateAndCommit(DbSession dbSession, RuleActivation activation, QProfileDto profile) {
+ List<ActiveRuleChange> changes = activate(dbSession, activation, profile);
+ activeRuleIndexer.commitAndIndex(dbSession, changes);
+ return changes;
+ }
+
public List<ActiveRuleChange> activate(DbSession dbSession, RuleActivation activation, QProfileDto profile) {
RuleActivatorContext context = contextFactory.create(dbSession, activation.getRuleKey(), profile, false);
return doActivate(dbSession, activation, context);
@@ -333,10 +339,9 @@ public class RuleActivator {
* Deactivate a rule on a Quality profile. Does nothing if the rule is not activated, but
* fails (fast) if the rule or the profile does not exist.
*/
- public void deactivateAndUpdateIndex(DbSession dbSession, QProfileDto profile, RuleKey ruleKey) {
+ public void deactivateAndCommit(DbSession dbSession, QProfileDto profile, RuleKey ruleKey) {
List<ActiveRuleChange> changes = deactivate(dbSession, profile, ruleKey);
- dbSession.commit();
- activeRuleIndexer.indexChanges(dbSession, changes);
+ activeRuleIndexer.commitAndIndex(dbSession, changes);
}
/**
@@ -418,7 +423,7 @@ public class RuleActivator {
return value;
}
- public BulkChangeResult bulkActivate(DbSession dbSession, RuleQuery ruleQuery, QProfileDto profile, @Nullable String severity) {
+ public BulkChangeResult bulkActivateAndCommit(DbSession dbSession, RuleQuery ruleQuery, QProfileDto profile, @Nullable String severity) {
BulkChangeResult result = new BulkChangeResult();
Iterator<RuleKey> rules = ruleIndex.searchAll(ruleQuery);
while (rules.hasNext()) {
@@ -437,12 +442,11 @@ public class RuleActivator {
result.getErrors().addAll(e.errors());
}
}
- dbSession.commit();
- activeRuleIndexer.indexChanges(dbSession, result.getChanges());
+ activeRuleIndexer.commitAndIndex(dbSession, result.getChanges());
return result;
}
- public BulkChangeResult bulkDeactivate(DbSession dbSession, RuleQuery ruleQuery, QProfileDto profile) {
+ public BulkChangeResult bulkDeactivateAndCommit(DbSession dbSession, RuleQuery ruleQuery, QProfileDto profile) {
BulkChangeResult result = new BulkChangeResult();
Iterator<RuleKey> rules = ruleIndex.searchAll(ruleQuery);
while (rules.hasNext()) {
@@ -459,12 +463,11 @@ public class RuleActivator {
result.getErrors().addAll(e.errors());
}
}
- dbSession.commit();
- activeRuleIndexer.indexChanges(dbSession, result.getChanges());
+ activeRuleIndexer.commitAndIndex(dbSession, result.getChanges());
return result;
}
- public List<ActiveRuleChange> setParent(DbSession dbSession, QProfileDto profile, @Nullable QProfileDto parent) {
+ public List<ActiveRuleChange> setParentAndCommit(DbSession dbSession, QProfileDto profile, @Nullable QProfileDto parent) {
checkRequest(
parent == null || profile.getLanguage().equals(parent.getLanguage()),
"Cannot set the profile '%s' as the parent of profile '%s' since their languages differ ('%s' != '%s')",
@@ -492,8 +495,7 @@ public class RuleActivator {
}
}
}
- dbSession.commit();
- activeRuleIndexer.indexChanges(dbSession, changes);
+ activeRuleIndexer.commitAndIndex(dbSession, changes);
return changes;
}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/index/ActiveRuleIndexer.java b/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/index/ActiveRuleIndexer.java
index f61b6089a68..5f768e0f88b 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/index/ActiveRuleIndexer.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/index/ActiveRuleIndexer.java
@@ -20,48 +20,64 @@
package org.sonar.server.qualityprofile.index;
import com.google.common.collect.ImmutableSet;
-import java.util.ArrayList;
import java.util.Collection;
+import java.util.HashMap;
import java.util.List;
+import java.util.Map;
import java.util.Set;
+import javax.annotation.Nullable;
import org.elasticsearch.action.index.IndexRequest;
import org.elasticsearch.action.search.SearchRequestBuilder;
import org.elasticsearch.index.query.QueryBuilders;
import org.sonar.api.rule.RuleKey;
+import org.sonar.api.utils.log.Logger;
+import org.sonar.api.utils.log.Loggers;
import org.sonar.db.DbClient;
import org.sonar.db.DbSession;
+import org.sonar.db.es.EsQueueDto;
+import org.sonar.db.qualityprofile.IndexedActiveRuleDto;
import org.sonar.db.qualityprofile.QProfileDto;
import org.sonar.db.qualityprofile.RulesProfileDto;
+import org.sonar.db.rule.SeverityUtil;
import org.sonar.server.es.BulkIndexer;
import org.sonar.server.es.BulkIndexer.Size;
import org.sonar.server.es.EsClient;
import org.sonar.server.es.IndexType;
+import org.sonar.server.es.IndexingListener;
+import org.sonar.server.es.IndexingResult;
+import org.sonar.server.es.ResiliencyIndexingListener;
+import org.sonar.server.es.ResilientIndexer;
import org.sonar.server.es.StartupIndexer;
+import org.sonar.server.qualityprofile.ActiveRule;
import org.sonar.server.qualityprofile.ActiveRuleChange;
import org.sonar.server.rule.index.RuleIndexDefinition;
import static org.elasticsearch.index.query.QueryBuilders.termQuery;
+import static org.sonar.core.util.stream.MoreCollectors.toArrayList;
import static org.sonar.server.rule.index.RuleIndexDefinition.FIELD_ACTIVE_RULE_PROFILE_UUID;
-import static org.sonar.server.rule.index.RuleIndexDefinition.FIELD_ACTIVE_RULE_RULE_KEY;
import static org.sonar.server.rule.index.RuleIndexDefinition.INDEX_TYPE_ACTIVE_RULE;
-public class ActiveRuleIndexer implements StartupIndexer {
+public class ActiveRuleIndexer implements StartupIndexer, ResilientIndexer {
+
+ private static final Logger LOGGER = Loggers.get(ActiveRuleIndexer.class);
+ private static final String ID_TYPE_ACTIVE_RULE_ID = "activeRuleId";
+ private static final String ID_TYPE_RULE_PROFILE_UUID = "ruleProfileUuid";
private final DbClient dbClient;
private final EsClient esClient;
- private final ActiveRuleIteratorFactory activeRuleIteratorFactory;
- public ActiveRuleIndexer(DbClient dbClient, EsClient esClient, ActiveRuleIteratorFactory activeRuleIteratorFactory) {
+ public ActiveRuleIndexer(DbClient dbClient, EsClient esClient) {
this.dbClient = dbClient;
this.esClient = esClient;
- this.activeRuleIteratorFactory = activeRuleIteratorFactory;
}
@Override
public void indexOnStartup(Set<IndexType> uninitializedIndexTypes) {
try (DbSession dbSession = dbClient.openSession(false)) {
- ActiveRuleIterator dbCursor = activeRuleIteratorFactory.createForAll(dbSession);
- scrollDbAndIndex(dbCursor, Size.LARGE);
+ BulkIndexer bulkIndexer = createBulkIndexer(Size.LARGE, IndexingListener.noop());
+ bulkIndexer.start();
+ dbClient.activeRuleDao().scrollAllForIndexing(dbSession, ar -> bulkIndexer.add(newIndexRequest(ar)));
+ bulkIndexer.stop();
}
}
@@ -70,79 +86,144 @@ public class ActiveRuleIndexer implements StartupIndexer {
return ImmutableSet.of(RuleIndexDefinition.INDEX_TYPE_ACTIVE_RULE);
}
+ public void commitAndIndex(DbSession dbSession, Collection<ActiveRuleChange> changes) {
+ List<EsQueueDto> items = changes.stream()
+ .map(ActiveRuleChange::getActiveRule)
+ .map(ar -> newQueueDto(String.valueOf(ar.getId()), ID_TYPE_ACTIVE_RULE_ID, ar.getRuleKey().toString()))
+ .collect(toArrayList());
+
+ dbClient.esQueueDao().insert(dbSession, items);
+ dbSession.commit();
+ postCommit(dbSession, items);
+ }
+
+ public void commitDeletionOfProfiles(DbSession dbSession, Collection<QProfileDto> profiles) {
+ List<EsQueueDto> items = profiles.stream()
+ .map(QProfileDto::getRulesProfileUuid)
+ .distinct()
+ .map(ruleProfileUuid -> newQueueDto(ruleProfileUuid, ID_TYPE_RULE_PROFILE_UUID, null))
+ .collect(toArrayList());
+
+ dbClient.esQueueDao().insert(dbSession, items);
+ dbSession.commit();
+ postCommit(dbSession, items);
+ }
+
+ /**
+ * Entry point for Byteman tests. See directory tests/resilience.
+ */
+ private void postCommit(DbSession dbSession, Collection<EsQueueDto> items) {
+ index(dbSession, items);
+ }
+
/**
- * Important - the existing documents are not deleted, so this method
- * does not guarantee consistency of index.
+ * @return the number of items that have been successfully indexed
*/
- public void indexRuleProfile(DbSession dbSession, RulesProfileDto ruleProfile) {
- try (ActiveRuleIterator dbCursor = activeRuleIteratorFactory.createForRuleProfile(dbSession, ruleProfile)) {
- scrollDbAndIndex(dbCursor, Size.REGULAR);
+ @Override
+ public IndexingResult index(DbSession dbSession, Collection<EsQueueDto> items) {
+ IndexingResult result = new IndexingResult();
+
+ if (items.isEmpty()) {
+ return result;
}
+
+ Map<Long, EsQueueDto> activeRuleItems = new HashMap<>();
+ Map<String, EsQueueDto> ruleProfileItems = new HashMap<>();
+ items.forEach(i -> {
+ if (ID_TYPE_RULE_PROFILE_UUID.equals(i.getDocIdType())) {
+ ruleProfileItems.put(i.getDocId(), i);
+ } else if (ID_TYPE_ACTIVE_RULE_ID.equals(i.getDocIdType())) {
+ activeRuleItems.put(Long.parseLong(i.getDocId()), i);
+ } else {
+ LOGGER.error("Unsupported es_queue.doc_id_type. Removing row from queue: " + i);
+ deleteQueueDto(dbSession, i);
+ }
+ });
+
+ if (!activeRuleItems.isEmpty()) {
+ result.add(doIndexActiveRules(dbSession, activeRuleItems));
+ }
+ if (!ruleProfileItems.isEmpty()) {
+ result.add(doIndexRuleProfiles(dbSession, ruleProfileItems));
+ }
+ return result;
}
- public void indexChanges(DbSession dbSession, List<ActiveRuleChange> changes) {
- BulkIndexer bulk = createBulkIndexer(Size.REGULAR);
- bulk.start();
- List<Integer> idsOfTouchedActiveRules = new ArrayList<>();
- changes.stream()
- .filter(c -> c.getActiveRule() != null)
- .forEach(c -> {
- if (c.getType().equals(ActiveRuleChange.Type.DEACTIVATED)) {
- bulk.addDeletion(INDEX_TYPE_ACTIVE_RULE, String.valueOf(c.getActiveRule().getId()), c.getKey().getRuleKey().toString());
- } else {
- idsOfTouchedActiveRules.add(c.getActiveRule().getId());
- }
+ private IndexingResult doIndexActiveRules(DbSession dbSession, Map<Long, EsQueueDto> activeRuleItems) {
+ BulkIndexer bulkIndexer = createBulkIndexer(Size.REGULAR, new ResiliencyIndexingListener(dbClient, dbSession, activeRuleItems.values()));
+ bulkIndexer.start();
+ Map<Long, EsQueueDto> remaining = new HashMap<>(activeRuleItems);
+ dbClient.activeRuleDao().scrollByIdsForIndexing(dbSession, activeRuleItems.keySet(),
+ // only index requests, no deletion requests.
+ // Deactivated users are not deleted but updated.
+ i -> {
+ remaining.remove(i.getId());
+ bulkIndexer.add(newIndexRequest(i));
});
- try (ActiveRuleIterator dbCursor = activeRuleIteratorFactory.createForActiveRules(dbSession, idsOfTouchedActiveRules)) {
- while (dbCursor.hasNext()) {
- ActiveRuleDoc activeRule = dbCursor.next();
- bulk.add(newIndexRequest(activeRule));
+
+ // the remaining ids reference rows that don't exist in db. They must
+ // be deleted from index.
+ remaining.values().forEach(item -> bulkIndexer.addDeletion(RuleIndexDefinition.INDEX_TYPE_ACTIVE_RULE,
+ item.getDocId(), item.getDocRouting()));
+ return bulkIndexer.stop();
+ }
+
+ private IndexingResult doIndexRuleProfiles(DbSession dbSession, Map<String, EsQueueDto> ruleProfileItems) {
+ IndexingResult result = new IndexingResult();
+
+ for (Map.Entry<String, EsQueueDto> entry : ruleProfileItems.entrySet()) {
+ String ruleProfileUUid = entry.getKey();
+ EsQueueDto item = entry.getValue();
+ IndexingResult profileResult;
+
+ RulesProfileDto profile = dbClient.qualityProfileDao().selectRuleProfile(dbSession, ruleProfileUUid);
+ if (profile == null) {
+ // profile does not exist anymore in db --> related documents must be deleted from index rules/activeRule
+ SearchRequestBuilder search = esClient.prepareSearch(INDEX_TYPE_ACTIVE_RULE)
+ .setQuery(QueryBuilders.boolQuery().must(termQuery(FIELD_ACTIVE_RULE_PROFILE_UUID, ruleProfileUUid)));
+ profileResult = BulkIndexer.delete(esClient, INDEX_TYPE_ACTIVE_RULE.getIndex(), search);
+
+ } else {
+ BulkIndexer bulkIndexer = createBulkIndexer(Size.REGULAR, IndexingListener.noop());
+ bulkIndexer.start();
+ dbClient.activeRuleDao().scrollByRuleProfileForIndexing(dbSession, ruleProfileUUid, i -> bulkIndexer.add(newIndexRequest(i)));
+ profileResult = bulkIndexer.stop();
+ }
+
+ if (profileResult.isSuccess()) {
+ deleteQueueDto(dbSession, item);
}
+ result.add(profileResult);
}
- bulk.stop();
- }
- public void deleteByProfiles(Collection<QProfileDto> profiles) {
- BulkIndexer bulk = createBulkIndexer(Size.REGULAR);
- bulk.start();
- profiles.forEach(profile -> {
- SearchRequestBuilder search = esClient.prepareSearch(INDEX_TYPE_ACTIVE_RULE)
- .setQuery(QueryBuilders.boolQuery().must(termQuery(FIELD_ACTIVE_RULE_PROFILE_UUID, profile.getRulesProfileUuid())));
- bulk.addDeletion(search);
- });
- bulk.stop();
+ return result;
}
- public void deleteByRuleKeys(Collection<RuleKey> ruleKeys) {
- BulkIndexer bulk = createBulkIndexer(Size.REGULAR);
- bulk.start();
- ruleKeys.forEach(ruleKey -> {
- SearchRequestBuilder search = esClient.prepareSearch(INDEX_TYPE_ACTIVE_RULE)
- .setQuery(QueryBuilders.boolQuery().must(termQuery(FIELD_ACTIVE_RULE_RULE_KEY, ruleKey.toString())));
- bulk.addDeletion(search);
- });
- bulk.stop();
+ private void deleteQueueDto(DbSession dbSession, EsQueueDto item) {
+ dbClient.esQueueDao().delete(dbSession, item);
+ dbSession.commit();
}
- private BulkIndexer createBulkIndexer(Size size) {
- return new BulkIndexer(esClient, INDEX_TYPE_ACTIVE_RULE.getIndex(), size);
+ private BulkIndexer createBulkIndexer(Size size, IndexingListener listener) {
+ return new BulkIndexer(esClient, INDEX_TYPE_ACTIVE_RULE.getIndex(), size, listener);
}
- private static IndexRequest newIndexRequest(ActiveRuleDoc doc) {
+ private static IndexRequest newIndexRequest(IndexedActiveRuleDto dto) {
+ ActiveRuleDoc doc = new ActiveRuleDoc(String.valueOf(dto.getId()));
+ doc.setRuleProfileUuid(dto.getRuleProfileUuid());
+ doc.setSeverity(SeverityUtil.getSeverityFromOrdinal(dto.getSeverity()));
+ doc.setRuleKey(RuleKey.of(dto.getRepository(), dto.getKey()));
+ // all the fields must be present, even if value is null
+ String inheritance = dto.getInheritance();
+ doc.setInheritance(inheritance == null ? ActiveRule.Inheritance.NONE.name() : inheritance);
return new IndexRequest(INDEX_TYPE_ACTIVE_RULE.getIndex(), INDEX_TYPE_ACTIVE_RULE.getType())
.id(doc.getId())
+ .parent(doc.getParent())
.routing(doc.getRouting())
- .parent(doc.getRuleKey().toString())
.source(doc.getFields());
}
- private void scrollDbAndIndex(ActiveRuleIterator dbCursor, Size size) {
- BulkIndexer bulk = new BulkIndexer(esClient, INDEX_TYPE_ACTIVE_RULE.getIndex(), size);
- bulk.start();
- while (dbCursor.hasNext()) {
- ActiveRuleDoc activeRule = dbCursor.next();
- bulk.add(newIndexRequest(activeRule));
- }
- bulk.stop();
+ private static EsQueueDto newQueueDto(String docId, String docIdType, @Nullable String routing) {
+ return EsQueueDto.create(EsQueueDto.Type.ACTIVE_RULE, docId, docIdType, routing);
}
}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/index/ActiveRuleIterator.java b/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/index/ActiveRuleIterator.java
deleted file mode 100644
index 617727a4208..00000000000
--- a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/index/ActiveRuleIterator.java
+++ /dev/null
@@ -1,27 +0,0 @@
-/*
- * 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 org.sonar.server.qualityprofile.index;
-
-import java.util.Iterator;
-
-public interface ActiveRuleIterator extends Iterator<ActiveRuleDoc>, AutoCloseable {
- @Override
- void close();
-}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/index/ActiveRuleIteratorFactory.java b/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/index/ActiveRuleIteratorFactory.java
deleted file mode 100644
index 9bf73ccef59..00000000000
--- a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/index/ActiveRuleIteratorFactory.java
+++ /dev/null
@@ -1,48 +0,0 @@
-/*
- * 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 org.sonar.server.qualityprofile.index;
-
-import java.util.Collection;
-import org.sonar.api.server.ServerSide;
-import org.sonar.db.DbClient;
-import org.sonar.db.DbSession;
-import org.sonar.db.qualityprofile.RulesProfileDto;
-
-@ServerSide
-public class ActiveRuleIteratorFactory {
-
- private final DbClient dbClient;
-
- public ActiveRuleIteratorFactory(DbClient dbClient) {
- this.dbClient = dbClient;
- }
-
- public ActiveRuleIterator createForAll(DbSession dbSession) {
- return new ActiveRuleIteratorForSingleChunk(dbClient, dbSession);
- }
-
- public ActiveRuleIterator createForRuleProfile(DbSession dbSession, RulesProfileDto ruleProfile) {
- return new ActiveRuleIteratorForSingleChunk(dbClient, dbSession, ruleProfile);
- }
-
- public ActiveRuleIterator createForActiveRules(DbSession dbSession, Collection<Integer> activeRuleIds) {
- return new ActiveRuleIteratorForMultipleChunks(dbClient, dbSession, activeRuleIds);
- }
-}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/index/ActiveRuleIteratorForMultipleChunks.java b/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/index/ActiveRuleIteratorForMultipleChunks.java
deleted file mode 100644
index 4ac2b92de88..00000000000
--- a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/index/ActiveRuleIteratorForMultipleChunks.java
+++ /dev/null
@@ -1,69 +0,0 @@
-/*
- * 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 org.sonar.server.qualityprofile.index;
-
-import java.util.Collection;
-import java.util.Iterator;
-import java.util.List;
-import org.sonar.db.DatabaseUtils;
-import org.sonar.db.DbClient;
-import org.sonar.db.DbSession;
-
-import static java.util.Optional.ofNullable;
-
-public class ActiveRuleIteratorForMultipleChunks implements ActiveRuleIterator {
-
- private final DbClient dbClient;
- private final DbSession dbSession;
- private final Iterator<List<Integer>> iteratorOverChunks;
- private ActiveRuleIteratorForSingleChunk currentChunk;
-
- public ActiveRuleIteratorForMultipleChunks(DbClient dbClient, DbSession dbSession, Collection<Integer> activeRuleIds) {
- this.dbClient = dbClient;
- this.dbSession = dbSession;
- this.iteratorOverChunks = DatabaseUtils.toUniqueAndSortedPartitions(activeRuleIds).iterator();
- }
-
- @Override
- public boolean hasNext() {
- if (currentChunk != null && currentChunk.hasNext()) {
- return true;
- }
- return iteratorOverChunks.hasNext();
- }
-
- @Override
- public ActiveRuleDoc next() {
- if (currentChunk == null || !currentChunk.hasNext()) {
- currentChunk = nextChunk();
- }
- return currentChunk.next();
- }
-
- private ActiveRuleIteratorForSingleChunk nextChunk() {
- List<Integer> nextInput = iteratorOverChunks.next();
- return new ActiveRuleIteratorForSingleChunk(dbClient, dbSession, nextInput);
- }
-
- @Override
- public void close() {
- ofNullable(currentChunk).ifPresent(ActiveRuleIterator::close);
- }
-}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/index/ActiveRuleIteratorForSingleChunk.java b/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/index/ActiveRuleIteratorForSingleChunk.java
deleted file mode 100644
index 0714ea34ef0..00000000000
--- a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/index/ActiveRuleIteratorForSingleChunk.java
+++ /dev/null
@@ -1,132 +0,0 @@
-/*
- * 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 org.sonar.server.qualityprofile.index;
-
-import java.sql.PreparedStatement;
-import java.sql.ResultSet;
-import java.sql.SQLException;
-import java.util.List;
-import org.apache.commons.lang.StringUtils;
-import org.sonar.api.rule.RuleKey;
-import org.sonar.db.DatabaseUtils;
-import org.sonar.db.DbClient;
-import org.sonar.db.DbSession;
-import org.sonar.db.ResultSetIterator;
-import org.sonar.db.qualityprofile.RulesProfileDto;
-import org.sonar.db.rule.SeverityUtil;
-import org.sonar.server.qualityprofile.ActiveRule;
-
-import static org.apache.commons.lang.StringUtils.repeat;
-
-/**
- * Scrolls over table ISSUES and reads documents to populate
- * the issues index
- */
-class ActiveRuleIteratorForSingleChunk implements ActiveRuleIterator {
-
- private static final String[] COLUMNS = {
- "ar.id",
- "ar.failure_level",
- "ar.inheritance",
- "r.plugin_name",
- "r.plugin_rule_key",
- "rp.kee"
- };
-
- private static final String SQL_ALL = "select " + StringUtils.join(COLUMNS, ",") + " from active_rules ar " +
- " inner join rules_profiles rp on rp.id = ar.profile_id " +
- " inner join rules r on r.id = ar.rule_id ";
-
- private final PreparedStatement stmt;
- private final ResultSetIterator<ActiveRuleDoc> iterator;
-
- ActiveRuleIteratorForSingleChunk(DbClient dbClient, DbSession dbSession) {
- try {
- stmt = dbClient.getMyBatis().newScrollingSelectStatement(dbSession, SQL_ALL);
- iterator = new ActiveRuleIteratorInternal(stmt);
- } catch (Exception e) {
- throw new IllegalStateException("Fail to prepare SQL request to select all active_rules", e);
- }
- }
-
- ActiveRuleIteratorForSingleChunk(DbClient dbClient, DbSession dbSession, RulesProfileDto ruleProfile) {
- try {
- stmt = dbClient.getMyBatis().newScrollingSelectStatement(dbSession, SQL_ALL + " where rp.kee = ?");
- stmt.setString(1, ruleProfile.getKee());
- iterator = new ActiveRuleIteratorInternal(stmt);
- } catch (Exception e) {
- throw new IllegalStateException("Fail to prepare SQL request to select active_rules of profile " + ruleProfile.getKee(), e);
- }
- }
-
- ActiveRuleIteratorForSingleChunk(DbClient dbClient, DbSession dbSession, List<Integer> activeRuleIds) {
- try {
- stmt = dbClient.getMyBatis().newScrollingSelectStatement(dbSession, SQL_ALL + " where ar.id in (" + repeat("?", ",", activeRuleIds.size()) + ")");
- for (int i = 0; i < activeRuleIds.size(); i++) {
- stmt.setInt(i + 1, activeRuleIds.get(i));
- }
- iterator = new ActiveRuleIteratorInternal(stmt);
- } catch (Exception e) {
- throw new IllegalStateException("Fail to prepare SQL request to select active_rules", e);
- }
- }
-
- @Override
- public boolean hasNext() {
- return iterator.hasNext();
- }
-
- @Override
- public ActiveRuleDoc next() {
- return iterator.next();
- }
-
- @Override
- public void close() {
- try {
- iterator.close();
- } finally {
- DatabaseUtils.closeQuietly(stmt);
- }
- }
-
- private static final class ActiveRuleIteratorInternal extends ResultSetIterator<ActiveRuleDoc> {
-
- ActiveRuleIteratorInternal(PreparedStatement stmt) throws SQLException {
- super(stmt);
- }
-
- @Override
- protected ActiveRuleDoc read(ResultSet rs) throws SQLException {
- long activeRuleId = rs.getLong(1);
- int severity = rs.getInt(2);
- String inheritance = rs.getString(3);
- RuleKey ruleKey = RuleKey.of(rs.getString(4), rs.getString(5));
- String ruleProfileUuid = rs.getString(6);
-
- return new ActiveRuleDoc(String.valueOf(activeRuleId))
- .setRuleProfileUuid(ruleProfileUuid)
- .setRuleKey(ruleKey)
- // all the fields must be present, even if value is null
- .setSeverity(SeverityUtil.getSeverityFromOrdinal(severity))
- .setInheritance(inheritance == null ? ActiveRule.Inheritance.NONE.name() : inheritance);
- }
- }
-}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/ws/ActivateRuleAction.java b/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/ws/ActivateRuleAction.java
index 9badee93750..efcb4fe8e7b 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/ws/ActivateRuleAction.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/ws/ActivateRuleAction.java
@@ -19,7 +19,6 @@
*/
package org.sonar.server.qualityprofile.ws;
-import java.util.List;
import java.util.Map;
import org.sonar.api.rule.RuleKey;
import org.sonar.api.rule.Severity;
@@ -30,10 +29,8 @@ import org.sonar.api.utils.KeyValueFormat;
import org.sonar.db.DbClient;
import org.sonar.db.DbSession;
import org.sonar.db.qualityprofile.QProfileDto;
-import org.sonar.server.qualityprofile.ActiveRuleChange;
import org.sonar.server.qualityprofile.RuleActivation;
import org.sonar.server.qualityprofile.RuleActivator;
-import org.sonar.server.qualityprofile.index.ActiveRuleIndexer;
import org.sonar.server.user.UserSession;
import static java.lang.String.format;
@@ -51,14 +48,12 @@ public class ActivateRuleAction implements QProfileWsAction {
private final RuleActivator ruleActivator;
private final UserSession userSession;
private final QProfileWsSupport wsSupport;
- private final ActiveRuleIndexer activeRuleIndexer;
- public ActivateRuleAction(DbClient dbClient, RuleActivator ruleActivator, UserSession userSession, QProfileWsSupport wsSupport, ActiveRuleIndexer activeRuleIndexer) {
+ public ActivateRuleAction(DbClient dbClient, RuleActivator ruleActivator, UserSession userSession, QProfileWsSupport wsSupport) {
this.dbClient = dbClient;
this.ruleActivator = ruleActivator;
this.userSession = userSession;
this.wsSupport = wsSupport;
- this.activeRuleIndexer = activeRuleIndexer;
}
public void define(WebService.NewController controller) {
@@ -104,9 +99,7 @@ public class ActivateRuleAction implements QProfileWsAction {
wsSupport.checkPermission(dbSession, profile);
wsSupport.checkNotBuiltInt(profile);
RuleActivation activation = readActivation(request);
- List<ActiveRuleChange> changes = ruleActivator.activate(dbSession, activation, profile);
- dbSession.commit();
- activeRuleIndexer.indexChanges(dbSession, changes);
+ ruleActivator.activateAndCommit(dbSession, activation, profile);
}
response.noContent();
diff --git a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/ws/ActivateRulesAction.java b/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/ws/ActivateRulesAction.java
index bb8918fcf48..fe1b61fc998 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/ws/ActivateRulesAction.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/ws/ActivateRulesAction.java
@@ -87,7 +87,7 @@ public class ActivateRulesAction implements QProfileWsAction {
QProfileDto profile = wsSupport.getProfile(dbSession, fromKey(qualityProfileKey));
wsSupport.checkPermission(dbSession, profile);
wsSupport.checkNotBuiltInt(profile);
- result = ruleActivator.bulkActivate(dbSession, ruleQueryFactory.createRuleQuery(dbSession, request), profile, request.param(PARAM_TARGET_SEVERITY));
+ result = ruleActivator.bulkActivateAndCommit(dbSession, ruleQueryFactory.createRuleQuery(dbSession, request), profile, request.param(PARAM_TARGET_SEVERITY));
}
writeResponse(result, response);
diff --git a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/ws/ChangeParentAction.java b/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/ws/ChangeParentAction.java
index 6ab45451fbb..8107ecd1b3f 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/ws/ChangeParentAction.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/ws/ChangeParentAction.java
@@ -98,13 +98,13 @@ public class ChangeParentAction implements QProfileWsAction {
String parentKey = request.param(PARAM_PARENT_PROFILE);
String parentName = request.param(PARAM_PARENT_NAME);
if (isEmpty(parentKey) && isEmpty(parentName)) {
- ruleActivator.setParent(dbSession, profile, null);
+ ruleActivator.setParentAndCommit(dbSession, profile, null);
} else {
String parentOrganizationKey = parentKey == null ? organization.getKey() : null;
String parentLanguage = parentKey == null ? request.param(PARAM_LANGUAGE) : null;
QProfileReference parentRef = QProfileReference.from(parentKey, parentOrganizationKey, parentLanguage, parentName);
QProfileDto parent = wsSupport.getProfile(dbSession, parentRef);
- ruleActivator.setParent(dbSession, profile, parent);
+ ruleActivator.setParentAndCommit(dbSession, profile, parent);
}
response.noContent();
diff --git a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/ws/CreateAction.java b/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/ws/CreateAction.java
index 8dc6bf90b30..26933ceb73d 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/ws/CreateAction.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/ws/CreateAction.java
@@ -132,8 +132,7 @@ public class CreateAction implements QProfileWsAction {
result.add(exporters.importXml(profile, importerKey, contentToImport, dbSession));
}
}
- dbSession.commit();
- activeRuleIndexer.indexChanges(dbSession, result.getChanges());
+ activeRuleIndexer.commitAndIndex(dbSession, result.getChanges());
return buildResponse(result, organization);
}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/ws/DeactivateRuleAction.java b/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/ws/DeactivateRuleAction.java
index fa7f99d76b7..3bb8ed16dcc 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/ws/DeactivateRuleAction.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/ws/DeactivateRuleAction.java
@@ -79,7 +79,7 @@ public class DeactivateRuleAction implements QProfileWsAction {
QProfileDto profile = wsSupport.getProfile(dbSession, QProfileReference.fromKey(qualityProfileKey));
wsSupport.checkPermission(dbSession, profile);
wsSupport.checkNotBuiltInt(profile);
- ruleActivator.deactivateAndUpdateIndex(dbSession, profile, ruleKey);
+ ruleActivator.deactivateAndCommit(dbSession, profile, ruleKey);
}
response.noContent();
}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/ws/DeactivateRulesAction.java b/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/ws/DeactivateRulesAction.java
index 243f8e7d196..cff201af44c 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/ws/DeactivateRulesAction.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/qualityprofile/ws/DeactivateRulesAction.java
@@ -80,7 +80,7 @@ public class DeactivateRulesAction implements QProfileWsAction {
QProfileDto profile = wsSupport.getProfile(dbSession, QProfileReference.fromKey(qualityProfileKey));
wsSupport.checkPermission(dbSession, profile);
wsSupport.checkNotBuiltInt(profile);
- result = ruleActivator.bulkDeactivate(dbSession, ruleQueryFactory.createRuleQuery(dbSession, request), profile);
+ result = ruleActivator.bulkDeactivateAndCommit(dbSession, ruleQueryFactory.createRuleQuery(dbSession, request), profile);
}
writeResponse(result, response);
}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/rule/RegisterRules.java b/server/sonar-server/src/main/java/org/sonar/server/rule/RegisterRules.java
index 822ef0b315f..a3750b27789 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/rule/RegisterRules.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/rule/RegisterRules.java
@@ -131,7 +131,7 @@ public class RegisterRules implements Startable {
persistRepositories(dbSession, context.repositories());
ruleIndexer.commitAndIndex(dbSession, keysToIndex);
- activeRuleIndexer.indexChanges(dbSession, changes);
+ activeRuleIndexer.commitAndIndex(dbSession, changes);
profiler.stopDebug();
webServerRuleFinder.startCaching();
diff --git a/server/sonar-server/src/main/java/org/sonar/server/rule/index/RuleDoc.java b/server/sonar-server/src/main/java/org/sonar/server/rule/index/RuleDoc.java
index 601e6118f79..95f4ce1105d 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/rule/index/RuleDoc.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/rule/index/RuleDoc.java
@@ -204,32 +204,32 @@ public class RuleDoc extends BaseDoc {
return ReflectionToStringBuilder.toString(this);
}
- public static RuleDoc of(RuleForIndexingDto ruleForIndexingDto) {
+ public static RuleDoc of(RuleForIndexingDto dto) {
RuleDoc ruleDoc = new RuleDoc()
- .setKey(ruleForIndexingDto.getRuleKey().toString())
- .setRepository(ruleForIndexingDto.getRepository())
- .setInternalKey(ruleForIndexingDto.getInternalKey())
- .setIsTemplate(ruleForIndexingDto.isTemplate())
- .setLanguage(ruleForIndexingDto.getLanguage())
- .setName(ruleForIndexingDto.getName())
- .setRuleKey(ruleForIndexingDto.getPluginRuleKey())
- .setSeverity(ruleForIndexingDto.getSeverityAsString())
- .setStatus(ruleForIndexingDto.getStatus().toString())
- .setType(ruleForIndexingDto.getTypeAsRuleType())
- .setCreatedAt(ruleForIndexingDto.getCreatedAt())
- .setUpdatedAt(ruleForIndexingDto.getUpdatedAt());
-
- if (ruleForIndexingDto.getPluginRuleKey() != null && ruleForIndexingDto.getRepository() != null) {
- ruleDoc.setTemplateKey(RuleKey.of(ruleForIndexingDto.getPluginRuleKey(), ruleForIndexingDto.getRepository()).toString());
+ .setKey(dto.getRuleKey().toString())
+ .setRepository(dto.getRepository())
+ .setInternalKey(dto.getInternalKey())
+ .setIsTemplate(dto.isTemplate())
+ .setLanguage(dto.getLanguage())
+ .setName(dto.getName())
+ .setRuleKey(dto.getPluginRuleKey())
+ .setSeverity(dto.getSeverityAsString())
+ .setStatus(dto.getStatus().toString())
+ .setType(dto.getTypeAsRuleType())
+ .setCreatedAt(dto.getCreatedAt())
+ .setUpdatedAt(dto.getUpdatedAt());
+
+ if (dto.getTemplateRuleKey() != null && dto.getTemplateRepository() != null) {
+ ruleDoc.setTemplateKey(RuleKey.of(dto.getTemplateRepository(), dto.getTemplateRuleKey()).toString());
} else {
ruleDoc.setTemplateKey(null);
}
- if (ruleForIndexingDto.getDescription() != null && ruleForIndexingDto.getDescriptionFormat() != null) {
- if (RuleDto.Format.HTML == ruleForIndexingDto.getDescriptionFormat()) {
- ruleDoc.setHtmlDescription(ruleForIndexingDto.getDescription());
+ if (dto.getDescription() != null && dto.getDescriptionFormat() != null) {
+ if (RuleDto.Format.HTML == dto.getDescriptionFormat()) {
+ ruleDoc.setHtmlDescription(dto.getDescription());
} else {
- ruleDoc.setHtmlDescription(Markdown.convertToHtml(ruleForIndexingDto.getDescription()));;
+ ruleDoc.setHtmlDescription(Markdown.convertToHtml(dto.getDescription()));
}
}
return ruleDoc;
diff --git a/server/sonar-server/src/main/java/org/sonar/server/rule/index/RuleIndexer.java b/server/sonar-server/src/main/java/org/sonar/server/rule/index/RuleIndexer.java
index 6f1c21d2d43..7c5db551866 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/rule/index/RuleIndexer.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/rule/index/RuleIndexer.java
@@ -22,16 +22,15 @@ package org.sonar.server.rule.index;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.ListMultimap;
import java.util.Collection;
-import java.util.Iterator;
import java.util.List;
import java.util.Set;
import org.elasticsearch.action.index.IndexRequest;
-import org.sonar.db.es.RuleExtensionId;
import org.sonar.api.rule.RuleKey;
import org.sonar.core.util.stream.MoreCollectors;
import org.sonar.db.DbClient;
import org.sonar.db.DbSession;
import org.sonar.db.es.EsQueueDto;
+import org.sonar.db.es.RuleExtensionId;
import org.sonar.db.organization.OrganizationDto;
import org.sonar.db.rule.RuleExtensionForIndexingDto;
import org.sonar.db.rule.RuleForIndexingDto;
@@ -39,15 +38,16 @@ import org.sonar.server.es.BulkIndexer;
import org.sonar.server.es.BulkIndexer.Size;
import org.sonar.server.es.EsClient;
import org.sonar.server.es.IndexType;
+import org.sonar.server.es.IndexingListener;
+import org.sonar.server.es.IndexingResult;
+import org.sonar.server.es.ResiliencyIndexingListener;
import org.sonar.server.es.ResilientIndexer;
-import org.sonar.server.es.ResilientIndexerResult;
import org.sonar.server.es.StartupIndexer;
import static com.google.common.base.Preconditions.checkArgument;
import static java.util.Collections.singletonList;
import static java.util.Objects.requireNonNull;
import static org.sonar.core.util.stream.MoreCollectors.toHashSet;
-import static org.sonar.server.rule.index.RuleIndexDefinition.INDEX;
import static org.sonar.server.rule.index.RuleIndexDefinition.INDEX_TYPE_RULE;
import static org.sonar.server.rule.index.RuleIndexDefinition.INDEX_TYPE_RULE_EXTENSION;
@@ -68,24 +68,27 @@ public class RuleIndexer implements StartupIndexer, ResilientIndexer {
@Override
public void indexOnStartup(Set<IndexType> uninitializedIndexTypes) {
- BulkIndexer bulk = new BulkIndexer(esClient, INDEX, Size.LARGE);
- bulk.start();
-
- // index all definitions and system extensions
- if (uninitializedIndexTypes.contains(INDEX_TYPE_RULE)) {
- try (RuleIterator rules = new RuleIteratorForSingleChunk(dbClient, null)) {
- doIndexRuleDefinitions(rules, bulk);
+ try (DbSession dbSession = dbClient.openSession(false)) {
+ BulkIndexer bulk = createBulkIndexer(Size.LARGE, IndexingListener.noop());
+ bulk.start();
+
+ // index all definitions and system extensions
+ if (uninitializedIndexTypes.contains(INDEX_TYPE_RULE)) {
+ dbClient.ruleDao().scrollIndexingRules(dbSession, dto -> {
+ bulk.add(newRuleDocIndexRequest(dto));
+ bulk.add(newRuleExtensionDocIndexRequest(dto));
+ });
}
- }
- // index all organization extensions
- if (uninitializedIndexTypes.contains(INDEX_TYPE_RULE_EXTENSION)) {
- try (RuleMetadataIterator metadatas = new RuleMetadataIterator(dbClient)) {
- doIndexRuleExtensions(metadatas, bulk);
+ // index all organization extensions
+ if (uninitializedIndexTypes.contains(INDEX_TYPE_RULE_EXTENSION)) {
+ dbClient.ruleDao().scrollIndexingRuleExtensions(dbSession, dto -> {
+ bulk.add(newRuleExtensionDocIndexRequest(dto));
+ });
}
- }
- bulk.stop();
+ bulk.stop();
+ }
}
public void commitAndIndex(DbSession dbSession, RuleKey ruleKey) {
@@ -123,12 +126,12 @@ public class RuleIndexer implements StartupIndexer, ResilientIndexer {
}
@Override
- public ResilientIndexerResult index(DbSession dbSession, Collection<EsQueueDto> items) {
+ public IndexingResult index(DbSession dbSession, Collection<EsQueueDto> items) {
if (items.isEmpty()) {
- return new ResilientIndexerResult();
+ return new IndexingResult();
}
- ResilientIndexerResult result = new ResilientIndexerResult();
+ IndexingResult result = new IndexingResult();
ListMultimap<EsQueueDto.Type, EsQueueDto> itemsByType = groupItemsByType(items);
@@ -138,9 +141,9 @@ public class RuleIndexer implements StartupIndexer, ResilientIndexer {
return result;
}
- private ResilientIndexerResult doIndexRules(DbSession dbSession, List<EsQueueDto> items) {
- BulkIndexer bulkIndexer = newBulkIndexerForRules(Size.REGULAR);
- bulkIndexer.start(dbSession, dbClient, items);
+ private IndexingResult doIndexRules(DbSession dbSession, List<EsQueueDto> items) {
+ BulkIndexer bulkIndexer = createBulkIndexer(Size.REGULAR, new ResiliencyIndexingListener(dbClient, dbSession, items));
+ bulkIndexer.start();
Set<RuleKey> rules = items
.stream()
@@ -151,7 +154,7 @@ public class RuleIndexer implements StartupIndexer, ResilientIndexer {
.map(i -> RuleKey.parse(i.getDocId()))
.collect(toHashSet(items.size()));
- dbClient.ruleDao().scrollRuleByRuleKeys(dbSession, rules,
+ dbClient.ruleDao().scrollIndexingRulesByKeys(dbSession, rules,
// only index requests, no deletion requests.
// Deactivated users are not deleted but updated.
r -> {
@@ -168,9 +171,9 @@ public class RuleIndexer implements StartupIndexer, ResilientIndexer {
return bulkIndexer.stop();
}
- private ResilientIndexerResult doIndexRuleExtensions(DbSession dbSession, List<EsQueueDto> items) {
- BulkIndexer bulkIndexer = newBulkIndexerForRules(Size.REGULAR);
- bulkIndexer.start(dbSession, dbClient, items);
+ private IndexingResult doIndexRuleExtensions(DbSession dbSession, List<EsQueueDto> items) {
+ BulkIndexer bulkIndexer = createBulkIndexer(Size.REGULAR, new ResiliencyIndexingListener(dbClient, dbSession, items));
+ bulkIndexer.start();
Set<RuleExtensionId> docIds = items
.stream()
@@ -181,15 +184,14 @@ public class RuleIndexer implements StartupIndexer, ResilientIndexer {
.map(RuleIndexer::explodeRuleExtensionDocId)
.collect(toHashSet(items.size()));
- dbClient.ruleDao().scrollRuleExtensionByRuleKeys(dbSession, docIds,
+ dbClient.ruleDao().scrollIndexingRuleExtensionsByIds(dbSession, docIds,
// only index requests, no deletion requests.
// Deactivated users are not deleted but updated.
r -> {
- docIds.remove(new RuleExtensionId(r.getOrganizationUuid(), r.getPluginName(), r.getPluginRuleKey()) );
+ docIds.remove(new RuleExtensionId(r.getOrganizationUuid(), r.getPluginName(), r.getPluginRuleKey()));
bulkIndexer.add(newRuleExtensionDocIndexRequest(r));
});
-
// the remaining items reference rows that don't exist in db. They must
// be deleted from index.
docIds.forEach(r -> bulkIndexer.addDeletion(RuleIndexDefinition.INDEX_TYPE_RULE_EXTENSION, r.getId()));
@@ -197,61 +199,36 @@ public class RuleIndexer implements StartupIndexer, ResilientIndexer {
return bulkIndexer.stop();
}
- private static void doIndexRuleDefinitions(Iterator<RuleDocWithSystemScope> rules, BulkIndexer bulk) {
- while (rules.hasNext()) {
- RuleDocWithSystemScope ruleWithExtension = rules.next();
- bulk.add(newIndexRequest(ruleWithExtension.getRuleDoc()));
- bulk.add(newIndexRequest(ruleWithExtension.getRuleExtensionDoc()));
- }
- }
-
- private static void doIndexRuleExtensions(Iterator<RuleExtensionDoc> metadatas, BulkIndexer bulk) {
- while (metadatas.hasNext()) {
- RuleExtensionDoc metadata = metadatas.next();
- bulk.add(newIndexRequest(metadata));
- }
- }
-
- private static IndexRequest newIndexRequest(RuleDoc rule) {
- return new IndexRequest(INDEX_TYPE_RULE.getIndex(), INDEX_TYPE_RULE.getType())
- .id(rule.getId())
- .routing(rule.getRouting())
- .source(rule.getFields());
- }
-
- private static IndexRequest newIndexRequest(RuleExtensionDoc ruleExtension) {
- return new IndexRequest(INDEX_TYPE_RULE_EXTENSION.getIndex(), INDEX_TYPE_RULE_EXTENSION.getType())
- .id(ruleExtension.getId())
- .routing(ruleExtension.getRouting())
- .source(ruleExtension.getFields())
- .parent(ruleExtension.getParent());
- }
-
private static IndexRequest newRuleDocIndexRequest(RuleForIndexingDto ruleForIndexingDto) {
- RuleDoc ruleDoc = RuleDoc.of(ruleForIndexingDto);
+ RuleDoc doc = RuleDoc.of(ruleForIndexingDto);
- return new IndexRequest(INDEX_TYPE_RULE.getIndex(), INDEX_TYPE_RULE.getType(), ruleDoc.key().toString())
- .source(ruleDoc.getFields());
+ return new IndexRequest(INDEX_TYPE_RULE.getIndex(), INDEX_TYPE_RULE.getType())
+ .id(doc.key().toString())
+ .routing(doc.getRouting())
+ .source(doc.getFields());
}
private static IndexRequest newRuleExtensionDocIndexRequest(RuleForIndexingDto ruleForIndexingDto) {
RuleExtensionDoc ruleExtensionDoc = RuleExtensionDoc.of(ruleForIndexingDto);
- return new IndexRequest(INDEX_TYPE_RULE_EXTENSION.getIndex(), INDEX_TYPE_RULE_EXTENSION.getType(), ruleExtensionDoc.getId())
- .source(ruleExtensionDoc.getFields())
- .parent(ruleExtensionDoc.getParent());
+ return new IndexRequest(INDEX_TYPE_RULE_EXTENSION.getIndex(), INDEX_TYPE_RULE_EXTENSION.getType())
+ .id(ruleExtensionDoc.getId())
+ .routing(ruleExtensionDoc.getRouting())
+ .parent(ruleExtensionDoc.getParent())
+ .source(ruleExtensionDoc.getFields());
}
private static IndexRequest newRuleExtensionDocIndexRequest(RuleExtensionForIndexingDto ruleExtensionForIndexingDto) {
- RuleExtensionDoc ruleExtensionDoc = RuleExtensionDoc.of(ruleExtensionForIndexingDto);
-
- return new IndexRequest(INDEX_TYPE_RULE_EXTENSION.getIndex(), INDEX_TYPE_RULE_EXTENSION.getType(), ruleExtensionDoc.getId())
- .source(ruleExtensionDoc.getFields())
- .parent(ruleExtensionDoc.getParent());
+ RuleExtensionDoc doc = RuleExtensionDoc.of(ruleExtensionForIndexingDto);
+ return new IndexRequest(INDEX_TYPE_RULE_EXTENSION.getIndex(), INDEX_TYPE_RULE_EXTENSION.getType())
+ .id(doc.getId())
+ .routing(doc.getRouting())
+ .parent(doc.getParent())
+ .source(doc.getFields());
}
- private BulkIndexer newBulkIndexerForRules(Size bulkSize) {
- return new BulkIndexer(esClient, INDEX_TYPE_RULE.getIndex(), bulkSize);
+ private BulkIndexer createBulkIndexer(Size bulkSize, IndexingListener listener) {
+ return new BulkIndexer(esClient, INDEX_TYPE_RULE.getIndex(), bulkSize, listener);
}
private static ListMultimap<EsQueueDto.Type, EsQueueDto> groupItemsByType(Collection<EsQueueDto> items) {
diff --git a/server/sonar-server/src/main/java/org/sonar/server/rule/index/RuleIteratorForMultipleChunks.java b/server/sonar-server/src/main/java/org/sonar/server/rule/index/RuleIteratorForMultipleChunks.java
deleted file mode 100644
index 8a86b4d8724..00000000000
--- a/server/sonar-server/src/main/java/org/sonar/server/rule/index/RuleIteratorForMultipleChunks.java
+++ /dev/null
@@ -1,80 +0,0 @@
-/*
- * 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 org.sonar.server.rule.index;
-
-import java.util.Collection;
-import java.util.Iterator;
-import java.util.List;
-import java.util.NoSuchElementException;
-import org.sonar.api.rule.RuleKey;
-import org.sonar.db.DatabaseUtils;
-import org.sonar.db.DbClient;
-
-import static java.util.Optional.ofNullable;
-
-public class RuleIteratorForMultipleChunks implements RuleIterator {
-
- private final DbClient dbClient;
- private final Iterator<List<RuleKey>> iteratorOverChunks;
- private RuleIteratorForSingleChunk currentChunk;
-
- public RuleIteratorForMultipleChunks(DbClient dbClient, Collection<RuleKey> keys) {
- this.dbClient = dbClient;
- iteratorOverChunks = DatabaseUtils.toUniqueAndSortedPartitions(keys).iterator();
- }
-
- @Override
- public boolean hasNext() {
- for (;;) {
- if (currentChunk != null && currentChunk.hasNext()) {
- return true;
- }
- if (iteratorOverChunks.hasNext()) {
- currentChunk = nextChunk();
- } else {
- return false;
- }
- }
- }
-
- @Override
- public RuleDocWithSystemScope next() {
- for (;;) {
- if (currentChunk != null && currentChunk.hasNext()) {
- return currentChunk.next();
- }
- if (iteratorOverChunks.hasNext()) {
- currentChunk = nextChunk();
- } else {
- throw new NoSuchElementException();
- }
- }
- }
-
- private RuleIteratorForSingleChunk nextChunk() {
- List<RuleKey> nextInput = iteratorOverChunks.next();
- return new RuleIteratorForSingleChunk(dbClient, nextInput);
- }
-
- @Override
- public void close() {
- ofNullable(currentChunk).ifPresent(RuleIterator::close);
- }
-}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/rule/index/RuleIteratorForSingleChunk.java b/server/sonar-server/src/main/java/org/sonar/server/rule/index/RuleIteratorForSingleChunk.java
deleted file mode 100644
index 30e9952014f..00000000000
--- a/server/sonar-server/src/main/java/org/sonar/server/rule/index/RuleIteratorForSingleChunk.java
+++ /dev/null
@@ -1,211 +0,0 @@
-/*
- * 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 org.sonar.server.rule.index;
-
-import com.google.common.base.Splitter;
-import com.google.common.collect.ImmutableSet;
-import java.sql.PreparedStatement;
-import java.sql.ResultSet;
-import java.sql.SQLException;
-import java.util.List;
-import java.util.Set;
-import java.util.concurrent.atomic.AtomicInteger;
-import javax.annotation.Nullable;
-import org.apache.commons.lang.StringUtils;
-import org.sonar.api.rule.RuleKey;
-import org.sonar.api.rules.RuleType;
-import org.sonar.db.DatabaseUtils;
-import org.sonar.db.DbClient;
-import org.sonar.db.DbSession;
-import org.sonar.db.ResultSetIterator;
-import org.sonar.db.rule.RuleDto;
-import org.sonar.db.rule.SeverityUtil;
-import org.sonar.markdown.Markdown;
-
-import static com.google.common.base.Preconditions.checkArgument;
-import static java.util.stream.Collectors.joining;
-
-/**
- * Scrolls over table RULES and reads documents to populate the rules index
- */
-public class RuleIteratorForSingleChunk implements RuleIterator {
-
- private static final String[] FIELDS = {
- // column 1
- "r.plugin_name",
- "r.plugin_rule_key",
- "r.name",
- "r.description",
- "r.description_format",
- "r.priority",
- "r.status",
- "r.is_template",
- "r.system_tags",
- "t.plugin_rule_key",
-
- // column 11
- "t.plugin_name",
- "r.plugin_config_key",
- "r.language",
- "r.rule_type",
- "r.created_at",
- "r.updated_at",
- };
-
- private static final String SQL_ALL = "SELECT " + StringUtils.join(FIELDS, ",") + " FROM rules r " +
- "LEFT OUTER JOIN rules t ON t.id=r.template_id";
- private static final Splitter TAGS_SPLITTER = Splitter.on(',').trimResults().omitEmptyStrings();
-
- private final DbSession session;
-
- private final List<RuleKey> ruleKeys;
-
- private final PreparedStatement stmt;
- private final ResultSetIterator<RuleDocWithSystemScope> iterator;
-
- RuleIteratorForSingleChunk(DbClient dbClient, @Nullable List<RuleKey> ruleKeys) {
- checkArgument(ruleKeys == null || ruleKeys.size() <= DatabaseUtils.PARTITION_SIZE_FOR_ORACLE,
- "Cannot search for more than " + DatabaseUtils.PARTITION_SIZE_FOR_ORACLE + " rule keys at once. Please provide the keys in smaller chunks.");
- this.ruleKeys = ruleKeys;
- this.session = dbClient.openSession(false);
-
- try {
- String sql = createSql();
- stmt = dbClient.getMyBatis().newScrollingSelectStatement(session, sql);
- iterator = createIterator();
- } catch (Exception e) {
- session.close();
- throw new IllegalStateException("Fail to prepare SQL request to select all rules", e);
- }
- }
-
- private RuleIteratorInternal createIterator() {
- try {
- setParameters(stmt);
- return new RuleIteratorInternal(stmt);
- } catch (SQLException e) {
- DatabaseUtils.closeQuietly(stmt);
- throw new IllegalStateException("Fail to prepare SQL request to select all rules", e);
- }
- }
-
- @Override
- public boolean hasNext() {
- return iterator.hasNext();
- }
-
- @Override
- public RuleDocWithSystemScope next() {
- return iterator.next();
- }
-
- private String createSql() {
- StringBuilder sql = new StringBuilder(SQL_ALL);
- if (ruleKeys != null && !ruleKeys.isEmpty()) {
- sql.append(" WHERE ");
- sql.append(ruleKeys.stream()
- .map(x -> "(r.plugin_name=? AND r.plugin_rule_key=?)")
- .collect(joining(" OR "))
- );
- }
- return sql.toString();
- }
-
- private void setParameters(PreparedStatement stmt) throws SQLException {
- AtomicInteger index = new AtomicInteger(1);
- if (ruleKeys != null && !ruleKeys.isEmpty()) {
- for (RuleKey ruleKey : ruleKeys) {
- stmt.setString(index.getAndIncrement(), ruleKey.repository());
- stmt.setString(index.getAndIncrement(), ruleKey.rule());
- }
- }
- }
-
- @Override
- public void close() {
- try {
- iterator.close();
- } finally {
- DatabaseUtils.closeQuietly(stmt);
- session.close();
- }
- }
-
- private static final class RuleIteratorInternal extends ResultSetIterator<RuleDocWithSystemScope> {
-
- public RuleIteratorInternal(PreparedStatement stmt) throws SQLException {
- super(stmt);
- }
-
- @Override
- protected RuleDocWithSystemScope read(ResultSet rs) throws SQLException {
- RuleDoc doc = new RuleDoc();
- RuleExtensionDoc extensionDoc = new RuleExtensionDoc().setScope(RuleExtensionScope.system());
-
- String repositoryKey = rs.getString(1);
- String ruleKey = rs.getString(2);
- RuleKey key = RuleKey.of(repositoryKey, ruleKey);
- extensionDoc.setRuleKey(key);
-
- // all the fields must be present, even if value is null
- doc.setKey(key.toString());
- doc.setRuleKey(ruleKey);
- doc.setRepository(repositoryKey);
- doc.setName(rs.getString(3));
-
- String description = rs.getString(4);
- String descriptionFormat = rs.getString(5);
- if (descriptionFormat != null && description != null) {
- String htmlDescription;
- if (RuleDto.Format.HTML == RuleDto.Format.valueOf(descriptionFormat)) {
- htmlDescription = description;
- } else {
- htmlDescription = Markdown.convertToHtml(description);
- }
- doc.setHtmlDescription(htmlDescription);
- }
-
- doc.setSeverity(SeverityUtil.getSeverityFromOrdinal(rs.getInt(6)));
- doc.setStatus(rs.getString(7));
- doc.setIsTemplate(rs.getBoolean(8));
- extensionDoc.setTags(stringTagsToSet(rs.getString(9)));
-
- String templateRuleKey = rs.getString(10);
- String templateRepoKey = rs.getString(11);
- if (templateRepoKey != null && templateRuleKey != null) {
- doc.setTemplateKey(RuleKey.of(templateRepoKey, templateRuleKey).toString());
- } else {
- doc.setTemplateKey(null);
- }
-
- doc.setInternalKey(rs.getString(12));
- doc.setLanguage(rs.getString(13));
- doc.setType(RuleType.valueOf(rs.getInt(14)));
- doc.setCreatedAt(rs.getLong(15));
- doc.setUpdatedAt(rs.getLong(16));
-
- return new RuleDocWithSystemScope(doc, extensionDoc);
- }
-
- private static Set<String> stringTagsToSet(@Nullable String tags) {
- return ImmutableSet.copyOf(TAGS_SPLITTER.split(tags == null ? "" : tags));
- }
- }
-}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/rule/index/RuleMetadataIterator.java b/server/sonar-server/src/main/java/org/sonar/server/rule/index/RuleMetadataIterator.java
deleted file mode 100644
index 5574c2a0a51..00000000000
--- a/server/sonar-server/src/main/java/org/sonar/server/rule/index/RuleMetadataIterator.java
+++ /dev/null
@@ -1,126 +0,0 @@
-/*
- * 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 org.sonar.server.rule.index;
-
-import com.google.common.base.Splitter;
-import com.google.common.collect.ImmutableSet;
-import java.sql.PreparedStatement;
-import java.sql.ResultSet;
-import java.sql.SQLException;
-import java.util.Iterator;
-import java.util.Set;
-import javax.annotation.Nullable;
-import org.apache.commons.lang.StringUtils;
-import org.sonar.api.rule.RuleKey;
-import org.sonar.db.DatabaseUtils;
-import org.sonar.db.DbClient;
-import org.sonar.db.DbSession;
-import org.sonar.db.ResultSetIterator;
-
-/**
- * Scrolls over table RULES_METADATA and reads documents to populate the rule extension index type
- */
-public class RuleMetadataIterator implements Iterator<RuleExtensionDoc>, AutoCloseable {
-
- private static final String[] FIELDS = {
- "r.plugin_name",
- "r.plugin_rule_key",
- "rm.organization_uuid",
- "rm.tags"
- };
-
- private static final String SQL_ALL = "SELECT " + StringUtils.join(FIELDS, ",") + " FROM rules r " +
- "INNER JOIN rules_metadata rm ON rm.rule_id = r.id " +
- "WHERE rm.tags is not null AND rm.tags != ''";
- private static final Splitter TAGS_SPLITTER = Splitter.on(',').trimResults().omitEmptyStrings();
-
- private final DbSession session;
-
- private final PreparedStatement stmt;
- private final ResultSetIterator<RuleExtensionDoc> iterator;
-
- RuleMetadataIterator(DbClient dbClient) {
- this.session = dbClient.openSession(false);
-
- try {
- String sql = SQL_ALL;
- stmt = dbClient.getMyBatis().newScrollingSelectStatement(session, sql);
- iterator = createIterator();
- } catch (Exception e) {
- session.close();
- throw new IllegalStateException("Fail to prepare SQL request to select all rules", e);
- }
- }
-
- private RuleMetadataIteratorInternal createIterator() {
- try {
- return new RuleMetadataIteratorInternal(stmt);
- } catch (SQLException e) {
- DatabaseUtils.closeQuietly(stmt);
- throw new IllegalStateException("Fail to prepare SQL request to select all rules", e);
- }
- }
-
- @Override
- public boolean hasNext() {
- return iterator.hasNext();
- }
-
- @Override
- public RuleExtensionDoc next() {
- return iterator.next();
- }
-
- @Override
- public void close() {
- try {
- iterator.close();
- } finally {
- DatabaseUtils.closeQuietly(stmt);
- session.close();
- }
- }
-
- private static final class RuleMetadataIteratorInternal extends ResultSetIterator<RuleExtensionDoc> {
-
- public RuleMetadataIteratorInternal(PreparedStatement stmt) throws SQLException {
- super(stmt);
- }
-
- @Override
- protected RuleExtensionDoc read(ResultSet rs) throws SQLException {
- RuleExtensionDoc doc = new RuleExtensionDoc();
-
- String repositoryKey = rs.getString(1);
- String ruleKey = rs.getString(2);
- RuleKey key = RuleKey.of(repositoryKey, ruleKey);
- doc.setRuleKey(key);
- doc.setScope(RuleExtensionScope.organization(rs.getString(3)));
- doc.setTags(stringTagsToSet(rs.getString(4)));
-
- return doc;
- }
-
- private static Set<String> stringTagsToSet(@Nullable String tags) {
- return ImmutableSet.copyOf(TAGS_SPLITTER.split(tags == null ? "" : tags));
- }
- }
-}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/user/index/UserIndexer.java b/server/sonar-server/src/main/java/org/sonar/server/user/index/UserIndexer.java
index b6ae239b4af..239a9bbcb60 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/user/index/UserIndexer.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/user/index/UserIndexer.java
@@ -37,8 +37,10 @@ import org.sonar.server.es.BulkIndexer;
import org.sonar.server.es.BulkIndexer.Size;
import org.sonar.server.es.EsClient;
import org.sonar.server.es.IndexType;
+import org.sonar.server.es.IndexingListener;
+import org.sonar.server.es.ResiliencyIndexingListener;
import org.sonar.server.es.ResilientIndexer;
-import org.sonar.server.es.ResilientIndexerResult;
+import org.sonar.server.es.IndexingResult;
import org.sonar.server.es.StartupIndexer;
import static java.util.Collections.singletonList;
@@ -67,7 +69,7 @@ public class UserIndexer implements StartupIndexer, ResilientIndexer {
ListMultimap<String, String> organizationUuidsByLogin = ArrayListMultimap.create();
dbClient.organizationMemberDao().selectAllForUserIndexing(dbSession, organizationUuidsByLogin::put);
- BulkIndexer bulkIndexer = newBulkIndexer(Size.LARGE);
+ BulkIndexer bulkIndexer = newBulkIndexer(Size.LARGE, IndexingListener.noop());
bulkIndexer.start();
dbClient.userDao().scrollAll(dbSession,
// only index requests, no deletion requests.
@@ -107,9 +109,9 @@ public class UserIndexer implements StartupIndexer, ResilientIndexer {
* @return the number of items that have been successfully indexed
*/
@Override
- public ResilientIndexerResult index(DbSession dbSession, Collection<EsQueueDto> items) {
+ public IndexingResult index(DbSession dbSession, Collection<EsQueueDto> items) {
if (items.isEmpty()) {
- return new ResilientIndexerResult();
+ return new IndexingResult();
}
Set<String> logins = items
.stream()
@@ -123,8 +125,8 @@ public class UserIndexer implements StartupIndexer, ResilientIndexer {
ListMultimap<String, String> organizationUuidsByLogin = ArrayListMultimap.create();
dbClient.organizationMemberDao().selectForUserIndexing(dbSession, logins, organizationUuidsByLogin::put);
- BulkIndexer bulkIndexer = newBulkIndexer(Size.REGULAR);
- bulkIndexer.start(dbSession, dbClient, items);
+ BulkIndexer bulkIndexer = newBulkIndexer(Size.REGULAR, new ResiliencyIndexingListener(dbClient, dbSession, items));
+ bulkIndexer.start();
dbClient.userDao().scrollByLogins(dbSession, logins,
// only index requests, no deletion requests.
// Deactivated users are not deleted but updated.
@@ -139,8 +141,8 @@ public class UserIndexer implements StartupIndexer, ResilientIndexer {
return bulkIndexer.stop();
}
- private BulkIndexer newBulkIndexer(Size bulkSize) {
- return new BulkIndexer(esClient, UserIndexDefinition.INDEX_TYPE_USER.getIndex(), bulkSize);
+ private BulkIndexer newBulkIndexer(Size bulkSize, IndexingListener listener) {
+ return new BulkIndexer(esClient, UserIndexDefinition.INDEX_TYPE_USER.getIndex(), bulkSize, listener);
}
private static IndexRequest newIndexRequest(UserDto user, ListMultimap<String, String> organizationUuidsByLogins) {
@@ -155,6 +157,7 @@ public class UserIndexer implements StartupIndexer, ResilientIndexer {
return new IndexRequest(UserIndexDefinition.INDEX_TYPE_USER.getIndex(), UserIndexDefinition.INDEX_TYPE_USER.getType())
.id(doc.getId())
+ .routing(doc.getRouting())
.source(doc.getFields());
}
}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/user/index/UserResultSetIterator.java b/server/sonar-server/src/main/java/org/sonar/server/user/index/UserResultSetIterator.java
deleted file mode 100644
index 22b07b634c7..00000000000
--- a/server/sonar-server/src/main/java/org/sonar/server/user/index/UserResultSetIterator.java
+++ /dev/null
@@ -1,121 +0,0 @@
-/*
- * 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 org.sonar.server.user.index;
-
-import com.google.common.collect.ArrayListMultimap;
-import com.google.common.collect.ListMultimap;
-import com.google.common.collect.Maps;
-import java.sql.PreparedStatement;
-import java.sql.ResultSet;
-import java.sql.SQLException;
-import java.util.Collection;
-import java.util.List;
-import javax.annotation.Nullable;
-import org.apache.commons.lang.StringUtils;
-import org.sonar.db.DbClient;
-import org.sonar.db.DbSession;
-import org.sonar.db.ResultSetIterator;
-import org.sonar.db.es.EsQueueDto;
-import org.sonar.db.user.UserDto;
-
-import static org.apache.commons.lang.StringUtils.repeat;
-import static org.sonar.core.util.stream.MoreCollectors.toArrayList;
-
-/**
- * Scrolls over table USERS and reads documents to populate the user index
- */
-class UserResultSetIterator extends ResultSetIterator<UserDoc> {
-
- private static final String[] FIELDS = {
- // column 1
- "u.login",
- "u.name",
- "u.email",
- "u.active",
- "u.scm_accounts",
- "u.created_at",
- "u.updated_at",
- };
-
- private static final String SQL_ALL = "select " + StringUtils.join(FIELDS, ",") + " from users u ";
-
- private final ListMultimap<String, String> organizationUuidsByLogins;
-
- private UserResultSetIterator(PreparedStatement stmt, ListMultimap<String, String> organizationUuidsByLogins) throws SQLException {
- super(stmt);
- this.organizationUuidsByLogins = organizationUuidsByLogins;
- }
-
- static UserResultSetIterator create(DbClient dbClient, DbSession session, @Nullable Collection<EsQueueDto> esQueueDtos) {
- try {
- String sql = SQL_ALL;
- List<String> logins = null;
- if (esQueueDtos != null) {
- logins = esQueueDtos.stream()
- .filter(i -> i.getDocType() == EsQueueDto.Type.USER)
- .map(EsQueueDto::getDocId).collect(toArrayList());
- sql += "where (" + repeat("u.login=?", " or ", logins.size()) + ")";
- }
-
- PreparedStatement stmt = dbClient.getMyBatis().newScrollingSelectStatement(session, sql);
- setParameters(stmt, logins);
-
- ListMultimap<String, String> organizationUuidsByLogin = ArrayListMultimap.create();
- if (esQueueDtos == null) {
- dbClient.organizationMemberDao().selectAllForUserIndexing(session, organizationUuidsByLogin::put);
- } else {
-
- dbClient.organizationMemberDao().selectForUserIndexing(session, logins, organizationUuidsByLogin::put);
- }
-
- return new UserResultSetIterator(stmt, organizationUuidsByLogin);
- } catch (SQLException e) {
- throw new IllegalStateException("Fail to prepare SQL request to select all users", e);
- }
- }
-
- private static void setParameters(PreparedStatement stmt, @Nullable Collection<String> logins) throws SQLException {
- if (logins == null) {
- return;
- }
-
- int paramIndex = 1;
- for (String login : logins) {
- stmt.setString(paramIndex, login);
- paramIndex++;
- }
- }
-
- @Override
- protected UserDoc read(ResultSet rs) throws SQLException {
- UserDoc doc = new UserDoc(Maps.newHashMapWithExpectedSize(6));
-
- String login = rs.getString(1);
-
- // all the keys must be present, even if value is null
- doc.setLogin(login);
- doc.setName(rs.getString(2));
- doc.setEmail(rs.getString(3));
- doc.setActive(rs.getBoolean(4));
- doc.setScmAccounts(UserDto.decodeScmAccounts(rs.getString(5)));
- doc.setOrganizationUuids(organizationUuidsByLogins.get(login));
- return doc;
- }
-}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/es/BulkIndexerTest.java b/server/sonar-server/src/test/java/org/sonar/server/es/BulkIndexerTest.java
index dd8abe6998a..8848ae6d5e3 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/es/BulkIndexerTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/es/BulkIndexerTest.java
@@ -82,10 +82,13 @@ public class BulkIndexerTest {
for (int i = 0; i < 10; i++) {
indexer.add(newIndexRequest(i));
}
- indexer.stop();
+ IndexingResult result = indexer.stop();
+ assertThat(result.isSuccess()).isTrue();
+ assertThat(result.getSuccess()).isEqualTo(10);
+ assertThat(result.getFailures()).isEqualTo(0);
+ assertThat(result.getTotal()).isEqualTo(10);
assertThat(count()).isEqualTo(10);
-
// replicas are re-enabled
assertThat(replicas()).isEqualTo(1);
}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/es/EsTester.java b/server/sonar-server/src/test/java/org/sonar/server/es/EsTester.java
index 2eb1269121a..3ae2027cad8 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/es/EsTester.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/es/EsTester.java
@@ -65,7 +65,7 @@ public class EsTester extends ExternalResource {
@Override
protected void before() throws Throwable {
- truncateIndices();
+ deleteIndices();
if (!indexDefinitions.isEmpty()) {
container = new ComponentContainer();
@@ -89,10 +89,14 @@ public class EsTester extends ExternalResource {
}
}
- private void truncateIndices() {
+ private void deleteIndices() {
client.nativeClient().admin().indices().prepareDelete("_all").get();
}
+ public void deleteIndex(String indexName) {
+ client.nativeClient().admin().indices().prepareDelete(indexName).get();
+ }
+
public void putDocuments(String index, String type, BaseDoc... docs) {
putDocuments(new IndexType(index, type), docs);
}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/es/IndexingResultTest.java b/server/sonar-server/src/test/java/org/sonar/server/es/IndexingResultTest.java
new file mode 100644
index 00000000000..2eb2c4ded86
--- /dev/null
+++ b/server/sonar-server/src/test/java/org/sonar/server/es/IndexingResultTest.java
@@ -0,0 +1,78 @@
+/*
+ * 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 org.sonar.server.es;
+
+import org.assertj.core.data.Offset;
+import org.junit.Test;
+
+import static org.assertj.core.api.Assertions.assertThat;
+
+public class IndexingResultTest {
+
+ private final IndexingResult underTest = new IndexingResult();
+
+ @Test
+ public void test_success() {
+ underTest.incrementRequests();
+ underTest.incrementRequests();
+ underTest.incrementSuccess();
+ underTest.incrementSuccess();
+
+ assertThat(underTest.getFailures()).isEqualTo(0);
+ assertThat(underTest.getSuccess()).isEqualTo(2);
+ assertThat(underTest.getTotal()).isEqualTo(2);
+ assertThat(underTest.getFailureRatio()).isEqualTo(0.0, Offset.offset(0.000001d));
+ assertThat(underTest.isSuccess()).isTrue();
+ }
+
+ @Test
+ public void test_failure() {
+ underTest.incrementRequests();
+ underTest.incrementRequests();
+
+ assertThat(underTest.getFailures()).isEqualTo(2);
+ assertThat(underTest.getSuccess()).isEqualTo(0);
+ assertThat(underTest.getTotal()).isEqualTo(2);
+ assertThat(underTest.getFailureRatio()).isEqualTo(1.0, Offset.offset(0.000001d));
+ assertThat(underTest.isSuccess()).isFalse();
+ }
+
+ @Test
+ public void test_partial_failure() {
+ underTest.incrementRequests();
+ underTest.incrementRequests();
+ underTest.incrementSuccess();
+
+ assertThat(underTest.getFailures()).isEqualTo(1);
+ assertThat(underTest.getSuccess()).isEqualTo(1);
+ assertThat(underTest.getTotal()).isEqualTo(2);
+ assertThat(underTest.getFailureRatio()).isEqualTo(0.5, Offset.offset(0.000001d));
+ assertThat(underTest.isSuccess()).isFalse();
+ }
+
+ @Test
+ public void correctness_even_with_no_data() {
+ assertThat(underTest.getFailures()).isEqualTo(0);
+ assertThat(underTest.getSuccess()).isEqualTo(0);
+ assertThat(underTest.getTotal()).isEqualTo(0);
+ assertThat(underTest.getFailureRatio()).isEqualTo(1);
+ assertThat(underTest.isSuccess()).isTrue();
+ }
+}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/es/RecoveryIndexerTest.java b/server/sonar-server/src/test/java/org/sonar/server/es/RecoveryIndexerTest.java
index f83015e5aac..0c8fa0f5f10 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/es/RecoveryIndexerTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/es/RecoveryIndexerTest.java
@@ -43,6 +43,7 @@ import org.sonar.db.DbTester;
import org.sonar.db.es.EsQueueDto;
import org.sonar.db.rule.RuleDto;
import org.sonar.db.user.UserDto;
+import org.sonar.server.qualityprofile.index.ActiveRuleIndexer;
import org.sonar.server.rule.index.RuleIndexDefinition;
import org.sonar.server.rule.index.RuleIndexer;
import org.sonar.server.user.index.UserIndexDefinition;
@@ -73,9 +74,9 @@ public class RecoveryIndexerTest {
@Rule
public TestRule safeguard = new DisableOnDebug(new Timeout(60, TimeUnit.SECONDS));
-
- private RuleIndexer mockedRuleIndexer = mock(RuleIndexer.class);
private UserIndexer mockedUserIndexer = mock(UserIndexer.class);
+ private RuleIndexer mockedRuleIndexer = mock(RuleIndexer.class);
+ private ActiveRuleIndexer mockedActiveRuleIndexer = mock(ActiveRuleIndexer.class);
private RecoveryIndexer underTest;
@After
@@ -103,7 +104,7 @@ public class RecoveryIndexerTest {
Settings settings = new MapSettings()
.setProperty("sonar.search.recovery.initialDelayInMs", "0")
.setProperty("sonar.search.recovery.delayInMs", "1");
- underTest = spy(new RecoveryIndexer(system2, settings, db.getDbClient(), mockedUserIndexer, mockedRuleIndexer));
+ underTest = spy(new RecoveryIndexer(system2, settings, db.getDbClient(), mockedUserIndexer, mockedRuleIndexer, mockedActiveRuleIndexer));
AtomicInteger calls = new AtomicInteger(0);
doAnswer(invocation -> {
calls.incrementAndGet();
@@ -234,7 +235,7 @@ public class RecoveryIndexerTest {
}
@Test
- public void stop_run_if_too_many_failures() throws Exception {
+ public void stop_run_if_too_many_failures() {
IntStream.range(0, 10).forEach(i -> createUnindexedUser());
advanceInTime();
@@ -254,7 +255,7 @@ public class RecoveryIndexerTest {
}
@Test
- public void do_not_stop_run_if_success_rate_is_greater_than_ratio() throws Exception {
+ public void do_not_stop_run_if_success_rate_is_greater_than_ratio() {
IntStream.range(0, 10).forEach(i -> createUnindexedUser());
advanceInTime();
@@ -294,7 +295,7 @@ public class RecoveryIndexerTest {
}
@Override
- public ResilientIndexerResult index(DbSession dbSession, Collection<EsQueueDto> items) {
+ public IndexingResult index(DbSession dbSession, Collection<EsQueueDto> items) {
called.addAll(items);
return super.index(dbSession, items);
}
@@ -308,7 +309,7 @@ public class RecoveryIndexerTest {
}
@Override
- public ResilientIndexerResult index(DbSession dbSession, Collection<EsQueueDto> items) {
+ public IndexingResult index(DbSession dbSession, Collection<EsQueueDto> items) {
called.addAll(items);
return super.index(dbSession, items);
}
@@ -322,7 +323,7 @@ public class RecoveryIndexerTest {
}
@Override
- public ResilientIndexerResult index(DbSession dbSession, Collection<EsQueueDto> items) {
+ public IndexingResult index(DbSession dbSession, Collection<EsQueueDto> items) {
called.addAll(items);
throw new RuntimeException("boom");
}
@@ -337,7 +338,7 @@ public class RecoveryIndexerTest {
}
@Override
- public ResilientIndexerResult index(DbSession dbSession, Collection<EsQueueDto> items) {
+ public IndexingResult index(DbSession dbSession, Collection<EsQueueDto> items) {
try {
if (counter.getCount() == 2) {
throw new RuntimeException("boom");
@@ -358,15 +359,15 @@ public class RecoveryIndexerTest {
}
@Override
- public ResilientIndexerResult index(DbSession dbSession, Collection<EsQueueDto> items) {
- ResilientIndexerResult result = new ResilientIndexerResult();
+ public IndexingResult index(DbSession dbSession, Collection<EsQueueDto> items) {
List<EsQueueDto> filteredItems = items.stream().filter(
i -> !i.getUuid().equals(failing.getUuid())).collect(toArrayList());
+ IndexingResult result = super.index(dbSession, filteredItems);
if (items.contains(failing)) {
- result.increaseFailure();
+ result.incrementRequests();
}
- return result.add(super.index(dbSession, filteredItems));
+ return result;
}
}
@@ -381,19 +382,17 @@ public class RecoveryIndexerTest {
}
@Override
- public ResilientIndexerResult index(DbSession dbSession, Collection<EsQueueDto> items) {
- System.out.println("called with " + items.size());
+ public IndexingResult index(DbSession dbSession, Collection<EsQueueDto> items) {
called.addAll(items);
int success = successfulReturns.next();
- ResilientIndexerResult result = new ResilientIndexerResult();
+ IndexingResult result = new IndexingResult();
items.stream().limit(success).forEach(i -> {
- System.out.println(" + success");
db.getDbClient().esQueueDao().delete(dbSession, i);
- result.increaseSuccess();
+ result.incrementSuccess();
indexed.add(i);
});
- rangeClosed(1, items.size() - success).forEach(i -> result.increaseFailure());
+ rangeClosed(1, items.size()).forEach(i -> result.incrementRequests());
dbSession.commit();
return result;
}
@@ -434,7 +433,7 @@ public class RecoveryIndexerTest {
}
private RecoveryIndexer newRecoveryIndexer(UserIndexer userIndexer, RuleIndexer ruleIndexer, Settings settings) {
- return new RecoveryIndexer(system2, settings, db.getDbClient(), userIndexer, ruleIndexer);
+ return new RecoveryIndexer(system2, settings, db.getDbClient(), userIndexer, ruleIndexer, mockedActiveRuleIndexer);
}
private EsQueueDto createUnindexedUser() {
diff --git a/server/sonar-server/src/test/java/org/sonar/server/es/ResilientIndexerResultTest.java b/server/sonar-server/src/test/java/org/sonar/server/es/ResilientIndexerResultTest.java
deleted file mode 100644
index 6a3a03c3ba1..00000000000
--- a/server/sonar-server/src/test/java/org/sonar/server/es/ResilientIndexerResultTest.java
+++ /dev/null
@@ -1,62 +0,0 @@
-package org.sonar.server.es;/*
- * 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.
- */
-
-import java.util.stream.IntStream;
-import org.apache.commons.lang.math.RandomUtils;
-import org.assertj.core.data.Offset;
-import org.junit.Before;
-import org.junit.Test;
-
-import static org.assertj.core.api.Assertions.assertThat;
-
-public class ResilientIndexerResultTest {
-
- private final ResilientIndexerResult underTest = new ResilientIndexerResult();
-
- @Before
- public void clear() {
- underTest.clear();
- }
-
- @Test
- public void ensure_correctness() {
- int success = 1 + RandomUtils.nextInt(100);
- int failures = RandomUtils.nextInt(100);
- IntStream.rangeClosed(1, success).forEach(i -> underTest.increaseSuccess());
- IntStream.rangeClosed(1, failures).forEach(i -> underTest.increaseFailure());
-
- assertThat(underTest.getFailures()).isEqualTo(failures);
- assertThat(underTest.getSuccess()).isEqualTo(success);
- assertThat(underTest.getTotal()).isEqualTo(success + failures);
- assertThat(underTest.getFailureRatio() + underTest.getSuccessRatio()).isEqualTo(1);
- assertThat(underTest.getFailureRatio()).isEqualTo(1.0d * failures / (success + failures), Offset.offset(0.000001d));
- assertThat(underTest.getSuccessRatio()).isEqualTo(1.0d * success / (success + failures), Offset.offset(0.000001d));
- }
-
- @Test
- public void correctness_even_with_no_data() {
- assertThat(underTest.getFailures()).isEqualTo(0);
- assertThat(underTest.getSuccess()).isEqualTo(0);
- assertThat(underTest.getTotal()).isEqualTo(0);
- assertThat(underTest.getFailureRatio() + underTest.getSuccessRatio()).isEqualTo(1);
- assertThat(underTest.getFailureRatio()).isEqualTo(1);
- assertThat(underTest.getSuccessRatio()).isEqualTo(0);
- }
-}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/issue/index/IssueIndexTest.java b/server/sonar-server/src/test/java/org/sonar/server/issue/index/IssueIndexTest.java
index 45ed66c7cab..16d46b6f3a7 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/issue/index/IssueIndexTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/issue/index/IssueIndexTest.java
@@ -86,7 +86,7 @@ public class IssueIndexTest {
public EsTester esTester = new EsTester(
new IssueIndexDefinition(settings.asConfig()),
new ViewIndexDefinition(settings.asConfig()),
- new RuleIndexDefinition(settings.asConfig());
+ new RuleIndexDefinition(settings.asConfig()));
@Rule
public DbTester dbTester = DbTester.create(system2);
@Rule
diff --git a/server/sonar-server/src/test/java/org/sonar/server/organization/ws/EnableSupportActionTest.java b/server/sonar-server/src/test/java/org/sonar/server/organization/ws/EnableSupportActionTest.java
index 8e7dd830227..38e336d4aff 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/organization/ws/EnableSupportActionTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/organization/ws/EnableSupportActionTest.java
@@ -27,7 +27,7 @@ import org.junit.Rule;
import org.junit.Test;
import org.junit.rules.ExpectedException;
import org.mockito.ArgumentCaptor;
-import org.sonar.api.config.MapSettings;
+import org.sonar.api.config.internal.MapSettings;
import org.sonar.api.rule.RuleKey;
import org.sonar.api.rule.RuleStatus;
import org.sonar.api.server.ws.WebService;
@@ -71,8 +71,7 @@ public class EnableSupportActionTest {
@Rule
public DbTester dbTester = DbTester.create();
@Rule
- public EsTester esTester = new EsTester(new RuleIndexDefinition(new MapSettings()));
-
+ public EsTester esTester = new EsTester(new RuleIndexDefinition(new MapSettings().asConfig()));
private DefaultOrganizationProvider defaultOrganizationProvider = TestDefaultOrganizationProvider.from(dbTester);
private OrganizationFlags organizationFlags = new OrganizationFlagsImpl(dbTester.getDbClient());
@@ -183,8 +182,7 @@ public class EnableSupportActionTest {
.containsExactlyInAnyOrder(
tuple(normal.getKey(), RuleStatus.READY),
tuple(template.getKey(), RuleStatus.READY),
- tuple(custom.getKey(), RuleStatus.READY)
- );
+ tuple(custom.getKey(), RuleStatus.READY));
call();
@@ -193,11 +191,10 @@ public class EnableSupportActionTest {
.containsExactlyInAnyOrder(
tuple(normal.getKey(), RuleStatus.READY),
tuple(template.getKey(), RuleStatus.REMOVED),
- tuple(custom.getKey(), RuleStatus.REMOVED)
- );
+ tuple(custom.getKey(), RuleStatus.REMOVED));
@SuppressWarnings("unchecked")
- Class<ArrayList<RuleKey>> listClass = (Class<ArrayList<RuleKey>>)(Class)ArrayList.class;
+ Class<ArrayList<RuleKey>> listClass = (Class<ArrayList<RuleKey>>) (Class) ArrayList.class;
ArgumentCaptor<ArrayList<RuleKey>> indexedRuleKeys = ArgumentCaptor.forClass(listClass);
verify(ruleIndexer).commitAndIndex(any(), indexedRuleKeys.capture());
assertThat(indexedRuleKeys.getValue()).containsExactlyInAnyOrder(template.getKey(), custom.getKey());
diff --git a/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/QProfileFactoryImplTest.java b/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/QProfileFactoryImplTest.java
index 95195117374..f2a0e92b8b2 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/QProfileFactoryImplTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/QProfileFactoryImplTest.java
@@ -49,7 +49,10 @@ import org.sonar.server.qualityprofile.index.ActiveRuleIndexer;
import static java.util.Arrays.asList;
import static org.assertj.core.api.Assertions.assertThat;
+import static org.mockito.Matchers.any;
+import static org.mockito.Matchers.anyCollection;
import static org.mockito.Mockito.mock;
+import static org.mockito.Mockito.never;
import static org.mockito.Mockito.verify;
import static org.mockito.Mockito.verifyZeroInteractions;
@@ -285,17 +288,17 @@ public class QProfileFactoryImplTest {
}
private void verifyNoCallsActiveRuleIndexerDelete() {
- verifyCallActiveRuleIndexerDelete();
+ verify(activeRuleIndexer, never()).commitDeletionOfProfiles(any(DbSession.class), anyCollection());
}
- private void verifyCallActiveRuleIndexerDelete(String... expectedProfileUuids) {
+ private void verifyCallActiveRuleIndexerDelete(String... expectedRuleProfileUuids) {
Class<Set<QProfileDto>> setClass = (Class<Set<QProfileDto>>) (Class) Set.class;
ArgumentCaptor<Set<QProfileDto>> setCaptor = ArgumentCaptor.forClass(setClass);
- verify(activeRuleIndexer).deleteByProfiles(setCaptor.capture());
+ verify(activeRuleIndexer).commitDeletionOfProfiles(any(DbSession.class), setCaptor.capture());
assertThat(setCaptor.getValue())
.extracting(QProfileDto::getKee)
- .containsExactlyInAnyOrder(expectedProfileUuids);
+ .containsExactlyInAnyOrder(expectedRuleProfileUuids);
}
private void assertThatRulesProfileExists(RulesProfileDto rulesProfile) {
@@ -327,14 +330,6 @@ public class QProfileFactoryImplTest {
// TODO active_rule_parameters
}
- private void assertThatCustomProfileIsDefault(OrganizationDto org, QProfileDto profile) {
- assertThat(db.getDbClient().qualityProfileDao().selectDefaultProfile(dbSession, org, profile.getLanguage())).isEqualTo(profile.getKee());
- }
-
- private void assertThatCustomProfileIsAssociatedToProject(OrganizationDto org, QProfileDto profile) {
- assertThat(db.getDbClient().qualityProfileDao().selectProjectAssociations(dbSession, org, profile, null)).isNotEmpty();
- }
-
private static void assertEqual(QProfileDto p1, QProfileDto p2) {
assertThat(p2.getOrganizationUuid()).isEqualTo(p1.getOrganizationUuid());
assertThat(p2.getName()).isEqualTo(p1.getName());
diff --git a/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/RuleActivatorTest.java b/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/RuleActivatorTest.java
index 81efcf72ab6..3e76d1a83f3 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/RuleActivatorTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/RuleActivatorTest.java
@@ -48,7 +48,6 @@ import org.sonar.server.es.EsTester;
import org.sonar.server.es.SearchOptions;
import org.sonar.server.exceptions.BadRequestException;
import org.sonar.server.qualityprofile.index.ActiveRuleIndexer;
-import org.sonar.server.qualityprofile.index.ActiveRuleIteratorFactory;
import org.sonar.server.rule.index.RuleIndex;
import org.sonar.server.rule.index.RuleIndexDefinition;
import org.sonar.server.rule.index.RuleIndexer;
@@ -86,8 +85,7 @@ public class RuleActivatorTest {
public UserSessionRule userSession = UserSessionRule.standalone();
private RuleIndex ruleIndex = new RuleIndex(es.client());
private RuleActivatorContextFactory contextFactory = new RuleActivatorContextFactory(db.getDbClient());
- private ActiveRuleIteratorFactory activeRuleIteratorFactory = new ActiveRuleIteratorFactory(db.getDbClient());
- private ActiveRuleIndexer activeRuleIndexer = new ActiveRuleIndexer(db.getDbClient(), es.client(), activeRuleIteratorFactory);
+ private ActiveRuleIndexer activeRuleIndexer = new ActiveRuleIndexer(db.getDbClient(), es.client());
private RuleIndexer ruleIndexer = new RuleIndexer(es.client(), db.getDbClient());
private TypeValidations typeValidations = new TypeValidations(asList(new StringTypeValidation(), new IntegerTypeValidation()));
@@ -730,7 +728,7 @@ public class RuleActivatorTest {
public void unset_parent_when_no_parent_does_not_fail() {
RuleDefinitionDto rule = createRule();
QProfileDto profile = createProfile(rule);
- underTest.setParent(db.getSession(), profile, null);
+ underTest.setParentAndCommit(db.getSession(), profile, null);
}
@Test
@@ -740,7 +738,7 @@ public class RuleActivatorTest {
expectedException.expect(BadRequestException.class);
expectedException.expectMessage(" can not be selected as parent of ");
- underTest.setParent(db.getSession(), profile, profile);
+ underTest.setParentAndCommit(db.getSession(), profile, profile);
}
@Test
@@ -751,7 +749,7 @@ public class RuleActivatorTest {
expectedException.expect(BadRequestException.class);
expectedException.expectMessage(" can not be selected as parent of ");
- underTest.setParent(db.getSession(), parentProfile, childProfile);
+ underTest.setParentAndCommit(db.getSession(), parentProfile, childProfile);
}
@Test
@@ -763,7 +761,7 @@ public class RuleActivatorTest {
expectedException.expect(BadRequestException.class);
expectedException.expectMessage(" can not be selected as parent of ");
- underTest.setParent(db.getSession(), parentProfile, grandchildProfile);
+ underTest.setParentAndCommit(db.getSession(), parentProfile, grandchildProfile);
}
@Test
@@ -782,7 +780,7 @@ public class RuleActivatorTest {
expectedException.expect(BadRequestException.class);
expectedException.expectMessage("Cannot set the profile");
- underTest.setParent(db.getSession(), childProfile, parentProfile);
+ underTest.setParentAndCommit(db.getSession(), childProfile, parentProfile);
}
@Test
@@ -798,12 +796,12 @@ public class RuleActivatorTest {
changes = activate(profile2, RuleActivation.create(rule2.getKey()));
assertThat(changes).hasSize(1);
- changes = underTest.setParent(db.getSession(), profile2, profile1);
+ changes = underTest.setParentAndCommit(db.getSession(), profile2, profile1);
assertThat(changes).hasSize(1);
assertThatRuleIsActivated(profile2, rule1, changes, rule1.getSeverityString(), INHERITED, emptyMap());
assertThatRuleIsActivated(profile2, rule2, null, rule2.getSeverityString(), null, emptyMap());
- changes = underTest.setParent(db.getSession(), profile2, null);
+ changes = underTest.setParentAndCommit(db.getSession(), profile2, null);
assertThat(changes).hasSize(1);
assertThatRuleIsActivated(profile2, rule2, null, rule2.getSeverityString(), null, emptyMap());
assertThatRuleIsNotPresent(profile2, rule1);
@@ -821,7 +819,7 @@ public class RuleActivatorTest {
changes = activate(profile2, RuleActivation.create(rule2.getKey()));
assertThat(changes).hasSize(1);
- changes = underTest.setParent(db.getSession(), profile2, profile1);
+ changes = underTest.setParentAndCommit(db.getSession(), profile2, profile1);
assertThat(changes).hasSize(1);
assertThatRuleIsActivated(profile2, rule1, changes, rule1.getSeverityString(), INHERITED, emptyMap());
assertThatRuleIsActivated(profile2, rule2, null, rule2.getSeverityString(), null, emptyMap());
@@ -832,7 +830,7 @@ public class RuleActivatorTest {
assertThatRuleIsUpdated(profile2, rule1, BLOCKER, ActiveRule.Inheritance.OVERRIDES, emptyMap());
assertThatRuleIsActivated(profile2, rule2, null, rule2.getSeverityString(), null, emptyMap());
- changes = underTest.setParent(db.getSession(), profile2, null);
+ changes = underTest.setParentAndCommit(db.getSession(), profile2, null);
assertThat(changes).hasSize(1);
// Not testing changes here since severity is not set in changelog
assertThatRuleIsActivated(profile2, rule1, null, BLOCKER, null, emptyMap());
@@ -856,7 +854,7 @@ public class RuleActivatorTest {
RuleQuery ruleQuery = new RuleQuery()
.setRepositories(singletonList(repositoryKey));
- BulkChangeResult bulkChangeResult = underTest.bulkActivate(db.getSession(), ruleQuery, profile, MINOR);
+ BulkChangeResult bulkChangeResult = underTest.bulkActivateAndCommit(db.getSession(), ruleQuery, profile, MINOR);
assertThat(bulkChangeResult.countFailed()).isEqualTo(0);
assertThat(bulkChangeResult.countSucceeded()).isEqualTo(bulkSize);
@@ -883,7 +881,7 @@ public class RuleActivatorTest {
RuleQuery ruleQuery = new RuleQuery()
.setRepositories(singletonList(repositoryKey));
- BulkChangeResult bulkChangeResult = underTest.bulkActivate(db.getSession(), ruleQuery, profile, MINOR);
+ BulkChangeResult bulkChangeResult = underTest.bulkActivateAndCommit(db.getSession(), ruleQuery, profile, MINOR);
assertThat(bulkChangeResult.countFailed()).isEqualTo(0);
assertThat(bulkChangeResult.countSucceeded()).isEqualTo(bulkSize);
@@ -891,7 +889,7 @@ public class RuleActivatorTest {
assertThat(db.getDbClient().activeRuleDao().selectByProfile(db.getSession(), profile)).hasSize(bulkSize);
// Now deactivate all rules
- bulkChangeResult = underTest.bulkDeactivate(db.getSession(), ruleQuery, profile);
+ bulkChangeResult = underTest.bulkDeactivateAndCommit(db.getSession(), ruleQuery, profile);
assertThat(bulkChangeResult.countFailed()).isEqualTo(0);
assertThat(bulkChangeResult.countSucceeded()).isEqualTo(bulkSize);
@@ -915,7 +913,7 @@ public class RuleActivatorTest {
RuleQuery ruleQuery = new RuleQuery()
.setQProfile(childProfile);
- BulkChangeResult bulkChangeResult = underTest.bulkDeactivate(db.getSession(), ruleQuery, childProfile);
+ BulkChangeResult bulkChangeResult = underTest.bulkDeactivateAndCommit(db.getSession(), ruleQuery, childProfile);
assertThat(bulkChangeResult.countFailed()).isEqualTo(1);
assertThat(bulkChangeResult.countSucceeded()).isEqualTo(0);
@@ -940,7 +938,7 @@ public class RuleActivatorTest {
RuleQuery query = new RuleQuery()
.setRuleKey(rule1.getRuleKey())
.setQProfile(parentProfile);
- BulkChangeResult result = underTest.bulkActivate(db.getSession(), query, parentProfile, "BLOCKER");
+ BulkChangeResult result = underTest.bulkActivateAndCommit(db.getSession(), query, parentProfile, "BLOCKER");
assertThat(result.getChanges()).hasSize(3);
assertThat(result.countSucceeded()).isEqualTo(1);
@@ -1057,7 +1055,7 @@ public class RuleActivatorTest {
db.rules().update(rule1);
QProfileDto childProfile = createProfile(rule1);
- List<ActiveRuleChange> changes = underTest.setParent(db.getSession(), childProfile, parentProfile);
+ List<ActiveRuleChange> changes = underTest.setParentAndCommit(db.getSession(), childProfile, parentProfile);
assertThatRuleIsNotPresent(childProfile, rule1);
assertThatRuleIsActivated(childProfile, rule2, changes, rule2.getSeverityString(), INHERITED, emptyMap());
diff --git a/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/index/ActiveRuleIndexerTest.java b/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/index/ActiveRuleIndexerTest.java
index 5c3094ccf5f..216c7f428e0 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/index/ActiveRuleIndexerTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/index/ActiveRuleIndexerTest.java
@@ -19,28 +19,31 @@
*/
package org.sonar.server.qualityprofile.index;
+import java.util.Collection;
+import java.util.Collections;
import java.util.List;
+import java.util.stream.Collectors;
+import org.assertj.core.groups.Tuple;
import org.junit.Before;
import org.junit.Rule;
import org.junit.Test;
import org.sonar.api.config.internal.MapSettings;
import org.sonar.api.utils.System2;
import org.sonar.db.DbTester;
+import org.sonar.db.es.EsQueueDto;
import org.sonar.db.organization.OrganizationDto;
import org.sonar.db.qualityprofile.ActiveRuleDto;
import org.sonar.db.qualityprofile.QProfileDto;
-import org.sonar.db.qualityprofile.RulesProfileDto;
import org.sonar.db.rule.RuleDefinitionDto;
import org.sonar.server.es.EsTester;
import org.sonar.server.qualityprofile.ActiveRuleChange;
import org.sonar.server.rule.index.RuleIndexDefinition;
import static java.util.Arrays.asList;
+import static java.util.Arrays.stream;
import static java.util.Collections.emptySet;
import static java.util.Collections.singletonList;
import static org.assertj.core.api.Assertions.assertThat;
-import static org.sonar.server.qualityprofile.ActiveRuleChange.Type.ACTIVATED;
-import static org.sonar.server.qualityprofile.ActiveRuleChange.Type.DEACTIVATED;
import static org.sonar.server.rule.index.RuleIndexDefinition.INDEX_TYPE_ACTIVE_RULE;
public class ActiveRuleIndexerTest {
@@ -53,7 +56,7 @@ public class ActiveRuleIndexerTest {
@Rule
public EsTester es = new EsTester(RuleIndexDefinition.createForTest(new MapSettings().asConfig()));
- private ActiveRuleIndexer underTest = new ActiveRuleIndexer(db.getDbClient(), es.client(), new ActiveRuleIteratorFactory(db.getDbClient()));
+ private ActiveRuleIndexer underTest = new ActiveRuleIndexer(db.getDbClient(), es.client());
private RuleDefinitionDto rule1;
private RuleDefinitionDto rule2;
private OrganizationDto org;
@@ -71,7 +74,7 @@ public class ActiveRuleIndexerTest {
@Test
public void getIndexTypes() {
- assertThat(underTest.getIndexTypes()).containsExactly(RuleIndexDefinition.INDEX_TYPE_ACTIVE_RULE);
+ assertThat(underTest.getIndexTypes()).containsExactly(INDEX_TYPE_ACTIVE_RULE);
}
@Test
@@ -89,78 +92,111 @@ public class ActiveRuleIndexerTest {
List<ActiveRuleDoc> docs = es.getDocuments(INDEX_TYPE_ACTIVE_RULE, ActiveRuleDoc.class);
assertThat(docs).hasSize(1);
verify(docs.get(0), rule1, profile1, activeRule);
+ assertThatEsQueueTableIsEmpty();
}
@Test
- public void deleteByProfiles() throws Exception {
- ActiveRuleDto activeRule1 = db.qualityProfiles().activateRule(profile1, rule1);
- ActiveRuleDto activeRule2 = db.qualityProfiles().activateRule(profile2, rule1);
- ActiveRuleDto activeRule3 = db.qualityProfiles().activateRule(profile2, rule2);
- index();
+ public void test_commitAndIndex() {
+ ActiveRuleDto ar1 = db.qualityProfiles().activateRule(profile1, rule1);
+ ActiveRuleDto ar2 = db.qualityProfiles().activateRule(profile2, rule1);
+ ActiveRuleDto ar3 = db.qualityProfiles().activateRule(profile2, rule2);
- underTest.deleteByProfiles(singletonList(profile2));
+ commitAndIndex(ar1, ar2);
- verifyOnlyIndexed(activeRule1);
+ verifyOnlyIndexed(ar1, ar2);
+ assertThatEsQueueTableIsEmpty();
}
@Test
- public void deleteByProfiles_does_nothing_if_profiles_are_not_indexed() throws Exception {
- ActiveRuleDto activeRule1 = db.qualityProfiles().activateRule(profile1, rule1);
- ActiveRuleDto activeRule2 = db.qualityProfiles().activateRule(profile2, rule1);
- ActiveRuleDto activeRule3 = db.qualityProfiles().activateRule(profile2, rule2);
- assertThat(es.countDocuments(INDEX_TYPE_ACTIVE_RULE)).isEqualTo(0);
+ public void commitAndIndex_empty_list() {
+ ActiveRuleDto ar = db.qualityProfiles().activateRule(profile1, rule1);
- underTest.deleteByProfiles(singletonList(profile2));
+ underTest.commitAndIndex(db.getSession(), Collections.emptyList());
assertThat(es.countDocuments(INDEX_TYPE_ACTIVE_RULE)).isEqualTo(0);
+ assertThatEsQueueTableIsEmpty();
}
@Test
- public void indexRuleProfile() throws Exception {
- ActiveRuleDto activeRule1 = db.qualityProfiles().activateRule(profile1, rule1);
- ActiveRuleDto activeRule2 = db.qualityProfiles().activateRule(profile2, rule1);
- ActiveRuleDto activeRule3 = db.qualityProfiles().activateRule(profile2, rule2);
+ public void commitAndIndex_keeps_elements_to_recover_in_ES_QUEUE_on_errors() {
+ ActiveRuleDto ar = db.qualityProfiles().activateRule(profile1, rule1);
+ // force error by deleting the index
+ deleteRulesIndex();
- indexProfile(profile2);
+ commitAndIndex(ar);
- verifyOnlyIndexed(activeRule2, activeRule3);
+ EsQueueDto expectedItem = EsQueueDto.create(EsQueueDto.Type.ACTIVE_RULE, "" + ar.getId(), "activeRuleId", ar.getRuleKey().toString());
+ assertThatEsQueueContainsExactly(expectedItem);
}
@Test
- public void indexChanges_puts_documents() throws Exception {
- ActiveRuleDto activeRule1 = db.qualityProfiles().activateRule(profile1, rule1);
- ActiveRuleDto activeRule2 = db.qualityProfiles().activateRule(profile2, rule1);
- ActiveRuleDto nonIndexed = db.qualityProfiles().activateRule(profile2, rule2);
+ public void commitAndIndex_deletes_the_documents_that_dont_exist_in_database() {
+ ActiveRuleDto ar = db.qualityProfiles().activateRule(profile1, rule1);
+ indexAll();
+ assertThat(es.countDocuments(INDEX_TYPE_ACTIVE_RULE)).isEqualTo(1);
- underTest.indexChanges(db.getSession(), asList(
- newChange(ACTIVATED, activeRule1), newChange(ACTIVATED, activeRule2)));
+ db.getDbClient().activeRuleDao().delete(db.getSession(), ar.getKey());
+ commitAndIndex(ar);
- verifyOnlyIndexed(activeRule1, activeRule2);
+ assertThat(es.countDocuments(INDEX_TYPE_ACTIVE_RULE)).isEqualTo(0);
+ assertThatEsQueueTableIsEmpty();
}
@Test
- public void indexChanges_deletes_documents_when_type_is_DEACTIVATED() throws Exception {
- ActiveRuleDto activeRule1 = db.qualityProfiles().activateRule(profile1, rule1);
- ActiveRuleDto activeRule2 = db.qualityProfiles().activateRule(profile2, rule1);
- underTest.indexChanges(db.getSession(), asList(
- newChange(ACTIVATED, activeRule1), newChange(ACTIVATED, activeRule2)));
- assertThat(es.countDocuments(RuleIndexDefinition.INDEX_TYPE_ACTIVE_RULE)).isEqualTo(2);
+ public void index_fails_and_deletes_doc_if_docIdType_is_unsupported() {
+ EsQueueDto item = EsQueueDto.create(EsQueueDto.Type.ACTIVE_RULE, "the_id", "unsupported", "the_routing");
+ db.getDbClient().esQueueDao().insert(db.getSession(), item);
- underTest.indexChanges(db.getSession(), singletonList(newChange(DEACTIVATED, activeRule1)));
+ underTest.index(db.getSession(), asList(item));
- verifyOnlyIndexed(activeRule2);
+ assertThatEsQueueTableIsEmpty();
+ assertThat(es.countDocuments(INDEX_TYPE_ACTIVE_RULE)).isEqualTo(0);
}
@Test
- public void deleteByRuleKeys() {
- ActiveRuleDto active1 = db.qualityProfiles().activateRule(profile1, rule1);
- ActiveRuleDto active2 = db.qualityProfiles().activateRule(profile2, rule1);
- ActiveRuleDto onRule2 = db.qualityProfiles().activateRule(profile2, rule2);
- index();
+ public void commitDeletionOfProfiles() {
+ ActiveRuleDto ar1 = db.qualityProfiles().activateRule(profile1, rule1);
+ ActiveRuleDto ar2 = db.qualityProfiles().activateRule(profile2, rule1);
+ ActiveRuleDto ar3 = db.qualityProfiles().activateRule(profile2, rule2);
+ indexAll();
+ db.getDbClient().qualityProfileDao().deleteRulesProfilesByUuids(db.getSession(), singletonList(profile2.getRulesProfileUuid()));
+
+ underTest.commitDeletionOfProfiles(db.getSession(), singletonList(profile2));
+
+ verifyOnlyIndexed(ar1);
+ }
+
+ @Test
+ public void commitDeletionOfProfiles_does_nothing_if_profiles_are_not_indexed() {
+ db.qualityProfiles().activateRule(profile1, rule1);
+ indexAll();
+ assertThat(es.countDocuments(INDEX_TYPE_ACTIVE_RULE)).isEqualTo(1);
+
+ underTest.commitDeletionOfProfiles(db.getSession(), singletonList(profile2));
+
+ assertThat(es.countDocuments(INDEX_TYPE_ACTIVE_RULE)).isEqualTo(1);
+ }
- underTest.deleteByRuleKeys(singletonList(rule2.getKey()));
- verifyOnlyIndexed(active1, active2);
+ private void deleteRulesIndex() {
+ es.deleteIndex(RuleIndexDefinition.INDEX_TYPE_RULE.getIndex());
+ }
+
+ private void assertThatEsQueueTableIsEmpty() {
+ assertThat(db.countRowsOfTable(db.getSession(), "es_queue")).isEqualTo(0);
+ }
+
+ private void assertThatEsQueueContainsExactly(EsQueueDto expected) {
+ Collection<EsQueueDto> items = db.getDbClient().esQueueDao().selectForRecovery(db.getSession(), system2.now() + 1_000, 10);
+ assertThat(items)
+ .extracting(EsQueueDto::getDocId, EsQueueDto::getDocIdType, EsQueueDto::getDocRouting)
+ .containsExactlyInAnyOrder(Tuple.tuple(expected.getDocId(), expected.getDocIdType(), expected.getDocRouting()));
+ }
+
+ private void commitAndIndex(ActiveRuleDto... ar) {
+ underTest.commitAndIndex(db.getSession(), stream(ar)
+ .map(a -> new ActiveRuleChange(ActiveRuleChange.Type.ACTIVATED, a))
+ .collect(Collectors.toList()));
}
private void verifyOnlyIndexed(ActiveRuleDto... expected) {
@@ -171,14 +207,6 @@ public class ActiveRuleIndexerTest {
}
}
- private ActiveRuleChange newChange(ActiveRuleChange.Type type, ActiveRuleDto activeRule) {
- return new ActiveRuleChange(type, activeRule);
- }
-
- private void indexProfile(QProfileDto profile) {
- underTest.indexRuleProfile(db.getSession(), RulesProfileDto.from(profile));
- }
-
private void verify(ActiveRuleDoc doc1, RuleDefinitionDto rule, QProfileDto profile, ActiveRuleDto activeRule) {
assertThat(doc1)
.matches(doc -> doc.getRuleKey().equals(rule.getKey()))
@@ -188,7 +216,7 @@ public class ActiveRuleIndexerTest {
.matches(doc -> doc.getSeverity().equals(activeRule.getSeverityString()));
}
- private void index() {
+ private void indexAll() {
underTest.indexOnStartup(emptySet());
}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/ws/ActivateRuleActionTest.java b/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/ws/ActivateRuleActionTest.java
index c2598bee161..a2bef4dbdb8 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/ws/ActivateRuleActionTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/ws/ActivateRuleActionTest.java
@@ -41,7 +41,6 @@ import org.sonar.server.exceptions.UnauthorizedException;
import org.sonar.server.organization.TestDefaultOrganizationProvider;
import org.sonar.server.qualityprofile.RuleActivation;
import org.sonar.server.qualityprofile.RuleActivator;
-import org.sonar.server.qualityprofile.index.ActiveRuleIndexer;
import org.sonar.server.tester.UserSessionRule;
import org.sonar.server.ws.TestRequest;
import org.sonar.server.ws.TestResponse;
@@ -68,9 +67,8 @@ public class ActivateRuleActionTest {
private DbClient dbClient = dbTester.getDbClient();
private RuleActivator ruleActivator = mock(RuleActivator.class);
private QProfileWsSupport wsSupport = new QProfileWsSupport(dbClient, userSession, TestDefaultOrganizationProvider.from(dbTester));
- private ActiveRuleIndexer activeRuleIndexer = mock(ActiveRuleIndexer.class);
- private WsActionTester ws = new WsActionTester(new ActivateRuleAction(dbClient, ruleActivator, userSession, wsSupport, activeRuleIndexer));
+ private WsActionTester ws = new WsActionTester(new ActivateRuleAction(dbClient, ruleActivator, userSession, wsSupport));
private OrganizationDto defaultOrganization;
private OrganizationDto organization;
@@ -152,7 +150,7 @@ public class ActivateRuleActionTest {
assertThat(response.getStatus()).isEqualTo(HttpURLConnection.HTTP_NO_CONTENT);
ArgumentCaptor<RuleActivation> captor = ArgumentCaptor.forClass(RuleActivation.class);
- verify(ruleActivator).activate(any(DbSession.class), captor.capture(), any(QProfileDto.class));
+ verify(ruleActivator).activateAndCommit(any(DbSession.class), captor.capture(), any(QProfileDto.class));
RuleActivation value = captor.getValue();
assertThat(value.getRuleKey()).isEqualTo(ruleKey);
assertThat(value.getSeverity()).isEqualTo(Severity.BLOCKER);
@@ -176,7 +174,7 @@ public class ActivateRuleActionTest {
assertThat(response.getStatus()).isEqualTo(HttpURLConnection.HTTP_NO_CONTENT);
ArgumentCaptor<RuleActivation> captor = ArgumentCaptor.forClass(RuleActivation.class);
- verify(ruleActivator).activate(any(DbSession.class), captor.capture(), any(QProfileDto.class));
+ verify(ruleActivator).activateAndCommit(any(DbSession.class), captor.capture(), any(QProfileDto.class));
assertThat(captor.getValue().getRuleKey()).isEqualTo(ruleKey);
assertThat(captor.getValue().getSeverity()).isEqualTo(Severity.BLOCKER);
assertThat(captor.getValue().isReset()).isFalse();
diff --git a/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/ws/ChangeParentActionTest.java b/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/ws/ChangeParentActionTest.java
index edcd0787fe9..8ca0a99b486 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/ws/ChangeParentActionTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/ws/ChangeParentActionTest.java
@@ -53,7 +53,6 @@ import org.sonar.server.organization.TestDefaultOrganizationProvider;
import org.sonar.server.qualityprofile.RuleActivator;
import org.sonar.server.qualityprofile.RuleActivatorContextFactory;
import org.sonar.server.qualityprofile.index.ActiveRuleIndexer;
-import org.sonar.server.qualityprofile.index.ActiveRuleIteratorFactory;
import org.sonar.server.rule.index.RuleIndex;
import org.sonar.server.rule.index.RuleIndexDefinition;
import org.sonar.server.rule.index.RuleIndexer;
@@ -104,7 +103,7 @@ public class ChangeParentActionTest {
EsClient esClient = esTester.client();
ruleIndex = new RuleIndex(esClient);
ruleIndexer = new RuleIndexer(esClient, dbClient);
- activeRuleIndexer = new ActiveRuleIndexer(dbClient, esClient, new ActiveRuleIteratorFactory(dbClient));
+ activeRuleIndexer = new ActiveRuleIndexer(dbClient, esClient);
RuleActivatorContextFactory ruleActivatorContextFactory = new RuleActivatorContextFactory(dbClient);
TypeValidations typeValidations = new TypeValidations(Collections.emptyList());
ruleActivator = new RuleActivator(System2.INSTANCE, dbClient, ruleIndex, ruleActivatorContextFactory, typeValidations, activeRuleIndexer, userSessionRule);
@@ -191,7 +190,7 @@ public class ChangeParentActionTest {
activeRuleIndexer.indexOnStartup(emptySet());
// Set parent 1
- ruleActivator.setParent(dbSession, child, parent1);
+ ruleActivator.setParentAndCommit(dbSession, child, parent1);
// Set parent 2 through WS
ws.newRequest()
@@ -219,7 +218,7 @@ public class ChangeParentActionTest {
activeRuleIndexer.indexOnStartup(emptySet());
// Set parent
- ruleActivator.setParent(dbSession, child, parent);
+ ruleActivator.setParentAndCommit(dbSession, child, parent);
// Remove parent through WS
ws.newRequest()
@@ -308,7 +307,7 @@ public class ChangeParentActionTest {
assertThat(dbClient.activeRuleDao().selectByProfileUuid(dbSession, child.getKee())).isEmpty();
// Set parent
- ruleActivator.setParent(dbSession, child, parent);
+ ruleActivator.setParentAndCommit(dbSession, child, parent);
// Remove parent
ws.newRequest()
diff --git a/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/ws/CreateActionTest.java b/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/ws/CreateActionTest.java
index 33c34e2daa3..a27c80aebe3 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/ws/CreateActionTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/ws/CreateActionTest.java
@@ -53,7 +53,6 @@ import org.sonar.server.qualityprofile.QProfileFactoryImpl;
import org.sonar.server.qualityprofile.RuleActivator;
import org.sonar.server.qualityprofile.RuleActivatorContextFactory;
import org.sonar.server.qualityprofile.index.ActiveRuleIndexer;
-import org.sonar.server.qualityprofile.index.ActiveRuleIteratorFactory;
import org.sonar.server.rule.index.RuleIndex;
import org.sonar.server.rule.index.RuleIndexDefinition;
import org.sonar.server.rule.index.RuleIndexer;
@@ -93,7 +92,7 @@ public class CreateActionTest {
private DbSession dbSession = db.getSession();
private RuleIndex ruleIndex = new RuleIndex(es.client());
private RuleIndexer ruleIndexer = new RuleIndexer(es.client(), dbClient);
- private ActiveRuleIndexer activeRuleIndexer = new ActiveRuleIndexer(dbClient, es.client(), new ActiveRuleIteratorFactory(dbClient));
+ private ActiveRuleIndexer activeRuleIndexer = new ActiveRuleIndexer(dbClient, es.client());
private ProfileImporter[] profileImporters = createImporters();
private QProfileExporters qProfileExporters = new QProfileExporters(dbClient, null,
new RuleActivator(mock(System2.class), dbClient, ruleIndex, new RuleActivatorContextFactory(dbClient), null, activeRuleIndexer, userSession),
diff --git a/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/ws/DeactivateRuleActionTest.java b/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/ws/DeactivateRuleActionTest.java
index 9ea0717b041..3c70df775d3 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/ws/DeactivateRuleActionTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/ws/DeactivateRuleActionTest.java
@@ -143,7 +143,7 @@ public class DeactivateRuleActionTest {
assertThat(response.getStatus()).isEqualTo(HttpURLConnection.HTTP_NO_CONTENT);
ArgumentCaptor<RuleKey> ruleKeyCaptor = ArgumentCaptor.forClass(RuleKey.class);
ArgumentCaptor<QProfileDto> qProfileDtoCaptor = ArgumentCaptor.forClass(QProfileDto.class);
- verify(ruleActivator).deactivateAndUpdateIndex(any(DbSession.class), qProfileDtoCaptor.capture(), ruleKeyCaptor.capture());
+ verify(ruleActivator).deactivateAndCommit(any(DbSession.class), qProfileDtoCaptor.capture(), ruleKeyCaptor.capture());
assertThat(ruleKeyCaptor.getValue()).isEqualTo(ruleKey);
assertThat(qProfileDtoCaptor.getValue().getKee()).isEqualTo(qualityProfile.getKee());
}
@@ -164,7 +164,7 @@ public class DeactivateRuleActionTest {
assertThat(response.getStatus()).isEqualTo(HttpURLConnection.HTTP_NO_CONTENT);
ArgumentCaptor<RuleKey> captor = ArgumentCaptor.forClass(RuleKey.class);
ArgumentCaptor<QProfileDto> qProfileDtoCaptor = ArgumentCaptor.forClass(QProfileDto.class);
- verify(ruleActivator).deactivateAndUpdateIndex(any(DbSession.class), qProfileDtoCaptor.capture(), captor.capture());
+ verify(ruleActivator).deactivateAndCommit(any(DbSession.class), qProfileDtoCaptor.capture(), captor.capture());
assertThat(captor.getValue()).isEqualTo(ruleKey);
assertThat(qProfileDtoCaptor.getValue().getKee()).isEqualTo(qualityProfile.getKee());
}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/ws/InheritanceActionTest.java b/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/ws/InheritanceActionTest.java
index 65841db6c72..beca53c8829 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/ws/InheritanceActionTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/ws/InheritanceActionTest.java
@@ -50,7 +50,6 @@ import org.sonar.server.qualityprofile.RuleActivation;
import org.sonar.server.qualityprofile.RuleActivator;
import org.sonar.server.qualityprofile.RuleActivatorContextFactory;
import org.sonar.server.qualityprofile.index.ActiveRuleIndexer;
-import org.sonar.server.qualityprofile.index.ActiveRuleIteratorFactory;
import org.sonar.server.rule.index.RuleIndex;
import org.sonar.server.rule.index.RuleIndexDefinition;
import org.sonar.server.rule.index.RuleIndexer;
@@ -92,7 +91,7 @@ public class InheritanceActionTest {
dbSession = dbTester.getSession();
esClient = es.client();
ruleIndexer = new RuleIndexer(esClient, dbClient);
- activeRuleIndexer = new ActiveRuleIndexer(dbClient, esClient, new ActiveRuleIteratorFactory(dbClient));
+ activeRuleIndexer = new ActiveRuleIndexer(dbClient, esClient);
TestDefaultOrganizationProvider defaultOrganizationProvider = TestDefaultOrganizationProvider.from(dbTester);
underTest = new InheritanceAction(
dbClient,
@@ -251,7 +250,7 @@ public class InheritanceActionTest {
}
private void setParent(QProfileDto profile, QProfileDto parent) {
- ruleActivator.setParent(dbSession, parent, profile);
+ ruleActivator.setParentAndCommit(dbSession, parent, profile);
}
private RuleDefinitionDto createRule(String lang, String id) {
diff --git a/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/ws/ShowActionTest.java b/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/ws/ShowActionTest.java
index f215112b365..8e605e81919 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/ws/ShowActionTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/qualityprofile/ws/ShowActionTest.java
@@ -35,7 +35,6 @@ import org.sonar.server.es.EsTester;
import org.sonar.server.exceptions.NotFoundException;
import org.sonar.server.organization.TestDefaultOrganizationProvider;
import org.sonar.server.qualityprofile.index.ActiveRuleIndexer;
-import org.sonar.server.qualityprofile.index.ActiveRuleIteratorFactory;
import org.sonar.server.rule.index.RuleIndex;
import org.sonar.server.rule.index.RuleIndexDefinition;
import org.sonar.server.rule.index.RuleIndexer;
@@ -73,7 +72,7 @@ public class ShowActionTest {
public ExpectedException expectedException = ExpectedException.none();
private RuleIndexer ruleIndexer = new RuleIndexer(es.client(), db.getDbClient());
- private ActiveRuleIndexer activeRuleIndexer = new ActiveRuleIndexer(db.getDbClient(), es.client(), new ActiveRuleIteratorFactory(db.getDbClient()));
+ private ActiveRuleIndexer activeRuleIndexer = new ActiveRuleIndexer(db.getDbClient(), es.client());
private RuleIndex ruleIndex = new RuleIndex(es.client());
private WsActionTester ws = new WsActionTester(
diff --git a/server/sonar-server/src/test/java/org/sonar/server/rule/RegisterRulesTest.java b/server/sonar-server/src/test/java/org/sonar/server/rule/RegisterRulesTest.java
index b91ed9fb686..50db488c3a3 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/rule/RegisterRulesTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/rule/RegisterRulesTest.java
@@ -49,7 +49,6 @@ import org.sonar.server.organization.OrganizationFlags;
import org.sonar.server.organization.TestOrganizationFlags;
import org.sonar.server.qualityprofile.RuleActivator;
import org.sonar.server.qualityprofile.index.ActiveRuleIndexer;
-import org.sonar.server.qualityprofile.index.ActiveRuleIteratorFactory;
import org.sonar.server.rule.index.RuleIndex;
import org.sonar.server.rule.index.RuleIndexDefinition;
import org.sonar.server.rule.index.RuleIndexer;
@@ -99,7 +98,7 @@ public class RegisterRulesTest {
when(system.now()).thenReturn(DATE1.getTime());
ruleIndexer = new RuleIndexer(esTester.client(), dbClient);
ruleIndex = new RuleIndex(esTester.client());
- activeRuleIndexer = new ActiveRuleIndexer(dbClient, esTester.client(), new ActiveRuleIteratorFactory(dbClient));
+ activeRuleIndexer = new ActiveRuleIndexer(dbClient, esTester.client());
defaultOrganization = dbTester.getDefaultOrganization();
}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/rule/index/RuleIndexTest.java b/server/sonar-server/src/test/java/org/sonar/server/rule/index/RuleIndexTest.java
index 465ab5f8e68..3e0afe24407 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/rule/index/RuleIndexTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/rule/index/RuleIndexTest.java
@@ -44,7 +44,6 @@ import org.sonar.server.es.Facets;
import org.sonar.server.es.SearchIdResult;
import org.sonar.server.es.SearchOptions;
import org.sonar.server.qualityprofile.index.ActiveRuleIndexer;
-import org.sonar.server.qualityprofile.index.ActiveRuleIteratorFactory;
import static com.google.common.collect.ImmutableSet.of;
import static java.util.Arrays.asList;
@@ -104,7 +103,7 @@ public class RuleIndexTest {
@Before
public void setUp() {
ruleIndexer = new RuleIndexer(es.client(), db.getDbClient());
- activeRuleIndexer = new ActiveRuleIndexer(db.getDbClient(), es.client(), new ActiveRuleIteratorFactory(db.getDbClient()));
+ activeRuleIndexer = new ActiveRuleIndexer(db.getDbClient(), es.client());
underTest = new RuleIndex(es.client());
}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/rule/index/RuleIteratorForSingleChunkTest.java b/server/sonar-server/src/test/java/org/sonar/server/rule/index/RuleIteratorForSingleChunkTest.java
deleted file mode 100644
index 0e78b373b40..00000000000
--- a/server/sonar-server/src/test/java/org/sonar/server/rule/index/RuleIteratorForSingleChunkTest.java
+++ /dev/null
@@ -1,201 +0,0 @@
-/*
- * 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 org.sonar.server.rule.index;
-
-import com.google.common.collect.Lists;
-import java.util.List;
-import org.junit.Before;
-import org.junit.Rule;
-import org.junit.Test;
-import org.sonar.api.rule.RuleKey;
-import org.sonar.api.rule.RuleStatus;
-import org.sonar.api.rule.Severity;
-import org.sonar.api.rules.RuleType;
-import org.sonar.api.utils.System2;
-import org.sonar.db.DbClient;
-import org.sonar.db.DbSession;
-import org.sonar.db.DbTester;
-import org.sonar.db.rule.RuleDefinitionDto;
-import org.sonar.db.rule.RuleDto;
-import org.sonar.server.exceptions.NotFoundException;
-
-import static com.google.common.collect.Sets.newHashSet;
-import static org.assertj.core.api.Assertions.assertThat;
-
-public class RuleIteratorForSingleChunkTest {
-
- @Rule
- public DbTester dbTester = DbTester.create(System2.INSTANCE);
-
- private DbClient dbClient = dbTester.getDbClient();
- private DbSession dbSession = dbTester.getSession();
- private RuleDefinitionDto templateRule;
- private RuleDefinitionDto customRule;
-
- @Before
- public void setUp() throws Exception {
- templateRule = new RuleDefinitionDto()
- .setRuleKey("S001")
- .setRepositoryKey("xoo")
- .setConfigKey("S1")
- .setName("Null Pointer")
- .setDescription("S001 desc")
- .setDescriptionFormat(RuleDto.Format.HTML)
- .setLanguage("xoo")
- .setSeverity(Severity.BLOCKER)
- .setStatus(RuleStatus.READY)
- .setIsTemplate(true)
- .setSystemTags(newHashSet("cwe"))
- .setType(RuleType.BUG)
- .setCreatedAt(1500000000000L)
- .setUpdatedAt(1600000000000L);
-
- customRule = new RuleDefinitionDto()
- .setRuleKey("S002")
- .setRepositoryKey("xoo")
- .setConfigKey("S2")
- .setName("Slow")
- .setDescription("*S002 desc*")
- .setDescriptionFormat(RuleDto.Format.MARKDOWN)
- .setLanguage("xoo")
- .setSeverity(Severity.MAJOR)
- .setStatus(RuleStatus.BETA)
- .setIsTemplate(false)
- .setType(RuleType.CODE_SMELL)
- .setCreatedAt(2000000000000L)
- .setUpdatedAt(2100000000000L);
- }
-
- @Test
- public void iterator_over_one_rule() {
- dbTester.rules().insert(templateRule);
-
- List<RuleDocWithSystemScope> results = getResults();
-
- assertThat(results).hasSize(1);
-
- RuleDocWithSystemScope ruleDocWithSystemScope = getRuleDoc(results, templateRule.getRuleKey());
- RuleDoc templateDoc = ruleDocWithSystemScope.getRuleDoc();
- RuleExtensionDoc templateExtensionDoc = ruleDocWithSystemScope.getRuleExtensionDoc();
- assertThat(templateDoc).isNotNull();
- assertThat(templateDoc.key()).isEqualTo(RuleKey.of("xoo", "S001"));
- assertThat(templateDoc.ruleKey()).isEqualTo("S001");
- assertThat(templateDoc.repository()).isEqualTo("xoo");
- assertThat(templateDoc.internalKey()).isEqualTo("S1");
- assertThat(templateDoc.name()).isEqualTo("Null Pointer");
- assertThat(templateDoc.htmlDescription()).isEqualTo("S001 desc");
- assertThat(templateDoc.language()).isEqualTo("xoo");
- assertThat(templateDoc.severity()).isEqualTo(Severity.BLOCKER);
- assertThat(templateDoc.status()).isEqualTo(RuleStatus.READY);
- assertThat(templateDoc.isTemplate()).isTrue();
- assertThat(templateExtensionDoc.getTags()).containsOnly("cwe");
- assertThat(templateDoc.createdAt()).isEqualTo(1500000000000L);
- assertThat(templateDoc.updatedAt()).isEqualTo(1600000000000L);
- }
-
- @Test
- public void iterator_over_rules() {
- dbTester.rules().insert(templateRule);
- dbClient.ruleDao().insert(dbSession, customRule);
- dbSession.commit();
-
- List<RuleDocWithSystemScope> results = getResults();
-
- assertThat(results).hasSize(2);
-
- RuleDocWithSystemScope templateDocWithSystemScope = getRuleDoc(results, templateRule.getRuleKey());
- RuleDoc templateDoc = templateDocWithSystemScope.getRuleDoc();
- RuleExtensionDoc templateExtensionDoc = templateDocWithSystemScope.getRuleExtensionDoc();
- assertThat(templateDoc.key()).isEqualTo(RuleKey.of("xoo", "S001"));
- assertThat(templateDoc.ruleKey()).isEqualTo("S001");
- assertThat(templateDoc.repository()).isEqualTo("xoo");
- assertThat(templateDoc.internalKey()).isEqualTo("S1");
- assertThat(templateDoc.name()).isEqualTo("Null Pointer");
- assertThat(templateDoc.htmlDescription()).isEqualTo("S001 desc");
- assertThat(templateDoc.language()).isEqualTo("xoo");
- assertThat(templateDoc.severity()).isEqualTo(Severity.BLOCKER);
- assertThat(templateDoc.status()).isEqualTo(RuleStatus.READY);
- assertThat(templateDoc.isTemplate()).isTrue();
- assertThat(templateExtensionDoc.getTags()).containsOnly("cwe");
- assertThat(templateDoc.createdAt()).isEqualTo(1500000000000L);
- assertThat(templateDoc.updatedAt()).isEqualTo(1600000000000L);
-
- RuleDocWithSystemScope customDocWithSystemScope = getRuleDoc(results, customRule.getRuleKey());
- RuleDoc customDoc = customDocWithSystemScope.getRuleDoc();
- RuleExtensionDoc customExtensionDoc = customDocWithSystemScope.getRuleExtensionDoc();
- assertThat(customDoc.key()).isEqualTo(RuleKey.of("xoo", "S002"));
- assertThat(customDoc.ruleKey()).isEqualTo("S002");
- assertThat(customDoc.repository()).isEqualTo("xoo");
- assertThat(customDoc.internalKey()).isEqualTo("S2");
- assertThat(customDoc.name()).isEqualTo("Slow");
- assertThat(customDoc.htmlDescription()).isEqualTo("<strong>S002 desc</strong>");
- assertThat(customDoc.language()).isEqualTo("xoo");
- assertThat(customDoc.severity()).isEqualTo(Severity.MAJOR);
- assertThat(customDoc.status()).isEqualTo(RuleStatus.BETA);
- assertThat(customDoc.isTemplate()).isFalse();
- assertThat(customExtensionDoc.getTags()).isEmpty();
- assertThat(customDoc.createdAt()).isEqualTo(2000000000000L);
- assertThat(customDoc.updatedAt()).isEqualTo(2100000000000L);
- }
-
- @Test
- public void custom_rule() {
- dbTester.rules().insert(templateRule);
- dbClient.ruleDao().insert(dbSession, customRule.setTemplateId(templateRule.getId()));
- dbSession.commit();
-
- List<RuleDocWithSystemScope> results = getResults();
-
- assertThat(results).hasSize(2);
-
- RuleDocWithSystemScope templateDocWithSystemScope = getRuleDoc(results, templateRule.getRuleKey());
- RuleDoc templateDoc = templateDocWithSystemScope.getRuleDoc();
- assertThat(templateDoc.isTemplate()).isTrue();
- assertThat(templateDoc.templateKey()).isNull();
-
- RuleDocWithSystemScope customDocWithSystemScope = getRuleDoc(results, customRule.getRuleKey());
- RuleDoc customDoc = customDocWithSystemScope.getRuleDoc();
- assertThat(customDoc.isTemplate()).isFalse();
- assertThat(customDoc.templateKey()).isEqualTo(RuleKey.of("xoo", "S001"));
- }
-
- @Test
- public void removed_rule_is_returned() {
- dbTester.rules().insert(templateRule.setStatus(RuleStatus.REMOVED));
- dbSession.commit();
-
- List<RuleDocWithSystemScope> results = getResults();
-
- assertThat(results).hasSize(1);
- }
-
- private List<RuleDocWithSystemScope> getResults() {
- return Lists.newArrayList(new RuleIteratorForSingleChunk(dbTester.getDbClient(), null));
- }
-
- private RuleDocWithSystemScope getRuleDoc(List<RuleDocWithSystemScope> results, String ruleKey) {
- RuleDocWithSystemScope rule;
- rule = results.stream()
- .filter(r -> ruleKey.equals(r.getRuleDoc().key().rule()))
- .findAny()
- .orElseThrow(() -> new NotFoundException("Rule not found in results"));
- return rule;
- }
-}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/rule/ws/DeleteActionTest.java b/server/sonar-server/src/test/java/org/sonar/server/rule/ws/DeleteActionTest.java
index 06971a1b467..d95fed22039 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/rule/ws/DeleteActionTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/rule/ws/DeleteActionTest.java
@@ -22,7 +22,7 @@ package org.sonar.server.rule.ws;
import org.junit.Rule;
import org.junit.Test;
import org.junit.rules.ExpectedException;
-import org.sonar.api.config.MapSettings;
+import org.sonar.api.config.internal.MapSettings;
import org.sonar.api.rule.RuleStatus;
import org.sonar.api.utils.System2;
import org.sonar.db.DbClient;
@@ -56,7 +56,7 @@ public class DeleteActionTest {
@Rule
public DbTester dbTester = DbTester.create();
@Rule
- public EsTester esTester = new EsTester(new RuleIndexDefinition(new MapSettings()));
+ public EsTester esTester = new EsTester(new RuleIndexDefinition(new MapSettings().asConfig()));
@Rule
public ExpectedException thrown = ExpectedException.none();
diff --git a/server/sonar-server/src/test/java/org/sonar/server/rule/ws/SearchActionTest.java b/server/sonar-server/src/test/java/org/sonar/server/rule/ws/SearchActionTest.java
index 9ce85332e56..e7dd9c99e82 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/rule/ws/SearchActionTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/rule/ws/SearchActionTest.java
@@ -55,7 +55,6 @@ import org.sonar.server.qualityprofile.RuleActivation;
import org.sonar.server.qualityprofile.RuleActivator;
import org.sonar.server.qualityprofile.RuleActivatorContextFactory;
import org.sonar.server.qualityprofile.index.ActiveRuleIndexer;
-import org.sonar.server.qualityprofile.index.ActiveRuleIteratorFactory;
import org.sonar.server.rule.index.RuleIndex;
import org.sonar.server.rule.index.RuleIndexDefinition;
import org.sonar.server.rule.index.RuleIndexer;
@@ -106,7 +105,7 @@ public class SearchActionTest {
private DefaultOrganizationProvider defaultOrganizationProvider = TestDefaultOrganizationProvider.from(db);
private RuleIndex ruleIndex = new RuleIndex(es.client());
private RuleIndexer ruleIndexer = new RuleIndexer(es.client(), db.getDbClient());
- private ActiveRuleIndexer activeRuleIndexer = new ActiveRuleIndexer(db.getDbClient(), es.client(), new ActiveRuleIteratorFactory(db.getDbClient()));
+ private ActiveRuleIndexer activeRuleIndexer = new ActiveRuleIndexer(db.getDbClient(), es.client());
private Languages languages = LanguageTesting.newLanguages(JAVA, "js");
private ActiveRuleCompleter activeRuleCompleter = new ActiveRuleCompleter(db.getDbClient(), languages);
private RuleWsSupport wsSupport = new RuleWsSupport(db.getDbClient(), userSession, defaultOrganizationProvider);
diff --git a/server/sonar-server/src/test/java/org/sonar/server/rule/ws/ShowActionTest.java b/server/sonar-server/src/test/java/org/sonar/server/rule/ws/ShowActionTest.java
index 22ff5c7c42a..df4b13920f0 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/rule/ws/ShowActionTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/rule/ws/ShowActionTest.java
@@ -44,7 +44,6 @@ import org.sonar.server.organization.DefaultOrganizationProvider;
import org.sonar.server.organization.TestDefaultOrganizationProvider;
import org.sonar.server.qualityprofile.QProfileTesting;
import org.sonar.server.qualityprofile.index.ActiveRuleIndexer;
-import org.sonar.server.qualityprofile.index.ActiveRuleIteratorFactory;
import org.sonar.server.rule.index.RuleIndexDefinition;
import org.sonar.server.rule.index.RuleIndexer;
import org.sonar.server.text.MacroInterpreter;
@@ -150,7 +149,7 @@ public class ShowActionTest {
.build();
Mockito.doReturn(singletonList(active)).when(activeRuleCompleter).completeShow(any(DbSession.class), orgCaptor.capture(), ruleCaptor.capture());
- ActiveRuleIndexer activeRuleIndexer = new ActiveRuleIndexer(dbClient, esClient, new ActiveRuleIteratorFactory(dbClient));
+ ActiveRuleIndexer activeRuleIndexer = new ActiveRuleIndexer(dbClient, esClient);
activeRuleIndexer.indexOnStartup(activeRuleIndexer.getIndexTypes());
TestResponse response = actionTester.newRequest().setMethod("GET")
@@ -185,7 +184,7 @@ public class ShowActionTest {
RuleMetadataDto ruleMetadata = dbTester.rules().insertOrUpdateMetadata(rule, organization);
dbTester.qualityProfiles().activateRule(profile, rule, a -> a.setSeverity("BLOCKER"));
- ActiveRuleIndexer activeRuleIndexer = new ActiveRuleIndexer(dbClient, esClient, new ActiveRuleIteratorFactory(dbClient));
+ ActiveRuleIndexer activeRuleIndexer = new ActiveRuleIndexer(dbClient, esClient);
activeRuleIndexer.indexOnStartup(activeRuleIndexer.getIndexTypes());
TestResponse response = actionTester.newRequest().setMethod("GET")
diff --git a/tests/resilience/active_rule_indexer.btm b/tests/resilience/active_rule_indexer.btm
new file mode 100644
index 00000000000..331b76d69e7
--- /dev/null
+++ b/tests/resilience/active_rule_indexer.btm
@@ -0,0 +1,12 @@
+# sonar.web.javaAdditionalOpts=-javaagent:/path/to/byteman-3.0.10/lib/byteman.jar=script:/path/to/active_rule_indexer.btm,boot:/path/to/byteman-3.0.10/lib/byteman.jar
+# sonar.search.recovery.delayInMs=10000
+# sonar.search.recovery.minAgeInMs=30000
+
+RULE make indexing of active rules silently fail
+CLASS ActiveRuleIndexer
+METHOD postCommit
+COMPILE
+AT ENTRY
+IF TRUE
+DO RETURN
+ENDRULE
diff --git a/tests/src/test/java/org/sonarqube/tests/Category5Suite.java b/tests/src/test/java/org/sonarqube/tests/Category5Suite.java
index 63b7c1f1628..49bc73e4c44 100644
--- a/tests/src/test/java/org/sonarqube/tests/Category5Suite.java
+++ b/tests/src/test/java/org/sonarqube/tests/Category5Suite.java
@@ -21,6 +21,7 @@ package org.sonarqube.tests;
import org.junit.runner.RunWith;
import org.junit.runners.Suite;
+import org.sonarqube.tests.qualityProfile.ActiveRuleEsResilienceTest;
import org.sonarqube.tests.qualityProfile.BuiltInQualityProfilesNotificationTest;
import org.sonarqube.tests.serverSystem.ClusterTest;
import org.sonarqube.tests.serverSystem.RestartTest;
@@ -51,6 +52,7 @@ import org.sonarqube.tests.user.UserEsResilienceTest;
SsoAuthenticationTest.class,
OnboardingTest.class,
BuiltInQualityProfilesNotificationTest.class,
+ ActiveRuleEsResilienceTest.class,
UserEsResilienceTest.class
})
public class Category5Suite {
diff --git a/tests/src/test/java/org/sonarqube/tests/qualityProfile/ActiveRuleEsResilienceTest.java b/tests/src/test/java/org/sonarqube/tests/qualityProfile/ActiveRuleEsResilienceTest.java
new file mode 100644
index 00000000000..6203a647054
--- /dev/null
+++ b/tests/src/test/java/org/sonarqube/tests/qualityProfile/ActiveRuleEsResilienceTest.java
@@ -0,0 +1,107 @@
+/*
+ * 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 org.sonarqube.tests.qualityProfile;
+
+import com.sonar.orchestrator.Orchestrator;
+import java.io.File;
+import java.util.concurrent.TimeUnit;
+import org.junit.ClassRule;
+import org.junit.Rule;
+import org.junit.Test;
+import org.junit.rules.DisableOnDebug;
+import org.junit.rules.TestRule;
+import org.junit.rules.Timeout;
+import org.sonarqube.tests.Tester;
+import org.sonarqube.ws.Organizations;
+import org.sonarqube.ws.QualityProfiles;
+import org.sonarqube.ws.client.rule.SearchWsRequest;
+import util.ItUtils;
+
+import static java.lang.String.format;
+import static org.assertj.core.api.Assertions.assertThat;
+
+public class ActiveRuleEsResilienceTest {
+ private static final String RULE_ONE_BUG_PER_LINE = "xoo:OneBugIssuePerLine";
+
+ @ClassRule
+ public static final Orchestrator orchestrator = Orchestrator.builderEnv()
+ .setServerProperty("sonar.web.javaAdditionalOpts",
+ format("-javaagent:%s=script:%s,boot:%s", findBytemanJar(), findBytemanScript(), findBytemanJar()))
+ .setServerProperty("sonar.search.recovery.delayInMs", "500")
+ .setServerProperty("sonar.search.recovery.minAgeInMs", "3000")
+ .addPlugin(ItUtils.xooPlugin())
+ .build();
+
+ @Rule
+ public TestRule timeout = new DisableOnDebug(Timeout.builder()
+ .withLookingForStuckThread(true)
+ .withTimeout(60L, TimeUnit.SECONDS)
+ .build());
+
+ @Rule
+ public Tester tester = new Tester(orchestrator);
+
+ @Test
+ public void activation_and_deactivation_of_rule_is_resilient_to_indexing_errors() throws Exception {
+ Organizations.Organization organization = tester.organizations().generate();
+ QualityProfiles.CreateWsResponse.QualityProfile profile = tester.qProfiles().createXooProfile(organization);
+
+ // step 1. activation
+ tester.qProfiles().activateRule(profile.getKey(), RULE_ONE_BUG_PER_LINE);
+
+ assertThat(searchActiveRules(profile)).isEqualTo(0);
+ while (searchActiveRules(profile) == 0) {
+ // rule is indexed by the recovery daemon, which runs every 3 seconds
+ Thread.sleep(500L);
+ }
+ assertThat(searchActiveRules(profile)).isEqualTo(1);
+
+ // step 2. deactivation
+ tester.qProfiles().deactivateRule(profile, RULE_ONE_BUG_PER_LINE);
+ while (searchActiveRules(profile) == 1) {
+ // rule is indexed by the recovery daemon, which runs every 3 seconds
+ Thread.sleep(500L);
+ }
+ assertThat(searchActiveRules(profile)).isEqualTo(0);
+ }
+
+ private long searchActiveRules(QualityProfiles.CreateWsResponse.QualityProfile profile) {
+ SearchWsRequest request = new SearchWsRequest().setActivation(true).setQProfile(profile.getKey());
+ return tester.wsClient().rules().search(request).getRulesCount();
+ }
+
+ private static String findBytemanJar() {
+ // see pom.xml, Maven copies and renames the artifact.
+ File jar = new File("target/byteman.jar");
+ if (!jar.exists()) {
+ throw new IllegalStateException("Can't find " + jar + ". Please execute 'mvn generate-test-resources' on integration tests once.");
+ }
+ return jar.getAbsolutePath();
+ }
+
+ private static String findBytemanScript() {
+ // see pom.xml, Maven copies and renames the artifact.
+ File script = new File("resilience/active_rule_indexer.btm");
+ if (!script.exists()) {
+ throw new IllegalStateException("Can't find " + script);
+ }
+ return script.getAbsolutePath();
+ }
+}