diff options
Diffstat (limited to 'plugins/sonar-cpd-plugin')
6 files changed, 267 insertions, 133 deletions
diff --git a/plugins/sonar-cpd-plugin/src/main/java/org/sonar/plugins/cpd/CpdAnalyser.java b/plugins/sonar-cpd-plugin/src/main/java/org/sonar/plugins/cpd/CpdAnalyser.java index 1d3b74d4dc3..2b67db65921 100644 --- a/plugins/sonar-cpd-plugin/src/main/java/org/sonar/plugins/cpd/CpdAnalyser.java +++ b/plugins/sonar-cpd-plugin/src/main/java/org/sonar/plugins/cpd/CpdAnalyser.java @@ -19,13 +19,7 @@ */ package org.sonar.plugins.cpd; -import java.io.File; -import java.util.HashMap; -import java.util.Iterator; -import java.util.Map; - import net.sourceforge.pmd.cpd.TokenEntry; - import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.sonar.api.batch.CpdMapping; @@ -34,6 +28,11 @@ import org.sonar.api.resources.Project; import org.sonar.api.resources.Resource; import org.sonar.duplications.cpd.Match; +import java.io.File; +import java.util.HashMap; +import java.util.Iterator; +import java.util.Map; + public class CpdAnalyser { private static final Logger LOG = LoggerFactory.getLogger(CpdAnalyser.class); @@ -78,20 +77,22 @@ public class CpdAnalyser { continue; } - firstFileData.cumulate(secondFile, secondLine, firstLine, match.getLineCount()); + String resourceKey = SonarEngine.getFullKey(project, secondFile); + firstFileData.cumulate(resourceKey, secondLine, firstLine, match.getLineCount()); } } } - for (DuplicationsData data : duplicationsData.values()) { - data.save(); + for (Map.Entry<Resource, DuplicationsData> entry : duplicationsData.entrySet()) { + entry.getValue().save(context, entry.getKey()); } } private DuplicationsData getDuplicationsData(Map<Resource, DuplicationsData> fileContainer, Resource file) { DuplicationsData data = fileContainer.get(file); if (data == null) { - data = new DuplicationsData(file, context); + String resourceKey = SonarEngine.getFullKey(project, file); + data = new DuplicationsData(resourceKey, context); fileContainer.put(file, data); } return data; diff --git a/plugins/sonar-cpd-plugin/src/main/java/org/sonar/plugins/cpd/DuplicationsData.java b/plugins/sonar-cpd-plugin/src/main/java/org/sonar/plugins/cpd/DuplicationsData.java index 32247cee88e..754b9678568 100644 --- a/plugins/sonar-cpd-plugin/src/main/java/org/sonar/plugins/cpd/DuplicationsData.java +++ b/plugins/sonar-cpd-plugin/src/main/java/org/sonar/plugins/cpd/DuplicationsData.java @@ -19,48 +19,42 @@ */ package org.sonar.plugins.cpd; -import java.util.ArrayList; -import java.util.Collections; -import java.util.Comparator; -import java.util.HashSet; -import java.util.List; -import java.util.Set; - +import com.google.common.collect.Lists; +import com.google.common.collect.Sets; import org.sonar.api.batch.SensorContext; import org.sonar.api.measures.CoreMetrics; import org.sonar.api.measures.Measure; import org.sonar.api.resources.Resource; +import java.util.Collections; +import java.util.Comparator; +import java.util.List; +import java.util.Set; + public class DuplicationsData { - private Resource resource; - private Set<Integer> duplicatedLines = new HashSet<Integer>(); - private double duplicatedBlocks; - private List<XmlEntry> duplicationXMLEntries = new ArrayList<XmlEntry>(); + private final String resourceKey; + private final Set<Integer> duplicatedLines = Sets.newHashSet(); + private final List<XmlEntry> duplicationXMLEntries = Lists.newArrayList(); - private SensorContext context; + private double duplicatedBlocks; - public DuplicationsData(Resource resource, SensorContext context) { - this.resource = resource; - this.context = context; + public DuplicationsData(String resourceKey, SensorContext context) { + this.resourceKey = resourceKey; } - public void cumulate(String targetResource, int targetDuplicationStartLine, int duplicationStartLine, int duplicatedLines) { - duplicationXMLEntries.add(new XmlEntry(targetResource, targetDuplicationStartLine, duplicationStartLine, duplicatedLines)); + public void cumulate(String targetResourceKey, int targetDuplicationStartLine, int duplicationStartLine, int duplicatedLines) { + duplicationXMLEntries.add(new XmlEntry(targetResourceKey, targetDuplicationStartLine, duplicationStartLine, duplicatedLines)); for (int duplicatedLine = duplicationStartLine; duplicatedLine < duplicationStartLine + duplicatedLines; duplicatedLine++) { this.duplicatedLines.add(duplicatedLine); } } - public void cumulate(Resource targetResource, int targetDuplicationStartLine, int duplicationStartLine, int duplicatedLines) { - cumulate(context.saveResource(targetResource), targetDuplicationStartLine, duplicationStartLine, duplicatedLines); - } - public void incrementDuplicatedBlock() { duplicatedBlocks++; } - public void save() { + public void save(SensorContext context, Resource resource) { context.saveMeasure(resource, CoreMetrics.DUPLICATED_FILES, 1d); context.saveMeasure(resource, CoreMetrics.DUPLICATED_LINES, (double) duplicatedLines.size()); context.saveMeasure(resource, CoreMetrics.DUPLICATED_BLOCKS, duplicatedBlocks); @@ -86,11 +80,11 @@ public class DuplicationsData { } }; - private static final class XmlEntry { - private String target; - private int targetStartLine; - private int startLine; - private int lines; + private final class XmlEntry { + private final String target; + private final int targetStartLine; + private final int startLine; + private final int lines; private XmlEntry(String target, int targetStartLine, int startLine, int lines) { this.target = target; @@ -101,10 +95,11 @@ public class DuplicationsData { @Override public String toString() { - return new StringBuilder().append("<duplication lines=\"").append(lines) - .append("\" start=\"").append(startLine) - .append("\" target-start=\"").append(targetStartLine) - .append("\" target-resource=\"").append(target).append("\"/>") + return new StringBuilder() + .append("<g>") + .append("<b s=\"").append(startLine).append("\" l=\"").append(lines).append("\" r=\"").append(resourceKey).append("\" />") + .append("<b s=\"").append(targetStartLine).append("\" l=\"").append(lines).append("\" r=\"").append(target).append("\" />") + .append("</g>") .toString(); } } diff --git a/plugins/sonar-cpd-plugin/src/main/java/org/sonar/plugins/cpd/decorators/package-info.java b/plugins/sonar-cpd-plugin/src/main/java/org/sonar/plugins/cpd/decorators/package-info.java new file mode 100644 index 00000000000..cf8081c7f81 --- /dev/null +++ b/plugins/sonar-cpd-plugin/src/main/java/org/sonar/plugins/cpd/decorators/package-info.java @@ -0,0 +1,25 @@ +/* + * Sonar, open source software quality management tool. + * Copyright (C) 2008-2012 SonarSource + * mailto:contact AT sonarsource DOT com + * + * Sonar 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. + * + * Sonar 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 Sonar; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02 + */ + +@ParametersAreNonnullByDefault +package org.sonar.plugins.cpd.decorators; + +import javax.annotation.ParametersAreNonnullByDefault; + diff --git a/plugins/sonar-cpd-plugin/src/main/java/org/sonar/plugins/cpd/index/package-info.java b/plugins/sonar-cpd-plugin/src/main/java/org/sonar/plugins/cpd/index/package-info.java new file mode 100644 index 00000000000..d5b116c13a4 --- /dev/null +++ b/plugins/sonar-cpd-plugin/src/main/java/org/sonar/plugins/cpd/index/package-info.java @@ -0,0 +1,25 @@ +/* + * Sonar, open source software quality management tool. + * Copyright (C) 2008-2012 SonarSource + * mailto:contact AT sonarsource DOT com + * + * Sonar 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. + * + * Sonar 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 Sonar; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02 + */ + +@ParametersAreNonnullByDefault +package org.sonar.plugins.cpd.index; + +import javax.annotation.ParametersAreNonnullByDefault; + diff --git a/plugins/sonar-cpd-plugin/src/main/java/org/sonar/plugins/cpd/package-info.java b/plugins/sonar-cpd-plugin/src/main/java/org/sonar/plugins/cpd/package-info.java new file mode 100644 index 00000000000..7e779ad6353 --- /dev/null +++ b/plugins/sonar-cpd-plugin/src/main/java/org/sonar/plugins/cpd/package-info.java @@ -0,0 +1,25 @@ +/* + * Sonar, open source software quality management tool. + * Copyright (C) 2008-2012 SonarSource + * mailto:contact AT sonarsource DOT com + * + * Sonar 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. + * + * Sonar 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 Sonar; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02 + */ + +@ParametersAreNonnullByDefault +package org.sonar.plugins.cpd; + +import javax.annotation.ParametersAreNonnullByDefault; + diff --git a/plugins/sonar-cpd-plugin/src/test/java/org/sonar/plugins/cpd/CpdAnalyserTest.java b/plugins/sonar-cpd-plugin/src/test/java/org/sonar/plugins/cpd/CpdAnalyserTest.java index c17b7797565..ada79ec7400 100644 --- a/plugins/sonar-cpd-plugin/src/test/java/org/sonar/plugins/cpd/CpdAnalyserTest.java +++ b/plugins/sonar-cpd-plugin/src/test/java/org/sonar/plugins/cpd/CpdAnalyserTest.java @@ -19,34 +19,30 @@ */ package org.sonar.plugins.cpd; -import static org.mockito.Matchers.anyList; -import static org.mockito.Matchers.anyObject; -import static org.mockito.Matchers.argThat; -import static org.mockito.Matchers.eq; -import static org.mockito.Mockito.atLeastOnce; -import static org.mockito.Mockito.mock; -import static org.mockito.Mockito.verify; -import static org.mockito.Mockito.when; - -import java.io.File; -import java.util.Arrays; -import java.util.Collections; -import java.util.LinkedHashSet; -import java.util.Set; - import net.sourceforge.pmd.cpd.TokenEntry; - import org.junit.Test; +import org.mockito.ArgumentCaptor; import org.sonar.api.batch.CpdMapping; import org.sonar.api.batch.SensorContext; import org.sonar.api.measures.CoreMetrics; +import org.sonar.api.measures.Measure; import org.sonar.api.resources.JavaFile; import org.sonar.api.resources.Project; import org.sonar.api.resources.ProjectFileSystem; import org.sonar.api.resources.Resource; -import org.sonar.api.test.IsMeasure; import org.sonar.duplications.cpd.Match; +import java.io.File; +import java.util.Arrays; +import java.util.Collections; +import java.util.LinkedHashSet; +import java.util.Set; + +import static org.hamcrest.Matchers.is; +import static org.junit.Assert.assertThat; +import static org.mockito.Matchers.*; +import static org.mockito.Mockito.*; + public class CpdAnalyserTest { @Test @@ -65,8 +61,6 @@ public class CpdAnalyserTest { Resource resource2 = new JavaFile("foo.Bar"); when(cpdMapping.createResource((File) anyObject(), anyList())).thenReturn(resource1).thenReturn(resource2).thenReturn(resource2) .thenReturn(resource1); - when(context.saveResource(resource1)).thenReturn("key1"); - when(context.saveResource(resource2)).thenReturn("key2"); Match match1 = new Match(5, new TokenEntry(null, file1.getAbsolutePath(), 5), new TokenEntry(null, file2.getAbsolutePath(), 15)); match1.setLineCount(200); @@ -74,24 +68,28 @@ public class CpdAnalyserTest { CpdAnalyser cpdAnalyser = new CpdAnalyser(project, context, cpdMapping); cpdAnalyser.analyse(Arrays.asList(match1).iterator()); + ArgumentCaptor<Measure> measureCaptor = ArgumentCaptor.forClass(Measure.class); + verify(context).saveMeasure(resource1, CoreMetrics.DUPLICATED_FILES, 1d); verify(context).saveMeasure(resource1, CoreMetrics.DUPLICATED_BLOCKS, 1d); verify(context).saveMeasure(resource1, CoreMetrics.DUPLICATED_LINES, 200d); - verify(context).saveMeasure( - eq(resource1), - argThat(new IsMeasure(CoreMetrics.DUPLICATIONS_DATA, "<duplications>" - + "<duplication lines=\"200\" start=\"5\" target-start=\"15\" target-resource=\"key2\"/>" + "</duplications>"))); + verify(context).saveMeasure(eq(resource1), measureCaptor.capture()); + Measure measure = measureCaptor.getValue(); + assertThat(measure.getMetric(), is(CoreMetrics.DUPLICATIONS_DATA)); + assertThat(measure.getData(), is("<duplications><g>" + + "<b s=\"5\" l=\"200\" r=\"key:foo.Foo\" />" + + "<b s=\"15\" l=\"200\" r=\"key:foo.Bar\" />" + + "</g></duplications>")); verify(context).saveMeasure(resource2, CoreMetrics.DUPLICATED_FILES, 1d); verify(context).saveMeasure(resource2, CoreMetrics.DUPLICATED_LINES, 200d); verify(context).saveMeasure(resource2, CoreMetrics.DUPLICATED_BLOCKS, 1d); - verify(context).saveMeasure( - eq(resource2), - argThat(new IsMeasure(CoreMetrics.DUPLICATIONS_DATA, - "<duplications><duplication lines=\"200\" start=\"15\" target-start=\"5\" target-resource=\"key1\"/></duplications>"))); - - verify(context, atLeastOnce()).saveResource(resource1); - verify(context, atLeastOnce()).saveResource(resource2); + verify(context).saveMeasure(eq(resource2), measureCaptor.capture()); + assertThat(measure.getMetric(), is(CoreMetrics.DUPLICATIONS_DATA)); + assertThat(measure.getData(), is("<duplications><g>" + + "<b s=\"5\" l=\"200\" r=\"key:foo.Foo\" />" + + "<b s=\"15\" l=\"200\" r=\"key:foo.Bar\" />" + + "</g></duplications>")); } @Test @@ -112,9 +110,6 @@ public class CpdAnalyserTest { Resource resource3 = new JavaFile("foo.Hotel"); when(cpdMapping.createResource((File) anyObject(), anyList())).thenReturn(resource1).thenReturn(resource2).thenReturn(resource2) .thenReturn(resource1).thenReturn(resource1).thenReturn(resource3).thenReturn(resource3).thenReturn(resource1); - when(context.saveResource(resource1)).thenReturn("key1"); - when(context.saveResource(resource2)).thenReturn("key2"); - when(context.saveResource(resource3)).thenReturn("key3"); Match match1 = new Match(5, new TokenEntry(null, file1.getAbsolutePath(), 5), new TokenEntry(null, file2.getAbsolutePath(), 15)); match1.setLineCount(200); @@ -124,35 +119,53 @@ public class CpdAnalyserTest { CpdAnalyser cpdAnalyser = new CpdAnalyser(project, context, cpdMapping); cpdAnalyser.analyse(Arrays.asList(match1, match2).iterator()); + ArgumentCaptor<Measure> measureCaptor = ArgumentCaptor.forClass(Measure.class); + verify(context).saveMeasure(resource1, CoreMetrics.DUPLICATED_FILES, 1d); verify(context).saveMeasure(resource1, CoreMetrics.DUPLICATED_BLOCKS, 2d); verify(context).saveMeasure(resource1, CoreMetrics.DUPLICATED_LINES, 200d); - verify(context).saveMeasure( - eq(resource1), - argThat(new IsMeasure(CoreMetrics.DUPLICATIONS_DATA, "<duplications>" - + "<duplication lines=\"100\" start=\"5\" target-start=\"15\" target-resource=\"key3\"/>" - + "<duplication lines=\"200\" start=\"5\" target-start=\"15\" target-resource=\"key2\"/>" - + "</duplications>"))); + verify(context).saveMeasure(eq(resource1), measureCaptor.capture()); + + Measure measure = measureCaptor.getValue(); + assertThat(measure.getMetric(), is(CoreMetrics.DUPLICATIONS_DATA)); + assertThat(measure.getData(), is("<duplications>" + + "<g>" + + "<b s=\"5\" l=\"100\" r=\"key:foo.Foo\" />" + + "<b s=\"15\" l=\"100\" r=\"key:foo.Hotel\" />" + + "</g>" + + "<g>" + + "<b s=\"5\" l=\"200\" r=\"key:foo.Foo\" />" + + "<b s=\"15\" l=\"200\" r=\"key:foo.Bar\" />" + + "</g>" + + "</duplications>")); verify(context).saveMeasure(resource2, CoreMetrics.DUPLICATED_FILES, 1d); verify(context).saveMeasure(resource2, CoreMetrics.DUPLICATED_LINES, 200d); verify(context).saveMeasure(resource2, CoreMetrics.DUPLICATED_BLOCKS, 1d); - verify(context).saveMeasure( - eq(resource2), - argThat(new IsMeasure(CoreMetrics.DUPLICATIONS_DATA, - "<duplications><duplication lines=\"200\" start=\"15\" target-start=\"5\" target-resource=\"key1\"/></duplications>"))); + verify(context).saveMeasure(eq(resource2), measureCaptor.capture()); + + measure = measureCaptor.getValue(); + assertThat(measure.getMetric(), is(CoreMetrics.DUPLICATIONS_DATA)); + assertThat(measure.getData(), is("<duplications>" + + "<g>" + + "<b s=\"15\" l=\"200\" r=\"key:foo.Bar\" />" + + "<b s=\"5\" l=\"200\" r=\"key:foo.Foo\" />" + + "</g>" + + "</duplications>")); verify(context).saveMeasure(resource3, CoreMetrics.DUPLICATED_FILES, 1d); verify(context).saveMeasure(resource3, CoreMetrics.DUPLICATED_LINES, 100d); verify(context).saveMeasure(resource3, CoreMetrics.DUPLICATED_BLOCKS, 1d); - verify(context).saveMeasure( - eq(resource3), - argThat(new IsMeasure(CoreMetrics.DUPLICATIONS_DATA, - "<duplications><duplication lines=\"100\" start=\"15\" target-start=\"5\" target-resource=\"key1\"/></duplications>"))); - - verify(context, atLeastOnce()).saveResource(resource1); - verify(context, atLeastOnce()).saveResource(resource2); - verify(context, atLeastOnce()).saveResource(resource3); + verify(context).saveMeasure(eq(resource3), measureCaptor.capture()); + + measure = measureCaptor.getValue(); + assertThat(measure.getMetric(), is(CoreMetrics.DUPLICATIONS_DATA)); + assertThat(measure.getData(), is("<duplications>" + + "<g>" + + "<b s=\"15\" l=\"100\" r=\"key:foo.Hotel\" />" + + "<b s=\"5\" l=\"100\" r=\"key:foo.Foo\" />" + + "</g>" + + "</duplications>")); } @Test @@ -177,10 +190,6 @@ public class CpdAnalyserTest { .thenReturn(resource4).thenReturn(resource2).thenReturn(resource1).thenReturn(resource3).thenReturn(resource4) .thenReturn(resource3).thenReturn(resource1).thenReturn(resource2).thenReturn(resource4).thenReturn(resource4) .thenReturn(resource1).thenReturn(resource2).thenReturn(resource3); - when(context.saveResource(resource1)).thenReturn("key1"); - when(context.saveResource(resource2)).thenReturn("key2"); - when(context.saveResource(resource3)).thenReturn("key3"); - when(context.saveResource(resource4)).thenReturn("key4"); Match match = new Match(5, createTokenEntry(file1.getAbsolutePath(), 5), createTokenEntry(file2.getAbsolutePath(), 15)); match.setLineCount(200); @@ -194,50 +203,95 @@ public class CpdAnalyserTest { CpdAnalyser cpdAnalyser = new CpdAnalyser(project, context, cpdMapping); cpdAnalyser.analyse(Arrays.asList(match).iterator()); + ArgumentCaptor<Measure> measureCaptor = ArgumentCaptor.forClass(Measure.class); + verify(context).saveMeasure(resource1, CoreMetrics.DUPLICATED_FILES, 1d); verify(context).saveMeasure(resource1, CoreMetrics.DUPLICATED_BLOCKS, 1d); verify(context).saveMeasure(resource1, CoreMetrics.DUPLICATED_LINES, 200d); - verify(context).saveMeasure( - eq(resource1), - argThat(new IsMeasure(CoreMetrics.DUPLICATIONS_DATA, "<duplications>" - + "<duplication lines=\"200\" start=\"5\" target-start=\"15\" target-resource=\"key2\"/>" - + "<duplication lines=\"200\" start=\"5\" target-start=\"7\" target-resource=\"key3\"/>" - + "<duplication lines=\"200\" start=\"5\" target-start=\"10\" target-resource=\"key4\"/>" + "</duplications>"))); + verify(context).saveMeasure(eq(resource1), measureCaptor.capture()); + + Measure measure = measureCaptor.getValue(); + assertThat(measure.getMetric(), is(CoreMetrics.DUPLICATIONS_DATA)); + assertThat(measure.getData(), is("<duplications>" + + "<g>" + + "<b s=\"5\" l=\"200\" r=\"key:foo.Foo\" />" + + "<b s=\"15\" l=\"200\" r=\"key:foo.Bar\" />" + + "</g>" + + "<g>" + + "<b s=\"5\" l=\"200\" r=\"key:foo.Foo\" />" + + "<b s=\"7\" l=\"200\" r=\"key:foo.Hotel\" />" + + "</g>" + + "<g>" + + "<b s=\"5\" l=\"200\" r=\"key:foo.Foo\" />" + + "<b s=\"10\" l=\"200\" r=\"key:foo.Coffee\" />" + + "</g>" + + "</duplications>")); verify(context).saveMeasure(resource3, CoreMetrics.DUPLICATED_FILES, 1d); verify(context).saveMeasure(resource3, CoreMetrics.DUPLICATED_LINES, 200d); verify(context).saveMeasure(resource3, CoreMetrics.DUPLICATED_BLOCKS, 1d); - verify(context).saveMeasure( - eq(resource2), - argThat(new IsMeasure(CoreMetrics.DUPLICATIONS_DATA, "<duplications>" - + "<duplication lines=\"200\" start=\"15\" target-start=\"5\" target-resource=\"key1\"/>" - + "<duplication lines=\"200\" start=\"15\" target-start=\"7\" target-resource=\"key3\"/>" - + "<duplication lines=\"200\" start=\"15\" target-start=\"10\" target-resource=\"key4\"/>" + "</duplications>"))); + verify(context).saveMeasure(eq(resource3), measureCaptor.capture()); + + measure = measureCaptor.getValue(); + assertThat(measure.getMetric(), is(CoreMetrics.DUPLICATIONS_DATA)); + assertThat(measure.getData(), is("<duplications>" + + "<g>" + + "<b s=\"7\" l=\"200\" r=\"key:foo.Hotel\" />" + + "<b s=\"5\" l=\"200\" r=\"key:foo.Foo\" />" + + "</g>" + + "<g>" + + "<b s=\"7\" l=\"200\" r=\"key:foo.Hotel\" />" + + "<b s=\"15\" l=\"200\" r=\"key:foo.Bar\" />" + + "</g>" + + "<g>" + + "<b s=\"7\" l=\"200\" r=\"key:foo.Hotel\" />" + + "<b s=\"10\" l=\"200\" r=\"key:foo.Coffee\" />" + + "</g>" + + "</duplications>")); verify(context).saveMeasure(resource2, CoreMetrics.DUPLICATED_FILES, 1d); verify(context).saveMeasure(resource2, CoreMetrics.DUPLICATED_LINES, 200d); verify(context).saveMeasure(resource2, CoreMetrics.DUPLICATED_BLOCKS, 1d); - verify(context).saveMeasure( - eq(resource3), - argThat(new IsMeasure(CoreMetrics.DUPLICATIONS_DATA, "<duplications>" - + "<duplication lines=\"200\" start=\"7\" target-start=\"5\" target-resource=\"key1\"/>" - + "<duplication lines=\"200\" start=\"7\" target-start=\"15\" target-resource=\"key2\"/>" - + "<duplication lines=\"200\" start=\"7\" target-start=\"10\" target-resource=\"key4\"/>" + "</duplications>"))); + verify(context).saveMeasure(eq(resource2), measureCaptor.capture()); + + measure = measureCaptor.getValue(); + assertThat(measure.getMetric(), is(CoreMetrics.DUPLICATIONS_DATA)); + assertThat(measure.getData(), is("<duplications>" + + "<g>" + + "<b s=\"15\" l=\"200\" r=\"key:foo.Bar\" />" + + "<b s=\"5\" l=\"200\" r=\"key:foo.Foo\" />" + + "</g>" + + "<g>" + + "<b s=\"15\" l=\"200\" r=\"key:foo.Bar\" />" + + "<b s=\"7\" l=\"200\" r=\"key:foo.Hotel\" />" + + "</g>" + + "<g>" + + "<b s=\"15\" l=\"200\" r=\"key:foo.Bar\" />" + + "<b s=\"10\" l=\"200\" r=\"key:foo.Coffee\" />" + + "</g>" + + "</duplications>")); verify(context).saveMeasure(resource4, CoreMetrics.DUPLICATED_LINES, 200d); verify(context).saveMeasure(resource4, CoreMetrics.DUPLICATED_FILES, 1d); verify(context).saveMeasure(resource4, CoreMetrics.DUPLICATED_BLOCKS, 1d); - verify(context).saveMeasure( - eq(resource4), - argThat(new IsMeasure(CoreMetrics.DUPLICATIONS_DATA, "<duplications>" - + "<duplication lines=\"200\" start=\"10\" target-start=\"5\" target-resource=\"key1\"/>" - + "<duplication lines=\"200\" start=\"10\" target-start=\"15\" target-resource=\"key2\"/>" - + "<duplication lines=\"200\" start=\"10\" target-start=\"7\" target-resource=\"key3\"/>" + "</duplications>"))); - - verify(context, atLeastOnce()).saveResource(resource1); - verify(context, atLeastOnce()).saveResource(resource2); - verify(context, atLeastOnce()).saveResource(resource3); - verify(context, atLeastOnce()).saveResource(resource4); + verify(context).saveMeasure(eq(resource4), measureCaptor.capture()); + + measure = measureCaptor.getValue(); + assertThat(measure.getMetric(), is(CoreMetrics.DUPLICATIONS_DATA)); + assertThat(measure.getData(), is("<duplications>" + + "<g>" + + "<b s=\"10\" l=\"200\" r=\"key:foo.Coffee\" />" + + "<b s=\"5\" l=\"200\" r=\"key:foo.Foo\" />" + + "</g>" + + "<g>" + + "<b s=\"10\" l=\"200\" r=\"key:foo.Coffee\" />" + + "<b s=\"15\" l=\"200\" r=\"key:foo.Bar\" />" + + "</g>" + + "<g>" + + "<b s=\"10\" l=\"200\" r=\"key:foo.Coffee\" />" + + "<b s=\"7\" l=\"200\" r=\"key:foo.Hotel\" />" + + "</g>" + + "</duplications>")); } @Test @@ -253,7 +307,6 @@ public class CpdAnalyserTest { CpdMapping cpdMapping = mock(CpdMapping.class); Resource resource1 = new JavaFile("foo.Foo"); when(cpdMapping.createResource((File) anyObject(), anyList())).thenReturn(resource1).thenReturn(resource1); - when(context.saveResource(resource1)).thenReturn("key1"); Match match1 = new Match(304, new TokenEntry(null, file1.getAbsolutePath(), 5), new TokenEntry(null, file1.getAbsolutePath(), 215)); match1.setLineCount(200); @@ -261,16 +314,26 @@ public class CpdAnalyserTest { CpdAnalyser cpdAnalyser = new CpdAnalyser(project, context, cpdMapping); cpdAnalyser.analyse(Arrays.asList(match1).iterator()); + ArgumentCaptor<Measure> measureCaptor = ArgumentCaptor.forClass(Measure.class); + verify(context).saveMeasure(resource1, CoreMetrics.DUPLICATED_FILES, 1d); verify(context).saveMeasure(resource1, CoreMetrics.DUPLICATED_LINES, 400d); verify(context).saveMeasure(resource1, CoreMetrics.DUPLICATED_BLOCKS, 2d); - verify(context).saveMeasure( - eq(resource1), - argThat(new IsMeasure(CoreMetrics.DUPLICATIONS_DATA, "<duplications>" - + "<duplication lines=\"200\" start=\"5\" target-start=\"215\" target-resource=\"key1\"/>" - + "<duplication lines=\"200\" start=\"215\" target-start=\"5\" target-resource=\"key1\"/>" + "</duplications>"))); - - verify(context, atLeastOnce()).saveResource(resource1); + verify(context).saveMeasure(eq(resource1), measureCaptor.capture()); + + Measure measure = measureCaptor.getValue(); + assertThat(measure.getMetric(), is(CoreMetrics.DUPLICATIONS_DATA)); + // FIXME in fact should be only one group - see SONAR-3131 + assertThat(measure.getData(), is("<duplications>" + + "<g>" + + "<b s=\"5\" l=\"200\" r=\"key:foo.Foo\" />" + + "<b s=\"215\" l=\"200\" r=\"key:foo.Foo\" />" + + "</g>" + + "<g>" + + "<b s=\"215\" l=\"200\" r=\"key:foo.Foo\" />" + + "<b s=\"5\" l=\"200\" r=\"key:foo.Foo\" />" + + "</g>" + + "</duplications>")); } private static TokenEntry createTokenEntry(String sourceId, int line) { |