aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/computation/source/DuplicationLineReader.java7
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/computation/step/PersistDuplicationsStep.java16
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/computation/source/DuplicationLineReaderTest.java28
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/computation/step/PersistDuplicationsStepTest.java29
-rw-r--r--sonar-batch-protocol/src/main/protobuf/batch_report.proto14
-rw-r--r--sonar-batch-protocol/src/test/java/org/sonar/batch/protocol/output/BatchReportReaderTest.java2
-rw-r--r--sonar-batch-protocol/src/test/java/org/sonar/batch/protocol/output/BatchReportWriterTest.java3
-rw-r--r--sonar-batch/src/main/java/org/sonar/batch/analysis/DefaultAnalysisMode.java12
-rw-r--r--sonar-batch/src/main/java/org/sonar/batch/cpd/CpdComponents.java4
-rw-r--r--sonar-batch/src/main/java/org/sonar/batch/cpd/DefaultCpdEngine.java52
-rw-r--r--sonar-batch/src/main/java/org/sonar/batch/cpd/JavaCpdEngine.java80
-rw-r--r--sonar-batch/src/main/java/org/sonar/batch/cpd/index/IndexFactory.java71
-rw-r--r--sonar-batch/src/main/java/org/sonar/batch/cpd/index/SonarDuplicationsIndex.java42
-rw-r--r--sonar-batch/src/main/java/org/sonar/batch/mediumtest/TaskResult.java45
-rw-r--r--sonar-batch/src/main/java/org/sonar/batch/report/DuplicationsPublisher.java3
-rw-r--r--sonar-batch/src/main/java/org/sonar/batch/report/MetadataPublisher.java7
-rw-r--r--sonar-batch/src/main/java/org/sonar/batch/sensor/DefaultSensorStorage.java14
-rw-r--r--sonar-batch/src/test/java/org/sonar/batch/cpd/CpdSensorTest.java10
-rw-r--r--sonar-batch/src/test/java/org/sonar/batch/cpd/DefaultCpdEngineTest.java14
-rw-r--r--sonar-batch/src/test/java/org/sonar/batch/cpd/index/IndexFactoryTest.java82
-rw-r--r--sonar-batch/src/test/java/org/sonar/batch/mediumtest/cpd/CpdMediumTest.java97
-rw-r--r--sonar-batch/src/test/java/org/sonar/batch/report/DuplicationsPublisherTest.java17
-rw-r--r--sonar-batch/src/test/java/org/sonar/batch/report/MetadataPublisherTest.java19
-rw-r--r--sonar-core/src/main/java/org/sonar/core/config/CorePropertyDefinitions.java57
-rw-r--r--sonar-plugin-api/src/main/java/org/sonar/api/CoreProperties.java10
25 files changed, 275 insertions, 460 deletions
diff --git a/server/sonar-server/src/main/java/org/sonar/server/computation/source/DuplicationLineReader.java b/server/sonar-server/src/main/java/org/sonar/server/computation/source/DuplicationLineReader.java
index 863bdc4c4b0..3c1773d1c86 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/computation/source/DuplicationLineReader.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/computation/source/DuplicationLineReader.java
@@ -20,15 +20,14 @@
package org.sonar.server.computation.source;
-import org.sonar.batch.protocol.output.BatchReport;
-import org.sonar.db.protobuf.DbFileSources;
-
import java.io.Serializable;
import java.util.Collections;
import java.util.Comparator;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
+import org.sonar.batch.protocol.output.BatchReport;
+import org.sonar.db.protobuf.DbFileSources;
import static com.google.common.collect.Lists.newArrayList;
import static com.google.common.collect.Maps.newHashMap;
@@ -72,7 +71,7 @@ public class DuplicationLineReader implements LineReader {
}
private static boolean isDuplicationOnSameFile(BatchReport.Duplicate duplicate) {
- return !duplicate.hasOtherFileKey() && !duplicate.hasOtherFileRef();
+ return !duplicate.hasOtherFileRef();
}
private static boolean matchLine(BatchReport.TextRange range, int line) {
diff --git a/server/sonar-server/src/main/java/org/sonar/server/computation/step/PersistDuplicationsStep.java b/server/sonar-server/src/main/java/org/sonar/server/computation/step/PersistDuplicationsStep.java
index 8f0daa93d6d..1eae2e88f52 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/computation/step/PersistDuplicationsStep.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/computation/step/PersistDuplicationsStep.java
@@ -121,18 +121,12 @@ public class PersistDuplicationsStep implements ComputationStep {
}
private void processDuplicationBlock(StringBuilder xml, BatchReport.Duplicate duplicate, String componentKey) {
- if (duplicate.hasOtherFileKey()) {
- // componentKey is only set for cross project duplications
- String crossProjectComponentKey = duplicate.getOtherFileKey();
- appendDuplication(xml, crossProjectComponentKey, duplicate);
+ if (duplicate.hasOtherFileRef()) {
+ // Duplication is on a different file
+ appendDuplication(xml, treeRootHolder.getComponentByRef(duplicate.getOtherFileRef()).getKey(), duplicate);
} else {
- if (duplicate.hasOtherFileRef()) {
- // Duplication is on a different file
- appendDuplication(xml, treeRootHolder.getComponentByRef(duplicate.getOtherFileRef()).getKey(), duplicate);
- } else {
- // Duplication is on a the same file
- appendDuplication(xml, componentKey, duplicate);
- }
+ // Duplication is on a the same file
+ appendDuplication(xml, componentKey, duplicate);
}
}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/computation/source/DuplicationLineReaderTest.java b/server/sonar-server/src/test/java/org/sonar/server/computation/source/DuplicationLineReaderTest.java
index c7bf7c127fe..44289f855ea 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/computation/source/DuplicationLineReaderTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/computation/source/DuplicationLineReaderTest.java
@@ -101,34 +101,6 @@ public class DuplicationLineReaderTest {
}
@Test
- public void read_duplication_with_duplicates_on_other_file_from_other_project() {
- DuplicationLineReader reader = new DuplicationLineReader(Iterators.forArray(
- BatchReport.Duplication.newBuilder()
- .setOriginPosition(BatchReport.TextRange.newBuilder()
- .setStartLine(1)
- .setEndLine(2)
- .build())
- .addDuplicate(BatchReport.Duplicate.newBuilder()
- .setOtherFileKey("other-component-key-from-another-project")
- .setRange(BatchReport.TextRange.newBuilder()
- .setStartLine(3)
- .setEndLine(4)
- .build())
- .build())
- .build()));
-
- reader.read(line1);
- reader.read(line2);
- reader.read(line3);
- reader.read(line4);
-
- assertThat(line1.getDuplicationList()).containsExactly(1);
- assertThat(line2.getDuplicationList()).containsExactly(1);
- assertThat(line3.getDuplicationList()).isEmpty();
- assertThat(line4.getDuplicationList()).isEmpty();
- }
-
- @Test
public void read_many_duplications() {
DuplicationLineReader reader = new DuplicationLineReader(Iterators.forArray(
BatchReport.Duplication.newBuilder()
diff --git a/server/sonar-server/src/test/java/org/sonar/server/computation/step/PersistDuplicationsStepTest.java b/server/sonar-server/src/test/java/org/sonar/server/computation/step/PersistDuplicationsStepTest.java
index 2a729e87cc8..40611e6367a 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/computation/step/PersistDuplicationsStepTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/computation/step/PersistDuplicationsStepTest.java
@@ -279,35 +279,6 @@ public class PersistDuplicationsStepTest extends BaseStepTest {
assertThat(dto.get("textValue")).isEqualTo("<duplications><g><b s=\"1\" l=\"5\" r=\"PROJECT_KEY:file\"/><b s=\"6\" l=\"5\" r=\"PROJECT_KEY:file2\"/></g></duplications>");
}
- @Test
- public void persist_duplications_on_different_projects() {
- saveDuplicationMetric();
- initReportWithProjectAndFile();
-
- BatchReport.Duplication duplication = BatchReport.Duplication.newBuilder()
- .setOriginPosition(BatchReport.TextRange.newBuilder()
- .setStartLine(1)
- .setEndLine(5)
- .build())
- .addDuplicate(BatchReport.Duplicate.newBuilder()
- .setOtherFileKey("PROJECT2_KEY:file2")
- .setRange(BatchReport.TextRange.newBuilder()
- .setStartLine(6)
- .setEndLine(10)
- .build())
- .build())
- .build();
- reportReader.putDuplications(2, newArrayList(duplication));
-
- underTest.execute();
-
- assertThat(dbTester.countRowsOfTable("project_measures")).isEqualTo(1);
-
- Map<String, Object> dto = dbTester.selectFirst("select snapshot_id as \"snapshotId\", text_value as \"textValue\" from project_measures");
- assertThat(dto.get("snapshotId")).isEqualTo(11L);
- assertThat(dto.get("textValue")).isEqualTo("<duplications><g><b s=\"1\" l=\"5\" r=\"PROJECT_KEY:file\"/><b s=\"6\" l=\"5\" r=\"PROJECT2_KEY:file2\"/></g></duplications>");
- }
-
private void initReportWithProjectAndFile() {
Component file = ReportComponent.builder(Component.Type.FILE, 2).setUuid("BCDE").setKey("PROJECT_KEY:file").build();
Component project = ReportComponent.builder(Component.Type.PROJECT, 1).setUuid("ABCD").setKey(PROJECT_KEY).addChildren(file).build();
diff --git a/sonar-batch-protocol/src/main/protobuf/batch_report.proto b/sonar-batch-protocol/src/main/protobuf/batch_report.proto
index bd5b8727821..7937353fe9f 100644
--- a/sonar-batch-protocol/src/main/protobuf/batch_report.proto
+++ b/sonar-batch-protocol/src/main/protobuf/batch_report.proto
@@ -134,9 +134,6 @@ message Duplicate {
// Will be null when duplicate is in the same file
optional int32 other_file_ref = 1;
optional TextRange range = 2;
-
- // temporary field during development of computation stack for cross project duplications
- optional string other_file_key = 3;
}
message Duplication {
@@ -147,12 +144,11 @@ message Duplication {
// Used for cross project duplication
message DuplicationBlock {
- repeated int32 hash = 1;
- optional int32 index_in_file = 2;
- optional int32 start_line = 3;
- optional int32 end_line = 4;
- optional int32 start_token_index = 5;
- optional int32 end_token_index = 6;
+ repeated int32 hash = 1 [packed = true];
+ optional int32 start_line = 2;
+ optional int32 end_line = 3;
+ optional int32 start_token_index = 4;
+ optional int32 end_token_index = 5;
}
// Lines start at 1 and line offsets start at 0
diff --git a/sonar-batch-protocol/src/test/java/org/sonar/batch/protocol/output/BatchReportReaderTest.java b/sonar-batch-protocol/src/test/java/org/sonar/batch/protocol/output/BatchReportReaderTest.java
index 50dfd49d6c3..3c2c5ac5da9 100644
--- a/sonar-batch-protocol/src/test/java/org/sonar/batch/protocol/output/BatchReportReaderTest.java
+++ b/sonar-batch-protocol/src/test/java/org/sonar/batch/protocol/output/BatchReportReaderTest.java
@@ -154,7 +154,6 @@ public class BatchReportReaderTest {
.setEndLine(5)
.build())
.addDuplicate(BatchReport.Duplicate.newBuilder()
- .setOtherFileKey("COMPONENT_A")
.setOtherFileRef(2)
.setRange(BatchReport.TextRange.newBuilder()
.setStartLine(6)
@@ -182,7 +181,6 @@ public class BatchReportReaderTest {
.setRef(1).build());
BatchReport.DuplicationBlock duplicationBlock = BatchReport.DuplicationBlock.newBuilder()
- .setIndexInFile(1)
.addAllHash(asList(1, 2, 3, 5, 7))
.setStartLine(1)
.setEndLine(2)
diff --git a/sonar-batch-protocol/src/test/java/org/sonar/batch/protocol/output/BatchReportWriterTest.java b/sonar-batch-protocol/src/test/java/org/sonar/batch/protocol/output/BatchReportWriterTest.java
index eda7000899b..fdd1ba2b67b 100644
--- a/sonar-batch-protocol/src/test/java/org/sonar/batch/protocol/output/BatchReportWriterTest.java
+++ b/sonar-batch-protocol/src/test/java/org/sonar/batch/protocol/output/BatchReportWriterTest.java
@@ -170,7 +170,6 @@ public class BatchReportWriterTest {
.setEndLine(5)
.build())
.addDuplicate(BatchReport.Duplicate.newBuilder()
- .setOtherFileKey("COMPONENT_A")
.setOtherFileRef(2)
.setRange(BatchReport.TextRange.newBuilder()
.setStartLine(6)
@@ -195,7 +194,6 @@ public class BatchReportWriterTest {
assertThat(underTest.hasComponentData(FileStructure.Domain.DUPLICATION_BLOCKS, 1)).isFalse();
BatchReport.DuplicationBlock duplicationBlock = BatchReport.DuplicationBlock.newBuilder()
- .setIndexInFile(1)
.addAllHash(asList(1, 2, 3, 5, 7))
.setStartLine(1)
.setEndLine(2)
@@ -209,7 +207,6 @@ public class BatchReportWriterTest {
assertThat(file).exists().isFile();
try (CloseableIterator<BatchReport.DuplicationBlock> duplicationBlocks = Protobuf.readStream(file, BatchReport.DuplicationBlock.parser())) {
BatchReport.DuplicationBlock duplicationBlockResult = duplicationBlocks.next();
- assertThat(duplicationBlockResult.getIndexInFile()).isEqualTo(1);
assertThat(duplicationBlockResult.getHashList()).containsOnly(1, 2, 3, 5, 7);
assertThat(duplicationBlockResult.getStartLine()).isEqualTo(1);
assertThat(duplicationBlockResult.getEndLine()).isEqualTo(2);
diff --git a/sonar-batch/src/main/java/org/sonar/batch/analysis/DefaultAnalysisMode.java b/sonar-batch/src/main/java/org/sonar/batch/analysis/DefaultAnalysisMode.java
index 31784aa1e80..34df18afaa8 100644
--- a/sonar-batch/src/main/java/org/sonar/batch/analysis/DefaultAnalysisMode.java
+++ b/sonar-batch/src/main/java/org/sonar/batch/analysis/DefaultAnalysisMode.java
@@ -19,16 +19,15 @@
*/
package org.sonar.batch.analysis;
-import org.sonar.batch.bootstrap.AbstractAnalysisMode;
-
-import org.sonar.batch.mediumtest.FakePluginInstaller;
-import org.sonar.batch.bootstrap.GlobalProperties;
+import java.util.Map;
+import javax.annotation.CheckForNull;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.sonar.api.CoreProperties;
import org.sonar.api.batch.AnalysisMode;
-
-import java.util.Map;
+import org.sonar.batch.bootstrap.AbstractAnalysisMode;
+import org.sonar.batch.bootstrap.GlobalProperties;
+import org.sonar.batch.mediumtest.FakePluginInstaller;
/**
* @since 4.0
@@ -99,6 +98,7 @@ public class DefaultAnalysisMode extends AbstractAnalysisMode implements Analysi
}
}
+ @CheckForNull
private static String getPropertyWithFallback(Map<String, String> props1, Map<String, String> props2, String key) {
if (props1.containsKey(key)) {
return props1.get(key);
diff --git a/sonar-batch/src/main/java/org/sonar/batch/cpd/CpdComponents.java b/sonar-batch/src/main/java/org/sonar/batch/cpd/CpdComponents.java
index e2c5555d4e2..ffac2cba012 100644
--- a/sonar-batch/src/main/java/org/sonar/batch/cpd/CpdComponents.java
+++ b/sonar-batch/src/main/java/org/sonar/batch/cpd/CpdComponents.java
@@ -20,11 +20,8 @@
package org.sonar.batch.cpd;
import com.google.common.collect.ImmutableList;
-
import java.util.List;
-import org.sonar.batch.cpd.index.IndexFactory;
-
public final class CpdComponents {
private CpdComponents() {
@@ -34,7 +31,6 @@ public final class CpdComponents {
return ImmutableList.of(
CpdSensor.class,
CpdMappings.class,
- IndexFactory.class,
JavaCpdEngine.class,
DefaultCpdEngine.class);
}
diff --git a/sonar-batch/src/main/java/org/sonar/batch/cpd/DefaultCpdEngine.java b/sonar-batch/src/main/java/org/sonar/batch/cpd/DefaultCpdEngine.java
index 93fc1726b46..a63b04f0006 100644
--- a/sonar-batch/src/main/java/org/sonar/batch/cpd/DefaultCpdEngine.java
+++ b/sonar-batch/src/main/java/org/sonar/batch/cpd/DefaultCpdEngine.java
@@ -23,6 +23,13 @@ import com.google.common.annotations.VisibleForTesting;
import com.google.common.base.Predicate;
import com.google.common.collect.Iterables;
import com.google.common.collect.Lists;
+import java.util.Collection;
+import java.util.List;
+import java.util.concurrent.ExecutionException;
+import java.util.concurrent.ExecutorService;
+import java.util.concurrent.Executors;
+import java.util.concurrent.TimeUnit;
+import java.util.concurrent.TimeoutException;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.sonar.api.CoreProperties;
@@ -33,24 +40,13 @@ import org.sonar.api.batch.fs.InputFile;
import org.sonar.api.batch.fs.internal.DefaultInputFile;
import org.sonar.api.batch.sensor.SensorContext;
import org.sonar.api.config.Settings;
-import org.sonar.api.resources.Project;
-import org.sonar.api.utils.SonarException;
-import org.sonar.batch.cpd.index.IndexFactory;
import org.sonar.batch.cpd.index.SonarDuplicationsIndex;
+import org.sonar.batch.index.BatchComponentCache;
+import org.sonar.batch.report.ReportPublisher;
import org.sonar.duplications.block.Block;
import org.sonar.duplications.index.CloneGroup;
import org.sonar.duplications.internal.pmd.TokenizerBridge;
-import javax.annotation.Nullable;
-
-import java.util.Collection;
-import java.util.List;
-import java.util.concurrent.ExecutionException;
-import java.util.concurrent.ExecutorService;
-import java.util.concurrent.Executors;
-import java.util.concurrent.TimeUnit;
-import java.util.concurrent.TimeoutException;
-
public class DefaultCpdEngine extends CpdEngine {
private static final Logger LOG = LoggerFactory.getLogger(DefaultCpdEngine.class);
@@ -60,22 +56,18 @@ public class DefaultCpdEngine extends CpdEngine {
*/
private static final int TIMEOUT = 5 * 60;
- private final IndexFactory indexFactory;
private final CpdMappings mappings;
private final FileSystem fs;
private final Settings settings;
- private final Project project;
+ private final ReportPublisher publisher;
+ private final BatchComponentCache batchComponentCache;
- public DefaultCpdEngine(@Nullable Project project, IndexFactory indexFactory, CpdMappings mappings, FileSystem fs, Settings settings) {
- this.project = project;
- this.indexFactory = indexFactory;
+ public DefaultCpdEngine(CpdMappings mappings, FileSystem fs, Settings settings, ReportPublisher publisher, BatchComponentCache batchComponentCache) {
this.mappings = mappings;
this.fs = fs;
this.settings = settings;
- }
-
- public DefaultCpdEngine(IndexFactory indexFactory, CpdMappings mappings, FileSystem fs, Settings settings) {
- this(null, indexFactory, mappings, fs, settings);
+ this.publisher = publisher;
+ this.batchComponentCache = batchComponentCache;
}
@Override
@@ -97,14 +89,13 @@ public class DefaultCpdEngine extends CpdEngine {
List<InputFile> sourceFiles = Lists.newArrayList(fs.inputFiles(p.and(
p.hasType(InputFile.Type.MAIN),
p.hasLanguage(languageKey),
- p.doesNotMatchPathPatterns(cpdExclusions)
- )));
+ p.doesNotMatchPathPatterns(cpdExclusions))));
if (sourceFiles.isEmpty()) {
return;
}
// Create index
- SonarDuplicationsIndex index = indexFactory.create(project, languageKey);
+ SonarDuplicationsIndex index = new SonarDuplicationsIndex(publisher, batchComponentCache, settings);
populateIndex(languageKey, sourceFiles, mapping, index);
// Detect
@@ -129,7 +120,7 @@ public class DefaultCpdEngine extends CpdEngine {
filtered = null;
LOG.warn("Timeout during detection of duplications for " + inputFile, e);
} catch (InterruptedException | ExecutionException e) {
- throw new SonarException("Fail during detection of duplication for " + inputFile, e);
+ throw new IllegalStateException("Fail during detection of duplication for " + inputFile, e);
}
JavaCpdEngine.save(context, inputFile, filtered);
@@ -144,8 +135,8 @@ public class DefaultCpdEngine extends CpdEngine {
for (InputFile inputFile : sourceFiles) {
LOG.debug("Populating index from {}", inputFile);
String resourceEffectiveKey = ((DefaultInputFile) inputFile).key();
- List<Block> blocks2 = bridge.chunk(resourceEffectiveKey, inputFile.file());
- index.insert(inputFile, blocks2);
+ List<Block> blocks = bridge.chunk(resourceEffectiveKey, inputFile.file());
+ index.insert(inputFile, blocks);
}
}
@@ -162,7 +153,7 @@ public class DefaultCpdEngine extends CpdEngine {
static int getDefaultBlockSize(String languageKey) {
if ("cobol".equals(languageKey)) {
return 30;
- } else if ("abap".equals(languageKey) || "natur".equals(languageKey)) {
+ } else if ("abap".equals(languageKey)) {
return 20;
} else {
return 10;
@@ -173,9 +164,6 @@ public class DefaultCpdEngine extends CpdEngine {
int getMinimumTokens(String languageKey) {
int minimumTokens = settings.getInt("sonar.cpd." + languageKey + ".minimumTokens");
if (minimumTokens == 0) {
- minimumTokens = settings.getInt(CoreProperties.CPD_MINIMUM_TOKENS_PROPERTY);
- }
- if (minimumTokens == 0) {
minimumTokens = 100;
}
diff --git a/sonar-batch/src/main/java/org/sonar/batch/cpd/JavaCpdEngine.java b/sonar-batch/src/main/java/org/sonar/batch/cpd/JavaCpdEngine.java
index 97059241af5..1adca4f11ca 100644
--- a/sonar-batch/src/main/java/org/sonar/batch/cpd/JavaCpdEngine.java
+++ b/sonar-batch/src/main/java/org/sonar/batch/cpd/JavaCpdEngine.java
@@ -22,6 +22,21 @@ package org.sonar.batch.cpd;
import com.google.common.collect.Iterables;
import com.google.common.collect.Lists;
+import java.io.FileInputStream;
+import java.io.FileNotFoundException;
+import java.io.InputStreamReader;
+import java.io.Reader;
+import java.util.Collection;
+import java.util.HashSet;
+import java.util.List;
+import java.util.Set;
+import java.util.concurrent.Callable;
+import java.util.concurrent.ExecutionException;
+import java.util.concurrent.ExecutorService;
+import java.util.concurrent.Executors;
+import java.util.concurrent.TimeUnit;
+import java.util.concurrent.TimeoutException;
+import javax.annotation.Nullable;
import org.apache.commons.io.IOUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
@@ -36,10 +51,9 @@ import org.sonar.api.batch.sensor.duplication.internal.DefaultDuplication;
import org.sonar.api.batch.sensor.measure.internal.DefaultMeasure;
import org.sonar.api.config.Settings;
import org.sonar.api.measures.CoreMetrics;
-import org.sonar.api.resources.Project;
-import org.sonar.api.utils.SonarException;
-import org.sonar.batch.cpd.index.IndexFactory;
import org.sonar.batch.cpd.index.SonarDuplicationsIndex;
+import org.sonar.batch.index.BatchComponentCache;
+import org.sonar.batch.report.ReportPublisher;
import org.sonar.duplications.block.Block;
import org.sonar.duplications.block.BlockChunker;
import org.sonar.duplications.detector.suffixtree.SuffixTreeCloneDetectionAlgorithm;
@@ -52,23 +66,6 @@ import org.sonar.duplications.statement.Statement;
import org.sonar.duplications.statement.StatementChunker;
import org.sonar.duplications.token.TokenChunker;
-import javax.annotation.Nullable;
-
-import java.io.FileInputStream;
-import java.io.FileNotFoundException;
-import java.io.InputStreamReader;
-import java.io.Reader;
-import java.util.Collection;
-import java.util.HashSet;
-import java.util.List;
-import java.util.Set;
-import java.util.concurrent.Callable;
-import java.util.concurrent.ExecutionException;
-import java.util.concurrent.ExecutorService;
-import java.util.concurrent.Executors;
-import java.util.concurrent.TimeUnit;
-import java.util.concurrent.TimeoutException;
-
public class JavaCpdEngine extends CpdEngine {
private static final Logger LOG = LoggerFactory.getLogger(JavaCpdEngine.class);
@@ -83,20 +80,16 @@ public class JavaCpdEngine extends CpdEngine {
private static final int MAX_CLONE_GROUP_PER_FILE = 100;
private static final int MAX_CLONE_PART_PER_GROUP = 100;
- private final IndexFactory indexFactory;
private final FileSystem fs;
private final Settings settings;
- private final Project project;
+ private final ReportPublisher publisher;
+ private final BatchComponentCache batchComponentCache;
- public JavaCpdEngine(@Nullable Project project, IndexFactory indexFactory, FileSystem fs, Settings settings) {
- this.project = project;
- this.indexFactory = indexFactory;
+ public JavaCpdEngine(FileSystem fs, Settings settings, ReportPublisher publisher, BatchComponentCache batchComponentCache) {
this.fs = fs;
this.settings = settings;
- }
-
- public JavaCpdEngine(IndexFactory indexFactory, FileSystem fs, Settings settings) {
- this(null, indexFactory, fs, settings);
+ this.publisher = publisher;
+ this.batchComponentCache = batchComponentCache;
}
@Override
@@ -112,17 +105,16 @@ public class JavaCpdEngine extends CpdEngine {
List<InputFile> sourceFiles = Lists.newArrayList(fs.inputFiles(p.and(
p.hasType(InputFile.Type.MAIN),
p.hasLanguage(languageKey),
- p.doesNotMatchPathPatterns(cpdExclusions)
- )));
+ p.doesNotMatchPathPatterns(cpdExclusions))));
if (sourceFiles.isEmpty()) {
return;
}
- SonarDuplicationsIndex index = createIndex(project, languageKey, sourceFiles);
+ SonarDuplicationsIndex index = createIndex(sourceFiles);
detect(index, context, sourceFiles);
}
- private SonarDuplicationsIndex createIndex(@Nullable Project project, String language, Iterable<InputFile> sourceFiles) {
- final SonarDuplicationsIndex index = indexFactory.create(project, language);
+ private SonarDuplicationsIndex createIndex(Iterable<InputFile> sourceFiles) {
+ final SonarDuplicationsIndex index = new SonarDuplicationsIndex(publisher, batchComponentCache, settings);
TokenChunker tokenChunker = JavaTokenProducer.build();
StatementChunker statementChunker = JavaStatementBuilder.build();
@@ -139,7 +131,7 @@ public class JavaCpdEngine extends CpdEngine {
reader = new InputStreamReader(new FileInputStream(inputFile.file()), fs.encoding());
statements = statementChunker.chunk(tokenChunker.chunk(reader));
} catch (FileNotFoundException e) {
- throw new SonarException("Cannot find file " + inputFile.file(), e);
+ throw new IllegalStateException("Cannot find file " + inputFile.file(), e);
} finally {
IOUtils.closeQuietly(reader);
}
@@ -166,10 +158,8 @@ public class JavaCpdEngine extends CpdEngine {
} catch (TimeoutException e) {
clones = null;
LOG.warn("Timeout during detection of duplications for " + inputFile, e);
- } catch (InterruptedException e) {
- throw new SonarException("Fail during detection of duplication for " + inputFile, e);
- } catch (ExecutionException e) {
- throw new SonarException("Fail during detection of duplication for " + inputFile, e);
+ } catch (InterruptedException | ExecutionException e) {
+ throw new IllegalStateException("Fail during detection of duplication for " + inputFile, e);
}
save(context, inputFile, clones);
@@ -209,20 +199,20 @@ public class JavaCpdEngine extends CpdEngine {
.forMetric(CoreMetrics.DUPLICATED_FILES)
.on(inputFile)
.withValue(1))
- .setFromCore()
- .save();
+ .setFromCore()
+ .save();
((DefaultMeasure<Integer>) context.<Integer>newMeasure()
.forMetric(CoreMetrics.DUPLICATED_LINES)
.on(inputFile)
.withValue(duplicatedLines))
- .setFromCore()
- .save();
+ .setFromCore()
+ .save();
((DefaultMeasure<Integer>) context.<Integer>newMeasure()
.forMetric(CoreMetrics.DUPLICATED_BLOCKS)
.on(inputFile)
.withValue(duplicatedBlocks))
- .setFromCore()
- .save();
+ .setFromCore()
+ .save();
}
private static void saveDuplications(org.sonar.api.batch.sensor.SensorContext context, InputFile inputFile, Iterable<CloneGroup> duplications) {
diff --git a/sonar-batch/src/main/java/org/sonar/batch/cpd/index/IndexFactory.java b/sonar-batch/src/main/java/org/sonar/batch/cpd/index/IndexFactory.java
deleted file mode 100644
index 31a4b03ce55..00000000000
--- a/sonar-batch/src/main/java/org/sonar/batch/cpd/index/IndexFactory.java
+++ /dev/null
@@ -1,71 +0,0 @@
-/*
- * SonarQube, open source software quality management tool.
- * Copyright (C) 2008-2014 SonarSource
- * mailto:contact AT sonarsource DOT com
- *
- * SonarQube 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.
- *
- * SonarQube 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.batch.cpd.index;
-
-import org.sonar.batch.analysis.DefaultAnalysisMode;
-
-import com.google.common.annotations.VisibleForTesting;
-
-import javax.annotation.Nullable;
-
-import org.apache.commons.lang.StringUtils;
-import org.slf4j.Logger;
-import org.sonar.api.CoreProperties;
-import org.sonar.api.batch.BatchSide;
-import org.sonar.api.config.Settings;
-import org.sonar.api.resources.Project;
-
-@BatchSide
-public class IndexFactory {
-
- private final Settings settings;
- private final DefaultAnalysisMode mode;
-
- public IndexFactory(DefaultAnalysisMode mode, Settings settings) {
- this.mode = mode;
- this.settings = settings;
- }
-
- public SonarDuplicationsIndex create(@Nullable Project project, String languageKey) {
- return new SonarDuplicationsIndex();
- }
-
- @VisibleForTesting
- boolean verifyCrossProject(@Nullable Project project, Logger logger) {
- boolean crossProject = false;
-
- if (settings.getBoolean(CoreProperties.CPD_CROSS_PROJECT)) {
- if (mode.isIssues()) {
- logger.info("Cross-project analysis disabled. Not supported in issues mode.");
- } else if (StringUtils.isNotBlank(settings.getString(CoreProperties.PROJECT_BRANCH_PROPERTY))) {
- logger.info("Cross-project analysis disabled. Not supported on project branches.");
- } else if (project == null) {
- // New sensor mode
- logger.info("Cross-project analysis disabled. Not supported in new sensor mode.");
- } else {
- logger.info("Cross-project analysis enabled");
- crossProject = true;
- }
- } else {
- logger.info("Cross-project analysis disabled");
- }
- return crossProject;
- }
-}
diff --git a/sonar-batch/src/main/java/org/sonar/batch/cpd/index/SonarDuplicationsIndex.java b/sonar-batch/src/main/java/org/sonar/batch/cpd/index/SonarDuplicationsIndex.java
index ab583d672dd..51a094a869e 100644
--- a/sonar-batch/src/main/java/org/sonar/batch/cpd/index/SonarDuplicationsIndex.java
+++ b/sonar-batch/src/main/java/org/sonar/batch/cpd/index/SonarDuplicationsIndex.java
@@ -19,8 +19,17 @@
*/
package org.sonar.batch.cpd.index;
+import com.google.common.base.Function;
+import com.google.common.collect.Iterables;
import java.util.Collection;
+import org.apache.commons.lang.StringUtils;
+import org.sonar.api.CoreProperties;
import org.sonar.api.batch.fs.InputFile;
+import org.sonar.api.config.Settings;
+import org.sonar.batch.index.BatchComponentCache;
+import org.sonar.batch.protocol.output.BatchReport;
+import org.sonar.batch.protocol.output.BatchReport.DuplicationBlock;
+import org.sonar.batch.report.ReportPublisher;
import org.sonar.duplications.block.Block;
import org.sonar.duplications.block.ByteArray;
import org.sonar.duplications.index.AbstractCloneIndex;
@@ -30,13 +39,46 @@ import org.sonar.duplications.index.PackedMemoryCloneIndex;
public class SonarDuplicationsIndex extends AbstractCloneIndex {
private final CloneIndex mem = new PackedMemoryCloneIndex();
+ private final ReportPublisher publisher;
+ private final BatchComponentCache batchComponentCache;
+ private final Settings settings;
+
+ public SonarDuplicationsIndex(ReportPublisher publisher, BatchComponentCache batchComponentCache, Settings settings) {
+ this.publisher = publisher;
+ this.batchComponentCache = batchComponentCache;
+ this.settings = settings;
+ }
public void insert(InputFile inputFile, Collection<Block> blocks) {
+ if (isCrossProjectDuplicationEnabled(settings)) {
+ int id = batchComponentCache.get(inputFile).batchId();
+ final BatchReport.DuplicationBlock.Builder builder = BatchReport.DuplicationBlock.newBuilder();
+ publisher.getWriter().writeDuplicationBlocks(id, Iterables.transform(blocks, new Function<Block, BatchReport.DuplicationBlock>() {
+ @Override
+ public DuplicationBlock apply(Block input) {
+ builder.clear();
+ builder.setStartLine(input.getStartLine());
+ builder.setEndLine(input.getEndLine());
+ builder.setStartTokenIndex(input.getStartUnit());
+ builder.setEndTokenIndex(input.getEndUnit());
+ for (int i : input.getBlockHash().toIntArray()) {
+ builder.addHash(i);
+ }
+ return builder.build();
+ }
+ }));
+ }
for (Block block : blocks) {
mem.insert(block);
}
}
+ public static boolean isCrossProjectDuplicationEnabled(Settings settings) {
+ return settings.getBoolean(CoreProperties.CPD_CROSS_PROJECT)
+ // No cross project duplication for branches
+ && StringUtils.isBlank(settings.getString(CoreProperties.PROJECT_BRANCH_PROPERTY));
+ }
+
public Collection<Block> getByInputFile(InputFile inputFile, String resourceKey) {
return mem.getByResourceId(resourceKey);
}
diff --git a/sonar-batch/src/main/java/org/sonar/batch/mediumtest/TaskResult.java b/sonar-batch/src/main/java/org/sonar/batch/mediumtest/TaskResult.java
index 04b5724b2c3..49c7f95736b 100644
--- a/sonar-batch/src/main/java/org/sonar/batch/mediumtest/TaskResult.java
+++ b/sonar-batch/src/main/java/org/sonar/batch/mediumtest/TaskResult.java
@@ -41,10 +41,8 @@ import org.sonar.api.batch.fs.TextPointer;
import org.sonar.api.batch.fs.TextRange;
import org.sonar.api.batch.fs.internal.DefaultInputDir;
import org.sonar.api.batch.fs.internal.DefaultInputFile;
-import org.sonar.api.batch.sensor.duplication.Duplication;
import org.sonar.api.batch.sensor.highlighting.TypeOfText;
import org.sonar.api.issue.Issue;
-import org.sonar.batch.duplication.DuplicationCache;
import org.sonar.batch.issue.IssueCache;
import org.sonar.batch.protocol.output.BatchReport;
import org.sonar.batch.protocol.output.BatchReport.Component;
@@ -63,7 +61,6 @@ public class TaskResult implements org.sonar.batch.mediumtest.ScanTaskObserver {
private static final Logger LOG = LoggerFactory.getLogger(TaskResult.class);
private List<Issue> issues = new ArrayList<>();
- private Map<String, List<Duplication>> duplications = new HashMap<>();
private Map<String, InputFile> inputFiles = new HashMap<>();
private Map<String, Component> reportComponents = new HashMap<>();
private Map<String, InputDir> inputDirs = new HashMap<>();
@@ -86,7 +83,6 @@ public class TaskResult implements org.sonar.batch.mediumtest.ScanTaskObserver {
storeFs(container);
- storeDuplication(container);
}
private void storeReportComponents(int componentRef, String parentModuleKey, @Nullable String branch) {
@@ -106,13 +102,6 @@ public class TaskResult implements org.sonar.batch.mediumtest.ScanTaskObserver {
return reader;
}
- private void storeDuplication(ProjectScanContainer container) {
- DuplicationCache duplicationCache = container.getComponentByType(DuplicationCache.class);
- for (String effectiveKey : duplicationCache.componentKeys()) {
- duplications.put(effectiveKey, Lists.<Duplication>newArrayList(duplicationCache.byComponent(effectiveKey)));
- }
- }
-
private void storeFs(ProjectScanContainer container) {
InputPathCache inputFileCache = container.getComponentByType(InputPathCache.class);
for (InputFile inputPath : inputFileCache.allFiles()) {
@@ -127,6 +116,10 @@ public class TaskResult implements org.sonar.batch.mediumtest.ScanTaskObserver {
return issues;
}
+ public Component getReportComponent(String key) {
+ return reportComponents.get(key);
+ }
+
public List<BatchReport.Issue> issuesFor(InputPath inputPath) {
List<BatchReport.Issue> result = Lists.newArrayList();
int ref = reportComponents.get(key(inputPath)).getRef();
@@ -162,10 +155,6 @@ public class TaskResult implements org.sonar.batch.mediumtest.ScanTaskObserver {
return inputDirs.get(relativePath);
}
- public List<Duplication> duplicationsFor(InputFile inputFile) {
- return duplications.get(((DefaultInputFile) inputFile).key());
- }
-
public Map<String, List<BatchReport.Measure>> allMeasures() {
Map<String, List<BatchReport.Measure>> result = new HashMap<>();
for (Map.Entry<String, Component> component : reportComponents.entrySet()) {
@@ -226,6 +215,32 @@ public class TaskResult implements org.sonar.batch.mediumtest.ScanTaskObserver {
return Collections.emptyList();
}
+ public List<BatchReport.Duplication> duplicationsFor(InputFile file) {
+ List<BatchReport.Duplication> result = new ArrayList<>();
+ int ref = reportComponents.get(((DefaultInputFile) file).key()).getRef();
+ try (CloseableIterator<BatchReport.Duplication> it = getReportReader().readComponentDuplications(ref)) {
+ while (it.hasNext()) {
+ result.add(it.next());
+ }
+ } catch (Exception e) {
+ throw new IllegalStateException(e);
+ }
+ return result;
+ }
+
+ public List<BatchReport.DuplicationBlock> duplicationBlocksFor(InputFile file) {
+ List<BatchReport.DuplicationBlock> result = new ArrayList<>();
+ int ref = reportComponents.get(((DefaultInputFile) file).key()).getRef();
+ try (CloseableIterator<BatchReport.DuplicationBlock> it = getReportReader().readComponentDuplicationBlocks(ref)) {
+ while (it.hasNext()) {
+ result.add(it.next());
+ }
+ } catch (Exception e) {
+ throw new IllegalStateException(e);
+ }
+ return result;
+ }
+
@CheckForNull
public BatchReport.Coverage coverageFor(InputFile file, int line) {
int ref = reportComponents.get(((DefaultInputFile) file).key()).getRef();
diff --git a/sonar-batch/src/main/java/org/sonar/batch/report/DuplicationsPublisher.java b/sonar-batch/src/main/java/org/sonar/batch/report/DuplicationsPublisher.java
index 6cf132a9f6e..068cb1c10e4 100644
--- a/sonar-batch/src/main/java/org/sonar/batch/report/DuplicationsPublisher.java
+++ b/sonar-batch/src/main/java/org/sonar/batch/report/DuplicationsPublisher.java
@@ -81,7 +81,8 @@ public class DuplicationsPublisher implements ReportPublisherStep {
if (sameProjectComponent != null) {
blockBuilder.setOtherFileRef(sameProjectComponent.batchId());
} else {
- blockBuilder.setOtherFileKey(componentKey);
+ // Should never happens
+ throw new IllegalStateException("No cross project duplication supported on batch side: " + componentKey);
}
}
dupBuilder.addDuplicate(blockBuilder
diff --git a/sonar-batch/src/main/java/org/sonar/batch/report/MetadataPublisher.java b/sonar-batch/src/main/java/org/sonar/batch/report/MetadataPublisher.java
index 52e98b53d81..9c1d2faca6d 100644
--- a/sonar-batch/src/main/java/org/sonar/batch/report/MetadataPublisher.java
+++ b/sonar-batch/src/main/java/org/sonar/batch/report/MetadataPublisher.java
@@ -21,7 +21,9 @@ package org.sonar.batch.report;
import org.sonar.api.CoreProperties;
import org.sonar.api.batch.bootstrap.ProjectDefinition;
+import org.sonar.api.config.Settings;
import org.sonar.api.resources.Project;
+import org.sonar.batch.cpd.index.SonarDuplicationsIndex;
import org.sonar.batch.index.BatchComponent;
import org.sonar.batch.index.BatchComponentCache;
import org.sonar.batch.protocol.output.BatchReport;
@@ -32,10 +34,12 @@ public class MetadataPublisher implements ReportPublisherStep {
private final BatchComponentCache componentCache;
private final ImmutableProjectReactor reactor;
+ private final Settings settings;
- public MetadataPublisher(BatchComponentCache componentCache, ImmutableProjectReactor reactor) {
+ public MetadataPublisher(BatchComponentCache componentCache, ImmutableProjectReactor reactor, Settings settings) {
this.componentCache = componentCache;
this.reactor = reactor;
+ this.settings = settings;
}
@Override
@@ -46,6 +50,7 @@ public class MetadataPublisher implements ReportPublisherStep {
.setAnalysisDate(((Project) rootProject.resource()).getAnalysisDate().getTime())
// Here we want key without branch
.setProjectKey(root.getKey())
+ .setCrossProjectDuplicationActivated(SonarDuplicationsIndex.isCrossProjectDuplicationEnabled(settings))
.setRootComponentRef(rootProject.batchId());
String branch = root.properties().get(CoreProperties.PROJECT_BRANCH_PROPERTY);
if (branch != null) {
diff --git a/sonar-batch/src/main/java/org/sonar/batch/sensor/DefaultSensorStorage.java b/sonar-batch/src/main/java/org/sonar/batch/sensor/DefaultSensorStorage.java
index 0c6a9d67fa2..bcb38b4958a 100644
--- a/sonar-batch/src/main/java/org/sonar/batch/sensor/DefaultSensorStorage.java
+++ b/sonar-batch/src/main/java/org/sonar/batch/sensor/DefaultSensorStorage.java
@@ -95,18 +95,18 @@ public class DefaultSensorStorage implements SensorStorage {
private final ModuleIssues moduleIssues;
private final CoverageExclusions coverageExclusions;
private final DuplicationCache duplicationCache;
- private final BatchComponentCache resourceCache;
+ private final BatchComponentCache componentCache;
private final ReportPublisher reportPublisher;
private final MeasureCache measureCache;
public DefaultSensorStorage(MetricFinder metricFinder, ModuleIssues moduleIssues,
Settings settings, FileSystem fs, ActiveRules activeRules, DuplicationCache duplicationCache,
- CoverageExclusions coverageExclusions, BatchComponentCache resourceCache, ReportPublisher reportPublisher, MeasureCache measureCache) {
+ CoverageExclusions coverageExclusions, BatchComponentCache componentCache, ReportPublisher reportPublisher, MeasureCache measureCache) {
this.metricFinder = metricFinder;
this.moduleIssues = moduleIssues;
this.coverageExclusions = coverageExclusions;
this.duplicationCache = duplicationCache;
- this.resourceCache = resourceCache;
+ this.componentCache = componentCache;
this.reportPublisher = reportPublisher;
this.measureCache = measureCache;
}
@@ -127,7 +127,7 @@ public class DefaultSensorStorage implements SensorStorage {
setValueAccordingToMetricType(newMeasure, m, measureToSave);
measureToSave.setFromCore(measure.isFromCore());
InputComponent inputComponent = newMeasure.inputComponent();
- Resource resource = resourceCache.get(inputComponent).resource();
+ Resource resource = componentCache.get(inputComponent).resource();
if (coverageExclusions.accept(resource, measureToSave)) {
saveMeasure(resource, measureToSave);
}
@@ -183,7 +183,7 @@ public class DefaultSensorStorage implements SensorStorage {
}
private File getFile(InputFile file) {
- BatchComponent r = resourceCache.get(file);
+ BatchComponent r = componentCache.get(file);
if (r == null) {
throw new IllegalStateException("Provided input file is not indexed");
}
@@ -199,13 +199,13 @@ public class DefaultSensorStorage implements SensorStorage {
public void store(DefaultHighlighting highlighting) {
BatchReportWriter writer = reportPublisher.getWriter();
DefaultInputFile inputFile = (DefaultInputFile) highlighting.inputFile();
- writer.writeComponentSyntaxHighlighting(resourceCache.get(inputFile).batchId(),
+ writer.writeComponentSyntaxHighlighting(componentCache.get(inputFile).batchId(),
Iterables.transform(highlighting.getSyntaxHighlightingRuleSet(), new BuildSyntaxHighlighting()));
}
public void store(DefaultInputFile inputFile, Map<Symbol, Set<TextRange>> referencesBySymbol) {
BatchReportWriter writer = reportPublisher.getWriter();
- writer.writeComponentSymbols(resourceCache.get(inputFile).batchId(),
+ writer.writeComponentSymbols(componentCache.get(inputFile).batchId(),
Iterables.transform(referencesBySymbol.entrySet(), new Function<Map.Entry<Symbol, Set<TextRange>>, BatchReport.Symbol>() {
private BatchReport.Symbol.Builder builder = BatchReport.Symbol.newBuilder();
private BatchReport.TextRange.Builder rangeBuilder = BatchReport.TextRange.newBuilder();
diff --git a/sonar-batch/src/test/java/org/sonar/batch/cpd/CpdSensorTest.java b/sonar-batch/src/test/java/org/sonar/batch/cpd/CpdSensorTest.java
index a99d356c87b..5c5adb3a4c2 100644
--- a/sonar-batch/src/test/java/org/sonar/batch/cpd/CpdSensorTest.java
+++ b/sonar-batch/src/test/java/org/sonar/batch/cpd/CpdSensorTest.java
@@ -19,6 +19,7 @@
*/
package org.sonar.batch.cpd;
+import java.io.IOException;
import org.junit.Before;
import org.junit.Rule;
import org.junit.Test;
@@ -27,12 +28,8 @@ import org.sonar.api.batch.fs.internal.DefaultFileSystem;
import org.sonar.api.config.PropertyDefinitions;
import org.sonar.api.config.Settings;
import org.sonar.api.resources.Java;
-import org.sonar.batch.cpd.index.IndexFactory;
-
-import java.io.IOException;
import static org.assertj.core.api.Assertions.assertThat;
-import static org.mockito.Mockito.mock;
public class CpdSensorTest {
@@ -46,9 +43,8 @@ public class CpdSensorTest {
@Before
public void setUp() throws IOException {
- IndexFactory indexFactory = mock(IndexFactory.class);
- sonarEngine = new JavaCpdEngine(indexFactory, null, null);
- sonarBridgeEngine = new DefaultCpdEngine(indexFactory, new CpdMappings(), null, null);
+ sonarEngine = new JavaCpdEngine(null, null, null, null);
+ sonarBridgeEngine = new DefaultCpdEngine(new CpdMappings(), null, null, null, null);
settings = new Settings(new PropertyDefinitions(CpdComponents.class));
DefaultFileSystem fs = new DefaultFileSystem(temp.newFolder().toPath());
diff --git a/sonar-batch/src/test/java/org/sonar/batch/cpd/DefaultCpdEngineTest.java b/sonar-batch/src/test/java/org/sonar/batch/cpd/DefaultCpdEngineTest.java
index fb2a632e9c9..427c62a1e4c 100644
--- a/sonar-batch/src/test/java/org/sonar/batch/cpd/DefaultCpdEngineTest.java
+++ b/sonar-batch/src/test/java/org/sonar/batch/cpd/DefaultCpdEngineTest.java
@@ -23,7 +23,6 @@ import org.junit.Before;
import org.junit.Test;
import org.slf4j.Logger;
import org.sonar.api.config.Settings;
-import org.sonar.api.resources.Project;
import static org.assertj.core.api.Assertions.assertThat;
import static org.mockito.Matchers.anyString;
@@ -40,7 +39,7 @@ public class DefaultCpdEngineTest {
@Before
public void init() {
settings = new Settings();
- engine = new DefaultCpdEngine(null, null, null, settings);
+ engine = new DefaultCpdEngine(null, null, settings, null, null);
}
@Test
@@ -61,7 +60,6 @@ public class DefaultCpdEngineTest {
@Test
public void shouldReturnDefaultBlockSize() {
assertThat(DefaultCpdEngine.getDefaultBlockSize("cobol")).isEqualTo(30);
- assertThat(DefaultCpdEngine.getDefaultBlockSize("natur")).isEqualTo(20);
assertThat(DefaultCpdEngine.getDefaultBlockSize("abap")).isEqualTo(20);
assertThat(DefaultCpdEngine.getDefaultBlockSize("other")).isEqualTo(10);
}
@@ -85,13 +83,6 @@ public class DefaultCpdEngineTest {
}
@Test
- public void generalMinimumTokens() {
- settings.setProperty("sonar.cpd.minimumTokens", 33);
-
- assertThat(engine.getMinimumTokens("java")).isEqualTo(33);
- }
-
- @Test
public void minimumTokensByLanguage() {
settings.setProperty("sonar.cpd.java.minimumTokens", "42");
settings.setProperty("sonar.cpd.php.minimumTokens", "33");
@@ -102,7 +93,4 @@ public class DefaultCpdEngineTest {
assertThat(engine.getMinimumTokens("php")).isEqualTo(33);
}
- private static Project newProject(String key) {
- return new Project(key).setAnalysisType(Project.AnalysisType.DYNAMIC);
- }
}
diff --git a/sonar-batch/src/test/java/org/sonar/batch/cpd/index/IndexFactoryTest.java b/sonar-batch/src/test/java/org/sonar/batch/cpd/index/IndexFactoryTest.java
deleted file mode 100644
index f33f40b6cba..00000000000
--- a/sonar-batch/src/test/java/org/sonar/batch/cpd/index/IndexFactoryTest.java
+++ /dev/null
@@ -1,82 +0,0 @@
-/*
- * SonarQube, open source software quality management tool.
- * Copyright (C) 2008-2014 SonarSource
- * mailto:contact AT sonarsource DOT com
- *
- * SonarQube 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.
- *
- * SonarQube 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.batch.cpd.index;
-
-import org.sonar.batch.analysis.DefaultAnalysisMode;
-
-import org.junit.Before;
-import org.junit.Test;
-import org.slf4j.Logger;
-import org.sonar.api.CoreProperties;
-import org.sonar.api.config.Settings;
-import org.sonar.api.resources.Project;
-import static org.assertj.core.api.Assertions.assertThat;
-import static org.mockito.Mockito.mock;
-import static org.mockito.Mockito.verify;
-import static org.mockito.Mockito.when;
-
-public class IndexFactoryTest {
-
- Project project;
- Settings settings;
- IndexFactory factory;
- Logger logger;
- private DefaultAnalysisMode analysisMode;
-
- @Before
- public void setUp() {
- project = new Project("foo");
- settings = new Settings();
- analysisMode = mock(DefaultAnalysisMode.class);
- factory = new IndexFactory(analysisMode, settings);
- logger = mock(Logger.class);
- }
-
- @Test
- public void crossProjectEnabled() {
- settings.setProperty(CoreProperties.CPD_CROSS_PROJECT, "true");
- assertThat(factory.verifyCrossProject(project, logger)).isTrue();
- verify(logger).info("Cross-project analysis enabled");
- }
-
- @Test
- public void noCrossProjectWithBranch() {
- settings.setProperty(CoreProperties.CPD_CROSS_PROJECT, "true");
- settings.setProperty(CoreProperties.PROJECT_BRANCH_PROPERTY, "branch");
- assertThat(factory.verifyCrossProject(project, logger)).isFalse();
- verify(logger).info("Cross-project analysis disabled. Not supported on project branches.");
- }
-
- @Test
- public void cross_project_should_be_disabled_on_issues_mode() {
- when(analysisMode.isIssues()).thenReturn(true);
- settings.setProperty(CoreProperties.CPD_CROSS_PROJECT, "true");
- assertThat(factory.verifyCrossProject(project, logger)).isFalse();
- verify(logger).info("Cross-project analysis disabled. Not supported in issues mode.");
- }
-
- @Test
- public void crossProjectDisabled() {
- settings.setProperty(CoreProperties.CPD_CROSS_PROJECT, "false");
- assertThat(factory.verifyCrossProject(project, logger)).isFalse();
- verify(logger).info("Cross-project analysis disabled");
- }
-
-}
diff --git a/sonar-batch/src/test/java/org/sonar/batch/mediumtest/cpd/CpdMediumTest.java b/sonar-batch/src/test/java/org/sonar/batch/mediumtest/cpd/CpdMediumTest.java
index 03b171ae31e..7a0c260e4f7 100644
--- a/sonar-batch/src/test/java/org/sonar/batch/mediumtest/cpd/CpdMediumTest.java
+++ b/sonar-batch/src/test/java/org/sonar/batch/mediumtest/cpd/CpdMediumTest.java
@@ -34,10 +34,10 @@ import org.junit.rules.ExpectedException;
import org.junit.rules.TemporaryFolder;
import org.sonar.api.batch.fs.InputFile;
import org.sonar.api.batch.fs.internal.DefaultInputFile;
-import org.sonar.api.batch.sensor.duplication.Duplication;
import org.sonar.api.measures.CoreMetrics;
import org.sonar.batch.mediumtest.BatchMediumTester;
import org.sonar.batch.mediumtest.TaskResult;
+import org.sonar.batch.protocol.output.BatchReport.DuplicationBlock;
import org.sonar.batch.protocol.output.BatchReport.Measure;
import org.sonar.xoo.XooPlugin;
@@ -112,28 +112,70 @@ public class CpdMediumTest {
InputFile inputFile1 = result.inputFile("src/sample1.xoo");
InputFile inputFile2 = result.inputFile("src/sample2.xoo");
+
// One clone group on each file
- List<Duplication> duplicationGroupsFile1 = result.duplicationsFor(inputFile1);
+ List<org.sonar.batch.protocol.output.BatchReport.Duplication> duplicationGroupsFile1 = result.duplicationsFor(inputFile1);
assertThat(duplicationGroupsFile1).hasSize(1);
- Duplication cloneGroupFile1 = duplicationGroupsFile1.get(0);
- assertThat(cloneGroupFile1.duplicates()).hasSize(1);
- assertThat(cloneGroupFile1.originBlock().startLine()).isEqualTo(1);
- assertThat(cloneGroupFile1.originBlock().length()).isEqualTo(17);
- assertThat(cloneGroupFile1.originBlock().resourceKey()).isEqualTo(((DefaultInputFile) inputFile1).key());
- assertThat(cloneGroupFile1.duplicates()).hasSize(1);
- assertThat(cloneGroupFile1.duplicates().get(0).resourceKey()).isEqualTo(((DefaultInputFile) inputFile2).key());
+ org.sonar.batch.protocol.output.BatchReport.Duplication cloneGroupFile1 = duplicationGroupsFile1.get(0);
+ assertThat(cloneGroupFile1.getOriginPosition().getStartLine()).isEqualTo(1);
+ assertThat(cloneGroupFile1.getOriginPosition().getEndLine()).isEqualTo(17);
+ assertThat(cloneGroupFile1.getDuplicateList()).hasSize(1);
+ assertThat(cloneGroupFile1.getDuplicate(0).getOtherFileRef()).isEqualTo(result.getReportComponent(((DefaultInputFile) inputFile2).key()).getRef());
- List<Duplication> duplicationGroupsFile2 = result.duplicationsFor(inputFile2);
+ List<org.sonar.batch.protocol.output.BatchReport.Duplication> duplicationGroupsFile2 = result.duplicationsFor(inputFile2);
assertThat(duplicationGroupsFile2).hasSize(1);
- Duplication cloneGroupFile2 = duplicationGroupsFile2.get(0);
- assertThat(cloneGroupFile2.duplicates()).hasSize(1);
- assertThat(cloneGroupFile2.originBlock().startLine()).isEqualTo(1);
- assertThat(cloneGroupFile2.originBlock().length()).isEqualTo(17);
- assertThat(cloneGroupFile2.originBlock().resourceKey()).isEqualTo(((DefaultInputFile) inputFile2).key());
- assertThat(cloneGroupFile2.duplicates()).hasSize(1);
- assertThat(cloneGroupFile2.duplicates().get(0).resourceKey()).isEqualTo(((DefaultInputFile) inputFile1).key());
+ org.sonar.batch.protocol.output.BatchReport.Duplication cloneGroupFile2 = duplicationGroupsFile2.get(0);
+ assertThat(cloneGroupFile2.getOriginPosition().getStartLine()).isEqualTo(1);
+ assertThat(cloneGroupFile2.getOriginPosition().getEndLine()).isEqualTo(17);
+ assertThat(cloneGroupFile2.getDuplicateList()).hasSize(1);
+ assertThat(cloneGroupFile2.getDuplicate(0).getOtherFileRef()).isEqualTo(result.getReportComponent(((DefaultInputFile) inputFile1).key()).getRef());
+
+ assertThat(result.duplicationBlocksFor(inputFile1)).isEmpty();
+ }
+
+ @Test
+ public void enableCrossProjectDuplication() throws IOException {
+ File srcDir = new File(baseDir, "src");
+ srcDir.mkdir();
+
+ String duplicatedStuff = "Sample xoo\ncontent\nfoo\nbar\ntoto\ntiti\nfoo";
+
+ File xooFile1 = new File(srcDir, "sample1.xoo");
+ FileUtils.write(xooFile1, duplicatedStuff);
+
+ TaskResult result = tester.newTask()
+ .properties(builder
+ .put("sonar.sources", "src")
+ .put("sonar.cpd.xoo.minimumTokens", "1")
+ .put("sonar.cpd.xoo.minimumLines", "5")
+ .put("sonar.verbose", "true")
+ .put("sonar.cpd.cross_project", "true")
+ .build())
+ .start();
+
+ InputFile inputFile1 = result.inputFile("src/sample1.xoo");
+
+ List<DuplicationBlock> duplicationBlocks = result.duplicationBlocksFor(inputFile1);
+ assertThat(duplicationBlocks).hasSize(3);
+ assertThat(duplicationBlocks.get(0).getStartLine()).isEqualTo(1);
+ assertThat(duplicationBlocks.get(0).getEndLine()).isEqualTo(5);
+ assertThat(duplicationBlocks.get(0).getStartTokenIndex()).isEqualTo(1);
+ assertThat(duplicationBlocks.get(0).getEndTokenIndex()).isEqualTo(6);
+ assertThat(duplicationBlocks.get(0).getHashList()).isNotEmpty();
+
+ assertThat(duplicationBlocks.get(1).getStartLine()).isEqualTo(2);
+ assertThat(duplicationBlocks.get(1).getEndLine()).isEqualTo(6);
+ assertThat(duplicationBlocks.get(1).getStartTokenIndex()).isEqualTo(3);
+ assertThat(duplicationBlocks.get(1).getEndTokenIndex()).isEqualTo(7);
+ assertThat(duplicationBlocks.get(0).getHashList()).isNotEmpty();
+
+ assertThat(duplicationBlocks.get(2).getStartLine()).isEqualTo(3);
+ assertThat(duplicationBlocks.get(2).getEndLine()).isEqualTo(7);
+ assertThat(duplicationBlocks.get(2).getStartTokenIndex()).isEqualTo(4);
+ assertThat(duplicationBlocks.get(2).getEndTokenIndex()).isEqualTo(8);
+ assertThat(duplicationBlocks.get(0).getHashList()).isNotEmpty();
}
// SONAR-6000
@@ -171,11 +213,11 @@ public class CpdMediumTest {
Tuple.tuple(CoreMetrics.DUPLICATED_BLOCKS_KEY, blockCount),
Tuple.tuple(CoreMetrics.DUPLICATED_LINES_KEY, blockCount));
- List<Duplication> duplicationGroups = result.duplicationsFor(result.inputFile("src/sample.xoo"));
+ List<org.sonar.batch.protocol.output.BatchReport.Duplication> duplicationGroups = result.duplicationsFor(result.inputFile("src/sample.xoo"));
assertThat(duplicationGroups).hasSize(1);
- Duplication cloneGroup = duplicationGroups.get(0);
- assertThat(cloneGroup.duplicates()).hasSize(100);
+ org.sonar.batch.protocol.output.BatchReport.Duplication cloneGroup = duplicationGroups.get(0);
+ assertThat(cloneGroup.getDuplicateList()).hasSize(100);
}
@Test
@@ -205,16 +247,15 @@ public class CpdMediumTest {
InputFile inputFile = result.inputFile("src/sample.xoo");
// One clone group
- List<Duplication> duplicationGroups = result.duplicationsFor(inputFile);
+ List<org.sonar.batch.protocol.output.BatchReport.Duplication> duplicationGroups = result.duplicationsFor(inputFile);
assertThat(duplicationGroups).hasSize(1);
- Duplication cloneGroup = duplicationGroups.get(0);
- assertThat(cloneGroup.duplicates()).hasSize(1);
- assertThat(cloneGroup.originBlock().startLine()).isEqualTo(1);
- assertThat(cloneGroup.originBlock().length()).isEqualTo(2);
- assertThat(cloneGroup.duplicates()).hasSize(1);
- assertThat(cloneGroup.duplicates().get(0).startLine()).isEqualTo(5);
- assertThat(cloneGroup.duplicates().get(0).length()).isEqualTo(2);
+ org.sonar.batch.protocol.output.BatchReport.Duplication cloneGroup = duplicationGroups.get(0);
+ assertThat(cloneGroup.getOriginPosition().getStartLine()).isEqualTo(1);
+ assertThat(cloneGroup.getOriginPosition().getEndLine()).isEqualTo(2);
+ assertThat(cloneGroup.getDuplicateList()).hasSize(1);
+ assertThat(cloneGroup.getDuplicate(0).getRange().getStartLine()).isEqualTo(5);
+ assertThat(cloneGroup.getDuplicate(0).getRange().getEndLine()).isEqualTo(6);
}
}
diff --git a/sonar-batch/src/test/java/org/sonar/batch/report/DuplicationsPublisherTest.java b/sonar-batch/src/test/java/org/sonar/batch/report/DuplicationsPublisherTest.java
index 731bad05db4..b57c18df8e2 100644
--- a/sonar-batch/src/test/java/org/sonar/batch/report/DuplicationsPublisherTest.java
+++ b/sonar-batch/src/test/java/org/sonar/batch/report/DuplicationsPublisherTest.java
@@ -73,11 +73,8 @@ public class DuplicationsPublisherTest {
.isDuplicatedBy("foo:src/Foo.php", 20, 50);
DefaultDuplication dup2 = new DefaultDuplication()
.setOriginBlock(new Duplication.Block("foo:src/Foo.php", 11, 10))
- .isDuplicatedBy("another", 20, 50);
- DefaultDuplication dup3 = new DefaultDuplication()
- .setOriginBlock(new Duplication.Block("foo:src/Foo.php", 11, 10))
.isDuplicatedBy("foo:src/Foo2.php", 20, 50);
- when(duplicationCache.byComponent("foo:src/Foo.php")).thenReturn(Arrays.asList(dup1, dup2, dup3));
+ when(duplicationCache.byComponent("foo:src/Foo.php")).thenReturn(Arrays.asList(dup1, dup2));
File outputDir = temp.newFolder();
BatchReportWriter writer = new BatchReportWriter(outputDir);
@@ -91,7 +88,6 @@ public class DuplicationsPublisherTest {
org.sonar.batch.protocol.output.BatchReport.Duplication savedDup1 = componentDuplications.next();
assertThat(savedDup1.getOriginPosition().getStartLine()).isEqualTo(1);
assertThat(savedDup1.getOriginPosition().getEndLine()).isEqualTo(10);
- assertThat(savedDup1.getDuplicate(0).hasOtherFileKey()).isFalse();
assertThat(savedDup1.getDuplicate(0).hasOtherFileRef()).isFalse();
assertThat(savedDup1.getDuplicate(0).getRange().getStartLine()).isEqualTo(20);
assertThat(savedDup1.getDuplicate(0).getRange().getEndLine()).isEqualTo(50);
@@ -99,19 +95,10 @@ public class DuplicationsPublisherTest {
org.sonar.batch.protocol.output.BatchReport.Duplication savedDup2 = componentDuplications.next();
assertThat(savedDup2.getOriginPosition().getStartLine()).isEqualTo(11);
assertThat(savedDup2.getOriginPosition().getEndLine()).isEqualTo(20);
- assertThat(savedDup2.getDuplicate(0).getOtherFileKey()).isEqualTo("another");
- assertThat(savedDup2.getDuplicate(0).hasOtherFileRef()).isFalse();
+ assertThat(savedDup2.getDuplicate(0).getOtherFileRef()).isEqualTo(3);
assertThat(savedDup2.getDuplicate(0).getRange().getStartLine()).isEqualTo(20);
assertThat(savedDup2.getDuplicate(0).getRange().getEndLine()).isEqualTo(50);
- org.sonar.batch.protocol.output.BatchReport.Duplication savedDup3 = componentDuplications.next();
- assertThat(savedDup3.getOriginPosition().getStartLine()).isEqualTo(11);
- assertThat(savedDup3.getOriginPosition().getEndLine()).isEqualTo(20);
- assertThat(savedDup3.getDuplicate(0).hasOtherFileKey()).isFalse();
- assertThat(savedDup3.getDuplicate(0).getOtherFileRef()).isEqualTo(3);
- assertThat(savedDup3.getDuplicate(0).getRange().getStartLine()).isEqualTo(20);
- assertThat(savedDup3.getDuplicate(0).getRange().getEndLine()).isEqualTo(50);
-
assertThat(componentDuplications.hasNext()).isFalse();
}
diff --git a/sonar-batch/src/test/java/org/sonar/batch/report/MetadataPublisherTest.java b/sonar-batch/src/test/java/org/sonar/batch/report/MetadataPublisherTest.java
index eab392c374f..eb93a5661c5 100644
--- a/sonar-batch/src/test/java/org/sonar/batch/report/MetadataPublisherTest.java
+++ b/sonar-batch/src/test/java/org/sonar/batch/report/MetadataPublisherTest.java
@@ -27,6 +27,7 @@ import org.junit.Test;
import org.junit.rules.TemporaryFolder;
import org.sonar.api.CoreProperties;
import org.sonar.api.batch.bootstrap.ProjectDefinition;
+import org.sonar.api.config.Settings;
import org.sonar.api.resources.Project;
import org.sonar.batch.index.BatchComponentCache;
import org.sonar.batch.protocol.output.BatchReport;
@@ -41,10 +42,10 @@ public class MetadataPublisherTest {
@Rule
public TemporaryFolder temp = new TemporaryFolder();
- ProjectDefinition projectDef;
- Project project;
-
- MetadataPublisher underTest;
+ private ProjectDefinition projectDef;
+ private Project project;
+ private MetadataPublisher underTest;
+ private Settings settings;
@Before
public void prepare() {
@@ -54,11 +55,13 @@ public class MetadataPublisherTest {
org.sonar.api.resources.Resource sampleFile = org.sonar.api.resources.File.create("src/Foo.php").setEffectiveKey("foo:src/Foo.php");
componentCache.add(project, null);
componentCache.add(sampleFile, project);
- underTest = new MetadataPublisher(componentCache, new ImmutableProjectReactor(projectDef));
+ settings = new Settings();
+ underTest = new MetadataPublisher(componentCache, new ImmutableProjectReactor(projectDef), settings);
}
@Test
public void write_metadata() throws Exception {
+ settings.setProperty(CoreProperties.CPD_CROSS_PROJECT, "true");
File outputDir = temp.newFolder();
BatchReportWriter writer = new BatchReportWriter(outputDir);
@@ -68,10 +71,14 @@ public class MetadataPublisherTest {
BatchReport.Metadata metadata = reader.readMetadata();
assertThat(metadata.getAnalysisDate()).isEqualTo(1234567L);
assertThat(metadata.getProjectKey()).isEqualTo("foo");
+ assertThat(metadata.getProjectKey()).isEqualTo("foo");
+ assertThat(metadata.getCrossProjectDuplicationActivated()).isTrue();
}
@Test
public void write_project_branch() throws Exception {
+ settings.setProperty(CoreProperties.CPD_CROSS_PROJECT, "true");
+ settings.setProperty(CoreProperties.PROJECT_BRANCH_PROPERTY, "myBranch");
projectDef.properties().put(CoreProperties.PROJECT_BRANCH_PROPERTY, "myBranch");
project.setKey("foo:myBranch");
project.setEffectiveKey("foo:myBranch");
@@ -86,6 +93,8 @@ public class MetadataPublisherTest {
assertThat(metadata.getAnalysisDate()).isEqualTo(1234567L);
assertThat(metadata.getProjectKey()).isEqualTo("foo");
assertThat(metadata.getBranch()).isEqualTo("myBranch");
+ // Cross project duplication disabled on branches
+ assertThat(metadata.getCrossProjectDuplicationActivated()).isFalse();
}
}
diff --git a/sonar-core/src/main/java/org/sonar/core/config/CorePropertyDefinitions.java b/sonar-core/src/main/java/org/sonar/core/config/CorePropertyDefinitions.java
index 554ca4b07ee..30f06ad2e7c 100644
--- a/sonar-core/src/main/java/org/sonar/core/config/CorePropertyDefinitions.java
+++ b/sonar-core/src/main/java/org/sonar/core/config/CorePropertyDefinitions.java
@@ -276,42 +276,31 @@ public class CorePropertyDefinitions {
.onlyOnQualifiers(Qualifiers.PROJECT)
.category(CoreProperties.CATEGORY_GENERAL)
.subCategory(CoreProperties.SUBCATEGORY_DIFFERENTIAL_VIEWS)
- .build()));
-
+ .build(),
// CPD
- // CPD properties disabled in 5.2, will be enabled back in 5.3 (see SONAR-6323)
-// PropertyDefinition.builder(CoreProperties.CPD_CROSS_PROJECT)
-// .defaultValue(Boolean.toString(CoreProperties.CPD_CROSS_RPOJECT_DEFAULT_VALUE))
-// .name("Cross project duplication detection")
-// .description("By default, SonarQube detects duplications at sub-project level. This means that a block "
-// + "duplicated on two sub-projects of the same project won't be reported. Setting this parameter to \"true\" "
-// + "allows to detect duplicates across sub-projects and more generally across projects. Note that activating "
-// + "this property will slightly increase each SonarQube analysis time.")
-// .onQualifiers(Qualifiers.PROJECT, Qualifiers.MODULE)
-// .category(CoreProperties.CATEGORY_GENERAL)
-// .subCategory(CoreProperties.SUBCATEGORY_DUPLICATIONS)
-// .type(PropertyType.BOOLEAN)
-// .build(),
-// PropertyDefinition.builder(CoreProperties.CPD_SKIP_PROPERTY)
-// .defaultValue(String.valueOf(false))
-// .name("Skip")
-// .description("Disable detection of duplications")
-// .hidden()
-// .category(CoreProperties.CATEGORY_GENERAL)
-// .subCategory(CoreProperties.SUBCATEGORY_DUPLICATIONS)
-// .type(PropertyType.BOOLEAN)
-// .build(),
-// PropertyDefinition.builder(CoreProperties.CPD_EXCLUSIONS)
-// .defaultValue("")
-// .name("Duplication Exclusions")
-// .description("Patterns used to exclude some source files from the duplication detection mechanism. " +
-// "See below to know how to use wildcards to specify this property.")
-// .onQualifiers(Qualifiers.PROJECT, Qualifiers.MODULE)
-// .category(CoreProperties.CATEGORY_EXCLUSIONS)
-// .subCategory(CoreProperties.SUBCATEGORY_DUPLICATIONS_EXCLUSIONS)
-// .multiValues(true)
-// .build()));
+ PropertyDefinition.builder(CoreProperties.CPD_CROSS_PROJECT)
+ .defaultValue(Boolean.toString(CoreProperties.CPD_CROSS_PROJECT_DEFAULT_VALUE))
+ .name("Cross project duplication detection")
+ .description("By default, SonarQube detects duplications at sub-project level. This means that a block "
+ + "duplicated on two sub-projects of the same project won't be reported. Setting this parameter to \"true\" "
+ + "allows to detect duplicates across sub-projects and more generally across projects. Note that activating "
+ + "this property will slightly increase each SonarQube analysis time.")
+ .onQualifiers(Qualifiers.PROJECT)
+ .category(CoreProperties.CATEGORY_GENERAL)
+ .subCategory(CoreProperties.SUBCATEGORY_DUPLICATIONS)
+ .type(PropertyType.BOOLEAN)
+ .build(),
+ PropertyDefinition.builder(CoreProperties.CPD_EXCLUSIONS)
+ .defaultValue("")
+ .name("Duplication Exclusions")
+ .description("Patterns used to exclude some source files from the duplication detection mechanism. " +
+ "See below to know how to use wildcards to specify this property.")
+ .onQualifiers(Qualifiers.PROJECT, Qualifiers.MODULE)
+ .category(CoreProperties.CATEGORY_EXCLUSIONS)
+ .subCategory(CoreProperties.SUBCATEGORY_DUPLICATIONS_EXCLUSIONS)
+ .multiValues(true)
+ .build()));
return defs;
}
}
diff --git a/sonar-plugin-api/src/main/java/org/sonar/api/CoreProperties.java b/sonar-plugin-api/src/main/java/org/sonar/api/CoreProperties.java
index a5f974c8101..9e4a625b626 100644
--- a/sonar-plugin-api/src/main/java/org/sonar/api/CoreProperties.java
+++ b/sonar-plugin-api/src/main/java/org/sonar/api/CoreProperties.java
@@ -266,12 +266,6 @@ public interface CoreProperties {
String CPD_PLUGIN = "cpd";
/**
- * @deprecated in 3.1
- */
- @Deprecated
- String CPD_MINIMUM_TOKENS_PROPERTY = "sonar.cpd.minimumTokens";
-
- /**
* @deprecated in 5.0
* @see <a href="https://jira.sonarsource.com/browse/SONAR-5339">SONAR-5339</a>
*/
@@ -287,7 +281,7 @@ public interface CoreProperties {
* @see #CPD_CROSS_PROJECT
* @since 2.11
*/
- boolean CPD_CROSS_RPOJECT_DEFAULT_VALUE = false;
+ boolean CPD_CROSS_PROJECT_DEFAULT_VALUE = false;
/**
* @since 3.5
@@ -434,7 +428,7 @@ public interface CoreProperties {
* @since 5.2
*/
String ANALYSIS_MODE_ISSUES = "issues";
-
+
/**
* @since 5.2
*/