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

org.apache.maven.plugin.checkstyle.AbstractCheckstyleReport Maven / Gradle / Ivy

Go to download

Generates a report on violations of code style and optionally fails the build if violations are detected.

There is a newer version: 3.5.0
Show newest version
package org.apache.maven.plugin.checkstyle;

/*
 * 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.ByteArrayOutputStream;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.util.ArrayList;
import java.util.Calendar;
import java.util.Collections;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.ResourceBundle;

import org.apache.maven.artifact.Artifact;
import org.apache.maven.doxia.tools.SiteTool;
import org.apache.maven.model.Dependency;
import org.apache.maven.model.Plugin;
import org.apache.maven.model.PluginManagement;
import org.apache.maven.model.ReportPlugin;
import org.apache.maven.model.Resource;
import org.apache.maven.plugin.MojoExecution;
import org.apache.maven.plugin.checkstyle.exec.CheckstyleExecutor;
import org.apache.maven.plugin.checkstyle.exec.CheckstyleExecutorException;
import org.apache.maven.plugin.checkstyle.exec.CheckstyleExecutorRequest;
import org.apache.maven.plugin.checkstyle.exec.CheckstyleResults;
import org.apache.maven.plugin.checkstyle.rss.CheckstyleRssGenerator;
import org.apache.maven.plugin.checkstyle.rss.CheckstyleRssGeneratorRequest;
import org.apache.maven.plugin.descriptor.PluginDescriptor;
import org.apache.maven.plugins.annotations.Component;
import org.apache.maven.plugins.annotations.Parameter;
import org.apache.maven.reporting.AbstractMavenReport;
import org.apache.maven.reporting.MavenReportException;
import org.codehaus.plexus.resource.ResourceManager;
import org.codehaus.plexus.resource.loader.FileResourceLoader;
import org.codehaus.plexus.util.FileUtils;
import org.codehaus.plexus.util.PathTool;
import org.codehaus.plexus.util.StringUtils;

import com.puppycrawl.tools.checkstyle.DefaultLogger;
import com.puppycrawl.tools.checkstyle.XMLLogger;
import com.puppycrawl.tools.checkstyle.api.AuditListener;
import com.puppycrawl.tools.checkstyle.api.CheckstyleException;

/**
 * Base abstract class for Checkstyle reports.
 *
 * @version $Id: AbstractCheckstyleReport.java 1663881 2015-03-04 08:10:45Z dennisl $
 */
