From 46fd585f40f0264b8889154e1924eb8df991b0f3 Mon Sep 17 00:00:00 2001 From: Martin Stockhammer Date: Thu, 11 Jun 2020 14:33:00 +0200 Subject: Improving repository API --- .../apache/archiva/common/utils/VersionUtil.java | 2 + .../core/repository/DaysOldRepositoryPurge.java | 140 ++++++++++----------- .../repository/RetentionCountRepositoryPurge.java | 104 ++++++++------- .../proxy/model/RepositoryProxyHandler.java | 16 +++ .../proxy/DefaultRepositoryProxyHandler.java | 72 ++++++++++- .../archiva/repository/RepositoryRequestInfo.java | 10 ++ .../content/ManagedDefaultRepositoryContent.java | 112 ++++++++++++----- .../maven/content/MavenRepositoryRequestInfo.java | 19 ++- .../metadata/storage/Maven2RepositoryStorage.java | 92 ++++++++++++++ .../ManagedDefaultRepositoryContentTest.java | 2 +- .../content/MavenRepositoryRequestInfoTest.java | 9 +- .../archiva/security/mock/MockBeanServices.java | 7 ++ .../archiva/webdav/ArchivaDavResourceFactory.java | 16 +-- .../repository/storage/RepositoryStorage.java | 16 +++ .../cassandra/MockRepositoryStorage.java | 7 ++ 15 files changed, 458 insertions(+), 166 deletions(-) diff --git a/archiva-modules/archiva-base/archiva-common/src/main/java/org/apache/archiva/common/utils/VersionUtil.java b/archiva-modules/archiva-base/archiva-common/src/main/java/org/apache/archiva/common/utils/VersionUtil.java index c1a3ac694..408d7b4e1 100644 --- a/archiva-modules/archiva-base/archiva-common/src/main/java/org/apache/archiva/common/utils/VersionUtil.java +++ b/archiva-modules/archiva-base/archiva-common/src/main/java/org/apache/archiva/common/utils/VersionUtil.java @@ -194,4 +194,6 @@ public class VersionUtil } return version; } + + } diff --git a/archiva-modules/archiva-base/archiva-consumers/archiva-core-consumers/src/main/java/org/apache/archiva/consumers/core/repository/DaysOldRepositoryPurge.java b/archiva-modules/archiva-base/archiva-consumers/archiva-core-consumers/src/main/java/org/apache/archiva/consumers/core/repository/DaysOldRepositoryPurge.java index fb42390bb..282f45c41 100644 --- a/archiva-modules/archiva-base/archiva-consumers/archiva-core-consumers/src/main/java/org/apache/archiva/consumers/core/repository/DaysOldRepositoryPurge.java +++ b/archiva-modules/archiva-base/archiva-consumers/archiva-core-consumers/src/main/java/org/apache/archiva/consumers/core/repository/DaysOldRepositoryPurge.java @@ -23,9 +23,9 @@ import org.apache.archiva.common.utils.VersionComparator; import org.apache.archiva.common.utils.VersionUtil; import org.apache.archiva.metadata.audit.RepositoryListener; import org.apache.archiva.metadata.repository.RepositorySession; -import org.apache.archiva.repository.ManagedRepositoryContent; -import org.apache.archiva.repository.LayoutException; import org.apache.archiva.repository.BaseRepositoryContentLayout; +import org.apache.archiva.repository.LayoutException; +import org.apache.archiva.repository.ManagedRepositoryContent; import org.apache.archiva.repository.content.Artifact; import org.apache.archiva.repository.content.ContentItem; import org.apache.archiva.repository.content.base.ArchivaItemSelector; @@ -74,94 +74,94 @@ public class DaysOldRepositoryPurge { ContentItem item = repository.toItem( path ); - if ( item instanceof Artifact ) + Artifact artifactItem = repository.getLayout( BaseRepositoryContentLayout.class ).adaptItem( Artifact.class, item ); + + if ( !artifactItem.exists( ) ) { - Artifact artifactItem = (Artifact) item; + return; + } - if ( !artifactItem.exists( ) ) - { - return; - } + // ArtifactReference artifact = repository.toArtifactReference( path ); - // ArtifactReference artifact = repository.toArtifactReference( path ); - - Calendar olderThanThisDate = Calendar.getInstance( TimeZone.getTimeZone( "UTC" ) ); - olderThanThisDate.add( Calendar.DATE, -retentionPeriod ); - - ArchivaItemSelector selector = ArchivaItemSelector.builder( ) - .withNamespace( artifactItem.getVersion( ).getProject( ).getNamespace( ).getId( ) ) - .withProjectId( artifactItem.getVersion( ).getProject( ).getId( ) ) - .withVersion( artifactItem.getVersion( ).getId( ) ) - .withClassifier( "*" ) - .includeRelatedArtifacts( ) - .build( ); - - List artifactVersions; - try( Stream stream = repository.getLayout( BaseRepositoryContentLayout.class ).newArtifactStream( selector )){ - artifactVersions = stream.map( a -> a.getArtifactVersion( ) ) - .filter( StringUtils::isNotEmpty ) - .distinct() - .collect( Collectors.toList( ) ); - } + Calendar olderThanThisDate = Calendar.getInstance( TimeZone.getTimeZone( "UTC" ) ); + olderThanThisDate.add( Calendar.DATE, -retentionPeriod ); - Collections.sort( artifactVersions, VersionComparator.getInstance( ) ); + ArchivaItemSelector selector = ArchivaItemSelector.builder( ) + .withNamespace( artifactItem.getVersion( ).getProject( ).getNamespace( ).getId( ) ) + .withProjectId( artifactItem.getVersion( ).getProject( ).getId( ) ) + .withVersion( artifactItem.getVersion( ).getId( ) ) + .withClassifier( "*" ) + .includeRelatedArtifacts( ) + .build( ); - if ( retentionCount > artifactVersions.size( ) ) - { - // Done. nothing to do here. skip it. - return; - } + List artifactVersions; + try ( Stream stream = repository.getLayout( BaseRepositoryContentLayout.class ).newArtifactStream( selector ) ) + { + artifactVersions = stream.map( a -> a.getArtifactVersion( ) ) + .filter( StringUtils::isNotEmpty ) + .distinct( ) + .collect( Collectors.toList( ) ); + } + + Collections.sort( artifactVersions, VersionComparator.getInstance( ) ); - int countToPurge = artifactVersions.size( ) - retentionCount; + if ( retentionCount > artifactVersions.size( ) ) + { + // Done. nothing to do here. skip it. + return; + } + int countToPurge = artifactVersions.size( ) - retentionCount; - ArchivaItemSelector.Builder artifactSelectorBuilder = ArchivaItemSelector.builder( ) - .withNamespace( artifactItem.getVersion( ).getProject( ).getNamespace( ).getId( ) ) - .withProjectId( artifactItem.getVersion( ).getProject( ).getId( ) ) - .withVersion( artifactItem.getVersion( ).getId( ) ) - .withArtifactId( artifactItem.getId() ) - .withClassifier( "*" ) - .includeRelatedArtifacts( ); - Set artifactsToDelete = new HashSet<>( ); - for ( String version : artifactVersions ) + ArchivaItemSelector.Builder artifactSelectorBuilder = ArchivaItemSelector.builder( ) + .withNamespace( artifactItem.getVersion( ).getProject( ).getNamespace( ).getId( ) ) + .withProjectId( artifactItem.getVersion( ).getProject( ).getId( ) ) + .withVersion( artifactItem.getVersion( ).getId( ) ) + .withArtifactId( artifactItem.getId( ) ) + .withClassifier( "*" ) + .includeRelatedArtifacts( ); + + Set artifactsToDelete = new HashSet<>( ); + for ( String version : artifactVersions ) + { + if ( countToPurge-- <= 0 ) { - if ( countToPurge-- <= 0 ) - { - break; - } + break; + } - ArchivaItemSelector artifactSelector = artifactSelectorBuilder.withArtifactVersion( version ).build( ); - try - { + ArchivaItemSelector artifactSelector = artifactSelectorBuilder.withArtifactVersion( version ).build( ); + try + { - // Is this a generic snapshot "1.0-SNAPSHOT" ? - if ( VersionUtil.isGenericSnapshot( version ) ) + // Is this a generic snapshot "1.0-SNAPSHOT" ? + if ( VersionUtil.isGenericSnapshot( version ) ) + { + List artifactList = repository.getLayout( BaseRepositoryContentLayout.class ).getArtifacts( artifactSelector ); + if ( artifactList.size( ) > 0 && artifactList.get( 0 ).getAsset( ).getModificationTime( ).toEpochMilli( ) < olderThanThisDate.getTimeInMillis( ) ) { - List artifactList = repository.getLayout( BaseRepositoryContentLayout.class ).getArtifacts( artifactSelector ); - if ( artifactList.size()>0 && artifactList.get(0).getAsset().getModificationTime( ).toEpochMilli( ) < olderThanThisDate.getTimeInMillis( ) ) - { - artifactsToDelete.addAll( artifactList ); - } + artifactsToDelete.addAll( artifactList ); } - // Is this a timestamp snapshot "1.0-20070822.123456-42" ? - else if ( VersionUtil.isUniqueSnapshot( version ) ) - { - Calendar timestampCal = uniqueSnapshotToCalendar( version ); + } + // Is this a timestamp snapshot "1.0-20070822.123456-42" ? + else if ( VersionUtil.isUniqueSnapshot( version ) ) + { + Calendar timestampCal = uniqueSnapshotToCalendar( version ); - if ( timestampCal.getTimeInMillis( ) < olderThanThisDate.getTimeInMillis( ) ) - { - artifactsToDelete.addAll( repository.getLayout( BaseRepositoryContentLayout.class ).getArtifacts( artifactSelector ) ); - } + if ( timestampCal.getTimeInMillis( ) < olderThanThisDate.getTimeInMillis( ) ) + { + artifactsToDelete.addAll( repository.getLayout( BaseRepositoryContentLayout.class ).getArtifacts( artifactSelector ) ); } - } catch ( IllegalArgumentException e ) { - log.error( "Bad selector for artifact: {}", e.getMessage( ), e ); - // continue } } - purge( artifactsToDelete ); + catch ( IllegalArgumentException e ) + { + log.error( "Bad selector for artifact: {}", e.getMessage( ), e ); + // continue + } } + purge( artifactsToDelete ); } catch ( LayoutException e ) { diff --git a/archiva-modules/archiva-base/archiva-consumers/archiva-core-consumers/src/main/java/org/apache/archiva/consumers/core/repository/RetentionCountRepositoryPurge.java b/archiva-modules/archiva-base/archiva-consumers/archiva-core-consumers/src/main/java/org/apache/archiva/consumers/core/repository/RetentionCountRepositoryPurge.java index ec3a242f4..74df6a679 100644 --- a/archiva-modules/archiva-base/archiva-consumers/archiva-core-consumers/src/main/java/org/apache/archiva/consumers/core/repository/RetentionCountRepositoryPurge.java +++ b/archiva-modules/archiva-base/archiva-consumers/archiva-core-consumers/src/main/java/org/apache/archiva/consumers/core/repository/RetentionCountRepositoryPurge.java @@ -21,12 +21,12 @@ package org.apache.archiva.consumers.core.repository; import org.apache.archiva.common.utils.VersionComparator; import org.apache.archiva.common.utils.VersionUtil; +import org.apache.archiva.metadata.audit.RepositoryListener; import org.apache.archiva.metadata.repository.RepositorySession; import org.apache.archiva.model.ArtifactReference; -import org.apache.archiva.repository.ManagedRepositoryContent; -import org.apache.archiva.repository.LayoutException; import org.apache.archiva.repository.BaseRepositoryContentLayout; -import org.apache.archiva.metadata.audit.RepositoryListener; +import org.apache.archiva.repository.LayoutException; +import org.apache.archiva.repository.ManagedRepositoryContent; import org.apache.archiva.repository.content.Artifact; import org.apache.archiva.repository.content.ContentItem; import org.apache.archiva.repository.content.base.ArchivaItemSelector; @@ -61,67 +61,65 @@ public class RetentionCountRepositoryPurge try { ContentItem item = repository.toItem( path ); - if (item instanceof Artifact ) + BaseRepositoryContentLayout layout = repository.getLayout( BaseRepositoryContentLayout.class ); + Artifact artifact = layout.adaptItem( Artifact.class, item ); + if ( !artifact.exists( ) ) { - Artifact artifact = (Artifact) item; - if (!artifact.exists()) { - return; - } + return; + } - if ( VersionUtil.isSnapshot( artifact.getVersion( ).getId() ) ) - { - ArchivaItemSelector selector = ArchivaItemSelector.builder( ) - .withNamespace( artifact.getVersion( ).getProject( ).getNamespace( ).getId( ) ) - .withProjectId( artifact.getVersion( ).getProject( ).getId( ) ) - .withArtifactId( artifact.getId( ) ) - .withVersion( artifact.getVersion( ).getId( ) ) - .withClassifier( "*" ) - .includeRelatedArtifacts() - .build( ); + if ( VersionUtil.isSnapshot( artifact.getVersion( ).getId( ) ) ) + { + ArchivaItemSelector selector = ArchivaItemSelector.builder( ) + .withNamespace( artifact.getVersion( ).getProject( ).getNamespace( ).getId( ) ) + .withProjectId( artifact.getVersion( ).getProject( ).getId( ) ) + .withArtifactId( artifact.getId( ) ) + .withVersion( artifact.getVersion( ).getId( ) ) + .withClassifier( "*" ) + .includeRelatedArtifacts( ) + .build( ); - List versions; - try( Stream stream = repository.getLayout( BaseRepositoryContentLayout.class ).newArtifactStream( selector) ){ - versions = stream.map( a -> a.getArtifactVersion( ) ) - .filter( StringUtils::isNotEmpty ) - .distinct() - .collect( Collectors.toList( ) ); - } + List versions; + try ( Stream stream = repository.getLayout( BaseRepositoryContentLayout.class ).newArtifactStream( selector ) ) + { + versions = stream.map( a -> a.getArtifactVersion( ) ) + .filter( StringUtils::isNotEmpty ) + .distinct( ) + .collect( Collectors.toList( ) ); + } + + Collections.sort( versions, VersionComparator.getInstance( ) ); - Collections.sort( versions, VersionComparator.getInstance( ) ); + if ( retentionCount > versions.size( ) ) + { + log.trace( "No deletion, because retention count is higher than actual number of artifacts." ); + // Done. nothing to do here. skip it. + return; + } - if ( retentionCount > versions.size( ) ) + ArchivaItemSelector.Builder selectorBuilder = ArchivaItemSelector.builder( ) + .withNamespace( artifact.getVersion( ).getProject( ).getNamespace( ).getId( ) ) + .withProjectId( artifact.getVersion( ).getProject( ).getId( ) ) + .withArtifactId( artifact.getId( ) ) + .withClassifier( "*" ) + .includeRelatedArtifacts( ) + .withVersion( artifact.getVersion( ).getId( ) ); + int countToPurge = versions.size( ) - retentionCount; + Set artifactsToDelete = new HashSet<>( ); + for ( String version : versions ) + { + if ( countToPurge-- <= 0 ) { - log.trace( "No deletion, because retention count is higher than actual number of artifacts." ); - // Done. nothing to do here. skip it. - return; + break; } - - ArchivaItemSelector.Builder selectorBuilder = ArchivaItemSelector.builder( ) - .withNamespace( artifact.getVersion( ).getProject( ).getNamespace( ).getId( ) ) - .withProjectId( artifact.getVersion( ).getProject( ).getId( ) ) - .withArtifactId( artifact.getId( ) ) - .withClassifier( "*" ) - .includeRelatedArtifacts() - .withVersion( artifact.getVersion( ).getId( ) ); - int countToPurge = versions.size( ) - retentionCount; - Set artifactsToDelete = new HashSet<>( ); - for ( String version : versions ) + List delArtifacts = repository.getLayout( BaseRepositoryContentLayout.class ).getArtifacts( selectorBuilder.withArtifactVersion( version ).build( ) ); + if ( delArtifacts != null && delArtifacts.size( ) > 0 ) { - if ( countToPurge-- <= 0 ) - { - break; - } - List delArtifacts = repository.getLayout( BaseRepositoryContentLayout.class ).getArtifacts( selectorBuilder.withArtifactVersion( version ).build( ) ); - if (delArtifacts!=null && delArtifacts.size()>0) - { - artifactsToDelete.addAll( delArtifacts ); - } + artifactsToDelete.addAll( delArtifacts ); } - purge( artifactsToDelete ); } - } else { - throw new RepositoryPurgeException( "Bad artifact path " + path ); + purge( artifactsToDelete ); } } catch ( LayoutException le ) diff --git a/archiva-modules/archiva-base/archiva-proxy-api/src/main/java/org/apache/archiva/proxy/model/RepositoryProxyHandler.java b/archiva-modules/archiva-base/archiva-proxy-api/src/main/java/org/apache/archiva/proxy/model/RepositoryProxyHandler.java index f775c3924..f6c9259a3 100644 --- a/archiva-modules/archiva-base/archiva-proxy-api/src/main/java/org/apache/archiva/proxy/model/RepositoryProxyHandler.java +++ b/archiva-modules/archiva-base/archiva-proxy-api/src/main/java/org/apache/archiva/proxy/model/RepositoryProxyHandler.java @@ -25,6 +25,7 @@ import org.apache.archiva.policies.ProxyDownloadException; import org.apache.archiva.repository.ManagedRepository; import org.apache.archiva.repository.RepositoryType; import org.apache.archiva.repository.content.Artifact; +import org.apache.archiva.repository.content.ItemSelector; import org.apache.archiva.repository.storage.StorageAsset; import java.util.List; @@ -89,6 +90,21 @@ public interface RepositoryProxyHandler StorageAsset fetchFromProxies( ManagedRepository repository, Artifact artifact ) throws ProxyDownloadException; + /** + * Performs the artifact fetch operation against the target repositories + * of the provided source repository. + *

+ * If the artifact is found, it is downloaded and placed into the source repository + * filesystem. + * + * @param repository the source repository to use. (must be a managed repository) + * @param artifactSelector the artifact to fetch. + * @return the file that was obtained, or null if no content was obtained + * @throws ProxyDownloadException if there was a problem fetching the content from the target repositories. + */ + StorageAsset fetchFromProxies( ManagedRepository repository, ItemSelector artifactSelector ) + throws ProxyDownloadException; + /** * Performs the metadata fetch operation against the target repositories * of the provided source repository. diff --git a/archiva-modules/archiva-base/archiva-proxy/src/main/java/org/apache/archiva/proxy/DefaultRepositoryProxyHandler.java b/archiva-modules/archiva-base/archiva-proxy/src/main/java/org/apache/archiva/proxy/DefaultRepositoryProxyHandler.java index 8cf2e97d2..2373fb938 100644 --- a/archiva-modules/archiva-base/archiva-proxy/src/main/java/org/apache/archiva/proxy/DefaultRepositoryProxyHandler.java +++ b/archiva-modules/archiva-base/archiva-proxy/src/main/java/org/apache/archiva/proxy/DefaultRepositoryProxyHandler.java @@ -50,6 +50,7 @@ import org.apache.archiva.repository.RemoteRepository; import org.apache.archiva.repository.RemoteRepositoryContent; import org.apache.archiva.repository.RepositoryType; import org.apache.archiva.repository.content.Artifact; +import org.apache.archiva.repository.content.ContentItem; import org.apache.archiva.repository.content.ItemSelector; import org.apache.archiva.repository.content.base.ArchivaItemSelector; import org.apache.archiva.repository.metadata.base.MetadataTools; @@ -206,6 +207,73 @@ public abstract class DefaultRepositoryProxyHandler implements RepositoryProxyHa return null; } + @Override + public StorageAsset fetchFromProxies( ManagedRepository repository, ItemSelector artifactSelector ) + throws ProxyDownloadException + { + Map previousExceptions = new LinkedHashMap<>(); + ContentItem item = repository.getContent( ).getItem( artifactSelector ); + StorageAsset localFile = item.getAsset( ); + + Properties requestProperties = new Properties(); + requestProperties.setProperty( "filetype", "artifact" ); + requestProperties.setProperty( "version", artifactSelector.getVersion() ); + requestProperties.setProperty( "managedRepositoryId", repository.getId() ); + + List connectors = getProxyConnectors( repository ); + for ( ProxyConnector connector : connectors ) + { + if ( !connector.isEnabled() ) + { + continue; + } + + RemoteRepository targetRepository = connector.getTargetRepository(); + requestProperties.setProperty( "remoteRepositoryId", targetRepository.getId() ); + + StorageAsset targetFile = targetRepository.getAsset( localFile.getPath( ) ); + // Removing the leading '/' from the path + String targetPath = targetFile.getPath( ).substring( 1 ); + try + { + StorageAsset downloadedFile = + transferFile( connector, targetRepository, targetPath, repository, localFile, requestProperties, + true ); + + if ( fileExists(downloadedFile) ) + { + log.debug( "Successfully transferred: {}", downloadedFile.getPath() ); + return downloadedFile; + } + } + catch ( NotFoundException e ) + { + log.debug( "Artifact {} not found on repository \"{}\".", item, + targetRepository.getId() ); + } + catch ( NotModifiedException e ) + { + log.debug( "Artifact {} not updated on repository \"{}\".", item, + targetRepository.getId() ); + } + catch ( ProxyException e ) + { + validatePolicies( this.downloadErrorPolicies, connector.getPolicies(), requestProperties, item, + targetRepository.getContent(), localFile, e, previousExceptions ); + } + } + + if ( !previousExceptions.isEmpty() ) + { + throw new ProxyDownloadException( "Failures occurred downloading from some remote repositories", + previousExceptions ); + } + + log.debug( "Exhausted all target repositories, artifact {} not found.", item ); + + return null; + } + @Override public StorageAsset fetchFromProxies( ManagedRepository repository, ArtifactReference artifact ) throws ProxyDownloadException @@ -736,7 +804,7 @@ public abstract class DefaultRepositoryProxyHandler implements RepositoryProxyHa } private void validatePolicies( Map policies, Map settings, - Properties request, Artifact artifact, RemoteRepositoryContent content, + Properties request, ContentItem artifact, RemoteRepositoryContent content, StorageAsset localFile, Exception exception, Map previousExceptions ) throws ProxyDownloadException { @@ -784,7 +852,7 @@ public abstract class DefaultRepositoryProxyHandler implements RepositoryProxyHa log.warn( "Transfer error from repository {} for artifact {} , continuing to next repository. Error message: {}", - content.getRepository().getId(), artifact.getId(), exception.getMessage() ); + content.getRepository().getId(), artifact, exception.getMessage() ); log.debug( "Full stack trace", exception ); } diff --git a/archiva-modules/archiva-base/archiva-repository-api/src/main/java/org/apache/archiva/repository/RepositoryRequestInfo.java b/archiva-modules/archiva-base/archiva-repository-api/src/main/java/org/apache/archiva/repository/RepositoryRequestInfo.java index 04700e76b..52cfb2c22 100644 --- a/archiva-modules/archiva-base/archiva-repository-api/src/main/java/org/apache/archiva/repository/RepositoryRequestInfo.java +++ b/archiva-modules/archiva-base/archiva-repository-api/src/main/java/org/apache/archiva/repository/RepositoryRequestInfo.java @@ -20,6 +20,7 @@ package org.apache.archiva.repository; */ import org.apache.archiva.model.ArtifactReference; +import org.apache.archiva.repository.content.ItemSelector; import org.apache.archiva.repository.features.RepositoryFeature; /** @@ -42,6 +43,15 @@ public interface RepositoryRequestInfo */ ArtifactReference toArtifactReference( String requestPath ) throws LayoutException; + + /** + * Returns the item selector that matches the given path. + * @param requestPath the request path which may be different from the filesystem structure + * @return the item selector + * @throws LayoutException if the path is not valid for the given repository layout + */ + ItemSelector toItemSelector( String requestPath ) throws LayoutException; + /** *

* Tests the path to see if it conforms to the expectations of a metadata request. diff --git a/archiva-modules/archiva-maven/archiva-maven-repository/src/main/java/org/apache/archiva/repository/maven/content/ManagedDefaultRepositoryContent.java b/archiva-modules/archiva-maven/archiva-maven-repository/src/main/java/org/apache/archiva/repository/maven/content/ManagedDefaultRepositoryContent.java index 734174732..b66207dd8 100644 --- a/archiva-modules/archiva-maven/archiva-maven-repository/src/main/java/org/apache/archiva/repository/maven/content/ManagedDefaultRepositoryContent.java +++ b/archiva-modules/archiva-maven/archiva-maven-repository/src/main/java/org/apache/archiva/repository/maven/content/ManagedDefaultRepositoryContent.java @@ -20,6 +20,7 @@ package org.apache.archiva.repository.maven.content; import org.apache.archiva.common.filelock.FileLockManager; import org.apache.archiva.common.utils.FileUtils; +import org.apache.archiva.common.utils.VersionUtil; import org.apache.archiva.configuration.FileTypes; import org.apache.archiva.metadata.maven.MavenMetadataReader; import org.apache.archiva.metadata.repository.storage.RepositoryPathTranslator; @@ -186,30 +187,42 @@ public class ManagedDefaultRepositoryContent @Override public T adaptItem( Class clazz, ContentItem item ) throws LayoutException { - if (clazz.isAssignableFrom( Version.class )) + try { - if ( !item.hasCharacteristic( Version.class ) ) + if ( clazz.isAssignableFrom( Version.class ) ) { - item.setCharacteristic( Version.class, createVersionFromPath( item.getAsset() ) ); + if ( !item.hasCharacteristic( Version.class ) ) + { + item.setCharacteristic( Version.class, createVersionFromPath( item.getAsset( ) ) ); + } + return (T) item.adapt( Version.class ); } - return (T) item.adapt( Version.class ); - } else if ( clazz.isAssignableFrom( Project.class )) { - if ( !item.hasCharacteristic( Project.class ) ) + else if ( clazz.isAssignableFrom( Project.class ) ) { - item.setCharacteristic( Project.class, createProjectFromPath( item.getAsset() ) ); + if ( !item.hasCharacteristic( Project.class ) ) + { + item.setCharacteristic( Project.class, createProjectFromPath( item.getAsset( ) ) ); + } + return (T) item.adapt( Project.class ); } - return (T) item.adapt( Project.class ); - } else if ( clazz.isAssignableFrom( Namespace.class )) { - if ( !item.hasCharacteristic( Namespace.class ) ) + else if ( clazz.isAssignableFrom( Namespace.class ) ) { - item.setCharacteristic( Namespace.class, createNamespaceFromPath( item.getAsset() ) ); + if ( !item.hasCharacteristic( Namespace.class ) ) + { + item.setCharacteristic( Namespace.class, createNamespaceFromPath( item.getAsset( ) ) ); + } + return (T) item.adapt( Namespace.class ); } - return (T) item.adapt( Namespace.class ); - } else if ( clazz.isAssignableFrom( Artifact.class )) { - if (!item.hasCharacteristic( Artifact.class )) { - item.setCharacteristic( Artifact.class, createArtifactFromPath( item.getAsset( ) ) ); + else if ( clazz.isAssignableFrom( Artifact.class ) ) + { + if ( !item.hasCharacteristic( Artifact.class ) ) + { + item.setCharacteristic( Artifact.class, createArtifactFromPath( item.getAsset( ) ) ); + } + return (T) item.adapt( Artifact.class ); } - return (T) item.adapt( Artifact.class ); + } catch (LayoutRuntimeException e) { + throw new LayoutException( e.getMessage( ), e ); } throw new LayoutException( "Could not convert item to class " + clazz); } @@ -593,6 +606,7 @@ public class ManagedDefaultRepositoryContent } } + private DataItem getDataItemFromPath( final StorageAsset artifactPath ) { final String contentType = getContentType( artifactPath ); @@ -644,13 +658,13 @@ public class ManagedDefaultRepositoryContent private ArtifactType artifactType = BaseArtifactTypes.MAIN; } - private ArtifactInfo getArtifactInfoFromPath( String genericVersion, StorageAsset path ) + private ArtifactInfo getArtifactInfoFromPath( final String genericVersion, final StorageAsset path ) { final ArtifactInfo info = new ArtifactInfo( ); info.asset = path; info.id = path.getParent( ).getParent( ).getName( ); final String fileName = path.getName( ); - if ( genericVersion.endsWith( "-" + SNAPSHOT ) ) + if ( VersionUtil.isGenericSnapshot( genericVersion ) ) { String baseVersion = StringUtils.substringBeforeLast( genericVersion, "-" + SNAPSHOT ); String prefix = info.id + "-" + baseVersion + "-"; @@ -722,7 +736,7 @@ public class ManagedDefaultRepositoryContent else { String prefix = info.id + "-" + genericVersion; - if ( fileName.startsWith( prefix ) ) + if ( fileName.startsWith( prefix + "-") ) { info.version = genericVersion; String classPostfix = StringUtils.removeStart( fileName, prefix ); @@ -737,6 +751,24 @@ public class ManagedDefaultRepositoryContent info.classifier = ""; info.remainder = classPostfix; } + } else if (fileName.startsWith(prefix + ".")) { + info.version = genericVersion; + info.remainder = StringUtils.removeStart( fileName, prefix ); + info.classifier = ""; + } else if (fileName.startsWith(info.id+"-")) { + String postFix = StringUtils.removeStart( fileName, info.id + "-" ); + String versionPart = StringUtils.substringBefore( postFix, "." ); + if (VersionUtil.isVersion(versionPart)) { + info.version = versionPart; + info.remainder = StringUtils.removeStart( postFix, versionPart ); + info.classifier = ""; + } else { + info.version = ""; + info.classifier = ""; + int dotPos = fileName.indexOf( "." ); + info.remainder = fileName.substring( dotPos ); + } + } else { @@ -747,10 +779,10 @@ public class ManagedDefaultRepositoryContent else { info.id = fileName; + info.version = ""; } log.debug( "Artifact does not match the version pattern {}", path ); info.artifactType = BaseArtifactTypes.UNKNOWN; - info.version = ""; info.classifier = ""; info.remainder = StringUtils.substringAfterLast( fileName, "." ); } @@ -1454,16 +1486,40 @@ public class ManagedDefaultRepositoryContent @Override public ContentItem toItem( String path ) throws LayoutException { + StorageAsset asset = getRepository( ).getAsset( path ); - if ( asset.isLeaf( ) ) - { - ItemSelector selector = getPathParser( ).toItemSelector( path ); - return getItem( selector ); - } - else - { - return getItemFromPath( asset ); + ContentItem item = getItemFromPath( asset ); + if (item instanceof DataItem) { + Artifact artifact = adaptItem( Artifact.class, item ); + if (asset.getParent()==null) { + throw new LayoutException( "Path too short for maven artifact "+path ); + } + String version = asset.getParent( ).getName( ); + if (asset.getParent().getParent()==null) { + throw new LayoutException( "Path too short for maven artifact " + path ); + } + String project = item.getAsset( ).getParent( ).getParent( ).getName( ); + DataItem dataItem = (DataItem) item; + if (StringUtils.isEmpty( dataItem.getExtension())) { + throw new LayoutException( "Missing type on maven artifact" ); + } + if (!project.equals(artifact.getId())) { + throw new LayoutException( "The maven artifact id "+artifact.getId() +" does not match the project id: " + project); + } + boolean versionIsGenericSnapshot = VersionUtil.isGenericSnapshot( version ); + boolean artifactVersionIsSnapshot = VersionUtil.isSnapshot( artifact.getArtifactVersion() ); + if ( versionIsGenericSnapshot && !artifactVersionIsSnapshot ) { + throw new LayoutException( "The maven artifact has no snapshot version in snapshot directory " + dataItem ); + } + if ( !versionIsGenericSnapshot && artifactVersionIsSnapshot) { + throw new LayoutException( "The maven artifact version " + artifact.getArtifactVersion() + " is a snapshot version but inside a non snapshot directory " + version ); + } + if ( !versionIsGenericSnapshot && !version.equals( artifact.getArtifactVersion() ) ) + { + throw new LayoutException( "The maven artifact version " + artifact.getArtifactVersion() + " does not match the version directory " + version ); + } } + return item; } @Override diff --git a/archiva-modules/archiva-maven/archiva-maven-repository/src/main/java/org/apache/archiva/repository/maven/content/MavenRepositoryRequestInfo.java b/archiva-modules/archiva-maven/archiva-maven-repository/src/main/java/org/apache/archiva/repository/maven/content/MavenRepositoryRequestInfo.java index 08afdcfbe..3a6145567 100644 --- a/archiva-modules/archiva-maven/archiva-maven-repository/src/main/java/org/apache/archiva/repository/maven/content/MavenRepositoryRequestInfo.java +++ b/archiva-modules/archiva-maven/archiva-maven-repository/src/main/java/org/apache/archiva/repository/maven/content/MavenRepositoryRequestInfo.java @@ -20,6 +20,7 @@ package org.apache.archiva.repository.maven.content; import org.apache.archiva.model.ArtifactReference; import org.apache.archiva.repository.*; +import org.apache.archiva.repository.content.ItemSelector; import org.apache.archiva.repository.content.PathParser; import org.apache.archiva.repository.features.RepositoryFeature; import org.apache.archiva.repository.metadata.base.MetadataTools; @@ -82,6 +83,12 @@ public class MavenRepositoryRequestInfo implements RepositoryRequestInfo } } + @Override + public ItemSelector toItemSelector( String requestPath ) throws LayoutException + { + return repository.getContent( ).toItemSelector( requestPath ); + } + /** *

* Tests the path to see if it conforms to the expectations of a metadata request. @@ -275,12 +282,18 @@ public class MavenRepositoryRequestInfo implements RepositoryRequestInfo * Default layout is the only layout that can contain maven-metadata.xml files, and * if the managedRepository is layout legacy, this request would never occur. */ - return requestedPath; + if (requestedPath.startsWith( "/" )) { + return requestedPath; + } else + { + return "/"+requestedPath; + } } + + // Treat as an artifact reference. - ArtifactReference ref = toArtifactReference( referencedResource ); - String adjustedPath = repository.getContent().toPath( ref ); + String adjustedPath = repository.getContent( ).toPath( repository.getContent( ).toItem( requestedPath ) ); return adjustedPath + supportfile; } diff --git a/archiva-modules/archiva-maven/archiva-maven-repository/src/main/java/org/apache/archiva/repository/maven/metadata/storage/Maven2RepositoryStorage.java b/archiva-modules/archiva-maven/archiva-maven-repository/src/main/java/org/apache/archiva/repository/maven/metadata/storage/Maven2RepositoryStorage.java index 51dec603f..e1c61aad2 100644 --- a/archiva-modules/archiva-maven/archiva-maven-repository/src/main/java/org/apache/archiva/repository/maven/metadata/storage/Maven2RepositoryStorage.java +++ b/archiva-modules/archiva-maven/archiva-maven-repository/src/main/java/org/apache/archiva/repository/maven/metadata/storage/Maven2RepositoryStorage.java @@ -644,6 +644,98 @@ public class Maven2RepositoryStorage } } + @Override + public ItemSelector applyServerSideRelocation(ManagedRepository managedRepository, ItemSelector artifactSelector) + throws ProxyDownloadException { + if ("pom".equals(artifactSelector.getType())) { + return artifactSelector; + } + + // Build the artifact POM reference + BaseRepositoryContentLayout layout; + try + { + layout = managedRepository.getContent( ).getLayout( BaseRepositoryContentLayout.class ); + } + catch ( LayoutException e ) + { + throw new ProxyDownloadException( "Could not set layout " + e.getMessage( ), new HashMap<>( ) ); + } + + RepositoryType repositoryType = managedRepository.getType(); + if (!proxyRegistry.hasHandler(repositoryType)) { + throw new ProxyDownloadException("No proxy handler found for repository type " + repositoryType, new HashMap<>()); + } + + + + ItemSelector selector = ArchivaItemSelector.builder( ) + .withNamespace( artifactSelector.getNamespace( ) ) + .withProjectId( artifactSelector.getArtifactId( ) ) + .withArtifactId( artifactSelector.getArtifactId( ) ) + .withVersion( artifactSelector.getVersion( ) ) + .withArtifactVersion( artifactSelector.getVersion( ) ) + .withType( "pom" ).build( ); + + Artifact pom = layout.getArtifact( selector ); + + RepositoryProxyHandler proxyHandler = proxyRegistry.getHandler(repositoryType).get(0); + + // Get the artifact POM from proxied repositories if needed + proxyHandler.fetchFromProxies(managedRepository, pom); + + // Open and read the POM from the managed repo + + if (!pom.exists()) { + return artifactSelector; + } + + try { + // MavenXpp3Reader leaves the file open, so we need to close it ourselves. + + Model model; + try (Reader reader = Channels.newReader(pom.getAsset().getReadChannel(), Charset.defaultCharset().name())) { + model = MAVEN_XPP_3_READER.read(reader); + } + + DistributionManagement dist = model.getDistributionManagement(); + if (dist != null) { + Relocation relocation = dist.getRelocation(); + if (relocation != null) { + ArchivaItemSelector.Builder relocatedBuilder = ArchivaItemSelector.builder( ); + // artifact is relocated : update the repositoryPath + if (relocation.getGroupId() != null) { + relocatedBuilder.withNamespace( relocation.getGroupId( ) ); + } else { + relocatedBuilder.withNamespace( artifactSelector.getNamespace( ) ); + } + if (relocation.getArtifactId() != null) { + relocatedBuilder.withArtifactId( relocation.getArtifactId( ) ); + } else { + relocatedBuilder.withArtifactId( artifactSelector.getArtifactId( ) ); + } + if (relocation.getVersion() != null) + { + relocatedBuilder.withVersion( relocation.getVersion( ) ); + } else { + relocatedBuilder.withVersion( artifactSelector.getVersion( ) ); + } + return relocatedBuilder.withArtifactVersion( artifactSelector.getArtifactVersion( ) ) + .withClassifier( artifactSelector.getClassifier( ) ) + .withType( artifactSelector.getType( ) ) + .withProjectId( artifactSelector.getProjectId( ) ) + .withExtension( artifactSelector.getExtension( ) ) + .build( ); + } + } + } catch (IOException e) { + // Unable to read POM : ignore. + } catch (XmlPullParserException e) { + // Invalid POM : ignore + } + return artifactSelector; + } + @Override public String getFilePath(String requestPath, org.apache.archiva.repository.ManagedRepository managedRepository) { diff --git a/archiva-modules/archiva-maven/archiva-maven-repository/src/test/java/org/apache/archiva/repository/maven/content/ManagedDefaultRepositoryContentTest.java b/archiva-modules/archiva-maven/archiva-maven-repository/src/test/java/org/apache/archiva/repository/maven/content/ManagedDefaultRepositoryContentTest.java index 532997b46..d86e75930 100644 --- a/archiva-modules/archiva-maven/archiva-maven-repository/src/test/java/org/apache/archiva/repository/maven/content/ManagedDefaultRepositoryContentTest.java +++ b/archiva-modules/archiva-maven/archiva-maven-repository/src/test/java/org/apache/archiva/repository/maven/content/ManagedDefaultRepositoryContentTest.java @@ -1014,7 +1014,7 @@ public class ManagedDefaultRepositoryContentTest path = "/org/apache/maven/shared/maven-downloader/1.1/maven-downloader-1.1.jar"; item = repoContent.toItem( path ); assertNotNull( item ); - assertTrue( item instanceof Artifact ); + assertTrue( item instanceof DataItem ); } diff --git a/archiva-modules/archiva-maven/archiva-maven-repository/src/test/java/org/apache/archiva/repository/maven/content/MavenRepositoryRequestInfoTest.java b/archiva-modules/archiva-maven/archiva-maven-repository/src/test/java/org/apache/archiva/repository/maven/content/MavenRepositoryRequestInfoTest.java index 6e8fe08b5..7317e57f3 100644 --- a/archiva-modules/archiva-maven/archiva-maven-repository/src/test/java/org/apache/archiva/repository/maven/content/MavenRepositoryRequestInfoTest.java +++ b/archiva-modules/archiva-maven/archiva-maven-repository/src/test/java/org/apache/archiva/repository/maven/content/MavenRepositoryRequestInfoTest.java @@ -72,6 +72,9 @@ public class MavenRepositoryRequestInfoTest @Inject FileLockManager fileLockManager; + @Inject + MavenContentHelper mavenContentHelper; + private MavenRepositoryRequestInfo repoRequest; @@ -109,6 +112,8 @@ public class MavenRepositoryRequestInfoTest ManagedDefaultRepositoryContent repoContent = new ManagedDefaultRepositoryContent(repository, artifactMappingProviders, fileTypes, fileLockManager); //repoContent = (ManagedRepositoryContent) lookup( ManagedRepositoryContent.class, "default" ); repository.setContent(repoContent); + repoContent.setMavenContentHelper( mavenContentHelper ); + repoRequest = new MavenRepositoryRequestInfo(repository); } @@ -430,7 +435,7 @@ public class MavenRepositoryRequestInfoTest ManagedRepositoryContent repository = createManagedRepo( "default" ); // Test (artifact) default to default - dual extension - assertEquals( "org/project/example-presentation/3.2/example-presentation-3.2.xml.zip", + assertEquals( "/org/project/example-presentation/3.2/example-presentation-3.2.xml.zip", repoRequest.toNativePath( "org/project/example-presentation/3.2/example-presentation-3.2.xml.zip") ); } @@ -442,7 +447,7 @@ public class MavenRepositoryRequestInfoTest ManagedRepositoryContent repository = createManagedRepo( "default" ); // Test (metadata) default to default - assertEquals( "org/apache/derby/derby/10.2.2.0/maven-metadata.xml.sha1", + assertEquals( "/org/apache/derby/derby/10.2.2.0/maven-metadata.xml.sha1", repoRequest.toNativePath( "org/apache/derby/derby/10.2.2.0/maven-metadata.xml.sha1") ); } diff --git a/archiva-modules/archiva-web/archiva-security/src/test/java/org/apache/archiva/security/mock/MockBeanServices.java b/archiva-modules/archiva-web/archiva-security/src/test/java/org/apache/archiva/security/mock/MockBeanServices.java index 9fb1184c6..edb686bff 100644 --- a/archiva-modules/archiva-web/archiva-security/src/test/java/org/apache/archiva/security/mock/MockBeanServices.java +++ b/archiva-modules/archiva-web/archiva-security/src/test/java/org/apache/archiva/security/mock/MockBeanServices.java @@ -31,6 +31,7 @@ import org.apache.archiva.components.taskqueue.TaskQueueException; import org.apache.archiva.repository.ManagedRepositoryContent; import org.apache.archiva.repository.ManagedRepository; import org.apache.archiva.metadata.audit.RepositoryListener; +import org.apache.archiva.repository.content.ItemSelector; import org.apache.archiva.scheduler.repository.model.RepositoryArchivaTaskScheduler; import org.apache.archiva.scheduler.repository.model.RepositoryTask; import org.apache.archiva.xml.XMLException; @@ -108,6 +109,12 @@ public class MockBeanServices } + @Override + public ItemSelector applyServerSideRelocation( ManagedRepository managedRepository, ItemSelector selector ) throws ProxyDownloadException + { + return null; + } + @Override public void deleteArtifact( MetadataRepository metadataRepository, String repositoryId, String namespace, diff --git a/archiva-modules/archiva-web/archiva-webdav/src/main/java/org/apache/archiva/webdav/ArchivaDavResourceFactory.java b/archiva-modules/archiva-web/archiva-webdav/src/main/java/org/apache/archiva/webdav/ArchivaDavResourceFactory.java index 8528b658c..b83dcdb61 100644 --- a/archiva-modules/archiva-web/archiva-webdav/src/main/java/org/apache/archiva/webdav/ArchivaDavResourceFactory.java +++ b/archiva-modules/archiva-web/archiva-webdav/src/main/java/org/apache/archiva/webdav/ArchivaDavResourceFactory.java @@ -68,6 +68,7 @@ import org.apache.archiva.repository.RepositoryRegistry; import org.apache.archiva.repository.RepositoryRequestInfo; import org.apache.archiva.repository.content.Artifact; import org.apache.archiva.repository.content.ContentItem; +import org.apache.archiva.repository.content.ItemSelector; import org.apache.archiva.repository.storage.fs.FilesystemStorage; import org.apache.archiva.repository.storage.StorageAsset; import org.apache.archiva.metadata.audit.AuditListener; @@ -791,22 +792,23 @@ public class ArchivaDavResourceFactory try { // Get the artifact reference in a layout neutral way. - ArtifactReference artifact = repositoryRequestInfo.toArtifactReference( path ); +// ArtifactReference artifact = repositoryRequestInfo.toArtifactReference( path ); + ItemSelector selector = repositoryRequestInfo.toItemSelector( path ); - if ( artifact != null ) + if ( selector != null ) { String repositoryLayout = managedRepository.getLayout(); RepositoryStorage repositoryStorage = this.applicationContext.getBean( "repositoryStorage#" + repositoryLayout, RepositoryStorage.class ); - repositoryStorage.applyServerSideRelocation( managedRepository, artifact ); + selector = repositoryStorage.applyServerSideRelocation( managedRepository, selector ); - StorageAsset proxiedFile = proxyHandler.fetchFromProxies( managedRepository, artifact ); + StorageAsset proxiedFile = proxyHandler.fetchFromProxies( managedRepository, selector ); - resource.setPath( managedRepository.getContent().toPath( artifact ) ); + resource.setPath( managedRepository.getContent().toPath( selector ) ); - log.debug( "Proxied artifact '{}:{}:{}'", artifact.getGroupId(), artifact.getArtifactId(), - artifact.getVersion() ); + log.debug( "Proxied artifact '{}:{}:{}:{}'", selector.getNamespace(), selector.getArtifactId(), + selector.getVersion(), selector.getArtifactVersion() ); return ( proxiedFile != null ); } diff --git a/archiva-modules/metadata/metadata-repository-api/src/main/java/org/apache/archiva/metadata/repository/storage/RepositoryStorage.java b/archiva-modules/metadata/metadata-repository-api/src/main/java/org/apache/archiva/metadata/repository/storage/RepositoryStorage.java index 6559be1c6..e56311506 100644 --- a/archiva-modules/metadata/metadata-repository-api/src/main/java/org/apache/archiva/metadata/repository/storage/RepositoryStorage.java +++ b/archiva-modules/metadata/metadata-repository-api/src/main/java/org/apache/archiva/metadata/repository/storage/RepositoryStorage.java @@ -27,6 +27,7 @@ import org.apache.archiva.model.ArtifactReference; import org.apache.archiva.policies.ProxyDownloadException; import org.apache.archiva.repository.ManagedRepositoryContent; import org.apache.archiva.repository.ManagedRepository; +import org.apache.archiva.repository.content.ItemSelector; import org.apache.archiva.xml.XMLException; import java.io.IOException; @@ -74,6 +75,21 @@ public interface RepositoryStorage void applyServerSideRelocation( ManagedRepository managedRepository, ArtifactReference artifact ) throws ProxyDownloadException; + /** + * A relocation capable client will request the POM prior to the artifact, and will then read meta-data and do + * client side relocation. A simplier client (like maven 1) will only request the artifact and not use the + * metadatas. + *

