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

org.apache.maven.plugin.changes.ChangesMojo Maven / Gradle / Ivy

Go to download

Creates a release history for inclusion into the site and assists in generating an announcement mail.

There is a newer version: 2.12.1
Show newest version
package org.apache.maven.plugin.changes;

/*
 * 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.net.URL;
import java.text.SimpleDateFormat;
import java.util.Collections;
import java.util.Date;
import java.util.Iterator;
import java.util.Locale;
import java.util.Map;
import java.util.Properties;
import java.util.ResourceBundle;

import org.apache.commons.collections.map.CaseInsensitiveMap;
import org.apache.maven.execution.MavenSession;
import org.apache.maven.reporting.MavenReportException;
import org.apache.maven.shared.filtering.MavenFileFilter;
import org.apache.maven.shared.filtering.MavenFileFilterRequest;
import org.apache.maven.shared.filtering.MavenFilteringException;
import org.codehaus.plexus.util.FileUtils;
import org.codehaus.plexus.util.IOUtil;
import org.codehaus.plexus.util.ReaderFactory;
import org.codehaus.plexus.util.StringUtils;
import org.codehaus.plexus.util.xml.XmlStreamReader;

/**
 * Goal which creates a nicely formatted Changes Report in html format from a changes.xml file.
 *
 * @goal changes-report
 * @author Johnny R. Ruiz III
 * @version $Id: ChangesMojo.java 1062681 2011-01-24 08:48:26Z krosenvold $
 * @threadSafe
 */
