diff --git a/common/graalvm-reachability-metadata/src/main/java/org/graalvm/reachability/Configuration.java b/common/graalvm-reachability-metadata/src/main/java/org/graalvm/reachability/Configuration.java new file mode 100644 index 000000000..59bcd56ce --- /dev/null +++ b/common/graalvm-reachability-metadata/src/main/java/org/graalvm/reachability/Configuration.java @@ -0,0 +1,63 @@ +/* + * Copyright (c) 2020, 2022 Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * The Universal Permissive License (UPL), Version 1.0 + * + * Subject to the condition set forth below, permission is hereby granted to any + * person obtaining a copy of this software, associated documentation and/or + * data (collectively the "Software"), free of charge and under any and all + * copyright rights in the Software, and any and all patent rights owned or + * freely licensable by each licensor hereunder covering either (i) the + * unmodified Software as contributed to or provided by such licensor, or (ii) + * the Larger Works (as defined below), to deal in both + * + * (a) the Software, and + * + * (b) any piece of software and/or hardware listed in the lrgrwrks.txt file if + * one is included with the Software each a "Larger Work" to which the Software + * is contributed by such licensors), + * + * without restriction, including without limitation the rights to copy, create + * derivative works of, display, perform, and distribute the Software and make, + * use, sell, offer for sale, import, export, have made, and have sold the + * Software and the Larger Work(s), and to sublicense the foregoing rights on + * either these or other terms. + * + * This license is subject to the following condition: + * + * The above copyright notice and either this complete permission notice or at a + * minimum a reference to the UPL must be included in all copies or substantial + * portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE + * SOFTWARE. + */ +package org.graalvm.reachability; + +import java.nio.file.Path; + +public class Configuration { + + private final Path directory; + + private final boolean override; + + public Configuration(Path directory, boolean override) { + this.directory = directory; + this.override = override; + } + + public Path getDirectory() { + return directory; + } + + public boolean isOverride() { + return override; + } +} diff --git a/common/graalvm-reachability-metadata/src/main/java/org/graalvm/reachability/GraalVMReachabilityMetadataRepository.java b/common/graalvm-reachability-metadata/src/main/java/org/graalvm/reachability/GraalVMReachabilityMetadataRepository.java index 74a2432b0..89ad5897c 100644 --- a/common/graalvm-reachability-metadata/src/main/java/org/graalvm/reachability/GraalVMReachabilityMetadataRepository.java +++ b/common/graalvm-reachability-metadata/src/main/java/org/graalvm/reachability/GraalVMReachabilityMetadataRepository.java @@ -40,7 +40,6 @@ */ package org.graalvm.reachability; -import java.nio.file.Path; import java.util.Collection; import java.util.Set; import java.util.function.Consumer; @@ -63,9 +62,9 @@ public interface GraalVMReachabilityMetadataRepository { * example if a configuration directory isn't available for a * particular artifact version. * @param queryBuilder the query builder - * @return the set of configuration directories matching the query + * @return the set of configuration matching the query */ - Set findConfigurationDirectoriesFor(Consumer queryBuilder); + Set findConfigurationsFor(Consumer queryBuilder); /** * Returns a list of configuration directories for the specified artifact. @@ -73,19 +72,19 @@ public interface GraalVMReachabilityMetadataRepository { * but the list may also be empty if the repository doesn't contain any. * Never null. * @param gavCoordinates the artifact GAV coordinates (group:artifact:version) - * @return a list of configuration directories + * @return a list of configuration */ - default Set findConfigurationDirectoriesFor(String gavCoordinates) { - return findConfigurationDirectoriesFor(q -> q.forArtifacts(gavCoordinates)); + default Set findConfigurationsFor(String gavCoordinates) { + return findConfigurationsFor(q -> q.forArtifacts(gavCoordinates)); } /** * Returns the set of configuration directories for all the modules supplied * as an argument. * @param modules the list of modules - * @return the set of configuration directories + * @return the set of configuration */ - default Set findConfigurationDirectoriesFor(Collection modules) { - return findConfigurationDirectoriesFor(q -> q.forArtifacts(modules)); + default Set findConfigurationsFor(Collection modules) { + return findConfigurationsFor(q -> q.forArtifacts(modules)); } } diff --git a/common/graalvm-reachability-metadata/src/main/java/org/graalvm/reachability/internal/FileSystemRepository.java b/common/graalvm-reachability-metadata/src/main/java/org/graalvm/reachability/internal/FileSystemRepository.java index 76dc0e26f..1a88282fe 100644 --- a/common/graalvm-reachability-metadata/src/main/java/org/graalvm/reachability/internal/FileSystemRepository.java +++ b/common/graalvm-reachability-metadata/src/main/java/org/graalvm/reachability/internal/FileSystemRepository.java @@ -40,6 +40,7 @@ */ package org.graalvm.reachability.internal; +import org.graalvm.reachability.Configuration; import org.graalvm.reachability.GraalVMReachabilityMetadataRepository; import org.graalvm.reachability.Query; import org.graalvm.reachability.internal.index.artifacts.SingleModuleJsonVersionToConfigDirectoryIndex; @@ -90,7 +91,7 @@ public static boolean isSupportedArchiveFormat(String path) { } @Override - public Set findConfigurationDirectoriesFor(Consumer queryBuilder) { + public Set findConfigurationsFor(Consumer queryBuilder) { DefaultQuery query = new DefaultQuery(); queryBuilder.accept(query); return query.getArtifacts() @@ -106,25 +107,25 @@ public Set findConfigurationDirectoriesFor(Consumer queryBu if (artifactQuery.getForcedConfig().isPresent()) { String configVersion = artifactQuery.getForcedConfig().get(); logger.log(groupId, artifactId, version, "Configuration is forced to version " + configVersion); - return index.findForcedConfiguration(configVersion); + return index.findConfiguration(groupId, artifactId, configVersion); } - Optional configurationDirectory = index.findConfigurationDirectory(groupId, artifactId, version); - if (!configurationDirectory.isPresent() && artifactQuery.isUseLatestVersion()) { + Optional configuration = index.findConfiguration(groupId, artifactId, version); + if (!configuration.isPresent() && artifactQuery.isUseLatestVersion()) { logger.log(groupId, artifactId, version, "Configuration directory not found. Trying latest version."); - configurationDirectory = index.findLatestConfigurationFor(groupId, artifactId); - if (!configurationDirectory.isPresent()) { + configuration = index.findLatestConfigurationFor(groupId, artifactId); + if (!configuration.isPresent()) { logger.log(groupId, artifactId, version, "Latest version not found!"); } } - Optional finalConfigurationDirectory = configurationDirectory; + Optional finalConfigurationDirectory = configuration; logger.log(groupId, artifactId, version, () -> { if (finalConfigurationDirectory.isPresent()) { - Path path = finalConfigurationDirectory.get(); + Path path = finalConfigurationDirectory.get().getDirectory(); return "Configuration directory is " + rootDirectory.relativize(path); } return "missing."; }); - return configurationDirectory; + return configuration; }) .filter(Optional::isPresent) .map(Optional::get); diff --git a/common/graalvm-reachability-metadata/src/main/java/org/graalvm/reachability/internal/index/artifacts/Artifact.java b/common/graalvm-reachability-metadata/src/main/java/org/graalvm/reachability/internal/index/artifacts/Artifact.java index 3641a67a7..248f11f9e 100644 --- a/common/graalvm-reachability-metadata/src/main/java/org/graalvm/reachability/internal/index/artifacts/Artifact.java +++ b/common/graalvm-reachability-metadata/src/main/java/org/graalvm/reachability/internal/index/artifacts/Artifact.java @@ -52,16 +52,19 @@ public class Artifact { private final Set versions; private final String directory; private final boolean latest; + private final boolean override; @JsonCreator public Artifact(@JsonProperty("module") String module, @JsonProperty("tested-versions") Set versions, @JsonProperty("metadata-version") String directory, - @JsonProperty(value = "latest", defaultValue = "false") boolean latest) { + @JsonProperty(value = "latest", defaultValue = "false") boolean latest, + @JsonProperty(value = "override", defaultValue = "false") boolean override) { this.module = module; this.versions = versions; this.directory = directory; this.latest = latest; + this.override = override; } public String getModule() { @@ -79,4 +82,8 @@ public String getDirectory() { public boolean isLatest() { return latest; } + + public boolean isOverride() { + return override; + } } diff --git a/common/graalvm-reachability-metadata/src/main/java/org/graalvm/reachability/internal/index/artifacts/SingleModuleJsonVersionToConfigDirectoryIndex.java b/common/graalvm-reachability-metadata/src/main/java/org/graalvm/reachability/internal/index/artifacts/SingleModuleJsonVersionToConfigDirectoryIndex.java index 25709d50c..048490fbe 100644 --- a/common/graalvm-reachability-metadata/src/main/java/org/graalvm/reachability/internal/index/artifacts/SingleModuleJsonVersionToConfigDirectoryIndex.java +++ b/common/graalvm-reachability-metadata/src/main/java/org/graalvm/reachability/internal/index/artifacts/SingleModuleJsonVersionToConfigDirectoryIndex.java @@ -42,6 +42,7 @@ import com.fasterxml.jackson.databind.ObjectMapper; import com.fasterxml.jackson.databind.type.TypeFactory; +import org.graalvm.reachability.Configuration; import org.graalvm.reachability.internal.UncheckedIOException; import java.io.BufferedReader; @@ -80,12 +81,6 @@ private Map> parseIndexFile(Path rootPath) { } - @Override - public Optional findForcedConfiguration(String version) { - Path configDir = moduleRoot.resolve(version); - return Files.isDirectory(configDir) ? Optional.of(configDir) : Optional.empty(); - } - /** * Returns the configuration directory for the requested artifact. * @@ -95,7 +90,7 @@ public Optional findForcedConfiguration(String version) { * @return a configuration directory, or empty if no configuration directory is available */ @Override - public Optional findConfigurationDirectory(String groupId, String artifactId, String version) { + public Optional findConfiguration(String groupId, String artifactId, String version) { return findConfigurationFor(groupId, artifactId, artifact -> artifact.getVersions().contains(version)); } @@ -107,11 +102,11 @@ public Optional findConfigurationDirectory(String groupId, String artifact * @return a configuration directory, or empty if no configuration directory is available */ @Override - public Optional findLatestConfigurationFor(String groupId, String artifactId) { + public Optional findLatestConfigurationFor(String groupId, String artifactId) { return findConfigurationFor(groupId, artifactId, Artifact::isLatest); } - private Optional findConfigurationFor(String groupId, String artifactId, Predicate predicate) { + private Optional findConfigurationFor(String groupId, String artifactId, Predicate predicate) { String module = groupId + ":" + artifactId; List artifacts = index.get(module); if (artifacts == null) { @@ -121,6 +116,7 @@ private Optional findConfigurationFor(String groupId, String artifactId, P .filter(artifact -> artifact.getModule().equals(module)) .filter(predicate) .findFirst() - .map(artifact -> moduleRoot.resolve(artifact.getDirectory())); + .map(artifact -> new Configuration(moduleRoot.resolve(artifact.getDirectory()), artifact.isOverride())); } + } diff --git a/common/graalvm-reachability-metadata/src/main/java/org/graalvm/reachability/internal/index/artifacts/VersionToConfigDirectoryIndex.java b/common/graalvm-reachability-metadata/src/main/java/org/graalvm/reachability/internal/index/artifacts/VersionToConfigDirectoryIndex.java index 3a712265d..02c44ac2d 100644 --- a/common/graalvm-reachability-metadata/src/main/java/org/graalvm/reachability/internal/index/artifacts/VersionToConfigDirectoryIndex.java +++ b/common/graalvm-reachability-metadata/src/main/java/org/graalvm/reachability/internal/index/artifacts/VersionToConfigDirectoryIndex.java @@ -40,33 +40,26 @@ */ package org.graalvm.reachability.internal.index.artifacts; -import java.nio.file.Path; import java.util.Optional; -public interface VersionToConfigDirectoryIndex { +import org.graalvm.reachability.Configuration; - /** - * Returns the specified configuration directory version, ignoring - * any existing configuration. - * @param version the requested version - * @return the configuration directory - */ - Optional findForcedConfiguration(String version); +public interface VersionToConfigDirectoryIndex { /** - * Returns the configuration directory for the requested artifact. + * Returns the configuration for the requested artifact. * @param groupId the group ID of the artifact * @param artifactId the artifact ID of the artifact * @param version the version of the artifact - * @return a configuration directory, or empty if no configuration directory is available + * @return a configuration, or empty if no configuration directory is available */ - Optional findConfigurationDirectory(String groupId, String artifactId, String version); + Optional findConfiguration(String groupId, String artifactId, String version); /** - * Returns the latest configuration directory for the requested artifact. + * Returns the latest configuration for the requested artifact. * @param groupId the group ID of the artifact * @param artifactId the artifact ID of the artifact - * @return a configuration directory, or empty if no configuration directory is available + * @return a configuration, or empty if no configuration directory is available */ - Optional findLatestConfigurationFor(String groupId, String artifactId); + Optional findLatestConfigurationFor(String groupId, String artifactId); } diff --git a/common/graalvm-reachability-metadata/src/test/java/org/graalvm/reachability/internal/FileSystemRepositoryTest.java b/common/graalvm-reachability-metadata/src/test/java/org/graalvm/reachability/internal/FileSystemRepositoryTest.java index 65c663116..d754a6529 100644 --- a/common/graalvm-reachability-metadata/src/test/java/org/graalvm/reachability/internal/FileSystemRepositoryTest.java +++ b/common/graalvm-reachability-metadata/src/test/java/org/graalvm/reachability/internal/FileSystemRepositoryTest.java @@ -41,6 +41,7 @@ package org.graalvm.reachability.internal; +import org.graalvm.reachability.Configuration; import org.graalvm.reachability.Query; import org.junit.jupiter.api.Test; @@ -51,6 +52,8 @@ import java.util.function.Consumer; import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertFalse; +import static org.junit.jupiter.api.Assertions.assertTrue; class FileSystemRepositoryTest { private FileSystemRepository repository; @@ -65,9 +68,11 @@ void testRepo1() { // then: result.hasSinglePath("org/foo/1.0"); + result.hasNoOverride(); // when: lookup("org:foo:1.1"); + result.hasOverride(); // then: result.hasSinglePath("org/foo/1.1"); @@ -87,6 +92,7 @@ void testRepo2() { // then: result.hasSinglePath("org/foo/1.1"); + result.hasNoOverride(); } @Test @@ -166,11 +172,11 @@ void canUseLatestConfigDir() { } private void lookup(Consumer builder) { - result = new Result(repository.findConfigurationDirectoriesFor(builder), repoPath); + result = new Result(repository.findConfigurationsFor(builder), repoPath); } private void lookup(String gav) { - result = new Result(repository.findConfigurationDirectoriesFor(gav), repoPath); + result = new Result(repository.findConfigurationsFor(gav), repoPath); } private void withRepo(String id) { @@ -184,20 +190,32 @@ private void withRepo(String id) { private static final class Result { private final Path repoPath; - private final Set configDirs; + private final Set configs; - private Result(Set configDirs, Path repoPath) { - this.configDirs = configDirs; + private Result(Set configs, Path repoPath) { + this.configs = configs; this.repoPath = repoPath; } public void isEmpty() { - assertEquals(0, configDirs.size()); + assertEquals(0, configs.size()); } public void hasSinglePath(String path) { - assertEquals(1, configDirs.size()); - assertEquals(repoPath.resolve(path), configDirs.iterator().next()); + assertEquals(1, configs.size()); + assertEquals(repoPath.resolve(path), configs.iterator().next().getDirectory()); + } + + public void hasOverride() { + for (Configuration config : configs) { + assertTrue(config.isOverride()); + } + } + + public void hasNoOverride() { + for (Configuration config : configs) { + assertFalse(config.isOverride()); + } } } } diff --git a/common/graalvm-reachability-metadata/src/test/java/org/graalvm/reachability/internal/index/artifacts/SingleModuleJsonVersionToConfigDirectoryIndexTest.java b/common/graalvm-reachability-metadata/src/test/java/org/graalvm/reachability/internal/index/artifacts/SingleModuleJsonVersionToConfigDirectoryIndexTest.java index 4cf3384e3..1a3a954b2 100644 --- a/common/graalvm-reachability-metadata/src/test/java/org/graalvm/reachability/internal/index/artifacts/SingleModuleJsonVersionToConfigDirectoryIndexTest.java +++ b/common/graalvm-reachability-metadata/src/test/java/org/graalvm/reachability/internal/index/artifacts/SingleModuleJsonVersionToConfigDirectoryIndexTest.java @@ -41,6 +41,7 @@ package org.graalvm.reachability.internal.index.artifacts; +import org.graalvm.reachability.Configuration; import org.junit.jupiter.api.Test; import java.io.File; @@ -61,31 +62,36 @@ class SingleModuleJsonVersionToConfigDirectoryIndexTest { void checkIndex() throws URISyntaxException { withIndex("artifact-1"); - Optional configDir = index.findConfigurationDirectory("com.foo", "bar", "1.0"); - assertTrue(configDir.isPresent()); - assertEquals(repoPath.resolve("1.0"), configDir.get()); + Optional config = index.findConfiguration("com.foo", "bar", "1.0"); + assertTrue(config.isPresent()); + assertEquals(repoPath.resolve("1.0"), config.get().getDirectory()); + assertFalse(config.get().isOverride()); - configDir = index.findConfigurationDirectory("com.foo", "bar", "1.3"); - assertTrue(configDir.isPresent()); - assertEquals(repoPath.resolve("1.0"), configDir.get()); + config = index.findConfiguration("com.foo", "bar", "1.3"); + assertTrue(config.isPresent()); + assertEquals(repoPath.resolve("1.0"), config.get().getDirectory()); + assertFalse(config.get().isOverride()); - configDir = index.findConfigurationDirectory("com.foo", "bar", "2.0"); - assertTrue(configDir.isPresent()); - assertEquals(repoPath.resolve("2.0"), configDir.get()); + config = index.findConfiguration("com.foo", "bar", "2.0"); + assertTrue(config.isPresent()); + assertEquals(repoPath.resolve("2.0"), config.get().getDirectory()); + assertTrue(config.get().isOverride()); - configDir = index.findConfigurationDirectory("com.foo", "bar", "2.5"); - assertFalse(configDir.isPresent()); + config = index.findConfiguration("com.foo", "bar", "2.5"); + assertFalse(config.isPresent()); - configDir = index.findConfigurationDirectory("com.foo", "bar-all", "2.0"); - assertTrue(configDir.isPresent()); - assertEquals(repoPath.resolve("2.0"), configDir.get()); + config = index.findConfiguration("com.foo", "bar-all", "2.0"); + assertTrue(config.isPresent()); + assertEquals(repoPath.resolve("2.0"), config.get().getDirectory()); + assertFalse(config.get().isOverride()); - configDir = index.findConfigurationDirectory("com.foo", "nope", "1.0"); - assertFalse(configDir.isPresent()); + config = index.findConfiguration("com.foo", "nope", "1.0"); + assertFalse(config.isPresent()); - Optional latest = index.findLatestConfigurationFor("com.foo", "bar"); + Optional latest = index.findLatestConfigurationFor("com.foo", "bar"); assertTrue(latest.isPresent()); - assertEquals(repoPath.resolve("2.0"), latest.get()); + assertEquals(repoPath.resolve("2.0"), latest.get().getDirectory()); + assertTrue(latest.get().isOverride()); } diff --git a/common/graalvm-reachability-metadata/src/test/resources/json/artifact-1/index.json b/common/graalvm-reachability-metadata/src/test/resources/json/artifact-1/index.json index 63370f676..d2f3b3d07 100644 --- a/common/graalvm-reachability-metadata/src/test/resources/json/artifact-1/index.json +++ b/common/graalvm-reachability-metadata/src/test/resources/json/artifact-1/index.json @@ -1,5 +1,5 @@ [ { "module": "com.foo:bar", "tested-versions": ["1.0", "1.1", "1.2", "1.3"], "metadata-version": "1.0" }, - { "module": "com.foo:bar", "tested-versions": ["2.0", "2.1"], "metadata-version": "2.0", "latest": true }, + { "module": "com.foo:bar", "tested-versions": ["2.0", "2.1"], "metadata-version": "2.0", "latest": true, "override": true }, { "module": "com.foo:bar-all", "tested-versions": ["2.0", "2.1"], "metadata-version": "2.0" } ] diff --git a/common/graalvm-reachability-metadata/src/test/resources/repos/repo1/org/foo/index.json b/common/graalvm-reachability-metadata/src/test/resources/repos/repo1/org/foo/index.json index 980fe87e7..70f6d9c33 100644 --- a/common/graalvm-reachability-metadata/src/test/resources/repos/repo1/org/foo/index.json +++ b/common/graalvm-reachability-metadata/src/test/resources/repos/repo1/org/foo/index.json @@ -12,6 +12,7 @@ "1.1" ], "metadata-version": "1.1", - "latest": true + "latest": true, + "override": true } ] diff --git a/native-gradle-plugin/src/main/java/org/graalvm/buildtools/gradle/NativeImagePlugin.java b/native-gradle-plugin/src/main/java/org/graalvm/buildtools/gradle/NativeImagePlugin.java index 1d240f784..1356e1ecd 100644 --- a/native-gradle-plugin/src/main/java/org/graalvm/buildtools/gradle/NativeImagePlugin.java +++ b/native-gradle-plugin/src/main/java/org/graalvm/buildtools/gradle/NativeImagePlugin.java @@ -126,6 +126,7 @@ import java.util.Set; import java.util.function.Predicate; import java.util.stream.Collectors; +import java.util.stream.Stream; import static org.graalvm.buildtools.gradle.internal.GradleUtils.transitiveProjectArtifacts; import static org.graalvm.buildtools.gradle.internal.NativeImageExecutableLocator.graalvmHomeProvider; @@ -307,6 +308,7 @@ private void configureAutomaticTaskCreation(Project project, t.getOutputFile().map(f -> f.getAsFile().getParentFile()) )); configureJvmReachabilityConfigurationDirectories(project, graalExtension, options, sourceSet); + configureJvmReachabilityExcludeConfigArgs(project, graalExtension, options, sourceSet); }); } @@ -326,7 +328,7 @@ private void configureJvmReachabilityConfigurationDirectories(Project project, G Configuration classpath = project.getConfigurations().getByName(sourceSet.getRuntimeClasspathConfigurationName()); Set excludedModules = repositoryExtension.getExcludedModules().getOrElse(Collections.emptySet()); Map forcedVersions = repositoryExtension.getModuleToConfigVersion().getOrElse(Collections.emptyMap()); - return serviceProvider.map(repo -> repo.findConfigurationDirectoriesFor(query -> classpath.getIncoming().getResolutionResult().allComponents(component -> { + return serviceProvider.map(repo -> repo.findConfigurationsFor(query -> classpath.getIncoming().getResolutionResult().allComponents(component -> { ModuleVersionIdentifier moduleVersion = component.getModuleVersion(); String module = Objects.requireNonNull(moduleVersion).getGroup() + ":" + moduleVersion.getName(); if (!excludedModules.contains(module)) { @@ -339,7 +341,7 @@ private void configureJvmReachabilityConfigurationDirectories(Project project, G } query.useLatestConfigWhenVersionIsUntested(); })).stream() - .map(Path::toAbsolutePath) + .map(configuration -> configuration.getDirectory().toAbsolutePath()) .map(Path::toFile) .collect(Collectors.toList())); } @@ -348,6 +350,48 @@ private void configureJvmReachabilityConfigurationDirectories(Project project, G })); } + private void configureJvmReachabilityExcludeConfigArgs(Project project, GraalVMExtension graalExtension, NativeImageOptions options, SourceSet sourceSet) { + GraalVMReachabilityMetadataRepositoryExtension repositoryExtension = reachabilityExtensionOn(graalExtension); + Provider serviceProvider = project.getGradle() + .getSharedServices() + .registerIfAbsent("nativeConfigurationService", GraalVMReachabilityMetadataService.class, spec -> { + LogLevel logLevel = determineLogLevel(); + spec.getParameters().getLogLevel().set(logLevel); + spec.getParameters().getUri().set(repositoryExtension.getUri()); + spec.getParameters().getCacheDir().set(new File(project.getGradle().getGradleUserHomeDir(), "native-build-tools/repositories")); + }); + options.getExcludeConfigArgs().addAll(repositoryExtension.getEnabled().flatMap(enabled -> { + if (enabled) { + if (repositoryExtension.getUri().isPresent()) { + Configuration classpath = project.getConfigurations().getByName(sourceSet.getRuntimeClasspathConfigurationName()); + Set excludedModules = repositoryExtension.getExcludedModules().getOrElse(Collections.emptySet()); + Map forcedVersions = repositoryExtension.getModuleToConfigVersion().getOrElse(Collections.emptyMap()); + return serviceProvider.map(repo -> classpath.getIncoming().getResolutionResult().getAllComponents().stream().flatMap(component -> { + ModuleVersionIdentifier moduleVersion = component.getModuleVersion(); + return repo.findConfigurationsFor(query -> { + String module = Objects.requireNonNull(moduleVersion).getGroup() + ":" + moduleVersion.getName(); + if (!excludedModules.contains(module)) { + query.forArtifact(artifact -> { + artifact.gav(module + ":" + moduleVersion.getVersion()); + if (forcedVersions.containsKey(module)) { + artifact.forceConfigVersion(forcedVersions.get(module)); + } + }); + } + query.useLatestConfigWhenVersionIsUntested(); + }).stream() + .filter(org.graalvm.reachability.Configuration::isOverride) + .flatMap(configuration -> Stream.of( + "--exclude-config", + "/" + moduleVersion.getName() + "-" + moduleVersion.getVersion() + "\\.jar$", + "^/META-INF/native-image/")); + }).collect(Collectors.toList())); + } + } + return project.getProviders().provider(Collections::emptyList); + })); + } + private static LogLevel determineLogLevel() { LogLevel logLevel = LogLevel.DEBUG; String loggingProperty = System.getProperty(CONFIG_REPO_LOGLEVEL); diff --git a/native-gradle-plugin/src/main/java/org/graalvm/buildtools/gradle/internal/GraalVMReachabilityMetadataService.java b/native-gradle-plugin/src/main/java/org/graalvm/buildtools/gradle/internal/GraalVMReachabilityMetadataService.java index d348f0b31..c6c482388 100644 --- a/native-gradle-plugin/src/main/java/org/graalvm/buildtools/gradle/internal/GraalVMReachabilityMetadataService.java +++ b/native-gradle-plugin/src/main/java/org/graalvm/buildtools/gradle/internal/GraalVMReachabilityMetadataService.java @@ -41,6 +41,7 @@ package org.graalvm.buildtools.gradle.internal; import org.graalvm.buildtools.utils.FileUtils; +import org.graalvm.reachability.Configuration; import org.graalvm.reachability.GraalVMReachabilityMetadataRepository; import org.graalvm.reachability.Query; import org.graalvm.reachability.internal.FileSystemRepository; @@ -169,17 +170,17 @@ public void log(String groupId, String artifactId, String version, Supplier findConfigurationDirectoriesFor(Consumer queryBuilder) { - return repository.findConfigurationDirectoriesFor(queryBuilder); + public Set findConfigurationsFor(Consumer queryBuilder) { + return repository.findConfigurationsFor(queryBuilder); } /** @@ -189,11 +190,11 @@ public Set findConfigurationDirectoriesFor(Consumer queryBu * Never null. * * @param gavCoordinates the artifact GAV coordinates (group:artifact:version) - * @return a list of configuration directories + * @return a list of configurations */ @Override - public Set findConfigurationDirectoriesFor(String gavCoordinates) { - return repository.findConfigurationDirectoriesFor(gavCoordinates); + public Set findConfigurationsFor(String gavCoordinates) { + return repository.findConfigurationsFor(gavCoordinates); } /** @@ -201,10 +202,10 @@ public Set findConfigurationDirectoriesFor(String gavCoordinates) { * as an argument. * * @param modules the list of modules - * @return the set of configuration directories + * @return the set of configurations */ @Override - public Set findConfigurationDirectoriesFor(Collection modules) { - return repository.findConfigurationDirectoriesFor(modules); + public Set findConfigurationsFor(Collection modules) { + return repository.findConfigurationsFor(modules); } } diff --git a/native-maven-plugin/src/main/java/org/graalvm/buildtools/maven/AbstractNativeMojo.java b/native-maven-plugin/src/main/java/org/graalvm/buildtools/maven/AbstractNativeMojo.java index 67162aabd..dd9499ed0 100644 --- a/native-maven-plugin/src/main/java/org/graalvm/buildtools/maven/AbstractNativeMojo.java +++ b/native-maven-plugin/src/main/java/org/graalvm/buildtools/maven/AbstractNativeMojo.java @@ -58,6 +58,7 @@ import org.graalvm.buildtools.utils.FileUtils; import org.graalvm.buildtools.utils.NativeImageUtils; import org.graalvm.buildtools.utils.SharedConstants; +import org.graalvm.reachability.Configuration; import org.graalvm.reachability.GraalVMReachabilityMetadataRepository; import org.graalvm.reachability.internal.FileSystemRepository; @@ -82,6 +83,7 @@ import java.util.Optional; import java.util.Set; import java.util.function.Supplier; +import java.util.regex.Pattern; import java.util.stream.Collectors; import java.util.stream.Stream; @@ -131,7 +133,7 @@ public abstract class AbstractNativeMojo extends AbstractMojo { protected final List imageClasspath; - protected final Set metadataRepositoryPaths; + protected final Set metadataRepositoryConfigurations; @Parameter(property = "debug", defaultValue = "false") protected boolean debug; @@ -192,7 +194,7 @@ public abstract class AbstractNativeMojo extends AbstractMojo { @Inject protected AbstractNativeMojo() { imageClasspath = new ArrayList<>(); - metadataRepositoryPaths = new HashSet<>(); + metadataRepositoryConfigurations = new HashSet<>(); useArgFile = SharedConstants.IS_WINDOWS; } @@ -526,9 +528,9 @@ public boolean isArtifactExcludedFromMetadataRepository(Artifact dependency) { } protected void maybeAddReachabilityMetadata(List configDirs) { - if (isMetadataRepositoryEnabled() && !metadataRepositoryPaths.isEmpty()) { - String arg = metadataRepositoryPaths.stream() - .map(Path::toAbsolutePath) + if (isMetadataRepositoryEnabled() && !metadataRepositoryConfigurations.isEmpty()) { + String arg = metadataRepositoryConfigurations.stream() + .map(configuration -> configuration.getDirectory().toAbsolutePath()) .map(Path::toFile) .map(File::getAbsolutePath) .collect(Collectors.joining(",")); @@ -540,7 +542,7 @@ protected void maybeAddReachabilityMetadata(List configDirs) { protected void maybeAddDependencyMetadata(Artifact dependency) { if (isMetadataRepositoryEnabled() && metadataRepository != null && !isArtifactExcludedFromMetadataRepository(dependency)) { - metadataRepositoryPaths.addAll(metadataRepository.findConfigurationDirectoriesFor(q -> { + Set configurations = metadataRepository.findConfigurationsFor(q -> { q.useLatestConfigWhenVersionIsUntested(); q.forArtifact(artifact -> { artifact.gav(String.join(":", @@ -549,7 +551,13 @@ protected void maybeAddDependencyMetadata(Artifact dependency) { dependency.getVersion())); getMetadataVersion(dependency).ifPresent(artifact::forceConfigVersion); }); - })); + }); + metadataRepositoryConfigurations.addAll(configurations); + if (configurations.stream().anyMatch(Configuration::isOverride)) { + buildArgs.add("--exclude-config"); + buildArgs.add(Pattern.quote(dependency.getFile().getAbsolutePath())); + buildArgs.add("^/META-INF/native-image/"); + } } }