
org.sonar.server.computation.step.DuplicationMeasuresStep 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.step;
import com.google.common.base.Optional;
import com.google.common.collect.ImmutableList;
import java.util.HashSet;
import java.util.Set;
import javax.annotation.CheckForNull;
import javax.annotation.Nullable;
import org.sonar.api.measures.CoreMetrics;
import org.sonar.server.computation.component.Component;
import org.sonar.server.computation.component.PathAwareCrawler;
import org.sonar.server.computation.component.TreeRootHolder;
import org.sonar.server.computation.duplication.Duplication;
import org.sonar.server.computation.duplication.DuplicationRepository;
import org.sonar.server.computation.duplication.InnerDuplicate;
import org.sonar.server.computation.duplication.TextBlock;
import org.sonar.server.computation.formula.Counter;
import org.sonar.server.computation.formula.CounterInitializationContext;
import org.sonar.server.computation.formula.CreateMeasureContext;
import org.sonar.server.computation.formula.Formula;
import org.sonar.server.computation.formula.FormulaExecutorComponentVisitor;
import org.sonar.server.computation.measure.Measure;
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.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;
import static org.sonar.api.measures.CoreMetrics.DUPLICATED_FILES_KEY;
import static org.sonar.api.measures.CoreMetrics.DUPLICATED_LINES_DENSITY_KEY;
import static org.sonar.api.measures.CoreMetrics.DUPLICATED_LINES_KEY;
import static org.sonar.api.measures.CoreMetrics.LINES_KEY;
import static org.sonar.api.measures.CoreMetrics.NCLOC_KEY;
/**
* Computes duplication measures on files and then aggregates them on higher components.
*
* This step must be executed after {@link CommentMeasuresStep} as it depends on {@link CoreMetrics#COMMENT_LINES}
*/
public class DuplicationMeasuresStep implements ComputationStep {
private final ImmutableList formulas;
private final TreeRootHolder treeRootHolder;
private final MetricRepository metricRepository;
private final MeasureRepository measureRepository;
public DuplicationMeasuresStep(TreeRootHolder treeRootHolder, MetricRepository metricRepository, MeasureRepository measureRepository,
@Nullable DuplicationRepository duplicationRepository) {
this.treeRootHolder = treeRootHolder;
this.metricRepository = metricRepository;
this.measureRepository = measureRepository;
this.formulas = ImmutableList.of(new DuplicationFormula(metricRepository, measureRepository, duplicationRepository));
}
/**
* Constructor used by Pico in Views where no DuplicationRepository is available.
*/
public DuplicationMeasuresStep(TreeRootHolder treeRootHolder, MetricRepository metricRepository, MeasureRepository measureRepository) {
this(treeRootHolder, metricRepository, measureRepository, null);
}
@Override
public String getDescription() {
return "Compute duplication measures";
}
@Override
public void execute() {
new PathAwareCrawler<>(
FormulaExecutorComponentVisitor.newBuilder(metricRepository, measureRepository).buildFor(formulas))
.visit(treeRootHolder.getRoot());
}
private static class DuplicationCounter implements Counter {
@CheckForNull
private final DuplicationRepository duplicationRepository;
private int fileCount = 0;
private int blockCount = 0;
private int lineCount = 0;
private DuplicationCounter(@Nullable DuplicationRepository duplicationRepository) {
this.duplicationRepository = duplicationRepository;
}
@Override
public void aggregate(DuplicationCounter counter) {
this.fileCount += counter.fileCount;
this.blockCount += counter.blockCount;
this.lineCount += counter.lineCount;
}
@Override
public void initialize(CounterInitializationContext context) {
Component leaf = context.getLeaf();
if (leaf.getType() == Component.Type.FILE) {
initializeForFile(leaf);
} else if (leaf.getType() == Component.Type.PROJECT_VIEW) {
initializeForProjectView(context);
}
}
private void initializeForFile(Component file) {
Iterable duplications = requireNonNull(this.duplicationRepository, "DuplicationRepository missing")
.getDuplications(file);
if (isEmpty(duplications)) {
return;
}
// use a set to count lines only once
Set duplicatedLineNumbers = new HashSet<>();
long blocks = 0;
for (Duplication duplication : duplications) {
blocks++;
addLines(duplication.getOriginal(), duplicatedLineNumbers);
for (InnerDuplicate innerDuplicate : from(duplication.getDuplicates()).filter(InnerDuplicate.class)) {
blocks++;
addLines(innerDuplicate.getTextBlock(), duplicatedLineNumbers);
}
}
this.fileCount += 1;
this.blockCount += blocks;
this.lineCount += duplicatedLineNumbers.size();
}
private static void addLines(TextBlock textBlock, Set duplicatedLineNumbers) {
for (int i = textBlock.getStart(); i <= textBlock.getEnd(); i++) {
duplicatedLineNumbers.add(i);
}
}
private void initializeForProjectView(CounterInitializationContext context) {
fileCount += getMeasure(context, DUPLICATED_FILES_KEY);
blockCount += getMeasure(context, DUPLICATED_BLOCKS_KEY);
lineCount += getMeasure(context, DUPLICATED_LINES_KEY);
}
private static int getMeasure(CounterInitializationContext context, String metricKey) {
Optional files = context.getMeasure(metricKey);
if (files.isPresent()) {
return files.get().getIntValue();
}
return 0;
}
}
private static final class DuplicationFormula implements Formula {
private final MeasureRepository measureRepository;
@CheckForNull
private final DuplicationRepository duplicationRepository;
private final Metric nclocMetric;
private final Metric linesMetric;
private final Metric commentLinesMetric;
private DuplicationFormula(MetricRepository metricRepository, MeasureRepository measureRepository, @Nullable DuplicationRepository duplicationRepository) {
this.measureRepository = measureRepository;
this.duplicationRepository = duplicationRepository;
this.nclocMetric = metricRepository.getByKey(NCLOC_KEY);
this.linesMetric = metricRepository.getByKey(LINES_KEY);
this.commentLinesMetric = metricRepository.getByKey(COMMENT_LINES_KEY);
}
@Override
public DuplicationCounter createNewCounter() {
return new DuplicationCounter(duplicationRepository);
}
@Override
public Optional createMeasure(DuplicationCounter counter, CreateMeasureContext context) {
switch (context.getMetric().getKey()) {
case DUPLICATED_FILES_KEY:
return Optional.of(Measure.newMeasureBuilder().create(counter.fileCount));
case DUPLICATED_LINES_KEY:
return Optional.of(Measure.newMeasureBuilder().create(counter.lineCount));
case DUPLICATED_LINES_DENSITY_KEY:
return createDuplicatedLinesDensityMeasure(counter, context);
case DUPLICATED_BLOCKS_KEY:
return Optional.of(Measure.newMeasureBuilder().create(counter.blockCount));
default:
throw new IllegalArgumentException("Unsupported metric " + context.getMetric());
}
}
private Optional createDuplicatedLinesDensityMeasure(DuplicationCounter counter, CreateMeasureContext context) {
int duplicatedLines = counter.lineCount;
Optional nbLines = getNbLinesFromLocOrNcloc(context);
if (nbLines.isPresent() && nbLines.get() > 0) {
double density = Math.min(100d, 100d * duplicatedLines / nbLines.get());
return Optional.of(Measure.newMeasureBuilder().create(density, context.getMetric().getDecimalScale()));
}
return Optional.absent();
}
private Optional getNbLinesFromLocOrNcloc(CreateMeasureContext context) {
Optional lines = measureRepository.getRawMeasure(context.getComponent(), linesMetric);
if (lines.isPresent()) {
return Optional.of(lines.get().getIntValue());
}
Optional nclocs = measureRepository.getRawMeasure(context.getComponent(), nclocMetric);
if (nclocs.isPresent()) {
Optional commentLines = measureRepository.getRawMeasure(context.getComponent(), commentLinesMetric);
int nbLines = nclocs.get().getIntValue();
return Optional.of(commentLines.isPresent() ? (nbLines + commentLines.get().getIntValue()) : nbLines);
}
return Optional.absent();
}
@Override
public String[] getOutputMetricKeys() {
return new String[] {DUPLICATED_FILES_KEY, DUPLICATED_LINES_KEY, DUPLICATED_LINES_DENSITY_KEY, DUPLICATED_BLOCKS_KEY};
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy