aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--sonar-duplications/src/test/java/org/sonar/duplications/java/JavaDuplicationsFunctionalTest.java151
1 files changed, 144 insertions, 7 deletions
diff --git a/sonar-duplications/src/test/java/org/sonar/duplications/java/JavaDuplicationsFunctionalTest.java b/sonar-duplications/src/test/java/org/sonar/duplications/java/JavaDuplicationsFunctionalTest.java
index c9a1c1b90c7..dd85d6adcb5 100644
--- a/sonar-duplications/src/test/java/org/sonar/duplications/java/JavaDuplicationsFunctionalTest.java
+++ b/sonar-duplications/src/test/java/org/sonar/duplications/java/JavaDuplicationsFunctionalTest.java
@@ -22,13 +22,16 @@ package org.sonar.duplications.java;
import static org.hamcrest.Matchers.is;
import static org.junit.Assert.assertThat;
+import java.util.Collection;
import java.util.List;
import org.junit.Test;
import org.sonar.duplications.block.Block;
import org.sonar.duplications.block.BlockChunker;
-import org.sonar.duplications.detector.original.OriginalCloneDetectionAlgorithm;
+import org.sonar.duplications.detector.suffixtree.SuffixTreeCloneDetectionAlgorithm;
import org.sonar.duplications.index.CloneGroup;
+import org.sonar.duplications.index.CloneIndex;
+import org.sonar.duplications.index.ClonePart;
import org.sonar.duplications.index.MemoryCloneIndex;
import org.sonar.duplications.statement.Statement;
import org.sonar.duplications.statement.StatementChunker;
@@ -36,23 +39,157 @@ import org.sonar.duplications.token.TokenChunker;
import com.google.common.base.Joiner;
+/**
+ * From <a href="http://research.cs.queensu.ca/TechReports/Reports/2007-541.pdf">A Survey on Software Clone Detection Research (2007 year)</a>:
+ * <ul>
+ * <li>Type 1: Identical code fragments except for variations in whitespace (may be also variations in layout) and comments.
+ * Type 1 is widely know as Exact clones.</li>
+ * <li>Type 2: Structurally/syntactically identical fragments except for variations in identifiers, literals, types, layout and comments.
+ * The reserved words and the sentence structures are essentially the same.</li>
+ * <li>Type 3: Copied fragments with further modifications. Statements can be changed,
+ * added and/or deleted in addition to variations in identifiers, literals, types, layout
+ * and comments.</li>
+ * </ul>
+ */
public class JavaDuplicationsFunctionalTest {
+ @Test
+ public void type1() {
+ String fragment0 = source(
+ "if (a >= b) {",
+ " c = d + b; // Comment1",
+ " d = d + 1;}",
+ "else",
+ " c = d - a; // Comment2");
+ String fragment1 = source(
+ "if (a>=b) {",
+ " // Comment1",
+ " c=d+b;",
+ " d=d+1;",
+ "} else // Comment2",
+ " c=d-a;");
+ List<CloneGroup> duplications = detect2(fragment0, fragment1);
+ assertThat(duplications.size(), is(1));
+ ClonePart part = duplications.get(0).getOriginPart();
+ assertThat(part.getLineStart(), is(1));
+ assertThat(part.getLineEnd(), is(5));
+ }
+
+ /**
+ * Supports only subset of Type 2.
+ *
+ * @see #type2
+ * @see #literalsNormalization()
+ */
+ @Test
+ public void type2_literals() {
+ String fragment0 = source(
+ "if (a >= b) {",
+ " c = b + 1; // Comment1",
+ " d = '1';}",
+ "else",
+ " c = d - a; // Comment2");
+ String fragment1 = source(
+ "if (a >= b) {",
+ " c = b + 2; // Comment1",
+ " d = '2';}",
+ "else",
+ " c = d - a; // Comment2");
+ List<CloneGroup> duplications = detect2(fragment0, fragment1);
+ assertThat(duplications.size(), is(1));
+ ClonePart part = duplications.get(0).getOriginPart();
+ assertThat(part.getLineStart(), is(1));
+ assertThat(part.getLineEnd(), is(5));
+ }
+
+ @Test
+ public void type2() {
+ String fragment0 = source(
+ "if (a >= b) {",
+ " c = d + b; // Comment1",
+ " d = d + 1;}",
+ "else",
+ " c = d - a; // Comment2");
+ String fragment1 = source(
+ "if (m >= n) {",
+ " // Comment3",
+ " y = x + n; // Comment1",
+ " x = x + 5;}",
+ "else",
+ " y = x - m; // Comment2");
+ List<CloneGroup> duplications = detect2(fragment0, fragment1);
+ assertThat(duplications.size(), is(0));
+ }
+
+ /**
+ * Does not support Type 3, however able to detect inner parts.
+ */
+ @Test
+ public void type3() {
+ String fragment0 = source(
+ "public int getSoLinger() throws SocketException {",
+ " Object o = impl.getOption( SocketOptions.SO_LINGER);",
+ " if (o instanceof Integer) {",
+ " return((Integer) o).intValue();",
+ " }",
+ " else return -1;",
+ "}");
+ String fragment1 = source(
+ "public synchronized int getSoTimeout() throws SocketException {",
+ " Object o = impl.getOption( SocketOptions.SO_TIMEOUT);",
+ " if (o instanceof Integer) {",
+ " return((Integer) o).intValue();",
+ " }",
+ " else return -0;",
+ "}"
+ );
+ List<CloneGroup> duplications = detect2(fragment0, fragment1);
+ assertThat(duplications.size(), is(1));
+ ClonePart part = duplications.get(0).getOriginPart();
+ assertThat(part.getLineStart(), is(3));
+ assertThat(part.getLineEnd(), is(6));
+ }
+
+ private String source(String... lines) {
+ return Joiner.on('\n').join(lines);
+ }
+
+ private static List<CloneGroup> detect2(String... fragments) {
+ MemoryCloneIndex index = new MemoryCloneIndex();
+ for (int i = 0; i < fragments.length; i++) {
+ addToIndex(index, "fragment" + i, fragments[i]);
+ }
+ return detect(index, index.getByResourceId("fragment0"));
+ }
+
+ private static void addToIndex(CloneIndex index, String resourceId, String sourceCode) {
+ List<Statement> statements = STATEMENT_CHUNKER.chunk(TOKEN_CHUNKER.chunk(sourceCode));
+ BlockChunker blockChunker = new BlockChunker(2);
+ List<Block> blocks = blockChunker.chunk(resourceId, statements);
+ for (Block block : blocks) {
+ index.insert(block);
+ }
+ }
+
+ private static List<CloneGroup> detect(CloneIndex index, Collection<Block> blocks) {
+ return SuffixTreeCloneDetectionAlgorithm.detect(index, blocks);
+ }
+
private static final int BLOCK_SIZE = 1;
- private TokenChunker tokenChunker = JavaTokenProducer.build();
- private StatementChunker statementChunker = JavaStatementBuilder.build();
- private BlockChunker blockChunker = new BlockChunker(BLOCK_SIZE);
+ private static TokenChunker TOKEN_CHUNKER = JavaTokenProducer.build();
+ private static StatementChunker STATEMENT_CHUNKER = JavaStatementBuilder.build();
+ private static BlockChunker BLOCK_CHUNKER = new BlockChunker(BLOCK_SIZE);
private List<CloneGroup> detect(String... lines) {
String sourceCode = Joiner.on('\n').join(lines);
MemoryCloneIndex index = new MemoryCloneIndex();
- List<Statement> statements = statementChunker.chunk(tokenChunker.chunk(sourceCode));
- List<Block> blocks = blockChunker.chunk("resourceId", statements);
+ List<Statement> statements = STATEMENT_CHUNKER.chunk(TOKEN_CHUNKER.chunk(sourceCode));
+ List<Block> blocks = BLOCK_CHUNKER.chunk("resourceId", statements);
for (Block block : blocks) {
index.insert(block);
}
- return OriginalCloneDetectionAlgorithm.detect(index, blocks);
+ return detect(index, blocks);
}
/**