org.sonar.plugins.php.phpunit.PhpUnitCoverageResultParser Maven / Gradle / Ivy
/*
* SonarQube PHP Plugin
* Copyright (C) 2010 SonarSource and Akram Ben Aissi
* [email protected]
*
* 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 02
*/
package org.sonar.plugins.php.phpunit;
import com.thoughtworks.xstream.XStream;
import org.apache.commons.io.IOUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.sonar.api.BatchExtension;
import org.sonar.api.batch.SensorContext;
import org.sonar.api.measures.CoreMetrics;
import org.sonar.api.measures.Measure;
import org.sonar.api.measures.PropertiesBuilder;
import org.sonar.api.resources.Project;
import org.sonar.api.resources.Resource;
import org.sonar.api.scan.filesystem.FileQuery;
import org.sonar.api.scan.filesystem.ModuleFileSystem;
import org.sonar.api.utils.ParsingUtils;
import org.sonar.api.utils.SonarException;
import org.sonar.plugins.php.api.Php;
import org.sonar.plugins.php.phpunit.xml.CoverageNode;
import org.sonar.plugins.php.phpunit.xml.FileNode;
import org.sonar.plugins.php.phpunit.xml.LineNode;
import org.sonar.plugins.php.phpunit.xml.MetricsNode;
import org.sonar.plugins.php.phpunit.xml.PackageNode;
import org.sonar.plugins.php.phpunit.xml.ProjectNode;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
/**
* The Class PhpUnitCoverageResultParser.
*/
public class PhpUnitCoverageResultParser implements BatchExtension {
// Used for debugging purposes to store measure by resource
private static final Map, Measure> MEASURES_BY_RESOURCE = new HashMap, Measure>();
private static final Logger LOG = LoggerFactory.getLogger(PhpUnitCoverageResultParser.class);
private final Project project;
private final SensorContext context;
private final ModuleFileSystem fileSystem;
/**
* Instantiates a new php unit coverage result parser.
*
* @param context the context
*/
public PhpUnitCoverageResultParser(Project project, SensorContext context, ModuleFileSystem fileSystem) {
super();
this.project = project;
this.context = context;
this.fileSystem = fileSystem;
}
/**
* Parses PHPUnit coverage file.
*
* @param coverageReportFile the coverage report file
*/
public void parse(File coverageReportFile) {
LOG.debug("Parsing file: " + coverageReportFile.getAbsolutePath());
parseFile(coverageReportFile);
}
/**
* Parses the file.
*
* @param coverageReportFile the coverage report file
*/
private void parseFile(File coverageReportFile) {
CoverageNode coverage = getCoverage(coverageReportFile);
List projects = coverage.getProjects();
if (projects != null && !projects.isEmpty()) {
ProjectNode projectNode = projects.get(0);
LOG.info("Project: " + projectNode.getName());
parseFileNodes(projectNode.getFiles());
parsePackagesNodes(projectNode.getPackages());
saveMeasureForMissingFiles();
}
}
/**
* Set default 0 value for files that do not have coverage metrics because they were not touched by any test,
* and thus not present in the coverage report file.
*/
private void saveMeasureForMissingFiles() {
for (File phpFile : fileSystem.files(FileQuery.onSource().onLanguage(Php.KEY))) {
org.sonar.api.resources.File resource = org.sonar.api.resources.File.fromIOFile(phpFile, project);
if (resource != null && context.getMeasure(resource, CoreMetrics.LINE_COVERAGE) == null) {
LOG.debug("Coverage metrics have not been set on '{}': default values will be inserted.", phpFile.getName());
context.saveMeasure(resource, CoreMetrics.LINE_COVERAGE, 0.0);
// for LINES_TO_COVER and UNCOVERED_LINES, we use NCLOC as an approximation
Measure ncloc = context.getMeasure(resource, CoreMetrics.NCLOC);
if (ncloc != null && context.getMeasure(CoreMetrics.LINES_TO_COVER) == null) {
context.saveMeasure(resource, CoreMetrics.LINES_TO_COVER, ncloc.getValue());
}
if (ncloc != null && context.getMeasure(CoreMetrics.UNCOVERED_LINES) == null) {
context.saveMeasure(resource, CoreMetrics.UNCOVERED_LINES, ncloc.getValue());
}
}
}
}
private void parsePackagesNodes(List packages) {
if (packages != null) {
for (PackageNode packageNode : packages) {
parseFileNodes(packageNode.getFiles());
}
}
}
private void parseFileNodes(List fileNodes) {
if (fileNodes != null) {
for (FileNode file : fileNodes) {
saveCoverageMeasure(file);
}
}
}
/**
* Saves the required metrics found on the fileNode
*
* @param fileNode the file
*/
protected void saveCoverageMeasure(FileNode fileNode) {
org.sonar.api.resources.File phpFile = org.sonar.api.resources.File.fromIOFile(new File(fileNode.getName()), project);
// Due to an unexpected behaviour in phpunit.coverage.xml containing references to covered source files, we have to check that the
// targeted file for coverage is not null.
if (phpFile != null) {
// Properties builder will generate the data associate with COVERAGE_LINE_HITS_DATA metrics.
// This should look like (lineNumner=Count) : 1=0;2=1;3=1....
PropertiesBuilder lineHits = new PropertiesBuilder(CoreMetrics.COVERAGE_LINE_HITS_DATA);
if (fileNode.getLines() != null) {
for (LineNode line : fileNode.getLines()) {
saveLineMeasure(line, lineHits);
}
}
MetricsNode metrics = fileNode.getMetrics();
Measure measure = lineHits.build();
logMeasureByResource(phpFile, measure);
context.saveMeasure(phpFile, measure);
// Save uncovered statements (lines)
double totalStatementsCount = metrics.getTotalStatementsCount();
double uncoveredLines = totalStatementsCount - metrics.getCoveredStatements();
double lineCoverage = 0;
if (metrics.getCoveredStatements() != 0) {
lineCoverage = metrics.getCoveredStatements() / totalStatementsCount;
}
context.saveMeasure(phpFile, CoreMetrics.LINES_TO_COVER, totalStatementsCount);
context.saveMeasure(phpFile, CoreMetrics.UNCOVERED_LINES, uncoveredLines);
context.saveMeasure(phpFile, CoreMetrics.LINE_COVERAGE, ParsingUtils.scaleValue(lineCoverage * 100.0));
}
}
private void logMeasureByResource(Resource resource, Measure measure) {
if (LOG.isDebugEnabled()) {
Measure alreadySaved = MEASURES_BY_RESOURCE.get(resource);
if (alreadySaved == null) {
MEASURES_BY_RESOURCE.put(resource, measure);
} else {
LOG.debug("Measure {} already saved for resoruce {}", measure, resource);
}
}
}
/**
* Save line measure.
*
* @param line the line
* @param lineHits the line hits
*/
private void saveLineMeasure(LineNode line, PropertiesBuilder lineHits) {
lineHits.add(line.getNum(), line.getCount());
}
/**
* Gets the coverage.
*
* @param coverageReportFile the coverage report file
* @return the coverage
*/
private CoverageNode getCoverage(File coverageReportFile) {
InputStream inputStream = null;
try {
XStream xstream = new XStream();
xstream.setClassLoader(getClass().getClassLoader());
xstream.aliasSystemAttribute("classType", "class");
xstream.processAnnotations(CoverageNode.class);
xstream.processAnnotations(ProjectNode.class);
xstream.processAnnotations(FileNode.class);
xstream.processAnnotations(MetricsNode.class);
xstream.processAnnotations(LineNode.class);
inputStream = new FileInputStream(coverageReportFile);
return (CoverageNode) xstream.fromXML(inputStream);
} catch (IOException e) {
throw new SonarException("Can't read phpUnit report: " + coverageReportFile.getName(), e);
} finally {
IOUtils.closeQuietly(inputStream);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy