blob: e9dfee53437db078063c3527bec4c555d61dc1ee [file] [log] [blame]
package org.apache.maven.plugins.artifact.buildinfo;
/*
* 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 java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Properties;
import java.util.Set;
import org.apache.maven.archiver.MavenArchiver;
import org.apache.maven.artifact.versioning.ArtifactVersion;
import org.apache.maven.artifact.versioning.DefaultArtifactVersion;
import org.apache.maven.execution.MavenSession;
import org.apache.maven.lifecycle.LifecycleExecutor;
import org.apache.maven.lifecycle.MavenExecutionPlan;
import org.apache.maven.model.Plugin;
import org.apache.maven.plugin.AbstractMojo;
import org.apache.maven.plugin.MojoExecution;
import org.apache.maven.plugin.MojoExecutionException;
import org.apache.maven.plugins.annotations.Component;
import org.apache.maven.plugins.annotations.Mojo;
import org.apache.maven.plugins.annotations.Parameter;
import org.apache.maven.project.MavenProject;
/**
* Check from buildplan that plugins used don't have known reproducible builds issues.
*
* @since 3.3.0
*/
@Mojo( name = "check-buildplan", threadSafe = true, requiresProject = true )
public class CheckBuildPlanMojo
extends AbstractMojo
{
@Parameter( defaultValue = "${reactorProjects}", required = true, readonly = true )
private List<MavenProject> reactorProjects;
@Parameter( defaultValue = "${project}", readonly = true )
private MavenProject project;
@Parameter( defaultValue = "${session}", readonly = true )
private MavenSession session;
@Component
private LifecycleExecutor lifecycleExecutor;
/** Allow to specify which goals/phases will be used to calculate execution plan. */
@Parameter( property = "check.buildplan.tasks", defaultValue = "deploy" )
private String[] tasks;
/**
* Timestamp for reproducible output archive entries, either formatted as ISO 8601
* <code>yyyy-MM-dd'T'HH:mm:ssXXX</code> or as an int representing seconds since the epoch (like
* <a href="https://reproducible-builds.org/docs/source-date-epoch/">SOURCE_DATE_EPOCH</a>).
*/
@Parameter( defaultValue = "${project.build.outputTimestamp}" )
private String outputTimestamp;
/**
* Provide a plugin issues property file to override plugin's <code>not-reproducible-plugins.properties</code>.
*/
@Parameter( property = "check.plugin-issues" )
private File pluginIssues;
/**
* Make build fail if execution plan contains non-reproducible plugins.
*/
@Parameter( property = "check.failOnNonReproducible", defaultValue = "true" )
private boolean failOnNonReproducible;
protected MavenExecutionPlan calculateExecutionPlan()
throws MojoExecutionException
{
try
{
return lifecycleExecutor.calculateExecutionPlan( session, tasks );
}
catch ( Exception e )
{
throw new MojoExecutionException( "Cannot calculate Maven execution plan" + e.getMessage(), e );
}
}
@Override
public void execute()
throws MojoExecutionException
{
boolean fail = hasBadOutputTimestamp();
// TODO check maven-jar-plugin module-info.class?
Properties issues = loadIssues();
MavenExecutionPlan plan = calculateExecutionPlan();
Set<String> plugins = new HashSet<>();
for ( MojoExecution exec : plan.getMojoExecutions() )
{
Plugin plugin = exec.getPlugin();
String id = plugin.getId();
if ( plugins.add( id ) )
{
// check reproducibility status
String issue = issues.getProperty( plugin.getKey() );
if ( issue == null )
{
getLog().info( "no known issue with " + id );
}
else if ( issue.startsWith( "fail:" ) )
{
String logMessage = "plugin without solution " + id + ", see " + issue.substring( 5 );
if ( failOnNonReproducible )
{
getLog().error( logMessage );
}
else
{
getLog().warn( logMessage );
}
fail = true;
}
else
{
ArtifactVersion minimum = new DefaultArtifactVersion( issue );
ArtifactVersion version = new DefaultArtifactVersion( plugin.getVersion() );
if ( version.compareTo( minimum ) < 0 )
{
String logMessage = "plugin with non-reproducible output: " + id + ", require minimum " + issue;
if ( failOnNonReproducible )
{
getLog().error( logMessage );
}
else
{
getLog().warn( logMessage );
}
fail = true;
}
else
{
getLog().info( "no known issue with " + id + " (>= " + issue + ")" );
}
}
}
}
if ( fail )
{
getLog().info( "current module pom.xml is " + project.getBasedir() + "/pom.xml" );
MavenProject parent = project;
while ( true )
{
parent = parent.getParent();
if ( ( parent == null ) || !reactorProjects.contains( parent ) )
{
break;
}
getLog().info( " parent pom.xml is " + parent.getBasedir() + "/pom.xml" );
}
String message = "non-reproducible plugin or configuration found with fix available";
if ( failOnNonReproducible )
{
throw new MojoExecutionException( message );
}
else
{
getLog().warn( message );
}
}
}
private boolean hasBadOutputTimestamp()
{
MavenArchiver archiver = new MavenArchiver();
Date timestamp = archiver.parseOutputTimestamp( outputTimestamp );
if ( timestamp == null )
{
getLog().error( "Reproducible Build not activated by project.build.outputTimestamp property: "
+ "see https://maven.apache.org/guides/mini/guide-reproducible-builds.html" );
return true;
}
else
{
if ( getLog().isDebugEnabled() )
{
getLog().debug( "project.build.outputTimestamp = \"" + outputTimestamp + "\" => "
+ new SimpleDateFormat( "yyyy-MM-dd'T'HH:mm:ssXXX" ).format( timestamp ) );
}
// check if timestamp well defined in a project from reactor
boolean parentInReactor = false;
MavenProject reactorParent = project;
while ( reactorProjects.contains( reactorParent.getParent() ) )
{
parentInReactor = true;
reactorParent = reactorParent.getParent();
}
String prop =
reactorParent.getOriginalModel().getProperties().getProperty( "project.build.outputTimestamp" );
if ( prop == null )
{
getLog().error( "project.build.outputTimestamp property should not be inherited but defined in "
+ ( parentInReactor ? "parent POM from reactor " : "POM " ) + reactorParent.getFile() );
return true;
}
}
return false;
}
private Properties loadIssues()
throws MojoExecutionException
{
try ( InputStream in =
( pluginIssues == null ) ? getClass().getResourceAsStream( "not-reproducible-plugins.properties" )
: new FileInputStream( pluginIssues ) )
{
Properties prop = new Properties();
prop.load( in );
Properties result = new Properties();
for ( Map.Entry<Object, Object> entry : prop.entrySet() )
{
String plugin = entry.getKey().toString().replace( '+', ':' );
if ( !plugin.contains( ":" ) )
{
plugin = "org.apache.maven.plugins:" + plugin;
}
result.put( plugin, entry.getValue() );
}
return result;
}
catch ( IOException ioe )
{
throw new MojoExecutionException( "Cannot load issues file", ioe );
}
}
}