Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fix: enforce plugin-required dependencies and log incompatibilities #20601

Merged
merged 1 commit into from
Dec 3, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@

import java.io.File;
import java.io.IOException;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.net.URI;
import java.net.URISyntaxException;
Expand Down Expand Up @@ -210,15 +211,37 @@ public void execute() throws MojoExecutionException, MojoFailureException {
try {
org.apache.maven.plugin.Mojo task = reflector.createMojo(this);
findExecuteMethod(task.getClass()).invoke(task);
reflector.logIncompatibilities(getLog()::debug);
} catch (MojoExecutionException | MojoFailureException e) {
logTroubleshootingHints(reflector, e);
throw e;
} catch (Exception e) {
logTroubleshootingHints(reflector, e);
throw new MojoFailureException(e.getMessage(), e);
} finally {
Thread.currentThread().setContextClassLoader(tccl);
}
}

private void logTroubleshootingHints(Reflector reflector, Throwable ex) {
reflector.logIncompatibilities(getLog()::warn);
if (ex instanceof InvocationTargetException) {
ex = ex.getCause();
}
StringBuilder errorMessage = new StringBuilder(ex.getMessage());
Throwable cause = ex.getCause();
while (cause != null) {
if (cause.getMessage() != null) {
errorMessage.append(" ").append(cause.getMessage());
}
cause = cause.getCause();
}
getLog().error(
"The build process encountered an error: " + errorMessage);
logError(
"To diagnose the issue, please re-run Maven with the -X option to enable detailed debug logging and identify the root cause.");
}

public void executeInternal() throws MojoFailureException {
long start = System.nanoTime();

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,7 @@
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.stream.Collectors;

Expand All @@ -52,8 +53,13 @@ public final class Reflector {
private static final Set<String> DEPENDENCIES_GROUP_EXCLUSIONS = Set.of(
"org.apache.maven", "org.codehaus.plexus", "org.slf4j",
"org.eclipse.sisu");
// Dependency required by the plugin but not provided by Flow at runtime
private static final Set<String> REQUIRED_PLUGIN_DEPENDENCIES = Set.of(
"org.reflections:reflections:jar",
"org.zeroturnaround:zt-exec:jar");

private final URLClassLoader isolatedClassLoader;
private List<String> dependenciesIncompatibility;
private Object classFinder;

/**
Expand All @@ -66,9 +72,11 @@ public Reflector(URLClassLoader isolatedClassLoader) {
this.isolatedClassLoader = isolatedClassLoader;
}

private Reflector(URLClassLoader isolatedClassLoader, Object classFinder) {
private Reflector(URLClassLoader isolatedClassLoader, Object classFinder,
List<String> dependenciesIncompatibility) {
this.isolatedClassLoader = isolatedClassLoader;
this.classFinder = classFinder;
this.dependenciesIncompatibility = dependenciesIncompatibility;
}

/**
Expand All @@ -91,6 +99,7 @@ private Reflector(URLClassLoader isolatedClassLoader, Object classFinder) {
* it is not possible to make a copy for it due to class
* definition incompatibilities.
*/
@SuppressWarnings("unchecked")
static Reflector adapt(Object reflector) {
if (reflector instanceof Reflector sameClassLoader) {
return sameClassLoader;
Expand All @@ -103,9 +112,13 @@ static Reflector adapt(Object reflector) {
findField(reflectorClass,
"isolatedClassLoader"),
URLClassLoader.class);
List<String> dependenciesIncompatibility = (List<String>) ReflectTools
.getJavaFieldValue(reflector, findField(reflectorClass,
"dependenciesIncompatibility"));
Object classFinder = ReflectTools.getJavaFieldValue(reflector,
findField(reflectorClass, "classFinder"));
return new Reflector(classLoader, classFinder);
return new Reflector(classLoader, classFinder,
dependenciesIncompatibility);
} catch (Exception e) {
throw new IllegalArgumentException(
"Object of type " + reflector.getClass().getName()
Expand Down Expand Up @@ -198,9 +211,27 @@ public Mojo createMojo(BuildDevBundleMojo sourceMojo) throws Exception {
*/
public static Reflector of(MavenProject project,
MojoExecution mojoExecution) {
List<String> dependenciesIncompatibility = new ArrayList<>();
URLClassLoader classLoader = createIsolatedClassLoader(project,
mojoExecution);
return new Reflector(classLoader);
mojoExecution, dependenciesIncompatibility);
Reflector reflector = new Reflector(classLoader);
reflector.dependenciesIncompatibility = dependenciesIncompatibility;
return reflector;
}

void logIncompatibilities(Consumer<String> logger) {
if (dependenciesIncompatibility != null) {
logger.accept(
"""
Found dependencies defined with different versions in project and Vaadin maven plugin.
Project dependencies are used, but plugin execution could fail if the versions are incompatible.
In case of build failure please analyze the project dependencies and update versions or configure exclusions for potential offending transitive dependencies.
You can use 'mvn dependency:tree -Dincludes=groupId:artifactId' to detect where the dependency is defined in the project.

"""
+ String.join(System.lineSeparator(),
dependenciesIncompatibility));
}
}

private synchronized Object getOrCreateClassFinder() throws Exception {
Expand All @@ -215,7 +246,8 @@ private synchronized Object getOrCreateClassFinder() throws Exception {
}

private static URLClassLoader createIsolatedClassLoader(
MavenProject project, MojoExecution mojoExecution) {
MavenProject project, MojoExecution mojoExecution,
List<String> dependenciesIncompatibility) {
List<URL> urls = new ArrayList<>();
String outputDirectory = project.getBuild().getOutputDirectory();
if (outputDirectory != null) {
Expand Down Expand Up @@ -246,17 +278,61 @@ private static URLClassLoader createIsolatedClassLoader(
&& artifact.getFile().getPath().matches(
INCLUDE_FROM_COMPILE_DEPS_REGEX))))
.collect(Collectors.toMap(keyMapper, Function.identity())));

if (mojoExecution != null) {
mojoExecution.getMojoDescriptor().getPluginDescriptor()
.getArtifacts().stream()

List<Artifact> pluginDependencies = mojoExecution
.getMojoDescriptor().getPluginDescriptor().getArtifacts()
.stream()
// Exclude all maven artifacts to prevent class loading
// clash with maven.api class realm
.filter(artifact -> !DEPENDENCIES_GROUP_EXCLUSIONS
.contains(artifact.getGroupId()))
.filter(artifact -> !projectDependencies
.containsKey(keyMapper.apply(artifact)))
.forEach(artifact -> projectDependencies
.put(keyMapper.apply(artifact), artifact));
.toList();

// Exclude project artifact that are also defined as mandatory
// plugin dependencies. The version provided by the plugin will be
// used to prevent failures during maven build.
pluginDependencies.stream().map(keyMapper)
.filter(REQUIRED_PLUGIN_DEPENDENCIES::contains)
.forEach(projectDependencies::remove);

// Preserve required plugin dependency that are not provided by Flow
// -1: dependency defined on both plugin and project, with different
// version
// 0: dependency defined on both plugin and project, with same
// version
// 1: dependency defined by the plugin only
Map<Integer, List<Artifact>> potentialDuplicates = pluginDependencies
.stream().collect(Collectors.groupingBy(pluginArtifact -> {
Artifact projectArtifact = projectDependencies
.get(keyMapper.apply(pluginArtifact));
if (projectArtifact == null) {
return 1;
} else if (projectArtifact.getId()
.equals(pluginArtifact.getId())) {
return 0;
}
return -1;
}));
// Log potential plugin and project dependency versions
// incompatibilities.
if (potentialDuplicates.containsKey(-1)) {
potentialDuplicates.get(-1).stream().map(pluginArtifact -> {
String key = keyMapper.apply(pluginArtifact);
return String.format(
"%s: project version [%s], plugin version [%s]",
key, projectDependencies.get(key).getBaseVersion(),
pluginArtifact.getBaseVersion());
}).forEach(dependenciesIncompatibility::add);
}

// Add dependencies defined only by the plugin
if (potentialDuplicates.containsKey(1)) {
potentialDuplicates.get(1)
.forEach(artifact -> projectDependencies
.put(keyMapper.apply(artifact), artifact));
}
}

projectDependencies.values().stream()
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
#
# Copyright 2000-2024 Vaadin Ltd.
#
# Licensed 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.
#

invoker.goals=clean package
invoker.buildResult=failure
Original file line number Diff line number Diff line change
@@ -0,0 +1,68 @@
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>

<groupId>com.vaadin.test.maven</groupId>
<artifactId>offending-dependency-project</artifactId>
<version>1.0</version>
<packaging>jar</packaging>

<description>
Tests that project dependencies does not override plugin required dependency.
</description>

<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
<maven.compiler.release>17</maven.compiler.release>
<maven.compiler.source>${maven.compiler.release}</maven.compiler.source>
<maven.compiler.target>${maven.compiler.release}</maven.compiler.target>
<maven.test.skip>true</maven.test.skip>

<flow.version>@project.version@</flow.version>
<maven.version>3.9.9</maven.version>
</properties>

<dependencies>
<dependency>
<groupId>com.vaadin</groupId>
<artifactId>flow-server</artifactId>
<version>${flow.version}</version>
</dependency>
<dependency>
<groupId>com.vaadin</groupId>
<artifactId>flow-client</artifactId>
<version>${flow.version}</version>
</dependency>
<!-- commons-io 2.6 is incompatible with Flow plugin -->
<dependency>
<groupId>commons-io</groupId>
<artifactId>commons-io</artifactId>
<version>2.6</version>
</dependency>
</dependencies>

<build>
<plugins>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
<version>3.13.0</version>
</plugin>
<plugin>
<groupId>com.vaadin</groupId>
<artifactId>flow-maven-plugin</artifactId>
<version>${flow.version}</version>
<executions>
<execution>
<goals>
<goal>prepare-frontend</goal>
<goal>build-frontend</goal>
</goals>
</execution>
</executions>
</plugin>
</plugins>
</build>

</project>
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
package com.vaadin.test;

import java.util.List;

import com.vaadin.flow.server.frontend.Options;
import com.vaadin.flow.server.frontend.TypeScriptBootstrapModifier;
import com.vaadin.flow.server.frontend.scanner.FrontendDependenciesScanner;

/**
* Hello world!
*/
public class ProjectFlowExtension implements TypeScriptBootstrapModifier {

@Override
public void modify(List<String> bootstrapTypeScript, Options options,
FrontendDependenciesScanner frontendDependenciesScanner) {
System.out.println("ProjectFlowExtension");
bootstrapTypeScript.add("(window as any).testProject=1;");
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
import java.nio.file.*;

flowTsx = basedir.toPath().resolve("build.log");
if ( !Files.exists(flowTsx, new LinkOption[0]) )
{
throw new RuntimeException("build.log not found");
}

lines = Files.readString(flowTsx);
if (
!lines.contains("Found dependencies defined with different versions in project and Vaadin maven plugin") &&
!lines.matches("^commons-io:commons-io.*\\[2\\.6\\],.*")
) {
throw new RuntimeException("Offending commons-io 2.6 dependency not detected");
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
#
# Copyright 2000-2024 Vaadin Ltd.
#
# Licensed 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.
#

invoker.goals=clean package
Loading
Loading