All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.maven.plugin.announcement.AnnouncementMojo Maven / Gradle / Ivy

package org.apache.maven.plugin.announcement;

/*
 * 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.FileOutputStream;
import java.io.OutputStreamWriter;
import java.io.Writer;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.Map;

import org.apache.maven.plugin.MojoExecutionException;
import org.apache.maven.plugin.changes.ChangesXML;
import org.apache.maven.plugin.changes.IssueAdapter;
import org.apache.maven.plugin.changes.ProjectUtils;
import org.apache.maven.plugin.changes.ReleaseUtils;
import org.apache.maven.plugin.github.GitHubDownloader;
import org.apache.maven.plugin.github.GitHubIssueManagementSystem;
import org.apache.maven.plugin.issues.Issue;
import org.apache.maven.plugin.issues.IssueManagementSystem;
import org.apache.maven.plugin.issues.IssueUtils;
import org.apache.maven.plugin.jira.AbstractJiraDownloader;
import org.apache.maven.plugin.jira.AdaptiveJiraDownloader;
import org.apache.maven.plugin.jira.JIRAIssueManagmentSystem;
import org.apache.maven.plugin.trac.TracDownloader;
import org.apache.maven.plugin.trac.TracIssueManagmentSystem;
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.plugins.changes.model.Release;
import org.apache.maven.project.MavenProject;
import org.apache.maven.settings.Settings;
import org.apache.velocity.Template;
import org.apache.velocity.app.VelocityEngine;
import org.apache.velocity.context.Context;
import org.apache.velocity.exception.ResourceNotFoundException;
import org.apache.velocity.exception.VelocityException;
import org.apache.velocity.tools.ToolManager;
import org.codehaus.plexus.util.ReaderFactory;
import org.codehaus.plexus.util.StringUtils;
import org.codehaus.plexus.velocity.VelocityComponent;

/**
 * Goal which generate an announcement from the announcement template.
 *
 * @author [email protected]
 * @version $Id: AnnouncementMojo.java 1765156 2016-10-16 13:54:35Z gboue $
 * @since 2.0-beta-2
 */
