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

org.apache.maven.plugin.eclipse.EclipsePlugin Maven / Gradle / Ivy

package org.apache.maven.plugin.eclipse;

/*
 * 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.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.URL;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.LinkedHashSet;
import java.util.LinkedList;
import java.util.List;
import java.util.ListIterator;
import java.util.Map;
import java.util.Set;

import org.apache.maven.artifact.Artifact;
import org.apache.maven.artifact.handler.ArtifactHandler;
import org.apache.maven.artifact.manager.WagonManager;
import org.apache.maven.model.Build;
import org.apache.maven.model.Plugin;
import org.apache.maven.model.Resource;
import org.apache.maven.plugin.MojoExecutionException;
import org.apache.maven.plugin.eclipse.reader.ReadWorkspaceLocations;
import org.apache.maven.plugin.eclipse.writers.EclipseClasspathWriter;
import org.apache.maven.plugin.eclipse.writers.EclipseProjectWriter;
import org.apache.maven.plugin.eclipse.writers.EclipseWriterConfig;
import org.apache.maven.plugin.eclipse.writers.workspace.EclipseSettingsWriter;
import org.apache.maven.plugin.eclipse.writers.wtp.EclipseWtpApplicationXMLWriter;
import org.apache.maven.plugin.eclipse.writers.wtp.EclipseWtpComponent15Writer;
import org.apache.maven.plugin.eclipse.writers.wtp.EclipseWtpComponentWriter;
import org.apache.maven.plugin.eclipse.writers.wtp.EclipseWtpFacetsWriter;
import org.apache.maven.plugin.eclipse.writers.wtp.EclipseWtpmodulesWriter;
import org.apache.maven.plugin.ide.AbstractIdeSupportMojo;
import org.apache.maven.plugin.ide.IdeDependency;
import org.apache.maven.plugin.ide.IdeUtils;
import org.apache.maven.plugin.ide.JeeUtils;
import org.apache.maven.plugins.annotations.Component;
import org.apache.maven.plugins.annotations.Execute;
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.project.MavenProject;
import org.apache.maven.settings.MavenSettingsBuilder;
import org.apache.maven.settings.Proxy;
import org.apache.maven.settings.Settings;
import org.apache.maven.wagon.Wagon;
import org.apache.maven.wagon.WagonException;
import org.apache.maven.wagon.observers.Debug;
import org.apache.maven.wagon.proxy.ProxyInfo;
import org.apache.maven.wagon.repository.Repository;
import org.codehaus.plexus.resource.ResourceManager;
import org.codehaus.plexus.resource.loader.FileResourceLoader;
import org.codehaus.plexus.resource.loader.ResourceNotFoundException;
import org.codehaus.plexus.util.FileUtils;
import org.codehaus.plexus.util.IOUtil;
import org.codehaus.plexus.util.StringUtils;
import org.codehaus.plexus.util.xml.Xpp3Dom;
import org.codehaus.plexus.util.xml.pull.XmlPullParserException;

/**
 * Generates the following eclipse configuration files:
 * 
    *
  • .project and .classpath files
  • *
  • .setting/org.eclipse.jdt.core.prefs with project specific compiler settings
  • *
  • various configuration files for WTP (Web Tools Project), if the parameter wtpversion is set to a * valid version (WTP configuration is not generated by default)
  • *
* If this goal is run on a multiproject root, dependencies between modules will be configured as direct project * dependencies in Eclipse (unless useProjectReferences is set to false). * * @author Trygve Laugstøl * @author Fabrizio Giustina * @version $Id$ */ @Mojo( name = "eclipse" ) @Execute( phase = LifecyclePhase.GENERATE_RESOURCES ) public class EclipsePlugin extends AbstractIdeSupportMojo { private static final String NATURE_WST_FACET_CORE_NATURE = "org.eclipse.wst.common.project.facet.core.nature"; private static final String BUILDER_WST_COMPONENT_STRUCTURAL_DEPENDENCY_RESOLVER = "org.eclipse.wst.common.modulecore.ComponentStructuralBuilderDependencyResolver"; protected static final String BUILDER_WST_VALIDATION = "org.eclipse.wst.validation.validationbuilder"; private static final String BUILDER_JDT_CORE_JAVA = "org.eclipse.jdt.core.javabuilder"; private static final String BUILDER_WST_COMPONENT_STRUCTURAL = "org.eclipse.wst.common.modulecore.ComponentStructuralBuilder"; private static final String BUILDER_WST_FACET = "org.eclipse.wst.common.project.facet.core.builder"; private static final String NATURE_WST_MODULE_CORE_NATURE = "org.eclipse.wst.common.modulecore.ModuleCoreNature"; private static final String NATURE_JDT_CORE_JAVA = "org.eclipse.jdt.core.javanature"; private static final String NATURE_JEM_WORKBENCH_JAVA_EMF = "org.eclipse.jem.workbench.JavaEMFNature"; protected static final String COMMON_PATH_JDT_LAUNCHING_JRE_CONTAINER = "org.eclipse.jdt.launching.JRE_CONTAINER"; // warning, order is important for binary search public static final String[] WTP_SUPPORTED_VERSIONS = new String[] { "1.0", "1.5", "2.0", "R7", "none" }; public static final String JAVA_FILE_PATTERN = "**/*.java"; /** * Constant for 'artifactId' element in POM.xml. */ private static final String POM_ELT_ARTIFACT_ID = "artifactId"; /** * Constant for 'groupId' element in POM.xml. */ private static final String POM_ELT_GROUP_ID = "groupId"; /** * List of eclipse project natures. By default the org.eclipse.jdt.core.javanature nature plus the * needed WTP natures are added. Natures added using this property replace the default list. * *
     * <projectnatures>
     *    <projectnature>org.eclipse.jdt.core.javanature</projectnature>
     *    <projectnature>org.eclipse.wst.common.modulecore.ModuleCoreNature</projectnature>
     * </projectnatures>
     * 
*/ @Parameter private List projectnatures; /** * List of artifacts, represented as groupId:artifactId, to exclude from the eclipse classpath, being * provided by some eclipse classPathContainer. * * @see http://jira.codehaus.org/browse/MECLIPSE-79 * @since 2.5 */ @Parameter private List excludes; /** * List of eclipse project natures to be added to the default ones. * *
     * <additionalProjectnatures>
     *    <projectnature>org.springframework.ide.eclipse.core.springnature</projectnature>
     * </additionalProjectnatures>
     * 
*/ @Parameter private List additionalProjectnatures; /** * List of eclipse project facets to be added to the default ones. * *
     * <additionalProjectFacets>
     *    <jst.jsf>1.1<jst.jsf/>
     * </additionalProjectFacets>
     * 
*/ @Parameter private Map additionalProjectFacets; // CHECKSTYLE_OFF: LineLength /** * List of eclipse build commands. By default the org.eclipse.jdt.core.javabuilder builder plus the * needed WTP builders are added. If you specify any configuration for this parameter, only those buildcommands * specified will be used; the defaults won't be added. Use the additionalBuildCommands parameter for * that. Configuration example: Old style: * *
     * <buildcommands>
     *    <buildcommand>org.eclipse.wst.common.modulecore.ComponentStructuralBuilder</buildcommand>
     *    <buildcommand>org.eclipse.jdt.core.javabuilder</buildcommand>
     *    <buildcommand>org.eclipse.wst.common.modulecore.ComponentStructuralBuilderDependencyResolver</buildcommand>
     * </buildcommands>
     * 
* * For new style, see additionalBuildCommands. */ // CHECKSTYLE_ON: LineLength @Parameter private List buildcommands; // CHECKSTYLE_OFF: LineLength /** * List of eclipse build commands to be added to the default ones. Old style: * *
     * <additionalBuildcommands>
     *    <buildcommand>org.springframework.ide.eclipse.core.springbuilder</buildcommand>
     * </additionalBuildcommands>
     * 
* * New style: * *
     * <additionalBuildcommands>
     *    <buildCommand>
     *      <name>org.eclipse.ui.externaltools.ExternalToolBuilder</name>
     *      <triggers>auto,full,incremental,</triggers>
     *      <arguments>
     *        <LaunchConfigHandle>&lt;project&gt;./externalToolBuilders/MavenBuilder.launch</LaunchConfighandle>
     *      </arguments>
     *    </buildCommand>
     * </additionalBuildcommands>
     * 
* * Note the difference between buildcommand and * buildCommand. You can mix and match old and new-style configuration entries. */ // CHECKSTYLE_ON: LineLength @Parameter private List additionalBuildcommands; /** * List of source folders to be added to the default ones. * *
     * <additionalSourceFolders>
     *    <additionalSourceFolder>target/generated-sources</additionalSourceFolder>
     * </additionalSourceFolders>
     * 
*/ @Parameter private List additionalSourceFolders; /** * List of optional source folders to be added to the default ones. * *
     * <additionalOptionalSourceFolders>
     *    <additionalOptionalSourceFolder>target/generated-sources</additionalOptionalSourceFolder>
     * </additionalOptionalSourceFolders>
     * 
*/ @Parameter private List additionalOptionalSourceFolders; /** * List of test source folders to be added to the default ones. * *
     * <additionalTestSourceFolders>
     *    <additionalTestSourceFolder>target/generated-test-sources</additionalTestSourceFolder>
     * </additionalTestSourceFolders>
     * 
*/ @Parameter private List additionalTestSourceFolders; /** * List of optional test source folders to be added to the default ones. * *
     * <additionalOptionalTestSourceFolders>
     *    <additionalOptionalTestSourceFolder>target/generated-test-sources</additionalOptionalTestSourceFolder>
     * </additionalOptionalTestSourceFolders>
     * 
*/ @Parameter private List additionalOptionalTestSourceFolders; // CHECKSTYLE_OFF: LineLength /** * List of container classpath entries. By default the org.eclipse.jdt.launching.JRE_CONTAINER * classpath container is added. Configuration example: * *
     * <classpathContainers>
     *    <classpathContainer>org.eclipse.jdt.launching.JRE_CONTAINER</classpathContainer>
     *    <classpathContainer>org.eclipse.jst.server.core.container/org.eclipse.jst.server.tomcat.runtimeTarget/Apache Tomcat v5.5</classpathContainer>
     *    <classpathContainer>org.eclipse.jst.j2ee.internal.web.container/artifact</classpathContainer>
     * </classpathContainers>
     * 
*/ // CHECKSTYLE_ON: LineLength @Parameter private List classpathContainers; /** * Enables/disables the downloading of source attachments. Defaults to false. DEPRECATED - use downloadSources * * @deprecated use downloadSources */ @Deprecated @Parameter( property = "eclipse.downloadSources" ) private boolean eclipseDownloadSources; /** * Eclipse workspace directory. */ @Parameter( property = "eclipse.projectDir", alias = "outputDir" ) private File eclipseProjectDir; /** * When set to false, the plugin will not create sub-projects and instead reference those sub-projects using the * installed package in the local repository */ @Parameter( property = "eclipse.useProjectReferences", defaultValue = "true", required = true ) private boolean useProjectReferences; /** * The default output directory */ @Parameter( property = "outputDirectory", alias = "outputDirectory", defaultValue = "${project.build.outputDirectory}", required = true ) private File buildOutputDirectory; /** * The version of WTP for which configuration files will be generated. The default value is "none" (don't generate * WTP configuration), supported versions are "R7", "1.0", "1.5" and "2.0" */ @Parameter( property = "wtpversion", defaultValue = "none" ) private String wtpversion; /** * JEE context name of the WTP module. ( ex. WEB context name ). You can use "ROOT" if you want to map the webapp to * the root context. */ @Parameter( property = "wtpContextName" ) private String wtpContextName; /** * The relative path of the manifest file */ @Parameter( property = "eclipse.manifest", defaultValue = "${basedir}/META-INF/MANIFEST.MF" ) private File manifest; // CHECKSTYLE_OFF: LineLength /** * Allow to configure additional generic configuration files for eclipse that will be written out to disk when * running eclipse:eclipse. FOr each file you can specify the name and the text content. * *
     * <plugin>
     *  <groupId>com.github.marschall</groupId>
     *  <artifactId>eclipse-maven-plugin</artifactId>
     *  <configuration>
     *   <additionalConfig>
     *    <file>
     *      <name>.checkstyle</name>
     *      <content>
     *        <![CDATA[<fileset-config file-format-version="1.2.0" simple-config="true">
     *          <fileset name="all" enabled="true" check-config-name="acme corporate style" local="false">
     *              <file-match-pattern match-pattern="." include-pattern="true"/>
     *          </fileset>
     *          <filter name="NonSrcDirs" enabled="true"/>
     *        </fileset-config>]]>
     *      </content>
     *    </file>
     *   </additionalConfig>
     *  </configuration>
     * </plugin>
     * 
* * Instead of the content you can also define (from version 2.5) an url to download the file : * *
     * <plugin>
     *  <groupId>com.github.marschall</groupId>
     *  <artifactId>eclipse-maven-plugin</artifactId>
     *  <configuration>
     *   <additionalConfig>
     *    <file>
     *      <name>.checkstyle</name>
     *      <url>http://some.place.org/path/to/file</url>
     *    </file>
     *   </additionalConfig>
     *  </configuration>
     * 
* * or a location : * *
     * <plugin>
     *  <groupId>com.github.marschall</groupId>
     *  <artifactId>eclipse-maven-plugin</artifactId>
     *  <configuration>
     *   <additionalConfig>
     *    <file>
     *     <name>.checkstyle</name>
     *     <location>/checkstyle-config.xml</location>
     *    </file>
     *   </additionalConfig>
     *  </configuration>
     *  <dependencies>
     *   <!-- The file defined in the location is stored in this dependency -->
     *   <dependency>
     *    <groupId>eclipsetest</groupId>
     *    <artifactId>checkstyle-config</artifactId>
     *    <version>1.0</version>
     *   </dependency>
     *  </dependencies>
     * </plugin>
     * 
*/ // CHECKSTYLE_ON: LineLength @Parameter private EclipseConfigFile[] additionalConfig; /** * If set to true, the version number of the artifact is appended to the name of the generated Eclipse * project. See projectNameTemplate for other options. */ @Parameter( property = "eclipse.addVersionToProjectName", defaultValue = "false" ) private boolean addVersionToProjectName; /** * If set to true, the groupId of the artifact is appended to the name of the generated Eclipse * project. See projectNameTemplate for other options. */ @Parameter( property = "eclipse.addGroupIdToProjectName", defaultValue = "false" ) private boolean addGroupIdToProjectName; /** * Allows configuring the name of the eclipse projects. This property if set wins over addVersionToProjectName and * addGroupIdToProjectName You can use [groupId], [artifactId] and [version] * variables. eg. [groupId].[artifactId]-[version] */ @Parameter( property = "eclipse.projectNameTemplate" ) private String projectNameTemplate; /** * Parsed wtp version. */ private float wtpVersionFloat; /** * Not a plugin parameter. Is this a java project? */ private boolean isJavaProject; /** * Must the application files be written for ear projects in a separate directory. */ @Parameter( property = "eclipse.wtpapplicationxml", defaultValue = "false" ) private boolean wtpapplicationxml; /** * What WTP defined server to use for deployment informations. */ @Parameter( property = "eclipse.wtpdefaultserver" ) private String wtpdefaultserver; /** * If set to true, the standard execution environment matching the compiler settings is set as JRE. If * set to false, the JRE matching the configured compiler-plugin executable or JAVA_HOME is selected * by name, if it is configured in the workspace. * * @since 2.10 */ @Parameter( property = "eclipse.preferStandardClasspathContainer", defaultValue = "false" ) private boolean preferStandardClasspathContainer; private WorkspaceConfiguration workspaceConfiguration; /** * ResourceManager for getting additonalConfig files from resources */ @Component private ResourceManager locator; /** * WagonManager for accessing internet resources. */ @Component private WagonManager wagonManager; /** * MavenSettingsBuilder for accessing settings.xml. */ @Component private MavenSettingsBuilder mavenSettingsBuilder; /** * This eclipse workspace is read and all artifacts detected there will be connected as eclipse projects and will * not be linked to the jars in the local repository. Requirement is that it was created with the similar wtp * settings as the reactor projects, but the project name template my differ. The pom's in the workspace projects * may not contain variables in the artefactId, groupId and version tags. If workspace is not defined, then an * attempt to locate it by checking up the directory hierarchy will be made. * * @since 2.5 */ @Parameter( property = "eclipse.workspace" ) protected File workspace; /** * Limit the use of project references to the current workspace. No project references will be created to projects * in the reactor when they are not available in the workspace. */ @Parameter( property = "eclipse.limitProjectReferencesToWorkspace", defaultValue = "false" ) protected boolean limitProjectReferencesToWorkspace; /** * List of exclusions to add to the source directories on the classpath. Adds excluding="" to the classpathentry of * the eclipse .classpath file. [MECLIPSE-104] * * @since 2.6.1 */ @Parameter private List sourceExcludes; /** * List of inclusions to add to the source directories on the classpath. Adds including="" to the classpathentry of * the eclipse .classpath file. *

* Java projects will always include "**/*.java" *

* [MECLIPSE-104] * * @since 2.6.1 */ @Parameter private List sourceIncludes; /** * A list of links to local files in the system. A configuration like this one in the pom : * *

     * <plugin>
     *   <groupId>com.github.marschall</groupId>
     *   <artifactId>eclipse-maven-plugin</artifactId>
     *   <configuration> 
     *     <linkedResources> 
     *       <linkedResource>
     *         <name>src/test/resources/oracle-ds.xml</name>
     *         <type>1</type>
     *         <location>C://jboss/server/default/deploy/oracle-ds.xml</location> 
     *       </linkedResource>
     *     </linkedResources> 
     *   </configuration> 
     * </plugin>
     * 
* * will produce in the .project : * *
     * <linkedResources>
     *   <link>
     *     <name>src/test/resources/oracle-ds.xml</name>
     *     <type>1</type> 
     *     <location>C://jboss/server/default/deploy/oracle-ds.xml</location>
     *   </link>
     * </linkedResources>
     * 
* * @since 2.8 */ @Parameter private List linkedResources; /** * Put classpath container entries last in eclipse classpath configuration. Note that this behaviour, although * useful in situations were you want to override resources found in classpath containers, will made JRE classes * loaded after 3rd party jars, so enabling it is not suggested. * * @since 2.9 */ @Parameter( property = "eclipse.classpathContainersLast", defaultValue = "false" ) protected boolean classpathContainersLast; /** * Whether to place test resources after main resources. Note that the default behavior of Maven version 2.0.8 or * later is to have test dirs before main dirs in classpath so this is discouraged if you need to reproduce the * maven behavior during tests. The default behavior is also changed in eclipse plugin version 2.6 in order to * better match the maven one. Switching to "test source last" can anyway be useful if you need to run your * application in eclipse, since there is no concept in eclipse of "phases" with different set of source dirs and * dependencies like we have in maven. * * @since 2.9 */ @Parameter( property = "eclipse.testSourcesLast", defaultValue = "false" ) protected boolean testSourcesLast; /** * The plugin is often capable in predicting the required Java EE version based on the dependencies of the project. * By setting this parameter to one of the {@code jeeversion} options the version will be locked. * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
Java EE API versions
jeeversionEJB versionServlet versionJSP version
8.03.24.02.3
7.03.23.12.3
6.03.13.02.2
5.03.02.52.1
1.42.12.42.0
1.32.02.31.2
1.21.12.21.1
* * @since 2.9 */ @Parameter( property = "eclipse.jeeversion" ) protected String jeeversion; protected final boolean isJavaProject() { return isJavaProject; } /** * Getter for buildcommands. * * @return Returns the buildcommands. */ public final List getBuildcommands() { return buildcommands; } /** * Setter for buildcommands. * * @param buildcommands The buildcommands to set. */ public final void setBuildcommands( List buildcommands ) { this.buildcommands = buildcommands; } /** * Getter for buildOutputDirectory. * * @return Returns the buildOutputDirectory. */ public final File getBuildOutputDirectory() { return buildOutputDirectory; } /** * Setter for buildOutputDirectory. * * @param buildOutputDirectory The buildOutputDirectory to set. */ public final void setBuildOutputDirectory( File buildOutputDirectory ) { this.buildOutputDirectory = buildOutputDirectory; } /** * Getter for classpathContainers. * * @return Returns the classpathContainers. */ public final List getClasspathContainers() { return classpathContainers; } /** * Setter for classpathContainers. * * @param classpathContainers The classpathContainers to set. */ public final void setClasspathContainers( List classpathContainers ) { this.classpathContainers = classpathContainers; } /** * Getter for eclipseProjectDir. * * @return Returns the eclipseProjectDir. */ public final File getEclipseProjectDir() { return eclipseProjectDir; } /** * Setter for eclipseProjectDir. * * @param eclipseProjectDir The eclipseProjectDir to set. */ public final void setEclipseProjectDir( File eclipseProjectDir ) { this.eclipseProjectDir = eclipseProjectDir; } /** * Getter for projectnatures. * * @return Returns the projectnatures. */ public final List getProjectnatures() { return projectnatures; } /** * Setter for projectnatures. * * @param projectnatures The projectnatures to set. */ public final void setProjectnatures( List projectnatures ) { this.projectnatures = projectnatures; } /** * Getter for useProjectReferences. * * @return Returns the useProjectReferences. */ @Override public final boolean getUseProjectReferences() { return useProjectReferences; } /** * Setter for useProjectReferences. * * @param useProjectReferences The useProjectReferences to set. */ public final void setUseProjectReferences( boolean useProjectReferences ) { this.useProjectReferences = useProjectReferences; } /** * Getter for wtpversion. * * @return Returns the wtpversion. */ public final String getWtpversion() { return wtpversion; } /** * Setter for wtpversion. * * @param wtpversion The wtpversion to set. */ public final void setWtpversion( String wtpversion ) { this.wtpversion = wtpversion; } /** * Getter for additionalBuildcommands. * * @return Returns the additionalBuildcommands. */ public final List getAdditionalBuildcommands() { return additionalBuildcommands; } /** * Setter for additionalBuildcommands. * * @param additionalBuildcommands The additionalBuildcommands to set. */ public final void setAdditionalBuildcommands( List additionalBuildcommands ) { this.additionalBuildcommands = additionalBuildcommands; } /** * Getter for additionalProjectnatures. * * @return the additionalProjectnatures. */ public final List getAdditionalSourceFolders() { return additionalSourceFolders; } /** * Setter for additionalSourceFolders. * * @param additionalSourceFolders The additionalProjectnatures to set. */ public final void setAdditionalSourceFolders( List additionalSourceFolders ) { this.additionalSourceFolders = additionalSourceFolders; } /** * Getter for additionalOptionalSourceFolders. * * @return the additionalOptionalSourceFolders. */ public final List getAdditionalOptionalSourceFolders() { return additionalOptionalSourceFolders; } /** * Setter for additionalOptionalSourceFolders. * * @param additionalOptionalSourceFolders The additionalOptionalSourceFolders to set. */ public final void setAdditionalOptionalSourceFolders( List additionalOptionalSourceFolders ) { this.additionalOptionalSourceFolders = additionalOptionalSourceFolders; } /** * Getter for additionalTestSourceFolders. * * @return the additionalTestSourceFolders. */ public final List getAdditionalTestSourceFolders() { return additionalTestSourceFolders; } /** * Setter for additionalTestSourceFolders. * * @param additionalTestSourceFolders The additionalTestSourceFolders to set. */ public final void setAdditionalTestSourceFolders( List additionalTestSourceFolders ) { this.additionalTestSourceFolders = additionalTestSourceFolders; } /** * Getter for additionalOptionalTestSourceFolders. * * @return the additionalOptionalTestSourceFolders. */ public final List getAdditionalOptionalTestSourceFolders() { return additionalOptionalTestSourceFolders; } /** * Setter for additionalOptionalTestSourceFolders. * * @param additionalOptionalTestSourceFolders The additionalOptionalTestSourceFolders to set. */ public final void setAdditionalOptionalTestSourceFolders( List additionalOptionalTestSourceFolders ) { this.additionalOptionalTestSourceFolders = additionalOptionalTestSourceFolders; } /** * Getter for additionalProjectnatures. * * @return the additionalProjectnatures. */ public final List getAdditionalProjectnatures() { return additionalProjectnatures; } /** * Setter for additionalProjectnatures. * * @param additionalProjectnatures The additionalProjectnatures to set. */ public final void setAdditionalProjectnatures( List additionalProjectnatures ) { this.additionalProjectnatures = additionalProjectnatures; } /** * Getter for addVersionToProjectName. * * @return the addVersionToProjectName. */ public final boolean isAddVersionToProjectName() { return addVersionToProjectName; } /** * Setter for addVersionToProjectName. * * @param addVersionToProjectName addVersionToProjectName */ public final void setAddVersionToProjectName( boolean addVersionToProjectName ) { this.addVersionToProjectName = addVersionToProjectName; } /** * Getter for addGroupIdToProjectName. * * @return the addGroupIdToProjectName. */ public final boolean isAddGroupIdToProjectName() { return addGroupIdToProjectName; } /** * Setter for addGroupIdToProjectName. * * @param addGroupIdToProjectName addGroupIdToProjectName */ public final void setAddGroupIdToProjectName( boolean addGroupIdToProjectName ) { this.addGroupIdToProjectName = addGroupIdToProjectName; } /** * Getter for projectNameTemplate * * @return projectNameTemplate */ public final String getProjectNameTemplate() { return projectNameTemplate; } /** * Setter for projectNameTemplate. * * @param projectNameTemplate projectNameTemplate */ public final void setProjectNameTemplate( String projectNameTemplate ) { this.projectNameTemplate = projectNameTemplate; } /** * @return the linkedResources */ public List getLinkedResources() { return linkedResources; } /** * @param linkedResources the linkedResources to set */ public void setLinkedResources( List linkedResources ) { this.linkedResources = linkedResources; } /** * @see org.apache.maven.plugin.Mojo#execute() */ @Override public final boolean setup() throws MojoExecutionException { boolean ready; checkDeprecations(); setProjectNameTemplate( IdeUtils.calculateProjectNameTemplate( getProjectNameTemplate(), isAddVersionToProjectName(), isAddGroupIdToProjectName(), getLog() ) ); ready = validate(); // TODO: Why are we using project in some places, and executedProject in others?? ArtifactHandler artifactHandler = project.getArtifact().getArtifactHandler(); // ear projects don't contain java sources isJavaProject = ( Constants.LANGUAGE_JAVA.equals( artifactHandler.getLanguage() ) && !Constants.PROJECT_PACKAGING_EAR.equals( packaging ) ); if ( sourceIncludes == null ) { sourceIncludes = new ArrayList<>(); } if ( isJavaProject ) { sourceIncludes.add( JAVA_FILE_PATTERN ); } if ( sourceExcludes == null ) { sourceExcludes = new ArrayList<>(); } parseConfigurationOptions(); // defaults if ( projectnatures == null ) { fillDefaultNatures( packaging ); } if ( additionalProjectnatures != null ) { projectnatures.addAll( additionalProjectnatures ); } if ( buildcommands == null ) { fillDefaultBuilders( packaging ); } else { convertBuildCommandList( buildcommands ); } if ( additionalBuildcommands != null ) { convertBuildCommandList( additionalBuildcommands ); buildcommands.addAll( additionalBuildcommands ); } if ( classpathContainers == null ) { fillDefaultClasspathContainers( packaging ); } else { verifyClasspathContainerListIsComplete(); } if ( linkedResources == null ) { linkedResources = new ArrayList<>(); } locator.addSearchPath( FileResourceLoader.ID, project.getFile().getParentFile().getAbsolutePath() ); locator.setOutputDirectory( new File( project.getBuild().getDirectory() ) ); // ready to start return ready; } /** * Convert any Strings in the commands List to BuildCommands. The conversion happens in * situ. * * @param commands a list of commands to convert into BuildCommand */ protected final void convertBuildCommandList( List commands ) { if ( commands != null ) { for ( ListIterator i = commands.listIterator(); i.hasNext(); ) { Object command = i.next(); if ( command instanceof String ) { command = new BuildCommand( (String) command ); i.set( command ); } } } } // CHECKSTYLE_OFF: MagicNumber private void parseConfigurationOptions() { if ( "R7".equalsIgnoreCase( wtpversion ) ) //$NON-NLS-1$ { wtpVersionFloat = 0.7f; } else if ( "1.0".equalsIgnoreCase( wtpversion ) ) //$NON-NLS-1$ { wtpVersionFloat = 1.0f; } else if ( "1.5".equalsIgnoreCase( wtpversion ) ) //$NON-NLS-1$ { wtpVersionFloat = 1.5f; } else if ( "2.0".equalsIgnoreCase( wtpversion ) ) //$NON-NLS-1$ { wtpVersionFloat = 2.0f; } if ( !"none".equalsIgnoreCase( wtpversion ) ) { getLog().info( Messages.getString( "EclipsePlugin.wtpversion", wtpversion ) ); } } // CHECKSTYLE_ON: MagicNumber private void verifyClasspathContainerListIsComplete() { boolean containsJREContainer = false; // Check if classpathContainer contains a JRE (default, alternate or // Execution Environment) for ( String classPathContainer : classpathContainers ) { if ( classPathContainer != null && classPathContainer.startsWith( COMMON_PATH_JDT_LAUNCHING_JRE_CONTAINER ) ) { containsJREContainer = true; break; } } if ( !containsJREContainer ) { getLog().warn( Messages.getString( "EclipsePlugin.missingjrecontainer" ) ); classpathContainers.add( 0, COMMON_PATH_JDT_LAUNCHING_JRE_CONTAINER ); } } private boolean validate() throws MojoExecutionException { // validate sanity of the current m2 project if ( Arrays.binarySearch( WTP_SUPPORTED_VERSIONS, wtpversion ) < 0 ) { throw new MojoExecutionException( Messages.getString( "EclipsePlugin.unsupportedwtp", new Object[] { wtpversion, StringUtils.join( WTP_SUPPORTED_VERSIONS, " " ) } ) ); } assertNotEmpty( executedProject.getGroupId(), POM_ELT_GROUP_ID ); assertNotEmpty( executedProject.getArtifactId(), POM_ELT_ARTIFACT_ID ); if ( executedProject.getFile() == null || !executedProject.getFile().exists() ) { throw new MojoExecutionException( Messages.getString( "EclipsePlugin.missingpom" ) ); } if ( "pom".equals( packaging ) && eclipseProjectDir == null ) { getLog().info( Messages.getString( "EclipsePlugin.pompackaging" ) ); return false; } if ( "eclipse-plugin".equals( packaging ) ) { getLog().info( Messages.getString( "EclipsePlugin.pdepackaging" ) ); return false; } if ( eclipseProjectDir == null ) { eclipseProjectDir = executedProject.getFile().getParentFile(); } if ( !eclipseProjectDir.exists() && !eclipseProjectDir.mkdirs() ) { throw new MojoExecutionException( Messages.getString( "EclipsePlugin.cantcreatedir", eclipseProjectDir ) ); } if ( !eclipseProjectDir.equals( executedProject.getFile().getParentFile() ) ) { if ( !eclipseProjectDir.isDirectory() ) { throw new MojoExecutionException( Messages.getString( "EclipsePlugin.notadir", eclipseProjectDir ) ); } eclipseProjectDir = new File( eclipseProjectDir, executedProject.getArtifactId() ); if ( !eclipseProjectDir.isDirectory() && !eclipseProjectDir.mkdirs() ) { throw new MojoExecutionException( Messages.getString( "EclipsePlugin.cantcreatedir", eclipseProjectDir ) ); } } validateExtras(); return true; } /** * Extension point for subclasses. *

* Called during setup and used to validate that the configuration is sane. * * @throws MojoExecutionException mojo failures. */ protected void validateExtras() throws MojoExecutionException { // provided for extension. } private void checkDeprecations() { if ( eclipseDownloadSources ) { // deprecated warning getLog().warn( Messages.getString( "EclipsePlugin.deprecatedpar", new Object[] { "eclipse.downloadSources", "downloadSources" } ) ); downloadSources = true; } checkDeprecationsExtras(); } /** * Extension point for subclasses. *

* Check for any extra deprecations and log warnings. Called during setup */ protected void checkDeprecationsExtras() { // provided for extension. } @Override public final void writeConfiguration( IdeDependency[] deps ) throws MojoExecutionException { EclipseWriterConfig config = createEclipseWriterConfig( deps ); // CHECKSTYLE_OFF: MagicNumber if ( wtpVersionFloat == 0.7f ) { new EclipseWtpmodulesWriter().init( getLog(), config ).write(); } if ( wtpVersionFloat >= 1.0f ) { new EclipseWtpFacetsWriter().init( getLog(), config ).write(); } if ( wtpVersionFloat == 1.0f ) { new EclipseWtpComponentWriter().init( getLog(), config ).write(); } if ( wtpVersionFloat >= 1.5 ) { new EclipseWtpComponent15Writer().init( getLog(), config ).write(); } // CHECKSTYLE_ON: MagicNumber new EclipseSettingsWriter().init( getLog(), config ).write(); if ( isJavaProject ) { new EclipseClasspathWriter().init( getLog(), config ).write(); } if ( wtpapplicationxml ) { new EclipseWtpApplicationXMLWriter().init( getLog(), config ).write(); } // NOTE: This one MUST be after EclipseClasspathwriter, and possibly others, // since currently EclipseClasspathWriter does some magic to detect nested // output folders and modifies the configuration by adding new (Ant) builders. // So the .project file must be written AFTER those have run! new EclipseProjectWriter().init( getLog(), config ).write(); writeAdditionalConfig(); getLog().info( Messages.getString( "EclipsePlugin.wrote", new Object[] { config.getEclipseProjectName(), eclipseProjectDir.getAbsolutePath() } ) ); } private void writeAdditionalConfig() throws MojoExecutionException { if ( additionalConfig != null ) { for ( EclipseConfigFile file : additionalConfig ) { File projectRelativeFile = new File( eclipseProjectDir, file.getName() ); if ( projectRelativeFile.isDirectory() ) { // just ignore? getLog().warn( Messages.getString( "EclipsePlugin.foundadir", projectRelativeFile.getAbsolutePath() ) ); } try { projectRelativeFile.getParentFile().mkdirs(); if ( file.getContent() == null ) { if ( file.getLocation() != null ) { InputStream inStream = locator.getResourceAsInputStream( file.getLocation() ); OutputStream outStream = new FileOutputStream( projectRelativeFile ); try { IOUtil.copy( inStream, outStream ); } finally { IOUtil.close( inStream ); IOUtil.close( outStream ); } } else { URL url = file.getURL(); String endPointUrl = url.getProtocol() + "://" + url.getAuthority(); // Repository Id should be ignored by Wagon ... Repository repository = new Repository( "additonal-configs", endPointUrl ); Wagon wagon = wagonManager.getWagon( repository ); if ( logger.isDebugEnabled() ) { Debug debug = new Debug(); wagon.addSessionListener( debug ); wagon.addTransferListener( debug ); } wagon.setTimeout( 1000 ); Settings settings = mavenSettingsBuilder.buildSettings(); ProxyInfo proxyInfo = null; if ( settings != null && settings.getActiveProxy() != null ) { Proxy settingsProxy = settings.getActiveProxy(); proxyInfo = new ProxyInfo(); proxyInfo.setHost( settingsProxy.getHost() ); proxyInfo.setType( settingsProxy.getProtocol() ); proxyInfo.setPort( settingsProxy.getPort() ); proxyInfo.setNonProxyHosts( settingsProxy.getNonProxyHosts() ); proxyInfo.setUserName( settingsProxy.getUsername() ); proxyInfo.setPassword( settingsProxy.getPassword() ); } if ( proxyInfo != null ) { wagon.connect( repository, wagonManager.getAuthenticationInfo( repository.getId() ), proxyInfo ); } else { wagon.connect( repository, wagonManager.getAuthenticationInfo( repository.getId() ) ); } wagon.get( url.getPath(), projectRelativeFile ); } } else { FileUtils.fileWrite( projectRelativeFile.getAbsolutePath(), file.getContent() ); } } catch ( WagonException e ) { throw new MojoExecutionException( Messages.getString( "EclipsePlugin.remoteexception", new Object[] { file.getURL(), e.getMessage() } ) ); } catch ( IOException e ) { throw new MojoExecutionException( Messages.getString( "EclipsePlugin.cantwritetofile", projectRelativeFile.getAbsolutePath() ) ); } catch ( ResourceNotFoundException e ) { throw new MojoExecutionException( Messages.getString( "EclipsePlugin.cantfindresource", file.getLocation() ) ); } catch ( XmlPullParserException e ) { throw new MojoExecutionException( Messages.getString( "EclipsePlugin.settingsxmlfailure", e.getMessage() ) ); } } } } /** * Create the EclipseWriterConfig for the specified dependencies. * * @param deps the project dependencies * @return a configured EclipseWriterConfig * @throws MojoExecutionException mojo failures. */ protected final EclipseWriterConfig createEclipseWriterConfig( IdeDependency[] deps ) throws MojoExecutionException { File projectBaseDir = executedProject.getFile().getParentFile(); // build a list of UNIQUE source dirs (both src and resources) to be // used in classpath and wtpmodules EclipseSourceDir[] sourceDirs = buildDirectoryList( executedProject, eclipseProjectDir, buildOutputDirectory ); EclipseWriterConfig config = new EclipseWriterConfig(); config.setWorkspaceConfiguration( getWorkspaceConfiguration() ); config.setProjectNameTemplate( getProjectNameTemplate() ); String projectName = IdeUtils.getProjectName( config.getProjectNameTemplate(), project ); config.setEclipseProjectName( projectName ); config.setWtpapplicationxml( wtpapplicationxml ); config.setWtpVersion( wtpVersionFloat ); Set convertedBuildCommands = new LinkedHashSet<>(); if ( buildcommands != null ) { for ( Object cmd : buildcommands ) { if ( cmd instanceof BuildCommand ) { convertedBuildCommands.add( (BuildCommand) cmd ); } else { convertedBuildCommands.add( new BuildCommand( (String) cmd ) ); } } } config.setBuildCommands( new LinkedList<>( convertedBuildCommands ) ); config.setBuildOutputDirectory( buildOutputDirectory ); config.setClasspathContainers( classpathContainers ); config.setDeps( deps ); config.setEclipseProjectDirectory( eclipseProjectDir ); config.setLocalRepository( localRepository ); config.setOSGIManifestFile( manifest ); config.setProject( project ); config.setProjectBaseDir( projectBaseDir ); config.setProjectnatures( projectnatures ); config.setProjectFacets( additionalProjectFacets ); config.setSourceDirs( sourceDirs ); config.setPackaging( packaging ); config.setLinkedResources( linkedResources ); config.setClasspathContainersLast( classpathContainersLast ); config.setJeeVersion( jeeversion ); collectWarContextRootsFromReactorEarConfiguration( config ); return config; } /** * If this is a war module peek into the reactor an search for an ear module that defines the context root of this * module. * * @param config config to save the context root. */ private void collectWarContextRootsFromReactorEarConfiguration( EclipseWriterConfig config ) { if ( reactorProjects != null && wtpContextName == null && Constants.PROJECT_PACKAGING_WAR.equals( project.getPackaging() ) ) { for ( MavenProject reactorProject : reactorProjects ) { if ( Constants.PROJECT_PACKAGING_EAR.equals( reactorProject.getPackaging() ) ) { Xpp3Dom[] warDefinitions = IdeUtils.getPluginConfigurationDom( reactorProject, JeeUtils.ARTIFACT_MAVEN_EAR_PLUGIN, new String[] { "modules", "webModule" } ); for ( Xpp3Dom warDefinition : warDefinitions ) { Xpp3Dom groupId = warDefinition.getChild( "groupId" ); Xpp3Dom artifactId = warDefinition.getChild( "artifactId" ); Xpp3Dom contextRoot = warDefinition.getChild( "contextRoot" ); if ( groupId != null && artifactId != null && contextRoot != null && groupId.getValue() != null && artifactId.getValue() != null && contextRoot.getValue() != null ) { getLog().info( "Found context root definition for " + groupId.getValue() + ":" + artifactId.getValue() + " " + contextRoot.getValue() ); if ( project.getArtifactId().equals( artifactId.getValue() ) && project.getGroupId().equals( groupId.getValue() ) ) { config.setContextName( contextRoot.getValue() ); } } else { getLog().info( "Found incomplete ear configuration in " + reactorProject.getGroupId() + ":" + reactorProject.getGroupId() + " found " + warDefinition.toString() ); } } } } } if ( config.getContextName() == null && Constants.PROJECT_PACKAGING_WAR.equals( project.getPackaging() ) ) { if ( wtpContextName == null ) { config.setContextName( project.getArtifactId() ); } else if ( "ROOT".equals( wtpContextName ) ) { config.setContextName( "" ); } else { config.setContextName( wtpContextName ); } } } private void assertNotEmpty( String string, String elementName ) throws MojoExecutionException { if ( string == null ) { throw new MojoExecutionException( Messages.getString( "EclipsePlugin.missingelement", elementName ) ); } } /** * Fill getProjectnatures() with values. *

* Subclasses should call super and then calculate their own additions and insert them via * getProjectnatures().addAll(). * * @param packaging the pom's packaging */ protected void fillDefaultNatures( String packaging ) { projectnatures = new ArrayList<>(); // CHECKSTYLE_OFF: MagicNumber if ( wtpVersionFloat >= 1.0f ) { projectnatures.add( NATURE_WST_FACET_CORE_NATURE ); // WTP 1.0 nature } if ( isJavaProject ) { projectnatures.add( NATURE_JDT_CORE_JAVA ); } if ( wtpVersionFloat >= 0.7f ) { projectnatures.add( NATURE_WST_MODULE_CORE_NATURE ); // WTP 0.7/1.0 nature if ( isJavaProject ) { projectnatures.add( NATURE_JEM_WORKBENCH_JAVA_EMF ); // WTP 0.7/1.0 nature } } // CHECKSTYLE_ON: MagicNumber } /** * Fill getClasspathContainers() with values. *

* Subclasses should call super and then calculate their own additions and insert them via * getClasspathContainers().addAll(). * * @param packaging the pom's packaging */ protected void fillDefaultClasspathContainers( String packaging ) { classpathContainers = new ArrayList<>(); if ( getWorkspaceConfiguration().getDefaultClasspathContainer() != null ) { getLog().info( "Adding default classpath container: " + getWorkspaceConfiguration().getDefaultClasspathContainer() ); classpathContainers.add( getWorkspaceConfiguration().getDefaultClasspathContainer() ); } } /** * Fill getBuildcommands() with values. *

* Subclasses should call super and then calculate their own additions and insert them via * getBuildcommands().addAll(). * * @param packaging the pom's packaging */ protected void fillDefaultBuilders( String packaging ) { buildcommands = new ArrayList(); // CHECKSTYLE_OFF: MagicNumber if ( wtpVersionFloat == 0.7f ) { buildcommands.add( new BuildCommand( BUILDER_WST_COMPONENT_STRUCTURAL ) ); // WTP 0.7 builder } if ( isJavaProject ) { buildcommands.add( new BuildCommand( BUILDER_JDT_CORE_JAVA ) ); } if ( wtpVersionFloat >= 1.5f ) { buildcommands.add( new BuildCommand( BUILDER_WST_FACET ) ); // WTP 1.5 builder } if ( wtpVersionFloat >= 0.7f ) { buildcommands.add( new BuildCommand( BUILDER_WST_VALIDATION ) ); // WTP 0.7/1.0 builder } if ( wtpVersionFloat == 0.7f ) { // WTP 0.7 builder buildcommands.add( new BuildCommand( BUILDER_WST_COMPONENT_STRUCTURAL_DEPENDENCY_RESOLVER ) ); } // CHECKSTYLE_ON: MagicNumber } /** * @param project the current Maven project * @param basedir the eclipse workspace directory * @param buildOutputDirectory the default output directory * @return the array of Eclipse source directories * @throws MojoExecutionException of failure */ public final EclipseSourceDir[] buildDirectoryList( MavenProject project, File basedir, File buildOutputDirectory ) throws MojoExecutionException { File projectBaseDir = project.getFile().getParentFile(); String mainOutput = IdeUtils.toRelativeAndFixSeparator( projectBaseDir, buildOutputDirectory, false ); // If using the standard output location, don't mix the test output into it. String testOutput = null; boolean useStandardOutputDir = buildOutputDirectory.equals( new File( project.getBuild().getOutputDirectory() ) ); if ( useStandardOutputDir ) { getLog().debug( "testOutput toRelativeAndFixSeparator " + projectBaseDir + " , " + project.getBuild().getTestOutputDirectory() ); testOutput = IdeUtils.toRelativeAndFixSeparator( projectBaseDir, new File( project.getBuild().getTestOutputDirectory() ), false ); getLog().debug( "testOutput after toRelative : " + testOutput ); } Set mainDirectories = new LinkedHashSet<>(); String sourceDirectory = project.getBuild().getSourceDirectory(); extractSourceDirs( mainDirectories, project.getCompileSourceRoots(), basedir, projectBaseDir, false, false, null, sourceDirectory ); extractAdditionalSourceDirs( mainDirectories, getAdditionalSourceFolders(), basedir, projectBaseDir, false, false, null, sourceDirectory ); extractAdditionalSourceDirs( mainDirectories, getAdditionalOptionalSourceFolders(), basedir, projectBaseDir, false, true, null, sourceDirectory ); extractResourceDirs( mainDirectories, project.getBuild().getResources(), basedir, projectBaseDir, false, mainOutput ); Set testDirectories = new LinkedHashSet<>(); String testSourceDirectory = project.getBuild().getTestSourceDirectory(); extractSourceDirs( testDirectories, project.getTestCompileSourceRoots(), basedir, projectBaseDir, true, false, testOutput, testSourceDirectory ); extractAdditionalSourceDirs( testDirectories, getAdditionalTestSourceFolders(), basedir, projectBaseDir, true, false, testOutput, testSourceDirectory ); extractAdditionalSourceDirs( testDirectories, getAdditionalOptionalTestSourceFolders(), basedir, projectBaseDir, true, true, testOutput, testSourceDirectory ); extractResourceDirs( testDirectories, project.getBuild().getTestResources(), basedir, projectBaseDir, true, testOutput ); // avoid duplicated entries Set directories = new LinkedHashSet<>(); // NOTE: Since MNG-3118, test classes come before main classes boolean testBeforeMain = isMavenVersion( "[2.0.8,)" ); // let users override this if needed, they need to simulate more than the test phase in eclipse if ( testSourcesLast ) { testBeforeMain = false; } if ( testBeforeMain ) { directories.addAll( testDirectories ); directories.removeAll( mainDirectories ); directories.addAll( mainDirectories ); } else { directories.addAll( mainDirectories ); directories.addAll( testDirectories ); } return directories.toArray( new EclipseSourceDir[directories.size()] ); } private void extractSourceDirs( Set directories, List sourceRoots, File basedir, File projectBaseDir, boolean test, boolean optional, String output, String defaultSourceRoot ) throws MojoExecutionException { if ( sourceRoots == null ) { return; } for ( String sourceRoot1 : sourceRoots ) { File sourceRootFile = new File( sourceRoot1 ); if ( sourceRootFile.isDirectory() ) { String sourceRoot = IdeUtils.toRelativeAndFixSeparator( projectBaseDir, sourceRootFile, !projectBaseDir.equals( basedir ) ); directories.add( new EclipseSourceDir( sourceRoot, output, false, test, optional, sourceIncludes, sourceExcludes, false, !defaultSourceRoot.equals( sourceRoot1 ) ) ); } } } private void extractAdditionalSourceDirs( Set directories, List sourceRoots, File basedir, File projectBaseDir, boolean test, boolean optional, String output, String defaultSourceRoot ) throws MojoExecutionException { if ( sourceRoots == null ) { return; } for ( String sourceRoot1 : sourceRoots ) { File sourceRootFile = new File( sourceRoot1 ); String sourceRoot = IdeUtils.toRelativeAndFixSeparator( projectBaseDir, sourceRootFile, !projectBaseDir.equals( basedir ) ); directories.add( new EclipseSourceDir( sourceRoot, output, false, test, optional, sourceIncludes, sourceExcludes, false, !defaultSourceRoot.equals( sourceRoot1 ) ) ); } } final void extractResourceDirs( Set directories, List resources, File basedir, File workspaceProjectBaseDir, boolean test, final String output ) throws MojoExecutionException { for ( Resource resource : resources ) { getLog().debug( "Processing resource dir: " + resource.getDirectory() ); List excludes = new ArrayList<>( resource.getExcludes() ); // automatically exclude java files: eclipse doesn't have the concept of resource directory so it will // try to compile any java file found in maven resource dirs excludes.add( JAVA_FILE_PATTERN ); // TODO: figure out how to merge if the same dir is specified twice // with different in/exclude patterns. File resourceDirectory = new File( /* basedir, */resource.getDirectory() ); if ( !resourceDirectory.exists() || !resourceDirectory.isDirectory() ) { getLog().debug( "Resource dir: " + resourceDirectory + " either missing or not a directory." ); continue; } String resourcePath = IdeUtils.toRelativeAndFixSeparator( workspaceProjectBaseDir, resourceDirectory, !workspaceProjectBaseDir.equals( basedir ) ); String thisOutput = output; if ( thisOutput != null ) { // sometimes thisOutput is already an absolute path File outputFile = new File( thisOutput ); if ( !outputFile.isAbsolute() ) { outputFile = new File( workspaceProjectBaseDir, thisOutput ); } // create output dir if it doesn't exist outputFile.mkdirs(); if ( !StringUtils.isEmpty( resource.getTargetPath() ) ) { outputFile = new File( outputFile, resource.getTargetPath() ); // create output dir if it doesn't exist outputFile.mkdirs(); } getLog().debug( "Making relative and fixing separator: { " + workspaceProjectBaseDir + ", " + outputFile + ", false }." ); thisOutput = IdeUtils.toRelativeAndFixSeparator( workspaceProjectBaseDir, outputFile, false ); } EclipseSourceDir resourceDir = new EclipseSourceDir( resourcePath, thisOutput, true, test, false, resource.getIncludes(), excludes, resource.isFiltering(), false ); if ( !directories.add( resourceDir ) ) { EclipseSourceDir originalDir = get( directories, resourceDir ); boolean merged = originalDir.merge( resourceDir ); if ( merged ) { getLog().info( "Resource directory's path matches an existing source directory. " + "Resources have been merged with the source directory " + originalDir.getPath() ); } else { getLog().info( "Resource directory's path matches an existing source directory but \"test\", " + "\"filtering\" or \"output\" were different. The resulting eclipse configuration " + "may not accurately reflect the project configuration for " + originalDir.getPath() ); } } } } /** * java.util.Set doesn't have a get() method that returns the matching object. Since we use objects that are * different by conceptually "equal" based on the path we need to locate the original object out of the Set. * * @param set the set to iterate over looking for the specified object * @param o the object to locate in the set * @return the object from the set, or null if not found in the set */ private static T get( Set set, Object o ) { for ( T item : set ) { if ( o.equals( item ) ) { return item; } } return null; } /** * {@inheritDoc} */ @Override public String getProjectNameForArifact( Artifact artifact ) { IdeDependency[] workspaceArtefacts = getWorkspaceArtefacts(); for ( int index = 0; workspaceArtefacts != null && index < workspaceArtefacts.length; index++ ) { IdeDependency workspaceArtefact = workspaceArtefacts[index]; if ( workspaceArtefact.isAddedToClasspath() && workspaceArtefact.getGroupId().equals( artifact.getGroupId() ) && workspaceArtefact.getArtifactId().equals( artifact.getArtifactId() ) ) { if ( workspaceArtefact.getVersion().equals( artifact.getBaseVersion() ) ) { return workspaceArtefact.getEclipseProjectName(); } } } MavenProject reactorProject = getReactorProject( artifact ); if ( reactorProject != null ) { return IdeUtils.getProjectName( getProjectNameTemplateForMavenProject( reactorProject ), artifact ); } return IdeUtils.getProjectName( getProjectNameTemplate(), artifact ); } /** * @param mavenProject the project to get the projectNameTemplate configuration from * @return the projectNameTemplate configuration from the specified MavenProject */ private String getProjectNameTemplateForMavenProject( MavenProject mavenProject ) { String projectNameTemplate = null; boolean addVersionToProjectName = false; boolean addGroupIdToProjectName = false; Build build = mavenProject.getBuild(); if ( build != null ) { String eclipsePlugin = "com.github.marschall:eclipse-maven-plugin"; Plugin plugin = (Plugin) build.getPluginsAsMap().get( eclipsePlugin ); if ( plugin == null && build.getPluginManagement() != null ) { plugin = (Plugin) build.getPluginManagement().getPluginsAsMap().get( eclipsePlugin ); } if ( plugin != null ) { Xpp3Dom config = (Xpp3Dom) plugin.getConfiguration(); if ( config != null ) { Xpp3Dom projectNameTemplateNode = config.getChild( "projectNameTemplate" ); if ( projectNameTemplateNode != null ) { projectNameTemplate = projectNameTemplateNode.getValue(); } Xpp3Dom addVersionToProjectNameNode = config.getChild( "addVersionToProjectName" ); addVersionToProjectName = addVersionToProjectNameNode != null; Xpp3Dom addGroupIdToProjectNameNode = config.getChild( "addGroupIdToProjectName" ); addGroupIdToProjectName = addGroupIdToProjectNameNode != null; } } } return IdeUtils.calculateProjectNameTemplate( projectNameTemplate, addVersionToProjectName, addGroupIdToProjectName, getLog() ); } /** * {@inheritDoc} */ @Override protected final IdeDependency[] getWorkspaceArtefacts() { return getWorkspaceConfiguration().getWorkspaceArtefacts(); } public final WorkspaceConfiguration getWorkspaceConfiguration() { if ( workspaceConfiguration == null ) { workspaceConfiguration = new WorkspaceConfiguration(); locateWorkspace(); getLog().info( Messages.getString( "EclipsePlugin.workspace", workspace ) ); workspaceConfiguration.setWorkspaceDirectory( workspace ); new ReadWorkspaceLocations().init( getLog(), workspaceConfiguration, project, wtpdefaultserver, preferStandardClasspathContainer ); } return workspaceConfiguration; } /** * If workspace is not defined, then attempt to locate it by checking up the directory hierarchy. */ private void locateWorkspace() { if ( workspace == null ) { File currentWorkingDirectory = new File( "." ).getAbsoluteFile(); while ( currentWorkingDirectory != null ) { File metadataDirectory = new File( currentWorkingDirectory, ".metadata" ); logger.debug( "Checking for eclipse workspace at " + currentWorkingDirectory ); if ( metadataDirectory.exists() && metadataDirectory.isDirectory() ) { logger.debug( " Found workspace at " + currentWorkingDirectory ); workspace = currentWorkingDirectory; return; } currentWorkingDirectory = currentWorkingDirectory.getParentFile(); } } } @Override public final List getExcludes() { return excludes; } /** * Utility method that locates a project producing the given artifact and verifies if it's a valid Eclipse project. * * @param artifact the artifact a project should produce. * @return true if the artifact is produced by a reactor projectart. */ @Override protected boolean isAvailableAsAReactorProject( Artifact artifact ) { MavenProject project = getReactorProject( artifact ); return ( project != null && new File( project.getBasedir(), ".project" ).exists() ); } /** * Utility method that locates a project in the workspace for the given artifact. * * @param artifact the artifact a project should produce. * @return true if the artifact is produced by a reactor projectart. */ private boolean isAvailableAsAWorkspaceProject( Artifact artifact ) { IdeDependency[] workspaceArtefacts = getWorkspaceArtefacts(); for ( int index = 0; workspaceArtefacts != null && index < workspaceArtefacts.length; index++ ) { IdeDependency workspaceArtefact = workspaceArtefacts[index]; if ( workspaceArtefact.getGroupId().equals( artifact.getGroupId() ) && workspaceArtefact.getArtifactId().equals( artifact.getArtifactId() ) ) { if ( workspaceArtefact.getVersion().equals( artifact.getBaseVersion() ) ) { workspaceArtefact.setAddedToClasspath( true ); getLog().debug( "Using workspace project: " + workspaceArtefact.getEclipseProjectName() ); return true; } else { getLog().info( "Artifact " + artifact.getId() + " already available as a workspace project, but with different version. " + "Expected: " + artifact.getBaseVersion() + ", found: " + workspaceArtefact.getVersion() ); } } } return false; } /** * Checks if jar has to be resolved for the given artifact * * @param art the artifact to check * @return true if resolution should happen */ @Override protected final boolean hasToResolveJar( Artifact art ) { return !( getUseProjectReferences() && isAvailableAsAReactorProject( art ) ) || ( limitProjectReferencesToWorkspace && !( getUseProjectReferences() && isAvailableAsAWorkspaceProject( art ) ) ); } /** * Checks if a projects reference has to be used for the given artifact * * @param art the artifact to check * @return true if a project reference has to be used. */ @Override protected final boolean useProjectReference( Artifact art ) { boolean isReactorProject = getUseProjectReferences() && isAvailableAsAReactorProject( art ); boolean isWorkspaceProject = getUseProjectReferences() && isAvailableAsAWorkspaceProject( art ); return ( isReactorProject && !limitProjectReferencesToWorkspace ) || // default ( limitProjectReferencesToWorkspace && isWorkspaceProject ) || // limitProjectReferencesToWorkspace ( !isReactorProject && isWorkspaceProject ); // default + workspace projects } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy