1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363 |
- package org.apache.archiva.rest.services;
- /*
- * 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 org.apache.archiva.admin.model.beans.ManagedRepository;
- import org.apache.archiva.common.utils.VersionComparator;
- import org.apache.archiva.common.utils.VersionUtil;
- import org.apache.archiva.repository.ManagedRepositoryContent;
- import org.apache.archiva.repository.content.base.ArchivaItemSelector;
- import org.apache.archiva.repository.maven.dependency.tree.DependencyTreeBuilder;
- import org.apache.archiva.maven.model.Artifact;
- import org.apache.archiva.maven.model.TreeEntry;
- import org.apache.archiva.metadata.generic.GenericMetadataFacet;
- import org.apache.archiva.metadata.model.ArtifactMetadata;
- import org.apache.archiva.metadata.model.MetadataFacet;
- import org.apache.archiva.metadata.model.ProjectVersionMetadata;
- import org.apache.archiva.metadata.model.ProjectVersionReference;
- import org.apache.archiva.metadata.repository.*;
- import org.apache.archiva.repository.maven.metadata.storage.ArtifactMetadataVersionComparator;
- import org.apache.archiva.repository.maven.metadata.storage.MavenProjectFacet;
- import org.apache.archiva.model.ArchivaRepositoryMetadata;
- import org.apache.archiva.proxy.ProxyRegistry;
- import org.apache.archiva.proxy.model.RepositoryProxyHandler;
- import org.apache.archiva.components.cache.Cache;
- import org.apache.archiva.repository.ReleaseScheme;
- import org.apache.archiva.repository.RepositoryException;
- import org.apache.archiva.repository.RepositoryNotFoundException;
- import org.apache.archiva.repository.RepositoryRegistry;
- import org.apache.archiva.repository.metadata.MetadataReader;
- import org.apache.archiva.repository.metadata.base.MetadataTools;
- import org.apache.archiva.repository.storage.fs.FsStorageUtil;
- import org.apache.archiva.repository.storage.StorageAsset;
- import org.apache.archiva.rest.api.model.*;
- import org.apache.archiva.rest.api.services.ArchivaRestServiceException;
- import org.apache.archiva.rest.api.services.BrowseService;
- import org.apache.archiva.rest.services.utils.ArtifactContentEntryComparator;
- import org.apache.archiva.security.ArchivaSecurityException;
- import org.apache.commons.collections4.CollectionUtils;
- import org.apache.commons.io.IOUtils;
- import org.apache.commons.lang3.StringUtils;
- import org.springframework.stereotype.Service;
-
- import javax.inject.Inject;
- import javax.inject.Named;
- import javax.ws.rs.core.Response;
- import java.io.IOException;
- import java.io.InputStream;
- import java.nio.charset.Charset;
- import java.nio.file.Files;
- import java.util.*;
- import java.util.jar.JarEntry;
- import java.util.jar.JarFile;
- import java.util.zip.ZipEntry;
-
- /**
- * @author Olivier Lamy
- * @since 1.4-M3
- */
- @Service( "browseService#rest" )
- public class DefaultBrowseService
- extends AbstractRestService
- implements BrowseService
- {
-
- private final Charset ARTIFACT_CONTENT_ENCODING=Charset.forName( "UTF-8" );
-
- @Inject
- private DependencyTreeBuilder dependencyTreeBuilder;
-
- @Inject
- ProxyRegistry proxyRegistry;
-
- @Inject
- RepositoryRegistry repositoryRegistry;
-
- @Inject
- @Named( value = "browse#versionMetadata" )
- private Cache<String, ProjectVersionMetadata> versionMetadataCache;
-
- private ManagedRepositoryContent getManagedRepositoryContent( String id) throws RepositoryException
- {
- org.apache.archiva.repository.ManagedRepository repo = repositoryRegistry.getManagedRepository( id );
- if (repo==null) {
- throw new RepositoryException( "Could not find repository "+id );
- }
- return repo.getContent();
- }
-
- @Override
- public BrowseResult getRootGroups( String repositoryId )
- throws ArchivaRestServiceException
- {
- List<String> selectedRepos = getSelectedRepos( repositoryId );
-
- Set<String> namespaces = new LinkedHashSet<String>();
-
- // TODO: this logic should be optional, particularly remembering we want to keep this code simple
- // it is located here to avoid the content repository implementation needing to do too much for what
- // is essentially presentation code
- Set<String> namespacesToCollapse = new LinkedHashSet<String>();
- RepositorySession repositorySession = null;
- try
- {
- repositorySession = repositorySessionFactory.createSession();
- }
- catch ( MetadataRepositoryException e )
- {
- e.printStackTrace( );
- }
- try
- {
- MetadataResolver metadataResolver = repositorySession.getResolver();
-
- for ( String repoId : selectedRepos )
- {
- namespacesToCollapse.addAll( metadataResolver.resolveRootNamespaces( repositorySession, repoId ) );
- }
- for ( String n : namespacesToCollapse )
- {
- // TODO: check performance of this
- namespaces.add( collapseNamespaces( repositorySession, metadataResolver, selectedRepos, n ) );
- }
- }
- catch ( MetadataResolutionException e )
- {
- throw new ArchivaRestServiceException( e.getMessage(),
- Response.Status.INTERNAL_SERVER_ERROR.getStatusCode(), e );
- }
- finally
- {
- repositorySession.close();
- }
-
- List<BrowseResultEntry> browseGroupResultEntries = new ArrayList<>( namespaces.size() );
- for ( String namespace : namespaces )
- {
- browseGroupResultEntries.add( new BrowseResultEntry( namespace, false ) );
- }
-
- Collections.sort( browseGroupResultEntries );
- return new BrowseResult( browseGroupResultEntries );
- }
-
- @Override
- public BrowseResult browseGroupId( String groupId, String repositoryId )
- throws ArchivaRestServiceException
- {
- List<String> selectedRepos = getSelectedRepos( repositoryId );
-
- Set<String> projects = new LinkedHashSet<>();
-
- RepositorySession repositorySession = null;
- try
- {
- repositorySession = repositorySessionFactory.createSession();
- }
- catch ( MetadataRepositoryException e )
- {
- e.printStackTrace( );
- }
- Set<String> namespaces;
- try
- {
- MetadataResolver metadataResolver = repositorySession.getResolver();
-
- Set<String> namespacesToCollapse = new LinkedHashSet<>();
- for ( String repoId : selectedRepos )
- {
- namespacesToCollapse.addAll( metadataResolver.resolveNamespaces( repositorySession, repoId, groupId ) );
-
- projects.addAll( metadataResolver.resolveProjects( repositorySession, repoId, groupId ) );
- }
-
- // TODO: this logic should be optional, particularly remembering we want to keep this code simple
- // it is located here to avoid the content repository implementation needing to do too much for what
- // is essentially presentation code
- namespaces = new LinkedHashSet<>();
- for ( String n : namespacesToCollapse )
- {
- // TODO: check performance of this
- namespaces.add(
- collapseNamespaces( repositorySession, metadataResolver, selectedRepos, groupId + "." + n ) );
- }
- }
- catch ( MetadataResolutionException e )
- {
- throw new ArchivaRestServiceException( e.getMessage(),
- Response.Status.INTERNAL_SERVER_ERROR.getStatusCode(), e );
- }
- finally
- {
- repositorySession.close();
- }
- List<BrowseResultEntry> browseGroupResultEntries = new ArrayList<>( namespaces.size() + projects.size() );
- for ( String namespace : namespaces )
- {
- browseGroupResultEntries.add( new BrowseResultEntry( namespace, false ).groupId( namespace ) );
- }
- for ( String project : projects )
- {
- browseGroupResultEntries.add(
- new BrowseResultEntry( groupId + '.' + project, true ).groupId( groupId ).artifactId( project ) );
- }
- Collections.sort( browseGroupResultEntries );
- return new BrowseResult( browseGroupResultEntries );
-
- }
-
- @Override
- public VersionsList getVersionsList( String groupId, String artifactId, String repositoryId )
- throws ArchivaRestServiceException
- {
- List<String> selectedRepos = getSelectedRepos( repositoryId );
-
- try
- {
- Collection<String> versions = getVersions( selectedRepos, groupId, artifactId );
- return new VersionsList( new ArrayList<>( versions ) );
- }
- catch ( MetadataResolutionException e )
- {
- throw new ArchivaRestServiceException( e.getMessage(),
- Response.Status.INTERNAL_SERVER_ERROR.getStatusCode(), e );
- }
-
- }
-
- private Collection<String> getVersions( List<String> selectedRepos, String groupId, String artifactId )
- throws MetadataResolutionException
-
- {
- RepositorySession repositorySession = null;
- try
- {
- repositorySession = repositorySessionFactory.createSession();
- }
- catch ( MetadataRepositoryException e )
- {
- e.printStackTrace( );
- }
- try
- {
- MetadataResolver metadataResolver = repositorySession.getResolver();
-
- Set<String> versions = new LinkedHashSet<String>();
-
- for ( String repoId : selectedRepos )
- {
- Collection<String> projectVersions =
- metadataResolver.resolveProjectVersions( repositorySession, repoId, groupId, artifactId );
- versions.addAll( projectVersions );
- }
-
- List<String> sortedVersions = new ArrayList<>( versions );
-
- Collections.sort( sortedVersions, VersionComparator.getInstance() );
-
- return sortedVersions;
- }
- finally
- {
- repositorySession.close();
- }
- }
-
- @Override
- public ProjectVersionMetadata getProjectMetadata( String groupId, String artifactId, String version,
- String repositoryId )
- throws ArchivaRestServiceException
- {
- List<String> selectedRepos = getSelectedRepos( repositoryId );
-
- RepositorySession repositorySession = null;
- try
- {
- repositorySession = repositorySessionFactory.createSession();
-
- MetadataResolver metadataResolver = repositorySession.getResolver();
-
- ProjectVersionMetadata versionMetadata = null;
- for ( String repoId : selectedRepos )
- {
- if ( versionMetadata == null || versionMetadata.isIncomplete() )
- {
- try
- {
- ProjectVersionMetadata versionMetadataTmp =
- metadataResolver.resolveProjectVersion( repositorySession, repoId, groupId, artifactId,
- version );
-
- if ( versionMetadata == null && versionMetadataTmp != null )
- {
- versionMetadata = versionMetadataTmp;
- }
-
-
- }
- catch ( MetadataResolutionException e )
- {
- log.warn( "Skipping invalid metadata while compiling shared model for {}:{} in repo {}: {}",
- groupId, artifactId, repoId, e.getMessage() );
- }
- }
- }
-
- return versionMetadata;
- } catch (MetadataRepositoryException e) {
- throw new ArchivaRestServiceException(e.getMessage(), e);
- } finally
- {
- if ( repositorySession != null )
- {
- repositorySession.close();
- }
- }
-
- }
-
- @Override
- public ProjectVersionMetadata getProjectVersionMetadata( String groupId, String artifactId, String repositoryId )
- throws ArchivaRestServiceException
- {
-
- List<String> selectedRepos = getSelectedRepos( repositoryId );
-
- RepositorySession repositorySession = null;
- try
- {
-
- Collection<String> projectVersions = getVersions( selectedRepos, groupId, artifactId );
-
- repositorySession = repositorySessionFactory.createSession();
-
- MetadataResolver metadataResolver = repositorySession.getResolver();
-
- ProjectVersionMetadata sharedModel = new ProjectVersionMetadata();
-
- MavenProjectFacet mavenFacet = new MavenProjectFacet();
- mavenFacet.setGroupId( groupId );
- mavenFacet.setArtifactId( artifactId );
- sharedModel.addFacet( mavenFacet );
-
- boolean isFirstVersion = true;
-
- for ( String version : projectVersions )
- {
- ProjectVersionMetadata versionMetadata = null;
- for ( String repoId : selectedRepos )
- {
- if ( versionMetadata == null || versionMetadata.isIncomplete() )
- {
- try
- {
- ProjectVersionMetadata projectVersionMetadataResolved = null;
- boolean useCache = !StringUtils.endsWith( version, VersionUtil.SNAPSHOT );
- String cacheKey = null;
- boolean cacheToUpdate = false;
- // FIXME a bit maven centric!!!
- // not a snapshot so get it from cache
- if ( useCache )
- {
- cacheKey = repoId + groupId + artifactId + version;
- projectVersionMetadataResolved = versionMetadataCache.get( cacheKey );
- }
- if ( useCache && projectVersionMetadataResolved != null )
- {
- versionMetadata = projectVersionMetadataResolved;
- }
- else
- {
- projectVersionMetadataResolved =
- metadataResolver.resolveProjectVersion( repositorySession, repoId, groupId,
- artifactId, version );
- versionMetadata = projectVersionMetadataResolved;
- cacheToUpdate = true;
- }
-
- if ( useCache && cacheToUpdate )
- {
- versionMetadataCache.put( cacheKey, projectVersionMetadataResolved );
- }
-
- }
- catch ( MetadataResolutionException e )
- {
- log.error( "Skipping invalid metadata while compiling shared model for " + groupId + ":"
- + artifactId + " in repo " + repoId + ": " + e.getMessage() );
- }
- }
- }
-
- if ( versionMetadata == null )
- {
- continue;
- }
-
- if ( isFirstVersion )
- {
- sharedModel = versionMetadata;
- sharedModel.setId( null );
- }
- else
- {
- MavenProjectFacet versionMetadataMavenFacet =
- (MavenProjectFacet) versionMetadata.getFacet( MavenProjectFacet.FACET_ID );
- if ( versionMetadataMavenFacet != null )
- {
- if ( mavenFacet.getPackaging() != null //
- && !StringUtils.equalsIgnoreCase( mavenFacet.getPackaging(),
- versionMetadataMavenFacet.getPackaging() ) )
- {
- mavenFacet.setPackaging( null );
- }
- }
-
- if ( StringUtils.isEmpty( sharedModel.getName() ) //
- && !StringUtils.isEmpty( versionMetadata.getName() ) )
- {
- sharedModel.setName( versionMetadata.getName() );
- }
-
- if ( sharedModel.getDescription() != null //
- && !StringUtils.equalsIgnoreCase( sharedModel.getDescription(),
- versionMetadata.getDescription() ) )
- {
- sharedModel.setDescription( StringUtils.isNotEmpty( versionMetadata.getDescription() )
- ? versionMetadata.getDescription()
- : "" );
- }
-
- if ( sharedModel.getIssueManagement() != null //
- && versionMetadata.getIssueManagement() != null //
- && !StringUtils.equalsIgnoreCase( sharedModel.getIssueManagement().getUrl(),
- versionMetadata.getIssueManagement().getUrl() ) )
- {
- sharedModel.setIssueManagement( versionMetadata.getIssueManagement() );
- }
-
- if ( sharedModel.getCiManagement() != null //
- && versionMetadata.getCiManagement() != null //
- && !StringUtils.equalsIgnoreCase( sharedModel.getCiManagement().getUrl(),
- versionMetadata.getCiManagement().getUrl() ) )
- {
- sharedModel.setCiManagement( versionMetadata.getCiManagement() );
- }
-
- if ( sharedModel.getOrganization() != null //
- && versionMetadata.getOrganization() != null //
- && !StringUtils.equalsIgnoreCase( sharedModel.getOrganization().getName(),
- versionMetadata.getOrganization().getName() ) )
- {
- sharedModel.setOrganization( versionMetadata.getOrganization() );
- }
-
- if ( sharedModel.getUrl() != null //
- && !StringUtils.equalsIgnoreCase( sharedModel.getUrl(), versionMetadata.getUrl() ) )
- {
- sharedModel.setUrl( versionMetadata.getUrl() );
- }
- }
-
- isFirstVersion = false;
- }
- return sharedModel;
- }
- catch (MetadataResolutionException | MetadataRepositoryException e )
- {
- throw new ArchivaRestServiceException( e.getMessage(),
- Response.Status.INTERNAL_SERVER_ERROR.getStatusCode(), e );
- }
- finally
- {
- if ( repositorySession != null )
- {
- repositorySession.close();
- }
- }
- }
-
- @Override
- public List<TreeEntry> getTreeEntries( String groupId, String artifactId, String version, String repositoryId )
- throws ArchivaRestServiceException
- {
- List<String> selectedRepos = getSelectedRepos( repositoryId );
-
- try
- {
- return dependencyTreeBuilder.buildDependencyTree( selectedRepos, groupId, artifactId, version );
- }
- catch ( Exception e )
- {
- log.error( e.getMessage(), e );
- }
-
- return Collections.emptyList();
- }
-
- @Override
- public List<ManagedRepository> getUserRepositories()
- throws ArchivaRestServiceException
- {
- try
- {
- return userRepositories.getAccessibleRepositories( getPrincipal() );
- }
- catch ( ArchivaSecurityException e )
- {
- throw new ArchivaRestServiceException( "repositories.read.observable.error",
- Response.Status.INTERNAL_SERVER_ERROR.getStatusCode(), e );
- }
- }
-
- @Override
- public List<ManagedRepository> getUserManagableRepositories() throws ArchivaRestServiceException {
- try
- {
- return userRepositories.getManagableRepositories( getPrincipal() );
- }
- catch ( ArchivaSecurityException e )
- {
- throw new ArchivaRestServiceException( "repositories.read.managable.error",
- Response.Status.INTERNAL_SERVER_ERROR.getStatusCode(), e );
- }
- }
-
- @Override
- public List<Artifact> getDependees( String groupId, String artifactId, String version, String repositoryId )
- throws ArchivaRestServiceException
- {
- List<ProjectVersionReference> references = new ArrayList<>();
- // TODO: what if we get duplicates across repositories?
- RepositorySession repositorySession = null;
- try
- {
- repositorySession = repositorySessionFactory.createSession();
- }
- catch ( MetadataRepositoryException e )
- {
- e.printStackTrace( );
- }
- try
- {
- MetadataResolver metadataResolver = repositorySession.getResolver();
- for ( String repoId : getObservableRepos() )
- {
- // TODO: what about if we want to see this irrespective of version?
- references.addAll(
- metadataResolver.resolveProjectReferences( repositorySession, repoId, groupId, artifactId,
- version ) );
- }
- }
- catch ( MetadataResolutionException e )
- {
- throw new ArchivaRestServiceException( e.getMessage(),
- Response.Status.INTERNAL_SERVER_ERROR.getStatusCode(), e );
- }
- finally
- {
- repositorySession.close();
- }
-
- List<Artifact> artifacts = new ArrayList<>( references.size() );
-
- for ( ProjectVersionReference projectVersionReference : references )
- {
- artifacts.add( new Artifact( projectVersionReference.getNamespace(), projectVersionReference.getProjectId(),
- projectVersionReference.getProjectVersion() ) );
- }
- return artifacts;
- }
-
- @Override
- public List<Entry> getMetadatas( String groupId, String artifactId, String version, String repositoryId )
- throws ArchivaRestServiceException
- {
- ProjectVersionMetadata projectVersionMetadata =
- getProjectMetadata( groupId, artifactId, version, repositoryId );
- if ( projectVersionMetadata == null )
- {
- return Collections.emptyList();
- }
- MetadataFacet metadataFacet = projectVersionMetadata.getFacet( GenericMetadataFacet.FACET_ID );
-
- if ( metadataFacet == null )
- {
- return Collections.emptyList();
- }
- Map<String, String> map = metadataFacet.toProperties();
- List<Entry> entries = new ArrayList<>( map.size() );
-
- for ( Map.Entry<String, String> entry : map.entrySet() )
- {
- entries.add( new Entry( entry.getKey(), entry.getValue() ) );
- }
-
- return entries;
- }
-
- @Override
- public ActionStatus addMetadata( String groupId, String artifactId, String version, String key, String value,
- String repositoryId )
- throws ArchivaRestServiceException
- {
- ProjectVersionMetadata projectVersionMetadata =
- getProjectMetadata( groupId, artifactId, version, repositoryId );
-
- if ( projectVersionMetadata == null )
- {
- return new ActionStatus( false );
- }
-
- Map<String, String> properties = new HashMap<>();
-
- MetadataFacet metadataFacet = projectVersionMetadata.getFacet( GenericMetadataFacet.FACET_ID );
-
- if ( metadataFacet != null && metadataFacet.toProperties() != null )
- {
- properties.putAll( metadataFacet.toProperties() );
- }
- else
- {
- metadataFacet = new GenericMetadataFacet();
- }
-
- properties.put( key, value );
-
- metadataFacet.fromProperties( properties );
-
- projectVersionMetadata.addFacet( metadataFacet );
-
- RepositorySession repositorySession = null;
- try
- {
- repositorySession = repositorySessionFactory.createSession();
- }
- catch ( MetadataRepositoryException e )
- {
- e.printStackTrace( );
- }
-
- try
- {
- MetadataRepository metadataRepository = repositorySession.getRepository();
-
- metadataRepository.updateProjectVersion(repositorySession , repositoryId, groupId, artifactId, projectVersionMetadata );
-
- repositorySession.save();
- }
- catch (MetadataRepositoryException | MetadataSessionException e )
- {
- log.error( e.getMessage(), e );
- throw new ArchivaRestServiceException( e.getMessage(),
- Response.Status.INTERNAL_SERVER_ERROR.getStatusCode(), e );
- }
- finally
- {
- repositorySession.close();
- }
- return new ActionStatus( true );
- }
-
- @Override
- public ActionStatus deleteMetadata( String groupId, String artifactId, String version, String key, String repositoryId )
- throws ArchivaRestServiceException
- {
- ProjectVersionMetadata projectVersionMetadata =
- getProjectMetadata( groupId, artifactId, version, repositoryId );
-
- if ( projectVersionMetadata == null )
- {
- return new ActionStatus( false );
- }
-
- GenericMetadataFacet metadataFacet =
- (GenericMetadataFacet) projectVersionMetadata.getFacet( GenericMetadataFacet.FACET_ID );
-
- if ( metadataFacet != null && metadataFacet.toProperties() != null )
- {
- Map<String, String> properties = metadataFacet.toProperties();
- properties.remove( key );
- metadataFacet.setAdditionalProperties( properties );
- }
- else
- {
- return new ActionStatus( true );
- }
-
- RepositorySession repositorySession = null;
- try
- {
- repositorySession = repositorySessionFactory.createSession();
- }
- catch ( MetadataRepositoryException e )
- {
- e.printStackTrace( );
- }
-
- try
- {
- MetadataRepository metadataRepository = repositorySession.getRepository();
-
- metadataRepository.updateProjectVersion(repositorySession , repositoryId, groupId, artifactId, projectVersionMetadata );
-
- repositorySession.save();
- }
- catch (MetadataRepositoryException | MetadataSessionException e )
- {
- log.error( e.getMessage(), e );
- throw new ArchivaRestServiceException( e.getMessage(),
- Response.Status.INTERNAL_SERVER_ERROR.getStatusCode(), e );
- }
- finally
- {
- repositorySession.close();
- }
- return new ActionStatus( true );
- }
-
- @Override
- public List<ArtifactContentEntry> getArtifactContentEntries( String groupId, String artifactId, String version,
- String classifier, String type, String path,
- String repositoryId )
- throws ArchivaRestServiceException
- {
- List<String> selectedRepos = getSelectedRepos( repositoryId );
- try
- {
- for ( String repoId : selectedRepos )
- {
-
- ManagedRepositoryContent managedRepositoryContent =
- getManagedRepositoryContent( repoId );
- ArchivaItemSelector itemSelector = ArchivaItemSelector.builder( ).withNamespace( groupId )
- .withProjectId( artifactId ).withVersion( version ).withClassifier( classifier )
- .withType( StringUtils.isEmpty( type ) ? "jar" : type )
- .withArtifactId( artifactId ).build();
- org.apache.archiva.repository.content.Artifact archivaArtifact = managedRepositoryContent.getItem( itemSelector ).adapt( org.apache.archiva.repository.content.Artifact.class );
- StorageAsset file = archivaArtifact.getAsset();
- if ( file.exists() )
- {
- return readFileEntries( file, path, repoId );
- }
- }
- }
- catch ( IOException e )
- {
- log.error( e.getMessage(), e );
- throw new ArchivaRestServiceException( e.getMessage(),
- Response.Status.INTERNAL_SERVER_ERROR.getStatusCode(), e );
- }
- catch ( RepositoryNotFoundException e )
- {
- log.error( e.getMessage(), e );
- throw new ArchivaRestServiceException( e.getMessage(),
- Response.Status.INTERNAL_SERVER_ERROR.getStatusCode(), e );
- }
- catch ( RepositoryException e )
- {
- log.error( e.getMessage(), e );
- throw new ArchivaRestServiceException( e.getMessage(),
- Response.Status.INTERNAL_SERVER_ERROR.getStatusCode(), e );
- }
- return Collections.emptyList();
- }
-
- @Override
- public List<Artifact> getArtifactDownloadInfos( String groupId, String artifactId, String version,
- String repositoryId )
- throws ArchivaRestServiceException
- {
- List<String> selectedRepos = getSelectedRepos( repositoryId );
-
- List<Artifact> artifactDownloadInfos = new ArrayList<>();
-
- try (RepositorySession session = repositorySessionFactory.createSession())
- {
- MetadataResolver metadataResolver = session.getResolver();
- for ( String repoId : selectedRepos )
- {
- List<ArtifactMetadata> artifacts = new ArrayList<>(
- metadataResolver.resolveArtifacts( session, repoId, groupId, artifactId, version ) );
- Collections.sort( artifacts, ArtifactMetadataVersionComparator.INSTANCE );
- if ( artifacts != null && !artifacts.isEmpty() )
- {
- return buildArtifacts( artifacts, repoId );
- }
- }
- }
- catch ( MetadataResolutionException e )
- {
- log.error( e.getMessage(), e );
- throw new ArchivaRestServiceException( e.getMessage(),
- Response.Status.INTERNAL_SERVER_ERROR.getStatusCode(), e );
- }
- catch ( MetadataRepositoryException e )
- {
- e.printStackTrace( );
- }
-
- return artifactDownloadInfos;
- }
-
- @Override
- public ArtifactContent getArtifactContentText( String groupId, String artifactId, String version, String classifier,
- String type, String path, String repositoryId )
- throws ArchivaRestServiceException
- {
- List<String> selectedRepos = getSelectedRepos( repositoryId );
- try
- {
- for ( String repoId : selectedRepos )
- {
-
- ManagedRepositoryContent managedRepositoryContent = null;
- try
- {
- managedRepositoryContent = getManagedRepositoryContent( repoId );
- }
- catch ( RepositoryException e )
- {
- log.error("No repository content found for "+repoId);
- continue;
- }
- ArchivaItemSelector itemSelector = ArchivaItemSelector.builder( ).withNamespace( groupId )
- .withProjectId( artifactId ).withVersion( version ).withClassifier( classifier )
- .withType( StringUtils.isEmpty( type ) ? "jar" : type )
- .withArtifactId( artifactId ).build();
- org.apache.archiva.repository.content.Artifact archivaArtifact = managedRepositoryContent.getItem( itemSelector ).adapt( org.apache.archiva.repository.content.Artifact.class );
- StorageAsset file = archivaArtifact.getAsset( );
- if ( !file.exists() )
- {
- log.debug( "file: {} not exists for repository: {} try next repository", file, repoId );
- continue;
- }
- if ( StringUtils.isNotBlank( path ) )
- {
- // zip entry of the path -> path must a real file entry of the archive
- FsStorageUtil.PathInformation pathInfo = FsStorageUtil.getAssetDataAsPath(file);
- JarFile jarFile = new JarFile( pathInfo.getPath().toFile());
- ZipEntry zipEntry = jarFile.getEntry( path );
- try (InputStream inputStream = jarFile.getInputStream( zipEntry ))
- {
- return new ArtifactContent( IOUtils.toString( inputStream, ARTIFACT_CONTENT_ENCODING ), repoId );
- }
- finally
- {
- closeQuietly( jarFile );
- if (pathInfo.isTmpFile()) {
- Files.deleteIfExists(pathInfo.getPath());
- }
- }
- }
- try(InputStream readStream = file.getReadStream()) {
- return new ArtifactContent(IOUtils.toString(readStream, ARTIFACT_CONTENT_ENCODING), repoId);
- }
- }
- }
- catch ( IOException e )
- {
- log.error( e.getMessage(), e );
- throw new ArchivaRestServiceException( e.getMessage(),
- Response.Status.INTERNAL_SERVER_ERROR.getStatusCode(), e );
- }
- log.debug( "artifact: {}:{}:{}:{}:{} not found", groupId, artifactId, version, classifier, type );
- // 404 ?
- return new ArtifactContent();
- }
-
- @Override
- public AvailabilityStatus artifactAvailable( String groupId, String artifactId, String version, String classifier,
- String repositoryId )
- throws ArchivaRestServiceException
- {
- List<String> selectedRepos = getSelectedRepos( repositoryId );
-
- boolean snapshot = VersionUtil.isSnapshot( version );
-
- try
- {
- for ( String repoId : selectedRepos )
- {
-
- org.apache.archiva.repository.ManagedRepository managedRepo = repositoryRegistry.getManagedRepository(repoId);
- if (!proxyRegistry.hasHandler(managedRepo.getType())) {
- throw new RepositoryException( "No proxy handler found for repository type "+managedRepo.getType());
- }
- RepositoryProxyHandler proxyHandler = proxyRegistry.getHandler(managedRepo.getType()).get(0);
- if ( ( snapshot && !managedRepo.getActiveReleaseSchemes().contains(ReleaseScheme.SNAPSHOT) ) || ( !snapshot
- && managedRepo.getActiveReleaseSchemes().contains(ReleaseScheme.SNAPSHOT) ) )
- {
- continue;
- }
- ManagedRepositoryContent managedRepositoryContent = getManagedRepositoryContent( repoId );
-
- // FIXME default to jar which can be wrong for war zip etc....
- ArchivaItemSelector itemSelector = ArchivaItemSelector.builder( ).withNamespace( groupId )
- .withProjectId( artifactId ).withVersion( version ).withClassifier( StringUtils.isEmpty( classifier )
- ? ""
- : classifier )
- .withType( "jar" )
- .withArtifactId( artifactId ).build();
- org.apache.archiva.repository.content.Artifact archivaArtifact = managedRepositoryContent.getItem( itemSelector ).adapt( org.apache.archiva.repository.content.Artifact.class );
- StorageAsset file = archivaArtifact.getAsset( );
-
- if ( file != null && file.exists() )
- {
- return new AvailabilityStatus( true );
- }
-
- // in case of SNAPSHOT we can have timestamped version locally !
- if ( StringUtils.endsWith( version, VersionUtil.SNAPSHOT ) )
- {
- StorageAsset metadataFile = file.getStorage().getAsset(file.getParent().getPath()+"/"+MetadataTools.MAVEN_METADATA );
- if ( metadataFile.exists() )
- {
- MetadataReader metadataReader = repositoryRegistry.getMetadataReader( managedRepositoryContent.getRepository( ).getType( ) );
- ArchivaRepositoryMetadata archivaRepositoryMetadata =
- metadataReader.read( metadataFile );
- int buildNumber = archivaRepositoryMetadata.getSnapshotVersion().getBuildNumber();
- String timeStamp = archivaRepositoryMetadata.getSnapshotVersion().getTimestamp();
- // rebuild file name with timestamped version and build number
- String timeStampFileName = new StringBuilder( artifactId ).append( '-' ) //
- .append( StringUtils.remove( version, "-" + VersionUtil.SNAPSHOT ) ) //
- .append( '-' ).append( timeStamp ) //
- .append( '-' ).append( Integer.toString( buildNumber ) ) //
- .append( ( StringUtils.isEmpty( classifier ) ? "" : "-" + classifier ) ) //
- .append( ".jar" ).toString();
-
- StorageAsset timeStampFile = file.getStorage().getAsset(file.getParent().getPath() + "/" + timeStampFileName );
- log.debug( "try to find timestamped snapshot version file: {}", timeStampFile.getPath() );
- if ( timeStampFile.exists() )
- {
- return new AvailabilityStatus( true );
- }
- }
- }
-
- String path = managedRepositoryContent.toPath( archivaArtifact );
-
- file = proxyHandler.fetchFromProxies( managedRepositoryContent.getRepository(), path );
-
- if ( file != null && file.exists() )
- {
- // download pom now
- String pomPath = StringUtils.substringBeforeLast( path, ".jar" ) + ".pom";
- proxyHandler.fetchFromProxies( managedRepositoryContent.getRepository(), pomPath );
- return new AvailabilityStatus( true );
- }
- }
- } catch ( RepositoryException e )
- {
- log.error( e.getMessage(), e );
- throw new ArchivaRestServiceException( e.getMessage(),
- Response.Status.INTERNAL_SERVER_ERROR.getStatusCode(), e );
- }
-
- return new AvailabilityStatus( false );
- }
-
- @Override
- public AvailabilityStatus artifactAvailable( String groupId, String artifactId, String version, String repositoryId )
- throws ArchivaRestServiceException
- {
- return artifactAvailable( groupId, artifactId, version, null, repositoryId );
- }
-
- @Override
- public List<Artifact> getArtifacts( String repositoryId )
- throws ArchivaRestServiceException
- {
- RepositorySession repositorySession = null;
- try
- {
- repositorySession = repositorySessionFactory.createSession();
- }
- catch ( MetadataRepositoryException e )
- {
- e.printStackTrace( );
- }
- try
- {
- List<ArtifactMetadata> artifactMetadatas = repositorySession.getRepository().getArtifacts(repositorySession , repositoryId );
- return buildArtifacts( artifactMetadatas, repositoryId );
- }
- catch ( MetadataRepositoryException e )
- {
- throw new ArchivaRestServiceException( e.getMessage(), e );
- }
- finally
- {
- repositorySession.close();
- }
- }
-
- @Override
- public List<Artifact> getArtifactsByProjectVersionMetadata( String key, String value, String repositoryId )
- throws ArchivaRestServiceException
- {
- RepositorySession repositorySession = null;
- try
- {
- repositorySession = repositorySessionFactory.createSession();
- }
- catch ( MetadataRepositoryException e )
- {
- e.printStackTrace( );
- }
- try
- {
- List<ArtifactMetadata> artifactMetadatas = repositorySession.getRepository().getArtifactsByProjectVersionFacet(repositorySession , key, value, repositoryId );
- return buildArtifacts( artifactMetadatas, repositoryId );
- }
- catch ( MetadataRepositoryException e )
- {
- throw new ArchivaRestServiceException( e.getMessage(), e );
- }
- finally
- {
- repositorySession.close();
- }
- }
-
- @Override
- public List<Artifact> getArtifactsByMetadata( String key, String value, String repositoryId )
- throws ArchivaRestServiceException
- {
- RepositorySession repositorySession = null;
- try
- {
- repositorySession = repositorySessionFactory.createSession();
- }
- catch ( MetadataRepositoryException e )
- {
- e.printStackTrace( );
- }
- try
- {
- List<ArtifactMetadata> artifactMetadatas = repositorySession.getRepository().getArtifactsByAttribute(repositorySession , key, value, repositoryId );
- return buildArtifacts( artifactMetadatas, repositoryId );
- }
- catch ( MetadataRepositoryException e )
- {
- throw new ArchivaRestServiceException( e.getMessage(), e );
- }
- finally
- {
- repositorySession.close();
- }
- }
-
- @Override
- public List<Artifact> getArtifactsByProperty( String key, String value, String repositoryId )
- throws ArchivaRestServiceException
- {
- RepositorySession repositorySession = null;
- try
- {
- repositorySession = repositorySessionFactory.createSession();
- }
- catch ( MetadataRepositoryException e )
- {
- e.printStackTrace( );
- }
- try
- {
- List<ArtifactMetadata> artifactMetadatas = repositorySession.getRepository().getArtifactsByProjectVersionAttribute(repositorySession , key, value, repositoryId );
- return buildArtifacts( artifactMetadatas, repositoryId );
- }
- catch ( MetadataRepositoryException e )
- {
- throw new ArchivaRestServiceException( e.getMessage(), e );
- }
- finally
- {
- repositorySession.close();
- }
- }
-
- @Override
- public ActionStatus importMetadata( MetadataAddRequest metadataAddRequest, String repositoryId )
- throws ArchivaRestServiceException
- {
- ActionStatus result = new ActionStatus( true );
- for ( Map.Entry<String, String> metadata : metadataAddRequest.getMetadatas().entrySet() )
- {
- result = addMetadata( metadataAddRequest.getGroupId(), metadataAddRequest.getArtifactId(),
- metadataAddRequest.getVersion(), metadata.getKey(), metadata.getValue(),
- repositoryId );
- if ( !result.isSuccess() )
- {
- break;
- }
- }
- return result;
- }
-
- @Override
- public List<Artifact> searchArtifacts( String text, String repositoryId, Boolean exact )
- throws ArchivaRestServiceException
- {
- try(RepositorySession repositorySession = repositorySessionFactory.createSession())
- {
- List<ArtifactMetadata> artifactMetadatas =
- repositorySession.getRepository().searchArtifacts(repositorySession , repositoryId, text, exact == null ? false : exact );
- return buildArtifacts( artifactMetadatas, repositoryId );
- }
- catch ( MetadataRepositoryException e )
- {
- throw new ArchivaRestServiceException( e.getMessage(), e );
- }
- }
-
- @Override
- public List<Artifact> searchArtifacts( String key, String text, String repositoryId, Boolean exact )
- throws ArchivaRestServiceException
- {
- RepositorySession repositorySession = null;
- try
- {
- repositorySession = repositorySessionFactory.createSession();
- }
- catch ( MetadataRepositoryException e )
- {
- e.printStackTrace( );
- }
- try
- {
- List<ArtifactMetadata> artifactMetadatas =
- repositorySession.getRepository().searchArtifacts(repositorySession , repositoryId, key, text, exact == null ? false : exact );
- return buildArtifacts( artifactMetadatas, repositoryId );
- }
- catch ( MetadataRepositoryException e )
- {
- throw new ArchivaRestServiceException( e.getMessage(), e );
- }
- finally
- {
- repositorySession.close();
- }
- }
-
- //---------------------------
- // internals
- //---------------------------
-
- private void closeQuietly( JarFile jarFile )
- {
- if ( jarFile != null )
- {
- try
- {
- jarFile.close();
- }
- catch ( IOException e )
- {
- log.warn( "ignore error closing jarFile {}", jarFile.getName() );
- }
- }
- }
-
- protected List<ArtifactContentEntry> readFileEntries(final StorageAsset file, final String filterPath, final String repoId )
- throws IOException
- {
- String cleanedfilterPath = filterPath==null ? "" : (StringUtils.startsWith(filterPath, "/") ?
- StringUtils.substringAfter(filterPath, "/") : filterPath);
- Map<String, ArtifactContentEntry> artifactContentEntryMap = new HashMap<>();
- int filterDepth = StringUtils.countMatches( cleanedfilterPath, "/" );
- if (!StringUtils.endsWith(cleanedfilterPath,"/") && !StringUtils.isEmpty(cleanedfilterPath)) {
- filterDepth++;
- }
-
- FsStorageUtil.PathInformation pathInfo = FsStorageUtil.getAssetDataAsPath(file);
- JarFile jarFile = new JarFile(pathInfo.getPath().toFile());
- try
- {
- Enumeration<JarEntry> jarEntryEnumeration = jarFile.entries();
- while ( jarEntryEnumeration.hasMoreElements() )
- {
- JarEntry currentEntry = jarEntryEnumeration.nextElement();
- String cleanedEntryName = StringUtils.endsWith( currentEntry.getName(), "/" ) ? //
- StringUtils.substringBeforeLast( currentEntry.getName(), "/" ) : currentEntry.getName();
- String entryRootPath = getRootPath( cleanedEntryName );
- int depth = StringUtils.countMatches( cleanedEntryName, "/" );
- if ( StringUtils.isEmpty( cleanedfilterPath ) //
- && !artifactContentEntryMap.containsKey( entryRootPath ) //
- && depth == filterDepth )
- {
-
- artifactContentEntryMap.put( entryRootPath,
- new ArtifactContentEntry( entryRootPath, !currentEntry.isDirectory(),
- depth, repoId ) );
- }
- else
- {
- if ( StringUtils.startsWith( cleanedEntryName, cleanedfilterPath ) //
- && ( depth == filterDepth || ( !currentEntry.isDirectory() && depth == filterDepth ) ) )
- {
- artifactContentEntryMap.put( cleanedEntryName, new ArtifactContentEntry( cleanedEntryName,
- !currentEntry.isDirectory(),
- depth, repoId ) );
- }
- }
- }
-
- if ( StringUtils.isNotEmpty( cleanedfilterPath ) )
- {
- Map<String, ArtifactContentEntry> filteredArtifactContentEntryMap = new HashMap<>();
-
- for ( Map.Entry<String, ArtifactContentEntry> entry : artifactContentEntryMap.entrySet() )
- {
- filteredArtifactContentEntryMap.put( entry.getKey(), entry.getValue() );
- }
-
- List<ArtifactContentEntry> sorted = getSmallerDepthEntries( filteredArtifactContentEntryMap );
- if ( sorted == null )
- {
- return Collections.emptyList();
- }
- Collections.sort( sorted, ArtifactContentEntryComparator.INSTANCE );
- return sorted;
- }
- }
- finally
- {
- if ( jarFile != null )
- {
- jarFile.close();
- }
- if (pathInfo.isTmpFile()) {
- Files.deleteIfExists(pathInfo.getPath());
- }
- }
- List<ArtifactContentEntry> sorted = new ArrayList<>( artifactContentEntryMap.values() );
- Collections.sort( sorted, ArtifactContentEntryComparator.INSTANCE );
- return sorted;
- }
-
- private List<ArtifactContentEntry> getSmallerDepthEntries( Map<String, ArtifactContentEntry> entries )
- {
- int smallestDepth = Integer.MAX_VALUE;
- Map<Integer, List<ArtifactContentEntry>> perDepthList = new HashMap<>();
- for ( Map.Entry<String, ArtifactContentEntry> entry : entries.entrySet() )
- {
-
- ArtifactContentEntry current = entry.getValue();
-
- if ( current.getDepth() < smallestDepth )
- {
- smallestDepth = current.getDepth();
- }
-
- List<ArtifactContentEntry> currentList = perDepthList.get( current.getDepth() );
-
- if ( currentList == null )
- {
- currentList = new ArrayList<>();
- currentList.add( current );
- perDepthList.put( current.getDepth(), currentList );
- }
- else
- {
- currentList.add( current );
- }
-
- }
-
- return perDepthList.get( smallestDepth );
- }
-
- /**
- * @param path
- * @return org/apache -> org , org -> org
- */
- private String getRootPath( String path )
- {
- if ( StringUtils.contains( path, '/' ) )
- {
- return StringUtils.substringBefore( path, "/" );
- }
- return path;
- }
-
- private List<String> getSelectedRepos( String repositoryId )
- throws ArchivaRestServiceException
- {
-
- List<String> selectedRepos = getObservableRepos();
-
- if ( CollectionUtils.isEmpty( selectedRepos ) )
- {
- return Collections.emptyList();
- }
-
- if ( StringUtils.isNotEmpty( repositoryId ) )
- {
- // check user has karma on the repository
- if ( !selectedRepos.contains( repositoryId ) )
- {
- throw new ArchivaRestServiceException( "browse.root.groups.repositoy.denied",
- Response.Status.FORBIDDEN.getStatusCode(), null );
- }
- selectedRepos = Collections.singletonList( repositoryId );
- }
- return selectedRepos;
- }
-
-
- private String collapseNamespaces( RepositorySession repositorySession, MetadataResolver metadataResolver,
- Collection<String> repoIds, String n )
- throws MetadataResolutionException
- {
- Set<String> subNamespaces = new LinkedHashSet<String>();
- for ( String repoId : repoIds )
- {
- subNamespaces.addAll( metadataResolver.resolveNamespaces( repositorySession, repoId, n ) );
- }
- if ( subNamespaces.size() != 1 )
- {
- log.debug( "{} is not collapsible as it has sub-namespaces: {}", n, subNamespaces );
- return n;
- }
- else
- {
- for ( String repoId : repoIds )
- {
- Collection<String> projects = metadataResolver.resolveProjects( repositorySession, repoId, n );
- if ( projects != null && !projects.isEmpty() )
- {
- log.debug( "{} is not collapsible as it has projects", n );
- return n;
- }
- }
- return collapseNamespaces( repositorySession, metadataResolver, repoIds,
- n + "." + subNamespaces.iterator().next() );
- }
- }
-
- public Cache<String, ProjectVersionMetadata> getVersionMetadataCache()
- {
- return versionMetadataCache;
- }
-
- public void setVersionMetadataCache( Cache<String, ProjectVersionMetadata> versionMetadataCache )
- {
- this.versionMetadataCache = versionMetadataCache;
- }
- }
|