From c06cdfe8f7e0daaff576ea8cd8ccb3584bde8aaf Mon Sep 17 00:00:00 2001 From: Bartosz Spyrko-Smietanko Date: Thu, 3 Oct 2024 14:20:00 +0100 Subject: [PATCH] [#292] During channel initialisation the version of resolved manifest is made available --- .../java/org/wildfly/channel/ChannelImpl.java | 95 +++++++++++++++++-- .../org/wildfly/channel/ChannelSession.java | 20 +++- .../channel/ChannelSessionInitTestCase.java | 29 ++++++ .../channel/ChannelWithBlocklistTestCase.java | 24 +++-- 4 files changed, 149 insertions(+), 19 deletions(-) diff --git a/core/src/main/java/org/wildfly/channel/ChannelImpl.java b/core/src/main/java/org/wildfly/channel/ChannelImpl.java index 380ea581..190aec45 100644 --- a/core/src/main/java/org/wildfly/channel/ChannelImpl.java +++ b/core/src/main/java/org/wildfly/channel/ChannelImpl.java @@ -16,6 +16,7 @@ */ package org.wildfly.channel; +import static java.util.Collections.singleton; import static java.util.Objects.requireNonNull; import static org.wildfly.channel.version.VersionMatcher.COMPARATOR; @@ -24,6 +25,7 @@ import java.util.ArrayList; import java.util.Collections; import java.util.HashMap; +import java.util.HashSet; import java.util.List; import java.util.Map; import java.util.Optional; @@ -41,7 +43,8 @@ class ChannelImpl implements AutoCloseable { private static final Logger LOG = Logger.getLogger(ChannelImpl.class); - private Channel channelDefinition; + private final Channel channelDefinition; + private Channel resolvedChannel; private List requiredChannels = Collections.emptyList(); @@ -53,6 +56,7 @@ class ChannelImpl implements AutoCloseable { private boolean dependency = false; public Optional blocklist = Optional.empty(); + private ChannelManifestCoordinate resolvedCoordinate; public ChannelManifest getManifest() { return channelManifest; @@ -78,8 +82,11 @@ void init(MavenVersionsResolver.Factory factory, List channels) { resolver = factory.create(channelDefinition.getRepositories()); + final Channel.Builder resolvedChannelBuilder = new Channel.Builder(channelDefinition); if (channelDefinition.getManifestCoordinate() != null) { - channelManifest = resolveManifest(channelDefinition.getManifestCoordinate()); + final ChannelManifestCoordinate coordinate = resolveManifestVersion(channelDefinition); + resolvedChannelBuilder.setManifestCoordinate(coordinate); + channelManifest = resolveManifest(coordinate); } else { channelManifest = new ChannelManifest(null, null, null, Collections.emptyList()); } @@ -95,11 +102,14 @@ void init(MavenVersionsResolver.Factory factory, List channels) { if (channelDefinition.getBlocklistCoordinate() != null) { BlocklistCoordinate blocklistCoordinate = channelDefinition.getBlocklistCoordinate(); + resolvedChannelBuilder.setBlocklistCoordinate(blocklistCoordinate); final List urls = resolver.resolveChannelMetadata(List.of(blocklistCoordinate)); this.blocklist = urls.stream() .map(Blocklist::from) .findFirst(); } + + this.resolvedChannel = resolvedChannelBuilder.build(); } private ChannelImpl findRequiredChannel(MavenVersionsResolver.Factory factory, List channels, ManifestRequirement manifestRequirement) { @@ -179,8 +189,8 @@ boolean isDependency() { return dependency; } - Channel getChannelDefinition() { - return channelDefinition; + Channel getResolvedChannelDefinition() { + return resolvedChannel; } static class ResolveLatestVersionResult { @@ -193,8 +203,79 @@ static class ResolveLatestVersionResult { } } + private Set attemptedRepositories() { + return new HashSet<>(channelDefinition.getRepositories()); + } + + private ChannelManifestCoordinate resolveManifestVersion(Channel baseDefinition) { + final ChannelManifestCoordinate manifestCoordinate = baseDefinition.getManifestCoordinate(); + + // if we already have a version or it is a URL blocklist, return it + if (manifestCoordinate.getUrl() != null || manifestCoordinate.getMaven().getVersion() != null) { + return manifestCoordinate; + } + + final Set allVersions = resolver.getAllVersions( + manifestCoordinate.getGroupId(), + manifestCoordinate.getArtifactId(), + manifestCoordinate.getExtension(), + manifestCoordinate.getClassifier() + ); + Optional latestVersion = VersionMatcher.getLatestVersion(allVersions); + String version = latestVersion.orElseThrow(() -> + new ArtifactTransferException(String.format("Unable to resolve the latest version of channel metadata %s:%s", manifestCoordinate.getGroupId(), manifestCoordinate.getArtifactId()), + singleton(new ArtifactCoordinate(manifestCoordinate.getGroupId(), manifestCoordinate.getArtifactId(), manifestCoordinate.getExtension(), manifestCoordinate.getClassifier(), "")), + attemptedRepositories())); + return new ChannelManifestCoordinate(manifestCoordinate.getGroupId(), manifestCoordinate.getArtifactId(), version); + + } + + private BlocklistCoordinate resolveBlocklistVersion(Channel baseDefinition) { + final BlocklistCoordinate blocklistCoordinate = baseDefinition.getBlocklistCoordinate(); + + if (blocklistCoordinate == null) { + return null; + } + + // if we already have a version or it is a URL blocklist, return it + if (blocklistCoordinate.getUrl() != null || blocklistCoordinate.getMaven().getVersion() != null) { + return blocklistCoordinate; + } + + + final Set allVersions = resolver.getAllVersions( + blocklistCoordinate.getGroupId(), + blocklistCoordinate.getArtifactId(), + blocklistCoordinate.getExtension(), + blocklistCoordinate.getClassifier() + ); + Optional latestVersion = VersionMatcher.getLatestVersion(allVersions); + String version = latestVersion.orElseThrow(() -> + new ArtifactTransferException(String.format("Unable to resolve the latest version of channel metadata %s:%s", blocklistCoordinate.getGroupId(), blocklistCoordinate.getArtifactId()), + singleton(new ArtifactCoordinate(blocklistCoordinate.getGroupId(), blocklistCoordinate.getArtifactId(), blocklistCoordinate.getExtension(), blocklistCoordinate.getClassifier(), "")), + attemptedRepositories())); + return new BlocklistCoordinate(blocklistCoordinate.getGroupId(), blocklistCoordinate.getArtifactId(), version); + + } + private ChannelManifest resolveManifest(ChannelManifestCoordinate manifestCoordinate) throws UnresolvedMavenArtifactException { - return resolver.resolveChannelMetadata(List.of(manifestCoordinate)) + if (manifestCoordinate.getUrl() == null && manifestCoordinate.getMaven().getVersion() == null) { + final Set allVersions = resolver.getAllVersions( + manifestCoordinate.getGroupId(), + manifestCoordinate.getArtifactId(), + manifestCoordinate.getExtension(), + manifestCoordinate.getClassifier() + ); + Optional latestVersion = VersionMatcher.getLatestVersion(allVersions); + String version = latestVersion.orElseThrow(() -> + new ArtifactTransferException(String.format("Unable to resolve the latest version of channel metadata %s:%s", manifestCoordinate.getGroupId(), manifestCoordinate.getArtifactId()), + singleton(new ArtifactCoordinate(manifestCoordinate.getGroupId(), manifestCoordinate.getArtifactId(), manifestCoordinate.getExtension(), manifestCoordinate.getClassifier(), "")), + attemptedRepositories())); + resolvedCoordinate = new ChannelManifestCoordinate(manifestCoordinate.getGroupId(), manifestCoordinate.getArtifactId(), version); + } else { + resolvedCoordinate = manifestCoordinate; + } + return resolver.resolveChannelMetadata(List.of(resolvedCoordinate)) .stream() .map(ChannelManifestMapper::from) .findFirst().orElseThrow(); @@ -248,7 +329,7 @@ Optional resolveLatestVersion(String groupId, String return Optional.of(new ResolveLatestVersionResult(latestMetadataVersion, this)); } catch (NoStreamFoundException e) { LOG.debugf(e, "Metadata resolution for %s:%s failed in channel %s", - groupId, artifactId, this.getChannelDefinition().getName()); + groupId, artifactId, this.getResolvedChannelDefinition().getName()); return Optional.empty(); } case MAVEN_RELEASE: @@ -260,7 +341,7 @@ Optional resolveLatestVersion(String groupId, String return Optional.of(new ResolveLatestVersionResult(releaseMetadataVersion, this)); } catch (NoStreamFoundException e) { LOG.debugf(e, "Metadata resolution for %s:%s failed in channel %s", - groupId, artifactId, this.getChannelDefinition().getName()); + groupId, artifactId, this.getResolvedChannelDefinition().getName()); return Optional.empty(); } default: diff --git a/core/src/main/java/org/wildfly/channel/ChannelSession.java b/core/src/main/java/org/wildfly/channel/ChannelSession.java index 1eea2a21..689c50b1 100644 --- a/core/src/main/java/org/wildfly/channel/ChannelSession.java +++ b/core/src/main/java/org/wildfly/channel/ChannelSession.java @@ -90,6 +90,18 @@ public ChannelSession(List channelDefinitions, MavenVersionsResolver.Fa validateNoDuplicatedManifests(); } + /** + * Get the definitions of channels used by this session. Returned version contains resolved versions + * of channel metadata (if applicable). + * + * @return List of {@code Channel}s used to resolve artifacts by this session + */ + public List getResolvedChannelDefinitions() { + return this.channels.stream() + .map(ChannelImpl::getResolvedChannelDefinition) + .collect(Collectors.toList()); + } + /** * Resolve the Maven artifact according to the session's channels. *

@@ -121,7 +133,7 @@ public MavenArtifact resolveMavenArtifact(String groupId, String artifactId, Str ChannelImpl.ResolveArtifactResult artifact = channel.resolveArtifact(groupId, artifactId, extension, classifier, latestVersion); recorder.recordStream(groupId, artifactId, latestVersion); - return new MavenArtifact(groupId, artifactId, extension, classifier, latestVersion, artifact.file, artifact.channel.getChannelDefinition().getName()); + return new MavenArtifact(groupId, artifactId, extension, classifier, latestVersion, artifact.file, artifact.channel.getResolvedChannelDefinition().getName()); } /** @@ -153,7 +165,7 @@ public List resolveMavenArtifacts(List coordi final MavenArtifact resolvedArtifact = new MavenArtifact(request.getGroupId(), request.getArtifactId(), request.getExtension(), request.getClassifier(), request.getVersion(), resolveArtifactResults.get(i).file, - resolveArtifactResults.get(i).channel.getChannelDefinition().getName()); + resolveArtifactResults.get(i).channel.getResolvedChannelDefinition().getName()); recorder.recordStream(resolvedArtifact.getGroupId(), resolvedArtifact.getArtifactId(), resolvedArtifact.getVersion()); res.add(resolvedArtifact); @@ -227,7 +239,7 @@ public List resolveDirectMavenArtifacts(List */ public VersionResult findLatestMavenArtifactVersion(String groupId, String artifactId, String extension, String classifier, String baseVersion) throws NoStreamFoundException { final ChannelImpl.ResolveLatestVersionResult channelWithLatestVersion = findChannelWithLatestVersion(groupId, artifactId, extension, classifier, baseVersion); - return new VersionResult(channelWithLatestVersion.version, channelWithLatestVersion.channel.getChannelDefinition().getName()); + return new VersionResult(channelWithLatestVersion.version, channelWithLatestVersion.channel.getResolvedChannelDefinition().getName()); } @Override @@ -274,7 +286,7 @@ private ChannelImpl.ResolveLatestVersionResult findChannelWithLatestVersion(Stri return foundVersions.get(foundLatestVersionInChannels.orElseThrow(() -> { final ArtifactCoordinate coord = new ArtifactCoordinate(groupId, artifactId, extension, classifier, ""); final Set repositories = channels.stream() - .map(ChannelImpl::getChannelDefinition) + .map(ChannelImpl::getResolvedChannelDefinition) .flatMap(d -> d.getRepositories().stream()) .collect(Collectors.toSet()); throw new NoStreamFoundException( diff --git a/core/src/test/java/org/wildfly/channel/ChannelSessionInitTestCase.java b/core/src/test/java/org/wildfly/channel/ChannelSessionInitTestCase.java index 713ef385..0da5698b 100644 --- a/core/src/test/java/org/wildfly/channel/ChannelSessionInitTestCase.java +++ b/core/src/test/java/org/wildfly/channel/ChannelSessionInitTestCase.java @@ -28,7 +28,9 @@ import java.nio.file.Files; import java.nio.file.Path; import java.util.List; +import java.util.Set; +import static org.assertj.core.api.Assertions.assertThat; import static org.junit.jupiter.api.Assertions.assertEquals; import static org.junit.jupiter.api.Assertions.assertNotNull; import static org.junit.jupiter.api.Assertions.assertNull; @@ -359,6 +361,33 @@ public void duplicatedManifestIDsAreDetected() throws Exception { assertThrows(RuntimeException.class, () -> new ChannelSession(channels, factory)); } + @Test + public void getVersionOfResolvedManifest() throws Exception { + final MavenVersionsResolver.Factory factory = mock(MavenVersionsResolver.Factory.class); + MavenVersionsResolver resolver = mock(MavenVersionsResolver.class); + when(factory.create(any())).thenReturn(resolver); + + final ChannelManifest requiredManifest = new ManifestBuilder() + .setId("manifest-one") + .build(); + mockManifest(resolver, requiredManifest, "test.channels:base-manifest:1.0.0"); + when(resolver.getAllVersions("test.channels", "base-manifest", ChannelManifest.EXTENSION, ChannelManifest.CLASSIFIER)) + .thenReturn(Set.of("1.0.0")); + + final List channels = List.of(new Channel.Builder() + .setName("channel one") + .addRepository("test", "test") + .setManifestCoordinate("test.channels", "base-manifest") + .build()); + try (ChannelSession channelSession = new ChannelSession(channels, factory)) { + assertThat(channelSession.getResolvedChannelDefinitions()) + .map(Channel::getManifestCoordinate) + .map(ChannelManifestCoordinate::getVersion) + .containsOnly("1.0.0"); + + } + } + private void mockManifest(MavenVersionsResolver resolver, ChannelManifest manifest, String gav) throws IOException { mockManifest(resolver, ChannelManifestMapper.toYaml(manifest), gav); } diff --git a/core/src/test/java/org/wildfly/channel/ChannelWithBlocklistTestCase.java b/core/src/test/java/org/wildfly/channel/ChannelWithBlocklistTestCase.java index dd7a8917..be0d3aed 100644 --- a/core/src/test/java/org/wildfly/channel/ChannelWithBlocklistTestCase.java +++ b/core/src/test/java/org/wildfly/channel/ChannelWithBlocklistTestCase.java @@ -64,6 +64,7 @@ public void testFindLatestMavenArtifactVersion() throws Exception { " maven:\n" + " groupId: test\n" + " artifactId: 'test.manifest'\n" + + " version: '1.0.0'\n" + "repositories:\n" + " - id: test\n" + " url: http://test.te"); @@ -79,7 +80,7 @@ public void testFindLatestMavenArtifactVersion() throws Exception { " - groupId: org.wildfly\n" + " artifactId: wildfly-ee-galleon-pack\n" + " versionPattern: .*"); - when(resolver.resolveChannelMetadata(List.of(new ChannelManifestCoordinate("test", "test.manifest")))) + when(resolver.resolveChannelMetadata(List.of(new ChannelManifestCoordinate("test", "test.manifest", "1.0.0")))) .thenReturn(List.of(tempDir.resolve("manifest.yaml").toUri().toURL())); when(resolver.resolveChannelMetadata(List.of(new BlocklistCoordinate("org.wildfly", "wildfly-blocklist")))) @@ -109,6 +110,7 @@ public void testFindLatestMavenArtifactVersionBlocklistDoesntExist() throws Exce " maven:\n" + " groupId: test\n" + " artifactId: 'test.manifest'\n" + + " version: '1.0.0'\n" + "repositories:\n" + " - id: test\n" + " url: http://test.te"); @@ -124,7 +126,7 @@ public void testFindLatestMavenArtifactVersionBlocklistDoesntExist() throws Exce " - groupId: org.wildfly\n" + " artifactId: '*'\n" + " versionPattern: '25\\.\\d+\\.\\d+.Final'"); - when(resolver.resolveChannelMetadata(List.of(new ChannelManifestCoordinate("test", "test.manifest")))) + when(resolver.resolveChannelMetadata(List.of(new ChannelManifestCoordinate("test", "test.manifest", "1.0.0")))) .thenReturn(List.of(tempDir.resolve("manifest.yaml").toUri().toURL())); when(resolver.resolveChannelMetadata(List.of(new BlocklistCoordinate("org.wildfly", "wildfly-blocklist")))) @@ -156,6 +158,7 @@ public void testFindLatestMavenArtifactVersionWithWildcardBlocklist() throws Exc " maven:\n" + " groupId: test\n" + " artifactId: 'test.manifest'\n" + + " version: '1.0.0'\n" + "repositories:\n" + " - id: test\n" + " url: http://test.te"); @@ -171,7 +174,7 @@ public void testFindLatestMavenArtifactVersionWithWildcardBlocklist() throws Exc " - groupId: org.wildfly\n" + " artifactId: '*'\n" + " versionPattern: '25\\.\\d+\\.\\d+.Final'"); - when(resolver.resolveChannelMetadata(List.of(new ChannelManifestCoordinate("test", "test.manifest")))) + when(resolver.resolveChannelMetadata(List.of(new ChannelManifestCoordinate("test", "test.manifest", "1.0.0")))) .thenReturn(List.of(tempDir.resolve("manifest.yaml").toUri().toURL())); when(resolver.resolveChannelMetadata(List.of(new BlocklistCoordinate("org.wildfly", "wildfly-blocklist")))) @@ -201,6 +204,7 @@ public void testFindLatestMavenArtifactVersionBlocklistsAllVersionsException() t " maven:\n" + " groupId: test\n" + " artifactId: 'test.manifest'\n" + + " version: '1.0.0'\n" + "repositories:\n" + " - id: test\n" + " url: http://test.te"); @@ -216,7 +220,7 @@ public void testFindLatestMavenArtifactVersionBlocklistsAllVersionsException() t " - groupId: org.wildfly\n" + " artifactId: '*'\n" + " versionPattern: '25\\.\\d+\\.\\d+.Final'"); - when(resolver.resolveChannelMetadata(List.of(new ChannelManifestCoordinate("test", "test.manifest")))) + when(resolver.resolveChannelMetadata(List.of(new ChannelManifestCoordinate("test", "test.manifest", "1.0.0")))) .thenReturn(List.of(tempDir.resolve("manifest.yaml").toUri().toURL())); when(resolver.resolveChannelMetadata(List.of(new BlocklistCoordinate("org.wildfly", "wildfly-blocklist")))) @@ -249,6 +253,7 @@ public void testResolveLatestMavenArtifact() throws Exception { " maven:\n" + " groupId: test\n" + " artifactId: 'test.manifest'\n" + + " version: '1.0.0'\n" + "repositories:\n" + " - id: test\n" + " url: http://test.te"); @@ -264,7 +269,7 @@ public void testResolveLatestMavenArtifact() throws Exception { " - groupId: org.wildfly\n" + " artifactId: '*'\n" + " versionPattern: '25\\.\\d+\\.\\d+.Final'"); - when(resolver.resolveChannelMetadata(List.of(new ChannelManifestCoordinate("test", "test.manifest")))) + when(resolver.resolveChannelMetadata(List.of(new ChannelManifestCoordinate("test", "test.manifest", "1.0.0")))) .thenReturn(List.of(tempDir.resolve("manifest.yaml").toUri().toURL())); when(resolver.resolveChannelMetadata(List.of(new BlocklistCoordinate("org.wildfly", "wildfly-blocklist")))) @@ -304,6 +309,7 @@ public void testResolveLatestMavenArtifactThrowUnresolvedMavenArtifactException( " maven:\n" + " groupId: test\n" + " artifactId: 'test.manifest'\n" + + " version: '1.0.0'\n" + "repositories:\n" + " - id: test\n" + " url: http://test.te"); @@ -319,7 +325,7 @@ public void testResolveLatestMavenArtifactThrowUnresolvedMavenArtifactException( " - groupId: org.wildfly\n" + " artifactId: '*'\n" + " versionPattern: '25\\.\\d+\\.\\d+.Final'"); - when(resolver.resolveChannelMetadata(List.of(new ChannelManifestCoordinate("test", "test.manifest")))) + when(resolver.resolveChannelMetadata(List.of(new ChannelManifestCoordinate("test", "test.manifest", "1.0.0")))) .thenReturn(List.of(tempDir.resolve("manifest.yaml").toUri().toURL())); when(resolver.resolveChannelMetadata(List.of(new BlocklistCoordinate("org.wildfly", "wildfly-blocklist")))) @@ -352,6 +358,7 @@ public void testResolveMavenArtifactsFromOneChannel() throws Exception { " maven:\n" + " groupId: test\n" + " artifactId: 'test.manifest'\n" + + " version: '1.0.0'\n" + "repositories:\n" + " - id: test\n" + " url: http://test.te"); @@ -370,7 +377,7 @@ public void testResolveMavenArtifactsFromOneChannel() throws Exception { " - groupId: org.wildfly\n" + " artifactId: wildfly-cli\n" + " version: \"26.0.0.Final\""); - when(resolver.resolveChannelMetadata(List.of(new ChannelManifestCoordinate("test", "test.manifest")))) + when(resolver.resolveChannelMetadata(List.of(new ChannelManifestCoordinate("test", "test.manifest", "1.0.0")))) .thenReturn(List.of(tempDir.resolve("manifest.yaml").toUri().toURL())); when(resolver.resolveChannelMetadata(List.of(new BlocklistCoordinate("org.wildfly", "wildfly-blocklist")))) @@ -476,6 +483,7 @@ public void testChannelWithInvalidBlacklist() throws Exception { " maven:\n" + " groupId: test\n" + " artifactId: 'test.manifest'\n" + + " version: '1.0.0'\n" + "repositories:\n" + " - id: test\n" + " url: http://test.te"); @@ -491,7 +499,7 @@ public void testChannelWithInvalidBlacklist() throws Exception { " - groupId: org.wildfly\n" + " artifactId: '*'\n" + " versionPattern: '25\\.\\d+\\.\\d+.Final'"); - when(resolver.resolveChannelMetadata(List.of(new ChannelManifestCoordinate("test", "test.manifest")))) + when(resolver.resolveChannelMetadata(List.of(new ChannelManifestCoordinate("test", "test.manifest", "1.0.0")))) .thenReturn(List.of(tempDir.resolve("manifest.yaml").toUri().toURL())); when(resolver.resolveChannelMetadata(List.of(new BlocklistCoordinate("org.wildfly", "wildfly-blocklist"))))