diff --git a/impl/maven-core/src/main/java/org/apache/maven/project/MavenProject.java b/impl/maven-core/src/main/java/org/apache/maven/project/MavenProject.java index e155609de55e..82a3f55c1034 100644 --- a/impl/maven-core/src/main/java/org/apache/maven/project/MavenProject.java +++ b/impl/maven-core/src/main/java/org/apache/maven/project/MavenProject.java @@ -23,6 +23,7 @@ import java.io.Writer; import java.nio.file.Path; import java.util.ArrayList; +import java.util.Collection; import java.util.Collections; import java.util.HashMap; import java.util.LinkedHashMap; @@ -30,11 +31,15 @@ import java.util.List; import java.util.Map; import java.util.Objects; +import java.util.Optional; import java.util.Properties; import java.util.Set; import java.util.function.Predicate; +import java.util.stream.Stream; import org.apache.maven.RepositoryUtils; +import org.apache.maven.api.annotations.Nonnull; +import org.apache.maven.api.annotations.Nullable; import org.apache.maven.artifact.Artifact; import org.apache.maven.artifact.ArtifactUtils; import org.apache.maven.artifact.DependencyResolutionRequiredException; @@ -56,6 +61,7 @@ import org.apache.maven.model.MailingList; import org.apache.maven.model.Model; import org.apache.maven.model.Organization; +import org.apache.maven.model.Parent; import org.apache.maven.model.Plugin; import org.apache.maven.model.PluginExecution; import org.apache.maven.model.PluginManagement; @@ -92,7 +98,6 @@ *

*/ public class MavenProject implements Cloneable { - private static final Logger LOGGER = LoggerFactory.getLogger(MavenProject.class); public static final String EMPTY_PROJECT_GROUP_ID = "unknown"; @@ -101,116 +106,345 @@ public class MavenProject implements Cloneable { public static final String EMPTY_PROJECT_VERSION = "0"; + @Nonnull private Model model; + @Nullable private MavenProject parent; + /** + * The path to the {@code pom.xml} file. By default, this file is located in {@link #basedir}. + */ + @Nullable private File file; + /** + * The project base directory. + * By default, this is the parent of the {@linkplain #getFile() POM file}. + */ + @Nullable private File basedir; + // TODO: what is the difference with {@code basedir}? + @Nullable private Path rootDirectory; + /** + * The artifacts specified by {@code setResolvedArtifacts(Set)}. This is for Maven internal usage only. + * This collection should be handled as if it was immutable, + * because it may be shared by cloned {@code MavenProject} instances. + */ + @Nullable private Set resolvedArtifacts; + /** + * The filter to apply on {@link #resolvedArtifacts} when {@link #artifacts} is lazily populated. + * May be {@code null} to exclude all resolved artifacts. + */ + @Nullable private ArtifactFilter artifactFilter; + /** + * All dependencies that this project has, including transitive ones. + * May be lazily computed from {@link #resolvedArtifacts}. + * This is an immutable collection potentially shared by cloned {@code MavenProject} instances. + */ + @Nullable // Computed when first requested. private Set artifacts; + @Nullable private Artifact parentArtifact; + /** + * The plugin dependencies that this project has. + * This is an immutable collection potentially shared by cloned {@code MavenProject} instances. + */ + @Nonnull private Set pluginArtifacts; + /** + * The artifact repositories of this project. + * This is an immutable collection potentially shared by cloned {@code MavenProject} instances. + */ + @Nonnull private List remoteArtifactRepositories; + /** + * The plugin repositories of this project. + * This is an immutable collection potentially shared by cloned {@code MavenProject} instances. + */ + @Nonnull private List pluginArtifactRepositories; + /** + * Cached value computed from {@link #remoteArtifactRepositories} when first requested. + * This is an immutable collection potentially shared by cloned {@code MavenProject} instances. + */ + @Nullable // Computed when first requested. private List remoteProjectRepositories; + /** + * Cached value computed from {@link #pluginArtifactRepositories} when first requested. + * This is an immutable collection potentially shared by cloned {@code MavenProject} instances. + */ + @Nullable // Computed when first requested. private List remotePluginRepositories; - private List attachedArtifacts = new ArrayList<>(); + /** + * List of the attached artifacts to this project. + * This is a modifiable collection which needs to be copied by {@link #deepCopy()}. + */ + @Nonnull + private List attachedArtifacts; + @Nullable private MavenProject executionProject; + /** + * The collected projects. + * This is an immutable collection potentially shared by cloned {@code MavenProject} instances. + */ + @Nonnull private List collectedProjects; - private List compileSourceRoots = new ArrayList<>(); + /** + * Root directories of source codes to compile. + * This is a modifiable collection which needs to be copied by {@link #deepCopy()}. + */ + @Nonnull + private List compileSourceRoots; - private List testCompileSourceRoots = new ArrayList<>(); + /** + * Root directories of test codes to compile. + * This is a modifiable collection which needs to be copied by {@link #deepCopy()}. + */ + @Nonnull + private List testCompileSourceRoots; - private List scriptSourceRoots = new ArrayList<>(); + /** + * Root directories of scriot codes. + * This is a modifiable collection which needs to be copied by {@link #deepCopy()}. + */ + @Nonnull + private List scriptSourceRoots; + @Nullable private ArtifactRepository releaseArtifactRepository; + @Nullable private ArtifactRepository snapshotArtifactRepository; - private List activeProfiles = new ArrayList<>(); + /** + * The active profiles. + * This is an immutable collection potentially shared by cloned {@code MavenProject} instances. + */ + @Nonnull + private List activeProfiles; - private Map> injectedProfileIds = new LinkedHashMap<>(); + /** + * The identifiers of all profiles that contributed to this project's effective model. + * This is a modifiable collection which needs to be copied by {@link #deepCopy()}. + */ + @Nonnull + private Map> injectedProfileIds; + /** + * Direct dependencies that this project has. + * This is an immutable collection potentially shared by cloned {@code MavenProject} instances. + */ + @Nullable // For compatibility with previous behavior. + @Deprecated private Set dependencyArtifacts; + @Nullable private Artifact artifact; - // calculated. + @Nullable // calculated when first requested. private Map artifactMap; + @Nullable private Model originalModel; + @Nullable // Computed when first requested. private Map pluginArtifactMap; + @Nonnull private Set reportArtifacts; + @Nullable private Map reportArtifactMap; + @Nonnull private Set extensionArtifacts; + @Nullable // Computed when first requested. private Map extensionArtifactMap; + @Nonnull private Map managedVersionMap; - private Map projectReferences = new HashMap<>(); + /** + * Projects by identifiers. + * This is a modifiable collection which needs to be copied by {@link #deepCopy()}. + */ + @Nonnull + private Map projectReferences; private boolean executionRoot; + @Nullable private File parentFile; + /** + * Key-value pairs providing context. + * This is a modifiable collection which needs to be copied by {@link #deepCopy()}. + */ + @Nonnull private Map context; + @Nullable private ClassRealm classRealm; + @Nullable private DependencyFilter extensionDependencyFilter; - private final Set lifecyclePhases = Collections.synchronizedSet(new LinkedHashSet<>()); + /** + * The life cycle phases. + * This is a modifiable collection which needs to be copied by {@link #deepCopy()}. + */ + @Nonnull + private Set lifecyclePhases; + /** + * Creates an initially empty project. + */ public MavenProject() { - Model model = new Model(); - + this(new Model()); model.setGroupId(EMPTY_PROJECT_GROUP_ID); model.setArtifactId(EMPTY_PROJECT_ARTIFACT_ID); model.setVersion(EMPTY_PROJECT_VERSION); - - setModel(model); } public MavenProject(org.apache.maven.api.model.Model model) { this(new Model(model)); } - public MavenProject(Model model) { - setModel(model); + /** + * Creates a project derived from the given model. + * + * @param model the model to wrap in a maven project + */ + public MavenProject(@Nonnull Model model) { + // Do not invoke `setModel(Model)` as escaping `this` is deprecated. + this.model = Objects.requireNonNull(model); + + // Immutable collections. + pluginArtifacts = Set.of(); + remoteArtifactRepositories = List.of(); + pluginArtifactRepositories = List.of(); + collectedProjects = List.of(); + activeProfiles = List.of(); + reportArtifacts = Set.of(); + extensionArtifacts = Set.of(); + managedVersionMap = Map.of(); + + // Mutable collections. + attachedArtifacts = new ArrayList<>(); + compileSourceRoots = new ArrayList<>(); + testCompileSourceRoots = new ArrayList<>(); + scriptSourceRoots = new ArrayList<>(); + injectedProfileIds = new LinkedHashMap<>(); + projectReferences = new HashMap<>(); + context = new HashMap<>(); + lifecyclePhases = Collections.synchronizedSet(new LinkedHashSet<>()); } - public MavenProject(MavenProject project) { - deepCopy(project); + /** + * Creates a copy of the given project. + * + * @param project the project to copy + * @see #clone() + */ + public MavenProject(@Nonnull MavenProject project) { + // Do not invoke setter methods. See "this-escaped" compiler warning. + model = project.getModel(); + parent = project.getParent(); + file = project.getFile(); + basedir = project.getBasedir(); + rootDirectory = project.getRootDirectory(); + artifactFilter = project.artifactFilter; // This internal property has no getter. + parentArtifact = project.getParentArtifact(); + executionProject = project.executionProject; // Intentionally avoid the getter. + releaseArtifactRepository = project.getReleaseArtifactRepository(); + snapshotArtifactRepository = project.getSnapshotArtifactRepository(); + artifact = project.getArtifact(); + originalModel = project.getOriginalModel(); + executionRoot = project.isExecutionRoot(); + parentFile = project.getParentFile(); + classRealm = project.getClassRealm(); + extensionDependencyFilter = project.getExtensionDependencyFilter(); + + // Immutable collections. + resolvedArtifacts = project.resolvedArtifacts; // This internal property has no getter. + artifacts = project.getArtifacts(); + pluginArtifacts = project.getPluginArtifacts(); + remoteArtifactRepositories = project.getRemoteArtifactRepositories(); + pluginArtifactRepositories = project.getPluginArtifactRepositories(); + remoteProjectRepositories = project.getRemoteProjectRepositories(); + remotePluginRepositories = project.getRemotePluginRepositories(); + collectedProjects = project.getCollectedProjects(); + activeProfiles = project.getActiveProfiles(); + dependencyArtifacts = project.getDependencyArtifacts(); + artifactMap = project.getArtifactMap(); + pluginArtifactMap = project.getPluginArtifactMap(); + reportArtifacts = project.getReportArtifacts(); + reportArtifactMap = project.getReportArtifactMap(); + extensionArtifacts = project.getExtensionArtifacts(); + extensionArtifactMap = project.getExtensionArtifactMap(); + managedVersionMap = project.getManagedVersionMap(); + + // Mutable collections. Will be copied by `deepCopy()`. + attachedArtifacts = project.getAttachedArtifacts(); + compileSourceRoots = project.getCompileSourceRoots(); + testCompileSourceRoots = project.getTestCompileSourceRoots(); + scriptSourceRoots = project.getScriptSourceRoots(); + injectedProfileIds = project.getInjectedProfileIds(); // Mutable, but will be copied by `deepCopy()`. + projectReferences = project.getProjectReferences(); + context = project.context; + lifecyclePhases = project.lifecyclePhases; + deepCopy(); } + /** + * Copies in-place the modifiable values of this {@code MavenProject} instance. + * This method should be invoked after a clone or after the copy constructor. + * This method should not invoke any user-overrideable method (e.g., no setter). + */ + private void deepCopy() { + model = model.clone(); + if (originalModel != null) { + originalModel = originalModel.clone(); + } + if (parentFile != null) { + parentFile = parentFile.getAbsoluteFile(); + } + attachedArtifacts = new ArrayList<>(attachedArtifacts); + compileSourceRoots = new ArrayList<>(compileSourceRoots); + testCompileSourceRoots = new ArrayList<>(testCompileSourceRoots); + scriptSourceRoots = new ArrayList<>(scriptSourceRoots); + injectedProfileIds = new LinkedHashMap<>(injectedProfileIds); + projectReferences = new LinkedHashMap<>(projectReferences); + context = new LinkedHashMap<>(context); + lifecyclePhases = Collections.synchronizedSet(new LinkedHashSet<>(lifecyclePhases)); + } + + // TODO: where is the difference with {@code basedir} and {@code rootDirectory}? + @Nullable public File getParentFile() { return parentFile; } - public void setParentFile(File parentFile) { + public void setParentFile(@Nullable File parentFile) { this.parentFile = parentFile; } @@ -218,15 +452,17 @@ public void setParentFile(File parentFile) { // Accessors // ---------------------------------------------------------------------- + @Nullable public Artifact getArtifact() { return artifact; } - public void setArtifact(Artifact artifact) { - this.artifact = artifact; + public void setArtifact(@Nullable Artifact target) { + artifact = target; } // TODO I would like to get rid of this. jvz. + @Nonnull public Model getModel() { return model; } @@ -236,11 +472,12 @@ public Model getModel() { * * @return the parent, or null if no parent is declared or there was an error building it */ + @Nullable public MavenProject getParent() { return parent; } - public void setParent(MavenProject parent) { + public void setParent(@Nullable MavenProject parent) { this.parent = parent; } @@ -248,11 +485,21 @@ public boolean hasParent() { return getParent() != null; } + /** + * {@return the path to the {@code pom.xml} file}. + * By default, this file is located in the {@linkplain #getBasedir() base directory}. + */ + @Nullable public File getFile() { return file; } - public void setFile(File file) { + /** + * Sets the {@code pom.xml} file to the given file, and the base directory to the parent of that file. + * + * @param file the new {@code pom.xml} file, as a file located in the new base directory + */ + public void setFile(@Nullable File file) { this.file = file; this.basedir = file != null ? file.getParentFile() : null; } @@ -260,12 +507,18 @@ public void setFile(File file) { /** * Sets project {@code file} without changing project {@code basedir}. * + * @param file the new {@code pom.xml} file * @since 3.2.4 */ - public void setPomFile(File file) { + public void setPomFile(@Nullable File file) { this.file = file; } + /** + * {@return the project base directory}. + * By default, this is the parent of the {@linkplain #getFile() POM file}. + */ + @Nullable public File getBasedir() { return basedir; } @@ -290,15 +543,14 @@ private void addPath(List paths, String path) { if (path != null) { path = path.trim(); if (!path.isEmpty()) { - File file = new File(path); - if (file.isAbsolute()) { - path = file.getAbsolutePath(); + File f = new File(path); + if (f.isAbsolute()) { + path = f.getAbsolutePath(); } else if (".".equals(path)) { path = getBasedir().getAbsolutePath(); } else { path = new File(getBasedir(), path).getAbsolutePath(); } - if (!paths.contains(path)) { paths.add(path); } @@ -306,20 +558,38 @@ private void addPath(List paths, String path) { } } - public void addCompileSourceRoot(String path) { - addPath(getCompileSourceRoots(), path); + /** + * If the given path is non-null, adds it to the source root directories. Otherwise, does nothing. + * + * @param path the path to add if non-null + */ + public void addCompileSourceRoot(@Nullable String path) { + addPath(compileSourceRoots, path); } - public void addTestCompileSourceRoot(String path) { - addPath(getTestCompileSourceRoots(), path); + /** + * If the given path is non-null, adds it to the test root directories. Otherwise, does nothing. + * + * @param path the path to add if non-null + */ + public void addTestCompileSourceRoot(@Nullable String path) { + addPath(testCompileSourceRoots, path); } + /** + * {@return the source root directories as an unmodifiable list}. + */ + @Nonnull public List getCompileSourceRoots() { - return compileSourceRoots; + return Collections.unmodifiableList(compileSourceRoots); } + /** + * {@return the test root directories as an unmodifiable list}. + */ + @Nonnull public List getTestCompileSourceRoots() { - return testCompileSourceRoots; + return Collections.unmodifiableList(testCompileSourceRoots); } // TODO let the scope handler deal with this @@ -350,18 +620,21 @@ private static boolean isTestPathElement(final String scope) { */ private List getClasspathElements(final Predicate scopeFilter, final boolean includeTestDir) throws DependencyResolutionRequiredException { - final List list = new ArrayList<>(getArtifacts().size() + 2); + @SuppressWarnings("LocalVariableHidesMemberVariable") // Usually the same content as the field. + Set artifacts = getArtifacts(); + final var list = new ArrayList(artifacts.size() + 2); + final var build = getBuild(); if (includeTestDir) { - String d = getBuild().getTestOutputDirectory(); + String d = build.getTestOutputDirectory(); if (d != null) { list.add(d); } } - String d = getBuild().getOutputDirectory(); + String d = build.getOutputDirectory(); if (d != null) { list.add(d); } - for (Artifact a : getArtifacts()) { + for (Artifact a : artifacts) { final File f = a.getFile(); if (f != null && scopeFilter.test(a.getScope())) { final ArtifactHandler h = a.getArtifactHandler(); @@ -374,7 +647,7 @@ private List getClasspathElements(final Predicate scopeFilter, f } /** - * Returns the elements placed on the classpath for compilation. + * {@return the elements placed on the classpath for compilation}. * This method can be invoked when the caller does not support module-path. * * @throws DependencyResolutionRequiredException if an artifact file is used, but has not been resolved @@ -388,7 +661,7 @@ public List getCompileClasspathElements() throws DependencyResolutionReq } /** - * Returns the elements placed on the classpath for tests. + * {@return the elements placed on the classpath for tests}. * This method can be invoked when the caller does not support module-path. * * @throws DependencyResolutionRequiredException if an artifact file is used, but has not been resolved @@ -402,7 +675,7 @@ public List getTestClasspathElements() throws DependencyResolutionRequir } /** - * Returns the elements placed on the classpath for runtime. + * {@return the elements placed on the classpath for runtime}. * This method can be invoked when the caller does not support module-path. * * @throws DependencyResolutionRequiredException if an artifact file is used, but has not been resolved @@ -436,12 +709,15 @@ public void setGroupId(String groupId) { } public String getGroupId() { - String groupId = getModel().getGroupId(); - - if ((groupId == null) && (getModel().getParent() != null)) { - groupId = getModel().getParent().getGroupId(); + @SuppressWarnings("LocalVariableHidesMemberVariable") // Usually the same value as the field. + Model model = getModel(); + String groupId = model.getGroupId(); + if (groupId == null) { + Parent mp = model.getParent(); + if (mp != null) { + groupId = mp.getGroupId(); + } } - return groupId; } @@ -459,8 +735,9 @@ public void setName(String name) { public String getName() { // TODO this should not be allowed to be null. - if (getModel().getName() != null) { - return getModel().getName(); + String name = getModel().getName(); + if (name != null) { + return name; } else { return getArtifactId(); } @@ -471,12 +748,15 @@ public void setVersion(String version) { } public String getVersion() { - String version = getModel().getVersion(); - - if ((version == null) && (getModel().getParent() != null)) { - version = getModel().getParent().getVersion(); + @SuppressWarnings("LocalVariableHidesMemberVariable") // Usually the same value as the field. + Model model = getModel(); + String version = model.getVersion(); + if (version == null) { + Parent mp = getModel().getParent(); + if (mp != null) { + version = mp.getVersion(); + } } - return version; } @@ -628,11 +908,52 @@ public void addLicense(License license) { getModel().addLicense(license); } - public void setArtifacts(Set artifacts) { - this.artifacts = artifacts; + private static Map copyWithSameOrder(Map items) { + switch (items.size()) { + case 0: + return Map.of(); + case 1: + return Map.copyOf(items); + default: + items = new LinkedHashMap<>(items); + if (items.containsKey(null)) { + throw new NullPointerException("The given map shall not contain the null key."); + } + if (items.containsValue(null)) { + throw new NullPointerException("The given map shall not contain the null values."); + } + return Collections.unmodifiableMap(items); + } + } + + private static Set copyWithSameOrder(Collection items) { + switch (items.size()) { + case 0: + return Set.of(); + case 1: + return Set.copyOf(items); + default: + var copy = new LinkedHashSet<>(items); + if (copy.contains(null)) { + throw new NullPointerException("The given set shall not contain the null element."); + } + return Collections.unmodifiableSet(copy); + } + } - // flush the calculated artifactMap - artifactMap = null; + /** + * Sets all dependencies that this project has, including transitive ones. + * The given set is copied: changes to the given set after this method call has no effect on this object. + * This copy is for ensuring that {@link #getArtifactMap()} stay consistent with the values given to this method. + * + *

Side effects

+ * Invoking this method will also modify the values returned by {@link #getArtifactMap()}. + * + * @param artifacts the new project dependencies + */ + public void setArtifacts(@Nonnull Set artifacts) { + this.artifacts = copyWithSameOrder(artifacts); + artifactMap = null; // flush the calculated artifactMap } /** @@ -640,54 +961,71 @@ public void setArtifacts(Set artifacts) { * what phases have run dependencies in some scopes won't be included. e.g. if only compile phase has run, * dependencies with scope test won't be included. * - * @return {@link Set} < {@link Artifact} > - * @see #getDependencyArtifacts() to get only direct dependencies + * @return unmodifiable set of dependencies */ + @Nonnull + @SuppressWarnings("ReturnOfCollectionOrArrayField") // The returned set is unmodifiable. public Set getArtifacts() { if (artifacts == null) { if (artifactFilter == null || resolvedArtifacts == null) { - artifacts = new LinkedHashSet<>(); + artifacts = Set.of(); } else { - artifacts = new LinkedHashSet<>(resolvedArtifacts.size() * 2); - for (Artifact artifact : resolvedArtifacts) { - if (artifactFilter.include(artifact)) { - artifacts.add(artifact); - } - } + artifacts = copyWithSameOrder(resolvedArtifacts.stream() + .filter((a) -> artifactFilter.include(a)) + .toList()); } } return artifacts; } + @Nonnull public Map getArtifactMap() { if (artifactMap == null) { artifactMap = ArtifactUtils.artifactMapByVersionlessId(getArtifacts()); } - return artifactMap; + return Collections.unmodifiableMap(artifactMap); } - public void setPluginArtifacts(Set pluginArtifacts) { - this.pluginArtifacts = pluginArtifacts; - + /** + * Sets all plugins that this project has, including transitive ones. + * The given set is copied: changes to the given set after this method call has no effect on this object. + * This copy is for ensuring that {@link #getPluginArtifactMap()} stay consistent with the values given + * to this method. + * + *

Side effects

+ * Invoking this method will also modify the values returned by {@link #getPluginArtifactMap()}. + * + * @param pluginArtifacts the new project plugins + */ + public void setPluginArtifacts(@Nonnull Set pluginArtifacts) { + this.pluginArtifacts = copyWithSameOrder(pluginArtifacts); this.pluginArtifactMap = null; } + /** + * All plugins that this project has. + * + * @return unmodifiable set of plugins + */ + @Nonnull + @SuppressWarnings("ReturnOfCollectionOrArrayField") // The returned set is unmodifiable. public Set getPluginArtifacts() { return pluginArtifacts; } + @Nonnull public Map getPluginArtifactMap() { if (pluginArtifactMap == null) { pluginArtifactMap = ArtifactUtils.artifactMapByVersionlessId(getPluginArtifacts()); } - - return pluginArtifactMap; + return Collections.unmodifiableMap(pluginArtifactMap); } - public void setParentArtifact(Artifact parentArtifact) { + public void setParentArtifact(@Nullable Artifact parentArtifact) { this.parentArtifact = parentArtifact; } + @Nullable public Artifact getParentArtifact() { return parentArtifact; } @@ -701,101 +1039,141 @@ public List getRepositories() { // ---------------------------------------------------------------------- public List getBuildPlugins() { - if (getModel().getBuild() == null) { + Build build = getModel().getBuild(); + if (build == null) { return Collections.emptyList(); } - return Collections.unmodifiableList(getModel().getBuild().getPlugins()); + return Collections.unmodifiableList(build.getPlugins()); } public List getModules() { - if (!getModel().getDelegate().getSubprojects().isEmpty()) { - return getModel().getDelegate().getSubprojects(); + @SuppressWarnings("LocalVariableHidesMemberVariable") // Usually the same value as the field. + Model model = getModel(); + List subprojects = model.getDelegate().getSubprojects(); + if (!subprojects.isEmpty()) { + return subprojects; } - return getModel().getModules(); + return model.getModules(); } public PluginManagement getPluginManagement() { PluginManagement pluginMgmt = null; - Build build = getModel().getBuild(); if (build != null) { pluginMgmt = build.getPluginManagement(); } - return pluginMgmt; } private Build getModelBuild() { - Build build = getModel().getBuild(); - + @SuppressWarnings("LocalVariableHidesMemberVariable") // Usually the same value as the field. + Model model = getModel(); + Build build = model.getBuild(); if (build == null) { build = new Build(); - - getModel().setBuild(build); + model.setBuild(build); } - return build; } - public void setRemoteArtifactRepositories(List remoteArtifactRepositories) { - this.remoteArtifactRepositories = remoteArtifactRepositories; - this.remoteProjectRepositories = RepositoryUtils.toRepos(getRemoteArtifactRepositories()); + /** + * Sets the artifact repositories of this project. + * The given list is copied: changes to the given list after this method call has no effect on this object. + * This copy is for ensuring that {@link #getRemoteProjectRepositories()} stay consistent with the values + * given to this method. + * + * @param remoteArtifactRepositories the new artifact repositories + */ + public void setRemoteArtifactRepositories(@Nonnull List remoteArtifactRepositories) { + this.remoteArtifactRepositories = List.copyOf(remoteArtifactRepositories); + this.remoteProjectRepositories = null; // Recompute when first requested. } + /** + * {@return the artifact repositories of this project}. The returned list is unmodifiable for ensuring + * that {@link #getRemoteProjectRepositories()} stay consistent with the values returned by this method. + */ + @Nonnull + @SuppressWarnings("ReturnOfCollectionOrArrayField") // The returned list is unmodifiable. public List getRemoteArtifactRepositories() { - if (remoteArtifactRepositories == null) { - remoteArtifactRepositories = new ArrayList<>(); - } - return remoteArtifactRepositories; } - public void setPluginArtifactRepositories(List pluginArtifactRepositories) { - this.pluginArtifactRepositories = pluginArtifactRepositories; - this.remotePluginRepositories = RepositoryUtils.toRepos(getPluginArtifactRepositories()); + /** + * Sets the plugin repositories of this project. + * The given list is copied: changes to the given list after this method call has no effect on this object. + * This copy is for ensuring that {@link #getRemotePluginRepositories()} stay consistent with the values + * given to this method. + * + * @param pluginArtifactRepositories the new artifact repositories + */ + public void setPluginArtifactRepositories(@Nonnull List pluginArtifactRepositories) { + this.pluginArtifactRepositories = List.copyOf(pluginArtifactRepositories); + this.remotePluginRepositories = null; // Recompute when first requested. } /** - * @return a list of ArtifactRepository objects constructed from the Repository objects returned by - * getPluginRepositories. + * {@return the plugin repositories of this project}. The returned list is unmodifiable for ensuring + * that {@link #getRemotePluginRepositories()} stay consistent with the values returned by this method. */ + @Nonnull + @SuppressWarnings("ReturnOfCollectionOrArrayField") // The returned list is unmodifiable. public List getPluginArtifactRepositories() { - if (pluginArtifactRepositories == null) { - pluginArtifactRepositories = new ArrayList<>(); - } - return pluginArtifactRepositories; } public ArtifactRepository getDistributionManagementArtifactRepository() { - return getArtifact().isSnapshot() && (getSnapshotArtifactRepository() != null) - ? getSnapshotArtifactRepository() - : getReleaseArtifactRepository(); + if (getArtifact().isSnapshot()) { + ArtifactRepository sar = getSnapshotArtifactRepository(); + if (sar != null) { + return sar; + } + } + return getReleaseArtifactRepository(); } public List getPluginRepositories() { return getModel().getPluginRepositories(); } + @Nonnull + @SuppressWarnings("ReturnOfCollectionOrArrayField") // The cached list is already unmodifiable. public List getRemoteProjectRepositories() { + if (remoteProjectRepositories == null) { + remoteProjectRepositories = RepositoryUtils.toRepos(getRemoteArtifactRepositories()); + } return remoteProjectRepositories; } + @Nonnull + @SuppressWarnings("ReturnOfCollectionOrArrayField") // The cached list is already unmodifiable. public List getRemotePluginRepositories() { + if (remotePluginRepositories == null) { + remotePluginRepositories = RepositoryUtils.toRepos(getPluginArtifactRepositories()); + } return remotePluginRepositories; } - public void setActiveProfiles(List activeProfiles) { - this.activeProfiles = activeProfiles; + /** + * Sets the active profiles of this project. + * The given list is copied: changes to the given list after this method call has no effect on this object. + * + * @param activeProfiles the new active profiles of this project + */ + public void setActiveProfiles(@Nonnull List activeProfiles) { + this.activeProfiles = List.copyOf(activeProfiles); } + @Nonnull + @SuppressWarnings("ReturnOfCollectionOrArrayField") // The list is already unmodifiable. public List getActiveProfiles() { return activeProfiles; } - public void setInjectedProfileIds(String source, List injectedProfileIds) { + public void setInjectedProfileIds(@Nonnull String source, @Nullable List injectedProfileIds) { + Objects.requireNonNull(source); if (injectedProfileIds != null) { - this.injectedProfileIds.put(source, new ArrayList<>(injectedProfileIds)); + this.injectedProfileIds.put(source, List.copyOf(injectedProfileIds)); } else { this.injectedProfileIds.remove(source); } @@ -808,11 +1186,11 @@ public void setInjectedProfileIds(String source, List injectedProfileIds * {@code ::} for a POM profile or {@code external} for profiles from the * {@code settings.xml}. * - * @return The identifiers of all injected profiles, indexed by the source from which the profiles originated, never - * {@code null}. + * @return The identifiers of all injected profiles, indexed by the source from which the profiles originated */ + @Nonnull public Map> getInjectedProfileIds() { - return this.injectedProfileIds; + return Collections.unmodifiableMap(injectedProfileIds); } /** @@ -826,7 +1204,11 @@ public Map> getInjectedProfileIds() { * @deprecated Please use {@link MavenProjectHelper} * @throws DuplicateArtifactAttachmentException will never happen but leave it for backward compatibility */ - public void addAttachedArtifact(Artifact artifact) throws DuplicateArtifactAttachmentException { + @Deprecated + public void addAttachedArtifact(@Nonnull Artifact artifact) throws DuplicateArtifactAttachmentException { + if (artifact == null) { + return; // While we document this method as non-null, we observe that some callers provide a null value. + } // if already there we remove it and add again int index = attachedArtifacts.indexOf(artifact); if (index >= 0) { @@ -842,22 +1224,19 @@ public void addAttachedArtifact(Artifact artifact) throws DuplicateArtifactAttac * * @return the attached artifacts of this project */ + @Nonnull public List getAttachedArtifacts() { - if (attachedArtifacts == null) { - attachedArtifacts = new ArrayList<>(); - } return Collections.unmodifiableList(attachedArtifacts); } public Xpp3Dom getGoalConfiguration( String pluginGroupId, String pluginArtifactId, String executionId, String goalId) { Xpp3Dom dom = null; - - if (getBuildPlugins() != null) { - for (Plugin plugin : getBuildPlugins()) { + List plugins = getBuildPlugins(); + if (plugins != null) { + for (Plugin plugin : plugins) { if (pluginGroupId.equals(plugin.getGroupId()) && pluginArtifactId.equals(plugin.getArtifactId())) { dom = (Xpp3Dom) plugin.getConfiguration(); - if (executionId != null) { for (PluginExecution execution : plugin.getExecutions()) { if (executionId.equals(execution.getId())) { @@ -871,67 +1250,81 @@ public Xpp3Dom getGoalConfiguration( } } } - if (dom != null) { // make a copy so the original in the POM doesn't get messed with dom = new Xpp3Dom(dom); } - return dom; } + @Nonnull public MavenProject getExecutionProject() { return (executionProject == null ? this : executionProject); } - public void setExecutionProject(MavenProject executionProject) { + public void setExecutionProject(@Nullable MavenProject executionProject) { this.executionProject = executionProject; } + @Nonnull + @SuppressWarnings("ReturnOfCollectionOrArrayField") // The list is already unmodifiable. public List getCollectedProjects() { return collectedProjects; } - public void setCollectedProjects(List collectedProjects) { - this.collectedProjects = collectedProjects; + /** + * Sets the collected project. + * The given list is copied: changes to the given list after this method call has no effect on this object. + * + * @param collectedProjects the collected projects + */ + public void setCollectedProjects(@Nonnull List collectedProjects) { + this.collectedProjects = List.copyOf(collectedProjects); } /** - * Direct dependencies that this project has. + * {@return the direct dependencies that this project has, or null if none}. + * Note that this method returns {@code null} instead of an empty set when there is no dependencies. + * This unusual behavior is kept for compatibility reasons with existing codes, which test for nullity instead + * of emptiness. * - * @return {@link Set} < {@link Artifact} > * @see #getArtifacts() to get all transitive dependencies */ @Deprecated + @Nullable // For compatibility with previous behavior. + @SuppressWarnings("ReturnOfCollectionOrArrayField") // The set is already unmodifiable. public Set getDependencyArtifacts() { return dependencyArtifacts; } @Deprecated - public void setDependencyArtifacts(Set dependencyArtifacts) { - this.dependencyArtifacts = dependencyArtifacts; + public void setDependencyArtifacts(@Nonnull Set dependencyArtifacts) { + this.dependencyArtifacts = copyWithSameOrder(dependencyArtifacts); } - public void setReleaseArtifactRepository(ArtifactRepository releaseArtifactRepository) { + public void setReleaseArtifactRepository(@Nullable ArtifactRepository releaseArtifactRepository) { this.releaseArtifactRepository = releaseArtifactRepository; } - public void setSnapshotArtifactRepository(ArtifactRepository snapshotArtifactRepository) { + public void setSnapshotArtifactRepository(@Nullable ArtifactRepository snapshotArtifactRepository) { this.snapshotArtifactRepository = snapshotArtifactRepository; } - public void setOriginalModel(Model originalModel) { + public void setOriginalModel(@Nullable Model originalModel) { this.originalModel = originalModel; } + @Nullable public Model getOriginalModel() { return originalModel; } - public void setManagedVersionMap(Map map) { - managedVersionMap = map; + public void setManagedVersionMap(@Nonnull Map map) { + managedVersionMap = copyWithSameOrder(map); } + @Nonnull + @SuppressWarnings("ReturnOfCollectionOrArrayField") // The map is already unmodifiable. public Map getManagedVersionMap() { return managedVersionMap; } @@ -943,9 +1336,7 @@ public boolean equals(Object other) { } else if (!(other instanceof MavenProject)) { return false; } - MavenProject that = (MavenProject) other; - return Objects.equals(getArtifactId(), that.getArtifactId()) && Objects.equals(getGroupId(), that.getGroupId()) && Objects.equals(getVersion(), that.getVersion()); @@ -958,14 +1349,16 @@ public int hashCode() { public List getBuildExtensions() { Build build = getBuild(); - if ((build == null) || (build.getExtensions() == null)) { - return Collections.emptyList(); - } else { - return Collections.unmodifiableList(build.getExtensions()); + if (build != null) { + List extensions = build.getExtensions(); + if (extensions != null) { + return Collections.unmodifiableList(extensions); + } } + return Collections.emptyList(); } - public void addProjectReference(MavenProject project) { + public void addProjectReference(@Nonnull MavenProject project) { projectReferences.put( getProjectReferenceId(project.getGroupId(), project.getArtifactId(), project.getVersion()), project); } @@ -978,8 +1371,9 @@ public List getFilters() { return getBuild().getFilters(); } + @Nonnull public Map getProjectReferences() { - return projectReferences; + return Collections.unmodifiableMap(projectReferences); } public boolean isExecutionRoot() { @@ -991,7 +1385,7 @@ public void setExecutionRoot(boolean executionRoot) { } public String getDefaultGoal() { - return getBuild() != null ? getBuild().getDefaultGoal() : null; + return getBuild().getDefaultGoal(); } public Plugin getPlugin(String pluginKey) { @@ -999,27 +1393,25 @@ public Plugin getPlugin(String pluginKey) { } /** - * Default toString + * {@return a string representation for debugging purposes}. */ @Override public String toString() { - StringBuilder sb = new StringBuilder(128); - sb.append("MavenProject: "); - sb.append(getGroupId()); - sb.append(':'); - sb.append(getArtifactId()); - sb.append(':'); - sb.append(getVersion()); - if (getFile() != null) { - sb.append(" @ "); - sb.append(getFile().getPath()); + StringBuilder sb = new StringBuilder(128).append("MavenProject: "); + sb.append(getGroupId()).append(':').append(getArtifactId()).append(':').append(getVersion()); + File f = getFile(); + if (f != null) { + sb.append(" @ ").append(f.getPath()); } - return sb.toString(); } /** + * {@return a deep copy of this object}. + * * @since 2.0.9 + * + * @see #MavenProject(MavenProject) */ @Override public MavenProject clone() { @@ -1027,124 +1419,59 @@ public MavenProject clone() { try { clone = (MavenProject) super.clone(); } catch (CloneNotSupportedException e) { - throw new UnsupportedOperationException(e); + throw new AssertionError(e); // Should never happen since this class is cloneable. } - - clone.deepCopy(this); - + clone.deepCopy(); return clone; } - public void setModel(Model model) { - this.model = model; + public void setModel(@Nonnull Model model) { + this.model = Objects.requireNonNull(model); } - protected void setAttachedArtifacts(List attachedArtifacts) { - this.attachedArtifacts = attachedArtifacts; + /** + * Sets the artifacts attached to this project. + * The given list is copied: changes to the given list after this method call has no effect on this object. + * + * @param attachedArtifacts the new artifacts attached to this project + */ + protected void setAttachedArtifacts(@Nonnull List attachedArtifacts) { + this.attachedArtifacts.clear(); + this.attachedArtifacts.addAll(attachedArtifacts); } - protected void setCompileSourceRoots(List compileSourceRoots) { - this.compileSourceRoots = compileSourceRoots; + /** + * Sets the source root directories of this project. + * The given list is copied: changes to the given list after this method call has no effect on this object. + * + * @param compileSourceRoots the new source root directories + */ + protected void setCompileSourceRoots(@Nonnull List compileSourceRoots) { + this.compileSourceRoots.clear(); + this.compileSourceRoots.addAll(compileSourceRoots); } - protected void setTestCompileSourceRoots(List testCompileSourceRoots) { - this.testCompileSourceRoots = testCompileSourceRoots; + /** + * Sets the test source directories of this project. + * The given list is copied: changes to the given list after this method call has no effect on this object. + * + * @param testCompileSourceRoots the new test source directories + */ + protected void setTestCompileSourceRoots(@Nonnull List testCompileSourceRoots) { + this.testCompileSourceRoots.clear(); + this.testCompileSourceRoots.addAll(testCompileSourceRoots); } + @Nullable protected ArtifactRepository getReleaseArtifactRepository() { return releaseArtifactRepository; } + @Nullable protected ArtifactRepository getSnapshotArtifactRepository() { return snapshotArtifactRepository; } - private void deepCopy(MavenProject project) { - // disown the parent - - // copy fields - file = project.file; - basedir = project.basedir; - - // don't need a deep copy, they don't get modified or added/removed to/from - but make them unmodifiable to be - // sure! - if (project.getDependencyArtifacts() != null) { - setDependencyArtifacts(Collections.unmodifiableSet(project.getDependencyArtifacts())); - } - - if (project.getArtifacts() != null) { - setArtifacts(Collections.unmodifiableSet(project.getArtifacts())); - } - - if (project.getParentFile() != null) { - parentFile = new File(project.getParentFile().getAbsolutePath()); - } - - if (project.getPluginArtifacts() != null) { - setPluginArtifacts(Collections.unmodifiableSet(project.getPluginArtifacts())); - } - - if (project.getReportArtifacts() != null) { - setReportArtifacts(Collections.unmodifiableSet(project.getReportArtifacts())); - } - - if (project.getExtensionArtifacts() != null) { - setExtensionArtifacts(Collections.unmodifiableSet(project.getExtensionArtifacts())); - } - - setParentArtifact((project.getParentArtifact())); - - if (project.getRemoteArtifactRepositories() != null) { - setRemoteArtifactRepositories(Collections.unmodifiableList(project.getRemoteArtifactRepositories())); - } - - if (project.getPluginArtifactRepositories() != null) { - setPluginArtifactRepositories(Collections.unmodifiableList(project.getPluginArtifactRepositories())); - } - - if (project.getActiveProfiles() != null) { - setActiveProfiles((Collections.unmodifiableList(project.getActiveProfiles()))); - } - - if (project.getAttachedArtifacts() != null) { - // clone properties modifiable by plugins in a forked lifecycle - setAttachedArtifacts(new ArrayList<>(project.getAttachedArtifacts())); - } - - if (project.getCompileSourceRoots() != null) { - // clone source roots - setCompileSourceRoots((new ArrayList<>(project.getCompileSourceRoots()))); - } - - if (project.getTestCompileSourceRoots() != null) { - setTestCompileSourceRoots((new ArrayList<>(project.getTestCompileSourceRoots()))); - } - - if (project.getScriptSourceRoots() != null) { - setScriptSourceRoots((new ArrayList<>(project.getScriptSourceRoots()))); - } - - if (project.getModel() != null) { - setModel(project.getModel().clone()); - } - - if (project.getOriginalModel() != null) { - setOriginalModel(project.getOriginalModel()); - } - - setExecutionRoot(project.isExecutionRoot()); - - if (project.getArtifact() != null) { - setArtifact(ArtifactUtils.copyArtifact(project.getArtifact())); - } - - if (project.getManagedVersionMap() != null) { - setManagedVersionMap(project.getManagedVersionMap()); - } - - lifecyclePhases.addAll(project.lifecyclePhases); - } - private static String getProjectReferenceId(String groupId, String artifactId, String version) { StringBuilder buffer = new StringBuilder(128); buffer.append(groupId).append(':').append(artifactId).append(':').append(version); @@ -1153,13 +1480,14 @@ private static String getProjectReferenceId(String groupId, String artifactId, S /** * Sets the value of the context value of this project identified by the given key. If the supplied value is - * null, the context value is removed from this project. Context values are intended to allow core + * {@code null}, the context value is removed from this project. Context values are intended to allow core * extensions to associate derived state with project instances. + * + * @param key the key to associate to a value + * @param value the value to associate to the given key, or {@code null} for removing the association */ - public void setContextValue(String key, Object value) { - if (context == null) { - context = new HashMap<>(); - } + public void setContextValue(@Nonnull String key, @Nullable Object value) { + Objects.requireNonNull(key); if (value != null) { context.put(key, value); } else { @@ -1169,12 +1497,12 @@ public void setContextValue(String key, Object value) { /** * Returns context value of this project associated with the given key or null if this project has no such value. + * + * @param key the key of the value to get + * @return the associated value, or {@code null} if none */ public Object getContextValue(String key) { - if (context == null) { - return null; - } - return context.get(key); + return context.get(Objects.requireNonNull(key)); } /** @@ -1183,6 +1511,7 @@ public Object getContextValue(String key) { * without prior notice and must not be used by plugins. * * @param classRealm The class realm hosting the build extensions of this project, may be {@code null}. + * @hidden */ public void setClassRealm(ClassRealm classRealm) { this.classRealm = classRealm; @@ -1195,6 +1524,7 @@ public void setClassRealm(ClassRealm classRealm) { * used by plugins. * * @return The project's class realm or {@code null}. + * @hidden */ public ClassRealm getClassRealm() { return classRealm; @@ -1206,6 +1536,7 @@ public ClassRealm getClassRealm() { * In particular, this method can be changed or deleted without prior notice and must not be used by plugins. * * @param extensionDependencyFilter The dependency filter to apply to plugins, may be {@code null}. + * @hidden */ public void setExtensionDependencyFilter(DependencyFilter extensionDependencyFilter) { this.extensionDependencyFilter = extensionDependencyFilter; @@ -1218,6 +1549,7 @@ public void setExtensionDependencyFilter(DependencyFilter extensionDependencyFil * used by plugins. * * @return The dependency filter or {@code null}. + * @hidden */ public DependencyFilter getExtensionDependencyFilter() { return extensionDependencyFilter; @@ -1229,10 +1561,11 @@ public DependencyFilter getExtensionDependencyFilter() { * part of the public API. In particular, this method can be changed or deleted without prior notice and must not be * used by plugins. * - * @param artifacts The set of artifacts, may be {@code null}. + * @param artifacts the set of artifacts, may be {@code null} + * @hidden */ - public void setResolvedArtifacts(Set artifacts) { - this.resolvedArtifacts = (artifacts != null) ? artifacts : Collections.emptySet(); + public void setResolvedArtifacts(@Nullable Set artifacts) { + this.resolvedArtifacts = copyWithSameOrder(artifacts); this.artifacts = null; this.artifactMap = null; } @@ -1243,9 +1576,10 @@ public void setResolvedArtifacts(Set artifacts) { * part of the public API. In particular, this method can be changed or deleted without prior notice and must not be * used by plugins. * - * @param artifactFilter The artifact filter, may be {@code null} to exclude all artifacts. + * @param artifactFilter the artifact filter, may be {@code null} to exclude all artifacts + * @hidden */ - public void setArtifactFilter(ArtifactFilter artifactFilter) { + public void setArtifactFilter(@Nullable ArtifactFilter artifactFilter) { this.artifactFilter = artifactFilter; this.artifacts = null; this.artifactMap = null; @@ -1258,6 +1592,7 @@ public void setArtifactFilter(ArtifactFilter artifactFilter) { * * @param phase The phase to check for, must not be {@code null}. * @return {@code true} if the phase has been seen. + * @hidden */ public boolean hasLifecyclePhase(String phase) { return lifecyclePhases.contains(phase); @@ -1269,6 +1604,7 @@ public boolean hasLifecyclePhase(String phase) { * used by plugins. * * @param lifecyclePhase The lifecycle phase to add, must not be {@code null}. + * @hidden */ public void addLifecyclePhase(String lifecyclePhase) { lifecyclePhases.add(lifecyclePhase); @@ -1295,45 +1631,33 @@ public String getModulePathAdjustment(MavenProject moduleProject) throws IOExcep // FIXME: This is hacky. What if module directory doesn't match artifactid, and parent // is coming from the repository?? String module = moduleProject.getArtifactId(); - File moduleFile = moduleProject.getFile(); - if (moduleFile != null) { File moduleDir = moduleFile.getCanonicalFile().getParentFile(); - module = moduleDir.getName(); } - if (moduleAdjustments == null) { moduleAdjustments = new HashMap<>(); - List modules = getModules(); if (modules != null) { for (String modulePath : modules) { String moduleName = modulePath; - if (moduleName.endsWith("/") || moduleName.endsWith("\\")) { moduleName = moduleName.substring(0, moduleName.length() - 1); } - int lastSlash = moduleName.lastIndexOf('/'); - if (lastSlash < 0) { lastSlash = moduleName.lastIndexOf('\\'); } - String adjustment = null; - if (lastSlash > -1) { moduleName = moduleName.substring(lastSlash + 1); adjustment = modulePath.substring(0, lastSlash); } - moduleAdjustments.put(moduleName, adjustment); } } } - return moduleAdjustments.get(module); } @@ -1344,216 +1668,138 @@ public Set createArtifacts(ArtifactFactory artifactFactory, String inh artifactFactory, getModel().getDependencies(), inheritedScope, filter, this); } + /** + * Sets the test script directories of this project. + * The given list is copied: changes to the given list after this method call has no effect on this object. + * + * @param scriptSourceRoots the new test script directories + */ @Deprecated - protected void setScriptSourceRoots(List scriptSourceRoots) { - this.scriptSourceRoots = scriptSourceRoots; + protected void setScriptSourceRoots(@Nonnull List scriptSourceRoots) { + this.scriptSourceRoots.clear(); + this.scriptSourceRoots.addAll(scriptSourceRoots); } @Deprecated - public void addScriptSourceRoot(String path) { - if (path != null) { - path = path.trim(); - if (path.length() != 0) { - if (!getScriptSourceRoots().contains(path)) { - getScriptSourceRoots().add(path); - } - } - } + public void addScriptSourceRoot(@Nullable String path) { + addPath(scriptSourceRoots, path); } + @Nonnull @Deprecated public List getScriptSourceRoots() { - return scriptSourceRoots; + return Collections.unmodifiableList(scriptSourceRoots); } + @Nonnull @Deprecated public List getCompileArtifacts() { - List list = new ArrayList<>(getArtifacts().size()); - - for (Artifact a : getArtifacts()) { - // TODO classpath check doesn't belong here - that's the other method - if (a.getArtifactHandler().isAddedToClasspath()) { - // TODO let the scope handler deal with this - if (isCompilePathElement(a.getScope())) { - list.add(a); - } - } - } - return list; + return getClasspathArtifacts(MavenProject::isCompilePathElement); } + @Nonnull @Deprecated public List getCompileDependencies() { - Set artifacts = getArtifacts(); - - if ((artifacts == null) || artifacts.isEmpty()) { - return Collections.emptyList(); - } - - List list = new ArrayList<>(artifacts.size()); - - for (Artifact a : getArtifacts()) { - // TODO let the scope handler deal with this - if (isCompilePathElement(a.getScope())) { - Dependency dependency = new Dependency(); - - dependency.setArtifactId(a.getArtifactId()); - dependency.setGroupId(a.getGroupId()); - dependency.setVersion(a.getVersion()); - dependency.setScope(a.getScope()); - dependency.setType(a.getType()); - dependency.setClassifier(a.getClassifier()); - - list.add(dependency); - } - } - return Collections.unmodifiableList(list); + return getDependencies(MavenProject::isCompilePathElement); } + @Nonnull @Deprecated public List getTestArtifacts() { - List list = new ArrayList<>(getArtifacts().size()); - - for (Artifact a : getArtifacts()) { - // TODO classpath check doesn't belong here - that's the other method - if (a.getArtifactHandler().isAddedToClasspath()) { - list.add(a); - } - } - return list; + return getClasspathArtifacts(MavenProject::isAddedToClasspath); } + @Nonnull @Deprecated public List getTestDependencies() { - Set artifacts = getArtifacts(); - - if ((artifacts == null) || artifacts.isEmpty()) { - return Collections.emptyList(); - } - - List list = new ArrayList<>(artifacts.size()); - - for (Artifact a : getArtifacts()) { - Dependency dependency = new Dependency(); - - dependency.setArtifactId(a.getArtifactId()); - dependency.setGroupId(a.getGroupId()); - dependency.setVersion(a.getVersion()); - dependency.setScope(a.getScope()); - dependency.setType(a.getType()); - dependency.setClassifier(a.getClassifier()); - - list.add(dependency); - } - return Collections.unmodifiableList(list); + return getDependencies(MavenProject::isTestPathElement); } + @Nonnull @Deprecated // used by the Maven ITs public List getRuntimeDependencies() { - Set artifacts = getArtifacts(); - - if ((artifacts == null) || artifacts.isEmpty()) { - return Collections.emptyList(); - } - - List list = new ArrayList<>(artifacts.size()); - - for (Artifact a : getArtifacts()) { - // TODO let the scope handler deal with this - if (isRuntimePathElement(a.getScope())) { - Dependency dependency = new Dependency(); - - dependency.setArtifactId(a.getArtifactId()); - dependency.setGroupId(a.getGroupId()); - dependency.setVersion(a.getVersion()); - dependency.setScope(a.getScope()); - dependency.setType(a.getType()); - dependency.setClassifier(a.getClassifier()); - - list.add(dependency); - } - } - return Collections.unmodifiableList(list); + return getDependencies(MavenProject::isRuntimePathElement); } + @Nonnull @Deprecated public List getRuntimeArtifacts() { - List list = new ArrayList<>(getArtifacts().size()); - - for (Artifact a : getArtifacts()) { - // TODO classpath check doesn't belong here - that's the other method - if (a.getArtifactHandler().isAddedToClasspath() && isRuntimePathElement(a.getScope())) { - list.add(a); - } - } - return list; + return getClasspathArtifacts(MavenProject::isRuntimePathElement); } + @Nonnull @Deprecated public List getSystemClasspathElements() throws DependencyResolutionRequiredException { - List list = new ArrayList<>(getArtifacts().size()); - - String d = getBuild().getOutputDirectory(); - if (d != null) { - list.add(d); - } - - for (Artifact a : getArtifacts()) { - if (a.getArtifactHandler().isAddedToClasspath()) { - // TODO let the scope handler deal with this - if (Artifact.SCOPE_SYSTEM.equals(a.getScope())) { - File f = a.getFile(); - if (f != null) { - list.add(f.getPath()); - } - } - } - } - return list; + Stream s1 = Optional.ofNullable(getBuild().getOutputDirectory()).stream(); + Stream s2 = getArtifacts().stream() + .filter(MavenProject::isAddedToClasspath) + .filter(MavenProject::isScopeSystem) + .map((Artifact::getFile)) + .filter(Objects::nonNull) + .map(File::getPath); + return Stream.concat(s1, s2).toList(); } + @Nonnull @Deprecated public List getSystemArtifacts() { - List list = new ArrayList<>(getArtifacts().size()); - - for (Artifact a : getArtifacts()) { - // TODO classpath check doesn't belong here - that's the other method - if (a.getArtifactHandler().isAddedToClasspath()) { - // TODO let the scope handler deal with this - if (Artifact.SCOPE_SYSTEM.equals(a.getScope())) { - list.add(a); - } - } - } - return list; + return getClasspathArtifacts(MavenProject::isScopeSystem); } + @Nonnull @Deprecated public List getSystemDependencies() { - Set artifacts = getArtifacts(); + return getDependencies(MavenProject::isScopeSystem); + } - if ((artifacts == null) || artifacts.isEmpty()) { - return Collections.emptyList(); - } + // TODO let the scope handler deal with this + private static boolean isCompilePathElement(Artifact a) { + return isCompilePathElement(a.getScope()); + } - List list = new ArrayList<>(artifacts.size()); + // TODO let the scope handler deal with this + private static boolean isRuntimePathElement(Artifact a) { + return isRuntimePathElement(a.getScope()); + } - for (Artifact a : getArtifacts()) { - // TODO let the scope handler deal with this - if (Artifact.SCOPE_SYSTEM.equals(a.getScope())) { - Dependency dependency = new Dependency(); + // TODO let the scope handler deal with this + private static boolean isTestPathElement(Artifact a) { + return isTestPathElement(a.getScope()); + } - dependency.setArtifactId(a.getArtifactId()); - dependency.setGroupId(a.getGroupId()); - dependency.setVersion(a.getVersion()); - dependency.setScope(a.getScope()); - dependency.setType(a.getType()); - dependency.setClassifier(a.getClassifier()); + // TODO let the scope handler deal with this + private static boolean isScopeSystem(Artifact a) { + return Artifact.SCOPE_SYSTEM.equals(a.getScope()); + } - list.add(dependency); - } - } - return Collections.unmodifiableList(list); + // TODO classpath check doesn't belong here - that's the other method + private static boolean isAddedToClasspath(Artifact a) { + return a.getArtifactHandler().isAddedToClasspath(); + } + + private List getClasspathArtifacts(final Predicate filter) { + return getArtifacts().stream() + .filter(MavenProject::isAddedToClasspath) + .filter(filter) + .toList(); + } + + private List getDependencies(final Predicate filter) { + return getArtifacts().stream() + .filter(filter) + .map(MavenProject::toDependency) + .toList(); + } + + private static Dependency toDependency(Artifact a) { + Dependency dependency = new Dependency(); + dependency.setArtifactId(a.getArtifactId()); + dependency.setGroupId(a.getGroupId()); + dependency.setVersion(a.getVersion()); + dependency.setScope(a.getScope()); + dependency.setType(a.getType()); + dependency.setClassifier(a.getClassifier()); + return dependency; } @Deprecated @@ -1567,45 +1813,45 @@ public Reporting getReporting() { } @Deprecated - public void setReportArtifacts(Set reportArtifacts) { - this.reportArtifacts = reportArtifacts; - + public void setReportArtifacts(@Nonnull Set reportArtifacts) { + this.reportArtifacts = copyWithSameOrder(reportArtifacts); reportArtifactMap = null; } @Deprecated public Set getReportArtifacts() { - return reportArtifacts; + return Collections.unmodifiableSet(reportArtifacts); } + @Nonnull @Deprecated public Map getReportArtifactMap() { if (reportArtifactMap == null) { reportArtifactMap = ArtifactUtils.artifactMapByVersionlessId(getReportArtifacts()); } - - return reportArtifactMap; + return Collections.unmodifiableMap(reportArtifactMap); } @Deprecated - public void setExtensionArtifacts(Set extensionArtifacts) { - this.extensionArtifacts = extensionArtifacts; - + public void setExtensionArtifacts(@Nonnull Set extensionArtifacts) { + this.extensionArtifacts = copyWithSameOrder(extensionArtifacts); extensionArtifactMap = null; } + @Nonnull @Deprecated + @SuppressWarnings("ReturnOfCollectionOrArrayField") // The set is already unmodifiable. public Set getExtensionArtifacts() { return extensionArtifacts; } + @Nonnull @Deprecated public Map getExtensionArtifactMap() { if (extensionArtifactMap == null) { extensionArtifactMap = ArtifactUtils.artifactMapByVersionlessId(getExtensionArtifacts()); } - - return extensionArtifactMap; + return Collections.unmodifiableMap(extensionArtifactMap); } @Deprecated @@ -1619,18 +1865,15 @@ public List getReportPlugins() { @Deprecated public Xpp3Dom getReportConfiguration(String pluginGroupId, String pluginArtifactId, String reportSetId) { Xpp3Dom dom = null; - // ---------------------------------------------------------------------- // I would like to be able to look up the Mojo object using a key but // we have a limitation in modello that will be remedied shortly. So // for now I have to iterate through and see what we have. // ---------------------------------------------------------------------- - if (getReportPlugins() != null) { for (ReportPlugin plugin : getReportPlugins()) { if (pluginGroupId.equals(plugin.getGroupId()) && pluginArtifactId.equals(plugin.getArtifactId())) { dom = (Xpp3Dom) plugin.getConfiguration(); - if (reportSetId != null) { ReportSet reportSet = plugin.getReportSetsAsMap().get(reportSetId); if (reportSet != null) { @@ -1645,12 +1888,10 @@ public Xpp3Dom getReportConfiguration(String pluginGroupId, String pluginArtifac } } } - if (dom != null) { // make a copy so the original in the POM doesn't get messed with dom = new Xpp3Dom(dom); } - return dom; } @@ -1702,17 +1943,19 @@ public ProjectBuildingRequest getProjectBuildingRequest() { * @param projectBuildingRequest The project building request, may be {@code null}. * @since 2.1 */ - // used by maven-dependency-tree - @Deprecated + @Deprecated // used by maven-dependency-tree public void setProjectBuildingRequest(ProjectBuildingRequest projectBuildingRequest) { this.projectBuilderConfiguration = projectBuildingRequest; } /** + * TODO: what is the difference with {@code basedir}? + * + * @return the root directory for this project + * @throws IllegalStateException if the root directory cannot be found * @since 4.0.0 - * @return the rootDirectory for this project - * @throws IllegalStateException if the rootDirectory cannot be found */ + @Nonnull public Path getRootDirectory() { if (rootDirectory == null) { throw new IllegalStateException(RootLocator.UNABLE_TO_FIND_ROOT_PROJECT_MESSAGE); @@ -1720,6 +1963,7 @@ public Path getRootDirectory() { return rootDirectory; } + @Nullable public void setRootDirectory(Path rootDirectory) { this.rootDirectory = rootDirectory; } diff --git a/impl/maven-core/src/test/java/org/apache/maven/project/MavenProjectTest.java b/impl/maven-core/src/test/java/org/apache/maven/project/MavenProjectTest.java index 402fd3014fcb..639b943415b5 100644 --- a/impl/maven-core/src/test/java/org/apache/maven/project/MavenProjectTest.java +++ b/impl/maven-core/src/test/java/org/apache/maven/project/MavenProjectTest.java @@ -32,7 +32,6 @@ import static org.junit.jupiter.api.Assertions.assertEquals; import static org.junit.jupiter.api.Assertions.assertFalse; import static org.junit.jupiter.api.Assertions.assertNotNull; -import static org.junit.jupiter.api.Assertions.assertNotSame; import static org.junit.jupiter.api.Assertions.assertTrue; class MavenProjectTest extends AbstractMavenProjectTestCase { @@ -173,10 +172,8 @@ void testCloneWithActiveProfile() throws Exception { assertEquals(1, activeProfilesClone.size(), "Expecting 1 active profile"); - assertNotSame( - activeProfilesOrig, - activeProfilesClone, - "The list of active profiles should have been cloned too but is same"); + // Note that the lists may be the same instance when unmodifiable. + assertEquals(activeProfilesOrig, activeProfilesClone); } @Test