Skip to content

Commit dba2f61

Browse files
committed
Add a bom packaging, trim down consumer pom, add build pom
The purpose of the BOM packaging is to more cleanly identity POM that will be consumed as BOM from POM that will be used as parents. The BOMs can be turned into a consumer POM, whereas parents can not loose any information. The consumer pom is flattened and trimmed down much more (parent, build, modules, properties, distributionManagement, repositories, pluginRepositories, reports, reporting and dependencies (for boms) or dependencyManagement (for other artifacts)).
1 parent c080f64 commit dba2f61

File tree

5 files changed

+220
-53
lines changed

5 files changed

+220
-53
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,46 @@
1+
/*
2+
* Licensed to the Apache Software Foundation (ASF) under one
3+
* or more contributor license agreements. See the NOTICE file
4+
* distributed with this work for additional information
5+
* regarding copyright ownership. The ASF licenses this file
6+
* to you under the Apache License, Version 2.0 (the
7+
* "License"); you may not use this file except in compliance
8+
* with the License. You may obtain a copy of the License at
9+
*
10+
* http://www.apache.org/licenses/LICENSE-2.0
11+
*
12+
* Unless required by applicable law or agreed to in writing,
13+
* software distributed under the License is distributed on an
14+
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
15+
* KIND, either express or implied. See the License for the
16+
* specific language governing permissions and limitations
17+
* under the License.
18+
*/
19+
package org.apache.maven.artifact.handler.providers;
20+
21+
import javax.inject.Inject;
22+
import javax.inject.Named;
23+
import javax.inject.Provider;
24+
import javax.inject.Singleton;
25+
26+
import org.apache.maven.artifact.handler.ArtifactHandler;
27+
import org.apache.maven.artifact.handler.DefaultArtifactHandler;
28+
29+
/**
30+
* {@code pom} artifact handler provider.
31+
*/
32+
@Named("bom")
33+
@Singleton
34+
public class BomArtifactHandlerProvider implements Provider<ArtifactHandler> {
35+
private final ArtifactHandler artifactHandler;
36+
37+
@Inject
38+
public BomArtifactHandlerProvider() {
39+
this.artifactHandler = new DefaultArtifactHandler("pom", null, null, null, null, false, "none", false);
40+
}
41+
42+
@Override
43+
public ArtifactHandler get() {
44+
return artifactHandler;
45+
}
46+
}

maven-core/src/main/java/org/apache/maven/internal/transformation/ConsumerPomArtifactTransformer.java

+125-44
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,7 @@
2424

2525
import java.io.IOException;
2626
import java.io.InputStream;
27+
import java.io.Writer;
2728
import java.nio.file.Files;
2829
import java.nio.file.Path;
2930
import java.nio.file.Paths;
@@ -32,13 +33,17 @@
3233
import java.util.Collection;
3334
import java.util.Set;
3435
import java.util.concurrent.CopyOnWriteArraySet;
35-
import java.util.function.BiConsumer;
36+
import java.util.stream.Collectors;
3637

