com.atlassian.maven.plugin.clover.CloverCheckMojo Maven / Gradle / Ivy
package com.atlassian.maven.plugin.clover;
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
import com.atlassian.clover.cfg.Percentage;
import com.atlassian.clover.ant.tasks.CloverPassTask;
import com.atlassian.maven.plugin.clover.internal.AbstractCloverMojo;
import org.apache.maven.plugin.MojoExecutionException;
import org.apache.tools.ant.BuildException;
import org.apache.tools.ant.Project;
import org.apache.tools.ant.types.FileSet;
import java.io.File;
import java.util.List;
import java.util.Iterator;
/**
* Verify Test Percentage Coverage (TPC) from an existing Clover database and fail the build if it is below the defined
* threshold. The check is done on main Clover databases and also on merged Clover databases when they exist.
*
* @goal check
* @phase verify
*
*/
public class CloverCheckMojo extends AbstractCloverMojo
{
/**
* The Test Percentage Coverage (TPC) threshold under which the plugin will report an error and fail the build.
* IMPORTANT: comparison of actual value with an expected percentage is performed with such numerical precision as
* number of fractional digits set for a targetPercentage.
* For example, if actual coverage value is 99.9% then for the targetPercentage="100%" it will PASS,
* whereas for the targetPercentage="100.000000%" it will FAIL.
*
* @parameter expression="${maven.clover.targetPercentage}"
*/
String targetPercentage;
/**
* The Test Percentage Method Coverage (TPC) threshold under which the plugin will report an error and fail the build.
* If maven.clover.targetPercentage is not specified, then this value is ignored.
*
* IMPORTANT: comparison of actual value with an expected percentage is performed with such numerical precision as
* number of fractional digits set for a methodPercentage.
*
* @parameter expression="${maven.clover.methodPercentage}"
*/
String methodPercentage;
/**
* The Test Percentage Statement Coverage (TPC) threshold under which the plugin will report an error and fail the build.
* If maven.clover.targetPercentage is not specified, then this value is ignored.
* IMPORTANT: comparison of actual value with an expected percentage is performed with such numerical precision as
* number of fractional digits set for a statementPercentage.
*
* @parameter expression="${maven.clover.statementPercentage}"
*/
String statementPercentage;
/**
* The Test Percentage Conditional Coverage (TPC) threshold under which the plugin will report an error and fail the build.
* If maven.clover.targetPercentage is not specified, then this value is ignored.
* IMPORTANT: comparison of actual value with an expected percentage is performed with such numerical precision as
* number of fractional digits set for a conditionalPercentage.
*
* @parameter expression="${maven.clover.conditionalPercentage}"
*/
String conditionalPercentage;
/**
* Comma or space separated list of Clover contexts (block, statement or method filers) to exclude before
* performing the check.
* @parameter expression="${maven.clover.contextFilters}"
*/
String contextFilters;
/**
* The type of code to log - APPLICATION, TEST or ALL code. Default is set to APPLICATION.
* @parameter expression="${maven.clover.codeType}"
*/
String codeType;
/**
* Do we fail the build on a violation? The default is true but there are some edge cases where you want to be
* able to check what would fail but without actually failing the build. For example you may want to let the build
* continue so that you can verify others checks that are executed after the Clover checks.
*
* @parameter expression="${maven.clover.failOnViolation}" default-value="true"
*/
boolean failOnViolation;
/**
* The location where historical Clover data is located.
*
*
* Allows you to specify a location for historical build data, along with a configurable threshold expressed as a percentage ?
* used to cause the build to fail if coverage has dropped.
* This attribute is passed down to specified packages, then the same test is done for these at the package level.
* This will only be used if there is no targetPercentage parameter set.
*
* @parameter expression="${maven.clover.historyDir}" default-value="${project.build.directory}/clover/history"
*/
File historyDir;
/**
* The percentage threshold to use if clover-check is checking coverage against historical clover data.
*
* This is the amount of leeway to use when comparing the current build's coverage with that of the last build.
*
* @parameter expression="${maven.clover.historyThreshold}" default-value="0%"
*
*/
String historyThreshold;
/**
* {@inheritDoc}
* @see com.atlassian.maven.plugin.clover.internal.AbstractCloverMojo#execute()
* @throws MojoExecutionException when the TPC is below the threshold
*/
public void execute()
throws MojoExecutionException
{
if (skip) {
getLog().debug("Skipping clover check.");
return;
}
if ( !isInCloverForkedLifecycle() )
{
if ( areCloverDatabasesAvailable() )
{
super.execute();
AbstractCloverMojo.waitForFlush( getWaitForFlush(), getFlushInterval() );
check();
}
else
{
getLog().info("No Clover database found, skipping test coverage verification");
}
}
}
/**
* Check both the main Clover database and the merged Clover database when they exist.
* @throws MojoExecutionException when the TPC is below the threshold
*/
private void check() throws MojoExecutionException
{
if ( new File( resolveCloverDatabase() ).exists() )
{
checkDatabase( resolveCloverDatabase() );
}
if ( new File( getCloverMergeDatabase() ).exists() )
{
checkDatabase( getCloverMergeDatabase() );
}
}
/**
* Check a Clover database and fail the build if the TPC is below the threshold.
*
* @param database the Clover database to verify
* @throws MojoExecutionException when the TPC is below the threshold
*/
private void checkDatabase(final String database) throws MojoExecutionException
{
final Project antProject = new Project();
antProject.init();
AbstractCloverMojo.registerCloverAntTasks(antProject, getLog());
CloverPassTask cloverPassTask = createCloverPassTask(database, antProject);
cloverPassTask.init();
cloverPassTask.setInitString(database);
cloverPassTask.setHaltOnFailure(true);
cloverPassTask.setFailureProperty("clovercheckproperty");
if (this.codeType != null) {
cloverPassTask.setCodeType(codeType);
}
if (this.targetPercentage != null) {
cloverPassTask.setTarget( new Percentage( this.targetPercentage ) );
getLog().info( "Checking for coverage of [" + targetPercentage + "] for database [" + database + "]");
if (this.methodPercentage != null)
{
cloverPassTask.setMethodTarget(new Percentage(this.methodPercentage));
getLog().info("Checking for method coverage of [" + methodPercentage + "] for database [" + database + "]");
}
if (this.conditionalPercentage != null)
{
cloverPassTask.setConditionalTarget(new Percentage(this.conditionalPercentage));
getLog().info("Checking for conditional coverage of [" + conditionalPercentage + "] for database [" + database + "]");
}
if (this.statementPercentage != null)
{
cloverPassTask.setStatementTarget(new Percentage(this.statementPercentage));
getLog().info("Checking for statement coverage of [" + statementPercentage + "] for database [" + database + "]");
}
} else if (this.historyDir.exists() && this.historyDir.isDirectory()) {
cloverPassTask.setHistorydir(this.historyDir);
cloverPassTask.setThreshold(new Percentage(this.historyThreshold));
getLog().info( "Checking coverage against historical data [" +
this.historyDir + " +/-" + this.historyThreshold +
" ] for database [" + database + "]");
} else {
getLog().warn("Skipping clover:check as 'maven.clover.targetPercentage' is not defined " +
"and 'maven.clover.historyDir' (" + this.historyDir.getPath() +
") does not exist or is not a directory.");
return;
}
if ( this.contextFilters != null )
{
cloverPassTask.setFilter( this.contextFilters );
}
setTestSourceRoots(cloverPassTask);
try
{
cloverPassTask.execute();
}
catch ( BuildException e )
{
getLog().error( antProject.getProperty( "clovercheckproperty" ) );
if ( this.failOnViolation )
{
throw new MojoExecutionException( e.getMessage(), e );
}
else
{
getLog().warn( "Clover test percentage coverage is below threshold but failOnViolation is set to "
+ " false, preventing the build from failing.");
}
}
}
private void setTestSourceRoots(final CloverPassTask cloverPassTask) {
final String originalSrcTestDir = CloverSetupMojo.getOriginalSrcTestDir(getProject().getId());
if (originalSrcTestDir != null) {
addTestSrcDir(cloverPassTask, originalSrcTestDir);
}
final List testSourceRoots = getProject().getTestCompileSourceRoots();
getLog().warn("has test file source dir = " + (testSourceRoots.size() > 0));
for (String testDir : testSourceRoots) {
getLog().warn("test file source dir = " + testDir);
addTestSrcDir(cloverPassTask, testDir);
}
}
private void addTestSrcDir(final CloverPassTask cloverPassTask, final String originalSrcTestDir) {
final FileSet testFiles = new FileSet();
final File dir = new File(originalSrcTestDir);
if (dir.exists()) {
testFiles.setDir(dir);
cloverPassTask.addTestSources(testFiles);
}
}
CloverPassTask createCloverPassTask(final String database, final Project antProject) {
return (CloverPassTask) antProject.createTask( "clover-check" );
}
/**
* @return true if the build is currently inside the custom build lifecycle forked by the
* clover:instrument
MOJO.
*/
private boolean isInCloverForkedLifecycle()
{
// We know we're in the forked lifecycle if the output directory is set to target/clover...
// TODO: Not perfect, need to find a better way. This is a hack!
return getProject().getBuild().getDirectory().endsWith( "clover" );
}
}