[MNG-7622] Maven Transformation and Consumer POM (#907)

Maven Consumer POM redone, it happens only in "maven3 realm" (before resolver), and allows use cases like m-gog-p and checksum-m-p work as before.

Key aspects:
* consumer POM is injected to build earliest possible as attached artifact
* it lives and is visible just like any other attached artifact (so m-gpg-p can process it)
* just before the install/deploy, they are "swapped out" to replace POM along with all "extras" it may have (checksum, signature)
* to support use cases like MNG-7067 (in memory model is changed, but not POM file), OnChangeTransformer could be extended to take into account both: file content and model content.

---

https://issues.apache.org/jira/browse/MNG-7622
This commit is contained in:
Tamas Cservenak 2023-01-25 11:27:00 +01:00 committed by GitHub
parent 74548dde8e
commit 8918c8144f
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
13 changed files with 568 additions and 120 deletions

View File

@ -1,47 +0,0 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package org.apache.maven.internal.aether;
import java.io.IOException;
import java.io.InputStream;
import java.nio.file.Files;
import java.nio.file.Path;
import org.apache.maven.model.building.DefaultBuildPomXMLFilterFactory;
import org.apache.maven.model.building.TransformerContext;
import org.apache.maven.model.transform.RawToConsumerPomXMLFilterFactory;
import org.apache.maven.model.transform.pull.XmlUtils;
import org.codehaus.plexus.util.ReaderFactory;
import org.codehaus.plexus.util.xml.XmlStreamReader;
import org.codehaus.plexus.util.xml.pull.EntityReplacementMap;
import org.codehaus.plexus.util.xml.pull.MXParser;
import org.codehaus.plexus.util.xml.pull.XmlPullParser;
import org.codehaus.plexus.util.xml.pull.XmlPullParserException;
class ConsumerModelSourceTransformer {
public InputStream transform(Path pomFile, TransformerContext context) throws IOException, XmlPullParserException {
XmlStreamReader reader = ReaderFactory.newXmlReader(Files.newInputStream(pomFile));
XmlPullParser parser = new MXParser(EntityReplacementMap.defaultEntityReplacementMap);
parser.setInput(reader);
parser = new RawToConsumerPomXMLFilterFactory(new DefaultBuildPomXMLFilterFactory(context, true))
.get(parser, pomFile);
return XmlUtils.writeDocument(reader, parser);
}
}

View File

@ -21,13 +21,8 @@ package org.apache.maven.internal.aether;
import javax.inject.Inject; import javax.inject.Inject;
import javax.inject.Named; import javax.inject.Named;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Arrays; import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap; import java.util.HashMap;
import java.util.HashSet; import java.util.HashSet;
import java.util.LinkedHashMap; import java.util.LinkedHashMap;
@ -41,11 +36,9 @@ import org.apache.maven.artifact.handler.manager.ArtifactHandlerManager;
import org.apache.maven.bridge.MavenRepositorySystem; import org.apache.maven.bridge.MavenRepositorySystem;
import org.apache.maven.eventspy.internal.EventSpyDispatcher; import org.apache.maven.eventspy.internal.EventSpyDispatcher;
import org.apache.maven.execution.MavenExecutionRequest; import org.apache.maven.execution.MavenExecutionRequest;
import org.apache.maven.feature.Features;
import org.apache.maven.internal.xml.XmlNodeImpl; import org.apache.maven.internal.xml.XmlNodeImpl;
import org.apache.maven.internal.xml.XmlPlexusConfiguration; import org.apache.maven.internal.xml.XmlPlexusConfiguration;
import org.apache.maven.model.ModelBase; import org.apache.maven.model.ModelBase;
import org.apache.maven.model.building.TransformerContext;
import org.apache.maven.repository.internal.MavenRepositorySystemUtils; import org.apache.maven.repository.internal.MavenRepositorySystemUtils;
import org.apache.maven.rtinfo.RuntimeInformation; import org.apache.maven.rtinfo.RuntimeInformation;
import org.apache.maven.settings.Mirror; import org.apache.maven.settings.Mirror;
@ -56,12 +49,9 @@ import org.apache.maven.settings.crypto.DefaultSettingsDecryptionRequest;
import org.apache.maven.settings.crypto.SettingsDecrypter; import org.apache.maven.settings.crypto.SettingsDecrypter;
import org.apache.maven.settings.crypto.SettingsDecryptionResult; import org.apache.maven.settings.crypto.SettingsDecryptionResult;
import org.codehaus.plexus.configuration.PlexusConfiguration; import org.codehaus.plexus.configuration.PlexusConfiguration;
import org.codehaus.plexus.util.xml.pull.XmlPullParserException;
import org.eclipse.aether.ConfigurationProperties; import org.eclipse.aether.ConfigurationProperties;
import org.eclipse.aether.DefaultRepositorySystemSession; import org.eclipse.aether.DefaultRepositorySystemSession;
import org.eclipse.aether.RepositorySystem; import org.eclipse.aether.RepositorySystem;
import org.eclipse.aether.SessionData;
import org.eclipse.aether.artifact.Artifact;
import org.eclipse.aether.repository.LocalRepository; import org.eclipse.aether.repository.LocalRepository;
import org.eclipse.aether.repository.LocalRepositoryManager; import org.eclipse.aether.repository.LocalRepositoryManager;
import org.eclipse.aether.repository.NoLocalRepositoryManagerException; import org.eclipse.aether.repository.NoLocalRepositoryManagerException;
@ -69,8 +59,6 @@ import org.eclipse.aether.repository.RepositoryPolicy;
import org.eclipse.aether.repository.WorkspaceReader; import org.eclipse.aether.repository.WorkspaceReader;
import org.eclipse.aether.resolution.ResolutionErrorPolicy; import org.eclipse.aether.resolution.ResolutionErrorPolicy;
import org.eclipse.aether.spi.localrepo.LocalRepositoryManagerFactory; import org.eclipse.aether.spi.localrepo.LocalRepositoryManagerFactory;
import org.eclipse.aether.transform.FileTransformer;
import org.eclipse.aether.transform.TransformException;
import org.eclipse.aether.util.ConfigUtils; import org.eclipse.aether.util.ConfigUtils;
import org.eclipse.aether.util.listener.ChainedRepositoryListener; import org.eclipse.aether.util.listener.ChainedRepositoryListener;
import org.eclipse.aether.util.repository.AuthenticationBuilder; import org.eclipse.aether.util.repository.AuthenticationBuilder;
@ -377,10 +365,6 @@ public class DefaultRepositorySystemSessionFactory {
setUpLocalRepositoryManager(request, session); setUpLocalRepositoryManager(request, session);
if (Features.buildConsumer(request.getUserProperties()).isActive()) {
session.setFileTransformerManager(a -> getTransformersForArtifact(a, session.getData()));
}
return session; return session;
} }
@ -437,30 +421,4 @@ public class DefaultRepositorySystemSessionFactory {
return "Apache-Maven" + version + " (Java " + System.getProperty("java.version") + "; " return "Apache-Maven" + version + " (Java " + System.getProperty("java.version") + "; "
+ System.getProperty("os.name") + " " + System.getProperty("os.version") + ")"; + System.getProperty("os.name") + " " + System.getProperty("os.version") + ")";
} }
private Collection<FileTransformer> getTransformersForArtifact(
final Artifact artifact, final SessionData sessionData) {
TransformerContext context = (TransformerContext) sessionData.get(TransformerContext.KEY);
Collection<FileTransformer> transformers = new ArrayList<>();
// In case of install:install-file there's no transformer context, as the goal is unrelated to the lifecycle.
if ("pom".equals(artifact.getExtension()) && context != null) {
transformers.add(new FileTransformer() {
@Override
public InputStream transformData(File pomFile) throws IOException, TransformException {
try {
return new ConsumerModelSourceTransformer().transform(pomFile.toPath(), context);
} catch (XmlPullParserException e) {
throw new TransformException(e);
}
}
@Override
public Artifact transformArtifact(Artifact artifact) {
return artifact;
}
});
}
return Collections.unmodifiableCollection(transformers);
}
} }

