
org.sonar.server.computation.source.DuplicationLineReader Maven / Gradle / Ivy
/*
* SonarQube
* Copyright (C) 2009-2016 SonarSource SA
* mailto:contact AT sonarsource DOT com
*
* This program 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.
*
* This program 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.source;
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.Collections;
import java.util.List;
import java.util.Map;
import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import org.sonar.db.protobuf.DbFileSources;
import org.sonar.server.computation.duplication.Duplication;
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 duplicatedTextBlockIndexByTextBlock;
public DuplicationLineReader(Iterable duplications) {
this.duplicatedTextBlockIndexByTextBlock = createIndexOfDuplicatedTextBlocks(duplications);
}
@Override
public void read(DbFileSources.Line.Builder lineBuilder) {
Predicate> containsLine = new TextBlockContainsLine(lineBuilder.getLine());
for (Integer textBlockIndex : from(duplicatedTextBlockIndexByTextBlock.entrySet())
.filter(containsLine)
.transform(MapEntryToBlockId.INSTANCE)
// list is sorted to cope with the non-guaranteed order of Map entries which would trigger false detection of changes
// in {@link DbFileSources.Line#getDuplicationList()}
.toSortedList(Ordering.natural())) {
lineBuilder.addDuplication(textBlockIndex);
}
}
/**
*
*
* This method uses the natural order of TextBlocks to ensure that given the same set of TextBlocks, they get the same
* index. It avoids false detections of changes in {@link DbFileSources.Line#getDuplicationList()}.
*
*/
private static Map createIndexOfDuplicatedTextBlocks(Iterable duplications) {
List duplicatedTextBlocks = extractAllDuplicatedTextBlocks(duplications);
Collections.sort(duplicatedTextBlocks);
return from(duplicatedTextBlocks)
.toMap(new TextBlockIndexGenerator());
}
/**
* Duplicated blocks in the current file are either {@link Duplication#getOriginal()} or {@link Duplication#getDuplicates()}
* when the {@link org.sonar.server.computation.duplication.Duplicate} is a {@link InnerDuplicate}.
*
* The returned list is mutable on purpose because it will be sorted.
*
*
* @see {@link #createIndexOfDuplicatedTextBlocks(Iterable)}
*/
private static List extractAllDuplicatedTextBlocks(Iterable duplications) {
List duplicatedBlock = new ArrayList<>(size(duplications));
for (Duplication duplication : duplications) {
duplicatedBlock.add(duplication.getOriginal());
for (InnerDuplicate duplicate : from(duplication.getDuplicates()).filter(InnerDuplicate.class)) {
duplicatedBlock.add(duplicate.getTextBlock());
}
}
return duplicatedBlock;
}
private static class TextBlockContainsLine implements Predicate> {
private final int line;
public TextBlockContainsLine(int line) {
this.line = line;
}
@Override
public boolean apply(@Nonnull Map.Entry input) {
return isLineInBlock(input.getKey(), line);
}
private static boolean isLineInBlock(TextBlock range, int line) {
return line >= range.getStart() && line <= range.getEnd();
}
}
private enum MapEntryToBlockId implements Function, Integer> {
INSTANCE;
@Override
@Nonnull
public Integer apply(@Nonnull Map.Entry input) {
return input.getValue();
}
}
private static class TextBlockIndexGenerator implements Function {
int i = 1;
@Nullable
@Override
public Integer apply(TextBlock input) {
return i++;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy