diff --git a/.dockerignore b/.dockerignore new file mode 100644 index 0000000..126708e --- /dev/null +++ b/.dockerignore @@ -0,0 +1,2 @@ +* +!target/ \ No newline at end of file diff --git a/.gitignore b/.gitignore index 6d8ca72..4621e2e 100644 --- a/.gitignore +++ b/.gitignore @@ -26,3 +26,5 @@ Icon? Thumbs.db # Folder config file Desktop.ini +its/plugin/projects/groovy-clover-sample/.sonar/* +its/plugin/projects/reuseReport/.sonar/* diff --git a/Dockerfile b/Dockerfile new file mode 100644 index 0000000..1e7efe3 --- /dev/null +++ b/Dockerfile @@ -0,0 +1,4 @@ +ARG VERSION=latest +FROM sonarqube:${VERSION} + +COPY target/sonar-clover-plugin.jar /opt/sonarqube/extensions/plugins/ \ No newline at end of file diff --git a/Makefile b/Makefile index cea34fd..12e75a6 100644 --- a/Makefile +++ b/Makefile @@ -1,6 +1,15 @@ run-test: ## Allows to run all unit tests @docker run --mount type=bind,src=$$(pwd),target=/usr/src -w /usr/src maven:alpine mvn test +run-integration-platform: ## Allows to run local integrations test with docker + @docker network create sonar || \ + docker build --build-arg VERSION=$$VERSION --tag test-instance . && \ + docker run -p 9000:9000 --name sonar-instance --net sonar test-instance + +run-integration-test: ## Allows to push a report in integration platform + @docker run --mount type=bind,src=$$(pwd)/its/integration,target=/usr/src -w /usr/src --net sonar maven:alpine \ + mvn clean clover:setup test clover:aggregate clover:clover sonar:sonar -Dsonar.sources=src -Dsonar.host.url=http://sonar-instance:9000 + build-package: ## Allows to build artifacts @docker run --mount type=bind,src=$$(pwd),target=/usr/src -w /usr/src maven:alpine mvn package diff --git a/README.md b/README.md index a471ee6..da2f627 100644 --- a/README.md +++ b/README.md @@ -3,10 +3,15 @@ Sonar Clover [![Build Status](https://travis-ci.org/sfeir-open-source/sonar-clover.svg?branch=master)](https://travis-ci.org/sfeir-open-source/sonar-clover) ## Description / Features -It provides the ability to feed SonarQube with code coverage data coming from Atlassian Clover. +It provides the ability to feed SonarQube with code coverage data coming from Atlassian Clover or it's new open source version: [OpenClover](http://openclover.org/). ## Usage To display code coverage data: -1. Prior to the SonarQube analysis, execute your unit tests and generate the Clover report. -1. Import this report while running the SonarQube analysis by setting the sonar.clover.reportPath property to the path to the Clover report. The path may be absolute or relative to the project base directory. +1. Prior to the SonarQube analysis, execute your unit tests and generate the [Clover report](http://openclover.org/doc/manual/latest/maven--quick-start-guide.html). + +1. Import this report while running the SonarQube analysis by setting the sonar.clover.reportPath (using prior version to sonarQube version 6) + or sonar.coverageReportPaths (sonarQube v7 or higher, FYI with this version, only the xml format is supported) property to the path to the Clover report. +The path may be absolute or relative to the project base directory. + +## diff --git a/contributing.md b/contributing.md new file mode 100644 index 0000000..3240803 --- /dev/null +++ b/contributing.md @@ -0,0 +1,19 @@ +## Contributions + +All contributions have to be test using a manual (for now) process + +## Test + +Each contributions should allows us to use this plugin in version LTS and Latest + +In order to test with different version of SonarQube, you could use: +```bash +VERSION= make run-integration-platform +``` +This command will start a sonarQube container with version : myVersion (`lts` or `latest` could be used) + +And run +```bash +make run-integration-test +``` +Which will do a mvn build on a test project and upload the report in the sonarQube container created above. diff --git a/its/integration/pom.xml b/its/integration/pom.xml new file mode 100644 index 0000000..248719b --- /dev/null +++ b/its/integration/pom.xml @@ -0,0 +1,41 @@ + + 4.0.0 + org.sonar.tests + integration + 1.0-SNAPSHOT + Sonar tests - clover reuse report + + + + + org.apache.maven.plugins + maven-compiler-plugin + + 1.8 + 1.8 + + + + org.openclover + clover-maven-plugin + 4.3.1 + + false + true + + + + + + + + + junit + junit + 4.7 + test + + + + diff --git a/its/integration/src/main/java/DTO.java b/its/integration/src/main/java/DTO.java new file mode 100644 index 0000000..845b8b5 --- /dev/null +++ b/its/integration/src/main/java/DTO.java @@ -0,0 +1,61 @@ +public class DTO { + private String item1; + + private String item2; + + private String item3; + + private int value1; + + private int value2; + + private int value3; + + public String getItem1() { + return item1; + } + + public void setItem1(String item1) { + this.item1 = item1; + } + + public String getItem2() { + return item2; + } + + public void setItem2(String item2) { + this.item2 = item2; + } + + public String getItem3() { + return item3; + } + + public void setItem3(String item3) { + this.item3 = item3; + } + + public int getValue1() { + return value1; + } + + public void setValue1(int value1) { + this.value1 = value1; + } + + public int getValue2() { + return value2; + } + + public void setValue2(int value2) { + this.value2 = value2; + } + + public int getValue3() { + return value3; + } + + public void setValue3(int value3) { + this.value3 = value3; + } +} diff --git a/its/integration/src/main/java/HelloWorld.java b/its/integration/src/main/java/HelloWorld.java new file mode 100644 index 0000000..29fd1e7 --- /dev/null +++ b/its/integration/src/main/java/HelloWorld.java @@ -0,0 +1,11 @@ +public class HelloWorld { + + public void neverCalled() { + System.out.println("Hello world!"); + } + + public boolean isPositive(int value) { + return value > 0 ? true : false; + } + +} diff --git a/its/integration/src/main/java/OmittedDTO.java b/its/integration/src/main/java/OmittedDTO.java new file mode 100644 index 0000000..f765f6e --- /dev/null +++ b/its/integration/src/main/java/OmittedDTO.java @@ -0,0 +1,61 @@ +public class OmittedDTO { + private String item1; + + private String item2; + + private String item3; + + private int value1; + + private int value2; + + private int value3; + + public String getItem1() { + return item1; + } + + public void setItem1(String item1) { + this.item1 = item1; + } + + public String getItem2() { + return item2; + } + + public void setItem2(String item2) { + this.item2 = item2; + } + + public String getItem3() { + return item3; + } + + public void setItem3(String item3) { + this.item3 = item3; + } + + public int getValue1() { + return value1; + } + + public void setValue1(int value1) { + this.value1 = value1; + } + + public int getValue2() { + return value2; + } + + public void setValue2(int value2) { + this.value2 = value2; + } + + public int getValue3() { + return value3; + } + + public void setValue3(int value3) { + this.value3 = value3; + } +} diff --git a/its/integration/src/test/java/HelloWorldTest.java b/its/integration/src/test/java/HelloWorldTest.java new file mode 100644 index 0000000..702a41f --- /dev/null +++ b/its/integration/src/test/java/HelloWorldTest.java @@ -0,0 +1,7 @@ +public class HelloWorldTest extends junit.framework.TestCase { + + public void testWillIncreaseCoverage() { + new HelloWorld().isPositive(0); + } + +} diff --git a/its/plugin/pom.xml b/its/plugin/pom.xml index ad9ec37..d929af1 100644 --- a/its/plugin/pom.xml +++ b/its/plugin/pom.xml @@ -22,18 +22,23 @@ org.sonarsource.orchestrator sonar-orchestrator - 3.10.1 + 3.15.2.1322 junit junit - 4.11 + 4.12 org.easytesting fest-assert 1.4 + + com.googlecode.json-simple + json-simple + 1.1.1 + diff --git a/its/plugin/projects/reuseReport/clover.xml b/its/plugin/projects/reuseReport/clover.xml index 5a8cbc1..c87799b 100644 --- a/its/plugin/projects/reuseReport/clover.xml +++ b/its/plugin/projects/reuseReport/clover.xml @@ -10,11 +10,27 @@ - - - - - + + + + + + + + + + + + + + + + + + + + + diff --git a/its/plugin/projects/reuseReport/pom.xml b/its/plugin/projects/reuseReport/pom.xml index 1a91ca7..89ae695 100644 --- a/its/plugin/projects/reuseReport/pom.xml +++ b/its/plugin/projects/reuseReport/pom.xml @@ -2,7 +2,7 @@ xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> 4.0.0 org.sonar.tests.cloverReuseReport - parent + reuseReport 1.0-SNAPSHOT Sonar tests - clover reuse report foo diff --git a/its/plugin/projects/reuseReport/src/main/java/DTO.java b/its/plugin/projects/reuseReport/src/main/java/DTO.java new file mode 100644 index 0000000..845b8b5 --- /dev/null +++ b/its/plugin/projects/reuseReport/src/main/java/DTO.java @@ -0,0 +1,61 @@ +public class DTO { + private String item1; + + private String item2; + + private String item3; + + private int value1; + + private int value2; + + private int value3; + + public String getItem1() { + return item1; + } + + public void setItem1(String item1) { + this.item1 = item1; + } + + public String getItem2() { + return item2; + } + + public void setItem2(String item2) { + this.item2 = item2; + } + + public String getItem3() { + return item3; + } + + public void setItem3(String item3) { + this.item3 = item3; + } + + public int getValue1() { + return value1; + } + + public void setValue1(int value1) { + this.value1 = value1; + } + + public int getValue2() { + return value2; + } + + public void setValue2(int value2) { + this.value2 = value2; + } + + public int getValue3() { + return value3; + } + + public void setValue3(int value3) { + this.value3 = value3; + } +} diff --git a/its/plugin/projects/reuseReport/src/main/java/OmittedDTO.java b/its/plugin/projects/reuseReport/src/main/java/OmittedDTO.java new file mode 100644 index 0000000..f765f6e --- /dev/null +++ b/its/plugin/projects/reuseReport/src/main/java/OmittedDTO.java @@ -0,0 +1,61 @@ +public class OmittedDTO { + private String item1; + + private String item2; + + private String item3; + + private int value1; + + private int value2; + + private int value3; + + public String getItem1() { + return item1; + } + + public void setItem1(String item1) { + this.item1 = item1; + } + + public String getItem2() { + return item2; + } + + public void setItem2(String item2) { + this.item2 = item2; + } + + public String getItem3() { + return item3; + } + + public void setItem3(String item3) { + this.item3 = item3; + } + + public int getValue1() { + return value1; + } + + public void setValue1(int value1) { + this.value1 = value1; + } + + public int getValue2() { + return value2; + } + + public void setValue2(int value2) { + this.value2 = value2; + } + + public int getValue3() { + return value3; + } + + public void setValue3(int value3) { + this.value3 = value3; + } +} diff --git a/its/plugin/src/test/java/com/sonar/clover/it/CloverTest.java b/its/plugin/src/test/java/com/sonar/clover/it/CloverTest.java index 0f01fe9..41ff1ea 100644 --- a/its/plugin/src/test/java/com/sonar/clover/it/CloverTest.java +++ b/its/plugin/src/test/java/com/sonar/clover/it/CloverTest.java @@ -22,10 +22,12 @@ import com.sonar.orchestrator.Orchestrator; import com.sonar.orchestrator.build.SonarScanner; import com.sonar.orchestrator.locator.FileLocation; +import org.json.simple.JSONArray; +import org.json.simple.JSONObject; +import org.json.simple.parser.JSONParser; +import org.json.simple.parser.ParseException; import org.junit.ClassRule; import org.junit.Test; -import org.sonar.wsclient.services.Resource; -import org.sonar.wsclient.services.ResourceQuery; import java.io.File; @@ -35,6 +37,8 @@ public class CloverTest { @ClassRule public static Orchestrator orchestrator = Orchestrator.builderEnv() + .setSonarVersion("6.7.5") + .setOrchestratorProperty("javaVersion", "LATEST_RELEASE") .addPlugin("java") .setOrchestratorProperty("groovyVersion", "LATEST_RELEASE") .addPlugin("groovy") @@ -56,13 +60,16 @@ public void reuse_report_project_java() { .setProjectVersion("1.0") .setSourceDirs("src/main/java") .setProjectDir(new File("projects/reuseReport")) - .setProperty("sonar.clover.reportPath", "clover.xml"); + .setProperty("sonar.clover.reportPath", "clover.xml") + .setProperty("sonar.java.binaries", "target/classes"); if (!orchestrator.getConfiguration().getPluginVersion("clover").isGreaterThan("2.9")) { analysis.setProperty("sonar.java.coveragePlugin", "clover"); } + orchestrator.executeBuild(analysis); - assertThat(getMeasure(project, "lines_to_cover")).isEqualTo(4); - assertThat(getMeasure(project, "uncovered_lines")).isEqualTo(2); + + assertThat(getMeasure(project, "lines_to_cover")).isEqualTo(8); + assertThat(getMeasure(project, "uncovered_lines")).isEqualTo(4); assertThat(getMeasure(file, "files")).isEqualTo(1); assertThat(getMeasure(file, "lines_to_cover")).isEqualTo(4); @@ -83,17 +90,26 @@ public void reuse_report_project_groovy() { .setLanguage("grvy") .setProjectDir(new File("projects/groovy-clover-sample")) .setProperty("sonar.clover.reportPath", "clover.xml"); + orchestrator.executeBuild(analysis); + assertThat(getMeasure(project, "lines_to_cover")).isEqualTo(2); assertThat(getMeasure(project, "uncovered_lines")).isEqualTo(0); + assertThat(getMeasure(groovyFile, "files")).isEqualTo(1); assertThat(getMeasure(groovyFile, "lines_to_cover")).isEqualTo(2); assertThat(getMeasure(groovyFile, "uncovered_lines")).isEqualTo(0); } private Integer getMeasure(String resourceKey, String metricKey) { - Resource resource = orchestrator.getServer().getWsClient().find(ResourceQuery.createForMetrics(resourceKey, metricKey)); - return resource != null ? resource.getMeasureIntValue(metricKey) : null; - } + final String body = orchestrator.getServer().newHttpCall("/api/measures/component").setParam("component", resourceKey).setParam("metricKeys", metricKey).execute().getBodyAsString(); + try { + final JSONObject componentJson = (JSONObject) ((JSONObject) new JSONParser().parse(body)).get("component"); + String result = (String) ((JSONObject) ((JSONArray) componentJson.get("measures")).get(0)).get("value"); + return result == null ? null : Integer.valueOf(result); + } catch (ParseException pe) { + return null; + } + } } diff --git a/pom.xml b/pom.xml index 6021f8c..cbf0fff 100644 --- a/pom.xml +++ b/pom.xml @@ -10,11 +10,10 @@ org.cheztone sonar-clover-plugin sonar-plugin - 3.2-SNAPSHOT + 4.0-SNAPSHOT Sonar Clover Plugin - - Atlassian Clover.]]> + OpenClover.]]> http://redirect.sonarsource.com/plugins/clover.html 2018 @@ -38,7 +37,7 @@ - 4.5.6 + 6.7.1 Clover org.sonar.plugins.clover.CloverPlugin SonarSource @@ -48,26 +47,39 @@ - org.codehaus.sonar + org.sonarsource.sonarqube sonar-plugin-api ${sonar.version} provided + + commons-lang + commons-lang + 2.6 + - - org.apache.maven - maven-project - 2.0.7 - provided + commons-io + commons-io + 2.4 + + + + org.slf4j + slf4j-api + 1.7.25 + + + + com.google.code.findbugs + jsr305 + 3.0.0 - org.codehaus.sonar + org.sonarsource.sonarqube sonar-testing-harness ${sonar.version} test diff --git a/src/main/java/org/sonar/plugins/clover/CloverPlugin.java b/src/main/java/org/sonar/plugins/clover/CloverPlugin.java index 5ecbaa3..4c992ef 100644 --- a/src/main/java/org/sonar/plugins/clover/CloverPlugin.java +++ b/src/main/java/org/sonar/plugins/clover/CloverPlugin.java @@ -19,23 +19,22 @@ */ package org.sonar.plugins.clover; +import org.sonar.api.Plugin; import org.sonar.api.Properties; import org.sonar.api.Property; -import org.sonar.api.SonarPlugin; -import java.util.Arrays; -import java.util.List; +import java.util.Collections; @Properties({ @Property( key = CloverSensor.REPORT_PATH_PROPERTY, - name = "Report path", - description = "Absolute or relative path to XML report file.", - project = true, global = true)}) -public final class CloverPlugin extends SonarPlugin { + defaultValue = "target/site/clover/clover.xml", + name = "Clover Report path", + description = "Absolute or relative path to the Clover XML report file.", + project = true)}) +public final class CloverPlugin implements Plugin { - @Override - public List getExtensions() { - return Arrays.asList(CloverSensor.class); + public void define(Context context) { + context.addExtensions(Collections.singleton(CloverSensor.class)); } } diff --git a/src/main/java/org/sonar/plugins/clover/CloverSensor.java b/src/main/java/org/sonar/plugins/clover/CloverSensor.java index 5590232..4e18331 100644 --- a/src/main/java/org/sonar/plugins/clover/CloverSensor.java +++ b/src/main/java/org/sonar/plugins/clover/CloverSensor.java @@ -19,57 +19,61 @@ */ package org.sonar.plugins.clover; -import org.codehaus.plexus.util.StringUtils; -import org.slf4j.LoggerFactory; -import org.sonar.api.batch.CoverageExtension; -import org.sonar.api.batch.Sensor; -import org.sonar.api.batch.SensorContext; +import org.apache.commons.lang.StringUtils; import org.sonar.api.batch.fs.FileSystem; -import org.sonar.api.config.Settings; -import org.sonar.api.resources.Project; +import org.sonar.api.batch.sensor.Sensor; +import org.sonar.api.batch.sensor.SensorContext; +import org.sonar.api.batch.sensor.SensorDescriptor; +import org.sonar.api.config.Configuration; import org.sonar.api.scan.filesystem.PathResolver; +import org.sonar.api.utils.log.Loggers; import javax.annotation.Nullable; import java.io.File; +import java.util.Optional; -public class CloverSensor implements Sensor, CoverageExtension { - public static final String REPORT_PATH_PROPERTY = "sonar.clover.reportPath"; +public class CloverSensor implements Sensor { + + static final String REPORT_PATH_PROPERTY = "sonar.clover.reportPath"; + static final String MISSING_FILE_MESSAGE = "Clover XML report not found"; private final FileSystem fs; private final PathResolver pathResolver; - private Settings settings; + private final Configuration configuration; - public CloverSensor(Settings settings, FileSystem fs, PathResolver pathResolver) { - this.settings = settings; + @SuppressWarnings("WeakerAccess") // brings compatibility with sonarQube v 6.x + public CloverSensor(Configuration configuration, FileSystem fs, PathResolver pathResolver) { + this.configuration = configuration; this.fs = fs; this.pathResolver = pathResolver; } + private File getReportFromProperty() { + Optional path = configuration.get(REPORT_PATH_PROPERTY); + if (path.isPresent() && StringUtils.isNotEmpty(path.get())) { + return pathResolver.relativeFile(fs.baseDir(), path.get()); + } + + return null; + } + + private static boolean reportExists(@Nullable File report) { + return report != null && report.isFile(); + } @Override - public boolean shouldExecuteOnProject(Project project) { - return StringUtils.isNotEmpty(settings.getString(REPORT_PATH_PROPERTY)); + public void describe(SensorDescriptor descriptor) { + descriptor.name("Clover Coverage Analysis"); + descriptor.onlyOnLanguages("java", "groovy"); } @Override - public void analyse(Project project, SensorContext context) { - File report = getReportFromProperty(); + public void execute(SensorContext context) { + final File report = getReportFromProperty(); if (reportExists(report)) { new CloverXmlReportParser(context, new InputFileProvider(fs)).collect(report); } else { - LoggerFactory.getLogger(getClass()).warn("Clover XML report not found"); + Loggers.get(getClass()).warn(MISSING_FILE_MESSAGE); } } - private File getReportFromProperty() { - String path = settings.getString(REPORT_PATH_PROPERTY); - if (StringUtils.isNotEmpty(path)) { - return pathResolver.relativeFile(fs.baseDir(), path); - } - return null; - } - - private static boolean reportExists(@Nullable File report) { - return report != null && report.isFile(); - } - } diff --git a/src/main/java/org/sonar/plugins/clover/CloverXmlReportParser.java b/src/main/java/org/sonar/plugins/clover/CloverXmlReportParser.java index 70d9b2a..6fcb388 100644 --- a/src/main/java/org/sonar/plugins/clover/CloverXmlReportParser.java +++ b/src/main/java/org/sonar/plugins/clover/CloverXmlReportParser.java @@ -22,173 +22,172 @@ import org.apache.commons.lang.StringUtils; import org.codehaus.staxmate.in.SMEvent; import org.codehaus.staxmate.in.SMFilterFactory; -import org.codehaus.staxmate.in.SMHierarchicCursor; import org.codehaus.staxmate.in.SMInputCursor; import org.codehaus.staxmate.in.SimpleFilter; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.sonar.api.batch.SensorContext; import org.sonar.api.batch.fs.InputFile; -import org.sonar.api.measures.CoverageMeasuresBuilder; -import org.sonar.api.measures.Measure; +import org.sonar.api.batch.fs.internal.DefaultInputFile; +import org.sonar.api.batch.sensor.SensorContext; +import org.sonar.api.batch.sensor.coverage.NewCoverage; +import org.sonar.api.utils.MessageException; import org.sonar.api.utils.ParsingUtils; -import org.sonar.api.utils.StaxParser; -import org.sonar.api.utils.XmlParserException; +import org.sonar.api.utils.log.Logger; +import org.sonar.api.utils.log.Loggers; import javax.annotation.Nullable; import javax.xml.stream.XMLStreamException; import java.io.File; import java.text.ParseException; -public class CloverXmlReportParser { - - private static final Logger LOG = LoggerFactory.getLogger(CloverXmlReportParser.class); - private SensorContext context; - private final InputFileProvider inputFileProvider; - private int files; - private int unmatchedFile; - private String unmatchedFiles; - final CoverageMeasuresBuilder fileMeasuresBuilder = CoverageMeasuresBuilder.create(); - - CloverXmlReportParser(SensorContext context, InputFileProvider inputFileProvider) { - this.context = context; - this.inputFileProvider = inputFileProvider; - } - - private static boolean reportExists(@Nullable File report) { - return report != null && report.exists() && report.isFile(); - } - - protected void collect(File xmlFile) { - try { - if (reportExists(xmlFile)) { - files = 0; - unmatchedFile = 0; - unmatchedFiles = ""; - LOG.info("Parsing " + xmlFile.getCanonicalPath()); - createStaxParser().parse(xmlFile); - LOG.info("Matched files in report : {}", getMatchedPercentage()); - if (!unmatchedFiles.isEmpty()) { - LOG.warn("{} files in clover report did not match any file in SonarQube Index : {}", unmatchedFile, unmatchedFiles); - } - } - } catch (IllegalStateException e) { - LOG.error("Format of clover report file is unexpected ", e); - throw new XmlParserException(e); - } catch (Exception e) { - LOG.error("An error occured while parsing clover xml report : ", e); - throw new XmlParserException(e); +class CloverXmlReportParser { + + private static final Logger LOG = Loggers.get(CloverXmlReportParser.class); + private SensorContext context; + private final InputFileProvider inputFileProvider; + private int files; + private int unmatchedFile; + private String unmatchedFiles; + + CloverXmlReportParser(SensorContext context, InputFileProvider inputFileProvider) { + this.context = context; + this.inputFileProvider = inputFileProvider; } - } - private StaxParser createStaxParser() { - return new StaxParser(new StaxParser.XmlStreamHandler() { - @Override - public void stream(SMHierarchicCursor rootCursor) throws XMLStreamException { + private static boolean reportExists(@Nullable File report) { + return report != null && report.exists() && report.isFile(); + } + + void collect(File xmlFile) { try { - collectProjectMeasures(rootCursor.advance()); - } catch (ParseException e) { - throw new XMLStreamException(e); + if (reportExists(xmlFile)) { + files = 0; + unmatchedFile = 0; + unmatchedFiles = ""; + LOG.info("Parsing " + xmlFile.getCanonicalPath()); + createStaxParser().parse(xmlFile); + LOG.info("Matched files in report : {}", getMatchedPercentage()); + if (!unmatchedFiles.isEmpty()) { + LOG.warn("{} files in Clover report did not match any file in SonarQube Index : {}", unmatchedFile, + unmatchedFiles); + } + } + } catch (IllegalArgumentException e) { + LOG.error("Format of clover report file is unexpected ", e); + throw e; + } catch (Exception e) { + LOG.error("An error occured while parsing Clover XML report : ", e); + throw MessageException.of("Clover XML report could not be parsed", e); } - } - }); - } - - private String getMatchedPercentage() { - if (files == 0) { - return "No files found in section of report"; } - return (files - unmatchedFile) * 100 / files+"%"; - } - - private void collectProjectMeasures(SMInputCursor rootCursor) throws ParseException, XMLStreamException { - SMInputCursor projectCursor = rootCursor.descendantElementCursor("project"); - SMInputCursor projectChildrenCursor = projectCursor.advance().childElementCursor(); - projectChildrenCursor.setFilter(new SimpleFilter(SMEvent.START_ELEMENT)); - //Skip the metrics tag. - projectChildrenCursor.advance(); - collectPackageMeasures(projectChildrenCursor); - } - - private void collectPackageMeasures(SMInputCursor packCursor) throws ParseException, XMLStreamException { - while (packCursor.getNext() != null) { - SMInputCursor packChildrenCursor = packCursor.descendantElementCursor(); - packChildrenCursor.setFilter(new SimpleFilter(SMEvent.START_ELEMENT)); - //Skip the metrics tag. - packChildrenCursor.advance(); - collectFileMeasures(packChildrenCursor); + + private StaxParser createStaxParser() { + return new StaxParser(rootCursor -> { + try { + collectProjectMeasures(rootCursor.advance()); + } catch (ParseException e) { + throw new XMLStreamException(e); + } + }); } - } - - private void collectFileMeasures(SMInputCursor fileCursor) throws ParseException, XMLStreamException { - fileCursor.setFilter(SMFilterFactory.getElementOnlyFilter("file")); - while (fileCursor.getNext() != null) { - if (fileCursor.asEvent().isStartElement()) { - String path = fileCursor.getAttrValue("path"); - if (path != null) { - SMInputCursor fileChildrenCursor = fileCursor.childCursor(new SimpleFilter(SMEvent.START_ELEMENT)); - // cursor should be on the metrics element - if (canBeIncludedInFileMetrics(fileChildrenCursor)) { - // cursor should be now on the line cursor - saveHitsData(getInputFile(path), fileChildrenCursor); - } + + private String getMatchedPercentage() { + if (files == 0) { + return "No files found in section of report"; } - } + return (files - unmatchedFile) * 100 / files + "%"; } - } - - private InputFile getInputFile(String path) { - files++; - InputFile resource = inputFileProvider.fromPath(path); - if (resource == null) { - unmatchedFile++; - LOG.warn("Resource " + path + " was not found."); - unmatchedFiles += path + ", "; + + private void collectProjectMeasures(SMInputCursor rootCursor) throws ParseException, XMLStreamException { + SMInputCursor projectCursor = rootCursor.descendantElementCursor("project"); + SMInputCursor projectChildrenCursor = projectCursor.advance().childElementCursor(); + projectChildrenCursor.setFilter(new SimpleFilter(SMEvent.START_ELEMENT)); + //Skip the metrics tag. + projectChildrenCursor.advance(); + collectPackageMeasures(projectChildrenCursor); } - return resource; - } - - private void saveHitsData(InputFile resource, SMInputCursor lineCursor) throws ParseException, XMLStreamException { - fileMeasuresBuilder.reset(); - - while (lineCursor.getNext() != null) { - // skip class elements on format 2_3_2 - if (isClass(lineCursor)) { - continue; - } - final int lineId = Integer.parseInt(lineCursor.getAttrValue("num")); - String count = lineCursor.getAttrValue("count"); - if (StringUtils.isNotBlank(count)) { - fileMeasuresBuilder.setHits(lineId, Integer.parseInt(count)); - - } else { - int trueCount = (int) ParsingUtils.parseNumber(lineCursor.getAttrValue("truecount")); - int falseCount = (int) ParsingUtils.parseNumber(lineCursor.getAttrValue("falsecount")); - int coveredConditions = 0; - if (trueCount > 0) { - coveredConditions++; + + private void collectPackageMeasures(SMInputCursor packCursor) throws ParseException, XMLStreamException { + while (packCursor.getNext() != null) { + SMInputCursor packChildrenCursor = packCursor.descendantElementCursor(); + packChildrenCursor.setFilter(new SimpleFilter(SMEvent.START_ELEMENT)); + //Skip the metrics tag. + packChildrenCursor.advance(); + collectFileMeasures(packChildrenCursor); } - if (falseCount > 0) { - coveredConditions++; + } + + private void collectFileMeasures(SMInputCursor fileCursor) throws ParseException, XMLStreamException { + fileCursor.setFilter(SMFilterFactory.getElementOnlyFilter("file")); + while (fileCursor.getNext() != null) { + if (fileCursor.asEvent().isStartElement()) { + String path = fileCursor.getAttrValue("path"); + if (path != null) { + SMInputCursor fileChildrenCursor = fileCursor.childCursor(new SimpleFilter(SMEvent.START_ELEMENT)); + InputFile resource = getInputFile(path); + if (resource != null) { + saveHitsData(resource, fileChildrenCursor); + } + } + } } - fileMeasuresBuilder.setConditions(lineId, 2, coveredConditions); - } } - if (resource != null) { - for (Measure measure : fileMeasuresBuilder.createMeasures()) { - context.saveMeasure(resource, measure); - } + + private InputFile getInputFile(String path) { + files++; + InputFile resource = inputFileProvider.fromPath(path); + if (resource == null) { + unmatchedFile++; + LOG.warn("Resource " + path + " was not found."); + unmatchedFiles += path + ", "; + } + return resource; } - } - private static boolean canBeIncludedInFileMetrics(SMInputCursor metricsCursor) throws ParseException, XMLStreamException { - while (metricsCursor.getNext() != null && isClass(metricsCursor)) { - // skip class elements on 1.x xml format + private void saveHitsData(InputFile resource, SMInputCursor lineCursor) throws ParseException, XMLStreamException { + final NewCoverage coverage = context.newCoverage().onFile(resource); + // cursor should be on the metrics element + if (!canBeIncludedInFileMetrics(lineCursor)) { + // cursor should now be on the line cursor; exclude this file if there are no elements to cover + ((DefaultInputFile) resource).setExcludedForCoverage(true); + } + + while (lineCursor.getNext() != null) { + // skip class elements on format 2_3_2 + if (isClass(lineCursor)) { + continue; + } + final int lineId = Integer.parseInt(lineCursor.getAttrValue("num")); + String count = lineCursor.getAttrValue("count"); + if (StringUtils.isNotBlank(count)) { + final int hits = Integer.parseInt(count); + coverage.lineHits(lineId, hits); + } else { + int trueCount = (int) ParsingUtils.parseNumber(lineCursor.getAttrValue("truecount")); + int falseCount = (int) ParsingUtils.parseNumber(lineCursor.getAttrValue("falsecount")); + int coveredConditions = 0; + if (trueCount > 0) { + coveredConditions++; + } + if (falseCount > 0) { + coveredConditions++; + } + + coverage.conditions(lineId, 2, coveredConditions); + } + } + + coverage.save(); } - return ParsingUtils.parseNumber(metricsCursor.getAttrValue("elements")) > 0; - } - private static boolean isClass(SMInputCursor cursor) throws XMLStreamException { - return "class".equals(cursor.getLocalName()); - } + private static boolean canBeIncludedInFileMetrics(SMInputCursor metricsCursor) + throws ParseException, XMLStreamException { + while (metricsCursor.getNext() != null && isClass(metricsCursor)) { + // skip class elements on 1.x xml format + } + + return ParsingUtils.parseNumber(metricsCursor.getAttrValue("elements")) > 0; + } + + private static boolean isClass(SMInputCursor cursor) throws XMLStreamException { + return "class".equals(cursor.getLocalName()); + } } diff --git a/src/main/java/org/sonar/plugins/clover/InputFileProvider.java b/src/main/java/org/sonar/plugins/clover/InputFileProvider.java index e81c379..e25583b 100644 --- a/src/main/java/org/sonar/plugins/clover/InputFileProvider.java +++ b/src/main/java/org/sonar/plugins/clover/InputFileProvider.java @@ -27,7 +27,7 @@ public class InputFileProvider { private final FileSystem fs; - public InputFileProvider(FileSystem fs) { + InputFileProvider(FileSystem fs) { this.fs = fs; } diff --git a/src/main/java/org/sonar/plugins/clover/StaxParser.java b/src/main/java/org/sonar/plugins/clover/StaxParser.java new file mode 100644 index 0000000..66a3e02 --- /dev/null +++ b/src/main/java/org/sonar/plugins/clover/StaxParser.java @@ -0,0 +1,96 @@ +/* + * Sonar Clover Plugin + * Copyright (C) 2008 SonarSource + * sonarqube@googlegroups.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.clover; + +import com.ctc.wstx.stax.WstxInputFactory; +import org.apache.commons.io.IOUtils; +import org.apache.commons.lang.StringUtils; +import org.codehaus.staxmate.SMInputFactory; +import org.codehaus.staxmate.in.SMHierarchicCursor; + +import javax.xml.stream.XMLInputFactory; +import javax.xml.stream.XMLResolver; +import javax.xml.stream.XMLStreamException; +import java.io.File; +import java.io.FileInputStream; +import java.io.IOException; +import java.io.InputStream; + +class StaxParser { + + private SMInputFactory inf; + private XmlStreamHandler streamHandler; + + StaxParser(XmlStreamHandler streamHandler) { + this.streamHandler = streamHandler; + XMLInputFactory xmlFactory = XMLInputFactory.newInstance(); + if (xmlFactory instanceof WstxInputFactory) { + WstxInputFactory wstxInputfactory = (WstxInputFactory) xmlFactory; + wstxInputfactory.configureForLowMemUsage(); + wstxInputfactory.getConfig().setUndeclaredEntityResolver(new UndeclaredEntitiesXMLResolver()); + } + xmlFactory.setProperty(XMLInputFactory.IS_VALIDATING, false); + xmlFactory.setProperty(XMLInputFactory.SUPPORT_DTD, false); + xmlFactory.setProperty(XMLInputFactory.IS_NAMESPACE_AWARE, false); + inf = new SMInputFactory(xmlFactory); + } + + void parse(File xmlFile) throws XMLStreamException { + try (FileInputStream input = new FileInputStream(xmlFile)) { + parse(input); + IOUtils.closeQuietly(input); + } catch (IOException e) { + throw new XMLStreamException(e); + } + } + + private void parse(InputStream xmlInput) throws XMLStreamException { + parse(inf.rootElementCursor(xmlInput)); + } + + private void parse(SMHierarchicCursor rootCursor) throws XMLStreamException { + try { + streamHandler.stream(rootCursor); + } finally { + rootCursor.getStreamReader().closeCompletely(); + } + } + + private static class UndeclaredEntitiesXMLResolver implements XMLResolver { + @Override + public Object resolveEntity(String arg0, String arg1, String fileName, String undeclaredEntity) { + // avoid problems with XML docs containing undeclared entities.. return the entity under its raw form if not an unicode expression + if (StringUtils.startsWithIgnoreCase(undeclaredEntity, "u") && undeclaredEntity.length() == 5) { + int unicodeCharHexValue = Integer.parseInt(undeclaredEntity.substring(1), 16); + if (Character.isDefined(unicodeCharHexValue)) { + undeclaredEntity = new String(new char[] { (char) unicodeCharHexValue }); + } + } + return undeclaredEntity; + } + } + + /** + * Simple interface for handling XML stream to parse + */ + public interface XmlStreamHandler { + void stream(SMHierarchicCursor rootCursor) throws XMLStreamException; + } +} diff --git a/src/test/java/org/sonar/plugins/clover/CloverPluginTest.java b/src/test/java/org/sonar/plugins/clover/CloverPluginTest.java index 251ffb2..7c62b57 100644 --- a/src/test/java/org/sonar/plugins/clover/CloverPluginTest.java +++ b/src/test/java/org/sonar/plugins/clover/CloverPluginTest.java @@ -20,13 +20,21 @@ package org.sonar.plugins.clover; import org.junit.Test; +import org.sonar.api.Plugin; +import org.sonar.api.SonarQubeSide; +import org.sonar.api.internal.SonarRuntimeImpl; +import org.sonar.api.utils.Version; import static org.fest.assertions.Assertions.assertThat; public class CloverPluginTest { @Test - public void test_getExtensions() { - assertThat(new CloverPlugin().getExtensions()).hasSize(1); + public void test_define() { + final Plugin.Context context = new Plugin.Context(SonarRuntimeImpl.forSonarQube( + Version.parse("6.7.4"), + SonarQubeSide.SCANNER)); + new CloverPlugin().define(context); + assertThat(context.getExtensions()).hasSize(1); } } diff --git a/src/test/java/org/sonar/plugins/clover/CloverSensorTest.java b/src/test/java/org/sonar/plugins/clover/CloverSensorTest.java index d4649fe..1b92519 100644 --- a/src/test/java/org/sonar/plugins/clover/CloverSensorTest.java +++ b/src/test/java/org/sonar/plugins/clover/CloverSensorTest.java @@ -19,63 +19,71 @@ */ package org.sonar.plugins.clover; -import org.junit.Before; +import org.junit.Rule; import org.junit.Test; -import org.sonar.api.batch.SensorContext; import org.sonar.api.batch.fs.internal.DefaultFileSystem; -import org.sonar.api.batch.fs.internal.DefaultInputFile; -import org.sonar.api.config.Settings; -import org.sonar.api.resources.Project; +import org.sonar.api.batch.fs.internal.TestInputFileBuilder; +import org.sonar.api.batch.sensor.internal.DefaultSensorDescriptor; +import org.sonar.api.batch.sensor.internal.SensorContextTester; +import org.sonar.api.config.Configuration; +import org.sonar.api.config.internal.ConfigurationBridge; +import org.sonar.api.config.internal.MapSettings; import org.sonar.api.scan.filesystem.PathResolver; +import org.sonar.api.utils.log.LogTester; +import org.sonar.api.utils.log.LoggerLevel; import java.io.File; import static org.fest.assertions.Assertions.assertThat; -import static org.mockito.Mockito.mock; -import static org.mockito.Mockito.verifyZeroInteractions; public class CloverSensorTest { - private Settings settings; - private CloverSensor sensor; - private Project project; - private SensorContext context; - private DefaultFileSystem fs; - - @Before - public void setUp() throws Exception { - settings = new Settings(); - fs = new DefaultFileSystem(new File("src/test/resources")); - sensor = new CloverSensor(settings, fs, new PathResolver()); - context = mock(SensorContext.class); - project = mock(Project.class); - } + private SensorContextTester context = SensorContextTester.create(new File("src/test/resources/").getAbsoluteFile()); - @Test - public void should_not_execute_if_report_path_empty() throws Exception { - settings.setProperty(CloverSensor.REPORT_PATH_PROPERTY, ""); - assertThat(sensor.shouldExecuteOnProject(project)).isFalse(); - } + @Rule + public LogTester logTester = new LogTester(); @Test - public void should_execute_if_report_path_set() throws Exception { - settings.setProperty(CloverSensor.REPORT_PATH_PROPERTY, "clover/clover.xml"); - assertThat(sensor.shouldExecuteOnProject(project)).isTrue(); + public void should_describe() { + final DefaultSensorDescriptor descriptor = new DefaultSensorDescriptor(); + + final CloverSensor sensor = new CloverSensor(new ConfigurationBridge(new MapSettings()), context.fileSystem(), new PathResolver()); + sensor.describe(descriptor); + + assertThat(descriptor.name()).isNotNull(); + assertThat(descriptor.languages()).hasSize(2); } @Test - public void should_not_interact_if_no_report_path() throws Exception { + public void should_not_interact_if_no_report_path() { + final MapSettings settings = new MapSettings(); settings.setProperty(CloverSensor.REPORT_PATH_PROPERTY, ""); - sensor.analyse(project, context); - verifyZeroInteractions(context); + + final Configuration configuration = new ConfigurationBridge(settings); + final CloverSensor sensor = new CloverSensor(configuration, context.fileSystem(), new PathResolver()); + + sensor.execute(context); + + assertThat(logTester.logs(LoggerLevel.WARN)).contains( + CloverSensor.MISSING_FILE_MESSAGE); } @Test - public void should_save_mesures() throws Exception { - String cloverFilePath = "org/sonar/plugins/clover/CloverXmlReportParserTest/clover.xml"; - fs.add(new DefaultInputFile(cloverFilePath)); - settings.setProperty(CloverSensor.REPORT_PATH_PROPERTY, cloverFilePath); - sensor.analyse(project, context); + public void should_process_file() throws Exception { + final String cloverFilePath = "org/sonar/plugins/clover/CloverXmlReportParserTest/clover_2_6_0.xml"; + final File cloverFile = TestUtils.getResource(cloverFilePath); + final MapSettings settings = new MapSettings(); + settings.setProperty(CloverSensor.REPORT_PATH_PROPERTY, cloverFile.getAbsolutePath()); + + final DefaultFileSystem fs = context.fileSystem(); + fs.add(new TestInputFileBuilder("", cloverFile.getAbsolutePath()).build()); + + final Configuration configuration = new ConfigurationBridge(settings); + final CloverSensor sensor = new CloverSensor(configuration, fs, new PathResolver()); + + sensor.execute(context); + assertThat(logTester.logs(LoggerLevel.INFO)).contains("Parsing " + cloverFile.getCanonicalPath()); + assertThat(logTester.logs(LoggerLevel.WARN).stream().anyMatch(s -> s.contains("14 files in Clover report did not match any file in SonarQube Index"))).isEqualTo(true); } } diff --git a/src/test/java/org/sonar/plugins/clover/CloverXmlReportParserTest.java b/src/test/java/org/sonar/plugins/clover/CloverXmlReportParserTest.java index b31989b..31576e5 100644 --- a/src/test/java/org/sonar/plugins/clover/CloverXmlReportParserTest.java +++ b/src/test/java/org/sonar/plugins/clover/CloverXmlReportParserTest.java @@ -19,97 +19,97 @@ */ package org.sonar.plugins.clover; -import org.apache.commons.lang.ObjectUtils; import org.apache.commons.lang.StringUtils; -import org.apache.commons.lang.math.NumberUtils; import org.hamcrest.BaseMatcher; import org.hamcrest.Description; import org.junit.Before; import org.junit.Test; -import org.sonar.api.batch.SensorContext; import org.sonar.api.batch.fs.InputFile; -import org.sonar.api.batch.fs.internal.DefaultInputFile; -import org.sonar.api.measures.CoreMetrics; -import org.sonar.api.measures.Measure; -import org.sonar.api.measures.Metric; -import org.sonar.api.resources.Resource; -import org.sonar.api.utils.XmlParserException; -import org.sonar.test.TestUtils; +import org.sonar.api.batch.fs.internal.TestInputFileBuilder; +import org.sonar.api.batch.sensor.internal.SensorContextTester; +import org.sonar.api.utils.MessageException; import java.io.File; import java.net.URISyntaxException; -import java.text.ParseException; -import static org.mockito.Matchers.anyDouble; -import static org.mockito.Matchers.anyObject; -import static org.mockito.Matchers.argThat; -import static org.mockito.Matchers.eq; -import static org.mockito.Mockito.mock; -import static org.mockito.Mockito.never; -import static org.mockito.Mockito.verify; +import static org.fest.assertions.Assertions.assertThat; public class CloverXmlReportParserTest { private CloverXmlReportParser reportParser; - private SensorContext context; - private File xmlFile; + private SensorContextTester context = SensorContextTester.create(new File("src/test/resources/")); private InputFileProvider provider; @Before - public void before() throws URISyntaxException { - xmlFile = TestUtils.getResource(getClass(), "clover.xml"); - context = mock(SensorContext.class); - //Return a sonar resource file with the name corresponding to invocation + public void before() { + // Return a sonar resource file with the name corresponding to invocation provider = new InputFileProvider(null) { @Override public InputFile fromPath(String path) { - DefaultInputFile inputFile = new DefaultInputFile(path); - inputFile.setAbsolutePath(path); - return inputFile; + return new TestInputFileBuilder("", path).setLines(1_000).build(); } }; - reportParser = new CloverXmlReportParser(context, provider); + reportParser = new CloverXmlReportParser(context, provider); } @Test - public void parseClover232Format() throws ParseException, URISyntaxException { + public void parseClover_2_3_2_Format() { reportParser.collect(TestUtils.getResource(getClass(), "clover_2_3_2.xml")); - verify(context).saveMeasure(argThat(new SonarFileMatcher("ASTSensor.java")), argThat(new IsMeasure(CoreMetrics.LINES_TO_COVER, 68.0))); - verify(context).saveMeasure(argThat(new SonarFileMatcher("ASTSensor.java")), argThat(new IsMeasure(CoreMetrics.UNCOVERED_LINES, 6.0))); + + final String testFileName = ":/Users/cmunger/dev/workspace/sonar/sonar-squid/src/main/java/org/sonar/squid/sensors/ASTSensor.java"; + assertThat(context.lineHits(testFileName, 44)).isEqualTo(1); + assertThat(context.conditions(testFileName, 157)).isEqualTo(2); + assertThat(context.coveredConditions(testFileName, 157)).isEqualTo(2); } @Test - public void parse_clover_3_2_2_Format() throws ParseException, URISyntaxException { + public void parse_clover_3_2_2_Format() { reportParser.collect(TestUtils.getResource(getClass(), "clover_3_2_2.xml")); - verify(context).saveMeasure(argThat(new SonarFileMatcher("SampleClass.java")), argThat(new IsMeasure(CoreMetrics.LINES_TO_COVER, 8.0))); - verify(context).saveMeasure(argThat(new SonarFileMatcher("SampleClass.java")), argThat(new IsMeasure(CoreMetrics.UNCOVERED_LINES, 3.0))); + + final String testFileName = ":/home/benzonico/Development/SonarSource/clover-sample/src/main/java/SampleClass.java"; + assertThat(context.lineHits(testFileName, 6)).isEqualTo(1); + assertThat(context.conditions(testFileName, 6)).isEqualTo(2); + assertThat(context.coveredConditions(testFileName, 6)).isEqualTo(1); } @Test - public void collectFileMeasures() throws Exception { - reportParser.collect(xmlFile); - verify(context).saveMeasure(argThat(new SonarFileMatcher("ClassUnderTest.java")), argThat(new IsMeasure(CoreMetrics.LINES_TO_COVER, 5.0))); - verify(context).saveMeasure(argThat(new SonarFileMatcher("ClassUnderTest.java")), argThat(new IsMeasure(CoreMetrics.UNCOVERED_LINES, 0.0))); - verify(context).saveMeasure(argThat(new SonarFileMatcher("ClassUnderTest.java")), argThat(new IsMeasure(CoreMetrics.COVERAGE_LINE_HITS_DATA, "4=1;5=1;6=2;8=1;9=1"))); + public void parse_clover_4_1_1_Format() { + reportParser.collect(TestUtils.getResource(getClass(), "clover_4_1_1.xml")); + + final String testFileName = ":/clover-examples/parameterized-junit4-example/src/test/java/Square.java"; + assertThat(context.lineHits(testFileName, 6)).isEqualTo(12); + assertThat(context.conditions(testFileName, 6)).isNull(); + assertThat(context.coveredConditions(testFileName, 6)).isNull(); + + final String omittedFileName = ":/clover-examples/parameterized-junit4-example/src/test/java/Omit.java"; + assertThat(context.lineHits(omittedFileName, 6)).isNull(); + assertThat(context.conditions(omittedFileName, 6)).isNull(); + assertThat(context.coveredConditions(omittedFileName, 6)).isNull(); } @Test - public void collectFileHitsData() throws Exception { - reportParser.collect(xmlFile); - verify(context).saveMeasure(argThat(new SonarFileMatcher("ClassUnderTest.java")), argThat(new IsMeasure(CoreMetrics.COVERAGE_LINE_HITS_DATA, "4=1;5=1;6=2;8=1;9=1"))); + public void parse_clover_2_6_0_Format() { + reportParser.collect(TestUtils.getResource(getClass(), "clover_2_6_0.xml")); + + final String testFileName = ":/Users/simon/projects/sonar/trunk/tests/integration/reference-projects/reference/src/main/java/org/sonar/samples/ClassUnderTest.java"; + assertThat(context.lineHits(testFileName, 4)).isEqualTo(1); + assertThat(context.conditions(testFileName, 9)).isNull(); + assertThat(context.coveredConditions(testFileName, 9)).isNull(); } @Test public void coverageShouldBeZeroWhenNoElements() throws URISyntaxException { File xmlFile = TestUtils.getResource(getClass(), "coverageShouldBeZeroWhenNoElements/clover.xml"); reportParser.collect(xmlFile); - verify(context, never()).saveMeasure((Resource) anyObject(), eq(CoreMetrics.COVERAGE), anyDouble()); - verify(context, never()).saveMeasure((Resource) anyObject(), eq(CoreMetrics.LINE_COVERAGE), anyDouble()); - verify(context, never()).saveMeasure((Resource) anyObject(), eq(CoreMetrics.BRANCH_COVERAGE), anyDouble()); + + final String testFileName = ":/sonar/sonar-commons/src/main/java/ch/hortis/sonar/model/MetricsClassType.java"; + assertThat(context.lineHits(testFileName, 1)).isNull(); + assertThat(context.conditions(testFileName, 1)).isNull(); + assertThat(context.coveredConditions(testFileName, 1)).isNull(); } - @Test(expected = XmlParserException.class) + @Test(expected = MessageException.class) public void bad_clover_should_throw_exception() throws Exception { reportParser.collect(TestUtils.getResource(getClass(), "bad_clover.xml")); } @@ -127,7 +127,7 @@ private SonarFileMatcher(String filename) { public boolean matches(Object o) { if (o instanceof InputFile) { InputFile file = (InputFile) o; - invokedFileName = extractFileName(file.absolutePath()); + invokedFileName = extractFileName(file.filename()); return filename.equals(invokedFileName); } return false; @@ -148,41 +148,4 @@ private String extractFileName(String filename) { return filename; } } - - public class IsMeasure extends BaseMatcher { - private Metric metric = null; - private Double value = null; - private String data = null; - private String mismatchTxt; - - public IsMeasure(Metric metric, Double value) { - this.metric = metric; - this.value = value; - } - - public IsMeasure(Metric metric, String data) { - this.metric = metric; - this.data = data; - } - - public boolean matches(Object o) { - Measure m = (Measure) o; - if (this.metric != null && !ObjectUtils.equals(this.metric, m.getMetric())) { - this.mismatchTxt = "metric: " + this.metric.getKey(); - return false; - } else if (this.value != null && NumberUtils.compare(this.value.doubleValue(), m.getValue().doubleValue()) != 0) { - this.mismatchTxt = "value: " + this.value; - return false; - } else if (this.data != null && !ObjectUtils.equals(this.data, m.getData())) { - this.mismatchTxt = "data: " + this.data; - return false; - } else { - return true; - } - } - - public void describeTo(Description description) { - description.appendText(this.mismatchTxt); - } - } } diff --git a/src/test/java/org/sonar/plugins/clover/TestUtils.java b/src/test/java/org/sonar/plugins/clover/TestUtils.java new file mode 100644 index 0000000..e64ee38 --- /dev/null +++ b/src/test/java/org/sonar/plugins/clover/TestUtils.java @@ -0,0 +1,93 @@ +/* + * Sonar Clover Plugin + * Copyright (C) 2008 SonarSource + * sonarqube@googlegroups.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.clover; + +import org.apache.commons.io.FileUtils; +import org.apache.commons.lang.StringUtils; + +import java.io.File; +import java.lang.reflect.Constructor; +import java.lang.reflect.Modifier; +import java.net.URL; + +/** + * Utilities for unit tests + * + * @since 2.2 + */ +public final class TestUtils { + + private TestUtils() { + } + + /** + * Search for a test resource in the classpath. For example getResource("org/sonar/MyClass/foo.txt"); + * + * @param path the starting slash is optional + * @return the resource. Null if resource not found + */ + public static File getResource(String path) { + String resourcePath = path; + if (!resourcePath.startsWith("/")) { + resourcePath = "/" + resourcePath; + } + URL url = TestUtils.class.getResource(resourcePath); + if (url != null) { + return FileUtils.toFile(url); + } + return null; + } + + /** + * Search for a resource in the classpath. For example calling the method getResource(getClass(), "myTestName/foo.txt") from + * the class org.sonar.Foo loads the file $basedir/src/test/resources/org/sonar/Foo/myTestName/foo.txt + * + * @return the resource. Null if resource not found + */ + public static File getResource(Class baseClass, String path) { + String resourcePath = StringUtils.replaceChars(baseClass.getCanonicalName(), '.', '/'); + if (!path.startsWith("/")) { + resourcePath += "/"; + } + resourcePath += path; + return getResource(resourcePath); + } + + /** + * Asserts that all constructors are private, usually for helper classes with + * only static methods. If a constructor does not have any parameters, then + * it's instantiated. + */ + public static boolean hasOnlyPrivateConstructors(Class clazz) { + boolean ok = true; + for (Constructor constructor : clazz.getDeclaredConstructors()) { + ok &= Modifier.isPrivate(constructor.getModifiers()); + if (constructor.getParameterTypes().length == 0) { + constructor.setAccessible(true); + try { + constructor.newInstance(); + } catch (Exception e) { + throw new IllegalStateException(String.format("Fail to instantiate %s", clazz), e); + } + } + } + return ok; + } +} diff --git a/src/test/resources/org/sonar/plugins/clover/CloverXmlReportParserTest/clover.xml b/src/test/resources/org/sonar/plugins/clover/CloverXmlReportParserTest/clover_2_6_0.xml similarity index 100% rename from src/test/resources/org/sonar/plugins/clover/CloverXmlReportParserTest/clover.xml rename to src/test/resources/org/sonar/plugins/clover/CloverXmlReportParserTest/clover_2_6_0.xml diff --git a/src/test/resources/org/sonar/plugins/clover/CloverXmlReportParserTest/clover_4_1_1.xml b/src/test/resources/org/sonar/plugins/clover/CloverXmlReportParserTest/clover_4_1_1.xml new file mode 100644 index 0000000..5ab6c18 --- /dev/null +++ b/src/test/resources/org/sonar/plugins/clover/CloverXmlReportParserTest/clover_4_1_1.xml @@ -0,0 +1,64 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file