aboutsummaryrefslogtreecommitdiffstats
path: root/sonar-core
diff options
context:
space:
mode:
Diffstat (limited to 'sonar-core')
-rw-r--r--sonar-core/src/main/java/org/sonar/core/issue/tracking/BlockRecognizer.java28
-rw-r--r--sonar-core/src/main/java/org/sonar/core/issue/tracking/Trackable.java3
-rw-r--r--sonar-core/src/main/java/org/sonar/core/issue/tracking/Tracking.java6
-rw-r--r--sonar-core/src/test/java/org/sonar/core/issue/tracking/TrackerTest.java35
4 files changed, 56 insertions, 16 deletions
diff --git a/sonar-core/src/main/java/org/sonar/core/issue/tracking/BlockRecognizer.java b/sonar-core/src/main/java/org/sonar/core/issue/tracking/BlockRecognizer.java
index abda1ae3bf5..c7570f9f67c 100644
--- a/sonar-core/src/main/java/org/sonar/core/issue/tracking/BlockRecognizer.java
+++ b/sonar-core/src/main/java/org/sonar/core/issue/tracking/BlockRecognizer.java
@@ -75,21 +75,23 @@ class BlockRecognizer<RAW extends Trackable, BASE extends Trackable> {
}
}
- // Check if remaining number of lines exceeds threshold
- if (basesByLine.keySet().size() * rawsByLine.keySet().size() < 250000) {
- List<LinePair> possibleLinePairs = Lists.newArrayList();
- for (Integer baseLine : basesByLine.keySet()) {
- for (Integer rawLine : rawsByLine.keySet()) {
- int weight = lengthOfMaximalBlock(baseInput.getLineHashSequence(), baseLine, rawInput.getLineHashSequence(), rawLine);
- possibleLinePairs.add(new LinePair(baseLine, rawLine, weight));
- }
- }
- Collections.sort(possibleLinePairs, LinePairComparator.INSTANCE);
- for (LinePair linePair : possibleLinePairs) {
- // High probability that baseLine has been moved to rawLine, so we can map all Issues on baseLine to all Issues on rawLine
- map(rawsByLine.get(linePair.rawLine), basesByLine.get(linePair.baseLine), tracking);
+ // Check if remaining number of lines exceeds threshold. It avoids processing too many combinations.
+ if (basesByLine.keySet().size() * rawsByLine.keySet().size() >= 250_000) {
+ return;
+ }
+
+ List<LinePair> possibleLinePairs = Lists.newArrayList();
+ for (Integer baseLine : basesByLine.keySet()) {
+ for (Integer rawLine : rawsByLine.keySet()) {
+ int weight = lengthOfMaximalBlock(baseInput.getLineHashSequence(), baseLine, rawInput.getLineHashSequence(), rawLine);
+ possibleLinePairs.add(new LinePair(baseLine, rawLine, weight));
}
}
+ Collections.sort(possibleLinePairs, LinePairComparator.INSTANCE);
+ for (LinePair linePair : possibleLinePairs) {
+ // High probability that baseLine has been moved to rawLine, so we can map all issues on baseLine to all issues on rawLine
+ map(rawsByLine.get(linePair.rawLine), basesByLine.get(linePair.baseLine), tracking);
+ }
}
/**
diff --git a/sonar-core/src/main/java/org/sonar/core/issue/tracking/Trackable.java b/sonar-core/src/main/java/org/sonar/core/issue/tracking/Trackable.java
index 1c5d6f067e0..33e3351254b 100644
--- a/sonar-core/src/main/java/org/sonar/core/issue/tracking/Trackable.java
+++ b/sonar-core/src/main/java/org/sonar/core/issue/tracking/Trackable.java
@@ -31,6 +31,9 @@ public interface Trackable {
@CheckForNull
Integer getLine();
+ /**
+ * Trimmed message of issue
+ */
String getMessage();
@CheckForNull
diff --git a/sonar-core/src/main/java/org/sonar/core/issue/tracking/Tracking.java b/sonar-core/src/main/java/org/sonar/core/issue/tracking/Tracking.java
index 214bf7afc79..2f7e326a5c6 100644
--- a/sonar-core/src/main/java/org/sonar/core/issue/tracking/Tracking.java
+++ b/sonar-core/src/main/java/org/sonar/core/issue/tracking/Tracking.java
@@ -88,8 +88,10 @@ public class Tracking<RAW extends Trackable, BASE extends Trackable> {
}
void match(RAW raw, BASE base) {
- rawToBase.put(raw, base);
- baseToRaw.put(base, raw);
+ if (!rawToBase.containsKey(raw)) {
+ rawToBase.put(raw, base);
+ baseToRaw.put(base, raw);
+ }
}
boolean isComplete() {
diff --git a/sonar-core/src/test/java/org/sonar/core/issue/tracking/TrackerTest.java b/sonar-core/src/test/java/org/sonar/core/issue/tracking/TrackerTest.java
index b55d2fdd8b8..ef87a41dcfb 100644
--- a/sonar-core/src/test/java/org/sonar/core/issue/tracking/TrackerTest.java
+++ b/sonar-core/src/test/java/org/sonar/core/issue/tracking/TrackerTest.java
@@ -30,6 +30,7 @@ import org.junit.rules.ExpectedException;
import org.sonar.api.rule.RuleKey;
import static java.util.Arrays.asList;
+import static org.apache.commons.lang.StringUtils.trim;
import static org.assertj.core.api.Assertions.assertThat;
public class TrackerTest {
@@ -38,6 +39,8 @@ public class TrackerTest {
public static final RuleKey RULE_UNUSED_LOCAL_VARIABLE = RuleKey.of("java", "UnusedLocalVariable");
public static final RuleKey RULE_UNUSED_PRIVATE_METHOD = RuleKey.of("java", "UnusedPrivateMethod");
public static final RuleKey RULE_NOT_DESIGNED_FOR_EXTENSION = RuleKey.of("java", "NotDesignedForExtension");
+ public static final RuleKey RULE_USE_DIAMOND = RuleKey.of("java", "UseDiamond");
+
@Rule
public ExpectedException thrown = ExpectedException.none();
@@ -348,6 +351,36 @@ public class TrackerTest {
assertThat(tracking.getUnmatchedBases()).containsOnly(base2);
}
+ /**
+ * https://jira.sonarsource.com/browse/SONAR-7595
+ */
+ @Test
+ public void match_only_one_issue_when_multiple_blocks_match_the_same_block() {
+ FakeInput baseInput = FakeInput.createForSourceLines(
+ "public class Toto {",
+ " private final Deque<Set<DataItem>> one = new ArrayDeque<Set<DataItem>>();",
+ " private final Deque<Set<DataItem>> two = new ArrayDeque<Set<DataItem>>();",
+ " private final Deque<Integer> three = new ArrayDeque<Integer>();",
+ " private final Deque<Set<Set<DataItem>>> four = new ArrayDeque<Set<DataItem>>();");
+ Issue base1 = baseInput.createIssueOnLine(2, RULE_USE_DIAMOND, "Use diamond");
+ baseInput.createIssueOnLine(3, RULE_USE_DIAMOND, "Use diamond");
+ baseInput.createIssueOnLine(4, RULE_USE_DIAMOND, "Use diamond");
+ baseInput.createIssueOnLine(5, RULE_USE_DIAMOND, "Use diamond");
+
+ FakeInput rawInput = FakeInput.createForSourceLines(
+ "public class Toto {",
+ " // move all lines",
+ " private final Deque<Set<DataItem>> one = new ArrayDeque<Set<DataItem>>();",
+ " private final Deque<Set<DataItem>> two = new ArrayDeque<>();",
+ " private final Deque<Integer> three = new ArrayDeque<>();",
+ " private final Deque<Set<Set<DataItem>>> four = new ArrayDeque<>();");
+ Issue raw1 = rawInput.createIssueOnLine(3, RULE_USE_DIAMOND, "Use diamond");
+
+ Tracking<Issue, Issue> tracking = tracker.track(rawInput, baseInput);
+ assertThat(tracking.getUnmatchedBases()).hasSize(3);
+ assertThat(tracking.baseFor(raw1)).isEqualTo(base1);
+ }
+
private static class Issue implements Trackable {
private final RuleKey ruleKey;
private final Integer line;
@@ -357,7 +390,7 @@ public class TrackerTest {
this.line = line;
this.lineHash = lineHash;
this.ruleKey = ruleKey;
- this.message = message;
+ this.message = trim(message);
}
@Override