aboutsummaryrefslogtreecommitdiffstats
path: root/server
diff options
context:
space:
mode:
Diffstat (limited to 'server')
-rw-r--r--server/sonar-server-benchmarks/src/test/java/org/sonar/server/benchmark/PersistFileSourcesStepTest.java14
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/computation/duplication/DetailedTextBlock.java65
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/computation/duplication/DuplicationRepository.java45
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/computation/duplication/DuplicationRepositoryImpl.java353
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/computation/duplication/IntegrateCrossProjectDuplications.java59
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/computation/duplication/TextBlock.java2
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/computation/source/DuplicationLineReader.java13
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/computation/step/DuplicationDataMeasuresStep.java6
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/computation/step/DuplicationMeasuresStep.java5
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/computation/step/LoadDuplicationsFromReportStep.java52
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/computation/duplication/DuplicationRepositoryImplTest.java362
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/computation/duplication/DuplicationRepositoryRule.java116
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/computation/duplication/IntegrateCrossProjectDuplicationsTest.java64
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/computation/step/LoadDuplicationsFromReportStepTest.java46
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/computation/step/PersistFileSourcesStepTest.java9
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/computation/step/ReportDuplicationMeasuresStepTest.java11
-rw-r--r--server/sonar-web/src/main/js/apps/overview/domains/coverage-domain.js2
-rw-r--r--server/sonar-web/src/main/js/apps/overview/domains/debt-domain.js2
-rw-r--r--server/sonar-web/src/main/js/apps/overview/domains/duplications-domain.js2
-rw-r--r--server/sonar-web/src/main/js/apps/overview/domains/structure-domain.js2
20 files changed, 403 insertions, 827 deletions
diff --git a/server/sonar-server-benchmarks/src/test/java/org/sonar/server/benchmark/PersistFileSourcesStepTest.java b/server/sonar-server-benchmarks/src/test/java/org/sonar/server/benchmark/PersistFileSourcesStepTest.java
index 722b99b9238..8cfa29b8274 100644
--- a/server/sonar-server-benchmarks/src/test/java/org/sonar/server/benchmark/PersistFileSourcesStepTest.java
+++ b/server/sonar-server-benchmarks/src/test/java/org/sonar/server/benchmark/PersistFileSourcesStepTest.java
@@ -23,6 +23,7 @@ package org.sonar.server.benchmark;
import java.io.File;
import java.io.IOException;
import java.util.ArrayList;
+import java.util.Arrays;
import java.util.List;
import org.apache.commons.io.FileUtils;
import org.junit.Rule;
@@ -43,7 +44,10 @@ import org.sonar.server.computation.batch.BatchReportReaderImpl;
import org.sonar.server.computation.batch.TreeRootHolderRule;
import org.sonar.server.computation.component.Component;
import org.sonar.server.computation.component.ReportComponent;
+import org.sonar.server.computation.duplication.Duplicate;
+import org.sonar.server.computation.duplication.Duplication;
import org.sonar.server.computation.duplication.DuplicationRepositoryRule;
+import org.sonar.server.computation.duplication.InnerDuplicate;
import org.sonar.server.computation.duplication.TextBlock;
import org.sonar.server.computation.scm.ScmInfoRepositoryImpl;
import org.sonar.server.computation.source.SourceHashRepositoryImpl;
@@ -141,9 +145,13 @@ public class PersistFileSourcesStepTest {
LineData lineData = new LineData();
for (int line = 1; line <= NUMBER_OF_LINES; line++) {
lineData.generateLineData(line);
-
- duplicationRepository.addDuplication(fileRef, new TextBlock(line, line), new TextBlock(line + 1, line + 1));
-
+ duplicationRepository.add(
+ fileRef,
+ new Duplication(
+ new TextBlock(line, line),
+ Arrays.<Duplicate>asList(new InnerDuplicate(new TextBlock(line + 1, line + 1)))
+ )
+ );
}
writer.writeComponent(BatchReport.Component.newBuilder()
.setRef(fileRef)
diff --git a/server/sonar-server/src/main/java/org/sonar/server/computation/duplication/DetailedTextBlock.java b/server/sonar-server/src/main/java/org/sonar/server/computation/duplication/DetailedTextBlock.java
new file mode 100644
index 00000000000..a096335c870
--- /dev/null
+++ b/server/sonar-server/src/main/java/org/sonar/server/computation/duplication/DetailedTextBlock.java
@@ -0,0 +1,65 @@
+/*
+ * 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.server.computation.duplication;
+
+import java.util.Objects;
+import javax.annotation.Nullable;
+
+/**
+ * Given that:
+ * <ul>
+ * <li>some language plugins are able to multiple duplication for the same textblock but with a
+ * different starting and/or ending offset</li>
+ * <li>there is no way to distinguish these block from each other as the offsets (or any other
+ * information) is not sent in the analysis report</li>
+ * </ul>,
+ * we are uniquely (and blindly) identifying each original block reported in the analysis report.
+ */
+public class DetailedTextBlock extends TextBlock {
+ private final int id;
+ /**
+ * @throws IllegalArgumentException if {@code start} is 0 or less
+ * @throws IllegalStateException if {@code end} is less than {@code start}
+ */
+ public DetailedTextBlock(int id, int start, int end) {
+ super(start, end);
+ this.id = id;
+ }
+
+ @Override
+ public boolean equals(@Nullable Object o) {
+ if (this == o) {
+ return true;
+ }
+ if (o == null || getClass() != o.getClass()) {
+ return false;
+ }
+ if (!super.equals(o)) {
+ return false;
+ }
+ DetailedTextBlock that = (DetailedTextBlock) o;
+ return id == that.id;
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(super.hashCode(), id);
+ }
+}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/computation/duplication/DuplicationRepository.java b/server/sonar-server/src/main/java/org/sonar/server/computation/duplication/DuplicationRepository.java
index 0bfc05a004a..a3f1721b3f3 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/computation/duplication/DuplicationRepository.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/computation/duplication/DuplicationRepository.java
@@ -19,7 +19,6 @@
*/
package org.sonar.server.computation.duplication;
-import java.util.Set;
import org.sonar.server.computation.component.Component;
/**
@@ -34,58 +33,20 @@ import org.sonar.server.computation.component.Component;
* </p>
*/
public interface DuplicationRepository {
-
/**
* Returns the duplications in the specified file {@link Component}, if any.
*
* @throws NullPointerException if {@code file} is {@code null}
* @throws IllegalArgumentException if the type of the {@link Component} argument is not {@link Component.Type#FILE}
*/
- Set<Duplication> getDuplications(Component file);
+ Iterable<Duplication> getDuplications(Component file);
/**
- * Adds a project duplication of the specified original {@link TextBlock} in the specified file {@link Component}
- * which is duplicated by the specified duplicate {@link TextBlock} in the same file.
- * <p>
- * This method can be called multiple times with the same {@code file} and {@code original} but a different
- * {@code duplicate} to add multiple duplications of the same block.
- * </p>
- * <p>
- * It must not, however, be called twice with {@code original} and {@code duplicate} swapped, this would raise
- * an {@link IllegalArgumentException} as the duplication already exists.
- * </p>
+ * Adds a project duplication in the specified file {@link Component} to the repository.
*
* @throws NullPointerException if any argument is {@code null}
* @throws IllegalArgumentException if the type of the {@link Component} argument is not {@link Component.Type#FILE}
- * @throws IllegalStateException if {@code original} and {@code duplicate} are the same
- * @throws IllegalStateException if the specified duplication already exists in the repository
- */
- void addDuplication(Component file, TextBlock original, TextBlock duplicate);
-
- /**
- * Adds a project duplication of the specified original {@link TextBlock} in the specified file {@link Component} as
- * duplicated by the duplicate {@link TextBlock} in the other file {@link Component}.
- * <p>
- * Note: the reverse duplication relationship between files is not added automatically (which leaves open the
- * possibility of inconsistent duplication information). This means that it is the responsibility of the repository's
- * user to call this method again with the {@link Component} arguments and the {@link TextBlock} arguments swapped.
- * </p>
- *
- * @throws NullPointerException if any argument is {@code null}
- * @throws IllegalArgumentException if the type of any of the {@link Component} arguments is not {@link Component.Type#FILE}
- * @throws IllegalArgumentException if {@code file} and {@code otherFile} are the same
- * @throws IllegalStateException if the specified duplication already exists in the repository
*/
- void addDuplication(Component file, TextBlock original, Component otherFile, TextBlock duplicate);
+ void add(Component file, Duplication duplication);
- /**
- * Adds a cross-project duplication of the specified original {@link TextBlock} in the specified file {@link Component},
- * as duplicated by the specified duplicate {@link TextBlock} in a file of another project identified by its key.
- *
- * @throws NullPointerException if any argument is {@code null}
- * @throws IllegalArgumentException if the type of the {@link Component} argument is not {@link Component.Type#FILE}
- * @throws IllegalArgumentException if {@code otherFileKey} is the key of a file in the project
- * @throws IllegalStateException if the specified duplication already exists in the repository
- */
- void addDuplication(Component file, TextBlock original, String otherFileKey, TextBlock duplicate);
}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/computation/duplication/DuplicationRepositoryImpl.java b/server/sonar-server/src/main/java/org/sonar/server/computation/duplication/DuplicationRepositoryImpl.java
index 49162a9518d..8659661d5fc 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/computation/duplication/DuplicationRepositoryImpl.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/computation/duplication/DuplicationRepositoryImpl.java
@@ -19,367 +19,44 @@
*/
package org.sonar.server.computation.duplication;
-import com.google.common.base.Function;
-import com.google.common.base.Optional;
+import com.google.common.collect.HashMultimap;
+import com.google.common.collect.Multimap;
+import java.util.Collection;
import java.util.Collections;
-import java.util.HashMap;
-import java.util.HashSet;
-import java.util.Map;
-import java.util.Set;
-import javax.annotation.Nonnull;
-import javax.annotation.Nullable;
-import javax.annotation.concurrent.Immutable;
import org.sonar.server.computation.component.Component;
-import org.sonar.server.computation.component.TreeRootHolder;
import static com.google.common.base.Preconditions.checkArgument;
-import static com.google.common.collect.FluentIterable.from;
-import static java.lang.String.format;
+import static com.google.common.base.Preconditions.checkNotNull;
import static java.util.Objects.requireNonNull;
/**
* In-memory implementation of {@link DuplicationRepository}.
*/
public class DuplicationRepositoryImpl implements DuplicationRepository {
- private final TreeRootHolder treeRootHolder;
- private final Map<String, Duplications> duplicationsByComponentUuid = new HashMap<>();
-
- public DuplicationRepositoryImpl(TreeRootHolder treeRootHolder) {
- this.treeRootHolder = treeRootHolder;
- }
-
- @Override
- public Set<Duplication> getDuplications(Component file) {
- checkFileComponentArgument(file);
-
- if (duplicationsByComponentUuid.containsKey(file.getUuid())) {
- return from(duplicationsByComponentUuid.get(file.getUuid()).getDuplicates())
- .transform(DuplicatesEntryToDuplication.INSTANCE)
- .toSet();
- }
- return Collections.emptySet();
- }
-
- @Override
- public void addDuplication(Component file, TextBlock original, TextBlock duplicate) {
- checkFileComponentArgument(file);
- checkOriginalTextBlockArgument(original);
- checkDuplicateTextBlockArgument(duplicate);
- checkArgument(!original.equals(duplicate), "original and duplicate TextBlocks can not be the same");
-
- Optional<Duplicates> duplicates = getDuplicates(file, original);
- if (duplicates.isPresent()) {
- checkDuplicationAlreadyExists(duplicates, file, original, duplicate);
- checkReverseDuplicationAlreadyExists(file, original, duplicate);
-
- duplicates.get().add(duplicate);
- } else {
- checkReverseDuplicationAlreadyExists(file, original, duplicate);
- getOrCreate(file, original).add(duplicate);
- }
- }
+ private Multimap<String, Duplication> duplications = HashMultimap.create();
@Override
- public void addDuplication(Component file, TextBlock original, Component otherFile, TextBlock duplicate) {
+ public Iterable<Duplication> getDuplications(Component file) {
checkFileComponentArgument(file);
- checkOriginalTextBlockArgument(original);
- checkComponentArgument(otherFile, "otherFile");
- checkDuplicateTextBlockArgument(duplicate);
- checkArgument(!file.equals(otherFile), "file and otherFile Components can not be the same");
- Optional<Duplicates> duplicates = getDuplicates(file, original);
- if (duplicates.isPresent()) {
- checkDuplicationAlreadyExists(duplicates, file, original, otherFile, duplicate);
-
- duplicates.get().add(otherFile, duplicate);
- } else {
- getOrCreate(file, original).add(otherFile, duplicate);
+ Collection<Duplication> duplications = this.duplications.asMap().get(file.getKey());
+ if (duplications == null) {
+ return Collections.emptyList();
}
+ return duplications;
}
@Override
- public void addDuplication(Component file, TextBlock original, String otherFileKey, TextBlock duplicate) {
+ public void add(Component file, Duplication duplication) {
checkFileComponentArgument(file);
- checkOriginalTextBlockArgument(original);
- requireNonNull(otherFileKey, "otherFileKey can not be null");
- checkDuplicateTextBlockArgument(duplicate);
- checkArgument(!treeRootHolder.hasComponentWithKey(otherFileKey), "otherFileKey '%s' can not be the key of a Component in the project", otherFileKey);
-
- Optional<Duplicates> duplicates = getDuplicates(file, original);
- if (duplicates.isPresent()) {
- checkDuplicationAlreadyExists(duplicates, file, original, otherFileKey, duplicate);
-
- duplicates.get().add(otherFileKey, duplicate);
- } else {
- getOrCreate(file, original).add(otherFileKey, duplicate);
- }
- }
-
- private Optional<Duplicates> getDuplicates(Component file, TextBlock original) {
- Duplications duplications = duplicationsByComponentUuid.get(file.getUuid());
- if (duplications == null) {
- return Optional.absent();
- }
- return duplications.get(original);
- }
-
- private void checkDuplicationAlreadyExists(Optional<Duplicates> duplicates, Component file, TextBlock original, TextBlock duplicate) {
- checkArgument(!duplicates.get().hasDuplicate(duplicate),
- "Inner duplicate %s is already associated to original %s in file %s", duplicate, original, file.getKey());
- }
-
- private void checkReverseDuplicationAlreadyExists(Component file, TextBlock original, TextBlock duplicate) {
- Optional<Duplicates> reverseDuplication = getDuplicates(file, duplicate);
- if (reverseDuplication.isPresent()) {
- checkArgument(!reverseDuplication.get().hasDuplicate(original),
- "Inner duplicate %s is already associated to original %s in file %s", duplicate, original, file.getKey());
- }
- }
-
- private static void checkDuplicationAlreadyExists(Optional<Duplicates> duplicates, Component file, TextBlock original, Component otherFile, TextBlock duplicate) {
- checkArgument(!duplicates.get().hasDuplicate(otherFile, duplicate),
- "In-project duplicate %s in file %s is already associated to original %s in file %s", duplicate, otherFile.getKey(), original, file.getKey());
- }
-
- private static void checkDuplicationAlreadyExists(Optional<Duplicates> duplicates, Component file, TextBlock original, String otherFileKey, TextBlock duplicate) {
- checkArgument(!duplicates.get().hasDuplicate(otherFileKey, duplicate),
- "Cross-project duplicate %s in file %s is already associated to original %s in file %s", duplicate, otherFileKey, original, file.getKey());
- }
-
- private Duplicates getOrCreate(Component file, TextBlock original) {
- Duplications duplications = duplicationsByComponentUuid.get(file.getUuid());
- if (duplications == null) {
- duplications = new Duplications();
- duplicationsByComponentUuid.put(file.getUuid(), duplications);
- }
-
- return duplications.getOrCreate(original);
- }
-
- /**
- * Represents the location of the file of one or more duplicate {@link TextBlock}.
- */
- private interface DuplicateLocation {
-
- }
-
- private enum InnerDuplicationLocation implements DuplicateLocation {
- INSTANCE
- }
-
- @Immutable
- private static final class InProjectDuplicationLocation implements DuplicateLocation {
- private final Component component;
-
- public InProjectDuplicationLocation(Component component) {
- this.component = component;
- }
-
- public Component getComponent() {
- return component;
- }
-
- @Override
- public boolean equals(@Nullable Object o) {
- if (this == o) {
- return true;
- }
- if (o == null || getClass() != o.getClass()) {
- return false;
- }
- InProjectDuplicationLocation that = (InProjectDuplicationLocation) o;
- return component.equals(that.component);
- }
-
- @Override
- public int hashCode() {
- return component.hashCode();
- }
- }
-
- @Immutable
- private static final class CrossProjectDuplicationLocation implements DuplicateLocation {
- private final String fileKey;
-
- private CrossProjectDuplicationLocation(String fileKey) {
- this.fileKey = fileKey;
- }
-
- public String getFileKey() {
- return fileKey;
- }
-
- @Override
- public boolean equals(@Nullable Object o) {
- if (this == o) {
- return true;
- }
- if (o == null || getClass() != o.getClass()) {
- return false;
- }
- CrossProjectDuplicationLocation that = (CrossProjectDuplicationLocation) o;
- return fileKey.equals(that.fileKey);
- }
-
- @Override
- public int hashCode() {
- return fileKey.hashCode();
- }
- }
-
- private static final class Duplications {
- private final Map<TextBlock, Duplicates> duplicatesByTextBlock = new HashMap<>();
-
- public Duplicates getOrCreate(TextBlock textBlock) {
- if (duplicatesByTextBlock.containsKey(textBlock)) {
- return duplicatesByTextBlock.get(textBlock);
- }
- Duplicates res = new Duplicates();
- duplicatesByTextBlock.put(textBlock, res);
- return res;
- }
-
- public Set<Map.Entry<TextBlock, Duplicates>> getDuplicates() {
- return duplicatesByTextBlock.entrySet();
- }
-
- public Optional<Duplicates> get(TextBlock original) {
- return Optional.fromNullable(duplicatesByTextBlock.get(original));
- }
- }
-
- private static final class Duplicates {
- private final Map<DuplicateLocation, Set<TextBlock>> duplicatesByLocation = new HashMap<>();
-
- public Iterable<DuplicateWithLocation> getDuplicates() {
- return from(duplicatesByLocation.entrySet())
- .transformAndConcat(MapEntryToDuplicateWithLocation.INSTANCE);
- }
-
- public void add(TextBlock duplicate) {
- add(InnerDuplicationLocation.INSTANCE, duplicate);
- }
-
- public void add(Component otherFile, TextBlock duplicate) {
- InProjectDuplicationLocation key = new InProjectDuplicationLocation(otherFile);
- add(key, duplicate);
- }
-
- public void add(String otherFileKey, TextBlock duplicate) {
- CrossProjectDuplicationLocation key = new CrossProjectDuplicationLocation(otherFileKey);
- add(key, duplicate);
- }
-
- private void add(DuplicateLocation duplicateLocation, TextBlock duplicate) {
- Set<TextBlock> textBlocks = duplicatesByLocation.get(duplicateLocation);
- if (textBlocks == null) {
- textBlocks = new HashSet<>(1);
- duplicatesByLocation.put(duplicateLocation, textBlocks);
- }
- textBlocks.add(duplicate);
- }
-
- public boolean hasDuplicate(TextBlock duplicate) {
- return containsEntry(InnerDuplicationLocation.INSTANCE, duplicate);
- }
-
- public boolean hasDuplicate(Component otherFile, TextBlock duplicate) {
- return containsEntry(new InProjectDuplicationLocation(otherFile), duplicate);
- }
-
- public boolean hasDuplicate(String otherFileKey, TextBlock duplicate) {
- return containsEntry(new CrossProjectDuplicationLocation(otherFileKey), duplicate);
- }
-
- private boolean containsEntry(DuplicateLocation duplicateLocation, TextBlock duplicate) {
- Set<TextBlock> textBlocks = duplicatesByLocation.get(duplicateLocation);
- return textBlocks != null && textBlocks.contains(duplicate);
- }
-
- private enum MapEntryToDuplicateWithLocation implements Function<Map.Entry<DuplicateLocation, Set<TextBlock>>, Iterable<DuplicateWithLocation>> {
- INSTANCE;
-
- @Override
- @Nonnull
- public Iterable<DuplicateWithLocation> apply(@Nonnull final Map.Entry<DuplicateLocation, Set<TextBlock>> entry) {
- return from(entry.getValue())
- .transform(new Function<TextBlock, DuplicateWithLocation>() {
- @Override
- @Nonnull
- public DuplicateWithLocation apply(@Nonnull TextBlock input) {
- return new DuplicateWithLocation(entry.getKey(), input);
- }
- });
- }
- }
- }
-
- @Immutable
- private static final class DuplicateWithLocation {
- private final DuplicateLocation location;
- private final TextBlock duplicate;
-
- private DuplicateWithLocation(DuplicateLocation location, TextBlock duplicate) {
- this.location = location;
- this.duplicate = duplicate;
- }
-
- public DuplicateLocation getLocation() {
- return location;
- }
-
- public TextBlock getDuplicate() {
- return duplicate;
- }
+ checkNotNull(duplication, "duplication can not be null");
+ duplications.put(file.getKey(), duplication);
}
private static void checkFileComponentArgument(Component file) {
- checkComponentArgument(file, "file");
- }
-
- private static void checkComponentArgument(Component file, String argName) {
- requireNonNull(file, format("%s can not be null", argName));
- checkArgument(file.getType() == Component.Type.FILE, "type of %s argument must be FILE", argName);
- }
-
- private static void checkDuplicateTextBlockArgument(TextBlock duplicate) {
- requireNonNull(duplicate, "duplicate can not be null");
- }
-
- private static void checkOriginalTextBlockArgument(TextBlock original) {
- requireNonNull(original, "original can not be null");
+ requireNonNull(file, "file can not be null");
+ checkArgument(file.getType() == Component.Type.FILE, "type of file must be FILE");
}
- private enum DuplicatesEntryToDuplication implements Function<Map.Entry<TextBlock, Duplicates>, Duplication> {
- INSTANCE;
-
- @Override
- @Nonnull
- public Duplication apply(@Nonnull Map.Entry<TextBlock, Duplicates> entry) {
- return new Duplication(
- entry.getKey(),
- from(entry.getValue().getDuplicates()).transform(DuplicateLocationEntryToDuplicate.INSTANCE));
- }
-
- private enum DuplicateLocationEntryToDuplicate implements Function<DuplicateWithLocation, Duplicate> {
- INSTANCE;
-
- @Override
- @Nonnull
- public Duplicate apply(@Nonnull DuplicateWithLocation input) {
- DuplicateLocation duplicateLocation = input.getLocation();
- if (duplicateLocation instanceof InnerDuplicationLocation) {
- return new InnerDuplicate(input.getDuplicate());
- }
- if (duplicateLocation instanceof InProjectDuplicationLocation) {
- return new InProjectDuplicate(((InProjectDuplicationLocation) duplicateLocation).getComponent(), input.getDuplicate());
- }
- if (duplicateLocation instanceof CrossProjectDuplicationLocation) {
- return new CrossProjectDuplicate(((CrossProjectDuplicationLocation) duplicateLocation).getFileKey(), input.getDuplicate());
- }
- throw new IllegalArgumentException("Unsupported DuplicationLocation type " + duplicateLocation.getClass());
- }
- }
- }
}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/computation/duplication/IntegrateCrossProjectDuplications.java b/server/sonar-server/src/main/java/org/sonar/server/computation/duplication/IntegrateCrossProjectDuplications.java
index efee7c5193d..2b2291a8685 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/computation/duplication/IntegrateCrossProjectDuplications.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/computation/duplication/IntegrateCrossProjectDuplications.java
@@ -20,7 +20,9 @@
package org.sonar.server.computation.duplication;
+import com.google.common.base.Function;
import com.google.common.base.Predicate;
+import com.google.common.collect.Iterables;
import java.util.Collection;
import java.util.HashMap;
import java.util.List;
@@ -91,21 +93,23 @@ public class IntegrateCrossProjectDuplications {
private void addDuplication(Component file, CloneGroup duplication) {
ClonePart originPart = duplication.getOriginPart();
- TextBlock originTextBlock = new TextBlock(originPart.getStartLine(), originPart.getEndLine());
- int clonePartCount = 0;
- for (ClonePart part : from(duplication.getCloneParts())
- .filter(new DoesNotMatchSameComponentKey(originPart.getResourceId()))) {
- clonePartCount++;
- if (clonePartCount > MAX_CLONE_PART_PER_GROUP) {
- LOGGER.warn("Too many duplication references on file {} for block at line {}. Keeping only the first {} references.",
- file.getKey(), originPart.getStartLine(), MAX_CLONE_PART_PER_GROUP);
- break;
- }
- duplicationRepository.addDuplication(file, originTextBlock, part.getResourceId(),
- new TextBlock(part.getStartLine(), part.getEndLine()));
+ Iterable<Duplicate> duplicates = convertClonePartsToDuplicates(file, duplication);
+ if (!Iterables.isEmpty(duplicates)) {
+ duplicationRepository.add(
+ file,
+ new Duplication(new TextBlock(originPart.getStartLine(), originPart.getEndLine()), duplicates)
+ );
}
}
+ private static Iterable<Duplicate> convertClonePartsToDuplicates(final Component file, CloneGroup duplication) {
+ final ClonePart originPart = duplication.getOriginPart();
+ return from(duplication.getCloneParts())
+ .filter(new DoesNotMatchSameComponentKey(originPart.getResourceId()))
+ .filter(new DuplicateLimiter(file, originPart))
+ .transform(ClonePartToCrossProjectDuplicate.INSTANCE);
+ }
+
private NumberOfUnitsNotLessThan getNumberOfUnitsNotLessThan(String language) {
NumberOfUnitsNotLessThan numberOfUnitsNotLessThan = numberOfUnitsByLanguage.get(language);
if (numberOfUnitsNotLessThan == null) {
@@ -153,4 +157,35 @@ public class IntegrateCrossProjectDuplications {
}
}
+ private static class DuplicateLimiter implements Predicate<ClonePart> {
+ private final Component file;
+ private final ClonePart originPart;
+ private int counter = 0;
+
+ public DuplicateLimiter(Component file, ClonePart originPart) {
+ this.file = file;
+ this.originPart = originPart;
+ }
+
+ @Override
+ public boolean apply(@Nonnull ClonePart input) {
+ if (counter == MAX_CLONE_PART_PER_GROUP) {
+ LOGGER.warn("Too many duplication references on file {} for block at line {}. Keeping only the first {} references.",
+ file.getKey(), originPart.getStartLine(), MAX_CLONE_PART_PER_GROUP);
+ }
+ return counter++ <= MAX_CLONE_GROUP_PER_FILE;
+ }
+ }
+
+ private enum ClonePartToCrossProjectDuplicate implements Function<ClonePart, Duplicate> {
+ INSTANCE;
+
+ @Override
+ @Nonnull
+ public Duplicate apply(@Nonnull ClonePart input) {
+ return new CrossProjectDuplicate(
+ input.getResourceId(),
+ new TextBlock(input.getStartLine(), input.getEndLine()));
+ }
+ }
}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/computation/duplication/TextBlock.java b/server/sonar-server/src/main/java/org/sonar/server/computation/duplication/TextBlock.java
index e317e7c9225..43a24c871e4 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/computation/duplication/TextBlock.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/computation/duplication/TextBlock.java
@@ -31,7 +31,7 @@ import static com.google.common.base.Preconditions.checkArgument;
* same start line, by smallest size (ie. lowest end line).
* </p>
*/
-public final class TextBlock implements Comparable<TextBlock> {
+public class TextBlock implements Comparable<TextBlock> {
private final int start;
private final int end;
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 a6a888731a4..2c655572357 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
@@ -24,11 +24,9 @@ import com.google.common.base.Function;
import com.google.common.base.Predicate;
import com.google.common.collect.Ordering;
import java.util.ArrayList;
-import java.util.Collection;
import java.util.Collections;
import java.util.List;
import java.util.Map;
-import java.util.Set;
import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import org.sonar.db.protobuf.DbFileSources;
@@ -37,12 +35,13 @@ import org.sonar.server.computation.duplication.InnerDuplicate;
import org.sonar.server.computation.duplication.TextBlock;
import static com.google.common.collect.FluentIterable.from;
+import static com.google.common.collect.Iterables.size;
public class DuplicationLineReader implements LineReader {
private final Map<TextBlock, Integer> duplicatedTextBlockIndexByTextBlock;
- public DuplicationLineReader(Set<Duplication> duplications) {
+ public DuplicationLineReader(Iterable<Duplication> duplications) {
this.duplicatedTextBlockIndexByTextBlock = createIndexOfDuplicatedTextBlocks(duplications);
}
@@ -70,7 +69,7 @@ public class DuplicationLineReader implements LineReader {
* index. It avoids false detections of changes in {@link DbFileSources.Line#getDuplicationList()}.
* </p>
*/
- private static Map<TextBlock, Integer> createIndexOfDuplicatedTextBlocks(Collection<Duplication> duplications) {
+ private static Map<TextBlock, Integer> createIndexOfDuplicatedTextBlocks(Iterable<Duplication> duplications) {
List<TextBlock> duplicatedTextBlocks = extractAllDuplicatedTextBlocks(duplications);
Collections.sort(duplicatedTextBlocks);
return from(duplicatedTextBlocks)
@@ -84,10 +83,10 @@ public class DuplicationLineReader implements LineReader {
* The returned list is mutable on purpose because it will be sorted.
* </p>
*
- * @see {@link #createIndexOfDuplicatedTextBlocks(Collection)}
+ * @see {@link #createIndexOfDuplicatedTextBlocks(Iterable)}
*/
- private static List<TextBlock> extractAllDuplicatedTextBlocks(Collection<Duplication> duplications) {
- List<TextBlock> duplicatedBlock = new ArrayList<>(duplications.size());
+ private static List<TextBlock> extractAllDuplicatedTextBlocks(Iterable<Duplication> duplications) {
+ List<TextBlock> duplicatedBlock = new ArrayList<>(size(duplications));
for (Duplication duplication : duplications) {
duplicatedBlock.add(duplication.getOriginal());
for (InnerDuplicate duplicate : from(duplication.getDuplicates()).filter(InnerDuplicate.class)) {
diff --git a/server/sonar-server/src/main/java/org/sonar/server/computation/step/DuplicationDataMeasuresStep.java b/server/sonar-server/src/main/java/org/sonar/server/computation/step/DuplicationDataMeasuresStep.java
index 0fbe43ff660..4cba9340e92 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/computation/step/DuplicationDataMeasuresStep.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/computation/step/DuplicationDataMeasuresStep.java
@@ -20,7 +20,6 @@
package org.sonar.server.computation.step;
-import java.util.Set;
import org.apache.commons.lang.StringEscapeUtils;
import org.sonar.server.computation.component.Component;
import org.sonar.server.computation.component.CrawlerDepthLimit;
@@ -39,6 +38,7 @@ import org.sonar.server.computation.measure.MeasureRepository;
import org.sonar.server.computation.metric.Metric;
import org.sonar.server.computation.metric.MetricRepository;
+import static com.google.common.collect.Iterables.isEmpty;
import static org.sonar.api.measures.CoreMetrics.DUPLICATIONS_DATA_KEY;
import static org.sonar.server.computation.component.ComponentVisitor.Order.PRE_ORDER;
@@ -75,8 +75,8 @@ public class DuplicationDataMeasuresStep implements ComputationStep {
@Override
public void visitFile(Component file) {
- Set<Duplication> duplications = duplicationRepository.getDuplications(file);
- if (!duplications.isEmpty()) {
+ Iterable<Duplication> duplications = duplicationRepository.getDuplications(file);
+ if (!isEmpty(duplications)) {
computeDuplications(file, duplications);
}
}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/computation/step/DuplicationMeasuresStep.java b/server/sonar-server/src/main/java/org/sonar/server/computation/step/DuplicationMeasuresStep.java
index 897d7f231b7..d7b448510d9 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/computation/step/DuplicationMeasuresStep.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/computation/step/DuplicationMeasuresStep.java
@@ -45,6 +45,7 @@ import org.sonar.server.computation.metric.Metric;
import org.sonar.server.computation.metric.MetricRepository;
import static com.google.common.collect.FluentIterable.from;
+import static com.google.common.collect.Iterables.isEmpty;
import static java.util.Objects.requireNonNull;
import static org.sonar.api.measures.CoreMetrics.COMMENT_LINES_KEY;
import static org.sonar.api.measures.CoreMetrics.DUPLICATED_BLOCKS_KEY;
@@ -123,9 +124,9 @@ public class DuplicationMeasuresStep implements ComputationStep {
}
private void initializeForFile(Component file) {
- Set<Duplication> duplications = requireNonNull(this.duplicationRepository, "DuplicationRepository missing")
+ Iterable<Duplication> duplications = requireNonNull(this.duplicationRepository, "DuplicationRepository missing")
.getDuplications(file);
- if (duplications.isEmpty()) {
+ if (isEmpty(duplications)) {
return;
}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/computation/step/LoadDuplicationsFromReportStep.java b/server/sonar-server/src/main/java/org/sonar/server/computation/step/LoadDuplicationsFromReportStep.java
index 3006266bfc2..540fc96fa07 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/computation/step/LoadDuplicationsFromReportStep.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/computation/step/LoadDuplicationsFromReportStep.java
@@ -19,6 +19,8 @@
*/
package org.sonar.server.computation.step;
+import com.google.common.base.Function;
+import javax.annotation.Nonnull;
import org.sonar.batch.protocol.output.BatchReport;
import org.sonar.core.util.CloseableIterator;
import org.sonar.server.computation.batch.BatchReportReader;
@@ -27,9 +29,16 @@ import org.sonar.server.computation.component.CrawlerDepthLimit;
import org.sonar.server.computation.component.DepthTraversalTypeAwareCrawler;
import org.sonar.server.computation.component.TreeRootHolder;
import org.sonar.server.computation.component.TypeAwareVisitorAdapter;
+import org.sonar.server.computation.duplication.DetailedTextBlock;
+import org.sonar.server.computation.duplication.Duplicate;
+import org.sonar.server.computation.duplication.Duplication;
import org.sonar.server.computation.duplication.DuplicationRepository;
+import org.sonar.server.computation.duplication.InProjectDuplicate;
+import org.sonar.server.computation.duplication.InnerDuplicate;
import org.sonar.server.computation.duplication.TextBlock;
+import static com.google.common.base.Preconditions.checkArgument;
+import static com.google.common.collect.FluentIterable.from;
import static org.sonar.server.computation.component.ComponentVisitor.Order.POST_ORDER;
/**
@@ -59,8 +68,9 @@ public class LoadDuplicationsFromReportStep implements ComputationStep {
public void visitFile(Component file) {
CloseableIterator<BatchReport.Duplication> duplications = batchReportReader.readComponentDuplications(file.getReportAttributes().getRef());
try {
+ int idGenerator = 1;
while (duplications.hasNext()) {
- loadDuplications(file, duplications.next());
+ loadDuplications(file, duplications.next(), idGenerator++);
}
} finally {
duplications.close();
@@ -69,18 +79,40 @@ public class LoadDuplicationsFromReportStep implements ComputationStep {
}).visit(treeRootHolder.getRoot());
}
- private void loadDuplications(Component file, BatchReport.Duplication duplication) {
- TextBlock original = convert(duplication.getOriginPosition());
- for (BatchReport.Duplicate duplicate : duplication.getDuplicateList()) {
- if (duplicate.hasOtherFileRef()) {
- duplicationRepository.addDuplication(file, original, treeRootHolder.getComponentByRef(duplicate.getOtherFileRef()), convert(duplicate.getRange()));
- } else {
- duplicationRepository.addDuplication(file, original, convert(duplicate.getRange()));
- }
- }
+ private void loadDuplications(Component file, BatchReport.Duplication duplication, int id) {
+ duplicationRepository.add(file,
+ new Duplication(
+ convert(duplication.getOriginPosition(), id),
+ from(duplication.getDuplicateList())
+ .transform(new BatchDuplicateToCeDuplicate(file))
+ ));
}
private static TextBlock convert(BatchReport.TextRange textRange) {
return new TextBlock(textRange.getStartLine(), textRange.getEndLine());
}
+
+ private static DetailedTextBlock convert(BatchReport.TextRange textRange, int id) {
+ return new DetailedTextBlock(id, textRange.getStartLine(), textRange.getEndLine());
+ }
+
+ private class BatchDuplicateToCeDuplicate implements Function<BatchReport.Duplicate, Duplicate> {
+ private final Component file;
+
+ private BatchDuplicateToCeDuplicate(Component file) {
+ this.file = file;
+ }
+
+ @Override
+ @Nonnull
+ public Duplicate apply(@Nonnull BatchReport.Duplicate input) {
+ if (input.hasOtherFileRef()) {
+ checkArgument(input.getOtherFileRef() != file.getReportAttributes().getRef(), "file and otherFile references can not be the same");
+ return new InProjectDuplicate(
+ treeRootHolder.getComponentByRef(input.getOtherFileRef()),
+ convert(input.getRange()));
+ }
+ return new InnerDuplicate(convert(input.getRange()));
+ }
+ }
}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/computation/duplication/DuplicationRepositoryImplTest.java b/server/sonar-server/src/test/java/org/sonar/server/computation/duplication/DuplicationRepositoryImplTest.java
index 65cab2715b5..72e4723aab4 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/computation/duplication/DuplicationRepositoryImplTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/computation/duplication/DuplicationRepositoryImplTest.java
@@ -23,12 +23,10 @@ import com.tngtech.java.junit.dataprovider.DataProvider;
import com.tngtech.java.junit.dataprovider.DataProviderRunner;
import com.tngtech.java.junit.dataprovider.UseDataProvider;
import java.util.Arrays;
-import java.util.Set;
import org.junit.Rule;
import org.junit.Test;
import org.junit.rules.ExpectedException;
import org.junit.runner.RunWith;
-import org.sonar.server.computation.batch.TreeRootHolderRule;
import org.sonar.server.computation.component.Component;
import org.sonar.server.computation.component.ReportComponent;
import org.sonar.server.util.WrapInSingleElementArray;
@@ -36,30 +34,20 @@ import org.sonar.server.util.WrapInSingleElementArray;
import static com.google.common.base.Predicates.equalTo;
import static com.google.common.base.Predicates.not;
import static com.google.common.collect.FluentIterable.from;
-import static java.lang.String.format;
import static org.assertj.core.api.Assertions.assertThat;
import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.when;
@RunWith(DataProviderRunner.class)
public class DuplicationRepositoryImplTest {
- @Rule
- public TreeRootHolderRule treeRootHolder = new TreeRootHolderRule();
-
private static final Component FILE_COMPONENT_1 = ReportComponent.builder(Component.Type.FILE, 1).build();
private static final Component FILE_COMPONENT_2 = ReportComponent.builder(Component.Type.FILE, 2).build();
- private static final Component FILE_COMPONENT_3 = ReportComponent.builder(Component.Type.FILE, 3).build();
- private static final TextBlock ORIGINAL_TEXTBLOCK = new TextBlock(1, 2);
- private static final TextBlock COPY_OF_ORIGINAL_TEXTBLOCK = new TextBlock(1, 2);
- private static final TextBlock DUPLICATE_TEXTBLOCK_1 = new TextBlock(15, 15);
- private static final TextBlock DUPLICATE_TEXTBLOCK_2 = new TextBlock(15, 16);
- private static final String FILE_KEY_1 = "1";
- private static final String FILE_KEY_2 = "2";
+ private static final Duplication SOME_DUPLICATION = createDuplication(1, 2);
@Rule
public ExpectedException expectedException = ExpectedException.none();
- private DuplicationRepository underTest = new DuplicationRepositoryImpl(treeRootHolder);
+ private DuplicationRepository underTest = new DuplicationRepositoryImpl();
@Test
public void getDuplications_throws_NPE_if_Component_argument_is_null() {
@@ -84,24 +72,18 @@ public class DuplicationRepositoryImplTest {
}
@Test
- public void addDuplication_inner_throws_NPE_if_file_argument_is_null() {
+ public void add_throws_NPE_if_file_argument_is_null() {
expectFileArgumentNPE();
- underTest.addDuplication(null, ORIGINAL_TEXTBLOCK, DUPLICATE_TEXTBLOCK_1);
- }
-
- @Test
- public void addDuplication_inner_throws_NPE_if_original_argument_is_null() {
- expectOriginalArgumentNPE();
-
- underTest.addDuplication(FILE_COMPONENT_1, null, DUPLICATE_TEXTBLOCK_1);
+ underTest.add(null, SOME_DUPLICATION);
}
@Test
- public void addDuplication_inner_throws_NPE_if_duplicate_argument_is_null() {
- expectDuplicateArgumentNPE();
+ public void addDuplication_inner_throws_NPE_if_duplication_argument_is_null() {
+ expectedException.expect(NullPointerException.class);
+ expectedException.expectMessage("duplication can not be null");
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, null);
+ underTest.add(FILE_COMPONENT_1, null);
}
@Test
@@ -111,317 +93,46 @@ public class DuplicationRepositoryImplTest {
Component component = mockComponentGetType(type);
- underTest.addDuplication(component, ORIGINAL_TEXTBLOCK, DUPLICATE_TEXTBLOCK_1);
+ underTest.add(component, SOME_DUPLICATION);
}
@Test
- public void addDuplication_inner_throws_IAE_if_original_and_duplicate_are_equal() {
- expectedException.expect(IllegalArgumentException.class);
- expectedException.expectMessage("original and duplicate TextBlocks can not be the same");
+ public void added_duplication_is_returned_as_is_by_getDuplications() {
+ underTest.add(FILE_COMPONENT_1, SOME_DUPLICATION);
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, COPY_OF_ORIGINAL_TEXTBLOCK);
- }
-
- @Test
- public void addDuplication_inner_throws_IAE_if_duplication_already_exists() {
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, DUPLICATE_TEXTBLOCK_1);
-
- expectedException.expect(IllegalArgumentException.class);
- expectedException.expectMessage(format(
- "Inner duplicate %s is already associated to original %s in file %s",
- DUPLICATE_TEXTBLOCK_1, ORIGINAL_TEXTBLOCK, FILE_COMPONENT_1.getKey()));
-
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, DUPLICATE_TEXTBLOCK_1);
- }
-
- @Test
- public void addDuplication_inner_throws_IAE_if_reverse_duplication_already_exists() {
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, DUPLICATE_TEXTBLOCK_1);
-
- expectedException.expect(IllegalArgumentException.class);
- expectedException.expectMessage(format(
- "Inner duplicate %s is already associated to original %s in file %s",
- ORIGINAL_TEXTBLOCK, DUPLICATE_TEXTBLOCK_1, FILE_COMPONENT_1.getKey()));
-
- underTest.addDuplication(FILE_COMPONENT_1, DUPLICATE_TEXTBLOCK_1, ORIGINAL_TEXTBLOCK);
- }
-
- @Test
- public void addDuplication_inner_throws_IAE_if_reverse_duplication_already_exists_and_duplicate_has_duplicates_of_its_own() {
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, DUPLICATE_TEXTBLOCK_1);
- underTest.addDuplication(FILE_COMPONENT_1, DUPLICATE_TEXTBLOCK_1, DUPLICATE_TEXTBLOCK_2);
-
- expectedException.expect(IllegalArgumentException.class);
- expectedException.expectMessage(format(
- "Inner duplicate %s is already associated to original %s in file %s",
- ORIGINAL_TEXTBLOCK, DUPLICATE_TEXTBLOCK_1, FILE_COMPONENT_1.getKey()));
-
- underTest.addDuplication(FILE_COMPONENT_1, DUPLICATE_TEXTBLOCK_1, ORIGINAL_TEXTBLOCK);
- }
-
- @Test
- public void addDuplication_inner_is_returned_by_getDuplications() {
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, DUPLICATE_TEXTBLOCK_1);
-
- Set<Duplication> duplications = underTest.getDuplications(FILE_COMPONENT_1);
+ Iterable<Duplication> duplications = underTest.getDuplications(FILE_COMPONENT_1);
assertThat(duplications).hasSize(1);
- assertDuplication(
- duplications.iterator().next(),
- ORIGINAL_TEXTBLOCK,
- new InnerDuplicate(DUPLICATE_TEXTBLOCK_1));
+ assertThat(duplications.iterator().next()).isSameAs(SOME_DUPLICATION);
assertNoDuplication(FILE_COMPONENT_2);
}
@Test
- public void addDuplication_inner_called_multiple_times_populate_a_single_Duplication() {
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, DUPLICATE_TEXTBLOCK_2);
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, DUPLICATE_TEXTBLOCK_1);
+ public void added_duplication_does_not_avoid_same_duplication_inserted_twice_but_only_one_is_returned() {
+ underTest.add(FILE_COMPONENT_1, SOME_DUPLICATION);
+ underTest.add(FILE_COMPONENT_1, SOME_DUPLICATION);
- Set<Duplication> duplications = underTest.getDuplications(FILE_COMPONENT_1);
+ Iterable<Duplication> duplications = underTest.getDuplications(FILE_COMPONENT_1);
assertThat(duplications).hasSize(1);
- assertDuplication(
- duplications.iterator().next(),
- ORIGINAL_TEXTBLOCK,
- new InnerDuplicate(DUPLICATE_TEXTBLOCK_1), new InnerDuplicate(DUPLICATE_TEXTBLOCK_2));
+ assertThat(duplications.iterator().next()).isSameAs(SOME_DUPLICATION);
assertNoDuplication(FILE_COMPONENT_2);
}
@Test
- public void addDuplication_inProject_throws_NPE_if_file_argument_is_null() {
- expectFileArgumentNPE();
-
- underTest.addDuplication(null, ORIGINAL_TEXTBLOCK, FILE_COMPONENT_2, DUPLICATE_TEXTBLOCK_1);
- }
-
- @Test
- public void addDuplication_inProject_throws_NPE_if_original_argument_is_null() {
- expectOriginalArgumentNPE();
+ public void added_duplications_are_returned_in_any_order_and_associated_to_the_right_file() {
+ underTest.add(FILE_COMPONENT_1, SOME_DUPLICATION);
+ underTest.add(FILE_COMPONENT_1, createDuplication(2, 4));
+ underTest.add(FILE_COMPONENT_1, createDuplication(2, 3));
+ underTest.add(FILE_COMPONENT_2, createDuplication(2, 3));
+ underTest.add(FILE_COMPONENT_2, createDuplication(1, 2));
- underTest.addDuplication(FILE_COMPONENT_1, null, FILE_COMPONENT_2, DUPLICATE_TEXTBLOCK_1);
+ assertThat(underTest.getDuplications(FILE_COMPONENT_1)).containsOnly(SOME_DUPLICATION, createDuplication(2, 3), createDuplication(2, 4));
+ assertThat(underTest.getDuplications(FILE_COMPONENT_2)).containsOnly(createDuplication(1, 2), createDuplication(2, 3));
}
- @Test
- public void addDuplication_inProject_throws_NPE_if_otherFile_argument_is_null() {
- expectedException.expect(NullPointerException.class);
- expectedException.expectMessage("otherFile can not be null");
-
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, (Component) null, DUPLICATE_TEXTBLOCK_1);
- }
-
- @Test
- @UseDataProvider("allComponentTypesButFile")
- public void addDuplication_inProject_throws_NPE_if_otherFile_type_is_not_FILE(Component.Type type) {
- expectedException.expect(IllegalArgumentException.class);
- expectedException.expectMessage("type of otherFile argument must be FILE");
-
- Component component = mockComponentGetType(type);
-
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, component, DUPLICATE_TEXTBLOCK_1);
- }
-
- @Test
- public void addDuplication_inProject_throws_NPE_if_duplicate_argument_is_null() {
- expectDuplicateArgumentNPE();
-
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, FILE_COMPONENT_2, null);
- }
-
- @Test
- @UseDataProvider("allComponentTypesButFile")
- public void addDuplication_inProject_throws_NPE_if_file_type_is_not_FILE(Component.Type type) {
- expectFileTypeIAE();
-
- Component component = mockComponentGetType(type);
-
- underTest.addDuplication(component, ORIGINAL_TEXTBLOCK, FILE_COMPONENT_2, DUPLICATE_TEXTBLOCK_1);
- }
-
- @Test
- public void addDuplication_inProject_throws_NPE_if_file_and_otherFile_are_the_same() {
- expectedException.expect(IllegalArgumentException.class);
- expectedException.expectMessage("file and otherFile Components can not be the same");
-
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, FILE_COMPONENT_1, DUPLICATE_TEXTBLOCK_1);
- }
-
- @Test
- public void addDuplication_inProject_throws_IAE_if_duplication_already_exists() {
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, FILE_COMPONENT_2, DUPLICATE_TEXTBLOCK_1);
-
- expectedException.expect(IllegalArgumentException.class);
- expectedException.expectMessage(format(
- "In-project duplicate %s in file %s is already associated to original %s in file %s",
- DUPLICATE_TEXTBLOCK_1, FILE_COMPONENT_2.getKey(), ORIGINAL_TEXTBLOCK, FILE_COMPONENT_1.getKey()));
-
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, FILE_COMPONENT_2, DUPLICATE_TEXTBLOCK_1);
- }
-
- @Test
- public void addDuplication_inProject_is_returned_by_getDuplications() {
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, FILE_COMPONENT_2, DUPLICATE_TEXTBLOCK_1);
-
- Set<Duplication> duplications = underTest.getDuplications(FILE_COMPONENT_1);
- assertThat(duplications).hasSize(1);
- assertDuplication(
- duplications.iterator().next(),
- ORIGINAL_TEXTBLOCK,
- new InProjectDuplicate(FILE_COMPONENT_2, DUPLICATE_TEXTBLOCK_1));
-
- assertNoDuplication(FILE_COMPONENT_2);
- }
-
- @Test
- public void addDuplication_inProject_called_multiple_times_populate_a_single_Duplication() {
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, FILE_COMPONENT_2, DUPLICATE_TEXTBLOCK_2);
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, FILE_COMPONENT_2, DUPLICATE_TEXTBLOCK_1);
-
- Set<Duplication> duplications = underTest.getDuplications(FILE_COMPONENT_1);
- assertThat(duplications).hasSize(1);
- assertDuplication(
- duplications.iterator().next(),
- ORIGINAL_TEXTBLOCK,
- new InProjectDuplicate(FILE_COMPONENT_2, DUPLICATE_TEXTBLOCK_1), new InProjectDuplicate(FILE_COMPONENT_2, DUPLICATE_TEXTBLOCK_2));
-
- assertNoDuplication(FILE_COMPONENT_2);
- }
-
- @Test
- public void addDuplication_inProject_called_multiple_times_with_different_components_populate_a_single_Duplication() {
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, FILE_COMPONENT_2, DUPLICATE_TEXTBLOCK_2);
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, FILE_COMPONENT_3, DUPLICATE_TEXTBLOCK_2);
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, FILE_COMPONENT_3, DUPLICATE_TEXTBLOCK_1);
-
- Set<Duplication> duplications = underTest.getDuplications(FILE_COMPONENT_1);
- assertThat(duplications).hasSize(1);
- assertDuplication(
- duplications.iterator().next(),
- ORIGINAL_TEXTBLOCK,
- new InProjectDuplicate(FILE_COMPONENT_2, DUPLICATE_TEXTBLOCK_2), new InProjectDuplicate(FILE_COMPONENT_3, DUPLICATE_TEXTBLOCK_1), new InProjectDuplicate(FILE_COMPONENT_3,
- DUPLICATE_TEXTBLOCK_2));
-
- assertNoDuplication(FILE_COMPONENT_2);
- assertNoDuplication(FILE_COMPONENT_3);
- }
-
- @Test
- public void addDuplication_crossProject_throws_NPE_if_file_argument_is_null() {
- expectFileArgumentNPE();
-
- underTest.addDuplication(null, ORIGINAL_TEXTBLOCK, FILE_KEY_1, DUPLICATE_TEXTBLOCK_1);
- }
-
- @Test
- public void addDuplication_crossProject_throws_NPE_if_original_argument_is_null() {
- expectOriginalArgumentNPE();
-
- underTest.addDuplication(FILE_COMPONENT_1, null, FILE_KEY_1, DUPLICATE_TEXTBLOCK_1);
- }
-
- @Test
- public void addDuplication_crossProject_throws_NPE_if_otherFileKey_argument_is_null() {
- expectedException.expect(NullPointerException.class);
- expectedException.expectMessage("otherFileKey can not be null");
-
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, (String) null, DUPLICATE_TEXTBLOCK_1);
- }
-
- @Test
- public void addDuplication_crossProject_throws_NPE_if_duplicate_argument_is_null() {
- expectDuplicateArgumentNPE();
-
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, FILE_KEY_1, null);
- }
-
- @Test
- public void addDuplication_crossProject_throws_IAE_if_otherFileKey_is_key_of_Component_in_the_project() {
- treeRootHolder.setRoot(ReportComponent.builder(Component.Type.PROJECT, 999)
- .addChildren(
- FILE_COMPONENT_1, FILE_COMPONENT_2
- )
- .build());
-
- expectedException.expect(IllegalArgumentException.class);
- expectedException.expectMessage(format("otherFileKey '%s' can not be the key of a Component in the project", FILE_COMPONENT_2.getKey()));
-
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, FILE_COMPONENT_2.getKey(), DUPLICATE_TEXTBLOCK_1);
- }
-
- @Test
- @UseDataProvider("allComponentTypesButFile")
- public void addDuplication_crossProject_throws_NPE_if_file_type_is_not_FILE(Component.Type type) {
- expectFileTypeIAE();
-
- Component component = mockComponentGetType(type);
-
- underTest.addDuplication(component, ORIGINAL_TEXTBLOCK, FILE_KEY_1, DUPLICATE_TEXTBLOCK_1);
- }
-
- @Test
- public void addDuplication_crossProject_throws_IAE_if_duplication_already_exists() {
- treeRootHolder.setRoot(FILE_COMPONENT_1);
-
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, FILE_KEY_1, DUPLICATE_TEXTBLOCK_1);
-
- expectedException.expect(IllegalArgumentException.class);
- expectedException.expectMessage(format(
- "Cross-project duplicate %s in file %s is already associated to original %s in file %s",
- DUPLICATE_TEXTBLOCK_1, FILE_KEY_1, ORIGINAL_TEXTBLOCK, FILE_COMPONENT_1.getKey()));
-
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, FILE_KEY_1, DUPLICATE_TEXTBLOCK_1);
- }
-
- @Test
- public void addDuplication_crossProject_is_returned_by_getDuplications() {
- treeRootHolder.setRoot(FILE_COMPONENT_1);
-
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, FILE_KEY_1, DUPLICATE_TEXTBLOCK_1);
-
- Set<Duplication> duplications = underTest.getDuplications(FILE_COMPONENT_1);
- assertThat(duplications).hasSize(1);
- assertDuplication(
- duplications.iterator().next(),
- ORIGINAL_TEXTBLOCK,
- new CrossProjectDuplicate(FILE_KEY_1, DUPLICATE_TEXTBLOCK_1));
-
- assertNoDuplication(FILE_COMPONENT_2);
- }
-
- @Test
- public void addDuplication_crossProject_called_multiple_times_populate_a_single_Duplication() {
- treeRootHolder.setRoot(FILE_COMPONENT_1);
-
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, FILE_KEY_1, DUPLICATE_TEXTBLOCK_2);
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, FILE_KEY_1, DUPLICATE_TEXTBLOCK_1);
-
- Set<Duplication> duplications = underTest.getDuplications(FILE_COMPONENT_1);
- assertThat(duplications).hasSize(1);
- assertDuplication(
- duplications.iterator().next(),
- ORIGINAL_TEXTBLOCK,
- new CrossProjectDuplicate(FILE_KEY_1, DUPLICATE_TEXTBLOCK_1), new CrossProjectDuplicate(FILE_KEY_1, DUPLICATE_TEXTBLOCK_2));
-
- assertNoDuplication(FILE_COMPONENT_2);
- }
-
- @Test
- public void addDuplication_crossProject_called_multiple_times_with_different_fileKeys_populate_a_single_Duplication() {
- treeRootHolder.setRoot(FILE_COMPONENT_1);
-
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, FILE_KEY_1, DUPLICATE_TEXTBLOCK_2);
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, FILE_KEY_2, DUPLICATE_TEXTBLOCK_2);
- underTest.addDuplication(FILE_COMPONENT_1, ORIGINAL_TEXTBLOCK, FILE_KEY_2, DUPLICATE_TEXTBLOCK_1);
-
- Set<Duplication> duplications = underTest.getDuplications(FILE_COMPONENT_1);
- assertThat(duplications).hasSize(1);
- assertDuplication(
- duplications.iterator().next(),
- ORIGINAL_TEXTBLOCK,
- new CrossProjectDuplicate(FILE_KEY_1, DUPLICATE_TEXTBLOCK_2), new CrossProjectDuplicate(FILE_KEY_2, DUPLICATE_TEXTBLOCK_1), new CrossProjectDuplicate(FILE_KEY_2,
- DUPLICATE_TEXTBLOCK_2));
-
- assertNoDuplication(FILE_COMPONENT_2);
+ private static Duplication createDuplication(int originalLine, int duplicateLine) {
+ return new Duplication(new TextBlock(originalLine, originalLine), Arrays.<Duplicate>asList(new InnerDuplicate(new TextBlock(duplicateLine, duplicateLine))));
}
@DataProvider
@@ -432,11 +143,6 @@ public class DuplicationRepositoryImplTest {
.toArray(Object[].class);
}
- private static void assertDuplication(Duplication duplication, TextBlock original, Duplicate... duplicates) {
- assertThat(duplication.getOriginal()).isEqualTo(original);
- assertThat(duplication.getDuplicates()).containsExactly(duplicates);
- }
-
private void assertNoDuplication(Component component) {
assertThat(underTest.getDuplications(component)).isEmpty();
}
@@ -446,19 +152,9 @@ public class DuplicationRepositoryImplTest {
expectedException.expectMessage("file can not be null");
}
- private void expectOriginalArgumentNPE() {
- expectedException.expect(NullPointerException.class);
- expectedException.expectMessage("original can not be null");
- }
-
- private void expectDuplicateArgumentNPE() {
- expectedException.expect(NullPointerException.class);
- expectedException.expectMessage("duplicate can not be null");
- }
-
private void expectFileTypeIAE() {
expectedException.expect(IllegalArgumentException.class);
- expectedException.expectMessage("type of file argument must be FILE");
+ expectedException.expectMessage("type of file must be FILE");
}
private Component mockComponentGetType(Component.Type type) {
diff --git a/server/sonar-server/src/test/java/org/sonar/server/computation/duplication/DuplicationRepositoryRule.java b/server/sonar-server/src/test/java/org/sonar/server/computation/duplication/DuplicationRepositoryRule.java
index 9feb25dcbf8..8d8ae1d2369 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/computation/duplication/DuplicationRepositoryRule.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/computation/duplication/DuplicationRepositoryRule.java
@@ -19,7 +19,12 @@
*/
package org.sonar.server.computation.duplication;
-import java.util.Set;
+import com.google.common.base.Function;
+import com.google.common.collect.ArrayListMultimap;
+import com.google.common.collect.Multimap;
+import java.util.Arrays;
+import javax.annotation.CheckForNull;
+import javax.annotation.Nonnull;
import org.junit.rules.ExternalResource;
import org.sonar.server.computation.batch.TreeRootHolderRule;
import org.sonar.server.computation.component.Component;
@@ -27,78 +32,139 @@ import org.sonar.server.computation.component.ComponentProvider;
import org.sonar.server.computation.component.TreeRootHolder;
import org.sonar.server.computation.component.TreeRootHolderComponentProvider;
+import static com.google.common.base.Preconditions.checkArgument;
+import static com.google.common.collect.FluentIterable.from;
+import static java.util.Objects.requireNonNull;
+
public class DuplicationRepositoryRule extends ExternalResource implements DuplicationRepository {
- private final TreeRootHolder treeRootHolder;
+ @CheckForNull
private final ComponentProvider componentProvider;
private DuplicationRepositoryImpl delegate;
+ private final Multimap<Component, TextBlock> componentRefsWithInnerDuplications = ArrayListMultimap.create();
+ private final Multimap<Component, TextBlock> componentRefsWithInProjectDuplications = ArrayListMultimap.create();
+ private final Multimap<Component, TextBlock> componentRefsWithCrossProjectDuplications = ArrayListMultimap.create();
private DuplicationRepositoryRule(TreeRootHolder treeRootHolder) {
- this.treeRootHolder = treeRootHolder;
this.componentProvider = new TreeRootHolderComponentProvider(treeRootHolder);
}
+ public DuplicationRepositoryRule() {
+ this.componentProvider = null;
+ }
+
public static DuplicationRepositoryRule create(TreeRootHolderRule treeRootHolder) {
return new DuplicationRepositoryRule(treeRootHolder);
}
+ public static DuplicationRepositoryRule create() {
+ return new DuplicationRepositoryRule();
+ }
+
@Override
protected void before() throws Throwable {
- this.delegate = new DuplicationRepositoryImpl(treeRootHolder);
+ this.delegate = new DuplicationRepositoryImpl();
}
@Override
protected void after() {
- this.componentProvider.reset();
+ if (this.componentProvider != null) {
+ this.componentProvider.reset();
+ }
+ this.componentRefsWithInnerDuplications.clear();
+ this.componentRefsWithInProjectDuplications.clear();
+ this.componentRefsWithCrossProjectDuplications.clear();
this.delegate = null;
}
- public Set<Duplication> getDuplications(int fileRef) {
- componentProvider.ensureInitialized();
+ public Iterable<Duplication> getDuplications(int fileRef) {
+ ensureComponentProviderInitialized();
return delegate.getDuplications(componentProvider.getByRef(fileRef));
}
- public DuplicationRepositoryRule addDuplication(int fileRef, TextBlock original, TextBlock duplicate) {
- componentProvider.ensureInitialized();
+ public void add(int fileRef, Duplication duplication) {
+ ensureComponentProviderInitialized();
+
+ delegate.add(componentProvider.getByRef(fileRef), duplication);
+ }
+
+ public DuplicationRepositoryRule addDuplication(int fileRef, TextBlock original, TextBlock... duplicates) {
+ ensureComponentProviderInitialized();
+ Component component = componentProvider.getByRef(fileRef);
+ checkArgument(!componentRefsWithInnerDuplications.containsEntry(component, original), "Inner duplications for file %s and original %s already set", fileRef, original);
+ checkArgument(!componentRefsWithInProjectDuplications.containsEntry(component, original), "InProject duplications for file %s and original %s already set. Use add(int, Duplication) instead", fileRef, original);
- delegate.addDuplication(componentProvider.getByRef(fileRef), original, duplicate);
+ componentRefsWithInnerDuplications.put(component, original);
+ delegate.add(
+ component,
+ new Duplication(
+ original,
+ from(Arrays.asList(duplicates)).transform(TextBlockToInnerDuplicate.INSTANCE))
+ );
return this;
}
public DuplicationRepositoryRule addDuplication(int fileRef, TextBlock original, int otherFileRef, TextBlock duplicate) {
- componentProvider.ensureInitialized();
-
- delegate.addDuplication(componentProvider.getByRef(fileRef), original, componentProvider.getByRef(otherFileRef), duplicate);
+ ensureComponentProviderInitialized();
+ Component component = componentProvider.getByRef(fileRef);
+ checkArgument(!componentRefsWithInProjectDuplications.containsEntry(component, original), "InProject duplications for file %s and original %s already set", fileRef, original);
+ checkArgument(!componentRefsWithInnerDuplications.containsEntry(component, original), "Inner duplications for file %s and original %s already set. Use add(int, Duplication) instead", fileRef, original);
+
+ componentRefsWithInProjectDuplications.put(component, original);
+ delegate.add(component,
+ new Duplication(
+ original,
+ Arrays.<Duplicate>asList(new InProjectDuplicate(componentProvider.getByRef(otherFileRef), duplicate))
+ )
+ );
return this;
}
public DuplicationRepositoryRule addDuplication(int fileRef, TextBlock original, String otherFileKey, TextBlock duplicate) {
- componentProvider.ensureInitialized();
-
- delegate.addDuplication(componentProvider.getByRef(fileRef), original, otherFileKey, duplicate);
+ ensureComponentProviderInitialized();
+ Component component = componentProvider.getByRef(fileRef);
+ checkArgument(!componentRefsWithCrossProjectDuplications.containsEntry(component, original), "CrossProject duplications for file %s and original %s already set", fileRef);
+
+ componentRefsWithCrossProjectDuplications.put(component, original);
+ delegate.add(componentProvider.getByRef(fileRef),
+ new Duplication(
+ original,
+ Arrays.<Duplicate>asList(new CrossProjectDuplicate(otherFileKey, duplicate))
+ )
+ );
return this;
}
@Override
- public Set<Duplication> getDuplications(Component file) {
+ public Iterable<Duplication> getDuplications(Component file) {
return delegate.getDuplications(file);
}
@Override
- public void addDuplication(Component file, TextBlock original, TextBlock duplicate) {
- delegate.addDuplication(file, original, duplicate);
+ public void add(Component file, Duplication duplication) {
+ TextBlock original = duplication.getOriginal();
+ checkArgument(!componentRefsWithInnerDuplications.containsEntry(file, original), "Inner duplications for file %s and original %s already set", file, original);
+ checkArgument(!componentRefsWithInProjectDuplications.containsEntry(file, original), "InProject duplications for file %s and original %s already set", file, original);
+ checkArgument(!componentRefsWithCrossProjectDuplications.containsEntry(file, original), "CrossProject duplications for file %s and original %s already set", file, original);
+
+ delegate.add(file, duplication);
}
- @Override
- public void addDuplication(Component file, TextBlock original, Component otherFile, TextBlock duplicate) {
- delegate.addDuplication(file, original, otherFile, duplicate);
+ private void ensureComponentProviderInitialized() {
+ requireNonNull(this.componentProvider, "Methods with component reference can not be used unless a TreeRootHolder has been provided when instantiating the rule");
+ this.componentProvider.ensureInitialized();
}
- @Override
- public void addDuplication(Component file, TextBlock original, String otherFileKey, TextBlock duplicate) {
- delegate.addDuplication(file, original, otherFileKey, duplicate);
+ private enum TextBlockToInnerDuplicate implements Function<TextBlock, Duplicate> {
+ INSTANCE;
+
+ @Override
+ @Nonnull
+ public Duplicate apply(@Nonnull TextBlock input) {
+ return new InnerDuplicate(input);
+ }
}
}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/computation/duplication/IntegrateCrossProjectDuplicationsTest.java b/server/sonar-server/src/test/java/org/sonar/server/computation/duplication/IntegrateCrossProjectDuplicationsTest.java
index 557426a572d..49d0c2f2f7e 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/computation/duplication/IntegrateCrossProjectDuplicationsTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/computation/duplication/IntegrateCrossProjectDuplicationsTest.java
@@ -21,6 +21,7 @@
package org.sonar.server.computation.duplication;
import java.util.ArrayList;
+import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import org.junit.Rule;
@@ -38,13 +39,6 @@ import static java.util.Arrays.asList;
import static java.util.Collections.singletonList;
import static org.apache.commons.lang.RandomStringUtils.randomAlphanumeric;
import static org.assertj.core.api.Assertions.assertThat;
-import static org.mockito.Matchers.any;
-import static org.mockito.Matchers.anyString;
-import static org.mockito.Matchers.eq;
-import static org.mockito.Mockito.mock;
-import static org.mockito.Mockito.times;
-import static org.mockito.Mockito.verify;
-import static org.mockito.Mockito.verifyNoMoreInteractions;
import static org.sonar.server.computation.component.Component.Type.FILE;
import static org.sonar.server.computation.component.ReportComponent.builder;
@@ -52,6 +46,8 @@ public class IntegrateCrossProjectDuplicationsTest {
@Rule
public LogTester logTester = new LogTester();
+ @Rule
+ public DuplicationRepositoryRule duplicationRepository = DuplicationRepositoryRule.create();
static final String XOO_LANGUAGE = "xoo";
@@ -65,8 +61,6 @@ public class IntegrateCrossProjectDuplicationsTest {
Settings settings = new Settings();
- DuplicationRepository duplicationRepository = mock(DuplicationRepository.class);
-
IntegrateCrossProjectDuplications underTest = new IntegrateCrossProjectDuplications(settings, duplicationRepository);
@Test
@@ -106,11 +100,9 @@ public class IntegrateCrossProjectDuplicationsTest {
underTest.computeCpd(ORIGIN_FILE, originBlocks, duplicatedBlocks);
- verify(duplicationRepository).addDuplication(
- ORIGIN_FILE,
- new TextBlock(30, 45),
- OTHER_FILE_KEY,
- new TextBlock(40, 55)
+ assertThat(duplicationRepository.getDuplications(ORIGIN_FILE))
+ .containsExactly(
+ crossProjectDuplication(new TextBlock(30, 45), OTHER_FILE_KEY, new TextBlock(40, 55))
);
}
@@ -140,11 +132,9 @@ public class IntegrateCrossProjectDuplicationsTest {
underTest.computeCpd(ORIGIN_FILE, originBlocks, duplicatedBlocks);
- verify(duplicationRepository).addDuplication(
- ORIGIN_FILE,
- new TextBlock(30, 45),
- OTHER_FILE_KEY,
- new TextBlock(40, 55)
+ assertThat(duplicationRepository.getDuplications(ORIGIN_FILE))
+ .containsExactly(
+ crossProjectDuplication(new TextBlock(30, 45), OTHER_FILE_KEY, new TextBlock(40, 55))
);
}
@@ -181,7 +171,7 @@ public class IntegrateCrossProjectDuplicationsTest {
underTest.computeCpd(ORIGIN_FILE, originBlocks, duplicatedBlocks);
- verifyNoMoreInteractions(duplicationRepository);
+ assertNoDuplicationAdded(ORIGIN_FILE);
}
@Test
@@ -210,7 +200,7 @@ public class IntegrateCrossProjectDuplicationsTest {
underTest.computeCpd(ORIGIN_FILE, originBlocks, duplicatedBlocks);
- verifyNoMoreInteractions(duplicationRepository);
+ assertNoDuplicationAdded(ORIGIN_FILE);
}
@Test
@@ -229,7 +219,7 @@ public class IntegrateCrossProjectDuplicationsTest {
underTest.computeCpd(ORIGIN_FILE, originBlocks, Collections.<Block>emptyList());
- verifyNoMoreInteractions(duplicationRepository);
+ assertNoDuplicationAdded(ORIGIN_FILE);
}
@Test
@@ -261,11 +251,9 @@ public class IntegrateCrossProjectDuplicationsTest {
underTest.computeCpd(javaFile, originBlocks, duplicatedBlocks);
- verify(duplicationRepository).addDuplication(
- ORIGIN_FILE,
- new TextBlock(30, 45),
- OTHER_FILE_KEY,
- new TextBlock(40, 55)
+ assertThat(duplicationRepository.getDuplications(ORIGIN_FILE))
+ .containsExactly(
+ crossProjectDuplication(new TextBlock(30, 45), OTHER_FILE_KEY, new TextBlock(40, 55))
);
}
@@ -294,11 +282,9 @@ public class IntegrateCrossProjectDuplicationsTest {
underTest.computeCpd(ORIGIN_FILE, originBlocks, duplicatedBlocks);
- verify(duplicationRepository).addDuplication(
- ORIGIN_FILE,
- new TextBlock(30, 45),
- OTHER_FILE_KEY,
- new TextBlock(40, 55)
+ assertThat(duplicationRepository.getDuplications(ORIGIN_FILE))
+ .containsExactly(
+ crossProjectDuplication(new TextBlock(30, 45), OTHER_FILE_KEY, new TextBlock(40, 55))
);
}
@@ -328,7 +314,9 @@ public class IntegrateCrossProjectDuplicationsTest {
assertThat(logTester.logs(LoggerLevel.WARN)).containsOnly(
"Too many duplication references on file " + ORIGIN_FILE_KEY + " for block at line 30. Keeping only the first 100 references.");
- verify(duplicationRepository, times(100)).addDuplication(eq(ORIGIN_FILE), any(TextBlock.class), anyString(), any(TextBlock.class));
+ Iterable<Duplication> duplications = duplicationRepository.getDuplications(ORIGIN_FILE);
+ assertThat(duplications).hasSize(1);
+ assertThat(duplications.iterator().next().getDuplicates()).hasSize(100);
}
@Test
@@ -359,8 +347,16 @@ public class IntegrateCrossProjectDuplicationsTest {
underTest.computeCpd(ORIGIN_FILE, originBlocks, duplicatedBlocks);
- verify(duplicationRepository, times(100)).addDuplication(eq(ORIGIN_FILE), any(TextBlock.class), anyString(), any(TextBlock.class));
+ assertThat(duplicationRepository.getDuplications(ORIGIN_FILE)).hasSize(100);
assertThat(logTester.logs(LoggerLevel.WARN)).containsOnly("Too many duplication groups on file " + ORIGIN_FILE_KEY + ". Keeping only the first 100 groups.");
}
+ private static Duplication crossProjectDuplication(TextBlock original, String otherFileKey, TextBlock duplicate) {
+ return new Duplication(original, Arrays.<Duplicate>asList(new CrossProjectDuplicate(otherFileKey, duplicate)));
+ }
+
+ private void assertNoDuplicationAdded(Component file) {
+ assertThat(duplicationRepository.getDuplications(file)).isEmpty();
+ }
+
}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/computation/step/LoadDuplicationsFromReportStepTest.java b/server/sonar-server/src/test/java/org/sonar/server/computation/step/LoadDuplicationsFromReportStepTest.java
index 5515924be2e..42b4c918a59 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/computation/step/LoadDuplicationsFromReportStepTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/computation/step/LoadDuplicationsFromReportStepTest.java
@@ -28,6 +28,7 @@ import org.sonar.server.computation.batch.BatchReportReaderRule;
import org.sonar.server.computation.batch.TreeRootHolderRule;
import org.sonar.server.computation.component.Component;
import org.sonar.server.computation.component.VisitException;
+import org.sonar.server.computation.duplication.DetailedTextBlock;
import org.sonar.server.computation.duplication.Duplicate;
import org.sonar.server.computation.duplication.Duplication;
import org.sonar.server.computation.duplication.DuplicationRepositoryRule;
@@ -85,7 +86,7 @@ public class LoadDuplicationsFromReportStepTest {
underTest.execute();
assertNoDuplication(FILE_1_REF);
- assertDuplications(FILE_2_REF, singleLineTextBlock(LINE), new InnerDuplicate(singleLineTextBlock(LINE + 1)));
+ assertDuplications(FILE_2_REF, singleLineDetailedTextBlock(1, LINE), new InnerDuplicate(singleLineTextBlock(LINE + 1)));
}
@Test
@@ -94,7 +95,7 @@ public class LoadDuplicationsFromReportStepTest {
underTest.execute();
- assertDuplications(FILE_1_REF, singleLineTextBlock(LINE), new InProjectDuplicate(treeRootHolder.getComponentByRef(FILE_2_REF), singleLineTextBlock(LINE + 1)));
+ assertDuplications(FILE_1_REF, singleLineDetailedTextBlock(1, LINE), new InProjectDuplicate(treeRootHolder.getComponentByRef(FILE_2_REF), singleLineTextBlock(LINE + 1)));
assertNoDuplication(FILE_2_REF);
}
@@ -118,21 +119,50 @@ public class LoadDuplicationsFromReportStepTest {
Component file1Component = treeRootHolder.getComponentByRef(FILE_1_REF);
assertThat(duplicationRepository.getDuplications(FILE_2_REF)).containsOnly(
duplication(
- singleLineTextBlock(LINE),
+ singleLineDetailedTextBlock(1, LINE),
new InnerDuplicate(singleLineTextBlock(LINE + 1)), new InnerDuplicate(singleLineTextBlock(LINE + 2)), new InProjectDuplicate(file1Component, singleLineTextBlock(LINE)),
new InProjectDuplicate(file1Component, singleLineTextBlock(LINE + 10))),
duplication(
- singleLineTextBlock(OTHER_LINE),
+ singleLineDetailedTextBlock(2, OTHER_LINE),
new InProjectDuplicate(file1Component, singleLineTextBlock(OTHER_LINE))
),
duplication(
- singleLineTextBlock(OTHER_LINE + 80),
+ singleLineDetailedTextBlock(3, OTHER_LINE + 80),
new InnerDuplicate(singleLineTextBlock(LINE)), new InnerDuplicate(singleLineTextBlock(LINE + 10))
)
);
}
@Test
+ public void loads_never_consider_originals_from_batch_on_same_lines_as_the_equals() {
+ reportReader.putDuplications(
+ FILE_2_REF,
+ createDuplication(
+ singleLineTextRange(LINE),
+ createInnerDuplicate(LINE + 1), createInnerDuplicate(LINE + 2), createInProjectDuplicate(FILE_1_REF, LINE + 2)),
+ createDuplication(
+ singleLineTextRange(LINE),
+ createInnerDuplicate(LINE + 2), createInnerDuplicate(LINE + 3), createInProjectDuplicate(FILE_1_REF, LINE + 2))
+ );
+
+ underTest.execute();
+
+ Component file1Component = treeRootHolder.getComponentByRef(FILE_1_REF);
+ assertThat(duplicationRepository.getDuplications(FILE_2_REF)).containsOnly(
+ duplication(
+ singleLineDetailedTextBlock(1, LINE),
+ new InnerDuplicate(singleLineTextBlock(LINE + 1)), new InnerDuplicate(singleLineTextBlock(LINE + 2)),
+ new InProjectDuplicate(file1Component, singleLineTextBlock(LINE + 2))
+ ),
+ duplication(
+ singleLineDetailedTextBlock(2, LINE),
+ new InnerDuplicate(singleLineTextBlock(LINE + 2)), new InnerDuplicate(singleLineTextBlock(LINE + 3)),
+ new InProjectDuplicate(file1Component, singleLineTextBlock(LINE + 2))
+ )
+ );
+ }
+
+ @Test
public void loads_duplication_with_otherFileRef_throws_IAE_if_component_does_not_exist() {
int line = 2;
reportReader.putDuplications(FILE_1_REF, createDuplication(singleLineTextRange(line), createInProjectDuplicate(666, line + 1)));
@@ -149,7 +179,7 @@ public class LoadDuplicationsFromReportStepTest {
reportReader.putDuplications(FILE_1_REF, createDuplication(singleLineTextRange(line), createInProjectDuplicate(FILE_1_REF, line + 1)));
expectedException.expect(VisitException.class);
- expectedException.expectCause(hasType(IllegalArgumentException.class).andMessage("file and otherFile Components can not be the same"));
+ expectedException.expectCause(hasType(IllegalArgumentException.class).andMessage("file and otherFile references can not be the same"));
underTest.execute();
}
@@ -166,6 +196,10 @@ public class LoadDuplicationsFromReportStepTest {
return new TextBlock(line, line);
}
+ private DetailedTextBlock singleLineDetailedTextBlock(int id, int line) {
+ return new DetailedTextBlock(id, line, line);
+ }
+
private static BatchReport.Duplication createDuplication(BatchReport.TextRange original, BatchReport.Duplicate... duplicates) {
BatchReport.Duplication.Builder builder = BatchReport.Duplication.newBuilder()
.setOriginPosition(original);
diff --git a/server/sonar-server/src/test/java/org/sonar/server/computation/step/PersistFileSourcesStepTest.java b/server/sonar-server/src/test/java/org/sonar/server/computation/step/PersistFileSourcesStepTest.java
index 83ba814824b..54749fb9011 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/computation/step/PersistFileSourcesStepTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/computation/step/PersistFileSourcesStepTest.java
@@ -20,6 +20,7 @@
package org.sonar.server.computation.step;
+import java.util.Arrays;
import org.junit.Before;
import org.junit.Rule;
import org.junit.Test;
@@ -38,7 +39,10 @@ import org.sonar.server.computation.batch.BatchReportReaderRule;
import org.sonar.server.computation.batch.TreeRootHolderRule;
import org.sonar.server.computation.component.Component;
import org.sonar.server.computation.component.ReportComponent;
+import org.sonar.server.computation.duplication.Duplicate;
+import org.sonar.server.computation.duplication.Duplication;
import org.sonar.server.computation.duplication.DuplicationRepositoryRule;
+import org.sonar.server.computation.duplication.InnerDuplicate;
import org.sonar.server.computation.duplication.TextBlock;
import org.sonar.server.computation.scm.Changeset;
import org.sonar.server.computation.scm.ScmInfoRepositoryRule;
@@ -242,7 +246,10 @@ public class PersistFileSourcesStepTest extends BaseStepTest {
public void persist_duplication() {
initBasicReport(1);
- duplicationRepository.addDuplication(FILE_REF, new TextBlock(1, 2), new TextBlock(3, 4));
+ duplicationRepository.add(
+ FILE_REF,
+ new Duplication(new TextBlock(1, 2), Arrays.<Duplicate>asList(new InnerDuplicate(new TextBlock(3, 4))))
+ );
underTest.execute();
diff --git a/server/sonar-server/src/test/java/org/sonar/server/computation/step/ReportDuplicationMeasuresStepTest.java b/server/sonar-server/src/test/java/org/sonar/server/computation/step/ReportDuplicationMeasuresStepTest.java
index 05f4ca5ffeb..10d6e802dbe 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/computation/step/ReportDuplicationMeasuresStepTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/computation/step/ReportDuplicationMeasuresStepTest.java
@@ -107,9 +107,7 @@ public class ReportDuplicationMeasuresStepTest {
@Test
public void compute_duplicated_blocks_one_for_original_one_for_each_InnerDuplicate() {
TextBlock original = new TextBlock(1, 1);
- duplicationRepository.addDuplication(FILE_1_REF, original, new TextBlock(2, 2));
- duplicationRepository.addDuplication(FILE_1_REF, original, new TextBlock(3, 3));
- duplicationRepository.addDuplication(FILE_1_REF, original, new TextBlock(2, 3));
+ duplicationRepository.addDuplication(FILE_1_REF, original, new TextBlock(2, 2), new TextBlock(3, 3), new TextBlock(2, 3));
underTest.execute();
@@ -213,8 +211,7 @@ public class ReportDuplicationMeasuresStepTest {
@Test
public void compute_duplicated_lines_counts_lines_from_original_and_InnerDuplicate_only_once() {
TextBlock original = new TextBlock(1, 12);
- duplicationRepository.addDuplication(FILE_1_REF, original, new TextBlock(10, 11));
- duplicationRepository.addDuplication(FILE_1_REF, original, new TextBlock(11, 15));
+ duplicationRepository.addDuplication(FILE_1_REF, original, new TextBlock(10, 11), new TextBlock(11, 15));
duplicationRepository.addDuplication(FILE_1_REF, new TextBlock(2, 2), new TextBlock(96, 96));
underTest.execute();
@@ -454,9 +451,11 @@ public class ReportDuplicationMeasuresStepTest {
private void addDuplicatedBlock(int fileRef, int blockCount) {
checkArgument(blockCount > 1, "BlockCount can not be less than 2");
TextBlock original = new TextBlock(1, 1);
+ TextBlock[] duplicates = new TextBlock[blockCount - 1];
for (int i = 10; i < blockCount + 9; i++) {
- duplicationRepository.addDuplication(fileRef, original, new TextBlock(i, i));
+ duplicates[i - 10] = new TextBlock(i, i);
}
+ duplicationRepository.addDuplication(fileRef, original, duplicates);
}
private void addRawMeasure(int componentRef, String metricKey, int value) {
diff --git a/server/sonar-web/src/main/js/apps/overview/domains/coverage-domain.js b/server/sonar-web/src/main/js/apps/overview/domains/coverage-domain.js
index 0b751209e84..0efb71a23a7 100644
--- a/server/sonar-web/src/main/js/apps/overview/domains/coverage-domain.js
+++ b/server/sonar-web/src/main/js/apps/overview/domains/coverage-domain.js
@@ -68,7 +68,7 @@ export const CoverageMain = React.createClass({
},
renderLoading () {
- return <div className="text-center">
+ return <div className="flex-1 text-center">
<i className="spinner spinner-margin"/>
</div>;
},
diff --git a/server/sonar-web/src/main/js/apps/overview/domains/debt-domain.js b/server/sonar-web/src/main/js/apps/overview/domains/debt-domain.js
index 392f7056abf..bde05fee367 100644
--- a/server/sonar-web/src/main/js/apps/overview/domains/debt-domain.js
+++ b/server/sonar-web/src/main/js/apps/overview/domains/debt-domain.js
@@ -98,7 +98,7 @@ export const DebtMain = React.createClass({
},
renderLoading () {
- return <div className="text-center">
+ return <div className="flex-1 text-center">
<i className="spinner spinner-margin"/>
</div>;
},
diff --git a/server/sonar-web/src/main/js/apps/overview/domains/duplications-domain.js b/server/sonar-web/src/main/js/apps/overview/domains/duplications-domain.js
index 861e82ad2b5..28a983f0b6a 100644
--- a/server/sonar-web/src/main/js/apps/overview/domains/duplications-domain.js
+++ b/server/sonar-web/src/main/js/apps/overview/domains/duplications-domain.js
@@ -62,7 +62,7 @@ export const DuplicationsMain = React.createClass({
},
renderLoading () {
- return <div className="text-center">
+ return <div className="flex-1 text-center">
<i className="spinner spinner-margin"/>
</div>;
},
diff --git a/server/sonar-web/src/main/js/apps/overview/domains/structure-domain.js b/server/sonar-web/src/main/js/apps/overview/domains/structure-domain.js
index 3c1563bde01..fc4c7013f76 100644
--- a/server/sonar-web/src/main/js/apps/overview/domains/structure-domain.js
+++ b/server/sonar-web/src/main/js/apps/overview/domains/structure-domain.js
@@ -58,7 +58,7 @@ export const StructureMain = React.createClass({
},
renderLoading () {
- return <div className="text-center">
+ return <div className="flex-1 text-center">
<i className="spinner spinner-margin"/>
</div>;
},