View File

@ -0,0 +1,58 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package org.apache.maven.internal.aether;
import javax.inject.Inject;
import javax.inject.Named;
import javax.inject.Singleton;
import org.apache.maven.internal.transformation.ConsumerPomArtifactTransformer;
import org.eclipse.aether.RepositorySystemSession;
import org.eclipse.aether.deployment.DeployRequest;
import org.eclipse.aether.deployment.DeployResult;
import org.eclipse.aether.deployment.DeploymentException;
import org.eclipse.aether.impl.Deployer;
import org.eclipse.aether.internal.impl.DefaultDeployer;
import org.eclipse.sisu.Priority;
import static java.util.Objects.requireNonNull;
/**
* Maven specific deployer.
*/
@Singleton
@Named
@Priority(100)
final class MavenDeployer implements Deployer {
private final DefaultDeployer deployer;
private final ConsumerPomArtifactTransformer consumerPomArtifactTransformer;
@Inject
MavenDeployer(DefaultDeployer deployer, ConsumerPomArtifactTransformer consumerPomArtifactTransformer) {
this.deployer = requireNonNull(deployer);
this.consumerPomArtifactTransformer = requireNonNull(consumerPomArtifactTransformer);
}
@Override
public DeployResult deploy(RepositorySystemSession session, DeployRequest request) throws DeploymentException {
return deployer.deploy(session, consumerPomArtifactTransformer.remapDeployArtifacts(session, request));
}
}

