mirror of https://github.com/apache/archiva.git
[MRM-1362] Add simple 'CRUD' pages for project-level metadata along with a "generic metadata" plugin
o action methods for updating and retrieving metadata properties o updated struts config for getting project metadata o added page for metadata in webapp artifact browse o added test for getting project metadata git-svn-id: https://svn.apache.org/repos/asf/archiva/trunk@940775 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
aad0ae13cf
commit
8f4f8c83e9
|
@ -25,6 +25,7 @@ import org.apache.archiva.metadata.model.Dependency;
|
|||
import org.apache.archiva.metadata.model.MailingList;
|
||||
import org.apache.archiva.metadata.model.ProjectVersionMetadata;
|
||||
import org.apache.archiva.metadata.model.ProjectVersionReference;
|
||||
import org.apache.archiva.metadata.repository.MetadataRepository;
|
||||
import org.apache.archiva.metadata.repository.MetadataResolutionException;
|
||||
import org.apache.archiva.metadata.repository.MetadataResolver;
|
||||
import org.apache.archiva.metadata.repository.storage.maven2.MavenArtifactFacet;
|
||||
|
@ -67,6 +68,11 @@ public class ShowArtifactAction
|
|||
*/
|
||||
private RepositoryContentFactory repositoryFactory;
|
||||
|
||||
/**
|
||||
* @plexus.requirement
|
||||
*/
|
||||
private MetadataRepository metadataRepository;
|
||||
|
||||
/* .\ Exposed Output Objects \.__________________________________ */
|
||||
|
||||
private String groupId;
|
||||
|
@ -95,20 +101,44 @@ public class ShowArtifactAction
|
|||
|
||||
private boolean dependencyTree = false;
|
||||
|
||||
private ProjectVersionMetadata projectMetadata;
|
||||
|
||||
/**
|
||||
* Show the versioned project information tab.
|
||||
* TODO: Change name to 'project' - we are showing project versions here, not specific artifact information (though
|
||||
* that is rendered in the download box).
|
||||
*/
|
||||
public String artifact()
|
||||
{
|
||||
|
||||
// In the future, this should be replaced by the repository grouping mechanism, so that we are only making
|
||||
// simple resource requests here and letting the resolver take care of it
|
||||
String errorMsg = null;
|
||||
ProjectVersionMetadata versionMetadata = getProjectVersionMetadata();
|
||||
|
||||
if ( versionMetadata == null )
|
||||
{
|
||||
addActionError( errorMsg != null ? errorMsg : "Artifact not found" );
|
||||
return ERROR;
|
||||
}
|
||||
|
||||
if ( versionMetadata.isIncomplete() )
|
||||
{
|
||||
addIncompleteModelWarning();
|
||||
}
|
||||
|
||||
model = versionMetadata;
|
||||
|
||||
return SUCCESS;
|
||||
}
|
||||
|
||||
private ProjectVersionMetadata getProjectVersionMetadata()
|
||||
{
|
||||
ProjectVersionMetadata versionMetadata = null;
|
||||
artifacts = new LinkedHashMap<String, List<ArtifactDownloadInfo>>();
|
||||
|
||||
List<String> repos = getObservableRepos();
|
||||
// In the future, this should be replaced by the repository grouping mechanism, so that we are only making
|
||||
// simple resource requests here and letting the resolver take care of it
|
||||
String errorMsg = null;
|
||||
|
||||
for ( String repoId : repos )
|
||||
{
|
||||
if ( versionMetadata == null )
|
||||
|
@ -122,7 +152,7 @@ public class ShowArtifactAction
|
|||
catch ( MetadataResolutionException e )
|
||||
{
|
||||
addIncompleteModelWarning();
|
||||
|
||||
|
||||
// TODO: need a consistent way to construct this - same in ArchivaMetadataCreationConsumer
|
||||
versionMetadata = new ProjectVersionMetadata();
|
||||
versionMetadata.setId( version );
|
||||
|
@ -138,7 +168,7 @@ public class ShowArtifactAction
|
|||
public int compare( ArtifactMetadata o1, ArtifactMetadata o2 )
|
||||
{
|
||||
// sort by version (reverse), then ID
|
||||
// TODO: move version sorting into repository handling (maven2 specific), and perhaps add a
|
||||
// TODO: move version sorting into repository handling (maven2 specific), and perhaps add a
|
||||
// way to get latest instead
|
||||
int result = new DefaultArtifactVersion( o2.getVersion() ).compareTo(
|
||||
new DefaultArtifactVersion( o1.getVersion() ) );
|
||||
|
@ -160,20 +190,7 @@ public class ShowArtifactAction
|
|||
}
|
||||
}
|
||||
|
||||
if ( versionMetadata == null )
|
||||
{
|
||||
addActionError( errorMsg != null ? errorMsg : "Artifact not found" );
|
||||
return ERROR;
|
||||
}
|
||||
|
||||
if ( versionMetadata.isIncomplete() )
|
||||
{
|
||||
addIncompleteModelWarning();
|
||||
}
|
||||
|
||||
model = versionMetadata;
|
||||
|
||||
return SUCCESS;
|
||||
return versionMetadata;
|
||||
}
|
||||
|
||||
private void addIncompleteModelWarning()
|
||||
|
@ -253,6 +270,32 @@ public class ShowArtifactAction
|
|||
return artifact();
|
||||
}
|
||||
|
||||
public String projectMetadata()
|
||||
{
|
||||
projectMetadata = getProjectVersionMetadata();
|
||||
String errorMsg = null;
|
||||
|
||||
if ( projectMetadata == null )
|
||||
{
|
||||
addActionError( errorMsg != null ? errorMsg : "Artifact not found" );
|
||||
return ERROR;
|
||||
}
|
||||
|
||||
if ( projectMetadata.isIncomplete() )
|
||||
{
|
||||
addIncompleteModelWarning();
|
||||
}
|
||||
|
||||
return SUCCESS;
|
||||
}
|
||||
|
||||
public String updateProjectMetadata()
|
||||
{
|
||||
metadataRepository.updateProjectVersion( repositoryId, groupId, artifactId, projectMetadata );
|
||||
|
||||
return SUCCESS;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void validate()
|
||||
{
|
||||
|
@ -357,6 +400,16 @@ public class ShowArtifactAction
|
|||
return dependencyTree;
|
||||
}
|
||||
|
||||
public ProjectVersionMetadata getProjectMetadata()
|
||||
{
|
||||
return projectMetadata;
|
||||
}
|
||||
|
||||
public void setProjectMetadata( ProjectVersionMetadata projectMetadata )
|
||||
{
|
||||
this.projectMetadata = projectMetadata;
|
||||
}
|
||||
|
||||
// TODO: move this into the artifact metadata itself via facets where necessary
|
||||
|
||||
public class ArtifactDownloadInfo
|
||||
|
@ -478,5 +531,7 @@ public class ShowArtifactAction
|
|||
{
|
||||
return path;
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
}
|
||||
|
|
|
@ -218,6 +218,10 @@
|
|||
<result>/WEB-INF/jsp/showArtifact.jsp</result>
|
||||
</action>
|
||||
|
||||
<action name="showProjectMetadata" class="showArtifactAction" method="projectMetadata">
|
||||
<result>/WEB-INF/jsp/showArtifact.jsp</result>
|
||||
</action>
|
||||
|
||||
</package>
|
||||
|
||||
<package name="components" namespace="/components" extends="struts-default">
|
||||
|
|
|
@ -0,0 +1,278 @@
|
|||
<%--
|
||||
~ 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.
|
||||
--%>
|
||||
|
||||
<%@ taglib prefix="s" uri="/struts-tags" %>
|
||||
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
|
||||
<%@ taglib prefix="archiva" uri="/WEB-INF/taglib.tld" %>
|
||||
|
||||
<script type="text/javascript">
|
||||
$(function() {
|
||||
$("#accordion2").accordion();
|
||||
});
|
||||
</script>
|
||||
|
||||
<p>
|
||||
<archiva:groupIdLink var="${groupId}" includeTop="true" />
|
||||
|
||||
<c:set var="url">
|
||||
<s:url action="browseArtifact" namespace="/">
|
||||
<s:param name="groupId" value="%{#attr.groupId}"/>
|
||||
<s:param name="artifactId" value="%{#attr.artifactId}"/>
|
||||
</s:url>
|
||||
</c:set>
|
||||
<a href="${url}">${artifactId}</a> /
|
||||
<strong>${version}</strong>
|
||||
</p>
|
||||
|
||||
<c:if test="${!empty (projectMetadata.description)}">
|
||||
<blockquote>${projectMetadata.description}</blockquote>
|
||||
</c:if>
|
||||
|
||||
<table class="infoTable">
|
||||
<tr>
|
||||
<th>Project Metadata ID</th>
|
||||
<td>${projectMetadata.id}</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th>URL</th>
|
||||
<td>${projectMetadata.url}</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th>Name</th>
|
||||
<td>${projectMetadata.name}</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th>Description</th>
|
||||
<td>${projectMetadata.description}</td>
|
||||
</tr>
|
||||
|
||||
<c:if test="${projectMetadata.organization != null || !empty (projectMetadata.licenses)
|
||||
|| projectMetadata.issueManagement != null || projectMetadata.ciManagement != null }">
|
||||
|
||||
<h2>Other Details</h2>
|
||||
<table class="infoTable">
|
||||
<c:if test="${projectMetadata.organization != null}">
|
||||
<tr>
|
||||
<th>Organisation</th>
|
||||
<td>
|
||||
<c:choose>
|
||||
<c:when test="${projectMetadata.organization.url != null}">
|
||||
<a href="${projectMetadata.organization.url}">${projectMetadata.organization.name}</a>
|
||||
</c:when>
|
||||
<c:otherwise>
|
||||
${projectMetadata.organization.name}
|
||||
</c:otherwise>
|
||||
</c:choose>
|
||||
</td>
|
||||
</tr>
|
||||
</c:if>
|
||||
<c:if test="${!empty (projectMetadata.licenses)}">
|
||||
<c:forEach items="${projectMetadata.licenses}" var="license">
|
||||
<tr>
|
||||
<th>License</th>
|
||||
<td>
|
||||
<c:choose>
|
||||
<c:when test="${!empty (license.url)}">
|
||||
<a href="${license.url}">${license.name}</a>
|
||||
</c:when>
|
||||
<c:otherwise>
|
||||
${license.name}
|
||||
</c:otherwise>
|
||||
</c:choose>
|
||||
</td>
|
||||
</tr>
|
||||
</c:forEach>
|
||||
</c:if>
|
||||
<c:if test="${projectMetadata.issueManagement != null}">
|
||||
<tr>
|
||||
<th>Issue Tracker</th>
|
||||
<td>
|
||||
<c:choose>
|
||||
<c:when test="${!empty (projectMetadata.issueManagement.url)}">
|
||||
<a href="${projectMetadata.issueManagement.url}">${projectMetadata.issueManagement.system}</a>
|
||||
</c:when>
|
||||
<c:otherwise>
|
||||
${projectMetadata.issueManagement.system}
|
||||
</c:otherwise>
|
||||
</c:choose>
|
||||
</td>
|
||||
</tr>
|
||||
</c:if>
|
||||
<c:if test="${projectMetadata.ciManagement != null}">
|
||||
<tr>
|
||||
<th>Continuous Integration</th>
|
||||
<td>
|
||||
<c:choose>
|
||||
<c:when test="${!empty (projectMetadata.ciManagement.url)}">
|
||||
<a href="${projectMetadata.ciManagement.url}">${projectMetadata.ciManagement.system}</a>
|
||||
</c:when>
|
||||
<c:otherwise>
|
||||
${projectMetadata.ciManagement.system}
|
||||
</c:otherwise>
|
||||
</c:choose>
|
||||
</td>
|
||||
</tr>
|
||||
</c:if>
|
||||
</table>
|
||||
</c:if>
|
||||
|
||||
<c:if test="${projectMetadata.scm != null}">
|
||||
<h2>SCM</h2>
|
||||
<table class="infoTable">
|
||||
<c:if test="${!empty (projectMetadata.scm.connection)}">
|
||||
<tr>
|
||||
<th>Connection</th>
|
||||
<td>
|
||||
<code>${projectMetadata.scm.connection}</code>
|
||||
</td>
|
||||
</tr>
|
||||
</c:if>
|
||||
<c:if test="${!empty (projectMetadata.scm.developerConnection)}">
|
||||
<tr>
|
||||
<th>Dev. Connection</th>
|
||||
<td>
|
||||
<code>${projectMetadata.scm.developerConnection}</code>
|
||||
</td>
|
||||
</tr>
|
||||
</c:if>
|
||||
<c:if test="${!empty (projectMetadata.scm.url)}">
|
||||
<tr>
|
||||
<th>Viewer</th>
|
||||
<td>
|
||||
<a href="${projectMetadata.scm.url}">${projectMetadata.scm.url}</a>
|
||||
</td>
|
||||
</tr>
|
||||
</c:if>
|
||||
</table>
|
||||
</c:if>
|
||||
|
||||
|
||||
<c:if test="${projectMetadata.mailingLists != null || projectMetadata.dependencies != null}">
|
||||
<div id="accordion2">
|
||||
<c:if test="${!empty (projectMetadata.mailingLists)}">
|
||||
<h2><a href="#">Mailing Lists</a></h2>
|
||||
<div>
|
||||
<c:forEach items="${projectMetadata.mailingLists}" var="mailingList">
|
||||
<h3>${mailingList.name}</h3>
|
||||
<table class="infoTable">
|
||||
<c:if test="${!empty (mailingList.subscribeAddress)}">
|
||||
<tr>
|
||||
<th>Subscribe</th>
|
||||
<td>
|
||||
<code>${mailingList.subscribeAddress}</code>
|
||||
</td>
|
||||
</tr>
|
||||
</c:if>
|
||||
<c:if test="${!empty (mailingList.postAddress)}">
|
||||
<tr>
|
||||
<th>Post</th>
|
||||
<td>
|
||||
<code>${mailingList.postAddress}</code>
|
||||
</td>
|
||||
</tr>
|
||||
</c:if>
|
||||
<c:if test="${!empty (mailingList.unsubscribeAddress)}">
|
||||
<tr>
|
||||
<th>Unsubscribe</th>
|
||||
<td>
|
||||
<code>${mailingList.unsubscribeAddress}</code>
|
||||
</td>
|
||||
</tr>
|
||||
</c:if>
|
||||
<c:if test="${!empty (mailingList.mainArchiveUrl)}">
|
||||
<tr>
|
||||
<th>Archive</th>
|
||||
<td>
|
||||
<code>${mailingList.mainArchiveUrl}</code>
|
||||
</td>
|
||||
</tr>
|
||||
</c:if>
|
||||
</table>
|
||||
</c:forEach>
|
||||
</div>
|
||||
</c:if>
|
||||
|
||||
<c:if test="${!empty (projectMetadata.dependencies)}">
|
||||
<h2><a href="#">Dependencies</a></h2>
|
||||
<div>
|
||||
<c:forEach items="${projectMetadata.dependencies}" var="dependency">
|
||||
<h3>Dependency</h3>
|
||||
<table class="infoTable">
|
||||
<c:if test="${!empty (dependency.groupId)}">
|
||||
<tr>
|
||||
<th>Group ID</th>
|
||||
<td>
|
||||
<code>${dependency.groupId}</code>
|
||||
</td>
|
||||
</tr>
|
||||
</c:if>
|
||||
<c:if test="${!empty (dependency.artifactId)}">
|
||||
<tr>
|
||||
<th>Artifact ID</th>
|
||||
<td>
|
||||
<code>${dependency.artifactId}</code>
|
||||
</td>
|
||||
</tr>
|
||||
</c:if>
|
||||
<c:if test="${!empty (dependency.version)}">
|
||||
<tr>
|
||||
<th>Version</th>
|
||||
<td>
|
||||
<code>${dependency.version}</code>
|
||||
</td>
|
||||
</tr>
|
||||
</c:if>
|
||||
<c:if test="${!empty (dependency.classifier)}">
|
||||
<tr>
|
||||
<th>Classifier</th>
|
||||
<td>
|
||||
<code>${dependency.classifier}</code>
|
||||
</td>
|
||||
</tr>
|
||||
</c:if>
|
||||
<c:if test="${!empty (dependency.type)}">
|
||||
<tr>
|
||||
<th>Type</th>
|
||||
<td>
|
||||
<code>${dependency.type}</code>
|
||||
</td>
|
||||
</tr>
|
||||
</c:if>
|
||||
<c:if test="${!empty (dependency.scope)}">
|
||||
<tr>
|
||||
<th>Scope</th>
|
||||
<td>
|
||||
<code>${dependency.scope}</code>
|
||||
</td>
|
||||
</tr>
|
||||
</c:if>
|
||||
<c:if test="${!empty (dependency.systemPath)}">
|
||||
<tr>
|
||||
<th>System Path</th>
|
||||
<td>
|
||||
<code>${dependency.systemPath}</code>
|
||||
</td>
|
||||
</tr>
|
||||
</c:if>
|
||||
</table>
|
||||
</c:forEach>
|
||||
</div>
|
||||
</c:if>
|
||||
</div>
|
||||
</c:if>
|
|
@ -126,6 +126,14 @@
|
|||
</s:url>
|
||||
</c:set>
|
||||
<my:currentWWUrl url="${url}">Mailing Lists</my:currentWWUrl>
|
||||
<c:set var="url">
|
||||
<s:url action="showProjectMetadata">
|
||||
<s:param name="groupId" value="%{groupId}"/>
|
||||
<s:param name="artifactId" value="%{artifactId}"/>
|
||||
<s:param name="version" value="%{version}"/>
|
||||
</s:url>
|
||||
</c:set>
|
||||
<my:currentWWUrl url="${url}">Metadata</my:currentWWUrl>
|
||||
<%-- TODO
|
||||
<redback:ifAnyAuthorized permissions="archiva-access-reports">
|
||||
<c:set var="url">
|
||||
|
@ -211,6 +219,9 @@
|
|||
<%-- TODO: panels? this is ugly as is --%>
|
||||
<div id="tabArea">
|
||||
<c:choose>
|
||||
<c:when test="${projectMetadata != null}">
|
||||
<%@ include file="/WEB-INF/jsp/include/projectMetadata.jspf" %>
|
||||
</c:when>
|
||||
<c:when test="${dependencies != null}">
|
||||
<%@ include file="/WEB-INF/jsp/include/artifactDependencies.jspf" %>
|
||||
</c:when>
|
||||
|
@ -229,6 +240,7 @@
|
|||
<c:otherwise>
|
||||
<%@ include file="/WEB-INF/jsp/include/artifactInfo.jspf" %>
|
||||
</c:otherwise>
|
||||
|
||||
</c:choose>
|
||||
|
||||
<s:if test="hasActionMessages()">
|
||||
|
|
|
@ -368,6 +368,41 @@ public class ShowArtifactActionTest
|
|||
assertTrue( action.getArtifacts().isEmpty() );
|
||||
}
|
||||
|
||||
public void testGetProjectMetadata()
|
||||
{
|
||||
ProjectVersionMetadata versionMetadata = createProjectModel( TEST_VERSION );
|
||||
Dependency dependency1 = createDependencyBasic( "artifactId1" );
|
||||
Dependency dependency2 = createDependencyExtended( "artifactId2" );
|
||||
versionMetadata.setDependencies( Arrays.asList( dependency1, dependency2 ) );
|
||||
|
||||
MailingList ml1 = createMailingList( "Users List", "users" );
|
||||
MailingList ml2 = createMailingList( "Developers List", "dev" );
|
||||
versionMetadata.setMailingLists( Arrays.asList( ml1, ml2 ) );
|
||||
|
||||
metadataResolver.setProjectVersion( TEST_REPO, TEST_GROUP_ID, TEST_ARTIFACT_ID, versionMetadata );
|
||||
|
||||
setActionParameters();
|
||||
|
||||
String result = action.projectMetadata();
|
||||
|
||||
assertActionSuccess( action, result );
|
||||
|
||||
assertActionParameters( action );
|
||||
ProjectVersionMetadata projectMetadata = action.getProjectMetadata();
|
||||
assertDefaultModel( projectMetadata );
|
||||
|
||||
assertNotNull( projectMetadata.getDependencies() );
|
||||
assertDependencyBasic( projectMetadata.getDependencies().get( 0 ), "artifactId1" );
|
||||
assertDependencyExtended( projectMetadata.getDependencies().get( 1 ), "artifactId2" );
|
||||
|
||||
assertEquals( TEST_REPO, action.getRepositoryId() );
|
||||
assertNull( action.getModel() );
|
||||
assertNull( action.getDependees() );
|
||||
assertNull( action.getDependencies() );
|
||||
assertNull( action.getMailingLists() );
|
||||
assertTrue( action.getArtifacts().isEmpty() );
|
||||
}
|
||||
|
||||
private void assertArtifacts( List<ArtifactMetadata> expectedArtifacts,
|
||||
Map<String, List<ShowArtifactAction.ArtifactDownloadInfo>> artifactMap )
|
||||
{
|
||||
|
|
Loading…
Reference in New Issue