public abstract class AbstractCheckstyleReport
    extends AbstractMavenReport
{
    public static final String PLUGIN_RESOURCES = "org/apache/maven/plugin/checkstyle";

    protected static final String JAVA_FILES = "**\\/*.java";

    /**
     * Specifies the cache file used to speed up Checkstyle on successive runs.
     */
    @Parameter( defaultValue = "${project.build.directory}/checkstyle-cachefile" )
    protected String cacheFile;

    /**
     * 

* Specifies the location of the XML configuration to use. *

*

*

* Potential values are a filesystem path, a URL, or a classpath resource. * This parameter expects that the contents of the location conform to the * xml format (Checkstyle Checker * module) configuration of rulesets. *

*

*

* This parameter is resolved as resource, URL, then file. If successfully * resolved, the contents of the configuration is copied into the * ${project.build.directory}/checkstyle-configuration.xml * file before being passed to Checkstyle as a configuration. *

*

*

* There are 2 predefined rulesets included in Maven Checkstyle Plugin: *

*
    *
  • config/sun_checks.xml: Sun Checks.
  • *
  • config/maven_checks.xml: Maven Source Checks.
  • *
*/ @Parameter( property = "checkstyle.config.location", defaultValue = "config/sun_checks.xml" ) protected String configLocation; /** * Output errors to console. */ @Parameter( property = "checkstyle.consoleOutput", defaultValue = "false" ) protected boolean consoleOutput; /** * The file encoding to use when reading the source files. If the property project.build.sourceEncoding * is not set, the platform default encoding is used. Note: This parameter always overrides the * property charset from Checkstyle's TreeWalker module. * * @since 2.2 */ @Parameter( property = "encoding", defaultValue = "${project.build.sourceEncoding}" ) protected String encoding; /** * Specifies if the build should fail upon a violation. */ @Parameter( defaultValue = "false" ) protected boolean failsOnError; /** *

* Specifies the location of the License file (a.k.a. the header file) that * can be used by Checkstyle to verify that source code has the correct * license header. *

*

* You need to use ${checkstyle.header.file} in your Checkstyle xml * configuration to reference the name of this header file. *

*

* For instance: *

*

* * <module name="RegexpHeader"> * <property name="headerFile" value="${checkstyle.header.file}"/> * </module> * *

* * @since 2.0-beta-2 */ @Parameter( property = "checkstyle.header.file", defaultValue = "LICENSE.txt" ) protected String headerLocation; /** * Skip entire check. * * @since 2.2 */ @Parameter( property = "checkstyle.skip", defaultValue = "false" ) protected boolean skip; /** * Specifies the path and filename to save the Checkstyle output. The format * of the output file is determined by the outputFileFormat * parameter. */ @Parameter( property = "checkstyle.output.file", defaultValue = "${project.build.directory}/checkstyle-result.xml" ) private File outputFile; /** *

* Specifies the location of the properties file. *

*

*

* This parameter is resolved as URL, File then resource. If successfully * resolved, the contents of the properties location is copied into the * ${project.build.directory}/checkstyle-checker.properties * file before being passed to Checkstyle for loading. *

*

*

* The contents of the propertiesLocation will be made * available to Checkstyle for specifying values for parameters within the * xml configuration (specified in the configLocation * parameter). *

* * @since 2.0-beta-2 */ @Parameter( property = "checkstyle.properties.location" ) protected String propertiesLocation; /** * Allows for specifying raw property expansion information. */ @Parameter protected String propertyExpansion; /** * Specifies the location of the resources to be used for Checkstyle. * * @since 2.10 */ @Parameter( defaultValue = "${project.resources}", readonly = true ) protected List resources; /** * Specifies the location of the test resources to be used for Checkstyle. * * @since 2.11 */ @Parameter( defaultValue = "${project.testResources}", readonly = true ) protected List testResources; /** * Specifies the names filter of the source files to be used for Checkstyle. */ @Parameter( property = "checkstyle.includes", defaultValue = JAVA_FILES, required = true ) protected String includes; /** * Specifies the names filter of the source files to be excluded for * Checkstyle. */ @Parameter( property = "checkstyle.excludes" ) protected String excludes; /** * Specifies the names filter of the resource files to be used for Checkstyle. * @since 2.11 */ @Parameter( property = "checkstyle.resourceIncludes", defaultValue = "**/*.properties", required = true ) protected String resourceIncludes; /** * Specifies the names filter of the resource files to be excluded for * Checkstyle. * @since 2.11 */ @Parameter( property = "checkstyle.resourceExcludes" ) protected String resourceExcludes; /** * Specifies whether to include the resource directories in the check. * @since 2.11 */ @Parameter( property = "checkstyle.includeResources", defaultValue = "true", required = true ) protected boolean includeResources; /** * Specifies whether to include the test resource directories in the check. * @since 2.11 */ @Parameter( property = "checkstyle.includeTestResources", defaultValue = "true", required = true ) protected boolean includeTestResources; /** * Specifies the location of the source directory to be used for Checkstyle. * * @deprecated instead use {@link #sourceDirectories} */ @Deprecated @Parameter private File sourceDirectory; /** * Specifies the location of the source directories to be used for Checkstyle. * @since 2.13 */ @Parameter( defaultValue = "${project.compileSourceRoots}" ) private List sourceDirectories; /** * Specifies the location of the test source directory to be used for * Checkstyle. * * @since 2.2 * @deprecated instead use {@link #testSourceDirectories} */ @Parameter @Deprecated private File testSourceDirectory; /** * Specifies the location of the test source directories to be used for Checkstyle. * @since 2.13 */ @Parameter( defaultValue = "${project.testCompileSourceRoots}" ) private List testSourceDirectories; /** * Include or not the test source directory/directories to be used for Checkstyle. * * @since 2.2 */ @Parameter( defaultValue = "false" ) protected boolean includeTestSourceDirectory; /** * The key to be used in the properties for the suppressions file. * * @since 2.1 */ @Parameter( property = "checkstyle.suppression.expression", defaultValue = "checkstyle.suppressions.file" ) protected String suppressionsFileExpression; /** *

* Specifies the location of the suppressions XML file to use. *

*

*

* This parameter is resolved as resource, URL, then file. If successfully * resolved, the contents of the suppressions XML is copied into the * ${project.build.directory}/checkstyle-supressions.xml file * before being passed to Checkstyle for loading. *

*

*

* See suppressionsFileExpression for the property that will * be made available to your Checkstyle configuration. *

* * @since 2.0-beta-2 */ @Parameter( property = "checkstyle.suppressions.location" ) protected String suppressionsLocation; /** * If null, the Checkstyle plugin will display violations on stdout. * Otherwise, a text file will be created with the violations. */ @Parameter private File useFile; /** * Specifies the format of the output to be used when writing to the output * file. Valid values are "plain" and "xml". */ @Parameter( property = "checkstyle.output.format", defaultValue = "xml" ) private String outputFileFormat; /** * Specifies if the Rules summary should be enabled or not. */ @Parameter( property = "checkstyle.enable.rules.summary", defaultValue = "true" ) private boolean enableRulesSummary; /** * Specifies if the Severity summary should be enabled or not. */ @Parameter( property = "checkstyle.enable.severity.summary", defaultValue = "true" ) private boolean enableSeveritySummary; /** * Specifies if the Files summary should be enabled or not. */ @Parameter( property = "checkstyle.enable.files.summary", defaultValue = "true" ) private boolean enableFilesSummary; /** * Specifies if the RSS should be enabled or not. */ @Parameter( property = "checkstyle.enable.rss", defaultValue = "true" ) private boolean enableRSS; /** * SiteTool. * * @since 2.2 */ @Component( role = SiteTool.class ) protected SiteTool siteTool; /** * The Plugin Descriptor */ @Parameter( defaultValue = "${plugin}", readonly = true, required = true ) private PluginDescriptor plugin; // remove when requiring Maven 3.x, just use #plugin @Parameter( defaultValue = "${mojoExecution}", readonly = true, required = true ) private MojoExecution mojoExecution; /** * Link the violation line numbers to the source xref. Will link * automatically if Maven JXR plugin is being used. * * @since 2.1 */ @Parameter( property = "linkXRef", defaultValue = "true" ) private boolean linkXRef; /** * Location of the Xrefs to link to. */ @Parameter( defaultValue = "${project.reporting.outputDirectory}/xref" ) private File xrefLocation; /** * When using custom treeWalkers, specify their names here so the checks * inside the treeWalker end up the the rule-summary. * * @since 2.11 */ @Parameter private List treeWalkerNames; /** */ @Component protected ResourceManager locator; /** * CheckstyleRssGenerator. * * @since 2.4 */ @Component( role = CheckstyleRssGenerator.class, hint = "default" ) protected CheckstyleRssGenerator checkstyleRssGenerator; /** * @since 2.5 */ @Component( role = CheckstyleExecutor.class, hint = "default" ) protected CheckstyleExecutor checkstyleExecutor; protected ByteArrayOutputStream stringOutputStream; /** {@inheritDoc} */ public String getName( Locale locale ) { return getBundle( locale ).getString( "report.checkstyle.name" ); } /** {@inheritDoc} */ public String getDescription( Locale locale ) { return getBundle( locale ).getString( "report.checkstyle.description" ); } /** {@inheritDoc} */ public void executeReport( Locale locale ) throws MavenReportException { locator.addSearchPath( FileResourceLoader.ID, project.getFile().getParentFile().getAbsolutePath() ); locator.addSearchPath( "url", "" ); locator.setOutputDirectory( new File( project.getBuild().getDirectory() ) ); // for when we start using maven-shared-io and maven-shared-monitor... // locator = new Locator( new MojoLogMonitorAdaptor( getLog() ) ); // locator = new Locator( getLog(), new File( project.getBuild().getDirectory() ) ); ClassLoader currentClassLoader = Thread.currentThread().getContextClassLoader(); try { CheckstyleExecutorRequest request = createRequest().setLicenseArtifacts( collectArtifacts( "license" ) ) .setConfigurationArtifacts( collectArtifacts( "configuration" ) ); CheckstyleResults results = checkstyleExecutor.executeCheckstyle( request ); ResourceBundle bundle = getBundle( locale ); generateReportStatics(); generateMainReport( results, bundle ); if ( enableRSS ) { CheckstyleRssGeneratorRequest checkstyleRssGeneratorRequest = new CheckstyleRssGeneratorRequest( this.project, this.getCopyright(), outputDirectory, getLog() ); checkstyleRssGenerator.generateRSS( results, checkstyleRssGeneratorRequest ); } } catch ( CheckstyleException e ) { throw new MavenReportException( "Failed during checkstyle configuration", e ); } catch ( CheckstyleExecutorException e ) { throw new MavenReportException( "Failed during checkstyle execution", e ); } finally { //be sure to restore original context classloader Thread.currentThread().setContextClassLoader( currentClassLoader ); } } /** * Create the Checkstyle executor request. * * @return The executor request. * @throws MavenReportException If something goes wrong during creation. */ protected abstract CheckstyleExecutorRequest createRequest() throws MavenReportException; @SuppressWarnings( "unchecked" ) private List collectArtifacts( String hint ) { if ( plugin == null || plugin.getGroupId() == null ) { // Maven 2.x workaround plugin = mojoExecution.getMojoDescriptor().getPluginDescriptor(); } List artifacts = new ArrayList(); PluginManagement pluginManagement = project.getBuild().getPluginManagement(); if ( pluginManagement != null ) { artifacts.addAll( getCheckstylePluginDependenciesAsArtifacts( pluginManagement.getPluginsAsMap(), hint ) ); } artifacts.addAll( getCheckstylePluginDependenciesAsArtifacts( project.getBuild().getPluginsAsMap(), hint ) ); return artifacts; } private List getCheckstylePluginDependenciesAsArtifacts( Map plugins, String hint ) { List artifacts = new ArrayList(); Plugin checkstylePlugin = plugins.get( plugin.getGroupId() + ":" + plugin.getArtifactId() ); if ( checkstylePlugin != null ) { for ( Dependency dep : checkstylePlugin.getDependencies() ) { // @todo if we can filter on hints, it should be done here... String depKey = dep.getGroupId() + ":" + dep.getArtifactId(); artifacts.add( (Artifact) plugin.getArtifactMap().get( depKey ) ); } } return artifacts; } /** * Creates and returns the report generation listener. * * @return The audit listener. * @throws MavenReportException If something goes wrong. */ protected AuditListener getListener() throws MavenReportException { AuditListener listener = null; if ( StringUtils.isNotEmpty( outputFileFormat ) ) { File resultFile = outputFile; OutputStream out = getOutputStream( resultFile ); if ( "xml".equals( outputFileFormat ) ) { listener = new XMLLogger( out, true ); } else if ( "plain".equals( outputFileFormat ) ) { listener = new DefaultLogger( out, true ); } else { // TODO: failure if not a report throw new MavenReportException( "Invalid output file format: (" + outputFileFormat + "). Must be 'plain' or 'xml'." ); } } return listener; } private OutputStream getOutputStream( File file ) throws MavenReportException { File parentFile = file.getAbsoluteFile().getParentFile(); if ( !parentFile.exists() ) { parentFile.mkdirs(); } FileOutputStream fileOutputStream; try { fileOutputStream = new FileOutputStream( file ); } catch ( FileNotFoundException e ) { throw new MavenReportException( "Unable to create output stream: " + file, e ); } return fileOutputStream; } /** * Creates and returns the console listener. * * @return The console listener. * @throws MavenReportException If something goes wrong. */ protected DefaultLogger getConsoleListener() throws MavenReportException { DefaultLogger consoleListener; if ( useFile == null ) { stringOutputStream = new ByteArrayOutputStream(); consoleListener = new DefaultLogger( stringOutputStream, false ); } else { OutputStream out = getOutputStream( useFile ); consoleListener = new DefaultLogger( out, true ); } return consoleListener; } private void generateReportStatics() throws MavenReportException { ReportResource rresource = new ReportResource( PLUGIN_RESOURCES, outputDirectory ); try { rresource.copy( "images/rss.png" ); } catch ( IOException e ) { throw new MavenReportException( "Unable to copy static resources.", e ); } } private String getCopyright() { String copyright; int currentYear = Calendar.getInstance().get( Calendar.YEAR ); if ( StringUtils.isNotEmpty( project.getInceptionYear() ) && !String.valueOf( currentYear ).equals( project.getInceptionYear() ) ) { copyright = project.getInceptionYear() + " - " + currentYear; } else { copyright = String.valueOf( currentYear ); } if ( ( project.getOrganization() != null ) && StringUtils.isNotEmpty( project.getOrganization().getName() ) ) { copyright = copyright + " " + project.getOrganization().getName(); } return copyright; } private void generateMainReport( CheckstyleResults results, ResourceBundle bundle ) { CheckstyleReportGenerator generator = new CheckstyleReportGenerator( getSink(), bundle, project.getBasedir(), siteTool, configLocation ); generator.setLog( getLog() ); generator.setEnableRulesSummary( enableRulesSummary ); generator.setEnableSeveritySummary( enableSeveritySummary ); generator.setEnableFilesSummary( enableFilesSummary ); generator.setEnableRSS( enableRSS ); generator.setCheckstyleConfig( results.getConfiguration() ); if ( linkXRef ) { String relativePath = PathTool.getRelativePath( getOutputDirectory(), xrefLocation.getAbsolutePath() ); if ( StringUtils.isEmpty( relativePath ) ) { relativePath = "."; } relativePath = relativePath + "/" + xrefLocation.getName(); if ( xrefLocation.exists() ) { // XRef was already generated by manual execution of a lifecycle // binding generator.setXrefLocation( relativePath ); } else { // Not yet generated - check if the report is on its way for ( ReportPlugin report : (Iterable) getProject().getReportPlugins() ) { String artifactId = report.getArtifactId(); if ( "maven-jxr-plugin".equals( artifactId ) || "jxr-maven-plugin".equals( artifactId ) ) { generator.setXrefLocation( relativePath ); } } } if ( generator.getXrefLocation() == null && results.getFileCount() > 0 ) { getLog().warn( "Unable to locate Source XRef to link to - DISABLED" ); } } if ( treeWalkerNames != null ) { generator.setTreeWalkerNames( treeWalkerNames ); } generator.generateReport( results ); } private static ResourceBundle getBundle( Locale locale ) { return ResourceBundle.getBundle( "checkstyle-report", locale, AbstractCheckstyleReport.class.getClassLoader() ); } protected List getSourceDirectories() { List sourceDirs = null; // if sourceDirectory is explicitly set, use it if ( sourceDirectory != null ) { sourceDirs = Collections.singletonList( sourceDirectory ); } else { sourceDirs = new ArrayList( sourceDirectories.size() ); for ( String sourceDir : sourceDirectories ) { sourceDirs.add( FileUtils.resolveFile( project.getBasedir(), sourceDir ) ); } } return sourceDirs; } protected List getTestSourceDirectories() { List testSourceDirs = null; // if testSourceDirectory is explicitly set, use it if ( testSourceDirectory != null ) { testSourceDirs = Collections.singletonList( testSourceDirectory ); } // probably null-check only required due to MavenProjectStubs else if ( testSourceDirectories != null ) { testSourceDirs = new ArrayList( testSourceDirectories.size() ); for ( String testSourceDir : testSourceDirectories ) { testSourceDirs.add( FileUtils.resolveFile( project.getBasedir(), testSourceDir ) ); } } return testSourceDirs; } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy