/* * #%L * Alfresco Remote API * %% * Copyright (C) 2005 - 2016 Alfresco Software Limited * %% * This file is part of the Alfresco software. * If the software was purchased under a paid Alfresco license, the terms of * the paid license agreement will prevail. Otherwise, the software is * provided under the following open source license terms: * * Alfresco is free software: you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * Alfresco is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with Alfresco. If not, see . * #L% */ package org.alfresco.rest.api.nodes; import org.alfresco.model.ContentModel; import org.alfresco.repo.security.permissions.AccessDeniedException; import org.alfresco.repo.version.Version2Model; import org.alfresco.repo.version.VersionModel; import org.alfresco.rest.api.Nodes; import org.alfresco.rest.api.model.Node; import org.alfresco.rest.api.model.NodeTarget; import org.alfresco.rest.api.model.UserInfo; import org.alfresco.rest.api.model.VersionOptions; import org.alfresco.rest.framework.BinaryProperties; import org.alfresco.rest.framework.Operation; import org.alfresco.rest.framework.WebApiDescription; import org.alfresco.rest.framework.core.exceptions.EntityNotFoundException; import org.alfresco.rest.framework.core.exceptions.PermissionDeniedException; import org.alfresco.rest.framework.resource.RelationshipResource; import org.alfresco.rest.framework.resource.actions.interfaces.RelationshipResourceAction; import org.alfresco.rest.framework.resource.actions.interfaces.RelationshipResourceBinaryAction; import org.alfresco.rest.framework.resource.content.BinaryResource; import org.alfresco.rest.framework.resource.parameters.CollectionWithPagingInfo; import org.alfresco.rest.framework.resource.parameters.Paging; import org.alfresco.rest.framework.resource.parameters.Parameters; import org.alfresco.rest.framework.webscripts.WithResponse; import org.alfresco.service.ServiceRegistry; import org.alfresco.service.cmr.coci.CheckOutCheckInService; import org.alfresco.service.cmr.repository.NodeRef; import org.alfresco.service.cmr.security.AccessStatus; import org.alfresco.service.cmr.security.PermissionService; import org.alfresco.service.cmr.version.Version; import org.alfresco.service.cmr.version.VersionHistory; import org.alfresco.service.cmr.version.VersionService; import org.alfresco.service.cmr.version.VersionType; import org.alfresco.service.namespace.QName; import org.alfresco.util.ParameterCheck; import org.alfresco.util.PropertyCheck; import org.springframework.beans.factory.InitializingBean; import javax.servlet.http.HttpServletResponse; import java.io.Serializable; import java.util.ArrayList; import java.util.HashMap; import java.util.List; import java.util.Map; /** * Node Versions - version history * * @author janv */ @RelationshipResource(name = "versions", entityResource = NodesEntityResource.class, title = "Node Versions") public class NodeVersionsRelation extends AbstractNodeRelation implements RelationshipResourceAction.Read, RelationshipResourceAction.ReadById, RelationshipResourceBinaryAction.Read, RelationshipResourceAction.Delete, InitializingBean { protected VersionService versionService; @Override public void afterPropertiesSet() { PropertyCheck.mandatory(this, "serviceRegistry", sr); ParameterCheck.mandatory("nodes", this.nodes); this.versionService = sr.getVersionService(); } /** * List version history * * @param nodeId String id of (live) node */ @Override @WebApiDescription(title = "Return version history as a paged list of version node infos") public CollectionWithPagingInfo readAll(String nodeId, Parameters parameters) { NodeRef nodeRef = nodes.validateOrLookupNode(nodeId, null); VersionHistory vh = versionService.getVersionHistory(nodeRef); Map mapUserInfo = new HashMap<>(10); List includeParam = parameters.getInclude(); List collection = null; if (vh != null) { collection = new ArrayList<>(vh.getAllVersions().size()); for (Version v : vh.getAllVersions()) { Node node = nodes.getFolderOrDocument(v.getFrozenStateNodeRef(), null, null, includeParam, mapUserInfo); mapVersionInfo(v, node); collection.add(node); } } return listPage(collection, parameters.getPaging()); } private void mapVersionInfo(Version v, Node aNode) { aNode.setNodeRef(new NodeRef("", "", v.getVersionLabel())); aNode.setVersionComment(v.getDescription()); Map props = aNode.getProperties(); if (props != null) { // special case (as per Version2Service) props.put("cm:"+Version2Model.PROP_VERSION_TYPE, v.getVersionProperty(Version2Model.PROP_VERSION_TYPE)); } //Don't show parentId, createdAt, createdByUser aNode.setParentId(null); aNode.setCreated(null); aNode.setCreatedByUser(null); } @Override @WebApiDescription(title="Get version node info", description = "Return metadata for a specific version node") public Node readById(String nodeId, String versionId, Parameters parameters) { Version v = findVersion(nodeId, versionId); if (v != null) { Node node = nodes.getFolderOrDocumentFullInfo(v.getFrozenStateNodeRef(), null, null, parameters, null); mapVersionInfo(v, node); return node; } throw new EntityNotFoundException(nodeId+"-"+versionId); } @WebApiDescription(title = "Download version content", description = "Download version content") @BinaryProperties({ "content" }) @Override public BinaryResource readProperty(String nodeId, String versionId, Parameters parameters) { Version v = findVersion(nodeId, versionId); if (v != null) { NodeRef versionNodeRef = v.getFrozenStateNodeRef(); return nodes.getContent(versionNodeRef, parameters, true); // TODO should we record version downloads ? } throw new EntityNotFoundException(nodeId+"-"+versionId); } @Operation("revert") @WebApiDescription(title = "Revert Version", description="Reverts (ie. promotes) specified version to become a new, most recent, version", successStatus = HttpServletResponse.SC_OK) public Node revertById(String nodeId, String versionId, VersionOptions versionOptions, Parameters parameters, WithResponse withResponse) { Version v = findVersion(nodeId, versionId); if (v != null) { CheckOutCheckInService cociService = sr.getCheckOutCheckInService(); NodeRef nodeRef = v.getVersionedNodeRef(); String versionComment = versionOptions.getComment(); VersionType versionType = VersionType.MINOR; Boolean versionMajor = versionOptions.getMajorVersion(); if ((versionMajor != null) && (versionMajor)) { versionType = VersionType.MAJOR; } Map versionProperties = new HashMap<>(2); versionProperties.put(VersionModel.PROP_VERSION_TYPE, versionType); if (versionComment != null) { versionProperties.put(VersionModel.PROP_DESCRIPTION, versionComment); } //cancel editing if we want to revert if (sr.getNodeService().hasAspect(nodeRef, ContentModel.ASPECT_WORKING_COPY)) { nodeRef = cociService.cancelCheckout(nodeRef); } // TODO review default for deep and/or whether we should make it an option versionService.revert(nodeRef, v, false); // Checkout/Checkin the node - to store the new version in version history NodeRef wcNodeRef = cociService.checkout(nodeRef); cociService.checkin(wcNodeRef, versionProperties); // get latest version v = versionService.getVersionHistory(nodeRef).getHeadVersion(); Node node = nodes.getFolderOrDocumentFullInfo(v.getFrozenStateNodeRef(), null, null, parameters, null); mapVersionInfo(v, node); return node; } throw new EntityNotFoundException(nodeId+"-"+versionId); } @Override @WebApiDescription(title = "Delete version") public void delete(String nodeId, String versionId, Parameters parameters) { Version v = findVersion(nodeId, versionId); NodeRef liveVersionedNodeRef = v.getVersionedNodeRef(); if (sr.getPermissionService().hasPermission(liveVersionedNodeRef, PermissionService.DELETE) != AccessStatus.ALLOWED) { throw new PermissionDeniedException("Cannot delete version"); } versionService.deleteVersion(liveVersionedNodeRef, v); Map props = sr.getNodeService().getProperties(liveVersionedNodeRef); if (props.get(ContentModel.PROP_VERSION_LABEL) == null) { // last version was deleted if (props.get(ContentModel.PROP_VERSION_TYPE) != null) { // minor fix up to versionable aspect - ie. remove versionType // TODO should we disable behaviours, eg. audit, versionable sr.getNodeService().removeProperty(liveVersionedNodeRef, ContentModel.PROP_VERSION_TYPE); } } } private Version findVersion(String nodeId, String versionLabelId) { NodeRef nodeRef = nodes.validateOrLookupNode(nodeId, null); return versionService.getVersionHistory(nodeRef).getVersion(versionLabelId); } }