All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.sonar.plugins.groovy.cobertura.CoberturaReportParser Maven / Gradle / Ivy

The newest version!
/*
 * Sonar Groovy Plugin
 * Copyright (C) 2010-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.plugins.groovy.cobertura;

import com.google.common.collect.Lists;
import com.google.common.collect.Maps;

import org.apache.commons.lang.StringUtils;
import org.codehaus.staxmate.in.SMInputCursor;
import org.sonar.api.batch.fs.FileSystem;
import org.sonar.api.batch.fs.InputFile;
import org.sonar.api.batch.sensor.SensorContext;
import org.sonar.api.batch.sensor.coverage.CoverageType;
import org.sonar.api.batch.sensor.coverage.NewCoverage;
import org.sonar.api.utils.MessageException;
import org.sonar.api.utils.log.Logger;
import org.sonar.api.utils.log.Loggers;
import org.sonar.plugins.groovy.foundation.Groovy;
import org.sonar.plugins.groovy.utils.StaxParser;

import javax.annotation.CheckForNull;
import javax.annotation.Nullable;
import javax.xml.stream.XMLStreamException;

import java.io.File;
import java.text.ParseException;
import java.util.List;
import java.util.Map;

import static java.util.Locale.ENGLISH;
import static org.sonar.api.utils.ParsingUtils.parseNumber;

public class CoberturaReportParser {

  private static final Logger LOG = Loggers.get(CoberturaReportParser.class);

  private final SensorContext context;
  private final FileSystem fileSystem;
  private List sourceDirs = Lists.newArrayList();

  public CoberturaReportParser(SensorContext context, final FileSystem fileSystem) {
    this.context = context;
    this.fileSystem = fileSystem;
  }

  /**
   * Parse a Cobertura xml report and create measures accordingly
   */
  public void parseReport(File xmlFile) {
    try {
      parseSources(xmlFile);
      parsePackages(xmlFile);
    } catch (XMLStreamException e) {
      throw MessageException.of("Unable to parse Cobertura report.", e);
    }
  }

  private void parseSources(File xmlFile) throws XMLStreamException {
    StaxParser sourceParser = new StaxParser(rootCursor -> {
      rootCursor.advance();
      sourceDirs = collectSourceDirs(rootCursor.descendantElementCursor("source"));
    });
    sourceParser.parse(xmlFile);
  }

  private static List collectSourceDirs(SMInputCursor source) throws XMLStreamException {
    List directories = Lists.newLinkedList();
    while (source.getNext() != null) {
      String sourceDir = cleanSourceDir(source.getElemStringValue());
      if (StringUtils.isNotBlank(sourceDir)) {
        directories.add(sourceDir);
      }
    }
    return directories;
  }

  private static String cleanSourceDir(String sourceDir) {
    if (StringUtils.isNotBlank(sourceDir)) {
      return sourceDir.trim();
    }
    return sourceDir;
  }

  private void parsePackages(File xmlFile) throws XMLStreamException {
    StaxParser fileParser = new StaxParser(rootCursor -> {
      rootCursor.advance();
      collectPackageMeasures(rootCursor.descendantElementCursor("package"));
    });
    fileParser.parse(xmlFile);
  }

  private void collectPackageMeasures(SMInputCursor pack) throws XMLStreamException {
    while (pack.getNext() != null) {
      Map resultByFilename = Maps.newHashMap();
      collectFileMeasures(pack.descendantElementCursor("class"), resultByFilename);
      handleFileMeasures(resultByFilename);
    }
  }

  private static void handleFileMeasures(Map resultByFilename) {
    for (ParsingResult parsingResult : resultByFilename.values()) {
      if (parsingResult.inputFile != null && Groovy.KEY.equals(parsingResult.inputFile.language())) {
        parsingResult.coverage.save();
      } else {
        LOG.warn("File not found: {}", parsingResult.filename);
      }
    }
  }

  @CheckForNull
  private InputFile getInputFile(String filename, List sourceDirs) {
    for (String sourceDir : sourceDirs) {
      String fileAbsolutePath = sourceDir + "/" + filename;
      InputFile file = fileSystem.inputFile(fileSystem.predicates().hasAbsolutePath(fileAbsolutePath));
      if (file != null) {
        return file;
      }
    }
    return null;
  }

  private void collectFileMeasures(SMInputCursor clazz, Map resultByFilename)
          throws XMLStreamException {
    while (clazz.getNext() != null) {
      String fileName = clazz.getAttrValue("filename");
      ParsingResult parsingResult = resultByFilename.get(fileName);
      if (parsingResult == null) {
        InputFile inputFile = getInputFile(fileName, sourceDirs);
        NewCoverage onFile = context.newCoverage().onFile(inputFile).ofType(CoverageType.UNIT);
        parsingResult = new ParsingResult(fileName, inputFile, onFile);
        resultByFilename.put(fileName, parsingResult);
      }
      collectFileData(clazz, parsingResult);
    }
  }

  private static void collectFileData(SMInputCursor clazz, ParsingResult parsingResult) throws XMLStreamException {
    SMInputCursor line = clazz.childElementCursor("lines").advance().childElementCursor("line");
    while (line.getNext() != null) {
      int lineId = Integer.parseInt(line.getAttrValue("number"));
      boolean validLine = parsingResult.isValidLine(lineId);
      if (!validLine && parsingResult.fileExists()) {
        LOG.info("Hit on invalid line for file " + parsingResult.filename + " (line: " + lineId + "/" + parsingResult.inputFile.lines() + ")");
      }
      try {
        int hits = (int) parseNumber(line.getAttrValue("hits"), ENGLISH);
        if (validLine) {
          parsingResult.coverage = parsingResult.coverage.lineHits(lineId, hits);
        }
      } catch (ParseException e) {
        throw MessageException.of("Unable to parse Cobertura report.", e);
      }

      String isBranch = line.getAttrValue("branch");
      String text = line.getAttrValue("condition-coverage");
      if (validLine && StringUtils.equals(isBranch, "true") && StringUtils.isNotBlank(text)) {
        String[] conditions = StringUtils.split(StringUtils.substringBetween(text, "(", ")"), "/");
        parsingResult.coverage = parsingResult.coverage.conditions(lineId, Integer.parseInt(conditions[1]), Integer.parseInt(conditions[0]));
      }
    }
  }

  private static class ParsingResult {
    private final String filename;
    @Nullable
    private final InputFile inputFile;
    private NewCoverage coverage;

    public ParsingResult(String filename, @Nullable InputFile inputFile, NewCoverage coverage) {
      this.filename = filename;
      this.inputFile = inputFile;
      this.coverage = coverage;
    }

    public boolean isValidLine(int lineId) {
      return fileExists() && lineId> 0 && lineId <= inputFile.lines();
    }

    public boolean fileExists() {
      return inputFile != null;
    }
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy