blob: 9ba4d7b8894feaff462187c08be1806412105448 [file] [log] [blame]
package org.apache.maven.plugin.assembly.archive.phase;
/*
* 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.IOException;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.Iterator;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import javax.annotation.Nonnull;
import org.apache.maven.artifact.Artifact;
import org.apache.maven.artifact.ArtifactUtils;
import org.apache.maven.plugin.assembly.AssemblerConfigurationSource;
import org.apache.maven.plugin.assembly.InvalidAssemblerConfigurationException;
import org.apache.maven.plugin.assembly.archive.ArchiveCreationException;
import org.apache.maven.plugin.assembly.archive.task.AddArtifactTask;
import org.apache.maven.plugin.assembly.archive.task.AddDependencySetsTask;
import org.apache.maven.plugin.assembly.archive.task.AddFileSetsTask;
import org.apache.maven.plugin.assembly.format.AssemblyFormattingException;
import org.apache.maven.plugin.assembly.model.DependencySet;
import org.apache.maven.plugin.assembly.model.FileSet;
import org.apache.maven.plugin.assembly.model.ModuleBinaries;
import org.apache.maven.plugin.assembly.model.ModuleSet;
import org.apache.maven.plugin.assembly.model.ModuleSources;
import org.apache.maven.plugin.assembly.resolved.ResolvedAssembly;
import org.apache.maven.plugin.assembly.resolved.ResolvedModuleSet;
import org.apache.maven.plugin.assembly.resolved.functions.ResolvedModuleSetConsumer;
import org.apache.maven.plugin.assembly.utils.AssemblyFormatUtils;
import org.apache.maven.plugin.assembly.utils.FilterUtils;
import org.apache.maven.plugin.assembly.utils.ProjectUtils;
import org.apache.maven.plugin.assembly.utils.TypeConversionUtils;
import org.apache.maven.project.MavenProject;
import org.apache.maven.project.MavenProjectBuilder;
import org.codehaus.plexus.archiver.Archiver;
import org.codehaus.plexus.archiver.manager.ArchiverManager;
import org.codehaus.plexus.component.annotations.Component;
import org.codehaus.plexus.component.annotations.Requirement;
import org.codehaus.plexus.logging.AbstractLogEnabled;
import org.codehaus.plexus.logging.Logger;
/**
* Handles the <moduleSets/> top-level section of the assembly descriptor.
*
* @version $Id$
*/
@Component( role = AssemblyArchiverPhase.class, hint = "module-sets" )
public class ModuleSetAssemblyPhase
extends AbstractLogEnabled
implements AssemblyArchiverPhase
{
// TODO: Remove if using something like commons-lang instead.
/**
* The line separator.
*/
private static final String LINE_SEPARATOR = System.getProperty( "line.separator" );
@Requirement
private MavenProjectBuilder projectBuilder;
@Requirement
private ArchiverManager archiverManager;
/**
* Create an instance.
*/
public ModuleSetAssemblyPhase()
{
// needed for plexus
}
/**
* @param projectBuilder The project builder.
* @param logger The logger.
*/
public ModuleSetAssemblyPhase( final MavenProjectBuilder projectBuilder, final Logger logger )
{
this.projectBuilder = projectBuilder;
enableLogging( logger );
}
/**
* {@inheritDoc}
*/
public void execute( final ResolvedAssembly assembly, final Archiver archiver,
final AssemblerConfigurationSource configSource )
throws ArchiveCreationException, AssemblyFormattingException, InvalidAssemblerConfigurationException
{
assembly.forEachResolvedModule( new ResolvedModuleSetConsumer()
{
public void accept( ResolvedModuleSet resolvedModule )
throws ArchiveCreationException, AssemblyFormattingException, InvalidAssemblerConfigurationException
{
validate( resolvedModule.getModuleSet(), configSource );
final Set<MavenProject> moduleProjects =
getModuleProjects( resolvedModule.getModuleSet(), configSource, getLogger() );
final ModuleSources sources = resolvedModule.getModuleSet().getSources();
addModuleSourceFileSets( sources, moduleProjects, archiver, configSource );
final ModuleBinaries binaries = resolvedModule.getModuleSet().getBinaries();
addModuleBinaries( resolvedModule, binaries, moduleProjects, archiver, configSource );
}
} );
}
private void validate( final ModuleSet moduleSet, final AssemblerConfigurationSource configSource )
{
if ( ( moduleSet.getSources() == null ) && ( moduleSet.getBinaries() == null ) )
{
getLogger().warn( "Encountered ModuleSet with no sources or binaries specified. Skipping." );
}
if ( moduleSet.isUseAllReactorProjects() && !moduleSet.isIncludeSubModules() )
{
getLogger().warn( "includeSubModules == false is incompatible with useAllReactorProjects. Ignoring."
+ "\n\nTo refactor, remove the <includeSubModules/> flag, and use the <includes/> "
+ "and <excludes/> sections to fine-tune the modules included." );
}
final List<MavenProject> projects = configSource.getReactorProjects();
if ( projects != null && projects.size() > 1 && projects.indexOf( configSource.getProject() ) == 0
&& moduleSet.getBinaries() != null )
{
getLogger().warn( "[DEPRECATION] moduleSet/binaries section detected in root-project assembly."
+ "\n\nMODULE BINARIES MAY NOT BE AVAILABLE FOR THIS ASSEMBLY!"
+ "\n\n To refactor, move this assembly into a child project and use the flag "
+ "<useAllReactorProjects>true</useAllReactorProjects> in each moduleSet." );
}
if ( moduleSet.getSources() != null )
{
final ModuleSources sources = moduleSet.getSources();
if ( isDeprecatedModuleSourcesConfigPresent( sources ) )
{
getLogger().warn( "[DEPRECATION] Use of <moduleSources/> as a file-set is deprecated. "
+ "Please use the <fileSets/> sub-element of <moduleSources/> instead." );
}
else if ( !sources.isUseDefaultExcludes() )
{
getLogger().warn( "[DEPRECATION] Use of directoryMode, fileMode, or useDefaultExcludes "
+ "elements directly within <moduleSources/> are all deprecated. "
+ "Please use the <fileSets/> sub-element of <moduleSources/> instead." );
}
}
}
void addModuleBinaries( ResolvedModuleSet resolvedModule, final ModuleBinaries binaries,
final Set<MavenProject> projects, final Archiver archiver,
final AssemblerConfigurationSource configSource )
throws ArchiveCreationException, AssemblyFormattingException, InvalidAssemblerConfigurationException
{
if ( binaries == null )
{
return;
}
final Set<MavenProject> moduleProjects = new LinkedHashSet<MavenProject>( projects );
for ( final Iterator<MavenProject> it = moduleProjects.iterator(); it.hasNext(); )
{
final MavenProject project = it.next();
if ( "pom".equals( project.getPackaging() ) )
{
final String projectId = ArtifactUtils.versionlessKey( project.getGroupId(), project.getArtifactId() );
getLogger().debug( "Excluding POM-packaging module: " + projectId );
it.remove();
}
}
final String classifier = binaries.getAttachmentClassifier();
final Map<MavenProject, Artifact> chosenModuleArtifacts = new HashMap<MavenProject, Artifact>();
for ( final MavenProject project : moduleProjects )
{
Artifact artifact = null;
if ( classifier == null )
{
getLogger().debug( "Processing binary artifact for module project: " + project.getId() );
artifact = project.getArtifact();
}
else
{
getLogger().debug( "Processing binary attachment: " + classifier + " for module project: "
+ project.getId() );
@SuppressWarnings( "unchecked" )
final List<Artifact> attachments = project.getAttachedArtifacts();
if ( ( attachments != null ) && !attachments.isEmpty() )
{
for ( final Artifact attachment : attachments )
{
if ( classifier.equals( attachment.getClassifier() ) )
{
artifact = attachment;
break;
}
}
}
if ( artifact == null )
{
throw new InvalidAssemblerConfigurationException( "Cannot find attachment with classifier: "
+ classifier + " in module project: " + project.getId()
+ ". Please exclude this module from the module-set." );
}
}
chosenModuleArtifacts.put( project, artifact );
addModuleArtifact( artifact, project, archiver, configSource, binaries );
}
final List<DependencySet> depSets = getDependencySets( binaries );
if ( depSets != null )
{
for ( final DependencySet ds : depSets )
{
// NOTE: Disabling useProjectArtifact flag, since module artifact has already been handled!
ds.setUseProjectArtifact( false );
}
// TODO: The following should be moved into a shared component, cause this
// test is the same as in maven-enforce rules ReactorModuleConvergence.
List<MavenProject> validateModuleVersions = validateModuleVersions( moduleProjects );
if ( !validateModuleVersions.isEmpty() )
{
StringBuilder sb =
new StringBuilder().append( "The current modules seemed to be having different versions." );
sb.append( LINE_SEPARATOR );
for ( MavenProject mavenProject : validateModuleVersions )
{
sb.append( " --> " );
sb.append( mavenProject.getId() );
sb.append( LINE_SEPARATOR );
}
getLogger().warn( sb.toString() );
}
for ( final MavenProject moduleProject : moduleProjects )
{
getLogger().debug( "Processing binary dependencies for module project: " + moduleProject.getId() );
final AddDependencySetsTask task =
new AddDependencySetsTask( depSets, resolvedModule.getArtifacts(), moduleProject, projectBuilder,
archiverManager, getLogger() );
task.setModuleProject( moduleProject );
task.setModuleArtifact( chosenModuleArtifacts.get( moduleProject ) );
task.setDefaultOutputDirectory( binaries.getOutputDirectory() );
task.setDefaultOutputFileNameMapping( binaries.getOutputFileNameMapping() );
task.execute( archiver, configSource );
}
}
}
private List<MavenProject> validateModuleVersions( Set<MavenProject> moduleProjects )
{
List<MavenProject> result = new ArrayList<MavenProject>();
if ( moduleProjects != null && !moduleProjects.isEmpty() )
{
String version = moduleProjects.iterator().next().getVersion();
getLogger().debug( "First version:" + version );
for ( MavenProject mavenProject : moduleProjects )
{
getLogger().debug( " -> checking " + mavenProject.getId() );
if ( !version.equals( mavenProject.getVersion() ) )
{
result.add( mavenProject );
}
}
}
return result;
}
public static List<DependencySet> getDependencySets( final ModuleBinaries binaries )
{
List<DependencySet> depSets = binaries.getDependencySets();
if ( ( ( depSets == null ) || depSets.isEmpty() ) && binaries.isIncludeDependencies() )
{
final DependencySet impliedDependencySet = new DependencySet();
impliedDependencySet.setOutputDirectory( binaries.getOutputDirectory() );
impliedDependencySet.setOutputFileNameMapping( binaries.getOutputFileNameMapping() );
impliedDependencySet.setFileMode( binaries.getFileMode() );
impliedDependencySet.setDirectoryMode( binaries.getDirectoryMode() );
impliedDependencySet.setExcludes( binaries.getExcludes() );
impliedDependencySet.setIncludes( binaries.getIncludes() );
impliedDependencySet.setUnpack( binaries.isUnpack() );
// unpackOptions is handled in the first stage of dependency-set handling, below.
depSets = Collections.singletonList( impliedDependencySet );
}
return depSets;
}
// protected List<String> collectExcludesFromQueuedArtifacts( final Set visitedArtifacts, final List binaryExcludes
// )
// {
// List excludes = binaryExcludes;
//
// if ( excludes == null )
// {
// excludes = new ArrayList();
// }
// else
// {
// excludes = new ArrayList( excludes );
// }
//
// for ( final Iterator it = visitedArtifacts.iterator(); it.hasNext(); )
// {
// excludes.add( it.next() );
// }
//
// return excludes;
// }
void addModuleArtifact( final Artifact artifact, final MavenProject project, final Archiver archiver,
final AssemblerConfigurationSource configSource, final ModuleBinaries binaries )
throws ArchiveCreationException, AssemblyFormattingException
{
if ( artifact.getFile() == null )
{
throw new ArchiveCreationException( "Artifact: " + artifact.getId()
+ " (included by module) does not have an artifact with a file. "
+ "Please ensure the package phase is run before the assembly is generated." );
}
final AddArtifactTask task = new AddArtifactTask( artifact, getLogger() );
task.setFileNameMapping( binaries.getOutputFileNameMapping() );
task.setOutputDirectory( binaries.getOutputDirectory() );
task.setProject( project );
task.setModuleProject( project );
task.setModuleArtifact( artifact );
final int dirMode = TypeConversionUtils.modeToInt( binaries.getDirectoryMode(), getLogger() );
if ( dirMode != -1 )
{
task.setDirectoryMode( dirMode );
}
final int fileMode = TypeConversionUtils.modeToInt( binaries.getFileMode(), getLogger() );
if ( fileMode != -1 )
{
task.setFileMode( fileMode );
}
task.setUnpack( binaries.isUnpack() );
if ( binaries.isUnpack() && binaries.getUnpackOptions() != null )
{
task.setIncludes( binaries.getUnpackOptions().getIncludes() );
task.setExcludes( binaries.getUnpackOptions().getExcludes() );
}
task.execute( archiver, configSource );
}
void addModuleSourceFileSets( final ModuleSources sources, final Set<MavenProject> moduleProjects,
final Archiver archiver, final AssemblerConfigurationSource configSource )
throws ArchiveCreationException, AssemblyFormattingException
{
if ( sources == null )
{
return;
}
final List<FileSet> fileSets = new ArrayList<FileSet>();
if ( isDeprecatedModuleSourcesConfigPresent( sources ) )
{
final FileSet fs = new FileSet();
fs.setOutputDirectory( sources.getOutputDirectory() );
fs.setIncludes( sources.getIncludes() );
fs.setExcludes( sources.getExcludes() );
fs.setUseDefaultExcludes( sources.isUseDefaultExcludes() );
fileSets.add( fs );
}
List<FileSet> subFileSets = sources.getFileSets();
if ( ( subFileSets == null ) || subFileSets.isEmpty() )
{
final FileSet fs = new FileSet();
fs.setDirectory( "src" );
subFileSets = Collections.singletonList( fs );
}
fileSets.addAll( subFileSets );
for ( final MavenProject moduleProject : moduleProjects )
{
getLogger().info( "Processing sources for module project: " + moduleProject.getId() );
final List<FileSet> moduleFileSets = new ArrayList<FileSet>();
for ( final FileSet fileSet : fileSets )
{
moduleFileSets.add( createFileSet( fileSet, sources, moduleProject, configSource ) );
}
final AddFileSetsTask task = new AddFileSetsTask( moduleFileSets );
task.setProject( moduleProject );
task.setModuleProject( moduleProject );
task.setLogger( getLogger() );
task.execute( archiver, configSource );
}
}
/**
* Determine whether the deprecated file-set configuration directly within the ModuleSources object is present.
*/
boolean isDeprecatedModuleSourcesConfigPresent( @Nonnull final ModuleSources sources )
{
boolean result = false;
if ( sources.getOutputDirectory() != null )
{
result = true;
}
else if ( ( sources.getIncludes() != null ) && !sources.getIncludes().isEmpty() )
{
result = true;
}
else if ( ( sources.getExcludes() != null ) && !sources.getExcludes().isEmpty() )
{
result = true;
}
return result;
}
@Nonnull
FileSet createFileSet( @Nonnull final FileSet fileSet, @Nonnull final ModuleSources sources,
@Nonnull final MavenProject moduleProject,
@Nonnull final AssemblerConfigurationSource configSource )
throws AssemblyFormattingException
{
final FileSet fs = new FileSet();
String sourcePath = fileSet.getDirectory();
final File moduleBasedir = moduleProject.getBasedir();
if ( sourcePath != null )
{
final File sourceDir = new File( sourcePath );
if ( !sourceDir.isAbsolute() )
{
sourcePath = new File( moduleBasedir, sourcePath ).getAbsolutePath();
}
}
else
{
sourcePath = moduleBasedir.getAbsolutePath();
}
fs.setDirectory( sourcePath );
fs.setDirectoryMode( fileSet.getDirectoryMode() );
final List<String> excludes = new ArrayList<String>();
final List<String> originalExcludes = fileSet.getExcludes();
if ( ( originalExcludes != null ) && !originalExcludes.isEmpty() )
{
excludes.addAll( originalExcludes );
}
if ( sources.isExcludeSubModuleDirectories() )
{
@SuppressWarnings( "unchecked" )
final List<String> modules = moduleProject.getModules();
for ( final String moduleSubPath : modules )
{
excludes.add( moduleSubPath + "/**" );
}
}
fs.setExcludes( excludes );
fs.setFiltered( fileSet.isFiltered() );
fs.setFileMode( fileSet.getFileMode() );
fs.setIncludes( fileSet.getIncludes() );
fs.setLineEnding( fileSet.getLineEnding() );
String destPathPrefix = "";
if ( sources.isIncludeModuleDirectory() )
{
destPathPrefix =
AssemblyFormatUtils.evaluateFileNameMapping( sources.getOutputDirectoryMapping(),
moduleProject.getArtifact(), configSource.getProject(),
moduleProject, moduleProject.getArtifact(), moduleProject,
configSource );
if ( !destPathPrefix.endsWith( "/" ) )
{
destPathPrefix += "/";
}
}
String destPath = fileSet.getOutputDirectory();
if ( destPath == null )
{
destPath = destPathPrefix;
}
else
{
destPath = destPathPrefix + destPath;
}
destPath =
AssemblyFormatUtils.getOutputDirectory( destPath, configSource.getProject(), moduleProject, moduleProject,
configSource.getFinalName(), configSource );
fs.setOutputDirectory( destPath );
getLogger().debug( "module source directory is: " + sourcePath );
getLogger().debug( "module dest directory is: " + destPath + " (assembly basedir may be prepended)" );
return fs;
}
@Nonnull
public static Set<MavenProject> getModuleProjects( final ModuleSet moduleSet,
final AssemblerConfigurationSource configSource,
final Logger logger )
throws ArchiveCreationException
{
MavenProject project = configSource.getProject();
Set<MavenProject> moduleProjects = null;
if ( moduleSet.isUseAllReactorProjects() )
{
if ( !moduleSet.isIncludeSubModules() )
{
moduleProjects = new LinkedHashSet<MavenProject>( configSource.getReactorProjects() );
}
project = configSource.getReactorProjects().get( 0 );
}
if ( moduleProjects == null )
{
try
{
moduleProjects =
ProjectUtils.getProjectModules( project, configSource.getReactorProjects(),
moduleSet.isIncludeSubModules(), logger );
}
catch ( final IOException e )
{
throw new ArchiveCreationException( "Error retrieving module-set for project: " + project.getId()
+ ": " + e.getMessage(), e );
}
}
FilterUtils.filterProjects( moduleProjects, moduleSet.getIncludes(), moduleSet.getExcludes(), true, logger );
return moduleProjects;
}
}