public class ChangesMojo
    extends AbstractChangesReport
{
    /**
     * A flag whether the report should also include the dates of individual actions. If set to false, only
     * the dates of releases will be written to the report.
     *
     * @parameter expression="${changes.addActionDate}" default-value="false"
     * @since 2.1
     */
    private boolean addActionDate;

    /**
     * Whether HTML code within an action should be escaped. By changing this to
     * false you can restore the behavior that was in version 2.2
     * of this plugin, allowing you to use HTML code to format the content of an
     * action.
     * 

* Note: If you use HTML code in an action you need to * place it inside a CDATA section. *

* Note: Putting any kind of markup inside a CDATA section * might mess up the Changes Report or other generated documents, such as * PDFs, that are based on your changes.xml file if you are not * careful. * * @parameter default-value="true" * @since 2.4 * @deprecated using markup inside CDATA sections does not work for all output formats! */ private boolean escapeHTML; /** * The directory for interpolated changes.xml. * * @parameter expression="${project.build.directory}/changes" * @required * @readonly * @since 2.2 */ private File filteredOutputDirectory; /** * applying filtering filtering "a la" resources plugin * * @parameter default-value="false" * @since 2.2 */ private boolean filteringChanges; /** * Template string that is used to discover the URL to use to display an issue report. * There are 2 template tokens you can use. %URL%: this is computed by getting the * <issueManagement>/<url> value from the POM, and removing the last '/' * and everything that comes after it. %ISSUE%: this is the issue number. *

* Note: In versions of this plugin prior to 2.0-beta-2 this parameter was called * link_template. *

* * @parameter expression="${changes.issueLinkTemplate}" default-value="%URL%/ViewIssue.jspa?key=%ISSUE%" * @since 2.0-beta-2 * @deprecated As of 2.1 use issueLinkTemplatePerSystem : this one will be with system default */ private String issueLinkTemplate; /** * Template strings per system that is used to discover the URL to use to display an issue report. Each key in this * map denotes the (case-insensitive) identifier of the issue tracking system and its value gives the URL template. *

* There are 2 template tokens you can use. %URL%: this is computed by getting the * <issueManagement>/<url> value from the POM, and removing the last '/' * and everything that comes after it. %ISSUE%: this is the issue number. *

*

* Note: The deprecated issueLinkTemplate will be used for a system called "default". *

*

* Note: Starting with version 2.4 you usually don't need * to specify this, unless you need to link to an issue management system in * your Changes report that isn't supported out of the box. See the * Usage page for more * information. *

* * @parameter * @since 2.1 */ private Map issueLinkTemplatePerSystem; /** * @component * @since 2.2 */ private MavenFileFilter mavenFileFilter; /** * Format to use for publishDate. The value will be available with the following expression ${publishDate} * * @see SimpleDateFormat * @parameter default-value="yyyy-MM-dd" * @since 2.2 */ private String publishDateFormat; /** * Locale to use for publishDate when formatting * * @see Locale * @parameter default-value="en" * @since 2.2 */ private String publishDateLocale; /** * @parameter expression="${session}" * @readonly * @required * @since 2.2 */ protected MavenSession session; /** * @parameter default-value="${project.issueManagement.system}" * @readonly * @since 2.4 */ private String system; /** * The URI of a file containing all the team members. If this is set to the * special value "none", no links will be generated for the team members. * * @parameter default-value="team-list.html" * @since 2.4 */ private String teamlist; /** * @parameter default-value="${project.issueManagement.url}" * @readonly */ private String url; /** * The path of the changes.xml file that will be converted into an HTML report. * * @parameter expression="${changes.xmlPath}" default-value="src/changes/changes.xml" */ private File xmlPath; private CaseInsensitiveMap caseInsensitiveIssueLinkTemplatePerSystem; /* --------------------------------------------------------------------- */ /* Public methods */ /* --------------------------------------------------------------------- */ public boolean canGenerateReport() { return xmlPath.isFile(); } public void executeReport( Locale locale ) throws MavenReportException { if ( !xmlPath.exists() ) { getLog().warn( "changes.xml file " + xmlPath.getAbsolutePath() + " does not exist." ); return; } if ( filteringChanges ) { if ( !filteredOutputDirectory.exists() ) { filteredOutputDirectory.mkdirs(); } XmlStreamReader xmlStreamReader = null; try { // so we get encoding from the file itself xmlStreamReader = ReaderFactory.newXmlReader( xmlPath ); String encoding = xmlStreamReader.getEncoding(); File resultFile = new File( filteredOutputDirectory, "changes.xml" ); Date now = new Date(); SimpleDateFormat simpleDateFormat = new SimpleDateFormat( publishDateFormat, new Locale( publishDateLocale ) ); Properties additionalProperties = new Properties(); additionalProperties.put( "publishDate", simpleDateFormat.format( now ) ); MavenFileFilterRequest mavenFileFilterRequest = new MavenFileFilterRequest( xmlPath, resultFile, true, project, Collections.EMPTY_LIST, false, encoding, session, additionalProperties ); mavenFileFilter.copyFile( mavenFileFilterRequest ); xmlPath = resultFile; } catch ( IOException e ) { throw new MavenReportException( "Exception during filtering changes file : " + e.getMessage(), e ); } catch ( MavenFilteringException e ) { throw new MavenReportException( "Exception during filtering changes file : " + e.getMessage(), e ); } finally { if ( xmlStreamReader != null ) { IOUtil.close( xmlStreamReader ); } } } ChangesXML changesXml = new ChangesXML( xmlPath, getLog() ); ChangesReportGenerator report = new ChangesReportGenerator( changesXml.getReleaseList() ); report.setAuthor( changesXml.getAuthor() ); report.setTitle( changesXml.getTitle() ); report.setEscapeHTML ( escapeHTML ); // Create a case insensitive version of issueLinkTemplatePerSystem // We need something case insensitive to maintain backward compatibility if ( issueLinkTemplatePerSystem == null ) { caseInsensitiveIssueLinkTemplatePerSystem = new CaseInsensitiveMap(); } else { caseInsensitiveIssueLinkTemplatePerSystem = new CaseInsensitiveMap( issueLinkTemplatePerSystem ); } // Set good default values for issue management systems here, but only // if they have not been configured already by the user addIssueLinkTemplate( ChangesReportGenerator.DEFAULT_ISSUE_SYSTEM_KEY, issueLinkTemplate ); addIssueLinkTemplate( "Bugzilla", "%URL%/show_bug.cgi?id=%ISSUE%" ); addIssueLinkTemplate( "GoogleCode", "%URL%/detail?id=%ISSUE%" ); addIssueLinkTemplate( "JIRA", "%URL%/%ISSUE%" ); addIssueLinkTemplate( "Mantis", "%URL%/view.php?id=%ISSUE%" ); addIssueLinkTemplate( "Redmine", "%URL%/issues/show/%ISSUE%" ); addIssueLinkTemplate( "Scarab", "%URL%/issues/id/%ISSUE%" ); addIssueLinkTemplate( "SourceForge", "http://sourceforge.net/support/tracker.php?aid=%ISSUE%" ); addIssueLinkTemplate( "Trac", "%URL%/ticket/%ISSUE%" ); addIssueLinkTemplate( "YouTrack", "%URL%/issue/%ISSUE%" ); // @todo Add more issue management systems here // Show the current issueLinkTemplatePerSystem configuration logIssueLinkTemplatePerSystem( caseInsensitiveIssueLinkTemplatePerSystem ); report.setIssueLinksPerSystem( caseInsensitiveIssueLinkTemplatePerSystem ); report.setSystem( system ); report.setTeamlist ( teamlist ); report.setUrl( url ); report.setAddActionDate( addActionDate ); if ( StringUtils.isEmpty( url ) ) { getLog().warn( "No issue management URL defined in POM. Links to your issues will not work correctly." ); } report.doGenerateReport( getBundle( locale ), getSink() ); // Copy the images copyStaticResources(); } public String getDescription( Locale locale ) { return getBundle( locale ).getString( "report.issues.description" ); } public String getName( Locale locale ) { return getBundle( locale ).getString( "report.issues.name" ); } public String getOutputName() { return "changes-report"; } /* --------------------------------------------------------------------- */ /* Private methods */ /* --------------------------------------------------------------------- */ /** * Add the issue link template for the given issue management system, * but only if it has not already been configured. * * @param system The issue management system * @param issueLinkTemplate The issue link template to use * @since 2.4 */ private void addIssueLinkTemplate( String system, String issueLinkTemplate ) { if ( caseInsensitiveIssueLinkTemplatePerSystem == null ) { caseInsensitiveIssueLinkTemplatePerSystem = new CaseInsensitiveMap(); } if ( !caseInsensitiveIssueLinkTemplatePerSystem.containsKey( system ) ) { caseInsensitiveIssueLinkTemplatePerSystem.put( system, issueLinkTemplate ); } } private void copyStaticResources() throws MavenReportException { final String pluginResourcesBase = "org/apache/maven/plugin/changes"; String resourceNames[] = { "images/add.gif", "images/fix.gif", "images/icon_help_sml.gif", "images/remove.gif", "images/rss.png", "images/update.gif" }; try { getLog().debug( "Copying static resources." ); for ( int i = 0; i < resourceNames.length; i++ ) { URL url = this.getClass().getClassLoader().getResource( pluginResourcesBase + "/" + resourceNames[i] ); FileUtils.copyURLToFile( url, new File( getReportOutputDirectory(), resourceNames[i] ) ); } } catch ( IOException e ) { throw new MavenReportException( "Unable to copy static resources." ); } } private ResourceBundle getBundle( Locale locale ) { return ResourceBundle.getBundle( "changes-report", locale, this.getClass().getClassLoader() ); } protected String getTeamlist() { return teamlist; } private void logIssueLinkTemplatePerSystem( Map issueLinkTemplatePerSystem ) { if ( getLog().isDebugEnabled() ) { if ( issueLinkTemplatePerSystem == null ) { getLog().debug( "No issueLinkTemplatePerSystem configuration was found" ); } else { Iterator iterator = issueLinkTemplatePerSystem.entrySet().iterator(); while ( iterator.hasNext() ) { Map.Entry entry = (Map.Entry) iterator.next(); getLog().debug( "issueLinkTemplatePerSystem[" + entry.getKey() + "] = " + entry.getValue() ); } } } } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy