View Javadoc

1   /*
2    * Copyright 2007 scala-tools.org
3    *
4    * Licensed under the Apache License, Version 2.0 (the "License");
5    * you may not use this file except in compliance with the License.
6    * You may obtain a copy of the License at
7    *
8    *    http://www.apache.org/licenses/LICENSE-2.0
9    *
10   * Unless required by applicable law or agreed to in writing,
11   * software distributed under the License is distributed on an "AS IS" BASIS,
12   * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13   * See the License for the specific language governing permissions
14   * and limitations under the License.
15   */
16  package org_scala_tools_maven;
17  
18  import java.util.ArrayList;
19  import java.util.HashSet;
20  import java.util.List;
21  import java.util.Set;
22  
23  import org.apache.maven.artifact.Artifact;
24  import org.apache.maven.artifact.ArtifactUtils;
25  import org.apache.maven.artifact.factory.ArtifactFactory;
26  import org.apache.maven.artifact.metadata.ArtifactMetadataSource;
27  import org.apache.maven.artifact.repository.ArtifactRepository;
28  import org.apache.maven.artifact.resolver.ArtifactCollector;
29  import org.apache.maven.artifact.resolver.ArtifactNotFoundException;
30  import org.apache.maven.artifact.resolver.ArtifactResolutionException;
31  import org.apache.maven.artifact.resolver.ArtifactResolver;
32  import org.apache.maven.artifact.resolver.filter.AndArtifactFilter;
33  import org.apache.maven.artifact.resolver.filter.ArtifactFilter;
34  import org.apache.maven.artifact.resolver.filter.ScopeArtifactFilter;
35  import org.apache.maven.model.Dependency;
36  import org.apache.maven.plugin.AbstractMojo;
37  import org.apache.maven.plugin.MojoExecutionException;
38  import org.apache.maven.plugin.MojoFailureException;
39  import org.apache.maven.project.MavenProject;
40  import org.apache.maven.project.MavenProjectBuilder;
41  import org.apache.maven.project.ProjectBuildingException;
42  import org.apache.maven.project.artifact.InvalidDependencyVersionException;
43  import org.apache.maven.shared.dependency.tree.DependencyNode;
44  import org.apache.maven.shared.dependency.tree.DependencyTreeBuilder;
45  import org.apache.maven.shared.dependency.tree.filter.AncestorOrSelfDependencyNodeFilter;
46  import org.apache.maven.shared.dependency.tree.filter.AndDependencyNodeFilter;
47  import org.apache.maven.shared.dependency.tree.filter.DependencyNodeFilter;
48  import org.apache.maven.shared.dependency.tree.traversal.CollectingDependencyNodeVisitor;
49  import org.apache.maven.shared.dependency.tree.traversal.DependencyNodeVisitor;
50  import org.apache.maven.shared.dependency.tree.traversal.FilteringDependencyNodeVisitor;
51  import org.codehaus.plexus.util.StringUtils;
52  import org_scala_tools_maven_dependency.CheckScalaVersionVisitor;
53  import org_scala_tools_maven_dependency.ScalaDistroArtifactFilter;
54  import org_scala_tools_maven_executions.JavaMainCaller;
55  import org_scala_tools_maven_executions.JavaMainCallerByFork;
56  import org_scala_tools_maven_executions.JavaMainCallerInProcess;
57  import org_scala_tools_maven_executions.MainHelper;
58  
59  public abstract class ScalaMojoSupport extends AbstractMojo {
60  
61      public static final String SCALA_GROUPID= "org.scala-lang";
62      public static final String SCALA_LIBRARY_ARTIFACTID= "scala-library";
63      /**
64       * @parameter expression="${project}"
65       * @required
66       * @readonly
67       */
68      protected MavenProject project;
69  
70      /**
71       * Used to look up Artifacts in the remote repository.
72       *
73       * @component
74       * @required
75       * @readonly
76       */
77      protected ArtifactFactory factory;
78  
79      /**
80       * Used to look up Artifacts in the remote repository.
81       *
82       * @component
83       * @required
84       * @readonly
85       */
86      protected ArtifactResolver resolver;
87      /**
88       * Location of the local repository.
89       *
90       * @parameter expression="${localRepository}"
91       * @readonly
92       * @required
93       */
94      protected ArtifactRepository localRepo;
95  
96      /**
97       * List of Remote Repositories used by the resolver
98       *
99       * @parameter expression="${project.remoteArtifactRepositories}"
100      * @readonly
101      * @required
102      */
103     protected List<?> remoteRepos;
104 
105     /**
106      * Additional dependencies/jar to add to classpath to run "scalaClassName" (scope and optional field not supported)
107      * ex :
108      * <pre>
109      *    &lt;dependencies>
110      *      &lt;dependency>
111      *        &lt;groupId>org.scala-tools&lt;/groupId>
112      *        &lt;artifactId>scala-compiler-addon&lt;/artifactId>
113      *        &lt;version>1.0-SNAPSHOT&lt;/version>
114      *      &lt;/dependency>
115      *    &lt;/dependencies>
116      * </pre>
117      * @parameter
118      */
119     protected BasicArtifact[] dependencies;
120 
121     /**
122      * Compiler plugin dependencies to use when compiling.
123      * ex:
124      * @parameter
125      * <xmp>
126      * <compilerPlugins>
127      * <compilerPlugin>
128      * <groupId>my.scala.plugin</groupId>
129      * <artifactId>amazingPlugin</artifactId>
130      * <version>1.0-SNAPSHOT</version>
131      * </compilerPlugin>
132      * </compilerPlugins>
133      * </xmp>
134      */
135     protected BasicArtifact[] compilerPlugins;
136 
137     /**
138      * Jvm Arguments.
139      *
140      * @parameter
141      */
142     protected String[] jvmArgs;
143 
144     /**
145      * compiler additionnals arguments
146      *
147      * @parameter
148      */
149     protected String[] args;
150 
151     /**
152      * className (FQN) of the scala tool to provide as
153      *
154      * @required
155      * @parameter expression="${maven.scala.className}"
156      *            default-value="scala.tools.nsc.Main"
157      */
158     protected String scalaClassName;
159 
160     /**
161      * Scala 's version to use.
162      * (property 'maven.scala.version' replaced by 'scala.version')
163      *
164      * @parameter expression="${scala.version}"
165      */
166     private String scalaVersion;
167 
168     /**
169      * Display the command line called ?
170      * (property 'maven.scala.displayCmd' replaced by 'displayCmd')
171      *
172      * @required
173      * @parameter expression="${displayCmd}"
174      *            default-value="false"
175      */
176     public boolean displayCmd;
177 
178     /**
179      * Forks the execution of scalac into a separate process.
180      *
181      * @parameter default-value="true"
182      */
183     protected boolean fork = true;
184 
185     /**
186      * Force the use of an external ArgFile to run any forked process.
187      *
188      * @parameter default-value="false"
189      */
190     protected boolean forceUseArgFile = false;
191 
192     /**
193      * Check if every dependencies use the same version of scala-library.
194      *
195      * @parameter expression="${maven.scala.checkConsistency}" default-value="true"
196      */
197     protected boolean checkMultipleScalaVersions;
198 
199     /**
200      * Determines if a detection of multiple scala versions in the dependencies will cause the build to fail.
201      *
202      * @parameter default-value="false"
203      */
204     protected boolean failOnMultipleScalaVersions = false;
205     /**
206      * Artifact factory, needed to download source jars.
207      *
208      * @component
209      * @required
210      * @readonly
211      */
212     protected MavenProjectBuilder mavenProjectBuilder;
213 
214     /**
215      * The artifact repository to use.
216      *
217      * @parameter expression="${localRepository}"
218      * @required
219      * @readonly
220      */
221     private ArtifactRepository localRepository;
222 
223     /**
224      * The artifact factory to use.
225      *
226      * @component
227      * @required
228      * @readonly
229      */
230     private ArtifactFactory artifactFactory;
231 
232     /**
233      * The artifact metadata source to use.
234      *
235      * @component
236      * @required
237      * @readonly
238      */
239     private ArtifactMetadataSource artifactMetadataSource;
240 
241     /**
242      * The artifact collector to use.
243      *
244      * @component
245      * @required
246      * @readonly
247      */
248     private ArtifactCollector artifactCollector;
249 
250     /**
251      * The dependency tree builder to use.
252      *
253      * @component
254      * @required
255      * @readonly
256      */
257     private DependencyTreeBuilder dependencyTreeBuilder;
258 
259     private VersionNumber _scalaVersionN;
260 
261     /**
262      * This method resolves the dependency artifacts from the project.
263      *
264      * @param theProject The POM.
265      * @return resolved set of dependency artifacts.
266      *
267      * @throws ArtifactResolutionException
268      * @throws ArtifactNotFoundException
269      * @throws InvalidDependencyVersionException
270      */
271     @SuppressWarnings("unchecked")
272     protected Set<Artifact> resolveDependencyArtifacts(MavenProject theProject) throws Exception {
273         AndArtifactFilter filter = new AndArtifactFilter();
274         filter.add(new ScopeArtifactFilter(Artifact.SCOPE_TEST));
275         filter.add(new ArtifactFilter(){
276             public boolean include(Artifact artifact) {
277                 return !artifact.isOptional();
278             }
279         });
280         //TODO follow the dependenciesManagement and override rules
281         Set<Artifact> artifacts = theProject.createArtifacts(factory, Artifact.SCOPE_RUNTIME, filter);
282         for (Artifact artifact : artifacts) {
283             resolver.resolve(artifact, remoteRepos, localRepo);
284         }
285         return artifacts;
286     }
287 
288     /**
289      * This method resolves all transitive dependencies of an artifact.
290      *
291      * @param artifact the artifact used to retrieve dependencies
292      *
293      * @return resolved set of dependencies
294      *
295      * @throws ArtifactResolutionException
296      * @throws ArtifactNotFoundException
297      * @throws ProjectBuildingException
298      * @throws InvalidDependencyVersionException
299      */
300     protected Set<Artifact> resolveArtifactDependencies(Artifact artifact) throws Exception {
301         Artifact pomArtifact = factory.createArtifact(artifact.getGroupId(), artifact.getArtifactId(), artifact.getVersion(), "", "pom");
302         MavenProject pomProject = mavenProjectBuilder.buildFromRepository(pomArtifact, remoteRepos, localRepo);
303         return resolveDependencyArtifacts(pomProject);
304     }
305 
306     public void addToClasspath(String groupId, String artifactId, String version, Set<String> classpath) throws Exception {
307         addToClasspath(groupId, artifactId, version, classpath, true);
308     }
309 
310 
311     public void addToClasspath(String groupId, String artifactId, String version, Set<String> classpath, boolean addDependencies) throws Exception {
312         addToClasspath(factory.createArtifact(groupId, artifactId, version, Artifact.SCOPE_RUNTIME, "jar"), classpath, addDependencies);
313     }
314 
315     protected void addToClasspath(Artifact artifact, Set<String> classpath, boolean addDependencies) throws Exception {
316         resolver.resolve(artifact, remoteRepos, localRepo);
317         classpath.add(artifact.getFile().getCanonicalPath());
318         if (addDependencies) {
319             for (Artifact dep : resolveArtifactDependencies(artifact)) {
320                 //classpath.add(dep.getFile().getCanonicalPath());
321                 addToClasspath(dep, classpath, addDependencies);
322             }
323         }
324     }
325 
326     public void execute() throws MojoExecutionException, MojoFailureException {
327         try {
328             String oldWay = System.getProperty("maven.scala.version");
329             if (oldWay != null) {
330                 getLog().warn("using 'maven.scala.version' is deprecated, use 'scala.version' instead");
331                 if (scalaVersion != null) {
332                     scalaVersion = oldWay;
333                 }
334             }
335 
336             oldWay = System.getProperty("maven.scala.displayCmd");
337             if (oldWay != null) {
338                 getLog().warn("using 'maven.scala.displayCmd' is deprecated, use 'displayCmd' instead");
339                 displayCmd = displayCmd || Boolean.parseBoolean(oldWay);
340             }
341             checkScalaVersion();
342             doExecute();
343         } catch (MojoExecutionException exc) {
344             throw exc;
345         } catch (MojoFailureException exc) {
346             throw exc;
347         } catch (RuntimeException exc) {
348             throw exc;
349         } catch (Exception exc) {
350             throw new MojoExecutionException("wrap: " + exc, exc);
351         }
352     }
353 
354     @SuppressWarnings("unchecked")
355     protected List<Dependency> getDependencies() {
356         return project.getCompileDependencies();
357     }
358 
359     protected VersionNumber findScalaVersion() throws Exception {
360         if (_scalaVersionN == null) {
361             String detectedScalaVersion = scalaVersion;
362             if (StringUtils.isEmpty(detectedScalaVersion)) {
363                 detectedScalaVersion = findScalaVersionFromDependencies();
364             }
365             if (StringUtils.isEmpty(detectedScalaVersion)) {
366                 if (!"pom".equals( project.getPackaging().toLowerCase() )) {
367                     getLog().warn("you don't define "+SCALA_GROUPID + ":" + SCALA_LIBRARY_ARTIFACTID + " as a dependency of the project");
368                 }
369                 detectedScalaVersion = "0.0.0";
370             } else {
371                 // grappy hack to retrieve the SNAPSHOT version without timestamp,...
372                 // because if version is -SNAPSHOT and artifact is deploy with uniqueValue then the version
373                 // get from dependency is with the timestamp and a build number (the resolved version)
374                 // but scala-compiler with the same version could have different resolved version (timestamp,...)
375                 boolean isSnapshot = ArtifactUtils.isSnapshot(detectedScalaVersion);
376                 if (isSnapshot && !detectedScalaVersion.endsWith("-SNAPSHOT")) {
377                     detectedScalaVersion = detectedScalaVersion.substring(0, detectedScalaVersion.lastIndexOf('-', detectedScalaVersion.lastIndexOf('-')-1)) + "-SNAPSHOT";
378                 }
379             }
380             if (StringUtils.isEmpty(detectedScalaVersion)) {
381                 throw new MojoFailureException("no scalaVersion detected or set");
382             }
383             if (StringUtils.isNotEmpty(scalaVersion)) {
384                 if (!scalaVersion.equals(detectedScalaVersion)) {
385                     getLog().warn("scala library version define in dependencies doesn't match the scalaVersion of the plugin");
386                 }
387                 //getLog().info("suggestion: remove the scalaVersion from pom.xml"); //scalaVersion could be define in a parent pom where lib is not required
388             }
389             _scalaVersionN = new VersionNumber(detectedScalaVersion);
390         }
391         return _scalaVersionN;
392     }
393 
394     //TODO refactor to do only one scan of dependency to find all scala-version
395     private String findScalaVersionFromDependencies() throws Exception {
396         String detectedScalaVersion = null;
397         for (Dependency dep : getDependencies()) {
398             if (SCALA_GROUPID.equals(dep.getGroupId()) && SCALA_LIBRARY_ARTIFACTID.equals(dep.getArtifactId())) {
399                 detectedScalaVersion = dep.getVersion();
400             }
401         }
402         if (StringUtils.isEmpty(detectedScalaVersion)) {
403             List<Dependency> deps = new ArrayList<Dependency>();
404             deps.addAll(project.getModel().getDependencies());
405             if (project.getModel().getDependencyManagement() != null) {
406                 deps.addAll(project.getModel().getDependencyManagement().getDependencies());
407             }
408             for (Dependency dep : deps) {
409                 if (SCALA_GROUPID.equals(dep.getGroupId()) && SCALA_LIBRARY_ARTIFACTID.equals(dep.getArtifactId())) {
410                     detectedScalaVersion = dep.getVersion();
411                 }
412             }
413         }
414         return detectedScalaVersion;
415     }
416 
417     protected void checkScalaVersion() throws Exception {
418         if (checkMultipleScalaVersions) {
419             checkCorrectVersionsOfScalaLibrary(findScalaVersion().toString());
420         }
421     }
422     
423     /** this method checks to see if there are multiple versions of the scala library
424      * @throws Exception */
425     private void checkCorrectVersionsOfScalaLibrary(String requiredScalaVersion) throws Exception {
426         getLog().info("Checking for multiple versions of scala");
427         //TODO - Make sure we handle bad artifacts....
428         // TODO: note that filter does not get applied due to MNG-3236
429             checkArtifactForScalaVersion(requiredScalaVersion, dependencyTreeBuilder.buildDependencyTree( project, localRepository, artifactFactory,
430                     artifactMetadataSource, null, artifactCollector ));
431     }
432 
433 
434     /** Visits a node (and all dependencies) to see if it contains duplicate scala versions */
435     private void checkArtifactForScalaVersion(String requiredScalaVersion, DependencyNode rootNode) throws Exception {
436         final CheckScalaVersionVisitor visitor = new CheckScalaVersionVisitor(requiredScalaVersion, getLog());
437 
438         CollectingDependencyNodeVisitor collectingVisitor = new CollectingDependencyNodeVisitor();
439         DependencyNodeVisitor firstPassVisitor = new FilteringDependencyNodeVisitor( collectingVisitor, createScalaDistroDependencyFilter() );
440         rootNode.accept( firstPassVisitor );
441 
442         DependencyNodeFilter secondPassFilter = new AncestorOrSelfDependencyNodeFilter( collectingVisitor.getNodes() );
443         DependencyNodeVisitor filteredVisitor = new FilteringDependencyNodeVisitor( visitor, secondPassFilter );
444 
445         rootNode.accept( filteredVisitor );
446 
447         if(visitor.isFailed()) {
448             visitor.logScalaDependents();
449             if(failOnMultipleScalaVersions) {
450                 getLog().error("Multiple versions of scala libraries detected!");
451                 throw new MojoFailureException("Multiple versions of scala libraries detected!");
452             }
453             getLog().warn("Multiple versions of scala libraries detected!");
454         }
455     }
456 
457     /**
458      * @return
459      *          A filter to only extract artifacts deployed from scala distributions
460      */
461     private DependencyNodeFilter createScalaDistroDependencyFilter() {
462         List<ArtifactFilter> filters = new ArrayList<ArtifactFilter>();
463         filters.add(new ScalaDistroArtifactFilter());
464         return new AndDependencyNodeFilter(filters);
465     }
466 
467 
468 
469     protected abstract void doExecute() throws Exception;
470 
471 
472     protected JavaMainCaller getScalaCommand() throws Exception {
473         JavaMainCaller cmd = getEmptyScalaCommand(scalaClassName);
474         cmd.addArgs(args);
475         addCompilerPluginOptions(cmd);
476         cmd.addJvmArgs(jvmArgs);
477         return cmd;
478     }
479 
480     protected JavaMainCaller getEmptyScalaCommand(String mainClass) throws Exception {
481         //TODO - Fork or not depending on configuration?
482         JavaMainCaller cmd;
483         if(fork) {
484            // scalac with args in files
485            // * works only since 2.8.0
486            // * is buggy (don't manage space in path on windows)
487             getLog().debug("use java command with args in file forced : " + forceUseArgFile);
488             cmd = new JavaMainCallerByFork(this, mainClass, getToolClasspath(), null, null, forceUseArgFile);
489         } else  {
490             cmd = new JavaMainCallerInProcess(this, mainClass, getToolClasspath(), null, null);
491         }
492         cmd.addJvmArgs("-Xbootclasspath/a:"+ getBootClasspath());
493         return cmd;
494     }
495 
496     private String getToolClasspath() throws Exception {
497         Set<String> classpath = new HashSet<String>();
498         addToClasspath(SCALA_GROUPID, "scala-compiler", findScalaVersion().toString(), classpath);
499 //        addToClasspath(SCALA_GROUPID, "scala-decoder", scalaVersion, classpath);
500 //        addToClasspath(SCALA_GROUPID, "scala-dbc", scalaVersion, classpath);
501         if (dependencies != null) {
502             for(BasicArtifact artifact: dependencies) {
503                 addToClasspath(artifact.groupId, artifact.artifactId, artifact.version, classpath);
504             }
505         }
506         return MainHelper.toMultiPath(classpath.toArray(new String[classpath.size()]));
507     }
508 
509     private String getBootClasspath() throws Exception {
510         Set<String> classpath = new HashSet<String>();
511         addToClasspath(SCALA_GROUPID, SCALA_LIBRARY_ARTIFACTID, findScalaVersion().toString(), classpath);
512         return MainHelper.toMultiPath(classpath.toArray(new String[classpath.size()]));
513     }
514 
515     /**
516      * @return
517      *           This returns whether or not the scala version can support having java sent into the compiler
518      */
519     protected boolean isJavaSupportedByCompiler() throws Exception {
520         return findScalaVersion().compareTo(new VersionNumber("2.7.2")) >= 0;
521     }
522 
523 
524     /**
525      * Adds appropriate compiler plugins to the scalac command.
526      * @param scalac
527      * @throws Exception
528      */
529     protected void addCompilerPluginOptions(JavaMainCaller scalac) throws Exception {
530         for (String plugin : getCompilerPlugins()) {
531             scalac.addArgs("-Xplugin:" + plugin);
532         }
533     }
534     /**
535      * Retrieves a list of paths to scala compiler plugins.
536      *
537      * @return The list of plugins
538      * @throws Exception
539      */
540     private Set<String> getCompilerPlugins() throws Exception {
541         Set<String> plugins = new HashSet<String>();
542         if (compilerPlugins != null) {
543             Set<String> ignoreClasspath = new HashSet<String>();
544             String sv = findScalaVersion().toString();
545             addToClasspath(SCALA_GROUPID, "scala-compiler", sv, ignoreClasspath);
546             addToClasspath(SCALA_GROUPID, SCALA_LIBRARY_ARTIFACTID, sv, ignoreClasspath);
547             for (BasicArtifact artifact : compilerPlugins) {
548                 getLog().info("compiler plugin: " + artifact.toString());
549                 // TODO - Ensure proper scala version for plugins
550                 Set<String> pluginClassPath = new HashSet<String>();
551                 //TODO - Pull in transitive dependencies.
552                 addToClasspath(artifact.groupId, artifact.artifactId, artifact.version, pluginClassPath, false);
553                 pluginClassPath.removeAll(ignoreClasspath);
554                 plugins.addAll(pluginClassPath);
555             }
556         }
557         return plugins;
558     }
559 
560 
561 }