View File

@ -0,0 +1,58 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package org.apache.maven.internal.aether;
import javax.inject.Inject;
import javax.inject.Named;
import javax.inject.Singleton;
import org.apache.maven.internal.transformation.ConsumerPomArtifactTransformer;
import org.eclipse.aether.RepositorySystemSession;
import org.eclipse.aether.impl.Installer;
import org.eclipse.aether.installation.InstallRequest;
import org.eclipse.aether.installation.InstallResult;
import org.eclipse.aether.installation.InstallationException;
import org.eclipse.aether.internal.impl.DefaultInstaller;
import org.eclipse.sisu.Priority;
import static java.util.Objects.requireNonNull;
/**
* Maven specific installer.
*/
@Singleton
@Named
@Priority(100)
final class MavenInstaller implements Installer {
private final DefaultInstaller installer;
private final ConsumerPomArtifactTransformer consumerPomArtifactTransformer;
@Inject
MavenInstaller(DefaultInstaller installer, ConsumerPomArtifactTransformer consumerPomArtifactTransformer) {
this.installer = requireNonNull(installer);
this.consumerPomArtifactTransformer = requireNonNull(consumerPomArtifactTransformer);
}
@Override
public InstallResult install(RepositorySystemSession session, InstallRequest request) throws InstallationException {
return installer.install(session, consumerPomArtifactTransformer.remapInstallArtifacts(session, request));
}
}

View File