+ * For such clients, archiva does server-side relocation by reading itself the <relocation> element in + * metadatas and serving the expected artifact. + * @param managedRepository the used managed repository + * @param artifact the artifact reference + * @throws org.apache.archiva.policies.ProxyDownloadException + */ + ItemSelector applyServerSideRelocation( ManagedRepository managedRepository, ItemSelector selector ) + throws ProxyDownloadException; + + /** * add an other method to evaluate real path as when receiving -SNAPSHOT (for maven storage) * request redirect to the last build diff --git a/archiva-modules/plugins/metadata-store-cassandra/src/test/java/org/apache/archiva/metadata/repository/cassandra/MockRepositoryStorage.java b/archiva-modules/plugins/metadata-store-cassandra/src/test/java/org/apache/archiva/metadata/repository/cassandra/MockRepositoryStorage.java index ab520236f..d320ecbb4 100644 --- a/archiva-modules/plugins/metadata-store-cassandra/src/test/java/org/apache/archiva/metadata/repository/cassandra/MockRepositoryStorage.java +++ b/archiva-modules/plugins/metadata-store-cassandra/src/test/java/org/apache/archiva/metadata/repository/cassandra/MockRepositoryStorage.java @@ -37,6 +37,7 @@ import org.apache.archiva.policies.ProxyDownloadException; import org.apache.archiva.repository.ManagedRepositoryContent; import org.apache.archiva.repository.ManagedRepository; import org.apache.archiva.metadata.audit.RepositoryListener; +import org.apache.archiva.repository.content.ItemSelector; import org.apache.archiva.xml.XMLException; import java.io.IOException; @@ -112,6 +113,12 @@ public class MockRepositoryStorage } + @Override + public ItemSelector applyServerSideRelocation( ManagedRepository managedRepository, ItemSelector selector ) throws ProxyDownloadException + { + return null; + } + @Override public String getFilePath( String requestPath, org.apache.archiva.repository.ManagedRepository managedRepository ) { -- cgit v1.2.3