/*
* Sonar C++ Plugin (Community)
* Copyright (C) 2010 Neticoa SAS France
* dev@sonar.codehaus.org
*
* 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.cxx.coverage;
import org.apache.commons.lang.StringUtils;
import org.codehaus.staxmate.in.SMHierarchicCursor;
import org.codehaus.staxmate.in.SMInputCursor;
import org.sonar.api.measures.CoverageMeasuresBuilder;
import org.sonar.api.utils.StaxParser;
import org.sonar.plugins.cxx.utils.CxxUtils;
import javax.xml.stream.XMLStreamException;
import java.io.File;
import java.util.Map;
/**
* {@inheritDoc}
*/
public class CoberturaParser implements CoverageParser {
/**
* {@inheritDoc}
*/
public void parseReport(File xmlFile, final Map<String, CoverageMeasuresBuilder> coverageData)
throws XMLStreamException
{
CxxUtils.LOG.info("Parsing report '{}'", xmlFile);
StaxParser parser = new StaxParser(new StaxParser.XmlStreamHandler() {
/**
* {@inheritDoc}
*/
public void stream(SMHierarchicCursor rootCursor) throws XMLStreamException {
rootCursor.advance();
collectPackageMeasures(rootCursor.descendantElementCursor("package"), coverageData);
}
});
parser.parse(xmlFile);
}
private void collectPackageMeasures(SMInputCursor pack, Map<String, CoverageMeasuresBuilder> coverageData)
throws XMLStreamException
{
while (pack.getNext() != null) {
collectFileMeasures(pack.descendantElementCursor("class"), coverageData);
}
}
private void collectFileMeasures(SMInputCursor clazz, Map<String, CoverageMeasuresBuilder> coverageData)
throws XMLStreamException
{
while (clazz.getNext() != null) {
String normalPath = CxxUtils.normalizePath(clazz.getAttrValue("filename"));
if(normalPath != null){
CoverageMeasuresBuilder builder = coverageData.get(normalPath);
if (builder == null) {
builder = CoverageMeasuresBuilder.create();
coverageData.put(normalPath, builder);
}
collectFileData(clazz, builder);
}
}
}
private void collectFileData(SMInputCursor clazz, CoverageMeasuresBuilder builder) throws XMLStreamException {
SMInputCursor line = clazz.childElementCursor("lines").advance().childElementCursor("line");
while (line.getNext() != null) {
int lineId = Integer.parseInt(line.getAttrValue("number"));
long noHits = Long.parseLong(line.getAttrValue("hits"));
if(noHits > Integer.MAX_VALUE){
CxxUtils.LOG.warn("Truncating the actual number of hits ({}) to the maximum number supported by Sonar ({})",
noHits, Integer.MAX_VALUE);
noHits = Integer.MAX_VALUE;
}
builder.setHits(lineId, (int)noHits);
String isBranch = line.getAttrValue("branch");
String text = line.getAttrValue("condition-coverage");
if (StringUtils.equals(isBranch, "true") && StringUtils.isNotBlank(text)) {
String[] conditions = StringUtils.split(StringUtils.substringBetween(text, "(", ")"), "/");
builder.setConditions(lineId, Integer.parseInt(conditions[1]), Integer.parseInt(conditions[0]));
}
}
}
@Override
public String toString() {
return getClass().getSimpleName();
}
}