org.ops4j.pax.url.mvn.internal.AetherBasedResolver Maven / Gradle / Ivy
/*
* Copyright (C) 2010 Toni Menzel
* Copyright (C) 2014 Guillaume Nodet
*
* 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.
*/
package org.ops4j.pax.url.mvn.internal;
import java.io.File;
import java.io.FileFilter;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.net.ConnectException;
import java.net.MalformedURLException;
import java.net.NoRouteToHostException;
import java.net.SocketTimeoutException;
import java.nio.file.Files;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.Comparator;
import java.util.Deque;
import java.util.HashMap;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.TreeSet;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentLinkedDeque;
import java.util.concurrent.ConcurrentMap;
import org.apache.http.impl.client.CloseableHttpClient;
import org.apache.maven.artifact.repository.metadata.SnapshotVersion;
import org.apache.maven.artifact.repository.metadata.Versioning;
import org.apache.maven.artifact.repository.metadata.io.xpp3.MetadataXpp3Reader;
import org.apache.maven.artifact.repository.metadata.io.xpp3.MetadataXpp3Writer;
import org.apache.maven.repository.internal.MavenRepositorySystemUtils;
import org.apache.maven.settings.Mirror;
import org.apache.maven.settings.Server;
import org.apache.maven.settings.Settings;
import org.apache.maven.settings.crypto.DefaultSettingsDecryptionRequest;
import org.apache.maven.settings.crypto.SettingsDecrypter;
import org.apache.maven.settings.crypto.SettingsDecryptionRequest;
import org.apache.maven.settings.crypto.SettingsDecryptionResult;
import org.codehaus.plexus.util.xml.Xpp3Dom;
import org.eclipse.aether.ConfigurationProperties;
import org.eclipse.aether.DefaultRepositorySystemSession;
import org.eclipse.aether.RepositoryException;
import org.eclipse.aether.RepositorySystem;
import org.eclipse.aether.RepositorySystemSession;
import org.eclipse.aether.artifact.Artifact;
import org.eclipse.aether.artifact.DefaultArtifact;
import org.eclipse.aether.connector.basic.BasicRepositoryConnectorFactory;
import org.eclipse.aether.impl.DefaultServiceLocator;
import org.eclipse.aether.installation.InstallRequest;
import org.eclipse.aether.internal.impl.PaxLocalRepositoryManager;
import org.eclipse.aether.internal.impl.slf4j.Slf4jLoggerFactory;
import org.eclipse.aether.metadata.DefaultMetadata;
import org.eclipse.aether.metadata.Metadata;
import org.eclipse.aether.repository.Authentication;
import org.eclipse.aether.repository.LocalMetadataRequest;
import org.eclipse.aether.repository.LocalRepository;
import org.eclipse.aether.repository.LocalRepositoryManager;
import org.eclipse.aether.repository.MirrorSelector;
import org.eclipse.aether.repository.Proxy;
import org.eclipse.aether.repository.ProxySelector;
import org.eclipse.aether.repository.RemoteRepository;
import org.eclipse.aether.repository.RepositoryPolicy;
import org.eclipse.aether.resolution.ArtifactRequest;
import org.eclipse.aether.resolution.ArtifactResolutionException;
import org.eclipse.aether.resolution.ArtifactResult;
import org.eclipse.aether.resolution.MetadataRequest;
import org.eclipse.aether.resolution.MetadataResult;
import org.eclipse.aether.resolution.VersionRangeRequest;
import org.eclipse.aether.resolution.VersionRangeResolutionException;
import org.eclipse.aether.resolution.VersionRangeResult;
import org.eclipse.aether.spi.connector.RepositoryConnectorFactory;
import org.eclipse.aether.spi.connector.transport.TransporterFactory;
import org.eclipse.aether.spi.localrepo.LocalRepositoryManagerFactory;
import org.eclipse.aether.transfer.ArtifactNotFoundException;
import org.eclipse.aether.transfer.ArtifactTransferException;
import org.eclipse.aether.transfer.MetadataNotFoundException;
import org.eclipse.aether.transfer.MetadataTransferException;
import org.eclipse.aether.transport.wagon.WagonProvider;
import org.eclipse.aether.transport.wagon.WagonTransporterFactory;
import org.eclipse.aether.util.repository.AuthenticationBuilder;
import org.eclipse.aether.util.repository.DefaultMirrorSelector;
import org.eclipse.aether.util.repository.DefaultProxySelector;
import org.eclipse.aether.util.version.GenericVersionScheme;
import org.eclipse.aether.version.InvalidVersionSpecificationException;
import org.eclipse.aether.version.Version;
import org.eclipse.aether.version.VersionConstraint;
import org.ops4j.lang.NullArgumentException;
import org.ops4j.pax.url.mvn.MavenResolver;
import org.ops4j.pax.url.mvn.MirrorInfo;
import org.ops4j.pax.url.mvn.ServiceConstants;
import org.ops4j.pax.url.mvn.internal.config.MavenConfiguration;
import org.ops4j.pax.url.mvn.internal.config.MavenRepositoryURL;
import org.slf4j.LoggerFactory;
import org.sonatype.plexus.components.cipher.DefaultPlexusCipher;
import org.sonatype.plexus.components.cipher.PlexusCipherException;
import static org.eclipse.aether.repository.RepositoryPolicy.CHECKSUM_POLICY_FAIL;
import static org.eclipse.aether.repository.RepositoryPolicy.CHECKSUM_POLICY_IGNORE;
import static org.eclipse.aether.repository.RepositoryPolicy.CHECKSUM_POLICY_WARN;
import static org.eclipse.aether.repository.RepositoryPolicy.UPDATE_POLICY_ALWAYS;
import static org.eclipse.aether.repository.RepositoryPolicy.UPDATE_POLICY_DAILY;
import static org.eclipse.aether.repository.RepositoryPolicy.UPDATE_POLICY_INTERVAL;
import static org.eclipse.aether.repository.RepositoryPolicy.UPDATE_POLICY_NEVER;
import static org.ops4j.pax.url.mvn.internal.Parser.VERSION_LATEST;
/**
* Aether based, drop in replacement for mvn protocol
*/
public class AetherBasedResolver implements MavenResolver {
private static final org.slf4j.Logger LOG = LoggerFactory.getLogger(AetherBasedResolver.class);
private static final String LATEST_VERSION_RANGE = "[0.0,)";
private static final String REPO_TYPE = "default";
private static final String SCHEMA_HTTP = "http";
private static final String SCHEMA_HTTPS = "https";
private static final String PROXY_HOST = "proxyHost";
private static final String PROXY_PORT = "proxyPort";
private static final String PROXY_USER = "proxyUser";
private static final String PROXY_PASSWORD = "proxyPassword";
private static final String NON_PROXY_HOSTS = "nonProxyHosts";
final private RepositorySystem m_repoSystem;
final private MavenConfiguration m_config;
final private MirrorSelector m_mirrorSelector;
final private ProxySelector m_proxySelector;
final private CloseableHttpClient m_client;
private Settings m_settings;
private ConfigurableSettingsDecrypter decrypter;
private LocalRepository localRepository;
private final ConcurrentMap> sessions
= new ConcurrentHashMap>();
/**
* Create a AetherBasedResolver
*
* @param configuration (must be not null)
*/
public AetherBasedResolver(final MavenConfiguration configuration) {
this(configuration, null);
}
/**
* Create a AetherBasedResolver
*
* @param configuration (must be not null)
*/
public AetherBasedResolver(final MavenConfiguration configuration, final MirrorInfo mirror) {
NullArgumentException.validateNotNull(configuration, "Maven configuration");
m_client = HttpClients.createClient(configuration.getPropertyResolver(), configuration.getPid());
m_config = configuration;
m_settings = configuration.getSettings();
m_repoSystem = newRepositorySystem();
decryptSettings();
m_proxySelector = selectProxies();
m_mirrorSelector = selectMirrors(mirror);
}
@Override
public void close() throws IOException {
m_client.close();
}
private void decryptSettings() {
SettingsDecryptionRequest request = new DefaultSettingsDecryptionRequest(m_settings);
SettingsDecryptionResult result = decrypter.decrypt(request);
m_settings.setProxies(result.getProxies());
m_settings.setServers(result.getServers());
}
private void assignProxyAndMirrors(List remoteRepos) {
Map> map = new HashMap>();
Map naming = new HashMap();
boolean aggregateReleaseEnabled = false, aggregateSnapshotEnabled = false;
String aggregateReleaseUpdateInterval = null, aggregateSnapshotUpdateInterval = null;
String aggregateReleaseChecksumPolicy = null, aggregateSnapshotChecksumPolicy = null;
List resultingRepos = new ArrayList();
for (RemoteRepository r : remoteRepos) {
naming.put(r.getId(), r);
RemoteRepository rProxy = new RemoteRepository.Builder(r).setProxy(
m_proxySelector.getProxy(r)).build();
resultingRepos.add(rProxy);
RemoteRepository mirror = m_mirrorSelector.getMirror(r);
if (mirror != null) {
String key = mirror.getId();
naming.put(key, mirror);
if (!map.containsKey(key)) {
map.put(key, new ArrayList());
}
List mirrored = map.get(key);
mirrored.add(r.getId());
// Aggregate policy settings of the mirror repos.
aggregateReleaseEnabled |= r.getPolicy(false).isEnabled();
aggregateSnapshotEnabled |= r.getPolicy(true).isEnabled();
aggregateReleaseUpdateInterval = minUpdateInterval(aggregateReleaseUpdateInterval, r.getPolicy(false).getUpdatePolicy());
aggregateSnapshotUpdateInterval = minUpdateInterval(aggregateSnapshotUpdateInterval, r.getPolicy(true).getUpdatePolicy());
aggregateReleaseChecksumPolicy = aggregateChecksumPolicy(aggregateReleaseChecksumPolicy, r.getPolicy(false).getChecksumPolicy());
aggregateSnapshotChecksumPolicy = aggregateChecksumPolicy(aggregateSnapshotChecksumPolicy, r.getPolicy(true).getChecksumPolicy());
}
}
for (String mirrorId : map.keySet()) {
RemoteRepository mirror = naming.get(mirrorId);
List mirroredRepos = new ArrayList();
for (String rep : map.get(mirrorId)) {
mirroredRepos.add(naming.get(rep));
}
RepositoryPolicy releasePolicy = new RepositoryPolicy(aggregateReleaseEnabled, aggregateReleaseUpdateInterval, aggregateReleaseChecksumPolicy);
RepositoryPolicy snapshotPolicy = new RepositoryPolicy(aggregateSnapshotEnabled, aggregateSnapshotUpdateInterval, aggregateSnapshotChecksumPolicy);
mirror = new RemoteRepository.Builder(mirror).setMirroredRepositories(mirroredRepos)
.setProxy(m_proxySelector.getProxy(mirror))
.setReleasePolicy(releasePolicy)
.setSnapshotPolicy(snapshotPolicy)
.build();
resultingRepos.removeAll(mirroredRepos);
resultingRepos.add(0, mirror);
}
remoteRepos.clear();
remoteRepos.addAll(resultingRepos);
}
private String minUpdateInterval(String interval1, String interval2) {
LOG.debug("interval1: {}, interval2: {}", interval1, interval2);
if (interval1 == null) {
return interval2;
} else if (interval2 == null) {
return interval1;
}
int interval1InMin = getIntervalInMinutes(interval1);
int interval2InMin = getIntervalInMinutes(interval2);
if (interval1InMin <= interval2InMin) {
return getUpdatePolicyInterval(interval1InMin);
} else {
return getUpdatePolicyInterval(interval2InMin);
}
}
private int getIntervalInMinutes(String interval) {
int intervalInMin;
if (interval.equals(UPDATE_POLICY_NEVER)) {
intervalInMin = Integer.MAX_VALUE;
} else if (interval.equals(UPDATE_POLICY_DAILY)) {
intervalInMin = 24 * 60;
} else if (interval.equals(UPDATE_POLICY_ALWAYS)) {
intervalInMin = Integer.MIN_VALUE;
} else if (interval.startsWith(UPDATE_POLICY_INTERVAL + ":")) {
try {
intervalInMin = Integer.parseInt(interval.substring(UPDATE_POLICY_INTERVAL.length() + 1));
} catch (NumberFormatException e) {
LOG.warn("unable to parse update policy interval: \"{}\"", interval);
intervalInMin = 24 * 60;
}
} else {
throw new IllegalArgumentException(String.format("Invalid update policy \"%s\"", interval));
}
return intervalInMin;
}
private String getUpdatePolicyInterval(int intervalInMin) {
switch (intervalInMin) {
case Integer.MAX_VALUE:
return UPDATE_POLICY_NEVER;
case Integer.MIN_VALUE:
return UPDATE_POLICY_ALWAYS;
case 24 * 60:
return UPDATE_POLICY_DAILY;
default:
return String.format("%s:%d", UPDATE_POLICY_INTERVAL, intervalInMin);
}
}
private String aggregateChecksumPolicy(String policy1, String policy2) {
if (policy1 == null) {
return policy2;
}
if (policy2 == null) {
return policy1;
}
if (policy1.equals(CHECKSUM_POLICY_FAIL) || policy2.equals(CHECKSUM_POLICY_FAIL)) {
return CHECKSUM_POLICY_FAIL;
} else if (policy1.equals(CHECKSUM_POLICY_WARN) || policy2.equals(CHECKSUM_POLICY_WARN)) {
return CHECKSUM_POLICY_WARN;
} else {
return CHECKSUM_POLICY_IGNORE;
}
}
private ProxySelector selectProxies() {
DefaultProxySelector proxySelector = new DefaultProxySelector();
for (org.apache.maven.settings.Proxy proxy : m_settings.getProxies()) {
if (!proxy.isActive()) {
continue;
}
String nonProxyHosts = proxy.getNonProxyHosts();
Proxy proxyObj = new Proxy(proxy.getProtocol(), proxy.getHost(), proxy.getPort(),
getAuthentication(proxy));
proxySelector.add(proxyObj, nonProxyHosts);
}
if (m_settings.getProxies().size() == 0) {
javaDefaultProxy(proxySelector);
}
return proxySelector;
}
private void javaDefaultProxy(DefaultProxySelector proxySelector) {
// Prefer https
String proxyHost = System.getProperty(SCHEMA_HTTPS + "." + PROXY_HOST);
String schema = (proxyHost != null) ? SCHEMA_HTTPS : SCHEMA_HTTP;
if (proxyHost == null) {
proxyHost = System.getProperty(schema + "." + PROXY_HOST);
}
if (proxyHost == null) {
return;
}
String proxyUser = System.getProperty(schema + "." + PROXY_USER);
String proxyPassword = System.getProperty(schema + "." + PROXY_PASSWORD);
int proxyPort = Integer.parseInt(System.getProperty(schema + "." + PROXY_PORT, "8080"));
String nonProxyHosts = System.getProperty(schema + "." + NON_PROXY_HOSTS);
Authentication authentication = createAuthentication(proxyUser, proxyPassword);
Proxy proxyObj = new Proxy(schema, proxyHost, proxyPort, authentication);
proxySelector.add(proxyObj, nonProxyHosts);
}
private Authentication createAuthentication(String proxyUser, String proxyPassword) {
Authentication authentication = null;
if (proxyUser != null) {
authentication = new AuthenticationBuilder()
.addUsername(proxyUser)
.addPassword(proxyPassword).build();
}
return authentication;
}
private MirrorSelector selectMirrors(MirrorInfo mirror) {
// configure mirror
// The class org.eclipse.aether.util.repository.DefaultMirrorSelector is final therefore it needs to be
// wrapped to fix PAXURL-289.
class DefaultMirrorSelectorWrapper implements MirrorSelector {
final DefaultMirrorSelector delegate = new DefaultMirrorSelector();
final Map authMap = new HashMap();
@Override
public RemoteRepository getMirror(RemoteRepository repository) {
RemoteRepository repo = delegate.getMirror(repository);
if (repo != null) {
Authentication mirrorAuth = authMap.get(repo.getId());
if (mirrorAuth != null) {
RemoteRepository.Builder builder = new RemoteRepository.Builder(repo);
repo = builder.setAuthentication(mirrorAuth).build();
}
}
return repo;
}
public DefaultMirrorSelector add(String id, String url, String type, boolean repositoryManager,
String mirrorOfIds, String mirrorOfTypes, Authentication authentication) {
LOG.trace("adding mirror {} auth = {}", id, authentication != null);
if (authentication != null) {
authMap.put(id, authentication);
}
return delegate.add(id, url, type, repositoryManager, mirrorOfIds, mirrorOfTypes);
}
}
DefaultMirrorSelectorWrapper selector = new DefaultMirrorSelectorWrapper();
for (Mirror m : m_settings.getMirrors()) {
selector.add(m.getId(), m.getUrl(), null, false, m.getMirrorOf(), "*", getAuthentication(m.getId()));
}
if (mirror != null) {
selector.add(mirror.getId(), mirror.getUrl(), null, false, mirror.getMirrorOf(), "*", getAuthentication(mirror.getId()));
}
return selector;
}
private List selectRepositories() {
List list = new ArrayList();
List urls = Collections.emptyList();
try {
urls = m_config.getRepositories();
} catch (MalformedURLException exc) {
LOG.error("invalid repository URLs", exc);
}
for (MavenRepositoryURL r : urls) {
if (r.isMulti()) {
addSubDirs(list, r.getFile());
} else {
addRepo(list, r);
}
}
return list;
}
List selectDefaultRepositories() {
List list = new ArrayList();
List urls = Collections.emptyList();
try {
urls = m_config.getDefaultRepositories();
} catch (MalformedURLException exc) {
LOG.error("invalid repository URLs", exc);
}
for (MavenRepositoryURL r : urls) {
if (r.isMulti()) {
addLocalSubDirs(list, r.getFile());
} else {
addLocalRepo(list, r);
}
}
return list;
}
private void addSubDirs(List list, File parentDir) {
if (!parentDir.isDirectory()) {
LOG.debug("Repository marked with @multi does not resolve to a directory: "
+ parentDir);
return;
}
for (File repo : getSortedChildDirectories(parentDir)) {
try {
String repoURI = repo.toURI().toString() + "@id=" + repo.getName();
LOG.debug("Adding repo from inside multi dir: " + repoURI);
addRepo(list, new MavenRepositoryURL(repoURI));
} catch (MalformedURLException e) {
LOG.error("Error resolving repo url of a multi repo " + repo.toURI());
}
}
}
/**
* For the given parent, we find all child files, and then
* sort those files by their name (not absolute path).
*
* The sorted list is returned, or an empty list if listFiles returns
* null.
* @param parent A non-null parent File for which you want to get the sorted list of child directories.
* @return The alphabetically sorted list of files, or an empty list if parent.listFiles() returns null.
*/
private static File[] getSortedChildDirectories(File parent) {
File[] files = parent.listFiles(new FileFilter() {
@Override
public boolean accept(File pathname) {
return pathname.isDirectory();
}
});
if (files == null) {
return new File[0];
}
Arrays.sort(files, new Comparator() {
@Override
public int compare(File o1, File o2) {
return o1.getName().compareTo(o2.getName());
}
});
return files;
}
private void addRepo(List list, MavenRepositoryURL repo) {
String releasesUpdatePolicy = repo.getReleasesUpdatePolicy();
if (releasesUpdatePolicy == null || releasesUpdatePolicy.isEmpty()) {
releasesUpdatePolicy = UPDATE_POLICY_DAILY;
}
String releasesChecksumPolicy = repo.getReleasesChecksumPolicy();
if (releasesChecksumPolicy == null || releasesChecksumPolicy.isEmpty()) {
releasesChecksumPolicy = CHECKSUM_POLICY_WARN;
}
String snapshotsUpdatePolicy = repo.getSnapshotsUpdatePolicy();
if (snapshotsUpdatePolicy == null || snapshotsUpdatePolicy.isEmpty()) {
snapshotsUpdatePolicy = UPDATE_POLICY_DAILY;
}
String snapshotsChecksumPolicy = repo.getSnapshotsChecksumPolicy();
if (snapshotsChecksumPolicy == null || snapshotsChecksumPolicy.isEmpty()) {
snapshotsChecksumPolicy = CHECKSUM_POLICY_WARN;
}
RemoteRepository.Builder builder = new RemoteRepository.Builder(repo.getId(), REPO_TYPE, repo.getURL().toExternalForm());
RepositoryPolicy releasePolicy = new RepositoryPolicy(repo.isReleasesEnabled(), releasesUpdatePolicy, releasesChecksumPolicy);
builder.setReleasePolicy(releasePolicy);
RepositoryPolicy snapshotPolicy = new RepositoryPolicy(repo.isSnapshotsEnabled(), snapshotsUpdatePolicy, snapshotsChecksumPolicy);
builder.setSnapshotPolicy(snapshotPolicy);
Authentication authentication = getAuthentication(repo.getId());
if (authentication != null) {
builder.setAuthentication(authentication);
}
list.add(builder.build());
}
private void addLocalSubDirs(List list, File parentDir) {
if (!parentDir.isDirectory()) {
LOG.debug("Repository marked with @multi does not resolve to a directory: "
+ parentDir);
return;
}
for (File repo : getSortedChildDirectories(parentDir)) {
try {
String repoURI = repo.toURI().toString() + "@id=" + repo.getName();
LOG.debug("Adding repo from inside multi dir: " + repoURI);
addLocalRepo(list, new MavenRepositoryURL(repoURI));
} catch (MalformedURLException e) {
LOG.error("Error resolving repo url of a multi repo " + repo.toURI());
}
}
}
private void addLocalRepo(List list, MavenRepositoryURL repo) {
if (repo.getFile() != null) {
LocalRepository local = new LocalRepository(repo.getFile(), "simple");
list.add(local);
}
}
public RepositorySystem getRepositorySystem() {
return m_repoSystem;
}
public List getRepositories() {
List repos = selectRepositories();
assignProxyAndMirrors(repos);
return repos;
}
@Override
public File resolve(String url) throws IOException {
return resolve(url, null);
}
@Override
public File resolve(String url, Exception previousException) throws IOException {
if (!url.startsWith(ServiceConstants.PROTOCOL + ":")) {
throw new IllegalArgumentException("url should be a mvn based url");
}
url = url.substring((ServiceConstants.PROTOCOL + ":").length());
Parser parser = new Parser(url);
return resolve(
parser.getGroup(),
parser.getArtifact(),
parser.getClassifier(),
parser.getType(),
parser.getVersion(),
parser.getRepositoryURL(),
previousException
);
}
/**
* Resolve maven artifact as file in repository.
*/
@Override
public File resolve(String groupId, String artifactId, String classifier,
String extension, String version) throws IOException {
return resolve(groupId, artifactId, classifier, extension, version, null, null);
}
@Override
public File resolve(String groupId, String artifactId, String classifier, String extension, String version, Exception previousException) throws IOException {
return resolve(groupId, artifactId, classifier, extension, version, null, previousException);
}
/**
* Resolve maven artifact as file in repository.
*/
public File resolve(String groupId, String artifactId, String classifier,
String extension, String version,
MavenRepositoryURL repositoryURL,
Exception previousException) throws IOException {
Artifact artifact = new DefaultArtifact(groupId, artifactId, classifier, extension, version);
return resolve(artifact, repositoryURL, previousException);
}
/**
* Resolve maven artifact as file in repository.
*/
public File resolve(Artifact artifact) throws IOException {
return resolve(artifact, null, null);
}
/**
* Resolve maven artifact as file in repository.
*/
public File resolve(Artifact artifact,
MavenRepositoryURL repositoryURL,
Exception previousException) throws IOException {
List defaultRepos = selectDefaultRepositories();
List remoteRepos = selectRepositories();
if (repositoryURL != null) {
addRepo(remoteRepos, repositoryURL);
}
// PAXURL-337: use previousException as hint to alter remote repositories to query
if (previousException != null) {
// we'll try using previous repositories, without these that will fail again anyway
List altered = new LinkedList<>();
RepositoryException repositoryException = findAetherException(previousException);
if (repositoryException instanceof ArtifactResolutionException) {
// check only this aggregate exception and assume it's related to current artifact
ArtifactResult result = ((ArtifactResolutionException) repositoryException).getResult();
if (result != null && result.getRequest() != null && result.getRequest().getArtifact().equals(artifact)) {
// one exception per repository checked
// consider only ArtifactTransferException:
// - they may be recoverable
// - these exceptions contain repository that was checked
for (Exception exception : result.getExceptions()) {
RepositoryException singleException = findAetherException(exception);
if (singleException instanceof ArtifactTransferException) {
RemoteRepository repository = ((ArtifactTransferException) singleException).getRepository();
if (repository != null) {
RetryChance chance = isRetryableException(singleException);
if (chance == RetryChance.NEVER) {
LOG.debug("Removing " + repository + " from list of repositories, previous exception: " +
singleException.getClass().getName() + ": " + singleException.getMessage());
} else {
altered.add(repository);
}
}
}
}
// swap list of repos now
remoteRepos = altered;
}
}
}
assignProxyAndMirrors(remoteRepos);
File resolved = resolve(defaultRepos, remoteRepos, artifact);
LOG.debug("Resolved ({}) as {}", artifact.toString(), resolved.getAbsolutePath());
return resolved;
}
private File resolve(List defaultRepos,
List remoteRepos,
Artifact artifact) throws IOException {
if (artifact.getExtension().isEmpty()) {
artifact = new DefaultArtifact(
artifact.getGroupId(),
artifact.getArtifactId(),
artifact.getClassifier(),
"jar",
artifact.getVersion()
);
}
if (artifact.getVersion().equals(VERSION_LATEST)) {
artifact = artifact.setVersion(LATEST_VERSION_RANGE);
}
// Try with default repositories
try {
GenericVersionScheme genericVersionScheme = new GenericVersionScheme();
VersionConstraint vc = genericVersionScheme.parseVersionConstraint(artifact.getVersion());
// first, each "default repo" will be treated as local repo and resolution will be performed
// without remote repositories
for (LocalRepository repo : defaultRepos) {
RepositorySystemSession session = newSession(repo);
try {
if (vc.getVersion() == null && vc.getRange() != null) {
// KARAF-6005: try to resolve version range against local repository (default repository)
Metadata metadata =
new DefaultMetadata(artifact.getGroupId(), artifact.getArtifactId(),
"maven-metadata.xml", Metadata.Nature.RELEASE_OR_SNAPSHOT);
new LocalMetadataRequest(metadata, null, null);
LocalRepositoryManager lrm = session.getLocalRepositoryManager();
String path = lrm.getPathForLocalMetadata(metadata);
File metadataLocation = new File(lrm.getRepository().getBasedir(), path).getParentFile();
Set versions = new TreeSet<>();
if (metadataLocation.isDirectory()) {
if (!new File(metadataLocation, "maven-metadata.xml").isFile()) {
// we will generate (kind of) maven-metadata.xml manually
String[] versionDirs = metadataLocation.list();
if (versionDirs != null) {
for (String vd : versionDirs) {
Version ver = genericVersionScheme.parseVersion(vd);
if (vc.containsVersion(ver)) {
versions.add(ver);
}
}
}
VersionRangeResult vrr = new VersionRangeResult(new VersionRangeRequest());
vrr.setVersions(new LinkedList<>(versions));
if (vrr.getHighestVersion() != null) {
if (LOG.isDebugEnabled()) {
LOG.debug("Resolved version range {} as {}", vc.getRange(), vrr.getHighestVersion().toString());
}
vc = new GenericVersionScheme().parseVersionConstraint(vrr.getHighestVersion().toString());
artifact = artifact.setVersion(vc.getVersion().toString());
}
} else {
// we can use normal metadata resolution algorithm
try {
VersionRangeResult versionResult = m_repoSystem.resolveVersionRange(session,
new VersionRangeRequest(artifact, null, null));
if (versionResult != null) {
Version v = versionResult.getHighestVersion();
if (v != null) {
if (LOG.isDebugEnabled()) {
LOG.debug("Resolved version range {} as {}", vc.getRange(), v.toString());
}
vc = new GenericVersionScheme().parseVersionConstraint(v.toString());
artifact = artifact.setVersion(vc.getVersion().toString());
}
}
} catch (VersionRangeResolutionException e) {
// Ignore
}
}
}
}
if (vc.getVersion() != null) {
// normal resolution without ranges
try {
return m_repoSystem
.resolveArtifact(session, new ArtifactRequest(artifact, null, null))
.getArtifact().getFile();
} catch (ArtifactResolutionException e) {
// Ignore
}
}
} finally {
releaseSession(session);
}
}
} catch (InvalidVersionSpecificationException e) {
// Should not happen
}
RepositorySystemSession session = newSession(null);
try {
artifact = resolveLatestVersionRange(session, remoteRepos, artifact);
return m_repoSystem
.resolveArtifact(session, new ArtifactRequest(artifact, remoteRepos, null))
.getArtifact().getFile();
} catch (ArtifactResolutionException e) {
// we know there's one ArtifactResult, because there was one ArtifactRequest
ArtifactResolutionException original = new ArtifactResolutionException(e.getResults(),
"Error resolving artifact " + artifact.toString(), null);
throw configureIOException(original, e, e.getResult().getExceptions());
} catch (VersionRangeResolutionException e) {
// we know there's one ArtifactResult, because there was one ArtifactRequest
VersionRangeResolutionException original = new VersionRangeResolutionException(e.getResult(),
"Error resolving artifact " + artifact.toString(), null);
throw configureIOException(original, e, e.getResult().getExceptions());
} finally {
releaseSession(session);
}
}
/**
* Take original maven exception's message and stack trace without suppressed exceptions. Suppressed
* exceptions will be taken from {@code ArtifactResult} or {@link VersionRangeResult}
* @param newMavenException exception with reconfigured suppressed exceptions
* @param cause original Maven exception
* @param resultExceptions
* @return
*/
private IOException configureIOException(Exception newMavenException, Exception cause, List resultExceptions) {
newMavenException.setStackTrace(cause.getStackTrace());
List messages = new ArrayList<>(resultExceptions.size());
List suppressed = new ArrayList<>();
for (Exception ex : resultExceptions) {
messages.add(ex.getMessage() == null ? ex.getClass().getName() : ex.getMessage());
suppressed.add(ex);
}
IOException exception = new IOException(newMavenException.getMessage() + ": " + messages, newMavenException);
for (Exception ex : suppressed) {
exception.addSuppressed(ex);
}
LOG.warn(exception.getMessage(), exception);
return exception;
}
@Override
public File resolveMetadata(String groupId, String artifactId, String type, String version) throws IOException {
return resolveMetadata(groupId, artifactId, type, version, null);
}
@Override
public File resolveMetadata(String groupId, String artifactId, String type, String version,
Exception previousException) throws IOException {
RepositorySystem system = getRepositorySystem();
RepositorySystemSession session = newSession();
try {
Metadata metadata = new DefaultMetadata(groupId, artifactId, version,
type, Metadata.Nature.RELEASE_OR_SNAPSHOT);
List requests = new ArrayList();
// TODO: previousException may be a hint to alter remote repository list to query
for (RemoteRepository repository : getRepositories()) {
MetadataRequest request = new MetadataRequest(metadata, repository, null);
request.setFavorLocalRepository(false);
requests.add(request);
}
MetadataRequest request = new MetadataRequest(metadata, null, null);
request.setFavorLocalRepository(true);
requests.add(request);
org.apache.maven.artifact.repository.metadata.Metadata mr = new org.apache.maven.artifact.repository.metadata.Metadata();
mr.setModelVersion("1.1.0");
mr.setGroupId(metadata.getGroupId());
mr.setArtifactId(metadata.getArtifactId());
mr.setVersioning(new Versioning());
boolean merged = false;
List results = system.resolveMetadata(session, requests);
for (MetadataResult result : results) {
if (result.getMetadata() != null && result.getMetadata().getFile() != null) {
FileInputStream fis = new FileInputStream(result.getMetadata().getFile());
org.apache.maven.artifact.repository.metadata.Metadata m = new MetadataXpp3Reader().read(fis, false);
fis.close();
if (m.getVersioning() != null) {
mr.getVersioning().setLastUpdated(latestTimestamp(mr.getVersioning().getLastUpdated(), m.getVersioning().getLastUpdated()));
mr.getVersioning().setLatest(latestVersion(mr.getVersioning().getLatest(), m.getVersioning().getLatest()));
mr.getVersioning().setRelease(latestVersion(mr.getVersioning().getRelease(), m.getVersioning().getRelease()));
for (String v : m.getVersioning().getVersions()) {
if (!mr.getVersioning().getVersions().contains(v)) {
mr.getVersioning().getVersions().add(v);
}
}
mr.getVersioning().getSnapshotVersions().addAll(m.getVersioning().getSnapshotVersions());
}
merged = true;
}
}
if (merged) {
Collections.sort(mr.getVersioning().getVersions(), VERSION_COMPARATOR);
Collections.sort(mr.getVersioning().getSnapshotVersions(), SNAPSHOT_VERSION_COMPARATOR);
File tmpFile = Files.createTempFile("mvn-", ".tmp").toFile();
try (FileOutputStream fos = new FileOutputStream(tmpFile)) {
new MetadataXpp3Writer().write(fos, mr);
}
return tmpFile;
}
return null;
} catch (Exception e) {
throw new IOException("Unable to resolve metadata", e);
} finally {
releaseSession(session);
}
}
@Override
public void upload(String groupId, String artifactId, String classifier, String extension, String version, File file) throws IOException {
RepositorySystem system = getRepositorySystem();
RepositorySystemSession session = newSession();
try {
Artifact artifact = new DefaultArtifact(groupId, artifactId, classifier, extension, version,
null, file);
InstallRequest request = new InstallRequest();
request.addArtifact(artifact);
system.install(session, request);
} catch (Exception e) {
throw new IOException("Unable to install artifact", e);
} finally {
releaseSession(session);
}
}
@Override
public void uploadMetadata(String groupId, String artifactId, String type, String version, File file) throws IOException {
RepositorySystem system = getRepositorySystem();
RepositorySystemSession session = newSession();
try {
Metadata metadata = new DefaultMetadata(groupId, artifactId, version,
type, Metadata.Nature.RELEASE_OR_SNAPSHOT,
file);
InstallRequest request = new InstallRequest();
request.addMetadata(metadata);
system.install(session, request);
} catch (Exception e) {
throw new IOException("Unable to install metadata", e);
} finally {
releaseSession(session);
}
}
@Override
public RetryChance isRetryableException(Exception exception) {
RetryChance retry = RetryChance.NEVER;
RepositoryException aetherException = findAetherException(exception);
if (aetherException instanceof ArtifactResolutionException) {
// aggregate case - exception that contains exceptions - usually per repository
ArtifactResolutionException resolutionException = (ArtifactResolutionException) aetherException;
if (resolutionException.getResult() != null) {
for (Exception ex : resolutionException.getResult().getExceptions()) {
RetryChance singleRetry = isRetryableException(ex);
if (retry.chance() < singleRetry.chance()) {
retry = singleRetry;
}
}
}
} else if (aetherException != null) {
// single exception case
if (aetherException instanceof ArtifactNotFoundException) {
// very little chance we'll find the artifact next time
retry = RetryChance.NEVER;
} else if (aetherException instanceof MetadataNotFoundException) {
retry = RetryChance.NEVER;
} else if (aetherException instanceof ArtifactTransferException
|| aetherException instanceof MetadataTransferException) {
// we could try again
Throwable root = rootException(aetherException);
if (root instanceof SocketTimeoutException) {
// we could try again - but without assuming we'll succeed eventually
retry = RetryChance.LOW;
} else if (root instanceof ConnectException) {
// "connection refused" - not retryable
retry = RetryChance.NEVER;
} else if (root instanceof NoRouteToHostException) {
// not retryable
retry = RetryChance.NEVER;
}
} else {
// general aether exception - let's fallback to NEVER, as retryable cases should be
// handled explicitly
retry = RetryChance.NEVER;
}
} else {
// we don't know about non-aether exceptions, so let's allow
retry = RetryChance.UNKNOWN;
}
return retry;
}
/**
* Find top-most Aether exception
* @param e
* @return
*/
protected RepositoryException findAetherException(Exception e) {
Throwable ex = e;
while (ex != null && !(ex instanceof RepositoryException)) {
ex = ex.getCause();
}
return ex == null ? null : (RepositoryException) ex;
}
/**
* Find root exception
* @param ex
* @return
*/
protected Throwable rootException(Exception ex) {
Throwable root = ex;
while (true) {
if (root.getCause() != null) {
root = root.getCause();
} else {
break;
}
}
return root;
}
private Comparator VERSION_COMPARATOR = new Comparator() {
@Override
public int compare(String v1, String v2) {
try {
Version vv1 = new GenericVersionScheme().parseVersion(v1);
Version vv2 = new GenericVersionScheme().parseVersion(v2);
return vv1.compareTo(vv2);
} catch (Exception e) {
return v1.compareTo(v2);
}
}
};
private Comparator SNAPSHOT_VERSION_COMPARATOR = new Comparator() {
@Override
public int compare(SnapshotVersion o1, SnapshotVersion o2) {
int c = VERSION_COMPARATOR.compare(o1.getVersion(), o2.getVersion());
if (c == 0) {
c = o1.getExtension().compareTo(o2.getExtension());
}
if (c == 0) {
c = o1.getClassifier().compareTo(o2.getClassifier());
}
return c;
}
};
private String latestTimestamp(String t1, String t2) {
if (t1 == null) {
return t2;
} else if (t2 == null) {
return t1;
} else {
return t1.compareTo(t2) < 0 ? t2 : t1;
}
}
private String latestVersion(String v1, String v2) {
if (v1 == null) {
return v2;
} else if (v2 == null) {
return v1;
} else {
return VERSION_COMPARATOR.compare(v1, v2) < 0 ? v2 : v1;
}
}
/**
* Tries to resolve versions = LATEST using an open range version query. If it succeeds, version
* of artifact is set to the highest available version.
*
* @param session
* to be used.
* @param artifact
* to be used
*
* @return an artifact with version set properly (highest if available)
*
* @throws org.eclipse.aether.resolution.VersionRangeResolutionException
* in case of resolver errors.
*/
private Artifact resolveLatestVersionRange(RepositorySystemSession session,
List remoteRepos, Artifact artifact)
throws VersionRangeResolutionException {
VersionRangeResult versionResult = m_repoSystem.resolveVersionRange(session,
new VersionRangeRequest(artifact, remoteRepos, null));
if (versionResult != null) {
Version v = versionResult.getHighestVersion();
if (v != null) {
artifact = artifact.setVersion(v.toString());
} else {
throw new VersionRangeResolutionException(versionResult,
"No highest version found for " + artifact);
}
}
return artifact;
}
public RepositorySystemSession newSession() {
return newSession(null);
}
private RepositorySystemSession newSession(LocalRepository repo) {
if (repo == null) {
repo = getLocalRepository();
}
Deque deque = sessions.get(repo);
RepositorySystemSession session = null;
if (deque != null) {
session = deque.pollFirst();
}
if (session == null) {
session = createSession(repo);
}
return session;
}
/**
* @see "org.eclipse.aether.internal.impl.DefaultUpdateCheckManager#SESSION_CHECKS"
*/
private static final String SESSION_CHECKS = "updateCheckManager.checks";
private void releaseSession(RepositorySystemSession session) {
LocalRepository repo = session.getLocalRepository();
Deque deque = sessions.get(repo);
if (deque == null) {
sessions.putIfAbsent(repo, new ConcurrentLinkedDeque());
deque = sessions.get(repo);
}
session.getData().set(SESSION_CHECKS, null);
deque.add(session);
}
private RepositorySystemSession createSession(LocalRepository repo) {
DefaultRepositorySystemSession session = MavenRepositorySystemUtils.newSession();
if (repo != null) {
session.setLocalRepositoryManager(m_repoSystem.newLocalRepositoryManager(session, repo));
} else {
session.setLocalRepositoryManager(m_repoSystem.newLocalRepositoryManager(session, getLocalRepository()));
}
session.setMirrorSelector(m_mirrorSelector);
session.setProxySelector(m_proxySelector);
String updatePolicy = m_config.getGlobalUpdatePolicy();
if (null != updatePolicy) {
session.setUpdatePolicy(updatePolicy);
}
String checksumPolicy = m_config.getGlobalChecksumPolicy();
if (null != checksumPolicy) {
session.setChecksumPolicy(checksumPolicy);
}
for (Server server : m_settings.getServers()) {
if (server.getConfiguration() != null
&& ((Xpp3Dom) server.getConfiguration()).getChild("httpHeaders") != null) {
addServerConfig(session, server);
}
}
// org.eclipse.aether.transport.wagon.WagonTransporter.connectWagon() sets connection timeout
// as max of connection timeout and request timeout taken from aether session config
// but on the other hand, request timeout is used in org.eclipse.aether.connector.basic.PartialFile.Factory
//
// because real socket read timeout is used again (correctly) in
// org.ops4j.pax.url.mvn.internal.wagon.ConfigurableHttpWagon.execute() - directly from wagon configuration
// (from org.apache.maven.wagon.AbstractWagon.getReadTimeout()), we explicitly configure aether session
// config with: PartialFile.Factory timeout == connection timeout
int defaultTimeut = m_config.getTimeout();
Integer timeout = m_config.getProperty(ServiceConstants.PROPERTY_SOCKET_CONNECTION_TIMEOUT, defaultTimeut, Integer.class);
session.setConfigProperty(ConfigurationProperties.CONNECT_TIMEOUT, timeout);
session.setConfigProperty(ConfigurationProperties.REQUEST_TIMEOUT, timeout);
session.setOffline(m_config.isOffline());
// PAXURL-322
boolean updateReleases = m_config.getProperty(ServiceConstants.PROPERTY_UPDATE_RELEASES, false, Boolean.class);
session.setConfigProperty(PaxLocalRepositoryManager.PROPERTY_UPDATE_RELEASES, updateReleases);
return session;
}
private LocalRepository getLocalRepository() {
if (localRepository == null) {
File local;
if (m_config.getLocalRepository() != null) {
local = m_config.getLocalRepository().getFile();
} else {
local = new File(System.getProperty("user.home"), ".m2/repository");
}
localRepository = new LocalRepository(local, "simple");
}
return localRepository;
}
private void addServerConfig(DefaultRepositorySystemSession session, Server server) {
Map headers = new HashMap();
Xpp3Dom configuration = (Xpp3Dom) server.getConfiguration();
Xpp3Dom httpHeaders = configuration.getChild("httpHeaders");
for (Xpp3Dom httpHeader : httpHeaders.getChildren("httpHeader")) {
Xpp3Dom name = httpHeader.getChild("name");
String headerName = name.getValue();
Xpp3Dom value = httpHeader.getChild("value");
String headerValue = value.getValue();
headers.put(headerName, headerValue);
}
session.setConfigProperty(String.format("%s.%s", ConfigurationProperties.HTTP_HEADERS, server.getId()), headers);
}
private Authentication getAuthentication(org.apache.maven.settings.Proxy proxy) {
// user, pass
if (proxy.getUsername() != null) {
return new AuthenticationBuilder().addUsername(proxy.getUsername())
.addPassword(proxy.getPassword()).build();
}
return null;
}
private Authentication getAuthentication(String repoId) {
Server server = m_settings.getServer(repoId);
if (server != null && server.getUsername() != null) {
AuthenticationBuilder authBuilder = new AuthenticationBuilder();
authBuilder.addUsername(server.getUsername()).addPassword(server.getPassword());
return authBuilder.build();
}
return null;
}
private RepositorySystem newRepositorySystem() {
DefaultServiceLocator locator = MavenRepositorySystemUtils.newServiceLocator();
// default timeout (both connection and read timeouts)
int defaultTimeout = m_config.getTimeout();
// connection timeout
int connectionTimeout = m_config.getProperty(ServiceConstants.PROPERTY_SOCKET_CONNECTION_TIMEOUT, defaultTimeout, Integer.class);
// read timeout
int soTimeout = m_config.getProperty(ServiceConstants.PROPERTY_SOCKET_SO_TIMEOUT, defaultTimeout, Integer.class);
locator.setServices(WagonProvider.class, new ManualWagonProvider(m_client, soTimeout, connectionTimeout));
locator.addService(TransporterFactory.class, WagonTransporterFactory.class);
locator.addService(RepositoryConnectorFactory.class, BasicRepositoryConnectorFactory.class);
decrypter = new ConfigurableSettingsDecrypter();
PaxUrlSecDispatcher secDispatcher = new PaxUrlSecDispatcher();
try {
secDispatcher.setCipher(new DefaultPlexusCipher());
} catch (PlexusCipherException exc) {
throw new IllegalStateException(exc);
}
secDispatcher.setConfigurationFile(m_config.getSecuritySettings());
decrypter.setSecurityDispatcher(secDispatcher);
locator.setServices(SettingsDecrypter.class, decrypter);
locator.setService(LocalRepositoryManagerFactory.class,
PaxLocalRepositoryManagerFactory.class);
locator.setService(org.eclipse.aether.spi.log.LoggerFactory.class,
Slf4jLoggerFactory.class);
return locator.getService(RepositorySystem.class);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy