2011-12-19 16:46:14 +00:00

517 lines
21 KiB
Java

/*
* Copyright (C) 2005-2010 Alfresco Software Limited.
*
* This file is part of Alfresco
*
* 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 <http://www.gnu.org/licenses/>.
*/
package org.alfresco.repo.rendition;
import static org.alfresco.model.ContentModel.PROP_NODE_REF;
import static org.alfresco.model.ContentModel.PROP_STORE_NAME;
import java.io.Serializable;
import java.util.Arrays;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import org.alfresco.model.ContentModel;
import org.alfresco.model.RenditionModel;
import org.alfresco.repo.policy.BehaviourFilter;
import org.alfresco.repo.rendition.executer.AbstractRenderingEngine;
import org.alfresco.service.cmr.rendition.RenditionDefinition;
import org.alfresco.service.cmr.rendition.RenditionService;
import org.alfresco.service.cmr.rendition.RenditionServiceException;
import org.alfresco.service.cmr.repository.ChildAssociationRef;
import org.alfresco.service.cmr.repository.NodeRef;
import org.alfresco.service.cmr.repository.NodeService;
import org.alfresco.service.namespace.NamespaceService;
import org.alfresco.service.namespace.QName;
import org.alfresco.service.namespace.QNamePattern;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
/**
* This class is responsible for placing a rendition node in the correct
* location given a temporary rendition, a source node, a rendition location and
* optionally an old rendition. This manages the complex logic of deciding
* whether to move an old rendition or orphan it and create a new one amongst
* other things.
*
* @author Nick Smith
*/
public class RenditionNodeManager
{
/** Logger */
private static Log logger = LogFactory.getLog(RenditionNodeManager.class);
private static final List<QName> unchangedProperties = Arrays.asList(PROP_NODE_REF, PROP_STORE_NAME);
private static final String LINE_BREAK = System.getProperty("line.separator", "\n");
/**
* The source node being rendered.
*/
private final NodeRef sourceNode;
private final NodeRef tempRenditionNode;
private final RenditionDefinition renditionDefinition;
private final RenditionLocation location;
private final NodeService nodeService;
private final BehaviourFilter behaviourFilter;
private final RenditionService renditionService;
/**
* This holds an existing rendition node if one exists and is linked to the source node
* by a correctly named rendition association.
*/
private final NodeRef existingLinkedRendition;
/**
*
* @param sourceNode the source node which is being rendered.
* @param tempRenditionNode the temporary rendition
* @param location the proposed location of the rendition node.
* @param renditionDefinition
* @param nodeService
* @param renditionService
*/
public RenditionNodeManager(NodeRef sourceNode, NodeRef tempRenditionNode, RenditionLocation location,
RenditionDefinition renditionDefinition, NodeService nodeService, RenditionService renditionService,
BehaviourFilter behaviourFilter)
{
this.sourceNode = sourceNode;
this.tempRenditionNode = tempRenditionNode;
this.location = location;
this.renditionDefinition = renditionDefinition;
this.nodeService = nodeService;
this.renditionService = renditionService;
this.behaviourFilter = behaviourFilter;
this.existingLinkedRendition = getExistingRendition();
if (logger.isDebugEnabled())
{
StringBuilder msg = new StringBuilder();
msg.append("Creating/updating rendition based on:").append(LINE_BREAK)
.append(" sourceNode: ").append(sourceNode).append(LINE_BREAK)
.append(" tempRendition: ").append(tempRenditionNode).append(LINE_BREAK)
.append(" parentNode: ").append(location.getParentRef()).append(LINE_BREAK)
.append(" childNode: ").append(location.getChildRef()).append(LINE_BREAK)
.append(" childName: ").append(location.getChildName()).append(LINE_BREAK)
.append(" renditionDefinition.name: ").append(renditionDefinition.getRenditionName());
logger.debug(msg.toString());
}
}
/**
* This method returns the {@link ChildAssociationRef} for the rendition node. In doing this
* it may reuse an existing rendition node, move an existing rendition node or create a new rendition node
* as appropriate.
* @return the primary parent association of the rendition node, which may not be the rendition association.
*/
public ChildAssociationRef findOrCreateRenditionNode()
{
QName renditionName = renditionDefinition.getRenditionName();
ChildAssociationRef result;
// If no rendition already exists create a new rendition node and association.
if (existingLinkedRendition == null)
{
if (logger.isDebugEnabled())
{
logger.debug("No existing rendition was found to be linked from the source node.");
}
result = getSpecifiedRenditionOrCreateNewRendition(renditionName);
}
else
{
// If a rendition exists and is already in the correct location then
// return that rendition's primary parent association
if (isOldRenditionInCorrectLocation())
{
result = nodeService.getPrimaryParent(existingLinkedRendition);
}
else
{
// If the old rendition is in the wrong location and the 'orphan
// existing rendition' param is set to true or the RenditionLocation
// specifies a destination NodeRef then delete the old
// rendition association and create a new rendition node.
if (isOrphaningRequired())
{
orphanOldRendition(renditionName);
result = getSpecifiedRenditionOrCreateNewRendition(renditionName);
}
// If the old rendition is in the wrong place and the 'orphan existing
// rendition' param is not set to true then move the existing rendition
// to the correct location.
result = moveOldRendition(renditionName);
}
}
transferNodeProperties(result);
return result;
}
/**
* This method moves the old rendition to the required location giving it the correct parent-assoc type and
* the specified association name.
*
* @param renditionName the name to put on the newly created association.
* @return the ChildAssociationRef of the moved nodeRef.
*/
private ChildAssociationRef moveOldRendition(QName renditionName)
{
NodeRef parent = location.getParentRef();
QName assocName = getAssociationName(parent.equals(sourceNode), renditionName);
QName assocType = sourceNode.equals(parent) ? RenditionModel.ASSOC_RENDITION : ContentModel.ASSOC_CONTAINS;
ChildAssociationRef result = nodeService.moveNode(existingLinkedRendition, parent, assocType, assocName);
if (logger.isDebugEnabled())
{
logger.debug("The old rendition was moved to " + result);
}
return result;
}
/**
* This method performs the 'orphaning' of the oldRendition. It removes the rendition aspect(s) and removes
* the child-association linking the old rendition to its source node. The old rendition node is not deleted.
*
* @param renditionName the name of the rendition.
* @throws RenditionServiceException if there was not exactly one parent assoc from the oldRendition having the specified renditionName
* or if the matching parent assoc was not to the correct source node.
*/
private void orphanOldRendition(QNamePattern renditionName)
{
// Get all parent assocs from the old rendition of the specified renditionName.
List<ChildAssociationRef> parents = nodeService.getParentAssocs(existingLinkedRendition, RenditionModel.ASSOC_RENDITION, renditionName);
// There should only be one matching assoc.
if(parents.size() == 1)
{
ChildAssociationRef parentAssoc = parents.get(0);
if(parentAssoc.getParentRef().equals(sourceNode))
{
if (logger.isDebugEnabled())
{
logger.debug("Orphaning old rendition node " + existingLinkedRendition);
}
behaviourFilter.disableBehaviour(existingLinkedRendition, ContentModel.ASPECT_AUDITABLE);
try
{
nodeService.removeAspect(existingLinkedRendition, RenditionModel.ASPECT_HIDDEN_RENDITION);
nodeService.removeAspect(existingLinkedRendition, RenditionModel.ASPECT_VISIBLE_RENDITION);
}
finally
{
behaviourFilter.enableBehaviour(existingLinkedRendition, ContentModel.ASPECT_AUDITABLE);
}
nodeService.removeChildAssociation(parentAssoc);
return;
}
}
String msg = "Node: " + existingLinkedRendition
+ " is not a rendition of type: " + renditionName
+ " for source node: " + sourceNode;
if (logger.isDebugEnabled())
{
logger.debug(msg);
}
throw new RenditionServiceException(msg);
}
/**
* This method determines whether or not orphaning of the old rendition is required.
*
* @return <code>true</code> if orphaning is required, else <code>false</code>.
*/
private boolean isOrphaningRequired()
{
boolean result = isOrphaningRequiredWithoutLog();
if (logger.isDebugEnabled())
{
StringBuilder msg = new StringBuilder();
msg.append("The old rendition does ");
if (result == false)
{
msg.append("not ");
}
msg.append("require orphaning.");
logger.debug(msg.toString());
}
return result;
}
/**
* This method determines whether or not orphaning of the old rendition is
* required.
*
* @return <code>true</code> if orphaning is required, else
* <code>false</code>.
*/
private boolean isOrphaningRequiredWithoutLog()
{
// Orphaning is required if the old rendition is in the wrong location and the 'orphan
// existing rendition' param is set to true or the RenditionLocation specifies a destination NodeRef.
if (location.getChildRef() != null)
return true;
else
return AbstractRenderingEngine.getParamWithDefault(RenditionService.PARAM_ORPHAN_EXISTING_RENDITION,
Boolean.FALSE, renditionDefinition);
}
/**
* This method determines whether or not the old rendition is already in the correct location.
*/
private boolean isOldRenditionInCorrectLocation()
{
boolean result = isOldRenditionInCorrectLocationWithoutLog();
if (logger.isDebugEnabled())
{
StringBuilder msg = new StringBuilder();
msg.append("The old rendition was ");
if (result == false)
{
msg.append("not ");
}
msg.append("in the correct location");
logger.debug(msg.toString());
}
return result;
}
private boolean isOldRenditionInCorrectLocationWithoutLog()
{
NodeRef destination = location.getChildRef();
if (destination != null)
{
return destination.equals(existingLinkedRendition);
}
ChildAssociationRef oldParentAssoc = nodeService.getPrimaryParent(existingLinkedRendition);
NodeRef oldParent = oldParentAssoc.getParentRef();
if (oldParent.equals(location.getParentRef()))
{
String childName = location.getChildName();
if (childName == null)
return true;
else
{
Serializable oldName = nodeService.getProperty(existingLinkedRendition, ContentModel.PROP_NAME);
return childName.equals(oldName);
}
}
return false;
}
private ChildAssociationRef getSpecifiedRenditionOrCreateNewRendition(QName renditionName)
{
ChildAssociationRef result;
NodeRef destination = location.getChildRef();
if (destination != null)
{
checkDestinationNodeIsAcceptable(destination);
result = nodeService.getPrimaryParent(destination);
if (logger.isDebugEnabled())
{
logger.debug("Destination was not null. Using primary parent of " + destination);
}
}
else
{
QName nodeType = null;
if (null != tempRenditionNode)
{
nodeType = nodeService.getType(tempRenditionNode);
}
result = createNewRendition(renditionName, nodeType);
}
if (logger.isDebugEnabled())
{
logger.debug("Using rendition " + result);
}
return result;
}
private void checkDestinationNodeIsAcceptable(NodeRef destination)
{
if (!nodeService.exists(destination))
{
return;
}
if (!renditionService.isRendition(destination))
{
throw new RenditionServiceException("Cannot perform a rendition to an existing node that is not a rendition.");
}
ChildAssociationRef sourceAssoc = renditionService.getSourceNode(destination);
if (!sourceAssoc.getParentRef().equals(sourceNode))
{
throw new RenditionServiceException("Cannot perform a rendition to an existing rendition node whose source is different.");
}
if (!sourceAssoc.getQName().equals(renditionDefinition.getRenditionName()))
{
throw new RenditionServiceException("Cannot perform a rendition to an existing rendition node whose rendition name is different.");
}
if (logger.isDebugEnabled())
{
logger.debug("Using destination node " + destination + " with existing srcNode: " + sourceAssoc);
}
}
/**
* This method creates a new rendition node. If the source node for this rendition is not
* the primary parent of the newly created rendition node, the rendition node is added as
* a child of the source node.
*
* @param renditionName
* @param nodeTypeQName
* @return the primary parent association of the newly created rendition node.
*/
private ChildAssociationRef createNewRendition(QName renditionName, QName nodeTypeQName)
{
NodeRef parentRef = location.getParentRef();
boolean parentIsSource = parentRef.equals(sourceNode);
QName renditionType = RenditionModel.ASSOC_RENDITION;
QName assocTypeQName = parentIsSource ? renditionType : ContentModel.ASSOC_CONTAINS;
if (null == nodeTypeQName)
{
nodeTypeQName = ContentModel.TYPE_CONTENT;
}
QName assocName = getAssociationName(parentIsSource, renditionName);
ChildAssociationRef primaryAssoc = nodeService.createNode(parentRef,
assocTypeQName, assocName, nodeTypeQName);
if (logger.isDebugEnabled())
{
StringBuilder msg = new StringBuilder();
msg.append("Created final rendition node ").append(primaryAssoc);
logger.debug(msg.toString());
}
// If the new rendition is not directly under the source node then add
// the rendition association.
if (parentIsSource == false)
{
NodeRef rendition = primaryAssoc.getChildRef();
ChildAssociationRef newChild = null;
behaviourFilter.disableBehaviour(sourceNode, ContentModel.ASPECT_AUDITABLE);
try
{
newChild = nodeService.addChild(sourceNode, rendition, renditionType, renditionName);
}
finally
{
behaviourFilter.enableBehaviour(sourceNode, ContentModel.ASPECT_AUDITABLE);
}
if (logger.isDebugEnabled())
{
StringBuilder msg = new StringBuilder();
msg.append("Added new rendition node as child of source node ").append(newChild);
logger.debug(msg.toString());
}
}
return primaryAssoc;
}
/**
* @param parentIsSource
* @param renditionName
* @return
*/
private QName getAssociationName(boolean parentIsSource, QName renditionName)
{
// If the parent is not the source node and the location has a child name then use that name.
QName assocName = renditionName;
String childName = location.getChildName();
if(parentIsSource==false
&& childName!=null
&& childName.length()>0)
{
assocName = QName.createQName(NamespaceService.CONTENT_MODEL_1_0_URI, childName);
}
return assocName;
}
/**
* This method returns the rendition on the given sourceNode with the given renditionDefinition, if such
* a rendition exists.
*
* @return the rendition node if one exists, else null.
*/
private NodeRef getExistingRendition()
{
QName renditionName = renditionDefinition.getRenditionName();
ChildAssociationRef renditionAssoc = renditionService.getRenditionByName(sourceNode, renditionName);
NodeRef result = (renditionAssoc == null) ? null : renditionAssoc.getChildRef();
if (logger.isDebugEnabled())
{
StringBuilder msg = new StringBuilder();
msg.append("Existing rendition with name ")
.append(renditionName)
.append(": ")
.append(result);
logger.debug(msg.toString());
}
return result;
}
/**
* This method copies properties from the temporary rendition node onto the targetNode. It also sets the node type.
* {@link #unchangedProperties Some properties} are not copied.
* @param finalRenditionAssoc
*/
private void transferNodeProperties(ChildAssociationRef finalRenditionAssoc)
{
NodeRef targetNode = finalRenditionAssoc.getChildRef();
if (logger.isDebugEnabled())
{
StringBuilder msg = new StringBuilder();
msg.append("Transferring some properties from ").append(tempRenditionNode).append(" to ").append(targetNode);
logger.debug(msg.toString());
}
// Copy the type from the temporary rendition to the real rendition, if required
QName type = nodeService.getType(tempRenditionNode);
if ((null != type) && !type.equals(nodeService.getType(targetNode)))
{
nodeService.setType(targetNode, type);
}
// Copy over all regular properties from the temporary rendition
Map<QName, Serializable> newProps = nodeService.getProperties(targetNode);
for(Entry<QName,Serializable> entry : nodeService.getProperties(tempRenditionNode).entrySet())
{
QName propKey = entry.getKey();
if(unchangedProperties.contains(propKey) ||
NamespaceService.SYSTEM_MODEL_1_0_URI.equals(propKey.getNamespaceURI()))
{
// These shouldn't be copied over
continue;
}
newProps.put(propKey, entry.getValue());
}
nodeService.setProperties(targetNode, newProps);
}
}