123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359 |
- 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.dependency.tree.maven2.DependencyTreeBuilder;
- import org.apache.archiva.maven2.metadata.MavenMetadataReader;
- import org.apache.archiva.maven2.model.Artifact;
- import org.apache.archiva.maven2.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.metadata.repository.storage.maven2.ArtifactMetadataVersionComparator;
- import org.apache.archiva.metadata.repository.storage.maven2.MavenProjectFacet;
- import org.apache.archiva.model.ArchivaArtifact;
- import org.apache.archiva.model.ArchivaRepositoryMetadata;
- import org.apache.archiva.proxy.ProxyRegistry;
- import org.apache.archiva.proxy.model.RepositoryProxyHandler;
- import org.apache.archiva.redback.components.cache.Cache;
- import org.apache.archiva.repository.ManagedRepositoryContent;
- import org.apache.archiva.repository.ReleaseScheme;
- import org.apache.archiva.repository.RepositoryException;
- import org.apache.archiva.repository.RepositoryNotFoundException;
- import org.apache.archiva.repository.metadata.base.MetadataTools;
- import org.apache.archiva.repository.storage.StorageAsset;
- import org.apache.archiva.repository.storage.StorageUtil;
- 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.archiva.xml.XMLException;
- 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
- @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 Boolean 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 Boolean.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 Boolean.TRUE;
- }
-
- @Override
- public Boolean deleteMetadata( String groupId, String artifactId, String version, String key, String repositoryId )
- throws ArchivaRestServiceException
- {
- ProjectVersionMetadata projectVersionMetadata =
- getProjectMetadata( groupId, artifactId, version, repositoryId );
-
- if ( projectVersionMetadata == null )
- {
- return Boolean.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 Boolean.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 Boolean.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 );
- ArchivaArtifact archivaArtifact = new ArchivaArtifact( groupId, artifactId, version, classifier,
- StringUtils.isEmpty( type ) ? "jar" : type,
- repoId );
- StorageAsset file = managedRepositoryContent.toFile( archivaArtifact );
- 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;
- }
- ArchivaArtifact archivaArtifact = new ArchivaArtifact( groupId, artifactId, version, classifier,
- StringUtils.isEmpty( type ) ? "jar" : type,
- repoId );
- StorageAsset file = managedRepositoryContent.toFile( archivaArtifact );
- 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
- StorageUtil.PathInformation pathInfo = StorageUtil.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 Boolean 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....
- ArchivaArtifact archivaArtifact = new ArchivaArtifact( groupId, artifactId, version,
- StringUtils.isEmpty( classifier )
- ? ""
- : classifier, "jar", repoId );
- StorageAsset file = managedRepositoryContent.toFile( archivaArtifact );
-
- if ( file != null && file.exists() )
- {
- return 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() )
- {
- try
- {
- ArchivaRepositoryMetadata archivaRepositoryMetadata =
- MavenMetadataReader.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 true;
- }
- }
- catch (XMLException | IOException e )
- {
- log.warn( "skip fail to find timestamped snapshot file: {}", e.getMessage() );
- }
- }
- }
-
- 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 true;
- }
- }
- } catch ( RepositoryException e )
- {
- log.error( e.getMessage(), e );
- throw new ArchivaRestServiceException( e.getMessage(),
- Response.Status.INTERNAL_SERVER_ERROR.getStatusCode(), e );
- }
-
- return false;
- }
-
- @Override
- public Boolean 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 Boolean importMetadata( MetadataAddRequest metadataAddRequest, String repositoryId )
- throws ArchivaRestServiceException
- {
- boolean result = 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 )
- {
- 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++;
- }
-
- StorageUtil.PathInformation pathInfo = StorageUtil.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;
- }
- }
|