38+
import org.apache.maven.api.model.Model;
39+
import org.apache.maven.api.model.ModelBase;
40+
import org.apache.maven.api.model.Profile;
3741
import org.apache.maven.feature.Features;
3842
import org.apache.maven.model.building.DefaultBuildPomXMLFilterFactory;
3943
import org.apache.maven.model.building.TransformerContext;
4044
import org.apache.maven.model.transform.RawToConsumerPomXMLFilterFactory;
4145
import org.apache.maven.model.transform.pull.XmlUtils;
46+
import org.apache.maven.model.v4.MavenXpp3Writer;
4247
import org.apache.maven.project.MavenProject;
4348
import org.apache.maven.project.artifact.ProjectArtifact;
4449
import org.codehaus.plexus.util.ReaderFactory;
@@ -64,6 +69,8 @@ public final class ConsumerPomArtifactTransformer {
6469

6570
private static final String CONSUMER_POM_CLASSIFIER = "consumer";
6671

72+
private static final String BUILD_POM_CLASSIFIER = "build";
73+
6774
private final Set<Path> toDelete = new CopyOnWriteArraySet<>();
6875

6976
public void injectTransformedArtifacts(MavenProject project, RepositorySystemSession session) throws IOException {
@@ -72,18 +79,23 @@ public void injectTransformedArtifacts(MavenProject project, RepositorySystemSes
7279
return;
7380
}
7481
if (isActive(session)) {
75-
Path generatedFile;
76-
String buildDirectory =
77-
project.getBuild() != null ? project.getBuild().getDirectory() : null;
78-
if (buildDirectory == null) {
79-
generatedFile = Files.createTempFile(CONSUMER_POM_CLASSIFIER, "pom");
80-
} else {
81-
Path buildDir = Paths.get(buildDirectory);
82+
Path buildDir =
83+
project.getBuild() != null ? Paths.get(project.getBuild().getDirectory()) : null;
84+
if (buildDir != null) {
8285
Files.createDirectories(buildDir);
83-
generatedFile = Files.createTempFile(buildDir, CONSUMER_POM_CLASSIFIER, "pom");
8486
}
85-
deferDeleteFile(generatedFile);
86-
project.addAttachedArtifact(new ConsumerPomArtifact(project, generatedFile, session));
87+
Path build = buildDir != null
88+
? Files.createTempFile(buildDir, BUILD_POM_CLASSIFIER + "-", ".pom")
89+
: Files.createTempFile(BUILD_POM_CLASSIFIER + "-", ".pom");
90+
deferDeleteFile(build);
91+
project.addAttachedArtifact(new BuildPomArtifact(project, build, session));
92+
if (!"pom".equals(project.getPackaging())) {
93+
Path consumer = buildDir != null
94+
? Files.createTempFile(buildDir, CONSUMER_POM_CLASSIFIER + "-", ".pom")
95+
: Files.createTempFile(CONSUMER_POM_CLASSIFIER + "-", ".pom");
96+
deferDeleteFile(consumer);
97+
project.addAttachedArtifact(new ConsumerPomArtifact(project, consumer));
98+
}
8799
} else if (project.getModel().isRoot()) {
88100
throw new IllegalStateException(
89101
"The use of the root attribute on the model requires the buildconsumer feature to be active");
@@ -124,61 +136,130 @@ private boolean isActive(RepositorySystemSession session) {
124136
}
125137

126138
private boolean consumerPomPresent(Collection<Artifact> artifacts) {
127-
return artifacts.stream().anyMatch(a -> CONSUMER_POM_CLASSIFIER.equals(a.getClassifier()));
139+
return artifacts.stream()
140+
.anyMatch(a -> "pom".equals(a.getExtension()) && BUILD_POM_CLASSIFIER.equals(a.getClassifier()));
128141
}
129142

130143
private Collection<Artifact> replacePom(Collection<Artifact> artifacts) {
131-
ArrayList<Artifact> result = new ArrayList<>(artifacts.size());
144+
Artifact consumer = null;
145+
Artifact build = null;
146+
Artifact main = null;
132147
for (Artifact artifact : artifacts) {
133-
if (CONSUMER_POM_CLASSIFIER.equals(artifact.getClassifier())) {
134-
// if under CONSUMER_POM_CLASSIFIER, move it to "" classifier
135-
DefaultArtifact remapped = new DefaultArtifact(
136-
artifact.getGroupId(),
137-
artifact.getArtifactId(),
148+
if ("pom".equals(artifact.getExtension())) {
149+
if (CONSUMER_POM_CLASSIFIER.equals(artifact.getClassifier())) {
150+
consumer = artifact;
151+
} else if (BUILD_POM_CLASSIFIER.equals(artifact.getClassifier())) {
152+
build = artifact;
153+
} else if ("".equals(artifact.getClassifier())) {
154+
main = artifact;
155+
}
156+
}
157+
}
158+
if (main != null) {
159+
if (consumer != null) {
160+
ArrayList<Artifact> result = new ArrayList<>(artifacts);
161+
result.remove(main);
162+
result.remove(consumer);
163+
result.add(new DefaultArtifact(
164+
consumer.getGroupId(),
165+
consumer.getArtifactId(),
138166
"",
139-
artifact.getExtension(),
140-
artifact.getVersion(),
141-
artifact.getProperties(),
142-
artifact.getFile());
143-
result.add(remapped);
144-
} else if ("".equals(artifact.getClassifier())
145-
&& (artifact.getExtension().equals("pom"))
146-
|| artifact.getExtension().startsWith("pom.")) {
147-
// skip POM and POM subordinates
148-
continue;
149-
} else {
150-
// everything else: add as is
151-
result.add(artifact);
167+
consumer.getExtension(),
168+
consumer.getVersion(),
169+
consumer.getProperties(),
170+
consumer.getFile()));
171+
artifacts = result;
172+
} else if (build != null) {
173+
ArrayList<Artifact> result = new ArrayList<>(artifacts);
174+
result.remove(main);
175+
result.remove(build);
176+
result.add(new DefaultArtifact(
177+
build.getGroupId(),
178+
build.getArtifactId(),
179+
"",
180+
build.getExtension(),
181+
build.getVersion(),
182+
build.getProperties(),
183+
build.getFile()));
184+
artifacts = result;
152185
}
153186
}
154-
return result;
187+
return artifacts;
155188
}
156189

157190
/**
158191
* Consumer POM is transformed from original POM.
159192
*/
160193
private static class ConsumerPomArtifact extends TransformedArtifact {
161194

162-
private ConsumerPomArtifact(MavenProject mavenProject, Path target, RepositorySystemSession session) {
195+
private MavenProject project;
196+
197+
private ConsumerPomArtifact(MavenProject mavenProject, Path target) {
163198
super(
164199
new ProjectArtifact(mavenProject),
165200
() -> mavenProject.getFile().toPath(),
166201
CONSUMER_POM_CLASSIFIER,
167202
"pom",
168-
target,
169-
transformer(session));
203+
target);
204+
this.project = mavenProject;
170205
}
171206

172-
private static BiConsumer<Path, Path> transformer(RepositorySystemSession session) {
207+
@Override
208+
public void transform(Path src, Path dest) {
209+
Model model = project.getModel().getDelegate();
210+
boolean isBom = "bom".equals(model.getPackaging());
211+
Model.Builder builder =
212+
prune(Model.newBuilder(model, true).root(false).parent(null), isBom);
213+
builder.profiles(model.getProfiles().stream()
214+
.map(p -> prune(Profile.newBuilder(p, true), isBom).build())
215+
.collect(Collectors.toList()));
216+
217+
Model consumer = builder.build();
218+
219+
try (Writer w = Files.newBufferedWriter(dest)) {
220+
new MavenXpp3Writer().write(w, consumer);
221+
} catch (IOException e) {
222+
throw new RuntimeException(e);
223+
}
224+
}
225+
226+
private <T extends ModelBase.Builder> T prune(T builder, boolean isBom) {
227+
builder.properties(null).reporting(null).pluginRepositories(null).distributionManagement(null);
228+
if (isBom) {
229+
builder.dependencies(null);
230+
} else {
231+
builder.dependencyManagement(null);
232+
}
233+
return builder;
234+
}
235+
}
236+
237+
/**
238+
* Consumer POM is transformed from original POM.
239+
*/
240+
private static class BuildPomArtifact extends TransformedArtifact {
241+
242+
private RepositorySystemSession session;
243+
244+
private BuildPomArtifact(MavenProject mavenProject, Path target, RepositorySystemSession session) {
245+
super(
246+
new ProjectArtifact(mavenProject),
247+
() -> mavenProject.getFile().toPath(),
248+
BUILD_POM_CLASSIFIER,
249+
"pom",
250+
target);
251+
this.session = session;
252+
}
253+
254+
@Override
255+
public void transform(Path src, Path dest) {
173256
TransformerContext context = (TransformerContext) session.getData().get(TransformerContext.KEY);
174-
return (src, dest) -> {
175-
try (InputStream inputStream = transform(src, context)) {
176-
Files.createDirectories(dest.getParent());
177-
Files.copy(inputStream, dest, StandardCopyOption.REPLACE_EXISTING);
178-
} catch (XmlPullParserException | IOException e) {
179-
throw new RuntimeException(e);
180-
}
181-
};
257+
try (InputStream inputStream = ConsumerPomArtifactTransformer.transform(src, context)) {
258+
Files.createDirectories(dest.getParent());
259+
Files.copy(inputStream, dest, StandardCopyOption.REPLACE_EXISTING);
260+
} catch (XmlPullParserException | IOException e) {
261+
throw new RuntimeException(e);
262+
}
182263
}
183264
}
184265

maven-core/src/main/java/org/apache/maven/internal/transformation/TransformedArtifact.java

+4-8
Original file line numberDiff line numberDiff line change
@@ -23,7 +23,6 @@
2323
import java.nio.file.Files;
2424
import java.nio.file.Path;
2525
import java.security.MessageDigest;
26-
import java.util.function.BiConsumer;
2726
import java.util.function.Supplier;
2827

2928
import org.apache.maven.artifact.Artifact;
@@ -42,12 +41,7 @@ abstract class TransformedArtifact extends DefaultArtifact {
4241
private final OnChangeTransformer onChangeTransformer;
4342

4443
TransformedArtifact(
45-
Artifact source,
46-
Supplier<Path> sourcePathProvider,
47-
String classifier,
48-
String extension,
49-
Path targetPath,
50-
BiConsumer<Path, Path> transformerConsumer) {
44+
Artifact source, Supplier<Path> sourcePathProvider, String classifier, String extension, Path targetPath) {
5145
super(
5246
source.getGroupId(),
5347
source.getArtifactId(),
@@ -58,7 +52,7 @@ abstract class TransformedArtifact extends DefaultArtifact {
5852
new TransformedArtifactHandler(
5953
classifier, extension, source.getArtifactHandler().getPackaging()));
6054
this.onChangeTransformer =
61-
new OnChangeTransformer(sourcePathProvider, targetPath, TransformedArtifact::sha1, transformerConsumer);
55+
new OnChangeTransformer(sourcePathProvider, targetPath, TransformedArtifact::sha1, this::transform);
6256
}
6357

6458
@Override
@@ -102,6 +96,8 @@ private static String sha1(Path path) {
10296
}
10397
}
10498

99+
protected abstract void transform(Path src, Path dst);
100+
105101
private static class TransformedArtifactHandler implements ArtifactHandler {
106102
private final String classifier;
107103

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
/*
2+
* Licensed to the Apache Software Foundation (ASF) under one
3+
* or more contributor license agreements. See the NOTICE file
4+
* distributed with this work for additional information
5+
* regarding copyright ownership. The ASF licenses this file
6+
* to you under the Apache License, Version 2.0 (the
7+
* "License"); you may not use this file except in compliance
8+
* with the License. You may obtain a copy of the License at
9+
*
10+
* http://www.apache.org/licenses/LICENSE-2.0
11+
*
12+
* Unless required by applicable law or agreed to in writing,
13+
* software distributed under the License is distributed on an
14+
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
15+
* KIND, either express or implied. See the License for the
16+
* specific language governing permissions and limitations
17+
* under the License.
18+
*/
19+
package org.apache.maven.lifecycle.providers.packaging;
20+
21+
import javax.inject.Inject;
22+
import javax.inject.Named;
23+
import javax.inject.Singleton;
24+
25+
/**
26+
* {@code bom} packaging plugins bindings provider for {@code default} lifecycle.
27+
*/
28+
@Named("bom")
29+
@Singleton
30+
public final class BomLifecycleMappingProvider extends AbstractLifecycleMappingProvider {
31+
// START SNIPPET: bom
32+
@SuppressWarnings("checkstyle:linelength")
33+
private static final String[] BINDINGS = {
34+
"install", "org.apache.maven.plugins:maven-install-plugin:" + INSTALL_PLUGIN_VERSION + ":install",
35+
"deploy", "org.apache.maven.plugins:maven-deploy-plugin:" + DEPLOY_PLUGIN_VERSION + ":deploy"
36+
};
37+
// END SNIPPET: bom
38+
39+
@Inject
40+
public BomLifecycleMappingProvider() {
41+
super(BINDINGS);
42+
}
43+
}

maven-model-builder/src/main/java/org/apache/maven/model/building/DefaultModelBuilder.java

+2-1
Original file line numberDiff line numberDiff line change
@@ -1602,7 +1602,8 @@ private void importDependencyManagement(
16021602
for (Iterator<Dependency> it = depMgmt.getDependencies().iterator(); it.hasNext(); ) {
16031603
Dependency dependency = it.next();
16041604

1605-
if (!"pom".equals(dependency.getType()) || !"import".equals(dependency.getScope())) {
1605+
if (!("pom".equals(dependency.getType()) && "import".equals(dependency.getScope()))
1606+
|| "bom".equals(dependency.getType())) {
16061607
continue;
16071608
}
16081609

0 commit comments

Comments
 (0)