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

org.apache.maven.plugin.compiler.CompilerMojo Maven / Gradle / Ivy

There is a newer version: 4.0.0-beta-1
Show newest version
package org.apache.maven.plugin.compiler;

/*
 * 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.Collection;
import java.util.Collections;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;

import org.apache.maven.artifact.Artifact;
import org.apache.maven.plugin.MojoExecutionException;
import org.apache.maven.plugin.compiler.module.JavaModuleDescriptor;
import org.apache.maven.plugin.compiler.module.ModuleInfoParser;
import org.apache.maven.plugin.compiler.module.ProjectAnalyzer;
import org.apache.maven.plugin.compiler.module.ProjectAnalyzerRequest;
import org.apache.maven.plugin.compiler.module.ProjectAnalyzerResult;
import org.apache.maven.plugins.annotations.Component;
import org.apache.maven.plugins.annotations.LifecyclePhase;
import org.apache.maven.plugins.annotations.Mojo;
import org.apache.maven.plugins.annotations.Parameter;
import org.apache.maven.plugins.annotations.ResolutionScope;
import org.apache.maven.project.MavenProject;
import org.apache.maven.shared.utils.StringUtils;
import org.apache.maven.shared.utils.logging.MessageUtils;
import org.codehaus.plexus.compiler.util.scan.SimpleSourceInclusionScanner;
import org.codehaus.plexus.compiler.util.scan.SourceInclusionScanner;
import org.codehaus.plexus.compiler.util.scan.StaleSourceScanner;

/**
 * Compiles application sources
 *
 * @author Jason van Zyl 
 * @version $Id: CompilerMojo.java 1795232 2017-05-15 19:03:52Z rfscholte $
 * @since 2.0
 */
@Mojo( name = "compile", defaultPhase = LifecyclePhase.COMPILE, threadSafe = true, 
    requiresDependencyResolution = ResolutionScope.COMPILE )