@ -35,11 +35,11 @@ import static java.util.Objects.requireNonNull;
*/ */
@Named @Named
@EagerSingleton @EagerSingleton
public final class ResolverLifecycle { final class ResolverLifecycle {
private final Provider<RepositorySystem> repositorySystemProvider; private final Provider<RepositorySystem> repositorySystemProvider;
@Inject @Inject
public ResolverLifecycle(Provider<RepositorySystem> repositorySystemProvider) { ResolverLifecycle(Provider<RepositorySystem> repositorySystemProvider) {
this.repositorySystemProvider = requireNonNull(repositorySystemProvider); this.repositorySystemProvider = requireNonNull(repositorySystemProvider);
} }

View File

@ -0,0 +1,169 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package org.apache.maven.internal.transformation;
import javax.inject.Named;
import javax.inject.Singleton;
import java.io.IOException;
import java.io.InputStream;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.nio.file.StandardCopyOption;
import java.util.ArrayList;
import java.util.Collection;
import java.util.function.BiConsumer;
import org.apache.maven.feature.Features;
import org.apache.maven.model.building.DefaultBuildPomXMLFilterFactory;
import org.apache.maven.model.building.TransformerContext;
import org.apache.maven.model.transform.RawToConsumerPomXMLFilterFactory;
import org.apache.maven.model.transform.pull.XmlUtils;
import org.apache.maven.project.MavenProject;
import org.apache.maven.project.artifact.ProjectArtifact;
import org.codehaus.plexus.util.ReaderFactory;
import org.codehaus.plexus.util.xml.XmlStreamReader;
import org.codehaus.plexus.util.xml.pull.EntityReplacementMap;
import org.codehaus.plexus.util.xml.pull.MXParser;
import org.codehaus.plexus.util.xml.pull.XmlPullParser;
import org.codehaus.plexus.util.xml.pull.XmlPullParserException;
import org.eclipse.aether.RepositorySystemSession;
import org.eclipse.aether.artifact.Artifact;
import org.eclipse.aether.artifact.DefaultArtifact;
import org.eclipse.aether.deployment.DeployRequest;
import org.eclipse.aether.installation.InstallRequest;
/**
* Consumer POM transformer.
*
* @since TBD
*/
@Singleton
@Named("consumer-pom")
public final class ConsumerPomArtifactTransformer {
private static final String CONSUMER_POM_CLASSIFIER = "consumer";
public void injectTransformedArtifacts(MavenProject project, RepositorySystemSession session) throws IOException {
if (isActive(session)) {
Path generatedFile;
String buildDirectory =
project.getBuild() != null ? project.getBuild().getDirectory() : null;
if (buildDirectory == null) {
generatedFile = Files.createTempFile(CONSUMER_POM_CLASSIFIER, "pom");
} else {
Path buildDir = Paths.get(buildDirectory);
Files.createDirectories(buildDir);
generatedFile = Files.createTempFile(buildDir, CONSUMER_POM_CLASSIFIER, "pom");
}
project.addAttachedArtifact(new ConsumerPomArtifact(project, generatedFile, session));
}
}
public InstallRequest remapInstallArtifacts(RepositorySystemSession session, InstallRequest request) {
if (isActive(session) && consumerPomPresent(request.getArtifacts())) {
request.setArtifacts(replacePom(request.getArtifacts()));
}
return request;
}
public DeployRequest remapDeployArtifacts(RepositorySystemSession session, DeployRequest request) {
if (isActive(session) && consumerPomPresent(request.getArtifacts())) {
request.setArtifacts(replacePom(request.getArtifacts()));
}
return request;
}
private boolean isActive(RepositorySystemSession session) {
return Features.buildConsumer(session.getUserProperties()).isActive();
}
private boolean consumerPomPresent(Collection<Artifact> artifacts) {
return artifacts.stream().anyMatch(a -> CONSUMER_POM_CLASSIFIER.equals(a.getClassifier()));
}
private Collection<Artifact> replacePom(Collection<Artifact> artifacts) {
ArrayList<Artifact> result = new ArrayList<>(artifacts.size());
for (Artifact artifact : artifacts) {
if (CONSUMER_POM_CLASSIFIER.equals(artifact.getClassifier())) {
// if under CONSUMER_POM_CLASSIFIER, move it to "" classifier
DefaultArtifact remapped = new DefaultArtifact(
artifact.getGroupId(),
artifact.getArtifactId(),
"",
artifact.getExtension(),
artifact.getVersion(),
artifact.getProperties(),
artifact.getFile());
result.add(remapped);
} else if ("".equals(artifact.getClassifier())
&& (artifact.getExtension().equals("pom"))
|| artifact.getExtension().startsWith("pom.")) {
// skip POM and POM subordinates
continue;
} else {
// everything else: add as is
result.add(artifact);
}
}
return result;
}
/**
* Consumer POM is transformed from original POM.
*/
private static class ConsumerPomArtifact extends TransformedArtifact {
private ConsumerPomArtifact(MavenProject mavenProject, Path target, RepositorySystemSession session) {
super(
new ProjectArtifact(mavenProject),
() -> mavenProject.getFile().toPath(),
CONSUMER_POM_CLASSIFIER,
"pom",
target,
transformer(session));
}
private static BiConsumer<Path, Path> transformer(RepositorySystemSession session) {
TransformerContext context = (TransformerContext) session.getData().get(TransformerContext.KEY);
return (src, dest) -> {
try (InputStream inputStream = transform(src, context)) {
Files.createDirectories(dest.getParent());
Files.copy(inputStream, dest, StandardCopyOption.REPLACE_EXISTING);
} catch (XmlPullParserException | IOException e) {
throw new RuntimeException(e);
}
};
}
}
/**
* The actual transformation: visible for testing.
*/
static InputStream transform(Path pomFile, TransformerContext context) throws IOException, XmlPullParserException {
XmlStreamReader reader = ReaderFactory.newXmlReader(Files.newInputStream(pomFile));
XmlPullParser parser = new MXParser(EntityReplacementMap.defaultEntityReplacementMap);
parser.setInput(reader);
parser = new RawToConsumerPomXMLFilterFactory(new DefaultBuildPomXMLFilterFactory(context, true))
.get(parser, pomFile);
return XmlUtils.writeDocument(reader, parser);
}
}

View File

@ -0,0 +1,104 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package org.apache.maven.internal.transformation;
import java.io.IOException;
import java.io.UncheckedIOException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.Objects;
import java.util.concurrent.atomic.AtomicReference;
import java.util.function.BiConsumer;
import java.util.function.Function;
import java.util.function.Supplier;
import static java.util.Objects.requireNonNull;
/**
* Keeps transformed file up-to-date relative to its source file. It manages state (i.e. hashing the content) using
* passed in stateFunction, and transforms when needed using passed in transformer bi-consumer.
* <p>
* Covered cases:
* <ul>
* <li>when source supplier returns {@code null}, this class will return {@code null}.</li>
* <li>when source supplier returns non existing path, this class will return non existing path.</li>
* <li>when source supplier returns existing path, this class will ensure transformation is in sync.</li>
* </ul>
*
* @since TBD
*/
final class OnChangeTransformer implements Supplier<Path> {
private final Supplier<Path> source;
private final Path target;
private final Function<Path, String> stateFunction;
private final BiConsumer<Path, Path> transformerConsumer;
private final AtomicReference<String> sourceState;
OnChangeTransformer(
Supplier<Path> source,
Path target,
Function<Path, String> stateFunction,
BiConsumer<Path, Path> transformerConsumer) {
this.source = requireNonNull(source);
this.target = requireNonNull(target);
this.stateFunction = requireNonNull(stateFunction);
this.transformerConsumer = requireNonNull(transformerConsumer);
this.sourceState = new AtomicReference<>(null);
}
@Override
public synchronized Path get() {
String state = mayUpdate();
if (state == null) {
return null;
}
return target;
}
private String mayUpdate() {
String result;
try {
Path src = source.get();
if (src == null) {
Files.deleteIfExists(target);
result = null;
} else if (!Files.exists(src)) {
Files.deleteIfExists(target);
result = "";
} else {
String current = stateFunction.apply(src);
String existing = sourceState.get();
if (!Objects.equals(current, existing)) {
transformerConsumer.accept(src, target);
Files.setLastModifiedTime(target, Files.getLastModifiedTime(src));
}
result = current;
}
} catch (IOException e) {
throw new UncheckedIOException(e);
}
sourceState.set(result);
return result;
}
}

View File

@ -0,0 +1,146 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package org.apache.maven.internal.transformation;
import java.io.File;
import java.io.InputStream;
import java.nio.file.Files;
import java.nio.file.Path;
import java.security.MessageDigest;
import java.util.function.BiConsumer;
import java.util.function.Supplier;
import org.apache.maven.artifact.Artifact;
import org.apache.maven.artifact.DefaultArtifact;
import org.apache.maven.artifact.handler.ArtifactHandler;
import static java.util.Objects.requireNonNull;
/**
* Transformed artifact is derived with some transformation from source artifact.
*
* @since TBD
*/
abstract class TransformedArtifact extends DefaultArtifact {
private final OnChangeTransformer onChangeTransformer;
TransformedArtifact(
Artifact source,
Supplier<Path> sourcePathProvider,
String classifier,
String extension,
Path targetPath,
BiConsumer<Path, Path> transformerConsumer) {
super(
source.getGroupId(),
source.getArtifactId(),
source.getVersionRange(),
source.getScope(),
extension,
classifier,
new TransformedArtifactHandler(
classifier, extension, source.getArtifactHandler().getPackaging()));
this.onChangeTransformer =
new OnChangeTransformer(sourcePathProvider, targetPath, TransformedArtifact::sha1, transformerConsumer);
}
@Override
public boolean isResolved() {
return getFile() != null;
}
@Override
public void setFile(File file) {
throw new IllegalStateException("transformed artifact file cannot be set");
}
@Override
public File getFile() {
Path result = onChangeTransformer.get();
if (result == null) {
return null;
}
return result.toFile();
}
private static final int BUFFER_SIZE = 8192;
private static String sha1(Path path) {
try {
MessageDigest md = MessageDigest.getInstance("SHA-1");
try (InputStream fis = Files.newInputStream(path)) {
byte[] buffer = new byte[BUFFER_SIZE];
int read;
while ((read = fis.read(buffer)) != -1) {
md.update(buffer, 0, read);
}
}
StringBuilder result = new StringBuilder();
for (byte b : md.digest()) {
result.append(String.format("%02x", b));
}
return result.toString();
} catch (Exception e) {
throw new RuntimeException(e);
}
}
private static class TransformedArtifactHandler implements ArtifactHandler {
private final String classifier;
private final String extension;
private final String packaging;
private TransformedArtifactHandler(String classifier, String extension, String packaging) {
this.classifier = classifier;
this.extension = requireNonNull(extension);
this.packaging = requireNonNull(packaging);
}
public String getClassifier() {
return classifier;
}
public String getDirectory() {
return null;
}
public String getExtension() {
return extension;
}
public String getLanguage() {
return "none";
}
public String getPackaging() {
return packaging;
}
public boolean isAddedToClasspath() {
return false;
}
public boolean isIncludesDependencies() {
return false;
}
}
}

View File

@ -30,6 +30,7 @@ import org.apache.maven.execution.ExecutionEvent;
import org.apache.maven.execution.MavenSession; import org.apache.maven.execution.MavenSession;
import org.apache.maven.execution.ProjectExecutionEvent; import org.apache.maven.execution.ProjectExecutionEvent;
import org.apache.maven.execution.ProjectExecutionListener; import org.apache.maven.execution.ProjectExecutionListener;
import org.apache.maven.internal.transformation.ConsumerPomArtifactTransformer;
import org.apache.maven.lifecycle.MavenExecutionPlan; import org.apache.maven.lifecycle.MavenExecutionPlan;
import org.apache.maven.lifecycle.internal.builder.BuilderCommon; import org.apache.maven.lifecycle.internal.builder.BuilderCommon;
import org.apache.maven.plugin.MojoExecution; import org.apache.maven.plugin.MojoExecution;
@ -55,6 +56,7 @@ public class LifecycleModuleBuilder {
private final BuilderCommon builderCommon; private final BuilderCommon builderCommon;
private final ExecutionEventCatapult eventCatapult; private final ExecutionEventCatapult eventCatapult;
private final ProjectExecutionListener projectExecutionListener; private final ProjectExecutionListener projectExecutionListener;
private final ConsumerPomArtifactTransformer consumerPomArtifactTransformer;
private final SessionScope sessionScope; private final SessionScope sessionScope;
@Inject @Inject
@ -63,11 +65,13 @@ public class LifecycleModuleBuilder {
BuilderCommon builderCommon, BuilderCommon builderCommon,
ExecutionEventCatapult eventCatapult, ExecutionEventCatapult eventCatapult,
List<ProjectExecutionListener> listeners, List<ProjectExecutionListener> listeners,
ConsumerPomArtifactTransformer consumerPomArtifactTransformer,
SessionScope sessionScope) { SessionScope sessionScope) {
this.mojoExecutor = mojoExecutor; this.mojoExecutor = mojoExecutor;
this.builderCommon = builderCommon; this.builderCommon = builderCommon;
this.eventCatapult = eventCatapult; this.eventCatapult = eventCatapult;
this.projectExecutionListener = new CompoundProjectExecutionListener(listeners); this.projectExecutionListener = new CompoundProjectExecutionListener(listeners);
this.consumerPomArtifactTransformer = consumerPomArtifactTransformer;
this.sessionScope = sessionScope; this.sessionScope = sessionScope;
} }
@ -93,6 +97,8 @@ public class LifecycleModuleBuilder {
return; return;
} }
consumerPomArtifactTransformer.injectTransformedArtifacts(currentProject, session.getRepositorySession());
BuilderCommon.attachToThread(currentProject); BuilderCommon.attachToThread(currentProject);
projectExecutionListener.beforeProjectExecution(new ProjectExecutionEvent(session, currentProject)); projectExecutionListener.beforeProjectExecution(new ProjectExecutionEvent(session, currentProject));

View File

@ -23,8 +23,6 @@ import javax.inject.Named;
import javax.inject.Singleton; import javax.inject.Singleton;
import java.util.List; import java.util.List;
import java.util.Optional;
import java.util.Properties;
import java.util.Set; import java.util.Set;
import java.util.stream.Collectors; import java.util.stream.Collectors;
@ -33,7 +31,6 @@ import org.apache.maven.execution.BuildFailure;
import org.apache.maven.execution.ExecutionEvent; import org.apache.maven.execution.ExecutionEvent;
import org.apache.maven.execution.MavenExecutionRequest; import org.apache.maven.execution.MavenExecutionRequest;
import org.apache.maven.execution.MavenSession; import org.apache.maven.execution.MavenSession;
import org.apache.maven.feature.Features;
import org.apache.maven.internal.MultilineMessageHelper; import org.apache.maven.internal.MultilineMessageHelper;
import org.apache.maven.lifecycle.LifecycleExecutionException; import org.apache.maven.lifecycle.LifecycleExecutionException;
import org.apache.maven.lifecycle.LifecycleNotFoundException; import org.apache.maven.lifecycle.LifecycleNotFoundException;
@ -112,24 +109,6 @@ public class BuilderCommon {
lifecycleDebugLogger.debugProjectPlan(project, executionPlan); lifecycleDebugLogger.debugProjectPlan(project, executionPlan);
// With Maven 4's build/consumer the POM will always rewrite during distribution.
// The maven-gpg-plugin uses the original POM, causing an invalid signature.
// Fail as long as there's no solution available yet
Properties userProperties = session.getUserProperties();
if (Features.buildConsumer(userProperties).isActive()) {
Optional<MojoExecution> gpgMojo = executionPlan.getMojoExecutions().stream()
.filter(m -> "maven-gpg-plugin".equals(m.getArtifactId())
&& "org.apache.maven.plugins".equals(m.getGroupId()))
.findAny();
if (gpgMojo.isPresent()) {
throw new LifecycleExecutionException("The maven-gpg-plugin is not supported by Maven 4."
+ " Verify if there is a compatible signing solution,"
+ " add -D" + Features.buildConsumer(userProperties).propertyName() + "=false"
+ " or use Maven 3.");
}
}
if (session.getRequest().getDegreeOfConcurrency() > 1 if (session.getRequest().getDegreeOfConcurrency() > 1
&& session.getProjects().size() > 1) { && session.getProjects().size() > 1) {
final Set<Plugin> unsafePlugins = executionPlan.getNonThreadSafePlugins(); final Set<Plugin> unsafePlugins = executionPlan.getNonThreadSafePlugins();

View File

@ -16,7 +16,7 @@
* specific language governing permissions and limitations * specific language governing permissions and limitations
* under the License. * under the License.
*/ */
package org.apache.maven.internal.aether; package org.apache.maven.internal.transformation;
import java.io.InputStream; import java.io.InputStream;
import java.nio.file.Files; import java.nio.file.Files;
@ -28,9 +28,7 @@ import org.apache.maven.model.building.TransformerContext;
import org.junit.jupiter.api.Test; import org.junit.jupiter.api.Test;
import org.xmlunit.assertj.XmlAssert; import org.xmlunit.assertj.XmlAssert;
public class ConsumerModelSourceTransformerTest { public class ConsumerPomArtifactTransformerTest {
private ConsumerModelSourceTransformer transformer = new ConsumerModelSourceTransformer();
@Test @Test
public void transform() throws Exception { public void transform() throws Exception {
Path beforePomFile = Path beforePomFile =
@ -39,7 +37,8 @@ public class ConsumerModelSourceTransformerTest {
Paths.get("src/test/resources/projects/transform/after.pom").toAbsolutePath(); Paths.get("src/test/resources/projects/transform/after.pom").toAbsolutePath();
try (InputStream expected = Files.newInputStream(afterPomFile); try (InputStream expected = Files.newInputStream(afterPomFile);
InputStream result = transformer.transform(beforePomFile, new NoTransformerContext())) { InputStream result =
ConsumerPomArtifactTransformer.transform(beforePomFile, new NoTransformerContext())) {
XmlAssert.assertThat(result).and(expected).areIdentical(); XmlAssert.assertThat(result).and(expected).areIdentical();
} }
} }

View File

@ -37,6 +37,7 @@ import org.apache.maven.plugin.MojoExecution;
import org.apache.maven.project.MavenProject; import org.apache.maven.project.MavenProject;
import org.codehaus.plexus.PlexusContainer; import org.codehaus.plexus.PlexusContainer;
import org.codehaus.plexus.testing.PlexusTest; import org.codehaus.plexus.testing.PlexusTest;
import org.eclipse.aether.DefaultRepositorySystemSession;
import org.junit.jupiter.api.Test; import org.junit.jupiter.api.Test;
import static org.junit.jupiter.api.Assertions.assertEquals; import static org.junit.jupiter.api.Assertions.assertEquals;
@ -63,7 +64,8 @@ public class LifecycleModuleBuilderTest {
MavenExecutionRequest mavenExecutionRequest = new DefaultMavenExecutionRequest(); MavenExecutionRequest mavenExecutionRequest = new DefaultMavenExecutionRequest();
mavenExecutionRequest.setExecutionListener(new AbstractExecutionListener()); mavenExecutionRequest.setExecutionListener(new AbstractExecutionListener());
mavenExecutionRequest.setGoals(Arrays.asList("clean")); mavenExecutionRequest.setGoals(Arrays.asList("clean"));
final MavenSession session = new MavenSession(null, null, mavenExecutionRequest, defaultMavenExecutionResult); final MavenSession session = new MavenSession(
null, new DefaultRepositorySystemSession(), mavenExecutionRequest, defaultMavenExecutionResult);
final ProjectDependencyGraphStub dependencyGraphStub = new ProjectDependencyGraphStub(); final ProjectDependencyGraphStub dependencyGraphStub = new ProjectDependencyGraphStub();
session.setProjectDependencyGraph(dependencyGraphStub); session.setProjectDependencyGraph(dependencyGraphStub);
session.setProjects(dependencyGraphStub.getSortedProjects()); session.setProjects(dependencyGraphStub.getSortedProjects());

View File

@ -18,7 +18,10 @@
*/ */
package org.apache.maven.feature; package org.apache.maven.feature;
import java.util.HashMap;
import java.util.Map;
import java.util.Properties; import java.util.Properties;
import java.util.stream.Collectors;
/** /**
* Centralized class for feature information * Centralized class for feature information
@ -30,9 +33,22 @@ public final class Features {
private Features() {} private Features() {}
public static Feature buildConsumer(Properties userProperties) { public static Feature buildConsumer(Properties userProperties) {
return buildConsumer(toMap(userProperties));
}
public static Feature buildConsumer(Map<String, String> userProperties) {
return new Feature(userProperties, "maven.experimental.buildconsumer", "true"); return new Feature(userProperties, "maven.experimental.buildconsumer", "true");
} }
private static Map<String, String> toMap(Properties properties) {
return properties.entrySet().stream()
.collect(Collectors.toMap(
e -> String.valueOf(e.getKey()),
e -> String.valueOf(e.getValue()),
(prev, next) -> next,
HashMap::new));
}
/** /**
* Represents some feature * Represents some feature
* *
@ -44,9 +60,9 @@ public final class Features {
private final String name; private final String name;
Feature(Properties userProperties, String name, String defaultValue) { Feature(Map<String, String> userProperties, String name, String defaultValue) {
this.name = name; this.name = name;
this.active = "true".equals(userProperties.getProperty(name, defaultValue)); this.active = "true".equals(userProperties.getOrDefault(name, defaultValue));
} }
public boolean isActive() { public boolean isActive() {