diff options
author | Martin Stockhammer <martin_s@apache.org> | 2019-06-10 23:33:34 +0200 |
---|---|---|
committer | Martin Stockhammer <martin_s@apache.org> | 2019-06-10 23:34:13 +0200 |
commit | 42257e90337da35c472d87f65aab065a6ec6f07d (patch) | |
tree | 332cfed1918c31d2810902eb6f15c937e80c8ef7 | |
parent | dac24551d0b9e31b21d23db1b1bb96c2e3a657e1 (diff) | |
download | archiva-42257e90337da35c472d87f65aab065a6ec6f07d.tar.gz archiva-42257e90337da35c472d87f65aab065a6ec6f07d.zip |
Using asset API in DAV
2 files changed, 174 insertions, 72 deletions
diff --git a/archiva-modules/archiva-base/archiva-checksum/src/main/java/org/apache/archiva/checksum/StreamingChecksum.java b/archiva-modules/archiva-base/archiva-checksum/src/main/java/org/apache/archiva/checksum/StreamingChecksum.java new file mode 100644 index 000000000..5e4aa7cfd --- /dev/null +++ b/archiva-modules/archiva-base/archiva-checksum/src/main/java/org/apache/archiva/checksum/StreamingChecksum.java @@ -0,0 +1,65 @@ +package org.apache.archiva.checksum; + +/* + * Licensed to the Apache Software Foundation (ASF) under one + * or more contributor license agreements. See the NOTICE file + * distributed with this work for additional information + * regarding copyright ownership. The ASF licenses this file + * to you 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. + */ + +import java.io.IOException; +import java.io.InputStream; +import java.io.OutputStream; +import java.util.List; +import java.util.stream.Collectors; + +/** + * + * Class that handles checksums with streams. + * + * @author Martin Stockhammer <martin_s@apache.org> + */ +public class StreamingChecksum +{ + static final int BUFFER_SIZE=4096; + + public static void updateChecksums( InputStream input, List<ChecksumAlgorithm> algorithms, List<OutputStream> checksumOutput) { + List<Checksum> checksums = algorithms.stream().map(a -> new Checksum( a )).collect( Collectors.toList()); + byte[] buffer = new byte[BUFFER_SIZE]; + int read; + try + { + while ( ( read = input.read( buffer ) ) >= 0 ) + { + for (Checksum cs : checksums ) { + cs.update( buffer, 0, read ); + } + } + int minIndex = Math.min(algorithms.size(), checksums.size()); + for (int csIndex = 0; csIndex<minIndex; csIndex++) { + Checksum cs = checksums.get(csIndex); + cs.finish(); + OutputStream os =checksumOutput.get(csIndex); + if (os!=null) + { + os.write( cs.getChecksum( ).getBytes( ) ); + } + } + + } catch ( IOException e ) { + + } + } +} 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 07ad2fdfa..d51b47f61 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 @@ -19,18 +19,19 @@ package org.apache.archiva.webdav; * under the License. */ -import org.apache.archiva.admin.model.RepositoryAdminException; -import org.apache.archiva.admin.model.beans.RemoteRepository; import org.apache.archiva.admin.model.managed.ManagedRepositoryAdmin; import org.apache.archiva.admin.model.remote.RemoteRepositoryAdmin; import org.apache.archiva.audit.Auditable; +import org.apache.archiva.checksum.ChecksumAlgorithm; +import org.apache.archiva.checksum.ChecksumUtil; +import org.apache.archiva.checksum.ChecksummedFile; +import org.apache.archiva.checksum.StreamingChecksum; import org.apache.archiva.common.filelock.FileLockManager; import org.apache.archiva.common.plexusbridge.PlexusSisuBridge; import org.apache.archiva.common.plexusbridge.PlexusSisuBridgeException; import org.apache.archiva.common.utils.PathUtil; import org.apache.archiva.common.utils.VersionUtil; import org.apache.archiva.configuration.ArchivaConfiguration; -import org.apache.archiva.configuration.RepositoryGroupConfiguration; import org.apache.archiva.indexer.ArchivaIndexingContext; import org.apache.archiva.indexer.merger.IndexMerger; import org.apache.archiva.indexer.merger.IndexMergerException; @@ -59,9 +60,15 @@ import org.apache.archiva.redback.policy.MustChangePasswordException; import org.apache.archiva.redback.system.SecuritySession; import org.apache.archiva.redback.users.User; import org.apache.archiva.redback.users.UserManager; -import org.apache.archiva.repository.*; +import org.apache.archiva.repository.LayoutException; +import org.apache.archiva.repository.ManagedRepository; +import org.apache.archiva.repository.ManagedRepositoryContent; +import org.apache.archiva.repository.ReleaseScheme; +import org.apache.archiva.repository.RepositoryGroup; +import org.apache.archiva.repository.RepositoryRegistry; +import org.apache.archiva.repository.RepositoryRequestInfo; import org.apache.archiva.repository.content.FilesystemAsset; -import org.apache.archiva.repository.content.maven2.MavenRepositoryRequestInfo; +import org.apache.archiva.repository.content.StorageAsset; import org.apache.archiva.repository.events.AuditListener; import org.apache.archiva.repository.features.IndexCreationFeature; import org.apache.archiva.repository.metadata.MetadataTools; @@ -86,8 +93,6 @@ import org.apache.jackrabbit.webdav.DavServletResponse; import org.apache.jackrabbit.webdav.DavSession; import org.apache.jackrabbit.webdav.lock.LockManager; import org.apache.jackrabbit.webdav.lock.SimpleLockManager; -import org.apache.maven.index.context.IndexingContext; -import org.codehaus.plexus.digest.ChecksumFile; import org.codehaus.plexus.digest.Digester; import org.codehaus.plexus.digest.DigesterException; import org.slf4j.Logger; @@ -102,6 +107,8 @@ import javax.inject.Named; import javax.servlet.http.HttpServletResponse; import javax.servlet.http.HttpSession; import java.io.IOException; +import java.io.OutputStream; +import java.io.OutputStreamWriter; import java.nio.file.Files; import java.nio.file.Path; import java.nio.file.Paths; @@ -111,7 +118,9 @@ import java.util.HashMap; import java.util.HashSet; import java.util.List; import java.util.Map; +import java.util.Objects; import java.util.Set; +import java.util.stream.Collectors; /** * @@ -167,7 +176,7 @@ public class ArchivaDavResourceFactory */ private final LockManager lockManager = new SimpleLockManager(); - private ChecksumFile checksum; + private ChecksummedFile checksum; private Digester digestSha1; @@ -190,10 +199,6 @@ public class ArchivaDavResourceFactory { this.archivaConfiguration = archivaConfiguration; this.applicationContext = applicationContext; - this.checksum = plexusSisuBridge.lookup( ChecksumFile.class ); - - this.digestMd5 = plexusSisuBridge.lookup( Digester.class, "md5" ); - this.digestSha1 = plexusSisuBridge.lookup( Digester.class, "sha1" ); } @@ -301,27 +306,29 @@ public class ArchivaDavResourceFactory { ArchivaDavResource res = (ArchivaDavResource) resource; - String filePath = - StringUtils.substringBeforeLast( res.getLocalResource().toAbsolutePath().toString().replace( '\\', '/' ), - "/" ); - filePath = filePath + "/maven-metadata-" + repoGroupConfig.getId() + ".xml"; - + String newPath = res.getAsset().getPath()+"/maven-metadata-" + sRepoId + ".xml"; // for MRM-872 handle checksums of the merged metadata files if ( repositoryRequestInfo.isSupportFile( requestedResource ) ) { - Path metadataChecksum = - Paths.get( filePath + "." + StringUtils.substringAfterLast( requestedResource, "." ) ); - - if ( Files.exists(metadataChecksum) ) + String metadataChecksumPath = newPath + "." + StringUtils.substringAfterLast( requestedResource, "." ); + StorageAsset metadataChecksum = repoGroup.getAsset( metadataChecksumPath ); + if ( repoGroup.getAsset( metadataChecksumPath ).exists() ) { LogicalResource logicalResource = new LogicalResource( getLogicalResource( archivaLocator, null, false ) ); - resource = - new ArchivaDavResource( metadataChecksum.toAbsolutePath().toString(), logicalResource.getPath(), null, - request.getRemoteAddr(), activePrincipal, request.getDavSession(), - archivaLocator, this, mimeTypes, auditListeners, scheduler, - fileLockManager ); + try + { + resource = + new ArchivaDavResource( metadataChecksum, logicalResource.getPath(), null, + request.getRemoteAddr(), activePrincipal, request.getDavSession(), + archivaLocator, this, mimeTypes, auditListeners, scheduler); + } + catch ( LayoutException e ) + { + log.error("Incompatible layout: {}", e.getMessage(), e); + throw new DavException( 500, e ); + } } } else @@ -352,16 +359,16 @@ public class ArchivaDavResourceFactory try { - Path resourceFile = writeMergedMetadataToFile( mergedMetadata, filePath ); + StorageAsset resourceFile = writeMergedMetadataToFile( repoGroup, mergedMetadata, newPath ); LogicalResource logicalResource = new LogicalResource( getLogicalResource( archivaLocator, null, false ) ); resource = - new ArchivaDavResource( resourceFile.toAbsolutePath().toString(), logicalResource.getPath(), null, + new ArchivaDavResource( resourceFile, logicalResource.getPath(), null, request.getRemoteAddr(), activePrincipal, request.getDavSession(), archivaLocator, this, mimeTypes, - auditListeners, scheduler, fileLockManager ); + auditListeners, scheduler); } catch ( RepositoryMetadataException r ) { @@ -379,6 +386,11 @@ public class ArchivaDavResourceFactory "Error occurred while generating checksum files." + de.getMessage() ); } + catch ( LayoutException e ) + { + log.error("Incompatible layout: {}", e.getMessage(), e); + throw new DavException(HttpServletResponse.SC_INTERNAL_SERVER_ERROR, "Incompatible layout for repository "+repoGroup.getId()); + } } } } @@ -414,11 +426,11 @@ public class ArchivaDavResourceFactory String requestedFileName = StringUtils.substringAfterLast( pathInfo, "/" ); Path temporaryIndexDirectory = buildMergedIndexDirectory( activePrincipal, request, repoGroup ); - asset = new FilesystemAsset() + FilesystemAsset asset = new FilesystemAsset( pathInfo, temporaryIndexDirectory.resolve(requestedFileName) ); Path resourceFile = temporaryIndexDirectory.resolve( requestedFileName ); try { - resource = new ArchivaDavResource( resourceFile.toAbsolutePath().toString(), requestedFileName, null, + resource = new ArchivaDavResource( asset, requestedFileName, null, request.getRemoteAddr(), activePrincipal, request.getDavSession(), archivaLocator, this, mimeTypes, auditListeners, scheduler ); } catch (LayoutException e) { @@ -429,8 +441,9 @@ public class ArchivaDavResourceFactory } else { - for ( String repositoryId : repositories ) + for ( ManagedRepository repository : repoGroup.getRepositories() ) { + String repositoryId = repository.getId(); ManagedRepositoryContent managedRepositoryContent; ManagedRepository managedRepository = repositoryRegistry.getManagedRepository( repositoryId ); if (managedRepository==null) { @@ -554,15 +567,24 @@ public class ArchivaDavResourceFactory path = path.substring( 1 ); } LogicalResource logicalResource = new LogicalResource( path ); - Path resourceFile = Paths.get( managedRepositoryContent.getRepoRoot(), path ); - resource = - new ArchivaDavResource( resourceFile.toAbsolutePath().toString(), path, managedRepositoryContent.getRepository(), - request.getRemoteAddr(), activePrincipal, request.getDavSession(), - archivaLocator, this, mimeTypes, auditListeners, scheduler, fileLockManager ); + StorageAsset repoAsset = managedRepositoryContent.getAsset( path ); + // Path resourceFile = Paths.get( managedRepositoryContent.getRepoRoot(), path ); + try + { + resource = + new ArchivaDavResource( repoAsset, path, managedRepositoryContent.getRepository(), + request.getRemoteAddr(), activePrincipal, request.getDavSession(), + archivaLocator, this, mimeTypes, auditListeners, scheduler ); + } + catch ( LayoutException e ) + { + log.error("Incompatible layout: {}", e.getMessage(), e); + throw new DavException( HttpServletResponse.SC_INTERNAL_SERVER_ERROR, e ); + } if ( WebdavMethodUtil.isReadMethod( request.getMethod() ) ) { - if ( archivaLocator.getHref( false ).endsWith( "/" ) && !Files.isDirectory( resourceFile ) ) + if ( archivaLocator.getHref( false ).endsWith( "/" ) && !repoAsset.isContainer() ) { // force a resource not found throw new DavException( HttpServletResponse.SC_NOT_FOUND, "Resource does not exist" ); @@ -571,10 +593,11 @@ public class ArchivaDavResourceFactory { if ( !resource.isCollection() ) { - boolean previouslyExisted = Files.exists(resourceFile); + boolean previouslyExisted = repoAsset.exists(); boolean fromProxy = fetchContentFromProxies( managedRepositoryContent, request, logicalResource ); + StorageAsset resourceAsset=null; // At this point the incoming request can either be in default or // legacy layout format. try @@ -582,17 +605,17 @@ public class ArchivaDavResourceFactory // Perform an adjustment of the resource to the managed // repository expected path. String localResourcePath = managedRepository.getRequestInfo().toNativePath( logicalResource.getPath() ); - resourceFile = Paths.get( managedRepositoryContent.getRepoRoot(), localResourcePath ); + resourceAsset = managedRepositoryContent.getAsset( localResourcePath ); resource = - new ArchivaDavResource( resourceFile.toAbsolutePath().toString(), logicalResource.getPath(), + new ArchivaDavResource( resourceAsset, logicalResource.getPath(), managedRepositoryContent.getRepository(), request.getRemoteAddr(), activePrincipal, request.getDavSession(), archivaLocator, this, mimeTypes, - auditListeners, scheduler, fileLockManager ); + auditListeners, scheduler ); } catch ( LayoutException e ) { - if ( !Files.exists(resourceFile) ) + if ( resourceAsset!=null && !resourceAsset.exists() ) { throw new DavException( HttpServletResponse.SC_NOT_FOUND, e ); } @@ -604,13 +627,13 @@ public class ArchivaDavResourceFactory + PROXIED_SUFFIX; log.debug( "Proxied artifact '{}' in repository '{}' (current user '{}')", - resourceFile.getFileName(), managedRepositoryContent.getId(), activePrincipal ); + resourceAsset.getName(), managedRepositoryContent.getId(), activePrincipal ); triggerAuditEvent( request.getRemoteAddr(), archivaLocator.getRepositoryId(), logicalResource.getPath(), action, activePrincipal ); } - if ( !Files.exists(resourceFile) ) + if ( !resourceAsset.exists() ) { throw new DavException( HttpServletResponse.SC_NOT_FOUND, "Resource does not exist" ); } @@ -710,10 +733,18 @@ public class ArchivaDavResourceFactory { logicalResource = logicalResource.substring( 1 ); } - Path resourceFile = Paths.get( managedRepositoryContent.getRepoRoot(), logicalResource ); - resource = new ArchivaDavResource( resourceFile.toAbsolutePath().toString(), logicalResource, - repo, davSession, archivaLocator, - this, mimeTypes, auditListeners, scheduler, fileLockManager ); + StorageAsset resourceAsset = managedRepositoryContent.getAsset( logicalResource ); + try + { + resource = new ArchivaDavResource( resourceAsset, logicalResource, + repo, davSession, archivaLocator, + this, mimeTypes, auditListeners, scheduler); + } + catch ( LayoutException e ) + { + log.error( "Incompatible layout: {}", e.getMessage( ), e ); + throw new DavException( HttpServletResponse.SC_INTERNAL_SERVER_ERROR, e ); + } resource.addLockManager( lockManager ); return resource; @@ -826,7 +857,7 @@ public class ArchivaDavResourceFactory // [MRM-503] - Metadata file need Pragma:no-cache response // header. if ( locator.getResourcePath().endsWith( "/maven-metadata.xml" ) || ( resource instanceof ArchivaDavResource - && ( Files.isDirectory( ArchivaDavResource.class.cast( resource ).getLocalResource()) ) ) ) + && ( ArchivaDavResource.class.cast( resource ).getAsset().isContainer() ) ) ) { response.setHeader( "Pragma", "no-cache" ); response.setHeader( "Cache-Control", "no-cache" ); @@ -1220,39 +1251,45 @@ public class ArchivaDavResourceFactory return allow; } - private Path writeMergedMetadataToFile( ArchivaRepositoryMetadata mergedMetadata, String outputFilename ) + private StorageAsset writeMergedMetadataToFile( RepositoryGroup repoGroup, ArchivaRepositoryMetadata mergedMetadata, String outputFilename ) throws RepositoryMetadataException, DigesterException, IOException { - Path outputFile = Paths.get( outputFilename ); - if ( Files.exists(outputFile) ) - { - org.apache.archiva.common.utils.FileUtils.deleteQuietly( outputFile ); - } - - Files.createDirectories(outputFile.getParent()); - RepositoryMetadataWriter.write( mergedMetadata, outputFile ); + StorageAsset asset = repoGroup.addAsset( outputFilename, false ); + OutputStream stream = asset.writeData( true ); + OutputStreamWriter sw = new OutputStreamWriter( stream, "UTF-8" ); + RepositoryMetadataWriter.write( mergedMetadata, sw ); - createChecksumFile( outputFilename, digestSha1 ); - createChecksumFile( outputFilename, digestMd5 ); - - return outputFile; + createChecksumFiles( repoGroup, outputFilename ); + return asset; } - private void createChecksumFile( String path, Digester digester ) - throws DigesterException, IOException - { - Path checksumFile = Paths.get( path + digester.getFilenameExtension() ); - if ( !Files.exists(checksumFile) ) + + private void createChecksumFiles(RepositoryGroup repo, String path) { + List<ChecksumAlgorithm> algorithms = ChecksumUtil.getAlgorithms( archivaConfiguration.getConfiguration( ).getArchivaRuntimeConfiguration( ).getChecksumTypes( ) ); + List<OutputStream> outStreams = algorithms.stream( ).map( algo -> { + String ext = algo.getDefaultExtension( ); + try + { + return repo.getAsset( path + "." + ext ).writeData( true ); + } + catch ( IOException e ) + { + e.printStackTrace( ); + return null; + } + } ).filter( Objects::nonNull ).collect( Collectors.toList( ) ); + try { - org.apache.archiva.common.utils.FileUtils.deleteQuietly( checksumFile ); - checksum.createChecksum( Paths.get( path ).toFile(), digester ); + StreamingChecksum.updateChecksums( repo.getAsset(path).getData(), algorithms, outStreams ); } - else if ( !Files.isRegularFile( checksumFile) ) + catch ( IOException e ) { - log.error( "Checksum file is not a file." ); + e.printStackTrace( ); } } + + private boolean isProjectReference( String requestedResource ) { try |