public class CompilerMojo
    extends AbstractCompilerMojo
{
    /**
     * The source directories containing the sources to be compiled.
     */
    @Parameter( defaultValue = "${project.compileSourceRoots}", readonly = true, required = true )
    private List compileSourceRoots;

    /**
     * The directory for compiled classes.
     */
    @Parameter( defaultValue = "${project.build.outputDirectory}", required = true, readonly = true )
    private File outputDirectory;

    /**
     * Projects main artifact.
     *
     * @todo this is an export variable, really
     */
    @Parameter( defaultValue = "${project.artifact}", readonly = true, required = true )
    private Artifact projectArtifact;

    /**
     * A list of inclusion filters for the compiler.
     */
    @Parameter
    private Set includes = new HashSet();

    /**
     * A list of exclusion filters for the compiler.
     */
    @Parameter
    private Set excludes = new HashSet();

    /**
     * 

* Specify where to place generated source files created by annotation processing. Only applies to JDK 1.6+ *

* * @since 2.2 */ @Parameter( defaultValue = "${project.build.directory}/generated-sources/annotations" ) private File generatedSourcesDirectory; /** * Set this to 'true' to bypass compilation of main sources. Its use is NOT RECOMMENDED, but quite convenient on * occasion. */ @Parameter( property = "maven.main.skip" ) private boolean skipMain; @Parameter( defaultValue = "${project.compileClasspathElements}", readonly = true, required = true ) private List compilePath; @Parameter private boolean allowPartialRequirements; @Component( hint = "qdox" ) private ModuleInfoParser moduleInfoParser; @Component private ProjectAnalyzer projectAnalyzer; private List classpathElements; private List modulepathElements; protected List getCompileSourceRoots() { return compileSourceRoots; } protected List getClasspathElements() { return classpathElements; } @Override protected List getModulepathElements() { return modulepathElements; } protected File getOutputDirectory() { return outputDirectory; } public void execute() throws MojoExecutionException, CompilationFailureException { if ( skipMain ) { getLog().info( "Not compiling main sources" ); return; } super.execute(); if ( outputDirectory.isDirectory() ) { projectArtifact.setFile( outputDirectory ); } } @Override protected void preparePaths( Set sourceFiles ) { assert compilePath != null; JavaModuleDescriptor moduleDescriptor = null; boolean hasModuleDescriptor = false; for ( File sourceFile : sourceFiles ) { if ( "module-info.java".equals( sourceFile.getName() ) ) { try { moduleDescriptor = moduleInfoParser.getModuleDescriptor( sourceFile.getParentFile() ); } catch ( IOException e ) { getLog().warn( "Failed to parse module-info.java: " + e.getMessage() ); } hasModuleDescriptor = true; break; } } if ( hasModuleDescriptor ) { // For now only allow named modules. Once we can create a graph with ASM we can specify exactly the modules // and we can detect if auto modules are used. In that case, MavenProject.setFile() should not be used, so // you cannot depend on this project and so it won't be distributed. modulepathElements = new ArrayList( compilePath.size() ); classpathElements = new ArrayList( compilePath.size() ); ProjectAnalyzerResult analyzerResult; try { Collection dependencyArtifacts = getCompileClasspathElements( getProject() ); ProjectAnalyzerRequest analyzerRequest = new ProjectAnalyzerRequest() .setBaseModuleDescriptor( moduleDescriptor ) .setDependencyArtifacts( dependencyArtifacts ); analyzerResult = projectAnalyzer.analyze( analyzerRequest ); if ( !analyzerResult.getRequiredAutomaticModules().isEmpty() ) { boolean filenameBased = false; for ( String automodule : analyzerResult.getRequiredAutomaticModules() ) { filenameBased = ProjectAnalyzerResult.ModuleNameSource.FILENAME.equals( analyzerResult.getModuleNameSource( automodule ) ); if ( filenameBased ) { final String message = "Required automodules detected. " + "Please don't publish this project to a public artifact repository!"; if ( moduleDescriptor.exports().isEmpty() ) { // application getLog().info( message ); } else { // library writeBoxedWarning( message ); } break; } } } for ( Map.Entry entry : analyzerResult.getPathElements().entrySet() ) { if ( !allowPartialRequirements ) { modulepathElements.add( entry.getKey().getPath() ); } else if ( entry.getValue() == null ) { classpathElements.add( entry.getKey().getPath() ); } else if ( analyzerResult.getRequiredNormalModules().contains( entry.getValue().name() ) ) { modulepathElements.add( entry.getKey().getPath() ); } else if ( analyzerResult.getRequiredAutomaticModules().contains( entry.getValue().name() ) ) { modulepathElements.add( entry.getKey().getPath() ); } else { classpathElements.add( entry.getKey().getPath() ); } } } catch ( IOException e ) { getLog().warn( e.getMessage() ); } // if ( !classpathElements.isEmpty() ) // { // if ( compilerArgs == null ) // { // compilerArgs = new ArrayList(); // } // compilerArgs.add( "--add-reads" ); // compilerArgs.add( moduleDescriptor.name() + "=ALL-UNNAMED" ); // // if ( !modulepathElements.isEmpty() ) // { // compilerArgs.add( "--add-reads" ); // compilerArgs.add( "ALL-MODULE-PATH=ALL-UNNAMED" ); // } // } } else { classpathElements = compilePath; modulepathElements = Collections.emptyList(); } } private List getCompileClasspathElements( MavenProject project ) { List list = new ArrayList( project.getArtifacts().size() + 1 ); list.add( new File( project.getBuild().getOutputDirectory() ) ); for ( Artifact a : project.getArtifacts() ) { list.add( a.getFile() ); } return list; } protected SourceInclusionScanner getSourceInclusionScanner( int staleMillis ) { SourceInclusionScanner scanner; if ( includes.isEmpty() && excludes.isEmpty() ) { scanner = new StaleSourceScanner( staleMillis ); } else { if ( includes.isEmpty() ) { includes.add( "**/*.java" ); } scanner = new StaleSourceScanner( staleMillis, includes, excludes ); } return scanner; } protected SourceInclusionScanner getSourceInclusionScanner( String inputFileEnding ) { SourceInclusionScanner scanner; // it's not defined if we get the ending with or without the dot '.' String defaultIncludePattern = "**/*" + ( inputFileEnding.startsWith( "." ) ? "" : "." ) + inputFileEnding; if ( includes.isEmpty() && excludes.isEmpty() ) { includes = Collections.singleton( defaultIncludePattern ); scanner = new SimpleSourceInclusionScanner( includes, Collections.emptySet() ); } else { if ( includes.isEmpty() ) { includes.add( defaultIncludePattern ); } scanner = new SimpleSourceInclusionScanner( includes, excludes ); } return scanner; } protected String getSource() { return source; } protected String getTarget() { return target; } @Override protected String getRelease() { return release; } protected String getCompilerArgument() { return compilerArgument; } protected Map getCompilerArguments() { return compilerArguments; } protected File getGeneratedSourcesDirectory() { return generatedSourcesDirectory; } private void writeBoxedWarning( String message ) { String line = StringUtils.repeat( "*", message.length() + 4 ); getLog().warn( line ); getLog().warn( "* " + MessageUtils.buffer().strong( message ) + " *" ); getLog().warn( line ); } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy