package org.codehaus.mojo.versions;
import org.apache.maven.artifact.ArtifactUtils;
import org.apache.maven.model.Model;
import org.apache.maven.model.Parent;
import org.apache.maven.plugin.MojoExecutionException;
import org.apache.maven.plugin.MojoFailureException;
import org.apache.maven.project.MavenProject;
import org.codehaus.mojo.versions.api.PomHelper;
import org.codehaus.mojo.versions.change.VersionChange;
import org.codehaus.mojo.versions.change.VersionChanger;
import org.codehaus.mojo.versions.change.VersionChangerFactory;
import org.codehaus.mojo.versions.ordering.ReactorDepthComparator;
import org.codehaus.mojo.versions.rewriting.ModifiedPomXMLEventReader;
import org.codehaus.mojo.versions.utils.ContextualLog;
import org.codehaus.mojo.versions.utils.DelegatingContextualLog;
import org.codehaus.plexus.util.StringUtils;
import javax.xml.stream.XMLStreamException;
import java.io.File;
import java.io.IOException;
import java.util.*;
/**
* Sets the current projects version, updating the details of any child modules as necessary.
*
* @author Stephen Connolly
* @aggregator
* @requiresProject true
* @requiresDirectInvocation true
* @since 1.0-beta-1
*/
public abstract class AbstractSetMojo extends AbstractVersionsUpdaterMojo
{
/**
* The groupId of the dependency/module to update.
*
* @parameter expression="${groupId}" default-value="${project.groupId}"
* @since 1.2
*/
protected String groupId;
/**
* The artifactId of the dependecy/module to update.
*
* @parameter expression="${artifactId}" default-value="${project.artifactId}"
* @since 1.2
*/
protected String artifactId;
/**
* Whether matching versions explicitly specified (as /project/version) in child modules should be updated.
*
* @parameter expression="${updateMatchingVersions}" default-value="true"
* @since 1.3
*/
protected Boolean updateMatchingVersions;
/**
* The changes to module coordinates. Guarded by this.
*/
protected final transient List sourceChanges = new ArrayList();
/**
* Updates the pom file.
*
* @param pom The pom file to update.
* @throws org.apache.maven.plugin.MojoExecutionException
* when things go wrong.
* @throws org.apache.maven.plugin.MojoFailureException
* when things go wrong.
* @throws javax.xml.stream.XMLStreamException
* when things go wrong.
*/
protected synchronized void update( ModifiedPomXMLEventReader pom )
throws MojoExecutionException, MojoFailureException, XMLStreamException
{
ContextualLog log = new DelegatingContextualLog( getLog() );
try
{
Model model = PomHelper.getRawModel(pom);
log.setContext( "Processing " + PomHelper.getGroupId( model ) + ":" + PomHelper.getArtifactId( model ) );
VersionChangerFactory versionChangerFactory = new VersionChangerFactory();
versionChangerFactory.setPom( pom );
versionChangerFactory.setLog( log );
versionChangerFactory.setModel( model );
VersionChanger changer = versionChangerFactory.newVersionChanger();
Iterator i = sourceChanges.iterator();
while ( i.hasNext() )
{
VersionChange versionChange = (VersionChange) i.next();
changer.apply( versionChange );
}
}
catch ( IOException e )
{
throw new MojoExecutionException( e.getMessage(), e );
}
log.clearContext();
}
private synchronized void addChange( String groupId, String artifactId, String oldVersion, String newVersion )
{
if ( !newVersion.equals( oldVersion ) )
{
sourceChanges.add( new VersionChange( groupId, artifactId, oldVersion, newVersion ) );
}
}
protected void setVersion(String newVersion, String oldVersion) throws MojoExecutionException, MojoFailureException
{
if ( StringUtils.isEmpty(newVersion) )
{
throw new MojoExecutionException( "You must specify the new version, either by using the newVersion "
+ "property (that is -DnewVersion=... on the command line) or run in interactive mode" );
}
// this is the triggering change
addChange( groupId, artifactId, oldVersion, newVersion );
try
{
final MavenProject project =
PomHelper.getLocalRoot(projectBuilder, getProject(), localRepository, null, getLog());
getLog().info( "Local aggregation root: " + project.getBasedir() );
final Map reactor = PomHelper.getReactorModels( project, getLog() );
// now fake out the triggering change
final Model current =
PomHelper.getModel( reactor, getProject().getGroupId(), getProject().getArtifactId() );
current.setVersion( newVersion );
final Set files = new LinkedHashSet();
files.add( getProject().getFile() );
final List order = new ArrayList( reactor.keySet() );
Collections.sort( order, new ReactorDepthComparator( reactor ) );
final Iterator i = order.iterator();
while ( i.hasNext() )
{
final String sourcePath = (String) i.next();
final Model sourceModel = (Model) reactor.get( sourcePath );
getLog().debug( sourcePath.length() == 0
? "Processing root module as parent"
: "Processing " + sourcePath + " as a parent." );
final String sourceGroupId = PomHelper.getGroupId( sourceModel );
if ( sourceGroupId == null )
{
getLog().warn( "Module " + sourcePath + " is missing a groupId." );
continue;
}
final String sourceArtifactId = PomHelper.getArtifactId( sourceModel );
if ( sourceArtifactId == null )
{
getLog().warn( "Module " + sourcePath + " is missing an artifactId." );
continue;
}
final String sourceVersion = PomHelper.getVersion( sourceModel );
if ( sourceVersion == null )
{
getLog().warn( "Module " + sourcePath + " is missing a version." );
continue;
}
final File moduleDir = new File( project.getBasedir(), sourcePath );
final File moduleProjectFile;
if ( moduleDir.isDirectory() )
{
moduleProjectFile = new File( moduleDir, "pom.xml" );
}
else
{
// i don't think this should ever happen... but just in case
// the module references the file-name
moduleProjectFile = moduleDir;
}
files.add( moduleProjectFile );
getLog().debug(
"Looking for modules which use " + ArtifactUtils.versionlessKey(sourceGroupId, sourceArtifactId)
+ " as their parent" );
final Iterator j =
PomHelper.getChildModels( reactor, sourceGroupId, sourceArtifactId ).entrySet().iterator();
while ( j.hasNext() )
{
final Map.Entry target = (Map.Entry) j.next();
final String targetPath = (String) target.getKey();
final Model targetModel = (Model) target.getValue();
final Parent parent = targetModel.getParent();
getLog().debug( "Module: " + targetPath );
if ( sourceVersion.equals( parent.getVersion() ) )
{
getLog().debug(
" parent already is " + ArtifactUtils.versionlessKey( sourceGroupId, sourceArtifactId )
+ ":" + sourceVersion );
}
else
{
getLog().debug(
" parent is " + ArtifactUtils.versionlessKey( sourceGroupId, sourceArtifactId ) + ":"
+ parent.getVersion() );
getLog().debug(
" will become " + ArtifactUtils.versionlessKey( sourceGroupId, sourceArtifactId ) + ":"
+ sourceVersion );
}
final boolean targetExplicit = PomHelper.isExplicitVersion( targetModel );
if ( ( updateMatchingVersions.booleanValue() || !targetExplicit ) && StringUtils.equals(
parent.getVersion(), PomHelper.getVersion( targetModel ) ) )
{
getLog().debug( " module is "
+ ArtifactUtils.versionlessKey( PomHelper.getGroupId( targetModel ),
PomHelper.getArtifactId( targetModel ) ) + ":"
+ PomHelper.getVersion( targetModel ) );
getLog().debug( " will become "
+ ArtifactUtils.versionlessKey( PomHelper.getGroupId( targetModel ),
PomHelper.getArtifactId( targetModel ) ) + ":"
+ sourceVersion );
addChange( PomHelper.getGroupId( targetModel ), PomHelper.getArtifactId( targetModel ),
PomHelper.getVersion( targetModel ), sourceVersion );
targetModel.setVersion( sourceVersion );
}
else
{
getLog().debug( " module is "
+ ArtifactUtils.versionlessKey( PomHelper.getGroupId( targetModel ),
PomHelper.getArtifactId( targetModel ) ) + ":"
+ PomHelper.getVersion( targetModel ) );
}
}
}
// now process all the updates
final Iterator k = files.iterator();
while ( k.hasNext() )
{
process( (File) k.next() );
}
}
catch ( IOException e )
{
throw new MojoExecutionException( e.getMessage(), e );
}
}
}