@Mojo( name = "announcement-generate", threadSafe = true )
public class AnnouncementMojo
    extends AbstractAnnouncementMojo
{
    private static final String CHANGES_XML = "changes.xml";

    private static final String JIRA = "JIRA";

    private static final String TRAC = "Trac";

    private static final String GIT_HUB = "GitHub";

    /**
     * The name of the file which will contain the generated announcement. If no value is specified the plugin will use
     * the name of the template.
     *
     * @since 2.4
     */
    @Parameter( property = "changes.announcementFile" )
    private String announcementFile;

    /**
     * Map of custom parameters for the announcement. This Map will be passed to the template.
     *
     * @since 2.1
     */
    @Parameter
    private Map announceParameters;

    /**
     */
    @Parameter( property = "project.artifactId", readonly = true )
    private String artifactId;

    /**
     * Name of the team that develops the artifact. This parameter will be passed to the template.
     */
    @Parameter( property = "changes.developmentTeam", defaultValue = "${project.name} team", required = true )
    private String developmentTeam;

    /**
     * The name of the artifact to be used in the announcement.
     */
    @Parameter( property = "changes.finalName", defaultValue = "${project.build.finalName}", required = true )
    private String finalName;

    /**
     */
    @Parameter( property = "project.groupId", readonly = true )
    private String groupId;

    /**
     * Short description or introduction of the released artifact. This parameter will be passed to the template.
     */
    @Parameter( defaultValue = "${project.description}" )
    private String introduction;

    /**
     * A list of issue management systems to fetch releases from. This parameter replaces the parameters
     * generateJiraAnnouncement and jiraMerge.
     * 

* Valid values are: changes.xml and JIRA. *

* Note: Only one issue management system that is configured in * <project>/<issueManagement> can be used. This currently means that you can combine a changes.xml file * with one other issue management system. * * @since 2.4 */ @Parameter private List issueManagementSystems; /** * Maps issues types to action types for grouping issues in announcements. If issue types are not defined for a * action type then the default issue type will be applied. *

* Valid action types: add, fix and update. *

* * @since 2.6 */ @Parameter private Map issueTypes; /** * Directory where the announcement file will be generated. * * @since 2.10 */ @Parameter( defaultValue = "${project.build.directory}/announcement", required = true ) private File announcementDirectory; /** * Directory where the announcement file will be generated. * * @deprecated Starting with version 2.10 this parameter is no longer used. You must use * {@link #announcementDirectory} instead. */ @Parameter private File outputDirectory; /** * Packaging structure for the artifact. */ @Parameter( property = "project.packaging", readonly = true ) private String packaging; /** * The Maven Project. */ @Parameter( defaultValue = "${project}", readonly = true, required = true ) private MavenProject project; /** * The Velocity template used to format the announcement. */ @Parameter( property = "changes.template", defaultValue = "announcement.vm", required = true ) private String template; /** * Directory that contains the template. *

* Note: This directory must be a subdirectory of * /src/main/resources/ or current project base directory. *

*/ // CHECKSTYLE_OFF: LineLength @Parameter( property = "changes.templateDirectory", defaultValue = "org/apache/maven/plugin/announcement", required = true ) private String templateDirectory; // CHECKSTYLE_ON: LineLength /** * The template encoding. * * @since 2.1 */ @Parameter( property = "changes.templateEncoding", defaultValue = "${project.build.sourceEncoding}" ) private String templateEncoding; /** * Use the JIRA query language instead of the JIRA query based on HTTP parameters. From JIRA 5.1 and up only JQL is * supported. JIRA 4.4 supports both JQL and URL parameter based queries. From 5.1.1 this is obsolete, since REST * queries only use JQL. * * @since 2.10 */ @Parameter( property = "changes.useJql", defaultValue = "false" ) private boolean useJql; /** * Distribution URL of the artifact. This parameter will be passed to the template. */ @Parameter( property = "project.url" ) private String url; /** * URL where the artifact can be downloaded. If not specified, no URL is used. This parameter will be passed to the * template. */ @Parameter private String urlDownload; /** * Velocity Component. */ @Component( role = VelocityComponent.class, hint = "maven-changes-plugin" ) private VelocityComponent velocity; /** * Version of the artifact. */ @Parameter( property = "changes.version", defaultValue = "${project.version}", required = true ) private String version; /** * The path of the changes.xml file. * * @parameter expression="${basedir}/src/changes/changes.xml" * @required */ @Parameter( defaultValue = "${basedir}/src/changes/changes.xml" ) private File xmlPath; // =======================================// // JIRA-Announcement Needed Parameters // // =======================================// /** * Defines the filter parameters to restrict which issues are retrieved from JIRA. The filter parameter uses the * same format of url parameters that is used in a JIRA search. * * @since 2.4 */ @Parameter( defaultValue = "" ) private String filter; /** * Flag to determine if the plugin will generate a JIRA announcement. * * @deprecated Since version 2.4 this parameter has been deprecated. Please use the issueManagementSystems parameter * instead. */ @Parameter( property = "generateJiraAnnouncement", defaultValue = "false", required = true ) private boolean generateJiraAnnouncement; /** * If releases from JIRA should be merged with the releases from a changes.xml file. * * @since 2.1 * @deprecated Since version 2.4 this parameter has been deprecated. Please use the issueManagementSystems parameter * instead. */ @Parameter( property = "changes.jiraMerge", defaultValue = "false" ) private boolean jiraMerge; /** * Defines the JIRA password for authentication into a private JIRA installation. * * @since 2.1 */ @Parameter( property = "changes.jiraPassword", defaultValue = "" ) private String jiraPassword; /** * Defines the JIRA username for authentication into a private JIRA installation. * * @since 2.1 */ @Parameter( property = "changes.jiraUser", defaultValue = "" ) private String jiraUser; /** * Path to the JIRA XML file, which will be parsed. */ @Parameter( defaultValue = "${project.build.directory}/jira-announcement.xml", required = true, readonly = true ) private File jiraXML; /** * The maximum number of issues to fetch from JIRA. *

* Note: In versions 2.0-beta-3 and earlier this parameter was called "nbEntries". *

*/ @Parameter( property = "changes.maxEntries", defaultValue = "25", required = true ) private int maxEntries; /** * Include issues from JIRA with these resolution ids. Multiple resolution ids can be specified as a comma separated * list of ids. *

* Note: In versions 2.0-beta-3 and earlier this parameter was called "resolutionId". *

*/ @Parameter( property = "changes.resolutionIds", defaultValue = "Fixed" ) private String resolutionIds; /** * Settings XML configuration. */ @Parameter( defaultValue = "${settings}", readonly = true, required = true ) private Settings settings; /** * Include issues from JIRA with these status ids. Multiple status ids can be specified as a comma separated list of * ids. *

* Note: In versions 2.0-beta-3 and earlier this parameter was called "statusId". *

*/ @Parameter( property = "changes.statusIds", defaultValue = "Closed" ) private String statusIds; /** * Defines the http user for basic authentication into the JIRA webserver. * * @since 2.4 */ @Parameter( property = "changes.webUser", defaultValue = "" ) private String webUser; /** * Defines the http password for basic authentication into the JIRA webserver. * * @since 2.4 */ @Parameter( property = "changes.webPassword", defaultValue = "" ) private String webPassword; /** * The prefix used when naming versions in JIRA. *

* If you have a project in JIRA with several components that have different release cycles, it is an often used * pattern to prefix the version with the name of the component, e.g. maven-filtering-1.0 etc. To fetch issues from * JIRA for a release of the "maven-filtering" component you would need to set this parameter to "maven-filtering-". *

* * @since 2.5 */ @Parameter( property = "changes.versionPrefix", defaultValue = "" ) private String versionPrefix; /** * Defines the connection timeout in milliseconds when accessing JIRA's REST-API. *

* Might help when you have a lot of different resolutions in your JIRA instance. *

* * @since 2.11 */ @Parameter( property = "changes.jiraConnectionTimeout", defaultValue = "36000" ) private int jiraConnectionTimeout; /** * Defines the receive timeout in milliseconds when accessing JIRA's REST-API. *

* Might help when you have a lot of different resolutions in your JIRA instance. *

* * @since 2.11 */ @Parameter( property = "changes.jiraReceiveTimout", defaultValue = "32000" ) private int jiraReceiveTimout; // =======================================// // Trac Parameters // // =======================================// /** * Defines the Trac password for authentication into a private Trac installation. * * @since 2.4 */ @Parameter( property = "changes.tracPassword", defaultValue = "" ) private String tracPassword; /** * Defines the Trac query for searching for tickets. * * @since 2.4 */ @Parameter( defaultValue = "order=id" ) private String tracQuery; /** * Defines the Trac username for authentication into a private Trac installation. * * @since 2.4 */ @Parameter( property = "changes.tracUser", defaultValue = "" ) private String tracUser; // =======================================// // Github Parameters // // =======================================// /** * The scheme of your github api domain. Only use if using github enterprise. * * @since 2.9 */ @Parameter( defaultValue = "http", property = "changes.githubAPIScheme" ) private String githubAPIScheme; /** * The port of your github api domain. Only use if using github enterprise. * * @since 2.9 */ @Parameter( defaultValue = "80", property = "changes.githubAPIPort" ) private int githubAPIPort; /** * The settings.xml server id to be used to authenticate into github api domain. Only use if using github * enterprise. * * @since 2.12 */ @Parameter( defaultValue = "github" ) private String githubAPIServerId; /** * Boolean which says if we should include open github issues in the announcement. */ @Parameter( defaultValue = "false" ) private boolean includeOpenIssues; private ReleaseUtils releaseUtils = new ReleaseUtils( getLog() ); private ChangesXML xml; // =======================================// // announcement-generate execution // // =======================================// /** * Generate the template * * @throws MojoExecutionException */ public void execute() throws MojoExecutionException { // Fail build fast if it is using deprecated parameters if ( outputDirectory != null ) { throw new MojoExecutionException( "You are using the old parameter 'outputDirectory'. " + "You must use 'announcementDirectory' instead." ); } // Run only at the execution root if ( runOnlyAtExecutionRoot && !isThisTheExecutionRoot() ) { getLog().info( "Skipping the announcement generation in this project because it's not the Execution Root" ); } else { if ( issueManagementSystems == null ) { issueManagementSystems = new ArrayList(); } // Handle deprecated parameters, in a backward compatible way if ( issueManagementSystems.isEmpty() ) { if ( this.jiraMerge ) { issueManagementSystems.add( CHANGES_XML ); issueManagementSystems.add( JIRA ); } else if ( generateJiraAnnouncement ) { issueManagementSystems.add( JIRA ); } else { issueManagementSystems.add( CHANGES_XML ); } } // Fetch releases from the configured issue management systems List releases = null; if ( issueManagementSystems.contains( CHANGES_XML ) ) { if ( getXmlPath().exists() ) { ChangesXML changesXML = new ChangesXML( getXmlPath(), getLog() ); List changesReleases = releaseUtils.convertReleaseList( changesXML.getReleaseList() ); releases = releaseUtils.mergeReleases( null, changesReleases ); getLog().info( "Including issues from file " + getXmlPath() + " in announcement..." ); } else { getLog().warn( "changes.xml file " + getXmlPath().getAbsolutePath() + " does not exist." ); } } if ( issueManagementSystems.contains( JIRA ) ) { String message = ProjectUtils.validateIssueManagement( project, JIRA, "JIRA announcement" ); if ( message == null ) { List jiraReleases = getJiraReleases(); releases = releaseUtils.mergeReleases( releases, jiraReleases ); getLog().info( "Including issues from JIRA in announcement..." ); } else { throw new MojoExecutionException( "Something is wrong with the Issue Management section. " + message ); } } if ( issueManagementSystems.contains( TRAC ) ) { String message = ProjectUtils.validateIssueManagement( project, TRAC, "Trac announcement" ); if ( message == null ) { List tracReleases = getTracReleases(); releases = releaseUtils.mergeReleases( releases, tracReleases ); getLog().info( "Including issues from Trac in announcement..." ); } else { throw new MojoExecutionException( "Something is wrong with the Issue Management section. " + message ); } } if ( issueManagementSystems.contains( GIT_HUB ) ) { String message = ProjectUtils.validateIssueManagement( project, GIT_HUB, "GitHub announcement" ); if ( message == null ) { List gitHubReleases = getGitHubReleases(); releases = releaseUtils.mergeReleases( releases, gitHubReleases ); getLog().info( "Including issues from GitHub in announcement..." ); } else { throw new MojoExecutionException( "Something is wrong with the Issue Management section. " + message ); } } // @todo Add more issue management systems here. // Follow these steps: // 1. Add a constant for the name of the issue management system // 2. Add the @parameters needed to configure the issue management system // 3. Add a protected List getReleases() method that retrieves a list of releases // 4. Merge those releases into the "releases" variable // For help with these steps, you can have a look at how this has been done for JIRA or Trac // Generate the report if ( releases == null || releases.isEmpty() ) { throw new MojoExecutionException( "No releases found in any of the " + "configured issue management systems." ); } else { doGenerate( releases ); } } } /** * Add the parameters to velocity context * * @param releases A List of Releases * @throws MojoExecutionException */ public void doGenerate( List releases ) throws MojoExecutionException { String version = ( versionPrefix == null ? "" : versionPrefix ) + getVersion(); getLog().debug( "Generating announcement for version [" + version + "]. Found these releases: " + ReleaseUtils.toString( releases ) ); doGenerate( releases, releaseUtils.getLatestRelease( releases, version ) ); } protected void doGenerate( List releases, Release release ) throws MojoExecutionException { try { ToolManager toolManager = new ToolManager( true ); Context context = toolManager.createContext(); if ( getIntroduction() == null || getIntroduction().equals( "" ) ) { setIntroduction( getUrl() ); } context.put( "releases", releases ); context.put( "groupId", getGroupId() ); context.put( "artifactId", getArtifactId() ); context.put( "version", getVersion() ); context.put( "packaging", getPackaging() ); context.put( "url", getUrl() ); context.put( "release", release ); context.put( "introduction", getIntroduction() ); context.put( "developmentTeam", getDevelopmentTeam() ); context.put( "finalName", getFinalName() ); context.put( "urlDownload", getUrlDownload() ); context.put( "project", project ); if ( announceParameters == null ) { // empty Map to prevent NPE in velocity execution context.put( "announceParameters", Collections.EMPTY_MAP ); } else { context.put( "announceParameters", announceParameters ); } processTemplate( context, announcementDirectory, template, announcementFile ); } catch ( ResourceNotFoundException rnfe ) { throw new MojoExecutionException( "Resource not found.", rnfe ); } catch ( VelocityException ve ) { throw new MojoExecutionException( ve.toString(), ve ); } } /** * Create the velocity template * * @param context velocity context that has the parameter values * @param outputDirectory directory where the file will be generated * @param template velocity template which will the context be merged * @param announcementFile The file name of the generated announcement * @throws ResourceNotFoundException, VelocityException, IOException */ public void processTemplate( Context context, File outputDirectory, String template, String announcementFile ) throws VelocityException, MojoExecutionException { File f; // Use the name of the template as a default value if ( StringUtils.isEmpty( announcementFile ) ) { announcementFile = template; } try { f = new File( outputDirectory, announcementFile ); if ( !f.getParentFile().exists() ) { f.getParentFile().mkdirs(); } VelocityEngine engine = velocity.getEngine(); engine.setApplicationAttribute( "baseDirectory", basedir ); if ( StringUtils.isEmpty( templateEncoding ) ) { templateEncoding = ReaderFactory.FILE_ENCODING; getLog().warn( "File encoding has not been set, using platform encoding " + templateEncoding + ", i.e. build is platform dependent!" ); } Writer writer = new OutputStreamWriter( new FileOutputStream( f ), templateEncoding ); Template velocityTemplate = engine.getTemplate( templateDirectory + "/" + template, templateEncoding ); velocityTemplate.merge( context, writer ); writer.flush(); writer.close(); getLog().info( "Created template " + f ); } catch ( ResourceNotFoundException rnfe ) { throw new ResourceNotFoundException( "Template not found. ( " + templateDirectory + "/" + template + " )" ); } catch ( VelocityException ve ) { throw new VelocityException( ve.toString() ); } catch ( Exception e ) { if ( e.getCause() != null ) { getLog().warn( e.getCause() ); } throw new MojoExecutionException( e.toString(), e.getCause() ); } } protected List getJiraReleases() throws MojoExecutionException { AbstractJiraDownloader jiraDownloader = new AdaptiveJiraDownloader(); File jiraXMLFile = jiraXML; jiraDownloader.setLog( getLog() ); jiraDownloader.setOutput( jiraXMLFile ); jiraDownloader.setStatusIds( statusIds ); jiraDownloader.setResolutionIds( resolutionIds ); jiraDownloader.setMavenProject( project ); jiraDownloader.setSettings( settings ); jiraDownloader.setNbEntries( maxEntries ); jiraDownloader.setFilter( filter ); jiraDownloader.setJiraUser( jiraUser ); jiraDownloader.setJiraPassword( jiraPassword ); jiraDownloader.setUseJql( useJql ); jiraDownloader.setWebUser( webUser ); jiraDownloader.setWebPassword( webPassword ); jiraDownloader.setConnectionTimeout( jiraConnectionTimeout ); jiraDownloader.setReceiveTimout( jiraReceiveTimout ); try { jiraDownloader.doExecute(); List issueList = jiraDownloader.getIssueList(); if ( StringUtils.isNotEmpty( versionPrefix ) ) { int originalNumberOfIssues = issueList.size(); issueList = IssueUtils.filterIssuesWithVersionPrefix( issueList, versionPrefix ); getLog().debug( "Filtered out " + issueList.size() + " issues of " + originalNumberOfIssues + " that matched the versionPrefix '" + versionPrefix + "'." ); } return getReleases( issueList, new JIRAIssueManagmentSystem() ); } catch ( Exception e ) { throw new MojoExecutionException( "Failed to extract issues from JIRA.", e ); } } private List getReleases( List issues, IssueManagementSystem ims ) throws MojoExecutionException { if ( issueTypes != null ) { ims.applyConfiguration( issueTypes ); } if ( issues.isEmpty() ) { return Collections.emptyList(); } else { IssueAdapter adapter = new IssueAdapter( ims ); return adapter.getReleases( issues ); } } protected List getTracReleases() throws MojoExecutionException { TracDownloader issueDownloader = new TracDownloader(); issueDownloader.setProject( project ); issueDownloader.setQuery( tracQuery ); issueDownloader.setTracPassword( tracPassword ); issueDownloader.setTracUser( tracUser ); try { return getReleases( issueDownloader.getIssueList(), new TracIssueManagmentSystem() ); } catch ( Exception e ) { throw new MojoExecutionException( "Failed to extract issues from Trac.", e ); } } protected List getGitHubReleases() throws MojoExecutionException { try { GitHubDownloader issueDownloader = new GitHubDownloader( project, githubAPIScheme, githubAPIPort, includeOpenIssues, true ); issueDownloader.configureAuthentication( githubAPIServerId, settings, getLog() ); return getReleases( issueDownloader.getIssueList(), new GitHubIssueManagementSystem() ); } catch ( Exception e ) { throw new MojoExecutionException( "Failed to extract issues from GitHub.", e ); } } /* * accessors */ public String getArtifactId() { return artifactId; } public void setArtifactId( String artifactId ) { this.artifactId = artifactId; } public String getDevelopmentTeam() { return developmentTeam; } public void setDevelopmentTeam( String developmentTeam ) { this.developmentTeam = developmentTeam; } public String getFinalName() { return finalName; } public void setFinalName( String finalName ) { this.finalName = finalName; } public String getGroupId() { return groupId; } public void setGroupId( String groupId ) { this.groupId = groupId; } public String getIntroduction() { return introduction; } public void setIntroduction( String introduction ) { this.introduction = introduction; } public void setIssueTypes( Map issueTypes ) { this.issueTypes = issueTypes; } public Map getIssueTypes() { return issueTypes; } public File getAnnouncementDirectory() { return announcementDirectory; } public void setAnnouncementDirectory( File announcementDirectory ) { this.announcementDirectory = announcementDirectory; } public String getPackaging() { return packaging; } public void setPackaging( String packaging ) { this.packaging = packaging; } public String getUrl() { return url; } public void setUrl( String url ) { this.url = url; } public String getUrlDownload() { return urlDownload; } public void setUrlDownload( String urlDownload ) { this.urlDownload = urlDownload; } public VelocityComponent getVelocity() { return velocity; } public void setVelocity( VelocityComponent velocity ) { this.velocity = velocity; } public String getVersion() { return version; } public void setVersion( String version ) { this.version = version; } public ChangesXML getXml() { return xml; } public void setXml( ChangesXML xml ) { this.xml = xml; } public File getXmlPath() { return xmlPath; } public void setXmlPath( File xmlPath ) { this.xmlPath = xmlPath; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy